diff options
Diffstat (limited to 'quantum/process_keycode')
| -rw-r--r-- | quantum/process_keycode/process_midi.c | 28 | ||||
| -rw-r--r-- | quantum/process_keycode/process_music.c | 7 | ||||
| -rw-r--r-- | quantum/process_keycode/process_tap_dance.c | 144 | ||||
| -rw-r--r-- | quantum/process_keycode/process_tap_dance.h | 52 | ||||
| -rw-r--r-- | quantum/process_keycode/process_unicode.c | 261 | ||||
| -rw-r--r-- | quantum/process_keycode/process_unicode.h | 54 | 
6 files changed, 430 insertions, 116 deletions
diff --git a/quantum/process_keycode/process_midi.c b/quantum/process_keycode/process_midi.c index d6ab9c6264..577dad43ac 100644 --- a/quantum/process_keycode/process_midi.c +++ b/quantum/process_keycode/process_midi.c @@ -1,13 +1,15 @@  #include "process_midi.h"  bool midi_activated = false; -uint8_t starting_note = 0x0C; -int offset = 7; +uint8_t midi_starting_note = 0x0C; +int midi_offset = 7;  bool process_midi(uint16_t keycode, keyrecord_t *record) {      if (keycode == MI_ON && record->event.pressed) {        midi_activated = true; +#ifdef AUDIO_ENABLE        music_scale_user(); +#endif        return false;      } @@ -20,42 +22,42 @@ bool process_midi(uint16_t keycode, keyrecord_t *record) {      if (midi_activated) {        if (record->event.key.col == (MATRIX_COLS - 1) && record->event.key.row == (MATRIX_ROWS - 1)) {            if (record->event.pressed) { -              starting_note++; // Change key +              midi_starting_note++; // Change key                midi_send_cc(&midi_device, 0, 0x7B, 0);            }            return false;        }        if (record->event.key.col == (MATRIX_COLS - 2) && record->event.key.row == (MATRIX_ROWS - 1)) {            if (record->event.pressed) { -              starting_note--; // Change key +              midi_starting_note--; // Change key                midi_send_cc(&midi_device, 0, 0x7B, 0);            }            return false;        }        if (record->event.key.col == (MATRIX_COLS - 3) && record->event.key.row == (MATRIX_ROWS - 1) && record->event.pressed) { -          offset++; // Change scale +          midi_offset++; // Change scale            midi_send_cc(&midi_device, 0, 0x7B, 0);            return false;        }        if (record->event.key.col == (MATRIX_COLS - 4) && record->event.key.row == (MATRIX_ROWS - 1) && record->event.pressed) { -          offset--; // Change scale +          midi_offset--; // Change scale            midi_send_cc(&midi_device, 0, 0x7B, 0);            return false;        }        // basic -      // uint8_t note = (starting_note + SCALE[record->event.key.col + offset])+12*(MATRIX_ROWS - record->event.key.row); +      // uint8_t note = (midi_starting_note + SCALE[record->event.key.col + midi_offset])+12*(MATRIX_ROWS - record->event.key.row);        // advanced -      // uint8_t note = (starting_note + record->event.key.col + offset)+12*(MATRIX_ROWS - record->event.key.row); +      // uint8_t note = (midi_starting_note + record->event.key.col + midi_offset)+12*(MATRIX_ROWS - record->event.key.row);        // guitar -      uint8_t note = (starting_note + record->event.key.col + offset)+5*(MATRIX_ROWS - record->event.key.row); +      uint8_t note = (midi_starting_note + record->event.key.col + midi_offset)+5*(MATRIX_ROWS - record->event.key.row);        // violin -      // uint8_t note = (starting_note + record->event.key.col + offset)+7*(MATRIX_ROWS - record->event.key.row); +      // uint8_t note = (midi_starting_note + record->event.key.col + midi_offset)+7*(MATRIX_ROWS - record->event.key.row);        if (record->event.pressed) { -        // midi_send_noteon(&midi_device, record->event.key.row, starting_note + SCALE[record->event.key.col], 127); +        // midi_send_noteon(&midi_device, record->event.key.row, midi_starting_note + SCALE[record->event.key.col], 127);          midi_send_noteon(&midi_device, 0, note, 127);        } else { -        // midi_send_noteoff(&midi_device, record->event.key.row, starting_note + SCALE[record->event.key.col], 127); +        // midi_send_noteoff(&midi_device, record->event.key.row, midi_starting_note + SCALE[record->event.key.col], 127);          midi_send_noteoff(&midi_device, 0, note, 127);        } @@ -63,4 +65,4 @@ bool process_midi(uint16_t keycode, keyrecord_t *record) {          return false;      }    return true; -}
\ No newline at end of file +} diff --git a/quantum/process_keycode/process_music.c b/quantum/process_keycode/process_music.c index c8f3ddb900..2d52e47a72 100644 --- a/quantum/process_keycode/process_music.c +++ b/quantum/process_keycode/process_music.c @@ -6,6 +6,7 @@ int offset = 7;  // music sequencer  static bool music_sequence_recording = false; +static bool music_sequence_recorded = false;  static bool music_sequence_playing = false;  static float music_sequence[16] = {0};  static uint8_t music_sequence_count = 0; @@ -77,6 +78,7 @@ bool process_music(uint16_t keycode, keyrecord_t *record) {        if (keycode == KC_LCTL && record->event.pressed) { // Start recording          stop_all_notes();          music_sequence_recording = true; +        music_sequence_recorded = false;          music_sequence_playing = false;          music_sequence_count = 0;          return false; @@ -84,12 +86,15 @@ bool process_music(uint16_t keycode, keyrecord_t *record) {        if (keycode == KC_LALT && record->event.pressed) { // Stop recording/playing          stop_all_notes(); +        if (music_sequence_recording) { // was recording +          music_sequence_recorded = true; +        }          music_sequence_recording = false;          music_sequence_playing = false;          return false;        } -      if (keycode == KC_LGUI && record->event.pressed) { // Start playing +      if (keycode == KC_LGUI && record->event.pressed && music_sequence_recorded) { // Start playing          stop_all_notes();          music_sequence_recording = false;          music_sequence_playing = true; diff --git a/quantum/process_keycode/process_tap_dance.c b/quantum/process_keycode/process_tap_dance.c index 9b172e1b6c..6ae362c4c2 100644 --- a/quantum/process_keycode/process_tap_dance.c +++ b/quantum/process_keycode/process_tap_dance.c @@ -1,90 +1,136 @@  #include "quantum.h" +#include "action_tapping.h" -static qk_tap_dance_state_t qk_tap_dance_state; +static uint16_t last_td; +static int8_t highest_td = -1; -static void _process_tap_dance_action_pair (qk_tap_dance_state_t *state, -                                            uint16_t kc1, uint16_t kc2) { -  uint16_t kc; +void qk_tap_dance_pair_finished (qk_tap_dance_state_t *state, void *user_data) { +  qk_tap_dance_pair_t *pair = (qk_tap_dance_pair_t *)user_data; -  if (state->count == 0) -    return; - -  kc = (state->count == 1) ? kc1 : kc2; +  if (state->count == 1) { +    register_code16 (pair->kc1); +  } else if (state->count == 2) { +    register_code16 (pair->kc2); +  } +} -  register_code (kc); -  unregister_code (kc); +void qk_tap_dance_pair_reset (qk_tap_dance_state_t *state, void *user_data) { +  qk_tap_dance_pair_t *pair = (qk_tap_dance_pair_t *)user_data; -  if (state->count >= 2) { -    reset_tap_dance (state); +  if (state->count == 1) { +    unregister_code16 (pair->kc1); +  } else if (state->count == 2) { +    unregister_code16 (pair->kc2);    }  } -static void _process_tap_dance_action_fn (qk_tap_dance_state_t *state, -                                          qk_tap_dance_user_fn_t fn) +static inline void _process_tap_dance_action_fn (qk_tap_dance_state_t *state, +                                                 void *user_data, +                                                 qk_tap_dance_user_fn_t fn)  { -  fn(state); +  if (fn) { +    fn(state, user_data); +  }  } -void process_tap_dance_action (uint16_t keycode) +static inline void process_tap_dance_action_on_each_tap (qk_tap_dance_action_t *action)  { -  uint16_t idx = keycode - QK_TAP_DANCE; -  qk_tap_dance_action_t action; - -  action = tap_dance_actions[idx]; +  _process_tap_dance_action_fn (&action->state, action->user_data, action->fn.on_each_tap); +} -  switch (action.type) { -  case QK_TAP_DANCE_TYPE_PAIR: -    _process_tap_dance_action_pair (&qk_tap_dance_state, -                                    action.pair.kc1, action.pair.kc2); -    break; -  case QK_TAP_DANCE_TYPE_FN: -    _process_tap_dance_action_fn (&qk_tap_dance_state, action.fn); -    break; +static inline void process_tap_dance_action_on_dance_finished (qk_tap_dance_action_t *action) +{ +  if (action->state.finished) +    return; +  action->state.finished = true; +  _process_tap_dance_action_fn (&action->state, action->user_data, action->fn.on_dance_finished); +} -  default: -    break; -  } +static inline void process_tap_dance_action_on_reset (qk_tap_dance_action_t *action) +{ +  _process_tap_dance_action_fn (&action->state, action->user_data, action->fn.on_reset);  }  bool process_tap_dance(uint16_t keycode, keyrecord_t *record) { -  bool r = true; +  uint16_t idx = keycode - QK_TAP_DANCE; +  qk_tap_dance_action_t *action; + +  if (last_td && last_td != keycode) { +    (&tap_dance_actions[last_td - QK_TAP_DANCE])->state.interrupted = true; +  }    switch(keycode) {    case QK_TAP_DANCE ... QK_TAP_DANCE_MAX: -    if (qk_tap_dance_state.keycode && qk_tap_dance_state.keycode != keycode) { -      process_tap_dance_action (qk_tap_dance_state.keycode); -    } else { -      r = false; -    } +    if ((int16_t)idx > highest_td) +      highest_td = idx; +    action = &tap_dance_actions[idx]; +    action->state.pressed = record->event.pressed;      if (record->event.pressed) { -      qk_tap_dance_state.keycode = keycode; -      qk_tap_dance_state.timer = timer_read (); -      qk_tap_dance_state.count++; +      action->state.keycode = keycode; +      action->state.count++; +      action->state.timer = timer_read(); +      process_tap_dance_action_on_each_tap (action); + +      if (last_td && last_td != keycode) { +        qk_tap_dance_action_t *paction = &tap_dance_actions[last_td - QK_TAP_DANCE]; +        paction->state.interrupted = true; +        process_tap_dance_action_on_dance_finished (paction); +        reset_tap_dance (&paction->state); +      } + +      last_td = keycode;      } +      break;    default: -    if (qk_tap_dance_state.keycode) { -      process_tap_dance_action (qk_tap_dance_state.keycode); - -      reset_tap_dance (&qk_tap_dance_state); +    if (!record->event.pressed) +      return true; + +    if (highest_td == -1) +      return true; + +    for (int i = 0; i <= highest_td; i++) { +      action = &tap_dance_actions[i]; +      if (action->state.count == 0) +        continue; +      action->state.interrupted = true; +      process_tap_dance_action_on_dance_finished (action); +      reset_tap_dance (&action->state);      }      break;    } -  return r; +  return true;  }  void matrix_scan_tap_dance () { -  if (qk_tap_dance_state.keycode && timer_elapsed (qk_tap_dance_state.timer) > TAPPING_TERM) { -    process_tap_dance_action (qk_tap_dance_state.keycode); +  if (highest_td == -1) +    return; -    reset_tap_dance (&qk_tap_dance_state); +  for (int i = 0; i <= highest_td; i++) { +    qk_tap_dance_action_t *action = &tap_dance_actions[i]; + +    if (action->state.count && timer_elapsed (action->state.timer) > TAPPING_TERM) { +      process_tap_dance_action_on_dance_finished (action); +      reset_tap_dance (&action->state); +    }    }  }  void reset_tap_dance (qk_tap_dance_state_t *state) { -  state->keycode = 0; +  qk_tap_dance_action_t *action; + +  if (state->pressed) +    return; + +  action = &tap_dance_actions[state->keycode - QK_TAP_DANCE]; + +  process_tap_dance_action_on_reset (action); +    state->count = 0; +  state->interrupted = false; +  state->finished = false; +  last_td = 0;  } diff --git a/quantum/process_keycode/process_tap_dance.h b/quantum/process_keycode/process_tap_dance.h index b9d7c7fcf4..f753cbba66 100644 --- a/quantum/process_keycode/process_tap_dance.h +++ b/quantum/process_keycode/process_tap_dance.h @@ -11,41 +11,48 @@ typedef struct    uint8_t count;    uint16_t keycode;    uint16_t timer; +  bool interrupted; +  bool pressed; +  bool finished;  } qk_tap_dance_state_t;  #define TD(n) (QK_TAP_DANCE + n) -typedef enum -{ -  QK_TAP_DANCE_TYPE_PAIR, -  QK_TAP_DANCE_TYPE_FN, -} qk_tap_dance_type_t; - -typedef void (*qk_tap_dance_user_fn_t) (qk_tap_dance_state_t *state); +typedef void (*qk_tap_dance_user_fn_t) (qk_tap_dance_state_t *state, void *user_data);  typedef struct  { -  qk_tap_dance_type_t type; -  union { -    struct { -      uint16_t kc1; -      uint16_t kc2; -    } pair; -    qk_tap_dance_user_fn_t fn; -  }; +  struct { +    qk_tap_dance_user_fn_t on_each_tap; +    qk_tap_dance_user_fn_t on_dance_finished; +    qk_tap_dance_user_fn_t on_reset; +  } fn; +  qk_tap_dance_state_t state; +  void *user_data;  } qk_tap_dance_action_t; +typedef struct +{ +  uint16_t kc1; +  uint16_t kc2; +} qk_tap_dance_pair_t; +  #define ACTION_TAP_DANCE_DOUBLE(kc1, kc2) { \ -    .type = QK_TAP_DANCE_TYPE_PAIR,         \ -    .pair = { kc1, kc2 }                    \ +    .fn = { NULL, qk_tap_dance_pair_finished, qk_tap_dance_pair_reset }, \ +    .user_data = (void *)&((qk_tap_dance_pair_t) { kc1, kc2 }),  \ +  } + +#define ACTION_TAP_DANCE_FN(user_fn) {  \ +    .fn = { NULL, user_fn, NULL }, \ +    .user_data = NULL, \    } -#define ACTION_TAP_DANCE_FN(user_fn) { \ -    .type = QK_TAP_DANCE_TYPE_FN, \ -    .fn = user_fn                 \ +#define ACTION_TAP_DANCE_FN_ADVANCED(user_fn_on_each_tap, user_fn_on_dance_finished, user_fn_on_dance_reset) { \ +    .fn = { user_fn_on_each_tap, user_fn_on_dance_finished, user_fn_on_dance_reset }, \ +    .user_data = NULL, \    } -extern const qk_tap_dance_action_t tap_dance_actions[]; +extern qk_tap_dance_action_t tap_dance_actions[];  /* To be used internally */ @@ -53,6 +60,9 @@ bool process_tap_dance(uint16_t keycode, keyrecord_t *record);  void matrix_scan_tap_dance (void);  void reset_tap_dance (qk_tap_dance_state_t *state); +void qk_tap_dance_pair_finished (qk_tap_dance_state_t *state, void *user_data); +void qk_tap_dance_pair_reset (qk_tap_dance_state_t *state, void *user_data); +  #else  #define TD(n) KC_NO diff --git a/quantum/process_keycode/process_unicode.c b/quantum/process_keycode/process_unicode.c index ad5d7f86b7..cd3a610b4d 100644 --- a/quantum/process_keycode/process_unicode.c +++ b/quantum/process_keycode/process_unicode.c @@ -2,6 +2,7 @@  static uint8_t input_mode; +__attribute__((weak))  uint16_t hex_to_keycode(uint8_t hex)  {    if (hex == 0x0) { @@ -13,45 +14,251 @@ uint16_t hex_to_keycode(uint8_t hex)    }  } -void set_unicode_mode(uint8_t os_target) +void set_unicode_input_mode(uint8_t os_target)  {    input_mode = os_target;  } +uint8_t get_unicode_input_mode(void) { +  return input_mode; +} + +__attribute__((weak)) +void unicode_input_start (void) { +  switch(input_mode) { +  case UC_OSX: +    register_code(KC_LALT); +    break; +  case UC_LNX: +    register_code(KC_LCTL); +    register_code(KC_LSFT); +    register_code(KC_U); +    unregister_code(KC_U); +    unregister_code(KC_LSFT); +    unregister_code(KC_LCTL); +    break; +  case UC_WIN: +    register_code(KC_LALT); +    register_code(KC_PPLS); +    unregister_code(KC_PPLS); +    break; +  case UC_WINC: +    register_code(KC_RALT); +    unregister_code(KC_RALT); +    register_code(KC_U); +    unregister_code(KC_U); +  } +  wait_ms(UNICODE_TYPE_DELAY); +} + +__attribute__((weak)) +void unicode_input_finish (void) { +  switch(input_mode) { +  case UC_OSX: +  case UC_WIN: +    unregister_code(KC_LALT); +    break; +  case UC_LNX: +    register_code(KC_SPC); +    unregister_code(KC_SPC); +    break; +  } +} + +void register_hex(uint16_t hex) { +  for(int i = 3; i >= 0; i--) { +    uint8_t digit = ((hex >> (i*4)) & 0xF); +    register_code(hex_to_keycode(digit)); +    unregister_code(hex_to_keycode(digit)); +  } +} +  bool process_unicode(uint16_t keycode, keyrecord_t *record) {    if (keycode > QK_UNICODE && record->event.pressed) {      uint16_t unicode = keycode & 0x7FFF; -    switch(input_mode) { -      case UC_OSX: -        register_code(KC_LALT); -        break; -      case UC_LNX: -        register_code(KC_LCTL); -        register_code(KC_LSFT); -        register_code(KC_U); -        unregister_code(KC_U); -        break; -      case UC_WIN: -        register_code(KC_LALT); -        register_code(KC_PPLS); -        unregister_code(KC_PPLS); -        break; +    unicode_input_start(); +    register_hex(unicode); +    unicode_input_finish(); +  } +  return true; +} + +#ifdef UNICODEMAP_ENABLE +__attribute__((weak)) +const uint32_t PROGMEM unicode_map[] = { +}; + +void register_hex32(uint32_t hex) { +  uint8_t onzerostart = 1; +  for(int i = 7; i >= 0; i--) { +    if (i <= 3) { +      onzerostart = 0;      } -    for(int i = 3; i >= 0; i--) { -        uint8_t digit = ((unicode >> (i*4)) & 0xF); +    uint8_t digit = ((hex >> (i*4)) & 0xF); +    if (digit == 0) { +      if (onzerostart == 0) {          register_code(hex_to_keycode(digit));          unregister_code(hex_to_keycode(digit)); +      } +    } else { +      register_code(hex_to_keycode(digit)); +      unregister_code(hex_to_keycode(digit)); +      onzerostart = 0;      } -    switch(input_mode) { -      case UC_OSX: -      case UC_WIN: -        unregister_code(KC_LALT); -        break; -      case UC_LNX: -        unregister_code(KC_LCTL); -        unregister_code(KC_LSFT); +  } +} + +__attribute__((weak)) +void unicode_map_input_error() {} + +bool process_unicode_map(uint16_t keycode, keyrecord_t *record) { +  if ((keycode & QK_UNICODE_MAP) == QK_UNICODE_MAP && record->event.pressed) { +    const uint32_t* map = unicode_map; +    uint16_t index = keycode & 0x7FF; +    uint32_t code = pgm_read_dword_far(&map[index]); +    if ((code > 0xFFFF && input_mode == UC_OSX) || (code > 0xFFFFF && input_mode == UC_LNX)) { +      // when character is out of range supported by the OS +      unicode_map_input_error(); +    } else { +      unicode_input_start(); +      register_hex32(code); +      unicode_input_finish(); +    } +  } +  return true; +} +#endif + +#ifdef UCIS_ENABLE +qk_ucis_state_t qk_ucis_state; + +void qk_ucis_start(void) { +  qk_ucis_state.count = 0; +  qk_ucis_state.in_progress = true; + +  qk_ucis_start_user(); +} + +__attribute__((weak)) +void qk_ucis_start_user(void) { +  unicode_input_start(); +  register_hex(0x2328); +  unicode_input_finish(); +} + +static bool is_uni_seq(char *seq) { +  uint8_t i; + +  for (i = 0; seq[i]; i++) { +    uint16_t code; +    if (('1' <= seq[i]) && (seq[i] <= '0')) +      code = seq[i] - '1' + KC_1; +    else +      code = seq[i] - 'a' + KC_A; + +    if (i > qk_ucis_state.count || qk_ucis_state.codes[i] != code) +      return false; +  } + +  return (qk_ucis_state.codes[i] == KC_ENT || +          qk_ucis_state.codes[i] == KC_SPC); +} + +__attribute__((weak)) +void qk_ucis_symbol_fallback (void) { +  for (uint8_t i = 0; i < qk_ucis_state.count - 1; i++) { +    uint8_t code = qk_ucis_state.codes[i]; +    register_code(code); +    unregister_code(code); +    wait_ms(UNICODE_TYPE_DELAY); +  } +} + +void register_ucis(const char *hex) { +  for(int i = 0; hex[i]; i++) { +    uint8_t kc = 0; +    char c = hex[i]; + +    switch (c) { +    case '0': +      kc = KC_0; +      break; +    case '1' ... '9': +      kc = c - '1' + KC_1; +      break; +    case 'a' ... 'f': +      kc = c - 'a' + KC_A; +      break; +    case 'A' ... 'F': +      kc = c - 'A' + KC_A; +      break; +    } + +    if (kc) { +      register_code (kc); +      unregister_code (kc); +      wait_ms (UNICODE_TYPE_DELAY); +    } +  } +} + +bool process_ucis (uint16_t keycode, keyrecord_t *record) { +  uint8_t i; + +  if (!qk_ucis_state.in_progress) +    return true; + +  if (qk_ucis_state.count >= UCIS_MAX_SYMBOL_LENGTH && +      !(keycode == KC_BSPC || keycode == KC_ESC || keycode == KC_SPC || keycode == KC_ENT)) { +    return false; +  } + +  if (!record->event.pressed) +    return true; + +  qk_ucis_state.codes[qk_ucis_state.count] = keycode; +  qk_ucis_state.count++; + +  if (keycode == KC_BSPC) { +    if (qk_ucis_state.count >= 2) { +      qk_ucis_state.count -= 2; +      return true; +    } else { +      qk_ucis_state.count--; +      return false; +    } +  } + +  if (keycode == KC_ENT || keycode == KC_SPC || keycode == KC_ESC) { +    bool symbol_found = false; + +    for (i = qk_ucis_state.count; i > 0; i--) { +      register_code (KC_BSPC); +      unregister_code (KC_BSPC); +      wait_ms(UNICODE_TYPE_DELAY); +    } + +    if (keycode == KC_ESC) { +      qk_ucis_state.in_progress = false; +      return false; +    } + +    unicode_input_start(); +    for (i = 0; ucis_symbol_table[i].symbol; i++) { +      if (is_uni_seq (ucis_symbol_table[i].symbol)) { +        symbol_found = true; +        register_ucis(ucis_symbol_table[i].code + 2);          break; +      } +    } +    if (!symbol_found) { +      qk_ucis_symbol_fallback();      } +    unicode_input_finish(); + +    qk_ucis_state.in_progress = false; +    return false;    }    return true; -}
\ No newline at end of file +} +#endif diff --git a/quantum/process_keycode/process_unicode.h b/quantum/process_keycode/process_unicode.h index ca17f8f669..065eeb5f6a 100644 --- a/quantum/process_keycode/process_unicode.h +++ b/quantum/process_keycode/process_unicode.h @@ -3,15 +3,59 @@  #include "quantum.h" -#define UC_OSX 0 -#define UC_LNX 1 -#define UC_WIN 2 -#define UC_BSD 3 +#define UC_OSX 0  // Mac OS X +#define UC_LNX 1  // Linux +#define UC_WIN 2  // Windows 'HexNumpad' +#define UC_BSD 3  // BSD (not implemented) +#define UC_WINC 4 // WinCompose https://github.com/samhocevar/wincompose + +#ifndef UNICODE_TYPE_DELAY +#define UNICODE_TYPE_DELAY 10 +#endif  void set_unicode_input_mode(uint8_t os_target); +uint8_t get_unicode_input_mode(void); +void unicode_input_start(void); +void unicode_input_finish(void); +void register_hex(uint16_t hex);  bool process_unicode(uint16_t keycode, keyrecord_t *record); +#ifdef UNICODEMAP_ENABLE +bool process_unicode_map(uint16_t keycode, keyrecord_t *record); +#endif + +#ifdef UCIS_ENABLE +#ifndef UCIS_MAX_SYMBOL_LENGTH +#define UCIS_MAX_SYMBOL_LENGTH 32 +#endif + +typedef struct { +  char *symbol; +  char *code; +} qk_ucis_symbol_t; + +typedef struct { +  uint8_t count; +  uint16_t codes[UCIS_MAX_SYMBOL_LENGTH]; +  bool in_progress:1; +} qk_ucis_state_t; + +extern qk_ucis_state_t qk_ucis_state; + +#define UCIS_TABLE(...) {__VA_ARGS__, {NULL, NULL}} +#define UCIS_SYM(name, code) {name, #code} + +extern const qk_ucis_symbol_t ucis_symbol_table[]; + +void qk_ucis_start(void); +void qk_ucis_start_user(void); +void qk_ucis_symbol_fallback (void); +void register_ucis(const char *hex); +bool process_ucis (uint16_t keycode, keyrecord_t *record); + +#endif +  #define UC_BSPC	UC(0x0008)  #define UC_SPC	UC(0x0020) @@ -119,4 +163,4 @@ bool process_unicode(uint16_t keycode, keyrecord_t *record);  #define UC_TILD	UC(0x007E)  #define UC_DEL	UC(0x007F) -#endif
\ No newline at end of file +#endif  | 
