Files

355 lines
11 KiB
C++
Raw Permalink Normal View History

2025-02-21 16:01:08 +09:00
/*
* SPDX-FileCopyrightText: 2026 M5Stack Technology CO LTD
2025-02-21 16:01:08 +09:00
*
* SPDX-License-Identifier: MIT
*/
/*
Example using M5UnitUnified for UnitCardKB/UnitCardKB2/UnitFacesQWERTY
2025-02-21 16:01:08 +09:00
*/
#include <M5Unified.h>
#include <M5UnitUnified.h>
#include <M5UnitUnifiedKEYBOARD.h>
#include <M5HAL.hpp>
2025-02-21 16:01:08 +09:00
#include <M5Utility.h>
#include <cctype>
#include <string>
// *************************************************************
// Choose one define symbol to match the unit you are using
// *************************************************************
#if !defined(USING_UNIT_CARDKB) && !defined(USING_UNIT_CARDKB2) && !defined(USING_UNIT_FACES_QWERTY)
2025-02-21 16:01:08 +09:00
// For CardKB
// #define USING_UNIT_CARDKB
// For CardKB2
// #define USING_UNIT_CARDKB2
#if defined(USING_UNIT_CARDKB2)
// Choose one communication mode for CardKB2
// For I2C
// #define USING_I2C_FOR_CARDKB2
// For UART
// #define USING_UART_FOR_CARDKB2
#endif
2025-02-21 16:01:08 +09:00
// For FacesQWERTY
// #define USING_UNIT_FACES_QWERTY
#endif
// *************************************************************
namespace {
auto& lcd = M5.Display;
m5::unit::UnitUnified Units;
2026-04-01 23:55:46 +09:00
const char* special_key_name(const char ch)
{
switch (ch) {
case '\b':
return "BS";
case '\t':
return "TAB";
case '\n':
return "LF";
case '\r':
return "CR";
case 0x1B:
return "ESC";
case 0x7F:
return "DEL";
default:
break;
}
#if defined(USING_UNIT_CARDKB)
using namespace m5::unit;
switch (ch) {
case UnitCardKB::SCHAR_LEFT:
return "LEFT";
case UnitCardKB::SCHAR_UP:
return "UP";
case UnitCardKB::SCHAR_DOWN:
return "DOWN";
case UnitCardKB::SCHAR_RIGHT:
return "RIGHT";
default:
break;
}
#elif defined(USING_UNIT_CARDKB2)
using namespace m5::unit::cardkb2;
switch (ch) {
case SCHAR_LEFT:
return "LEFT";
case SCHAR_UP:
return "UP";
case SCHAR_DOWN:
return "DOWN";
case SCHAR_RIGHT:
return "RIGHT";
default:
break;
}
#elif defined(USING_UNIT_FACES_QWERTY)
using namespace m5::unit;
switch (ch) {
case UnitFacesQWERTY::SCHAR_UP:
return "UP";
case UnitFacesQWERTY::SCHAR_INS:
return "INS";
case UnitFacesQWERTY::SCHAR_HOME:
return "HOME";
case UnitFacesQWERTY::SCHAR_END:
return "END";
case UnitFacesQWERTY::SCHAR_PAGE_UP:
return "PGUP";
case UnitFacesQWERTY::SCHAR_PAGE_DOWN:
return "PGDN";
case UnitFacesQWERTY::SCHAR_LEFT:
return "LEFT";
case UnitFacesQWERTY::SCHAR_DOWN:
return "DOWN";
case UnitFacesQWERTY::SCHAR_RIGHT:
return "RIGHT";
case UnitFacesQWERTY::SCHAR_SPEAKER:
return "SPK";
default:
break;
}
#endif
return nullptr;
}
2025-02-21 16:01:08 +09:00
#if defined(USING_UNIT_CARDKB)
#pragma message "Using UnitCardKB (I2C)"
2025-02-21 16:01:08 +09:00
m5::unit::UnitCardKB unit;
#elif defined(USING_UNIT_CARDKB2)
#if defined(USING_UART_FOR_CARDKB2)
#pragma message "Using UnitCardKB2UART (UART)"
m5::unit::UnitCardKB2UART unit;
#else
#pragma message "Using UnitCardKB2 (I2C)"
m5::unit::UnitCardKB2 unit;
#endif
2025-02-21 16:01:08 +09:00
#elif defined(USING_UNIT_FACES_QWERTY)
#pragma message "Using UnitFacesQWERTY (I2C)"
2025-02-21 16:01:08 +09:00
m5::unit::UnitFacesQWERTY unit;
#else
#error Must choose unit define, USING_UNIT_CARDKB, USING_UNIT_CARDKB2, or USING_UNIT_FACES_QWERTY
2025-02-21 16:01:08 +09:00
#endif
#if defined(USING_UNIT_CARDKB) || defined(USING_UNIT_FACES_QWERTY)
2025-02-21 16:01:08 +09:00
bool scan_mode{};
#endif
2025-02-21 16:01:08 +09:00
2026-04-01 20:06:47 +09:00
// NessoN1: Arduino Wire (I2C_NUM_0) cannot be used for GROVE port.
// Wire is used by M5Unified In_I2C for internal devices (IOExpander etc.).
// Wire1 exists but is reserved for HatPort — cannot be used for GROVE.
// Reconfiguring Wire to GROVE pins breaks In_I2C, causing ESP_ERR_INVALID_STATE in M5.update().
// Solution: Use SoftwareI2C via M5HAL (bit-banging) for the GROVE port.
// NanoC6: Wire.begin() on GROVE pins conflicts with m5::I2C_Class registered by Ex_I2C.setPort()
// on the same I2C_NUM_0, causing sporadic NACK errors.
// Solution: Use M5.Ex_I2C (m5::I2C_Class) directly instead of Arduino Wire.
bool setup_i2c()
2025-02-21 16:01:08 +09:00
{
auto board = M5.getBoard();
if (board == m5::board_t::board_ArduinoNessoN1) {
2026-04-01 20:06:47 +09:00
// NessoN1: GROVE is on port_b (GPIO 5/4), not port_a (which maps to Wire pins 8/10)
auto pin_num_sda = M5.getPin(m5::pin_name_t::port_b_out);
auto pin_num_scl = M5.getPin(m5::pin_name_t::port_b_in);
M5_LOGI("getPin(M5HAL): SDA:%u SCL:%u", pin_num_sda, pin_num_scl);
m5::hal::bus::I2CBusConfig i2c_cfg;
i2c_cfg.pin_sda = m5::hal::gpio::getPin(pin_num_sda);
i2c_cfg.pin_scl = m5::hal::gpio::getPin(pin_num_scl);
auto i2c_bus = m5::hal::bus::i2c::getBus(i2c_cfg);
M5_LOGI("Bus:%d", i2c_bus.has_value());
return Units.add(unit, i2c_bus ? i2c_bus.value() : nullptr) && Units.begin();
} else if (board == m5::board_t::board_M5NanoC6) {
2026-04-01 20:06:47 +09:00
// NanoC6: Use M5.Ex_I2C (m5::I2C_Class, not Arduino Wire)
M5_LOGI("Using M5.Ex_I2C");
return Units.add(unit, M5.Ex_I2C) && Units.begin();
} else {
auto pin_num_sda = M5.getPin(m5::pin_name_t::port_a_sda);
auto pin_num_scl = M5.getPin(m5::pin_name_t::port_a_scl);
M5_LOGI("getPin: SDA:%u SCL:%u", pin_num_sda, pin_num_scl);
Wire.end();
Wire.begin(pin_num_sda, pin_num_scl, 100 * 1000U);
return Units.add(unit, Wire) && Units.begin();
}
}
#if defined(USING_UNIT_CARDKB)
bool setup_cardkb()
{
if (!setup_i2c()) {
return false;
}
M5.Log.printf("Hardware:%02X Firmware:%02X\n", unit.hardwareType(), unit.firmwareVersion());
return true;
}
void loop_cardkb()
{
if (unit.firmwareVersion() && M5.BtnA.wasClicked()) {
scan_mode = !scan_mode;
unit.writeMode(scan_mode ? m5::unit::keyboard::Mode::M5UnitUnified : m5::unit::keyboard::Mode::Conventional);
M5.Log.printf("======== Change behavior %s mode\n", scan_mode ? "M5UnitUnified" : "Conventional");
}
}
#endif
#if defined(USING_UNIT_CARDKB2) && !defined(USING_UART_FOR_CARDKB2)
bool setup_cardkb2_i2c()
{
if (!setup_i2c()) {
return false;
}
M5.Log.printf("Firmware:%02X\n", unit.firmwareVersion());
return true;
}
void loop_cardkb2_i2c()
{
}
#endif
2025-02-21 16:01:08 +09:00
#if defined(USING_UART_FOR_CARDKB2)
bool setup_cardkb2_uart()
{
// UART mode: CardKB2 must be switched to UART mode first (Fn+Sym+2 on the device)
2026-04-01 20:06:47 +09:00
// Port C primary, Port A fallback (NessoN1: Port B fallback — Port A is Wire pins)
auto board = M5.getBoard();
auto pin_num_rx = M5.getPin(m5::pin_name_t::port_c_rxd);
auto pin_num_tx = M5.getPin(m5::pin_name_t::port_c_txd);
if (pin_num_rx < 0 || pin_num_tx < 0) {
2026-04-01 20:06:47 +09:00
if (board == m5::board_t::board_ArduinoNessoN1) {
M5_LOGW("PortC is not available, using PortB");
pin_num_rx = M5.getPin(m5::pin_name_t::port_b_in);
pin_num_tx = M5.getPin(m5::pin_name_t::port_b_out);
} else {
M5_LOGW("PortC is not available, using PortA");
Wire.end();
pin_num_rx = M5.getPin(m5::pin_name_t::port_a_pin1);
pin_num_tx = M5.getPin(m5::pin_name_t::port_a_pin2);
}
}
M5_LOGI("getPin: RX:%d TX:%d", pin_num_rx, pin_num_tx);
2025-02-21 16:01:08 +09:00
// NOTE: setExtPower does not fully reset CardKB2 (Sym state may persist).
// Only works on boards with AXP power management (Core2, CoreS3).
// Press RST button on CardKB2 if Sym LED remains after mode switch.
M5.Power.setExtPower(false);
m5::utility::delay(100);
M5.Power.setExtPower(true);
m5::utility::delay(100);
#if defined(CONFIG_IDF_TARGET_ESP32C6)
auto& serial = Serial1;
#elif SOC_UART_NUM > 2
auto& serial = Serial2;
#elif SOC_UART_NUM > 1
auto& serial = Serial1;
#else
#error "Not enough Serial"
#endif
serial.begin(115200, SERIAL_8N1, pin_num_rx, pin_num_tx);
if (!Units.add(unit, serial) || !Units.begin()) {
return false;
}
M5.Log.printf("Firmware:Unknown (UART mode)\n");
return true;
2025-02-21 16:01:08 +09:00
}
void loop_cardkb2_uart()
2025-02-21 16:01:08 +09:00
{
static uint64_t prev_now{}, prev_holding{}, prev_repeating{};
if (unit.nowBits() != prev_now) {
M5.Log.printf("NOW:%016llX WP:%016llX WR:%016llX\n", unit.nowBits(), unit.pressedBits(), unit.releasedBits());
prev_now = unit.nowBits();
}
if (unit.holdingBits() != prev_holding) {
M5.Log.printf("HOLD:%016llX wasHold:%016llX\n", unit.holdingBits(), unit.wasHoldBits());
prev_holding = unit.holdingBits();
}
if (unit.repeatingBits() != prev_repeating) {
M5.Log.printf("RPT:%016llX\n", unit.repeatingBits());
prev_repeating = unit.repeatingBits();
}
}
#endif
2025-02-21 16:01:08 +09:00
#if defined(USING_UNIT_FACES_QWERTY)
bool setup_faces()
{
2026-04-01 16:27:01 +09:00
// FacesQWERTY connects via M-BUS (internal I2C), not GROVE
if (!Units.add(unit, M5.In_I2C) || !Units.begin()) {
return false;
}
M5.Log.printf("FacesType:%02X Firmware:%02X\n", unit.facesType(), unit.firmwareVersion());
return true;
}
void loop_faces()
{
if (unit.firmwareVersion() && M5.BtnA.wasClicked()) {
scan_mode = !scan_mode;
unit.writeMode(scan_mode ? m5::unit::keyboard::Mode::M5UnitUnified : m5::unit::keyboard::Mode::Conventional);
M5.Log.printf("======== Change behavior %s mode\n", scan_mode ? "M5UnitUnified" : "Conventional");
}
}
#endif
} // namespace
using namespace m5::unit::keyboard;
void setup()
{
M5.begin();
M5.setTouchButtonHeightByRatio(100);
bool unit_ready{};
#if defined(USING_UART_FOR_CARDKB2)
unit_ready = setup_cardkb2_uart();
#elif defined(USING_UNIT_CARDKB2)
unit_ready = setup_cardkb2_i2c();
#elif defined(USING_UNIT_CARDKB)
unit_ready = setup_cardkb();
#elif defined(USING_UNIT_FACES_QWERTY)
unit_ready = setup_faces();
#endif
if (!unit_ready) {
M5_LOGE("Failed to begin");
#if defined(USING_UNIT_CARDKB2)
M5_LOGE("Check CardKB2 communication mode (Fn+Sym+1:I2C, Fn+Sym+2:UART)");
#endif
lcd.fillScreen(TFT_RED);
while (true) {
m5::utility::delay(10000);
}
}
M5_LOGI("M5UnitUnified has been begun");
M5_LOGI("%s", Units.debugInfo().c_str());
lcd.fillScreen(TFT_DARKGREEN);
}
void loop()
{
M5.update();
Units.update();
#if defined(USING_UART_FOR_CARDKB2)
loop_cardkb2_uart();
#elif defined(USING_UNIT_CARDKB2)
loop_cardkb2_i2c();
#elif defined(USING_UNIT_CARDKB)
loop_cardkb();
#elif defined(USING_UNIT_FACES_QWERTY)
loop_faces();
#endif
// Common: get input characters
2025-02-21 16:01:08 +09:00
if (unit.updated()) {
while (unit.available()) {
2026-04-01 23:55:46 +09:00
char ch = unit.getchar();
auto sname = special_key_name(ch);
M5.Log.printf("Char:[%02X %s]\n", (uint8_t)ch, sname ? sname : m5::utility::formatString("%c", ch).c_str());
2025-02-21 18:34:56 +09:00
M5.Speaker.tone(1000, 20);
unit.discard();
2025-02-21 16:01:08 +09:00
}
}
}