]> git.donarmstrong.com Git - qmk_firmware.git/blob - quantum/process_keycode/process_unicode.c
process_unicode: Handle too long UCIS symbol names
[qmk_firmware.git] / quantum / process_keycode / process_unicode.c
1 #include "process_unicode.h"
2
3 static uint8_t input_mode;
4
5 uint16_t hex_to_keycode(uint8_t hex)
6 {
7   if (hex == 0x0) {
8     return KC_0;
9   } else if (hex < 0xA) {
10     return KC_1 + (hex - 0x1);
11   } else {
12     return KC_A + (hex - 0xA);
13   }
14 }
15
16 void set_unicode_input_mode(uint8_t os_target)
17 {
18   input_mode = os_target;
19 }
20
21 void unicode_input_start (void) {
22   switch(input_mode) {
23   case UC_OSX:
24     register_code(KC_LALT);
25     break;
26   case UC_LNX:
27     register_code(KC_LCTL);
28     register_code(KC_LSFT);
29     register_code(KC_U);
30     unregister_code(KC_U);
31     unregister_code(KC_LSFT);
32     unregister_code(KC_LCTL);
33     break;
34   case UC_WIN:
35     register_code(KC_LALT);
36     register_code(KC_PPLS);
37     unregister_code(KC_PPLS);
38     break;
39   }
40 }
41
42 void unicode_input_finish (void) {
43   switch(input_mode) {
44   case UC_OSX:
45   case UC_WIN:
46     unregister_code(KC_LALT);
47     break;
48   case UC_LNX:
49     register_code(KC_SPC);
50     unregister_code(KC_SPC);
51     break;
52   }
53 }
54
55 void register_hex(uint16_t hex) {
56   for(int i = 3; i >= 0; i--) {
57     uint8_t digit = ((hex >> (i*4)) & 0xF);
58     register_code(hex_to_keycode(digit));
59     unregister_code(hex_to_keycode(digit));
60   }
61 }
62
63 bool process_unicode(uint16_t keycode, keyrecord_t *record) {
64   if (keycode > QK_UNICODE && record->event.pressed) {
65     uint16_t unicode = keycode & 0x7FFF;
66     unicode_input_start();
67     register_hex(unicode);
68     unicode_input_finish();
69   }
70   return true;
71 }
72
73 #ifdef UCIS_ENABLE
74 void qk_ucis_start(void) {
75   qk_ucis_state.count = 0;
76   qk_ucis_state.in_progress = true;
77
78   unicode_input_start();
79   register_hex(0x2328);
80   unicode_input_finish();
81 }
82
83 static bool is_uni_seq(char *seq) {
84   uint8_t i;
85
86   for (i = 0; seq[i]; i++) {
87     uint16_t code;
88     if (('1' <= seq[i]) && (seq[i] <= '0'))
89       code = seq[i] - '1' + KC_1;
90     else
91       code = seq[i] - 'a' + KC_A;
92
93     if (i > qk_ucis_state.count || qk_ucis_state.codes[i] != code)
94       return false;
95   }
96
97   return (qk_ucis_state.codes[i] == KC_ENT ||
98           qk_ucis_state.codes[i] == KC_SPC);
99 }
100
101 __attribute__((weak))
102 void qk_ucis_symbol_fallback (void) {
103   for (uint8_t i = 0; i < qk_ucis_state.count - 1; i++) {
104     uint8_t code = qk_ucis_state.codes[i];
105     register_code(code);
106     unregister_code(code);
107   }
108 }
109
110 bool process_record_ucis (uint16_t keycode, keyrecord_t *record) {
111   uint8_t i;
112
113   if (!qk_ucis_state.in_progress)
114     return true;
115
116   if (qk_ucis_state.count >= UCIS_MAX_SYMBOL_LENGTH &&
117       !(keycode == KC_BSPC || keycode == KC_ESC || keycode == KC_SPC || keycode == KC_ENT)) {
118     return false;
119   }
120
121   if (!record->event.pressed)
122     return true;
123
124   qk_ucis_state.codes[qk_ucis_state.count] = keycode;
125   qk_ucis_state.count++;
126
127   if (keycode == KC_BSPC) {
128     if (qk_ucis_state.count >= 2) {
129       qk_ucis_state.count -= 2;
130       return true;
131     } else {
132       qk_ucis_state.count--;
133       return false;
134     }
135   }
136
137   if (keycode == KC_ENT || keycode == KC_SPC || keycode == KC_ESC) {
138     bool symbol_found = false;
139
140     for (i = qk_ucis_state.count; i > 0; i--) {
141       register_code (KC_BSPC);
142       unregister_code (KC_BSPC);
143     }
144
145     if (keycode == KC_ESC) {
146       qk_ucis_state.in_progress = false;
147       return false;
148     }
149
150     unicode_input_start();
151     for (i = 0; ucis_symbol_table[i].symbol; i++) {
152       if (is_uni_seq (ucis_symbol_table[i].symbol)) {
153         symbol_found = true;
154         for (uint8_t j = 0; ucis_symbol_table[i].codes[j]; j++) {
155           register_hex(ucis_symbol_table[i].codes[j]);
156         }
157         break;
158       }
159     }
160     if (!symbol_found) {
161       qk_ucis_symbol_fallback();
162     }
163     unicode_input_finish();
164
165     qk_ucis_state.in_progress = false;
166     return false;
167   }
168   return true;
169 }
170 #endif