mirror of
https://github.com/meshtastic/firmware.git
synced 2025-02-08 21:51:46 +00:00
137 lines
4.2 KiB
C++
137 lines
4.2 KiB
C++
#include "TouchScreenBase.h"
|
|
#include "main.h"
|
|
|
|
#ifndef TIME_LONG_PRESS
|
|
#define TIME_LONG_PRESS 400
|
|
#endif
|
|
|
|
// move a minimum distance over the screen to detect a "swipe"
|
|
#ifndef TOUCH_THRESHOLD_X
|
|
#define TOUCH_THRESHOLD_X 30
|
|
#endif
|
|
|
|
#ifndef TOUCH_THRESHOLD_Y
|
|
#define TOUCH_THRESHOLD_Y 20
|
|
#endif
|
|
|
|
TouchScreenBase::TouchScreenBase(const char *name, uint16_t width, uint16_t height)
|
|
: concurrency::OSThread(name), _display_width(width), _display_height(height), _first_x(0), _last_x(0), _first_y(0),
|
|
_last_y(0), _start(0), _tapped(false), _originName(name)
|
|
{
|
|
}
|
|
|
|
void TouchScreenBase::init(bool hasTouch)
|
|
{
|
|
if (hasTouch) {
|
|
LOG_INFO("TouchScreen initialized %d %d\n", TOUCH_THRESHOLD_X, TOUCH_THRESHOLD_Y);
|
|
this->setInterval(100);
|
|
} else {
|
|
disable();
|
|
this->setInterval(UINT_MAX);
|
|
}
|
|
}
|
|
|
|
int32_t TouchScreenBase::runOnce()
|
|
{
|
|
TouchEvent e;
|
|
e.touchEvent = static_cast<char>(TOUCH_ACTION_NONE);
|
|
|
|
// process touch events
|
|
int16_t x, y;
|
|
bool touched = getTouch(x, y);
|
|
if (touched) {
|
|
this->setInterval(20);
|
|
_last_x = x;
|
|
_last_y = y;
|
|
}
|
|
if (touched != _touchedOld) {
|
|
if (touched) {
|
|
hapticFeedback();
|
|
_state = TOUCH_EVENT_OCCURRED;
|
|
_start = millis();
|
|
_first_x = x;
|
|
_first_y = y;
|
|
} else {
|
|
_state = TOUCH_EVENT_CLEARED;
|
|
time_t duration = millis() - _start;
|
|
x = _last_x;
|
|
y = _last_y;
|
|
this->setInterval(50);
|
|
|
|
// compute distance
|
|
int16_t dx = x - _first_x;
|
|
int16_t dy = y - _first_y;
|
|
uint16_t adx = abs(dx);
|
|
uint16_t ady = abs(dy);
|
|
|
|
// swipe horizontal
|
|
if (adx > ady && adx > TOUCH_THRESHOLD_X) {
|
|
if (0 > dx) { // swipe right to left
|
|
e.touchEvent = static_cast<char>(TOUCH_ACTION_LEFT);
|
|
LOG_DEBUG("action SWIPE: right to left\n");
|
|
} else { // swipe left to right
|
|
e.touchEvent = static_cast<char>(TOUCH_ACTION_RIGHT);
|
|
LOG_DEBUG("action SWIPE: left to right\n");
|
|
}
|
|
}
|
|
// swipe vertical
|
|
else if (ady > adx && ady > TOUCH_THRESHOLD_Y) {
|
|
if (0 > dy) { // swipe bottom to top
|
|
e.touchEvent = static_cast<char>(TOUCH_ACTION_UP);
|
|
LOG_DEBUG("action SWIPE: bottom to top\n");
|
|
} else { // swipe top to bottom
|
|
e.touchEvent = static_cast<char>(TOUCH_ACTION_DOWN);
|
|
LOG_DEBUG("action SWIPE: top to bottom\n");
|
|
}
|
|
}
|
|
// tap
|
|
else {
|
|
if (duration > 0 && duration < TIME_LONG_PRESS) {
|
|
if (_tapped) {
|
|
_tapped = false;
|
|
e.touchEvent = static_cast<char>(TOUCH_ACTION_DOUBLE_TAP);
|
|
LOG_DEBUG("action DOUBLE TAP(%d/%d)\n", x, y);
|
|
} else {
|
|
_tapped = true;
|
|
}
|
|
} else {
|
|
_tapped = false;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
_touchedOld = touched;
|
|
|
|
// fire TAP event when no 2nd tap occured within time
|
|
if (_tapped && (time_t(millis()) - _start) > TIME_LONG_PRESS - 50) {
|
|
_tapped = false;
|
|
e.touchEvent = static_cast<char>(TOUCH_ACTION_TAP);
|
|
LOG_DEBUG("action TAP(%d/%d)\n", _last_x, _last_y);
|
|
}
|
|
|
|
// fire LONG_PRESS event without the need for release
|
|
if (touched && (time_t(millis()) - _start) > TIME_LONG_PRESS) {
|
|
// tricky: prevent reoccurring events and another touch event when releasing
|
|
_start = millis() + 30000;
|
|
e.touchEvent = static_cast<char>(TOUCH_ACTION_LONG_PRESS);
|
|
LOG_DEBUG("action LONG PRESS(%d/%d)\n", _last_x, _last_y);
|
|
}
|
|
|
|
if (e.touchEvent != TOUCH_ACTION_NONE) {
|
|
e.source = this->_originName;
|
|
e.x = _last_x;
|
|
e.y = _last_y;
|
|
onEvent(e);
|
|
}
|
|
|
|
return interval;
|
|
}
|
|
|
|
void TouchScreenBase::hapticFeedback()
|
|
{
|
|
#ifdef T_WATCH_S3
|
|
drv.setWaveform(0, 75);
|
|
drv.setWaveform(1, 0); // end waveform
|
|
drv.go();
|
|
#endif
|
|
} |