mirror of
https://github.com/zsa/qmk_firmware.git
synced 2026-01-10 15:42:25 +00:00
* add 75_(ansi|iso) Community Layouts to mechlovin/olly/octagon (#22459) * expand mechlovin/olly/octagon * Update info.json * Rename info.json to keyboard.json * correct matrix position for key * remove VIA * [Core] get_keycode_string(): function to format keycodes as strings, for more readable debug logging. (#24787) * keycode_string(): Format keycodes as strings. This adds the `keycode_string()` function described in https://getreuer.info/posts/keyboards/keycode-string/index.html as a core feature. * Fix formatting. * keycode_string review revisions. * Rename keycode_string() -> get_keycode_string() for consistency with existing string utils like get_u8_str(). * Revise custom keycode names with separate _user and _kb tables. * Correct indent in builddefs/generic_features.mk. Co-authored-by: Ryan <fauxpark@gmail.com> * Add KC_NUHS, KC_NUBS, and KC_CAPS. * Fix linking error with custom names. * Attempt at simplifying interface. * Formatting fix. * Several fixes and revisions. * Don't use PSTR in KEYCODE_STRING_NAME, since this fails to build on AVR. Store custom names in RAM. * Revise the internal table of common keycode names to use its own storage representation, still in PROGMEM, and now more efficiently stored flat in 8 bytes per entry. * Support Swap Hands keycodes and a few other keycodes. * Revert "Formatting fix." This reverts commit 2a2771068c7ee545ffac4103aa07e847a9ec3816. * Revert "Attempt at simplifying interface." This reverts commit 8eaf67de76e75bc92d106a8b0decc893fbc65fa5. * Simplify custom names API by sigprof's suggestion. * Support more keycodes. * Add QK_LOCK keycode. * Add Secure keycodes. * Add Joystick keycodes. * Add Programmable Button keycodes. * Add macro MC_ keycodes. * For remaining keys in known code ranges, stringify them as "QK_<feature>+<number>". For instance, "QK_MIDI+7". * Bug fix and a few improvements. * Fix missing right-hand bit when displaying 5-bit mods numerically. * Support KC_HYPR, KC_MEH, HYPR_T(kc), MEH_T(kc). * Exclude one-shot keycodes when NO_ACTION_ONESHOT is defined. --------- Co-authored-by: Ryan <fauxpark@gmail.com> * Align to latest CLI dependencies (#24553) * Align to latest CLI dependencies * Update docs * Add Community Layout support to daskeyboard4 (#23884) add ansi CL * Add the plywrks ply8x hotswap variant. (#23558) * Add hotswap variant * Update RGB matrix * Move files around to target develop * Revert rules.mk for keyboards/jaykeeb/joker/rules.mk * Update keyboards/plywrks/ply8x/hotswap/keyboard.json Co-authored-by: Drashna Jaelre <drashna@live.com> * Apply suggestions from code review Co-authored-by: Duncan Sutherland <dunk2k_2000@hotmail.com> * Add missing community layouts * Delete keyboards/plywrks/ply8x/rules.mk * Update missing keys in RGB matrix * Add missing key in RGB matrix for hotswap ver * Remove via keymaps * Add keyboard alias for plywrks/ply8x to plywrks/ply8x/solder * Fix typo * Fix another typo --------- Co-authored-by: Drashna Jaelre <drashna@live.com> Co-authored-by: Duncan Sutherland <dunk2k_2000@hotmail.com> * [Core] use `keycode_string` in unit tests (#25042) * tests: use keycode_string feature With a proper keycode to string implementation in qmk there is no need to use the unit tests only implementation anymore. Signed-off-by: Stefan Kerkmann <karlk90@pm.me> * tests: remove keycode_util feature This feature is no longer used as we switched the tests to the keycode string implementation. Signed-off-by: Stefan Kerkmann <karlk90@pm.me> * Non-volatile memory data repository pattern (#24356) * First batch of eeconfig conversions. * Offset and length for datablocks. * `via`, `dynamic_keymap`. * Fix filename. * Commentary. * wilba leds * satisfaction75 * satisfaction75 * more keyboard whack-a-mole * satisfaction75 * omnikeyish * more whack-a-mole * `generic_features.mk` to automatically pick up nvm repositories * thievery * deferred variable resolve * whitespace * convert api to structs/unions * convert api to structs/unions * convert api to structs/unions * fixups * code-side docs * code size fix * rollback * nvm_xxxxx_erase * Updated location of eeconfig magic numbers so non-EEPROM nvm drivers can use them too. * Fixup build. * Fixup compilation error with encoders. * Build fixes. * Add `via_ci` keymap to onekey to exercise VIA bindings (and thus dynamic keymap et.al.), fixup compilation errors based on preprocessor+sizeof. * Build failure rectification. * Migrate remaining `split.soft_serial_pin` to `split.serial.pin` (#25046) * Migrate keyboards/bastardkb * Migrate keyboards/handwired * Migrate keyboards/helix * Fix duplicate serial key * Fix outdated GPIO control function usage (#25060) * [Modules] Provide access to current path in `rules.mk`. (#25061) * Update keymap for keycult 1800 (#25070) Update keymap Co-authored-by: yiancar <yiancar@gmail.com> * Add handwired/erikpeyronson/erkbd (#25030) Co-authored-by: Erik Peyronson <erik.peyronson@gmail.com> Co-authored-by: jack <jack@pngu.org> Co-authored-by: Drashna Jaelre <drashna@live.com> * Add support for Starry FRL (#24626) Co-authored-by: jack <jack@pngu.org> Co-authored-by: Drashna Jaelre <drashna@live.com> Co-authored-by: Ryan <fauxpark@gmail.com> * Add "Large Lad" keyboard (#24727) Co-authored-by: jack <jack@pngu.org> * Bump vite from 5.4.12 to 5.4.15 in /builddefs/docsgen (#25065) Bumps [vite](https://github.com/vitejs/vite/tree/HEAD/packages/vite) from 5.4.12 to 5.4.15. - [Release notes](https://github.com/vitejs/vite/releases) - [Changelog](https://github.com/vitejs/vite/blob/v5.4.15/packages/vite/CHANGELOG.md) - [Commits](https://github.com/vitejs/vite/commits/v5.4.15/packages/vite) --- updated-dependencies: - dependency-name: vite dependency-type: direct:development ... Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Allow AnnePro2 to reboot (#24886) Without this, the QK_REBOOT key did nothing. * Fix path typo related RP2040 (#25069) Fix path typo * Update onekey example for nucleo f446re (#25067) * use accessible pins for nucleo f446re onekey example * remove pin collision with matrix in keyboard.json * use accessible pins for LED * remove pin collision with matrix * Update readme.md to reflect pin changes * Module documentation typo correction (#25073) * Fix lockups on AVR with `qmk/hello_world` module (#25074) Fix lockups on AVR. * At101ish (#25072) * Dell AT101 replacement pcb support * Update keyboards/at101ish/readme.md Co-Authored-By: fauxpark <fauxpark@gmail.com> * remove empty src clause in makefile * feature: Update at101ish to qmk v0.28 * feature: Add osdetecting keymap variant. * refactor: Move at101ish keyboard to handwired folder. * fix: Adjust at101ish readme- * fix: review changes. * chore: Remove unneeded feature. --------- Co-authored-by: fauxpark <fauxpark@gmail.com> * Add "license" field to Community Module JSON schema. (#25085) Add "license" field to community module schema. * Add kt60HS-T v2 PCB (#25080) * Add kt60HS-Tv2 * Update keyboards/keyten/kt60hs_t/readme.md Co-authored-by: Sergey Vlasov <sigprof@gmail.com> * Update keyboards/keyten/kt60hs_t/v1/readme.md Co-authored-by: Sergey Vlasov <sigprof@gmail.com> * Update keyboards/keyten/kt60hs_t/v2/keyboard.json Co-authored-by: Sergey Vlasov <sigprof@gmail.com> * Update keyboards/keyten/kt60hs_t/v2/readme.md Co-authored-by: Sergey Vlasov <sigprof@gmail.com> * Update keyboards/keyten/kt60hs_t/info.json Co-authored-by: Sergey Vlasov <sigprof@gmail.com> * Change of structure * Moving the keyboard * Update data/mappings/keyboard_aliases.hjson Co-authored-by: Sergey Vlasov <sigprof@gmail.com> * Update keyboards/keyten/kt60hs_t/v1/keyboard.json Co-authored-by: Duncan Sutherland <dunk2k_2000@hotmail.com> * Update keyboards/keyten/kt60hs_t/v2/keyboard.json Co-authored-by: Duncan Sutherland <dunk2k_2000@hotmail.com> --------- Co-authored-by: Sergey Vlasov <sigprof@gmail.com> Co-authored-by: Duncan Sutherland <dunk2k_2000@hotmail.com> * Make sure that unit tests run on all release versions * [ErgoDox EZ] Fix complication issues due to updates * [ErgoDox EZ] Fix compilication errors and warnings We want all green! * Fix 'qmk lint -kb' argument handling (#25093) * Refactor Deemen17 Works DE60 (#25088) * Add Coban Pad 12A (#25039) Co-authored-by: jack <jack@pngu.org> * [Keyboard] Add PHDesign PH60/Multi Keyboard PCB (#25086) * Add PH60/Multi Support * Add PCB PIcture for README * Remove MO(_FN2) * README Typo Fix * Layout and README Adjustment * Add README for PHDesign Main Folder * Keymap Improvement * Update README.md * [Keyboard] Add Ortho Slayer (#25099) * Add Ortho Slayer * Update keyboards/keyten/ortho_slayer/keymaps/default/keymap.c Co-authored-by: jack <jack@pngu.org> * Update keyboards/keyten/ortho_slayer/readme.md Co-authored-by: jack <jack@pngu.org> --------- Co-authored-by: jack <jack@pngu.org> * Fix coban pad9a wrong layout in keyboard.json (#25100) * Remove `CTPC`/`CONVERT_TO_PROTON_C` options (#25111) * Remove direct docs.qmk.fm links from docs (#25113) * Add warning when deprecated 'promicro_rp2040' is used (#25112) * Add Vida to QMK (#24225) Co-authored-by: jack <jack@pngu.org> Co-authored-by: Duncan Sutherland <dunk2k_2000@hotmail.com> * More Windows->Unix style path fixes. (#25119) * Include `math.h` where necessary. (#25122) * Cater for use of `__errno_r()` in ChibiOS syscalls.c with newer picolibc revisions (#25121) * chore: Allow disabling underglow on Work Louder devices (#25123) (#25120) * Allow disabling Underglow on Work Louder devices Allows disabling Underglow on Work Louder devices by using `RGBLIGHT_ENABLE = no` on rules.mk * Update keyboards/work_louder/rgb_functions.c Suggested by @zvecr on review. Co-authored-by: Joel Challis <git@zvecr.com> --------- Co-authored-by: Joel Challis <git@zvecr.com> * Exclude external userspace from lint checking (#24680) * fix: Fix startup sound for Preonic (#25132) (#25133) Add `AUDIO_INIT_DELAY ` to config.h to resolve * New standard layout for Savage65 (65_ansi_blocker_tsangan_split_bs) (#24690) * Added a default firmware and layout for the WindStudio Wind X R1 keyboard. * Wind X R1: cleaned-up the folders to make clear that this firmware is for the release 1 of this keyboard. * Delete keyboards/windstudio/wind_x/R1 directory Removing the uppercase R1 folder * feat(cannonkeys/savage65): Added layout to keyboard.json - Added the layout LAYOUT_65_ansi_blocker_tsangan_split_bs to the community layouts. * kradoindustries_promenade: add LAYOUT_1x2u (#25090) * Update shuguet/shu89 (#24780) * Update keyboard.json Update mod keys location in RGB layout. * Update keyboard.json * Update keyboards/shuguet/shu89/keyboard.json Co-authored-by: Ryan <fauxpark@gmail.com> --------- Co-authored-by: Ryan <fauxpark@gmail.com> * [Keyboard] Add suika83opti (#24991) * [chore]: move and rename mouse/scroll min/max defines (#25141) * protocol: move {XY/HV}_REPORT_{MIN,MAX} into report.h ..to allow easier re-use in other code implementations. * protocol: rename {XY/HV}_REPORT_{MIN/MAX} to MOUSE_REPORT_{XY/HV}_{MIN/MAX} ..to avoid naming collisions. * [Core] Flow Tap tap-hold option to disable HRMs during fast typing (#25125) aka Global Quick Tap, Require Prior Idle * Add Link keyboard (#25058) Co-authored-by: jack <jack@pngu.org> Co-authored-by: Drashna Jaelre <drashna@live.com> * Remove Sofle `rgb_default` keymap & tidy readme's (#25010) * Added Keyboard LumPy27 (#24967) Co-authored-by: jack <jack@pngu.org> Co-authored-by: Drashna Jaelre <drashna@live.com> * [Keyboard] Kobold r1 (#25161) * Kobold r1 * Apply suggestions from code review Co-authored-by: jack <jack@pngu.org> * `board_init` => `early_hardware_init_post`. --------- Co-authored-by: jack <jack@pngu.org> * [Docs] Unify lighting step descriptions (#25167) unify lighting step descriptions and defaults across docs * [Keyboard] Add voidhhkb-hotswap (#25007) * Added files for voidhhkb-hotswap * Updated keyboard name to resolve build errors * Implement suggestions from PR. Use 60_hhkb community layout. * Update keyboards/void/voidhhkb_hotswap/readme.md Co-authored-by: Ryan <fauxpark@gmail.com> * Apply suggestions from code review Co-authored-by: jack <jack@pngu.org> --------- Co-authored-by: Ryan <fauxpark@gmail.com> Co-authored-by: jack <jack@pngu.org> * Remove duplication of RGB Matrix defaults (#25146) * Remove duplication of RGB Matrix defaults * Remove more duplication of defaults * fix * Fix missing and extra commas in JSON schema (#25057) * Remove duplication of RGBLight defaults (#25169) * Ignore the Layer Lock key in Repeat Key and Caps Word. (#25171) * Allow for disabling EEPROM subsystem entirely. (#25173) * [keyboard] ymdk/id75/rp2040 (#25157) Co-authored-by: tao heihei <> * Remove `bluefruit_le_read_battery_voltage` function (#25129) * Fix 'Would you like to clone the submodules?' prompt under msys (#24958) * Fixup eeconfig lighting reset. (#25166) * DOCS: `qmk-hid` missing in bootloaders list? (#25177) * Fix for `.clangd`. (#25180) * Update develop branch to Pico SDK 1.5.1 (#25178) * Add lint warning for empty url (#25182) * Implement connection keycode logic (#25176) * Add handwired/footy (#25151) Co-authored-by: jack <jack@pngu.org> * Decrease firmware size for `anavi/macropad8`. (#25185) Preparation for bootstrapper. * Align ChibiOS `USB_WAIT_FOR_ENUMERATION` implementation (#25184) * [Bug][Core] Fix for Flow Tap: fix handling of distinct taps and timer updates. (#25175) * Flow Tap bug fix. As reported by @amarz45 and @mwpardue, there is a bug where if two tap-hold keys are pressed in distinct taps back to back, then Flow Tap is not applied on the second tap-hold key, but it should be. In a related bug reported by @NikGovorov, if a tap-hold key is held followed by a tap of a tap-hold key, then Flow Tap updates its timer on the release of the held tap-hold key, but it should be ignored. The problem common to both these bugs is that I incorrectly assumed `tapping_key` is cleared to noevent once it is released, when actually `tapping_key` is still maintained for `TAPPING_TERM` ms after release (for Quick Tap). This commit fixes that. Thanks to @amarz45, @mwpardue, and @NikGovorov for reporting! Details: * Logic for converting the current tap-hold event to a tap is extracted to `flow_tap_key_if_within_term()`, which is now invoked also in the post-release "interfered with other tap key" case. This fixes the distinct taps bug. * The Flow Tap timer is now updated at the beginning of each call to `process_record()`, provided that there is no unsettled tap-hold key at that time and that the record is not for a mod or layer switch key. By moving this update logic to `process_record()`, it is conceptually simpler and more robust. * Unit tests extended to cover the reported scenarios. * Fix formatting. * Revision to fix @NikGovorov's scenario. The issue is that when another key is pressed while a layer-tap hasn't been settled yet, the `prev_keycode` remembers the keycode from before the layer switched. This can then enable Flow Tap for the following key when it shouldn't, or vice versa. Thanks to @NikGovorov for reporting! This commit revises Flow Tap in the following ways: * The previous key and timer are both updated from `process_record()`. This is slightly later in the sequence of processing than before, and by this point, a just-settled layer-tap should have taken effect so that the keycode from the correct layer is remembered. * The Flow Tap previous key and timer are updated now also on key release events, except for releases of modifiers and held layer switches. * The Flow Tap previous key and timer are now updated together, for simplicity. This makes the logic easier to think about. * A few additional unit tests, including @NikGovorov's scenario as "layer_tap_ignored_with_disabled_key_complex." * Remove empty `url` fields (#25181) * Prompt for converter when creating new keymap (#25116) * High resolution scrolling (without feature report parsing) (#24423) * hires scrolling without feature report parsing * fix valid range for exponent * fix incorrect minimum exponent value documentation * Avoid duplication in generated community modules `rules.mk` (#25135) * Bump rlespinasse/github-slug-action from 3 to 5 (#25021) * Remove `"console":false` from keyboards (#25190) * Update 'qmk generate-api' to only publish pure DD keymaps (#24782) * Remove more duplication of defaults (#25189) * Align `new-keyboard` template to current standards (#25191) * Bump vite from 5.4.15 to 5.4.18 in /builddefs/docsgen (#25192) Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Fix `boardsource/beiwagon` RGB Matrix coordinates (#25018) * Remove `"command":false` from keyboards (#25193) * Extend lint checks to reject duplication of defaults (#25149) * modelh: add prerequisites for via support (#24932) * First TypeK support (#22876) * Add Lemokey X0 keyboard (#24994) * keyboards/annepro2/ld: Add per-variant linker scripts (#24999) C18 has an MCU with 16K SRAM, up from C15's 8K. Split the linker script into C15 and C18 variants to make use of the larger RAM capacity of C18. * Add new keyboard MirageiX (#25054) Co-authored-by: Drashna Jaelre <drashna@live.com> Co-authored-by: jack <0x6a73@protonmail.com> * [Keymap] Sofle RGB - fixed stuck on numpad layer and layout comments (#24942) * Add handwired 4x14 LuMaWing keyboard (#24885) * Add jcpm2 (JC Pro Macro 2) (#24816) Co-authored-by: jack <jack@pngu.org> Co-authored-by: Drashna Jaelre <drashna@live.com> Co-authored-by: Joel Challis <git@zvecr.com> * [Keyboard] Add splitkb.com's Halcyon Elora rev2 (#24790) Co-authored-by: Drashna Jaelre <drashna@live.com> * [Keyboard] mzmkb/slimdash/rev1 (#24804) Co-authored-by: Drashna Jaelre <drashna@live.com> Co-authored-by: jack <jack@pngu.org> Co-authored-by: Duncan Sutherland <dunk2k_2000@hotmail.com> Co-authored-by: Ryan <fauxpark@gmail.com> * Added new keyboard epssp75 (#24756) Co-authored-by: Drashna Jaelre <drashna@live.com> * Addition of OK-1 (#24646) Co-authored-by: Joel Challis <git@zvecr.com> Co-authored-by: Drashna Jaelre <drashna@live.com> * Add Umbra keyboard (#24569) Co-authored-by: Duncan Sutherland <dunk2k_2000@hotmail.com> Co-authored-by: Ryan <fauxpark@gmail.com> Co-authored-by: Drashna Jaelre <drashna@live.com> * Amptrics 0420 keyboard addition (#24744) Co-authored-by: jack <jack@pngu.org> Co-authored-by: Drashna Jaelre <drashna@live.com> Co-authored-by: Ryan <fauxpark@gmail.com> * [Core] Enhance Flow Tap to work better for rolls over multiple tap-hold keys. (#25200) * Flow Tap revision for rolling press. * Remove debugging cruft. * Formatting fix. * amptrics/0422 - Prevent OOB in `update_leds_for_layer` (#25209) * [Keyboard] Add Gravity-45(#25206) * add gravity-45 * readme.md * fix readme * Update keyboards/green_keys/gravity_45/keyboard.json Co-authored-by: jack <0x6a73@protonmail.com> * run qmk format-json -i keyboards/green_keys/gravity_45/keyboard.json * add url * Update keyboards/green_keys/gravity_45/keyboard.json Co-authored-by: Duncan Sutherland <dunk2k_2000@hotmail.com> * Update keyboard.json * Update keyboard.json * Update keyboards/green_keys/gravity_45/keyboard.json Co-authored-by: Drashna Jaelre <drashna@live.com> --------- Co-authored-by: jack <0x6a73@protonmail.com> Co-authored-by: Duncan Sutherland <dunk2k_2000@hotmail.com> Co-authored-by: Drashna Jaelre <drashna@live.com> * Remove redundant keyboard headers (#25208) * Fix Spleeb compile when pointing device is enabled (#25016) * [Bug] Minimise force-included files (#25194) * Add additional hooks for Community modules (#25050) * Workaround for resolving keyboard alias for config file values (#25228) * Generate versions to keycode headers (#25219) * Resolve alias for `qmk new-keymap` keyboard prompts (#25210) * [Keyboard] Add Binepad KN01 (#25224) * Add Binepad NeoKnob KN01 * post @waffle87 recommendations * Add battery changed callbacks (#25207) * Ensure `qmk_userspace_paths` maintains detected order (#25204) * Bind Bluetooth driver to `host_driver_t` (#25199) * Deprecate `qmk generate-compilation-database`. (#25237) * Remove force disable of NKRO when Bluetooth enabled (#25201) * Keycult 60 (#25213) Co-authored-by: Duncan Sutherland <dunk2k_2000@hotmail.com> Co-authored-by: jack <jack@pngu.org> * [Keyboard] Add Binepad KnobX1 (#25222) Co-authored-by: Drashna Jaelre <drashna@live.com> * [Keyboard] Update Tractyl Manuform and add F405 (weact) variant (#24764) * Layout corrections: Zed60 (#25003) * Fixed print statement after enabling 32-bit layers (#25027) * Fix Aurora sweep default keymap configuration (#25148) * Docs update for installing qmk with uv (#24995) * CXT Studio 12E3: Fix encoder resolutions not applying (#25242) * add resolution to encoders so they apply * Tweak default keymap * replace KC_UNDO with C(KC_Z) as well * Add debounce to duplicated defaults check (#25246) * [Docs] Fix typos introduced by PR #25050 (#25250) It isn't a drashna PR if there aren't some typos in it somewhere. * Allow LVGL onekey keymap to be able compile for other board (#25005) * [Keyboard] Add Jason Hazel’s Bad Wings v2 (#25252) Co-authored-by: Florent Allard <florent.allard@savoirfairelinux.com> * Add raw_hid support to host driver (#25255) * Fix Wear Leveling compilation (#25254) * [New Feature/Core] New RGB Matrix Animation "Starlight Smooth" (#25203) * Enable community modules to define LED matrix and RGB matrix effects. (#25187) Co-authored-by: Joel Challis <git@zvecr.com> * Fix OS_DETECTION_KEYBOARD_RESET (#25015) Co-authored-by: Nick Brassel <nick@tzarc.org> * Fixes the numlock indicator for Magic Force MF17 numpad (#25260) * Add Harite v2 keyboard (#24975) * dlip/haritev2 - Post merge fixes (#25264) * Remove more USB only branches from NKRO handling (#25263) * Deprecate `usb.force_nkro`/`FORCE_NKRO` (#25262) * Add BDN9 Rev. 3 (#25261) * Remove duplicate of SPI default config from keyboards (#25266) * Resolve miscellaneous keyboard lint warnings (#25268) * Add Zeropad (#24737) Co-authored-by: Drashna Jaelre <drashna@live.com> Co-authored-by: Joel Challis <git@zvecr.com> * Add Waveshare RP2040-Keyboard-3 support (#25269) * Update PR checklist notes on custom matrix (#25240) * Chew folders (#24785) * gcc15 AVR compilation fixes (#25238) * [Core] STM32G0x1 support (#24301) * Use relative paths for schemas, instead of $id. Enables VScode validation. (#25251) * add doio/kb03 (#24815) Co-authored-by: Drashna Jaelre <drashna@live.com> Co-authored-by: Joel Challis <git@zvecr.com> * Move rookiebwoy to ivndbt (#25142) * [Chore] use {rgblight,rgb_matrix}_hsv_to_rgb overrides (#25271) * Remove outdated `nix` support due to bit-rot. (#25280) * Add `compiler_support.h` (#25274) * Configure boards to use development_board - 0-9 (#25287) * [Fix] lib8tion: enable fixed scale8 and blend functions (#25272) lib8tion: enable fixed scale8 and blend functions These FastLED derived lib8tion functions have been fixed and enabled by default in FastLED. QMK just never set these defines, there is no reason to keep the buggy implementation. It is assumed that nobody relied on the buggy behavior. * Configure boards to use development_board - UVWXYZ (#25288) * [Docs] Fix tap_hold code blocks (#25298) * salicylic_acid3/getta25 - Fix oled keymap (#25295) * Configure boards to use development_board - S (#25293) * Configure boards to use development_board - T (#25294) * 2025 Q2 changelog (#25297) Co-authored-by: Drashna Jaelre <drashna@live.com> Co-authored-by: Nick Brassel <nick@tzarc.org> * Update Oryx module for newer code * Soft reset matrix * Use improved i2c reset for voyager and moonlander matrix * Fix formatting * Remove labeler action (unneeded) * Fix module API version for Oryx module * Use i2cStop instead of trying to work around --------- Signed-off-by: Stefan Kerkmann <karlk90@pm.me> Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: Duncan Sutherland <dunk2k_2000@hotmail.com> Co-authored-by: QMK Bot <hello@qmk.fm> Co-authored-by: Pascal Getreuer <50221757+getreuer@users.noreply.github.com> Co-authored-by: Ryan <fauxpark@gmail.com> Co-authored-by: Joel Challis <git@zvecr.com> Co-authored-by: Ramon Imbao <ramonimbao@gmail.com> Co-authored-by: Stefan Kerkmann <karlk90@pm.me> Co-authored-by: Nick Brassel <nick@tzarc.org> Co-authored-by: jack <jack@pngu.org> Co-authored-by: yiancar <yiangosyiangou@cytanet.com.cy> Co-authored-by: yiancar <yiancar@gmail.com> Co-authored-by: Erik Peyronson <erikpeyronson@gmail.com> Co-authored-by: Erik Peyronson <erik.peyronson@gmail.com> Co-authored-by: Sắn <59417802+MaiTheSan@users.noreply.github.com> Co-authored-by: Hyphen-ated <Hyphen-ated@users.noreply.github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: Geoffrey Frogeye <s+github@frogeye.fr> Co-authored-by: lsh4711 <120231876+lsh4711@users.noreply.github.com> Co-authored-by: Ben Green <bengreen.uk@gmail.com> Co-authored-by: フィルターペーパー <76888457+filterpaper@users.noreply.github.com> Co-authored-by: henrikosorensen <henrik.sorensen@gmail.com> Co-authored-by: Ivan Gromov <38141348+key10iq@users.noreply.github.com> Co-authored-by: Sergey Vlasov <sigprof@gmail.com> Co-authored-by: Pham Duc Minh <95753855+Deemen17@users.noreply.github.com> Co-authored-by: Dam Vu Duy <RyanDam@users.noreply.github.com> Co-authored-by: nonameCCC <79012391+nonameCCC@users.noreply.github.com> Co-authored-by: sudo pacman -Syu <hauvipapro@gmail.com> Co-authored-by: Andrew Kannan <andrew.kannan@gmail.com> Co-authored-by: Luis Garcia <luis@bitjester.com> Co-authored-by: Christian C. Berclaz <christian.berclaz@mac.com> Co-authored-by: Olivier Mehani <shtrom-github@ssji.net> Co-authored-by: Sylvain Huguet <sylvain@huguet.me> Co-authored-by: suikagiken <115451678+suikagiken@users.noreply.github.com> Co-authored-by: Daniel Reisch <danieljreisch@gmail.com> Co-authored-by: ClownFish <177758267+clownfish-og@users.noreply.github.com> Co-authored-by: JamesWilson1996 <47866504+JamesWilson1996@users.noreply.github.com> Co-authored-by: Less/Rikki <86894501+lesshonor@users.noreply.github.com> Co-authored-by: Jan Bláha <blaha.j502@gmail.com> Co-authored-by: Eric Molitor <534583+emolitor@users.noreply.github.com> Co-authored-by: CJ Pais <cj@cjpais.com> Co-authored-by: eynsai <47629346+eynsai@users.noreply.github.com> Co-authored-by: Joel Beckmeyer <joel@beckmeyer.us> Co-authored-by: Álvaro A. Volpato <alvaro.augusto.volpato@gmail.com> Co-authored-by: Aidan Gauland <aidalgol@users.noreply.github.com> Co-authored-by: Michał Kopeć <michal@nozomi.space> Co-authored-by: takashicompany <t@kashi.company> Co-authored-by: jack <0x6a73@protonmail.com> Co-authored-by: Matheus Marques <matheusmbar@gmail.com> Co-authored-by: LucasMateijsen <l.mateijsen@outlook.com> Co-authored-by: Jeremy Cook <jscook55@gmail.com> Co-authored-by: VeyPatch <126267034+VeyPatch@users.noreply.github.com> Co-authored-by: mizma <omoikane@path-works.net> Co-authored-by: hen-des <141591967+hen-des@users.noreply.github.com> Co-authored-by: Cipulot <40441626+Cipulot@users.noreply.github.com> Co-authored-by: josephawilliamsiv <31166673+josephawilliamsiv@users.noreply.github.com> Co-authored-by: vchowl <vchowl@outlook.com> Co-authored-by: Christopher Hoage <iam@chrishoage.com> Co-authored-by: Silvino R. <366673+silvinor@users.noreply.github.com> Co-authored-by: dabstractor <dustindschultz@gmail.com> Co-authored-by: Nathan Cain <13713501+nathanscain@users.noreply.github.com> Co-authored-by: muge <221161+muge@users.noreply.github.com> Co-authored-by: HorrorTroll <sonicvipduc@gmail.com> Co-authored-by: cyxae <cyxae@amphitryon.nrst.fr> Co-authored-by: Florent Allard <florent.allard@savoirfairelinux.com> Co-authored-by: art-was-here <mail@buckles.email> Co-authored-by: Matti Hiljanen <170205+qvr@users.noreply.github.com> Co-authored-by: Wasteland Fluttershy <ingvardm@gmail.com> Co-authored-by: Dane Lipscombe <danelipscombe@gmail.com> Co-authored-by: Danny <nooges@users.noreply.github.com> Co-authored-by: Infos <136488157+diffrentGuesser@users.noreply.github.com> Co-authored-by: Florent Linguenheld <f@linguenheld.fr> Co-authored-by: ivan <81021475+ivndbt@users.noreply.github.com> Co-authored-by: Pablo Martínez <58857054+elpekenin@users.noreply.github.com>
932 lines
37 KiB
C
932 lines
37 KiB
C
#include <stdint.h>
|
|
#include <stdbool.h>
|
|
|
|
#include "action.h"
|
|
#include "action_layer.h"
|
|
#include "action_tapping.h"
|
|
#include "action_util.h"
|
|
#include "keycode.h"
|
|
#include "quantum_keycodes.h"
|
|
#include "timer.h"
|
|
|
|
#ifndef NO_ACTION_TAPPING
|
|
|
|
# if defined(IGNORE_MOD_TAP_INTERRUPT_PER_KEY)
|
|
# error "IGNORE_MOD_TAP_INTERRUPT_PER_KEY has been removed; the code needs to be ported to use HOLD_ON_OTHER_KEY_PRESS_PER_KEY instead."
|
|
# elif defined(IGNORE_MOD_TAP_INTERRUPT)
|
|
# error "IGNORE_MOD_TAP_INTERRUPT is no longer necessary as it is now the default behavior of mod-tap keys. Please remove it from your config."
|
|
# endif
|
|
|
|
# ifndef COMBO_ENABLE
|
|
# define IS_TAPPING_RECORD(r) (KEYEQ(tapping_key.event.key, (r->event.key)))
|
|
# else
|
|
# define IS_TAPPING_RECORD(r) (KEYEQ(tapping_key.event.key, (r->event.key)) && tapping_key.keycode == r->keycode)
|
|
# endif
|
|
# define WITHIN_TAPPING_TERM(e) (TIMER_DIFF_16(e.time, tapping_key.event.time) < GET_TAPPING_TERM(get_record_keycode(&tapping_key, false), &tapping_key))
|
|
# define WITHIN_QUICK_TAP_TERM(e) (TIMER_DIFF_16(e.time, tapping_key.event.time) < GET_QUICK_TAP_TERM(get_record_keycode(&tapping_key, false), &tapping_key))
|
|
|
|
# ifdef DYNAMIC_TAPPING_TERM_ENABLE
|
|
uint16_t g_tapping_term = TAPPING_TERM;
|
|
# endif
|
|
|
|
# ifdef TAPPING_TERM_PER_KEY
|
|
__attribute__((weak)) uint16_t get_tapping_term(uint16_t keycode, keyrecord_t *record) {
|
|
# ifdef DYNAMIC_TAPPING_TERM_ENABLE
|
|
return g_tapping_term;
|
|
# else
|
|
return TAPPING_TERM;
|
|
# endif
|
|
}
|
|
# endif
|
|
|
|
# ifdef QUICK_TAP_TERM_PER_KEY
|
|
__attribute__((weak)) uint16_t get_quick_tap_term(uint16_t keycode, keyrecord_t *record) {
|
|
return QUICK_TAP_TERM;
|
|
}
|
|
# endif
|
|
|
|
# ifdef PERMISSIVE_HOLD_PER_KEY
|
|
__attribute__((weak)) bool get_permissive_hold(uint16_t keycode, keyrecord_t *record) {
|
|
return false;
|
|
}
|
|
# endif
|
|
|
|
# if defined(CHORDAL_HOLD) || defined(FLOW_TAP_TERM)
|
|
# define REGISTERED_TAPS_SIZE 8
|
|
// Array of tap-hold keys that have been settled as tapped but not yet released.
|
|
static keypos_t registered_taps[REGISTERED_TAPS_SIZE] = {};
|
|
static uint8_t num_registered_taps = 0;
|
|
|
|
/** Adds `key` to the registered_taps array. */
|
|
static void registered_taps_add(keypos_t key);
|
|
/** Returns the index of `key` in registered_taps, or -1 if not found. */
|
|
static int8_t registered_tap_find(keypos_t key);
|
|
/** Removes index `i` from the registered_taps array. */
|
|
static void registered_taps_del_index(uint8_t i);
|
|
/** Logs the registered_taps array for debugging. */
|
|
static void debug_registered_taps(void);
|
|
|
|
static bool is_mt_or_lt(uint16_t keycode) {
|
|
return IS_QK_MOD_TAP(keycode) || IS_QK_LAYER_TAP(keycode);
|
|
}
|
|
# endif // defined(CHORDAL_HOLD) || defined(FLOW_TAP_TERM)
|
|
|
|
# if defined(CHORDAL_HOLD)
|
|
extern const char chordal_hold_layout[MATRIX_ROWS][MATRIX_COLS] PROGMEM;
|
|
|
|
/** \brief Finds which queued events should be held according to Chordal Hold.
|
|
*
|
|
* In a situation with multiple unsettled tap-hold key presses, scan the queue
|
|
* up until the first release, non-tap-hold, or one-shot event and find the
|
|
* latest event in the queue that settles as held according to
|
|
* get_chordal_hold().
|
|
*
|
|
* \return Index of the first tap, or equivalently, one past the latest hold.
|
|
*/
|
|
static uint8_t waiting_buffer_find_chordal_hold_tap(void);
|
|
|
|
/** Processes queued events up to and including `key` as tapped. */
|
|
static void waiting_buffer_chordal_hold_taps_until(keypos_t key);
|
|
|
|
/** \brief Processes and pops buffered events until the first tap-hold event. */
|
|
static void waiting_buffer_process_regular(void);
|
|
# endif // CHORDAL_HOLD
|
|
|
|
# ifdef HOLD_ON_OTHER_KEY_PRESS_PER_KEY
|
|
__attribute__((weak)) bool get_hold_on_other_key_press(uint16_t keycode, keyrecord_t *record) {
|
|
return false;
|
|
}
|
|
# endif
|
|
|
|
# if defined(AUTO_SHIFT_ENABLE) && defined(RETRO_SHIFT)
|
|
# include "process_auto_shift.h"
|
|
# endif
|
|
|
|
# if defined(FLOW_TAP_TERM)
|
|
static uint16_t flow_tap_prev_keycode = KC_NO;
|
|
static uint16_t flow_tap_prev_time = 0;
|
|
static bool flow_tap_expired = true;
|
|
|
|
static bool flow_tap_key_if_within_term(keyrecord_t *record, uint16_t prev_time);
|
|
# endif // defined(FLOW_TAP_TERM)
|
|
|
|
static keyrecord_t tapping_key = {};
|
|
static keyrecord_t waiting_buffer[WAITING_BUFFER_SIZE] = {};
|
|
static uint8_t waiting_buffer_head = 0;
|
|
static uint8_t waiting_buffer_tail = 0;
|
|
|
|
static bool process_tapping(keyrecord_t *record);
|
|
static bool waiting_buffer_enq(keyrecord_t record);
|
|
static void waiting_buffer_clear(void);
|
|
static bool waiting_buffer_typed(keyevent_t event);
|
|
static bool waiting_buffer_has_anykey_pressed(void);
|
|
static void waiting_buffer_scan_tap(void);
|
|
static void debug_tapping_key(void);
|
|
static void debug_waiting_buffer(void);
|
|
|
|
/** \brief Action Tapping Process
|
|
*
|
|
* FIXME: Needs doc
|
|
*/
|
|
void action_tapping_process(keyrecord_t record) {
|
|
if (process_tapping(&record)) {
|
|
if (IS_EVENT(record.event)) {
|
|
ac_dprintf("processed: ");
|
|
debug_record(record);
|
|
ac_dprintf("\n");
|
|
}
|
|
} else {
|
|
if (!waiting_buffer_enq(record)) {
|
|
// clear all in case of overflow.
|
|
ac_dprintf("OVERFLOW: CLEAR ALL STATES\n");
|
|
clear_keyboard();
|
|
waiting_buffer_clear();
|
|
tapping_key = (keyrecord_t){0};
|
|
}
|
|
}
|
|
|
|
// process waiting_buffer
|
|
if (IS_EVENT(record.event) && waiting_buffer_head != waiting_buffer_tail) {
|
|
ac_dprintf("---- action_exec: process waiting_buffer -----\n");
|
|
}
|
|
for (; waiting_buffer_tail != waiting_buffer_head; waiting_buffer_tail = (waiting_buffer_tail + 1) % WAITING_BUFFER_SIZE) {
|
|
if (process_tapping(&waiting_buffer[waiting_buffer_tail])) {
|
|
ac_dprintf("processed: waiting_buffer[%u] =", waiting_buffer_tail);
|
|
debug_record(waiting_buffer[waiting_buffer_tail]);
|
|
ac_dprintf("\n\n");
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
if (IS_EVENT(record.event)) {
|
|
ac_dprintf("\n");
|
|
} else {
|
|
# ifdef FLOW_TAP_TERM
|
|
if (!flow_tap_expired && TIMER_DIFF_16(record.event.time, flow_tap_prev_time) >= INT16_MAX / 2) {
|
|
flow_tap_expired = true;
|
|
}
|
|
# endif // FLOW_TAP_TERM
|
|
}
|
|
}
|
|
|
|
/* Some conditionally defined helper macros to keep process_tapping more
|
|
* readable. The conditional definition of tapping_keycode and all the
|
|
* conditional uses of it are hidden inside macros named TAP_...
|
|
*/
|
|
# define TAP_DEFINE_KEYCODE const uint16_t tapping_keycode = get_record_keycode(&tapping_key, false)
|
|
|
|
# if defined(AUTO_SHIFT_ENABLE) && defined(RETRO_SHIFT)
|
|
# ifdef RETRO_TAPPING_PER_KEY
|
|
# define TAP_GET_RETRO_TAPPING(keyp) get_auto_shifted_key(tapping_keycode, keyp) && get_retro_tapping(tapping_keycode, &tapping_key)
|
|
# else
|
|
# define TAP_GET_RETRO_TAPPING(keyp) get_auto_shifted_key(tapping_keycode, keyp)
|
|
# endif
|
|
/* Used to extend TAPPING_TERM:
|
|
* indefinitely if RETRO_SHIFT does not have a value
|
|
* to RETRO_SHIFT if RETRO_SHIFT is set
|
|
* for possibly retro shifted keys.
|
|
*/
|
|
# define MAYBE_RETRO_SHIFTING(ev, keyp) (get_auto_shifted_key(tapping_keycode, keyp) && TAP_GET_RETRO_TAPPING(keyp) && ((RETRO_SHIFT + 0) == 0 || TIMER_DIFF_16((ev).time, tapping_key.event.time) < (RETRO_SHIFT + 0)))
|
|
# define TAP_IS_LT IS_QK_LAYER_TAP(tapping_keycode)
|
|
# define TAP_IS_MT IS_QK_MOD_TAP(tapping_keycode)
|
|
# define TAP_IS_RETRO IS_RETRO(tapping_keycode)
|
|
# else
|
|
# define TAP_GET_RETRO_TAPPING(keyp) false
|
|
# define MAYBE_RETRO_SHIFTING(ev, kp) false
|
|
# define TAP_IS_LT false
|
|
# define TAP_IS_MT false
|
|
# define TAP_IS_RETRO false
|
|
# endif
|
|
|
|
# ifdef PERMISSIVE_HOLD_PER_KEY
|
|
# define TAP_GET_PERMISSIVE_HOLD get_permissive_hold(tapping_keycode, &tapping_key)
|
|
# elif defined(PERMISSIVE_HOLD)
|
|
# define TAP_GET_PERMISSIVE_HOLD true
|
|
# else
|
|
# define TAP_GET_PERMISSIVE_HOLD false
|
|
# endif
|
|
|
|
# ifdef HOLD_ON_OTHER_KEY_PRESS_PER_KEY
|
|
# define TAP_GET_HOLD_ON_OTHER_KEY_PRESS get_hold_on_other_key_press(tapping_keycode, &tapping_key)
|
|
# elif defined(HOLD_ON_OTHER_KEY_PRESS)
|
|
# define TAP_GET_HOLD_ON_OTHER_KEY_PRESS true
|
|
# else
|
|
# define TAP_GET_HOLD_ON_OTHER_KEY_PRESS false
|
|
# endif
|
|
|
|
/** \brief Tapping
|
|
*
|
|
* Rule: Tap key is typed(pressed and released) within TAPPING_TERM.
|
|
* (without interfering by typing other key)
|
|
*/
|
|
/* return true when key event is processed or consumed. */
|
|
bool process_tapping(keyrecord_t *keyp) {
|
|
const keyevent_t event = keyp->event;
|
|
|
|
# if defined(CHORDAL_HOLD) || defined(FLOW_TAP_TERM)
|
|
if (!event.pressed) {
|
|
const int8_t i = registered_tap_find(event.key);
|
|
if (i != -1) {
|
|
// If a tap-hold key was previously settled as tapped, set its
|
|
// tap.count correspondingly on release.
|
|
keyp->tap.count = 1;
|
|
registered_taps_del_index(i);
|
|
ac_dprintf("Found tap release for [%d]\n", i);
|
|
debug_registered_taps();
|
|
}
|
|
}
|
|
# endif // defined(CHORDAL_HOLD) || defined(FLOW_TAP_TERM)
|
|
|
|
// state machine is in the "reset" state, no tapping key is to be
|
|
// processed
|
|
if (IS_NOEVENT(tapping_key.event)) {
|
|
if (!IS_EVENT(event)) {
|
|
// early return for tick events
|
|
} else if (event.pressed && is_tap_record(keyp)) {
|
|
// the currently pressed key is a tapping key, therefore transition
|
|
// into the "pressed" tapping key state
|
|
|
|
# if defined(FLOW_TAP_TERM)
|
|
if (flow_tap_key_if_within_term(keyp, flow_tap_prev_time)) {
|
|
return true;
|
|
}
|
|
# endif // defined(FLOW_TAP_TERM)
|
|
|
|
ac_dprintf("Tapping: Start(Press tap key).\n");
|
|
tapping_key = *keyp;
|
|
process_record_tap_hint(&tapping_key);
|
|
waiting_buffer_scan_tap();
|
|
debug_tapping_key();
|
|
} else {
|
|
// the current key is just a regular key, pass it on for regular
|
|
// processing
|
|
process_record(keyp);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
# if (defined(AUTO_SHIFT_ENABLE) && defined(RETRO_SHIFT)) || defined(PERMISSIVE_HOLD_PER_KEY) || defined(CHORDAL_HOLD) || defined(HOLD_ON_OTHER_KEY_PRESS_PER_KEY)
|
|
TAP_DEFINE_KEYCODE;
|
|
# endif
|
|
|
|
// process "pressed" tapping key state
|
|
if (tapping_key.event.pressed) {
|
|
if (WITHIN_TAPPING_TERM(event) || MAYBE_RETRO_SHIFTING(event, keyp)) {
|
|
if (IS_NOEVENT(event)) {
|
|
// early return for tick events
|
|
return true;
|
|
}
|
|
|
|
if (tapping_key.tap.count == 0) {
|
|
if (IS_TAPPING_RECORD(keyp) && !event.pressed) {
|
|
// first tap!
|
|
ac_dprintf("Tapping: First tap(0->1).\n");
|
|
tapping_key.tap.count = 1;
|
|
debug_tapping_key();
|
|
process_record(&tapping_key);
|
|
|
|
// copy tapping state
|
|
keyp->tap = tapping_key.tap;
|
|
|
|
# if defined(FLOW_TAP_TERM)
|
|
// Now that tapping_key has settled as tapped, check whether
|
|
// Flow Tap applies to following yet-unsettled keys.
|
|
uint16_t prev_time = tapping_key.event.time;
|
|
for (; waiting_buffer_tail != waiting_buffer_head; waiting_buffer_tail = (waiting_buffer_tail + 1) % WAITING_BUFFER_SIZE) {
|
|
keyrecord_t *record = &waiting_buffer[waiting_buffer_tail];
|
|
if (!record->event.pressed) {
|
|
break;
|
|
}
|
|
const int16_t next_time = record->event.time;
|
|
if (!is_tap_record(record)) {
|
|
process_record(record);
|
|
} else if (!flow_tap_key_if_within_term(record, prev_time)) {
|
|
break;
|
|
}
|
|
prev_time = next_time;
|
|
}
|
|
debug_waiting_buffer();
|
|
# endif // defined(FLOW_TAP_TERM)
|
|
|
|
// enqueue
|
|
return false;
|
|
}
|
|
# if defined(CHORDAL_HOLD)
|
|
else if (is_mt_or_lt(tapping_keycode) && !event.pressed && waiting_buffer_typed(event) && !get_chordal_hold(tapping_keycode, &tapping_key, get_record_keycode(keyp, false), keyp)) {
|
|
// Key release that is not a chord with the tapping key.
|
|
// Settle the tapping key and any other pending tap-hold
|
|
// keys preceding the press of this key as tapped.
|
|
|
|
ac_dprintf("Tapping: End. Chord considered a tap\n");
|
|
tapping_key.tap.count = 1;
|
|
registered_taps_add(tapping_key.event.key);
|
|
process_record(&tapping_key);
|
|
tapping_key = (keyrecord_t){0};
|
|
|
|
waiting_buffer_chordal_hold_taps_until(event.key);
|
|
debug_registered_taps();
|
|
debug_waiting_buffer();
|
|
// enqueue
|
|
return false;
|
|
}
|
|
# endif // CHORDAL_HOLD
|
|
/* Process a key typed within TAPPING_TERM
|
|
* This can register the key before settlement of tapping,
|
|
* useful for long TAPPING_TERM but may prevent fast typing.
|
|
*/
|
|
// clang-format off
|
|
else if (
|
|
!event.pressed && waiting_buffer_typed(event) &&
|
|
(
|
|
TAP_GET_PERMISSIVE_HOLD ||
|
|
// Causes nested taps to not wait past TAPPING_TERM/RETRO_SHIFT
|
|
// unnecessarily and fixes them for Layer Taps.
|
|
TAP_GET_RETRO_TAPPING(keyp)
|
|
)
|
|
) {
|
|
// clang-format on
|
|
ac_dprintf("Tapping: End. No tap. Interfered by typing key\n");
|
|
process_record(&tapping_key);
|
|
|
|
# if defined(CHORDAL_HOLD)
|
|
uint8_t first_tap = waiting_buffer_find_chordal_hold_tap();
|
|
ac_dprintf("first_tap = %u\n", first_tap);
|
|
if (first_tap < WAITING_BUFFER_SIZE) {
|
|
for (; waiting_buffer_tail != first_tap; waiting_buffer_tail = (waiting_buffer_tail + 1) % WAITING_BUFFER_SIZE) {
|
|
ac_dprintf("Processing [%u]\n", waiting_buffer_tail);
|
|
process_record(&waiting_buffer[waiting_buffer_tail]);
|
|
}
|
|
}
|
|
|
|
waiting_buffer_chordal_hold_taps_until(event.key);
|
|
debug_registered_taps();
|
|
debug_waiting_buffer();
|
|
# endif // CHORDAL_HOLD
|
|
|
|
tapping_key = (keyrecord_t){0};
|
|
debug_tapping_key();
|
|
// enqueue
|
|
return false;
|
|
}
|
|
/* Process release event of a key pressed before tapping starts
|
|
* Without this unexpected repeating will occur with having fast repeating setting
|
|
* https://github.com/tmk/tmk_keyboard/issues/60
|
|
*
|
|
* NOTE: This workaround causes events to process out of order,
|
|
* e.g. in a rolled press of three tap-hold keys like
|
|
*
|
|
* "A down, B down, C down, A up, B up, C up"
|
|
*
|
|
* events are processed as
|
|
*
|
|
* "A down, B down, A up, B up, C down, C up"
|
|
*
|
|
* It seems incorrect to process keyp before the tapping key.
|
|
* This workaround is old, from 2013. This might no longer
|
|
* be needed for the original problem it was meant to address.
|
|
*/
|
|
else if (!event.pressed && !waiting_buffer_typed(event)) {
|
|
// Modifier/Layer should be retained till end of this tapping.
|
|
action_t action = layer_switch_get_action(event.key);
|
|
switch (action.kind.id) {
|
|
case ACT_LMODS:
|
|
case ACT_RMODS:
|
|
if (action.key.mods && !action.key.code) return false;
|
|
if (IS_MODIFIER_KEYCODE(action.key.code)) return false;
|
|
break;
|
|
case ACT_LMODS_TAP:
|
|
case ACT_RMODS_TAP:
|
|
if (action.key.mods && keyp->tap.count == 0) return false;
|
|
if (IS_MODIFIER_KEYCODE(action.key.code)) return false;
|
|
break;
|
|
case ACT_LAYER_TAP:
|
|
case ACT_LAYER_TAP_EXT:
|
|
switch (action.layer_tap.code) {
|
|
case 0 ...(OP_TAP_TOGGLE - 1):
|
|
case OP_ON_OFF:
|
|
case OP_OFF_ON:
|
|
case OP_SET_CLEAR:
|
|
return false;
|
|
}
|
|
break;
|
|
}
|
|
// Release of key should be process immediately.
|
|
ac_dprintf("Tapping: release event of a key pressed before tapping\n");
|
|
process_record(keyp);
|
|
return true;
|
|
} else {
|
|
// set interrupted flag when other key pressed during tapping
|
|
if (event.pressed) {
|
|
tapping_key.tap.interrupted = true;
|
|
|
|
# if defined(CHORDAL_HOLD)
|
|
if (is_mt_or_lt(tapping_keycode) && !get_chordal_hold(tapping_keycode, &tapping_key, get_record_keycode(keyp, false), keyp)) {
|
|
// In process_action(), HOLD_ON_OTHER_KEY_PRESS
|
|
// will revert interrupted events to holds, so
|
|
// this needs to be set false.
|
|
tapping_key.tap.interrupted = false;
|
|
|
|
if (!is_tap_record(keyp)) {
|
|
ac_dprintf("Tapping: End. Chord considered a tap\n");
|
|
tapping_key.tap.count = 1;
|
|
registered_taps_add(tapping_key.event.key);
|
|
debug_registered_taps();
|
|
process_record(&tapping_key);
|
|
tapping_key = (keyrecord_t){0};
|
|
}
|
|
} else
|
|
# endif // CHORDAL_HOLD
|
|
if (TAP_GET_HOLD_ON_OTHER_KEY_PRESS
|
|
# if defined(AUTO_SHIFT_ENABLE) && defined(RETRO_SHIFT)
|
|
// Auto Shift cannot evaluate this early
|
|
// Retro Shift uses the hold action for all nested taps even without HOLD_ON_OTHER_KEY_PRESS, so this is fine to skip
|
|
&& !(MAYBE_RETRO_SHIFTING(event, keyp) && get_auto_shifted_key(get_record_keycode(keyp, false), keyp))
|
|
# endif
|
|
) {
|
|
// Settle the tapping key as *held*, since
|
|
// HOLD_ON_OTHER_KEY_PRESS is enabled for this key.
|
|
ac_dprintf("Tapping: End. No tap. Interfered by pressed key\n");
|
|
process_record(&tapping_key);
|
|
|
|
# if defined(CHORDAL_HOLD)
|
|
if (waiting_buffer_tail != waiting_buffer_head && is_tap_record(&waiting_buffer[waiting_buffer_tail])) {
|
|
tapping_key = waiting_buffer[waiting_buffer_tail];
|
|
// Pop tail from the queue.
|
|
waiting_buffer_tail = (waiting_buffer_tail + 1) % WAITING_BUFFER_SIZE;
|
|
debug_waiting_buffer();
|
|
} else
|
|
# endif // CHORDAL_HOLD
|
|
{
|
|
tapping_key = (keyrecord_t){0};
|
|
}
|
|
debug_tapping_key();
|
|
|
|
# if defined(CHORDAL_HOLD)
|
|
waiting_buffer_process_regular();
|
|
# endif // CHORDAL_HOLD
|
|
}
|
|
}
|
|
// enqueue
|
|
return false;
|
|
}
|
|
}
|
|
// tap_count > 0
|
|
else {
|
|
if (IS_TAPPING_RECORD(keyp) && !event.pressed) {
|
|
ac_dprintf("Tapping: Tap release(%u)\n", tapping_key.tap.count);
|
|
keyp->tap = tapping_key.tap;
|
|
process_record(keyp);
|
|
tapping_key = *keyp;
|
|
debug_tapping_key();
|
|
return true;
|
|
} else if (is_tap_record(keyp) && event.pressed) {
|
|
if (tapping_key.tap.count > 1) {
|
|
ac_dprintf("Tapping: Start new tap with releasing last tap(>1).\n");
|
|
// unregister key
|
|
process_record(&(keyrecord_t){
|
|
.tap = tapping_key.tap,
|
|
.event.key = tapping_key.event.key,
|
|
.event.time = event.time,
|
|
.event.pressed = false,
|
|
.event.type = tapping_key.event.type,
|
|
# ifdef COMBO_ENABLE
|
|
.keycode = tapping_key.keycode,
|
|
# endif
|
|
});
|
|
} else {
|
|
ac_dprintf("Tapping: Start while last tap(1).\n");
|
|
}
|
|
tapping_key = *keyp;
|
|
waiting_buffer_scan_tap();
|
|
debug_tapping_key();
|
|
return true;
|
|
} else {
|
|
ac_dprintf("Tapping: key event while last tap(>0).\n");
|
|
# if defined(AUTO_SHIFT_ENABLE) && defined(RETRO_SHIFT)
|
|
retroshift_swap_times();
|
|
# endif
|
|
process_record(keyp);
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
// after TAPPING_TERM
|
|
else {
|
|
if (tapping_key.tap.count == 0) {
|
|
ac_dprintf("Tapping: End. Timeout. Not tap(0): ");
|
|
debug_event(event);
|
|
ac_dprintf("\n");
|
|
process_record(&tapping_key);
|
|
tapping_key = (keyrecord_t){0};
|
|
debug_tapping_key();
|
|
return false;
|
|
} else {
|
|
if (IS_NOEVENT(event)) {
|
|
return true;
|
|
}
|
|
if (IS_TAPPING_RECORD(keyp) && !event.pressed) {
|
|
ac_dprintf("Tapping: End. last timeout tap release(>0).");
|
|
keyp->tap = tapping_key.tap;
|
|
process_record(keyp);
|
|
tapping_key = (keyrecord_t){0};
|
|
return true;
|
|
} else if (is_tap_record(keyp) && event.pressed) {
|
|
if (tapping_key.tap.count > 1) {
|
|
ac_dprintf("Tapping: Start new tap with releasing last timeout tap(>1).\n");
|
|
// unregister key
|
|
process_record(&(keyrecord_t){
|
|
.tap = tapping_key.tap,
|
|
.event.key = tapping_key.event.key,
|
|
.event.time = event.time,
|
|
.event.pressed = false,
|
|
.event.type = tapping_key.event.type,
|
|
# ifdef COMBO_ENABLE
|
|
.keycode = tapping_key.keycode,
|
|
# endif
|
|
});
|
|
} else {
|
|
ac_dprintf("Tapping: Start while last timeout tap(1).\n");
|
|
}
|
|
tapping_key = *keyp;
|
|
waiting_buffer_scan_tap();
|
|
debug_tapping_key();
|
|
return true;
|
|
} else {
|
|
ac_dprintf("Tapping: key event while last timeout tap(>0).\n");
|
|
process_record(keyp);
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// process "released" tapping key state
|
|
else {
|
|
if (WITHIN_TAPPING_TERM(event) || MAYBE_RETRO_SHIFTING(event, keyp)) {
|
|
if (IS_NOEVENT(event)) {
|
|
// early return for tick events
|
|
return true;
|
|
}
|
|
if (event.pressed) {
|
|
if (IS_TAPPING_RECORD(keyp)) {
|
|
if (WITHIN_QUICK_TAP_TERM(event) && !tapping_key.tap.interrupted && tapping_key.tap.count > 0) {
|
|
// sequential tap.
|
|
keyp->tap = tapping_key.tap;
|
|
if (keyp->tap.count < 15) keyp->tap.count += 1;
|
|
ac_dprintf("Tapping: Tap press(%u)\n", keyp->tap.count);
|
|
process_record(keyp);
|
|
tapping_key = *keyp;
|
|
debug_tapping_key();
|
|
return true;
|
|
}
|
|
// FIX: start new tap again
|
|
tapping_key = *keyp;
|
|
return true;
|
|
} else if (is_tap_record(keyp)) {
|
|
// Sequential tap can be interfered with other tap key.
|
|
# if defined(FLOW_TAP_TERM)
|
|
if (flow_tap_key_if_within_term(keyp, flow_tap_prev_time)) {
|
|
tapping_key = (keyrecord_t){0};
|
|
debug_tapping_key();
|
|
return true;
|
|
}
|
|
# endif // defined(FLOW_TAP_TERM)
|
|
ac_dprintf("Tapping: Start with interfering other tap.\n");
|
|
tapping_key = *keyp;
|
|
waiting_buffer_scan_tap();
|
|
debug_tapping_key();
|
|
return true;
|
|
} else {
|
|
// should none in buffer
|
|
// FIX: interrupted when other key is pressed
|
|
tapping_key.tap.interrupted = true;
|
|
process_record(keyp);
|
|
return true;
|
|
}
|
|
} else {
|
|
ac_dprintf("Tapping: other key just after tap.\n");
|
|
process_record(keyp);
|
|
return true;
|
|
}
|
|
} else {
|
|
// Timeout - reset state machine.
|
|
ac_dprintf("Tapping: End(Timeout after releasing last tap): ");
|
|
debug_event(event);
|
|
ac_dprintf("\n");
|
|
tapping_key = (keyrecord_t){0};
|
|
debug_tapping_key();
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
/** \brief Waiting buffer enq
|
|
*
|
|
* FIXME: Needs docs
|
|
*/
|
|
bool waiting_buffer_enq(keyrecord_t record) {
|
|
if (IS_NOEVENT(record.event)) {
|
|
return true;
|
|
}
|
|
|
|
if ((waiting_buffer_head + 1) % WAITING_BUFFER_SIZE == waiting_buffer_tail) {
|
|
ac_dprintf("waiting_buffer_enq: Over flow.\n");
|
|
return false;
|
|
}
|
|
|
|
waiting_buffer[waiting_buffer_head] = record;
|
|
waiting_buffer_head = (waiting_buffer_head + 1) % WAITING_BUFFER_SIZE;
|
|
|
|
ac_dprintf("waiting_buffer_enq: ");
|
|
debug_waiting_buffer();
|
|
return true;
|
|
}
|
|
|
|
/** \brief Waiting buffer clear
|
|
*
|
|
* FIXME: Needs docs
|
|
*/
|
|
void waiting_buffer_clear(void) {
|
|
waiting_buffer_head = 0;
|
|
waiting_buffer_tail = 0;
|
|
}
|
|
|
|
/** \brief Waiting buffer typed
|
|
*
|
|
* FIXME: Needs docs
|
|
*/
|
|
bool waiting_buffer_typed(keyevent_t event) {
|
|
for (uint8_t i = waiting_buffer_tail; i != waiting_buffer_head; i = (i + 1) % WAITING_BUFFER_SIZE) {
|
|
if (KEYEQ(event.key, waiting_buffer[i].event.key) && event.pressed != waiting_buffer[i].event.pressed) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/** \brief Waiting buffer has anykey pressed
|
|
*
|
|
* FIXME: Needs docs
|
|
*/
|
|
__attribute__((unused)) bool waiting_buffer_has_anykey_pressed(void) {
|
|
for (uint8_t i = waiting_buffer_tail; i != waiting_buffer_head; i = (i + 1) % WAITING_BUFFER_SIZE) {
|
|
if (waiting_buffer[i].event.pressed) return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/** \brief Scan buffer for tapping
|
|
*
|
|
* FIXME: Needs docs
|
|
*/
|
|
void waiting_buffer_scan_tap(void) {
|
|
// early return if:
|
|
// - tapping already is settled
|
|
// - invalid state: tapping_key released && tap.count == 0
|
|
if ((tapping_key.tap.count > 0) || !tapping_key.event.pressed) {
|
|
return;
|
|
}
|
|
|
|
# if (defined(AUTO_SHIFT_ENABLE) && defined(RETRO_SHIFT))
|
|
TAP_DEFINE_KEYCODE;
|
|
# endif
|
|
for (uint8_t i = waiting_buffer_tail; i != waiting_buffer_head; i = (i + 1) % WAITING_BUFFER_SIZE) {
|
|
keyrecord_t *candidate = &waiting_buffer[i];
|
|
// clang-format off
|
|
if (IS_EVENT(candidate->event) && KEYEQ(candidate->event.key, tapping_key.event.key) && !candidate->event.pressed && (
|
|
WITHIN_TAPPING_TERM(waiting_buffer[i].event) || MAYBE_RETRO_SHIFTING(waiting_buffer[i].event, &tapping_key)
|
|
)) {
|
|
// clang-format on
|
|
tapping_key.tap.count = 1;
|
|
candidate->tap.count = 1;
|
|
process_record(&tapping_key);
|
|
|
|
ac_dprintf("waiting_buffer_scan_tap: found at [%u]\n", i);
|
|
debug_waiting_buffer();
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
# if defined(CHORDAL_HOLD) || defined(FLOW_TAP_TERM)
|
|
static void registered_taps_add(keypos_t key) {
|
|
if (num_registered_taps >= REGISTERED_TAPS_SIZE) {
|
|
ac_dprintf("TAPS OVERFLOW: CLEAR ALL STATES\n");
|
|
clear_keyboard();
|
|
num_registered_taps = 0;
|
|
}
|
|
|
|
registered_taps[num_registered_taps] = key;
|
|
++num_registered_taps;
|
|
}
|
|
|
|
static int8_t registered_tap_find(keypos_t key) {
|
|
for (int8_t i = 0; i < num_registered_taps; ++i) {
|
|
if (KEYEQ(registered_taps[i], key)) {
|
|
return i;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
static void registered_taps_del_index(uint8_t i) {
|
|
if (i < num_registered_taps) {
|
|
--num_registered_taps;
|
|
if (i < num_registered_taps) {
|
|
registered_taps[i] = registered_taps[num_registered_taps];
|
|
}
|
|
}
|
|
}
|
|
|
|
static void debug_registered_taps(void) {
|
|
ac_dprintf("registered_taps = { ");
|
|
for (int8_t i = 0; i < num_registered_taps; ++i) {
|
|
ac_dprintf("%02X%02X ", registered_taps[i].row, registered_taps[i].col);
|
|
}
|
|
ac_dprintf("}\n");
|
|
}
|
|
|
|
# endif // defined(CHORDAL_HOLD) || defined(FLOW_TAP_TERM)
|
|
|
|
# ifdef CHORDAL_HOLD
|
|
__attribute__((weak)) bool get_chordal_hold(uint16_t tap_hold_keycode, keyrecord_t *tap_hold_record, uint16_t other_keycode, keyrecord_t *other_record) {
|
|
return get_chordal_hold_default(tap_hold_record, other_record);
|
|
}
|
|
|
|
bool get_chordal_hold_default(keyrecord_t *tap_hold_record, keyrecord_t *other_record) {
|
|
if (tap_hold_record->event.type != KEY_EVENT || other_record->event.type != KEY_EVENT) {
|
|
return true; // Return true on combos or other non-key events.
|
|
}
|
|
|
|
char tap_hold_hand = chordal_hold_handedness(tap_hold_record->event.key);
|
|
if (tap_hold_hand == '*') {
|
|
return true;
|
|
}
|
|
char other_hand = chordal_hold_handedness(other_record->event.key);
|
|
return other_hand == '*' || tap_hold_hand != other_hand;
|
|
}
|
|
|
|
__attribute__((weak)) char chordal_hold_handedness(keypos_t key) {
|
|
return (char)pgm_read_byte(&chordal_hold_layout[key.row][key.col]);
|
|
}
|
|
|
|
static uint8_t waiting_buffer_find_chordal_hold_tap(void) {
|
|
keyrecord_t *prev = &tapping_key;
|
|
uint16_t prev_keycode = get_record_keycode(&tapping_key, false);
|
|
uint8_t first_tap = WAITING_BUFFER_SIZE;
|
|
for (uint8_t i = waiting_buffer_tail; i != waiting_buffer_head; i = (i + 1) % WAITING_BUFFER_SIZE) {
|
|
keyrecord_t * cur = &waiting_buffer[i];
|
|
const uint16_t cur_keycode = get_record_keycode(cur, false);
|
|
if (!cur->event.pressed || !is_mt_or_lt(prev_keycode)) {
|
|
break;
|
|
} else if (get_chordal_hold(prev_keycode, prev, cur_keycode, cur)) {
|
|
first_tap = i; // Track one index past the latest hold.
|
|
}
|
|
prev = cur;
|
|
prev_keycode = cur_keycode;
|
|
}
|
|
return first_tap;
|
|
}
|
|
|
|
static void waiting_buffer_chordal_hold_taps_until(keypos_t key) {
|
|
while (waiting_buffer_tail != waiting_buffer_head) {
|
|
keyrecord_t *record = &waiting_buffer[waiting_buffer_tail];
|
|
ac_dprintf("waiting_buffer_chordal_hold_taps_until: processing [%u]\n", waiting_buffer_tail);
|
|
if (record->event.pressed && is_tap_record(record)) {
|
|
record->tap.count = 1;
|
|
registered_taps_add(record->event.key);
|
|
}
|
|
process_record(record);
|
|
waiting_buffer_tail = (waiting_buffer_tail + 1) % WAITING_BUFFER_SIZE;
|
|
|
|
if (KEYEQ(key, record->event.key) && record->event.pressed) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void waiting_buffer_process_regular(void) {
|
|
for (; waiting_buffer_tail != waiting_buffer_head; waiting_buffer_tail = (waiting_buffer_tail + 1) % WAITING_BUFFER_SIZE) {
|
|
if (is_tap_record(&waiting_buffer[waiting_buffer_tail])) {
|
|
break; // Stop once a tap-hold key event is reached.
|
|
}
|
|
ac_dprintf("waiting_buffer_process_regular: processing [%u]\n", waiting_buffer_tail);
|
|
process_record(&waiting_buffer[waiting_buffer_tail]);
|
|
}
|
|
debug_waiting_buffer();
|
|
}
|
|
# endif // CHORDAL_HOLD
|
|
|
|
# ifdef FLOW_TAP_TERM
|
|
void flow_tap_update_last_event(keyrecord_t *record) {
|
|
const uint16_t keycode = get_record_keycode(record, false);
|
|
// Don't update while a tap-hold key is unsettled.
|
|
if (record->tap.count == 0 && (waiting_buffer_tail != waiting_buffer_head || (tapping_key.event.pressed && tapping_key.tap.count == 0))) {
|
|
return;
|
|
}
|
|
// Ignore releases of modifiers and held layer switches.
|
|
if (!record->event.pressed) {
|
|
switch (keycode) {
|
|
case MODIFIER_KEYCODE_RANGE:
|
|
case QK_MOMENTARY ... QK_MOMENTARY_MAX:
|
|
case QK_LAYER_TAP_TOGGLE ... QK_LAYER_TAP_TOGGLE_MAX:
|
|
# ifndef NO_ACTION_ONESHOT // Ignore one-shot keys.
|
|
case QK_ONE_SHOT_MOD ... QK_ONE_SHOT_MOD_MAX:
|
|
case QK_ONE_SHOT_LAYER ... QK_ONE_SHOT_LAYER_MAX:
|
|
# endif // NO_ACTION_ONESHOT
|
|
# ifdef TRI_LAYER_ENABLE // Ignore Tri Layer keys.
|
|
case QK_TRI_LAYER_LOWER:
|
|
case QK_TRI_LAYER_UPPER:
|
|
# endif // TRI_LAYER_ENABLE
|
|
return;
|
|
case QK_MODS ... QK_MODS_MAX:
|
|
if (QK_MODS_GET_BASIC_KEYCODE(keycode) == KC_NO) {
|
|
return;
|
|
}
|
|
break;
|
|
case QK_MOD_TAP ... QK_MOD_TAP_MAX:
|
|
case QK_LAYER_TAP ... QK_LAYER_TAP_MAX:
|
|
if (record->tap.count == 0) {
|
|
return;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
flow_tap_prev_keycode = keycode;
|
|
flow_tap_prev_time = record->event.time;
|
|
flow_tap_expired = false;
|
|
}
|
|
|
|
static bool flow_tap_key_if_within_term(keyrecord_t *record, uint16_t prev_time) {
|
|
const uint16_t idle_time = TIMER_DIFF_16(record->event.time, prev_time);
|
|
if (flow_tap_expired || idle_time >= 500) {
|
|
return false;
|
|
}
|
|
|
|
const uint16_t keycode = get_record_keycode(record, false);
|
|
if (is_mt_or_lt(keycode)) {
|
|
uint16_t term = get_flow_tap_term(keycode, record, flow_tap_prev_keycode);
|
|
if (term > 500) {
|
|
term = 500;
|
|
}
|
|
if (idle_time < term) {
|
|
debug_event(record->event);
|
|
ac_dprintf(" within flow tap term (%u < %u) considered a tap\n", idle_time, term);
|
|
record->tap.count = 1;
|
|
registered_taps_add(record->event.key);
|
|
debug_registered_taps();
|
|
process_record(record);
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// By default, enable Flow Tap for the keys in the main alphas area and Space.
|
|
// This should work reasonably even if the layout is remapped on the host to an
|
|
// alt layout or international layout (e.g. Dvorak or AZERTY), where these same
|
|
// key positions are mostly used for typing letters.
|
|
__attribute__((weak)) bool is_flow_tap_key(uint16_t keycode) {
|
|
if ((get_mods() & (MOD_MASK_CG | MOD_BIT_LALT)) != 0) {
|
|
return false; // Disable Flow Tap on hotkeys.
|
|
}
|
|
switch (get_tap_keycode(keycode)) {
|
|
case KC_SPC:
|
|
case KC_A ... KC_Z:
|
|
case KC_DOT:
|
|
case KC_COMM:
|
|
case KC_SCLN:
|
|
case KC_SLSH:
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
__attribute__((weak)) uint16_t get_flow_tap_term(uint16_t keycode, keyrecord_t *record, uint16_t prev_keycode) {
|
|
if (is_flow_tap_key(keycode) && is_flow_tap_key(prev_keycode)) {
|
|
return FLOW_TAP_TERM;
|
|
}
|
|
return 0;
|
|
}
|
|
# endif // FLOW_TAP_TERM
|
|
|
|
/** \brief Logs tapping key if ACTION_DEBUG is enabled. */
|
|
static void debug_tapping_key(void) {
|
|
ac_dprintf("TAPPING_KEY=");
|
|
debug_record(tapping_key);
|
|
ac_dprintf("\n");
|
|
}
|
|
|
|
/** \brief Logs waiting buffer if ACTION_DEBUG is enabled. */
|
|
static void debug_waiting_buffer(void) {
|
|
ac_dprintf("{");
|
|
for (uint8_t i = waiting_buffer_tail; i != waiting_buffer_head; i = (i + 1) % WAITING_BUFFER_SIZE) {
|
|
ac_dprintf(" [%u]=", i);
|
|
debug_record(waiting_buffer[i]);
|
|
}
|
|
ac_dprintf("}\n");
|
|
}
|
|
|
|
#endif
|