logo

qmk_firmware

custom branch of QMK firmware git clone https://anongit.hacktivis.me/git/qmk_firmware.git

test_alt_repeat_key.cpp (18170B)


  1. // Copyright 2023 Google LLC
  2. //
  3. // This program is free software: you can redistribute it and/or modify
  4. // it under the terms of the GNU General Public License as published by
  5. // the Free Software Foundation, either version 2 of the License, or
  6. // (at your option) any later version.
  7. //
  8. // This program is distributed in the hope that it will be useful,
  9. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. // GNU General Public License for more details.
  12. //
  13. // You should have received a copy of the GNU General Public License
  14. // along with this program. If not, see <http://www.gnu.org/licenses/>.
  15. #include <functional>
  16. #include "keyboard_report_util.hpp"
  17. #include "keycode.h"
  18. #include "test_common.hpp"
  19. #include "test_fixture.hpp"
  20. #include "test_keymap_key.hpp"
  21. using ::testing::AnyNumber;
  22. using ::testing::InSequence;
  23. namespace {
  24. bool process_record_user_default(uint16_t keycode, keyrecord_t* record) {
  25. return true;
  26. }
  27. bool remember_last_key_user_default(uint16_t keycode, keyrecord_t* record, uint8_t* remembered_mods) {
  28. return true;
  29. }
  30. uint16_t get_alt_repeat_key_keycode_user_default(uint16_t keycode, uint8_t mods) {
  31. return KC_TRNS;
  32. }
  33. // Indirections so that process_record_user() can be replaced with other
  34. // functions in the test cases below.
  35. std::function<bool(uint16_t, keyrecord_t*)> process_record_user_fun = process_record_user_default;
  36. std::function<bool(uint16_t, keyrecord_t*, uint8_t*)> remember_last_key_user_fun = remember_last_key_user_default;
  37. std::function<uint16_t(uint16_t, uint8_t)> get_alt_repeat_key_keycode_user_fun = get_alt_repeat_key_keycode_user_default;
  38. extern "C" bool process_record_user(uint16_t keycode, keyrecord_t* record) {
  39. return process_record_user_fun(keycode, record);
  40. }
  41. extern "C" bool remember_last_key_user(uint16_t keycode, keyrecord_t* record, uint8_t* remembered_mods) {
  42. return remember_last_key_user_fun(keycode, record, remembered_mods);
  43. }
  44. extern "C" uint16_t get_alt_repeat_key_keycode_user(uint16_t keycode, uint8_t mods) {
  45. return get_alt_repeat_key_keycode_user_fun(keycode, mods);
  46. }
  47. class AltRepeatKey : public TestFixture {
  48. public:
  49. bool process_record_user_was_called_;
  50. void SetUp() override {
  51. process_record_user_fun = process_record_user_default;
  52. remember_last_key_user_fun = remember_last_key_user_default;
  53. get_alt_repeat_key_keycode_user_fun = get_alt_repeat_key_keycode_user_default;
  54. }
  55. void ExpectProcessRecordUserCalledWith(bool expected_press, uint16_t expected_keycode, int8_t expected_repeat_key_count) {
  56. process_record_user_was_called_ = false;
  57. process_record_user_fun = [=](uint16_t keycode, keyrecord_t* record) {
  58. EXPECT_EQ(record->event.pressed, expected_press);
  59. EXPECT_KEYCODE_EQ(keycode, expected_keycode);
  60. EXPECT_EQ(get_repeat_key_count(), expected_repeat_key_count);
  61. // Tests below use this to verify process_record_user() was called.
  62. process_record_user_was_called_ = true;
  63. return true;
  64. };
  65. }
  66. // Expects that the characters of `s` are sent.
  67. // NOTE: This implementation is limited to chars a-z, A-Z.
  68. void ExpectString(TestDriver& driver, const std::string& s) {
  69. InSequence seq;
  70. for (int c : s) {
  71. switch (c) {
  72. case 'a' ... 'z': { // Lowercase letter.
  73. uint16_t keycode = c - ('a' - KC_A);
  74. EXPECT_REPORT(driver, (keycode));
  75. } break;
  76. case 'A' ... 'Z': { // Capital letter = KC_LSFT + letter key.
  77. uint16_t keycode = c - ('A' - KC_A);
  78. EXPECT_REPORT(driver, (KC_LSFT, keycode));
  79. } break;
  80. }
  81. }
  82. }
  83. };
  84. TEST_F(AltRepeatKey, AlternateBasic) {
  85. TestDriver driver;
  86. KeymapKey key_bspc(0, 0, 0, KC_BSPC);
  87. KeymapKey key_pgdn(0, 1, 0, KC_PGDN);
  88. KeymapKey key_pgup(0, 2, 0, KC_PGUP);
  89. KeymapKey key_repeat(0, 4, 0, QK_REP);
  90. KeymapKey key_alt_repeat(0, 5, 0, QK_AREP);
  91. set_keymap({key_bspc, key_pgdn, key_pgup, key_repeat, key_alt_repeat});
  92. // Allow any number of empty reports.
  93. EXPECT_EMPTY_REPORT(driver).Times(AnyNumber());
  94. {
  95. InSequence seq;
  96. EXPECT_REPORT(driver, (KC_BSPC));
  97. EXPECT_REPORT(driver, (KC_DEL));
  98. EXPECT_REPORT(driver, (KC_DEL));
  99. EXPECT_REPORT(driver, (KC_BSPC));
  100. EXPECT_REPORT(driver, (KC_DEL));
  101. EXPECT_REPORT(driver, (KC_PGDN));
  102. EXPECT_REPORT(driver, (KC_PGUP));
  103. EXPECT_REPORT(driver, (KC_PGUP));
  104. EXPECT_REPORT(driver, (KC_PGDN));
  105. }
  106. tap_key(key_bspc);
  107. for (int n = 1; n <= 2; ++n) { // Tap the Alternate Repeat Key twice.
  108. ExpectProcessRecordUserCalledWith(true, KC_DEL, -n);
  109. key_alt_repeat.press(); // Press the Alternate Repeat Key.
  110. run_one_scan_loop();
  111. EXPECT_TRUE(process_record_user_was_called_);
  112. // Expect the corresponding release event.
  113. ExpectProcessRecordUserCalledWith(false, KC_DEL, -n);
  114. key_alt_repeat.release(); // Release the Repeat Key.
  115. run_one_scan_loop();
  116. EXPECT_TRUE(process_record_user_was_called_);
  117. }
  118. process_record_user_fun = process_record_user_default;
  119. tap_keys(key_repeat, key_alt_repeat);
  120. tap_keys(key_pgdn, key_alt_repeat);
  121. tap_keys(key_pgup, key_alt_repeat);
  122. testing::Mock::VerifyAndClearExpectations(&driver);
  123. }
  124. struct TestParamsAlternateKeyCodes {
  125. uint16_t keycode;
  126. uint8_t mods;
  127. uint16_t expected_alt_keycode;
  128. };
  129. // Tests `get_alt_repeat_key_keycode()` for various keycodes.
  130. TEST_F(AltRepeatKey, GetAltRepeatKeyKeycode) {
  131. for (const auto& params : std::vector<TestParamsAlternateKeyCodes>({
  132. // clang-format off
  133. // Each line tests one call to `get_alt_repeat_key_keycode()`:
  134. // {keycode, mods, expected_alt_keycode}.
  135. // Arrows.
  136. {KC_LEFT, 0, KC_RGHT},
  137. {KC_RGHT, 0, KC_LEFT},
  138. {KC_LEFT, MOD_BIT(KC_LSFT), LSFT(KC_RGHT)},
  139. {KC_LEFT, MOD_BIT(KC_RSFT), RSFT(KC_RGHT)},
  140. {KC_LEFT, MOD_BIT(KC_LCTL) | MOD_BIT(KC_LSFT), C(S(KC_RGHT))},
  141. {KC_LEFT, MOD_BIT(KC_LGUI), LGUI(KC_RGHT)},
  142. {C(KC_LEFT), MOD_BIT(KC_LSFT), C(S(KC_RGHT))},
  143. {KC_UP, 0, KC_DOWN},
  144. // Navigation keys.
  145. {KC_PGUP, 0, KC_PGDN},
  146. {KC_HOME, 0, KC_END },
  147. // Media keys.
  148. {KC_WBAK, 0, KC_WFWD},
  149. {KC_MNXT, 0, KC_MPRV},
  150. {KC_MRWD, 0, KC_MFFD},
  151. {KC_VOLU, 0, KC_VOLD},
  152. {KC_BRIU, 0, KC_BRID},
  153. // Emacs navigation.
  154. {KC_N, MOD_BIT(KC_LCTL), C(KC_P)},
  155. {KC_B, MOD_BIT(KC_LCTL), LCTL(KC_F)},
  156. {KC_B, MOD_BIT(KC_RCTL), RCTL(KC_F)},
  157. {KC_B, MOD_BIT(KC_LALT), LALT(KC_F)},
  158. {KC_F, MOD_BIT(KC_LCTL), C(KC_B)},
  159. {KC_A, MOD_BIT(KC_LCTL), C(KC_E)},
  160. {KC_D, MOD_BIT(KC_LCTL), C(KC_U)},
  161. // Vim navigation.
  162. {KC_J, 0, KC_K},
  163. {KC_K, 0, KC_J},
  164. {KC_H, 0, KC_L},
  165. {KC_B, 0, KC_W},
  166. {KC_W, 0, KC_B},
  167. {KC_E, 0, KC_B},
  168. {KC_B, MOD_BIT(KC_LSFT), S(KC_W)},
  169. {KC_W, MOD_BIT(KC_LSFT), S(KC_B)},
  170. {KC_E, MOD_BIT(KC_LSFT), S(KC_B)},
  171. {KC_O, MOD_BIT(KC_LCTL), C(KC_I)},
  172. {KC_I, MOD_BIT(KC_LCTL), C(KC_O)},
  173. // Other.
  174. {KC_DEL, 0, KC_BSPC},
  175. {KC_LBRC, 0, KC_RBRC},
  176. {KC_LCBR, 0, KC_RCBR},
  177. // Some keys where the last key is a tap-hold key.
  178. {LSFT_T(KC_F), MOD_BIT(KC_RCTL), RCTL(KC_B)},
  179. {LT(1, KC_A), MOD_BIT(KC_RGUI), RGUI(KC_E)},
  180. {RALT_T(KC_J), 0, KC_K},
  181. // Some keys where no alternate is defined.
  182. {KC_A, 0, KC_NO},
  183. {KC_F1, 0, KC_NO},
  184. {QK_LEAD, 0, KC_NO},
  185. {MO(1), 0, KC_NO},
  186. // clang-format on
  187. })) {
  188. SCOPED_TRACE(std::string("Input keycode: ") + get_keycode_string(params.keycode));
  189. set_last_keycode(params.keycode);
  190. set_last_mods(params.mods);
  191. const uint16_t actual = get_alt_repeat_key_keycode();
  192. EXPECT_KEYCODE_EQ(get_alt_repeat_key_keycode(), params.expected_alt_keycode);
  193. }
  194. }
  195. // Test adding to and overriding the above through the
  196. // `get_alt_repeat_key_keycode_user()` callback.
  197. TEST_F(AltRepeatKey, GetAltRepeatKeyKeycodeUser) {
  198. get_alt_repeat_key_keycode_user_fun = [](uint16_t keycode, uint8_t mods) -> uint16_t {
  199. bool shifted = (mods & MOD_MASK_SHIFT);
  200. switch (keycode) {
  201. case KC_LEFT:
  202. return KC_ENT;
  203. case MO(1):
  204. return TG(1);
  205. case KC_TAB: // Tab <-> Shift + Tab example.
  206. if (shifted) {
  207. return KC_TAB;
  208. } else {
  209. return S(KC_TAB);
  210. }
  211. }
  212. // Ctrl + Y <-> Ctrl + Z example.
  213. if ((mods & MOD_MASK_CTRL)) {
  214. switch (keycode) {
  215. case KC_Y:
  216. return C(KC_Z);
  217. case KC_Z:
  218. return C(KC_Y);
  219. }
  220. }
  221. return KC_NO;
  222. };
  223. set_last_keycode(KC_LEFT);
  224. EXPECT_KEYCODE_EQ(get_alt_repeat_key_keycode(), KC_ENT);
  225. set_last_keycode(MO(1));
  226. EXPECT_KEYCODE_EQ(get_alt_repeat_key_keycode(), TG(1));
  227. set_last_keycode(KC_TAB);
  228. EXPECT_KEYCODE_EQ(get_alt_repeat_key_keycode(), S(KC_TAB));
  229. set_last_keycode(KC_TAB);
  230. set_last_mods(MOD_BIT(KC_LSFT));
  231. EXPECT_KEYCODE_EQ(get_alt_repeat_key_keycode(), KC_TAB);
  232. set_last_keycode(KC_Z);
  233. set_last_mods(MOD_BIT(KC_LCTL));
  234. EXPECT_KEYCODE_EQ(get_alt_repeat_key_keycode(), C(KC_Y));
  235. set_last_keycode(KC_Y);
  236. set_last_mods(MOD_BIT(KC_LCTL));
  237. EXPECT_KEYCODE_EQ(get_alt_repeat_key_keycode(), C(KC_Z));
  238. }
  239. // Tests rolling from a key to Alternate Repeat.
  240. TEST_F(AltRepeatKey, RollingToAltRepeat) {
  241. TestDriver driver;
  242. KeymapKey key_left(0, 0, 0, KC_LEFT);
  243. KeymapKey key_alt_repeat(0, 1, 0, QK_AREP);
  244. set_keymap({key_left, key_alt_repeat});
  245. {
  246. InSequence seq;
  247. EXPECT_REPORT(driver, (KC_LEFT));
  248. EXPECT_REPORT(driver, (KC_LEFT, KC_RGHT));
  249. EXPECT_REPORT(driver, (KC_RGHT));
  250. EXPECT_EMPTY_REPORT(driver);
  251. EXPECT_REPORT(driver, (KC_RGHT));
  252. EXPECT_EMPTY_REPORT(driver);
  253. }
  254. // Perform a rolled press from Left to Alternate Repeat.
  255. ExpectProcessRecordUserCalledWith(true, KC_LEFT, 0);
  256. key_left.press();
  257. run_one_scan_loop();
  258. EXPECT_TRUE(process_record_user_was_called_);
  259. ExpectProcessRecordUserCalledWith(true, KC_RGHT, -1);
  260. key_alt_repeat.press(); // Press the Alternate Repeat Key.
  261. run_one_scan_loop();
  262. EXPECT_TRUE(process_record_user_was_called_);
  263. ExpectProcessRecordUserCalledWith(false, KC_LEFT, 0);
  264. key_left.release();
  265. run_one_scan_loop();
  266. EXPECT_TRUE(process_record_user_was_called_);
  267. ExpectProcessRecordUserCalledWith(false, KC_RGHT, -1);
  268. key_alt_repeat.release(); // Release the Alternate Repeat Key.
  269. run_one_scan_loop();
  270. EXPECT_TRUE(process_record_user_was_called_);
  271. process_record_user_fun = process_record_user_default;
  272. tap_key(key_alt_repeat);
  273. testing::Mock::VerifyAndClearExpectations(&driver);
  274. }
  275. // Tests rolling from Alternate Repeat to another key.
  276. TEST_F(AltRepeatKey, RollingFromAltRepeat) {
  277. TestDriver driver;
  278. KeymapKey key_left(0, 0, 0, KC_LEFT);
  279. KeymapKey key_up(0, 1, 0, KC_UP);
  280. KeymapKey key_alt_repeat(0, 2, 0, QK_AREP);
  281. set_keymap({key_left, key_up, key_alt_repeat});
  282. {
  283. InSequence seq;
  284. EXPECT_REPORT(driver, (KC_LEFT));
  285. EXPECT_EMPTY_REPORT(driver);
  286. EXPECT_REPORT(driver, (KC_RGHT));
  287. EXPECT_REPORT(driver, (KC_RGHT, KC_UP));
  288. EXPECT_REPORT(driver, (KC_UP));
  289. EXPECT_EMPTY_REPORT(driver);
  290. EXPECT_REPORT(driver, (KC_DOWN));
  291. EXPECT_EMPTY_REPORT(driver);
  292. }
  293. tap_key(key_left);
  294. // Perform a rolled press from Alternate Repeat to Up.
  295. ExpectProcessRecordUserCalledWith(true, KC_RGHT, -1);
  296. key_alt_repeat.press(); // Press the Alternate Repeat Key.
  297. run_one_scan_loop();
  298. EXPECT_TRUE(process_record_user_was_called_);
  299. ExpectProcessRecordUserCalledWith(true, KC_UP, 0);
  300. key_up.press();
  301. run_one_scan_loop();
  302. EXPECT_TRUE(process_record_user_was_called_);
  303. EXPECT_KEYCODE_EQ(get_last_keycode(), KC_UP);
  304. ExpectProcessRecordUserCalledWith(false, KC_RGHT, -1);
  305. key_alt_repeat.release(); // Release the Alternate Repeat Key.
  306. run_one_scan_loop();
  307. EXPECT_TRUE(process_record_user_was_called_);
  308. ExpectProcessRecordUserCalledWith(false, KC_UP, 0);
  309. key_up.release();
  310. run_one_scan_loop();
  311. EXPECT_TRUE(process_record_user_was_called_);
  312. process_record_user_fun = process_record_user_default;
  313. tap_key(key_alt_repeat);
  314. testing::Mock::VerifyAndClearExpectations(&driver);
  315. }
  316. // Tests using the Alternate Repeat Key on a macro that doesn't have an
  317. // alternate keycode defined.
  318. TEST_F(AltRepeatKey, AlternateUnsupportedMacro) {
  319. TestDriver driver;
  320. KeymapKey key_foo(0, 0, 0, QK_USER_0);
  321. KeymapKey key_alt_repeat(0, 1, 0, QK_AREP);
  322. set_keymap({key_foo, key_alt_repeat});
  323. process_record_user_fun = [=](uint16_t keycode, keyrecord_t* record) {
  324. process_record_user_was_called_ = true;
  325. switch (keycode) {
  326. case QK_USER_0:
  327. if (record->event.pressed) {
  328. SEND_STRING("foo");
  329. }
  330. break;
  331. }
  332. return true;
  333. };
  334. // Allow any number of empty reports.
  335. EXPECT_EMPTY_REPORT(driver).Times(AnyNumber());
  336. ExpectString(driver, "foofoo");
  337. process_record_user_was_called_ = false;
  338. tap_key(key_foo);
  339. EXPECT_TRUE(process_record_user_was_called_);
  340. EXPECT_KEYCODE_EQ(get_last_keycode(), QK_USER_0);
  341. EXPECT_KEYCODE_EQ(get_alt_repeat_key_keycode(), KC_NO);
  342. process_record_user_was_called_ = false;
  343. key_alt_repeat.press(); // Press Alternate Repeat.
  344. run_one_scan_loop();
  345. EXPECT_FALSE(process_record_user_was_called_);
  346. process_record_user_was_called_ = false;
  347. key_alt_repeat.release(); // Release Alternate Repeat.
  348. run_one_scan_loop();
  349. EXPECT_FALSE(process_record_user_was_called_);
  350. process_record_user_was_called_ = false;
  351. tap_key(key_foo);
  352. EXPECT_TRUE(process_record_user_was_called_);
  353. testing::Mock::VerifyAndClearExpectations(&driver);
  354. }
  355. // Tests a macro with custom alternate behavior.
  356. TEST_F(AltRepeatKey, MacroCustomAlternate) {
  357. TestDriver driver;
  358. KeymapKey key_foo(0, 0, 0, QK_USER_0);
  359. KeymapKey key_alt_repeat(0, 1, 0, QK_AREP);
  360. set_keymap({key_foo, key_alt_repeat});
  361. get_alt_repeat_key_keycode_user_fun = [](uint16_t keycode, uint8_t mods) -> uint16_t {
  362. switch (keycode) {
  363. case QK_USER_0:
  364. return QK_USER_0; // QK_USER_0 handles its own alternate.
  365. default:
  366. return KC_NO; // No key by default.
  367. }
  368. };
  369. process_record_user_fun = [=](uint16_t keycode, keyrecord_t* record) {
  370. process_record_user_was_called_ = true;
  371. switch (keycode) {
  372. case QK_USER_0:
  373. if (record->event.pressed) {
  374. if (get_repeat_key_count() >= 0) {
  375. SEND_STRING("foo");
  376. } else { // Key is being alternate repeated.
  377. SEND_STRING("bar");
  378. }
  379. }
  380. break;
  381. }
  382. return true;
  383. };
  384. // Allow any number of empty reports.
  385. EXPECT_EMPTY_REPORT(driver).Times(AnyNumber());
  386. ExpectString(driver, "foobarbar");
  387. tap_keys(key_foo, key_alt_repeat, key_alt_repeat);
  388. testing::Mock::VerifyAndClearExpectations(&driver);
  389. }
  390. // Tests the Additional "Alternate" keys example from the documentation page.
  391. TEST_F(AltRepeatKey, AdditionalAlternateKeysExample) {
  392. TestDriver driver;
  393. KeymapKey key_a(0, 0, 0, KC_A);
  394. KeymapKey key_w(0, 1, 0, KC_W);
  395. KeymapKey key_altrep2(0, 2, 0, QK_USER_0);
  396. KeymapKey key_altrep3(0, 3, 0, QK_USER_1);
  397. set_keymap({key_a, key_w, key_altrep2, key_altrep3});
  398. remember_last_key_user_fun = [](uint16_t keycode, keyrecord_t* record, uint8_t* remembered_mods) {
  399. switch (keycode) {
  400. case QK_USER_0:
  401. case QK_USER_1:
  402. return false; // Ignore ALTREP keys.
  403. }
  404. return true; // Other keys can be repeated.
  405. };
  406. process_record_user_fun = [=](uint16_t keycode, keyrecord_t* record) {
  407. switch (keycode) {
  408. case QK_USER_0:
  409. if (record->event.pressed) {
  410. const uint16_t last_key = get_last_keycode();
  411. switch (last_key) {
  412. case KC_A:
  413. SEND_STRING(/*a*/ "tion");
  414. break;
  415. case KC_W:
  416. SEND_STRING(/*w*/ "hich");
  417. break;
  418. }
  419. }
  420. return false;
  421. case QK_USER_1:
  422. if (record->event.pressed) {
  423. const uint16_t last_key = get_last_keycode();
  424. switch (last_key) {
  425. case KC_A:
  426. SEND_STRING(/*a*/ "bout");
  427. break;
  428. case KC_W:
  429. SEND_STRING(/*w*/ "ould");
  430. break;
  431. }
  432. }
  433. return false;
  434. }
  435. return true;
  436. };
  437. // Allow any number of empty reports.
  438. EXPECT_EMPTY_REPORT(driver).Times(AnyNumber());
  439. ExpectString(driver, "ationwhichaboutwould");
  440. tap_keys(key_a, key_altrep2, key_w, key_altrep2);
  441. tap_keys(key_a, key_altrep3, key_w, key_altrep3);
  442. testing::Mock::VerifyAndClearExpectations(&driver);
  443. }
  444. } // namespace