2nd-Level Encoder Debouncing for scrappy encoder dials (#1838)

* Experimental encoder debouncing

* Experimental encoder debouncing

* Clang

* Comment changed
This commit is contained in:
Mark Thompson 2024-02-01 14:13:28 -06:00 committed by GitHub
parent a2a5fb166e
commit ee2e57d702
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
3 changed files with 45 additions and 71 deletions

View File

@ -26,77 +26,51 @@
#include "portapack.hpp" #include "portapack.hpp"
#include "portapack_persistent_memory.hpp" #include "portapack_persistent_memory.hpp"
// Now supporting multiple levels of rotary encoder dial sensitivity // Transition map for rotary encoder phase bits
// // 00 -> 01 -> 11 -> 10 cw
// Portapack H2 normally has a 30-step encoder, meaning one step (pulse) every // 00 -> 10 -> 11 -> 01 ccw
// 12 degrees of rotation. // NB: Bits are swapped versus older FW versions
// static const int8_t transition_map[16] = {
// For each encoder "pulse" there are 4 state transitions, and we can choose 0, // 00->00: noop
// between looking at all of them (high sensitivity), half of them (medium/default), 1, // 00->01: cw start
// or one quarter of them (low sensitivity). -1, // 00->10: ccw start
static const int8_t transition_map[][16] = { 0, // 00->11: rate
// Normal (Medium) Sensitivity -- default -1, // 01->00: ccw end
{ 0, // 01->01: noop
0, // 0000: noop 0, // 01->10: rate
0, // 0001: ccw start 1, // 01->11: cw end
0, // 0010: cw start 1, // 10->00: cw end
0, // 0011: rate 0, // 10->01: rate
1, // 0100: cw end 0, // 10->10: noop
0, // 0101: noop -1, // 10->11: ccw end
0, // 0110: rate 0, // 11->00: rate
-1, // 0111: ccw end -1, // 11->01: ccw start
-1, // 1000: ccw end 1, // 11->10: cw start
0, // 1001: rate 0, // 11->11: noop
0, // 1010: noop };
1, // 1011: cw end
0, // 1100: rate // Rotary encoder dial sensitivity (transition ignored if bit is 0)
0, // 1101: cw start static const uint16_t sensitivity_map[] = {
0, // 1110: ccw start 0x0990, // DIAL_SENSITIVITY_NORMAL
0, // 1111: noop 0x0110, // DIAL_SENSITIVITY_LOW
}, 0x6996, // DIAL_SENSITIVITY_HIGH
// Low Sensitivity
{
0, // 0000: noop
0, // 0001: ccw start
0, // 0010: cw start
0, // 0011: rate
1, // 0100: cw end
0, // 0101: noop
0, // 0110: rate
0, // 0111: ccw end
-1, // 1000: ccw end
0, // 1001: rate
0, // 1010: noop
0, // 1011: cw end
0, // 1100: rate
0, // 1101: cw start
0, // 1110: ccw start
0, // 1111: noop
},
// High Sensitivity
{
0, // 0000: noop
-1, // 0001: ccw start
1, // 0010: cw start
0, // 0011: rate
1, // 0100: cw end
0, // 0101: noop
0, // 0110: rate
-1, // 0111: ccw end
-1, // 1000: ccw end
0, // 1001: rate
0, // 1010: noop
1, // 1011: cw end
0, // 1100: rate
1, // 1101: cw start
-1, // 1110: ccw start
0, // 1111: noop
},
}; };
int_fast8_t Encoder::update(const uint_fast8_t phase_bits) { int_fast8_t Encoder::update(const uint_fast8_t phase_bits) {
state = (state << 2) | phase_bits; state = ((state << 2) | phase_bits) & 0x0F;
// dial sensitivity setting is stored in pmem int_fast8_t direction = transition_map[state];
return transition_map[portapack::persistent_memory::config_encoder_dial_sensitivity()][state & 0xf];
// Require 2 state changes in same direction to register movement -- for additional level of contact switch debouncing
if (direction == prev_direction) {
if ((sensitivity_map[portapack::persistent_memory::config_encoder_dial_sensitivity()] & (1 << state)) == 0)
return 0;
return direction;
}
// It's normal for transition map to return 0 between every +1/-1, so discarding those
if (direction != 0)
prev_direction = direction;
return 0;
} }

View File

@ -30,6 +30,7 @@ class Encoder {
private: private:
uint_fast8_t state{0}; uint_fast8_t state{0};
int_fast8_t prev_direction{0};
}; };
#endif /*__ENCODER_H__*/ #endif /*__ENCODER_H__*/

View File

@ -162,8 +162,7 @@ static bool switches_update(const uint8_t raw) {
} }
static bool encoder_update(const uint8_t raw) { static bool encoder_update(const uint8_t raw) {
// TODO: swap +1/-1 directions in encoder.update() to avoid needless swizzing of phase bits here return encoder_debounce.feed(raw >> 6);
return encoder_debounce.feed(((raw >> 7) & 0x01) | ((raw >> 5) & 0x02));
} }
static bool encoder_read() { static bool encoder_read() {