From 3aa6bcc40319ac1bff92779a5ed4d0be3810ade1 Mon Sep 17 00:00:00 2001 From: jonny Date: Thu, 29 Feb 2024 23:50:12 +0100 Subject: [PATCH 1/6] Adjust button commands (dont toggle modes) Adjust button actions to match the current README --- board_single/main/button.cpp | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/board_single/main/button.cpp b/board_single/main/button.cpp index ae4a36e..e62a692 100644 --- a/board_single/main/button.cpp +++ b/board_single/main/button.cpp @@ -96,8 +96,8 @@ void buttonCommands::action (uint8_t count, bool lastPressLong){ // ## switch to ADJUST_CHAIR mode ## if (lastPressLong) { - ESP_LOGW(TAG, "cmd %d: toggle ADJUST_CHAIR", count); - control->toggleMode(controlMode_t::ADJUST_CHAIR); + ESP_LOGW(TAG, "cmd %d: switch to ADJUST_CHAIR", count); + control->changeMode(controlMode_t::ADJUST_CHAIR); } // ## toggle IDLE ## else { @@ -114,14 +114,14 @@ void buttonCommands::action (uint8_t count, bool lastPressLong){ case 4: // ## switch to HTTP mode ## - ESP_LOGW(TAG, "cmd %d: toggle between HTTP and JOYSTICK", count); - control->toggleModes(controlMode_t::HTTP, controlMode_t::JOYSTICK); //toggle between HTTP and JOYSTICK mode + ESP_LOGW(TAG, "cmd %d: switch to HTTP", count); + control->changeMode(controlMode_t::HTTP); //switch to HTTP mode break; case 6: // ## switch to MASSAGE mode ## - ESP_LOGW(TAG, "cmd %d: toggle between MASSAGE and JOYSTICK", count); - control->toggleModes(controlMode_t::MASSAGE, controlMode_t::JOYSTICK); //toggle between MASSAGE and JOYSTICK mode + ESP_LOGW(TAG, "switch to MASSAGE"); + control->changeMode(controlMode_t::MASSAGE); //switch to MASSAGE mode break; case 8: @@ -129,7 +129,7 @@ void buttonCommands::action (uint8_t count, bool lastPressLong){ //toggle deceleration fading between on and off //decelEnabled = motorLeft->toggleFade(fadeType_t::DECEL); //motorRight->toggleFade(fadeType_t::DECEL); - decelEnabled = motorLeft->toggleFade(fadeType_t::ACCEL); + decelEnabled = motorLeft->toggleFade(fadeType_t::ACCEL); //TODO remove/simplify this using less functions motorRight->toggleFade(fadeType_t::ACCEL); ESP_LOGW(TAG, "cmd %d: toggle deceleration fading to: %d", count, (int)decelEnabled); if (decelEnabled){ From bc9504d4ab5f7b7f143526ea46a0d48c62f62110 Mon Sep 17 00:00:00 2001 From: jonny Date: Fri, 1 Mar 2024 13:55:51 +0100 Subject: [PATCH 2/6] Add Timeout / Notification on "forgot to turn off" control: - reset timeout on user input only -> drop reset on changed motor duty - add timeoutNotifyPowerStillOnMs -> when forgotten to turn off the power the buzzer beeps a few times every 30 minutes - Add/fix JOYSTICK_LOG_IN_IDLE option remove empty config.hpp --- board_single/main/CMakeLists.txt | 1 - board_single/main/auto.cpp | 1 - board_single/main/button.cpp | 17 ++-- board_single/main/button.hpp | 1 - board_single/main/config.cpp | 8 +- board_single/main/config.h | 6 ++ board_single/main/config.hpp | 57 ------------ board_single/main/control.cpp | 154 ++++++++++++++++--------------- board_single/main/control.hpp | 9 +- board_single/main/main.cpp | 1 - board_single/main/menu.cpp | 11 +-- 11 files changed, 106 insertions(+), 160 deletions(-) create mode 100644 board_single/main/config.h delete mode 100644 board_single/main/config.hpp diff --git a/board_single/main/CMakeLists.txt b/board_single/main/CMakeLists.txt index cbbf984..52ed876 100644 --- a/board_single/main/CMakeLists.txt +++ b/board_single/main/CMakeLists.txt @@ -1,7 +1,6 @@ idf_component_register( SRCS "main.cpp" - "config.cpp" "control.cpp" "button.cpp" "fan.cpp" diff --git a/board_single/main/auto.cpp b/board_single/main/auto.cpp index 8882c74..a3f2d3c 100644 --- a/board_single/main/auto.cpp +++ b/board_single/main/auto.cpp @@ -1,5 +1,4 @@ #include "auto.hpp" -#include "config.hpp" //tag for logging static const char * TAG = "automatedArmchair"; diff --git a/board_single/main/button.cpp b/board_single/main/button.cpp index e62a692..4b7c2c0 100644 --- a/board_single/main/button.cpp +++ b/board_single/main/button.cpp @@ -98,13 +98,14 @@ void buttonCommands::action (uint8_t count, bool lastPressLong){ { ESP_LOGW(TAG, "cmd %d: switch to ADJUST_CHAIR", count); control->changeMode(controlMode_t::ADJUST_CHAIR); - } - // ## toggle IDLE ## - else { + } + // ## toggle IDLE ## + else + { ESP_LOGW(TAG, "cmd %d: toggle IDLE", count); - control->toggleIdle(); //toggle between idle and previous/default mode - } - break; + control->toggleIdle(); // toggle between idle and previous/default mode + } + break; case 3: // ## switch to JOYSTICK mode ## @@ -155,7 +156,7 @@ void buttonCommands::action (uint8_t count, bool lastPressLong){ // when not in MENU mode, repeatedly receives events from encoder button // and takes the corresponding action // this function has to be started once in a separate task -#define INPUT_TIMEOUT 800 // duration of no button events, after which action is run (implicitly also is 'long-press' time) +#define INPUT_TIMEOUT 700 // duration of no button events, after which action is run (implicitly also is 'long-press' time) void buttonCommands::startHandleLoop() { //-- variables -- @@ -178,7 +179,7 @@ void buttonCommands::startHandleLoop() //-- get events from encoder -- if (xQueueReceive(encoderQueue, &ev, INPUT_TIMEOUT / portTICK_PERIOD_MS)) { - control->resetTimeout(); //reset inactivity IDLE timeout + control->resetTimeout(); // user input -> reset switch to IDLE timeout switch (ev.type) { break; diff --git a/board_single/main/button.hpp b/board_single/main/button.hpp index 30b52b0..8b5e523 100644 --- a/board_single/main/button.hpp +++ b/board_single/main/button.hpp @@ -5,7 +5,6 @@ #include "control.hpp" #include "motorctl.hpp" #include "auto.hpp" -#include "config.hpp" #include "joystick.hpp" diff --git a/board_single/main/config.cpp b/board_single/main/config.cpp index c029fa5..5598afe 100644 --- a/board_single/main/config.cpp +++ b/board_single/main/config.cpp @@ -121,11 +121,9 @@ motorctl_config_t configMotorControlRight = { //------------------------------ control_config_t configControl = { .defaultMode = controlMode_t::JOYSTICK, // default mode after startup and toggling IDLE - //--- timeout --- - .timeoutMs = 3 * 60 * 1000, // time of inactivity after which the mode gets switched to IDLE - .timeoutTolerancePer = 5, // percentage the duty can vary between timeout checks considered still inactive - //--- http mode --- - + //--- timeouts --- + .timeoutSwitchToIdleMs = 5 * 60 * 1000, // time of inactivity after which the mode gets switched to IDLE + .timeoutNotifyPowerStillOnMs = 6 * 60 * 60 * 1000 // time in IDLE after which buzzer beeps in certain interval (notify "forgot to turn off") }; //------------------------------- diff --git a/board_single/main/config.h b/board_single/main/config.h new file mode 100644 index 0000000..9cc2aab --- /dev/null +++ b/board_single/main/config.h @@ -0,0 +1,6 @@ +#pragma once + +// outsourced macros / definitions + +//-- control.cpp -- +//#define JOYSTICK_LOG_IN_IDLE \ No newline at end of file diff --git a/board_single/main/config.hpp b/board_single/main/config.hpp deleted file mode 100644 index 697d4f4..0000000 --- a/board_single/main/config.hpp +++ /dev/null @@ -1,57 +0,0 @@ -#pragma once - -#include "motordrivers.hpp" -#include "motorctl.hpp" -#include "joystick.hpp" - -#include "gpio_evaluateSwitch.hpp" -#include "buzzer.hpp" -#include "control.hpp" -#include "fan.hpp" -#include "http.hpp" -#include "auto.hpp" -#include "speedsensor.hpp" -#include "chairAdjust.hpp" - -// -////in IDLE mode: set loglevel for evaluatedJoystick to DEBUG -////and repeatedly read joystick e.g. for manually calibrating / testing joystick -////#define JOYSTICK_LOG_IN_IDLE -// -// -////TODO outsource global variables to e.g. global.cpp and only config options here? -// -////create global controlledMotor instances for both motors -//extern controlledMotor motorLeft; -//extern controlledMotor motorRight; -// -////create global joystic instance -//extern evaluatedJoystick joystick; -// -////create global evaluated switch instance for button next to joystick -//extern gpio_evaluatedSwitch buttonJoystick; -// -////create global buzzer object -//extern buzzer_t buzzer; -// -////create global control object -//extern controlledArmchair control; -// -////create global automatedArmchair object (for auto-mode) -//extern automatedArmchair_c armchair; -// -////create global httpJoystick object -////extern httpJoystick httpJoystickMain; -// -////configuration for fans / cooling -//extern fan_config_t configCooling; -// -////create global objects for measuring speed -//extern speedSensor speedLeft; -//extern speedSensor speedRight; -// -////create global objects for controlling the chair position -//extern cControlledRest legRest; -//extern cControlledRest backRest; -// -// \ No newline at end of file diff --git a/board_single/main/control.cpp b/board_single/main/control.cpp index 0eb617f..bf21154 100644 --- a/board_single/main/control.cpp +++ b/board_single/main/control.cpp @@ -9,13 +9,13 @@ extern "C" #include "wifi.h" } -#include "config.hpp" +#include "config.h" #include "control.hpp" #include "chairAdjust.hpp" -//used definitions moved from config.hpp: -//#define JOYSTICK_TEST +//used definitions moved from config.h: +//#define JOYSTICK_LOG_IN_IDLE //tag for logging @@ -70,7 +70,6 @@ controlledArmchair::controlledArmchair( void task_control( void * pvParameters ){ controlledArmchair * control = (controlledArmchair *)pvParameters; ESP_LOGW(TAG, "Initializing controlledArmchair and starting handle loop"); - //start handle loop (control object declared in config.hpp) control->startHandleLoop(); } @@ -95,13 +94,15 @@ void controlledArmchair::startHandleLoop() { //motorLeft->setTarget(commands.left.state, commands.left.duty); vTaskDelay(500 / portTICK_PERIOD_MS); #ifdef JOYSTICK_LOG_IN_IDLE - //get joystick data here (without using it) - //since loglevel is DEBUG, calculation details are output - joystick_l->getData(); + // get joystick data and log it + joystickData_t data joystick_l->getData(); + ESP_LOGI("JOYSTICK_LOG_IN_IDLE", "x=%.3f, y=%.3f, radius=%.3f, angle=%.3f, pos=%s, adcx=%d, adcy=%d", + data.x, data.y, data.radius, data.angle, + joystickPosStr[(int)data.position], + objects->joystick->getRawX(), objects->joystick->getRawY()); #endif - break; - - + break; + //------- handle JOYSTICK mode ------- case controlMode_t::JOYSTICK: vTaskDelay(50 / portTICK_PERIOD_MS); //get current joystick data with getData method of evaluatedJoystick @@ -113,6 +114,7 @@ void controlledArmchair::startHandleLoop() { // only generate when the stick data actually changed (e.g. stick stayed in center) if (stickData.x != stickDataLast.x || stickData.y != stickDataLast.y) { + resetTimeout(); //user input -> reset switch to IDLE timeout commands = joystick_generateCommandsDriving(stickData, &joystickGenerateCommands_config); // apply motor commands motorRight->setTarget(commands.right); @@ -121,27 +123,34 @@ void controlledArmchair::startHandleLoop() { else { vTaskDelay(20 / portTICK_PERIOD_MS); - ESP_LOGD(TAG, "analog joystick data unchanged at %s not updating commands", joystickPosStr[(int)stickData.position]); + ESP_LOGV(TAG, "analog joystick data unchanged at %s not updating commands", joystickPosStr[(int)stickData.position]); } break; + //------- handle MASSAGE mode ------- case controlMode_t::MASSAGE: vTaskDelay(10 / portTICK_PERIOD_MS); //--- read joystick --- - //only update joystick data when input not frozen - if (!freezeInput){ + // only update joystick data when input not frozen + stickDataLast = stickData; + if (!freezeInput) stickData = joystick_l->getData(); - } //--- generate motor commands --- - //pass joystick data from getData method of evaluatedJoystick to generateCommandsShaking function - commands = joystick_generateCommandsShaking(stickData); - //apply motor commands - motorRight->setTarget(commands.right); - motorLeft->setTarget(commands.left); + // only generate when the stick data actually changed (e.g. stick stayed in center) + if (stickData.x != stickDataLast.x || stickData.y != stickDataLast.y) + { + resetTimeout(); // user input -> reset switch to IDLE timeout + // pass joystick data from getData method of evaluatedJoystick to generateCommandsShaking function + commands = joystick_generateCommandsShaking(stickData); + // apply motor commands + motorRight->setTarget(commands.right); + motorLeft->setTarget(commands.left); + } break; + //------- handle HTTP mode ------- case controlMode_t::HTTP: //--- get joystick data from queue --- stickDataLast = stickData; @@ -152,6 +161,7 @@ void controlledArmchair::startHandleLoop() { //--- generate motor commands --- //only generate when the stick data actually changed (e.g. no new data recevied via http) if (stickData.x != stickDataLast.x || stickData.y != stickDataLast.y ){ + resetTimeout(); // user input -> reset switch to IDLE timeout // Note: timeout (no data received) is handled in getData method commands = joystick_generateCommandsDriving(stickData, &joystickGenerateCommands_config); @@ -166,6 +176,7 @@ void controlledArmchair::startHandleLoop() { break; + //------- handle AUTO mode ------- case controlMode_t::AUTO: vTaskDelay(20 / portTICK_PERIOD_MS); //generate commands @@ -206,26 +217,26 @@ void controlledArmchair::startHandleLoop() { break; + //------- handle ADJUST_CHAIR mode ------- case controlMode_t::ADJUST_CHAIR: vTaskDelay(100 / portTICK_PERIOD_MS); //--- read joystick --- + stickDataLast = stickData; stickData = joystick_l->getData(); - //--- idle motors --- - //commands = cmds_bothMotorsIdle; - now done once at mode change - //motorRight->setTarget(commands.right.state, commands.right.duty); - //motorLeft->setTarget(commands.left.state, commands.left.duty); //--- control armchair position with joystick input --- - controlChairAdjustment(joystick_l->getData(), legRest, backRest); + // dont update when stick data did not change + if (stickData.x != stickDataLast.x || stickData.y != stickDataLast.y) + { + resetTimeout(); // user input -> reset switch to IDLE timeout + controlChairAdjustment(joystick_l->getData(), legRest, backRest); + } break; + //------- handle MENU mode ------- case controlMode_t::MENU: - vTaskDelay(1000 / portTICK_PERIOD_MS); //nothing to do here, display task handles the menu - //--- idle motors --- - //commands = cmds_bothMotorsIdle; - now done once at mode change - //motorRight->setTarget(commands.right.state, commands.right.duty); - //motorLeft->setTarget(commands.left.state, commands.left.duty); + vTaskDelay(1000 / portTICK_PERIOD_MS); break; //TODO: add other modes here @@ -238,8 +249,7 @@ void controlledArmchair::startHandleLoop() { if (esp_log_timestamp() - timestamp_SlowLoopLastRun > 5000) { ESP_LOGV(TAG, "running slow loop... time since last run: %.1fs", (float)(esp_log_timestamp() - timestamp_SlowLoopLastRun)/1000); timestamp_SlowLoopLastRun = esp_log_timestamp(); - - //run function that detects timeout (switch to idle) + //run function that detects timeout (switch to idle, or notify "forgot to turn off") handleTimeout(); } @@ -310,59 +320,52 @@ void controlledArmchair::idleBothMotors(){ motorLeft->setTarget(cmd_motorIdle); } + //----------------------------------- //---------- resetTimeout ----------- //----------------------------------- void controlledArmchair::resetTimeout(){ //TODO mutex timestamp_lastActivity = esp_log_timestamp(); + ESP_LOGV(TAG, "timeout: activity detected, resetting timeout"); } - //------------------------------------ //---------- handleTimeout ----------- //------------------------------------ -//percentage the duty can vary since last timeout check and still counts as incative -//TODO: add this to config -float inactivityTolerance = 10; +// 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 30 * 60 * 1000 // beep every 30 minutes for someone to notice +// note: timeout durations are configured in config.cpp +void controlledArmchair::handleTimeout() +{ + uint32_t noActivityDurationMs = esp_log_timestamp() - timestamp_lastActivity; + // log current inactivity and configured timeouts + ESP_LOGD(TAG, "timeout check: last activity %dmin and %ds ago - timeout IDLE after %ds - notify after power on after %dh", + noActivityDurationMs / 1000 / 60, + noActivityDurationMs / 1000 % 60, + config.timeoutSwitchToIdleMs / 1000, + config.timeoutNotifyPowerStillOnMs / 1000); -//local function that checks whether two values differ more than a given tolerance -bool validateActivity(float dutyOld, float dutyNow, float tolerance){ - float dutyDelta = dutyNow - dutyOld; - if (fabs(dutyDelta) < tolerance) { - return false; //no significant activity detected - } else { - return true; //there was activity + // timeout to IDLE when not idling already + 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); + changeMode(controlMode_t::IDLE); } -} - -//function that evaluates whether there is no activity/change on the motor duty for a certain time. If so, a switch to IDLE is issued. - has to be run repeatedly in a slow interval -void controlledArmchair::handleTimeout(){ - //check for timeout only when not idling already - if (mode != controlMode_t::IDLE) { - //get current duty from controlled motor objects - float dutyLeftNow = motorLeft->getStatus().duty; - float dutyRightNow = motorRight->getStatus().duty; - - //activity detected on any of the two motors - if (validateActivity(dutyLeft_lastActivity, dutyLeftNow, inactivityTolerance) - || validateActivity(dutyRight_lastActivity, dutyRightNow, inactivityTolerance) - ){ - ESP_LOGD(TAG, "timeout check: [activity] detected since last check -> reset"); - //reset last duty and timestamp - dutyLeft_lastActivity = dutyLeftNow; - dutyRight_lastActivity = dutyRightNow; - resetTimeout(); - } - //no activity on any motor and msTimeout exceeded - else if (esp_log_timestamp() - timestamp_lastActivity > config.timeoutMs){ - ESP_LOGI(TAG, "timeout check: [TIMEOUT], no activity for more than %.ds -> switch to idle", config.timeoutMs/1000); - //toggle to idle mode - toggleIdle(); - } - else { - ESP_LOGD(TAG, "timeout check: [inactive], last activity %.1f s ago, timeout after %d s", (float)(esp_log_timestamp() - timestamp_lastActivity)/1000, config.timeoutMs/1000); + // 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 + else if (esp_log_timestamp() - timestamp_lastModeChange > config.timeoutNotifyPowerStillOnMs) + { + // beep in certain intervals + 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); + // TODO dont beep at certain times ranges (e.g. at night) + timestamp_lastTimeoutBeep = esp_log_timestamp(); + buzzer->beep(4, 100, 50); } } } @@ -385,6 +388,8 @@ void controlledArmchair::changeMode(controlMode_t modeNew) { //copy previous mode modePrevious = mode; + //store time changed (needed for timeout) + timestamp_lastModeChange = esp_log_timestamp(); ESP_LOGW(TAG, "=== changing mode from %s to %s ===", controlModeStr[(int)mode], controlModeStr[(int)modeNew]); @@ -448,9 +453,6 @@ void controlledArmchair::changeMode(controlMode_t modeNew) { ESP_LOGW(TAG, "switching to IDLE mode: turning both motors off, beep"); idleBothMotors(); buzzer->beep(1, 900, 0); -#ifdef JOYSTICK_LOG_IN_IDLE - esp_log_level_set("evaluatedJoystick", ESP_LOG_DEBUG); -#endif break; case controlMode_t::ADJUST_CHAIR: @@ -540,9 +542,9 @@ void controlledArmchair::toggleMode(controlMode_t modePrimary){ -//------------------------------- -//------ loadDecelDuration ------ -//------------------------------- +//----------------------------- +//-------- loadMaxDuty -------- +//----------------------------- // update local config value when maxDuty is stored in nvs void controlledArmchair::loadMaxDuty(void) { diff --git a/board_single/main/control.hpp b/board_single/main/control.hpp index 37ae943..55e122f 100644 --- a/board_single/main/control.hpp +++ b/board_single/main/control.hpp @@ -27,7 +27,8 @@ extern const char* controlModeStr[9]; typedef struct control_config_t { controlMode_t defaultMode; //default mode after startup and toggling IDLE //timeout options - uint32_t timeoutMs; //time of inactivity after which the mode gets switched to IDLE + 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; @@ -169,10 +170,10 @@ class controlledArmchair { //variable for slow loop uint32_t timestamp_SlowLoopLastRun = 0; - //variables for detecting timeout (switch to idle, after inactivity) - float dutyLeft_lastActivity = 0; - float dutyRight_lastActivity = 0; + //variables for detecting timeout (switch to idle, or notify "forgot to turn off" after inactivity + uint32_t timestamp_lastModeChange = 0; uint32_t timestamp_lastActivity = 0; + uint32_t timestamp_lastTimeoutBeep = 0; }; diff --git a/board_single/main/main.cpp b/board_single/main/main.cpp index 008d741..0a121c1 100644 --- a/board_single/main/main.cpp +++ b/board_single/main/main.cpp @@ -27,7 +27,6 @@ extern "C" #include "motorctl.hpp" //folder single_board -#include "config.hpp" #include "control.hpp" #include "button.hpp" #include "display.hpp" diff --git a/board_single/main/menu.cpp b/board_single/main/menu.cpp index 46100d4..4bce5a4 100644 --- a/board_single/main/menu.cpp +++ b/board_single/main/menu.cpp @@ -11,7 +11,6 @@ extern "C"{ #include "menu.hpp" #include "encoder.hpp" -#include "config.hpp" #include "motorctl.hpp" @@ -105,7 +104,7 @@ void item_calibrateJoystick_action(display_task_parameters_t *objects, SSD1306_t // save and next when button clicked, exit when long pressed if (xQueueReceive(objects->encoderQueue, &event, CALIBRATE_JOYSTICK_UPDATE_INTERVAL / portTICK_PERIOD_MS)) { - objects->control->resetTimeout(); + objects->control->resetTimeout(); // user input -> reset switch to IDLE timeout switch (event.type) { case RE_ET_BTN_CLICKED: @@ -215,7 +214,7 @@ void item_debugJoystick_action(display_task_parameters_t * objects, SSD1306_t * // exit when button pressed if (xQueueReceive(objects->encoderQueue, &event, DEBUG_JOYSTICK_UPDATE_INTERVAL / portTICK_PERIOD_MS)) { - objects->control->resetTimeout(); + objects->control->resetTimeout(); // user input -> reset switch to IDLE timeout switch (event.type) { case RE_ET_BTN_CLICKED: @@ -621,7 +620,7 @@ void handleMenu(display_task_parameters_t * objects, SSD1306_t *display) { // reset menu- and control-timeout on any encoder event lastActivity = esp_log_timestamp(); - objects->control->resetTimeout(); + objects->control->resetTimeout(); // user input -> reset switch to IDLE timeout switch (event.type) { case RE_ET_CHANGED: @@ -692,7 +691,7 @@ void handleMenu(display_task_parameters_t * objects, SSD1306_t *display) // wait for encoder event if (xQueueReceive(objects->encoderQueue, &event, QUEUE_TIMEOUT / portTICK_PERIOD_MS)) { - objects->control->resetTimeout(); + objects->control->resetTimeout(); // user input -> reset switch to IDLE timeout switch (event.type) { case RE_ET_CHANGED: @@ -732,7 +731,7 @@ void handleMenu(display_task_parameters_t * objects, SSD1306_t *display) } // reset menu- and control-timeout on any encoder event lastActivity = esp_log_timestamp(); - objects->control->resetTimeout(); + objects->control->resetTimeout(); // user input -> reset switch to IDLE timeout } break; } From 0d082a52d8b27ea84258b43f2c66368db26bf07f Mon Sep 17 00:00:00 2001 From: jonny_l480 Date: Fri, 1 Mar 2024 23:59:00 +0100 Subject: [PATCH 3/6] 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 --- board_single/main/control.cpp | 13 +++++----- board_single/main/control.hpp | 2 ++ board_single/main/display.cpp | 47 +++++++++++++++++++++++++++++++++-- board_single/main/display.hpp | 2 +- 4 files changed, 55 insertions(+), 9 deletions(-) diff --git a/board_single/main/control.cpp b/board_single/main/control.cpp index bf21154..1998aa2 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 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 void controlledArmchair::handleTimeout() { @@ -347,25 +347,26 @@ void controlledArmchair::handleTimeout() noActivityDurationMs / 1000 / 60, noActivityDurationMs / 1000 % 60, config.timeoutSwitchToIdleMs / 1000, - config.timeoutNotifyPowerStillOnMs / 1000); + config.timeoutNotifyPowerStillOnMs / 1000 / 60 / 60); // timeout to IDLE when not idling already 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); 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") // 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 - 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) timestamp_lastTimeoutBeep = esp_log_timestamp(); - buzzer->beep(4, 100, 50); + buzzer->beep(6, 100, 50); } } } diff --git a/board_single/main/control.hpp b/board_single/main/control.hpp index 55e122f..daf40f0 100644 --- a/board_single/main/control.hpp +++ b/board_single/main/control.hpp @@ -97,6 +97,8 @@ class controlledArmchair { void setMaxDuty(float maxDutyNew) { writeMaxDuty(maxDutyNew); }; float getMaxDuty() const {return joystickGenerateCommands_config.maxDuty; }; + uint32_t getInactivityDurationMs() {return esp_log_timestamp() - timestamp_lastActivity;}; + private: //--- functions --- diff --git a/board_single/main/display.cpp b/board_single/main/display.cpp index c664e06..c8806e1 100644 --- a/board_single/main/display.cpp +++ b/board_single/main/display.cpp @@ -234,7 +234,7 @@ float getBatteryPercent() //shows overview on entire display: //Battery percentage, voltage, current, mode, rpm, speed #define STATUS_SCREEN_OVERVIEW_UPDATE_INTERVAL 500 -void showStatusScreenOverview(display_task_parameters_t * objects) +void showStatusScreenOverview(display_task_parameters_t *objects) { //-- battery percentage -- // 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); } - //############################ //##### 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 #### //######################## @@ -363,6 +398,8 @@ void display_selectStatusPage(displayStatusPage_t newStatusPage){ //======= display task ======= //============================ // TODO: separate task for each loop? +#define DISPLAY_IDLE_TIMEOUT_SCREENSAVER_MS 15*60*1000 + void display_task(void *pvParameters) { ESP_LOGW(TAG, "Initializing display and starting handle loop"); @@ -403,7 +440,13 @@ void display_task(void *pvParameters) case STATUS_SCREEN_MOTORS: showStatusScreenMotors(objects); 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 } diff --git a/board_single/main/display.hpp b/board_single/main/display.hpp index 13c554a..776861b 100644 --- a/board_single/main/display.hpp +++ b/board_single/main/display.hpp @@ -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) -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) float getBatteryVoltage(); From 3514dd6bf208b7ec635b6d9901fffab2db652eb6 Mon Sep 17 00:00:00 2001 From: jonny Date: Sat, 2 Mar 2024 13:01:29 +0100 Subject: [PATCH 4/6] 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; From fbae02b328c628a42734971cdd555b6d2f702fb1 Mon Sep 17 00:00:00 2001 From: jonny Date: Sun, 3 Mar 2024 11:47:32 +0100 Subject: [PATCH 5/6] Add software-scrolling to Screensaver Compared to the white-text display used for testing, the actually used blue-text display does not support hardware-scrolling (the text was just static) display: - Add code that continously scrolls screensaver text 1 character to the right and wraps the truncating part to the start - Add option to enable/disable hardware-scrolling if available --- board_single/main/display.cpp | 63 ++++++++++++++++++++++++++--------- 1 file changed, 48 insertions(+), 15 deletions(-) diff --git a/board_single/main/display.cpp b/board_single/main/display.cpp index c73491d..7c3e875 100644 --- a/board_single/main/display.cpp +++ b/board_single/main/display.cpp @@ -15,6 +15,9 @@ extern "C"{ // continously vary display contrast from 0 to 250 in OVERVIEW status screen //#define BRIGHTNESS_TEST +// if display and driver support hardware scrolling the SCREENSAVER status-screen will be smoother: +//#define HARDWARE_SCROLL_AVAILABLE + //=== variables === // every function can access the display configuration from config.cpp @@ -349,45 +352,71 @@ void showStatusScreenMotors(display_task_parameters_t *objects) } -//############################### -//#### showScreen Sreensaver #### -//############################### -// 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 +// ################################ +// #### showScreen Screensaver #### +// ################################ +// 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 +#define DISPLAY_HORIZONTAL_CHARACTER_COUNT 16 +#define DISPLAY_VERTICAL_LINE_COUNT 8 void showStatusScreenScreensaver(display_task_parameters_t *objects) { - // note: scrolling is enabled at screen change (display_selectStatusPage()) + //-- variables for line rotation -- static int msPassed = 0; static int currentLine = 0; static bool lineChanging = false; // clear display once when rotating to next line - if (lineChanging) { + if (lineChanging) + { ssd1306_clear_screen(&dev, false); lineChanging = false; } + //-- print 2 lines scrolling horizontally -- +#ifdef HARDWARE_SCROLL_AVAILABLE // when display supports hardware scrolling -> only the content has to be updated + // note: scrolling is enabled at screen change (display_selectStatusPage()) // 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()); - + // note: scrolling is disabled at screen change (display_selectStatusPage()) +#else // custom implementation to scroll the text 1 character to the right every iteration (also wraps over the end to beginning) + static int offset = DISPLAY_HORIZONTAL_CHARACTER_COUNT; + char buf1[64], buf2[64]; + // scroll text left to right (taken window of the string moves to the left => offset 16->0, 16->0 ...) + offset -= 1; + if (offset < 0) + offset = DISPLAY_HORIZONTAL_CHARACTER_COUNT - 1; // 0 = no crop -> start over with crop + // note: these strings have to be symetrical and 2x display character count long + snprintf(buf1, 64, "IDLE since: IDLE since: "); + snprintf(buf2, 64, "%.1fh, B:%02.0f%% %.1fh, B:%02.0f%% ", + (float)objects->control->getInactivityDurationMs() / 1000 / 60 / 60, + getBatteryPercent(), + (float)objects->control->getInactivityDurationMs() / 1000 / 60 / 60, + getBatteryPercent()); + // print strings on display while limiting to certain window (ignore certain count of characters at start) + displayTextLine(&dev, currentLine, false, false, "%s", buf1 + offset); + displayTextLine(&dev, currentLine + 1, false, false, "%s", buf2 + offset); +#endif + //-- handle line rotation -- // 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 + if (++currentLine >= DISPLAY_VERTICAL_LINE_COUNT - 1) // rotate to next line currentLine = 0; - lineChanging = true; //clear screen in next run + lineChanging = true; // clear screen in next run } - // wait update-update interval and increment passed time after each run + //-- wait update interval -- + // wait 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()) } + //######################## //#### showStartupMsg #### //######################## @@ -415,9 +444,11 @@ 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); +#ifdef HARDWARE_SCROLL_AVAILABLE + case STATUS_SCREEN_SCREENSAVER: + ssd1306_hardware_scroll(&dev, SCROLL_STOP); // disable scrolling when exiting screensaver break; +#endif default: break; } @@ -429,8 +460,10 @@ void display_selectStatusPage(displayStatusPage_t newStatusPage) switch (selectedStatusPage) { case STATUS_SCREEN_SCREENSAVER: - ssd1306_clear_screen(&dev, false); + ssd1306_clear_screen(&dev, false); // clear screen when switching +#ifdef HARDWARE_SCROLL_AVAILABLE ssd1306_hardware_scroll(&dev, SCROLL_RIGHT); +#endif break; default: break; From 77ba81e99634954874e5548993faabb07cc676e5 Mon Sep 17 00:00:00 2001 From: jonny Date: Sun, 3 Mar 2024 22:04:27 +0100 Subject: [PATCH 6/6] WiFi on in HTTP-mode only, Add batt-threshold to timeout As initially planned starting wifi only when needed, at change to HTTP mode. Tested this on breakout board: ESP32 is not getting hot anymore! Also optimized timeout: only notify "forgot to turn off" when battery is below certain threshold (e.g. disable when charger is connected over night) main: - dont start wifi at startup anymore - remove double-initialization of nvs already initialized in wifi.c, thus removed some code - optimize comments, logging control: - timeout "forgot to turn off": Add battery threshold - start/stop wifi when switching to/from HTTP mode wifi: - split init function to separate functions for NVS and NETIF (more clear in main) - optimize log output at nvs init (moved from main) - rename start/stop functions, formatting --- board_single/main/control.cpp | 22 +++++++-- board_single/main/display.hpp | 3 ++ board_single/main/main.cpp | 35 ++++++--------- common/wifi.c | 84 ++++++++++++++++++++++------------- common/wifi.h | 20 ++++----- 5 files changed, 100 insertions(+), 64 deletions(-) diff --git a/board_single/main/control.cpp b/board_single/main/control.cpp index 9d9c914..7ebd887 100644 --- a/board_single/main/control.cpp +++ b/board_single/main/control.cpp @@ -12,6 +12,7 @@ extern "C" #include "config.h" #include "control.hpp" #include "chairAdjust.hpp" +#include "display.hpp" // needed for getBatteryPercent() //used definitions moved from config.h: @@ -338,6 +339,7 @@ void controlledArmchair::resetTimeout(){ // 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 5 * 60 * 1000 // beep every 5 minutes for someone to notice +#define TIMEOUT_POWER_STILL_ON_BATTERY_THRESHOLD_PERCENT 96 // only notify/beep when below certain percentage (prevent beeping when connected to charger) // note: timeout durations are configured in config.cpp void controlledArmchair::handleTimeout() { @@ -349,6 +351,7 @@ void controlledArmchair::handleTimeout() config.timeoutSwitchToIdleMs / 1000, config.timeoutNotifyPowerStillOnMs / 1000 / 60 / 60); + // -- timeout switch to IDLE -- // timeout to IDLE when not idling already if (mode != controlMode_t::IDLE && noActivityDurationMs > config.timeoutSwitchToIdleMs) { @@ -356,15 +359,18 @@ void controlledArmchair::handleTimeout() changeMode(controlMode_t::IDLE); //TODO switch to previous status-screen when activity detected } + + // -- timeout notify "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 - else if ((esp_log_timestamp() - timestamp_lastModeChange) > config.timeoutNotifyPowerStillOnMs) + // also battery charge-level has to be below certain threshold to prevent beeping in case connected to charger + // note: ignores user input while in IDLE (e.g. encoder rotation) + else if ((esp_log_timestamp() - timestamp_lastModeChange) > config.timeoutNotifyPowerStillOnMs && getBatteryPercent() < TIMEOUT_POWER_STILL_ON_BATTERY_THRESHOLD_PERCENT) { // beep in certain intervals if ((esp_log_timestamp() - timestamp_lastTimeoutBeep) > TIMEOUT_POWER_STILL_ON_BEEP_INTERVAL_MS) { 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 time ranges (e.g. at night) timestamp_lastTimeoutBeep = esp_log_timestamp(); buzzer->beep(6, 100, 50); } @@ -407,6 +413,11 @@ void controlledArmchair::changeMode(controlMode_t modeNew) { buzzer->beep(1,200,100); break; + case controlMode_t::HTTP: + ESP_LOGW(TAG, "switching from HTTP mode -> stopping wifi-ap"); + wifi_stop_ap(); + break; + case controlMode_t::MASSAGE: ESP_LOGW(TAG, "switching from MASSAGE mode -> restoring fading, reset frozen input"); //TODO: fix issue when downfading was disabled before switching to massage mode - currently it gets enabled again here... @@ -454,6 +465,11 @@ void controlledArmchair::changeMode(controlMode_t modeNew) { buzzer->beep(1, 900, 0); break; + case controlMode_t::HTTP: + ESP_LOGW(TAG, "switching to HTTP mode -> starting wifi-ap"); + wifi_start_ap(); + break; + case controlMode_t::ADJUST_CHAIR: ESP_LOGW(TAG, "switching to ADJUST_CHAIR mode: turning both motors off, beep"); idleBothMotors(); diff --git a/board_single/main/display.hpp b/board_single/main/display.hpp index 4646dc0..a4fb8c6 100644 --- a/board_single/main/display.hpp +++ b/board_single/main/display.hpp @@ -59,6 +59,9 @@ typedef enum displayStatusPage_t {STATUS_SCREEN_OVERVIEW=0, STATUS_SCREEN_SPEED, // get precise battery voltage (using lookup table) float getBatteryVoltage(); +// get battery charge level in percent (using lookup table as discharge curve) +float getBatteryPercent(); + // function to select one of the defined status screens which are shown on display when not in MENU mode void display_selectStatusPage(displayStatusPage_t newStatusPage); diff --git a/board_single/main/main.cpp b/board_single/main/main.cpp index 0a121c1..57d10c5 100644 --- a/board_single/main/main.cpp +++ b/board_single/main/main.cpp @@ -107,7 +107,6 @@ nvs_handle_t nvsHandle; //================================= //initialize spi flash filesystem (used for webserver) void init_spiffs(){ - ESP_LOGW(TAG, "initializing spiffs..."); esp_vfs_spiffs_conf_t esp_vfs_spiffs_conf = { .base_path = "/spiffs", .partition_label = NULL, @@ -194,35 +193,29 @@ extern "C" void app_main(void) { gpio_set_direction(GPIO_NUM_17, GPIO_MODE_OUTPUT); gpio_set_level(GPIO_NUM_17, 1); - //--- initialize nvs-flash and netif (needed for wifi) --- - ESP_LOGW(TAG,"initializing wifi..."); - wifi_initNvs_initNetif(); + //--- initialize nvs-flash and netif --- + ESP_LOGW(TAG,"initializing NVS..."); + wifi_initNvs(); //needed for wifi and persistent config variables + ESP_LOGW(TAG,"initializing NETIF..."); + wifi_initNetif(); // needed for wifi //--- initialize spiffs --- - init_spiffs(); + ESP_LOGW(TAG, "initializing SPIFFS..."); + init_spiffs(); // used by httpd server //--- initialize and start wifi --- - ESP_LOGW(TAG,"starting wifi..."); - //wifi_init_client(); //connect to existing wifi - wifi_init_ap(); //start access point - ESP_LOGD(TAG,"done starting wifi"); + // Note: now started only when switching to HTTP mode in control.cpp + // ESP_LOGW(TAG,"starting wifi..."); + // wifi_start_client(); //connect to existing wifi (dropped) + // wifi_start_ap(); //start access point //--- initialize encoder --- const QueueHandle_t encoderQueue = encoder_init(&encoder_config); - //--- initialize nvs-flash --- (for persistant config values) - ESP_LOGW(TAG, "initializing nvs-flash..."); - esp_err_t err = nvs_flash_init(); - if (err == ESP_ERR_NVS_NO_FREE_PAGES || err == ESP_ERR_NVS_NEW_VERSION_FOUND) - { - ESP_LOGE(TAG, "NVS truncated -> deleting flash"); - // Retry nvs_flash_init - ESP_ERROR_CHECK(nvs_flash_erase()); - err = nvs_flash_init(); - } - ESP_ERROR_CHECK(err); //--- open nvs-flash --- - err = nvs_open("storage", NVS_READWRITE, &nvsHandle); + // note: nvs already initialized in wifi_initNvs() + ESP_LOGW(TAG, "opening NVS-handle..."); + esp_err_t err = nvs_open("storage", NVS_READWRITE, &nvsHandle); // this handle is passed to all tasks for accessing nvs if (err != ESP_OK) ESP_LOGE(TAG, "Error (%s) opening NVS handle!\n", esp_err_to_name(err)); diff --git a/common/wifi.c b/common/wifi.c index 27eb3db..557abfc 100644 --- a/common/wifi.c +++ b/common/wifi.c @@ -21,26 +21,45 @@ static const char *TAG = "wifi"; static esp_event_handler_instance_t instance_any_id; -//============================================ -//============ init nvs and netif ============ -//============================================ -//initialize nvs-flash and netif (needed for both AP and CLIENT) +//########################################## +//############ common functions ############ +//########################################## + +//============================ +//========= init nvs ========= +//============================ +//initialize nvs-flash (needed for both AP and CLIENT) //has to be run once at startup -void wifi_initNvs_initNetif(){ +void wifi_initNvs(){ //Initialize NVS (needed for wifi) - esp_err_t ret = nvs_flash_init(); - if (ret == ESP_ERR_NVS_NO_FREE_PAGES || ret == ESP_ERR_NVS_NEW_VERSION_FOUND) { - ESP_ERROR_CHECK(nvs_flash_erase()); - ret = nvs_flash_init(); - } + esp_err_t err = nvs_flash_init(); + if (err == ESP_ERR_NVS_NO_FREE_PAGES || err == ESP_ERR_NVS_NEW_VERSION_FOUND) + { + ESP_LOGE(TAG, "NVS truncated -> deleting flash"); + // Retry nvs_flash_init + ESP_ERROR_CHECK(nvs_flash_erase()); + err = nvs_flash_init(); + } + ESP_ERROR_CHECK(err); +} + + +//============================== +//========= init netif ========= +//============================== +//initialize netif (needed for both AP and CLIENT) +//has to be run once at startup +void wifi_initNetif(){ ESP_ERROR_CHECK(esp_netif_init()); ESP_ERROR_CHECK(esp_event_loop_create_default()); } -//=========================================== -//============ init access point ============ -//=========================================== + + +//############################################ +//############### access point ############### +//############################################ //-------------------------------------------- //------ configuration / declarations -------- @@ -66,10 +85,12 @@ static void wifi_event_handler_ap(void* arg, esp_event_base_t event_base, } } -//----------------------- -//------ init ap -------- -//----------------------- -void wifi_init_ap(void) + + +//======================== +//====== start AP ======== +//======================== +void wifi_start_ap(void) { ap = esp_netif_create_default_wifi_ap(); @@ -107,9 +128,9 @@ void wifi_init_ap(void) //============================= -//========= deinit AP ========= +//========== stop AP ========== //============================= -void wifi_deinit_ap(void) +void wifi_stop_ap(void) { ESP_ERROR_CHECK(esp_event_handler_instance_unregister(WIFI_EVENT, ESP_EVENT_ANY_ID, instance_any_id)); ESP_ERROR_CHECK(esp_event_handler_instance_unregister(WIFI_EVENT, ESP_EVENT_ANY_ID, instance_any_id)); @@ -123,9 +144,9 @@ void wifi_deinit_ap(void) -//=========================================== -//=============== init client =============== -//=========================================== +//########################################## +//################# client ################# +//########################################## //-------------------------------------------- //------ configuration / declarations -------- @@ -168,10 +189,13 @@ static void event_handler(void* arg, esp_event_base_t event_base, xEventGroupSetBits(s_wifi_event_group, WIFI_CONNECTED_BIT); } } -//--------------------------- -//------ init client -------- -//--------------------------- -void wifi_init_client(void) + + + +//=========================== +//====== init client ======== +//=========================== +void wifi_start_client(void) { s_wifi_event_group = xEventGroupCreate(); sta = esp_netif_create_default_wifi_sta(); @@ -249,10 +273,10 @@ void wifi_init_client(void) -//================================= -//========= deinit client ========= -//================================= -void wifi_deinit_client(void) +//=============================== +//========= stop client ========= +//=============================== +void wifi_stop_client(void) { /* The event will not be processed after unregister */ ESP_ERROR_CHECK(esp_event_handler_instance_unregister(IP_EVENT, IP_EVENT_STA_GOT_IP, instance_got_ip)); diff --git a/common/wifi.h b/common/wifi.h index 39eecee..d65a3bf 100644 --- a/common/wifi.h +++ b/common/wifi.h @@ -3,20 +3,20 @@ //TODO: currently wifi names and passwords are configured in wifi.c -> move this to config? //initialize nvs-flash and netif (needed for both AP and CLIENT) -//has to be run once at startup -//Note: this cant be put in wifi_init functions because this may not be in deinit functions -void wifi_initNvs_initNetif(); +//both functions have to be run once at startup +void wifi_initNvs(); +void wifi_initNetif(); -//function to start an access point -void wifi_init_ap(void); -//function to disable/deinit access point -void wifi_deinit_ap(void); +//function to start an access point (config in wifi.c) +void wifi_start_ap(void); +//function to disable/stop access point +void wifi_stop_ap(void); -//function to connect to existing wifi network -void wifi_init_client(void); +//function to connect to existing wifi network (config in wifi.c) +void wifi_start_client(void); //function to disable/deinit client -void wifi_deinit_client(void); +void wifi_stop_client(void);