From cac22ca4e1f44c0a53252b5fd046f78f489956f0 Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Sun, 24 Jul 2022 12:21:11 +0200 Subject: [PATCH 1/7] Add functions to send button-event to control task - control.cpp: - Add method sendButtonEvent to control class - joystick mode: define joystick center at button event - massage mode: toggle freezing of joystick input at button event - button.cpp: remove reboot command (1x press) - button.cpp: add command sendButtonEvent to control task (1x press) --- main/button.cpp | 10 +++++---- main/control.cpp | 55 +++++++++++++++++++++++++++++++++++++++++++----- main/control.hpp | 11 ++++++++++ 3 files changed, 67 insertions(+), 9 deletions(-) diff --git a/main/button.cpp b/main/button.cpp index 927dac3..7b6f5cd 100644 --- a/main/button.cpp +++ b/main/button.cpp @@ -44,11 +44,13 @@ void buttonCommands::action (uint8_t count){ break; case 1: - ESP_LOGW(TAG, "RESTART"); - buzzer->beep(1,1000,1); - vTaskDelay(1000 / portTICK_PERIOD_MS); - esp_restart(); + //ESP_LOGW(TAG, "RESTART"); + //buzzer->beep(1,1000,1); + //vTaskDelay(1000 / portTICK_PERIOD_MS); + //esp_restart(); + ESP_LOGW(TAG, "cmd %d: sending button event to control task", count); + control->sendButtonEvent(count); break; case 2: diff --git a/main/control.cpp b/main/control.cpp index ff77931..0497a9b 100644 --- a/main/control.cpp +++ b/main/control.cpp @@ -61,6 +61,7 @@ void controlledArmchair::startHandleLoop() { mode = controlMode_t::IDLE; break; + case controlMode_t::IDLE: //copy preset commands for idling both motors commands = cmds_bothMotorsIdle; @@ -69,7 +70,9 @@ void controlledArmchair::startHandleLoop() { vTaskDelay(200 / portTICK_PERIOD_MS); break; + case controlMode_t::JOYSTICK: + vTaskDelay(20 / portTICK_PERIOD_MS); //get current joystick data with getData method of evaluatedJoystick stickData = joystick_l->getData(); //additionaly scale coordinates (more detail in slower area) @@ -80,19 +83,39 @@ void controlledArmchair::startHandleLoop() { motorRight->setTarget(commands.right.state, commands.right.duty); motorLeft->setTarget(commands.left.state, commands.left.duty); //TODO make motorctl.setTarget also accept motorcommand struct directly - vTaskDelay(20 / portTICK_PERIOD_MS); + + //--- button event --- + if (buttonEvent){ + joystick_l->defineCenter(); + buzzer->beep(2, 200, 100); + } break; + case controlMode_t::MASSAGE: - //generate motor commands + vTaskDelay(20 / portTICK_PERIOD_MS); + //--- read joystick --- + //only update joystick data when input not frozen + if (!freezeInput){ + stickData = joystick_l->getData(); + } + + //--- generate motor commands --- //pass joystick data from getData method of evaluatedJoystick to generateCommandsShaking function - commands = joystick_generateCommandsShaking(joystick_l->getData()); + commands = joystick_generateCommandsShaking(stickData); //apply motor commands motorRight->setTarget(commands.right.state, commands.right.duty); motorLeft->setTarget(commands.left.state, commands.left.duty); - vTaskDelay(20 / portTICK_PERIOD_MS); + + //--- button event --- + if (buttonEvent){ + //toggle freeze of input (lock joystick at current values) + freezeInput = !freezeInput; + buzzer->beep(2, 200, 100); + } break; + case controlMode_t::HTTP: //--- get joystick data from queue --- //Note this function waits several seconds (httpconfig.timeoutMs) for data to arrive, otherwise Center data or NULL is returned @@ -111,9 +134,17 @@ void controlledArmchair::startHandleLoop() { motorLeft->setTarget(commands.left.state, commands.left.duty); break; + // //TODO: add other modes here } + //--- reset button event --- (only one run of handle loop) + //TODO: what if variable gets set durin above code? -> mutex around entire handle loop + if (buttonEvent == true){ + ESP_LOGI(TAG, "resetting button event"); + buttonEvent = false; + } + //----------------------- //------ slow loop ------ @@ -142,10 +173,24 @@ void controlledArmchair::resetTimeout(){ +//------------------------------------ +//--------- sendButtonEvent ---------- +//------------------------------------ +void controlledArmchair::sendButtonEvent(uint8_t count){ + //TODO mutex - if not replaced with queue + ESP_LOGI(TAG, "setting button event"); + buttonEvent = true; + buttonCount = count; +} + + + //------------------------------------ //---------- handleTimeout ----------- //------------------------------------ -float inactivityTolerance = 10; //percentage the duty can vary since last timeout check and still counts as incative +//percentage the duty can vary since last timeout check and still counts as incative +//TODO: add this to config +float inactivityTolerance = 10; //local function that checks whether two values differ more than a given tolerance bool validateActivity(float dutyOld, float dutyNow, float tolerance){ diff --git a/main/control.hpp b/main/control.hpp index 52dc4b1..67a62b3 100644 --- a/main/control.hpp +++ b/main/control.hpp @@ -58,6 +58,10 @@ class controlledArmchair { //function that restarts timer which initiates the automatic timeout (switch to IDLE) after certain time of inactivity void resetTimeout(); + //function for sending a button event (e.g. from button task at event) to control task + //TODO: use queue instead? + void sendButtonEvent(uint8_t count); + private: //--- functions --- @@ -83,6 +87,13 @@ class controlledArmchair { //variables for http mode uint32_t http_timestamp_lastData = 0; + //variables for MASSAGE mode + bool freezeInput = false; + + //variable to store button event + bool buttonEvent = false; + uint8_t buttonCount = 0; + //definition of mode enum controlMode_t mode = controlMode_t::IDLE; From fea8c94a73f70b36cca6c0cedd1023a09d708133 Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Sun, 24 Jul 2022 17:13:12 +0200 Subject: [PATCH 2/7] Modify fading when switching from/to MASSAGE mode In massage mode it is required that the motors react quicker, Added the following functionality to control.cpp: - when changing to MASSAGE mode the downfading is disabled and upfading is reduced - when changing from MASSAGE mode downfading is enabled and upfading reset to default value --- main/control.cpp | 25 ++++++++++++++++++++++++- 1 file changed, 24 insertions(+), 1 deletion(-) diff --git a/main/control.cpp b/main/control.cpp index 0497a9b..95165bd 100644 --- a/main/control.cpp +++ b/main/control.cpp @@ -259,7 +259,6 @@ void controlledArmchair::changeMode(controlMode_t modeNew) { ESP_LOGI(TAG, "disabling http server..."); http_stop_server(); - //FIXME: make wifi function work here - currently starting wifi at startup (see notes main.cpp) //stop wifi //TODO: decide whether ap or client is currently used - which has to be disabled? @@ -268,6 +267,17 @@ void controlledArmchair::changeMode(controlMode_t modeNew) { //wifi_deinit_ap(); ESP_LOGI(TAG, "done stopping http mode"); break; + + case controlMode_t::MASSAGE: + ESP_LOGW(TAG, "switching from MASSAGE mode -> restoring fading"); + //TODO: fix issue when downfading was disabled before switching to massage mode - currently it gets enabled again here... + //enable downfading (set to default value) + motorLeft->setFade(fadeType_t::DECEL, true); + motorRight->setFade(fadeType_t::DECEL, true); + //set upfading to default value + motorLeft->setFade(fadeType_t::ACCEL, true); + motorRight->setFade(fadeType_t::ACCEL, true); + break; } @@ -300,6 +310,19 @@ void controlledArmchair::changeMode(controlMode_t modeNew) { http_init_server(); ESP_LOGI(TAG, "done initializing http mode"); break; + + case controlMode_t::MASSAGE: + ESP_LOGW(TAG, "switching to MASSAGE mode -> reducing fading"); + uint32_t shake_msFadeAccel = 400; //TODO: move this to config + + //disable downfading (max. deceleration) + motorLeft->setFade(fadeType_t::DECEL, false); + motorRight->setFade(fadeType_t::DECEL, false); + //reduce upfading (increase acceleration) + motorLeft->setFade(fadeType_t::ACCEL, shake_msFadeAccel); + motorRight->setFade(fadeType_t::ACCEL, shake_msFadeAccel); + break; + } //--- update mode to new mode --- From ad0723a766413334c8b136db8a1272e9f605b78d Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Mon, 25 Jul 2022 11:02:52 +0200 Subject: [PATCH 3/7] Update MASSAGE: 4 modes (each quadrant) with hysteresis Modified function that generates commands in massage mode (joystick.cpp) - define variable stickQuadrant with axis as hysteresis from actual joystick data/pos - prepare switch case for implementation of 4 modes (each quadrant) - add basic content to modes for testing (different motor directions) TODO: remove old code with pulsing shake variable and implement actual modes --- main/joystick.cpp | 133 +++++++++++++++++++++++++++++++--------------- 1 file changed, 91 insertions(+), 42 deletions(-) diff --git a/main/joystick.cpp b/main/joystick.cpp index fa4bd12..27c709f 100644 --- a/main/joystick.cpp +++ b/main/joystick.cpp @@ -376,6 +376,9 @@ motorCommands_t joystick_generateCommandsDriving(joystickData_t data){ uint32_t shake_timestamp_turnedOn = 0; uint32_t shake_timestamp_turnedOff = 0; bool shake_state = false; +joystickPos_t lastStickPos = joystickPos_t::CENTER; +//stick position quadrant only with "X_AXIS and Y_AXIS" as hysteresis +joystickPos_t stickQuadrant = joystickPos_t::CENTER; //--- configure shake mode --- TODO: move this to config uint32_t shake_msOffMax = 90; @@ -385,25 +388,17 @@ float dutyShake = 60; //function that generates commands for both motors from the joystick data motorCommands_t joystick_generateCommandsShaking(joystickData_t data){ - - //struct with current data of the joystick - //typedef struct joystickData_t { - // joystickPos_t position; - // float x; - // float y; - // float radius; - // float angle; - //} joystickData_t; - - + //--- handle pulsing shake variable --- + //TODO remove this, make individual per mode? + //TODO only run this when not CENTER anyways? motorCommands_t commands; float ratio = fabs(data.angle) / 90; //90degree = x=0 || 0degree = y=0 - //--- calculate on/off duration --- + //calculate on/off duration uint32_t msOn = shake_msOnMax * data.radius; uint32_t msOff = shake_msOffMax * data.radius; - //--- evaluate state (on/off) --- + //evaluate state (on/off) if (data.radius > 0 ){ //currently off if (shake_state == false){ @@ -433,44 +428,97 @@ motorCommands_t joystick_generateCommandsShaking(joystickData_t data){ + //struct with current data of the joystick + //typedef struct joystickData_t { + // joystickPos_t position; + // float x; + // float y; + // float radius; + // float angle; + //} joystickData_t; - if (shake_state){ - switch (data.position){ - default: - commands.left.state = motorstate_t::IDLE; - commands.right.state = motorstate_t::IDLE; - commands.left.duty = 0; - commands.right.duty = 0; - break; + //--- evaluate stick position --- + //4 quadrants and center only - with X and Y axis as hysteresis + switch (data.position){ - case joystickPos_t::Y_AXIS: + case joystickPos_t::CENTER: + //immediately set to center at center + stickQuadrant = joystickPos_t::CENTER; + break; + + case joystickPos_t::Y_AXIS: + //when moving from center to axis initially start in a certain quadrant + if (stickQuadrant == joystickPos_t::CENTER) { if (data.y > 0){ - commands.left.state = motorstate_t::FWD; - commands.right.state = motorstate_t::FWD; + stickQuadrant = joystickPos_t::TOP_RIGHT; } else { - commands.left.state = motorstate_t::REV; - commands.right.state = motorstate_t::REV; + stickQuadrant = joystickPos_t::BOTTOM_RIGHT; } - //set duty to shake - commands.left.duty = dutyShake; - commands.right.duty = dutyShake; - break; + } - case joystickPos_t::X_AXIS: - if (data.x > 0) { - commands.left.state = motorstate_t::FWD; - commands.right.state = motorstate_t::REV; + case joystickPos_t::X_AXIS: + //when moving from center to axis initially start in a certain quadrant + if (stickQuadrant == joystickPos_t::CENTER) { + if (data.x > 0){ + stickQuadrant = joystickPos_t::TOP_RIGHT; } else { - commands.left.state = motorstate_t::REV; - commands.right.state = motorstate_t::FWD; + stickQuadrant = joystickPos_t::TOP_LEFT; } - //set duty to shake - commands.left.duty = dutyShake; - commands.right.duty = dutyShake; - break; - } - } else { //shake state off + } + break; + + case joystickPos_t::TOP_RIGHT: + case joystickPos_t::TOP_LEFT: + case joystickPos_t::BOTTOM_LEFT: + case joystickPos_t::BOTTOM_RIGHT: + //update/change evaluated pos when in one of the 4 quadrants + stickQuadrant = data.position; + //TODO: maybe beep when switching mode? (difficult because beep object has to be passed to function) + break; + } + + + + //--- handle different modes (joystich in any of 4 quadrants) --- + switch (stickQuadrant){ + case joystickPos_t::CENTER: + case joystickPos_t::X_AXIS: //never true + case joystickPos_t::Y_AXIS: //never true + commands.left.state = motorstate_t::IDLE; + commands.right.state = motorstate_t::IDLE; + commands.left.duty = 0; + commands.right.duty = 0; + ESP_LOGI(TAG_CMD, "generate shake commands: CENTER -> idle"); + return commands; + break; + //4 different modes + case joystickPos_t::TOP_RIGHT: + commands.left.state = motorstate_t::FWD; + commands.right.state = motorstate_t::FWD; + break; + case joystickPos_t::TOP_LEFT: + commands.left.state = motorstate_t::FWD; + commands.right.state = motorstate_t::REV; + break; + case joystickPos_t::BOTTOM_LEFT: + commands.left.state = motorstate_t::REV; + commands.right.state = motorstate_t::FWD; + break; + case joystickPos_t::BOTTOM_RIGHT: + commands.left.state = motorstate_t::REV; + commands.right.state = motorstate_t::REV; + break; + } + + + //--- turn motors on/off depending on pulsing shake variable --- + if (shake_state == true){ + //set duty to shake + commands.left.duty = dutyShake; + commands.right.duty = dutyShake; + //directions are defined above depending on mode + } else { commands.left.state = motorstate_t::IDLE; commands.right.state = motorstate_t::IDLE; commands.left.duty = 0; @@ -482,5 +530,6 @@ motorCommands_t joystick_generateCommandsShaking(joystickData_t data){ joystickPosStr[(int)data.position], data.angle, ratio, (1-ratio), data.radius, data.x, data.y); ESP_LOGI(TAG_CMD, "motor left: state=%s, duty=%.3f", motorstateStr[(int)commands.left.state], commands.left.duty); ESP_LOGI(TAG_CMD, "motor right: state=%s, duty=%.3f", motorstateStr[(int)commands.right.state], commands.right.duty); + return commands; } From 3fa2f17a72d238ef706168d9e9dfb21ad5011584 Mon Sep 17 00:00:00 2001 From: jonny_l480 Date: Mon, 25 Jul 2022 12:54:05 +0200 Subject: [PATCH 4/7] Fix Bug unexpected movement MASSAGE; Adjustments Fix bug where armchair moved too long every once in a while (especially at start). Minor adjustments of parameters. main.cpp: - set individual task priorities for each task e.g. fan or buzzer task has very low priority (did not have any immediate effect though while testing) - increase handle interval of motorctl change from 20 to 10 (same frequency as generation of massage commands) -> this fixed the bug with unexpected movement (motorctl could not process every command from massage mode) control.cpp: - decrease delay in massage mode for more detail/levels at joystick radius (from 20 to 10ms) - increase fading in massage mode (400ms to 500ms) for slightly less hard shaking joystick.cpp: - reduce max shaking amount - swap modes - top left shake backward - top right shake forward - bottom left/right shake rotating --- main/control.cpp | 4 ++-- main/joystick.cpp | 8 ++++---- main/main.cpp | 12 ++++++------ 3 files changed, 12 insertions(+), 12 deletions(-) diff --git a/main/control.cpp b/main/control.cpp index 95165bd..cbb1ae9 100644 --- a/main/control.cpp +++ b/main/control.cpp @@ -93,7 +93,7 @@ void controlledArmchair::startHandleLoop() { case controlMode_t::MASSAGE: - vTaskDelay(20 / portTICK_PERIOD_MS); + vTaskDelay(10 / portTICK_PERIOD_MS); //--- read joystick --- //only update joystick data when input not frozen if (!freezeInput){ @@ -313,7 +313,7 @@ void controlledArmchair::changeMode(controlMode_t modeNew) { case controlMode_t::MASSAGE: ESP_LOGW(TAG, "switching to MASSAGE mode -> reducing fading"); - uint32_t shake_msFadeAccel = 400; //TODO: move this to config + uint32_t shake_msFadeAccel = 500; //TODO: move this to config //disable downfading (max. deceleration) motorLeft->setFade(fadeType_t::DECEL, false); diff --git a/main/joystick.cpp b/main/joystick.cpp index 27c709f..64bbf1c 100644 --- a/main/joystick.cpp +++ b/main/joystick.cpp @@ -381,8 +381,8 @@ joystickPos_t lastStickPos = joystickPos_t::CENTER; joystickPos_t stickQuadrant = joystickPos_t::CENTER; //--- configure shake mode --- TODO: move this to config -uint32_t shake_msOffMax = 90; -uint32_t shake_msOnMax = 180; +uint32_t shake_msOffMax = 80; +uint32_t shake_msOnMax = 120; float dutyShake = 60; //function that generates commands for both motors from the joystick data @@ -498,7 +498,7 @@ motorCommands_t joystick_generateCommandsShaking(joystickData_t data){ commands.right.state = motorstate_t::FWD; break; case joystickPos_t::TOP_LEFT: - commands.left.state = motorstate_t::FWD; + commands.left.state = motorstate_t::REV; commands.right.state = motorstate_t::REV; break; case joystickPos_t::BOTTOM_LEFT: @@ -506,7 +506,7 @@ motorCommands_t joystick_generateCommandsShaking(joystickData_t data){ commands.right.state = motorstate_t::FWD; break; case joystickPos_t::BOTTOM_RIGHT: - commands.left.state = motorstate_t::REV; + commands.left.state = motorstate_t::FWD; commands.right.state = motorstate_t::REV; break; } diff --git a/main/main.cpp b/main/main.cpp index f5bff17..49ec363 100644 --- a/main/main.cpp +++ b/main/main.cpp @@ -38,7 +38,7 @@ void task_motorctl( void * pvParameters ){ motorRight.handle(); motorLeft.handle(); //10khz -> T=100us - vTaskDelay(20 / portTICK_PERIOD_MS); + vTaskDelay(10 / portTICK_PERIOD_MS); } } @@ -148,7 +148,7 @@ extern "C" void app_main(void) { //esp_log_level_set("evaluatedJoystick", ESP_LOG_DEBUG); //esp_log_level_set("joystickCommands", ESP_LOG_DEBUG); esp_log_level_set("button", ESP_LOG_INFO); - esp_log_level_set("control", ESP_LOG_DEBUG); + esp_log_level_set("control", ESP_LOG_INFO); esp_log_level_set("fan-control", ESP_LOG_INFO); esp_log_level_set("wifi", ESP_LOG_INFO); esp_log_level_set("http", ESP_LOG_INFO); @@ -158,12 +158,12 @@ extern "C" void app_main(void) { //--- create task for controlling the motors --- //---------------------------------------------- //task that receives commands, handles ramp and current limit and executes commands using the motordriver function - xTaskCreate(&task_motorctl, "task_motor-control", 2048, NULL, 5, NULL); + xTaskCreate(&task_motorctl, "task_motor-control", 2048, NULL, 6, NULL); //------------------------------ //--- create task for buzzer --- //------------------------------ - xTaskCreate(&task_buzzer, "task_buzzer", 2048, NULL, 5, NULL); + xTaskCreate(&task_buzzer, "task_buzzer", 2048, NULL, 2, NULL); //------------------------------- //--- create task for control --- @@ -175,13 +175,13 @@ 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", 2048, NULL, 5, NULL); + xTaskCreate(&task_button, "task_button", 2048, NULL, 4, NULL); //----------------------------------- //--- create task for fan control --- //----------------------------------- //task that evaluates and processes the button input and runs the configured commands - xTaskCreate(&task_fans, "task_fans", 2048, NULL, 5, NULL); + xTaskCreate(&task_fans, "task_fans", 2048, NULL, 1, NULL); //beep at startup From 1c3576206afc9d85687ecada2a14452e45e08c79 Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Mon, 25 Jul 2022 22:45:59 +0200 Subject: [PATCH 5/7] Add button commands; Update Readme button.cpp: - Add command: reboot when long button press - Add command: switch to mode JOYSTICK with 3x press Readme: - rework table with switch functions - add new commands to table - add drink holder to planned features joystick.cpp: - fix bug: missing 'break;' for switch case in function joystick_CommandsShaking --- README.md | 23 ++++++++++++++--------- main/button.cpp | 29 +++++++++++++++++++++-------- main/joystick.cpp | 3 ++- 3 files changed, 37 insertions(+), 18 deletions(-) diff --git a/README.md b/README.md index 3224b46..b5c95d5 100644 --- a/README.md +++ b/README.md @@ -92,21 +92,26 @@ A diagram which shows what components are connected to which terminals of the pc - Anti slip regulation - Self driving algorithm - Lights +- drinks holder # Usage ## Switch functions **Currently implemented** -| Count | Action | -| --- | ---| -| 1 | | -| 2 | toggle IDLE mode | -| 3 | | -| 4 | toggle between HTTP and JOYSTICK mode| -| 5 | | -| 6 | toggle between MASSAGE and JOYSTICK mode | -| 7 | | +| Count | Type | Action | Description | +| --- | --- | --- | --- | +| 1x | configure | [JOYSTICK] **calibrate stick** | when in joystick mode: set joystick center to current joystick pos | +| 1x | control | [MASSAGE] **freeze** input | when in massage mode: lock or unlock joystick input at current position | +| 2x | toggle mode | **IDLE** <=> previous | enable/disable chair armchair e.g. enable after startup or timeout | +| 3x | switch mode | **JOYSTICK** | switch to default mode JOYSTICK | +| 4x | toggle mode | **HTTP** <=> JOYSTICK | switch to '**remote control** via web-app' or back to JOYSTICK mode | +| 5x | | | | +| 6x | toggle mode | **MASSAGE** <=> JOYSTICK | switch to MASSAGE mode or back to JOYSTICK mode | +| 7x | | | | +| 8x | toggle option | **deceleration limit** | disable/enable deceleration limit (default on) => more responsive | +| >1s | system | **restart** | Restart the controller when pressing the button longer than 1 second | + **previous functions - not implemented** diff --git a/main/button.cpp b/main/button.cpp index 7b6f5cd..5eada13 100644 --- a/main/button.cpp +++ b/main/button.cpp @@ -43,14 +43,20 @@ void buttonCommands::action (uint8_t count){ buzzer->beep(3, 400, 100); break; - case 1: - //ESP_LOGW(TAG, "RESTART"); - //buzzer->beep(1,1000,1); - //vTaskDelay(1000 / portTICK_PERIOD_MS); - //esp_restart(); + case 0: //special case when last button press is longer than timeout (>1s) + ESP_LOGW(TAG, "RESTART"); + buzzer->beep(1,1000,1); + vTaskDelay(1000 / portTICK_PERIOD_MS); + esp_restart(); + case 1: ESP_LOGW(TAG, "cmd %d: sending button event to control task", count); - control->sendButtonEvent(count); + control->sendButtonEvent(count); //TODO: always send button event to control task (not just at count=1) -> control.cpp has to be changed + break; + + case 3: + ESP_LOGW(TAG, "cmd %d: switch to JOYSTICK", count); + control->changeMode(controlMode_t::JOYSTICK); //switch to JOYSTICK mode break; case 2: @@ -123,9 +129,16 @@ void buttonCommands::startHandleLoop() { state = inputState_t::IDLE; buzzer->beep(count, 50, 50); //TODO: add optional "bool wait" parameter to beep function to delay until finished beeping - //run action function with current count of button presses ESP_LOGI(TAG, "timeout - running action function for count=%d", count); - action(count); + //--- run action function --- + //check if still pressed + if (button->state == true){ + //run special case when last press was longer than timeout + action(0); + } else { + //run action function with current count of button presses + action(count); + } } break; } diff --git a/main/joystick.cpp b/main/joystick.cpp index 64bbf1c..cb9c90b 100644 --- a/main/joystick.cpp +++ b/main/joystick.cpp @@ -456,6 +456,7 @@ motorCommands_t joystick_generateCommandsShaking(joystickData_t data){ stickQuadrant = joystickPos_t::BOTTOM_RIGHT; } } + break; case joystickPos_t::X_AXIS: //when moving from center to axis initially start in a certain quadrant @@ -480,7 +481,7 @@ motorCommands_t joystick_generateCommandsShaking(joystickData_t data){ - //--- handle different modes (joystich in any of 4 quadrants) --- + //--- handle different modes (joystick in any of 4 quadrants) --- switch (stickQuadrant){ case joystickPos_t::CENTER: case joystickPos_t::X_AXIS: //never true From bbe26a3823a6601224f3fe1c092d987b5435f1b1 Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Tue, 26 Jul 2022 09:36:46 +0200 Subject: [PATCH 6/7] Add button cmd x12: alternative joystick mapping New command and feature also made some general changes to control and button necessary joystick.cpp: - add optional parameter to joystick_CommandsDriving function bool altStickMapping (default false) if true this currently swaps BOTTOM_LEFT with BOTTOM_RIGHT for experimental different joystick mapping button.cpp: - fix/move variable declaration outside of switch - add 12x case: send count to control task control.cpp: - create new section in handle loop for button events - move x1 commands from JOYSTICK and MASSAGE case to new button section - remove unnecessary variable buttonEvent (only using buttonCount now) - add functionality to 12x button event -> toggle alternative stick mapping readme.md: - add new x12 button command --- README.md | 2 ++ main/button.cpp | 13 ++++++++++- main/control.cpp | 57 +++++++++++++++++++++++++++++------------------ main/control.hpp | 2 +- main/joystick.cpp | 17 +++++++++++--- main/joystick.hpp | 2 +- 6 files changed, 65 insertions(+), 28 deletions(-) diff --git a/README.md b/README.md index b5c95d5..d91fbe0 100644 --- a/README.md +++ b/README.md @@ -110,6 +110,8 @@ A diagram which shows what components are connected to which terminals of the pc | 6x | toggle mode | **MASSAGE** <=> JOYSTICK | switch to MASSAGE mode or back to JOYSTICK mode | | 7x | | | | | 8x | toggle option | **deceleration limit** | disable/enable deceleration limit (default on) => more responsive | +| | | | | +| 12x | toggle option | **alt stick mapping** | toggle between default and alternative stick mapping (reverse swapped) | | >1s | system | **restart** | Restart the controller when pressing the button longer than 1 second | diff --git a/main/button.cpp b/main/button.cpp index 5eada13..b717328 100644 --- a/main/button.cpp +++ b/main/button.cpp @@ -36,6 +36,10 @@ buttonCommands::buttonCommands(gpio_evaluatedSwitch * button_f, controlledArmcha //---------------------------- //function that runs commands depending on a count value void buttonCommands::action (uint8_t count){ + //--- variable declarations --- + bool decelEnabled; //for different beeping when toggling + + //--- actions based on count --- switch (count){ //no such command default: @@ -51,6 +55,7 @@ void buttonCommands::action (uint8_t count){ case 1: 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; @@ -76,7 +81,7 @@ void buttonCommands::action (uint8_t count){ case 8: //toggle deceleration fading between on and off - bool decelEnabled = motorLeft->toggleFade(fadeType_t::DECEL); + 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){ @@ -84,7 +89,13 @@ void buttonCommands::action (uint8_t count){ } 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; } } diff --git a/main/control.cpp b/main/control.cpp index cbb1ae9..061e063 100644 --- a/main/control.cpp +++ b/main/control.cpp @@ -78,17 +78,11 @@ void controlledArmchair::startHandleLoop() { //additionaly scale coordinates (more detail in slower area) joystick_scaleCoordinatesLinear(&stickData, 0.6, 0.35); //TODO: add scaling parameters to config //generate motor commands - commands = joystick_generateCommandsDriving(stickData); + commands = joystick_generateCommandsDriving(stickData, altStickMapping); //apply motor commands motorRight->setTarget(commands.right.state, commands.right.duty); motorLeft->setTarget(commands.left.state, commands.left.duty); //TODO make motorctl.setTarget also accept motorcommand struct directly - - //--- button event --- - if (buttonEvent){ - joystick_l->defineCenter(); - buzzer->beep(2, 200, 100); - } break; @@ -99,7 +93,6 @@ void controlledArmchair::startHandleLoop() { if (!freezeInput){ stickData = joystick_l->getData(); } - //--- generate motor commands --- //pass joystick data from getData method of evaluatedJoystick to generateCommandsShaking function commands = joystick_generateCommandsShaking(stickData); @@ -107,12 +100,6 @@ void controlledArmchair::startHandleLoop() { motorRight->setTarget(commands.right.state, commands.right.duty); motorLeft->setTarget(commands.left.state, commands.left.duty); - //--- button event --- - if (buttonEvent){ - //toggle freeze of input (lock joystick at current values) - freezeInput = !freezeInput; - buzzer->beep(2, 200, 100); - } break; @@ -126,7 +113,7 @@ void controlledArmchair::startHandleLoop() { ESP_LOGD(TAG, "generating commands from x=%.3f y=%.3f radius=%.3f angle=%.3f", stickData.x, stickData.y, stickData.radius, stickData.angle); //--- generate motor commands --- //Note: timeout (no data received) is handled in getData method - commands = joystick_generateCommandsDriving(stickData); + commands = joystick_generateCommandsDriving(stickData, altStickMapping); //--- apply commands to motors --- //TODO make motorctl.setTarget also accept motorcommand struct directly @@ -134,16 +121,43 @@ void controlledArmchair::startHandleLoop() { motorLeft->setTarget(commands.left.state, commands.left.duty); break; - // //TODO: add other modes here } - //--- reset button event --- (only one run of handle loop) - //TODO: what if variable gets set durin above code? -> mutex around entire handle loop - if (buttonEvent == true){ - ESP_LOGI(TAG, "resetting button event"); - buttonEvent = false; + + //--- run actions based on received button button event --- + //TODO: what if variable gets set from other task during this code? -> mutex around this code + switch (buttonCount) { + case 1: //define joystick center or freeze input + if (mode == controlMode_t::JOYSTICK){ + //joystick mode: calibrate joystick + joystick_l->defineCenter(); + } else if (mode == controlMode_t::MASSAGE){ + //massage mode: toggle freeze of input (lock joystick at current values) + freezeInput = !freezeInput; + if (freezeInput){ + buzzer->beep(5, 40, 25); + } else { + buzzer->beep(1, 300, 100); + } + } + break; + + case 12: //toggle alternative joystick mapping (reverse swapped) + altStickMapping = !altStickMapping; + if (altStickMapping){ + buzzer->beep(6, 70, 50); + } else { + buzzer->beep(1, 500, 100); + } + break; } + //--- reset button event --- (only one action per run) + if (buttonCount > 0){ + ESP_LOGI(TAG, "resetting button event/count"); + buttonCount = 0; + } + //----------------------- @@ -179,7 +193,6 @@ void controlledArmchair::resetTimeout(){ void controlledArmchair::sendButtonEvent(uint8_t count){ //TODO mutex - if not replaced with queue ESP_LOGI(TAG, "setting button event"); - buttonEvent = true; buttonCount = count; } diff --git a/main/control.hpp b/main/control.hpp index 67a62b3..69d492a 100644 --- a/main/control.hpp +++ b/main/control.hpp @@ -83,6 +83,7 @@ class controlledArmchair { //store joystick data joystickData_t stickData; + bool altStickMapping; //alternative joystick mapping (reverse mapped differently) //variables for http mode uint32_t http_timestamp_lastData = 0; @@ -91,7 +92,6 @@ class controlledArmchair { bool freezeInput = false; //variable to store button event - bool buttonEvent = false; uint8_t buttonCount = 0; //definition of mode enum diff --git a/main/joystick.cpp b/main/joystick.cpp index cb9c90b..746547e 100644 --- a/main/joystick.cpp +++ b/main/joystick.cpp @@ -278,7 +278,7 @@ joystickPos_t joystick_evaluatePosition(float x, float y){ //========= joystick_CommandsDriving ========= //============================================ //function that generates commands for both motors from the joystick data -motorCommands_t joystick_generateCommandsDriving(joystickData_t data){ +motorCommands_t joystick_generateCommandsDriving(joystickData_t data, bool altStickMapping){ //struct with current data of the joystick @@ -292,12 +292,23 @@ motorCommands_t joystick_generateCommandsDriving(joystickData_t data){ motorCommands_t commands; - float dutyMax = 94; //TODO add this to config, make changeable during runtime + float dutyMax = 95; //TODO add this to config, make changeable during runtime - float dutyOffset = 10; //immedeately starts with this duty, TODO add this to config + float dutyOffset = 10; //immediately starts with this duty, TODO add this to config float dutyRange = dutyMax - dutyOffset; float ratio = fabs(data.angle) / 90; //90degree = x=0 || 0degree = y=0 + //experimental alternative control mode + if (altStickMapping == true){ + //swap BOTTOM_LEFT and BOTTOM_RIGHT + if (data.position == joystickPos_t::BOTTOM_LEFT){ + data.position = joystickPos_t::BOTTOM_RIGHT; + } + else if (data.position == joystickPos_t::BOTTOM_RIGHT){ + data.position = joystickPos_t::BOTTOM_LEFT; + } + } + switch (data.position){ case joystickPos_t::CENTER: diff --git a/main/joystick.hpp b/main/joystick.hpp index c631003..80572b5 100644 --- a/main/joystick.hpp +++ b/main/joystick.hpp @@ -107,7 +107,7 @@ class evaluatedJoystick { //============================================ //function that generates commands for both motors from the joystick data //motorCommands_t joystick_generateCommandsDriving(evaluatedJoystick joystick); -motorCommands_t joystick_generateCommandsDriving(joystickData_t data ); +motorCommands_t joystick_generateCommandsDriving(joystickData_t data, bool altStickMapping = false); From 66756b9b75690202bf02db42cc773901e43f80f2 Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Thu, 28 Jul 2022 17:54:44 +0200 Subject: [PATCH 7/7] Fix bug: massage stick freeze not resetting When switching mode from MASSAGE to other mode while joystick input frozen (button). It was still frozen when switching back to MASSAGE mode later. This is confusing and is seen as a bug -> reset locked input variable to false on mode change --- main/control.cpp | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/main/control.cpp b/main/control.cpp index 061e063..7bc9faf 100644 --- a/main/control.cpp +++ b/main/control.cpp @@ -282,7 +282,7 @@ void controlledArmchair::changeMode(controlMode_t modeNew) { break; case controlMode_t::MASSAGE: - ESP_LOGW(TAG, "switching from MASSAGE mode -> restoring fading"); + 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... //enable downfading (set to default value) motorLeft->setFade(fadeType_t::DECEL, true); @@ -290,6 +290,8 @@ void controlledArmchair::changeMode(controlMode_t modeNew) { //set upfading to default value motorLeft->setFade(fadeType_t::ACCEL, true); motorRight->setFade(fadeType_t::ACCEL, true); + //reset frozen input state + freezeInput = false; break; }