Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Allow passing debounce_threshold in keypad.py #1044

Merged
merged 18 commits into from
Nov 8, 2024
Merged

Conversation

regicidalplutophage
Copy link
Member

This is needed for proper debouncing

@regicidalplutophage
Copy link
Member Author

I'm testing this on a direct pin keyboard and I've got no complaints. Maybe new defaults are still too conservative even? But then I didn't have any issues with key chatter anyway, so I'm not the target audience for debouncing...

@@ -30,7 +30,8 @@ class MyKeyboard(KMKKeyboard):
row_pins=self.row_pins,
# optional arguments with defaults:
columns_to_anodes=DiodeOrientation.COL2ROW,
interval=0.02, # Debounce time in floating point seconds
interval=0.01, # How often the matrix is sampled
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
interval=0.01, # How often the matrix is sampled
interval=0.01, # Matrix sampling interval in ms

Same for the other parameter lists. ("how often" means frequency btw, which is the inverse).

@@ -30,7 +30,8 @@ class MyKeyboard(KMKKeyboard):
row_pins=self.row_pins,
# optional arguments with defaults:
columns_to_anodes=DiodeOrientation.COL2ROW,
interval=0.02, # Debounce time in floating point seconds
interval=0.01, # How often the matrix is sampled
debounce_threshold=5, # Number of samples needed to change state
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I have a strong opinion on keeping the upstream default, even if it's "maybe suboptimal". Make a note in the docs that values around 5 are tested and should work.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The way KMK is used we'd need this information in getting started... I'd prefer the minimum viable config to remain within its current scope.

@regicidalplutophage
Copy link
Member Author

Unless you budge on defaults, getting started still needs some rewriting to point to porting guide and scanners. I'm pooped though.

Comment on lines 52 to 65
]
for i in args:
if i is None:
args.pop(i)
kwargs = {
'columns_to_anodes': columns_to_anodes,
'interval': interval,
'debounce_threshold': debounce_threshold,
'max_events': max_events,
}
for key, value in kwargs.items():
if value is None:
kwargs.pop(key)
self.keypad = keypad.Keys(*args, **kwargs)
Copy link
Collaborator

@xs5871 xs5871 Nov 8, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

oh dear. That is much too complicated for my taste. Do we want backwards compatibility with CP8?

  1. No: none if this is necessary
  2. Yes: Since we're not doing any validation nor any abstraction, just use pokemon arguments:
def __init__(self, *args, **kwargs):
    self.keypad = keypad.Keys(*args, **kwargs)

style sidenote: in any case, this screams comprehension:

args = [v for v in args if v is not None]
kwargs = {k:v for k,v in kwargs.items() if v is not None}

Copy link
Member Author

@regicidalplutophage regicidalplutophage Nov 8, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I didn't go with comprehension because init had explicit arguments. I do like the pokemon arguments though, not sure why it wasn't that in the first place -- referring both to my PR and the main branch.


Right, the too complicated stuff came from the place of wanting to provide non-upstream defaults then have the ability to None-out those defaults if it has to work in earlier versions of circuitpython.

docs/en/scanners.md Show resolved Hide resolved
@regicidalplutophage regicidalplutophage changed the title Expose debounce_threshold in keypad.py Allow passing debounce_threshold in keypad.py Nov 8, 2024
@xs5871 xs5871 merged commit 643afc9 into main Nov 8, 2024
2 checks passed
@xs5871 xs5871 deleted the regicidalplutophage-patch-1 branch November 8, 2024 23:32
@DaVarga
Copy link

DaVarga commented Nov 10, 2024

This line seems to break the main branch
https://github.com/KMKfw/kmk_firmware/pull/1044/files#diff-47931159c3b3d0e39f59c6f47a3d0e133ebffb04b8dafd078d79d5ba0f3bb5f8R52
Right now it throws "TypeError: 'pins' argument required".
Maybe this line should be self.keypad = keypad.KeyMatrix(*args, **kwargs) ?

Same goes for ShiftRegisterKeys

@xs5871 xs5871 mentioned this pull request Nov 10, 2024
@regicidalplutophage
Copy link
Member Author

Well, that's embarassing. The fix is merged now!

@Esnos33
Copy link

Esnos33 commented Dec 15, 2024

Hi @regicidalplutophage I wanted to try out your pull request regarding sticky keys, but when I updated my kmk and circuitpython, I got new error that is related to this pr, do you know how to fix this?
Minimal config for this example

import board

from kmk.keys import KC
from storage import getmount
from kmk.kmk_keyboard import KMKKeyboard as _KMKKeyboard
from kmk.scanners.keypad import KeysScanner

from kmk.modules.layers import Layers
from kmk.modules.split import Split, SplitSide
from kmk.modules.sticky_keys import StickyKeys

name = str(getmount("/").label)
is_right = True if name.endswith("R") else False

_KEY_CFG_LEFT = [
    board.GP1, board.GP2, board.GP5, board.GP4, board.GP3,
    board.GP6, board.GP27, board.GP7, board.GP26, board.GP8,
    board.GP9, board.GP21, board.GP10, board.GP20, board.GP11,
    board.GP19, board.GP12, board.GP18,
]

_KEY_CFG_RIGHT = [
    board.GP3, board.GP4, board.GP5, board.GP2, board.GP1,
    board.GP8, board.GP26, board.GP7, board.GP27, board.GP6,
    board.GP11, board.GP20, board.GP10, board.GP21, board.GP9,
    board.GP18, board.GP12, board.GP19,
]

class KMKKeyboard(_KMKKeyboard):
    def __init__(self):
        self.matrix = KeysScanner(
            pins=_KEY_CFG_RIGHT if is_right else _KEY_CFG_LEFT
        )

    # fmt: off
    coord_mapping = [
        0, 1, 2, 3, 4,      18, 19, 20, 21, 22,
        5, 6, 7, 8, 9,      23, 24, 25, 26, 27,
        10, 11, 12, 13, 14,  28, 29, 30, 31, 32, # noqa
        15, 16, 17,  33, 34, 35 # noqa

    ]

keyboard = KMKKeyboard() # adding "value_when_pressed=False" as argument doesn't work
keyboard.debug_enabled = True

# Layer
keyboard.modules.append(Layers())

# one-shot
sticky_keys = StickyKeys(release_after=2000)
keyboard.modules.append(sticky_keys)

# Split
split_side = SplitSide.RIGHT if is_right else SplitSide.LEFT
split = Split(
    split_side=split_side,
    split_target_left=True,
    uart_interval=20,
    data_pin=board.GP17,  # RX
    data_pin2=board.GP16,  # TX
    uart_flip=False,
    use_pio=False,
)
keyboard.modules.append(split)

def sticky(x):
    return KC.STICKY(x, defer_release=True, retap_cancel=True)

keyboard.keymap = [
    # layer 0
    [
        KC.Q, KC.W, KC.E, KC.R, KC.T, KC.Y, KC.U, KC.I, KC.O, KC.P,
        KC.A, KC.S, KC.D, KC.F, KC.G, KC.H, KC.J, KC.K, KC.L, KC.ESC,
        KC.Z, KC.X, KC.C, KC.V, KC.B, KC.N, KC.M, KC.COMM, KC.DOT, KC.SLSH,
        KC.LCTRL, KC.MO(1), KC.MO(4), KC.SPC, KC.MO(2), KC.MO(5),
    ],
    # layer 1
    [
        KC.LGUI, KC.LCTL(KC.W), KC.LCTL(KC.D), KC.LCTL(KC.U), KC.LCTL(KC.T), KC.PGUP, KC.HOME, KC.UP, KC.END, KC.TO(0),
        sticky(KC.LALT), sticky(KC.LGUI), sticky(KC.LSFT), sticky(KC.LCTL), KC.LCTL(KC.F), KC.PGDN, KC.LEFT, KC.DOWN, KC.RIGHT, KC.TRNS,
        KC.LCTL(KC.Z), KC.LCTL(KC.X), KC.LCTL(KC.C), KC.LCTL(KC.V), KC.PSCR, KC.SPC, KC.ENT, KC.TAB, KC.DEL, KC.TRNS,
        KC.TRNS, KC.TRNS, KC.TRNS, KC.BSPC, KC.MO(3), KC.TRNS,
    ],
]

if __name__ == '__main__':
    keyboard.go()

I get error:

Traceback (most recent call last):
  File "main.py", line 44, in <module>
  File "main.py", line 32, in __init__
  File "kmk/scanners/keypad.py", line 52, in __init__
TypeError: 'value_when_pressed' argument required

Code done running.

@regicidalplutophage
Copy link
Member Author

Hi @Esnos33!

The value_when_pressed is an argument for the keypad KeysScanner. This PR has made it required, we need to update it in the docs. Good catch!

In your case the line 29 should be:

class KMKKeyboard(_KMKKeyboard):
    def __init__(self):
        self.matrix = KeysScanner(
            pins=_KEY_CFG_RIGHT if is_right else _KEY_CFG_LEFT,
            value_when_pressed=False,
        )

And if you want to make use of this PR:

class KMKKeyboard(_KMKKeyboard):
    def __init__(self):
        self.matrix = KeysScanner(
            pins=_KEY_CFG_RIGHT if is_right else _KEY_CFG_LEFT,
            value_when_pressed=False,
            interval=0.01, 
            debounce_threshold=3,
        )

@Esnos33
Copy link

Esnos33 commented Dec 16, 2024

@regicidalplutophage with this change everything works now, thank you for your contributions!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants