Add Display-notification, default position, Push+scroll

This commit is contained in:
jonny 2024-09-04 23:04:54 +02:00
parent 43cc3ce3e0
commit 23cf6e6f3e
6 changed files with 75 additions and 35 deletions

View File

@ -186,7 +186,7 @@ void buttonCommands::action (uint8_t count, bool lastPressLong){
void buttonCommands::startHandleLoop() void buttonCommands::startHandleLoop()
{ {
//-- variables -- //-- variables --
bool isPressed = false; static bool isPressed = false;
static rotary_encoder_event_t event; // store event data static rotary_encoder_event_t event; // store event data
int rotateCount = 0; // temporary count clicks encoder was rotated int rotateCount = 0; // temporary count clicks encoder was rotated
// int count = 0; (from class) // int count = 0; (from class)
@ -221,27 +221,39 @@ void buttonCommands::startHandleLoop()
ESP_LOGD(TAG, "Button released"); ESP_LOGD(TAG, "Button released");
isPressed = false; // rest stored state isPressed = false; // rest stored state
break; break;
case RE_ET_CHANGED: // scroll through status pages when simply rotating encoder case RE_ET_CHANGED:
if (isPressed){
if (event.diff > 0) /////### scroll through status pages when PRESSED + ROTATED ###
legRest->setTargetPercent(legRest->getTargetPercent() - 5); ///if (event.diff > 0)
//TODO display notification /// display_rotateStatusPage(true, true); // select NEXT status screen, stay at last element (dont rotate to first)
///else
/// display_rotateStatusPage(false, true); // select PREVIOUS status screen, stay at first element (dont rotate to last)
//### adjust back support when PRESSED + ROTATED ###
if (event.diff > 0)
backRest->setTargetPercent(backRest->getTargetPercent() - 5);
else
backRest->setTargetPercent(backRest->getTargetPercent() + 5);
// show temporary notification on display
char buf[8];
snprintf(buf, 8, "%.1f%%", backRest->getTargetPercent());
display_showNotification(1000, "moving Rest:", "LEG", buf);
}
//### adjust leg support when ROTATED ###
else else
legRest->setTargetPercent(legRest->getTargetPercent() + 5); {
// increment target position each click
//## switch status page with rotate - disabled // TODO: ignore first 2 clicks
///rotateCount++; // TODO: control back rest too?
///if (rotateCount >= 2) // at least two rotate-clicks necessary for one page switch if (event.diff > 0)
///{ legRest->setTargetPercent(legRest->getTargetPercent() - 5);
/// if (event.diff > 0) else
/// display_rotateStatusPage(true, true); // select NEXT status screen, stay at last element (dont rotate to first) legRest->setTargetPercent(legRest->getTargetPercent() + 5);
/// else // show temporary notification on display
/// display_rotateStatusPage(false, true); // select PREVIOUS status screen, stay at first element (dont rotate to last) char buf[8];
/// rotateCount = 0; snprintf(buf, 8, "%.1f%%", legRest->getTargetPercent());
/// buzzer->beep(1, 90, 0); display_showNotification(1000, "moving Rest:", "LEG", buf);
///} }
///else buzzer->beep(1, 90, 0);
/// buzzer->beep(1, 65, 0);
break; break;
case RE_ET_BTN_LONG_PRESSED: case RE_ET_BTN_LONG_PRESSED:
case RE_ET_BTN_CLICKED: case RE_ET_BTN_CLICKED:
@ -251,16 +263,6 @@ void buttonCommands::startHandleLoop()
} }
else // timeout (no event received within TIMEOUT) else // timeout (no event received within TIMEOUT)
{ {
//## switch status page with rotate - disabled
// switch back to default status screen in case less than 2 rotate-clicks received
///if (rotateCount != 0)
///{
/// rotateCount = 0;
/// display_selectStatusPage(STATUS_SCREEN_OVERVIEW);
/// //TODO only change/beep if not already at STATUS_SCREEN_OVERVIEW
/// //buzzer->beep(1, 100, 0);
///}
// encoder was pressed // encoder was pressed
if (count > 0) if (count > 0)
{ {

View File

@ -23,6 +23,11 @@ extern "C"{
// every function can access the display configuration from config.cpp // every function can access the display configuration from config.cpp
static display_config_t displayConfig; static display_config_t displayConfig;
// communicate with display task to block+restore display writing, when a notification was triggered
uint32_t timestampNotificationStop = 0;
bool notificationIsActive = false;
//-------------------------- //--------------------------
//------- getVoltage ------- //------- getVoltage -------
@ -584,6 +589,25 @@ void handleStatusScreen(display_task_parameters_t *objects)
} }
//==============================
//====== showNotification ======
//==============================
// trigger custom notification that is shown in any mode for set amount of time
void display_showNotification(uint32_t showDurationMs, const char *line1, const char *line2Large, const char *line3Large)
{
timestampNotificationStop = esp_log_timestamp() + showDurationMs;
notificationIsActive = true;
// void displayTextLineCentered(SSD1306_t *display, int line, bool isLarge, bool inverted, const char *format, ...);
displayTextLineCentered(&dev, 0, false, false, "%s", line1);
displayTextLineCentered(&dev, 1, true, false, "%s", line2Large);
displayTextLineCentered(&dev, 4, true, false, "%s", line3Large);
displayTextLine(&dev, 7, false, false, " ");
ESP_LOGW(TAG, "start showing notification '%s' '%s' for %d ms", line1, line2Large, showDurationMs);
}
//============================ //============================
//======= display task ======= //======= display task =======
//============================ //============================
@ -606,6 +630,15 @@ void display_task(void *pvParameters)
// repeatedly update display with content depending on current mode // repeatedly update display with content depending on current mode
while (1) while (1)
{ {
// dont update anything when a notification is active + check timeout
if (notificationIsActive){
if (esp_log_timestamp() >= timestampNotificationStop)
notificationIsActive = false;
vTaskDelay(100 / portTICK_PERIOD_MS);
continue;
}
//update display according to current control mode
switch (objects->control->getCurrentMode()) switch (objects->control->getCurrentMode())
{ {
case controlMode_t::MENU_SETTINGS: case controlMode_t::MENU_SETTINGS:

View File

@ -67,6 +67,9 @@ void display_selectStatusPage(displayStatusPage_t newStatusPage);
// select next/previous status screen to be shown, when noRotate is set is stays at first/last screen // select next/previous status screen to be shown, when noRotate is set is stays at first/last screen
void display_rotateStatusPage(bool reverseDirection = false, bool noRotate = false); void display_rotateStatusPage(bool reverseDirection = false, bool noRotate = false);
// show content for certain time in any mode
void display_showNotification(uint32_t showDurationMs, const char *line1, const char * line2Large, const char * line3Large );
//task that inititialized the display, displays welcome message //task that inititialized the display, displays welcome message
//and releatedly updates the display with certain content //and releatedly updates the display with certain content
void display_task( void * pvParameters ); void display_task( void * pvParameters );

View File

@ -161,9 +161,9 @@ void createObjects()
buzzer = new buzzer_t(GPIO_NUM_12, 1); buzzer = new buzzer_t(GPIO_NUM_12, 1);
// create objects for controlling the chair position // create objects for controlling the chair position
// gpio_up, gpio_down, name // gpio_up, gpio_down, travelDuration, name, defaultPosition
legRest = new cControlledRest(GPIO_NUM_2, GPIO_NUM_15, 12000, "legRest"); legRest = new cControlledRest(GPIO_NUM_2, GPIO_NUM_15, 12000, "legRest");
backRest = new cControlledRest(GPIO_NUM_16, GPIO_NUM_4, 12000, "backRest"); backRest = new cControlledRest(GPIO_NUM_16, GPIO_NUM_4, 12000, "backRest", 100); //default position "100% up"
// create control object (control.hpp) // create control object (control.hpp)
// with configuration from config.cpp // with configuration from config.cpp

View File

@ -24,10 +24,12 @@ static const char * TAG = "chair-adjustment";
//============================= //=============================
//======== constructor ======== //======== constructor ========
//============================= //=============================
cControlledRest::cControlledRest(gpio_num_t gpio_up_f, gpio_num_t gpio_down_f, uint32_t travelDurationMs, const char * name_f):travelDuration(travelDurationMs){ cControlledRest::cControlledRest(gpio_num_t gpio_up_f, gpio_num_t gpio_down_f, uint32_t travelDurationMs, const char * name_f, float defaultPosition):travelDuration(travelDurationMs){
strcpy(name, name_f); strcpy(name, name_f);
gpio_up = gpio_up_f; gpio_up = gpio_up_f;
gpio_down = gpio_down_f; gpio_down = gpio_down_f;
positionNow = defaultPosition;
positionTarget = positionNow;
init(); init();
} }

View File

@ -18,7 +18,7 @@ extern const char* restStateStr[];
//2 instances will be created one for back and one for leg rest //2 instances will be created one for back and one for leg rest
class cControlledRest { class cControlledRest {
public: public:
cControlledRest(gpio_num_t gpio_up, gpio_num_t gpio_down, uint32_t travelDurationMs, const char *name); cControlledRest(gpio_num_t gpio_up, gpio_num_t gpio_down, uint32_t travelDurationMs, const char *name, float defaultPosition = 0);
void setState(restState_t targetState); void setState(restState_t targetState);
float getPercent(); //TODO update position first float getPercent(); //TODO update position first
void setTargetPercent(float targetPercent); void setTargetPercent(float targetPercent);