2019-12-05 20:19:47 +00:00
|
|
|
#include "DisplayApp.h"
|
|
|
|
#include <FreeRTOS.h>
|
|
|
|
#include <task.h>
|
|
|
|
#include <libraries/log/nrf_log.h>
|
|
|
|
#include <boards.h>
|
2019-12-07 16:11:50 +00:00
|
|
|
#include <nrf_font.h>
|
2019-12-07 18:15:33 +00:00
|
|
|
#include <hal/nrf_rtc.h>
|
2019-12-07 16:11:50 +00:00
|
|
|
#include "Components/Gfx/Gfx.h"
|
2019-12-26 17:33:40 +00:00
|
|
|
#include <queue.h>
|
2019-12-28 13:34:50 +00:00
|
|
|
#include <Components/DateTime/DateTimeController.h>
|
2019-12-05 20:19:47 +00:00
|
|
|
|
|
|
|
using namespace Pinetime::Applications;
|
|
|
|
|
2019-12-28 13:34:50 +00:00
|
|
|
char const *DisplayApp::DaysString[] = {
|
|
|
|
"",
|
|
|
|
"MONDAY",
|
|
|
|
"TUESDAY",
|
|
|
|
"WEDNESDAY",
|
|
|
|
"THURSDAY",
|
|
|
|
"FRIDAY",
|
|
|
|
"SATURDAY",
|
|
|
|
"SUNDAY"
|
|
|
|
};
|
|
|
|
|
|
|
|
char const *DisplayApp::MonthsString[] = {
|
|
|
|
"",
|
|
|
|
"JAN",
|
|
|
|
"FEB",
|
|
|
|
"MAR",
|
|
|
|
"APR",
|
|
|
|
"MAY",
|
|
|
|
"JUN",
|
|
|
|
"JUL",
|
|
|
|
"AUG",
|
|
|
|
"SEP",
|
|
|
|
"OCT",
|
|
|
|
"NOV",
|
|
|
|
"DEC"
|
|
|
|
};
|
|
|
|
|
|
|
|
DisplayApp::DisplayApp(Controllers::Battery &batteryController,
|
|
|
|
Controllers::Ble &bleController,
|
|
|
|
Controllers::DateTime &dateTimeController) :
|
2019-12-27 16:05:49 +00:00
|
|
|
batteryController{batteryController},
|
2019-12-28 13:34:50 +00:00
|
|
|
bleController{bleController},
|
|
|
|
dateTimeController{dateTimeController} {
|
2019-12-27 15:05:35 +00:00
|
|
|
msgQueue = xQueueCreate(queueSize, itemSize);
|
|
|
|
}
|
|
|
|
|
2019-12-05 20:19:47 +00:00
|
|
|
void DisplayApp::Start() {
|
|
|
|
if (pdPASS != xTaskCreate(DisplayApp::Process, "DisplayApp", 256, this, 0, &taskHandle))
|
|
|
|
APP_ERROR_HANDLER(NRF_ERROR_NO_MEM);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DisplayApp::Process(void *instance) {
|
2019-12-28 13:34:50 +00:00
|
|
|
auto *app = static_cast<DisplayApp *>(instance);
|
2019-12-05 20:19:47 +00:00
|
|
|
NRF_LOG_INFO("DisplayApp task started!");
|
2019-12-07 16:11:50 +00:00
|
|
|
app->InitHw();
|
|
|
|
|
2019-12-05 20:19:47 +00:00
|
|
|
while (1) {
|
2019-12-07 18:15:33 +00:00
|
|
|
app->Refresh();
|
2019-12-05 20:19:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-07 16:11:50 +00:00
|
|
|
void DisplayApp::InitHw() {
|
2019-12-05 20:19:47 +00:00
|
|
|
nrf_gpio_cfg_output(14);
|
|
|
|
nrf_gpio_cfg_output(22);
|
|
|
|
nrf_gpio_cfg_output(23);
|
|
|
|
nrf_gpio_pin_clear(14);
|
2019-12-07 16:11:50 +00:00
|
|
|
nrf_gpio_pin_clear(22);
|
|
|
|
nrf_gpio_pin_clear(23);
|
|
|
|
|
|
|
|
Drivers::SpiMaster::Parameters params;
|
|
|
|
params.bitOrder = Drivers::SpiMaster::BitOrder::Msb_Lsb;
|
|
|
|
params.mode = Drivers::SpiMaster::Modes::Mode3;
|
|
|
|
params.Frequency = Drivers::SpiMaster::Frequencies::Freq8Mhz;
|
|
|
|
params.pinCSN = 25;
|
|
|
|
params.pinMISO = 4;
|
|
|
|
params.pinMOSI = 3;
|
|
|
|
params.pinSCK = 2;
|
|
|
|
spi.Init(Drivers::SpiMaster::SpiModule::SPI0, params);
|
2019-12-05 20:19:47 +00:00
|
|
|
|
2019-12-07 16:11:50 +00:00
|
|
|
lcd.reset(new Drivers::St7789(spi, 18));
|
|
|
|
gfx.reset(new Components::Gfx(*lcd.get()));
|
|
|
|
gfx->ClearScreen();
|
2019-12-05 20:19:47 +00:00
|
|
|
|
2019-12-07 16:11:50 +00:00
|
|
|
uint8_t x = 7;
|
2019-12-28 13:34:50 +00:00
|
|
|
gfx->DrawChar(&largeFont, '0', &x, 78, 0xffff);
|
2019-12-05 20:19:47 +00:00
|
|
|
|
2019-12-07 16:11:50 +00:00
|
|
|
x = 61;
|
2019-12-07 18:15:33 +00:00
|
|
|
gfx->DrawChar(&largeFont, '0', &x, 78, 0xffff);
|
|
|
|
|
|
|
|
x = 94;
|
|
|
|
gfx->DrawChar(&largeFont, ':', &x, 78, 0xffff);
|
|
|
|
|
|
|
|
x = 127;
|
|
|
|
gfx->DrawChar(&largeFont, '0', &x, 78, 0xffff);
|
|
|
|
|
|
|
|
x = 181;
|
|
|
|
gfx->DrawChar(&largeFont, '0', &x, 78, 0xffff);
|
2019-12-27 15:05:35 +00:00
|
|
|
|
2019-12-27 16:05:49 +00:00
|
|
|
gfx->DrawString(10, 0, 0x0000, "BLE", &smallFont, false);
|
2019-12-28 13:34:50 +00:00
|
|
|
gfx->DrawString(20, 180, 0xffff, "", &smallFont, false);
|
2019-12-07 18:15:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void DisplayApp::Refresh() {
|
2019-12-26 17:33:40 +00:00
|
|
|
TickType_t queueTimeout;
|
2019-12-28 13:34:50 +00:00
|
|
|
switch (state) {
|
2019-12-26 17:33:40 +00:00
|
|
|
case States::Idle:
|
|
|
|
IdleState();
|
|
|
|
queueTimeout = portMAX_DELAY;
|
|
|
|
break;
|
|
|
|
case States::Running:
|
|
|
|
RunningState();
|
|
|
|
queueTimeout = 1000;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
Messages msg;
|
2019-12-28 13:34:50 +00:00
|
|
|
if (xQueueReceive(msgQueue, &msg, queueTimeout)) {
|
|
|
|
switch (msg) {
|
2019-12-26 17:33:40 +00:00
|
|
|
case Messages::GoToSleep:
|
|
|
|
nrf_gpio_pin_set(23);
|
|
|
|
vTaskDelay(100);
|
|
|
|
nrf_gpio_pin_set(22);
|
|
|
|
vTaskDelay(100);
|
|
|
|
nrf_gpio_pin_set(14);
|
2020-01-02 13:48:33 +00:00
|
|
|
lcd->DisplayOff();
|
2019-12-26 17:33:40 +00:00
|
|
|
state = States::Idle;
|
|
|
|
break;
|
|
|
|
case Messages::GoToRunning:
|
2020-01-02 13:48:33 +00:00
|
|
|
lcd->DisplayOn();
|
2019-12-26 17:33:40 +00:00
|
|
|
nrf_gpio_pin_clear(23);
|
|
|
|
nrf_gpio_pin_clear(22);
|
|
|
|
nrf_gpio_pin_clear(14);
|
|
|
|
state = States::Running;
|
|
|
|
break;
|
2019-12-28 13:34:50 +00:00
|
|
|
case Messages::UpdateDateTime:
|
|
|
|
deltaSeconds = nrf_rtc_counter_get(portNRF_RTC_REG) / 1000;
|
|
|
|
this->seconds = dateTimeController.Seconds();
|
|
|
|
this->minutes = dateTimeController.Minutes();
|
|
|
|
this->hours = dateTimeController.Hours();
|
|
|
|
dateUpdated = true;
|
|
|
|
break;
|
|
|
|
case Messages::UpdateBleConnection:
|
|
|
|
bleConnectionUpdated = true;
|
|
|
|
break;
|
|
|
|
case Messages::UpdateBatteryLevel:
|
|
|
|
batteryLevelUpdated = true;
|
|
|
|
break;
|
2019-12-26 17:33:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DisplayApp::RunningState() {
|
2019-12-21 21:31:06 +00:00
|
|
|
uint32_t systick_counter = nrf_rtc_counter_get(portNRF_RTC_REG);
|
|
|
|
|
2019-12-28 13:34:50 +00:00
|
|
|
if (batteryLevelUpdated) {
|
|
|
|
char batteryChar[11];
|
|
|
|
uint16_t newBatteryValue = batteryController.PercentRemaining();
|
|
|
|
newBatteryValue = (newBatteryValue > 100) ? 100 : newBatteryValue;
|
|
|
|
newBatteryValue = (newBatteryValue < 0) ? 0 : newBatteryValue;
|
|
|
|
|
|
|
|
batteryLevelUpdated = false;
|
|
|
|
sprintf(batteryChar, "BAT: %d%%", newBatteryValue);
|
|
|
|
gfx->DrawString((240 - 108), 0, 0xffff, batteryChar, &smallFont, false);
|
2019-12-27 15:05:35 +00:00
|
|
|
}
|
2019-12-27 14:12:09 +00:00
|
|
|
|
2019-12-28 13:34:50 +00:00
|
|
|
if (bleConnectionUpdated) {
|
|
|
|
bleConnectionUpdated = false;
|
|
|
|
uint16_t color = (bleController.IsConnected()) ? 0xffff : 0x0000;
|
2019-12-27 16:05:49 +00:00
|
|
|
gfx->DrawString(10, 0, color, "BLE", &smallFont, false);
|
|
|
|
}
|
|
|
|
|
2019-12-21 21:31:06 +00:00
|
|
|
auto raw = systick_counter / 1000;
|
|
|
|
auto currentDeltaSeconds = raw - deltaSeconds;
|
|
|
|
|
|
|
|
auto deltaMinutes = (currentDeltaSeconds / 60);
|
|
|
|
auto currentMinutes = minutes + deltaMinutes;
|
|
|
|
|
|
|
|
auto deltaHours = currentMinutes / 60;
|
|
|
|
currentMinutes -= (deltaHours * 60);
|
|
|
|
auto currentHours = hours + deltaHours;
|
|
|
|
|
2019-12-07 18:15:33 +00:00
|
|
|
char minutesChar[3];
|
2019-12-21 21:31:06 +00:00
|
|
|
sprintf(minutesChar, "%02d", currentMinutes);
|
2019-12-07 18:15:33 +00:00
|
|
|
|
2019-12-21 16:58:00 +00:00
|
|
|
char hoursChar[3];
|
2019-12-21 21:31:06 +00:00
|
|
|
sprintf(hoursChar, "%02d", currentHours);
|
2019-12-21 16:58:00 +00:00
|
|
|
|
2019-12-07 18:15:33 +00:00
|
|
|
uint8_t x = 7;
|
2019-12-28 13:34:50 +00:00
|
|
|
if (hoursChar[0] != currentChar[0]) {
|
2019-12-21 16:58:00 +00:00
|
|
|
gfx->DrawChar(&largeFont, hoursChar[0], &x, 78, 0xffff);
|
|
|
|
currentChar[0] = hoursChar[0];
|
2019-12-07 18:15:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
x = 61;
|
2019-12-28 13:34:50 +00:00
|
|
|
if (hoursChar[1] != currentChar[1]) {
|
2019-12-21 16:58:00 +00:00
|
|
|
gfx->DrawChar(&largeFont, hoursChar[1], &x, 78, 0xffff);
|
|
|
|
currentChar[1] = hoursChar[1];
|
2019-12-07 18:15:33 +00:00
|
|
|
}
|
2019-12-05 20:19:47 +00:00
|
|
|
|
2019-12-07 16:11:50 +00:00
|
|
|
x = 127;
|
2019-12-28 13:34:50 +00:00
|
|
|
if (minutesChar[0] != currentChar[2]) {
|
2019-12-21 16:58:00 +00:00
|
|
|
gfx->DrawChar(&largeFont, minutesChar[0], &x, 78, 0xffff);
|
|
|
|
currentChar[2] = minutesChar[0];
|
2019-12-07 18:15:33 +00:00
|
|
|
}
|
2019-12-05 20:19:47 +00:00
|
|
|
|
2019-12-07 16:11:50 +00:00
|
|
|
x = 181;
|
2019-12-28 13:34:50 +00:00
|
|
|
if (minutesChar[1] != currentChar[3]) {
|
2019-12-21 16:58:00 +00:00
|
|
|
gfx->DrawChar(&largeFont, minutesChar[1], &x, 78, 0xffff);
|
|
|
|
currentChar[3] = minutesChar[1];
|
2019-12-07 18:15:33 +00:00
|
|
|
}
|
2019-12-28 13:34:50 +00:00
|
|
|
|
|
|
|
if (dateUpdated) {
|
|
|
|
auto year = dateTimeController.Year();
|
|
|
|
auto month = dateTimeController.Month();
|
|
|
|
auto day = dateTimeController.Day();
|
|
|
|
auto dayOfWeek = dateTimeController.DayOfWeek();
|
|
|
|
|
|
|
|
char dateStr[22];
|
|
|
|
sprintf(dateStr, "%s %d %s %d", DayOfWeekToString(dayOfWeek), day, MonthToString(month), year);
|
|
|
|
gfx->DrawString(10, 180, 0xffff, dateStr, &smallFont, false);
|
|
|
|
dateUpdated = false;
|
|
|
|
}
|
2019-12-05 20:19:47 +00:00
|
|
|
}
|
2019-12-21 16:58:00 +00:00
|
|
|
|
2019-12-28 13:34:50 +00:00
|
|
|
const char *DisplayApp::MonthToString(Pinetime::Controllers::DateTime::Months month) {
|
|
|
|
return DisplayApp::MonthsString[static_cast<uint8_t>(month)];
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *DisplayApp::DayOfWeekToString(Pinetime::Controllers::DateTime::Days dayOfWeek) {
|
|
|
|
return DisplayApp::DaysString[static_cast<uint8_t>(dayOfWeek)];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-26 17:33:40 +00:00
|
|
|
void DisplayApp::IdleState() {
|
|
|
|
|
2019-12-21 16:58:00 +00:00
|
|
|
}
|
|
|
|
|
2019-12-26 17:33:40 +00:00
|
|
|
void DisplayApp::PushMessage(DisplayApp::Messages msg) {
|
|
|
|
BaseType_t xHigherPriorityTaskWoken;
|
|
|
|
xHigherPriorityTaskWoken = pdFALSE;
|
2019-12-28 13:34:50 +00:00
|
|
|
xQueueSendFromISR(msgQueue, &msg, &xHigherPriorityTaskWoken);
|
|
|
|
if (xHigherPriorityTaskWoken) {
|
2019-12-26 17:33:40 +00:00
|
|
|
/* Actual macro used here is port specific. */
|
|
|
|
// TODO : should I do something here?
|
|
|
|
}
|
2019-12-21 21:31:06 +00:00
|
|
|
}
|