mirror of
https://github.com/daveallie/crosspoint-reader.git
synced 2025-12-19 07:37:41 +03:00
167 lines
4.9 KiB
C++
167 lines
4.9 KiB
C++
#include "SettingsScreen.h"
|
|
|
|
#include <GfxRenderer.h>
|
|
|
|
#include "CrossPointSettings.h"
|
|
#include "config.h"
|
|
|
|
// Define the static settings list
|
|
|
|
const SettingInfo SettingsScreen::settingsList[SettingsScreen::settingsCount] = {
|
|
{"White Sleep Screen", SettingType::TOGGLE, &CrossPointSettings::whiteSleepScreen},
|
|
{"Extra Paragraph Spacing", SettingType::TOGGLE, &CrossPointSettings::extraParagraphSpacing},
|
|
{"WiFi", SettingType::ACTION, nullptr}};
|
|
|
|
void SettingsScreen::taskTrampoline(void* param) {
|
|
auto* self = static_cast<SettingsScreen*>(param);
|
|
self->displayTaskLoop();
|
|
}
|
|
|
|
void SettingsScreen::onEnter() {
|
|
renderingMutex = xSemaphoreCreateMutex();
|
|
|
|
// Reset selection to first item
|
|
selectedSettingIndex = 0;
|
|
|
|
// Trigger first update
|
|
updateRequired = true;
|
|
|
|
xTaskCreate(&SettingsScreen::taskTrampoline, "SettingsScreenTask",
|
|
2048, // Stack size
|
|
this, // Parameters
|
|
1, // Priority
|
|
&displayTaskHandle // Task handle
|
|
);
|
|
}
|
|
|
|
void SettingsScreen::onExit() {
|
|
// Wait until not rendering to delete task to avoid killing mid-instruction to EPD
|
|
xSemaphoreTake(renderingMutex, portMAX_DELAY);
|
|
if (displayTaskHandle) {
|
|
vTaskDelete(displayTaskHandle);
|
|
displayTaskHandle = nullptr;
|
|
}
|
|
vSemaphoreDelete(renderingMutex);
|
|
renderingMutex = nullptr;
|
|
}
|
|
|
|
void SettingsScreen::handleInput() {
|
|
// Handle actions with early return
|
|
if (inputManager.wasPressed(InputManager::BTN_CONFIRM)) {
|
|
activateCurrentSetting();
|
|
updateRequired = true;
|
|
return;
|
|
}
|
|
|
|
if (inputManager.wasPressed(InputManager::BTN_BACK)) {
|
|
SETTINGS.saveToFile();
|
|
onGoHome();
|
|
return;
|
|
}
|
|
|
|
// Handle navigation
|
|
if (inputManager.wasPressed(InputManager::BTN_UP) || inputManager.wasPressed(InputManager::BTN_LEFT)) {
|
|
// Move selection up (with wrap-around)
|
|
selectedSettingIndex = (selectedSettingIndex > 0) ? (selectedSettingIndex - 1) : (settingsCount - 1);
|
|
updateRequired = true;
|
|
} else if (inputManager.wasPressed(InputManager::BTN_DOWN) || inputManager.wasPressed(InputManager::BTN_RIGHT)) {
|
|
// Move selection down
|
|
if (selectedSettingIndex < settingsCount - 1) {
|
|
selectedSettingIndex++;
|
|
updateRequired = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
void SettingsScreen::activateCurrentSetting() {
|
|
// Validate index
|
|
if (selectedSettingIndex < 0 || selectedSettingIndex >= settingsCount) {
|
|
return;
|
|
}
|
|
|
|
const auto& setting = settingsList[selectedSettingIndex];
|
|
|
|
if (setting.type == SettingType::TOGGLE) {
|
|
toggleCurrentSetting();
|
|
// Trigger a redraw of the entire screen
|
|
updateRequired = true;
|
|
} else if (setting.type == SettingType::ACTION) {
|
|
// Handle action settings
|
|
if (std::string(setting.name) == "WiFi") {
|
|
onGoWifi();
|
|
}
|
|
}
|
|
}
|
|
|
|
void SettingsScreen::toggleCurrentSetting() {
|
|
// Validate index
|
|
if (selectedSettingIndex < 0 || selectedSettingIndex >= settingsCount) {
|
|
return;
|
|
}
|
|
|
|
const auto& setting = settingsList[selectedSettingIndex];
|
|
|
|
// Only toggle if it's a toggle type and has a value pointer
|
|
if (setting.type != SettingType::TOGGLE || setting.valuePtr == nullptr) {
|
|
return;
|
|
}
|
|
|
|
// Toggle the boolean value using the member pointer
|
|
bool currentValue = SETTINGS.*(setting.valuePtr);
|
|
SETTINGS.*(setting.valuePtr) = !currentValue;
|
|
|
|
// Save settings when they change
|
|
SETTINGS.saveToFile();
|
|
}
|
|
|
|
void SettingsScreen::displayTaskLoop() {
|
|
while (true) {
|
|
if (updateRequired) {
|
|
updateRequired = false;
|
|
xSemaphoreTake(renderingMutex, portMAX_DELAY);
|
|
render();
|
|
xSemaphoreGive(renderingMutex);
|
|
}
|
|
vTaskDelay(10 / portTICK_PERIOD_MS);
|
|
}
|
|
}
|
|
|
|
void SettingsScreen::render() const {
|
|
renderer.clearScreen();
|
|
|
|
const auto pageWidth = GfxRenderer::getScreenWidth();
|
|
const auto pageHeight = GfxRenderer::getScreenHeight();
|
|
|
|
// Draw header
|
|
renderer.drawCenteredText(READER_FONT_ID, 10, "Settings", true, BOLD);
|
|
|
|
// We always have at least one setting
|
|
|
|
// Draw all settings
|
|
for (int i = 0; i < settingsCount; i++) {
|
|
const int settingY = 60 + i * 30; // 30 pixels between settings
|
|
|
|
// Draw selection indicator for the selected setting
|
|
if (i == selectedSettingIndex) {
|
|
renderer.drawText(UI_FONT_ID, 5, settingY, ">");
|
|
}
|
|
|
|
// Draw setting name
|
|
renderer.drawText(UI_FONT_ID, 20, settingY, settingsList[i].name);
|
|
|
|
// Draw value based on setting type
|
|
if (settingsList[i].type == SettingType::TOGGLE && settingsList[i].valuePtr != nullptr) {
|
|
bool value = SETTINGS.*(settingsList[i].valuePtr);
|
|
renderer.drawText(UI_FONT_ID, pageWidth - 80, settingY, value ? "ON" : "OFF");
|
|
} else if (settingsList[i].type == SettingType::ACTION) {
|
|
renderer.drawText(UI_FONT_ID, pageWidth - 80, settingY, ">");
|
|
}
|
|
}
|
|
|
|
// Draw help text
|
|
renderer.drawText(SMALL_FONT_ID, 20, pageHeight - 30, "Press OK to select, BACK to save & exit");
|
|
|
|
// Always use standard refresh for settings screen
|
|
renderer.displayBuffer();
|
|
}
|