mirror of
https://github.com/meshtastic/firmware.git
synced 2025-06-12 16:12:07 +00:00
t-watch-updates: Add canned message free text via touch keyboard and watch face frames to T-Watch S3 (#3941)
Co-authored-by: Ben Meadors <benmmeadors@gmail.com>
This commit is contained in:
parent
1a253dccc3
commit
2f9dc813d3
4
src/graphics/PointStruct.h
Normal file
4
src/graphics/PointStruct.h
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
struct PointStruct {
|
||||||
|
int x;
|
||||||
|
int y;
|
||||||
|
};
|
@ -419,6 +419,466 @@ static bool shouldDrawMessage(const meshtastic_MeshPacket *packet)
|
|||||||
return packet->from != 0 && !moduleConfig.store_forward.enabled;
|
return packet->from != 0 && !moduleConfig.store_forward.enabled;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Draw power bars or a charging indicator on an image of a battery, determined by battery charge voltage or percentage.
|
||||||
|
static void drawBattery(OLEDDisplay *display, int16_t x, int16_t y, uint8_t *imgBuffer, const PowerStatus *powerStatus)
|
||||||
|
{
|
||||||
|
static const uint8_t powerBar[3] = {0x81, 0xBD, 0xBD};
|
||||||
|
static const uint8_t lightning[8] = {0xA1, 0xA1, 0xA5, 0xAD, 0xB5, 0xA5, 0x85, 0x85};
|
||||||
|
// Clear the bar area on the battery image
|
||||||
|
for (int i = 1; i < 14; i++) {
|
||||||
|
imgBuffer[i] = 0x81;
|
||||||
|
}
|
||||||
|
// If charging, draw a charging indicator
|
||||||
|
if (powerStatus->getIsCharging()) {
|
||||||
|
memcpy(imgBuffer + 3, lightning, 8);
|
||||||
|
// If not charging, Draw power bars
|
||||||
|
} else {
|
||||||
|
for (int i = 0; i < 4; i++) {
|
||||||
|
if (powerStatus->getBatteryChargePercent() >= 25 * i)
|
||||||
|
memcpy(imgBuffer + 1 + (i * 3), powerBar, 3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
display->drawFastImage(x, y, 16, 8, imgBuffer);
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef T_WATCH_S3
|
||||||
|
|
||||||
|
void Screen::drawWatchFaceToggleButton(OLEDDisplay *display, int16_t x, int16_t y, bool digitalMode, float scale)
|
||||||
|
{
|
||||||
|
uint16_t segmentWidth = SEGMENT_WIDTH * scale;
|
||||||
|
uint16_t segmentHeight = SEGMENT_HEIGHT * scale;
|
||||||
|
|
||||||
|
if (digitalMode) {
|
||||||
|
uint16_t radius = (segmentWidth + (segmentHeight * 2) + 4) / 2;
|
||||||
|
uint16_t centerX = (x + segmentHeight + 2) + (radius / 2);
|
||||||
|
uint16_t centerY = (y + segmentHeight + 2) + (radius / 2);
|
||||||
|
|
||||||
|
display->drawCircle(centerX, centerY, radius);
|
||||||
|
display->drawCircle(centerX, centerY, radius + 1);
|
||||||
|
display->drawLine(centerX, centerY, centerX, centerY - radius + 3);
|
||||||
|
display->drawLine(centerX, centerY, centerX + radius - 3, centerY);
|
||||||
|
} else {
|
||||||
|
uint16_t segmentOneX = x + segmentHeight + 2;
|
||||||
|
uint16_t segmentOneY = y;
|
||||||
|
|
||||||
|
uint16_t segmentTwoX = segmentOneX + segmentWidth + 2;
|
||||||
|
uint16_t segmentTwoY = segmentOneY + segmentHeight + 2;
|
||||||
|
|
||||||
|
uint16_t segmentThreeX = segmentOneX;
|
||||||
|
uint16_t segmentThreeY = segmentTwoY + segmentWidth + 2;
|
||||||
|
|
||||||
|
uint16_t segmentFourX = x;
|
||||||
|
uint16_t segmentFourY = y + segmentHeight + 2;
|
||||||
|
|
||||||
|
drawHorizontalSegment(display, segmentOneX, segmentOneY, segmentWidth, segmentHeight);
|
||||||
|
drawVerticalSegment(display, segmentTwoX, segmentTwoY, segmentWidth, segmentHeight);
|
||||||
|
drawHorizontalSegment(display, segmentThreeX, segmentThreeY, segmentWidth, segmentHeight);
|
||||||
|
drawVerticalSegment(display, segmentFourX, segmentFourY, segmentWidth, segmentHeight);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Draw a digital clock
|
||||||
|
void Screen::drawDigitalClockFrame(OLEDDisplay *display, OLEDDisplayUiState *state, int16_t x, int16_t y)
|
||||||
|
{
|
||||||
|
display->setTextAlignment(TEXT_ALIGN_LEFT);
|
||||||
|
|
||||||
|
drawBattery(display, x, y + 7, imgBattery, powerStatus);
|
||||||
|
|
||||||
|
if (powerStatus->getHasBattery()) {
|
||||||
|
String batteryPercent = String(powerStatus->getBatteryChargePercent()) + "%";
|
||||||
|
|
||||||
|
display->setFont(FONT_SMALL);
|
||||||
|
|
||||||
|
display->drawString(x + 20, y + 2, batteryPercent);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (nimbleBluetooth->isConnected()) {
|
||||||
|
drawBluetoothConnectedIcon(display, display->getWidth() - 18, y + 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
drawWatchFaceToggleButton(display, display->getWidth() - 36, display->getHeight() - 36, screen->digitalWatchFace, 1);
|
||||||
|
|
||||||
|
display->setColor(OLEDDISPLAY_COLOR::WHITE);
|
||||||
|
|
||||||
|
uint32_t rtc_sec = getValidTime(RTCQuality::RTCQualityDevice, true); // Display local timezone
|
||||||
|
if (rtc_sec > 0) {
|
||||||
|
long hms = rtc_sec % SEC_PER_DAY;
|
||||||
|
hms = (hms + SEC_PER_DAY) % SEC_PER_DAY;
|
||||||
|
|
||||||
|
int hour = hms / SEC_PER_HOUR;
|
||||||
|
int minute = (hms % SEC_PER_HOUR) / SEC_PER_MIN;
|
||||||
|
int second = (hms % SEC_PER_HOUR) % SEC_PER_MIN; // or hms % SEC_PER_MIN
|
||||||
|
|
||||||
|
hour = hour > 12 ? hour - 12 : hour;
|
||||||
|
|
||||||
|
if (hour == 0) {
|
||||||
|
hour = 12;
|
||||||
|
}
|
||||||
|
|
||||||
|
// hours string
|
||||||
|
String hourString = String(hour);
|
||||||
|
|
||||||
|
// minutes string
|
||||||
|
String minuteString = minute < 10 ? "0" + String(minute) : String(minute);
|
||||||
|
|
||||||
|
String timeString = hourString + ":" + minuteString;
|
||||||
|
|
||||||
|
// seconds string
|
||||||
|
String secondString = second < 10 ? "0" + String(second) : String(second);
|
||||||
|
|
||||||
|
float scale = 1.5;
|
||||||
|
|
||||||
|
uint16_t segmentWidth = SEGMENT_WIDTH * scale;
|
||||||
|
uint16_t segmentHeight = SEGMENT_HEIGHT * scale;
|
||||||
|
|
||||||
|
// calculate hours:minutes string width
|
||||||
|
uint16_t timeStringWidth = timeString.length() * 5;
|
||||||
|
|
||||||
|
for (uint8_t i = 0; i < timeString.length(); i++) {
|
||||||
|
String character = String(timeString[i]);
|
||||||
|
|
||||||
|
if (character == ":") {
|
||||||
|
timeStringWidth += segmentHeight;
|
||||||
|
} else {
|
||||||
|
timeStringWidth += segmentWidth + (segmentHeight * 2) + 4;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// calculate seconds string width
|
||||||
|
uint16_t secondStringWidth = (secondString.length() * 12) + 4;
|
||||||
|
|
||||||
|
// sum these to get total string width
|
||||||
|
uint16_t totalWidth = timeStringWidth + secondStringWidth;
|
||||||
|
|
||||||
|
uint16_t hourMinuteTextX = (display->getWidth() / 2) - (totalWidth / 2);
|
||||||
|
|
||||||
|
uint16_t startingHourMinuteTextX = hourMinuteTextX;
|
||||||
|
|
||||||
|
uint16_t hourMinuteTextY = (display->getHeight() / 2) - (((segmentWidth * 2) + (segmentHeight * 3) + 8) / 2);
|
||||||
|
|
||||||
|
// iterate over characters in hours:minutes string and draw segmented characters
|
||||||
|
for (uint8_t i = 0; i < timeString.length(); i++) {
|
||||||
|
String character = String(timeString[i]);
|
||||||
|
|
||||||
|
if (character == ":") {
|
||||||
|
drawSegmentedDisplayColon(display, hourMinuteTextX, hourMinuteTextY, scale);
|
||||||
|
|
||||||
|
hourMinuteTextX += segmentHeight + 6;
|
||||||
|
} else {
|
||||||
|
drawSegmentedDisplayCharacter(display, hourMinuteTextX, hourMinuteTextY, character.toInt(), scale);
|
||||||
|
|
||||||
|
hourMinuteTextX += segmentWidth + (segmentHeight * 2) + 4;
|
||||||
|
}
|
||||||
|
|
||||||
|
hourMinuteTextX += 5;
|
||||||
|
}
|
||||||
|
|
||||||
|
// draw seconds string
|
||||||
|
display->setFont(FONT_MEDIUM);
|
||||||
|
display->drawString(startingHourMinuteTextX + timeStringWidth + 4,
|
||||||
|
(display->getHeight() - hourMinuteTextY) - FONT_HEIGHT_MEDIUM + 6, secondString);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Screen::drawSegmentedDisplayColon(OLEDDisplay *display, int x, int y, float scale)
|
||||||
|
{
|
||||||
|
uint16_t segmentWidth = SEGMENT_WIDTH * scale;
|
||||||
|
uint16_t segmentHeight = SEGMENT_HEIGHT * scale;
|
||||||
|
|
||||||
|
uint16_t cellHeight = (segmentWidth * 2) + (segmentHeight * 3) + 8;
|
||||||
|
|
||||||
|
uint16_t topAndBottomX = x + (4 * scale);
|
||||||
|
|
||||||
|
uint16_t quarterCellHeight = cellHeight / 4;
|
||||||
|
|
||||||
|
uint16_t topY = y + quarterCellHeight;
|
||||||
|
uint16_t bottomY = y + (quarterCellHeight * 3);
|
||||||
|
|
||||||
|
display->fillRect(topAndBottomX, topY, segmentHeight, segmentHeight);
|
||||||
|
display->fillRect(topAndBottomX, bottomY, segmentHeight, segmentHeight);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Screen::drawSegmentedDisplayCharacter(OLEDDisplay *display, int x, int y, uint8_t number, float scale)
|
||||||
|
{
|
||||||
|
// the numbers 0-9, each expressed as an array of seven boolean (0|1) values encoding the on/off state of
|
||||||
|
// segment {innerIndex + 1}
|
||||||
|
// e.g., to display the numeral '0', segments 1-6 are on, and segment 7 is off.
|
||||||
|
uint8_t numbers[10][7] = {
|
||||||
|
{1, 1, 1, 1, 1, 1, 0}, // 0 Display segment key
|
||||||
|
{0, 1, 1, 0, 0, 0, 0}, // 1 1
|
||||||
|
{1, 1, 0, 1, 1, 0, 1}, // 2 ___
|
||||||
|
{1, 1, 1, 1, 0, 0, 1}, // 3 6 | | 2
|
||||||
|
{0, 1, 1, 0, 0, 1, 1}, // 4 |_7̲_|
|
||||||
|
{1, 0, 1, 1, 0, 1, 1}, // 5 5 | | 3
|
||||||
|
{1, 0, 1, 1, 1, 1, 1}, // 6 |___|
|
||||||
|
{1, 1, 1, 0, 0, 1, 0}, // 7
|
||||||
|
{1, 1, 1, 1, 1, 1, 1}, // 8 4
|
||||||
|
{1, 1, 1, 1, 0, 1, 1}, // 9
|
||||||
|
};
|
||||||
|
|
||||||
|
// the width and height of each segment's central rectangle:
|
||||||
|
// _____________________
|
||||||
|
// ⋰| (only this part, |⋱
|
||||||
|
// ⋰ | not including | ⋱
|
||||||
|
// ⋱ | the triangles | ⋰
|
||||||
|
// ⋱| on the ends) |⋰
|
||||||
|
// ‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾
|
||||||
|
|
||||||
|
uint16_t segmentWidth = SEGMENT_WIDTH * scale;
|
||||||
|
uint16_t segmentHeight = SEGMENT_HEIGHT * scale;
|
||||||
|
|
||||||
|
// segment x and y coordinates
|
||||||
|
uint16_t segmentOneX = x + segmentHeight + 2;
|
||||||
|
uint16_t segmentOneY = y;
|
||||||
|
|
||||||
|
uint16_t segmentTwoX = segmentOneX + segmentWidth + 2;
|
||||||
|
uint16_t segmentTwoY = segmentOneY + segmentHeight + 2;
|
||||||
|
|
||||||
|
uint16_t segmentThreeX = segmentTwoX;
|
||||||
|
uint16_t segmentThreeY = segmentTwoY + segmentWidth + 2 + segmentHeight + 2;
|
||||||
|
|
||||||
|
uint16_t segmentFourX = segmentOneX;
|
||||||
|
uint16_t segmentFourY = segmentThreeY + segmentWidth + 2;
|
||||||
|
|
||||||
|
uint16_t segmentFiveX = x;
|
||||||
|
uint16_t segmentFiveY = segmentThreeY;
|
||||||
|
|
||||||
|
uint16_t segmentSixX = x;
|
||||||
|
uint16_t segmentSixY = segmentTwoY;
|
||||||
|
|
||||||
|
uint16_t segmentSevenX = segmentOneX;
|
||||||
|
uint16_t segmentSevenY = segmentTwoY + segmentWidth + 2;
|
||||||
|
|
||||||
|
if (numbers[number][0]) {
|
||||||
|
drawHorizontalSegment(display, segmentOneX, segmentOneY, segmentWidth, segmentHeight);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (numbers[number][1]) {
|
||||||
|
drawVerticalSegment(display, segmentTwoX, segmentTwoY, segmentWidth, segmentHeight);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (numbers[number][2]) {
|
||||||
|
drawVerticalSegment(display, segmentThreeX, segmentThreeY, segmentWidth, segmentHeight);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (numbers[number][3]) {
|
||||||
|
drawHorizontalSegment(display, segmentFourX, segmentFourY, segmentWidth, segmentHeight);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (numbers[number][4]) {
|
||||||
|
drawVerticalSegment(display, segmentFiveX, segmentFiveY, segmentWidth, segmentHeight);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (numbers[number][5]) {
|
||||||
|
drawVerticalSegment(display, segmentSixX, segmentSixY, segmentWidth, segmentHeight);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (numbers[number][6]) {
|
||||||
|
drawHorizontalSegment(display, segmentSevenX, segmentSevenY, segmentWidth, segmentHeight);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Screen::drawHorizontalSegment(OLEDDisplay *display, int x, int y, int width, int height)
|
||||||
|
{
|
||||||
|
int halfHeight = height / 2;
|
||||||
|
|
||||||
|
// draw central rectangle
|
||||||
|
display->fillRect(x, y, width, height);
|
||||||
|
|
||||||
|
// draw end triangles
|
||||||
|
display->fillTriangle(x, y, x, y + height - 1, x - halfHeight, y + halfHeight);
|
||||||
|
|
||||||
|
display->fillTriangle(x + width, y, x + width + halfHeight, y + halfHeight, x + width, y + height - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Screen::drawVerticalSegment(OLEDDisplay *display, int x, int y, int width, int height)
|
||||||
|
{
|
||||||
|
int halfHeight = height / 2;
|
||||||
|
|
||||||
|
// draw central rectangle
|
||||||
|
display->fillRect(x, y, height, width);
|
||||||
|
|
||||||
|
// draw end triangles
|
||||||
|
display->fillTriangle(x + halfHeight, y - halfHeight, x + height - 1, y, x, y);
|
||||||
|
|
||||||
|
display->fillTriangle(x, y + width, x + height - 1, y + width, x + halfHeight, y + width + halfHeight);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Screen::drawBluetoothConnectedIcon(OLEDDisplay *display, int16_t x, int16_t y)
|
||||||
|
{
|
||||||
|
display->drawFastImage(x, y, 18, 14, bluetoothConnectedIcon);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Draw an analog clock
|
||||||
|
void Screen::drawAnalogClockFrame(OLEDDisplay *display, OLEDDisplayUiState *state, int16_t x, int16_t y)
|
||||||
|
{
|
||||||
|
display->setTextAlignment(TEXT_ALIGN_LEFT);
|
||||||
|
|
||||||
|
drawBattery(display, x, y + 7, imgBattery, powerStatus);
|
||||||
|
|
||||||
|
if (powerStatus->getHasBattery()) {
|
||||||
|
String batteryPercent = String(powerStatus->getBatteryChargePercent()) + "%";
|
||||||
|
|
||||||
|
display->setFont(FONT_SMALL);
|
||||||
|
|
||||||
|
display->drawString(x + 20, y + 2, batteryPercent);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (nimbleBluetooth->isConnected()) {
|
||||||
|
drawBluetoothConnectedIcon(display, display->getWidth() - 18, y + 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
drawWatchFaceToggleButton(display, display->getWidth() - 36, display->getHeight() - 36, screen->digitalWatchFace, 1);
|
||||||
|
|
||||||
|
// clock face center coordinates
|
||||||
|
int16_t centerX = display->getWidth() / 2;
|
||||||
|
int16_t centerY = display->getHeight() / 2;
|
||||||
|
|
||||||
|
// clock face radius
|
||||||
|
int16_t radius = (display->getWidth() / 2) * 0.8;
|
||||||
|
|
||||||
|
// noon (0 deg) coordinates (outermost circle)
|
||||||
|
int16_t noonX = centerX;
|
||||||
|
int16_t noonY = centerY - radius;
|
||||||
|
|
||||||
|
// second hand radius and y coordinate (outermost circle)
|
||||||
|
int16_t secondHandNoonY = noonY + 1;
|
||||||
|
|
||||||
|
// tick mark outer y coordinate; (first nested circle)
|
||||||
|
int16_t tickMarkOuterNoonY = secondHandNoonY;
|
||||||
|
|
||||||
|
// seconds tick mark inner y coordinate; (second nested circle)
|
||||||
|
double secondsTickMarkInnerNoonY = (double)noonY + 8;
|
||||||
|
|
||||||
|
// hours tick mark inner y coordinate; (third nested circle)
|
||||||
|
double hoursTickMarkInnerNoonY = (double)noonY + 16;
|
||||||
|
|
||||||
|
// minute hand y coordinate
|
||||||
|
int16_t minuteHandNoonY = secondsTickMarkInnerNoonY + 4;
|
||||||
|
|
||||||
|
// hour string y coordinate
|
||||||
|
int16_t hourStringNoonY = minuteHandNoonY + 18;
|
||||||
|
|
||||||
|
// hour hand radius and y coordinate
|
||||||
|
int16_t hourHandRadius = radius * 0.55;
|
||||||
|
int16_t hourHandNoonY = centerY - hourHandRadius;
|
||||||
|
|
||||||
|
display->setColor(OLEDDISPLAY_COLOR::WHITE);
|
||||||
|
display->drawCircle(centerX, centerY, radius);
|
||||||
|
|
||||||
|
uint32_t rtc_sec = getValidTime(RTCQuality::RTCQualityDevice, true); // Display local timezone
|
||||||
|
if (rtc_sec > 0) {
|
||||||
|
long hms = rtc_sec % SEC_PER_DAY;
|
||||||
|
hms = (hms + SEC_PER_DAY) % SEC_PER_DAY;
|
||||||
|
|
||||||
|
// Tear apart hms into h:m:s
|
||||||
|
int hour = hms / SEC_PER_HOUR;
|
||||||
|
int minute = (hms % SEC_PER_HOUR) / SEC_PER_MIN;
|
||||||
|
int second = (hms % SEC_PER_HOUR) % SEC_PER_MIN; // or hms % SEC_PER_MIN
|
||||||
|
|
||||||
|
hour = hour > 12 ? hour - 12 : hour;
|
||||||
|
|
||||||
|
int16_t degreesPerHour = 30;
|
||||||
|
int16_t degreesPerMinuteOrSecond = 6;
|
||||||
|
|
||||||
|
double hourBaseAngle = hour * degreesPerHour;
|
||||||
|
double hourAngleOffset = ((double)minute / 60) * degreesPerHour;
|
||||||
|
double hourAngle = radians(hourBaseAngle + hourAngleOffset);
|
||||||
|
|
||||||
|
double minuteBaseAngle = minute * degreesPerMinuteOrSecond;
|
||||||
|
double minuteAngleOffset = ((double)second / 60) * degreesPerMinuteOrSecond;
|
||||||
|
double minuteAngle = radians(minuteBaseAngle + minuteAngleOffset);
|
||||||
|
|
||||||
|
double secondAngle = radians(second * degreesPerMinuteOrSecond);
|
||||||
|
|
||||||
|
double hourX = sin(-hourAngle) * (hourHandNoonY - centerY) + noonX;
|
||||||
|
double hourY = cos(-hourAngle) * (hourHandNoonY - centerY) + centerY;
|
||||||
|
|
||||||
|
double minuteX = sin(-minuteAngle) * (minuteHandNoonY - centerY) + noonX;
|
||||||
|
double minuteY = cos(-minuteAngle) * (minuteHandNoonY - centerY) + centerY;
|
||||||
|
|
||||||
|
double secondX = sin(-secondAngle) * (secondHandNoonY - centerY) + noonX;
|
||||||
|
double secondY = cos(-secondAngle) * (secondHandNoonY - centerY) + centerY;
|
||||||
|
|
||||||
|
display->setFont(FONT_MEDIUM);
|
||||||
|
|
||||||
|
// draw minute and hour tick marks and hour numbers
|
||||||
|
for (uint16_t angle = 0; angle < 360; angle += 6) {
|
||||||
|
double angleInRadians = radians(angle);
|
||||||
|
|
||||||
|
double sineAngleInRadians = sin(-angleInRadians);
|
||||||
|
double cosineAngleInRadians = cos(-angleInRadians);
|
||||||
|
|
||||||
|
double endX = sineAngleInRadians * (tickMarkOuterNoonY - centerY) + noonX;
|
||||||
|
double endY = cosineAngleInRadians * (tickMarkOuterNoonY - centerY) + centerY;
|
||||||
|
|
||||||
|
if (angle % degreesPerHour == 0) {
|
||||||
|
double startX = sineAngleInRadians * (hoursTickMarkInnerNoonY - centerY) + noonX;
|
||||||
|
double startY = cosineAngleInRadians * (hoursTickMarkInnerNoonY - centerY) + centerY;
|
||||||
|
|
||||||
|
// draw hour tick mark
|
||||||
|
display->drawLine(startX, startY, endX, endY);
|
||||||
|
|
||||||
|
static char buffer[2];
|
||||||
|
|
||||||
|
uint8_t hourInt = (angle / 30);
|
||||||
|
|
||||||
|
if (hourInt == 0) {
|
||||||
|
hourInt = 12;
|
||||||
|
}
|
||||||
|
|
||||||
|
// hour number x offset needs to be adjusted for some cases
|
||||||
|
int8_t hourStringXOffset;
|
||||||
|
int8_t hourStringYOffset = 13;
|
||||||
|
|
||||||
|
switch (hourInt) {
|
||||||
|
case 3:
|
||||||
|
hourStringXOffset = 5;
|
||||||
|
break;
|
||||||
|
case 9:
|
||||||
|
hourStringXOffset = 7;
|
||||||
|
break;
|
||||||
|
case 10:
|
||||||
|
case 11:
|
||||||
|
hourStringXOffset = 8;
|
||||||
|
break;
|
||||||
|
case 12:
|
||||||
|
hourStringXOffset = 13;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
hourStringXOffset = 6;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
double hourStringX = (sineAngleInRadians * (hourStringNoonY - centerY) + noonX) - hourStringXOffset;
|
||||||
|
double hourStringY = (cosineAngleInRadians * (hourStringNoonY - centerY) + centerY) - hourStringYOffset;
|
||||||
|
|
||||||
|
// draw hour number
|
||||||
|
display->drawStringf(hourStringX, hourStringY, buffer, "%d", hourInt);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (angle % degreesPerMinuteOrSecond == 0) {
|
||||||
|
double startX = sineAngleInRadians * (secondsTickMarkInnerNoonY - centerY) + noonX;
|
||||||
|
double startY = cosineAngleInRadians * (secondsTickMarkInnerNoonY - centerY) + centerY;
|
||||||
|
|
||||||
|
// draw minute tick mark
|
||||||
|
display->drawLine(startX, startY, endX, endY);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// draw hour hand
|
||||||
|
display->drawLine(centerX, centerY, hourX, hourY);
|
||||||
|
|
||||||
|
// draw minute hand
|
||||||
|
display->drawLine(centerX, centerY, minuteX, minuteY);
|
||||||
|
|
||||||
|
// draw second hand
|
||||||
|
display->drawLine(centerX, centerY, secondX, secondY);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
// Get an absolute time from "seconds ago" info. Returns false if no valid timestamp possible
|
// Get an absolute time from "seconds ago" info. Returns false if no valid timestamp possible
|
||||||
bool deltaToTimestamp(uint32_t secondsAgo, uint8_t *hours, uint8_t *minutes, int32_t *daysAgo)
|
bool deltaToTimestamp(uint32_t secondsAgo, uint8_t *hours, uint8_t *minutes, int32_t *daysAgo)
|
||||||
{
|
{
|
||||||
@ -664,28 +1124,6 @@ static void drawColumns(OLEDDisplay *display, int16_t x, int16_t y, const char *
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Draw power bars or a charging indicator on an image of a battery, determined by battery charge voltage or percentage.
|
|
||||||
static void drawBattery(OLEDDisplay *display, int16_t x, int16_t y, uint8_t *imgBuffer, const PowerStatus *powerStatus)
|
|
||||||
{
|
|
||||||
static const uint8_t powerBar[3] = {0x81, 0xBD, 0xBD};
|
|
||||||
static const uint8_t lightning[8] = {0xA1, 0xA1, 0xA5, 0xAD, 0xB5, 0xA5, 0x85, 0x85};
|
|
||||||
// Clear the bar area on the battery image
|
|
||||||
for (int i = 1; i < 14; i++) {
|
|
||||||
imgBuffer[i] = 0x81;
|
|
||||||
}
|
|
||||||
// If charging, draw a charging indicator
|
|
||||||
if (powerStatus->getIsCharging()) {
|
|
||||||
memcpy(imgBuffer + 3, lightning, 8);
|
|
||||||
// If not charging, Draw power bars
|
|
||||||
} else {
|
|
||||||
for (int i = 0; i < 4; i++) {
|
|
||||||
if (powerStatus->getBatteryChargePercent() >= 25 * i)
|
|
||||||
memcpy(imgBuffer + 1 + (i * 3), powerBar, 3);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
display->drawFastImage(x, y, 16, 8, imgBuffer);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Draw nodes status
|
// Draw nodes status
|
||||||
static void drawNodes(OLEDDisplay *display, int16_t x, int16_t y, const NodeStatus *nodeStatus)
|
static void drawNodes(OLEDDisplay *display, int16_t x, int16_t y, const NodeStatus *nodeStatus)
|
||||||
{
|
{
|
||||||
@ -1377,6 +1815,10 @@ int32_t Screen::runOnce()
|
|||||||
return RUN_SAME;
|
return RUN_SAME;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (displayHeight == 0) {
|
||||||
|
displayHeight = dispdev->getHeight();
|
||||||
|
}
|
||||||
|
|
||||||
// Show boot screen for first logo_timeout seconds, then switch to normal operation.
|
// Show boot screen for first logo_timeout seconds, then switch to normal operation.
|
||||||
// serialSinceMsec adjusts for additional serial wait time during nRF52 bootup
|
// serialSinceMsec adjusts for additional serial wait time during nRF52 bootup
|
||||||
static bool showingBootScreen = true;
|
static bool showingBootScreen = true;
|
||||||
@ -1655,6 +2097,10 @@ void Screen::setFrames()
|
|||||||
normalFrames[numframes++] = drawWaypointFrame;
|
normalFrames[numframes++] = drawWaypointFrame;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef T_WATCH_S3
|
||||||
|
normalFrames[numframes++] = screen->digitalWatchFace ? &Screen::drawDigitalClockFrame : &Screen::drawAnalogClockFrame;
|
||||||
|
#endif
|
||||||
|
|
||||||
// then all the nodes
|
// then all the nodes
|
||||||
// We only show a few nodes in our scrolling list - because meshes with many nodes would have too many screens
|
// We only show a few nodes in our scrolling list - because meshes with many nodes would have too many screens
|
||||||
size_t numToShow = min(numMeshNodes, 4U);
|
size_t numToShow = min(numMeshNodes, 4U);
|
||||||
@ -2226,6 +2672,19 @@ int Screen::handleUIFrameEvent(const UIFrameEvent *event)
|
|||||||
|
|
||||||
int Screen::handleInputEvent(const InputEvent *event)
|
int Screen::handleInputEvent(const InputEvent *event)
|
||||||
{
|
{
|
||||||
|
|
||||||
|
#ifdef T_WATCH_S3
|
||||||
|
// For the T-Watch, intercept touches to the 'toggle digital/analog watch face' button
|
||||||
|
if (this->ui->getUiState()->currentFrame == 0 && event->touchX >= 204 && event->touchX <= 240 && event->touchY >= 204 &&
|
||||||
|
event->touchY <= 240) {
|
||||||
|
screen->digitalWatchFace = !screen->digitalWatchFace;
|
||||||
|
|
||||||
|
setFrames();
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
if (showingNormalScreen && moduleFrames.size() == 0) {
|
if (showingNormalScreen && moduleFrames.size() == 0) {
|
||||||
// LOG_DEBUG("Screen::handleInputEvent from %s\n", event->source);
|
// LOG_DEBUG("Screen::handleInputEvent from %s\n", event->source);
|
||||||
if (event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_LEFT)) {
|
if (event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_LEFT)) {
|
||||||
|
@ -48,6 +48,7 @@ class Screen
|
|||||||
|
|
||||||
#include "EInkDisplay2.h"
|
#include "EInkDisplay2.h"
|
||||||
#include "EInkDynamicDisplay.h"
|
#include "EInkDynamicDisplay.h"
|
||||||
|
#include "PointStruct.h"
|
||||||
#include "TFTDisplay.h"
|
#include "TFTDisplay.h"
|
||||||
#include "TypedQueue.h"
|
#include "TypedQueue.h"
|
||||||
#include "commands.h"
|
#include "commands.h"
|
||||||
@ -77,6 +78,10 @@ class Screen
|
|||||||
#define EINK_BLACK OLEDDISPLAY_COLOR::WHITE
|
#define EINK_BLACK OLEDDISPLAY_COLOR::WHITE
|
||||||
#define EINK_WHITE OLEDDISPLAY_COLOR::BLACK
|
#define EINK_WHITE OLEDDISPLAY_COLOR::BLACK
|
||||||
|
|
||||||
|
// Base segment dimensions for T-Watch segmented display
|
||||||
|
#define SEGMENT_WIDTH 16
|
||||||
|
#define SEGMENT_HEIGHT 4
|
||||||
|
|
||||||
namespace graphics
|
namespace graphics
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -389,6 +394,27 @@ class Screen : public concurrency::OSThread
|
|||||||
|
|
||||||
static void drawDebugInfoWiFiTrampoline(OLEDDisplay *display, OLEDDisplayUiState *state, int16_t x, int16_t y);
|
static void drawDebugInfoWiFiTrampoline(OLEDDisplay *display, OLEDDisplayUiState *state, int16_t x, int16_t y);
|
||||||
|
|
||||||
|
#ifdef T_WATCH_S3
|
||||||
|
static void drawAnalogClockFrame(OLEDDisplay *display, OLEDDisplayUiState *state, int16_t x, int16_t y);
|
||||||
|
|
||||||
|
static void drawDigitalClockFrame(OLEDDisplay *display, OLEDDisplayUiState *state, int16_t x, int16_t y);
|
||||||
|
|
||||||
|
static void drawSegmentedDisplayCharacter(OLEDDisplay *display, int x, int y, uint8_t number, float scale = 1);
|
||||||
|
|
||||||
|
static void drawHorizontalSegment(OLEDDisplay *display, int x, int y, int width, int height);
|
||||||
|
|
||||||
|
static void drawVerticalSegment(OLEDDisplay *display, int x, int y, int width, int height);
|
||||||
|
|
||||||
|
static void drawSegmentedDisplayColon(OLEDDisplay *display, int x, int y, float scale = 1);
|
||||||
|
|
||||||
|
static void drawWatchFaceToggleButton(OLEDDisplay *display, int16_t x, int16_t y, bool digitalMode = true, float scale = 1);
|
||||||
|
|
||||||
|
static void drawBluetoothConnectedIcon(OLEDDisplay *display, int16_t x, int16_t y);
|
||||||
|
|
||||||
|
// Whether we are showing the digital watch face or the analog one
|
||||||
|
bool digitalWatchFace = true;
|
||||||
|
#endif
|
||||||
|
|
||||||
/// Queue of commands to execute in doTask.
|
/// Queue of commands to execute in doTask.
|
||||||
TypedQueue<ScreenCmd> cmdQueue;
|
TypedQueue<ScreenCmd> cmdQueue;
|
||||||
/// Whether we are using a display
|
/// Whether we are using a display
|
||||||
|
@ -14,6 +14,12 @@ const uint8_t imgUser[] PROGMEM = {0x3C, 0x42, 0x99, 0xA5, 0xA5, 0x99, 0x42, 0x3
|
|||||||
const uint8_t imgPositionEmpty[] PROGMEM = {0x20, 0x30, 0x28, 0x24, 0x42, 0xFF};
|
const uint8_t imgPositionEmpty[] PROGMEM = {0x20, 0x30, 0x28, 0x24, 0x42, 0xFF};
|
||||||
const uint8_t imgPositionSolid[] PROGMEM = {0x20, 0x30, 0x38, 0x3C, 0x7E, 0xFF};
|
const uint8_t imgPositionSolid[] PROGMEM = {0x20, 0x30, 0x38, 0x3C, 0x7E, 0xFF};
|
||||||
|
|
||||||
|
#ifdef T_WATCH_S3
|
||||||
|
const uint8_t bluetoothConnectedIcon[36] PROGMEM = {0xfe, 0x01, 0xff, 0x03, 0x03, 0x03, 0x03, 0x03, 0x03, 0x00, 0xe3, 0x1f,
|
||||||
|
0xf3, 0x3f, 0x33, 0x30, 0x33, 0x33, 0x33, 0x33, 0x03, 0x33, 0xff, 0x33,
|
||||||
|
0xfe, 0x31, 0x00, 0x30, 0x30, 0x30, 0x30, 0x30, 0xf0, 0x3f, 0xe0, 0x1f};
|
||||||
|
#endif
|
||||||
|
|
||||||
#if (defined(USE_EINK) || defined(ILI9341_DRIVER) || defined(ST7735_CS) || defined(ST7789_CS) || defined(HX8357_CS) || \
|
#if (defined(USE_EINK) || defined(ILI9341_DRIVER) || defined(ST7735_CS) || defined(ST7789_CS) || defined(HX8357_CS) || \
|
||||||
ARCH_PORTDUINO) && \
|
ARCH_PORTDUINO) && \
|
||||||
!defined(DISPLAY_FORCE_SMALL_FONTS)
|
!defined(DISPLAY_FORCE_SMALL_FONTS)
|
||||||
|
@ -8,6 +8,8 @@ typedef struct _InputEvent {
|
|||||||
const char *source;
|
const char *source;
|
||||||
char inputEvent;
|
char inputEvent;
|
||||||
char kbchar;
|
char kbchar;
|
||||||
|
uint16_t touchX;
|
||||||
|
uint16_t touchY;
|
||||||
} InputEvent;
|
} InputEvent;
|
||||||
class InputBroker : public Observable<const InputEvent *>
|
class InputBroker : public Observable<const InputEvent *>
|
||||||
{
|
{
|
||||||
|
@ -49,6 +49,10 @@ void TouchScreenImpl1::onEvent(const TouchEvent &event)
|
|||||||
{
|
{
|
||||||
InputEvent e;
|
InputEvent e;
|
||||||
e.source = event.source;
|
e.source = event.source;
|
||||||
|
|
||||||
|
e.touchX = event.x;
|
||||||
|
e.touchY = event.y;
|
||||||
|
|
||||||
switch (event.touchEvent) {
|
switch (event.touchEvent) {
|
||||||
case TOUCH_ACTION_LEFT: {
|
case TOUCH_ACTION_LEFT: {
|
||||||
e.inputEvent = static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_RIGHT);
|
e.inputEvent = static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_RIGHT);
|
||||||
|
@ -47,6 +47,12 @@ CannedMessageModule::CannedMessageModule()
|
|||||||
disable();
|
disable();
|
||||||
} else {
|
} else {
|
||||||
LOG_INFO("CannedMessageModule is enabled\n");
|
LOG_INFO("CannedMessageModule is enabled\n");
|
||||||
|
|
||||||
|
// T-Watch interface currently has no way to select destination type, so default to 'node'
|
||||||
|
#ifdef T_WATCH_S3
|
||||||
|
this->destSelect = CANNED_MESSAGE_DESTINATION_TYPE_NODE;
|
||||||
|
#endif
|
||||||
|
|
||||||
this->inputObserver.observe(inputBroker);
|
this->inputObserver.observe(inputBroker);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@ -67,8 +73,14 @@ int CannedMessageModule::splitConfiguredMessages()
|
|||||||
int messageIndex = 0;
|
int messageIndex = 0;
|
||||||
int i = 0;
|
int i = 0;
|
||||||
|
|
||||||
|
String messages = cannedMessageModuleConfig.messages;
|
||||||
|
|
||||||
|
String separator = messages.length() ? "|" : "";
|
||||||
|
|
||||||
|
messages = "[---- Free Text ----]" + separator + messages;
|
||||||
|
|
||||||
// collect all the message parts
|
// collect all the message parts
|
||||||
strncpy(this->messageStore, cannedMessageModuleConfig.messages, sizeof(this->messageStore));
|
strncpy(this->messageStore, messages.c_str(), sizeof(this->messageStore));
|
||||||
|
|
||||||
// The first message points to the beginning of the store.
|
// The first message points to the beginning of the store.
|
||||||
this->messages[messageIndex++] = this->messageStore;
|
this->messages[messageIndex++] = this->messageStore;
|
||||||
@ -78,7 +90,6 @@ int CannedMessageModule::splitConfiguredMessages()
|
|||||||
if (this->messageStore[i] == '|') {
|
if (this->messageStore[i] == '|') {
|
||||||
// Message ending found, replace it with string-end character.
|
// Message ending found, replace it with string-end character.
|
||||||
this->messageStore[i] = '\0';
|
this->messageStore[i] = '\0';
|
||||||
LOG_DEBUG("CannedMessage %d is: '%s'\n", messageIndex - 1, this->messages[messageIndex - 1]);
|
|
||||||
|
|
||||||
// hit our max messages, bail
|
// hit our max messages, bail
|
||||||
if (messageIndex >= CANNED_MESSAGE_MODULE_MESSAGE_MAX_COUNT) {
|
if (messageIndex >= CANNED_MESSAGE_MODULE_MESSAGE_MAX_COUNT) {
|
||||||
@ -119,20 +130,27 @@ int CannedMessageModule::handleInputEvent(const InputEvent *event)
|
|||||||
bool validEvent = false;
|
bool validEvent = false;
|
||||||
if (event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_UP)) {
|
if (event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_UP)) {
|
||||||
if (this->messagesCount > 0) {
|
if (this->messagesCount > 0) {
|
||||||
// LOG_DEBUG("Canned message event UP\n");
|
|
||||||
this->runState = CANNED_MESSAGE_RUN_STATE_ACTION_UP;
|
this->runState = CANNED_MESSAGE_RUN_STATE_ACTION_UP;
|
||||||
validEvent = true;
|
validEvent = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_DOWN)) {
|
if (event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_DOWN)) {
|
||||||
if (this->messagesCount > 0) {
|
if (this->messagesCount > 0) {
|
||||||
// LOG_DEBUG("Canned message event DOWN\n");
|
|
||||||
this->runState = CANNED_MESSAGE_RUN_STATE_ACTION_DOWN;
|
this->runState = CANNED_MESSAGE_RUN_STATE_ACTION_DOWN;
|
||||||
validEvent = true;
|
validEvent = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_SELECT)) {
|
if (event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_SELECT)) {
|
||||||
LOG_DEBUG("Canned message event Select\n");
|
if (this->currentMessageIndex == 0) {
|
||||||
|
this->runState = CANNED_MESSAGE_RUN_STATE_FREETEXT;
|
||||||
|
|
||||||
|
UIFrameEvent e = {false, true};
|
||||||
|
e.frameChanged = true;
|
||||||
|
this->notifyObservers(&e);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
// when inactive, call the onebutton shortpress instead. Activate Module only on up/down
|
// when inactive, call the onebutton shortpress instead. Activate Module only on up/down
|
||||||
if ((this->runState == CANNED_MESSAGE_RUN_STATE_INACTIVE) || (this->runState == CANNED_MESSAGE_RUN_STATE_DISABLED)) {
|
if ((this->runState == CANNED_MESSAGE_RUN_STATE_INACTIVE) || (this->runState == CANNED_MESSAGE_RUN_STATE_DISABLED)) {
|
||||||
powerFSM.trigger(EVENT_PRESS);
|
powerFSM.trigger(EVENT_PRESS);
|
||||||
@ -143,38 +161,47 @@ int CannedMessageModule::handleInputEvent(const InputEvent *event)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_CANCEL)) {
|
if (event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_CANCEL)) {
|
||||||
LOG_DEBUG("Canned message event Cancel\n");
|
|
||||||
UIFrameEvent e = {false, true};
|
UIFrameEvent e = {false, true};
|
||||||
e.frameChanged = true;
|
e.frameChanged = true;
|
||||||
this->currentMessageIndex = -1;
|
this->currentMessageIndex = -1;
|
||||||
|
|
||||||
|
#ifndef T_WATCH_S3
|
||||||
this->freetext = ""; // clear freetext
|
this->freetext = ""; // clear freetext
|
||||||
this->cursor = 0;
|
this->cursor = 0;
|
||||||
this->destSelect = CANNED_MESSAGE_DESTINATION_TYPE_NONE;
|
this->destSelect = CANNED_MESSAGE_DESTINATION_TYPE_NONE;
|
||||||
|
#endif
|
||||||
|
|
||||||
this->runState = CANNED_MESSAGE_RUN_STATE_INACTIVE;
|
this->runState = CANNED_MESSAGE_RUN_STATE_INACTIVE;
|
||||||
this->notifyObservers(&e);
|
this->notifyObservers(&e);
|
||||||
}
|
}
|
||||||
if ((event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_BACK)) ||
|
if ((event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_BACK)) ||
|
||||||
(event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_LEFT)) ||
|
(event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_LEFT)) ||
|
||||||
(event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_RIGHT))) {
|
(event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_RIGHT))) {
|
||||||
// LOG_DEBUG("Canned message event (%x)\n", event->kbchar);
|
|
||||||
|
#ifdef T_WATCH_S3
|
||||||
|
if (event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_LEFT)) {
|
||||||
|
this->payload = 0xb4;
|
||||||
|
} else if (event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_RIGHT)) {
|
||||||
|
this->payload = 0xb7;
|
||||||
|
}
|
||||||
|
#else
|
||||||
// tweak for left/right events generated via trackball/touch with empty kbchar
|
// tweak for left/right events generated via trackball/touch with empty kbchar
|
||||||
if (!event->kbchar) {
|
if (!event->kbchar) {
|
||||||
if (event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_LEFT)) {
|
if (event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_LEFT)) {
|
||||||
this->payload = 0xb4;
|
this->payload = 0xb4;
|
||||||
// this->destSelect = CANNED_MESSAGE_DESTINATION_TYPE_NODE;
|
|
||||||
} else if (event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_RIGHT)) {
|
} else if (event->inputEvent == static_cast<char>(meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_RIGHT)) {
|
||||||
this->payload = 0xb7;
|
this->payload = 0xb7;
|
||||||
// this->destSelect = CANNED_MESSAGE_DESTINATION_TYPE_NODE;
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// pass the pressed key
|
// pass the pressed key
|
||||||
this->payload = event->kbchar;
|
this->payload = event->kbchar;
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
this->lastTouchMillis = millis();
|
this->lastTouchMillis = millis();
|
||||||
validEvent = true;
|
validEvent = true;
|
||||||
}
|
}
|
||||||
if (event->inputEvent == static_cast<char>(ANYKEY)) {
|
if (event->inputEvent == static_cast<char>(ANYKEY)) {
|
||||||
LOG_DEBUG("Canned message event any key pressed\n");
|
|
||||||
// when inactive, this will switch to the freetext mode
|
// when inactive, this will switch to the freetext mode
|
||||||
if ((this->runState == CANNED_MESSAGE_RUN_STATE_INACTIVE) || (this->runState == CANNED_MESSAGE_RUN_STATE_ACTIVE) ||
|
if ((this->runState == CANNED_MESSAGE_RUN_STATE_INACTIVE) || (this->runState == CANNED_MESSAGE_RUN_STATE_ACTIVE) ||
|
||||||
(this->runState == CANNED_MESSAGE_RUN_STATE_DISABLED)) {
|
(this->runState == CANNED_MESSAGE_RUN_STATE_DISABLED)) {
|
||||||
@ -250,8 +277,68 @@ int CannedMessageModule::handleInputEvent(const InputEvent *event)
|
|||||||
screen->removeFunctionSymbal("Fn"); // remove modifier (function) symbal
|
screen->removeFunctionSymbal("Fn"); // remove modifier (function) symbal
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef T_WATCH_S3
|
||||||
|
if (this->runState == CANNED_MESSAGE_RUN_STATE_FREETEXT) {
|
||||||
|
String keyTapped = keyForCoordinates(event->touchX, event->touchY);
|
||||||
|
|
||||||
|
if (keyTapped == "⇧") {
|
||||||
|
this->highlight = -1;
|
||||||
|
|
||||||
|
this->payload = 0x00;
|
||||||
|
|
||||||
|
validEvent = true;
|
||||||
|
|
||||||
|
this->shift = !this->shift;
|
||||||
|
} else if (keyTapped == "⌫") {
|
||||||
|
this->highlight = keyTapped[0];
|
||||||
|
|
||||||
|
this->payload = 0x08;
|
||||||
|
|
||||||
|
validEvent = true;
|
||||||
|
|
||||||
|
this->shift = false;
|
||||||
|
} else if (keyTapped == "123" || keyTapped == "ABC") {
|
||||||
|
this->highlight = -1;
|
||||||
|
|
||||||
|
this->payload = 0x00;
|
||||||
|
|
||||||
|
this->charSet = this->charSet == 0 ? 1 : 0;
|
||||||
|
|
||||||
|
validEvent = true;
|
||||||
|
} else if (keyTapped == " ") {
|
||||||
|
this->highlight = keyTapped[0];
|
||||||
|
|
||||||
|
this->payload = keyTapped[0];
|
||||||
|
|
||||||
|
validEvent = true;
|
||||||
|
|
||||||
|
this->shift = false;
|
||||||
|
} else if (keyTapped == "↵") {
|
||||||
|
this->highlight = 0x00;
|
||||||
|
|
||||||
|
this->runState = CANNED_MESSAGE_RUN_STATE_ACTION_SELECT;
|
||||||
|
|
||||||
|
this->payload = CANNED_MESSAGE_RUN_STATE_FREETEXT;
|
||||||
|
|
||||||
|
this->currentMessageIndex = event->kbchar - 1;
|
||||||
|
|
||||||
|
validEvent = true;
|
||||||
|
|
||||||
|
this->shift = false;
|
||||||
|
} else if (keyTapped != "") {
|
||||||
|
this->highlight = keyTapped[0];
|
||||||
|
|
||||||
|
this->payload = this->shift ? keyTapped[0] : std::tolower(keyTapped[0]);
|
||||||
|
|
||||||
|
validEvent = true;
|
||||||
|
|
||||||
|
this->shift = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
if (event->inputEvent == static_cast<char>(MATRIXKEY)) {
|
if (event->inputEvent == static_cast<char>(MATRIXKEY)) {
|
||||||
LOG_DEBUG("Canned message event Matrix key pressed\n");
|
|
||||||
// this will send the text immediately on matrix press
|
// this will send the text immediately on matrix press
|
||||||
this->runState = CANNED_MESSAGE_RUN_STATE_ACTION_SELECT;
|
this->runState = CANNED_MESSAGE_RUN_STATE_ACTION_SELECT;
|
||||||
this->payload = MATRIXKEY;
|
this->payload = MATRIXKEY;
|
||||||
@ -311,17 +398,24 @@ int32_t CannedMessageModule::runOnce()
|
|||||||
this->currentMessageIndex = -1;
|
this->currentMessageIndex = -1;
|
||||||
this->freetext = ""; // clear freetext
|
this->freetext = ""; // clear freetext
|
||||||
this->cursor = 0;
|
this->cursor = 0;
|
||||||
|
|
||||||
|
#ifndef T_WATCH_S3
|
||||||
this->destSelect = CANNED_MESSAGE_DESTINATION_TYPE_NONE;
|
this->destSelect = CANNED_MESSAGE_DESTINATION_TYPE_NONE;
|
||||||
|
#endif
|
||||||
|
|
||||||
this->notifyObservers(&e);
|
this->notifyObservers(&e);
|
||||||
} else if (((this->runState == CANNED_MESSAGE_RUN_STATE_ACTIVE) || (this->runState == CANNED_MESSAGE_RUN_STATE_FREETEXT)) &&
|
} else if (((this->runState == CANNED_MESSAGE_RUN_STATE_ACTIVE) || (this->runState == CANNED_MESSAGE_RUN_STATE_FREETEXT)) &&
|
||||||
((millis() - this->lastTouchMillis) > INACTIVATE_AFTER_MS)) {
|
((millis() - this->lastTouchMillis) > INACTIVATE_AFTER_MS)) {
|
||||||
// Reset module
|
// Reset module
|
||||||
LOG_DEBUG("Reset due to lack of activity.\n");
|
|
||||||
e.frameChanged = true;
|
e.frameChanged = true;
|
||||||
this->currentMessageIndex = -1;
|
this->currentMessageIndex = -1;
|
||||||
this->freetext = ""; // clear freetext
|
this->freetext = ""; // clear freetext
|
||||||
this->cursor = 0;
|
this->cursor = 0;
|
||||||
|
|
||||||
|
#ifndef T_WATCH_S3
|
||||||
this->destSelect = CANNED_MESSAGE_DESTINATION_TYPE_NONE;
|
this->destSelect = CANNED_MESSAGE_DESTINATION_TYPE_NONE;
|
||||||
|
#endif
|
||||||
|
|
||||||
this->runState = CANNED_MESSAGE_RUN_STATE_INACTIVE;
|
this->runState = CANNED_MESSAGE_RUN_STATE_INACTIVE;
|
||||||
this->notifyObservers(&e);
|
this->notifyObservers(&e);
|
||||||
} else if (this->runState == CANNED_MESSAGE_RUN_STATE_ACTION_SELECT) {
|
} else if (this->runState == CANNED_MESSAGE_RUN_STATE_ACTION_SELECT) {
|
||||||
@ -330,7 +424,6 @@ int32_t CannedMessageModule::runOnce()
|
|||||||
sendText(this->dest, indexChannels[this->channel], this->freetext.c_str(), true);
|
sendText(this->dest, indexChannels[this->channel], this->freetext.c_str(), true);
|
||||||
this->runState = CANNED_MESSAGE_RUN_STATE_SENDING_ACTIVE;
|
this->runState = CANNED_MESSAGE_RUN_STATE_SENDING_ACTIVE;
|
||||||
} else {
|
} else {
|
||||||
LOG_DEBUG("Reset message is empty.\n");
|
|
||||||
this->runState = CANNED_MESSAGE_RUN_STATE_INACTIVE;
|
this->runState = CANNED_MESSAGE_RUN_STATE_INACTIVE;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@ -339,11 +432,15 @@ int32_t CannedMessageModule::runOnce()
|
|||||||
powerFSM.trigger(EVENT_PRESS);
|
powerFSM.trigger(EVENT_PRESS);
|
||||||
return INT32_MAX;
|
return INT32_MAX;
|
||||||
} else {
|
} else {
|
||||||
|
#ifdef T_WATCH_S3
|
||||||
|
sendText(this->dest, indexChannels[this->channel], this->messages[this->currentMessageIndex], true);
|
||||||
|
#else
|
||||||
sendText(NODENUM_BROADCAST, channels.getPrimaryIndex(), this->messages[this->currentMessageIndex], true);
|
sendText(NODENUM_BROADCAST, channels.getPrimaryIndex(), this->messages[this->currentMessageIndex], true);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
this->runState = CANNED_MESSAGE_RUN_STATE_SENDING_ACTIVE;
|
this->runState = CANNED_MESSAGE_RUN_STATE_SENDING_ACTIVE;
|
||||||
} else {
|
} else {
|
||||||
LOG_DEBUG("Reset message is empty.\n");
|
// LOG_DEBUG("Reset message is empty.\n");
|
||||||
this->runState = CANNED_MESSAGE_RUN_STATE_INACTIVE;
|
this->runState = CANNED_MESSAGE_RUN_STATE_INACTIVE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -351,7 +448,11 @@ int32_t CannedMessageModule::runOnce()
|
|||||||
this->currentMessageIndex = -1;
|
this->currentMessageIndex = -1;
|
||||||
this->freetext = ""; // clear freetext
|
this->freetext = ""; // clear freetext
|
||||||
this->cursor = 0;
|
this->cursor = 0;
|
||||||
|
|
||||||
|
#ifndef T_WATCH_S3
|
||||||
this->destSelect = CANNED_MESSAGE_DESTINATION_TYPE_NONE;
|
this->destSelect = CANNED_MESSAGE_DESTINATION_TYPE_NONE;
|
||||||
|
#endif
|
||||||
|
|
||||||
this->notifyObservers(&e);
|
this->notifyObservers(&e);
|
||||||
return 2000;
|
return 2000;
|
||||||
} else if ((this->runState != CANNED_MESSAGE_RUN_STATE_FREETEXT) && (this->currentMessageIndex == -1)) {
|
} else if ((this->runState != CANNED_MESSAGE_RUN_STATE_FREETEXT) && (this->currentMessageIndex == -1)) {
|
||||||
@ -364,7 +465,11 @@ int32_t CannedMessageModule::runOnce()
|
|||||||
this->currentMessageIndex = getPrevIndex();
|
this->currentMessageIndex = getPrevIndex();
|
||||||
this->freetext = ""; // clear freetext
|
this->freetext = ""; // clear freetext
|
||||||
this->cursor = 0;
|
this->cursor = 0;
|
||||||
|
|
||||||
|
#ifndef T_WATCH_S3
|
||||||
this->destSelect = CANNED_MESSAGE_DESTINATION_TYPE_NONE;
|
this->destSelect = CANNED_MESSAGE_DESTINATION_TYPE_NONE;
|
||||||
|
#endif
|
||||||
|
|
||||||
this->runState = CANNED_MESSAGE_RUN_STATE_ACTIVE;
|
this->runState = CANNED_MESSAGE_RUN_STATE_ACTIVE;
|
||||||
LOG_DEBUG("MOVE UP (%d):%s\n", this->currentMessageIndex, this->getCurrentMessage());
|
LOG_DEBUG("MOVE UP (%d):%s\n", this->currentMessageIndex, this->getCurrentMessage());
|
||||||
}
|
}
|
||||||
@ -373,7 +478,11 @@ int32_t CannedMessageModule::runOnce()
|
|||||||
this->currentMessageIndex = this->getNextIndex();
|
this->currentMessageIndex = this->getNextIndex();
|
||||||
this->freetext = ""; // clear freetext
|
this->freetext = ""; // clear freetext
|
||||||
this->cursor = 0;
|
this->cursor = 0;
|
||||||
|
|
||||||
|
#ifndef T_WATCH_S3
|
||||||
this->destSelect = CANNED_MESSAGE_DESTINATION_TYPE_NONE;
|
this->destSelect = CANNED_MESSAGE_DESTINATION_TYPE_NONE;
|
||||||
|
#endif
|
||||||
|
|
||||||
this->runState = CANNED_MESSAGE_RUN_STATE_ACTIVE;
|
this->runState = CANNED_MESSAGE_RUN_STATE_ACTIVE;
|
||||||
LOG_DEBUG("MOVE DOWN (%d):%s\n", this->currentMessageIndex, this->getCurrentMessage());
|
LOG_DEBUG("MOVE DOWN (%d):%s\n", this->currentMessageIndex, this->getCurrentMessage());
|
||||||
}
|
}
|
||||||
@ -457,7 +566,7 @@ int32_t CannedMessageModule::runOnce()
|
|||||||
switch (this->payload) { // code below all trigger the freetext window (where you type to send a message) or reset the
|
switch (this->payload) { // code below all trigger the freetext window (where you type to send a message) or reset the
|
||||||
// display back to the default window
|
// display back to the default window
|
||||||
case 0x08: // backspace
|
case 0x08: // backspace
|
||||||
if (this->freetext.length() > 0) {
|
if (this->freetext.length() > 0 && this->highlight == 0x00) {
|
||||||
if (this->cursor == this->freetext.length()) {
|
if (this->cursor == this->freetext.length()) {
|
||||||
this->freetext = this->freetext.substring(0, this->freetext.length() - 1);
|
this->freetext = this->freetext.substring(0, this->freetext.length() - 1);
|
||||||
} else {
|
} else {
|
||||||
@ -495,13 +604,19 @@ int32_t CannedMessageModule::runOnce()
|
|||||||
runState = CANNED_MESSAGE_RUN_STATE_INACTIVE;
|
runState = CANNED_MESSAGE_RUN_STATE_INACTIVE;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
if (this->highlight != 0x00) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
if (this->cursor == this->freetext.length()) {
|
if (this->cursor == this->freetext.length()) {
|
||||||
this->freetext += this->payload;
|
this->freetext += this->payload;
|
||||||
} else {
|
} else {
|
||||||
this->freetext =
|
this->freetext =
|
||||||
this->freetext.substring(0, this->cursor) + this->payload + this->freetext.substring(this->cursor);
|
this->freetext.substring(0, this->cursor) + this->payload + this->freetext.substring(this->cursor);
|
||||||
}
|
}
|
||||||
|
|
||||||
this->cursor += 1;
|
this->cursor += 1;
|
||||||
|
|
||||||
uint16_t maxChars = meshtastic_Constants_DATA_PAYLOAD_LEN - (moduleConfig.canned_message.send_bell ? 1 : 0);
|
uint16_t maxChars = meshtastic_Constants_DATA_PAYLOAD_LEN - (moduleConfig.canned_message.send_bell ? 1 : 0);
|
||||||
if (this->freetext.length() > maxChars) {
|
if (this->freetext.length() > maxChars) {
|
||||||
this->cursor = maxChars;
|
this->cursor = maxChars;
|
||||||
@ -594,6 +709,201 @@ void CannedMessageModule::showTemporaryMessage(const String &message)
|
|||||||
setIntervalFromNow(2000);
|
setIntervalFromNow(2000);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef T_WATCH_S3
|
||||||
|
|
||||||
|
String CannedMessageModule::keyForCoordinates(uint x, uint y)
|
||||||
|
{
|
||||||
|
int outerSize = *(&this->keyboard[this->charSet] + 1) - this->keyboard[this->charSet];
|
||||||
|
|
||||||
|
for (int8_t outerIndex = 0; outerIndex < outerSize; outerIndex++) {
|
||||||
|
int innerSize = *(&this->keyboard[this->charSet][outerIndex] + 1) - this->keyboard[this->charSet][outerIndex];
|
||||||
|
|
||||||
|
for (int8_t innerIndex = 0; innerIndex < innerSize; innerIndex++) {
|
||||||
|
Letter letter = this->keyboard[this->charSet][outerIndex][innerIndex];
|
||||||
|
|
||||||
|
if (x > letter.rectX && x < (letter.rectX + letter.rectWidth) && y > letter.rectY &&
|
||||||
|
y < (letter.rectY + letter.rectHeight)) {
|
||||||
|
return letter.character;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
void CannedMessageModule::drawKeyboard(OLEDDisplay *display, OLEDDisplayUiState *state, int16_t x, int16_t y)
|
||||||
|
{
|
||||||
|
int outerSize = *(&this->keyboard[this->charSet] + 1) - this->keyboard[this->charSet];
|
||||||
|
|
||||||
|
int xOffset = 0;
|
||||||
|
|
||||||
|
int yOffset = 56;
|
||||||
|
|
||||||
|
display->setTextAlignment(TEXT_ALIGN_LEFT);
|
||||||
|
|
||||||
|
display->setFont(FONT_SMALL);
|
||||||
|
|
||||||
|
display->setColor(OLEDDISPLAY_COLOR::WHITE);
|
||||||
|
|
||||||
|
display->drawStringMaxWidth(0, 0, display->getWidth(),
|
||||||
|
cannedMessageModule->drawWithCursor(cannedMessageModule->freetext, cannedMessageModule->cursor));
|
||||||
|
|
||||||
|
display->setFont(FONT_MEDIUM);
|
||||||
|
|
||||||
|
int cellHeight = round((display->height() - 64) / outerSize);
|
||||||
|
|
||||||
|
int yCorrection = 8;
|
||||||
|
|
||||||
|
for (int8_t outerIndex = 0; outerIndex < outerSize; outerIndex++) {
|
||||||
|
yOffset += outerIndex > 0 ? cellHeight : 0;
|
||||||
|
|
||||||
|
int innerSizeBound = *(&this->keyboard[this->charSet][outerIndex] + 1) - this->keyboard[this->charSet][outerIndex];
|
||||||
|
|
||||||
|
int innerSize = 0;
|
||||||
|
|
||||||
|
for (int8_t innerIndex = 0; innerIndex < innerSizeBound; innerIndex++) {
|
||||||
|
if (this->keyboard[this->charSet][outerIndex][innerIndex].character != "") {
|
||||||
|
innerSize++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int cellWidth = display->width() / innerSize;
|
||||||
|
|
||||||
|
for (int8_t innerIndex = 0; innerIndex < innerSize; innerIndex++) {
|
||||||
|
xOffset += innerIndex > 0 ? cellWidth : 0;
|
||||||
|
|
||||||
|
Letter letter = this->keyboard[this->charSet][outerIndex][innerIndex];
|
||||||
|
|
||||||
|
Letter updatedLetter = {letter.character, letter.width, xOffset, yOffset, cellWidth, cellHeight};
|
||||||
|
|
||||||
|
this->keyboard[this->charSet][outerIndex][innerIndex] = updatedLetter;
|
||||||
|
|
||||||
|
float characterOffset = ((cellWidth / 2) - (letter.width / 2));
|
||||||
|
|
||||||
|
if (letter.character == "⇧") {
|
||||||
|
if (this->shift) {
|
||||||
|
display->fillRect(xOffset, yOffset, cellWidth, cellHeight);
|
||||||
|
|
||||||
|
display->setColor(OLEDDISPLAY_COLOR::BLACK);
|
||||||
|
|
||||||
|
drawShiftIcon(display, xOffset + characterOffset, yOffset + yCorrection + 5, 1.2);
|
||||||
|
|
||||||
|
display->setColor(OLEDDISPLAY_COLOR::WHITE);
|
||||||
|
} else {
|
||||||
|
display->drawRect(xOffset, yOffset, cellWidth, cellHeight);
|
||||||
|
|
||||||
|
drawShiftIcon(display, xOffset + characterOffset, yOffset + yCorrection + 5, 1.2);
|
||||||
|
}
|
||||||
|
} else if (letter.character == "⌫") {
|
||||||
|
if (this->highlight == letter.character[0]) {
|
||||||
|
display->fillRect(xOffset, yOffset, cellWidth, cellHeight);
|
||||||
|
|
||||||
|
display->setColor(OLEDDISPLAY_COLOR::BLACK);
|
||||||
|
|
||||||
|
drawBackspaceIcon(display, xOffset + characterOffset, yOffset + yCorrection + 5, 1.2);
|
||||||
|
|
||||||
|
display->setColor(OLEDDISPLAY_COLOR::WHITE);
|
||||||
|
|
||||||
|
setIntervalFromNow(0);
|
||||||
|
} else {
|
||||||
|
display->drawRect(xOffset, yOffset, cellWidth, cellHeight);
|
||||||
|
|
||||||
|
drawBackspaceIcon(display, xOffset + characterOffset, yOffset + yCorrection + 5, 1.2);
|
||||||
|
}
|
||||||
|
} else if (letter.character == "↵") {
|
||||||
|
display->drawRect(xOffset, yOffset, cellWidth, cellHeight);
|
||||||
|
|
||||||
|
drawEnterIcon(display, xOffset + characterOffset, yOffset + yCorrection + 5, 1.7);
|
||||||
|
} else {
|
||||||
|
if (this->highlight == letter.character[0]) {
|
||||||
|
display->fillRect(xOffset, yOffset, cellWidth, cellHeight);
|
||||||
|
|
||||||
|
display->setColor(OLEDDISPLAY_COLOR::BLACK);
|
||||||
|
|
||||||
|
display->drawString(xOffset + characterOffset, yOffset + yCorrection,
|
||||||
|
letter.character == " " ? "space" : letter.character);
|
||||||
|
|
||||||
|
display->setColor(OLEDDISPLAY_COLOR::WHITE);
|
||||||
|
|
||||||
|
setIntervalFromNow(0);
|
||||||
|
} else {
|
||||||
|
display->drawRect(xOffset, yOffset, cellWidth, cellHeight);
|
||||||
|
|
||||||
|
display->drawString(xOffset + characterOffset, yOffset + yCorrection,
|
||||||
|
letter.character == " " ? "space" : letter.character);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
xOffset = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
this->highlight = 0x00;
|
||||||
|
}
|
||||||
|
|
||||||
|
void CannedMessageModule::drawShiftIcon(OLEDDisplay *display, int x, int y, float scale)
|
||||||
|
{
|
||||||
|
PointStruct shiftIcon[10] = {{8, 0}, {15, 7}, {15, 8}, {12, 8}, {12, 12}, {4, 12}, {4, 8}, {1, 8}, {1, 7}, {8, 0}};
|
||||||
|
|
||||||
|
int size = 10;
|
||||||
|
|
||||||
|
for (int i = 0; i < size - 1; i++) {
|
||||||
|
int x0 = x + (shiftIcon[i].x * scale);
|
||||||
|
int y0 = y + (shiftIcon[i].y * scale);
|
||||||
|
int x1 = x + (shiftIcon[i + 1].x * scale);
|
||||||
|
int y1 = y + (shiftIcon[i + 1].y * scale);
|
||||||
|
|
||||||
|
display->drawLine(x0, y0, x1, y1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void CannedMessageModule::drawBackspaceIcon(OLEDDisplay *display, int x, int y, float scale)
|
||||||
|
{
|
||||||
|
PointStruct backspaceIcon[6] = {{0, 7}, {5, 2}, {15, 2}, {15, 12}, {5, 12}, {0, 7}};
|
||||||
|
|
||||||
|
int size = 6;
|
||||||
|
|
||||||
|
for (int i = 0; i < size - 1; i++) {
|
||||||
|
int x0 = x + (backspaceIcon[i].x * scale);
|
||||||
|
int y0 = y + (backspaceIcon[i].y * scale);
|
||||||
|
int x1 = x + (backspaceIcon[i + 1].x * scale);
|
||||||
|
int y1 = y + (backspaceIcon[i + 1].y * scale);
|
||||||
|
|
||||||
|
display->drawLine(x0, y0, x1, y1);
|
||||||
|
}
|
||||||
|
|
||||||
|
PointStruct backspaceIconX[4] = {{7, 4}, {13, 10}, {7, 10}, {13, 4}};
|
||||||
|
|
||||||
|
size = 4;
|
||||||
|
|
||||||
|
for (int i = 0; i < size - 1; i++) {
|
||||||
|
int x0 = x + (backspaceIconX[i].x * scale);
|
||||||
|
int y0 = y + (backspaceIconX[i].y * scale);
|
||||||
|
int x1 = x + (backspaceIconX[i + 1].x * scale);
|
||||||
|
int y1 = y + (backspaceIconX[i + 1].y * scale);
|
||||||
|
|
||||||
|
display->drawLine(x0, y0, x1, y1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void CannedMessageModule::drawEnterIcon(OLEDDisplay *display, int x, int y, float scale)
|
||||||
|
{
|
||||||
|
PointStruct enterIcon[6] = {{0, 7}, {4, 3}, {4, 11}, {0, 7}, {15, 7}, {15, 0}};
|
||||||
|
|
||||||
|
int size = 6;
|
||||||
|
|
||||||
|
for (int i = 0; i < size - 1; i++) {
|
||||||
|
int x0 = x + (enterIcon[i].x * scale);
|
||||||
|
int y0 = y + (enterIcon[i].y * scale);
|
||||||
|
int x1 = x + (enterIcon[i + 1].x * scale);
|
||||||
|
int y1 = y + (enterIcon[i + 1].y * scale);
|
||||||
|
|
||||||
|
display->drawLine(x0, y0, x1, y1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
void CannedMessageModule::drawFrame(OLEDDisplay *display, OLEDDisplayUiState *state, int16_t x, int16_t y)
|
void CannedMessageModule::drawFrame(OLEDDisplay *display, OLEDDisplayUiState *state, int16_t x, int16_t y)
|
||||||
{
|
{
|
||||||
char buffer[50];
|
char buffer[50];
|
||||||
@ -614,6 +924,16 @@ void CannedMessageModule::drawFrame(OLEDDisplay *display, OLEDDisplayUiState *st
|
|||||||
}
|
}
|
||||||
display->drawStringf(display->getWidth() / 2 + x, 0 + y + 12, buffer, displayString,
|
display->drawStringf(display->getWidth() / 2 + x, 0 + y + 12, buffer, displayString,
|
||||||
cannedMessageModule->getNodeName(this->incoming));
|
cannedMessageModule->getNodeName(this->incoming));
|
||||||
|
|
||||||
|
display->setFont(FONT_SMALL);
|
||||||
|
|
||||||
|
String snrString = "Last Rx SNR: %f";
|
||||||
|
String rssiString = "Last Rx RSSI: %d";
|
||||||
|
|
||||||
|
if (this->ack) {
|
||||||
|
display->drawStringf(display->getWidth() / 2 + x, y + 100, buffer, snrString, this->lastRxSnr);
|
||||||
|
display->drawStringf(display->getWidth() / 2 + x, y + 130, buffer, rssiString, this->lastRxRssi);
|
||||||
|
}
|
||||||
} else if (cannedMessageModule->runState == CANNED_MESSAGE_RUN_STATE_SENDING_ACTIVE) {
|
} else if (cannedMessageModule->runState == CANNED_MESSAGE_RUN_STATE_SENDING_ACTIVE) {
|
||||||
display->setTextAlignment(TEXT_ALIGN_CENTER);
|
display->setTextAlignment(TEXT_ALIGN_CENTER);
|
||||||
display->setFont(FONT_MEDIUM);
|
display->setFont(FONT_MEDIUM);
|
||||||
@ -623,6 +943,11 @@ void CannedMessageModule::drawFrame(OLEDDisplay *display, OLEDDisplayUiState *st
|
|||||||
display->setFont(FONT_SMALL);
|
display->setFont(FONT_SMALL);
|
||||||
display->drawString(10 + x, 0 + y + FONT_HEIGHT_SMALL, "Canned Message\nModule disabled.");
|
display->drawString(10 + x, 0 + y + FONT_HEIGHT_SMALL, "Canned Message\nModule disabled.");
|
||||||
} else if (cannedMessageModule->runState == CANNED_MESSAGE_RUN_STATE_FREETEXT) {
|
} else if (cannedMessageModule->runState == CANNED_MESSAGE_RUN_STATE_FREETEXT) {
|
||||||
|
|
||||||
|
#ifdef T_WATCH_S3
|
||||||
|
drawKeyboard(display, state, 0, 0);
|
||||||
|
#else
|
||||||
|
|
||||||
display->setTextAlignment(TEXT_ALIGN_LEFT);
|
display->setTextAlignment(TEXT_ALIGN_LEFT);
|
||||||
display->setFont(FONT_SMALL);
|
display->setFont(FONT_SMALL);
|
||||||
if (this->destSelect != CANNED_MESSAGE_DESTINATION_TYPE_NONE) {
|
if (this->destSelect != CANNED_MESSAGE_DESTINATION_TYPE_NONE) {
|
||||||
@ -663,6 +988,7 @@ void CannedMessageModule::drawFrame(OLEDDisplay *display, OLEDDisplayUiState *st
|
|||||||
display->drawStringMaxWidth(
|
display->drawStringMaxWidth(
|
||||||
0 + x, 0 + y + FONT_HEIGHT_SMALL, x + display->getWidth(),
|
0 + x, 0 + y + FONT_HEIGHT_SMALL, x + display->getWidth(),
|
||||||
cannedMessageModule->drawWithCursor(cannedMessageModule->freetext, cannedMessageModule->cursor));
|
cannedMessageModule->drawWithCursor(cannedMessageModule->freetext, cannedMessageModule->cursor));
|
||||||
|
#endif
|
||||||
} else {
|
} else {
|
||||||
if (this->messagesCount > 0) {
|
if (this->messagesCount > 0) {
|
||||||
display->setTextAlignment(TEXT_ALIGN_LEFT);
|
display->setTextAlignment(TEXT_ALIGN_LEFT);
|
||||||
|
@ -22,6 +22,17 @@ enum cannedMessageDestinationType {
|
|||||||
CANNED_MESSAGE_DESTINATION_TYPE_CHANNEL
|
CANNED_MESSAGE_DESTINATION_TYPE_CHANNEL
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum CannedMessageModuleIconType { shift, backspace, space, enter };
|
||||||
|
|
||||||
|
struct Letter {
|
||||||
|
String character;
|
||||||
|
float width;
|
||||||
|
int rectX;
|
||||||
|
int rectY;
|
||||||
|
int rectWidth;
|
||||||
|
int rectHeight;
|
||||||
|
};
|
||||||
|
|
||||||
#define CANNED_MESSAGE_MODULE_MESSAGE_MAX_COUNT 50
|
#define CANNED_MESSAGE_MODULE_MESSAGE_MAX_COUNT 50
|
||||||
/**
|
/**
|
||||||
* Sum of CannedMessageModuleConfig part sizes.
|
* Sum of CannedMessageModuleConfig part sizes.
|
||||||
@ -61,6 +72,14 @@ class CannedMessageModule : public SinglePortModule, public Observable<const UIF
|
|||||||
*/
|
*/
|
||||||
virtual bool wantPacket(const meshtastic_MeshPacket *p) override
|
virtual bool wantPacket(const meshtastic_MeshPacket *p) override
|
||||||
{
|
{
|
||||||
|
if (p->rx_rssi != 0) {
|
||||||
|
this->lastRxRssi = p->rx_rssi;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (p->rx_snr > 0) {
|
||||||
|
this->lastRxSnr = p->rx_snr;
|
||||||
|
}
|
||||||
|
|
||||||
switch (p->decoded.portnum) {
|
switch (p->decoded.portnum) {
|
||||||
case meshtastic_PortNum_TEXT_MESSAGE_APP:
|
case meshtastic_PortNum_TEXT_MESSAGE_APP:
|
||||||
case meshtastic_PortNum_ROUTING_APP:
|
case meshtastic_PortNum_ROUTING_APP:
|
||||||
@ -79,6 +98,18 @@ class CannedMessageModule : public SinglePortModule, public Observable<const UIF
|
|||||||
int getNextIndex();
|
int getNextIndex();
|
||||||
int getPrevIndex();
|
int getPrevIndex();
|
||||||
|
|
||||||
|
#ifdef T_WATCH_S3
|
||||||
|
void drawKeyboard(OLEDDisplay *display, OLEDDisplayUiState *state, int16_t x, int16_t y);
|
||||||
|
String keyForCoordinates(uint x, uint y);
|
||||||
|
bool shift = false;
|
||||||
|
int charSet = 0;
|
||||||
|
void drawShiftIcon(OLEDDisplay *display, int x, int y, float scale = 1);
|
||||||
|
void drawBackspaceIcon(OLEDDisplay *display, int x, int y, float scale = 1);
|
||||||
|
void drawEnterIcon(OLEDDisplay *display, int x, int y, float scale = 1);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
char highlight = 0x00;
|
||||||
|
|
||||||
int handleInputEvent(const InputEvent *event);
|
int handleInputEvent(const InputEvent *event);
|
||||||
virtual bool wantUIFrame() override { return this->shouldDraw(); }
|
virtual bool wantUIFrame() override { return this->shouldDraw(); }
|
||||||
virtual Observable<const UIFrameEvent *> *getUIFrameObservable() override { return this; }
|
virtual Observable<const UIFrameEvent *> *getUIFrameObservable() override { return this; }
|
||||||
@ -110,12 +141,84 @@ class CannedMessageModule : public SinglePortModule, public Observable<const UIF
|
|||||||
ChannelIndex indexChannels[MAX_NUM_CHANNELS] = {0};
|
ChannelIndex indexChannels[MAX_NUM_CHANNELS] = {0};
|
||||||
NodeNum incoming = NODENUM_BROADCAST;
|
NodeNum incoming = NODENUM_BROADCAST;
|
||||||
bool ack = false; // True means ACK, false means NAK (error_reason != NONE)
|
bool ack = false; // True means ACK, false means NAK (error_reason != NONE)
|
||||||
|
float lastRxSnr = 0;
|
||||||
|
int32_t lastRxRssi = 0;
|
||||||
|
|
||||||
char messageStore[CANNED_MESSAGE_MODULE_MESSAGES_SIZE + 1];
|
char messageStore[CANNED_MESSAGE_MODULE_MESSAGES_SIZE + 1];
|
||||||
char *messages[CANNED_MESSAGE_MODULE_MESSAGE_MAX_COUNT];
|
char *messages[CANNED_MESSAGE_MODULE_MESSAGE_MAX_COUNT];
|
||||||
int messagesCount = 0;
|
int messagesCount = 0;
|
||||||
unsigned long lastTouchMillis = 0;
|
unsigned long lastTouchMillis = 0;
|
||||||
String temporaryMessage;
|
String temporaryMessage;
|
||||||
|
|
||||||
|
#ifdef T_WATCH_S3
|
||||||
|
Letter keyboard[2][4][10] = {{{{"Q", 20, 0, 0, 0, 0},
|
||||||
|
{"W", 22, 0, 0, 0, 0},
|
||||||
|
{"E", 17, 0, 0, 0, 0},
|
||||||
|
{"R", 16.5, 0, 0, 0, 0},
|
||||||
|
{"T", 14, 0, 0, 0, 0},
|
||||||
|
{"Y", 15, 0, 0, 0, 0},
|
||||||
|
{"U", 16.5, 0, 0, 0, 0},
|
||||||
|
{"I", 5, 0, 0, 0, 0},
|
||||||
|
{"O", 19.5, 0, 0, 0, 0},
|
||||||
|
{"P", 15.5, 0, 0, 0, 0}},
|
||||||
|
{{"A", 14, 0, 0, 0, 0},
|
||||||
|
{"S", 15, 0, 0, 0, 0},
|
||||||
|
{"D", 16.5, 0, 0, 0, 0},
|
||||||
|
{"F", 15, 0, 0, 0, 0},
|
||||||
|
{"G", 17, 0, 0, 0, 0},
|
||||||
|
{"H", 15.5, 0, 0, 0, 0},
|
||||||
|
{"J", 12, 0, 0, 0, 0},
|
||||||
|
{"K", 15.5, 0, 0, 0, 0},
|
||||||
|
{"L", 14, 0, 0, 0, 0},
|
||||||
|
{"", 0, 0, 0, 0, 0}},
|
||||||
|
{{"⇧", 20, 0, 0, 0, 0},
|
||||||
|
{"Z", 14, 0, 0, 0, 0},
|
||||||
|
{"X", 14.5, 0, 0, 0, 0},
|
||||||
|
{"C", 15.5, 0, 0, 0, 0},
|
||||||
|
{"V", 13.5, 0, 0, 0, 0},
|
||||||
|
{"B", 15, 0, 0, 0, 0},
|
||||||
|
{"N", 15, 0, 0, 0, 0},
|
||||||
|
{"M", 17, 0, 0, 0, 0},
|
||||||
|
{"⌫", 20, 0, 0, 0, 0},
|
||||||
|
{"", 0, 0, 0, 0, 0}},
|
||||||
|
{{"123", 42, 0, 0, 0, 0},
|
||||||
|
{" ", 64, 0, 0, 0, 0},
|
||||||
|
{"↵", 36, 0, 0, 0, 0},
|
||||||
|
{"", 0, 0, 0, 0, 0},
|
||||||
|
{"", 0, 0, 0, 0, 0},
|
||||||
|
{"", 0, 0, 0, 0, 0},
|
||||||
|
{"", 0, 0, 0, 0, 0},
|
||||||
|
{"", 0, 0, 0, 0, 0},
|
||||||
|
{"", 0, 0, 0, 0, 0},
|
||||||
|
{"", 0, 0, 0, 0, 0}}},
|
||||||
|
{{{"1", 12, 0, 0, 0, 0},
|
||||||
|
{"2", 13.5, 0, 0, 0, 0},
|
||||||
|
{"3", 12.5, 0, 0, 0, 0},
|
||||||
|
{"4", 14, 0, 0, 0, 0},
|
||||||
|
{"5", 14, 0, 0, 0, 0},
|
||||||
|
{"6", 14, 0, 0, 0, 0},
|
||||||
|
{"7", 13.5, 0, 0, 0, 0},
|
||||||
|
{"8", 14, 0, 0, 0, 0},
|
||||||
|
{"9", 14, 0, 0, 0, 0},
|
||||||
|
{"0", 14, 0, 0, 0, 0}},
|
||||||
|
{{"-", 8, 0, 0, 0, 0},
|
||||||
|
{"/", 8, 0, 0, 0, 0},
|
||||||
|
{":", 4.5, 0, 0, 0, 0},
|
||||||
|
{";", 4.5, 0, 0, 0, 0},
|
||||||
|
{"(", 7, 0, 0, 0, 0},
|
||||||
|
{")", 6.5, 0, 0, 0, 0},
|
||||||
|
{"$", 12.5, 0, 0, 0, 0},
|
||||||
|
{"&", 15, 0, 0, 0, 0},
|
||||||
|
{"@", 21.5, 0, 0, 0, 0},
|
||||||
|
{"\"", 8, 0, 0, 0, 0}},
|
||||||
|
{{".", 8, 0, 0, 0, 0},
|
||||||
|
{",", 8, 0, 0, 0, 0},
|
||||||
|
{"?", 10, 0, 0, 0, 0},
|
||||||
|
{"!", 10, 0, 0, 0, 0},
|
||||||
|
{"'", 10, 0, 0, 0, 0},
|
||||||
|
{"⌫", 20, 0, 0, 0, 0}},
|
||||||
|
{{"ABC", 50, 0, 0, 0, 0}, {" ", 64, 0, 0, 0, 0}, {"↵", 36, 0, 0, 0, 0}}}};
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
extern CannedMessageModule *cannedMessageModule;
|
extern CannedMessageModule *cannedMessageModule;
|
||||||
|
Loading…
Reference in New Issue
Block a user