1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
|
/* Copyright 2021 Gun Pinyo
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include QMK_KEYBOARD_H
extern keymap_config_t keymap_config;
enum planck_layers {
LY_0000, LY_0001, LY_0010, LY_0011,
LY_0100, LY_0101, LY_0110, LY_0111,
LY_1000, LY_1001, LY_1010, LY_1011,
LY_1100, LY_1101, LY_1110, LY_1111,
LY_THAI_A, LY_THAI_B, LY_THAI_C,
LY_SANDBOX, LY_STICK,
};
enum planck_keycodes {
MIN_KC = SAFE_RANGE,
FUNC_A, FUNC_B, FUNC_C, FUNC_D,
SANDBOX, STICK,
LTHAI_A, LTHAI_B, LTHAI_C,
LSW0110, LSW1111, LSW0100,
USER_NAME, USER_EMAIL,
MAX_KC,
DYNAMIC_MACRO_RANGE,
};
#define KC_L2_0 KC_LSFT
#define KC_L3_0 KC_LCTL
#define KC_L3_1 KC_LALT
#define KC_L3_2 KC_LGUI
#define KC_R2_1 FUNC_A
#define KC_R2_2 FUNC_B
#define KC_R2_3 FUNC_C
#define KC_R2_4 FUNC_D
#define KC_R2_5 KC_RSFT
#define KC_R3_0 KC_R2_4
#define KC_R3_1 KC_SPC
#define LAYOUT_gunp( k00, k01, k02, k03, k04, k05, \
k10, k11, k12, k13, k14, k15, \
k20, k21, k22, k23, k24, k25, \
k30, k31, k32, k33, k34, k35) \
LAYOUT_ortho_4x12( \
k00,k01,k02,k03,k04,k05, LSW0110, KC_LEFT, KC_BSPC, KC_RIGHT,XXXXXXX, XXXXXXX,\
k10,k11,k12,k13,k14,k15, LSW1111, LTHAI_A, LTHAI_B, LTHAI_C, XXXXXXX, XXXXXXX,\
k20,k21,k22,k23,k24,k25, LSW0100, KC_R2_1, KC_R2_2, KC_R2_3, KC_R2_4, KC_R2_5,\
k30,k31,k32,k33,k34,k35, KC_R3_0, KC_R3_1, KC_ENT, KC_RGUI, KC_RALT, KC_RCTL)
#define UNUSED_LAYER LAYOUT_gunp(\
XXXXXXX, XXXXXXX, XXXXXXX, XXXXXXX, XXXXXXX, XXXXXXX,\
XXXXXXX, XXXXXXX, XXXXXXX, XXXXXXX, XXXXXXX, XXXXXXX,\
XXXXXXX, XXXXXXX, XXXXXXX, XXXXXXX, XXXXXXX, XXXXXXX,\
XXXXXXX, XXXXXXX, XXXXXXX, XXXXXXX, XXXXXXX, XXXXXXX),
#define F1_F12_LAYER(M) LAYOUT_gunp(\
S(M(KC_F7 )), S(M(KC_F8 )), M(KC_F7 ), M(KC_F8 ), M(KC_F9 ), S(M(KC_F9 )),\
S(M(KC_F4 )), S(M(KC_F5 )), M(KC_F4 ), M(KC_F5 ), M(KC_F6 ), S(M(KC_F6 )),\
S(M(KC_F1 )), S(M(KC_F2 )), M(KC_F1 ), M(KC_F2 ), M(KC_F3 ), S(M(KC_F3 )),\
S(M(KC_F10)), S(M(KC_F11)), M(KC_F10), M(KC_F11), M(KC_F12), S(M(KC_F12))),
#define MC(kc) LCTL(kc)
#define MA(kc) LALT(kc)
#define MG(kc) LGUI(kc)
#define MSC(kc) LSFT(LCTL(kc))
const uint16_t PROGMEM keymaps[][MATRIX_ROWS][MATRIX_COLS] = {
[LY_0000] = LAYOUT_ortho_4x12(
KC_PGUP, KC_Q, KC_W, KC_F, KC_P, KC_G, KC_J, KC_L, KC_U, KC_Y, KC_K, KC_TAB,
KC_PGDN, KC_A, KC_R, KC_S, KC_T, KC_D, KC_H, KC_N, KC_E, KC_I, KC_O, STICK,
KC_L2_0, KC_Z, KC_X, KC_C, KC_V, KC_B, KC_M,
KC_R2_1, KC_R2_2, KC_R2_3, KC_UP, KC_R2_5,
KC_L3_0, KC_L3_1, KC_L3_2, KC_ESCAPE, KC_BSPACE, KC_ENTER,
KC_R3_0, KC_R3_1, KC_PSCREEN, KC_LEFT, KC_DOWN, KC_RIGHT
),
[LY_1000] = LAYOUT_gunp(
XXXXXXX, KC_PERC, KC_QUES, KC_EXLM, KC_GRV, XXXXXXX,
XXXXXXX, KC_AT, KC_COMM, KC_DOT, KC_QUOT, KC_EQL,
KC_L2_0, KC_DLR, KC_SCLN, KC_COLN, KC_DQUO, KC_UNDS,
KC_L3_0, KC_L3_1, KC_L3_2, XXXXXXX, XXXXXXX, XXXXXXX
),
[LY_0010] = LAYOUT_gunp(
XXXXXXX, KC_AMPR, KC_PIPE, KC_TILD, KC_CIRC, XXXXXXX,
XXXXXXX, KC_LCBR, KC_LBRC, KC_LPRN, KC_LT, KC_SLSH,
KC_L2_0, KC_RCBR, KC_RBRC, KC_RPRN, KC_GT, KC_BSLS,
KC_L3_0, KC_L3_1, KC_L3_2, XXXXXXX, XXXXXXX, XXXXXXX
),
[LY_1010] = LAYOUT_gunp(
XXXXXXX, KC_HASH, KC_7, KC_8, KC_9, XXXXXXX,
XXXXXXX, KC_ASTR, KC_4, KC_5, KC_6, KC_PLUS,
KC_L2_0, KC_0, KC_1, KC_2, KC_3, KC_MINS,
KC_L3_0, KC_L3_1, KC_L3_2, XXXXXXX, XXXXXXX, XXXXXXX
),
[LY_0100] = LAYOUT_gunp(
XXXXXXX, MC(KC_G), KC_HOME, KC_UP, KC_END, MC(KC_UP),
XXXXXXX, KC_PGUP, KC_LEFT, KC_DOWN, KC_RIGHT, MC(KC_DOWN),
XXXXXXX, KC_PGDN, MC(KC_S), MC(KC_A), MC(KC_Z), MSC(KC_Z),
XXXXXXX, XXXXXXX, MC(KC_LEFT), MC(KC_F), MC(KC_RIGHT), XXXXXXX
),
[LY_1100] = LAYOUT_gunp(
XXXXXXX, MSC(KC_G), S(KC_HOME), S(KC_UP), S(KC_END), MSC(KC_UP),
XXXXXXX, S(KC_PGUP), S(KC_LEFT), S(KC_DOWN), S(KC_RIGHT), MSC(KC_DOWN),
XXXXXXX, S(KC_PGDN), MC(KC_X), MC(KC_C), MC(KC_V), MC(KC_L),
XXXXXXX, XXXXXXX, MSC(KC_LEFT), MC(KC_R), MSC(KC_RIGHT), XXXXXXX
),
[LY_0110] = F1_F12_LAYER()
[LY_1110] = F1_F12_LAYER(MG)
[LY_0001] = LAYOUT_gunp(
MC(KC_H), MC(KC_0), MSC(KC_PGUP), KC_BRIU, MSC(KC_PGDN), KC_VOLU,
MSC(KC_T), MC(KC_PLUS), MC(KC_PGUP), KC_BRID, MC(KC_PGDN), KC_VOLD,
OSM(MOD_LSFT), MC(KC_MINS), MA(KC_LEFT), MC(KC_T), MA(KC_RIGHT), KC_MUTE,
OSM(MOD_LCTL), OSM(MOD_LALT), OSM(MOD_LGUI), MC(KC_W), KC_DELETE, KC_APP
),
[LY_1101] = LAYOUT_gunp(
SANDBOX, XXXXXXX, AU_TOG, KC_LOCK, RGB_TOG, RESET,
KC_WAKE, KC_CLCK, USER_NAME, USER_EMAIL, RGB_MOD, DEBUG,
KC_SLEP, KC_NLCK, DM_REC1, DM_PLY1, XXXXXXX, EEP_RST,
KC_PWR, KC_SLCK, DM_REC2, DM_PLY2, DM_RSTP, KC_INSERT
),
[LY_1111] = LAYOUT_gunp(
KC_ACL1, KC_ACL0, KC_WH_L, KC_MS_U, KC_WH_R, KC_WH_U,
KC_ACL2, XXXXXXX, KC_MS_L, KC_MS_D, KC_MS_R, KC_WH_D,
KC_MPRV, KC_BTN5, KC_MRWD, KC_MPLY, KC_MFFD, KC_BTN3,
KC_MNXT, KC_BTN4, KC_PAUSE, KC_MSTP, KC_BTN1, KC_BTN2
),
[LY_0111] = UNUSED_LAYER
[LY_1011] = UNUSED_LAYER
[LY_1001] = UNUSED_LAYER
[LY_0101] = UNUSED_LAYER
[LY_0011] = UNUSED_LAYER
/* mapping from US QWERTY to TH Kedmanee */
[LY_THAI_A] = LAYOUT_gunp(
S(KC_G), KC_PIPE, KC_8, KC_COMM, S(KC_A), KC_QUES,
S(KC_I), KC_SCLN, KC_QUOT, KC_O, KC_I, KC_RBRC,
S(KC_P), S(KC_T), KC_M, KC_P, KC_EQL, KC_COLN,
KC_LT, S(KC_S), S(KC_R), KC_GT, KC_R, KC_A
),
[LY_THAI_B] = LAYOUT_gunp(
S(KC_N), S(KC_H), KC_J, KC_H, S(KC_U), S(KC_J),
S(KC_F), KC_C, KC_G, KC_K, KC_Y, KC_T,
KC_DOT, KC_W, KC_B, KC_U, KC_7, KC_N,
KC_1, S(KC_B), S(KC_Y), KC_E, KC_6, KC_CIRC
),
[LY_THAI_C] = LAYOUT_gunp(
KC_4, KC_BSLS, KC_MINS, S(KC_V), S(KC_E), S(KC_D),
KC_5, KC_0, KC_D, KC_V, KC_F, KC_9,
S(KC_L), S(KC_K), KC_L, KC_S, KC_LBRC, KC_X,
S(KC_O), KC_Q, KC_LCBR, S(KC_C), KC_Z, KC_SLSH
),
[LY_SANDBOX] = LAYOUT_ortho_4x12(
KC_LCBR, KC_Q, KC_W, KC_F, KC_P, KC_G, KC_J, KC_L, KC_U, KC_Y, KC_K, KC_RCBR,
KC_LBRC, KC_A, KC_R, KC_S, KC_T, KC_D, KC_H, KC_N, KC_E, KC_I, KC_O, KC_RBRC,
KC_LPRN,KC_Z,KC_X,KC_C,KC_V,KC_B,KC_M,KC_COMM,KC_DOT,KC_SCLN,KC_COLN,KC_RPRN,
KC_LT , KC_1, KC_2, KC_3, KC_4, KC_5, KC_6, KC_7, KC_8, KC_9, KC_0, KC_GT
),
[LY_STICK] = UNUSED_LAYER // as a gateway to other persistant layers
};
#define NUM_LANGS 2
#define LANG_ENG 0
#define LANG_THAI 1
uint16_t cur_layer = LY_0000;
uint16_t cur_lang = LANG_ENG;
bool is_layer_persistant = false;
void change_layer(uint16_t new_layer) {
if(cur_layer != new_layer) {
layer_off(cur_layer);
layer_on(new_layer);
cur_layer = new_layer;
}
}
void change_lang(uint16_t lang) {
while(lang != cur_lang) {
SEND_STRING(SS_LGUI(" "));
cur_lang = (cur_lang + 1) % NUM_LANGS;
}
}
void user_panic(void) {
SEND_STRING("Planck Keyboard: User Panic!");
}
void update_env_thai(void) {
if(LY_THAI_A <= cur_layer && cur_layer <= LY_THAI_C) {
change_lang(LANG_THAI);
} else {
change_lang(LANG_ENG);
}
}
uint16_t get_persistant_layer_from_keycode(uint16_t keycode) {
switch(keycode) {
case FUNC_A: return LY_1000;
case FUNC_B: return LY_1010;
case FUNC_C: return LY_0010;
case FUNC_D: return LY_0000;
case LTHAI_A: return LY_THAI_A;
case LTHAI_B: return LY_THAI_B;
case LTHAI_C: return LY_THAI_C;
case LSW0110: return LY_0110;
case LSW1111: return LY_1111;
case LSW0100: return LY_0100;
}
return 0; // this line is unreachable but be here to make the complier happy
}
bool process_record_user(uint16_t keycode, keyrecord_t *record) {
if(!is_layer_persistant && FUNC_A <= keycode && keycode <= FUNC_D) {
uint16_t mask = 1 << (3 - (keycode - FUNC_A));
uint16_t cur_layer_code = cur_layer - LY_0000;
if(!(LY_0000 <= cur_layer && cur_layer <= LY_1111))
user_panic();
if(!(cur_layer_code & mask) == record->event.pressed)
change_layer((mask ^ cur_layer_code) + LY_0000);
return false;
}
if(!(record->event.pressed))
return !(MIN_KC <= keycode && keycode <= MAX_KC);
switch(keycode) {
case USER_NAME:
SEND_STRING("Gun Pinyo");
return false;
case USER_EMAIL:
SEND_STRING("gunpinyo@gmail.com");
return false;
case SANDBOX:
case STICK:
change_layer(keycode == STICK ? LY_STICK : LY_SANDBOX);
change_lang(LANG_ENG);
is_layer_persistant = true;
return false;
case FUNC_A: case FUNC_B: case FUNC_C: case FUNC_D:
case LTHAI_A: case LTHAI_B: case LTHAI_C:
case LSW0110: case LSW1111: case LSW0100:
if(is_layer_persistant) {
change_layer(get_persistant_layer_from_keycode(keycode));
update_env_thai();
// `FUNC_D` resets the layer configuration when `is_layer_persistant`
is_layer_persistant = keycode != FUNC_D;
if(!is_layer_persistant) {
clear_keyboard();
layer_clear();
}
}
return false;
}
return true;
}
#ifdef RGB_MATRIX_ENABLE
void rgb_matrix_indicators_kb(void) {
// `42` is the index of the middle light at the bottom row (in planck light)
// it is disabled because it does not have a cover, hence irritates my eyes
rgb_matrix_set_color(42, 0, 0, 0);
}
#endif
#ifdef AUDIO_ENABLE
float tone_startup[][2] = SONG(STARTUP_SOUND);
float tone_goodbye[][2] = SONG(GOODBYE_SOUND);
float tone_ly_normal[][2] = SONG(QWERTY_SOUND);
float tone_ly_spacial[][2] = SONG(DVORAK_SOUND);
float music_scale[][2] = SONG(MUSIC_SCALE_SOUND);
#endif
void startup_user() {
#ifdef AUDIO_ENABLE
PLAY_SONG(tone_startup);
#endif
}
void shutdown_user() {
#ifdef AUDIO_ENABLE
PLAY_SONG(tone_goodbye);
stop_all_notes();
#endif
}
|