~ruther/qmk_firmware

f3ffd6ad50f0a4bf24f0a0453cc5502b4b88f390 — epaew 6 years ago 0031e46
Keymap: Refactor edvorakjp user library (#4480)

* Refactor edvorakjp user library

* add tap dance support

* update keymaps

* edvorakjp: add SWAP_SCLN option

* fix behavior of SWAP_SCLN
M keyboards/helix/rev2/keymaps/edvorakjp/config.h => keyboards/helix/rev2/keymaps/edvorakjp/config.h +1 -0
@@ 4,6 4,7 @@
#undef TAPPING_FORCE_HOLD
#undef TAPPING_TERM
#define TAPPING_TERM 120
#define SWAP_SCLN

// If you need more program area, try select and reduce rgblight modes to use.


M keyboards/helix/rev2/keymaps/edvorakjp/keymap.c => keyboards/helix/rev2/keymaps/edvorakjp/keymap.c +1 -1
@@ 11,7 11,7 @@ bool process_record_keymap(uint16_t keycode, keyrecord_t *record) {
  switch(keycode) {
    case KC_LOCK:
      if (record->event.pressed) {
        if (edvorakjp_config.enable_kc_lang) {
        if (get_enable_kc_lang()) {
          SEND_STRING( SS_LCTRL(SS_LSFT(SS_TAP(X_POWER))) );
        } else {
          SEND_STRING( SS_LGUI("l") );

M keyboards/helix/rev2/keymaps/edvorakjp/keymap_4rows.c => keyboards/helix/rev2/keymaps/edvorakjp/keymap_4rows.c +1 -1
@@ 5,7 5,7 @@ const uint16_t PROGMEM keymaps[][MATRIX_ROWS][MATRIX_COLS] = {

  [_EDVORAK] = LAYOUT_kc(
  //,----+----+----+----+----+----.              ,----+----+----+----+----+----.
     LBRC,RBRC,COMM,DOT , P  , Y  ,                F  , G  , R  , W  , Q  ,BSLS,
     LBRC,RBRC,COMM,DOT , Y  , P  ,                F  , G  , R  , W  , Q  ,BSLS,
  //|----+----+----+----+----+----|              |----+----+----+----+----+----|
     EQL , A  , O  , E  , I  , U  ,                D  , T  , N  , S  , M  ,MINS,
  //|----+----+----+----+----+----|              |----+----+----+----+----+----|

M keyboards/helix/rev2/keymaps/edvorakjp/keymap_5rows.c => keyboards/helix/rev2/keymaps/edvorakjp/keymap_5rows.c +1 -1
@@ 7,7 7,7 @@ const uint16_t PROGMEM keymaps[][MATRIX_ROWS][MATRIX_COLS] = {
  //,----+----+----+----+----+----.              ,----+----+----+----+----+----.
     GRV ,EXLM, AT ,HASH,DLR ,PERC,               CIRC,AMPR,ASTR,LPRN,RPRN,BSPC,
  //|----+----+----+----+----+----|              |----+----+----+----+----+----|
     LBRC,RBRC,COMM,DOT , P  , Y  ,                F  , G  , R  , W  , Q  ,BSLS,
     LBRC,RBRC,COMM,DOT , Y  , P  ,                F  , G  , R  , W  , Q  ,BSLS,
  //|----+----+----+----+----+----|              |----+----+----+----+----+----|
     EQL , A  , O  , E  , I  , U  ,                D  , T  , N  , S  , M  ,MINS,
  //|----+----+----+----+----+----+----.    ,----|----+----+----+----+----+----|

M keyboards/helix/rev2/keymaps/edvorakjp/oled.c => keyboards/helix/rev2/keymaps/edvorakjp/oled.c +5 -10
@@ 23,15 23,10 @@ void render_status(struct CharacterMatrix *matrix) {

  // Render to mode icon
  static char logo[][2][3] = {{{0x95,0x96,0},{0xb5,0xb6,0}},{{0x97,0x98,0},{0xb7,0xb8,0}}};
  if (edvorakjp_config.enable_kc_lang) {
    matrix_write(matrix, logo[0][0]);
    matrix_write_P(matrix, PSTR("\n"));
    matrix_write(matrix, logo[0][1]);
  } else {
    matrix_write(matrix, logo[1][0]);
    matrix_write_P(matrix, PSTR("\n"));
    matrix_write(matrix, logo[1][1]);
  }
  int mode_number = get_enable_kc_lang() ? 0 : 1;
  matrix_write(matrix, logo[mode_number][0]);
  matrix_write(matrix, "\n");
  matrix_write(matrix, logo[mode_number][1]);

  // Define layers here, Have not worked out how to have text displayed for each layer. Copy down the number you see and add a case for it below
  char buf[40];


@@ 63,7 58,7 @@ void render_status(struct CharacterMatrix *matrix) {
  // Host Keyboard LED Status
  char led[40];
  snprintf(led, sizeof(led), "\n%s %s %s %s",
      edvorakjp_config.enable_jp_extra_layer && japanese_mode ? "EXT" : "   ",
      get_enable_jp_extra_layer() && get_japanese_mode() ? "EXT" : "   ",
      (host_keyboard_leds() & (1<<USB_LED_NUM_LOCK)) ? "NMLK" : "    ",
      (host_keyboard_leds() & (1<<USB_LED_CAPS_LOCK)) ? "CAPS" : "    ",
      (host_keyboard_leds() & (1<<USB_LED_SCROLL_LOCK)) ? "SCLK" : "    ");

M keyboards/iris/keymaps/edvorakjp/config.h => keyboards/iris/keymaps/edvorakjp/config.h +1 -0
@@ 26,5 26,6 @@
#define MOUSEKEY_WHEEL_TIME_TO_MAX 20

#define TAPPING_TERM 120
#define SWAP_SCLN

#endif

M keyboards/iris/keymaps/edvorakjp/keymap.c => keyboards/iris/keymaps/edvorakjp/keymap.c +2 -5
@@ 1,7 1,4 @@
#include QMK_KEYBOARD_H
#include "action_layer.h"
#include "eeconfig.h"

#include "edvorakjp.h"

enum custom_keycodes {


@@ 32,7 29,7 @@ const uint16_t PROGMEM keymaps[][MATRIX_ROWS][MATRIX_COLS] = {
  //,----+----+----+----+----+----.              ,----+----+----+----+----+----.
     GRV ,EXLM, AT ,HASH,DLR ,PERC,               CIRC,AMPR,ASTR,LPRN,RPRN,BSPC,
  //|----+----+----+----+----+----|              |----+----+----+----+----+----|
     LBRC,RBRC,COMM,DOT , P  , Y  ,                F  , G  , R  , W  , Q  ,BSLS,
     LBRC,RBRC,COMM,DOT , Y  , P  ,                F  , G  , R  , W  , Q  ,BSLS,
  //|----+----+----+----+----+----|              |----+----+----+----+----+----|
     EQL , A  , O  , E  , I  , U  ,                D  , T  , N  , S  , M  ,MINS,
  //|----+----+----+----+----+----+----.    ,----|----+----+----+----+----+----|


@@ 134,7 131,7 @@ bool process_record_keymap(uint16_t keycode, keyrecord_t *record) {
  switch(keycode) {
    case KC_LOCK:
      if (record->event.pressed) {
        if (edvorakjp_config.enable_kc_lang) {
        if (get_enable_kc_lang()) {
          SEND_STRING( SS_LCTRL(SS_LSFT(SS_TAP(X_POWER))) );
        } else {
          SEND_STRING( SS_LGUI("l") );

M users/edvorakjp/edvorakjp.c => users/edvorakjp/edvorakjp.c +7 -215
@@ 1,47 1,12 @@
#include "eeprom.h"
#include "edvorakjp.h"

bool japanese_mode;
uint16_t time_on_pressed;

edvorakjp_config_t edvorakjp_config;

uint8_t eeconfig_read_edvorakjp(void) {
  return eeprom_read_byte(EECONFIG_EDVORAK);
}

void eeconfig_update_edvorakjp(uint8_t val) {
  eeprom_update_byte(EECONFIG_EDVORAK, val);
}

void dvorakj_layer_off(void) {
  layer_off(_EDVORAKJ1);
  layer_off(_EDVORAKJ2);
}

void update_japanese_mode(bool new_state) {
  japanese_mode = new_state;
  if (japanese_mode) {
    if (edvorakjp_config.enable_kc_lang) {
      SEND_STRING(SS_TAP(X_LANG1));
    } else {
      SEND_STRING(SS_LALT("`"));
    }
  } else {
    dvorakj_layer_off();
    if (edvorakjp_config.enable_kc_lang) {
      SEND_STRING(SS_TAP(X_LANG2));
    } else {
      SEND_STRING(SS_LALT("`"));
    }
  }
}

void matrix_init_user(void) {
  japanese_mode = false;
  time_on_pressed = 0;
  edvorakjp_config.raw = eeconfig_read_edvorakjp();

  edvorakjp_status_init();
  matrix_init_keymap();
}



@@ 58,189 23,16 @@ uint32_t layer_state_set_keymap(uint32_t state) {
  return state;
}

/*
 * Each process_record_* methods defined here are
 * return false if handle edvorak_keycodes, or return true others.
 */
__attribute__ ((weak))
bool process_record_keymap(uint16_t keycode, keyrecord_t *record) {
  return true;
}

bool process_record_edvorakjp_ext(uint16_t keycode, keyrecord_t *record) {
  if (!(edvorakjp_config.enable_jp_extra_layer &&\
        (default_layer_state == 1UL<<_EDVORAK) &&\
        japanese_mode &&\
        record->event.pressed)) {
    return true;
  }

  // consonant keys
  // layer_on(J1) or layer_on(J2) are defined based on key positions.
  switch (keycode) {
    // right hand's left side w/o N
    case KC_F:
    case KC_G:
    case KC_R:
    case KC_D:
    case KC_T:
    case KC_B:
    case KC_H:
    case KC_J:
      layer_on(_EDVORAKJ1);
      register_code(keycode);
      unregister_code(keycode);
      return false;

    // N: toggle layer
    case KC_N:
      biton32(layer_state) == _EDVORAK ? layer_on(_EDVORAKJ1) : dvorakj_layer_off();
      register_code(keycode);
      unregister_code(keycode);
      return false;

    // left hand and right hand's right side
    case KC_X:
    case KC_C:
    case KC_V:
    case KC_Z:
    case KC_P:
    case KC_Y:
    case KC_W:
    case KC_Q:
    case KC_S:
    case KC_M:
    case KC_K:
    case KC_L:
      layer_on(_EDVORAKJ2);
      register_code(keycode);
      unregister_code(keycode);
      return false;
  }

  // vowel keys, symbol keys and modifier keys
  dvorakj_layer_off();
  switch (keycode) {
    // combination vowel keys
    case KC_AI:
      SEND_STRING("ai");
      return false;
    case KC_OU:
      SEND_STRING("ou");
      return false;
    case KC_EI:
      SEND_STRING("ei");
      return false;
    case KC_ANN:
      SEND_STRING("ann");
      return false;
    case KC_ONN:
      SEND_STRING("onn");
      return false;
    case KC_ENN:
      SEND_STRING("enn");
      return false;
    case KC_INN:
      SEND_STRING("inn");
      return false;
    case KC_UNN:
      SEND_STRING("unn");
      return false;

    // AOEIU and other (symbol, modifier) keys
    default:
      return true;
  }
}

bool process_record_edvorakjp_config(uint16_t keycode, keyrecord_t *record) {
  switch (keycode) {
    case KC_MAC:
      edvorakjp_config.enable_kc_lang = true;
      eeconfig_update_edvorakjp(edvorakjp_config.raw);
      return false;
    case KC_WIN:
      edvorakjp_config.enable_kc_lang = false;
      eeconfig_update_edvorakjp(edvorakjp_config.raw);
      return false;
    case KC_EXTON:
      edvorakjp_config.enable_jp_extra_layer = true;
      eeconfig_update_edvorakjp(edvorakjp_config.raw);
      return false;
    case KC_EXTOFF:
      edvorakjp_config.enable_jp_extra_layer = false;
      eeconfig_update_edvorakjp(edvorakjp_config.raw);
      return false;
  }
  return true;
}

bool process_record_layer(uint16_t keycode, keyrecord_t *record) {
  switch (keycode) {
    case EDVORAK:
      if (record->event.pressed) {
        set_single_persistent_default_layer(_EDVORAK);
      }
      return false;
    case QWERTY:
      if (record->event.pressed) {
        dvorakj_layer_off();
        set_single_persistent_default_layer(_QWERTY);
      }
      return false;
    case LOWER:
      if (record->event.pressed) {
        layer_on(_LOWER);
        time_on_pressed = record->event.time;
      } else {
        layer_off(_LOWER);

        if (TIMER_DIFF_16(record->event.time, time_on_pressed) < TAPPING_TERM) {
          update_japanese_mode(false);
        }
        time_on_pressed = 0;
      }
      return false;
    case RAISE:
      if (record->event.pressed) {
        layer_on(_RAISE);
        time_on_pressed = record->event.time;
      } else {
        layer_off(_RAISE);

        if (TIMER_DIFF_16(record->event.time, time_on_pressed) < TAPPING_TERM) {
          update_japanese_mode(true);
        }
        time_on_pressed = 0;
      }
      return false;
    default:
      return true;
  }
}

bool process_record_ime(uint16_t keycode, keyrecord_t *record) {
  switch (keycode) {
    case KC_JPN:
      if (record->event.pressed) {
        update_japanese_mode(true);
      }
      return false;
    case KC_ENG:
      if (record->event.pressed) {
        update_japanese_mode(false);
      }
      return false;
    default:
      return true;
  }
}

bool process_record_user(uint16_t keycode, keyrecord_t *record) {

  return process_record_keymap(keycode, record) &&\
         process_record_edvorakjp_ext(keycode, record) &&\
         process_record_edvorakjp_swap_scln(keycode, record) &&\
         process_record_edvorakjp_config(keycode, record) &&\
         process_record_layer(keycode, record) &&\
         process_record_ime(keycode, record);
}

__attribute__ ((weak))
bool process_record_keymap(uint16_t keycode, keyrecord_t *record) {
  return true;
}

M users/edvorakjp/edvorakjp.h => users/edvorakjp/edvorakjp.h +20 -17
@@ 1,5 1,5 @@
#ifndef USERSPACE
#define USERSPACE
#ifndef EDVORAKJP
#define EDVORAKJP

#include "quantum.h"
#include "action_layer.h"


@@ 8,15 8,6 @@

extern keymap_config_t keymap_config;

typedef union {
  uint8_t raw;
  struct {
    bool enable_jp_extra_layer : 1;
    bool enable_kc_lang        : 1;  // for macOS
  };
} edvorakjp_config_t;
extern edvorakjp_config_t edvorakjp_config;

enum edvorakjp_layers {
  _EDVORAK = 0,
  _EDVORAKJ1,


@@ 50,26 41,38 @@ enum edvorakjp_keycodes {
  NEW_SAFE_RANGE
};

uint8_t eeconfig_read_edvorakjp(void);
void eeconfig_update_edvorakjp(uint8_t val);
enum tap_dance_code {
  TD_LOWER = 0,
  TD_RAISE
};

// base
void dvorakj_layer_off(void);
void update_japanese_mode(bool new_state);
void matrix_init_user(void);
void matrix_init_keymap(void);
uint32_t layer_state_set_user(uint32_t state);
uint32_t layer_state_set_keymap(uint32_t state);
bool process_record_user(uint16_t keycode, keyrecord_t *record);
bool process_record_keymap(uint16_t keycode, keyrecord_t *record);

// status
void edvorakjp_status_init(void);
bool get_enable_jp_extra_layer(void);
void set_enable_jp_extra_layer(bool new_state);
bool get_enable_kc_lang(void);
void set_enable_kc_lang(bool new_state);
bool get_japanese_mode(void);
void set_japanese_mode(bool new_state);

/*
 * Each process_record_* methods defined here are
 * return false if processed, or return true if not processed.
 * You can add your original macros in process_record_keymap() in keymap.c.
 */
bool process_record_keymap(uint16_t keycode, keyrecord_t *record);
bool process_record_edvorakjp_ext(uint16_t keycode, keyrecord_t *record);
bool process_record_edvorakjp_swap_scln(uint16_t keycode, keyrecord_t *record);
bool process_record_edvorakjp_config(uint16_t keycode, keyrecord_t *record);
bool process_record_layer(uint16_t keycode, keyrecord_t *record);
bool process_record_ime(uint16_t keycode, keyrecord_t *record);
bool process_record_user(uint16_t keycode, keyrecord_t *record);

#endif
#endif // EDVORAKJP

A users/edvorakjp/edvorakjp_process_record.c => users/edvorakjp/edvorakjp_process_record.c +206 -0
@@ 0,0 1,206 @@
#include "edvorakjp.h"

#if TAP_DANCE_ENABLE != yes
static uint16_t time_on_pressed;
#endif
/*
 * Each process_record_* methods defined here are
 * return false if handle edvorak_keycodes, or return true others.
 */
bool process_record_edvorakjp_ext(uint16_t keycode, keyrecord_t *record) {
  if (!(default_layer_state == 1UL<<_EDVORAK &&
        get_enable_jp_extra_layer() && get_japanese_mode())) {
    return true;
  }

  // consonant keys
  // layer_on(J1) or layer_on(J2) are defined based on key positions.
  switch (keycode) {
    // right hand's left side w/o N
    case KC_F:
    case KC_G:
    case KC_R:
    case KC_D:
    case KC_T:
    case KC_B:
    case KC_H:
    case KC_J:
      if (record->event.pressed) {
        layer_on(_EDVORAKJ1);
      }
      return true;

    // N: toggle layer
    case KC_N:
      if (record->event.pressed) {
        biton32(layer_state) == _EDVORAK ? layer_on(_EDVORAKJ1) : dvorakj_layer_off();
      }
      return true;

    // left hand and right hand's right side
    case KC_X:
    case KC_C:
    case KC_V:
    case KC_Z:
    case KC_Y:
    case KC_P:
    case KC_W:
    case KC_Q:
    case KC_S:
    case KC_M:
    case KC_K:
    case KC_L:
      if (record->event.pressed) {
        layer_on(_EDVORAKJ2);
      }
      return true;
  }

  // vowel keys, symbol keys and modifier keys
  if (record->event.pressed) {
    dvorakj_layer_off();
  }
  switch (keycode) {
    // combination vowel keys
    case KC_AI:
      if (record->event.pressed) {
        SEND_STRING("ai");
      }
      return false;
    case KC_OU:
      if (record->event.pressed) {
        SEND_STRING("ou");
      }
      return false;
    case KC_EI:
      if (record->event.pressed) {
        SEND_STRING("ei");
      }
      return false;
    case KC_ANN:
      if (record->event.pressed) {
        SEND_STRING("ann");
      }
      return false;
    case KC_ONN:
      if (record->event.pressed) {
        SEND_STRING("onn");
      }
      return false;
    case KC_ENN:
      if (record->event.pressed) {
        SEND_STRING("enn");
      }
      return false;
    case KC_INN:
      if (record->event.pressed) {
        SEND_STRING("inn");
      }
      return false;
    case KC_UNN:
      if (record->event.pressed) {
        SEND_STRING("unn");
      }
      return false;
  }
  // AOEIU and other (symbol, modifier) keys
  return true;
}

bool process_record_edvorakjp_swap_scln(uint16_t keycode, keyrecord_t *record) {
#ifdef SWAP_SCLN
  static const uint8_t shift_bits = MOD_BIT(KC_LSFT) | MOD_BIT(KC_RSFT);
  static uint8_t last_mods_status;
  if (keycode == KC_SCLN) {
    if (record->event.pressed) {
      last_mods_status = get_mods();

      // invert shift_bits
      if (last_mods_status & shift_bits) {
        set_mods(last_mods_status & ~shift_bits);
      } else {
        set_mods(last_mods_status | MOD_BIT(KC_LSFT));
      }
    } else {
      set_mods(last_mods_status);
      last_mods_status = 0;
    }
  }
#endif
  return true;
}

bool process_record_edvorakjp_config(uint16_t keycode, keyrecord_t *record) {
  switch (keycode) {
    case KC_MAC:
    case KC_WIN:
      if (record->event.pressed) {
        set_enable_kc_lang(keycode == KC_MAC);
      }
      return false;
    case KC_EXTON:
    case KC_EXTOFF:
      if (record->event.pressed) {
        set_enable_jp_extra_layer(keycode == KC_EXTON);
      }
      return false;
  }
  return true;
}

bool process_record_layer(uint16_t keycode, keyrecord_t *record) {
  switch (keycode) {
    case EDVORAK:
      if (record->event.pressed) {
        set_single_persistent_default_layer(_EDVORAK);
      }
      return false;
    case QWERTY:
      if (record->event.pressed) {
        dvorakj_layer_off();
        set_single_persistent_default_layer(_QWERTY);
      }
      return false;
#if TAP_DANCE_ENABLE != yes
    case LOWER:
      if (record->event.pressed) {
        layer_on(_LOWER);
        time_on_pressed = record->event.time;
      } else {
        layer_off(_LOWER);

        if (TIMER_DIFF_16(record->event.time, time_on_pressed) < TAPPING_TERM) {
          set_japanese_mode(false);
        }
        time_on_pressed = 0;
      }
      return false;
    case RAISE:
      if (record->event.pressed) {
        layer_on(_RAISE);
        time_on_pressed = record->event.time;
      } else {
        layer_off(_RAISE);

        if (TIMER_DIFF_16(record->event.time, time_on_pressed) < TAPPING_TERM) {
          set_japanese_mode(true);
        }
        time_on_pressed = 0;
      }
      return false;
#endif
  }
  return true;
}

bool process_record_ime(uint16_t keycode, keyrecord_t *record) {
  switch (keycode) {
    case KC_JPN:
    case KC_ENG:
      if (record->event.pressed) {
        set_japanese_mode(keycode == KC_JPN);
      }
      return false;
  }
  return true;
}

A users/edvorakjp/edvorakjp_status.c => users/edvorakjp/edvorakjp_status.c +75 -0
@@ 0,0 1,75 @@
#include "eeprom.h"
#include "edvorakjp.h"

typedef union {
  uint8_t raw;
  struct {
    bool enable_jp_extra_layer : 1;
    bool enable_kc_lang        : 1;  // for macOS
  };
} edvorakjp_config_t;
static edvorakjp_config_t edvorakjp_config;

typedef struct {
  bool japanese_mode;
} edvorakjp_state_t;
static edvorakjp_state_t edvorakjp_state;

/*
 * private methods
 */
uint8_t eeconfig_read_edvorakjp(void) {
  return eeprom_read_byte(EECONFIG_EDVORAK);
}

void eeconfig_update_edvorakjp(uint8_t val) {
  eeprom_update_byte(EECONFIG_EDVORAK, val);
}

/*
 * public methods
 */
void edvorakjp_status_init(void) {
  edvorakjp_state.japanese_mode = false;
  edvorakjp_config.raw = eeconfig_read_edvorakjp();
}

bool get_enable_jp_extra_layer(void) {
  return edvorakjp_config.enable_jp_extra_layer;
}

void set_enable_jp_extra_layer(bool new_state) {
  edvorakjp_config.enable_jp_extra_layer = new_state;
  eeconfig_update_edvorakjp(edvorakjp_config.raw);
}

bool get_enable_kc_lang(void) {
  return edvorakjp_config.enable_kc_lang;
}

void set_enable_kc_lang(bool new_state) {
  edvorakjp_config.enable_kc_lang = new_state;
  eeconfig_update_edvorakjp(edvorakjp_config.raw);
}

bool get_japanese_mode(void) {
  return edvorakjp_state.japanese_mode;
}

void set_japanese_mode(bool new_state) {
  edvorakjp_state.japanese_mode = new_state;
  if (edvorakjp_state.japanese_mode) {
    if (edvorakjp_config.enable_kc_lang) {
      SEND_STRING(SS_TAP(X_LANG1));
    } else {
      SEND_STRING(SS_LALT("`"));
    }
  } else {
    dvorakj_layer_off();
    if (edvorakjp_config.enable_kc_lang) {
      SEND_STRING(SS_TAP(X_LANG2));
    } else {
      SEND_STRING(SS_LALT("`"));
    }
  }
}

A users/edvorakjp/edvorakjp_tap_dance.c => users/edvorakjp/edvorakjp_tap_dance.c +71 -0
@@ 0,0 1,71 @@
#include "edvorakjp.h"
#include "process_keycode/process_tap_dance.h"

enum tap_state {
  NONE = 0,
  SINGLE_TAP = 1,
  DOUBLE_TAP = 2,
  HOLD
};

static int td_status_lower = NONE;
static int td_status_raise = NONE;

int cur_dance(qk_tap_dance_state_t *state) {
  if (state->interrupted || !state->pressed) {
    return state->count == 1 ? SINGLE_TAP : DOUBLE_TAP;
  } else {
    return HOLD;
  }
}

void td_lower_finished(qk_tap_dance_state_t *state, void *user_data) {
  td_status_lower = cur_dance(state);
  switch(td_status_lower) {
    case SINGLE_TAP:
      set_japanese_mode(false);
      register_code(KC_ESC);
      break;
    case DOUBLE_TAP:
      set_japanese_mode(false);
      break;
    case HOLD:
      break;
  }
  layer_on(_LOWER);
}

void td_lower_reset(qk_tap_dance_state_t *state, void *user_data) {
  if (td_status_lower == SINGLE_TAP) {
    unregister_code(KC_ESC);
  }
  layer_off(_LOWER);
  td_status_lower = NONE;
}

void td_raise_finished(qk_tap_dance_state_t *state, void *user_data) {
  td_status_raise = cur_dance(state);
  switch(td_status_raise) {
    case SINGLE_TAP:
    case DOUBLE_TAP:
      set_japanese_mode(true);
      break;
    case HOLD:
      break;
  }
  layer_on(_RAISE);
}

void td_raise_reset(qk_tap_dance_state_t *state, void *user_data) {
  layer_off(_RAISE);
  td_status_raise = NONE;
}

qk_tap_dance_action_t tap_dance_actions[] = {
  [TD_LOWER] = ACTION_TAP_DANCE_FN_ADVANCED_TIME(
      NULL, td_lower_finished, td_lower_reset, TAPPING_TERM * 1.5
      ),
  [TD_RAISE] = ACTION_TAP_DANCE_FN_ADVANCED_TIME(
      NULL, td_raise_finished, td_raise_reset, TAPPING_TERM * 1.5
      )
};

M users/edvorakjp/readme.md => users/edvorakjp/readme.md +8 -4
@@ 10,7 10,7 @@ This is a sample. You can swap any symbol keys and modifier keys.
  //+----+----+----+----+----+----+----+----+----+----+----+----+----+---------+
      `  , !  , @  , #  , $  , %  , ^  , &  , *  , (  , )  , [  , ]  ,  BSPC   ,
  //+----+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-+-------+
      TAB  , '  , ,  , .  , P  , Y  , F  , G  , R  , W  , Q  , /  , =  ,   \   ,
      TAB  , '  , ,  , .  , Y  , P  , F  , G  , R  , W  , Q  , /  , =  ,   \   ,
  //+------++---++---++---++---++---++---++---++---++---++---++---++---+-------+
      CAPS  , A  , O  , E  , I  , U  , D  , T  , N  , S  , M  , -  ,    ENT    ,
  //+-------+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-----------+


@@ 25,7 25,7 @@ This is a sample. You can swap any symbol keys and modifier keys.
  //+----+----+----+----+----+----+----+----+----+----+----+----+----+---------+
      `  , !  , @  , #  , $  , %  , ^  , &  , *  , (  , )  , [  , ]  ,  BSPC   ,
  //+----+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-+-------+
      TAB  , '  , ,  , .  , P  , Y  , F  , G  , R  , W  , C  , /  , =  ,
      TAB  , '  , ,  , .  , Y  , P  , F  , G  , R  , W  , C  , /  , =  ,
  //+------++---++---++---++---++---++---++---++---++---++---++---++---++
      CAPS  , A  , O  , E  , I  , U  , D  , T  , N  , S  , M  , ;  , -  , ENT  ,
  //+-------+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-+--+-+--+------+


@@ 78,11 78,15 @@ This is a sample. You can swap any symbol keys and modifier keys.
## for Programmer

- Dvorak 配列をベースに、ショートカットでよく利用される XCV は QWERTY 配列の位置を維持
- Vimユーザのために、HJKL キーを横並びで配置
- 一部にVimユーザ用のキー配置を実施
    - HJKL キーを横並びで配置
    - Shift押下時と非押下時で、";"キーの挙動を入れ替え(`config.h` 内で `#define SWAP_SCLN` の宣言が必要です)
- デフォルトレイヤーには、数字キーの代わりに記号 `!@#$%^&*()` を配置

- mainly based on Dvorak layout, but XCV is available in the same position of QWERTY layout
- HJKL is lining side by side, for Vim users
- for Vim users
    - HJKL is lining side by side
    - swap the ";" key behavior. i.e. send ":" normally and send ";" when you hold shift. (need `#define SWAP_SCLN` in your `config.h`)
- we can type `!@#$%^&*()` keys without shift keys in base layer

## License

M users/edvorakjp/rules.mk => users/edvorakjp/rules.mk +7 -1
@@ 1,1 1,7 @@
SRC += edvorakjp.c
SRC += edvorakjp.c \
       edvorakjp_process_record.c \
       edvorakjp_status.c

ifeq ($(TAP_DANCE_ENABLE), yes)
SRC += edvorakjp_tap_dance.c
endif