Control: Rework button functionality - standalone

- re-enable button functionality
- make button independent of controlledmotor objects
- disable 2x eject support command
- disable 8x toggle fading command
This commit is contained in:
jonny_jr9 2023-08-30 20:09:19 +02:00
parent 455b6f0456
commit d25f8cd381
3 changed files with 248 additions and 246 deletions

View File

@ -1,3 +1,4 @@
#include "auto.hpp"
extern "C"
{
#include <stdio.h>
@ -17,200 +18,205 @@ static const char * TAG = "button";
//FIXME needs rework, motorleft/right objectn not available anymore
// //-----------------------------
// //-------- constructor --------
// //-----------------------------
// buttonCommands::buttonCommands(gpio_evaluatedSwitch * button_f, evaluatedJoystick * joystick_f, controlledArmchair * control_f, buzzer_t * buzzer_f, controlledMotor * motorLeft_f, controlledMotor * motorRight_f){
// //copy object pointers
// button = button_f;
// joystick = joystick_f;
// control = control_f;
// buzzer = buzzer_f;
// motorLeft = motorLeft_f;
// motorRight = motorRight_f;
// //TODO declare / configure evaluatedSwitch here instead of config (unnecessary that button object is globally available - only used here)?
// }
//
//
//
// //----------------------------
// //--------- action -----------
// //----------------------------
// //function that runs commands depending on a count value
// void buttonCommands::action (uint8_t count, bool lastPressLong){
// //--- variable declarations ---
// bool decelEnabled; //for different beeping when toggling
// commandSimple_t cmds[8]; //array for commands for automatedArmchair
//
// //--- get joystick position ---
// //joystickData_t stickData = joystick->getData();
//
// //--- actions based on count ---
// switch (count){
// //no such command
// default:
// ESP_LOGE(TAG, "no command for count=%d defined", count);
// buzzer->beep(3, 400, 100);
// break;
//
// case 1:
// //restart contoller when 1x long pressed
// if (lastPressLong){
// ESP_LOGW(TAG, "RESTART");
// buzzer->beep(1,1000,1);
// vTaskDelay(1000 / portTICK_PERIOD_MS);
// esp_restart();
// return;
// }
//
// ESP_LOGW(TAG, "cmd %d: sending button event to control task", count);
// //-> define joystick center or toggle freeze input (executed in control task)
// control->sendButtonEvent(count); //TODO: always send button event to control task (not just at count=1) -> control.cpp has to be changed
// break;
// case 2:
// //run automatic commands to lift leg support when pressed 1x short 1x long
// if (lastPressLong){
// //define commands
// cmds[0] =
// {
// .motorCmds = {
// .left = {motorstate_t::REV, 90},
// .right = {motorstate_t::REV, 90}
// },
// .msDuration = 1200,
// .fadeDecel = 800,
// .fadeAccel = 1300,
// .instruction = auto_instruction_t::NONE
// };
// cmds[1] =
// {
// .motorCmds = {
// .left = {motorstate_t::FWD, 70},
// .right = {motorstate_t::FWD, 70}
// },
// .msDuration = 70,
// .fadeDecel = 0,
// .fadeAccel = 300,
// .instruction = auto_instruction_t::NONE
// };
// cmds[2] =
// {
// .motorCmds = {
// .left = {motorstate_t::IDLE, 0},
// .right = {motorstate_t::IDLE, 0}
// },
// .msDuration = 10,
// .fadeDecel = 800,
// .fadeAccel = 1300,
// .instruction = auto_instruction_t::SWITCH_JOYSTICK_MODE
// };
//
// //send commands to automatedArmchair command queue
// armchair.addCommands(cmds, 3);
//
// //change mode to AUTO
// control->changeMode(controlMode_t::AUTO);
// return;
// }
//
// //toggle idle when 2x pressed
// ESP_LOGW(TAG, "cmd %d: toggle IDLE", count);
// control->toggleIdle(); //toggle between idle and previous/default mode
// break;
//
//
// case 3:
// ESP_LOGW(TAG, "cmd %d: switch to JOYSTICK", count);
// control->changeMode(controlMode_t::JOYSTICK); //switch to JOYSTICK mode
// break;
//
// case 4:
// 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
// break;
//
// case 6:
// 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
// break;
//
// case 8:
// //toggle deceleration fading between on and off
// decelEnabled = motorLeft->toggleFade(fadeType_t::DECEL);
// motorRight->toggleFade(fadeType_t::DECEL);
// ESP_LOGW(TAG, "cmd %d: toggle deceleration fading to: %d", count, (int)decelEnabled);
// if (decelEnabled){
// buzzer->beep(3, 60, 50);
// } else {
// buzzer->beep(1, 1000, 1);
// }
// break;
//
// case 12:
// ESP_LOGW(TAG, "cmd %d: sending button event to control task", count);
// //-> toggle altStickMapping (executed in control task)
// control->sendButtonEvent(count); //TODO: always send button event to control task (not just at count=1)?
// break;
//
// }
// }
//
//
//
//
// //-----------------------------
// //------ startHandleLoop ------
// //-----------------------------
// //this function has to be started once in a separate task
// //repeatedly evaluates and processes button events then takes the corresponding action
// void buttonCommands::startHandleLoop() {
//
// while(1) {
// vTaskDelay(20 / portTICK_PERIOD_MS);
// //run handle function of evaluatedSwitch object
// button->handle();
//
// //--- count button presses and run action ---
// switch(state) {
// case inputState_t::IDLE: //wait for initial button press
// if (button->risingEdge) {
// count = 1;
// buzzer->beep(1, 65, 0);
// timestamp_lastAction = esp_log_timestamp();
// state = inputState_t::WAIT_FOR_INPUT;
// ESP_LOGI(TAG, "first button press detected -> waiting for further events");
// }
// break;
//
// case inputState_t::WAIT_FOR_INPUT: //wait for further presses
// //button pressed again
// if (button->risingEdge){
// count++;
// buzzer->beep(1, 65, 0);
// timestamp_lastAction = esp_log_timestamp();
// ESP_LOGI(TAG, "another press detected -> count=%d -> waiting for further events", count);
// }
// //timeout
// else if (esp_log_timestamp() - timestamp_lastAction > 1000) {
// state = inputState_t::IDLE;
// buzzer->beep(count, 50, 50);
// //TODO: add optional "bool wait" parameter to beep function to delay until finished beeping
// ESP_LOGI(TAG, "timeout - running action function for count=%d", count);
// //--- run action function ---
// //check if still pressed
// bool lastPressLong = false;
// if (button->state == true){
// //run special case when last press was longer than timeout
// lastPressLong = true;
// }
// //run action function with current count of button presses
// action(count, lastPressLong);
// }
// break;
// }
// }
// }
//
//
//
//
//-----------------------------
//-------- constructor --------
//-----------------------------
buttonCommands::buttonCommands(
gpio_evaluatedSwitch * button_f,
evaluatedJoystick * joystick_f,
controlledArmchair * control_f,
buzzer_t * buzzer_f
){
//copy object pointers
button = button_f;
joystick = joystick_f;
control = control_f;
buzzer = buzzer_f;
//TODO declare / configure evaluatedSwitch here instead of config (unnecessary that button object is globally available - only used here)?
}
//----------------------------
//--------- action -----------
//----------------------------
//function that runs commands depending on a count value
void buttonCommands::action (uint8_t count, bool lastPressLong){
//--- variable declarations ---
bool decelEnabled; //for different beeping when toggling
commandSimple_t cmds[8]; //array for commands for automatedArmchair
//--- get joystick position ---
//joystickData_t stickData = joystick->getData();
//--- actions based on count ---
switch (count){
//no such command
default:
ESP_LOGE(TAG, "no command for count=%d defined", count);
buzzer->beep(3, 400, 100);
break;
case 1:
//restart contoller when 1x long pressed
if (lastPressLong){
ESP_LOGW(TAG, "RESTART");
buzzer->beep(1,1000,1);
vTaskDelay(1000 / portTICK_PERIOD_MS);
esp_restart();
return;
}
ESP_LOGW(TAG, "cmd %d: sending button event to control task", count);
//-> define joystick center or toggle freeze input (executed in control task)
control->sendButtonEvent(count); //TODO: always send button event to control task (not just at count=1) -> control.cpp has to be changed
break;
case 2:
//TODO no use for leg support from 1.0 anymore, add other functionality
//run automatic commands to lift leg support when pressed 1x short 1x long
//if (lastPressLong){
// //define commands
// cmds[0] =
// {
// .motorCmds = {
// .left = {motorstate_t::REV, 90},
// .right = {motorstate_t::REV, 90}
// },
// .msDuration = 1200,
// .fadeDecel = 800,
// .fadeAccel = 1300,
// .instruction = auto_instruction_t::NONE
// };
// cmds[1] =
// {
// .motorCmds = {
// .left = {motorstate_t::FWD, 70},
// .right = {motorstate_t::FWD, 70}
// },
// .msDuration = 70,
// .fadeDecel = 0,
// .fadeAccel = 300,
// .instruction = auto_instruction_t::NONE
// };
// cmds[2] =
// {
// .motorCmds = {
// .left = {motorstate_t::IDLE, 0},
// .right = {motorstate_t::IDLE, 0}
// },
// .msDuration = 10,
// .fadeDecel = 800,
// .fadeAccel = 1300,
// .instruction = auto_instruction_t::SWITCH_JOYSTICK_MODE
// };
// //send commands to automatedArmchair command queue
// armchair.addCommands(cmds, 3);
// //change mode to AUTO
// control->changeMode(controlMode_t::AUTO);
// return;
//}
//toggle idle when 2x pressed
ESP_LOGW(TAG, "cmd %d: toggle IDLE", count);
control->toggleIdle(); //toggle between idle and previous/default mode
break;
case 3:
ESP_LOGW(TAG, "cmd %d: switch to JOYSTICK", count);
control->changeMode(controlMode_t::JOYSTICK); //switch to JOYSTICK mode
break;
case 4:
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
break;
case 6:
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
break;
// case 8:
// //TODO rework this with uart config
// //toggle deceleration fading between on and off
// decelEnabled = motorLeft->toggleFade(fadeType_t::DECEL);
// motorRight->toggleFade(fadeType_t::DECEL);
// ESP_LOGW(TAG, "cmd %d: toggle deceleration fading to: %d", count, (int)decelEnabled);
// if (decelEnabled){
// buzzer->beep(3, 60, 50);
// } else {
// buzzer->beep(1, 1000, 1);
// }
break;
case 12:
ESP_LOGW(TAG, "cmd %d: sending button event to control task", count);
//-> toggle altStickMapping (executed in control task)
control->sendButtonEvent(count); //TODO: always send button event to control task (not just at count=1)?
break;
}
}
//-----------------------------
//------ startHandleLoop ------
//-----------------------------
//this function has to be started once in a separate task
//repeatedly evaluates and processes button events then takes the corresponding action
void buttonCommands::startHandleLoop() {
while(1) {
vTaskDelay(20 / portTICK_PERIOD_MS);
//run handle function of evaluatedSwitch object
button->handle();
//--- count button presses and run action ---
switch(state) {
case inputState_t::IDLE: //wait for initial button press
if (button->risingEdge) {
count = 1;
buzzer->beep(1, 65, 0);
timestamp_lastAction = esp_log_timestamp();
state = inputState_t::WAIT_FOR_INPUT;
ESP_LOGI(TAG, "first button press detected -> waiting for further events");
}
break;
case inputState_t::WAIT_FOR_INPUT: //wait for further presses
//button pressed again
if (button->risingEdge){
count++;
buzzer->beep(1, 65, 0);
timestamp_lastAction = esp_log_timestamp();
ESP_LOGI(TAG, "another press detected -> count=%d -> waiting for further events", count);
}
//timeout
else if (esp_log_timestamp() - timestamp_lastAction > 1000) {
state = inputState_t::IDLE;
buzzer->beep(count, 50, 50);
//TODO: add optional "bool wait" parameter to beep function to delay until finished beeping
ESP_LOGI(TAG, "timeout - running action function for count=%d", count);
//--- run action function ---
//check if still pressed
bool lastPressLong = false;
if (button->state == true){
//run special case when last press was longer than timeout
lastPressLong = true;
}
//run action function with current count of button presses
action(count, lastPressLong);
}
break;
}
}
}

View File

@ -11,44 +11,40 @@
// //===================================
// //====== buttonCommands class =======
// //===================================
// //class which runs commands depending on the count a button was pressed
// class buttonCommands {
// public:
// //--- constructor ---
// buttonCommands (
// gpio_evaluatedSwitch * button_f,
// evaluatedJoystick * joystick_f,
// controlledArmchair * control_f,
// buzzer_t * buzzer_f,
// controlledMotor * motorLeft_f,
// controlledMotor * motorRight_f
// );
//
// //--- functions ---
// //the following function has to be started once in a separate task.
// //repeatedly evaluates and processes button events then takes the corresponding action
// void startHandleLoop();
//
// private:
// //--- functions ---
// void action(uint8_t count, bool lastPressLong);
//
// //--- objects ---
// gpio_evaluatedSwitch* button;
// evaluatedJoystick* joystick;
// controlledArmchair * control;
// buzzer_t* buzzer;
// controlledMotor * motorLeft;
// controlledMotor * motorRight;
//
// //--- variables ---
// uint8_t count = 0;
// uint32_t timestamp_lastAction = 0;
// enum class inputState_t {IDLE, WAIT_FOR_INPUT};
// inputState_t state = inputState_t::IDLE;
//
// };
//===================================
//====== buttonCommands class =======
//===================================
//class which runs commands depending on the count a button was pressed
class buttonCommands {
public:
//--- constructor ---
buttonCommands (
gpio_evaluatedSwitch * button_f,
evaluatedJoystick * joystick_f,
controlledArmchair * control_f,
buzzer_t * buzzer_f
);
//--- functions ---
//the following function has to be started once in a separate task.
//repeatedly evaluates and processes button events then takes the corresponding action
void startHandleLoop();
private:
//--- functions ---
void action(uint8_t count, bool lastPressLong);
//--- objects ---
gpio_evaluatedSwitch* button;
evaluatedJoystick* joystick;
controlledArmchair * control;
buzzer_t* buzzer;
//--- variables ---
uint8_t count = 0;
uint32_t timestamp_lastAction = 0;
enum class inputState_t {IDLE, WAIT_FOR_INPUT};
inputState_t state = inputState_t::IDLE;
};

View File

@ -71,13 +71,13 @@ void task_control( void * pvParameters ){
//============ button task =============
//======================================
//task that handles the button interface/commands
//void task_button( void * pvParameters ){
// ESP_LOGI(TAG, "Initializing command-button and starting handle loop");
// //create button instance
// buttonCommands commandButton(&buttonJoystick, &joystick, &control, &buzzer, &motorLeft, &motorRight);
// //start handle loop
// commandButton.startHandleLoop();
//}
void task_button( void * pvParameters ){
ESP_LOGI(TAG, "Initializing command-button and starting handle loop");
//create button instance
buttonCommands commandButton(&buttonJoystick, &joystick, &control, &buzzer);
//start handle loop
commandButton.startHandleLoop();
}
@ -169,7 +169,7 @@ extern "C" void app_main(void) {
//------------------------------
//--- create task for buzzer ---
//------------------------------
//xTaskCreate(&task_buzzer, "task_buzzer", 2048, NULL, 2, NULL);
xTaskCreate(&task_buzzer, "task_buzzer", 2048, NULL, 2, NULL);
//-------------------------------
//--- create task for control ---
@ -181,7 +181,7 @@ extern "C" void app_main(void) {
//--- create task for button ---
//------------------------------
//task that evaluates and processes the button input and runs the configured commands
//xTaskCreate(&task_button, "task_button", 4096, NULL, 4, NULL);
xTaskCreate(&task_button, "task_button", 4096, NULL, 4, NULL);
//-----------------------------------
//--- create task for fan control ---