You've already forked M5Unit-GESTURE
mirror of
https://github.com/m5stack/M5Unit-GESTURE.git
synced 2026-05-20 11:28:10 -07:00
170 lines
5.6 KiB
C++
170 lines
5.6 KiB
C++
/*
|
|
* SPDX-FileCopyrightText: 2026 M5Stack Technology CO LTD
|
|
*
|
|
* SPDX-License-Identifier: MIT
|
|
*/
|
|
/*
|
|
Example of sending identified gestures as keyboard commands via BLE
|
|
|
|
Required:
|
|
- https://github.com/wakwak-koba/ESP32-NimBLE-Keyboard
|
|
- https://github.com/h2zero/NimBLE-Arduino
|
|
*/
|
|
#include <M5Unified.h>
|
|
#include <M5UnitUnified.h>
|
|
#include <M5UnitUnifiedGESTURE.h>
|
|
#include <Wire.h>
|
|
#include <M5HAL.hpp> // For NessoN1
|
|
|
|
#define USE_NIMBLE // Define it if using NIMBLE
|
|
#include <BleKeyboard.h>
|
|
|
|
using gesture_t = m5::unit::paj7620u2::Gesture;
|
|
|
|
namespace {
|
|
auto& lcd = M5.Display;
|
|
|
|
m5::unit::UnitUnified Units;
|
|
m5::unit::UnitGesture unit;
|
|
|
|
BleKeyboard bleKeyboard{"PagerKB", "M5UU", 100};
|
|
unsigned long inactive_to{};
|
|
constexpr decltype(inactive_to) INACTIVE_TIME{1500}; // Period of inactivity (ms)
|
|
|
|
constexpr const char* gstr[] = {
|
|
"None", "Up", "Down", "Left", "Right", "Forward", "Backward", "Clockwise", "CounterClockwise",
|
|
"Wave", "Approach", "HasObject", "WakeupTrigger", "Confirm", "Abort", "Reserve", "NoObject",
|
|
};
|
|
|
|
const char* gesture_to_string(const gesture_t g)
|
|
{
|
|
auto gg = m5::stl::to_underlying(g);
|
|
|
|
uint32_t idx = (gg == 0) ? 0 : __builtin_ctz(gg) + 1;
|
|
return idx < m5::stl::size(gstr) ? gstr[idx] : "ERR";
|
|
}
|
|
|
|
// Gesture and keycode correspondence table
|
|
constexpr uint8_t key_table[] = {
|
|
0, // None
|
|
0, // Up
|
|
0, // Down
|
|
0, // Left
|
|
0, // Right
|
|
0, // Forward
|
|
0, // Backward
|
|
KEY_DOWN_ARROW, // Clockwise
|
|
KEY_UP_ARROW, // CounterClockwise
|
|
0, // Wave
|
|
0, // Approach
|
|
0, // HasObject
|
|
0, // WakeupTrigger
|
|
0, // Confirm
|
|
0, // Abort
|
|
0, // Reserve
|
|
0, // NoObject
|
|
};
|
|
|
|
uint8_t gesture_to_key(const gesture_t g)
|
|
{
|
|
auto gg = m5::stl::to_underlying(g);
|
|
uint32_t idx = (gg == 0) ? 0 : __builtin_ctz(gg) + 1;
|
|
return idx < m5::stl::size(key_table) ? key_table[idx] : 0;
|
|
}
|
|
|
|
} // namespace
|
|
|
|
void setup()
|
|
{
|
|
M5.begin();
|
|
M5.setTouchButtonHeightByRatio(100);
|
|
|
|
// The screen shall be in landscape mode
|
|
if (lcd.height() > lcd.width()) {
|
|
lcd.setRotation(1);
|
|
}
|
|
|
|
auto board = M5.getBoard();
|
|
|
|
// 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 unit_ready{};
|
|
if (board == m5::board_t::board_ArduinoNessoN1) {
|
|
// 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());
|
|
unit_ready = Units.add(unit, i2c_bus ? i2c_bus.value() : nullptr) && Units.begin();
|
|
} else if (board == m5::board_t::board_M5NanoC6) {
|
|
// NanoC6: Use M5.Ex_I2C (m5::I2C_Class, not Arduino Wire)
|
|
M5_LOGI("Using M5.Ex_I2C");
|
|
unit_ready = 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, 400 * 1000U);
|
|
unit_ready = Units.add(unit, Wire) && Units.begin();
|
|
}
|
|
if (!unit_ready) {
|
|
M5_LOGE("Failed to begin");
|
|
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_DARKGRAY);
|
|
bleKeyboard.begin();
|
|
}
|
|
|
|
void loop()
|
|
{
|
|
static bool connected{};
|
|
|
|
M5.update();
|
|
Units.update();
|
|
if (connected != bleKeyboard.isConnected()) {
|
|
connected = bleKeyboard.isConnected();
|
|
M5.Log.printf("Change BLE connection:%u\n", connected);
|
|
lcd.fillScreen(connected ? TFT_DARKGREEN : TFT_DARKGRAY);
|
|
}
|
|
if (connected) {
|
|
if (inactive_to) {
|
|
if (m5::utility::millis() < inactive_to) {
|
|
return;
|
|
}
|
|
inactive_to = 0;
|
|
lcd.fillScreen(TFT_DARKGREEN);
|
|
}
|
|
|
|
if (unit.updated()) {
|
|
auto key = gesture_to_key(unit.gesture());
|
|
if (key) {
|
|
M5.Log.printf("Send [0X%X] Gesture:%s\n", key, gesture_to_string(unit.gesture()));
|
|
bleKeyboard.write(key);
|
|
// Continuous input prevention period
|
|
inactive_to = m5::utility::millis() + INACTIVE_TIME;
|
|
lcd.fillScreen(TFT_ORANGE);
|
|
}
|
|
}
|
|
} else {
|
|
m5::utility::delay(1000);
|
|
}
|
|
}
|