tbeam-helium-mapper/main/screen.cpp

285 wiersze
8.3 KiB
C++

/*
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 "screen.h"
#include <Wire.h>
#include "OLEDDisplay.h"
#include "SH1106Wire.h"
#include "SSD1306Wire.h"
#include "configuration.h"
#include "credentials.h"
#include "fonts.h"
#include "gps.h"
#include "images.h"
#define SCREEN_HEADER_HEIGHT 24
OLEDDisplay *display;
uint8_t _screen_line = SCREEN_HEADER_HEIGHT - 1;
enum display_types { DISPLAY_UNKNOWN, DISPLAY_SSD1306, DISPLAY_SH1106 };
enum display_types display_type = DISPLAY_UNKNOWN;
void screen_show_logo() {
if (!display)
return;
uint8_t x = (display->getWidth() - TTN_IMAGE_WIDTH) / 2;
uint8_t y = SCREEN_HEADER_HEIGHT + (display->getHeight() - SCREEN_HEADER_HEIGHT - TTN_IMAGE_HEIGHT) / 2 + 1;
display->drawXbm(x, y, TTN_IMAGE_WIDTH, TTN_IMAGE_HEIGHT, TTN_IMAGE);
}
void screen_off() {
if (!display)
return;
display->displayOff();
}
void screen_on() {
if (!display)
return;
display->displayOn();
}
void screen_clear() {
if (!display)
return;
display->clear();
}
void screen_print(const char *text, uint8_t x, uint8_t y, uint8_t alignment) {
DEBUG_MSG(text);
if (!display)
return;
display->setTextAlignment((OLEDDISPLAY_TEXT_ALIGNMENT)alignment);
display->drawString(x, y, text);
}
void screen_print(const char *text, uint8_t x, uint8_t y) {
screen_print(text, x, y, TEXT_ALIGN_LEFT);
}
void screen_print(const char *text) {
Serial.printf("Screen: %s\n", text);
if (!display)
return;
display->print(text);
if (_screen_line + 8 > display->getHeight()) {
// scroll
}
_screen_line += 8;
// screen_loop();
}
void screen_update() {
if (display)
display->display();
}
/*
* The SSD1306 and SH1106 controllers are almost the same, but different.
* Most importantly here, the SH1106 allows reading from the frame buffer, while the SSD1306 does not.
* We exploit this by writing two bytes and reading them back. A mismatch probably means SSD1306.
* Probably.
*/
enum display_types display_get_type(uint8_t id) {
uint8_t err;
uint8_t b1, b2;
Wire.begin(I2C_SDA, I2C_SCL);
Wire.setClock(7000000);
Wire.beginTransmission(id);
uint8_t a[] = {0, 0, 0x10, 0xB0};
Wire.write(a, sizeof(a));
if ((err = Wire.endTransmission(false)) != 0) {
Serial.printf("err=%d EndTransmission=%d(%s)", err, Wire.lastError(), Wire.getErrorText(Wire.lastError()));
return DISPLAY_UNKNOWN;
}
Wire.beginTransmission(id);
uint8_t b[] = {0x40, 'M', 'P'};
Wire.write(b, sizeof(b));
if ((err = Wire.endTransmission(false)) != 0) {
Serial.printf("err=%d EndTransmission=%d(%s)", err, Wire.lastError(), Wire.getErrorText(Wire.lastError()));
return DISPLAY_UNKNOWN;
}
Wire.beginTransmission(id);
uint8_t c[] = {0, 0, 0x10};
Wire.write(c, sizeof(c));
if ((err = Wire.endTransmission(false)) != 0) {
Serial.printf("err=%d EndTransmission=%d(%s)", err, Wire.lastError(), Wire.getErrorText(Wire.lastError()));
return DISPLAY_UNKNOWN;
}
Wire.beginTransmission(id);
Wire.write(0x40);
if ((err = Wire.endTransmission(false)) != 0) {
Serial.printf("err=%d EndTransmission=%d(%s)", err, Wire.lastError(), Wire.getErrorText(Wire.lastError()));
return DISPLAY_UNKNOWN;
}
err = Wire.requestFrom((int)id, (int)3, (int)1);
if (err != 3) {
return DISPLAY_UNKNOWN;
}
Wire.read(); // Discard
b1 = Wire.read();
b2 = Wire.read();
Wire.endTransmission();
if (b1 == 'M' && b2 == 'P')
return DISPLAY_SH1106;
else
return DISPLAY_SSD1306;
}
void screen_setup(uint8_t addr) {
/* Attempt to determine which kind of display we're dealing with */
if (display_type == DISPLAY_UNKNOWN)
display_type = display_get_type(addr);
// Display instance
if (display_type == DISPLAY_SSD1306)
display = new SSD1306Wire(addr, I2C_SDA, I2C_SCL);
else if (display_type == DISPLAY_SH1106)
display = new SH1106Wire(addr, I2C_SDA, I2C_SCL);
else
return;
display->init();
display->flipScreenVertically();
display->setFont(Custom_ArialMT_Plain_10);
// Scroll buffer
display->setLogBuffer(4, 30);
}
void screen_end() {
if (display) {
screen_off();
display->end();
delete display;
}
}
#include <axp20x.h>
extern AXP20X_Class axp; // TODO: This is evil
void screen_header(unsigned int tx_interval_s, float min_dist_moved, char *cached_sf_name, boolean in_deadzone,
boolean stay_on, boolean never_rest) {
if (!display)
return;
char buffer[40];
uint32_t sats = tGPS.satellites.value();
boolean no_gps = (sats < 3);
uint16_t devid_hint = ((DEVEUI[7] << 4) | (DEVEUI[6] & 0xF0) >> 4);
display->clear();
// Cycle display every 3 seconds
if (millis() % 6000 < 3000) {
// Voltage and Current
snprintf(buffer, sizeof(buffer), "%.2fV %.0fmA", axp.getBattVoltage() / 1000,
axp.getBattChargeCurrent() - axp.getBattDischargeCurrent());
// display->setTextAlignment(TEXT_ALIGN_CENTER);
// display->drawString(display->getWidth() / 2, 2, buffer);
display->setTextAlignment(TEXT_ALIGN_LEFT);
display->drawString(0, 2, buffer);
} else {
// ID & Time
if (no_gps) {
snprintf(buffer, sizeof(buffer), "#%03X", devid_hint);
display->setTextAlignment(TEXT_ALIGN_LEFT);
display->drawString(0, 2, buffer);
display->setTextAlignment(TEXT_ALIGN_CENTER);
display->drawString(display->getWidth() / 2, 2, "*** NO GPS ***");
snprintf(buffer, sizeof(buffer), "(%d)", sats);
display->setTextAlignment(TEXT_ALIGN_RIGHT);
display->drawString(display->getWidth(), 2, buffer);
} else {
snprintf(buffer, sizeof(buffer), "#%03X %02d:%02d:%02d", devid_hint, tGPS.time.hour(), tGPS.time.minute(),
tGPS.time.second());
display->setTextAlignment(TEXT_ALIGN_LEFT);
display->drawString(0, 2, buffer);
}
}
// HDOP & Satellite count
if (!no_gps) {
snprintf(buffer, sizeof(buffer), "%2.1f %d", tGPS.hdop.hdop(), sats);
display->setTextAlignment(TEXT_ALIGN_RIGHT);
display->drawString(display->getWidth() - SATELLITE_IMAGE_WIDTH - 4, 2, buffer);
display->drawXbm(display->getWidth() - SATELLITE_IMAGE_WIDTH, 0, SATELLITE_IMAGE_WIDTH, SATELLITE_IMAGE_HEIGHT,
SATELLITE_IMAGE);
}
// Second status row:
snprintf(buffer, sizeof(buffer), "%us %.0fm %c%c%c", tx_interval_s, min_dist_moved, in_deadzone ? 'D' : ' ',
stay_on ? 'S' : ' ', never_rest ? 'N' : ' ');
display->setTextAlignment(TEXT_ALIGN_LEFT);
display->drawString(0, 12, buffer);
display->setTextAlignment(TEXT_ALIGN_RIGHT);
display->drawString(display->getWidth(), 12, cached_sf_name);
display->drawHorizontalLine(0, SCREEN_HEADER_HEIGHT, display->getWidth());
}
#define MARGIN 15
void screen_body(boolean in_menu, const char *menu_prev, const char *menu_cur, const char *menu_next,
boolean highlighted) {
if (!display)
return;
if (in_menu) {
char buffer[40];
display->setTextAlignment(TEXT_ALIGN_CENTER);
display->drawString(display->getWidth() / 2, SCREEN_HEADER_HEIGHT + 5, menu_prev);
display->drawString(display->getWidth() / 2, SCREEN_HEADER_HEIGHT + 28, menu_next);
if (highlighted)
display->clear();
display->drawHorizontalLine(MARGIN, SCREEN_HEADER_HEIGHT + 16, display->getWidth() - MARGIN * 2);
snprintf(buffer, sizeof(buffer), highlighted ? ">>> %s <<<" : "%s", menu_cur);
display->drawString(display->getWidth() / 2, SCREEN_HEADER_HEIGHT + 16, buffer);
display->drawHorizontalLine(MARGIN, SCREEN_HEADER_HEIGHT + 28, display->getWidth() - MARGIN * 2);
display->drawVerticalLine(MARGIN, SCREEN_HEADER_HEIGHT + 16, 28 - 16);
display->drawVerticalLine(display->getWidth() - MARGIN, SCREEN_HEADER_HEIGHT + 16, 28 - 16);
} else
display->drawLogBuffer(0, SCREEN_HEADER_HEIGHT);
display->display();
}