mirror of
https://github.com/zsa/qmk_firmware.git
synced 2026-04-18 06:22:14 +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>
1477 lines
49 KiB
C
1477 lines
49 KiB
C
/* Copyright 2016-2017 Yang Liu
|
|
*
|
|
* 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 <math.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include "progmem.h"
|
|
#include "sync_timer.h"
|
|
#include "rgblight.h"
|
|
#include "color.h"
|
|
#include "debug.h"
|
|
#include "util.h"
|
|
#include "led_tables.h"
|
|
#include <lib/lib8tion/lib8tion.h>
|
|
#include "eeconfig.h"
|
|
|
|
#ifdef RGBLIGHT_SPLIT
|
|
/* for split keyboard */
|
|
# define RGBLIGHT_SPLIT_SET_CHANGE_MODE rgblight_status.change_flags |= RGBLIGHT_STATUS_CHANGE_MODE
|
|
# define RGBLIGHT_SPLIT_SET_CHANGE_HSVS rgblight_status.change_flags |= RGBLIGHT_STATUS_CHANGE_HSVS
|
|
# define RGBLIGHT_SPLIT_SET_CHANGE_MODEHSVS rgblight_status.change_flags |= (RGBLIGHT_STATUS_CHANGE_MODE | RGBLIGHT_STATUS_CHANGE_HSVS)
|
|
# define RGBLIGHT_SPLIT_SET_CHANGE_LAYERS rgblight_status.change_flags |= RGBLIGHT_STATUS_CHANGE_LAYERS
|
|
# define RGBLIGHT_SPLIT_SET_CHANGE_TIMER_ENABLE rgblight_status.change_flags |= RGBLIGHT_STATUS_CHANGE_TIMER
|
|
# define RGBLIGHT_SPLIT_ANIMATION_TICK rgblight_status.change_flags |= RGBLIGHT_STATUS_ANIMATION_TICK
|
|
#else
|
|
# define RGBLIGHT_SPLIT_SET_CHANGE_MODE
|
|
# define RGBLIGHT_SPLIT_SET_CHANGE_HSVS
|
|
# define RGBLIGHT_SPLIT_SET_CHANGE_MODEHSVS
|
|
# define RGBLIGHT_SPLIT_SET_CHANGE_LAYERS
|
|
# define RGBLIGHT_SPLIT_SET_CHANGE_TIMER_ENABLE
|
|
# define RGBLIGHT_SPLIT_ANIMATION_TICK
|
|
#endif
|
|
|
|
#define _RGBM_SINGLE_STATIC(sym) RGBLIGHT_MODE_##sym,
|
|
#define _RGBM_SINGLE_DYNAMIC(sym)
|
|
#define _RGBM_MULTI_STATIC(sym) RGBLIGHT_MODE_##sym,
|
|
#define _RGBM_MULTI_DYNAMIC(sym)
|
|
#define _RGBM_TMP_STATIC(sym, msym) RGBLIGHT_MODE_##sym,
|
|
#define _RGBM_TMP_DYNAMIC(sym, msym)
|
|
static uint8_t static_effect_table[] = {
|
|
#include "rgblight_modes.h"
|
|
};
|
|
|
|
#define _RGBM_SINGLE_STATIC(sym) RGBLIGHT_MODE_##sym,
|
|
#define _RGBM_SINGLE_DYNAMIC(sym) RGBLIGHT_MODE_##sym,
|
|
#define _RGBM_MULTI_STATIC(sym) RGBLIGHT_MODE_##sym,
|
|
#define _RGBM_MULTI_DYNAMIC(sym) RGBLIGHT_MODE_##sym,
|
|
#define _RGBM_TMP_STATIC(sym, msym) RGBLIGHT_MODE_##msym,
|
|
#define _RGBM_TMP_DYNAMIC(sym, msym) RGBLIGHT_MODE_##msym,
|
|
static uint8_t mode_base_table[] = {
|
|
0, // RGBLIGHT_MODE_zero
|
|
#include "rgblight_modes.h"
|
|
};
|
|
|
|
#if !defined(RGBLIGHT_DEFAULT_MODE)
|
|
# define RGBLIGHT_DEFAULT_MODE RGBLIGHT_MODE_STATIC_LIGHT
|
|
#endif
|
|
|
|
#if !defined(RGBLIGHT_DEFAULT_HUE)
|
|
# define RGBLIGHT_DEFAULT_HUE 0
|
|
#endif
|
|
|
|
#if !defined(RGBLIGHT_DEFAULT_SAT)
|
|
# define RGBLIGHT_DEFAULT_SAT UINT8_MAX
|
|
#endif
|
|
|
|
#if !defined(RGBLIGHT_DEFAULT_VAL)
|
|
# define RGBLIGHT_DEFAULT_VAL RGBLIGHT_LIMIT_VAL
|
|
#endif
|
|
|
|
#if !defined(RGBLIGHT_DEFAULT_SPD)
|
|
# define RGBLIGHT_DEFAULT_SPD 0
|
|
#endif
|
|
|
|
#if !defined(RGBLIGHT_DEFAULT_ON)
|
|
# define RGBLIGHT_DEFAULT_ON true
|
|
#endif
|
|
|
|
static inline int is_static_effect(uint8_t mode) {
|
|
return memchr(static_effect_table, mode, sizeof(static_effect_table)) != NULL;
|
|
}
|
|
|
|
#ifdef RGBLIGHT_LED_MAP
|
|
const uint8_t led_map[] PROGMEM = RGBLIGHT_LED_MAP;
|
|
#endif
|
|
|
|
#ifdef RGBLIGHT_EFFECT_STATIC_GRADIENT
|
|
__attribute__((weak)) const uint8_t RGBLED_GRADIENT_RANGES[] PROGMEM = {255, 170, 127, 85, 64};
|
|
#endif
|
|
|
|
rgblight_config_t rgblight_config;
|
|
rgblight_status_t rgblight_status = {.timer_enabled = false};
|
|
bool is_rgblight_initialized = false;
|
|
|
|
#ifdef RGBLIGHT_SLEEP
|
|
static bool is_suspended;
|
|
static bool pre_suspend_enabled;
|
|
#endif
|
|
|
|
#ifdef RGBLIGHT_USE_TIMER
|
|
animation_status_t animation_status = {};
|
|
#endif
|
|
|
|
#ifdef RGBLIGHT_LAYERS
|
|
rgblight_segment_t const *const *rgblight_layers = NULL;
|
|
|
|
static bool deferred_set_layer_state = false;
|
|
#endif
|
|
|
|
rgblight_ranges_t rgblight_ranges = {0, RGBLIGHT_LED_COUNT, 0, RGBLIGHT_LED_COUNT, RGBLIGHT_LED_COUNT};
|
|
|
|
void rgblight_set_clipping_range(uint8_t start_pos, uint8_t num_leds) {
|
|
rgblight_ranges.clipping_start_pos = start_pos;
|
|
rgblight_ranges.clipping_num_leds = num_leds;
|
|
}
|
|
|
|
void rgblight_set_effect_range(uint8_t start_pos, uint8_t num_leds) {
|
|
if (start_pos >= RGBLIGHT_LED_COUNT) return;
|
|
if (start_pos + num_leds > RGBLIGHT_LED_COUNT) return;
|
|
rgblight_ranges.effect_start_pos = start_pos;
|
|
rgblight_ranges.effect_end_pos = start_pos + num_leds;
|
|
rgblight_ranges.effect_num_leds = num_leds;
|
|
}
|
|
|
|
__attribute__((weak)) rgb_t rgblight_hsv_to_rgb(hsv_t hsv) {
|
|
return hsv_to_rgb(hsv);
|
|
}
|
|
|
|
uint8_t rgblight_led_index(uint8_t index) {
|
|
#if defined(RGBLIGHT_LED_MAP)
|
|
return pgm_read_byte(&led_map[index]) - rgblight_ranges.clipping_start_pos;
|
|
#else
|
|
return index - rgblight_ranges.clipping_start_pos;
|
|
#endif
|
|
}
|
|
|
|
void setrgb(uint8_t r, uint8_t g, uint8_t b, int index) {
|
|
rgblight_driver.set_color(rgblight_led_index(index), r, g, b);
|
|
}
|
|
|
|
void sethsv_raw(uint8_t hue, uint8_t sat, uint8_t val, int index) {
|
|
hsv_t hsv = {hue, sat, val};
|
|
rgb_t rgb = rgblight_hsv_to_rgb(hsv);
|
|
setrgb(rgb.r, rgb.g, rgb.b, index);
|
|
}
|
|
|
|
void sethsv(uint8_t hue, uint8_t sat, uint8_t val, int index) {
|
|
sethsv_raw(hue, sat, val > RGBLIGHT_LIMIT_VAL ? RGBLIGHT_LIMIT_VAL : val, index);
|
|
}
|
|
|
|
void rgblight_check_config(void) {
|
|
/* Add some out of bound checks for RGB light config */
|
|
|
|
if (rgblight_config.mode < RGBLIGHT_MODE_STATIC_LIGHT) {
|
|
rgblight_config.mode = RGBLIGHT_MODE_STATIC_LIGHT;
|
|
} else if (rgblight_config.mode > RGBLIGHT_MODES) {
|
|
rgblight_config.mode = RGBLIGHT_MODES;
|
|
}
|
|
|
|
if (rgblight_config.val > RGBLIGHT_LIMIT_VAL) {
|
|
rgblight_config.val = RGBLIGHT_LIMIT_VAL;
|
|
}
|
|
}
|
|
|
|
void eeconfig_update_rgblight_current(void) {
|
|
rgblight_check_config();
|
|
eeconfig_update_rgblight(&rgblight_config);
|
|
}
|
|
|
|
void eeconfig_update_rgblight_default(void) {
|
|
rgblight_config.enable = RGBLIGHT_DEFAULT_ON;
|
|
rgblight_config.velocikey = 0;
|
|
rgblight_config.mode = RGBLIGHT_DEFAULT_MODE;
|
|
rgblight_config.hue = RGBLIGHT_DEFAULT_HUE;
|
|
rgblight_config.sat = RGBLIGHT_DEFAULT_SAT;
|
|
rgblight_config.val = RGBLIGHT_DEFAULT_VAL;
|
|
rgblight_config.speed = RGBLIGHT_DEFAULT_SPD;
|
|
RGBLIGHT_SPLIT_SET_CHANGE_MODEHSVS;
|
|
eeconfig_update_rgblight(&rgblight_config);
|
|
}
|
|
|
|
void eeconfig_debug_rgblight(void) {
|
|
dprintf("rgblight_config EEPROM:\n");
|
|
dprintf("rgblight_config.enable = %d\n", rgblight_config.enable);
|
|
dprintf("rgblight_config.velocikey = %d\n", rgblight_config.velocikey);
|
|
dprintf("rghlight_config.mode = %d\n", rgblight_config.mode);
|
|
dprintf("rgblight_config.hue = %d\n", rgblight_config.hue);
|
|
dprintf("rgblight_config.sat = %d\n", rgblight_config.sat);
|
|
dprintf("rgblight_config.val = %d\n", rgblight_config.val);
|
|
dprintf("rgblight_config.speed = %d\n", rgblight_config.speed);
|
|
}
|
|
|
|
void rgblight_init(void) {
|
|
/* if already initialized, don't do it again.
|
|
If you must do it again, extern this and set to false, first.
|
|
This is a dirty, dirty hack until proper hooks can be added for keyboard startup. */
|
|
if (is_rgblight_initialized) {
|
|
return;
|
|
}
|
|
|
|
dprintf("rgblight_init start!\n");
|
|
eeconfig_read_rgblight(&rgblight_config);
|
|
RGBLIGHT_SPLIT_SET_CHANGE_MODEHSVS;
|
|
if (!rgblight_config.mode) {
|
|
dprintf("rgblight_init rgblight_config.mode = 0. Write default values to EEPROM.\n");
|
|
eeconfig_update_rgblight_default();
|
|
eeconfig_read_rgblight(&rgblight_config);
|
|
}
|
|
rgblight_check_config();
|
|
|
|
eeconfig_debug_rgblight(); // display current eeprom values
|
|
|
|
rgblight_timer_init(); // setup the timer
|
|
|
|
rgblight_driver.init();
|
|
|
|
if (rgblight_config.enable) {
|
|
rgblight_mode_noeeprom(rgblight_config.mode);
|
|
}
|
|
|
|
is_rgblight_initialized = true;
|
|
}
|
|
|
|
void rgblight_reload_from_eeprom(void) {
|
|
/* Reset back to what we have in eeprom */
|
|
eeconfig_read_rgblight(&rgblight_config);
|
|
RGBLIGHT_SPLIT_SET_CHANGE_MODEHSVS;
|
|
rgblight_check_config();
|
|
eeconfig_debug_rgblight(); // display current eeprom values
|
|
if (rgblight_config.enable) {
|
|
rgblight_mode_noeeprom(rgblight_config.mode);
|
|
}
|
|
}
|
|
|
|
uint64_t rgblight_read_qword(void) {
|
|
return rgblight_config.raw;
|
|
}
|
|
|
|
void rgblight_update_qword(uint64_t qword) {
|
|
RGBLIGHT_SPLIT_SET_CHANGE_MODEHSVS;
|
|
rgblight_config.raw = qword;
|
|
if (rgblight_config.enable)
|
|
rgblight_mode_noeeprom(rgblight_config.mode);
|
|
else {
|
|
rgblight_timer_disable();
|
|
rgblight_set();
|
|
}
|
|
}
|
|
|
|
void rgblight_increase(void) {
|
|
uint8_t mode = 0;
|
|
if (rgblight_config.mode < RGBLIGHT_MODES) {
|
|
mode = rgblight_config.mode + 1;
|
|
}
|
|
rgblight_mode(mode);
|
|
}
|
|
void rgblight_decrease(void) {
|
|
uint8_t mode = 0;
|
|
// Mode will never be < 1. If it ever is, eeprom needs to be initialized.
|
|
if (rgblight_config.mode > RGBLIGHT_MODE_STATIC_LIGHT) {
|
|
mode = rgblight_config.mode - 1;
|
|
}
|
|
rgblight_mode(mode);
|
|
}
|
|
void rgblight_step_helper(bool write_to_eeprom) {
|
|
uint8_t mode = 0;
|
|
mode = rgblight_config.mode + 1;
|
|
if (mode > RGBLIGHT_MODES) {
|
|
mode = 1;
|
|
}
|
|
rgblight_mode_eeprom_helper(mode, write_to_eeprom);
|
|
}
|
|
void rgblight_step_noeeprom(void) {
|
|
rgblight_step_helper(false);
|
|
}
|
|
void rgblight_step(void) {
|
|
rgblight_step_helper(true);
|
|
}
|
|
void rgblight_step_reverse_helper(bool write_to_eeprom) {
|
|
uint8_t mode = 0;
|
|
mode = rgblight_config.mode - 1;
|
|
if (mode < 1) {
|
|
mode = RGBLIGHT_MODES;
|
|
}
|
|
rgblight_mode_eeprom_helper(mode, write_to_eeprom);
|
|
}
|
|
void rgblight_step_reverse_noeeprom(void) {
|
|
rgblight_step_reverse_helper(false);
|
|
}
|
|
void rgblight_step_reverse(void) {
|
|
rgblight_step_reverse_helper(true);
|
|
}
|
|
|
|
uint8_t rgblight_get_mode(void) {
|
|
if (!rgblight_config.enable) {
|
|
return false;
|
|
}
|
|
|
|
return rgblight_config.mode;
|
|
}
|
|
|
|
void rgblight_mode_eeprom_helper(uint8_t mode, bool write_to_eeprom) {
|
|
if (!rgblight_config.enable) {
|
|
return;
|
|
}
|
|
if (mode < RGBLIGHT_MODE_STATIC_LIGHT) {
|
|
rgblight_config.mode = RGBLIGHT_MODE_STATIC_LIGHT;
|
|
} else if (mode > RGBLIGHT_MODES) {
|
|
rgblight_config.mode = RGBLIGHT_MODES;
|
|
} else {
|
|
rgblight_config.mode = mode;
|
|
}
|
|
RGBLIGHT_SPLIT_SET_CHANGE_MODE;
|
|
if (write_to_eeprom) {
|
|
eeconfig_update_rgblight(&rgblight_config);
|
|
dprintf("rgblight mode [EEPROM]: %u\n", rgblight_config.mode);
|
|
} else {
|
|
dprintf("rgblight mode [NOEEPROM]: %u\n", rgblight_config.mode);
|
|
}
|
|
if (is_static_effect(rgblight_config.mode)) {
|
|
rgblight_timer_disable();
|
|
} else {
|
|
rgblight_timer_enable();
|
|
}
|
|
#ifdef RGBLIGHT_USE_TIMER
|
|
animation_status.restart = true;
|
|
#endif
|
|
rgblight_sethsv_noeeprom(rgblight_config.hue, rgblight_config.sat, rgblight_config.val);
|
|
}
|
|
|
|
void rgblight_mode(uint8_t mode) {
|
|
rgblight_mode_eeprom_helper(mode, true);
|
|
}
|
|
|
|
void rgblight_mode_noeeprom(uint8_t mode) {
|
|
rgblight_mode_eeprom_helper(mode, false);
|
|
}
|
|
|
|
void rgblight_toggle(void) {
|
|
dprintf("rgblight toggle [EEPROM]: rgblight_config.enable = %u\n", !rgblight_config.enable);
|
|
if (rgblight_config.enable) {
|
|
rgblight_disable();
|
|
} else {
|
|
rgblight_enable();
|
|
}
|
|
}
|
|
|
|
void rgblight_toggle_noeeprom(void) {
|
|
dprintf("rgblight toggle [NOEEPROM]: rgblight_config.enable = %u\n", !rgblight_config.enable);
|
|
if (rgblight_config.enable) {
|
|
rgblight_disable_noeeprom();
|
|
} else {
|
|
rgblight_enable_noeeprom();
|
|
}
|
|
}
|
|
|
|
void rgblight_enable(void) {
|
|
rgblight_config.enable = 1;
|
|
// No need to update EEPROM here. rgblight_mode() will do that, actually
|
|
// eeconfig_update_rgblight(&rgblight_config);
|
|
dprintf("rgblight enable [EEPROM]: rgblight_config.enable = %u\n", rgblight_config.enable);
|
|
rgblight_mode(rgblight_config.mode);
|
|
}
|
|
|
|
void rgblight_enable_noeeprom(void) {
|
|
rgblight_config.enable = 1;
|
|
dprintf("rgblight enable [NOEEPROM]: rgblight_config.enable = %u\n", rgblight_config.enable);
|
|
rgblight_mode_noeeprom(rgblight_config.mode);
|
|
}
|
|
|
|
void rgblight_disable(void) {
|
|
rgblight_config.enable = 0;
|
|
eeconfig_update_rgblight(&rgblight_config);
|
|
dprintf("rgblight disable [EEPROM]: rgblight_config.enable = %u\n", rgblight_config.enable);
|
|
rgblight_timer_disable();
|
|
RGBLIGHT_SPLIT_SET_CHANGE_MODE;
|
|
rgblight_set();
|
|
}
|
|
|
|
void rgblight_disable_noeeprom(void) {
|
|
rgblight_config.enable = 0;
|
|
dprintf("rgblight disable [NOEEPROM]: rgblight_config.enable = %u\n", rgblight_config.enable);
|
|
rgblight_timer_disable();
|
|
RGBLIGHT_SPLIT_SET_CHANGE_MODE;
|
|
rgblight_set();
|
|
}
|
|
|
|
void rgblight_enabled_noeeprom(bool state) {
|
|
state ? rgblight_enable_noeeprom() : rgblight_disable_noeeprom();
|
|
}
|
|
|
|
bool rgblight_is_enabled(void) {
|
|
return rgblight_config.enable;
|
|
}
|
|
|
|
void rgblight_increase_hue_helper(bool write_to_eeprom) {
|
|
uint8_t hue = rgblight_config.hue + RGBLIGHT_HUE_STEP;
|
|
rgblight_sethsv_eeprom_helper(hue, rgblight_config.sat, rgblight_config.val, write_to_eeprom);
|
|
}
|
|
void rgblight_increase_hue_noeeprom(void) {
|
|
rgblight_increase_hue_helper(false);
|
|
}
|
|
void rgblight_increase_hue(void) {
|
|
rgblight_increase_hue_helper(true);
|
|
}
|
|
void rgblight_decrease_hue_helper(bool write_to_eeprom) {
|
|
uint8_t hue = rgblight_config.hue - RGBLIGHT_HUE_STEP;
|
|
rgblight_sethsv_eeprom_helper(hue, rgblight_config.sat, rgblight_config.val, write_to_eeprom);
|
|
}
|
|
void rgblight_decrease_hue_noeeprom(void) {
|
|
rgblight_decrease_hue_helper(false);
|
|
}
|
|
void rgblight_decrease_hue(void) {
|
|
rgblight_decrease_hue_helper(true);
|
|
}
|
|
void rgblight_increase_sat_helper(bool write_to_eeprom) {
|
|
uint8_t sat = qadd8(rgblight_config.sat, RGBLIGHT_SAT_STEP);
|
|
rgblight_sethsv_eeprom_helper(rgblight_config.hue, sat, rgblight_config.val, write_to_eeprom);
|
|
}
|
|
void rgblight_increase_sat_noeeprom(void) {
|
|
rgblight_increase_sat_helper(false);
|
|
}
|
|
void rgblight_increase_sat(void) {
|
|
rgblight_increase_sat_helper(true);
|
|
}
|
|
void rgblight_decrease_sat_helper(bool write_to_eeprom) {
|
|
uint8_t sat = qsub8(rgblight_config.sat, RGBLIGHT_SAT_STEP);
|
|
rgblight_sethsv_eeprom_helper(rgblight_config.hue, sat, rgblight_config.val, write_to_eeprom);
|
|
}
|
|
void rgblight_decrease_sat_noeeprom(void) {
|
|
rgblight_decrease_sat_helper(false);
|
|
}
|
|
void rgblight_decrease_sat(void) {
|
|
rgblight_decrease_sat_helper(true);
|
|
}
|
|
void rgblight_increase_val_helper(bool write_to_eeprom) {
|
|
uint8_t val = qadd8(rgblight_config.val, RGBLIGHT_VAL_STEP);
|
|
rgblight_sethsv_eeprom_helper(rgblight_config.hue, rgblight_config.sat, val, write_to_eeprom);
|
|
}
|
|
void rgblight_increase_val_noeeprom(void) {
|
|
rgblight_increase_val_helper(false);
|
|
}
|
|
void rgblight_increase_val(void) {
|
|
rgblight_increase_val_helper(true);
|
|
}
|
|
void rgblight_decrease_val_helper(bool write_to_eeprom) {
|
|
uint8_t val = qsub8(rgblight_config.val, RGBLIGHT_VAL_STEP);
|
|
rgblight_sethsv_eeprom_helper(rgblight_config.hue, rgblight_config.sat, val, write_to_eeprom);
|
|
}
|
|
void rgblight_decrease_val_noeeprom(void) {
|
|
rgblight_decrease_val_helper(false);
|
|
}
|
|
void rgblight_decrease_val(void) {
|
|
rgblight_decrease_val_helper(true);
|
|
}
|
|
|
|
void rgblight_increase_speed_helper(bool write_to_eeprom) {
|
|
if (rgblight_config.speed < 3) rgblight_config.speed++;
|
|
// RGBLIGHT_SPLIT_SET_CHANGE_HSVS; // NEED?
|
|
if (write_to_eeprom) {
|
|
eeconfig_update_rgblight(&rgblight_config);
|
|
}
|
|
}
|
|
void rgblight_increase_speed(void) {
|
|
rgblight_increase_speed_helper(true);
|
|
}
|
|
void rgblight_increase_speed_noeeprom(void) {
|
|
rgblight_increase_speed_helper(false);
|
|
}
|
|
|
|
void rgblight_decrease_speed_helper(bool write_to_eeprom) {
|
|
if (rgblight_config.speed > 0) rgblight_config.speed--;
|
|
// RGBLIGHT_SPLIT_SET_CHANGE_HSVS; // NEED??
|
|
if (write_to_eeprom) {
|
|
eeconfig_update_rgblight(&rgblight_config);
|
|
}
|
|
}
|
|
void rgblight_decrease_speed(void) {
|
|
rgblight_decrease_speed_helper(true);
|
|
}
|
|
void rgblight_decrease_speed_noeeprom(void) {
|
|
rgblight_decrease_speed_helper(false);
|
|
}
|
|
|
|
void rgblight_sethsv_noeeprom_old(uint8_t hue, uint8_t sat, uint8_t val) {
|
|
if (rgblight_config.enable) {
|
|
rgb_t rgb = rgblight_hsv_to_rgb((hsv_t){hue, sat, val > RGBLIGHT_LIMIT_VAL ? RGBLIGHT_LIMIT_VAL : val});
|
|
rgblight_setrgb(rgb.r, rgb.g, rgb.b);
|
|
}
|
|
}
|
|
|
|
void rgblight_sethsv_eeprom_helper(uint8_t hue, uint8_t sat, uint8_t val, bool write_to_eeprom) {
|
|
if (rgblight_config.enable) {
|
|
#ifdef RGBLIGHT_SPLIT
|
|
if (rgblight_config.hue != hue || rgblight_config.sat != sat || rgblight_config.val != val) {
|
|
RGBLIGHT_SPLIT_SET_CHANGE_HSVS;
|
|
}
|
|
#endif
|
|
rgblight_status.base_mode = mode_base_table[rgblight_config.mode];
|
|
if (rgblight_config.mode == RGBLIGHT_MODE_STATIC_LIGHT) {
|
|
// same static color
|
|
#ifdef RGBLIGHT_LAYERS_RETAIN_VAL
|
|
// needed for rgblight_layers_write() to get the new val, since it reads rgblight_config.val
|
|
rgblight_config.val = val;
|
|
#endif
|
|
rgb_t rgb = rgblight_hsv_to_rgb((hsv_t){hue, sat, val > RGBLIGHT_LIMIT_VAL ? RGBLIGHT_LIMIT_VAL : val});
|
|
rgblight_setrgb(rgb.r, rgb.g, rgb.b);
|
|
} else {
|
|
// all LEDs in same color
|
|
if (1 == 0) { // dummy
|
|
}
|
|
#ifdef RGBLIGHT_EFFECT_BREATHING
|
|
else if (rgblight_status.base_mode == RGBLIGHT_MODE_BREATHING) {
|
|
// breathing mode, ignore the change of val, use in memory value instead
|
|
val = rgblight_config.val;
|
|
}
|
|
#endif
|
|
#ifdef RGBLIGHT_EFFECT_RAINBOW_MOOD
|
|
else if (rgblight_status.base_mode == RGBLIGHT_MODE_RAINBOW_MOOD) {
|
|
// rainbow mood, ignore the change of hue
|
|
hue = rgblight_config.hue;
|
|
}
|
|
#endif
|
|
#ifdef RGBLIGHT_EFFECT_RAINBOW_SWIRL
|
|
else if (rgblight_status.base_mode == RGBLIGHT_MODE_RAINBOW_SWIRL) {
|
|
// rainbow swirl, ignore the change of hue
|
|
hue = rgblight_config.hue;
|
|
}
|
|
#endif
|
|
#ifdef RGBLIGHT_EFFECT_STATIC_GRADIENT
|
|
else if (rgblight_status.base_mode == RGBLIGHT_MODE_STATIC_GRADIENT) {
|
|
// static gradient
|
|
uint8_t delta = rgblight_config.mode - rgblight_status.base_mode;
|
|
bool direction = (delta % 2) == 0;
|
|
|
|
uint8_t range = pgm_read_byte(&RGBLED_GRADIENT_RANGES[delta / 2]);
|
|
for (uint8_t i = 0; i < rgblight_ranges.effect_num_leds; i++) {
|
|
uint8_t _hue = ((uint16_t)i * (uint16_t)range) / rgblight_ranges.effect_num_leds;
|
|
if (direction) {
|
|
_hue = hue + _hue;
|
|
} else {
|
|
_hue = hue - _hue;
|
|
}
|
|
dprintf("rgblight rainbow set hsv: %d,%d,%d,%u\n", i, _hue, direction, range);
|
|
sethsv(_hue, sat, val, i + rgblight_ranges.effect_start_pos);
|
|
}
|
|
# ifdef RGBLIGHT_LAYERS_RETAIN_VAL
|
|
// needed for rgblight_layers_write() to get the new val, since it reads rgblight_config.val
|
|
rgblight_config.val = val;
|
|
# endif
|
|
rgblight_set();
|
|
}
|
|
#endif
|
|
}
|
|
rgblight_config.hue = hue;
|
|
rgblight_config.sat = sat;
|
|
rgblight_config.val = val;
|
|
if (write_to_eeprom) {
|
|
eeconfig_update_rgblight(&rgblight_config);
|
|
dprintf("rgblight set hsv [EEPROM]: %u,%u,%u\n", rgblight_config.hue, rgblight_config.sat, rgblight_config.val);
|
|
} else {
|
|
dprintf("rgblight set hsv [NOEEPROM]: %u,%u,%u\n", rgblight_config.hue, rgblight_config.sat, rgblight_config.val);
|
|
}
|
|
}
|
|
}
|
|
|
|
void rgblight_sethsv(uint8_t hue, uint8_t sat, uint8_t val) {
|
|
rgblight_sethsv_eeprom_helper(hue, sat, val, true);
|
|
}
|
|
|
|
void rgblight_sethsv_noeeprom(uint8_t hue, uint8_t sat, uint8_t val) {
|
|
rgblight_sethsv_eeprom_helper(hue, sat, val, false);
|
|
}
|
|
|
|
uint8_t rgblight_get_speed(void) {
|
|
return rgblight_config.speed;
|
|
}
|
|
|
|
void rgblight_set_speed_eeprom_helper(uint8_t speed, bool write_to_eeprom) {
|
|
rgblight_config.speed = speed;
|
|
if (write_to_eeprom) {
|
|
eeconfig_update_rgblight(&rgblight_config);
|
|
dprintf("rgblight set speed [EEPROM]: %u\n", rgblight_config.speed);
|
|
} else {
|
|
dprintf("rgblight set speed [NOEEPROM]: %u\n", rgblight_config.speed);
|
|
}
|
|
}
|
|
|
|
void rgblight_set_speed(uint8_t speed) {
|
|
rgblight_set_speed_eeprom_helper(speed, true);
|
|
}
|
|
|
|
void rgblight_set_speed_noeeprom(uint8_t speed) {
|
|
rgblight_set_speed_eeprom_helper(speed, false);
|
|
}
|
|
|
|
uint8_t rgblight_get_hue(void) {
|
|
return rgblight_config.hue;
|
|
}
|
|
|
|
uint8_t rgblight_get_sat(void) {
|
|
return rgblight_config.sat;
|
|
}
|
|
|
|
uint8_t rgblight_get_val(void) {
|
|
return rgblight_config.val;
|
|
}
|
|
|
|
hsv_t rgblight_get_hsv(void) {
|
|
return (hsv_t){rgblight_config.hue, rgblight_config.sat, rgblight_config.val};
|
|
}
|
|
|
|
void rgblight_setrgb(uint8_t r, uint8_t g, uint8_t b) {
|
|
if (!rgblight_config.enable) {
|
|
return;
|
|
}
|
|
|
|
for (uint8_t i = rgblight_ranges.effect_start_pos; i < rgblight_ranges.effect_end_pos; i++) {
|
|
rgblight_driver.set_color(rgblight_led_index(i), r, g, b);
|
|
}
|
|
rgblight_set();
|
|
}
|
|
|
|
void rgblight_setrgb_at(uint8_t r, uint8_t g, uint8_t b, uint8_t index) {
|
|
if (!rgblight_config.enable || index >= RGBLIGHT_LED_COUNT) {
|
|
return;
|
|
}
|
|
|
|
rgblight_driver.set_color(rgblight_led_index(index), r, g, b);
|
|
rgblight_set();
|
|
}
|
|
|
|
void rgblight_sethsv_at(uint8_t hue, uint8_t sat, uint8_t val, uint8_t index) {
|
|
if (!rgblight_config.enable) {
|
|
return;
|
|
}
|
|
|
|
rgb_t rgb = rgblight_hsv_to_rgb((hsv_t){hue, sat, val > RGBLIGHT_LIMIT_VAL ? RGBLIGHT_LIMIT_VAL : val});
|
|
rgblight_setrgb_at(rgb.r, rgb.g, rgb.b, index);
|
|
}
|
|
|
|
#if defined(RGBLIGHT_EFFECT_BREATHING) || defined(RGBLIGHT_EFFECT_RAINBOW_MOOD) || defined(RGBLIGHT_EFFECT_RAINBOW_SWIRL) || defined(RGBLIGHT_EFFECT_SNAKE) || defined(RGBLIGHT_EFFECT_KNIGHT) || defined(RGBLIGHT_EFFECT_TWINKLE)
|
|
|
|
static uint8_t get_interval_time(const uint8_t *default_interval_address, uint8_t velocikey_min, uint8_t velocikey_max) {
|
|
return
|
|
# ifdef VELOCIKEY_ENABLE
|
|
rgblight_velocikey_enabled() ? rgblight_velocikey_match_speed(velocikey_min, velocikey_max) :
|
|
# endif
|
|
pgm_read_byte(default_interval_address);
|
|
}
|
|
|
|
#endif
|
|
|
|
void rgblight_setrgb_range(uint8_t r, uint8_t g, uint8_t b, uint8_t start, uint8_t end) {
|
|
if (!rgblight_config.enable || start < 0 || start >= end || end > RGBLIGHT_LED_COUNT) {
|
|
return;
|
|
}
|
|
|
|
for (uint8_t i = start; i < end; i++) {
|
|
rgblight_driver.set_color(rgblight_led_index(i), r, g, b);
|
|
}
|
|
rgblight_set();
|
|
}
|
|
|
|
void rgblight_sethsv_range(uint8_t hue, uint8_t sat, uint8_t val, uint8_t start, uint8_t end) {
|
|
if (!rgblight_config.enable) {
|
|
return;
|
|
}
|
|
|
|
rgb_t rgb = rgblight_hsv_to_rgb((hsv_t){hue, sat, val > RGBLIGHT_LIMIT_VAL ? RGBLIGHT_LIMIT_VAL : val});
|
|
rgblight_setrgb_range(rgb.r, rgb.g, rgb.b, start, end);
|
|
}
|
|
|
|
#ifndef RGBLIGHT_SPLIT
|
|
void rgblight_setrgb_master(uint8_t r, uint8_t g, uint8_t b) {
|
|
rgblight_setrgb_range(r, g, b, 0, (uint8_t)RGBLIGHT_LED_COUNT / 2);
|
|
}
|
|
|
|
void rgblight_setrgb_slave(uint8_t r, uint8_t g, uint8_t b) {
|
|
rgblight_setrgb_range(r, g, b, (uint8_t)RGBLIGHT_LED_COUNT / 2, (uint8_t)RGBLIGHT_LED_COUNT);
|
|
}
|
|
|
|
void rgblight_sethsv_master(uint8_t hue, uint8_t sat, uint8_t val) {
|
|
rgblight_sethsv_range(hue, sat, val, 0, (uint8_t)RGBLIGHT_LED_COUNT / 2);
|
|
}
|
|
|
|
void rgblight_sethsv_slave(uint8_t hue, uint8_t sat, uint8_t val) {
|
|
rgblight_sethsv_range(hue, sat, val, (uint8_t)RGBLIGHT_LED_COUNT / 2, (uint8_t)RGBLIGHT_LED_COUNT);
|
|
}
|
|
#endif // ifndef RGBLIGHT_SPLIT
|
|
|
|
#ifdef RGBLIGHT_LAYERS
|
|
void rgblight_set_layer_state(uint8_t layer, bool enabled) {
|
|
rgblight_layer_mask_t mask = (rgblight_layer_mask_t)1 << layer;
|
|
if (enabled) {
|
|
rgblight_status.enabled_layer_mask |= mask;
|
|
} else {
|
|
rgblight_status.enabled_layer_mask &= ~mask;
|
|
}
|
|
RGBLIGHT_SPLIT_SET_CHANGE_LAYERS;
|
|
|
|
// Calling rgblight_set() here (directly or indirectly) could
|
|
// potentially cause timing issues when there are multiple
|
|
// successive calls to rgblight_set_layer_state(). Instead,
|
|
// set a flag and do it the next time rgblight_task() runs.
|
|
|
|
deferred_set_layer_state = true;
|
|
}
|
|
|
|
bool rgblight_get_layer_state(uint8_t layer) {
|
|
rgblight_layer_mask_t mask = (rgblight_layer_mask_t)1 << layer;
|
|
return (rgblight_status.enabled_layer_mask & mask) != 0;
|
|
}
|
|
|
|
// Write any enabled LED layers into the buffer
|
|
static void rgblight_layers_write(void) {
|
|
# ifdef RGBLIGHT_LAYERS_RETAIN_VAL
|
|
uint8_t current_val = rgblight_get_val();
|
|
# endif
|
|
uint8_t i = 0;
|
|
// For each layer
|
|
for (const rgblight_segment_t *const *layer_ptr = rgblight_layers; i < RGBLIGHT_MAX_LAYERS; layer_ptr++, i++) {
|
|
if (!rgblight_get_layer_state(i)) {
|
|
continue; // Layer is disabled
|
|
}
|
|
const rgblight_segment_t *segment_ptr = pgm_read_ptr(layer_ptr);
|
|
if (segment_ptr == NULL) {
|
|
break; // No more layers
|
|
}
|
|
// For each segment
|
|
while (1) {
|
|
rgblight_segment_t segment;
|
|
memcpy_P(&segment, segment_ptr, sizeof(rgblight_segment_t));
|
|
if (segment.index == RGBLIGHT_END_SEGMENT_INDEX) {
|
|
break; // No more segments
|
|
}
|
|
// Write segment.count LEDs
|
|
int limit = MIN(segment.index + segment.count, RGBLIGHT_LED_COUNT);
|
|
for (int i = segment.index; i < limit; i++) {
|
|
# ifdef RGBLIGHT_LAYERS_RETAIN_VAL
|
|
sethsv(segment.hue, segment.sat, current_val, i);
|
|
# else
|
|
sethsv(segment.hue, segment.sat, segment.val, i);
|
|
# endif
|
|
}
|
|
segment_ptr++;
|
|
}
|
|
}
|
|
}
|
|
|
|
# ifdef RGBLIGHT_LAYER_BLINK
|
|
rgblight_layer_mask_t _blinking_layer_mask = 0;
|
|
static uint16_t _repeat_timer;
|
|
static uint8_t _times_remaining;
|
|
static uint16_t _dur;
|
|
|
|
void rgblight_blink_layer(uint8_t layer, uint16_t duration_ms) {
|
|
rgblight_blink_layer_repeat(layer, duration_ms, 1);
|
|
}
|
|
|
|
void rgblight_blink_layer_repeat(uint8_t layer, uint16_t duration_ms, uint8_t times) {
|
|
if (times > UINT8_MAX / 2) {
|
|
times = UINT8_MAX / 2;
|
|
}
|
|
|
|
_times_remaining = times * 2;
|
|
_dur = duration_ms;
|
|
|
|
rgblight_set_layer_state(layer, true);
|
|
_times_remaining--;
|
|
_blinking_layer_mask |= (rgblight_layer_mask_t)1 << layer;
|
|
_repeat_timer = sync_timer_read() + duration_ms;
|
|
}
|
|
|
|
void rgblight_unblink_layer(uint8_t layer) {
|
|
rgblight_set_layer_state(layer, false);
|
|
_blinking_layer_mask &= ~((rgblight_layer_mask_t)1 << layer);
|
|
}
|
|
|
|
void rgblight_unblink_all_but_layer(uint8_t layer) {
|
|
for (uint8_t i = 0; i < RGBLIGHT_MAX_LAYERS; i++) {
|
|
if (i != layer) {
|
|
if ((_blinking_layer_mask & (rgblight_layer_mask_t)1 << i) != 0) {
|
|
rgblight_unblink_layer(i);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void rgblight_blink_layer_repeat_helper(void) {
|
|
if (_blinking_layer_mask != 0 && timer_expired(sync_timer_read(), _repeat_timer)) {
|
|
for (uint8_t layer = 0; layer < RGBLIGHT_MAX_LAYERS; layer++) {
|
|
if ((_blinking_layer_mask & (rgblight_layer_mask_t)1 << layer) != 0) {
|
|
if (_times_remaining % 2 == 1) {
|
|
rgblight_set_layer_state(layer, false);
|
|
} else {
|
|
rgblight_set_layer_state(layer, true);
|
|
}
|
|
}
|
|
}
|
|
_times_remaining--;
|
|
if (_times_remaining <= 0) {
|
|
_blinking_layer_mask = 0;
|
|
} else {
|
|
_repeat_timer = sync_timer_read() + _dur;
|
|
}
|
|
}
|
|
}
|
|
# endif
|
|
|
|
#endif
|
|
|
|
#ifdef RGBLIGHT_SLEEP
|
|
|
|
void rgblight_suspend(void) {
|
|
rgblight_timer_disable();
|
|
if (!is_suspended) {
|
|
is_suspended = true;
|
|
pre_suspend_enabled = rgblight_config.enable;
|
|
|
|
# ifdef RGBLIGHT_LAYER_BLINK
|
|
// make sure any layer blinks don't come back after suspend
|
|
rgblight_status.enabled_layer_mask &= ~_blinking_layer_mask;
|
|
_blinking_layer_mask = 0;
|
|
# endif
|
|
|
|
rgblight_disable_noeeprom();
|
|
}
|
|
}
|
|
|
|
void rgblight_wakeup(void) {
|
|
is_suspended = false;
|
|
|
|
if (pre_suspend_enabled) {
|
|
rgblight_enable_noeeprom();
|
|
}
|
|
# ifdef RGBLIGHT_LAYERS_OVERRIDE_RGB_OFF
|
|
// Need this or else the LEDs won't be set
|
|
else if (rgblight_status.enabled_layer_mask != 0) {
|
|
rgblight_set();
|
|
}
|
|
# endif
|
|
|
|
rgblight_timer_enable();
|
|
}
|
|
|
|
#endif
|
|
|
|
void rgblight_set(void) {
|
|
if (!rgblight_config.enable) {
|
|
for (uint8_t i = rgblight_ranges.effect_start_pos; i < rgblight_ranges.effect_end_pos; i++) {
|
|
rgblight_driver.set_color(rgblight_led_index(i), 0, 0, 0);
|
|
}
|
|
}
|
|
|
|
#ifdef RGBLIGHT_LAYERS
|
|
if (rgblight_layers != NULL
|
|
# if !defined(RGBLIGHT_LAYERS_OVERRIDE_RGB_OFF)
|
|
&& rgblight_config.enable
|
|
# elif defined(RGBLIGHT_SLEEP)
|
|
&& !is_suspended
|
|
# endif
|
|
) {
|
|
rgblight_layers_write();
|
|
}
|
|
#endif
|
|
|
|
rgblight_driver.flush();
|
|
}
|
|
|
|
#ifdef RGBLIGHT_SPLIT
|
|
/* for split keyboard master side */
|
|
uint8_t rgblight_get_change_flags(void) {
|
|
return rgblight_status.change_flags;
|
|
}
|
|
|
|
void rgblight_clear_change_flags(void) {
|
|
rgblight_status.change_flags = 0;
|
|
}
|
|
|
|
void rgblight_get_syncinfo(rgblight_syncinfo_t *syncinfo) {
|
|
syncinfo->config = rgblight_config;
|
|
syncinfo->status = rgblight_status;
|
|
}
|
|
|
|
/* for split keyboard slave side */
|
|
void rgblight_update_sync(rgblight_syncinfo_t *syncinfo, bool write_to_eeprom) {
|
|
# ifdef RGBLIGHT_LAYERS
|
|
if (syncinfo->status.change_flags & RGBLIGHT_STATUS_CHANGE_LAYERS) {
|
|
rgblight_status.enabled_layer_mask = syncinfo->status.enabled_layer_mask;
|
|
}
|
|
# endif
|
|
if (syncinfo->status.change_flags & RGBLIGHT_STATUS_CHANGE_MODE) {
|
|
if (syncinfo->config.enable) {
|
|
rgblight_config.enable = 1; // == rgblight_enable_noeeprom();
|
|
rgblight_mode_eeprom_helper(syncinfo->config.mode, write_to_eeprom);
|
|
} else {
|
|
rgblight_disable_noeeprom();
|
|
}
|
|
}
|
|
if (syncinfo->status.change_flags & RGBLIGHT_STATUS_CHANGE_HSVS) {
|
|
rgblight_sethsv_eeprom_helper(syncinfo->config.hue, syncinfo->config.sat, syncinfo->config.val, write_to_eeprom);
|
|
// rgblight_config.speed = config->speed; // NEED???
|
|
}
|
|
# ifdef RGBLIGHT_USE_TIMER
|
|
if (syncinfo->status.change_flags & RGBLIGHT_STATUS_CHANGE_TIMER) {
|
|
if (syncinfo->status.timer_enabled) {
|
|
rgblight_timer_enable();
|
|
} else {
|
|
rgblight_timer_disable();
|
|
}
|
|
}
|
|
# ifndef RGBLIGHT_SPLIT_NO_ANIMATION_SYNC
|
|
if (syncinfo->status.change_flags & RGBLIGHT_STATUS_ANIMATION_TICK) {
|
|
animation_status.restart = true;
|
|
}
|
|
# endif /* RGBLIGHT_SPLIT_NO_ANIMATION_SYNC */
|
|
# endif /* RGBLIGHT_USE_TIMER */
|
|
}
|
|
#endif /* RGBLIGHT_SPLIT */
|
|
|
|
#ifdef RGBLIGHT_USE_TIMER
|
|
|
|
typedef void (*effect_func_t)(animation_status_t *anim);
|
|
|
|
// Animation timer -- use system timer (AVR Timer0)
|
|
void rgblight_timer_init(void) {
|
|
rgblight_status.timer_enabled = false;
|
|
RGBLIGHT_SPLIT_SET_CHANGE_TIMER_ENABLE;
|
|
}
|
|
void rgblight_timer_enable(void) {
|
|
if (!is_static_effect(rgblight_config.mode)) {
|
|
rgblight_status.timer_enabled = true;
|
|
}
|
|
animation_status.last_timer = sync_timer_read();
|
|
RGBLIGHT_SPLIT_SET_CHANGE_TIMER_ENABLE;
|
|
dprintf("rgblight timer enabled.\n");
|
|
}
|
|
void rgblight_timer_disable(void) {
|
|
rgblight_status.timer_enabled = false;
|
|
RGBLIGHT_SPLIT_SET_CHANGE_TIMER_ENABLE;
|
|
dprintf("rgblight timer disable.\n");
|
|
}
|
|
void rgblight_timer_toggle(void) {
|
|
dprintf("rgblight timer toggle.\n");
|
|
if (rgblight_status.timer_enabled) {
|
|
rgblight_timer_disable();
|
|
} else {
|
|
rgblight_timer_enable();
|
|
}
|
|
}
|
|
|
|
void rgblight_show_solid_color(uint8_t r, uint8_t g, uint8_t b) {
|
|
rgblight_enable();
|
|
rgblight_mode(RGBLIGHT_MODE_STATIC_LIGHT);
|
|
rgblight_setrgb(r, g, b);
|
|
}
|
|
|
|
static void rgblight_effect_dummy(animation_status_t *anim) {
|
|
// do nothing
|
|
/********
|
|
dprintf("rgblight_task() what happened?\n");
|
|
dprintf("is_static_effect %d\n", is_static_effect(rgblight_config.mode));
|
|
dprintf("mode = %d, base_mode = %d, timer_enabled %d, ",
|
|
rgblight_config.mode, rgblight_status.base_mode,
|
|
rgblight_status.timer_enabled);
|
|
dprintf("last_timer = %d\n",anim->last_timer);
|
|
**/
|
|
}
|
|
|
|
void rgblight_timer_task(void) {
|
|
if (rgblight_status.timer_enabled) {
|
|
effect_func_t effect_func = rgblight_effect_dummy;
|
|
uint16_t interval_time = 2000; // dummy interval
|
|
uint8_t delta = rgblight_config.mode - rgblight_status.base_mode;
|
|
animation_status.delta = delta;
|
|
|
|
// static light mode, do nothing here
|
|
if (1 == 0) { // dummy
|
|
}
|
|
# ifdef RGBLIGHT_EFFECT_BREATHING
|
|
else if (rgblight_status.base_mode == RGBLIGHT_MODE_BREATHING) {
|
|
// breathing mode
|
|
interval_time = get_interval_time(&RGBLED_BREATHING_INTERVALS[delta], 1, 100);
|
|
effect_func = rgblight_effect_breathing;
|
|
}
|
|
# endif
|
|
# ifdef RGBLIGHT_EFFECT_RAINBOW_MOOD
|
|
else if (rgblight_status.base_mode == RGBLIGHT_MODE_RAINBOW_MOOD) {
|
|
// rainbow mood mode
|
|
interval_time = get_interval_time(&RGBLED_RAINBOW_MOOD_INTERVALS[delta], 5, 100);
|
|
effect_func = rgblight_effect_rainbow_mood;
|
|
}
|
|
# endif
|
|
# ifdef RGBLIGHT_EFFECT_RAINBOW_SWIRL
|
|
else if (rgblight_status.base_mode == RGBLIGHT_MODE_RAINBOW_SWIRL) {
|
|
// rainbow swirl mode
|
|
interval_time = get_interval_time(&RGBLED_RAINBOW_SWIRL_INTERVALS[delta / 2], 1, 100);
|
|
effect_func = rgblight_effect_rainbow_swirl;
|
|
}
|
|
# endif
|
|
# ifdef RGBLIGHT_EFFECT_SNAKE
|
|
else if (rgblight_status.base_mode == RGBLIGHT_MODE_SNAKE) {
|
|
// snake mode
|
|
interval_time = get_interval_time(&RGBLED_SNAKE_INTERVALS[delta / 2], 1, 200);
|
|
effect_func = rgblight_effect_snake;
|
|
}
|
|
# endif
|
|
# ifdef RGBLIGHT_EFFECT_KNIGHT
|
|
else if (rgblight_status.base_mode == RGBLIGHT_MODE_KNIGHT) {
|
|
// knight mode
|
|
interval_time = get_interval_time(&RGBLED_KNIGHT_INTERVALS[delta], 5, 100);
|
|
effect_func = rgblight_effect_knight;
|
|
}
|
|
# endif
|
|
# ifdef RGBLIGHT_EFFECT_CHRISTMAS
|
|
else if (rgblight_status.base_mode == RGBLIGHT_MODE_CHRISTMAS) {
|
|
// christmas mode
|
|
interval_time = RGBLIGHT_EFFECT_CHRISTMAS_INTERVAL;
|
|
effect_func = (effect_func_t)rgblight_effect_christmas;
|
|
}
|
|
# endif
|
|
# ifdef RGBLIGHT_EFFECT_RGB_TEST
|
|
else if (rgblight_status.base_mode == RGBLIGHT_MODE_RGB_TEST) {
|
|
// RGB test mode
|
|
interval_time = pgm_read_word(&RGBLED_RGBTEST_INTERVALS[0]);
|
|
effect_func = (effect_func_t)rgblight_effect_rgbtest;
|
|
}
|
|
# endif
|
|
# ifdef RGBLIGHT_EFFECT_ALTERNATING
|
|
else if (rgblight_status.base_mode == RGBLIGHT_MODE_ALTERNATING) {
|
|
interval_time = 500;
|
|
effect_func = (effect_func_t)rgblight_effect_alternating;
|
|
}
|
|
# endif
|
|
# ifdef RGBLIGHT_EFFECT_TWINKLE
|
|
else if (rgblight_status.base_mode == RGBLIGHT_MODE_TWINKLE) {
|
|
interval_time = get_interval_time(&RGBLED_TWINKLE_INTERVALS[delta % 3], 5, 30);
|
|
effect_func = (effect_func_t)rgblight_effect_twinkle;
|
|
}
|
|
# endif
|
|
if (animation_status.restart) {
|
|
animation_status.restart = false;
|
|
animation_status.last_timer = sync_timer_read();
|
|
animation_status.pos16 = 0; // restart signal to local each effect
|
|
}
|
|
uint16_t now = sync_timer_read();
|
|
if (timer_expired(now, animation_status.last_timer)) {
|
|
# if defined(RGBLIGHT_SPLIT) && !defined(RGBLIGHT_SPLIT_NO_ANIMATION_SYNC)
|
|
static uint16_t report_last_timer = 0;
|
|
static bool tick_flag = false;
|
|
uint16_t oldpos16;
|
|
if (tick_flag) {
|
|
tick_flag = false;
|
|
if (timer_expired(now, report_last_timer)) {
|
|
report_last_timer += 30000;
|
|
dprintf("rgblight animation tick report to slave\n");
|
|
RGBLIGHT_SPLIT_ANIMATION_TICK;
|
|
}
|
|
}
|
|
oldpos16 = animation_status.pos16;
|
|
# endif
|
|
animation_status.last_timer += interval_time;
|
|
effect_func(&animation_status);
|
|
# if defined(RGBLIGHT_SPLIT) && !defined(RGBLIGHT_SPLIT_NO_ANIMATION_SYNC)
|
|
if (animation_status.pos16 == 0 && oldpos16 != 0) {
|
|
tick_flag = true;
|
|
}
|
|
# endif
|
|
}
|
|
}
|
|
|
|
# ifdef RGBLIGHT_LAYERS
|
|
# ifdef RGBLIGHT_LAYER_BLINK
|
|
rgblight_blink_layer_repeat_helper();
|
|
# endif
|
|
|
|
if (deferred_set_layer_state) {
|
|
deferred_set_layer_state = false;
|
|
|
|
// Static modes don't have a ticker running to update the LEDs
|
|
if (rgblight_status.timer_enabled == false) {
|
|
rgblight_mode_noeeprom(rgblight_config.mode);
|
|
}
|
|
|
|
# ifdef RGBLIGHT_LAYERS_OVERRIDE_RGB_OFF
|
|
// If not enabled, then nothing else will actually set the LEDs...
|
|
if (!rgblight_config.enable) {
|
|
rgblight_set();
|
|
}
|
|
# endif
|
|
}
|
|
# endif
|
|
}
|
|
|
|
#endif /* RGBLIGHT_USE_TIMER */
|
|
|
|
#if defined(RGBLIGHT_EFFECT_BREATHING) || defined(RGBLIGHT_EFFECT_TWINKLE)
|
|
|
|
# ifndef RGBLIGHT_EFFECT_BREATHE_CENTER
|
|
# ifndef RGBLIGHT_BREATHE_TABLE_SIZE
|
|
# define RGBLIGHT_BREATHE_TABLE_SIZE 256 // 256 or 128 or 64
|
|
# endif
|
|
# include <rgblight_breathe_table.h>
|
|
# endif
|
|
|
|
static uint8_t breathe_calc(uint8_t pos) {
|
|
// http://sean.voisen.org/blog/2011/10/breathing-led-with-arduino/
|
|
# ifdef RGBLIGHT_EFFECT_BREATHE_TABLE
|
|
return pgm_read_byte(&rgblight_effect_breathe_table[pos / table_scale]);
|
|
# else
|
|
return (exp(sin((pos / 255.0) * M_PI)) - RGBLIGHT_EFFECT_BREATHE_CENTER / M_E) * (RGBLIGHT_EFFECT_BREATHE_MAX / (M_E - 1 / M_E));
|
|
# endif
|
|
}
|
|
|
|
#endif
|
|
|
|
// Effects
|
|
#ifdef RGBLIGHT_EFFECT_BREATHING
|
|
|
|
__attribute__((weak)) const uint8_t RGBLED_BREATHING_INTERVALS[] PROGMEM = {30, 20, 10, 5};
|
|
|
|
void rgblight_effect_breathing(animation_status_t *anim) {
|
|
uint8_t val = breathe_calc(anim->pos);
|
|
rgblight_sethsv_noeeprom_old(rgblight_config.hue, rgblight_config.sat, val);
|
|
anim->pos = (anim->pos + 1);
|
|
}
|
|
#endif
|
|
|
|
#ifdef RGBLIGHT_EFFECT_RAINBOW_MOOD
|
|
__attribute__((weak)) const uint8_t RGBLED_RAINBOW_MOOD_INTERVALS[] PROGMEM = {120, 60, 30};
|
|
|
|
void rgblight_effect_rainbow_mood(animation_status_t *anim) {
|
|
rgblight_sethsv_noeeprom_old(anim->current_hue, rgblight_config.sat, rgblight_config.val);
|
|
anim->current_hue++;
|
|
}
|
|
#endif
|
|
|
|
#ifdef RGBLIGHT_EFFECT_RAINBOW_SWIRL
|
|
# ifndef RGBLIGHT_RAINBOW_SWIRL_RANGE
|
|
# define RGBLIGHT_RAINBOW_SWIRL_RANGE 255
|
|
# endif
|
|
|
|
__attribute__((weak)) const uint8_t RGBLED_RAINBOW_SWIRL_INTERVALS[] PROGMEM = {100, 50, 20};
|
|
|
|
void rgblight_effect_rainbow_swirl(animation_status_t *anim) {
|
|
uint8_t hue;
|
|
uint8_t i;
|
|
|
|
for (i = 0; i < rgblight_ranges.effect_num_leds; i++) {
|
|
hue = (RGBLIGHT_RAINBOW_SWIRL_RANGE / rgblight_ranges.effect_num_leds * i + anim->current_hue);
|
|
sethsv(hue, rgblight_config.sat, rgblight_config.val, i + rgblight_ranges.effect_start_pos);
|
|
}
|
|
rgblight_set();
|
|
|
|
if (anim->delta % 2) {
|
|
anim->current_hue++;
|
|
} else {
|
|
anim->current_hue--;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#ifdef RGBLIGHT_EFFECT_SNAKE
|
|
__attribute__((weak)) const uint8_t RGBLED_SNAKE_INTERVALS[] PROGMEM = {100, 50, 20};
|
|
|
|
void rgblight_effect_snake(animation_status_t *anim) {
|
|
static uint8_t pos = 0;
|
|
uint8_t i, j;
|
|
int8_t k;
|
|
int8_t increment = 1;
|
|
|
|
if (anim->delta % 2) {
|
|
increment = -1;
|
|
}
|
|
|
|
# if defined(RGBLIGHT_SPLIT) && !defined(RGBLIGHT_SPLIT_NO_ANIMATION_SYNC)
|
|
if (anim->pos == 0) { // restart signal
|
|
if (increment == 1) {
|
|
pos = rgblight_ranges.effect_num_leds - 1;
|
|
} else {
|
|
pos = 0;
|
|
}
|
|
anim->pos = 1;
|
|
}
|
|
# endif
|
|
|
|
for (i = 0; i < rgblight_ranges.effect_num_leds; i++) {
|
|
rgblight_driver.set_color(rgblight_led_index(i + rgblight_ranges.effect_start_pos), 0, 0, 0);
|
|
|
|
for (j = 0; j < RGBLIGHT_EFFECT_SNAKE_LENGTH; j++) {
|
|
k = pos + j * increment;
|
|
if (k > RGBLIGHT_LED_COUNT) {
|
|
k = k % (RGBLIGHT_LED_COUNT);
|
|
}
|
|
if (k < 0) {
|
|
k = k + rgblight_ranges.effect_num_leds;
|
|
}
|
|
if (i == k) {
|
|
sethsv(rgblight_config.hue, rgblight_config.sat, (uint8_t)(rgblight_config.val * (RGBLIGHT_EFFECT_SNAKE_LENGTH - j) / RGBLIGHT_EFFECT_SNAKE_LENGTH), i + rgblight_ranges.effect_start_pos);
|
|
}
|
|
}
|
|
}
|
|
rgblight_set();
|
|
if (increment == 1) {
|
|
if (pos - RGBLIGHT_EFFECT_SNAKE_INCREMENT < 0) {
|
|
pos = rgblight_ranges.effect_num_leds - 1;
|
|
# if defined(RGBLIGHT_SPLIT) && !defined(RGBLIGHT_SPLIT_NO_ANIMATION_SYNC)
|
|
anim->pos = 0;
|
|
# endif
|
|
} else {
|
|
pos -= RGBLIGHT_EFFECT_SNAKE_INCREMENT;
|
|
# if defined(RGBLIGHT_SPLIT) && !defined(RGBLIGHT_SPLIT_NO_ANIMATION_SYNC)
|
|
anim->pos = 1;
|
|
# endif
|
|
}
|
|
} else {
|
|
pos = (pos + RGBLIGHT_EFFECT_SNAKE_INCREMENT) % rgblight_ranges.effect_num_leds;
|
|
# if defined(RGBLIGHT_SPLIT) && !defined(RGBLIGHT_SPLIT_NO_ANIMATION_SYNC)
|
|
anim->pos = pos;
|
|
# endif
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#ifdef RGBLIGHT_EFFECT_KNIGHT
|
|
__attribute__((weak)) const uint8_t RGBLED_KNIGHT_INTERVALS[] PROGMEM = {127, 63, 31};
|
|
|
|
void rgblight_effect_knight(animation_status_t *anim) {
|
|
static int8_t low_bound = 0;
|
|
static int8_t high_bound = RGBLIGHT_EFFECT_KNIGHT_LENGTH - 1;
|
|
static int8_t increment = RGBLIGHT_EFFECT_KNIGHT_INCREMENT;
|
|
uint8_t i, cur;
|
|
|
|
# if defined(RGBLIGHT_SPLIT) && !defined(RGBLIGHT_SPLIT_NO_ANIMATION_SYNC)
|
|
if (anim->pos == 0) { // restart signal
|
|
anim->pos = 1;
|
|
low_bound = 0;
|
|
high_bound = RGBLIGHT_EFFECT_KNIGHT_LENGTH - 1;
|
|
increment = 1;
|
|
}
|
|
# endif
|
|
// Set all the LEDs to 0
|
|
for (i = rgblight_ranges.effect_start_pos; i < rgblight_ranges.effect_end_pos; i++) {
|
|
rgblight_driver.set_color(rgblight_led_index(i), 0, 0, 0);
|
|
}
|
|
// Determine which LEDs should be lit up
|
|
for (i = 0; i < RGBLIGHT_EFFECT_KNIGHT_LED_NUM; i++) {
|
|
cur = (i + RGBLIGHT_EFFECT_KNIGHT_OFFSET) % rgblight_ranges.effect_num_leds + rgblight_ranges.effect_start_pos;
|
|
|
|
if (i >= low_bound && i <= high_bound) {
|
|
sethsv(rgblight_config.hue, rgblight_config.sat, rgblight_config.val, cur);
|
|
} else {
|
|
rgblight_driver.set_color(rgblight_led_index(cur), 0, 0, 0);
|
|
}
|
|
}
|
|
rgblight_set();
|
|
|
|
// Move from low_bound to high_bound changing the direction we increment each
|
|
// time a boundary is hit.
|
|
low_bound += increment;
|
|
high_bound += increment;
|
|
|
|
if (high_bound <= 0 || low_bound >= RGBLIGHT_EFFECT_KNIGHT_LED_NUM - 1) {
|
|
increment = -increment;
|
|
# if defined(RGBLIGHT_SPLIT) && !defined(RGBLIGHT_SPLIT_NO_ANIMATION_SYNC)
|
|
if (increment == 1) {
|
|
anim->pos = 0;
|
|
}
|
|
# endif
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#ifdef RGBLIGHT_EFFECT_CHRISTMAS
|
|
# define CUBED(x) ((x) * (x) * (x))
|
|
|
|
/**
|
|
* Christmas lights effect, with a smooth animation between red & green.
|
|
*/
|
|
void rgblight_effect_christmas(animation_status_t *anim) {
|
|
static int8_t increment = 1;
|
|
const uint8_t max_pos = 32;
|
|
const uint8_t hue_green = 85;
|
|
|
|
uint32_t xa;
|
|
uint8_t hue, val;
|
|
uint8_t i;
|
|
|
|
// The effect works by animating anim->pos from 0 to 32 and back to 0.
|
|
// The pos is used in a cubic bezier formula to ease-in-out between red and green, leaving the interpolated colors visible as short as possible.
|
|
xa = CUBED((uint32_t)anim->pos);
|
|
hue = ((uint32_t)hue_green) * xa / (xa + CUBED((uint32_t)(max_pos - anim->pos)));
|
|
// Additionally, these interpolated colors get shown with a slightly darker value, to make them less prominent than the main colors.
|
|
val = 255 - (3 * (hue < hue_green / 2 ? hue : hue_green - hue) / 2);
|
|
|
|
for (i = 0; i < rgblight_ranges.effect_num_leds; i++) {
|
|
uint8_t local_hue = (i / RGBLIGHT_EFFECT_CHRISTMAS_STEP) % 2 ? hue : hue_green - hue;
|
|
sethsv(local_hue, rgblight_config.sat, val, i + rgblight_ranges.effect_start_pos);
|
|
}
|
|
rgblight_set();
|
|
|
|
if (anim->pos == 0) {
|
|
increment = 1;
|
|
} else if (anim->pos == max_pos) {
|
|
increment = -1;
|
|
}
|
|
anim->pos += increment;
|
|
}
|
|
#endif
|
|
|
|
#ifdef RGBLIGHT_EFFECT_RGB_TEST
|
|
__attribute__((weak)) const uint16_t RGBLED_RGBTEST_INTERVALS[] PROGMEM = {1024};
|
|
|
|
void rgblight_effect_rgbtest(animation_status_t *anim) {
|
|
uint8_t val = rgblight_get_val();
|
|
|
|
uint8_t r = anim->pos & 1 ? val : 0;
|
|
uint8_t g = anim->pos & 2 ? val : 0;
|
|
uint8_t b = anim->pos & 4 ? val : 0;
|
|
rgblight_setrgb(r, g, b);
|
|
anim->pos = (anim->pos + 1) % 8;
|
|
}
|
|
#endif
|
|
|
|
#ifdef RGBLIGHT_EFFECT_ALTERNATING
|
|
void rgblight_effect_alternating(animation_status_t *anim) {
|
|
for (int i = 0; i < rgblight_ranges.effect_num_leds; i++) {
|
|
if (i < rgblight_ranges.effect_num_leds / 2 && anim->pos) {
|
|
sethsv(rgblight_config.hue, rgblight_config.sat, rgblight_config.val, i + rgblight_ranges.effect_start_pos);
|
|
} else if (i >= rgblight_ranges.effect_num_leds / 2 && !anim->pos) {
|
|
sethsv(rgblight_config.hue, rgblight_config.sat, rgblight_config.val, i + rgblight_ranges.effect_start_pos);
|
|
} else {
|
|
sethsv(rgblight_config.hue, rgblight_config.sat, 0, i + rgblight_ranges.effect_start_pos);
|
|
}
|
|
}
|
|
rgblight_set();
|
|
anim->pos = (anim->pos + 1) % 2;
|
|
}
|
|
#endif
|
|
|
|
#ifdef RGBLIGHT_EFFECT_TWINKLE
|
|
__attribute__((weak)) const uint8_t RGBLED_TWINKLE_INTERVALS[] PROGMEM = {30, 15, 5};
|
|
|
|
typedef struct PACKED {
|
|
hsv_t hsv;
|
|
uint8_t life;
|
|
uint8_t max_life;
|
|
} TwinkleState;
|
|
|
|
static TwinkleState led_twinkle_state[RGBLIGHT_LED_COUNT];
|
|
|
|
void rgblight_effect_twinkle(animation_status_t *anim) {
|
|
const bool random_color = anim->delta / 3;
|
|
const bool restart = anim->pos == 0;
|
|
anim->pos = 1;
|
|
|
|
const uint8_t bottom = breathe_calc(0);
|
|
const uint8_t top = breathe_calc(127);
|
|
|
|
uint8_t frac(uint8_t n, uint8_t d) {
|
|
return (uint16_t)255 * n / d;
|
|
}
|
|
uint8_t scale(uint16_t v, uint8_t scale) {
|
|
return (v * scale) >> 8;
|
|
}
|
|
|
|
const uint8_t trigger = scale((uint16_t)0xFF * RGBLIGHT_EFFECT_TWINKLE_PROBABILITY, 127 + rgblight_config.val / 2);
|
|
|
|
for (uint8_t i = 0; i < rgblight_ranges.effect_num_leds; i++) {
|
|
TwinkleState *t = &(led_twinkle_state[i]);
|
|
hsv_t * c = &(t->hsv);
|
|
|
|
if (!random_color) {
|
|
c->h = rgblight_config.hue;
|
|
c->s = rgblight_config.sat;
|
|
}
|
|
|
|
if (restart) {
|
|
// Restart
|
|
t->life = 0;
|
|
c->v = 0;
|
|
} else if (t->life) {
|
|
// This LED is already on, either brightening or dimming
|
|
t->life--;
|
|
uint8_t unscaled = frac(breathe_calc(frac(t->life, t->max_life)) - bottom, top - bottom);
|
|
c->v = scale(rgblight_config.val, unscaled);
|
|
} else if ((rand() % 0xFF) < trigger) {
|
|
// This LED is off, but was randomly selected to start brightening
|
|
if (random_color) {
|
|
c->h = rand() % 0xFF;
|
|
c->s = (rand() % (rgblight_config.sat / 2)) + (rgblight_config.sat / 2);
|
|
}
|
|
c->v = 0;
|
|
t->max_life = MAX(20, MIN(RGBLIGHT_EFFECT_TWINKLE_LIFE, rgblight_config.val));
|
|
t->life = t->max_life;
|
|
} else {
|
|
// This LED is off, and was NOT selected to start brightening
|
|
}
|
|
|
|
sethsv(c->h, c->s, c->v, i + rgblight_ranges.effect_start_pos);
|
|
}
|
|
|
|
rgblight_set();
|
|
}
|
|
#endif
|
|
|
|
void preprocess_rgblight(void) {
|
|
#ifdef VELOCIKEY_ENABLE
|
|
if (rgblight_velocikey_enabled()) {
|
|
rgblight_velocikey_accelerate();
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void rgblight_task(void) {
|
|
#ifdef RGBLIGHT_USE_TIMER
|
|
rgblight_timer_task();
|
|
#endif
|
|
|
|
#ifdef VELOCIKEY_ENABLE
|
|
if (rgblight_velocikey_enabled()) {
|
|
rgblight_velocikey_decelerate();
|
|
}
|
|
#endif
|
|
}
|
|
|
|
#ifdef VELOCIKEY_ENABLE
|
|
# define TYPING_SPEED_MAX_VALUE 200
|
|
|
|
static uint8_t typing_speed = 0;
|
|
|
|
bool rgblight_velocikey_enabled(void) {
|
|
return rgblight_config.velocikey;
|
|
}
|
|
|
|
void rgblight_velocikey_toggle(void) {
|
|
dprintf("rgblight velocikey toggle [EEPROM]: rgblight_config.velocikey = %u\n", !rgblight_config.velocikey);
|
|
rgblight_config.velocikey = !rgblight_config.velocikey;
|
|
eeconfig_update_rgblight_current();
|
|
}
|
|
|
|
void rgblight_velocikey_accelerate(void) {
|
|
if (typing_speed < TYPING_SPEED_MAX_VALUE) typing_speed += (TYPING_SPEED_MAX_VALUE / 100);
|
|
}
|
|
|
|
void rgblight_velocikey_decelerate(void) {
|
|
static uint16_t decay_timer = 0;
|
|
|
|
if (timer_elapsed(decay_timer) > 500 || decay_timer == 0) {
|
|
if (typing_speed > 0) typing_speed -= 1;
|
|
// Decay a little faster at half of max speed
|
|
if (typing_speed > TYPING_SPEED_MAX_VALUE / 2) typing_speed -= 1;
|
|
// Decay even faster at 3/4 of max speed
|
|
if (typing_speed > TYPING_SPEED_MAX_VALUE / 4 * 3) typing_speed -= 2;
|
|
decay_timer = timer_read();
|
|
}
|
|
}
|
|
|
|
uint8_t rgblight_velocikey_match_speed(uint8_t minValue, uint8_t maxValue) {
|
|
return MAX(minValue, maxValue - (maxValue - minValue) * ((float)typing_speed / TYPING_SPEED_MAX_VALUE));
|
|
}
|
|
|
|
#endif
|