From 3514dd6bf208b7ec635b6d9901fffab2db652eb6 Mon Sep 17 00:00:00 2001 From: jonny Date: Sat, 2 Mar 2024 13:01:29 +0100 Subject: [PATCH] Add Brightness reduction at inactive, Optimize timeout display.cpp: - add timeout where display brightness gets reduced - rework display-task loop to handle timeouts - run commands when changing from/to status-page (toggle scrolling) - add BRIGHTNESS_TEST option config: Add display timeouts to config.cpp control: - dont reset timeout at mode change (happens not always at user input) - remove deprecated config option --- board_single/main/config.cpp | 13 ++- board_single/main/control.cpp | 4 +- board_single/main/control.hpp | 1 - board_single/main/display.cpp | 151 ++++++++++++++++++++++++++-------- board_single/main/display.hpp | 7 +- 5 files changed, 131 insertions(+), 45 deletions(-) diff --git a/board_single/main/config.cpp b/board_single/main/config.cpp index 5598afe..061851d 100644 --- a/board_single/main/config.cpp +++ b/board_single/main/config.cpp @@ -201,16 +201,21 @@ speedSensor_config_t speedRight_config{ //------------------------- //-------- display -------- //------------------------- -display_config_t display_config { +display_config_t display_config{ + // hardware initialization .gpio_scl = GPIO_NUM_22, .gpio_sda = GPIO_NUM_23, - .gpio_reset = -1, //negative number disables reset feature + .gpio_reset = -1, // negative number disables reset feature .width = 128, .height = 64, .offsetX = 2, .flip = false, - .contrast = 0xff, //max: 255 -}; + .contrastNormal = 170, // max: 255 + // display task + .contrastReduced = 30, // max: 255 + .timeoutReduceContrastMs = 5 * 60 * 1000, // actions at certain inactivity + .timeoutSwitchToScreensaverMs = 30 * 60 * 1000 + }; diff --git a/board_single/main/control.cpp b/board_single/main/control.cpp index 1998aa2..9d9c914 100644 --- a/board_single/main/control.cpp +++ b/board_single/main/control.cpp @@ -337,7 +337,7 @@ void controlledArmchair::resetTimeout(){ // 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) // this function has to be run repeatedly (can be slow interval) -#define TIMEOUT_POWER_STILL_ON_BEEP_INTERVAL_MS 10 * 60 * 1000 // beep every 30 minutes for someone to notice +#define TIMEOUT_POWER_STILL_ON_BEEP_INTERVAL_MS 5 * 60 * 1000 // beep every 5 minutes for someone to notice // note: timeout durations are configured in config.cpp void controlledArmchair::handleTimeout() { @@ -378,8 +378,6 @@ void controlledArmchair::handleTimeout() //----------------------------------- //function to change to a specified control mode void controlledArmchair::changeMode(controlMode_t modeNew) { - //reset timeout timer - resetTimeout(); //exit if target mode is already active if (mode == modeNew) { diff --git a/board_single/main/control.hpp b/board_single/main/control.hpp index daf40f0..39dc8c7 100644 --- a/board_single/main/control.hpp +++ b/board_single/main/control.hpp @@ -29,7 +29,6 @@ typedef struct control_config_t { //timeout options uint32_t timeoutSwitchToIdleMs; //time of inactivity after which the mode gets switched to IDLE uint32_t timeoutNotifyPowerStillOnMs; - float timeoutTolerancePer; //percentage the duty can vary between timeout checks considered still inactive } control_config_t; diff --git a/board_single/main/display.cpp b/board_single/main/display.cpp index c8806e1..c73491d 100644 --- a/board_single/main/display.cpp +++ b/board_single/main/display.cpp @@ -12,8 +12,14 @@ extern "C"{ #define STARTUP_MSG_TIMEOUT 2000 #define ADC_BATT_VOLTAGE ADC1_CHANNEL_6 #define BAT_CELL_COUNT 7 +// continously vary display contrast from 0 to 250 in OVERVIEW status screen +//#define BRIGHTNESS_TEST +//=== variables === +// every function can access the display configuration from config.cpp +static display_config_t displayConfig; + //-------------------------- //------- getVoltage ------- @@ -61,7 +67,10 @@ void display_init(display_config_t config){ ssd1306_init(&dev, config.width, config.height, config.offsetX); ssd1306_clear_screen(&dev, false); - ssd1306_contrast(&dev, config.contrast); + ssd1306_contrast(&dev, config.contrastNormal); + + //store configuration locally (e.g. for accessing timeouts) + displayConfig = config; } @@ -233,7 +242,7 @@ float getBatteryPercent() //############################# //shows overview on entire display: //Battery percentage, voltage, current, mode, rpm, speed -#define STATUS_SCREEN_OVERVIEW_UPDATE_INTERVAL 500 +#define STATUS_SCREEN_OVERVIEW_UPDATE_INTERVAL 400 void showStatusScreenOverview(display_task_parameters_t *objects) { //-- battery percentage -- @@ -263,8 +272,20 @@ void showStatusScreenOverview(display_task_parameters_t *objects) objects->speedLeft->getRpm(), objects->speedRight->getRpm()); vTaskDelay(STATUS_SCREEN_OVERVIEW_UPDATE_INTERVAL / portTICK_PERIOD_MS); + + //-- brightness test -- +#ifdef BRIGHTNESS_TEST + // continously vary brightness/contrast for testing + displayConfig.contrastNormal += 10; + if (displayConfig.contrastNormal > 255) + displayConfig.contrastNormal = 0; + ssd1306_contrast(&dev, displayConfig.contrastNormal); + vTaskDelay(100 / portTICK_PERIOD_MS); + ESP_LOGW(TAG, "TEST BRIGHTNESS, setting to %d", displayConfig.contrastNormal); +#endif } + //############################ //##### showScreen Speed ##### //############################ @@ -314,8 +335,6 @@ void showStatusScreenJoystick(display_task_parameters_t * objects) #define STATUS_SCREEN_MOTORS_UPDATE_INTERVAL 150 void showStatusScreenMotors(display_task_parameters_t *objects) { - // print all joystick data - joystickData_t data = objects->joystick->getData(); displayTextLine(&dev, 0, true, false, "%-4.0fW ", fabs(objects->motorLeft->getCurrentA()) * getBatteryVoltage()); displayTextLine(&dev, 3, true, false, "%-4.0fW ", fabs(objects->motorRight->getCurrentA()) * getBatteryVoltage()); //displayTextLine(&dev, 0, true, false, "L:%02.0f%%", objects->motorLeft->getStatus().duty); @@ -333,37 +352,40 @@ 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 +// show inactivity duration and battery perventage scrolling across screen the entire screen to prevent burn in +#define STATUS_SCREEN_SCREENSAVER_DELAY_NEXT_LINE_MS 10*1000 +#define STATUS_SCREEN_SCREENSAVER_UPDATE_INTERVAL 500 void showStatusScreenScreensaver(display_task_parameters_t *objects) { - // to prevent burn-in only showing minimal and scrolling text + // note: scrolling is enabled at screen change (display_selectStatusPage()) + static int msPassed = 0; + static int currentLine = 0; + static bool lineChanging = false; + // clear display once when rotating to next line + if (lineChanging) { ssd1306_clear_screen(&dev, false); - ssd1306_hardware_scroll(&dev, SCROLL_RIGHT); + lineChanging = false; + } + // update text every iteration to prevent empty screen at start + displayTextLine(&dev, currentLine, false, false, "IDLE since:"); + displayTextLine(&dev, currentLine + 1, false, false, "%.1fh, B:%02.0f%%", + (float)objects->control->getInactivityDurationMs() / 1000 / 60 / 60, + getBatteryPercent()); - // 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); + // to not block the display task for several seconds returning every e.g. 500ms here + // -> ensures detection of activity (exit condition) in task loop is handled regularly + if (msPassed > STATUS_SCREEN_SCREENSAVER_DELAY_NEXT_LINE_MS) // switch to next line is due + { + msPassed = 0; // rest seconds count + // increment / rotate to next line + if (++currentLine >= 7) // rotate to next line + currentLine = 0; + lineChanging = true; //clear screen in next run + } + // wait update-update interval and increment passed time after each run + vTaskDelay(STATUS_SCREEN_SCREENSAVER_UPDATE_INTERVAL / portTICK_PERIOD_MS); + msPassed += STATUS_SCREEN_SCREENSAVER_UPDATE_INTERVAL; + // note: scrolling is disabled at screen change (display_selectStatusPage()) } //######################## @@ -388,17 +410,39 @@ void showStartupMsg(){ //============================ //===== selectStatusPage ===== //============================ -void display_selectStatusPage(displayStatusPage_t newStatusPage){ +void display_selectStatusPage(displayStatusPage_t newStatusPage) +{ + //-- run commands when switching FROM certain mode -- + switch (selectedStatusPage) + { + case STATUS_SCREEN_SCREENSAVER: // disable scrolling when exiting screensaver + ssd1306_hardware_scroll(&dev, SCROLL_STOP); + break; + default: + break; + } + ESP_LOGW(TAG, "switching statusPage from %d to %d", (int)selectedStatusPage, (int)newStatusPage); selectedStatusPage = newStatusPage; + + //-- run commands when switching TO certain mode -- + switch (selectedStatusPage) + { + case STATUS_SCREEN_SCREENSAVER: + ssd1306_clear_screen(&dev, false); + ssd1306_hardware_scroll(&dev, SCROLL_RIGHT); + break; + default: + break; + } } + //============================ //======= display task ======= //============================ // TODO: separate task for each loop? -#define DISPLAY_IDLE_TIMEOUT_SCREENSAVER_MS 15*60*1000 void display_task(void *pvParameters) { @@ -444,9 +488,44 @@ void display_task(void *pvParameters) 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; + + //--- handle timeouts --- + uint32_t inactiveMs = objects->control->getInactivityDurationMs(); + //-- screensaver -- + // handle switch to screensaver when no user input for a long time + if (inactiveMs > displayConfig.timeoutSwitchToScreensaverMs) // timeout - switch to screensaver is due + { + if (selectedStatusPage != STATUS_SCREEN_SCREENSAVER){ // switch/log only once at change + ESP_LOGW(TAG, "no activity for more than %d min, switching to screensaver", inactiveMs / 1000 / 60); + display_selectStatusPage(STATUS_SCREEN_SCREENSAVER); + } + } + else if (selectedStatusPage == STATUS_SCREEN_SCREENSAVER) // exit screensaver when there was recent activity + { + ESP_LOGW(TAG, "recent activity detected, disabling screensaver"); + display_selectStatusPage(STATUS_SCREEN_OVERVIEW); + } + + //-- reduce brightness -- + // handle brightness reduction when no user input for some time + static bool brightnessIsReduced = false; + if (inactiveMs > displayConfig.timeoutReduceContrastMs) // threshold exceeded - reduction of brightness is due + { + if (!brightnessIsReduced) //change / log only once at change + { + // reduce display brightness (less burn in) + ESP_LOGW(TAG, "no activity for more than %d min, reducing display brightness to %d/255", inactiveMs / 1000 / 60, displayConfig.contrastReduced); + ssd1306_contrast(&dev, displayConfig.contrastReduced); + brightnessIsReduced = true; + } + } + else if (brightnessIsReduced) // threshold not exceeded anymore, but still reduced + { + // increase display brighness again + ESP_LOGW(TAG, "recent activity detected, increasing brightness again"); + ssd1306_contrast(&dev, displayConfig.contrastNormal); + brightnessIsReduced = false; + } } // TODO add pages and menus } diff --git a/board_single/main/display.hpp b/board_single/main/display.hpp index 776861b..4646dc0 100644 --- a/board_single/main/display.hpp +++ b/board_single/main/display.hpp @@ -22,6 +22,7 @@ extern "C" { // configuration for initializing display (passed to task as well) typedef struct display_config_t { + // initialization gpio_num_t gpio_scl; gpio_num_t gpio_sda; int gpio_reset; // negative number means reset pin is not connected or not used @@ -29,7 +30,11 @@ typedef struct display_config_t { int height; int offsetX; bool flip; - int contrast; + // display-task + int contrastNormal; + int contrastReduced; + uint32_t timeoutReduceContrastMs; + uint32_t timeoutSwitchToScreensaverMs; } display_config_t;