2020-02-07 21:51:17 +00:00
|
|
|
/*
|
|
|
|
|
|
|
|
SSD1306 - Screen module
|
|
|
|
|
|
|
|
Copyright (C) 2018 by Xose Pérez <xose dot perez at gmail dot com>
|
|
|
|
|
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <Wire.h>
|
|
|
|
#include "SSD1306Wire.h"
|
|
|
|
#include "OLEDDisplay.h"
|
|
|
|
#include "images.h"
|
|
|
|
#include "fonts.h"
|
|
|
|
#include "GPS.h"
|
|
|
|
#include "OLEDDisplayUi.h"
|
|
|
|
#include "screen.h"
|
|
|
|
|
2020-02-07 22:52:45 +00:00
|
|
|
#define FONT_HEIGHT 14 // actually 13 for "ariel 10" but want a little extra space
|
|
|
|
|
2020-02-07 23:37:25 +00:00
|
|
|
#define SCREEN_WIDTH 128
|
|
|
|
#define SCREEN_HEIGHT 64
|
2020-02-07 21:51:17 +00:00
|
|
|
|
|
|
|
#ifdef I2C_SDA
|
2020-02-07 23:37:25 +00:00
|
|
|
SSD1306Wire dispdev(SSD1306_ADDRESS, I2C_SDA, I2C_SCL);
|
2020-02-07 21:51:17 +00:00
|
|
|
#else
|
2020-02-07 23:37:25 +00:00
|
|
|
SSD1306Wire dispdev(SSD1306_ADDRESS, 0, 0); // fake values to keep build happy, we won't ever init
|
|
|
|
#endif
|
2020-02-07 21:51:17 +00:00
|
|
|
|
|
|
|
bool disp; // true if we are using display
|
|
|
|
|
|
|
|
OLEDDisplayUi ui(&dispdev);
|
|
|
|
|
|
|
|
void msOverlay(OLEDDisplay *display, OLEDDisplayUiState *state)
|
|
|
|
{
|
|
|
|
display->setTextAlignment(TEXT_ALIGN_RIGHT);
|
|
|
|
display->setFont(ArialMT_Plain_10);
|
|
|
|
display->drawString(128, 0, String(millis()));
|
|
|
|
}
|
|
|
|
|
2020-02-07 22:52:45 +00:00
|
|
|
void drawBootScreen(OLEDDisplay *display, OLEDDisplayUiState *state, int16_t x, int16_t y)
|
2020-02-07 21:51:17 +00:00
|
|
|
{
|
|
|
|
// draw an xbm image.
|
|
|
|
// Please note that everything that should be transitioned
|
|
|
|
// needs to be drawn relative to x and y
|
|
|
|
|
2020-02-07 23:37:25 +00:00
|
|
|
display->drawXbm(x + 32, y, icon_width, icon_height, (const uint8_t *)icon_bits);
|
2020-02-07 22:52:45 +00:00
|
|
|
|
|
|
|
display->setFont(ArialMT_Plain_10);
|
|
|
|
display->setTextAlignment(TEXT_ALIGN_CENTER);
|
|
|
|
display->drawString(64 + x, SCREEN_HEIGHT - FONT_HEIGHT, APP_NAME " " APP_VERSION);
|
|
|
|
|
|
|
|
ui.disableIndicator();
|
2020-02-07 21:51:17 +00:00
|
|
|
}
|
|
|
|
|
2020-02-08 04:59:21 +00:00
|
|
|
static char btPIN[16] = "888888";
|
|
|
|
|
|
|
|
void drawFrameBluetooth(OLEDDisplay *display, OLEDDisplayUiState *state, int16_t x, int16_t y)
|
|
|
|
{
|
|
|
|
// Demonstrates the 3 included default sizes. The fonts come from SSD1306Fonts.h file
|
|
|
|
// Besides the default fonts there will be a program to convert TrueType fonts into this format
|
|
|
|
display->setTextAlignment(TEXT_ALIGN_CENTER);
|
|
|
|
display->setFont(ArialMT_Plain_16);
|
|
|
|
display->drawString(64 + x, 2 + y, "Bluetooth");
|
|
|
|
|
|
|
|
display->setFont(ArialMT_Plain_10);
|
|
|
|
display->drawString(64 + x, SCREEN_HEIGHT - FONT_HEIGHT + y, "Enter this code");
|
|
|
|
|
|
|
|
display->setTextAlignment(TEXT_ALIGN_CENTER);
|
|
|
|
display->setFont(ArialMT_Plain_24);
|
|
|
|
display->drawString(64 + x, 22 + y, btPIN);
|
|
|
|
|
|
|
|
ui.disableIndicator();
|
|
|
|
}
|
|
|
|
|
2020-02-07 21:51:17 +00:00
|
|
|
void drawFrame2(OLEDDisplay *display, OLEDDisplayUiState *state, int16_t x, int16_t y)
|
|
|
|
{
|
|
|
|
// Demonstrates the 3 included default sizes. The fonts come from SSD1306Fonts.h file
|
|
|
|
// Besides the default fonts there will be a program to convert TrueType fonts into this format
|
|
|
|
display->setTextAlignment(TEXT_ALIGN_LEFT);
|
|
|
|
display->setFont(ArialMT_Plain_10);
|
|
|
|
display->drawString(0 + x, 10 + y, "Arial 10");
|
|
|
|
|
|
|
|
display->setFont(ArialMT_Plain_16);
|
|
|
|
display->drawString(0 + x, 20 + y, "Arial 16");
|
|
|
|
|
|
|
|
display->setFont(ArialMT_Plain_24);
|
|
|
|
display->drawString(0 + x, 34 + y, "Arial 24");
|
|
|
|
}
|
|
|
|
|
|
|
|
void drawFrame3(OLEDDisplay *display, OLEDDisplayUiState *state, int16_t x, int16_t y)
|
|
|
|
{
|
|
|
|
// Text alignment demo
|
|
|
|
display->setFont(ArialMT_Plain_10);
|
|
|
|
|
|
|
|
// The coordinates define the left starting point of the text
|
|
|
|
display->setTextAlignment(TEXT_ALIGN_LEFT);
|
|
|
|
display->drawString(0 + x, 11 + y, "Left aligned (0,10)");
|
|
|
|
|
|
|
|
// The coordinates define the center of the text
|
|
|
|
display->setTextAlignment(TEXT_ALIGN_CENTER);
|
|
|
|
display->drawString(64 + x, 22 + y, "Center aligned (64,22)");
|
|
|
|
|
|
|
|
// The coordinates define the right end of the text
|
|
|
|
display->setTextAlignment(TEXT_ALIGN_RIGHT);
|
|
|
|
display->drawString(128 + x, 33 + y, "Right aligned (128,33)");
|
|
|
|
}
|
|
|
|
|
2020-02-07 22:52:45 +00:00
|
|
|
/// Draw the last text message we received
|
2020-02-07 23:37:25 +00:00
|
|
|
void drawTextMessageFrame(OLEDDisplay *display, OLEDDisplayUiState *state, int16_t x, int16_t y)
|
2020-02-07 21:51:17 +00:00
|
|
|
{
|
|
|
|
// Demo for drawStringMaxWidth:
|
|
|
|
// with the third parameter you can define the width after which words will be wrapped.
|
|
|
|
// Currently only spaces and "-" are allowed for wrapping
|
|
|
|
display->setTextAlignment(TEXT_ALIGN_LEFT);
|
2020-02-07 23:37:25 +00:00
|
|
|
display->setFont(ArialMT_Plain_16);
|
|
|
|
String sender = "KH:";
|
|
|
|
display->drawString(0 + x, 0 + y, sender);
|
2020-02-07 21:51:17 +00:00
|
|
|
display->setFont(ArialMT_Plain_10);
|
2020-02-08 01:26:42 +00:00
|
|
|
display->drawStringMaxWidth(4 + x, 10 + y, 128, " Lorem ipsum\n dolor sit amet, consetetur sadipscing elitr, sed diam");
|
2020-02-07 23:37:25 +00:00
|
|
|
|
2020-02-08 01:26:42 +00:00
|
|
|
// ui.disableIndicator();
|
2020-02-07 21:51:17 +00:00
|
|
|
}
|
|
|
|
|
2020-02-08 01:26:42 +00:00
|
|
|
/// Draw a series of fields in a column, wrapping to multiple colums if needed
|
|
|
|
void drawColumns(OLEDDisplay *display, int16_t x, int16_t y, const char **fields)
|
2020-02-07 21:51:17 +00:00
|
|
|
{
|
2020-02-08 01:26:42 +00:00
|
|
|
// The coordinates define the left starting point of the text
|
|
|
|
display->setTextAlignment(TEXT_ALIGN_LEFT);
|
|
|
|
|
|
|
|
const char **f = fields;
|
|
|
|
int xo = x, yo = y;
|
|
|
|
while (*f)
|
|
|
|
{
|
|
|
|
display->drawString(xo, yo, *f);
|
|
|
|
yo += FONT_HEIGHT;
|
|
|
|
if (yo > SCREEN_HEIGHT - FONT_HEIGHT)
|
|
|
|
{
|
|
|
|
xo += SCREEN_WIDTH / 2;
|
|
|
|
yo = 0;
|
|
|
|
}
|
|
|
|
f++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Draw a series of fields in a row, wrapping to multiple rows if needed
|
|
|
|
/// @return the max y we ended up printing to
|
|
|
|
uint32_t drawRows(OLEDDisplay *display, int16_t x, int16_t y, const char **fields)
|
|
|
|
{
|
|
|
|
// The coordinates define the left starting point of the text
|
|
|
|
display->setTextAlignment(TEXT_ALIGN_LEFT);
|
|
|
|
|
|
|
|
const char **f = fields;
|
|
|
|
int xo = x, yo = y;
|
|
|
|
while (*f)
|
|
|
|
{
|
|
|
|
display->drawString(xo, yo, *f);
|
|
|
|
xo += SCREEN_WIDTH / 2; // hardwired for two columns per row....
|
|
|
|
if (xo >= SCREEN_WIDTH)
|
|
|
|
{
|
|
|
|
yo += FONT_HEIGHT;
|
|
|
|
xo = 0;
|
|
|
|
}
|
|
|
|
f++;
|
|
|
|
}
|
|
|
|
|
|
|
|
yo += FONT_HEIGHT; // include the last line in our total
|
|
|
|
|
|
|
|
return yo;
|
|
|
|
}
|
|
|
|
|
|
|
|
void drawNodeInfo(OLEDDisplay *display, OLEDDisplayUiState *state, int16_t x, int16_t y)
|
|
|
|
{
|
|
|
|
display->setFont(ArialMT_Plain_10);
|
|
|
|
|
|
|
|
// The coordinates define the left starting point of the text
|
|
|
|
display->setTextAlignment(TEXT_ALIGN_LEFT);
|
|
|
|
|
|
|
|
const char *fields[] = {
|
|
|
|
"Kevin Hester (KH)",
|
|
|
|
"4.2 mi",
|
|
|
|
"Signal: good",
|
|
|
|
"24 minutes ago",
|
|
|
|
NULL};
|
|
|
|
drawColumns(display, x, y, fields);
|
|
|
|
|
|
|
|
display->drawXbm(x + (SCREEN_WIDTH - compass_width), y + (SCREEN_HEIGHT - compass_height) / 2, compass_width, compass_height, (const uint8_t *)compass_bits);
|
|
|
|
}
|
|
|
|
|
|
|
|
void drawDebugInfo(OLEDDisplay *display, OLEDDisplayUiState *state, int16_t x, int16_t y)
|
|
|
|
{
|
|
|
|
display->setFont(ArialMT_Plain_10);
|
|
|
|
|
|
|
|
// The coordinates define the left starting point of the text
|
|
|
|
display->setTextAlignment(TEXT_ALIGN_LEFT);
|
|
|
|
|
|
|
|
const char *fields[] = {
|
|
|
|
"Batt 89%",
|
|
|
|
"GPS 75%",
|
|
|
|
"Users 4/12",
|
|
|
|
NULL};
|
|
|
|
uint32_t yo = drawRows(display, x, y, fields);
|
|
|
|
|
|
|
|
display->drawLogBuffer(x, yo);
|
2020-02-07 21:51:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// This array keeps function pointers to all frames
|
|
|
|
// frames are the single views that slide in
|
2020-02-08 01:26:42 +00:00
|
|
|
FrameCallback frames[] = {drawBootScreen, drawTextMessageFrame, drawNodeInfo, drawDebugInfo};
|
2020-02-07 23:37:25 +00:00
|
|
|
FrameCallback *nonBootFrames = frames + 1;
|
2020-02-07 21:51:17 +00:00
|
|
|
|
|
|
|
// Overlays are statically drawn on top of a frame eg. a clock
|
2020-02-08 01:26:42 +00:00
|
|
|
OverlayCallback overlays[] = {/* msOverlay */};
|
2020-02-07 21:51:17 +00:00
|
|
|
|
2020-02-07 23:37:25 +00:00
|
|
|
// how many frames are there?
|
|
|
|
const int frameCount = sizeof(frames) / sizeof(frames[0]);
|
|
|
|
const int overlaysCount = sizeof(overlays) / sizeof(overlays[0]);
|
|
|
|
|
2020-02-08 04:59:21 +00:00
|
|
|
#if 0
|
2020-02-07 21:51:17 +00:00
|
|
|
void _screen_header()
|
|
|
|
{
|
|
|
|
if (!disp)
|
|
|
|
return;
|
|
|
|
|
2020-02-08 04:59:21 +00:00
|
|
|
|
2020-02-07 21:51:17 +00:00
|
|
|
// Message count
|
|
|
|
//snprintf(buffer, sizeof(buffer), "#%03d", ttn_get_count() % 1000);
|
|
|
|
//display->setTextAlignment(TEXT_ALIGN_LEFT);
|
|
|
|
//display->drawString(0, 2, buffer);
|
|
|
|
|
|
|
|
// Datetime
|
|
|
|
display->setTextAlignment(TEXT_ALIGN_CENTER);
|
|
|
|
display->drawString(display->getWidth()/2, 2, gps.getTimeStr());
|
|
|
|
|
|
|
|
// Satellite count
|
|
|
|
display->setTextAlignment(TEXT_ALIGN_RIGHT);
|
|
|
|
char buffer[10];
|
|
|
|
display->drawString(display->getWidth() - SATELLITE_IMAGE_WIDTH - 4, 2, itoa(gps.satellites.value(), buffer, 10));
|
|
|
|
display->drawXbm(display->getWidth() - SATELLITE_IMAGE_WIDTH, 0, SATELLITE_IMAGE_WIDTH, SATELLITE_IMAGE_HEIGHT, SATELLITE_IMAGE);
|
|
|
|
}
|
2020-02-08 04:59:21 +00:00
|
|
|
#endif
|
2020-02-07 21:51:17 +00:00
|
|
|
|
|
|
|
void screen_off()
|
|
|
|
{
|
|
|
|
if (!disp)
|
|
|
|
return;
|
|
|
|
|
|
|
|
dispdev.displayOff();
|
|
|
|
}
|
|
|
|
|
|
|
|
void screen_on()
|
|
|
|
{
|
|
|
|
if (!disp)
|
|
|
|
return;
|
|
|
|
|
|
|
|
dispdev.displayOn();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void screen_print(const char *text, uint8_t x, uint8_t y, uint8_t alignment)
|
|
|
|
{
|
|
|
|
DEBUG_MSG(text);
|
|
|
|
|
|
|
|
if (!disp)
|
|
|
|
return;
|
|
|
|
|
|
|
|
dispdev.setTextAlignment((OLEDDISPLAY_TEXT_ALIGNMENT)alignment);
|
|
|
|
dispdev.drawString(x, y, text);
|
|
|
|
}
|
|
|
|
|
|
|
|
void screen_print(const char *text)
|
|
|
|
{
|
|
|
|
DEBUG_MSG("Screen: %s\n", text);
|
|
|
|
if (!disp)
|
|
|
|
return;
|
|
|
|
|
|
|
|
dispdev.print(text);
|
2020-02-08 01:26:42 +00:00
|
|
|
// ui.update();
|
2020-02-07 21:51:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void screen_setup()
|
|
|
|
{
|
|
|
|
#ifdef I2C_SDA
|
|
|
|
// Display instance
|
|
|
|
disp = true;
|
|
|
|
|
|
|
|
// The ESP is capable of rendering 60fps in 80Mhz mode
|
|
|
|
// but that won't give you much time for anything else
|
|
|
|
// run it in 160Mhz mode or just set it to 30 fps
|
|
|
|
ui.setTargetFPS(30);
|
|
|
|
|
|
|
|
// Customize the active and inactive symbol
|
|
|
|
//ui.setActiveSymbol(activeSymbol);
|
|
|
|
//ui.setInactiveSymbol(inactiveSymbol);
|
|
|
|
|
|
|
|
// You can change this to
|
|
|
|
// TOP, LEFT, BOTTOM, RIGHT
|
|
|
|
ui.setIndicatorPosition(BOTTOM);
|
|
|
|
|
|
|
|
// Defines where the first frame is located in the bar.
|
|
|
|
ui.setIndicatorDirection(LEFT_RIGHT);
|
|
|
|
|
|
|
|
// You can change the transition that is used
|
|
|
|
// SLIDE_LEFT, SLIDE_RIGHT, SLIDE_UP, SLIDE_DOWN
|
|
|
|
ui.setFrameAnimation(SLIDE_LEFT);
|
|
|
|
|
2020-02-08 01:26:42 +00:00
|
|
|
// Add frames - we subtract one from the framecount so there won't be a visual glitch when we take the boot screen out of the sequence.
|
|
|
|
ui.setFrames(frames, frameCount - 1);
|
2020-02-07 21:51:17 +00:00
|
|
|
|
|
|
|
// Add overlays
|
|
|
|
ui.setOverlays(overlays, overlaysCount);
|
|
|
|
|
|
|
|
// Initialising the UI will init the display too.
|
|
|
|
ui.init();
|
|
|
|
|
|
|
|
// Scroll buffer
|
2020-02-08 01:26:42 +00:00
|
|
|
dispdev.setLogBuffer(5, 32);
|
2020-02-07 21:51:17 +00:00
|
|
|
|
2020-02-07 22:52:45 +00:00
|
|
|
// dispdev.flipScreenVertically(); // looks better without this on lora32
|
2020-02-07 23:37:25 +00:00
|
|
|
// dispdev.setFont(Custom_ArialMT_Plain_10);
|
2020-02-07 21:51:17 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2020-02-08 04:59:21 +00:00
|
|
|
static bool showingBluetooth;
|
|
|
|
|
2020-02-08 15:38:08 +00:00
|
|
|
/// If set to true (possibly from an ISR), we should turn on the screen the next time our idle loop runs.
|
|
|
|
static bool wakeScreen;
|
|
|
|
|
2020-02-08 00:12:55 +00:00
|
|
|
uint32_t screen_loop()
|
2020-02-07 21:51:17 +00:00
|
|
|
{
|
|
|
|
if (!disp)
|
2020-02-08 00:12:55 +00:00
|
|
|
return 30 * 1000;
|
2020-02-07 21:51:17 +00:00
|
|
|
|
2020-02-08 15:55:12 +00:00
|
|
|
if (wakeScreen)
|
2020-02-07 21:51:17 +00:00
|
|
|
{
|
2020-02-08 15:55:12 +00:00
|
|
|
screen_on(); // make sure the screen is not asleep
|
|
|
|
wakeScreen = false;
|
2020-02-07 21:51:17 +00:00
|
|
|
}
|
2020-02-08 15:38:08 +00:00
|
|
|
|
2020-02-07 23:37:25 +00:00
|
|
|
static bool showingBootScreen = true;
|
|
|
|
|
2020-02-08 01:26:42 +00:00
|
|
|
ui.update();
|
2020-02-07 23:37:25 +00:00
|
|
|
|
|
|
|
// Once we finish showing the bootscreen, remove it from the loop
|
2020-02-08 04:59:21 +00:00
|
|
|
if (showingBootScreen && !showingBluetooth && ui.getUiState()->currentFrame == 1)
|
2020-02-07 23:37:25 +00:00
|
|
|
{
|
2020-02-08 01:26:42 +00:00
|
|
|
showingBootScreen = false;
|
2020-02-08 04:59:21 +00:00
|
|
|
screen_set_frames();
|
2020-02-07 23:37:25 +00:00
|
|
|
}
|
2020-02-08 00:12:55 +00:00
|
|
|
|
|
|
|
// If we are scrolling do 30fps, otherwise just 1 fps (to save CPU)
|
|
|
|
return (ui.getUiState()->frameState == IN_TRANSITION ? 10 : 500);
|
2020-02-07 21:51:17 +00:00
|
|
|
}
|
2020-02-08 01:48:12 +00:00
|
|
|
|
2020-02-08 04:59:21 +00:00
|
|
|
// Show the bluetooth PIN screen
|
2020-02-08 15:55:12 +00:00
|
|
|
void screen_start_bluetooth(uint32_t pin)
|
|
|
|
{
|
|
|
|
static FrameCallback btFrames[] = {drawFrameBluetooth};
|
2020-02-08 04:59:21 +00:00
|
|
|
|
|
|
|
snprintf(btPIN, sizeof(btPIN), "%06d", pin);
|
|
|
|
|
|
|
|
DEBUG_MSG("showing bluetooth screen\n");
|
|
|
|
showingBluetooth = true;
|
2020-02-08 15:55:12 +00:00
|
|
|
wakeScreen = true;
|
2020-02-08 04:59:21 +00:00
|
|
|
|
2020-02-08 15:55:12 +00:00
|
|
|
ui.disableAutoTransition(); // we now require presses
|
|
|
|
ui.setFrames(btFrames, 1); // Just show the bluetooth frame
|
2020-02-08 04:59:21 +00:00
|
|
|
// we rely on our main loop to show this screen (because we are invoked deep inside of bluetooth callbacks)
|
|
|
|
// ui.update(); // manually draw once, because I'm not sure if loop is getting called
|
|
|
|
}
|
|
|
|
|
|
|
|
// restore our regular frame list
|
2020-02-08 15:55:12 +00:00
|
|
|
void screen_set_frames()
|
|
|
|
{
|
2020-02-08 04:59:21 +00:00
|
|
|
DEBUG_MSG("showing standard frames\n");
|
2020-02-08 15:55:12 +00:00
|
|
|
ui.setFrames(nonBootFrames, frameCount - 1);
|
2020-02-08 04:59:21 +00:00
|
|
|
showingBluetooth = false;
|
|
|
|
}
|
2020-02-08 01:48:12 +00:00
|
|
|
|
|
|
|
/// handle press of the button
|
2020-02-08 15:55:12 +00:00
|
|
|
void screen_press()
|
|
|
|
{
|
2020-02-08 15:38:08 +00:00
|
|
|
// screen_start_bluetooth(123456);
|
2020-02-08 04:59:21 +00:00
|
|
|
|
2020-02-08 01:48:12 +00:00
|
|
|
// Once the user presses a button, stop auto scrolling between screens
|
2020-02-08 15:55:12 +00:00
|
|
|
ui.disableAutoTransition(); // we now require presses
|
2020-02-08 01:48:12 +00:00
|
|
|
ui.nextFrame();
|
|
|
|
}
|