From 20873b41753ac749f29e3c5250b11d76bf6964a1 Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Sun, 19 Jun 2022 15:37:42 +0200 Subject: [PATCH 1/7] Move coordinate scaling from app to controller - joystick.cpp/hpp: - move method scaleCoordinate from joystick class to public function - modify scaleCoordinate function to accept float values instead of ADC pin, change tolerance parameters to percent instead of absolute number - change method getData to use the public function now - control.cpp: - use scaleCoordinate function in http mode - calculate radius in http mode - config.cpp - adjust tolerance thresholds for joystick to percent - App.js - disable "snap to zero" feature -> just scale joystick output to value of -1 to 1 --- main/config.cpp | 43 ++++++++++++-------- main/control.cpp | 14 ++++++- main/joystick.cpp | 96 +++++++++++++++++++++++--------------------- main/joystick.hpp | 14 ++++--- react-app/src/App.js | 21 ++++++++-- 5 files changed, 116 insertions(+), 72 deletions(-) diff --git a/main/config.cpp b/main/config.cpp index d22ffa0..b9d1e38 100644 --- a/main/config.cpp +++ b/main/config.cpp @@ -27,7 +27,7 @@ single100a_config_t configDriverRight = { .pwmFreq = 10000 }; -//configure motor contol +//--- configure motor contol --- motorctl_config_t configMotorControl = { .msFade = 900, .currentMax = 10 @@ -40,16 +40,17 @@ controlledMotor motorRight(configDriverRight, configMotorControl); + //-------------------------------------- //------- joystick configuration ------- //-------------------------------------- joystick_config_t configJoystick = { .adc_x = ADC1_CHANNEL_3, //GPIO39 .adc_y = ADC1_CHANNEL_0, //GPIO36 - //range around center-threshold of each axis the coordinates stays at 0 (adc value 0-4095) - .tolerance_zero = 100, - //threshold the coordinate snaps to -1 or 1 before configured "_max" or "_min" threshold (mechanical end) is reached (adc value 0-4095) - .tolerance_end = 80, + //range around center-threshold of each axis the coordinates stays at 0 (percentage of available range 0-100) + .tolerance_zero = 7, + //threshold the coordinate snaps to -1 or 1 before configured "_max" or "_min" threshold (mechanical end) is reached (percentage of available range 0-100) + .tolerance_end = 5, //threshold the radius jumps to 1 before the stick is at max radius (range 0-1) .tolerance_radius = 0.05, @@ -63,6 +64,27 @@ joystick_config_t configJoystick = { .y_inverted = true }; + + +//---------------------------- +//--- configure fan contol --- +//---------------------------- +fan_config_t configFanLeft = { + .gpio_fan = GPIO_NUM_2, + .msRun = 5000, + .dutyThreshold = 35 +}; +fan_config_t configFanRight = { + .gpio_fan = GPIO_NUM_15, + .msRun = 5000, + .dutyThreshold = 35 +}; + + + +//================================= +//===== create global objects ===== +//================================= //create global joystic instance evaluatedJoystick joystick(configJoystick); @@ -76,15 +98,4 @@ buzzer_t buzzer(GPIO_NUM_12, 100); controlledArmchair control(&buzzer, &motorLeft, &motorRight); -//configure fan contol -fan_config_t configFanLeft = { - .gpio_fan = GPIO_NUM_2, - .msRun = 5000, - .dutyThreshold = 35 -}; -fan_config_t configFanRight = { - .gpio_fan = GPIO_NUM_15, - .msRun = 5000, - .dutyThreshold = 35 -}; diff --git a/main/control.cpp b/main/control.cpp index 9ebd302..5c54231 100644 --- a/main/control.cpp +++ b/main/control.cpp @@ -87,11 +87,21 @@ void controlledArmchair::startHandleLoop() { //reset timestamp lastAction http_timestamp_lastData = esp_log_timestamp(); - ESP_LOGD(TAG, "received data from http queue -> generating commands\n x=%.3f y=%.3f radius=%.3f angle=%.3f", + ESP_LOGD(TAG, "received data (from queue): x=%.3f y=%.3f radius=%.3f angle=%.3f", dataRead.x, dataRead.y, dataRead.radius, dataRead.angle); + //--- scale coordinates --- + //note: scaleCoordinate function currently can not handle negative input -> add offset to input + // scaleCoordinate(input, min, max, center, tolerance_zero_per, tolerance_end_per) + dataRead.x = scaleCoordinate(dataRead.x+1, 0, 2, 1, 5, 2); //TODO: move tolerance to config (virtualJoystick or control_Config_t?) + dataRead.y = scaleCoordinate(dataRead.y+1, 0, 2, 1, 5, 2); + //--- calculate radius with new coordinates --- + dataRead.radius = sqrt(pow(dataRead.x,2) + pow(dataRead.y,2)); + ESP_LOGD(TAG, "processed/scaled data: x=%.3f y=%.3f radius=%.3f", dataRead.x, dataRead.y, dataRead.radius); + //--- generate motor commands --- //pass received joystick data from http queue to generatecommands function from joystick.hpp + ESP_LOGV(TAG, "generating commands..."); commands = joystick_generateCommandsDriving(dataRead); //--- apply commands to motors --- @@ -103,7 +113,7 @@ void controlledArmchair::startHandleLoop() { //--- timeout --- //turn off motors when motor still on and no new data received for some time if ( - (esp_log_timestamp() - http_timestamp_lastData > 3000) //no data received for x seconds + (esp_log_timestamp() - http_timestamp_lastData > 3000) //no data received for x seconds //TODO: move timeout to config && (commands.left.state != motorstate_t::IDLE || commands.right.state != motorstate_t::IDLE) //at least one motor is still running ){ ESP_LOGE(TAG, "TIMEOUT - no data received for 3s -> stopping motors"); diff --git a/main/joystick.cpp b/main/joystick.cpp index 62997c3..11e63aa 100644 --- a/main/joystick.cpp +++ b/main/joystick.cpp @@ -68,50 +68,6 @@ int evaluatedJoystick::readAdc(adc1_channel_t adc_channel, bool inverted) { -//---------------------------- -//------ getCoordinate ------- -//---------------------------- -//function to read voltage at a gpio pin and scale it to a value from -1 to 1 using the given thresholds and tolerances -float evaluatedJoystick::getCoordinate(adc1_channel_t adc_channel, bool inverted, int min, int max, int center, int tolerance_zero, int tolerance_end) { - - float coordinate = 0; - - //read voltage from adc - int input = readAdc(adc_channel, inverted); - - //define coordinate value considering the different tolerances - //--- center --- - if ((input < center+tolerance_zero) && (input > center-tolerance_zero) ) { //adc value is inside tolerance around center threshold - coordinate = 0; - } - //--- maximum --- - else if (input > max-tolerance_end) { - coordinate = 1; - } - //--- minimum --- - else if (input < min+tolerance_end) { - coordinate = -1; - } - //--- positive area --- - else if (input > center) { - float range = max - center - tolerance_zero - tolerance_end; - coordinate = (input - center - tolerance_end) / range; - } - //--- negative area --- - else if (input < center) { - float range = (center - min - tolerance_zero - tolerance_end); - coordinate = -(center-input - tolerance_end) / range; - } - - ESP_LOGD(TAG, "coordinate=%.3f, input=%d/4095, isInverted=%d", coordinate, input, inverted); - //return coordinate (-1 to 1) - return coordinate; - -} - - - - //------------------------------- //---------- getData ------------ //------------------------------- @@ -120,10 +76,11 @@ joystickData_t evaluatedJoystick::getData() { //get coordinates //TODO individual tolerances for each axis? Otherwise some parameters can be removed ESP_LOGD(TAG, "getting X coodrinate..."); - float x = getCoordinate(config.adc_x, config.x_inverted, config.x_min, config.x_max, x_center, config.tolerance_zero, config.tolerance_end); + float x = scaleCoordinate(readAdc(config.adc_x, config.x_inverted), config.x_min, config.x_max, x_center, config.tolerance_zero, config.tolerance_end); data.x = x; + ESP_LOGD(TAG, "getting Y coodrinate..."); - float y = getCoordinate(config.adc_y, config.y_inverted, config.y_min, config.y_max, y_center, config.tolerance_zero, config.tolerance_end); + float y = scaleCoordinate(readAdc(config.adc_y, config.y_inverted), config.y_min, config.y_max, y_center, config.tolerance_zero, config.tolerance_end); data.y = y; //calculate radius @@ -159,6 +116,53 @@ void evaluatedJoystick::defineCenter(){ + + +//============================== +//====== scaleCoordinate ======= +//============================== +//function that scales an input value (e.g. from adc pin) to a value from -1 to 1 using the given thresholds and tolerances +float scaleCoordinate(float input, float min, float max, float center, float tolerance_zero_per, float tolerance_end_per) { + + float coordinate = 0; + + //convert tolerance percentages to actual values of range + double tolerance_zero = (max-min) * tolerance_zero_per / 100; + double tolerance_end = (max-min) * tolerance_end_per / 100; + + //define coordinate value considering the different tolerances + //--- center --- + if ((input < center+tolerance_zero) && (input > center-tolerance_zero) ) { //adc value is inside tolerance around center threshold + coordinate = 0; + } + //--- maximum --- + else if (input > max-tolerance_end) { + coordinate = 1; + } + //--- minimum --- + else if (input < min+tolerance_end) { + coordinate = -1; + } + //--- positive area --- + else if (input > center) { + float range = max - center - tolerance_zero - tolerance_end; + coordinate = (input - center - tolerance_zero - tolerance_end) / range; + } + //--- negative area --- + else if (input < center) { + float range = (center - min - tolerance_zero - tolerance_end); + coordinate = -(center-input - tolerance_zero - tolerance_end) / range; + } + + ESP_LOGD(TAG, "scaled coordinate from %.3f to %.3f", input, coordinate); + //return coordinate (-1 to 1) + return coordinate; + +} + + + + //============================================= //========= joystick_evaluatePosition ========= //============================================= diff --git a/main/joystick.hpp b/main/joystick.hpp index f7f3a87..050c51d 100644 --- a/main/joystick.hpp +++ b/main/joystick.hpp @@ -86,8 +86,6 @@ class evaluatedJoystick { void init(); //read adc while making multiple samples with option to invert the result int readAdc(adc1_channel_t adc_channel, bool inverted = false); - //read input voltage and scale to value from -1 to 1 using the given thresholds and tolerances - float getCoordinate(adc1_channel_t adc_channel, bool inverted, int min, int max, int center, int tolerance_zero, int tolerance_end); //--- variables --- joystick_config_t config; @@ -111,9 +109,15 @@ class evaluatedJoystick { motorCommands_t joystick_generateCommandsDriving(joystickData_t data ); +//============================== +//====== scaleCoordinate ======= +//============================== +//function that scales an input value (e.g. from adc pin) to a value from -1 to 1 using the giben thresholds and tolerances +float scaleCoordinate(float input, float min, float max, float center, float tolerance_zero_per, float tolerance_end_per); -//============================================ -//========= joystick_CommandsDriving ========= -//============================================ + +//============================================= +//========= joystick_evaluatePosition ========= +//============================================= //function that defines and returns enum joystickPos from x and y coordinates joystickPos_t joystick_evaluatePosition(float x, float y); diff --git a/react-app/src/App.js b/react-app/src/App.js index 9326d2a..76921c4 100644 --- a/react-app/src/App.js +++ b/react-app/src/App.js @@ -33,7 +33,7 @@ function App() { // - snaps 0 zero for a given tolerance in percent // - rounds value do given decimal places // - TODO: add threshold it snaps to 1 / -1 (100%) toleranceEnd - const ScaleCoordinate = (input) => { + const ScaleCoordinateTolerance = (input) => { //calc tolerance threshold and available range const tolerance = joystickSize/2 * toleranceSnapToZeroPer/100; const range = joystickSize/2 - tolerance; @@ -61,6 +61,15 @@ function App() { + //---------------------------------------- + //----------- Scale coordinate ----------- + //---------------------------------------- + //simply scale coordinate from joystick to a value of -1 to 1 without applying any tolerances + const ScaleCoordinate = (input) => { + return ( input / (joystickSize/2) ).toFixed(decimalPlaces); + } + + //------------------------------------------- //------- Senda data via POST request ------- @@ -106,13 +115,19 @@ function App() { //evaluate coordinates and send to esp32 const handleMove = (e) => { //console.log("data from joystick-element X:" + e.x + " Y:" + e.y + " distance:" + e.distance); - //calculate needed variables + + //--- convert coordinates --- + //Note: tolerance (snap to zero) now handled by controller -> send raw coordinates + //const x = ScaleCoordinateTolerance(e.x); + //const y = ScaleCoordinateTolerance(e.y); const x = ScaleCoordinate(e.x); const y = ScaleCoordinate(e.y); + //--- scale radius --- const radius = (e.distance / 100).toFixed(5); + //--- calculate angle --- const angle = ( Math.atan( y / x ) * 180 / Math.PI ).toFixed(2); - //crate object with necessary data + //create object with necessary data const joystick_data={ x: x, y: y, From d32b0c567124fc835f0954629b5551c56141e28b Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Mon, 20 Jun 2022 12:31:33 +0200 Subject: [PATCH 2/7] Add timeout feature to control Add feature that switches to mode IDLE when duty of both motors did not change over certain time. control.cpp/hpp: - add private function that handles timeout - add public function that resets timeout - add slow loop with timeout handle inside control handle loop --- main/control.cpp | 84 ++++++++++++++++++++++++++++++++++++++++++++++-- main/control.hpp | 15 +++++++++ 2 files changed, 96 insertions(+), 3 deletions(-) diff --git a/main/control.cpp b/main/control.cpp index 5c54231..0cefc9f 100644 --- a/main/control.cpp +++ b/main/control.cpp @@ -126,6 +126,78 @@ void controlledArmchair::startHandleLoop() { break; //TODO: add other modes here } + + + //----------------------- + //------ slow loop ------ + //----------------------- + //this section is run about every 5s (+500ms) + 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 which detects timeout (switch to idle) + handleTimeout(); + } + + }//end while(1) +}//end startHandleLoop + + + +//----------------------------------- +//---------- resetTimeout ----------- +//----------------------------------- +void controlledArmchair::resetTimeout(){ + //TODO mutex + timestamp_lastActivity = esp_log_timestamp(); +} + + + +//------------------------------------ +//---------- handleTimeout ----------- +//------------------------------------ +uint32_t msTimeout = 30000; //TODO move this to config ##################### +float inactivityTolerance = 10; //percentage the duty can vary since last timeout check and still counts as incative + +//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 + } +} + +//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: detected [activity] since last check -> reset"); + //reset last duty and timestamp + timestamp_lastActivity = esp_log_timestamp(); + dutyLeft_lastActivity = dutyLeftNow; + dutyRight_lastActivity = dutyRightNow; + } + //no activity on any motor and msTimeout exceeded + else if (esp_log_timestamp() - timestamp_lastActivity > msTimeout){ + ESP_LOGI(TAG, "timeout check: [TIMEOUT], no activity for more than %.ds -> switch to idle", msTimeout/1000); + //toggle to idle mode + toggleIdle(); + } + else { + ESP_LOGD(TAG, "timeout check: [inactive], last activity %.1f seconds ago", (float)(esp_log_timestamp() - timestamp_lastActivity)/1000); + } } } @@ -136,6 +208,9 @@ void controlledArmchair::startHandleLoop() { //----------------------------------- //function to change to a specified control mode void controlledArmchair::changeMode(controlMode_t modeNew) { + //reset timeout timer + resetTimeout(); + //copy previous mode controlMode_t modePrevious = mode; @@ -171,6 +246,10 @@ void controlledArmchair::changeMode(controlMode_t modeNew) { ESP_LOGI(TAG, "noting to execute when changing TO this mode"); break; + case controlMode_t::IDLE: + buzzer->beep(1, 1500, 0); + break; + case controlMode_t::HTTP: ESP_LOGW(TAG, "switching to http mode -> enabling http and wifi"); //start wifi @@ -207,12 +286,11 @@ void controlledArmchair::toggleIdle() { if (mode == controlMode_t::IDLE){ changeMode(modePrevious); //restore previous mode, or default if not switched yet buzzer->beep(2, 200, 100); - ESP_LOGW(TAG, "switched mode from IDLE to %s", controlModeStr[(int)mode]); + ESP_LOGW(TAG, "toggle idle: switched mode from IDLE to %s", controlModeStr[(int)mode]); } else { modePrevious = mode; //store current mode changeMode(controlMode_t::IDLE); //set mode to IDLE - buzzer->beep(1, 1000, 0); - ESP_LOGW(TAG, "switched mode from IDLE to %s", controlModeStr[(int)mode]); + ESP_LOGW(TAG, "toggle idle: switched mode from %s to IDLE", controlModeStr[(int)mode]); } } diff --git a/main/control.hpp b/main/control.hpp index e7dbefb..87e878c 100644 --- a/main/control.hpp +++ b/main/control.hpp @@ -39,8 +39,15 @@ class controlledArmchair { //function that toggles between two modes, but prefers first argument if entirely different mode is currently active void toggleModes(controlMode_t modePrimary, controlMode_t modeSecondary); + //function that restarts timer which initiates the automatic timeout (switch to IDLE) after certain time of inactivity + void resetTimeout(); + private: + //--- functions --- + //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 handleTimeout(); + //--- objects --- buzzer_t* buzzer; controlledMotor* motorLeft; @@ -68,6 +75,14 @@ class controlledArmchair { .left = cmd_motorIdle, .right = cmd_motorIdle }; + + //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; + uint32_t timestamp_lastActivity = 0; }; From f38940b7bfa0e51ef38f3b0aecef7f23e707d6f8 Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Tue, 21 Jun 2022 10:32:08 +0200 Subject: [PATCH 3/7] Move config parameters from control to config - create new struct control_config_t with several variables previously hardcoded in control.cpp - modified constructor: add config parameter - add definition of config struct in config.cpp typedef struct control_config_t { controlMode_t defaultMode; //default mode after startup and toggling IDLE //--- timeout --- uint32_t timeoutMs; //time of inactivity after which the mode gets switched to IDLE float timeoutTolerancePer; //percentage the duty can vary between timeout checks considered still inactive //--- http mode --- float http_toleranceZeroPer;//percentage around joystick axis the coordinate snaps to 0 float http_toleranceEndPer; //percentage before joystick end the coordinate snaps to 1/-1 uint32_t http_timeoutMs; //time no new data was received before the motors get turned off --- main/config.cpp | 17 ++++++++++++++++- main/control.cpp | 21 ++++++++++++++------- main/control.hpp | 23 +++++++++++++++++++++-- 3 files changed, 51 insertions(+), 10 deletions(-) diff --git a/main/config.cpp b/main/config.cpp index b9d1e38..dd3fc05 100644 --- a/main/config.cpp +++ b/main/config.cpp @@ -39,6 +39,21 @@ controlledMotor motorRight(configDriverRight, configMotorControl); +//------------------------------ +//------- control config ------- +//------------------------------ +control_config_t configControl = { + .defaultMode = controlMode_t::JOYSTICK, //default mode after startup and toggling IDLE + //--- timeout --- + .timeoutMs = 30*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 --- + .http_toleranceZeroPer = 5, //percentage around joystick axis the coordinate snaps to 0 + .http_toleranceEndPer = 2, //percentage before joystick end the coordinate snaps to 1/-1 + .http_timeoutMs = 3000 //time no new data was received before the motors get turned off + +}; + //-------------------------------------- @@ -95,7 +110,7 @@ gpio_evaluatedSwitch buttonJoystick(GPIO_NUM_33, true, false); //pullup true, no buzzer_t buzzer(GPIO_NUM_12, 100); //create global control object -controlledArmchair control(&buzzer, &motorLeft, &motorRight); +controlledArmchair control(configControl, &buzzer, &motorLeft, &motorRight); diff --git a/main/control.cpp b/main/control.cpp index 0cefc9f..90bd4cb 100644 --- a/main/control.cpp +++ b/main/control.cpp @@ -24,15 +24,21 @@ const char* controlModeStr[7] = {"IDLE", "JOYSTICK", "MASSAGE", "HTTP", "MQTT", //-------- constructor -------- //----------------------------- controlledArmchair::controlledArmchair ( + control_config_t config_f, buzzer_t * buzzer_f, controlledMotor* motorLeft_f, controlledMotor* motorRight_f ){ + //copy configuration + config = config_f; //copy object pointers buzzer = buzzer_f; motorLeft = motorLeft_f; motorRight = motorRight_f; + //set default mode from config + modePrevious = config.defaultMode; + //TODO declare / configure controlled motors here instead of config (unnecessary that button object is globally available - only used here)? } @@ -79,6 +85,7 @@ void controlledArmchair::startHandleLoop() { break; case controlMode_t::HTTP: + //TODO: outsource this code to http.cpp? //create emptry struct for receiving data from http function joystickData_t dataRead = { }; @@ -91,10 +98,10 @@ void controlledArmchair::startHandleLoop() { dataRead.x, dataRead.y, dataRead.radius, dataRead.angle); //--- scale coordinates --- - //note: scaleCoordinate function currently can not handle negative input -> add offset to input + //note: scaleCoordinate function currently can not handle negative input -> added offset to input // scaleCoordinate(input, min, max, center, tolerance_zero_per, tolerance_end_per) - dataRead.x = scaleCoordinate(dataRead.x+1, 0, 2, 1, 5, 2); //TODO: move tolerance to config (virtualJoystick or control_Config_t?) - dataRead.y = scaleCoordinate(dataRead.y+1, 0, 2, 1, 5, 2); + dataRead.x = scaleCoordinate(dataRead.x+1, 0, 2, 1, config.http_toleranceZeroPer, config.http_toleranceEndPer); + dataRead.y = scaleCoordinate(dataRead.y+1, 0, 2, 1, config.http_toleranceZeroPer, config.http_toleranceEndPer); //--- calculate radius with new coordinates --- dataRead.radius = sqrt(pow(dataRead.x,2) + pow(dataRead.y,2)); ESP_LOGD(TAG, "processed/scaled data: x=%.3f y=%.3f radius=%.3f", dataRead.x, dataRead.y, dataRead.radius); @@ -113,7 +120,7 @@ void controlledArmchair::startHandleLoop() { //--- timeout --- //turn off motors when motor still on and no new data received for some time if ( - (esp_log_timestamp() - http_timestamp_lastData > 3000) //no data received for x seconds //TODO: move timeout to config + (esp_log_timestamp() - http_timestamp_lastData > config.http_timeoutMs) //no data received for x seconds && (commands.left.state != motorstate_t::IDLE || commands.right.state != motorstate_t::IDLE) //at least one motor is still running ){ ESP_LOGE(TAG, "TIMEOUT - no data received for 3s -> stopping motors"); @@ -124,6 +131,7 @@ void controlledArmchair::startHandleLoop() { } break; + //TODO: add other modes here } @@ -158,7 +166,6 @@ void controlledArmchair::resetTimeout(){ //------------------------------------ //---------- handleTimeout ----------- //------------------------------------ -uint32_t msTimeout = 30000; //TODO move this to config ##################### float inactivityTolerance = 10; //percentage the duty can vary since last timeout check and still counts as incative //local function that checks whether two values differ more than a given tolerance @@ -190,8 +197,8 @@ void controlledArmchair::handleTimeout(){ dutyRight_lastActivity = dutyRightNow; } //no activity on any motor and msTimeout exceeded - else if (esp_log_timestamp() - timestamp_lastActivity > msTimeout){ - ESP_LOGI(TAG, "timeout check: [TIMEOUT], no activity for more than %.ds -> switch to idle", msTimeout/1000); + 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(); } diff --git a/main/control.hpp b/main/control.hpp index 87e878c..649f14c 100644 --- a/main/control.hpp +++ b/main/control.hpp @@ -5,11 +5,27 @@ #include "buzzer.hpp" +//-------------------------------------------- +//---- struct, enum, variable declarations --- +//-------------------------------------------- //enum that decides how the motors get controlled enum class controlMode_t {IDLE, JOYSTICK, MASSAGE, HTTP, MQTT, BLUETOOTH, AUTO}; -//extern controlMode_t mode; +//string array representing the mode enum (for printing the state as string) extern const char* controlModeStr[7]; +//struct with config parameters +typedef struct control_config_t { + controlMode_t defaultMode; //default mode after startup and toggling IDLE + //--- timeout --- + uint32_t timeoutMs; //time of inactivity after which the mode gets switched to IDLE + float timeoutTolerancePer; //percentage the duty can vary between timeout checks considered still inactive + //--- http mode --- + float http_toleranceZeroPer;//percentage around joystick axis the coordinate snaps to 0 + float http_toleranceEndPer; //percentage before joystick end the coordinate snaps to 1/-1 + uint32_t http_timeoutMs; //time no new data was received before the motors get turned off +} control_config_t; + + //================================== @@ -21,6 +37,7 @@ class controlledArmchair { public: //--- constructor --- controlledArmchair ( + control_config_t config_f, buzzer_t* buzzer_f, controlledMotor* motorLeft_f, controlledMotor* motorRight_f @@ -56,6 +73,8 @@ class controlledArmchair { //---variables --- //struct for motor commands returned by generate functions of each mode motorCommands_t commands; + //struct with config parameters + control_config_t config; //variables for http mode uint32_t http_timestamp_lastData = 0; @@ -64,7 +83,7 @@ class controlledArmchair { controlMode_t mode = controlMode_t::IDLE; //variable to store mode when toggling IDLE mode - controlMode_t modePrevious = controlMode_t::JOYSTICK; //default mode + controlMode_t modePrevious; //default mode //command preset for idling motors const motorCommand_t cmd_motorIdle = { From 73325e08ce0d3ed1072aceff08bcb5913db67c91 Mon Sep 17 00:00:00 2001 From: jonny_l480 Date: Tue, 21 Jun 2022 12:15:01 +0200 Subject: [PATCH 4/7] Fix bug in scaling, Fix http cmd gen, Fix overflow Fix several bugs noticed while testing the preceding commits in dev branch: - Fix bug in function scaleCoordinate - scaling was wrong resulting in negative/inverted values at start of axis - Adjust timeout value from 30s to 5min - Fix http joystick behaivor - calculate angle, radius and evaluate position AFTER the coordinates have been scaled in control.cpp (bug introduced when switching applying tolerance on controller instead of in the app) - Add independent toleranceZero for X and Y axis -> unnecessary to have large tolerance for x axis... makes turning more sensitive - Fix stack overflow in control task - controller crashed repeatedly when logging output was enabled in control task - -> doubled stack size at task creation in main.cpp currently works, position hast to be evaluated AFTER coordinate scaling --- main/config.cpp | 5 +++-- main/control.cpp | 10 +++++++--- main/control.hpp | 3 ++- main/http.cpp | 2 +- main/joystick.cpp | 6 +++--- main/main.cpp | 2 +- 6 files changed, 17 insertions(+), 11 deletions(-) diff --git a/main/config.cpp b/main/config.cpp index dd3fc05..fdef38c 100644 --- a/main/config.cpp +++ b/main/config.cpp @@ -45,10 +45,11 @@ controlledMotor motorRight(configDriverRight, configMotorControl); control_config_t configControl = { .defaultMode = controlMode_t::JOYSTICK, //default mode after startup and toggling IDLE //--- timeout --- - .timeoutMs = 30*1000, //time of inactivity after which the mode gets switched to IDLE + .timeoutMs = 5*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 --- - .http_toleranceZeroPer = 5, //percentage around joystick axis the coordinate snaps to 0 + .http_toleranceZeroX_Per = 3, //percentage around joystick axis the coordinate snaps to 0 + .http_toleranceZeroY_Per = 10, .http_toleranceEndPer = 2, //percentage before joystick end the coordinate snaps to 1/-1 .http_timeoutMs = 3000 //time no new data was received before the motors get turned off diff --git a/main/control.cpp b/main/control.cpp index 90bd4cb..0e7fb60 100644 --- a/main/control.cpp +++ b/main/control.cpp @@ -100,15 +100,19 @@ void controlledArmchair::startHandleLoop() { //--- scale coordinates --- //note: scaleCoordinate function currently can not handle negative input -> added offset to input // scaleCoordinate(input, min, max, center, tolerance_zero_per, tolerance_end_per) - dataRead.x = scaleCoordinate(dataRead.x+1, 0, 2, 1, config.http_toleranceZeroPer, config.http_toleranceEndPer); - dataRead.y = scaleCoordinate(dataRead.y+1, 0, 2, 1, config.http_toleranceZeroPer, config.http_toleranceEndPer); - //--- calculate radius with new coordinates --- + dataRead.x = scaleCoordinate(dataRead.x+1, 0, 2, 1, config.http_toleranceZeroX_Per, config.http_toleranceEndPer); + dataRead.y = scaleCoordinate(dataRead.y+1, 0, 2, 1, config.http_toleranceZeroY_Per, config.http_toleranceEndPer); + //--- re-calculate radius, angle and position with new/scaled coordinates --- dataRead.radius = sqrt(pow(dataRead.x,2) + pow(dataRead.y,2)); + dataRead.angle = (atan(dataRead.y/dataRead.x) * 180) / 3.141; + dataRead.position = joystick_evaluatePosition(dataRead.x, dataRead.y); + ESP_LOGD(TAG, "processed/scaled data: x=%.3f y=%.3f radius=%.3f", dataRead.x, dataRead.y, dataRead.radius); //--- generate motor commands --- //pass received joystick data from http queue to generatecommands function from joystick.hpp ESP_LOGV(TAG, "generating commands..."); + ESP_LOGD(TAG, "generating commands from x=%.3f y=%.3f radius=%.3f angle=%.3f", dataRead.x, dataRead.y, dataRead.radius, dataRead.angle); commands = joystick_generateCommandsDriving(dataRead); //--- apply commands to motors --- diff --git a/main/control.hpp b/main/control.hpp index 649f14c..a053470 100644 --- a/main/control.hpp +++ b/main/control.hpp @@ -20,7 +20,8 @@ typedef struct control_config_t { uint32_t timeoutMs; //time of inactivity after which the mode gets switched to IDLE float timeoutTolerancePer; //percentage the duty can vary between timeout checks considered still inactive //--- http mode --- - float http_toleranceZeroPer;//percentage around joystick axis the coordinate snaps to 0 + float http_toleranceZeroX_Per;//percentage around joystick axis the coordinate snaps to 0 + float http_toleranceZeroY_Per; float http_toleranceEndPer; //percentage before joystick end the coordinate snaps to 1/-1 uint32_t http_timeoutMs; //time no new data was received before the motors get turned off } control_config_t; diff --git a/main/http.cpp b/main/http.cpp index 5c370d5..d885f63 100644 --- a/main/http.cpp +++ b/main/http.cpp @@ -136,7 +136,7 @@ esp_err_t on_joystick_url(httpd_req_t *req) data.angle = static_cast(angle_json->valuedouble); //--- evaluate joystick position enum --- - data.position = joystick_evaluatePosition(data.x, data.y); + //data.position = joystick_evaluatePosition(data.x, data.y); //log received and parsed values diff --git a/main/joystick.cpp b/main/joystick.cpp index 11e63aa..7322a8e 100644 --- a/main/joystick.cpp +++ b/main/joystick.cpp @@ -146,15 +146,15 @@ float scaleCoordinate(float input, float min, float max, float center, float tol //--- positive area --- else if (input > center) { float range = max - center - tolerance_zero - tolerance_end; - coordinate = (input - center - tolerance_zero - tolerance_end) / range; + coordinate = (input - center - tolerance_zero) / range; } //--- negative area --- else if (input < center) { float range = (center - min - tolerance_zero - tolerance_end); - coordinate = -(center-input - tolerance_zero - tolerance_end) / range; + coordinate = -(center-input - tolerance_zero) / range; } - ESP_LOGD(TAG, "scaled coordinate from %.3f to %.3f", input, coordinate); + ESP_LOGD(TAG, "scaled coordinate from %.3f to %.3f, tolZero=%.3f, tolEnd=%.3f", input, coordinate, tolerance_zero, tolerance_end); //return coordinate (-1 to 1) return coordinate; diff --git a/main/main.cpp b/main/main.cpp index 1c076d4..8b701a6 100644 --- a/main/main.cpp +++ b/main/main.cpp @@ -169,7 +169,7 @@ extern "C" void app_main(void) { //--- create task for control --- //------------------------------- //task that generates motor commands depending on the current mode and sends those to motorctl task - xTaskCreate(&task_control, "task_control", 2048, NULL, 5, NULL); + xTaskCreate(&task_control, "task_control", 4096, NULL, 5, NULL); //------------------------------ //--- create task for button --- From 06d0fda8ee912e625a9f6cbd760a42aa937009d7 Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Wed, 22 Jun 2022 12:29:08 +0200 Subject: [PATCH 5/7] Create httpJoystick class create new class for http mode -> move code for http mode from control.cpp to http.cpp this makes control.cpp and config.cpp more clear --- main/config.cpp | 21 ++++++++--- main/config.hpp | 4 ++ main/control.cpp | 64 ++++++++------------------------ main/control.hpp | 10 ++--- main/http.cpp | 95 ++++++++++++++++++++++++++++++++++++++---------- main/http.hpp | 50 ++++++++++++++++++++++++- 6 files changed, 164 insertions(+), 80 deletions(-) diff --git a/main/config.cpp b/main/config.cpp index fdef38c..30f6e12 100644 --- a/main/config.cpp +++ b/main/config.cpp @@ -48,15 +48,23 @@ control_config_t configControl = { .timeoutMs = 5*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 --- - .http_toleranceZeroX_Per = 3, //percentage around joystick axis the coordinate snaps to 0 - .http_toleranceZeroY_Per = 10, - .http_toleranceEndPer = 2, //percentage before joystick end the coordinate snaps to 1/-1 - .http_timeoutMs = 3000 //time no new data was received before the motors get turned off }; +//------------------------------- +//----- httpJoystick config ----- +//------------------------------- +httpJoystick_config_t configHttpJoystickMain{ + .toleranceZeroX_Per = 3, //percentage around joystick axis the coordinate snaps to 0 + .toleranceZeroY_Per = 10, + .toleranceEndPer = 2, //percentage before joystick end the coordinate snaps to 1/-1 + .timeoutMs = 3000 //time no new data was received before the motors get turned off +}; + + + //-------------------------------------- //------- joystick configuration ------- //-------------------------------------- @@ -110,8 +118,11 @@ gpio_evaluatedSwitch buttonJoystick(GPIO_NUM_33, true, false); //pullup true, no //create buzzer object on pin 12 with gap between queued events of 100ms buzzer_t buzzer(GPIO_NUM_12, 100); +//create global httpJoystick object +httpJoystick httpJoystickMain(configHttpJoystickMain); + //create global control object -controlledArmchair control(configControl, &buzzer, &motorLeft, &motorRight); +controlledArmchair control(configControl, &buzzer, &motorLeft, &motorRight, &httpJoystickMain); diff --git a/main/config.hpp b/main/config.hpp index 2dce508..20db9bc 100644 --- a/main/config.hpp +++ b/main/config.hpp @@ -8,6 +8,7 @@ #include "buzzer.hpp" #include "control.hpp" #include "fan.hpp" +#include "http.hpp" //create global controlledMotor instances for both motors @@ -26,6 +27,9 @@ extern buzzer_t buzzer; //create global control object extern controlledArmchair control; +//create global httpJoystick object +extern httpJoystick httpJoystickMain; + //configuration for fans extern fan_config_t configFanLeft; extern fan_config_t configFanRight; diff --git a/main/control.cpp b/main/control.cpp index 0e7fb60..b672977 100644 --- a/main/control.cpp +++ b/main/control.cpp @@ -12,7 +12,6 @@ extern "C" #include "config.hpp" #include "control.hpp" -#include "http.hpp" //tag for logging @@ -27,7 +26,8 @@ controlledArmchair::controlledArmchair ( control_config_t config_f, buzzer_t * buzzer_f, controlledMotor* motorLeft_f, - controlledMotor* motorRight_f + controlledMotor* motorRight_f, + httpJoystick* httpJoystick_f ){ //copy configuration @@ -36,6 +36,7 @@ controlledArmchair::controlledArmchair ( buzzer = buzzer_f; motorLeft = motorLeft_f; motorRight = motorRight_f; + httpJoystickMain_l = httpJoystick_f; //set default mode from config modePrevious = config.defaultMode; @@ -85,58 +86,25 @@ void controlledArmchair::startHandleLoop() { break; case controlMode_t::HTTP: - //TODO: outsource this code to http.cpp? //create emptry struct for receiving data from http function joystickData_t dataRead = { }; //--- get joystick data from queue --- - if( xQueueReceive( joystickDataQueue, &dataRead, pdMS_TO_TICKS(500) ) ) { - //reset timestamp lastAction - http_timestamp_lastData = esp_log_timestamp(); + //Note this function waits several seconds (httpconfig.timeoutMs) for data to arrive, otherwise Center data or NULL is returned + //TODO: as described above, when changing modes it might delay a few seconds for the change to apply + dataRead = httpJoystickMain_l->getData(); + //--- generate motor commands --- + ESP_LOGD(TAG, "generating commands from x=%.3f y=%.3f radius=%.3f angle=%.3f", dataRead.x, dataRead.y, dataRead.radius, dataRead.angle); + //Note: timeout (no data received) is handled in getData method + commands = joystick_generateCommandsDriving(dataRead); - ESP_LOGD(TAG, "received data (from queue): x=%.3f y=%.3f radius=%.3f angle=%.3f", - dataRead.x, dataRead.y, dataRead.radius, dataRead.angle); + //--- apply commands to motors --- + //TODO make motorctl.setTarget also accept motorcommand struct directly + motorRight->setTarget(commands.right.state, commands.right.duty); + motorLeft->setTarget(commands.left.state, commands.left.duty); + break; - //--- scale coordinates --- - //note: scaleCoordinate function currently can not handle negative input -> added offset to input - // scaleCoordinate(input, min, max, center, tolerance_zero_per, tolerance_end_per) - dataRead.x = scaleCoordinate(dataRead.x+1, 0, 2, 1, config.http_toleranceZeroX_Per, config.http_toleranceEndPer); - dataRead.y = scaleCoordinate(dataRead.y+1, 0, 2, 1, config.http_toleranceZeroY_Per, config.http_toleranceEndPer); - //--- re-calculate radius, angle and position with new/scaled coordinates --- - dataRead.radius = sqrt(pow(dataRead.x,2) + pow(dataRead.y,2)); - dataRead.angle = (atan(dataRead.y/dataRead.x) * 180) / 3.141; - dataRead.position = joystick_evaluatePosition(dataRead.x, dataRead.y); - - ESP_LOGD(TAG, "processed/scaled data: x=%.3f y=%.3f radius=%.3f", dataRead.x, dataRead.y, dataRead.radius); - - //--- generate motor commands --- - //pass received joystick data from http queue to generatecommands function from joystick.hpp - ESP_LOGV(TAG, "generating commands..."); - ESP_LOGD(TAG, "generating commands from x=%.3f y=%.3f radius=%.3f angle=%.3f", dataRead.x, dataRead.y, dataRead.radius, dataRead.angle); - commands = joystick_generateCommandsDriving(dataRead); - - //--- apply commands to motors --- - //TODO make motorctl.setTarget also accept motorcommand struct directly - motorRight->setTarget(commands.right.state, commands.right.duty); - motorLeft->setTarget(commands.left.state, commands.left.duty); - } - - //--- timeout --- - //turn off motors when motor still on and no new data received for some time - if ( - (esp_log_timestamp() - http_timestamp_lastData > config.http_timeoutMs) //no data received for x seconds - && (commands.left.state != motorstate_t::IDLE || commands.right.state != motorstate_t::IDLE) //at least one motor is still running - ){ - ESP_LOGE(TAG, "TIMEOUT - no data received for 3s -> stopping motors"); - //copy preset commands for idling both motors - commands = cmds_bothMotorsIdle; - motorRight->setTarget(commands.right.state, commands.right.duty); - motorLeft->setTarget(commands.left.state, commands.left.duty); - } - - break; - - //TODO: add other modes here + // //TODO: add other modes here } diff --git a/main/control.hpp b/main/control.hpp index a053470..cc83964 100644 --- a/main/control.hpp +++ b/main/control.hpp @@ -3,6 +3,7 @@ #include "motordrivers.hpp" #include "motorctl.hpp" #include "buzzer.hpp" +#include "http.hpp" //-------------------------------------------- @@ -19,11 +20,6 @@ typedef struct control_config_t { //--- timeout --- uint32_t timeoutMs; //time of inactivity after which the mode gets switched to IDLE float timeoutTolerancePer; //percentage the duty can vary between timeout checks considered still inactive - //--- http mode --- - float http_toleranceZeroX_Per;//percentage around joystick axis the coordinate snaps to 0 - float http_toleranceZeroY_Per; - float http_toleranceEndPer; //percentage before joystick end the coordinate snaps to 1/-1 - uint32_t http_timeoutMs; //time no new data was received before the motors get turned off } control_config_t; @@ -41,7 +37,8 @@ class controlledArmchair { control_config_t config_f, buzzer_t* buzzer_f, controlledMotor* motorLeft_f, - controlledMotor* motorRight_f + controlledMotor* motorRight_f, + httpJoystick* httpJoystick_f ); //--- functions --- @@ -70,6 +67,7 @@ class controlledArmchair { buzzer_t* buzzer; controlledMotor* motorLeft; controlledMotor* motorRight; + httpJoystick* httpJoystickMain_l; //---variables --- //struct for motor commands returned by generate functions of each mode diff --git a/main/http.cpp b/main/http.cpp index d885f63..a544c54 100644 --- a/main/http.cpp +++ b/main/http.cpp @@ -3,7 +3,6 @@ extern "C" #include #include "mdns.h" #include "cJSON.h" -#include "esp_http_server.h" #include "esp_spiffs.h" #include "esp_wifi.h" #include "freertos/FreeRTOS.h" @@ -14,14 +13,13 @@ extern "C" } #include "http.hpp" -#include "joystick.hpp" +#include "config.hpp" //tag for logging static const char * TAG = "http"; static httpd_handle_t server = NULL; -QueueHandle_t joystickDataQueue = xQueueCreate( 1, sizeof( struct joystickData_t ) ); //joystickData_t http_readFromJoystickQueue @@ -99,12 +97,25 @@ static esp_err_t on_default_url(httpd_req_t *req) -//=============================== -//====== joystick endpoint ====== -//=============================== -//function that is called when data is received with post request at /api/joystick -esp_err_t on_joystick_url(httpd_req_t *req) -{ +//============================== +//===== httpJoystick class ===== +//============================== +//----------------------- +//----- constructor ----- +//----------------------- +httpJoystick::httpJoystick( httpJoystick_config_t config_f ){ + //copy config struct + config = config_f; + //initialize queue for joystick data + QueueHandle_t joystickDataQueue = xQueueCreate( 1, sizeof( struct joystickData_t ) ); +} + + +//-------------------------- +//---- receiveHttpData ----- +//-------------------------- +//joystick endpoint - function that is called when data is received with post request at /api/joystick +esp_err_t httpJoystick::receiveHttpData(httpd_req_t *req){ //--- add header --- //to allow cross origin (otherwise browser fails when app is running on another host) httpd_resp_set_hdr(req, "Access-Control-Allow-Origin", "*"); @@ -134,32 +145,79 @@ esp_err_t on_joystick_url(httpd_req_t *req) data.y = static_cast(y_json->valuedouble); data.radius = static_cast(radius_json->valuedouble); data.angle = static_cast(angle_json->valuedouble); - - //--- evaluate joystick position enum --- - //data.position = joystick_evaluatePosition(data.x, data.y); - - //log received and parsed values - ESP_LOGI(TAG, "parsed values from received json: \n x=%.3f y=%.3f radius=%.3f angle=%.3f", + ESP_LOGI(TAG, "received values: x=%.3f y=%.3f radius=%.3f angle=%.3f", data.x, data.y, data.radius, data.angle); + // scaleCoordinate(input, min, max, center, tolerance_zero_per, tolerance_end_per) + data.x = scaleCoordinate(data.x+1, 0, 2, 1, config.toleranceZeroX_Per, config.toleranceEndPer); + data.y = scaleCoordinate(data.y+1, 0, 2, 1, config.toleranceZeroY_Per, config.toleranceEndPer); + + //--- re-calculate radius, angle and position with new/scaled coordinates --- + data.radius = sqrt(pow(data.x,2) + pow(data.y,2)); + data.angle = (atan(data.y/data.x) * 180) / 3.141; + data.position = joystick_evaluatePosition(data.x, data.y); + //log processed values + ESP_LOGI(TAG, "processed values: x=%.3f y=%.3f radius=%.3f angle=%.3f pos=%s", + data.x, data.y, data.radius, data.angle, joystickPosStr[(int)data.position]); + //--- free memory --- cJSON_Delete(payload); - //--- send data to control task via queue --- //xQueueSend( joystickDataQueue, ( void * )&data, ( TickType_t ) 0 ); //changed to length = 1 -> overwrite - older values are no longer relevant xQueueOverwrite( joystickDataQueue, ( void * )&data ); - //--- return http response --- httpd_resp_set_status(req, "204 NO CONTENT"); httpd_resp_send(req, NULL, 0); + return ESP_OK; } +//------------------- +//----- getData ----- +//------------------- +//wait for and return joystick data from queue, if timeout return NULL +joystickData_t httpJoystick::getData(){ + + //--- get joystick data from queue --- + if( xQueueReceive( joystickDataQueue, &dataRead, pdMS_TO_TICKS(config.timeoutMs) ) ) { + + ESP_LOGD(TAG, "getData: received data (from queue): x=%.3f y=%.3f radius=%.3f angle=%.3f", + dataRead.x, dataRead.y, dataRead.radius, dataRead.angle); + } + //--- timeout --- + //no new data received within configured timeout + else { + //send error message when last received data did NOT result in CENTER position + if (dataRead.position != joystickPos_t::CENTER) { + //change data to "joystick center" data to stop the motors + dataRead = dataCenter; + ESP_LOGE(TAG, "TIMEOUT - no data received for 3s -> set to center"); + } + } + return dataRead; +} + + + +//-------------------------------------------- +//--- receiveHttpData for httpJoystickMain --- +//-------------------------------------------- +//function that wraps pointer to member function of httpJoystickMain instance in a "normal" function which the webserver can run on joystick URL + +//declare pointer to receiveHttpData method of httpJoystick class +esp_err_t (httpJoystick::*pointerToReceiveFunc)(httpd_req_t *req) = &httpJoystick::receiveHttpData; + +esp_err_t on_joystick_url(httpd_req_t *req){ + //run pointer to receiveHttpData function of httpJoystickMain instance + return (httpJoystickMain.*pointerToReceiveFunc)(req); +} + + //============================ @@ -169,9 +227,6 @@ esp_err_t on_joystick_url(httpd_req_t *req) void http_init_server() { - - - //configure webserver httpd_config_t config = HTTPD_DEFAULT_CONFIG(); config.uri_match_fn = httpd_uri_match_wildcard; diff --git a/main/http.hpp b/main/http.hpp index 6723c13..eab53ef 100644 --- a/main/http.hpp +++ b/main/http.hpp @@ -1,6 +1,14 @@ #pragma once -extern QueueHandle_t joystickDataQueue; +extern "C" +{ +#include "esp_http_server.h" +} + +#include "joystick.hpp" + + + //============================ //===== init http server ===== @@ -21,3 +29,43 @@ void start_mdns_service(); //============================ //function that destroys the http server void http_stop_server(); + + +//============================== +//===== httpJoystick class ===== +//============================== +//class that receices that from a HTTP post request, generates and scales joystick data and provides the data in a queue + +//struct with configuration parameters +typedef struct httpJoystick_config_t { + float toleranceZeroX_Per;//percentage around joystick axis the coordinate snaps to 0 + float toleranceZeroY_Per; + float toleranceEndPer; //percentage before joystick end the coordinate snaps to 1/-1 + uint32_t timeoutMs; //time no new data was received before the motors get turned off +} httpJoystick_config_t; + + +class httpJoystick{ + public: + //--- constructor --- + httpJoystick( httpJoystick_config_t config_f ); + + //--- functions --- + joystickData_t getData(); //wait for and return joystick data from queue, if timeout return CENTER + + esp_err_t receiveHttpData(httpd_req_t *req); //function that is called when data is received with post request at /api/joystick + + private: + //--- variables --- + httpJoystick_config_t config; + QueueHandle_t joystickDataQueue = xQueueCreate( 1, sizeof( struct joystickData_t ) ); + //struct for receiving data from http function, and storing data of last update + joystickData_t dataRead; + const joystickData_t dataCenter = { + .position = joystickPos_t::CENTER, + .x = 0, + .y = 0, + .radius = 0, + .angle = 0 + }; +}; From 1da03e9429c0e4b30f796eb667f789e554ac7bd3 Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Fri, 24 Jun 2022 09:06:26 +0200 Subject: [PATCH 6/7] Add different tolerances for X and Y coordinate As already did for http joystick: - Add different config options tolerance zero for X and Y axis for normal/actual joystick. This makes it possible to set Y tolerance to a lower value resulting in a more responsive turning action, with still having a large range around X axis for turning mode --- main/config.cpp | 9 +++++---- main/joystick.cpp | 4 ++-- main/joystick.hpp | 9 +++++---- 3 files changed, 12 insertions(+), 10 deletions(-) diff --git a/main/config.cpp b/main/config.cpp index 30f6e12..c6ef330 100644 --- a/main/config.cpp +++ b/main/config.cpp @@ -71,10 +71,11 @@ httpJoystick_config_t configHttpJoystickMain{ joystick_config_t configJoystick = { .adc_x = ADC1_CHANNEL_3, //GPIO39 .adc_y = ADC1_CHANNEL_0, //GPIO36 - //range around center-threshold of each axis the coordinates stays at 0 (percentage of available range 0-100) - .tolerance_zero = 7, - //threshold the coordinate snaps to -1 or 1 before configured "_max" or "_min" threshold (mechanical end) is reached (percentage of available range 0-100) - .tolerance_end = 5, + //percentage of joystick range the coordinate of the axis snaps to 0 (0-100) + .tolerance_zeroX_per = 7, + .tolerance_zeroY_per = 3, + //percentage of joystick range the coordinate snaps to -1 or 1 before configured "_max" or "_min" threshold (mechanical end) is reached (0-100) + .tolerance_end_per = 5, //threshold the radius jumps to 1 before the stick is at max radius (range 0-1) .tolerance_radius = 0.05, diff --git a/main/joystick.cpp b/main/joystick.cpp index 7322a8e..9796a55 100644 --- a/main/joystick.cpp +++ b/main/joystick.cpp @@ -76,11 +76,11 @@ joystickData_t evaluatedJoystick::getData() { //get coordinates //TODO individual tolerances for each axis? Otherwise some parameters can be removed ESP_LOGD(TAG, "getting X coodrinate..."); - float x = scaleCoordinate(readAdc(config.adc_x, config.x_inverted), config.x_min, config.x_max, x_center, config.tolerance_zero, config.tolerance_end); + float x = scaleCoordinate(readAdc(config.adc_x, config.x_inverted), config.x_min, config.x_max, x_center, config.tolerance_zeroX_per, config.tolerance_end_per); data.x = x; ESP_LOGD(TAG, "getting Y coodrinate..."); - float y = scaleCoordinate(readAdc(config.adc_y, config.y_inverted), config.y_min, config.y_max, y_center, config.tolerance_zero, config.tolerance_end); + float y = scaleCoordinate(readAdc(config.adc_y, config.y_inverted), config.y_min, config.y_max, y_center, config.tolerance_zeroY_per, config.tolerance_end_per); data.y = y; //calculate radius diff --git a/main/joystick.hpp b/main/joystick.hpp index 050c51d..663a6da 100644 --- a/main/joystick.hpp +++ b/main/joystick.hpp @@ -31,10 +31,11 @@ typedef struct joystick_config_t { adc1_channel_t adc_x; adc1_channel_t adc_y; - //range around center-threshold of each axis the coordinates stays at 0 (adc value 0-4095) - int tolerance_zero; - //threshold the coordinate snaps to -1 or 1 before configured "_max" or "_min" threshold (mechanical end) is reached (adc value 0-4095) - int tolerance_end; + //percentage of joystick range the coordinate of the axis snaps to 0 (0-100) + int tolerance_zeroX_per; + int tolerance_zeroY_per; + //percentage of joystick range the coordinate snaps to -1 or 1 before configured "_max" or "_min" threshold (mechanical end) is reached (0-100) + int tolerance_end_per; //threshold the radius jumps to 1 before the stick is at max radius (range 0-1) float tolerance_radius; From b811f77c8ea4f45b295c804452060d3feb6ca780 Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Fri, 24 Jun 2022 21:38:40 +0200 Subject: [PATCH 7/7] Update App: Dark theme, remove radius and angle Update react web app: - add dark background color - change joystick colors - increase joystick size by 50px - change heading - remove angle and radius - not displaying anymore - not sending to api anymore Update http.cpp: - remove radius and angle from json parsing - limit radius to 1 --- main/http.cpp | 18 +++++++++++------- react-app/public/index.html | 2 +- react-app/src/App.js | 29 +++++++---------------------- 3 files changed, 19 insertions(+), 30 deletions(-) diff --git a/main/http.cpp b/main/http.cpp index a544c54..0c76184 100644 --- a/main/http.cpp +++ b/main/http.cpp @@ -133,8 +133,6 @@ esp_err_t httpJoystick::receiveHttpData(httpd_req_t *req){ //--- extract relevant items from json object --- cJSON *x_json = cJSON_GetObjectItem(payload, "x"); cJSON *y_json = cJSON_GetObjectItem(payload, "y"); - cJSON *radius_json = cJSON_GetObjectItem(payload, "radius"); - cJSON *angle_json = cJSON_GetObjectItem(payload, "angle"); //--- save items to struct --- joystickData_t data = { }; @@ -143,20 +141,26 @@ esp_err_t httpJoystick::receiveHttpData(httpd_req_t *req){ //convert json to double to float data.x = static_cast(x_json->valuedouble); data.y = static_cast(y_json->valuedouble); - data.radius = static_cast(radius_json->valuedouble); - data.angle = static_cast(angle_json->valuedouble); //log received and parsed values - ESP_LOGI(TAG, "received values: x=%.3f y=%.3f radius=%.3f angle=%.3f", - data.x, data.y, data.radius, data.angle); + ESP_LOGI(TAG, "received values: x=%.3f y=%.3f", + data.x, data.y); // scaleCoordinate(input, min, max, center, tolerance_zero_per, tolerance_end_per) data.x = scaleCoordinate(data.x+1, 0, 2, 1, config.toleranceZeroX_Per, config.toleranceEndPer); data.y = scaleCoordinate(data.y+1, 0, 2, 1, config.toleranceZeroY_Per, config.toleranceEndPer); - //--- re-calculate radius, angle and position with new/scaled coordinates --- + //--- calculate radius with new/scaled coordinates --- data.radius = sqrt(pow(data.x,2) + pow(data.y,2)); + //TODO: radius tolerance? (as in original joystick func) + //limit radius to 1 + if (data.radius > 1) { + data.radius = 1; + } + //--- calculate angle --- data.angle = (atan(data.y/data.x) * 180) / 3.141; + //--- evaluate position --- data.position = joystick_evaluatePosition(data.x, data.y); + //log processed values ESP_LOGI(TAG, "processed values: x=%.3f y=%.3f radius=%.3f angle=%.3f pos=%s", data.x, data.y, data.radius, data.angle, joystickPosStr[(int)data.position]); diff --git a/react-app/public/index.html b/react-app/public/index.html index fc4c89c..41b6646 100644 --- a/react-app/public/index.html +++ b/react-app/public/index.html @@ -10,7 +10,7 @@ /> armchair ctl - +