Add Screensaver, Fix timeout
Add screensaver status screen to prevent oled burn-in: Display task switches to screensaver status-screen when certain time of inactivity is exceeded
This commit is contained in:
parent
bc9504d4ab
commit
0d082a52d8
@ -337,7 +337,7 @@ void controlledArmchair::resetTimeout(){
|
|||||||
// switch to IDLE when no activity (prevent accidential movement)
|
// switch to IDLE when no activity (prevent accidential movement)
|
||||||
// notify "power still on" when in IDLE for a very long time (prevent battery drain when forgotten to turn off)
|
// notify "power still on" when in IDLE for a very long time (prevent battery drain when forgotten to turn off)
|
||||||
// this function has to be run repeatedly (can be slow interval)
|
// this function has to be run repeatedly (can be slow interval)
|
||||||
#define TIMEOUT_POWER_STILL_ON_BEEP_INTERVAL_MS 30 * 60 * 1000 // beep every 30 minutes for someone to notice
|
#define TIMEOUT_POWER_STILL_ON_BEEP_INTERVAL_MS 10 * 60 * 1000 // beep every 30 minutes for someone to notice
|
||||||
// note: timeout durations are configured in config.cpp
|
// note: timeout durations are configured in config.cpp
|
||||||
void controlledArmchair::handleTimeout()
|
void controlledArmchair::handleTimeout()
|
||||||
{
|
{
|
||||||
@ -347,25 +347,26 @@ void controlledArmchair::handleTimeout()
|
|||||||
noActivityDurationMs / 1000 / 60,
|
noActivityDurationMs / 1000 / 60,
|
||||||
noActivityDurationMs / 1000 % 60,
|
noActivityDurationMs / 1000 % 60,
|
||||||
config.timeoutSwitchToIdleMs / 1000,
|
config.timeoutSwitchToIdleMs / 1000,
|
||||||
config.timeoutNotifyPowerStillOnMs / 1000);
|
config.timeoutNotifyPowerStillOnMs / 1000 / 60 / 60);
|
||||||
|
|
||||||
// timeout to IDLE when not idling already
|
// timeout to IDLE when not idling already
|
||||||
if (mode != controlMode_t::IDLE && noActivityDurationMs > config.timeoutSwitchToIdleMs)
|
if (mode != controlMode_t::IDLE && noActivityDurationMs > config.timeoutSwitchToIdleMs)
|
||||||
{
|
{
|
||||||
ESP_LOGW(TAG, "timeout check: [TIMEOUT], no activity for more than %ds -> switch to IDLE", config.timeoutSwitchToIdleMs / 1000);
|
ESP_LOGW(TAG, "timeout check: [TIMEOUT], no activity for more than %ds -> switch to IDLE", config.timeoutSwitchToIdleMs / 1000);
|
||||||
changeMode(controlMode_t::IDLE);
|
changeMode(controlMode_t::IDLE);
|
||||||
|
//TODO switch to previous status-screen when activity detected
|
||||||
}
|
}
|
||||||
// repeatedly notify via buzzer when in IDLE for a very long time to prevent battery drain ("forgot to turn off")
|
// repeatedly notify via buzzer when in IDLE for a very long time to prevent battery drain ("forgot to turn off")
|
||||||
// note: ignores user input while in IDLE
|
// note: ignores user input while in IDLE
|
||||||
else if (esp_log_timestamp() - timestamp_lastModeChange > config.timeoutNotifyPowerStillOnMs)
|
else if ((esp_log_timestamp() - timestamp_lastModeChange) > config.timeoutNotifyPowerStillOnMs)
|
||||||
{
|
{
|
||||||
// beep in certain intervals
|
// beep in certain intervals
|
||||||
if (esp_log_timestamp() - timestamp_lastTimeoutBeep > TIMEOUT_POWER_STILL_ON_BEEP_INTERVAL_MS)
|
if ((esp_log_timestamp() - timestamp_lastTimeoutBeep) > TIMEOUT_POWER_STILL_ON_BEEP_INTERVAL_MS)
|
||||||
{
|
{
|
||||||
ESP_LOGD(TAG, "timeout: [TIMEOUT] in IDLE for more than %.3f hours", (float)(esp_log_timestamp() - timestamp_lastModeChange) / 1000 / 60 / 60);
|
ESP_LOGW(TAG, "timeout: [TIMEOUT] in IDLE since %.3f hours -> beeping", (float)(esp_log_timestamp() - timestamp_lastModeChange) / 1000 / 60 / 60);
|
||||||
// TODO dont beep at certain times ranges (e.g. at night)
|
// TODO dont beep at certain times ranges (e.g. at night)
|
||||||
timestamp_lastTimeoutBeep = esp_log_timestamp();
|
timestamp_lastTimeoutBeep = esp_log_timestamp();
|
||||||
buzzer->beep(4, 100, 50);
|
buzzer->beep(6, 100, 50);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -97,6 +97,8 @@ class controlledArmchair {
|
|||||||
void setMaxDuty(float maxDutyNew) { writeMaxDuty(maxDutyNew); };
|
void setMaxDuty(float maxDutyNew) { writeMaxDuty(maxDutyNew); };
|
||||||
float getMaxDuty() const {return joystickGenerateCommands_config.maxDuty; };
|
float getMaxDuty() const {return joystickGenerateCommands_config.maxDuty; };
|
||||||
|
|
||||||
|
uint32_t getInactivityDurationMs() {return esp_log_timestamp() - timestamp_lastActivity;};
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
||||||
//--- functions ---
|
//--- functions ---
|
||||||
|
@ -234,7 +234,7 @@ float getBatteryPercent()
|
|||||||
//shows overview on entire display:
|
//shows overview on entire display:
|
||||||
//Battery percentage, voltage, current, mode, rpm, speed
|
//Battery percentage, voltage, current, mode, rpm, speed
|
||||||
#define STATUS_SCREEN_OVERVIEW_UPDATE_INTERVAL 500
|
#define STATUS_SCREEN_OVERVIEW_UPDATE_INTERVAL 500
|
||||||
void showStatusScreenOverview(display_task_parameters_t * objects)
|
void showStatusScreenOverview(display_task_parameters_t *objects)
|
||||||
{
|
{
|
||||||
//-- battery percentage --
|
//-- battery percentage --
|
||||||
// TODO update when no load (currentsensors = ~0A) only
|
// TODO update when no load (currentsensors = ~0A) only
|
||||||
@ -265,7 +265,6 @@ void showStatusScreenOverview(display_task_parameters_t * objects)
|
|||||||
vTaskDelay(STATUS_SCREEN_OVERVIEW_UPDATE_INTERVAL / portTICK_PERIOD_MS);
|
vTaskDelay(STATUS_SCREEN_OVERVIEW_UPDATE_INTERVAL / portTICK_PERIOD_MS);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
//############################
|
//############################
|
||||||
//##### showScreen Speed #####
|
//##### showScreen Speed #####
|
||||||
//############################
|
//############################
|
||||||
@ -331,6 +330,42 @@ void showStatusScreenMotors(display_task_parameters_t *objects)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//###############################
|
||||||
|
//#### showScreen Sreensaver ####
|
||||||
|
//###############################
|
||||||
|
// show minimal text scrolling across screen to prevent burn in
|
||||||
|
// indicates that armchair is still on (probably "forgotten to be turned off")
|
||||||
|
#define STATUS_SCREEN_TIMEOUT_NEXT_LINE_SEC 6
|
||||||
|
void showStatusScreenScreensaver(display_task_parameters_t *objects)
|
||||||
|
{
|
||||||
|
// to prevent burn-in only showing minimal and scrolling text
|
||||||
|
ssd1306_clear_screen(&dev, false);
|
||||||
|
ssd1306_hardware_scroll(&dev, SCROLL_RIGHT);
|
||||||
|
|
||||||
|
// loop through all lines (also scroll down)
|
||||||
|
for (int line = 0; line < 7; line++)
|
||||||
|
{
|
||||||
|
ssd1306_clear_screen(&dev, false);
|
||||||
|
displayTextLine(&dev, line, false, false, "IDLE since");
|
||||||
|
displayTextLine(&dev, line + 1, false, false, "%.1fh, B:%02.0f%%", (float)objects->control->getInactivityDurationMs() / 1000 / 60 / 60, getBatteryPercent());
|
||||||
|
// check exit condition while waiting some time before switching to next line
|
||||||
|
int secondsPassed = 0;
|
||||||
|
while (secondsPassed < STATUS_SCREEN_TIMEOUT_NEXT_LINE_SEC)
|
||||||
|
{
|
||||||
|
secondsPassed ++;
|
||||||
|
vTaskDelay(1000 / portTICK_PERIOD_MS);
|
||||||
|
// switch to default status screen, when IDLE mode is exited (timeout has ended)
|
||||||
|
if (objects->control->getCurrentMode() != controlMode_t::IDLE)
|
||||||
|
{
|
||||||
|
display_selectStatusPage(STATUS_SCREEN_OVERVIEW);
|
||||||
|
ssd1306_hardware_scroll(&dev, SCROLL_STOP);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ssd1306_hardware_scroll(&dev, SCROLL_STOP);
|
||||||
|
}
|
||||||
|
|
||||||
//########################
|
//########################
|
||||||
//#### showStartupMsg ####
|
//#### showStartupMsg ####
|
||||||
//########################
|
//########################
|
||||||
@ -363,6 +398,8 @@ void display_selectStatusPage(displayStatusPage_t newStatusPage){
|
|||||||
//======= display task =======
|
//======= display task =======
|
||||||
//============================
|
//============================
|
||||||
// TODO: separate task for each loop?
|
// TODO: separate task for each loop?
|
||||||
|
#define DISPLAY_IDLE_TIMEOUT_SCREENSAVER_MS 15*60*1000
|
||||||
|
|
||||||
void display_task(void *pvParameters)
|
void display_task(void *pvParameters)
|
||||||
{
|
{
|
||||||
ESP_LOGW(TAG, "Initializing display and starting handle loop");
|
ESP_LOGW(TAG, "Initializing display and starting handle loop");
|
||||||
@ -403,7 +440,13 @@ void display_task(void *pvParameters)
|
|||||||
case STATUS_SCREEN_MOTORS:
|
case STATUS_SCREEN_MOTORS:
|
||||||
showStatusScreenMotors(objects);
|
showStatusScreenMotors(objects);
|
||||||
break;
|
break;
|
||||||
|
case STATUS_SCREEN_SCREENSAVER:
|
||||||
|
showStatusScreenScreensaver(objects);
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
// switch to screensaver when no user activity for a long time, to prevent burn in
|
||||||
|
if (objects->control->getInactivityDurationMs() > DISPLAY_IDLE_TIMEOUT_SCREENSAVER_MS)
|
||||||
|
selectedStatusPage = STATUS_SCREEN_SCREENSAVER;
|
||||||
}
|
}
|
||||||
// TODO add pages and menus
|
// TODO add pages and menus
|
||||||
}
|
}
|
||||||
|
@ -49,7 +49,7 @@ typedef struct display_task_parameters_t {
|
|||||||
|
|
||||||
|
|
||||||
// enum for selecting the currently shown status page (display content when not in MENU mode)
|
// enum for selecting the currently shown status page (display content when not in MENU mode)
|
||||||
typedef enum displayStatusPage_t {STATUS_SCREEN_OVERVIEW=0, STATUS_SCREEN_SPEED, STATUS_SCREEN_JOYSTICK, STATUS_SCREEN_MOTORS} displayStatusPage_t;
|
typedef enum displayStatusPage_t {STATUS_SCREEN_OVERVIEW=0, STATUS_SCREEN_SPEED, STATUS_SCREEN_JOYSTICK, STATUS_SCREEN_MOTORS, STATUS_SCREEN_SCREENSAVER} displayStatusPage_t;
|
||||||
|
|
||||||
// get precise battery voltage (using lookup table)
|
// get precise battery voltage (using lookup table)
|
||||||
float getBatteryVoltage();
|
float getBatteryVoltage();
|
||||||
|
Loading…
x
Reference in New Issue
Block a user