]> git.donarmstrong.com Git - qmk_firmware.git/blob - quantum/process_keycode/process_unicode.c
process_unicode: Use uint32_t for UCIS purposes
[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 void register_hex32(uint32_t hex) {
64   for(int i = 7; i >= 0; i--) {
65     uint8_t digit = ((hex >> (i*8)) & 0xF);
66     register_code(hex_to_keycode(digit));
67     unregister_code(hex_to_keycode(digit));
68   }
69 }
70
71 bool process_unicode(uint16_t keycode, keyrecord_t *record) {
72   if (keycode > QK_UNICODE && record->event.pressed) {
73     uint16_t unicode = keycode & 0x7FFF;
74     unicode_input_start();
75     register_hex(unicode);
76     unicode_input_finish();
77   }
78   return true;
79 }
80
81 #ifdef UCIS_ENABLE
82 void qk_ucis_start(void) {
83   qk_ucis_state.count = 0;
84   qk_ucis_state.in_progress = true;
85
86   qk_ucis_start_user();
87 }
88
89 __attribute__((weak))
90 void qk_ucis_start_user(void) {
91   unicode_input_start();
92   register_hex(0x2328);
93   unicode_input_finish();
94 }
95
96 static bool is_uni_seq(char *seq) {
97   uint8_t i;
98
99   for (i = 0; seq[i]; i++) {
100     uint16_t code;
101     if (('1' <= seq[i]) && (seq[i] <= '0'))
102       code = seq[i] - '1' + KC_1;
103     else
104       code = seq[i] - 'a' + KC_A;
105
106     if (i > qk_ucis_state.count || qk_ucis_state.codes[i] != code)
107       return false;
108   }
109
110   return (qk_ucis_state.codes[i] == KC_ENT ||
111           qk_ucis_state.codes[i] == KC_SPC);
112 }
113
114 __attribute__((weak))
115 void qk_ucis_symbol_fallback (void) {
116   for (uint8_t i = 0; i < qk_ucis_state.count - 1; i++) {
117     uint8_t code = qk_ucis_state.codes[i];
118     register_code(code);
119     unregister_code(code);
120   }
121 }
122
123 bool process_ucis (uint16_t keycode, keyrecord_t *record) {
124   uint8_t i;
125
126   if (!qk_ucis_state.in_progress)
127     return true;
128
129   if (qk_ucis_state.count >= UCIS_MAX_SYMBOL_LENGTH &&
130       !(keycode == KC_BSPC || keycode == KC_ESC || keycode == KC_SPC || keycode == KC_ENT)) {
131     return false;
132   }
133
134   if (!record->event.pressed)
135     return true;
136
137   qk_ucis_state.codes[qk_ucis_state.count] = keycode;
138   qk_ucis_state.count++;
139
140   if (keycode == KC_BSPC) {
141     if (qk_ucis_state.count >= 2) {
142       qk_ucis_state.count -= 2;
143       return true;
144     } else {
145       qk_ucis_state.count--;
146       return false;
147     }
148   }
149
150   if (keycode == KC_ENT || keycode == KC_SPC || keycode == KC_ESC) {
151     bool symbol_found = false;
152
153     for (i = qk_ucis_state.count; i > 0; i--) {
154       register_code (KC_BSPC);
155       unregister_code (KC_BSPC);
156     }
157
158     if (keycode == KC_ESC) {
159       qk_ucis_state.in_progress = false;
160       return false;
161     }
162
163     unicode_input_start();
164     for (i = 0; ucis_symbol_table[i].symbol; i++) {
165       if (is_uni_seq (ucis_symbol_table[i].symbol)) {
166         symbol_found = true;
167         register_hex32(ucis_symbol_table[i].code);
168         break;
169       }
170     }
171     if (!symbol_found) {
172       qk_ucis_symbol_fallback();
173     }
174     unicode_input_finish();
175
176     qk_ucis_state.in_progress = false;
177     return false;
178   }
179   return true;
180 }
181 #endif