|
35 | 35 | #include "opentx.h"
|
36 | 36 | #endif
|
37 | 37 |
|
| 38 | +#if ROTARY_ENCODER_GRANULARITY == 2 |
| 39 | + #define ON_DETENT(p) ((p == 3) || (p == 0)) |
| 40 | +#elif ROTARY_ENCODER_GRANULARITY == 4 |
| 41 | + #define ON_DETENT(p) (p == 3) |
| 42 | +#elif |
| 43 | +#error "Unknown ROTARY_ENCODER_GRANULARITY" |
| 44 | +#endif |
| 45 | + |
38 | 46 | volatile rotenc_t rotencValue = 0;
|
39 | 47 | volatile uint32_t rotencDt = 0;
|
40 | 48 |
|
41 |
| -#if defined(BOOT) |
42 |
| -#define INC_ROT 1 |
43 |
| -#define INC_ROT_2 2 |
44 |
| -#else |
45 |
| -#define INC_ROT \ |
46 |
| - (g_eeGeneral.rotEncMode == ROTARY_ENCODER_MODE_INVERT_BOTH ? -1 : 1); |
47 |
| -#define INC_ROT_2 \ |
48 |
| - (g_eeGeneral.rotEncMode == ROTARY_ENCODER_MODE_INVERT_BOTH ? -2 : 2); |
49 |
| -#endif |
| 49 | +// Last encoder pins state |
| 50 | +static uint8_t lastPins = 0; |
| 51 | +// Record encoder position change between detents |
| 52 | +int8_t reChgPos = 0; |
| 53 | +// Used on start to ignore movement until encoder position on detent |
| 54 | +bool skipUntilDetent = false; |
50 | 55 |
|
51 | 56 | rotenc_t rotaryEncoderGetValue()
|
52 |
| -{ |
53 |
| - return rotencValue / ROTARY_ENCODER_GRANULARITY; |
54 |
| -} |
55 |
| - |
56 |
| -rotenc_t rotaryEncoderGetRawValue() |
57 | 57 | {
|
58 | 58 | return rotencValue;
|
59 | 59 | }
|
60 | 60 |
|
61 | 61 | void rotaryEncoderCheck()
|
62 | 62 | {
|
63 |
| - static uint8_t state = 0; |
64 |
| - static uint8_t re_count = 0; |
| 63 | + // Value increment for each state transition of the RE pins |
| 64 | +#if defined(ROTARY_ENCODER_INVERTED) |
| 65 | + static int8_t reInc[4][4] = { |
| 66 | + // Prev = 0 |
| 67 | + { 0, -1, 1, -2 }, |
| 68 | + // Prev = 1 |
| 69 | + { 1, 0, 0, -1 }, |
| 70 | + // Prev = 2 |
| 71 | + { -1, 0, 0, 1 }, |
| 72 | + // Prev = 3 |
| 73 | + { 2, 1, -1, 0 }, |
| 74 | + }; |
| 75 | +#else |
| 76 | + static int8_t reInc[4][4] = { |
| 77 | + // Prev = 0 |
| 78 | + { 0, 1, -1, 2 }, |
| 79 | + // Prev = 1 |
| 80 | + { -1, 0, 0, 1 }, |
| 81 | + // Prev = 2 |
| 82 | + { 1, 0, 0, -1 }, |
| 83 | + // Prev = 3 |
| 84 | + { -2, -1, 1, 0 }, |
| 85 | + }; |
| 86 | +#endif |
| 87 | + |
65 | 88 | uint8_t pins = ROTARY_ENCODER_POSITION();
|
66 | 89 |
|
67 |
| -#if defined(ROTARY_ENCODER_SUPPORT_BUGGY_WIRING) |
68 |
| - if (pins != (state & 0x03) && !(readKeys() & (1 << KEY_ENTER))) { |
69 |
| - if (re_count == 0) { |
70 |
| - // Need at least 2 values to correctly determine initial direction |
71 |
| - re_count = 1; |
72 |
| - } else { |
73 |
| - if ((pins ^ (state & 0x03)) == 0x03) { |
74 |
| - if (pins == 3) { |
75 |
| - rotencValue += INC_ROT_2; |
76 |
| - } else { |
77 |
| - rotencValue -= INC_ROT_2; |
78 |
| - } |
79 |
| - } else { |
80 |
| - if ((state & 0x01) ^ ((pins & 0x02) >> 1)) { |
81 |
| - rotencValue -= INC_ROT; |
82 |
| - } else { |
83 |
| - rotencValue += INC_ROT; |
84 |
| - } |
85 |
| - } |
86 |
| - |
87 |
| - if (re_count == 1) |
88 |
| - { |
89 |
| - re_count = 2; |
90 |
| - // Assume 1st value is same direction as 2nd value |
91 |
| - rotencValue = rotencValue * 2; |
92 |
| - } |
| 90 | + // No change - do nothing |
| 91 | + if (pins == lastPins) { |
| 92 | + return; |
| 93 | + } |
| 94 | + |
| 95 | + // Handle case where radio started with encoder not on detent position |
| 96 | + if (skipUntilDetent) { |
| 97 | + if (ON_DETENT(pins)) { |
| 98 | + lastPins = pins; |
| 99 | + skipUntilDetent = false; |
93 | 100 | }
|
94 |
| - state &= ~0x03; |
95 |
| - state |= pins; |
| 101 | + return; |
96 | 102 | }
|
97 |
| -#else |
98 |
| - if (pins != state && !(readKeys() & (1 << KEY_ENTER))) { |
99 |
| - if (re_count == 0) { |
100 |
| - // Need at least 2 values to correctly determine initial direction |
101 |
| - re_count = 1; |
102 |
| - } else { |
103 |
| -#if defined(ROTARY_ENCODER_INVERTED) |
104 |
| - if (!(state & 0x01) ^ ((pins & 0x02) >> 1)) { |
105 |
| -#else |
106 |
| - if ((state & 0x01) ^ ((pins & 0x02) >> 1)) { |
| 103 | + |
| 104 | + // Get increment value for pin state transition |
| 105 | + int inc = reInc[lastPins][pins]; |
| 106 | + |
| 107 | +#if !defined(BOOT) |
| 108 | + if (g_eeGeneral.rotEncMode == ROTARY_ENCODER_MODE_INVERT_BOTH) |
| 109 | + inc = -inc; |
107 | 110 | #endif
|
108 |
| - rotencValue -= INC_ROT; |
109 |
| - } else { |
110 |
| - rotencValue += INC_ROT; |
111 |
| - } |
112 |
| - |
113 |
| - if (re_count == 1) |
114 |
| - { |
115 |
| - re_count = 2; |
116 |
| - // Assume 1st value is same direction as 2nd value |
117 |
| - rotencValue = rotencValue * 2; |
118 |
| - } |
| 111 | + |
| 112 | + // Update position change between detents |
| 113 | + reChgPos += inc; |
| 114 | + |
| 115 | + // Update reported value on full detent change |
| 116 | + if (reChgPos >= ROTARY_ENCODER_GRANULARITY) { |
| 117 | + // If ENTER pressed - ignore scrolling |
| 118 | + if ((readKeys() & (1 << KEY_ENTER)) == 0) { |
| 119 | + rotencValue += 1; |
| 120 | + } |
| 121 | + reChgPos -= ROTARY_ENCODER_GRANULARITY; |
| 122 | + } else if (reChgPos <= -ROTARY_ENCODER_GRANULARITY) { |
| 123 | + // If ENTER pressed - ignore scrolling |
| 124 | + if ((readKeys() & (1 << KEY_ENTER)) == 0) { |
| 125 | + rotencValue -= 1; |
119 | 126 | }
|
120 |
| - state = pins; |
| 127 | + reChgPos += ROTARY_ENCODER_GRANULARITY; |
121 | 128 | }
|
122 |
| -#endif |
| 129 | + |
| 130 | + lastPins = pins; |
123 | 131 |
|
124 | 132 | #if !defined(BOOT) && defined(COLORLCD)
|
125 | 133 | static uint32_t last_tick = 0;
|
126 | 134 | static rotenc_t last_value = 0;
|
127 | 135 |
|
128 | 136 | rotenc_t value = rotencValue;
|
129 |
| - rotenc_t diff = (value - last_value) / ROTARY_ENCODER_GRANULARITY; |
| 137 | + rotenc_t diff = (value - last_value); |
130 | 138 |
|
131 | 139 | if (diff != 0) {
|
132 | 140 | uint32_t now = RTOS_GET_MS();
|
133 | 141 | uint32_t dt = now - last_tick;
|
134 | 142 | // pre-compute accumulated dt (dx/dt is done later in LVGL driver)
|
135 | 143 | rotencDt += dt;
|
136 | 144 | last_tick = now;
|
137 |
| - last_value += diff * ROTARY_ENCODER_GRANULARITY; |
| 145 | + last_value = value; |
138 | 146 | }
|
139 | 147 | #endif
|
140 | 148 | }
|
@@ -181,6 +189,10 @@ void rotaryEncoderInit()
|
181 | 189 |
|
182 | 190 | NVIC_EnableIRQ(ROTARY_ENCODER_TIMER_IRQn);
|
183 | 191 | NVIC_SetPriority(ROTARY_ENCODER_TIMER_IRQn, 7);
|
| 192 | + |
| 193 | + // Get initial position |
| 194 | + lastPins = ROTARY_ENCODER_POSITION(); |
| 195 | + skipUntilDetent = !ON_DETENT(lastPins); |
184 | 196 | }
|
185 | 197 |
|
186 | 198 | extern "C" void ROTARY_ENCODER_TIMER_IRQHandler(void)
|
|
0 commit comments