2023-11-07 00:11:12 +00:00
|
|
|
#include "screen_handler.hpp"
|
|
|
|
|
2023-11-12 11:38:28 +00:00
|
|
|
// TaskHandle_t priceUpdateTaskHandle;
|
|
|
|
// TaskHandle_t blockUpdateTaskHandle;
|
|
|
|
// TaskHandle_t timeUpdateTaskHandle;
|
2023-11-07 20:26:15 +00:00
|
|
|
TaskHandle_t taskScreenRotateTaskHandle;
|
2023-11-12 11:38:28 +00:00
|
|
|
TaskHandle_t workerTaskHandle;
|
2023-11-07 20:26:15 +00:00
|
|
|
esp_timer_handle_t screenRotateTimer;
|
|
|
|
esp_timer_handle_t minuteTimer;
|
2023-11-07 00:11:12 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
std::array<std::string, NUM_SCREENS> taskEpdContent = {"", "", "", "",
|
|
|
|
"", "", ""};
|
2023-11-07 00:11:12 +00:00
|
|
|
std::string priceString;
|
2023-11-13 11:27:34 +00:00
|
|
|
|
2023-11-12 11:38:28 +00:00
|
|
|
#define WORK_QUEUE_SIZE 10
|
2023-11-12 23:33:48 +00:00
|
|
|
QueueHandle_t workQueue = NULL;
|
2023-11-07 20:26:15 +00:00
|
|
|
|
|
|
|
uint currentScreen;
|
2023-11-07 00:11:12 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void workerTask(void *pvParameters) {
|
|
|
|
WorkItem receivedItem;
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
// Wait for a work item to be available in the queue
|
|
|
|
if (xQueueReceive(workQueue, &receivedItem, portMAX_DELAY)) {
|
|
|
|
uint firstIndex = 0;
|
|
|
|
|
|
|
|
// Process the work item based on its type
|
|
|
|
switch (receivedItem.type) {
|
2023-11-30 21:56:50 +00:00
|
|
|
case TASK_PRICE_UPDATE: {
|
|
|
|
uint price = getPrice();
|
|
|
|
char priceSymbol = '$';
|
|
|
|
if (preferences.getBool("fetchEurPrice", false)) {
|
|
|
|
priceSymbol = '[';
|
|
|
|
}
|
|
|
|
if (getCurrentScreen() == SCREEN_BTC_TICKER) {
|
|
|
|
taskEpdContent = parsePriceData(price, priceSymbol);
|
|
|
|
} else if (getCurrentScreen() == SCREEN_MSCW_TIME) {
|
2024-03-10 11:35:20 +00:00
|
|
|
taskEpdContent = parseSatsPerCurrency(price, priceSymbol, preferences.getBool("useSatsSymbol", false));
|
2023-11-30 21:56:50 +00:00
|
|
|
} else {
|
|
|
|
taskEpdContent =
|
|
|
|
parseMarketCap(getBlockHeight(), price, priceSymbol,
|
|
|
|
preferences.getBool("mcapBigChar", true));
|
|
|
|
}
|
2023-11-08 11:18:59 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
setEpdContent(taskEpdContent);
|
2023-11-30 21:56:50 +00:00
|
|
|
break;
|
2023-11-30 21:38:01 +00:00
|
|
|
}
|
2024-03-10 19:24:55 +00:00
|
|
|
case TASK_FEE_UPDATE: {
|
|
|
|
if (getCurrentScreen() == SCREEN_BLOCK_FEE_RATE) {
|
|
|
|
taskEpdContent = parseBlockFees(static_cast<std::uint16_t>(getBlockMedianFee()));
|
|
|
|
setEpdContent(taskEpdContent);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2023-11-30 21:56:50 +00:00
|
|
|
case TASK_BLOCK_UPDATE: {
|
|
|
|
if (getCurrentScreen() != SCREEN_HALVING_COUNTDOWN) {
|
|
|
|
taskEpdContent = parseBlockHeight(getBlockHeight());
|
|
|
|
} else {
|
2024-03-10 11:35:20 +00:00
|
|
|
taskEpdContent = parseHalvingCountdown(getBlockHeight(), preferences.getBool("useBlkCountdown", false));
|
2023-11-30 21:38:01 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:56:50 +00:00
|
|
|
if (getCurrentScreen() == SCREEN_HALVING_COUNTDOWN ||
|
|
|
|
getCurrentScreen() == SCREEN_BLOCK_HEIGHT) {
|
|
|
|
setEpdContent(taskEpdContent);
|
2023-11-30 21:38:01 +00:00
|
|
|
}
|
2023-11-30 21:56:50 +00:00
|
|
|
break;
|
2023-11-08 11:18:59 +00:00
|
|
|
}
|
2023-11-30 21:56:50 +00:00
|
|
|
case TASK_TIME_UPDATE: {
|
|
|
|
if (getCurrentScreen() == SCREEN_TIME) {
|
|
|
|
time_t currentTime;
|
|
|
|
struct tm timeinfo;
|
|
|
|
time(¤tTime);
|
|
|
|
localtime_r(¤tTime, &timeinfo);
|
|
|
|
std::string timeString;
|
|
|
|
|
|
|
|
String minute = String(timeinfo.tm_min);
|
|
|
|
if (minute.length() < 2) {
|
|
|
|
minute = "0" + minute;
|
|
|
|
}
|
|
|
|
|
|
|
|
timeString =
|
|
|
|
std::to_string(timeinfo.tm_hour) + ":" + minute.c_str();
|
|
|
|
timeString.insert(timeString.begin(),
|
|
|
|
NUM_SCREENS - timeString.length(), ' ');
|
|
|
|
taskEpdContent[0] = std::to_string(timeinfo.tm_mday) + "/" +
|
|
|
|
std::to_string(timeinfo.tm_mon + 1);
|
|
|
|
|
|
|
|
for (uint i = 1; i < NUM_SCREENS; i++) {
|
|
|
|
taskEpdContent[i] = timeString[i];
|
|
|
|
}
|
|
|
|
setEpdContent(taskEpdContent);
|
|
|
|
}
|
2023-11-08 11:18:59 +00:00
|
|
|
|
2023-11-30 21:56:50 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
// Add more cases for additional task types
|
2023-11-30 21:38:01 +00:00
|
|
|
}
|
2023-11-07 20:26:15 +00:00
|
|
|
}
|
2023-11-30 21:38:01 +00:00
|
|
|
}
|
2023-11-07 20:26:15 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void taskScreenRotate(void *pvParameters) {
|
|
|
|
for (;;) {
|
|
|
|
ulTaskNotifyTake(pdTRUE, portMAX_DELAY);
|
|
|
|
|
|
|
|
int nextScreen = (currentScreen + 1) % SCREEN_COUNT;
|
|
|
|
String key = "screen" + String(nextScreen) + "Visible";
|
|
|
|
|
|
|
|
while (!preferences.getBool(key.c_str(), true)) {
|
|
|
|
nextScreen = (nextScreen + 1) % SCREEN_COUNT;
|
|
|
|
key = "screen" + String(nextScreen) + "Visible";
|
2023-11-07 20:26:15 +00:00
|
|
|
}
|
2023-11-30 21:38:01 +00:00
|
|
|
|
|
|
|
setCurrentScreen(nextScreen);
|
|
|
|
}
|
2023-11-07 20:26:15 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void IRAM_ATTR minuteTimerISR(void *arg) {
|
|
|
|
BaseType_t xHigherPriorityTaskWoken = pdFALSE;
|
|
|
|
// vTaskNotifyGiveFromISR(timeUpdateTaskHandle, &xHigherPriorityTaskWoken);
|
|
|
|
WorkItem timeUpdate = {TASK_TIME_UPDATE, 0};
|
|
|
|
xQueueSendFromISR(workQueue, &timeUpdate, &xHigherPriorityTaskWoken);
|
|
|
|
if (priceFetchTaskHandle != NULL) {
|
|
|
|
vTaskNotifyGiveFromISR(priceFetchTaskHandle, &xHigherPriorityTaskWoken);
|
|
|
|
}
|
|
|
|
if (xHigherPriorityTaskWoken == pdTRUE) {
|
|
|
|
portYIELD_FROM_ISR();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void IRAM_ATTR screenRotateTimerISR(void *arg) {
|
|
|
|
BaseType_t xHigherPriorityTaskWoken = pdFALSE;
|
|
|
|
vTaskNotifyGiveFromISR(taskScreenRotateTaskHandle, &xHigherPriorityTaskWoken);
|
|
|
|
if (xHigherPriorityTaskWoken == pdTRUE) {
|
|
|
|
portYIELD_FROM_ISR();
|
|
|
|
}
|
2023-11-07 20:26:15 +00:00
|
|
|
}
|
2023-11-07 00:11:12 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void setupTasks() {
|
|
|
|
workQueue = xQueueCreate(WORK_QUEUE_SIZE, sizeof(WorkItem));
|
2023-11-12 11:38:28 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
xTaskCreate(workerTask, "workerTask", 4096, NULL, tskIDLE_PRIORITY,
|
|
|
|
&workerTaskHandle);
|
2023-11-12 11:38:28 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
xTaskCreate(taskScreenRotate, "rotateScreen", 2048, NULL, tskIDLE_PRIORITY,
|
|
|
|
&taskScreenRotateTaskHandle);
|
2023-11-08 14:27:22 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
waitUntilNoneBusy();
|
|
|
|
setCurrentScreen(preferences.getUInt("currentScreen", 0));
|
2023-11-07 20:26:15 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void setupTimeUpdateTimer(void *pvParameters) {
|
|
|
|
const esp_timer_create_args_t minuteTimerConfig = {
|
|
|
|
.callback = &minuteTimerISR, .name = "minute_timer"};
|
2023-11-07 20:26:15 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
esp_timer_create(&minuteTimerConfig, &minuteTimer);
|
2023-11-07 20:26:15 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
time_t currentTime;
|
|
|
|
struct tm timeinfo;
|
|
|
|
time(¤tTime);
|
|
|
|
localtime_r(¤tTime, &timeinfo);
|
|
|
|
uint32_t secondsUntilNextMinute = 60 - timeinfo.tm_sec;
|
2023-11-07 20:26:15 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
if (secondsUntilNextMinute > 0)
|
|
|
|
vTaskDelay(pdMS_TO_TICKS((secondsUntilNextMinute * 1000)));
|
2023-11-07 20:26:15 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
esp_timer_start_periodic(minuteTimer, usPerMinute);
|
2023-11-13 16:51:10 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
WorkItem timeUpdate = {TASK_TIME_UPDATE, 0};
|
|
|
|
xQueueSend(workQueue, &timeUpdate, portMAX_DELAY);
|
|
|
|
// xTaskNotifyGive(timeUpdateTaskHandle);
|
2023-11-07 20:26:15 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
vTaskDelete(NULL);
|
2023-11-07 20:26:15 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void setupScreenRotateTimer(void *pvParameters) {
|
|
|
|
const esp_timer_create_args_t screenRotateTimerConfig = {
|
|
|
|
.callback = &screenRotateTimerISR, .name = "screen_rotate_timer"};
|
2023-11-07 20:26:15 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
esp_timer_create(&screenRotateTimerConfig, &screenRotateTimer);
|
2023-11-08 14:27:22 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
if (preferences.getBool("timerActive", true)) {
|
|
|
|
esp_timer_start_periodic(screenRotateTimer,
|
|
|
|
getTimerSeconds() * usPerSecond);
|
|
|
|
}
|
2023-11-07 20:26:15 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
vTaskDelete(NULL);
|
2023-11-07 20:26:15 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
uint getTimerSeconds() { return preferences.getUInt("timerSeconds", 1800); }
|
2023-11-07 20:26:15 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
bool isTimerActive() { return esp_timer_is_active(screenRotateTimer); }
|
2023-11-08 19:29:06 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void setTimerActive(bool status) {
|
|
|
|
if (status) {
|
|
|
|
esp_timer_start_periodic(screenRotateTimer,
|
|
|
|
getTimerSeconds() * usPerSecond);
|
|
|
|
queueLedEffect(LED_EFFECT_START_TIMER);
|
|
|
|
preferences.putBool("timerActive", true);
|
|
|
|
} else {
|
|
|
|
esp_timer_stop(screenRotateTimer);
|
|
|
|
queueLedEffect(LED_EFFECT_PAUSE_TIMER);
|
|
|
|
preferences.putBool("timerActive", false);
|
|
|
|
}
|
2023-11-07 20:26:15 +00:00
|
|
|
|
2023-11-30 21:56:50 +00:00
|
|
|
if (eventSourceTaskHandle != NULL) xTaskNotifyGive(eventSourceTaskHandle);
|
2023-11-08 11:18:59 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void toggleTimerActive() { setTimerActive(!isTimerActive()); }
|
2023-11-07 20:26:15 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
uint getCurrentScreen() { return currentScreen; }
|
2023-11-07 20:26:15 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void setCurrentScreen(uint newScreen) {
|
|
|
|
if (newScreen != SCREEN_CUSTOM) {
|
|
|
|
preferences.putUInt("currentScreen", newScreen);
|
|
|
|
}
|
2023-11-07 20:26:15 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
currentScreen = newScreen;
|
2023-11-08 19:29:06 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
switch (currentScreen) {
|
2023-11-30 21:56:50 +00:00
|
|
|
case SCREEN_TIME: {
|
|
|
|
WorkItem timeUpdate = {TASK_TIME_UPDATE, 0};
|
|
|
|
xQueueSend(workQueue, &timeUpdate, portMAX_DELAY);
|
|
|
|
// xTaskNotifyGive(timeUpdateTaskHandle);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case SCREEN_HALVING_COUNTDOWN:
|
|
|
|
case SCREEN_BLOCK_HEIGHT: {
|
|
|
|
WorkItem blockUpdate = {TASK_BLOCK_UPDATE, 0};
|
|
|
|
xQueueSend(workQueue, &blockUpdate, portMAX_DELAY);
|
|
|
|
// xTaskNotifyGive(blockUpdateTaskHandle);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case SCREEN_MARKET_CAP:
|
|
|
|
case SCREEN_MSCW_TIME:
|
|
|
|
case SCREEN_BTC_TICKER: {
|
|
|
|
WorkItem priceUpdate = {TASK_PRICE_UPDATE, 0};
|
|
|
|
xQueueSend(workQueue, &priceUpdate, portMAX_DELAY);
|
|
|
|
// xTaskNotifyGive(priceUpdateTaskHandle);
|
|
|
|
break;
|
|
|
|
}
|
2024-03-10 19:37:45 +00:00
|
|
|
case SCREEN_BLOCK_FEE_RATE: {
|
|
|
|
WorkItem blockUpdate = {TASK_FEE_UPDATE, 0};
|
|
|
|
xQueueSend(workQueue, &blockUpdate, portMAX_DELAY);
|
|
|
|
break;
|
|
|
|
}
|
2023-11-30 21:38:01 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:56:50 +00:00
|
|
|
if (eventSourceTaskHandle != NULL) xTaskNotifyGive(eventSourceTaskHandle);
|
2023-11-08 11:18:59 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void nextScreen() {
|
|
|
|
int newCurrentScreen = (getCurrentScreen() + 1) % SCREEN_COUNT;
|
|
|
|
String key = "screen" + String(newCurrentScreen) + "Visible";
|
2023-11-08 11:18:59 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
while (!preferences.getBool(key.c_str(), true)) {
|
|
|
|
newCurrentScreen = (newCurrentScreen + 1) % SCREEN_COUNT;
|
|
|
|
key = "screen" + String(newCurrentScreen) + "Visible";
|
|
|
|
}
|
|
|
|
setCurrentScreen(newCurrentScreen);
|
2023-11-08 11:18:59 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void previousScreen() {
|
|
|
|
int newCurrentScreen = modulo(getCurrentScreen() - 1, SCREEN_COUNT);
|
|
|
|
String key = "screen" + String(newCurrentScreen) + "Visible";
|
2023-11-08 11:18:59 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
while (!preferences.getBool(key.c_str(), true)) {
|
|
|
|
newCurrentScreen = modulo(newCurrentScreen - 1, SCREEN_COUNT);
|
|
|
|
key = "screen" + String(newCurrentScreen) + "Visible";
|
|
|
|
}
|
|
|
|
setCurrentScreen(newCurrentScreen);
|
2023-11-08 11:18:59 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void showSystemStatusScreen() {
|
|
|
|
std::array<String, NUM_SCREENS> sysStatusEpdContent = {"", "", "", "",
|
|
|
|
"", "", ""};
|
|
|
|
|
|
|
|
String ipAddr = WiFi.localIP().toString();
|
|
|
|
String subNet = WiFi.subnetMask().toString();
|
|
|
|
|
|
|
|
sysStatusEpdContent[0] = "IP/Subnet";
|
|
|
|
|
|
|
|
int ipAddrPos = 0;
|
|
|
|
int subnetPos = 0;
|
|
|
|
for (int i = 0; i < 4; i++) {
|
|
|
|
sysStatusEpdContent[1 + i] = ipAddr.substring(0, ipAddr.indexOf('.')) +
|
|
|
|
"/" + subNet.substring(0, subNet.indexOf('.'));
|
|
|
|
ipAddrPos = ipAddr.indexOf('.') + 1;
|
|
|
|
subnetPos = subNet.indexOf('.') + 1;
|
|
|
|
ipAddr = ipAddr.substring(ipAddrPos);
|
|
|
|
subNet = subNet.substring(subnetPos);
|
|
|
|
}
|
|
|
|
sysStatusEpdContent[NUM_SCREENS - 2] = "RAM/Status";
|
|
|
|
|
|
|
|
sysStatusEpdContent[NUM_SCREENS - 1] =
|
|
|
|
String((int)round(ESP.getFreeHeap() / 1024)) + "/" +
|
|
|
|
(int)round(ESP.getHeapSize() / 1024);
|
|
|
|
setCurrentScreen(SCREEN_CUSTOM);
|
|
|
|
setEpdContent(sysStatusEpdContent);
|
2023-11-07 00:11:12 +00:00
|
|
|
}
|