diff --git a/board_control/main/button.cpp b/board_control/main/button.cpp index f90a8a4..5ad9b59 100644 --- a/board_control/main/button.cpp +++ b/board_control/main/button.cpp @@ -1,3 +1,4 @@ +#include "auto.hpp" extern "C" { #include @@ -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; + } + } + } + + + + diff --git a/board_control/main/button.hpp b/board_control/main/button.hpp index 8f44b16..495f1dd 100644 --- a/board_control/main/button.hpp +++ b/board_control/main/button.hpp @@ -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; + +}; diff --git a/board_control/main/main.cpp b/board_control/main/main.cpp index 03d66bc..260310a 100644 --- a/board_control/main/main.cpp +++ b/board_control/main/main.cpp @@ -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 ---