2022-03-28 16:55:58 +02:00
|
|
|
#include "kbI2cBase.h"
|
2022-05-07 20:31:21 +10:00
|
|
|
#include "configuration.h"
|
2023-03-08 19:13:46 -08:00
|
|
|
#include "detect/ScanI2C.h"
|
2024-05-19 19:32:08 +02:00
|
|
|
#include "detect/ScanI2CTwoWire.h"
|
2022-03-28 16:55:58 +02:00
|
|
|
|
2025-07-21 19:33:24 +02:00
|
|
|
#if defined(T_DECK_PRO)
|
|
|
|
|
#include "TDeckProKeyboard.h"
|
|
|
|
|
#elif defined(T_LORA_PAGER)
|
|
|
|
|
#include "TLoraPagerKeyboard.h"
|
2025-11-30 17:21:10 -06:00
|
|
|
#elif defined(HACKADAY_COMMUNICATOR)
|
|
|
|
|
#include "HackadayCommunicatorKeyboard.h"
|
2025-07-21 19:33:24 +02:00
|
|
|
#else
|
|
|
|
|
#include "TCA8418Keyboard.h"
|
|
|
|
|
#endif
|
|
|
|
|
|
2023-03-08 19:13:46 -08:00
|
|
|
extern ScanI2C::DeviceAddress cardkb_found;
|
2022-11-13 14:56:52 +01:00
|
|
|
extern uint8_t kb_model;
|
2022-09-06 15:58:33 +08:00
|
|
|
|
2025-07-21 19:33:24 +02:00
|
|
|
KbI2cBase::KbI2cBase(const char *name)
|
|
|
|
|
: concurrency::OSThread(name),
|
|
|
|
|
#if defined(T_DECK_PRO)
|
|
|
|
|
TCAKeyboard(*(new TDeckProKeyboard()))
|
|
|
|
|
#elif defined(T_LORA_PAGER)
|
|
|
|
|
TCAKeyboard(*(new TLoraPagerKeyboard()))
|
2025-11-30 17:21:10 -06:00
|
|
|
#elif defined(HACKADAY_COMMUNICATOR)
|
|
|
|
|
TCAKeyboard(*(new HackadayCommunicatorKeyboard()))
|
2025-07-21 19:33:24 +02:00
|
|
|
#else
|
|
|
|
|
TCAKeyboard(*(new TCA8418Keyboard()))
|
|
|
|
|
#endif
|
2022-03-28 16:55:58 +02:00
|
|
|
{
|
2026-01-03 21:19:24 +01:00
|
|
|
this->_originName = name;
|
2022-03-28 16:55:58 +02:00
|
|
|
}
|
|
|
|
|
|
2026-01-03 21:19:24 +01:00
|
|
|
uint8_t read_from_14004(TwoWire *i2cBus, uint8_t reg, uint8_t *data, uint8_t length) {
|
|
|
|
|
uint8_t readflag = 0;
|
|
|
|
|
i2cBus->beginTransmission(CARDKB_ADDR);
|
|
|
|
|
i2cBus->write(reg);
|
|
|
|
|
i2cBus->endTransmission(); // stop transmitting
|
|
|
|
|
delay(20);
|
|
|
|
|
i2cBus->requestFrom(CARDKB_ADDR, (int)length);
|
|
|
|
|
int i = 0;
|
|
|
|
|
while (i2cBus->available()) // slave may send less than requested
|
|
|
|
|
{
|
|
|
|
|
data[i++] = i2cBus->read(); // receive a byte as a proper uint8_t
|
|
|
|
|
readflag = 1;
|
|
|
|
|
}
|
|
|
|
|
return readflag;
|
2022-10-06 16:00:33 +02:00
|
|
|
}
|
|
|
|
|
|
2026-01-03 21:19:24 +01:00
|
|
|
int32_t KbI2cBase::runOnce() {
|
|
|
|
|
if (!i2cBus) {
|
|
|
|
|
switch (cardkb_found.port) {
|
|
|
|
|
case ScanI2C::WIRE1:
|
2024-09-18 03:28:23 +12:00
|
|
|
#if WIRE_INTERFACES_COUNT == 2
|
2026-01-03 21:19:24 +01:00
|
|
|
LOG_DEBUG("Use I2C Bus 1 (the second one)");
|
|
|
|
|
i2cBus = &Wire1;
|
|
|
|
|
if (cardkb_found.address == BBQ10_KB_ADDR) {
|
|
|
|
|
Q10keyboard.begin(BBQ10_KB_ADDR, &Wire1);
|
|
|
|
|
Q10keyboard.setBacklight(0);
|
|
|
|
|
}
|
|
|
|
|
if (cardkb_found.address == MPR121_KB_ADDR) {
|
|
|
|
|
MPRkeyboard.begin(MPR121_KB_ADDR, &Wire1);
|
|
|
|
|
}
|
|
|
|
|
if (cardkb_found.address == TCA8418_KB_ADDR) {
|
|
|
|
|
TCAKeyboard.begin(TCA8418_KB_ADDR, &Wire1);
|
|
|
|
|
}
|
|
|
|
|
break;
|
2023-03-08 19:13:46 -08:00
|
|
|
#endif
|
2026-01-03 21:19:24 +01:00
|
|
|
case ScanI2C::WIRE:
|
|
|
|
|
LOG_DEBUG("Use I2C Bus 0 (the first one)");
|
|
|
|
|
i2cBus = &Wire;
|
|
|
|
|
if (cardkb_found.address == BBQ10_KB_ADDR) {
|
|
|
|
|
Q10keyboard.begin(BBQ10_KB_ADDR, &Wire);
|
|
|
|
|
Q10keyboard.setBacklight(0);
|
|
|
|
|
}
|
|
|
|
|
if (cardkb_found.address == MPR121_KB_ADDR) {
|
|
|
|
|
MPRkeyboard.begin(MPR121_KB_ADDR, &Wire);
|
|
|
|
|
}
|
|
|
|
|
if (cardkb_found.address == TCA8418_KB_ADDR) {
|
|
|
|
|
TCAKeyboard.begin(TCA8418_KB_ADDR, &Wire);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case ScanI2C::NO_I2C:
|
|
|
|
|
default:
|
|
|
|
|
i2cBus = 0;
|
2023-03-08 19:13:46 -08:00
|
|
|
}
|
2026-01-03 21:19:24 +01:00
|
|
|
}
|
2023-03-08 19:13:46 -08:00
|
|
|
|
2026-01-03 21:19:24 +01:00
|
|
|
switch (kb_model) {
|
|
|
|
|
case 0x11: { // BB Q10
|
|
|
|
|
int keyCount = Q10keyboard.keyCount();
|
|
|
|
|
while (keyCount--) {
|
|
|
|
|
const BBQ10Keyboard::KeyEvent key = Q10keyboard.keyEvent();
|
|
|
|
|
if ((key.key != 0x00) && (key.state == BBQ10Keyboard::StateRelease)) {
|
|
|
|
|
InputEvent e = {};
|
|
|
|
|
e.inputEvent = INPUT_BROKER_NONE;
|
|
|
|
|
e.source = this->_originName;
|
|
|
|
|
switch (key.key) {
|
|
|
|
|
case 'p': // TAB
|
|
|
|
|
case 't': // TAB as well
|
|
|
|
|
if (is_sym) {
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = 0x09; // TAB Scancode
|
|
|
|
|
is_sym = false; // reset sym state after second keypress
|
|
|
|
|
} else {
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = key.key;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 'q': // ESC
|
|
|
|
|
if (is_sym) {
|
|
|
|
|
e.inputEvent = INPUT_BROKER_CANCEL;
|
|
|
|
|
e.kbchar = 0;
|
|
|
|
|
is_sym = false; // reset sym state after second keypress
|
|
|
|
|
} else {
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = key.key;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 0x08: // Back
|
|
|
|
|
e.inputEvent = INPUT_BROKER_BACK;
|
|
|
|
|
e.kbchar = key.key;
|
|
|
|
|
break;
|
|
|
|
|
case 'e': // sym e
|
|
|
|
|
if (is_sym) {
|
|
|
|
|
e.inputEvent = INPUT_BROKER_UP;
|
|
|
|
|
e.kbchar = INPUT_BROKER_UP;
|
|
|
|
|
is_sym = false; // reset sym state after second keypress
|
|
|
|
|
} else {
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = key.key;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 'x': // sym x
|
|
|
|
|
if (is_sym) {
|
|
|
|
|
e.inputEvent = INPUT_BROKER_DOWN;
|
|
|
|
|
e.kbchar = 0;
|
|
|
|
|
is_sym = false; // reset sym state after second keypress
|
|
|
|
|
} else {
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = key.key;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 's': // sym s
|
|
|
|
|
if (is_sym) {
|
|
|
|
|
e.inputEvent = INPUT_BROKER_LEFT;
|
|
|
|
|
e.kbchar = 0x00; // tweak for destSelect
|
|
|
|
|
is_sym = false; // reset sym state after second keypress
|
|
|
|
|
} else {
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = key.key;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 'f': // sym f
|
|
|
|
|
if (is_sym) {
|
|
|
|
|
e.inputEvent = INPUT_BROKER_RIGHT;
|
|
|
|
|
e.kbchar = 0x00; // tweak for destSelect
|
|
|
|
|
is_sym = false; // reset sym state after second keypress
|
|
|
|
|
} else {
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = key.key;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 0x13: // Code scanner says the SYM key is 0x13
|
|
|
|
|
is_sym = !is_sym;
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = is_sym ? INPUT_BROKER_MSG_FN_SYMBOL_ON // send 0xf1 to tell CannedMessages to display that
|
|
|
|
|
: INPUT_BROKER_MSG_FN_SYMBOL_OFF; // the modifier key is active
|
|
|
|
|
break;
|
|
|
|
|
case 0x0a: // apparently Enter on Q10 is a line feed instead of carriage return
|
|
|
|
|
e.inputEvent = INPUT_BROKER_SELECT;
|
|
|
|
|
break;
|
|
|
|
|
case 0x00: // nopress
|
|
|
|
|
e.inputEvent = INPUT_BROKER_NONE;
|
|
|
|
|
break;
|
|
|
|
|
default: // all other keys
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = key.key;
|
|
|
|
|
is_sym = false; // reset sym state after second keypress
|
|
|
|
|
break;
|
|
|
|
|
}
|
2023-08-19 15:37:42 +02:00
|
|
|
|
2026-01-03 21:19:24 +01:00
|
|
|
if (e.inputEvent != INPUT_BROKER_NONE) {
|
|
|
|
|
this->notifyObservers(&e);
|
2023-08-19 15:37:42 +02:00
|
|
|
}
|
2026-01-03 21:19:24 +01:00
|
|
|
}
|
2023-08-19 15:37:42 +02:00
|
|
|
}
|
2026-01-03 21:19:24 +01:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case 0x37: { // MPR121
|
|
|
|
|
MPRkeyboard.trigger();
|
|
|
|
|
InputEvent e = {};
|
2024-10-21 16:53:36 +10:00
|
|
|
|
2026-01-03 21:19:24 +01:00
|
|
|
while (MPRkeyboard.hasEvent()) {
|
|
|
|
|
char nextEvent = MPRkeyboard.dequeueEvent();
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = 0x00;
|
|
|
|
|
e.source = this->_originName;
|
|
|
|
|
switch (nextEvent) {
|
|
|
|
|
case 0x00: // MPR121_NONE
|
|
|
|
|
e.inputEvent = INPUT_BROKER_NONE;
|
|
|
|
|
e.kbchar = 0x00;
|
|
|
|
|
break;
|
|
|
|
|
case 0x90: // MPR121_REBOOT
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = INPUT_BROKER_MSG_REBOOT;
|
|
|
|
|
break;
|
|
|
|
|
case 0xb4: // MPR121_LEFT
|
|
|
|
|
e.inputEvent = INPUT_BROKER_LEFT;
|
|
|
|
|
e.kbchar = 0x00;
|
|
|
|
|
break;
|
|
|
|
|
case 0xb5: // MPR121_UP
|
|
|
|
|
e.inputEvent = INPUT_BROKER_UP;
|
|
|
|
|
e.kbchar = 0x00;
|
|
|
|
|
break;
|
|
|
|
|
case 0xb6: // MPR121_DOWN
|
|
|
|
|
e.inputEvent = INPUT_BROKER_DOWN;
|
|
|
|
|
e.kbchar = 0x00;
|
|
|
|
|
break;
|
|
|
|
|
case 0xb7: // MPR121_RIGHT
|
|
|
|
|
e.inputEvent = INPUT_BROKER_RIGHT;
|
|
|
|
|
e.kbchar = 0x00;
|
|
|
|
|
break;
|
|
|
|
|
case 0x1b: // MPR121_ESC
|
|
|
|
|
e.inputEvent = INPUT_BROKER_CANCEL;
|
|
|
|
|
e.kbchar = 0;
|
|
|
|
|
break;
|
|
|
|
|
case 0x08: // MPR121_BSP
|
|
|
|
|
e.inputEvent = INPUT_BROKER_BACK;
|
|
|
|
|
e.kbchar = 0x08;
|
|
|
|
|
break;
|
|
|
|
|
case 0x0d: // MPR121_SELECT
|
|
|
|
|
e.inputEvent = INPUT_BROKER_SELECT;
|
|
|
|
|
e.kbchar = 0x00;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
if (nextEvent > 127) {
|
|
|
|
|
e.inputEvent = INPUT_BROKER_NONE;
|
|
|
|
|
e.kbchar = 0x00;
|
|
|
|
|
break;
|
2024-10-21 16:53:36 +10:00
|
|
|
}
|
2026-01-03 21:19:24 +01:00
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = nextEvent;
|
2024-10-21 16:53:36 +10:00
|
|
|
break;
|
2026-01-03 21:19:24 +01:00
|
|
|
}
|
|
|
|
|
if (e.inputEvent != INPUT_BROKER_NONE) {
|
|
|
|
|
LOG_DEBUG("MP121 Notifying: %i Char: %i", e.inputEvent, e.kbchar);
|
|
|
|
|
this->notifyObservers(&e);
|
|
|
|
|
}
|
2024-10-21 16:53:36 +10:00
|
|
|
}
|
2026-01-03 21:19:24 +01:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case 0x84: { // Adafruit TCA8418
|
|
|
|
|
TCAKeyboard.trigger();
|
|
|
|
|
InputEvent e = {};
|
|
|
|
|
while (TCAKeyboard.hasEvent()) {
|
|
|
|
|
char nextEvent = TCAKeyboard.dequeueEvent();
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = 0x00;
|
|
|
|
|
e.source = this->_originName;
|
|
|
|
|
switch (nextEvent) {
|
|
|
|
|
case TCA8418KeyboardBase::NONE:
|
|
|
|
|
e.inputEvent = INPUT_BROKER_NONE;
|
|
|
|
|
e.kbchar = 0x00;
|
|
|
|
|
break;
|
|
|
|
|
case TCA8418KeyboardBase::REBOOT:
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = INPUT_BROKER_MSG_REBOOT;
|
|
|
|
|
break;
|
|
|
|
|
case TCA8418KeyboardBase::LEFT:
|
|
|
|
|
e.inputEvent = INPUT_BROKER_LEFT;
|
|
|
|
|
e.kbchar = 0x00;
|
|
|
|
|
break;
|
|
|
|
|
case TCA8418KeyboardBase::UP:
|
|
|
|
|
e.inputEvent = INPUT_BROKER_UP;
|
|
|
|
|
e.kbchar = 0x00;
|
|
|
|
|
break;
|
|
|
|
|
case TCA8418KeyboardBase::DOWN:
|
|
|
|
|
e.inputEvent = INPUT_BROKER_DOWN;
|
|
|
|
|
e.kbchar = 0x00;
|
|
|
|
|
break;
|
|
|
|
|
case TCA8418KeyboardBase::RIGHT:
|
|
|
|
|
e.inputEvent = INPUT_BROKER_RIGHT;
|
|
|
|
|
e.kbchar = 0x00;
|
|
|
|
|
break;
|
|
|
|
|
case TCA8418KeyboardBase::BSP:
|
|
|
|
|
e.inputEvent = INPUT_BROKER_BACK;
|
|
|
|
|
e.kbchar = 0x08;
|
|
|
|
|
break;
|
|
|
|
|
case TCA8418KeyboardBase::SELECT:
|
|
|
|
|
e.inputEvent = INPUT_BROKER_SELECT;
|
|
|
|
|
e.kbchar = 0x00;
|
|
|
|
|
break;
|
|
|
|
|
case TCA8418KeyboardBase::ESC:
|
|
|
|
|
e.inputEvent = INPUT_BROKER_CANCEL;
|
|
|
|
|
e.kbchar = 0x00;
|
|
|
|
|
break;
|
|
|
|
|
case TCA8418KeyboardBase::GPS_TOGGLE:
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = INPUT_BROKER_GPS_TOGGLE;
|
|
|
|
|
break;
|
|
|
|
|
case TCA8418KeyboardBase::SEND_PING:
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = INPUT_BROKER_SEND_PING;
|
|
|
|
|
break;
|
|
|
|
|
case TCA8418KeyboardBase::MUTE_TOGGLE:
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = INPUT_BROKER_MSG_MUTE_TOGGLE;
|
|
|
|
|
break;
|
|
|
|
|
case TCA8418KeyboardBase::BT_TOGGLE:
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = INPUT_BROKER_MSG_BLUETOOTH_TOGGLE;
|
|
|
|
|
break;
|
|
|
|
|
case TCA8418KeyboardBase::BL_TOGGLE:
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = INPUT_BROKER_MSG_BLUETOOTH_TOGGLE;
|
|
|
|
|
break;
|
|
|
|
|
case TCA8418KeyboardBase::TAB:
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = INPUT_BROKER_MSG_TAB;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
if (nextEvent > 127) {
|
|
|
|
|
e.inputEvent = INPUT_BROKER_NONE;
|
|
|
|
|
e.kbchar = 0x00;
|
|
|
|
|
break;
|
2025-04-03 19:18:18 +00:00
|
|
|
}
|
2026-01-03 21:19:24 +01:00
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = nextEvent;
|
2025-04-03 19:18:18 +00:00
|
|
|
break;
|
2026-01-03 21:19:24 +01:00
|
|
|
}
|
|
|
|
|
if (e.inputEvent != INPUT_BROKER_NONE) {
|
|
|
|
|
// LOG_DEBUG("TCA8418 Notifying: %i Char: %c", e.inputEvent, e.kbchar);
|
|
|
|
|
this->notifyObservers(&e);
|
|
|
|
|
}
|
|
|
|
|
TCAKeyboard.trigger();
|
2025-04-03 19:18:18 +00:00
|
|
|
}
|
2026-01-03 21:19:24 +01:00
|
|
|
TCAKeyboard.clearInt();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case 0x02: {
|
|
|
|
|
// RAK14004
|
|
|
|
|
uint8_t rDataBuf[8] = {0};
|
|
|
|
|
uint8_t PrintDataBuf = 0;
|
|
|
|
|
if (read_from_14004(i2cBus, 0x01, rDataBuf, 0x04) == 1) {
|
|
|
|
|
for (uint8_t aCount = 0; aCount < 0x04; aCount++) {
|
|
|
|
|
for (uint8_t bCount = 0; bCount < 0x04; bCount++) {
|
|
|
|
|
if (((rDataBuf[aCount] >> bCount) & 0x01) == 0x01) {
|
|
|
|
|
PrintDataBuf = aCount * 0x04 + bCount + 1;
|
|
|
|
|
}
|
2022-10-06 16:00:33 +02:00
|
|
|
}
|
2026-01-03 21:19:24 +01:00
|
|
|
}
|
2023-08-19 15:37:42 +02:00
|
|
|
}
|
2026-01-03 21:19:24 +01:00
|
|
|
if (PrintDataBuf != 0) {
|
|
|
|
|
LOG_DEBUG("RAK14004 key 0x%x pressed", PrintDataBuf);
|
|
|
|
|
InputEvent e = {};
|
|
|
|
|
e.inputEvent = INPUT_BROKER_MATRIXKEY;
|
|
|
|
|
e.source = this->_originName;
|
|
|
|
|
e.kbchar = PrintDataBuf;
|
|
|
|
|
this->notifyObservers(&e);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case 0x00: // CARDKB
|
|
|
|
|
case 0x10: { // T-DECK
|
2022-05-07 20:31:21 +10:00
|
|
|
|
2026-01-03 21:19:24 +01:00
|
|
|
i2cBus->requestFrom((int)cardkb_found.address, 1);
|
Unify the native display config between legacy display and MUI (#6838)
* Add missed include
* Another Warning fix
* Add another HAS_SCREEN
* Namespace fixes
* Removed depricated destination types and re-factored destination screen
* Get rid of Arduino Strings
* Clean up after Copilot
* SixthLine Def, Screen Rename
Added Sixth Line Definition Screen Rename, and Automatic Line Adjustment
* Consistency is hard - fixed "Sixth"
* System Frame Updates
Adjusted line construction to ensure we fit maximum content per screen.
* Fix up notifications
* Add a couple more ifdef HAS_SCREEN lines
* Add screen->isOverlayBannerShowing()
* Don't forget the invert!
* Adjust Nodelist Center Divider
Adjust Nodelist Center Divider
* Fix variable casting
* Fix entryText variable as empty before update to fix validation
* Altitude is int32_t
* Update PowerTelemetry to have correct data type
* Fix cppcheck warnings (#6945)
* Fix cppcheck warnings
* Adjust logic in Power.cpp for power sensor
---------
Co-authored-by: Jason P <applewiz@mac.com>
* More pixel wrangling so things line up NodeList edition
* Adjust NodeList alignments and plumb some background padding for a possible title fix
* Better alignment for banner notifications
* Move title into drawCommonHeader; initial screen tested
* Fonts make spacing items difficult
* Improved beeping booping and other buzzer based feedback (#6947)
* Improved beeping booping and other buzzer based feedback
* audible button feedback (#6949)
* Refactor
---------
Co-authored-by: todd-herbert <herbert.todd@gmail.com>
* Sandpapered the corners of the notification popup
* Finalize drawCommonHeader migration
* Update Title of Favorite Node Screens
* Update node metric alignment on LoRa screen
* Update the border for popups to separate it from background
* Update PaxcounterModule.cpp with CommonHeader
* Update WiFi screen with CommonHeader and related data reflow
* It was not, in fact, pointing up
* Fix build on wismeshtap
* T-deck trackball debounce
* Fix uptime on Device Focused page to actually detail
* Update Sys screen for new uptime, add label to Freq/Chan on LoRa
* Don't display DOP any longer, make Uptime consistent
* Revert Uptime change on Favorites, Apply to Device Focused
* Label the satelite number to avoid confusion
* Boop boop boop boop
* Correct GPS positioning and string consistency across strings for GPS
* Fix GPS text alignment
* Enable canned messages by default
* Don't wake screen on new nodes
* Cannedmessage list emote support added
* Fn+e emote picker for freetext screen
* Actually block CannedInput actions while display is shown
* Add selection menu to bannerOverlay
* Off by one
* Move to unified text layouts and spacing
* Still my Fav without an "e"
* Fully remove EVENT_NODEDB_UPDATED
* Simply LoRa screen
* Make some char pointers const to fix compilation on native targets
* Update drawCompassNorth to include radius
* Fix warning
* button thread cleanup
* Pull OneButton handling from PowerFSM and add MUI switch (#6973)
* Trunk
* Onebutton Menu Support
* Add temporary clock icon
* Add gps location to fsi
* Banner message state reset
* Cast to char to satisfy compiler
* Better fast handling of input during banner
* Fix warning
* Derp
* oops
* Update ref
* Wire buzzer_mode
* remove legacy string->print()
* Only init screen if one found
* Unsigned Char
* More buttonThread cleaning
* screen.cpp button handling cleanup
* The Great Event Rename of 2025
* Fix the Radiomaster
* Missed trackball type change
* Remove unused function
* Make ButtonThread an InputBroker
* Coffee hadn't kicked in yet
* Add clock icon for Navigation Bar
* Restore clock screen definition code - whoops
* ExternalNotifications now observe inputBroker
* Clock rework (#6992)
* Move Clock bits into ClockRenderer space
* Rework clock into all device navigation
* T-Watch Actually Builds Different
* Compile fix
---------
Co-authored-by: Jonathan Bennett <jbennett@incomsystems.biz>
* Add AM/PM to Digital Clock
* Flip Seconds and AM/PM on Clock Display
* Tik-tok pixels are hard
* Fix builds on Thinknode M1
* Check for GPS and don't crash
* Don't endif til the end
* Rework the OneButton thread to be much less of a mess. (#6997)
* Rework the OneButton thread to be much less of a mess. And break lots of targets temporarily
* Update src/input/ButtonThread.h
Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
* fix GPS toggle
* Send the shutdown event, not just the kbchar
* Honor the back button in a notificaiton popup
* Draw the right size box for popup with options
* Try to un-break all the things
---------
Co-authored-by: Ben Meadors <benmmeadors@gmail.com>
Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
* 24-hour Clock Should have leading zero, but not 12-hour
* Fixup some compile errors
* Add intRoutine to ButtonThread init, to get more responsive user button back
* Add Timezone picker
* Fix Warning
* Optionally set the initial selection for the chooser popup
* Make back buttons work in canned messages
* Drop the wrapper classes
* LonPressTime now configurable
* Clock Frame can not longer be blank; just add valid time
* Back buttons everywhere!
* Key Verification confirm banner
* Make Elecrow M* top button a back button
* Add settings saves
* EInk responsiveness fixes
* Linux Input Fixes
* Add Native Trackball/Joystick support, and move UserButton to Input
* No Flight Stick Mode
* Send input event
* Add Channel Utilization to Device Focused frame
* Don't shift screens when we draw new ones
* Add showOverlayBanner arguments to no-op
* trunk
* Default Native trackball to NC
* Fix crash in simulator mode
* Add longLong button press
* Get the args right
* Adjust Bluetooth Pairing Screen to account for bottom navigation.
* Trackball everywhere, and unPhone buttons
* Remap visionmaster secondary button to TB_UP
* Kill ScanAndSelect
* trunk
* No longer need the canned messages input filter
* All Canned All the time
* Fix stm32 compile error regarding inputBroker
* Unify tft lineheights (#7033)
* Create variable line heights based upon SCREEN_HEIGHT
* Refactor textPositions into method -> getTextPositions
* Update SharedUIDisplay.h
---------
Co-authored-by: Jason P <applewiz@mac.com>
* Adjust top distance for larger displays
* Adjust icon sizes for larger displays
* Fix Paxcounter compile errors after code updates
* Pixel wrangling to make larger screens fit better
* Alert frame has precedence over banner -- for now
* Unify on ALT_BUTTON
* Align AM/PM to the digit, not the segment on larger displays
* Move some global pin defines into configuration.h
* Scaffolding for BMM150 9-axis gyro
* Alt button behavior
* Don't add the blank GPS frames without HAS_GPS
* EVENT_NODEDB_UPDATED has been retired
* Clean out LOG_WARN messages from debugging
* Add dismiss message function
* Minor buttonThread cleanup
* Add BMM150 support
* Clean up last warning from dev
* Simplify bmm150 init return logic
* Add option to reply to messages
* Add minimal menu upon selecting home screen
* Move Messages to slot 2, rename GPS to Position, move variables nearer functional usage in Screen.cpp
* Properly dismiss message
* T-Deck Trackball press is not user button
* Add select on favorite frame to launch cannedMessage DM
* Minor wording change
* Less capital letters
* Fix empty message check, time isn't reliable
* drop dead code
* Make UIRenderer a static class instead of namespace
* Fix the select on favorite
* Check if message is empty early and then 'return'
* Add kb_found, and show the option to launch freetype if appropriate
* Ignore impossible touchscreen touches
* Auto scroll fix
* Move linebreak after "from" for banners to maximize screen usage.
* Center "No messages to show" on Message frame
* Start consolidating buzzer behavior
* Fixed signed / unsigned warning
* Cast second parameter of max() to make some targets happy
* Cast kbchar to (char) to make arduino string happy
* Shorten the notice of "No messages"
* Add buzzer mode chooser
* Add regionPicker to Lora icon
* Reduce line spacing and reorder Position screen to resolve overlapping issues
* Update message titles, fix GPS icons, add Back options
* Leftover boops
* Remove chirp
* Make the region selection dismissable when a region is already set
* Add read-aloud functionality on messages w/ esp8266sam
* "Last Heard" is a better label
* tweak the beep
* 5 options
* properly tear down freetext upon cancel
* de-convelute canned messages just a bit
* Correct height of Mail icon in navigation bar
* Remove unused warning
* Consolidate time methods into TimeFormatters
* Oops
* Change LoRa Picker Cancel to Back
* Tweak selection characters on Banner
* Message render not scrolling on 5th line
* More fixes for message scrolling
* Remove the safety next on text overflow - we found that root cause
* Add pin definitions to fix compilation for obscure target
* Don't let the touchscreen send unitialized kbchar values
* Make virtual KB just a bit quicker
* No more double tap, swipe!
* Left is left, and Right is right
* Update horizontal lightning bolt design
* Move from solid to dashed separator for Message Frame
* Single emote feature fix
* Manually sort overlapping elements for now
* Freetext and clearer choices
* Fix ESP32 InkHUD builds on the unify-tft branch (#7087)
* Remove BaseUI branding
* Capitalization is fun
* Revert Meshtastic Boot Frame Changes
* Add ANZ_433 LoRa region to picker
* Update settings.json
---------
Co-authored-by: HarukiToreda <116696711+HarukiToreda@users.noreply.github.com>
Co-authored-by: Ben Meadors <benmmeadors@gmail.com>
Co-authored-by: Jason P <applewiz@mac.com>
Co-authored-by: todd-herbert <herbert.todd@gmail.com>
Co-authored-by: Thomas Göttgens <tgoettgens@gmail.com>
Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
2025-06-21 06:36:04 -05:00
|
|
|
|
2026-01-03 21:19:24 +01:00
|
|
|
if (i2cBus->available()) {
|
|
|
|
|
char c = i2cBus->read();
|
|
|
|
|
InputEvent e = {};
|
|
|
|
|
e.inputEvent = INPUT_BROKER_NONE;
|
|
|
|
|
e.source = this->_originName;
|
|
|
|
|
switch (c) {
|
|
|
|
|
case 0x71: // This is the button q. If modifier and q pressed, it cancels the input
|
|
|
|
|
if (is_sym) {
|
|
|
|
|
is_sym = false;
|
|
|
|
|
e.inputEvent = INPUT_BROKER_CANCEL;
|
|
|
|
|
} else {
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = c;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 0x74: // letter t. if modifier and t pressed call 'tab'
|
|
|
|
|
if (is_sym) {
|
|
|
|
|
is_sym = false;
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = 0x09; // TAB Scancode
|
|
|
|
|
} else {
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = c;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 0x6d: // letter m. Modifier makes it mute notifications
|
|
|
|
|
if (is_sym) {
|
|
|
|
|
is_sym = false;
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = INPUT_BROKER_MSG_MUTE_TOGGLE; // mute notifications
|
|
|
|
|
} else {
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = c;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 0x6f: // letter o(+). Modifier makes screen increase in brightness
|
|
|
|
|
if (is_sym) {
|
|
|
|
|
is_sym = false;
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = INPUT_BROKER_MSG_BRIGHTNESS_UP; // Increase Brightness code
|
|
|
|
|
} else {
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = c;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 0x69: // letter i(-). Modifier makes screen decrease in brightness
|
|
|
|
|
if (is_sym) {
|
|
|
|
|
is_sym = false;
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = INPUT_BROKER_MSG_BRIGHTNESS_DOWN; // Decrease Brightness code
|
|
|
|
|
} else {
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = c;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 0x20: // Space. Send network ping like double press does
|
|
|
|
|
if (is_sym) {
|
|
|
|
|
is_sym = false;
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = INPUT_BROKER_SEND_PING; // (fn + space)
|
|
|
|
|
} else {
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = c;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 0x67: // letter g. toggle gps
|
|
|
|
|
if (is_sym) {
|
|
|
|
|
is_sym = false;
|
|
|
|
|
e.inputEvent = INPUT_BROKER_GPS_TOGGLE;
|
|
|
|
|
e.kbchar = INPUT_BROKER_GPS_TOGGLE;
|
|
|
|
|
} else {
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = c;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 0x1b: // ESC
|
|
|
|
|
e.inputEvent = INPUT_BROKER_CANCEL;
|
|
|
|
|
break;
|
|
|
|
|
case 0x08: // Back
|
|
|
|
|
e.inputEvent = INPUT_BROKER_BACK;
|
|
|
|
|
e.kbchar = 0;
|
|
|
|
|
break;
|
|
|
|
|
case 0xb5: // Up
|
|
|
|
|
e.inputEvent = INPUT_BROKER_UP;
|
|
|
|
|
e.kbchar = 0;
|
|
|
|
|
break;
|
|
|
|
|
case 0xb6: // Down
|
|
|
|
|
e.inputEvent = INPUT_BROKER_DOWN;
|
|
|
|
|
e.kbchar = 0;
|
|
|
|
|
break;
|
|
|
|
|
case 0xb4: // Left
|
|
|
|
|
e.inputEvent = INPUT_BROKER_LEFT;
|
|
|
|
|
e.kbchar = 0;
|
|
|
|
|
break;
|
|
|
|
|
case 0xb7: // Right
|
|
|
|
|
e.inputEvent = INPUT_BROKER_RIGHT;
|
|
|
|
|
e.kbchar = 0;
|
|
|
|
|
break;
|
|
|
|
|
case 0xc: // Modifier key: 0xc is alt+c (Other options could be: 0xea = shift+mic button or 0x4 shift+$(speaker))
|
|
|
|
|
// toggle moddifiers button.
|
|
|
|
|
is_sym = !is_sym;
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = is_sym ? INPUT_BROKER_MSG_FN_SYMBOL_ON // send 0xf1 to tell CannedMessages to display that the
|
|
|
|
|
: INPUT_BROKER_MSG_FN_SYMBOL_OFF; // modifier key is active
|
|
|
|
|
break;
|
|
|
|
|
case 0x9e: // fn+g INPUT_BROKER_GPS_TOGGLE
|
|
|
|
|
e.inputEvent = INPUT_BROKER_GPS_TOGGLE;
|
|
|
|
|
e.kbchar = c;
|
|
|
|
|
break;
|
|
|
|
|
case 0xaf: // fn+space INPUT_BROKER_SEND_PING
|
|
|
|
|
e.inputEvent = INPUT_BROKER_SEND_PING;
|
|
|
|
|
e.kbchar = c;
|
|
|
|
|
break;
|
|
|
|
|
case 0x9b: // fn+s INPUT_BROKER_MSG_SHUTDOWN
|
|
|
|
|
e.inputEvent = INPUT_BROKER_SHUTDOWN;
|
|
|
|
|
e.kbchar = c;
|
|
|
|
|
break;
|
2022-03-28 16:55:58 +02:00
|
|
|
|
2026-01-03 21:19:24 +01:00
|
|
|
case 0x90: // fn+r INPUT_BROKER_MSG_REBOOT
|
|
|
|
|
case 0x91: // fn+t
|
|
|
|
|
case 0xac: // fn+m INPUT_BROKER_MSG_MUTE_TOGGLE
|
|
|
|
|
case 0xAA: // fn+b INPUT_BROKER_MSG_BLUETOOTH_TOGGLE
|
|
|
|
|
case 0x8F: // fn+e INPUT_BROKER_MSG_EMOTE_LIST
|
|
|
|
|
// just pass those unmodified
|
|
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = c;
|
|
|
|
|
break;
|
|
|
|
|
case 0x0d: // Enter
|
|
|
|
|
e.inputEvent = INPUT_BROKER_SELECT;
|
|
|
|
|
break;
|
|
|
|
|
case 0x00: // nopress
|
|
|
|
|
e.inputEvent = INPUT_BROKER_NONE;
|
|
|
|
|
break;
|
|
|
|
|
default: // all other keys
|
|
|
|
|
if (c > 127) { // bogus key value
|
|
|
|
|
e.inputEvent = INPUT_BROKER_NONE;
|
|
|
|
|
break;
|
2022-09-20 14:28:42 +02:00
|
|
|
}
|
2026-01-03 21:19:24 +01:00
|
|
|
e.inputEvent = INPUT_BROKER_ANYKEY;
|
|
|
|
|
e.kbchar = c;
|
|
|
|
|
is_sym = false;
|
2023-08-19 15:37:42 +02:00
|
|
|
break;
|
2026-01-03 21:19:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (e.inputEvent != INPUT_BROKER_NONE) {
|
|
|
|
|
this->notifyObservers(&e);
|
|
|
|
|
}
|
2023-08-19 15:37:42 +02:00
|
|
|
}
|
2026-01-03 21:19:24 +01:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
LOG_WARN("Unknown kb_model 0x%02x", kb_model);
|
|
|
|
|
}
|
|
|
|
|
return 300;
|
2025-09-22 21:06:23 -05:00
|
|
|
}
|
|
|
|
|
|
2026-01-03 21:19:24 +01:00
|
|
|
void KbI2cBase::toggleBacklight(bool on) {
|
2025-09-22 21:06:23 -05:00
|
|
|
#if defined(T_LORA_PAGER)
|
2026-01-03 21:19:24 +01:00
|
|
|
TCAKeyboard.setBacklight(on);
|
2025-09-22 21:06:23 -05:00
|
|
|
#endif
|
|
|
|
|
}
|