From 6fe6185b1953e3d24a94b58f2bc679e429e32ae7 Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Fri, 17 Jan 2020 18:53:18 -0800 Subject: [PATCH 01/29] Remove unused extern --- setup.cpp | 3 --- 1 file changed, 3 deletions(-) diff --git a/setup.cpp b/setup.cpp index 10bace7..fefd615 100644 --- a/setup.cpp +++ b/setup.cpp @@ -22,9 +22,6 @@ void setupExit(){ menuOn = 0; } - //this is used by the si5351 routines in the ubitx_5351 file -extern uint32_t si5351bx_vcoa; - static const unsigned int COLOR_TEXT = DISPLAY_WHITE; static const unsigned int COLOR_BACKGROUND = DISPLAY_BLACK; static const unsigned int COLOR_TITLE_BACKGROUND = DISPLAY_NAVY; From f0e06401150a2d5609e15eb4445ec3b7718ee01b Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Fri, 17 Jan 2020 19:05:16 -0800 Subject: [PATCH 02/29] Swap tone button for menu button for faster dev --- ubitx_ui.cpp | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/ubitx_ui.cpp b/ubitx_ui.cpp index 5e4e52b..14f8536 100644 --- a/ubitx_ui.cpp +++ b/ubitx_ui.cpp @@ -66,7 +66,7 @@ enum btn_set_e { BUTTON_15, BUTTON_10, BUTTON_WPM, - BUTTON_TON, + BUTTON_MNU, BUTTON_FRQ, BUTTON_TOTAL }; @@ -85,7 +85,7 @@ constexpr Button btn_set[BUTTON_TOTAL] PROGMEM = { {LAYOUT_BUTTON_X + 0*LAYOUT_BUTTON_PITCH_X, LAYOUT_BUTTON_Y + 0*LAYOUT_BUTTON_PITCH_Y, LAYOUT_BUTTON_WIDTH, LAYOUT_BUTTON_HEIGHT, BUTTON_RIT, "RIT", 'R'}, {LAYOUT_BUTTON_X + 1*LAYOUT_BUTTON_PITCH_X, LAYOUT_BUTTON_Y + 0*LAYOUT_BUTTON_PITCH_Y, LAYOUT_BUTTON_WIDTH, LAYOUT_BUTTON_HEIGHT, BUTTON_USB, "USB", 'U'}, {LAYOUT_BUTTON_X + 2*LAYOUT_BUTTON_PITCH_X, LAYOUT_BUTTON_Y + 0*LAYOUT_BUTTON_PITCH_Y, LAYOUT_BUTTON_WIDTH, LAYOUT_BUTTON_HEIGHT, BUTTON_LSB, "LSB", 'L'}, - {LAYOUT_BUTTON_X + 3*LAYOUT_BUTTON_PITCH_X, LAYOUT_BUTTON_Y + 0*LAYOUT_BUTTON_PITCH_Y, LAYOUT_BUTTON_WIDTH, LAYOUT_BUTTON_HEIGHT, BUTTON_CW , "CW", 'M'}, + {LAYOUT_BUTTON_X + 3*LAYOUT_BUTTON_PITCH_X, LAYOUT_BUTTON_Y + 0*LAYOUT_BUTTON_PITCH_Y, LAYOUT_BUTTON_WIDTH, LAYOUT_BUTTON_HEIGHT, BUTTON_CW , "CW", 'C'}, {LAYOUT_BUTTON_X + 4*LAYOUT_BUTTON_PITCH_X, LAYOUT_BUTTON_Y + 0*LAYOUT_BUTTON_PITCH_Y, LAYOUT_BUTTON_WIDTH, LAYOUT_BUTTON_HEIGHT, BUTTON_SPL, "SPL", 'S'}, {LAYOUT_BUTTON_X + 0*LAYOUT_BUTTON_PITCH_X, LAYOUT_BUTTON_Y + 1*LAYOUT_BUTTON_PITCH_Y, LAYOUT_BUTTON_WIDTH, LAYOUT_BUTTON_HEIGHT, BUTTON_80, "80", '8'}, @@ -97,7 +97,7 @@ constexpr Button btn_set[BUTTON_TOTAL] PROGMEM = { {LAYOUT_BUTTON_X + 0*LAYOUT_BUTTON_PITCH_X, LAYOUT_BUTTON_Y + 2*LAYOUT_BUTTON_PITCH_Y, LAYOUT_BUTTON_WIDTH, LAYOUT_BUTTON_HEIGHT, BUTTON_15 , "15", '5'}, {LAYOUT_BUTTON_X + 1*LAYOUT_BUTTON_PITCH_X, LAYOUT_BUTTON_Y + 2*LAYOUT_BUTTON_PITCH_Y, LAYOUT_BUTTON_WIDTH, LAYOUT_BUTTON_HEIGHT, BUTTON_10 , "10", '1'}, {LAYOUT_BUTTON_X + 2*LAYOUT_BUTTON_PITCH_X, LAYOUT_BUTTON_Y + 2*LAYOUT_BUTTON_PITCH_Y, LAYOUT_BUTTON_WIDTH, LAYOUT_BUTTON_HEIGHT, BUTTON_WPM, "WPM", 'W'}, - {LAYOUT_BUTTON_X + 3*LAYOUT_BUTTON_PITCH_X, LAYOUT_BUTTON_Y + 2*LAYOUT_BUTTON_PITCH_Y, LAYOUT_BUTTON_WIDTH, LAYOUT_BUTTON_HEIGHT, BUTTON_TON, "TON", 'T'}, + {LAYOUT_BUTTON_X + 3*LAYOUT_BUTTON_PITCH_X, LAYOUT_BUTTON_Y + 2*LAYOUT_BUTTON_PITCH_Y, LAYOUT_BUTTON_WIDTH, LAYOUT_BUTTON_HEIGHT, BUTTON_MNU, "MNU", 'M'}, {LAYOUT_BUTTON_X + 4*LAYOUT_BUTTON_PITCH_X, LAYOUT_BUTTON_Y + 2*LAYOUT_BUTTON_PITCH_Y, LAYOUT_BUTTON_WIDTH, LAYOUT_BUTTON_HEIGHT, BUTTON_FRQ, "FRQ", 'F'}, }; @@ -962,9 +962,9 @@ void doCommand(Button* button){ setCwSpeed(); break; } - case BUTTON_TON: + case BUTTON_MNU: { - setCwTone(); + doSetup2(); break; } default: From ebaf7c44de4e3476f962c3b50d3d0cc4bb338cc0 Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Fri, 17 Jan 2020 21:41:52 -0800 Subject: [PATCH 03/29] Do most of the refactoring required to make menus easy to expand --- setup.cpp | 190 ++++++++++++++++++++++++++---------------------------- 1 file changed, 91 insertions(+), 99 deletions(-) diff --git a/setup.cpp b/setup.cpp index fefd615..097feb0 100644 --- a/setup.cpp +++ b/setup.cpp @@ -18,10 +18,6 @@ * - If the menu item is NOT clicked on, then the menu's prompt is to be displayed */ -void setupExit(){ - menuOn = 0; -} - static const unsigned int COLOR_TEXT = DISPLAY_WHITE; static const unsigned int COLOR_BACKGROUND = DISPLAY_BLACK; static const unsigned int COLOR_TITLE_BACKGROUND = DISPLAY_NAVY; @@ -190,7 +186,6 @@ void setupCwDelay(){ SaveSettingsToEeprom(); active_delay(500); - setupExit(); } void setupKeyer(){ @@ -243,88 +238,107 @@ void setupKeyer(){ globalSettings.keyerMode = tmp_mode; SaveSettingsToEeprom(); - - setupExit(); } +struct MenuItem_t { + const char* const ItemName; + const void (*OnSelect)(); +}; + +void runMenu(const MenuItem_t* const menu_items, const uint16_t num_items); +#define RUN_MENU(menu) runMenu(menu,sizeof(menu)/sizeof(menu[0])) + +const char MT_CAL [] PROGMEM = "Calibrations"; const char MI_SET_FREQ [] PROGMEM = "Set Freq..."; const char MI_SET_BFO [] PROGMEM = "Set BFO..."; +const char MI_TOUCH [] PROGMEM = "Touch Screen..."; +const MenuItem_t calibrationMenu [] PROGMEM { + {MT_CAL,nullptr},//Title + {MI_SET_FREQ,setupFreq}, + {MI_SET_BFO,setupBFO}, + {MI_TOUCH,setupTouch}, +}; +void runCalibrationMenu(){RUN_MENU(calibrationMenu);} + +const char MT_CW [] PROGMEM = "CW/Morse Setup"; const char MI_CW_DELAY [] PROGMEM = "CW Delay..."; const char MI_CW_KEYER [] PROGMEM = "CW Keyer..."; -const char MI_TOUCH [] PROGMEM = "Touch Screen..."; +const MenuItem_t cwMenu [] PROGMEM { + {MT_CW,nullptr},//Title + {MI_CW_DELAY,setupCwDelay}, + {MI_CW_KEYER,setupKeyer}, +}; +void runCwMenu(){RUN_MENU(cwMenu);} + +const char MT_SETTINGS [] PROGMEM = "Settings"; +const MenuItem_t mainMenu [] PROGMEM { + {MT_SETTINGS,nullptr},//Title + {MT_CAL,runCalibrationMenu}, + {MT_CW,runCwMenu}, +}; + const char MI_EXIT [] PROGMEM = "Exit"; +const MenuItem_t exitMenu PROGMEM = {MI_EXIT,nullptr}; -enum MenuIds { - MENU_SET_FREQ, - MENU_SET_BFO, - MENU_CW_DELAY, - MENU_CW_KEYER, - MENU_TOUCH, - MENU_EXIT, - MENU_TOTAL -}; - -const char* const menuItems [MENU_TOTAL] PROGMEM { - MI_SET_FREQ, - MI_SET_BFO, - MI_CW_DELAY, - MI_CW_KEYER, - MI_TOUCH, - MI_EXIT -}; - -void drawSetupMenu(){ +void drawMenu(const MenuItem_t* const items, const uint16_t num_items) +{ displayClear(COLOR_BACKGROUND); - strcpy_P(b,(const char*)F("Setup")); + MenuItem_t mi = {"",nullptr}; + memcpy_P(&mi,&items[0],sizeof(mi)); + strcpy_P(b,mi.ItemName); displayText(b, LAYOUT_TITLE_X, LAYOUT_TITLE_Y, LAYOUT_TITLE_WIDTH, LAYOUT_TITLE_HEIGHT, COLOR_TEXT, COLOR_TITLE_BACKGROUND, COLOR_ACTIVE_BORDER); - for(unsigned int i = 0; i < MENU_TOTAL; ++i){ - strcpy_P(b,(const char*)pgm_read_word(&(menuItems[i]))); - displayText(b, LAYOUT_ITEM_X, LAYOUT_ITEM_Y + i*LAYOUT_ITEM_PITCH_Y, LAYOUT_ITEM_WIDTH, LAYOUT_ITEM_HEIGHT, COLOR_TEXT, COLOR_BACKGROUND, COLOR_INACTIVE_BORDER); + for(unsigned int i = 1; i < num_items; ++i){ + memcpy_P(&mi,&items[i],sizeof(mi)); + strcpy_P(b,mi.ItemName); + displayText(b, LAYOUT_ITEM_X, LAYOUT_ITEM_Y + (i-1)*LAYOUT_ITEM_PITCH_Y, LAYOUT_ITEM_WIDTH, LAYOUT_ITEM_HEIGHT, COLOR_TEXT, COLOR_BACKGROUND, COLOR_INACTIVE_BORDER); } + memcpy_P(&mi,&exitMenu,sizeof(mi)); + strcpy_P(b,mi.ItemName); + displayText(b, LAYOUT_ITEM_X, LAYOUT_ITEM_Y + (num_items-1)*LAYOUT_ITEM_PITCH_Y, LAYOUT_ITEM_WIDTH, LAYOUT_ITEM_HEIGHT, COLOR_TEXT, COLOR_BACKGROUND, COLOR_INACTIVE_BORDER); } -void movePuck(int i){ - static int prevPuck = 1;//Start value at 1 so that on init, when we get called with 0, we'll update - +void movePuck(unsigned int old_index, unsigned int new_index) +{ //Don't update if we're already on the right selection - if(prevPuck == i){ + if(old_index == new_index){ return; } - - //Clear old - displayRect(LAYOUT_ITEM_X, LAYOUT_ITEM_Y + (prevPuck*LAYOUT_ITEM_PITCH_Y), LAYOUT_ITEM_WIDTH, LAYOUT_ITEM_HEIGHT, COLOR_INACTIVE_BORDER); + else if(((unsigned int)-1) != old_index){ + //Clear old + displayRect(LAYOUT_ITEM_X, LAYOUT_ITEM_Y + (old_index*LAYOUT_ITEM_PITCH_Y), LAYOUT_ITEM_WIDTH, LAYOUT_ITEM_HEIGHT, COLOR_INACTIVE_BORDER); + } //Draw new - displayRect(LAYOUT_ITEM_X, LAYOUT_ITEM_Y + (i*LAYOUT_ITEM_PITCH_Y), LAYOUT_ITEM_WIDTH, LAYOUT_ITEM_HEIGHT, COLOR_ACTIVE_BORDER); - prevPuck = i; - + displayRect(LAYOUT_ITEM_X, LAYOUT_ITEM_Y + (new_index*LAYOUT_ITEM_PITCH_Y), LAYOUT_ITEM_WIDTH, LAYOUT_ITEM_HEIGHT, COLOR_ACTIVE_BORDER); } -void doSetup2(){ +void runMenu(const MenuItem_t* const menu_items, const uint16_t num_items) +{ static const unsigned int COUNTS_PER_ITEM = 10; - int select=0, i, btnState; + const unsigned int MAX_KNOB_VALUE = num_items*COUNTS_PER_ITEM - 1; + int knob_sum = 0; + unsigned int old_index = 0; - drawSetupMenu(); - movePuck(select); + drawMenu(menu_items,num_items); + movePuck(1,0);//Force draw of puck //wait for the button to be raised up - while(btnDown()) + while(btnDown()){ active_delay(50); + } active_delay(50); //debounce - - menuOn = 2; - while (menuOn){ - i = enc_read(); + while (true){ + knob_sum += enc_read(); + if(knob_sum < 0){ + knob_sum = 0; + } + else if(MAX_KNOB_VALUE < knob_sum){ + knob_sum = MAX_KNOB_VALUE; + } - if (i > 0){ - if (select + i < MENU_TOTAL*COUNTS_PER_ITEM) - select += i; - movePuck(select/COUNTS_PER_ITEM); - } - if (i < 0 && select + i >= 0){ - select += i; //caught ya, i is already -ve here, so you add it - movePuck(select/COUNTS_PER_ITEM); - } + uint16_t index = knob_sum/COUNTS_PER_ITEM; + movePuck(old_index,index); + old_index = index; if (!btnDown()){ active_delay(50); @@ -335,50 +349,28 @@ void doSetup2(){ while(btnDown()){ active_delay(50); } - active_delay(300); + active_delay(50);//debounce - switch(select/COUNTS_PER_ITEM){ - case MENU_SET_FREQ: - { - setupFreq(); - break; - } - case MENU_SET_BFO: - { - setupBFO(); - break; - } - case MENU_CW_DELAY: - { - setupCwDelay(); - break; - } - case MENU_CW_KEYER: - { - setupKeyer(); - break; - } - case MENU_TOUCH: - { - setupTouch(); - break; - } - case MENU_EXIT: - default: - { - menuOn = 0; - break; - } - }//switch - //redraw - drawSetupMenu(); + if(num_items-1 > index){ + MenuItem_t mi = {"",nullptr}; + memcpy_P(&mi,&menu_items[index+1],sizeof(mi));//The 0th element in the array is the title, so offset by 1 + mi.OnSelect(); + drawMenu(menu_items,num_items);//Need to re-render, since whatever ran just now is assumed to have drawn something + old_index = -1;//Force redraw + } + else{ + break; + } } //debounce the button - while(btnDown()) + while(btnDown()){ active_delay(50); - active_delay(50); + } + active_delay(50);//debounce +} - checkCAT(); +void doSetup2(){ + RUN_MENU(mainMenu); guiUpdate(); } From e0bc5999c8bd9e73cca19ac52aa6d5c9e1716e2d Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Fri, 17 Jan 2020 22:22:06 -0800 Subject: [PATCH 04/29] De-duplicate keyer strings --- setup.cpp | 39 ++++++++++++++++----------------------- 1 file changed, 16 insertions(+), 23 deletions(-) diff --git a/setup.cpp b/setup.cpp index 097feb0..c1e1208 100644 --- a/setup.cpp +++ b/setup.cpp @@ -188,24 +188,27 @@ void setupCwDelay(){ active_delay(500); } -void setupKeyer(){ - displayDialog(F("Set CW Keyer"),F("Press tune to Save")); - - if(KeyerMode_e::KEYER_STRAIGHT == globalSettings.keyerMode){ - strcpy_P(c,(const char*)F("< Hand Key >")); - displayText(c, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); +void formatKeyerEnum(char* output, const KeyerMode_e mode) +{ + if(KeyerMode_e::KEYER_STRAIGHT == mode){ + strcpy_P(output,(const char*)F("< Hand Key >")); } - else if(KeyerMode_e::KEYER_IAMBIC_A == globalSettings.keyerMode){ - strcpy_P(c,(const char*)F("< Iambic A >")); - displayText(c, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); + else if(KeyerMode_e::KEYER_IAMBIC_A == mode){ + strcpy_P(output,(const char*)F("< Iambic A >")); } else{ - strcpy_P(c,(const char*)F("< Iambic B >")); - displayText(c, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); + strcpy_P(output,(const char*)F("< Iambic B >")); } +} + +void setupKeyer(){ + displayDialog(F("Set CW Keyer"),F("Press tune to Save")); int knob = 0; uint32_t tmp_mode = globalSettings.keyerMode; + formatKeyerEnum(c, tmp_mode); + displayText(c, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); + while (!btnDown()) { knob = enc_read(); @@ -220,18 +223,8 @@ void setupKeyer(){ tmp_mode++; } - if (KeyerMode_e::KEYER_STRAIGHT == tmp_mode){ - strcpy_P(c,(const char*)F("< Hand Key >")); - displayText(c, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); - } - else if(KeyerMode_e::KEYER_IAMBIC_A == tmp_mode){ - strcpy_P(c,(const char*)F("< Iambic A >")); - displayText(c, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); - } - else if (KeyerMode_e::KEYER_IAMBIC_B == tmp_mode){ - strcpy_P(c,(const char*)F("< Iambic B >")); - displayText(c, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); - } + formatKeyerEnum(c,tmp_mode); + displayText(c, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); } active_delay(500); From b100da145ecac69476a435bf020746b81ae04f3c Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Fri, 17 Jan 2020 22:27:12 -0800 Subject: [PATCH 05/29] Rename some of the menu items --- setup.cpp | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/setup.cpp b/setup.cpp index c1e1208..145edf7 100644 --- a/setup.cpp +++ b/setup.cpp @@ -242,9 +242,9 @@ void runMenu(const MenuItem_t* const menu_items, const uint16_t num_items); #define RUN_MENU(menu) runMenu(menu,sizeof(menu)/sizeof(menu[0])) const char MT_CAL [] PROGMEM = "Calibrations"; -const char MI_SET_FREQ [] PROGMEM = "Set Freq..."; -const char MI_SET_BFO [] PROGMEM = "Set BFO..."; -const char MI_TOUCH [] PROGMEM = "Touch Screen..."; +const char MI_SET_FREQ [] PROGMEM = "Local Osc Frequency"; +const char MI_SET_BFO [] PROGMEM = "Beat Osc Frequency"; +const char MI_TOUCH [] PROGMEM = "Touch Screen"; const MenuItem_t calibrationMenu [] PROGMEM { {MT_CAL,nullptr},//Title {MI_SET_FREQ,setupFreq}, @@ -254,8 +254,8 @@ const MenuItem_t calibrationMenu [] PROGMEM { void runCalibrationMenu(){RUN_MENU(calibrationMenu);} const char MT_CW [] PROGMEM = "CW/Morse Setup"; -const char MI_CW_DELAY [] PROGMEM = "CW Delay..."; -const char MI_CW_KEYER [] PROGMEM = "CW Keyer..."; +const char MI_CW_DELAY [] PROGMEM = "Tx/Rx Switching Delay"; +const char MI_CW_KEYER [] PROGMEM = "Keyer Type"; const MenuItem_t cwMenu [] PROGMEM { {MT_CW,nullptr},//Title {MI_CW_DELAY,setupCwDelay}, From bc83bcd5d185f3c7d88f9e37c0d8be986ee40ede Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Fri, 17 Jan 2020 22:57:41 -0800 Subject: [PATCH 06/29] Move CW WPM and Hz to setup menu --- setup.cpp | 63 ++++++++++++++++++++++++++++++++++++++++++++---- ubitx_ui.cpp | 68 ++-------------------------------------------------- 2 files changed, 61 insertions(+), 70 deletions(-) diff --git a/setup.cpp b/setup.cpp index 145edf7..97d6f5f 100644 --- a/setup.cpp +++ b/setup.cpp @@ -161,7 +161,6 @@ void setupCwDelay(){ displayDialog(F("Set CW T/R Delay"),F("Press tune to Save")); - active_delay(500); prev_cw_delay = globalSettings.cwActiveTimeoutMs; ltoa(globalSettings.cwActiveTimeoutMs, b, 10); @@ -185,7 +184,6 @@ void setupCwDelay(){ } SaveSettingsToEeprom(); - active_delay(500); } void formatKeyerEnum(char* output, const KeyerMode_e mode) @@ -233,6 +231,59 @@ void setupKeyer(){ SaveSettingsToEeprom(); } +void setupCwSpeed() +{ + displayDialog(F("Set CW Speed (WPM)"),F("Press tune to Save")); + + unsigned int wpm = 1200/globalSettings.cwDitDurationMs; + + itoa(wpm, b, 10); + displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); + + while (!btnDown()){ + int knob = enc_read(); + + if (knob < 0 && wpm > 1) + --wpm; + else if (knob > 0 && wpm < 100) + ++wpm; + else + continue;//don't update the frequency or the display + + itoa(wpm, b, 10); + displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); + } + + globalSettings.cwDitDurationMs = 1200/wpm; + SaveSettingsToEeprom(); +} + +void setupCwTone(){ + displayDialog(F("Set CW Tone (Hz)"),F("Press tune to Save")); + + tone(CW_TONE, globalSettings.cwSideToneFreq); + itoa(globalSettings.cwSideToneFreq, b, 10); + displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); + + while(!btnDown()){ + int knob = enc_read(); + + if (knob > 0 && globalSettings.cwSideToneFreq < 2000) + globalSettings.cwSideToneFreq += 10; + else if (knob < 0 && globalSettings.cwSideToneFreq > 100 ) + globalSettings.cwSideToneFreq -= 10; + else + continue; //don't update the frequency or the display + + tone(CW_TONE, globalSettings.cwSideToneFreq); + itoa(globalSettings.cwSideToneFreq, b, 10); + displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); + } + noTone(CW_TONE); + + SaveSettingsToEeprom(); +} + struct MenuItem_t { const char* const ItemName; const void (*OnSelect)(); @@ -242,8 +293,8 @@ void runMenu(const MenuItem_t* const menu_items, const uint16_t num_items); #define RUN_MENU(menu) runMenu(menu,sizeof(menu)/sizeof(menu[0])) const char MT_CAL [] PROGMEM = "Calibrations"; -const char MI_SET_FREQ [] PROGMEM = "Local Osc Frequency"; -const char MI_SET_BFO [] PROGMEM = "Beat Osc Frequency"; +const char MI_SET_FREQ [] PROGMEM = "Local Oscillator"; +const char MI_SET_BFO [] PROGMEM = "Beat Frequency Osc (BFO)"; const char MI_TOUCH [] PROGMEM = "Touch Screen"; const MenuItem_t calibrationMenu [] PROGMEM { {MT_CAL,nullptr},//Title @@ -254,10 +305,14 @@ const MenuItem_t calibrationMenu [] PROGMEM { void runCalibrationMenu(){RUN_MENU(calibrationMenu);} const char MT_CW [] PROGMEM = "CW/Morse Setup"; +const char MI_CW_SPEED [] PROGMEM = "Play Speed (WPM)"; +const char MI_CW_TONE [] PROGMEM = "Tone Frequency"; const char MI_CW_DELAY [] PROGMEM = "Tx/Rx Switching Delay"; const char MI_CW_KEYER [] PROGMEM = "Keyer Type"; const MenuItem_t cwMenu [] PROGMEM { {MT_CW,nullptr},//Title + {MI_CW_SPEED,setupCwSpeed}, + {MI_CW_TONE,setupCwTone}, {MI_CW_DELAY,setupCwDelay}, {MI_CW_KEYER,setupKeyer}, }; diff --git a/ubitx_ui.cpp b/ubitx_ui.cpp index 14f8536..8457f11 100644 --- a/ubitx_ui.cpp +++ b/ubitx_ui.cpp @@ -65,7 +65,7 @@ enum btn_set_e { BUTTON_17, BUTTON_15, BUTTON_10, - BUTTON_WPM, + BUTTON_BLANK_1, BUTTON_MNU, BUTTON_FRQ, BUTTON_TOTAL @@ -96,7 +96,7 @@ constexpr Button btn_set[BUTTON_TOTAL] PROGMEM = { {LAYOUT_BUTTON_X + 0*LAYOUT_BUTTON_PITCH_X, LAYOUT_BUTTON_Y + 2*LAYOUT_BUTTON_PITCH_Y, LAYOUT_BUTTON_WIDTH, LAYOUT_BUTTON_HEIGHT, BUTTON_15 , "15", '5'}, {LAYOUT_BUTTON_X + 1*LAYOUT_BUTTON_PITCH_X, LAYOUT_BUTTON_Y + 2*LAYOUT_BUTTON_PITCH_Y, LAYOUT_BUTTON_WIDTH, LAYOUT_BUTTON_HEIGHT, BUTTON_10 , "10", '1'}, - {LAYOUT_BUTTON_X + 2*LAYOUT_BUTTON_PITCH_X, LAYOUT_BUTTON_Y + 2*LAYOUT_BUTTON_PITCH_Y, LAYOUT_BUTTON_WIDTH, LAYOUT_BUTTON_HEIGHT, BUTTON_WPM, "WPM", 'W'}, + {LAYOUT_BUTTON_X + 2*LAYOUT_BUTTON_PITCH_X, LAYOUT_BUTTON_Y + 2*LAYOUT_BUTTON_PITCH_Y, LAYOUT_BUTTON_WIDTH, LAYOUT_BUTTON_HEIGHT, BUTTON_BLANK_1, "", '\0'}, {LAYOUT_BUTTON_X + 3*LAYOUT_BUTTON_PITCH_X, LAYOUT_BUTTON_Y + 2*LAYOUT_BUTTON_PITCH_Y, LAYOUT_BUTTON_WIDTH, LAYOUT_BUTTON_HEIGHT, BUTTON_MNU, "MNU", 'M'}, {LAYOUT_BUTTON_X + 4*LAYOUT_BUTTON_PITCH_X, LAYOUT_BUTTON_Y + 2*LAYOUT_BUTTON_PITCH_Y, LAYOUT_BUTTON_WIDTH, LAYOUT_BUTTON_HEIGHT, BUTTON_FRQ, "FRQ", 'F'}, }; @@ -808,65 +808,6 @@ void switchBand(uint32_t bandfreq){ saveVFOs(); } -void setCwSpeed() -{ - int wpm = 1200/globalSettings.cwDitDurationMs; - - wpm = getValueByKnob(1, 100, 1, wpm,F("CW: "),F(" WPM")); - - globalSettings.cwDitDurationMs = 1200/wpm; - SaveSettingsToEeprom(); - active_delay(500); - drawStatusbar(); - //printLine2(""); - //updateDisplay(); -} - -void setCwTone(){ - int knob = 0; - int prev_sideTone; - - tone(CW_TONE, globalSettings.cwSideToneFreq); - itoa(globalSettings.cwSideToneFreq, c, 10); - strcpy_P(b,(const char*)F("CW Tone: ")); - strcat(b, c); - strcat_P(b,(const char*)F(" Hz")); - drawCommandbar(b); - - //disable all clock 1 and clock 2 - while (digitalRead(PTT) == HIGH && !btnDown()) - { - knob = enc_read(); - - if (knob > 0 && globalSettings.cwSideToneFreq < 2000) - globalSettings.cwSideToneFreq += 10; - else if (knob < 0 && globalSettings.cwSideToneFreq > 100 ) - globalSettings.cwSideToneFreq -= 10; - else - continue; //don't update the frequency or the display - - tone(CW_TONE, globalSettings.cwSideToneFreq); - itoa(globalSettings.cwSideToneFreq, c, 10); - strcpy_P(b,(const char*)F("CW Tone: ")); - strcat(b, c); - strcat_P(b,(const char*)F(" Hz")); - drawCommandbar(b); - //printLine2(b); - - checkCAT(); - active_delay(20); - } - noTone(CW_TONE); - - SaveSettingsToEeprom(); - - b[0] = 0; - drawCommandbar(b); - drawStatusbar(); - //printLine2(""); - //updateDisplay(); -} - void doCommand(Button* button){ //Serial.print(F("Doing command: ")); //Serial.print(button->text); @@ -957,11 +898,6 @@ void doCommand(Button* button){ enterFreq(); break; } - case BUTTON_WPM: - { - setCwSpeed(); - break; - } case BUTTON_MNU: { doSetup2(); From e1d0ee1c4343defb37e8a943d5d6cd162e75de1e Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Fri, 17 Jan 2020 23:10:38 -0800 Subject: [PATCH 07/29] If button is down, load defaults and go from there --- ubitx_v6.3.1_code.ino | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/ubitx_v6.3.1_code.ino b/ubitx_v6.3.1_code.ino index 8871f4f..0d18816 100644 --- a/ubitx_v6.3.1_code.ino +++ b/ubitx_v6.3.1_code.ino @@ -473,12 +473,13 @@ void setup() //Run initial calibration routine if button is pressed during power up if(btnDown()){ + LoadDefaultSettings(); setupTouch(); SetActiveVfoMode(VfoMode_e::VFO_MODE_USB); - setFrequency(10000000l); + setFrequency(10000000L); setupFreq(); SetActiveVfoMode(VfoMode_e::VFO_MODE_LSB); - setFrequency(7100000l); + setFrequency(7100000L); setupBFO(); } From 897b5c5de55f60c68251a854b3b58cafa511e597 Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Fri, 17 Jan 2020 23:10:53 -0800 Subject: [PATCH 08/29] Add reset option --- setup.cpp | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/setup.cpp b/setup.cpp index 97d6f5f..037d926 100644 --- a/setup.cpp +++ b/setup.cpp @@ -284,6 +284,17 @@ void setupCwTone(){ SaveSettingsToEeprom(); } +void setupResetAll() +{ + LoadDefaultSettings(); + SaveSettingsToEeprom(); + while(btnDown()){ + active_delay(50); + } + active_delay(50); + setup(); +} + struct MenuItem_t { const char* const ItemName; const void (*OnSelect)(); @@ -319,10 +330,12 @@ const MenuItem_t cwMenu [] PROGMEM { void runCwMenu(){RUN_MENU(cwMenu);} const char MT_SETTINGS [] PROGMEM = "Settings"; +const char MI_RESET [] PROGMEM = "Reset all Cals/Settings"; const MenuItem_t mainMenu [] PROGMEM { {MT_SETTINGS,nullptr},//Title {MT_CAL,runCalibrationMenu}, {MT_CW,runCwMenu}, + {MI_RESET,setupResetAll}, }; const char MI_EXIT [] PROGMEM = "Exit"; From f79ca7c693623966434ccf4339cb6221cce00ad1 Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Fri, 17 Jan 2020 23:23:56 -0800 Subject: [PATCH 09/29] Give a reset dialog so it can't be a single mistake to reset --- setup.cpp | 37 ++++++++++++++++++++++++++++++++++--- 1 file changed, 34 insertions(+), 3 deletions(-) diff --git a/setup.cpp b/setup.cpp index 037d926..12e7afd 100644 --- a/setup.cpp +++ b/setup.cpp @@ -286,13 +286,44 @@ void setupCwTone(){ void setupResetAll() { - LoadDefaultSettings(); - SaveSettingsToEeprom(); + displayDialog(F("Reset all cals and settings?"),F("Press tune to Confirm")); + strcpy_P(b,(const char*)F("No")); + displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); + + bool reset_all = false; + while(!btnDown()){ + int knob = enc_read(); + + if(knob > 0){ + reset_all = true; + } + else if(knob < 0){ + reset_all = false; + } + else{ + continue; + } + + if(reset_all){ + strcpy_P(b,(const char*)F("Yes")); + } + else{ + strcpy_P(b,(const char*)F("No")); + } + + displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); + } + while(btnDown()){ active_delay(50); } active_delay(50); - setup(); + + if(reset_all){ + LoadDefaultSettings(); + SaveSettingsToEeprom(); + setup(); + } } struct MenuItem_t { From 8212ecf1e9f1286d474548b8a49046354c80d09a Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Sat, 18 Jan 2020 23:34:41 -0800 Subject: [PATCH 10/29] Proof of concept generic setting menu for cw tone --- setup.cpp | 88 ++++++++++++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 87 insertions(+), 1 deletion(-) diff --git a/setup.cpp b/setup.cpp index 12e7afd..158b0c7 100644 --- a/setup.cpp +++ b/setup.cpp @@ -66,6 +66,92 @@ void displayDialog(const __FlashStringHelper* title, const __FlashStringHelper* displayText(c, LAYOUT_INSTRUCTION_TEXT_X, LAYOUT_INSTRUCTION_TEXT_Y, LAYOUT_INSTRUCTION_TEXT_WIDTH, LAYOUT_INSTRUCTION_TEXT_HEIGHT, COLOR_TEXT, COLOR_BACKGROUND, COLOR_BACKGROUND); } +struct SettingScreen_t { + const char* const Title; + const char* const AdditionalText; + const uint16_t KnobDivider; + const uint16_t StepSize; + void (*Initialize)(long int* start_value_out); + void (*Validate)(const long int candidate_value_in, long int* validated_value_out); + void (*OnValueChange)(const long int new_value, char* buff_out, const size_t buff_out_size); + void (*Finalize)(const long int final_value); +}; + +#define LIMIT(val,min,max) ((val) < (min)) ? (min) : (((max) < (val)) ? (max) : (val)) +const char SS_EMPTY [] PROGMEM = ""; +const char SS_CW_TONE [] PROGMEM = "Set CW Tone (Hz)"; +const SettingScreen_t SettingCwTone = { + SS_CW_TONE, + SS_EMPTY, + 1, + 10, + [](long int* svo){*svo = globalSettings.cwSideToneFreq;}, + [](const long int cvi, long int* vvo){*vvo = LIMIT(cvi,100,2000);}, + [](const long int nv, char* buf, const size_t buf_size){ + globalSettings.cwSideToneFreq = nv; + tone(CW_TONE, globalSettings.cwSideToneFreq); + ltoa(globalSettings.cwSideToneFreq,buf,10); + }, + [](const long int fv){ + noTone(CW_TONE); + globalSettings.cwSideToneFreq = fv; + SaveSettingsToEeprom(); + } +}; +void runSetting(const SettingScreen_t* const screen); +void runToneSetting(){runSetting(&SettingCwTone);} + +void runSetting(const SettingScreen_t* const screen) +{ + displayDialog(reinterpret_cast(screen->Title),F("Push Tune to Save")); + + //Wait for button to stop being pressed + while(btnDown()){ + active_delay(10); + } + active_delay(10); + + long int raw_value = 0; + long int last_value = 0; + + screen->Initialize(&last_value); + screen->OnValueChange(last_value,b,sizeof(b)); + displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_TITLE_BACKGROUND, COLOR_BACKGROUND); + + raw_value = last_value * screen->KnobDivider; + + while (!btnDown()) + { + int knob = enc_read(); + if(knob != 0){ + raw_value += knob * (int32_t)screen->StepSize; + } + else{ + continue; + } + + const long int candidate_value = raw_value / (int32_t)screen->KnobDivider; + long int value = 0; + screen->Validate(candidate_value,&value); + + //If we're going out of bounds, prevent the raw value from going too far out + if(candidate_value != value){ + raw_value = value * (int32_t)screen->KnobDivider; + } + + if(value == last_value){ + continue; + } + else{ + screen->OnValueChange(value,b,sizeof(b)); + displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_TITLE_BACKGROUND, COLOR_BACKGROUND); + last_value = value; + } + } + + screen->Finalize(last_value); +} + void printCarrierFreq(unsigned long freq) { formatFreq(freq,c,sizeof(c)); @@ -354,7 +440,7 @@ const char MI_CW_KEYER [] PROGMEM = "Keyer Type"; const MenuItem_t cwMenu [] PROGMEM { {MT_CW,nullptr},//Title {MI_CW_SPEED,setupCwSpeed}, - {MI_CW_TONE,setupCwTone}, + {MI_CW_TONE,runToneSetting}, {MI_CW_DELAY,setupCwDelay}, {MI_CW_KEYER,setupKeyer}, }; From 03fd8463eb8ef37216a4f01deea1e77941809d12 Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Sun, 19 Jan 2020 10:48:07 -0800 Subject: [PATCH 11/29] Switch to named functions instead of lambdas, because apparently PROGMEM doesn't like lambdas --- setup.cpp | 64 +++++++++++++++++++++++++++++++++---------------------- 1 file changed, 39 insertions(+), 25 deletions(-) diff --git a/setup.cpp b/setup.cpp index 158b0c7..f09724f 100644 --- a/setup.cpp +++ b/setup.cpp @@ -78,32 +78,46 @@ struct SettingScreen_t { }; #define LIMIT(val,min,max) ((val) < (min)) ? (min) : (((max) < (val)) ? (max) : (val)) +ssCwToneInitialize(long int* start_value_out) +{ + *start_value_out = globalSettings.cwSideToneFreq; +} +ssCwToneValidate(const long int candidate_value_in, long int* validated_value_out) +{ + *validated_value_out = LIMIT(candidate_value_in,100,2000); +} +ssCwToneChange(const long int new_value, char* buff_out, const size_t buff_out_size) +{ + globalSettings.cwSideToneFreq = new_value; + tone(CW_TONE, globalSettings.cwSideToneFreq); + ltoa(globalSettings.cwSideToneFreq,buff_out,10); +} +ssCwToneFinalize(const long int final_value) +{ + noTone(CW_TONE); + globalSettings.cwSideToneFreq = final_value; + SaveSettingsToEeprom(); +} const char SS_EMPTY [] PROGMEM = ""; const char SS_CW_TONE [] PROGMEM = "Set CW Tone (Hz)"; -const SettingScreen_t SettingCwTone = { +const SettingScreen_t settingScreens [] PROGMEM = { SS_CW_TONE, SS_EMPTY, 1, 10, - [](long int* svo){*svo = globalSettings.cwSideToneFreq;}, - [](const long int cvi, long int* vvo){*vvo = LIMIT(cvi,100,2000);}, - [](const long int nv, char* buf, const size_t buf_size){ - globalSettings.cwSideToneFreq = nv; - tone(CW_TONE, globalSettings.cwSideToneFreq); - ltoa(globalSettings.cwSideToneFreq,buf,10); - }, - [](const long int fv){ - noTone(CW_TONE); - globalSettings.cwSideToneFreq = fv; - SaveSettingsToEeprom(); - } + ssCwToneInitialize, + ssCwToneValidate, + ssCwToneChange, + ssCwToneFinalize }; void runSetting(const SettingScreen_t* const screen); -void runToneSetting(){runSetting(&SettingCwTone);} +void runToneSetting(){runSetting(&settingScreens[0]);} -void runSetting(const SettingScreen_t* const screen) +void runSetting(const SettingScreen_t* const p_screen) { - displayDialog(reinterpret_cast(screen->Title),F("Push Tune to Save")); + SettingScreen_t screen = {0}; + memcpy_P(&screen,p_screen,sizeof(screen)); + displayDialog(reinterpret_cast(screen.Title),F("Push Tune to Save")); //Wait for button to stop being pressed while(btnDown()){ @@ -114,42 +128,42 @@ void runSetting(const SettingScreen_t* const screen) long int raw_value = 0; long int last_value = 0; - screen->Initialize(&last_value); - screen->OnValueChange(last_value,b,sizeof(b)); + screen.Initialize(&last_value); + screen.OnValueChange(last_value,b,sizeof(b)); displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_TITLE_BACKGROUND, COLOR_BACKGROUND); - raw_value = last_value * screen->KnobDivider; + raw_value = last_value * screen.KnobDivider; while (!btnDown()) { int knob = enc_read(); if(knob != 0){ - raw_value += knob * (int32_t)screen->StepSize; + raw_value += knob * (int32_t)screen.StepSize; } else{ continue; } - const long int candidate_value = raw_value / (int32_t)screen->KnobDivider; + const long int candidate_value = raw_value / (int32_t)screen.KnobDivider; long int value = 0; - screen->Validate(candidate_value,&value); + screen.Validate(candidate_value,&value); //If we're going out of bounds, prevent the raw value from going too far out if(candidate_value != value){ - raw_value = value * (int32_t)screen->KnobDivider; + raw_value = value * (int32_t)screen.KnobDivider; } if(value == last_value){ continue; } else{ - screen->OnValueChange(value,b,sizeof(b)); + screen.OnValueChange(value,b,sizeof(b)); displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_TITLE_BACKGROUND, COLOR_BACKGROUND); last_value = value; } } - screen->Finalize(last_value); + screen.Finalize(last_value); } void printCarrierFreq(unsigned long freq) From e65f4d57a4e8c54c7c126f96c47da779c0a07df8 Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Sun, 19 Jan 2020 12:56:14 -0800 Subject: [PATCH 12/29] Implement the instructions text --- setup.cpp | 44 +++++++++++++++++++++++++++----------------- 1 file changed, 27 insertions(+), 17 deletions(-) diff --git a/setup.cpp b/setup.cpp index f09724f..953a0dd 100644 --- a/setup.cpp +++ b/setup.cpp @@ -51,19 +51,27 @@ static const unsigned int LAYOUT_SETTING_VALUE_Y = LAYOUT_ITEM_Y + 3*LAYOUT_ITEM static const unsigned int LAYOUT_SETTING_VALUE_WIDTH = LAYOUT_ITEM_WIDTH; static const unsigned int LAYOUT_SETTING_VALUE_HEIGHT = LAYOUT_ITEM_HEIGHT; -static const unsigned int LAYOUT_INSTRUCTION_TEXT_X = 20; -static const unsigned int LAYOUT_INSTRUCTION_TEXT_Y = LAYOUT_ITEM_Y + 5*LAYOUT_ITEM_PITCH_Y; -static const unsigned int LAYOUT_INSTRUCTION_TEXT_WIDTH = LAYOUT_ITEM_WIDTH; -static const unsigned int LAYOUT_INSTRUCTION_TEXT_HEIGHT = LAYOUT_ITEM_HEIGHT; +static const unsigned int LAYOUT_INSTRUCTIONS_TEXT_X = 20; +static const unsigned int LAYOUT_INSTRUCTIONS_TEXT_Y = LAYOUT_ITEM_Y; +static const unsigned int LAYOUT_INSTRUCTIONS_TEXT_WIDTH = LAYOUT_ITEM_WIDTH; +static const unsigned int LAYOUT_INSTRUCTIONS_TEXT_HEIGHT = LAYOUT_SETTING_VALUE_Y - LAYOUT_ITEM_Y - 1; -void displayDialog(const __FlashStringHelper* title, const __FlashStringHelper* instructions){ - strcpy_P(b,(const char*)title); - strcpy_P(c,(const char*)instructions); +static const unsigned int LAYOUT_CONFIRM_TEXT_X = 20; +static const unsigned int LAYOUT_CONFIRM_TEXT_Y = LAYOUT_ITEM_Y + 5*LAYOUT_ITEM_PITCH_Y; +static const unsigned int LAYOUT_CONFIRM_TEXT_WIDTH = LAYOUT_ITEM_WIDTH; +static const unsigned int LAYOUT_CONFIRM_TEXT_HEIGHT = LAYOUT_ITEM_HEIGHT; + +void displayDialog(const char* title, + const char* instructions){ displayClear(COLOR_BACKGROUND); displayRect(LAYOUT_OUTER_BORDER_X,LAYOUT_OUTER_BORDER_Y,LAYOUT_OUTER_BORDER_WIDTH,LAYOUT_OUTER_BORDER_HEIGHT, COLOR_ACTIVE_BORDER); displayRect(LAYOUT_INNER_BORDER_X,LAYOUT_INNER_BORDER_Y,LAYOUT_INNER_BORDER_WIDTH,LAYOUT_INNER_BORDER_HEIGHT, COLOR_ACTIVE_BORDER); + strcpy_P(b,title); displayText(b, LAYOUT_TITLE_X, LAYOUT_TITLE_Y, LAYOUT_TITLE_WIDTH, LAYOUT_TITLE_HEIGHT, COLOR_TEXT, COLOR_TITLE_BACKGROUND, COLOR_ACTIVE_BORDER); - displayText(c, LAYOUT_INSTRUCTION_TEXT_X, LAYOUT_INSTRUCTION_TEXT_Y, LAYOUT_INSTRUCTION_TEXT_WIDTH, LAYOUT_INSTRUCTION_TEXT_HEIGHT, COLOR_TEXT, COLOR_BACKGROUND, COLOR_BACKGROUND); + strcpy_P(b,instructions); + displayText(b, LAYOUT_INSTRUCTIONS_TEXT_X, LAYOUT_INSTRUCTIONS_TEXT_Y, LAYOUT_INSTRUCTIONS_TEXT_WIDTH, LAYOUT_INSTRUCTIONS_TEXT_HEIGHT, COLOR_TEXT, COLOR_BACKGROUND, COLOR_BACKGROUND); + strcpy_P(b,(const char*)F("Push Tune to Save")); + displayText(b, LAYOUT_CONFIRM_TEXT_X, LAYOUT_CONFIRM_TEXT_Y, LAYOUT_CONFIRM_TEXT_WIDTH, LAYOUT_CONFIRM_TEXT_HEIGHT, COLOR_TEXT, COLOR_BACKGROUND, COLOR_BACKGROUND); } struct SettingScreen_t { @@ -100,9 +108,10 @@ ssCwToneFinalize(const long int final_value) } const char SS_EMPTY [] PROGMEM = ""; const char SS_CW_TONE [] PROGMEM = "Set CW Tone (Hz)"; +const char SS_LONG_TEXT [] PROGMEM = "This is a long string of text that should be wrapped at least once, possibly more."; const SettingScreen_t settingScreens [] PROGMEM = { SS_CW_TONE, - SS_EMPTY, + SS_LONG_TEXT, 1, 10, ssCwToneInitialize, @@ -117,7 +126,8 @@ void runSetting(const SettingScreen_t* const p_screen) { SettingScreen_t screen = {0}; memcpy_P(&screen,p_screen,sizeof(screen)); - displayDialog(reinterpret_cast(screen.Title),F("Push Tune to Save")); + displayDialog(screen.Title, + screen.AdditionalText); //Wait for button to stop being pressed while(btnDown()){ @@ -173,7 +183,7 @@ void printCarrierFreq(unsigned long freq) } void setupFreq(){ - displayDialog(F("Set Frequency"),F("Push TUNE to Save")); + //displayDialog(F("Set Frequency"),F("Push TUNE to Save")); //round off the the nearest khz { @@ -229,7 +239,7 @@ void setupFreq(){ } void setupBFO(){ - displayDialog(F("Set BFO"),F("Press TUNE to Save")); + //displayDialog(F("Set BFO"),F("Press TUNE to Save")); si5351bx_setfreq(0, globalSettings.usbCarrierFreq); printCarrierFreq(globalSettings.usbCarrierFreq); @@ -259,7 +269,7 @@ void setupCwDelay(){ int knob = 0; int prev_cw_delay; - displayDialog(F("Set CW T/R Delay"),F("Press tune to Save")); + //displayDialog(F("Set CW T/R Delay"),F("Press tune to Save")); prev_cw_delay = globalSettings.cwActiveTimeoutMs; @@ -300,7 +310,7 @@ void formatKeyerEnum(char* output, const KeyerMode_e mode) } void setupKeyer(){ - displayDialog(F("Set CW Keyer"),F("Press tune to Save")); + //displayDialog(F("Set CW Keyer"),F("Press tune to Save")); int knob = 0; uint32_t tmp_mode = globalSettings.keyerMode; @@ -333,7 +343,7 @@ void setupKeyer(){ void setupCwSpeed() { - displayDialog(F("Set CW Speed (WPM)"),F("Press tune to Save")); + //displayDialog(F("Set CW Speed (WPM)"),F("Press tune to Save")); unsigned int wpm = 1200/globalSettings.cwDitDurationMs; @@ -359,7 +369,7 @@ void setupCwSpeed() } void setupCwTone(){ - displayDialog(F("Set CW Tone (Hz)"),F("Press tune to Save")); + //displayDialog(F("Set CW Tone (Hz)"),F("Press tune to Save")); tone(CW_TONE, globalSettings.cwSideToneFreq); itoa(globalSettings.cwSideToneFreq, b, 10); @@ -386,7 +396,7 @@ void setupCwTone(){ void setupResetAll() { - displayDialog(F("Reset all cals and settings?"),F("Press tune to Confirm")); + //displayDialog(F("Reset all cals and settings?"),F("Press tune to Confirm")); strcpy_P(b,(const char*)F("No")); displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); From 00f5d2691d2dcbf4ba2ea4d1019815e2a928dd92 Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Sun, 19 Jan 2020 12:56:47 -0800 Subject: [PATCH 13/29] Make text wrap work nicely --- PDQ_MinLib/PDQ_GFX.h | 44 ++++++++++++++++++++++++++++++++------------ nano_gui.cpp | 35 ++++++++++++++++++++++++++++++----- 2 files changed, 62 insertions(+), 17 deletions(-) diff --git a/PDQ_MinLib/PDQ_GFX.h b/PDQ_MinLib/PDQ_GFX.h index 5b225ef..49370f0 100644 --- a/PDQ_MinLib/PDQ_GFX.h +++ b/PDQ_MinLib/PDQ_GFX.h @@ -145,6 +145,7 @@ public: static void drawChar(coord_t x, coord_t y, unsigned char c, color_t color, color_t bg, uint8_t size); static void drawCharGFX(coord_t x, coord_t y, unsigned char c, color_t color, color_t bg, uint8_t size); static inline void setCursor(coord_t x, coord_t y); + static inline void setBound(coord_t x, coord_t y); static inline void setTextColor(color_t c); static inline void setTextColor(color_t c, color_t bg); static inline void setTextSize(uint8_t s); @@ -157,8 +158,8 @@ public: static inline uint8_t getRotation() __attribute__ ((always_inline)) { return rotation; } static inline coord_t getCursorX() __attribute__ ((always_inline)) { return cursor_x; } static inline coord_t getCursorY() __attribute__ ((always_inline)) { return cursor_y; } - static inline void getTextBounds(char *string, coord_t x, coord_t y, int16_t *x1, int16_t *y1, uint16_t *w, uint16_t *h); - static inline void getTextBounds(const __FlashStringHelper *s, coord_t x, coord_t y, int16_t *x1, int16_t *y1, uint16_t *w, uint16_t *h); + static inline void getTextBounds(char *string, coord_t x, coord_t y, int16_t *x1, int16_t *y1, uint16_t *w, uint16_t *h, coord_t wi = _width); + static inline void getTextBounds(const __FlashStringHelper *s, coord_t x, coord_t y, int16_t *x1, int16_t *y1, uint16_t *w, uint16_t *h, coord_t wi = _width); virtual size_t write(uint8_t); // used by Arduino "Print.h" (and the one required virtual function) @@ -167,6 +168,7 @@ protected: static coord_t WIDTH, HEIGHT; // This is the 'raw' display w/h - never changes static coord_t _width, _height; // Display w/h as modified by current rotation static coord_t cursor_x, cursor_y; + static coord_t bound_x1, bound_x2; static color_t textcolor, textbgcolor; static uint8_t textsize; static uint8_t rotation; @@ -216,6 +218,10 @@ int16_t PDQ_GFX::cursor_x; template int16_t PDQ_GFX::cursor_y; template +int16_t PDQ_GFX::bound_x1; +template +int16_t PDQ_GFX::bound_x2; +template color_t PDQ_GFX::textcolor; template color_t PDQ_GFX::textbgcolor; @@ -239,6 +245,8 @@ PDQ_GFX::PDQ_GFX(coord_t w, coord_t h) _height = (int16_t)h; cursor_x = 0; cursor_y = 0; + bound_x1 = 0; + bound_x2 = WIDTH; rotation = 0; textsize = 1; textcolor = 0xffff; @@ -756,10 +764,10 @@ size_t PDQ_GFX::write(uint8_t c) if ((w > 0) && (h > 0)) { coord_t xo = (int8_t)pgm_read_byte(&glyph->xOffset); // sic - if(wrap && ((cursor_x + textsize * (xo + w)) >= _width)) + if(wrap && ((cursor_x + textsize * (xo + w)) >= bound_x2)) { // Drawing character would go off right edge; wrap to new line - cursor_x = 0; + cursor_x = bound_x1; cursor_y += (coord_t)textsize * (uint8_t)pgm_read_byte(&gfxFont->yAdvance); } @@ -970,6 +978,13 @@ void PDQ_GFX::setCursor(coord_t x, coord_t y) cursor_y = (int16_t)y; } +template +void PDQ_GFX::setBound(coord_t x1, coord_t x2) +{ + bound_x1 = (int16_t)x1; + bound_x2 = (int16_t)x2; +} + template void PDQ_GFX::setTextSize(uint8_t s) { @@ -1046,10 +1061,13 @@ void PDQ_GFX::setFont(const GFXfont *f) // Pass string and a cursor position, returns UL corner and W,H. template -void PDQ_GFX::getTextBounds(char *str, coord_t x, coord_t y, int16_t *x1, int16_t *y1, uint16_t *w, uint16_t *h) +void PDQ_GFX::getTextBounds(char *str, coord_t x, coord_t y, int16_t *x1, int16_t *y1, uint16_t *w, uint16_t *h, coord_t wi) { uint8_t c; // Current character + coord_t xs = x; + coord_t xe = xs+wi; + *x1 = x; *y1 = y; *w = *h = 0; @@ -1080,9 +1098,9 @@ void PDQ_GFX::getTextBounds(char *str, coord_t x, coord_t y, int16_t *x1, in xa = pgm_read_byte(&glyph->xAdvance); xo = pgm_read_byte(&glyph->xOffset); yo = pgm_read_byte(&glyph->yOffset); - if (wrap && ((x + (((int16_t)xo + gw) * ts)) >= _width)) // Line wrap + if (wrap && ((x + (((int16_t)xo + gw) * ts)) >= xe)) // Line wrap { - x = 0; // Reset x to 0 + x = xs; // Reset x to 0 y += ya; // Advance y by 1 line } gx1 = x + xo * ts; @@ -1103,7 +1121,7 @@ void PDQ_GFX::getTextBounds(char *str, coord_t x, coord_t y, int16_t *x1, in } else // Newline { - x = 0; // Reset x + x = xs; // Reset x y += ya; // Advance y by 1 line } } @@ -1158,10 +1176,12 @@ void PDQ_GFX::getTextBounds(char *str, coord_t x, coord_t y, int16_t *x1, in // Same as above, but for PROGMEM strings template -void PDQ_GFX::getTextBounds(const __FlashStringHelper *str, coord_t x, coord_t y, int16_t *x1, int16_t *y1, uint16_t *w, uint16_t *h) +void PDQ_GFX::getTextBounds(const __FlashStringHelper *str, coord_t x, coord_t y, int16_t *x1, int16_t *y1, uint16_t *w, uint16_t *h, coord_t wi) { uint8_t *s = (uint8_t *)str; uint8_t c; + coord_t xs = x; + coord_t xe = xs+wi; *x1 = x; *y1 = y; @@ -1194,9 +1214,9 @@ void PDQ_GFX::getTextBounds(const __FlashStringHelper *str, coord_t x, coord xa = pgm_read_byte(&glyph->xAdvance); xo = pgm_read_byte(&glyph->xOffset); yo = pgm_read_byte(&glyph->yOffset); - if (wrap && ((x + (((int16_t)xo + gw) * ts)) >= _width)) // Line wrap + if (wrap && ((x + (((int16_t)xo + gw) * ts)) >= xe)) // Line wrap { - x = 0; // Reset x to 0 + x = xs; // Reset x to 0 y += ya; // Advance y by 1 line } gx1 = x + xo * ts; @@ -1217,7 +1237,7 @@ void PDQ_GFX::getTextBounds(const __FlashStringHelper *str, coord_t x, coord } else // Newline { - x = 0; // Reset x + x = xs; // Reset x y += ya; // Advance y by 1 line } } diff --git a/nano_gui.cpp b/nano_gui.cpp index 6ca1ddb..b27b091 100644 --- a/nano_gui.cpp +++ b/nano_gui.cpp @@ -178,7 +178,7 @@ void displayInit(void){ tft.begin(); tft.setFont(ubitx_font); - tft.setTextWrap(false); + tft.setTextWrap(true); tft.setTextColor(DISPLAY_GREEN,DISPLAY_BLACK); tft.setTextSize(1); tft.setRotation(1); @@ -217,6 +217,14 @@ void displayChar(int16_t x, int16_t y, unsigned char c, uint16_t color, uint16_t void displayRawText(char *text, int x1, int y1, int color, int background){ tft.setTextColor(color,background); tft.setCursor(x1,y1); + tft.setBound(0,320); + tft.print(text); +} + +void displayRawText(char *text, int x1, int y1, int w, int color, int background){ + tft.setTextColor(color,background); + tft.setCursor(x1,y1); + tft.setBound(x1,x1+w); tft.print(text); } @@ -228,10 +236,27 @@ void displayText(char *text, int x1, int y1, int w, int h, int color, int backgr int16_t y1_out; uint16_t width_out; uint16_t height_out; - tft.getTextBounds(text,x1,y1,&x1_out,&y1_out,&width_out,&height_out); - x1 += (w - ( width_out + (x1_out-x1)))/2; - y1 += h - (h - height_out)/2; - displayRawText(text,x1,y1,color,background); + tft.getTextBounds(text,x1,y1,&x1_out,&y1_out,&width_out,&height_out,w); + Serial.println(text); + Serial.print(F("Calc w/h:")); + Serial.print(width_out); + Serial.print(F(",")); + Serial.println(height_out); + Serial.print(F("w/h:")); + Serial.print(w); + Serial.print(F(",")); + Serial.println(h); + Serial.print(F("Old x1/y1:")); + Serial.print(x1); + Serial.print(F(",")); + Serial.println(y1); + x1 += (w - ( (int32_t)width_out + (x1_out-x1)))/2; + y1 += (ubitx_font->yAdvance + h - ( (int32_t)height_out + (y1_out-y1)))/2; + Serial.print(F("New x1/y1:")); + Serial.print(x1); + Serial.print(F(",")); + Serial.println(y1); + displayRawText(text,x1,y1,w,color,background); } void setupTouch(){ From a8e616637c394856d9fdafc7d1750dc7b7835205 Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Sun, 19 Jan 2020 12:57:53 -0800 Subject: [PATCH 14/29] Remove a bunch of debugging statements --- nano_gui.cpp | 17 ----------------- 1 file changed, 17 deletions(-) diff --git a/nano_gui.cpp b/nano_gui.cpp index b27b091..902f2ad 100644 --- a/nano_gui.cpp +++ b/nano_gui.cpp @@ -237,25 +237,8 @@ void displayText(char *text, int x1, int y1, int w, int h, int color, int backgr uint16_t width_out; uint16_t height_out; tft.getTextBounds(text,x1,y1,&x1_out,&y1_out,&width_out,&height_out,w); - Serial.println(text); - Serial.print(F("Calc w/h:")); - Serial.print(width_out); - Serial.print(F(",")); - Serial.println(height_out); - Serial.print(F("w/h:")); - Serial.print(w); - Serial.print(F(",")); - Serial.println(h); - Serial.print(F("Old x1/y1:")); - Serial.print(x1); - Serial.print(F(",")); - Serial.println(y1); x1 += (w - ( (int32_t)width_out + (x1_out-x1)))/2; y1 += (ubitx_font->yAdvance + h - ( (int32_t)height_out + (y1_out-y1)))/2; - Serial.print(F("New x1/y1:")); - Serial.print(x1); - Serial.print(F(",")); - Serial.println(y1); displayRawText(text,x1,y1,w,color,background); } From ec409d8805d4f80f8c581a7a2abd2b3e07ca14c5 Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Sun, 19 Jan 2020 13:05:02 -0800 Subject: [PATCH 15/29] Fix reset x for manual newlines --- PDQ_MinLib/PDQ_GFX.h | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/PDQ_MinLib/PDQ_GFX.h b/PDQ_MinLib/PDQ_GFX.h index 49370f0..cb9f152 100644 --- a/PDQ_MinLib/PDQ_GFX.h +++ b/PDQ_MinLib/PDQ_GFX.h @@ -748,7 +748,7 @@ size_t PDQ_GFX::write(uint8_t c) { if(c == '\n') { - cursor_x = 0; + cursor_x = bound_x1; cursor_y += (coord_t)textsize * (uint8_t)pgm_read_byte(&gfxFont->yAdvance); } else if (c != '\r') From d0900aa392d05e92c508c40f7e503960e8076a04 Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Sun, 19 Jan 2020 18:40:11 -0800 Subject: [PATCH 16/29] Refactor local oscillator cal --- nano_gui.h | 1 - setup.cpp | 171 +++++++++++++++++++----------------------- ubitx.h | 9 --- ubitx_ui.cpp | 35 +++++---- ubitx_v6.3.1_code.ino | 11 +-- 5 files changed, 102 insertions(+), 125 deletions(-) diff --git a/nano_gui.h b/nano_gui.h index f58cf02..b6d24ac 100644 --- a/nano_gui.h +++ b/nano_gui.h @@ -23,7 +23,6 @@ void formatFreq(uint32_t freq, char* buff, uint16_t buff_size); /* touch functions */ boolean readTouch(); -void setupTouch(); void scaleTouch(struct Point *p); // Color definitions diff --git a/setup.cpp b/setup.cpp index 953a0dd..329a41d 100644 --- a/setup.cpp +++ b/setup.cpp @@ -1,9 +1,10 @@ #include -#include +#include "freeRam.h" #include "morse.h" +#include "nano_gui.h" +#include "setup.h" #include "settings.h" #include "ubitx.h" -#include "nano_gui.h" /** Menus * The Radio menus are accessed by tapping on the function button. @@ -85,43 +86,6 @@ struct SettingScreen_t { void (*Finalize)(const long int final_value); }; -#define LIMIT(val,min,max) ((val) < (min)) ? (min) : (((max) < (val)) ? (max) : (val)) -ssCwToneInitialize(long int* start_value_out) -{ - *start_value_out = globalSettings.cwSideToneFreq; -} -ssCwToneValidate(const long int candidate_value_in, long int* validated_value_out) -{ - *validated_value_out = LIMIT(candidate_value_in,100,2000); -} -ssCwToneChange(const long int new_value, char* buff_out, const size_t buff_out_size) -{ - globalSettings.cwSideToneFreq = new_value; - tone(CW_TONE, globalSettings.cwSideToneFreq); - ltoa(globalSettings.cwSideToneFreq,buff_out,10); -} -ssCwToneFinalize(const long int final_value) -{ - noTone(CW_TONE); - globalSettings.cwSideToneFreq = final_value; - SaveSettingsToEeprom(); -} -const char SS_EMPTY [] PROGMEM = ""; -const char SS_CW_TONE [] PROGMEM = "Set CW Tone (Hz)"; -const char SS_LONG_TEXT [] PROGMEM = "This is a long string of text that should be wrapped at least once, possibly more."; -const SettingScreen_t settingScreens [] PROGMEM = { - SS_CW_TONE, - SS_LONG_TEXT, - 1, - 10, - ssCwToneInitialize, - ssCwToneValidate, - ssCwToneChange, - ssCwToneFinalize -}; -void runSetting(const SettingScreen_t* const screen); -void runToneSetting(){runSetting(&settingScreens[0]);} - void runSetting(const SettingScreen_t* const p_screen) { SettingScreen_t screen = {0}; @@ -142,7 +106,7 @@ void runSetting(const SettingScreen_t* const p_screen) screen.OnValueChange(last_value,b,sizeof(b)); displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_TITLE_BACKGROUND, COLOR_BACKGROUND); - raw_value = last_value * screen.KnobDivider; + raw_value = last_value * (int32_t)screen.KnobDivider; while (!btnDown()) { @@ -176,73 +140,96 @@ void runSetting(const SettingScreen_t* const p_screen) screen.Finalize(last_value); } -void printCarrierFreq(unsigned long freq) +#define LIMIT(val,min,max) ((val) < (min)) ? (min) : (((max) < (val)) ? (max) : (val)) + +ssCwToneInitialize(long int* start_value_out) { - formatFreq(freq,c,sizeof(c)); - displayText(c, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_TITLE_BACKGROUND, COLOR_BACKGROUND); + *start_value_out = globalSettings.cwSideToneFreq; } +ssCwToneValidate(const long int candidate_value_in, long int* validated_value_out) +{ + *validated_value_out = LIMIT(candidate_value_in,100,2000); +} +ssCwToneChange(const long int new_value, char* buff_out, const size_t buff_out_size) +{ + globalSettings.cwSideToneFreq = new_value; + tone(CW_TONE, globalSettings.cwSideToneFreq); + ltoa(globalSettings.cwSideToneFreq,buff_out,10); + strcat_P(buff_out,(const char*)F("Hz")); +} +ssCwToneFinalize(const long int final_value) +{ + noTone(CW_TONE); + globalSettings.cwSideToneFreq = final_value; + SaveSettingsToEeprom(); +} +const char SS_CW_TONE_T [] PROGMEM = "Set CW Tone"; +const char SS_CW_TONE_A [] PROGMEM = "Select a frequency that\nCW mode to tune for"; +const SettingScreen_t ssTone PROGMEM = { + SS_CW_TONE_T, + SS_CW_TONE_A, + 1, + 10, + ssCwToneInitialize, + ssCwToneValidate, + ssCwToneChange, + ssCwToneFinalize +}; +void runToneSetting(){runSetting(&ssTone);} -void setupFreq(){ - //displayDialog(F("Set Frequency"),F("Push TUNE to Save")); - - //round off the the nearest khz +void ssLocalOscInitialize(long int* start_value_out){ + //round off the current frequency the nearest kHz { uint32_t freq = GetActiveVfoFreq(); - freq = (freq/1000l)* 1000l; + freq = (freq/1000L) * 1000L; setFrequency(freq); + si5351bx_setfreq(0, globalSettings.usbCarrierFreq); //set back the carrier oscillator, cw tx switches it off } - - strcpy_P(c,(const char*)F("You should have a")); - displayText(c, LAYOUT_SETTING_VALUE_X, LAYOUT_ITEM_Y, LAYOUT_ITEM_WIDTH, LAYOUT_ITEM_HEIGHT, COLOR_TEXT, COLOR_BACKGROUND, COLOR_BACKGROUND); - strcpy_P(c,(const char*)F("signal exactly at")); - displayText(c, LAYOUT_SETTING_VALUE_X, LAYOUT_ITEM_Y + 1*LAYOUT_ITEM_PITCH_Y, LAYOUT_ITEM_WIDTH, LAYOUT_ITEM_HEIGHT, COLOR_TEXT, COLOR_BACKGROUND, COLOR_BACKGROUND); - ltoa(GetActiveVfoFreq()/1000L, c, 10); - strcat_P(c,(const char*)F(" KHz")); - displayText(c, LAYOUT_SETTING_VALUE_X, LAYOUT_ITEM_Y + 2*LAYOUT_ITEM_PITCH_Y, LAYOUT_ITEM_WIDTH, LAYOUT_ITEM_HEIGHT, COLOR_TEXT, COLOR_BACKGROUND, COLOR_BACKGROUND); - strcpy_P(c,(const char*)F("Rotate to zerobeat")); - displayText(c, LAYOUT_SETTING_VALUE_X, LAYOUT_ITEM_Y + 4*LAYOUT_ITEM_PITCH_Y, LAYOUT_ITEM_WIDTH, LAYOUT_ITEM_HEIGHT, COLOR_TEXT, COLOR_BACKGROUND, COLOR_BACKGROUND); - - ltoa(globalSettings.oscillatorCal, b, 10); - displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_TITLE_BACKGROUND, COLOR_BACKGROUND); - //keep clear of any previous button press - while (btnDown()) - active_delay(100); - active_delay(100); - - while (!btnDown()) - { - int knob = enc_read(); - if(knob != 0){ - globalSettings.oscillatorCal += knob * 875; - } - else{ - continue; //don't update the frequency or the display - } - - si5351bx_setfreq(0, globalSettings.usbCarrierFreq); //set back the carrier oscillator anyway, cw tx switches it off - si5351_set_calibration(globalSettings.oscillatorCal); - setFrequency(GetActiveVfoFreq()); - - ltoa(globalSettings.oscillatorCal, b, 10); - displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_TITLE_BACKGROUND, COLOR_BACKGROUND); + *start_value_out = globalSettings.oscillatorCal; + Serial.println(freeRam()); +} +void ssLocalOscValidate(const long int candidate_value_in, long int* validated_value_out) +{ + *validated_value_out = candidate_value_in;//No check - allow anything +} +void ssLocalOscChange(const long int new_value, char* buff_out, const size_t buff_out_size) +{ + si5351_set_calibration(new_value); + setFrequency(GetActiveVfoFreq()); + const long int u = abs(new_value); + if(new_value != u){ + strcpy_P(buff_out,(const char*)F("-")); + ++buff_out; } - + formatFreq(u,buff_out,buff_out_size); + strcat_P(buff_out,(const char*)F("Hz")); +} +void ssLocalOscFinalize(const long int final_value) +{ + globalSettings.oscillatorCal = final_value; SaveSettingsToEeprom(); - initOscillators(); si5351_set_calibration(globalSettings.oscillatorCal); setFrequency(GetActiveVfoFreq()); - - //debounce and delay - while(btnDown()) - active_delay(50); - active_delay(100); } +const char SS_LOCAL_OSC_T [] PROGMEM = "Set Local Osc Calibration"; +const char SS_LOCAL_OSC_A [] PROGMEM = "Exit menu, tune so that the\ndial displays the desired freq,\nthen tune here until the\nsignal is zerobeat"; +const SettingScreen_t ssLocalOsc PROGMEM = { + SS_LOCAL_OSC_T, + SS_LOCAL_OSC_A, + 1, + 875, + ssLocalOscInitialize, + ssLocalOscValidate, + ssLocalOscChange, + ssLocalOscFinalize +}; +void runLocalOscSetting(){runSetting(&ssLocalOsc);} void setupBFO(){ //displayDialog(F("Set BFO"),F("Press TUNE to Save")); si5351bx_setfreq(0, globalSettings.usbCarrierFreq); - printCarrierFreq(globalSettings.usbCarrierFreq); + //printCarrierFreq(globalSettings.usbCarrierFreq); while (!btnDown()){ int knob = enc_read(); @@ -255,7 +242,7 @@ void setupBFO(){ si5351bx_setfreq(0, globalSettings.usbCarrierFreq); setFrequency(GetActiveVfoFreq()); - printCarrierFreq(globalSettings.usbCarrierFreq); + //printCarrierFreq(globalSettings.usbCarrierFreq); active_delay(100); } @@ -450,7 +437,7 @@ const char MI_SET_BFO [] PROGMEM = "Beat Frequency Osc (BFO)"; const char MI_TOUCH [] PROGMEM = "Touch Screen"; const MenuItem_t calibrationMenu [] PROGMEM { {MT_CAL,nullptr},//Title - {MI_SET_FREQ,setupFreq}, + {MI_SET_FREQ,runLocalOscSetting}, {MI_SET_BFO,setupBFO}, {MI_TOUCH,setupTouch}, }; diff --git a/ubitx.h b/ubitx.h index 3944a6d..e9aa36e 100644 --- a/ubitx.h +++ b/ubitx.h @@ -92,10 +92,6 @@ extern char c[30], b[30]; #define HIGHEST_FREQ (30000000l) static const uint32_t THRESHOLD_USB_LSB = 10000000L; -extern unsigned long firstIF; - -extern uint8_t menuOn; - /* these are functions implemented in the main file named as ubitx_xxx.ino */ void active_delay(int delay_by); void saveVFOs(); @@ -122,11 +118,6 @@ void drawTx(); //are useful to concatanate the values with text like "Set Freq to " x " KHz" int getValueByKnob(int minimum, int maximum, int step_size, int initial, char* prefix, char *postfix); -//functions of the setup menu. implemented in seteup.cpp -void doSetup2(); //main setup function, displays the setup menu, calls various dialog boxes -void setupBFO(); -void setupFreq(); - //main functions to check if any button is pressed and other user interface events void doCommands(); //does the commands with encoder to jump from button to button void checkTouch(); //does the commands with a touch on the buttons diff --git a/ubitx_ui.cpp b/ubitx_ui.cpp index 8457f11..ad0f9ee 100644 --- a/ubitx_ui.cpp +++ b/ubitx_ui.cpp @@ -1,7 +1,8 @@ #include -#include +#include "freeRam.h" #include "morse.h" #include "settings.h" +#include "setup.h" #include "ubitx.h" #include "nano_gui.h" @@ -900,6 +901,7 @@ void doCommand(Button* button){ } case BUTTON_MNU: { + Serial.println(freeRam()); doSetup2(); break; } @@ -953,20 +955,17 @@ void drawFocus(int ibtn, int color){ } void doCommands(){ - int select=0, i, prevButton, btnState; + int select = 0; + int prev_button = 0; //wait for the button to be raised up while(btnDown()) active_delay(50); active_delay(50); //debounce - menuOn = 2; - - while (menuOn){ - + while (true){ //check if the knob's button was pressed - btnState = btnDown(); - if (btnState){ + if (btnDown()){ Button button; memcpy_P(&button, &(btn_set[select/10]), sizeof(Button)); @@ -986,27 +985,27 @@ void doCommands(){ return; } - i = enc_read(); + int knob = enc_read(); - if (i == 0){ + if (knob == 0){ active_delay(50); continue; } - if (i > 0){ - if (select + i < BUTTON_TOTAL * 10) - select += i; + if (knob > 0){ + if (select + knob < BUTTON_TOTAL * 10) + select += knob; } - if (i < 0 && select + i >= 0) - select += i; //caught ya, i is already -ve here, so you add it + if (knob < 0 && select + knob >= 0) + select += knob; //caught ya, i is already -ve here, so you add it - if (prevButton == select / 10) + if (prev_button == select / 10) continue; //we are on a new button - drawFocus(prevButton, COLOR_INACTIVE_BORDER); + drawFocus(prev_button, COLOR_INACTIVE_BORDER); drawFocus(select/10, COLOR_ACTIVE_BORDER); - prevButton = select/10; + prev_button = select/10; } // guiUpdate(); diff --git a/ubitx_v6.3.1_code.ino b/ubitx_v6.3.1_code.ino index 0d18816..bce908d 100644 --- a/ubitx_v6.3.1_code.ino +++ b/ubitx_v6.3.1_code.ino @@ -31,6 +31,7 @@ */ #include #include "settings.h" +#include "setup.h" #include "ubitx.h" #include "nano_gui.h" @@ -40,16 +41,16 @@ * created in a memory region called the stack. The stack has just a few bytes of space on the Arduino * if you declare large strings inside functions, they can easily exceed the capacity of the stack * and mess up your programs. - * We circumvent this by declaring a few global buffers as kitchen counters where we can + * We circumvent this by declaring a few global buffers as kitchen counters where we can * slice and dice our strings. These strings are mostly used to control the display or handle * the input and output from the USB port. We must keep a count of the bytes used while reading * the serial port as we can easily run out of buffer space. This is done in the serial_in_count variable. */ -char c[30], b[30]; +char b[30]; +char c[30]; //during CAT commands, we will freeeze the display until CAT is disengaged unsigned char doingCAT = 0; -byte menuOn = 0; //set to 1 when the menu is being displayed, if a menu item sets it to zero, the menu is exited /** @@ -463,7 +464,7 @@ void initPorts(){ void setup() { Serial.begin(38400); - Serial.flush(); + Serial.flush(); initSettings(); displayInit(); @@ -477,7 +478,7 @@ void setup() setupTouch(); SetActiveVfoMode(VfoMode_e::VFO_MODE_USB); setFrequency(10000000L); - setupFreq(); + runLocalOscSetting(); SetActiveVfoMode(VfoMode_e::VFO_MODE_LSB); setFrequency(7100000L); setupBFO(); From 9b224699ee46af8222990b6e2e072aca822ec3bf Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Sun, 19 Jan 2020 18:41:36 -0800 Subject: [PATCH 17/29] increase buffer b size to avoid heap smashing --- ubitx.h | 2 +- ubitx_v6.3.1_code.ino | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/ubitx.h b/ubitx.h index e9aa36e..0800e75 100644 --- a/ubitx.h +++ b/ubitx.h @@ -56,7 +56,7 @@ it uses an ILI9341 display controller and an XPT2046 touch controller. * the input and output from the USB port. We must keep a count of the bytes used while reading * the serial port as we can easily run out of buffer space. This is done in the serial_in_count variable. */ -extern char c[30], b[30]; +extern char c[30], b[128]; /** * The second set of 16 pins on the Raduino's bottom connector are have the three clock outputs and the digital lines to control the rig. diff --git a/ubitx_v6.3.1_code.ino b/ubitx_v6.3.1_code.ino index bce908d..21abdaf 100644 --- a/ubitx_v6.3.1_code.ino +++ b/ubitx_v6.3.1_code.ino @@ -46,7 +46,7 @@ * the input and output from the USB port. We must keep a count of the bytes used while reading * the serial port as we can easily run out of buffer space. This is done in the serial_in_count variable. */ -char b[30]; +char b[128]; char c[30]; //during CAT commands, we will freeeze the display until CAT is disengaged From 315d9348f06503ff4f18cdb21754b186a37b03b2 Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Sun, 19 Jan 2020 19:00:45 -0800 Subject: [PATCH 18/29] Swap most string operations to bounded versions --- setup.cpp | 32 +++++++++++++++----------------- ubitx_cat.cpp | 11 ----------- ubitx_ui.cpp | 38 ++++++++++++++++++-------------------- 3 files changed, 33 insertions(+), 48 deletions(-) diff --git a/setup.cpp b/setup.cpp index 329a41d..6a90477 100644 --- a/setup.cpp +++ b/setup.cpp @@ -1,5 +1,4 @@ #include -#include "freeRam.h" #include "morse.h" #include "nano_gui.h" #include "setup.h" @@ -67,11 +66,11 @@ void displayDialog(const char* title, displayClear(COLOR_BACKGROUND); displayRect(LAYOUT_OUTER_BORDER_X,LAYOUT_OUTER_BORDER_Y,LAYOUT_OUTER_BORDER_WIDTH,LAYOUT_OUTER_BORDER_HEIGHT, COLOR_ACTIVE_BORDER); displayRect(LAYOUT_INNER_BORDER_X,LAYOUT_INNER_BORDER_Y,LAYOUT_INNER_BORDER_WIDTH,LAYOUT_INNER_BORDER_HEIGHT, COLOR_ACTIVE_BORDER); - strcpy_P(b,title); + strncpy_P(b,title,sizeof(b)); displayText(b, LAYOUT_TITLE_X, LAYOUT_TITLE_Y, LAYOUT_TITLE_WIDTH, LAYOUT_TITLE_HEIGHT, COLOR_TEXT, COLOR_TITLE_BACKGROUND, COLOR_ACTIVE_BORDER); - strcpy_P(b,instructions); + strncpy_P(b,instructions,sizeof(b)); displayText(b, LAYOUT_INSTRUCTIONS_TEXT_X, LAYOUT_INSTRUCTIONS_TEXT_Y, LAYOUT_INSTRUCTIONS_TEXT_WIDTH, LAYOUT_INSTRUCTIONS_TEXT_HEIGHT, COLOR_TEXT, COLOR_BACKGROUND, COLOR_BACKGROUND); - strcpy_P(b,(const char*)F("Push Tune to Save")); + strncpy_P(b,(const char*)F("Push Tune to Save"),sizeof(b)); displayText(b, LAYOUT_CONFIRM_TEXT_X, LAYOUT_CONFIRM_TEXT_Y, LAYOUT_CONFIRM_TEXT_WIDTH, LAYOUT_CONFIRM_TEXT_HEIGHT, COLOR_TEXT, COLOR_BACKGROUND, COLOR_BACKGROUND); } @@ -155,7 +154,7 @@ ssCwToneChange(const long int new_value, char* buff_out, const size_t buff_out_s globalSettings.cwSideToneFreq = new_value; tone(CW_TONE, globalSettings.cwSideToneFreq); ltoa(globalSettings.cwSideToneFreq,buff_out,10); - strcat_P(buff_out,(const char*)F("Hz")); + strncat_P(buff_out,(const char*)F("Hz"),buff_out_size - strlen(buff_out)); } ssCwToneFinalize(const long int final_value) { @@ -186,7 +185,6 @@ void ssLocalOscInitialize(long int* start_value_out){ si5351bx_setfreq(0, globalSettings.usbCarrierFreq); //set back the carrier oscillator, cw tx switches it off } *start_value_out = globalSettings.oscillatorCal; - Serial.println(freeRam()); } void ssLocalOscValidate(const long int candidate_value_in, long int* validated_value_out) { @@ -198,11 +196,11 @@ void ssLocalOscChange(const long int new_value, char* buff_out, const size_t buf setFrequency(GetActiveVfoFreq()); const long int u = abs(new_value); if(new_value != u){ - strcpy_P(buff_out,(const char*)F("-")); + strncpy_P(buff_out,(const char*)F("-"),buff_out_size); ++buff_out; } - formatFreq(u,buff_out,buff_out_size); - strcat_P(buff_out,(const char*)F("Hz")); + formatFreq(u,buff_out,buff_out_size - strlen(buff_out)); + strncat_P(buff_out,(const char*)F("Hz"),buff_out_size - strlen(buff_out)); } void ssLocalOscFinalize(const long int final_value) { @@ -261,7 +259,7 @@ void setupCwDelay(){ prev_cw_delay = globalSettings.cwActiveTimeoutMs; ltoa(globalSettings.cwActiveTimeoutMs, b, 10); - strcat_P(b,(const char*)F(" msec")); + strncat_P(b,(const char*)F(" msec"),sizeof(b) - strlen(b)); displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); while (!btnDown()){ @@ -275,7 +273,7 @@ void setupCwDelay(){ continue; //don't update the frequency or the display ltoa(globalSettings.cwActiveTimeoutMs, b, 10); - strcat_P(b,(const char*)F(" msec")); + strncat_P(b,(const char*)F(" msec"),sizeof(b) - strlen(b)); displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); } @@ -384,7 +382,7 @@ void setupCwTone(){ void setupResetAll() { //displayDialog(F("Reset all cals and settings?"),F("Press tune to Confirm")); - strcpy_P(b,(const char*)F("No")); + strncpy_P(b,(const char*)F("No"),sizeof(b)); displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); bool reset_all = false; @@ -402,10 +400,10 @@ void setupResetAll() } if(reset_all){ - strcpy_P(b,(const char*)F("Yes")); + strncpy_P(b,(const char*)F("Yes"),sizeof(b)); } else{ - strcpy_P(b,(const char*)F("No")); + strncpy_P(b,(const char*)F("No"),sizeof(b)); } displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); @@ -474,15 +472,15 @@ void drawMenu(const MenuItem_t* const items, const uint16_t num_items) displayClear(COLOR_BACKGROUND); MenuItem_t mi = {"",nullptr}; memcpy_P(&mi,&items[0],sizeof(mi)); - strcpy_P(b,mi.ItemName); + strncpy_P(b,mi.ItemName,sizeof(b)); displayText(b, LAYOUT_TITLE_X, LAYOUT_TITLE_Y, LAYOUT_TITLE_WIDTH, LAYOUT_TITLE_HEIGHT, COLOR_TEXT, COLOR_TITLE_BACKGROUND, COLOR_ACTIVE_BORDER); for(unsigned int i = 1; i < num_items; ++i){ memcpy_P(&mi,&items[i],sizeof(mi)); - strcpy_P(b,mi.ItemName); + strncpy_P(b,mi.ItemName,sizeof(b)); displayText(b, LAYOUT_ITEM_X, LAYOUT_ITEM_Y + (i-1)*LAYOUT_ITEM_PITCH_Y, LAYOUT_ITEM_WIDTH, LAYOUT_ITEM_HEIGHT, COLOR_TEXT, COLOR_BACKGROUND, COLOR_INACTIVE_BORDER); } memcpy_P(&mi,&exitMenu,sizeof(mi)); - strcpy_P(b,mi.ItemName); + strncpy_P(b,mi.ItemName,sizeof(b)); displayText(b, LAYOUT_ITEM_X, LAYOUT_ITEM_Y + (num_items-1)*LAYOUT_ITEM_PITCH_Y, LAYOUT_ITEM_WIDTH, LAYOUT_ITEM_HEIGHT, COLOR_TEXT, COLOR_BACKGROUND, COLOR_INACTIVE_BORDER); } diff --git a/ubitx_cat.cpp b/ubitx_cat.cpp index 815c18b..91c7f49 100644 --- a/ubitx_cat.cpp +++ b/ubitx_cat.cpp @@ -123,13 +123,6 @@ void catReadEEPRom(void) //for remove warnings byte temp0 = cat[0]; byte temp1 = cat[1]; -/* - itoa((int) cat[0], b, 16); - strcat(b, ":"); - itoa((int) cat[1], c, 16); - strcat(b, c); - printLine2(b); -*/ cat[0] = 0; cat[1] = 0; @@ -385,10 +378,6 @@ void processCATCommand2(byte* cmd) { default: //somehow, get this to print the four bytes ultoa(*((unsigned long *)cmd), c, 16); - /*itoa(cmd[4], b, 16); - strcat(b, ">"); - strcat(b, c); - printLine2(b);*/ response[0] = 0x00; Serial.write(response[0]); } diff --git a/ubitx_ui.cpp b/ubitx_ui.cpp index ad0f9ee..624705f 100644 --- a/ubitx_ui.cpp +++ b/ubitx_ui.cpp @@ -1,5 +1,4 @@ #include -#include "freeRam.h" #include "morse.h" #include "settings.h" #include "setup.h" @@ -166,10 +165,10 @@ int getValueByKnob(int minimum, int maximum, int step_size, int initial, const active_delay(200); knob_value = initial; - strcpy_P(b,(const char*)prefix); + strncpy_P(b,(const char*)prefix,sizeof(b)); itoa(knob_value, c, 10); - strcat(b, c); - strcat_P(b, (const char*)postfix); + strncat(b, c, sizeof(b) - strlen(b)); + strncat_P(b, (const char*)postfix, sizeof(b) - strlen(b)); drawCommandbar(b); while(!btnDown() && digitalRead(PTT) == HIGH){ @@ -180,10 +179,10 @@ int getValueByKnob(int minimum, int maximum, int step_size, int initial, const if (knob_value < maximum && knob > 0) knob_value += step_size; - strcpy_P(b,(const char*)prefix); + strncpy_P(b,(const char*)prefix,sizeof(b)); itoa(knob_value, c, 10); - strcat(b, c); - strcat_P(b,(const char*)postfix); + strncat(b, c, sizeof(b) - strlen(b)); + strncat_P(b,(const char*)postfix, sizeof(b) - strlen(b)); drawCommandbar(b); } checkCAT(); @@ -335,8 +334,8 @@ void displayRIT(){ c[0] = 0; displayFillrect(LAYOUT_MODE_TEXT_X,LAYOUT_MODE_TEXT_Y,LAYOUT_MODE_TEXT_WIDTH,LAYOUT_MODE_TEXT_HEIGHT, COLOR_BACKGROUND); if(globalSettings.ritOn){ - strcpy_P(c,(const char*)F("TX:")); - formatFreq(globalSettings.ritFrequency, c+3, sizeof(c)-3); + strncpy_P(c,(const char*)F("TX:"),sizeof(c)); + formatFreq(globalSettings.ritFrequency, c+3, sizeof(c)-strlen(c)); if (VFO_A == globalSettings.activeVfo) displayText(c, LAYOUT_VFO_LABEL_X + 0*LAYOUT_VFO_LABEL_PITCH_X, LAYOUT_MODE_TEXT_Y, LAYOUT_VFO_LABEL_WIDTH, LAYOUT_MODE_TEXT_HEIGHT, COLOR_TEXT, COLOR_BACKGROUND, COLOR_BACKGROUND); else @@ -350,9 +349,9 @@ void fastTune(){ active_delay(50); active_delay(300); - strcpy_P(c,(const char*)F("Fast tune")); + strncpy_P(c,(const char*)F("Fast tune"),sizeof(c)); displayText(c, LAYOUT_MODE_TEXT_X, LAYOUT_MODE_TEXT_Y, LAYOUT_MODE_TEXT_WIDTH, LAYOUT_MODE_TEXT_HEIGHT, COLOR_TEXT, COLOR_BACKGROUND, COLOR_BACKGROUND); - while(1){ + while(true){ checkCAT(); //exit after debouncing the btnDown @@ -472,8 +471,8 @@ void enterFreq(){ }//switch }//if button hit test }// end of the button scanning loop - strcpy(b, c); - strcat_P(b,(const char*)F(" KHz")); + strncpy(b, c, sizeof(b)); + strncat_P(b,(const char*)F(" KHz"),sizeof(b) - strlen(b)); displayText(b, LAYOUT_MODE_TEXT_X, LAYOUT_MODE_TEXT_Y, LAYOUT_MODE_TEXT_WIDTH, LAYOUT_MODE_TEXT_HEIGHT, COLOR_TEXT, COLOR_BACKGROUND, COLOR_BACKGROUND); active_delay(300); while(readTouch()) @@ -482,21 +481,21 @@ void enterFreq(){ } void drawCWStatus(){ - strcpy_P(b,(const char*)F(" cw: ")); + strncpy_P(b,(const char*)F(" cw: "),sizeof(b)); int wpm = 1200/globalSettings.cwDitDurationMs; itoa(wpm,c, 10); - strcat(b, c); - strcat_P(b,(const char*)F("wpm, ")); + strncat(b, c, sizeof(b) - strlen(b)); + strncat_P(b,(const char*)F("wpm, "), sizeof(b) - strlen(b)); itoa(globalSettings.cwSideToneFreq, c, 10); - strcat(b, c); - strcat_P(b,(const char*)F("hz")); + strncat(b, c, sizeof(b) - strlen(b)); + strncat_P(b,(const char*)F("hz"), sizeof(b) - strlen(b)); displayText(b, LAYOUT_CW_TEXT_X, LAYOUT_CW_TEXT_Y, LAYOUT_CW_TEXT_WIDTH, LAYOUT_CW_TEXT_HEIGHT, COLOR_TEXT, COLOR_BACKGROUND, COLOR_BACKGROUND); } void drawTx(){ if (globalSettings.txActive){ - strcpy_P(b,(const char*)F("TX")); + strncpy_P(b,(const char*)F("TX"), sizeof(b)); displayText(b, LAYOUT_TX_X, LAYOUT_TX_Y, LAYOUT_TX_WIDTH, LAYOUT_TX_HEIGHT, COLOR_ACTIVE_TEXT, COLOR_ACTIVE_BACKGROUND, COLOR_BACKGROUND); } else{ @@ -901,7 +900,6 @@ void doCommand(Button* button){ } case BUTTON_MNU: { - Serial.println(freeRam()); doSetup2(); break; } From b99e13eff4b211f73b68b8ab3441037c65ac15a3 Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Sun, 19 Jan 2020 19:24:26 -0800 Subject: [PATCH 19/29] Refactor BFO settings menu --- setup.cpp | 63 +++++++++++++++++++++++++------------------ setup.h | 6 +++++ ubitx_v6.3.1_code.ino | 2 +- 3 files changed, 44 insertions(+), 27 deletions(-) create mode 100644 setup.h diff --git a/setup.cpp b/setup.cpp index 6a90477..fcc43a8 100644 --- a/setup.cpp +++ b/setup.cpp @@ -78,7 +78,7 @@ struct SettingScreen_t { const char* const Title; const char* const AdditionalText; const uint16_t KnobDivider; - const uint16_t StepSize; + const int16_t StepSize;//int so that it can be negative void (*Initialize)(long int* start_value_out); void (*Validate)(const long int candidate_value_in, long int* validated_value_out); void (*OnValueChange)(const long int new_value, char* buff_out, const size_t buff_out_size); @@ -111,7 +111,7 @@ void runSetting(const SettingScreen_t* const p_screen) { int knob = enc_read(); if(knob != 0){ - raw_value += knob * (int32_t)screen.StepSize; + raw_value += knob * screen.StepSize; } else{ continue; @@ -141,6 +141,7 @@ void runSetting(const SettingScreen_t* const p_screen) #define LIMIT(val,min,max) ((val) < (min)) ? (min) : (((max) < (val)) ? (max) : (val)) +//CW Tone ssCwToneInitialize(long int* start_value_out) { *start_value_out = globalSettings.cwSideToneFreq; @@ -176,11 +177,11 @@ const SettingScreen_t ssTone PROGMEM = { }; void runToneSetting(){runSetting(&ssTone);} +//Local Oscillator void ssLocalOscInitialize(long int* start_value_out){ - //round off the current frequency the nearest kHz { uint32_t freq = GetActiveVfoFreq(); - freq = (freq/1000L) * 1000L; + freq = (freq/1000L) * 1000L;//round off the current frequency the nearest kHz setFrequency(freq); si5351bx_setfreq(0, globalSettings.usbCarrierFreq); //set back the carrier oscillator, cw tx switches it off } @@ -223,32 +224,42 @@ const SettingScreen_t ssLocalOsc PROGMEM = { }; void runLocalOscSetting(){runSetting(&ssLocalOsc);} -void setupBFO(){ - //displayDialog(F("Set BFO"),F("Press TUNE to Save")); - +//BFO +void ssBfoInitialize(long int* start_value_out){ si5351bx_setfreq(0, globalSettings.usbCarrierFreq); - //printCarrierFreq(globalSettings.usbCarrierFreq); - - while (!btnDown()){ - int knob = enc_read(); - if(knob != 0){ - globalSettings.usbCarrierFreq -= 50 * knob; - } - else{ - continue; //don't update the frequency or the display - } - - si5351bx_setfreq(0, globalSettings.usbCarrierFreq); - setFrequency(GetActiveVfoFreq()); - //printCarrierFreq(globalSettings.usbCarrierFreq); - - active_delay(100); - } - + *start_value_out = globalSettings.usbCarrierFreq; +} +void ssBfoValidate(const long int candidate_value_in, long int* validated_value_out) +{ + *validated_value_out = LIMIT(candidate_value_in,11048000L,11060000L); +} +void ssBfoChange(const long int new_value, char* buff_out, const size_t buff_out_size) +{ + si5351bx_setfreq(0, new_value); + setFrequency(GetActiveVfoFreq()); + formatFreq(new_value,buff_out,buff_out_size); + strncat_P(buff_out,(const char*)F("Hz"),buff_out_size - strlen(buff_out)); +} +void ssBfoFinalize(const long int final_value) +{ + globalSettings.usbCarrierFreq = final_value; SaveSettingsToEeprom(); si5351bx_setfreq(0, globalSettings.usbCarrierFreq); setFrequency(GetActiveVfoFreq()); } +const char SS_BFO_T [] PROGMEM = "Set BFO Calibration"; +const char SS_BFO_A [] PROGMEM = "Exit menu, tune to an unused\nfrequency, then tune here\nuntil the audio is between\n300-3000Hz"; +const SettingScreen_t ssBfo PROGMEM = { + SS_BFO_T, + SS_BFO_A, + 1, + -50,//Negative to make dial more intuitive: turning clockwise increases the perceived audio frequency + ssBfoInitialize, + ssBfoValidate, + ssBfoChange, + ssBfoFinalize +}; +void runBfoSetting(){runSetting(&ssBfo);} void setupCwDelay(){ int knob = 0; @@ -436,7 +447,7 @@ const char MI_TOUCH [] PROGMEM = "Touch Screen"; const MenuItem_t calibrationMenu [] PROGMEM { {MT_CAL,nullptr},//Title {MI_SET_FREQ,runLocalOscSetting}, - {MI_SET_BFO,setupBFO}, + {MI_SET_BFO,runBfoSetting}, {MI_TOUCH,setupTouch}, }; void runCalibrationMenu(){RUN_MENU(calibrationMenu);} diff --git a/setup.h b/setup.h new file mode 100644 index 0000000..16b5b0d --- /dev/null +++ b/setup.h @@ -0,0 +1,6 @@ +#pragma once + +void doSetup2(); //main setup function, displays the setup menu, calls various dialog boxes +void setupTouch(); +void runLocalOscSetting(); +void runBfoSetting(); \ No newline at end of file diff --git a/ubitx_v6.3.1_code.ino b/ubitx_v6.3.1_code.ino index 21abdaf..617977e 100644 --- a/ubitx_v6.3.1_code.ino +++ b/ubitx_v6.3.1_code.ino @@ -481,7 +481,7 @@ void setup() runLocalOscSetting(); SetActiveVfoMode(VfoMode_e::VFO_MODE_LSB); setFrequency(7100000L); - setupBFO(); + runBfoSetting(); } guiUpdate(); From 58bfe556207f6f3bdaef3b2db4f98ac51b7571c0 Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Sun, 19 Jan 2020 19:26:31 -0800 Subject: [PATCH 20/29] Move tone down to keep the menus in a somewhat orderly fashion --- setup.cpp | 72 +++++++++++++++++++++++++++---------------------------- 1 file changed, 36 insertions(+), 36 deletions(-) diff --git a/setup.cpp b/setup.cpp index fcc43a8..a2d91f0 100644 --- a/setup.cpp +++ b/setup.cpp @@ -141,42 +141,6 @@ void runSetting(const SettingScreen_t* const p_screen) #define LIMIT(val,min,max) ((val) < (min)) ? (min) : (((max) < (val)) ? (max) : (val)) -//CW Tone -ssCwToneInitialize(long int* start_value_out) -{ - *start_value_out = globalSettings.cwSideToneFreq; -} -ssCwToneValidate(const long int candidate_value_in, long int* validated_value_out) -{ - *validated_value_out = LIMIT(candidate_value_in,100,2000); -} -ssCwToneChange(const long int new_value, char* buff_out, const size_t buff_out_size) -{ - globalSettings.cwSideToneFreq = new_value; - tone(CW_TONE, globalSettings.cwSideToneFreq); - ltoa(globalSettings.cwSideToneFreq,buff_out,10); - strncat_P(buff_out,(const char*)F("Hz"),buff_out_size - strlen(buff_out)); -} -ssCwToneFinalize(const long int final_value) -{ - noTone(CW_TONE); - globalSettings.cwSideToneFreq = final_value; - SaveSettingsToEeprom(); -} -const char SS_CW_TONE_T [] PROGMEM = "Set CW Tone"; -const char SS_CW_TONE_A [] PROGMEM = "Select a frequency that\nCW mode to tune for"; -const SettingScreen_t ssTone PROGMEM = { - SS_CW_TONE_T, - SS_CW_TONE_A, - 1, - 10, - ssCwToneInitialize, - ssCwToneValidate, - ssCwToneChange, - ssCwToneFinalize -}; -void runToneSetting(){runSetting(&ssTone);} - //Local Oscillator void ssLocalOscInitialize(long int* start_value_out){ { @@ -261,6 +225,42 @@ const SettingScreen_t ssBfo PROGMEM = { }; void runBfoSetting(){runSetting(&ssBfo);} +//CW Tone +ssCwToneInitialize(long int* start_value_out) +{ + *start_value_out = globalSettings.cwSideToneFreq; +} +ssCwToneValidate(const long int candidate_value_in, long int* validated_value_out) +{ + *validated_value_out = LIMIT(candidate_value_in,100,2000); +} +ssCwToneChange(const long int new_value, char* buff_out, const size_t buff_out_size) +{ + globalSettings.cwSideToneFreq = new_value; + tone(CW_TONE, globalSettings.cwSideToneFreq); + ltoa(globalSettings.cwSideToneFreq,buff_out,10); + strncat_P(buff_out,(const char*)F("Hz"),buff_out_size - strlen(buff_out)); +} +ssCwToneFinalize(const long int final_value) +{ + noTone(CW_TONE); + globalSettings.cwSideToneFreq = final_value; + SaveSettingsToEeprom(); +} +const char SS_CW_TONE_T [] PROGMEM = "Set CW Tone"; +const char SS_CW_TONE_A [] PROGMEM = "Select a frequency that\nCW mode to tune for"; +const SettingScreen_t ssTone PROGMEM = { + SS_CW_TONE_T, + SS_CW_TONE_A, + 1, + 10, + ssCwToneInitialize, + ssCwToneValidate, + ssCwToneChange, + ssCwToneFinalize +}; +void runToneSetting(){runSetting(&ssTone);} + void setupCwDelay(){ int knob = 0; int prev_cw_delay; From dc1da20b13072ed632ea6d6b3f37620653e65208 Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Sun, 19 Jan 2020 19:40:03 -0800 Subject: [PATCH 21/29] Refactor cw switch delay --- setup.cpp | 68 ++++++++++++++++++++++++++++--------------------------- 1 file changed, 35 insertions(+), 33 deletions(-) diff --git a/setup.cpp b/setup.cpp index a2d91f0..994d589 100644 --- a/setup.cpp +++ b/setup.cpp @@ -226,22 +226,22 @@ const SettingScreen_t ssBfo PROGMEM = { void runBfoSetting(){runSetting(&ssBfo);} //CW Tone -ssCwToneInitialize(long int* start_value_out) +void ssCwToneInitialize(long int* start_value_out) { *start_value_out = globalSettings.cwSideToneFreq; } -ssCwToneValidate(const long int candidate_value_in, long int* validated_value_out) +void ssCwToneValidate(const long int candidate_value_in, long int* validated_value_out) { *validated_value_out = LIMIT(candidate_value_in,100,2000); } -ssCwToneChange(const long int new_value, char* buff_out, const size_t buff_out_size) +void ssCwToneChange(const long int new_value, char* buff_out, const size_t buff_out_size) { globalSettings.cwSideToneFreq = new_value; tone(CW_TONE, globalSettings.cwSideToneFreq); ltoa(globalSettings.cwSideToneFreq,buff_out,10); strncat_P(buff_out,(const char*)F("Hz"),buff_out_size - strlen(buff_out)); } -ssCwToneFinalize(const long int final_value) +void ssCwToneFinalize(const long int final_value) { noTone(CW_TONE); globalSettings.cwSideToneFreq = final_value; @@ -261,36 +261,38 @@ const SettingScreen_t ssTone PROGMEM = { }; void runToneSetting(){runSetting(&ssTone);} -void setupCwDelay(){ - int knob = 0; - int prev_cw_delay; - - //displayDialog(F("Set CW T/R Delay"),F("Press tune to Save")); - - prev_cw_delay = globalSettings.cwActiveTimeoutMs; - - ltoa(globalSettings.cwActiveTimeoutMs, b, 10); - strncat_P(b,(const char*)F(" msec"),sizeof(b) - strlen(b)); - displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); - - while (!btnDown()){ - knob = enc_read(); - - if (knob < 0 && globalSettings.cwActiveTimeoutMs > 100) - globalSettings.cwActiveTimeoutMs -= 100; - else if (knob > 0 && globalSettings.cwActiveTimeoutMs < 1000) - globalSettings.cwActiveTimeoutMs += 100; - else - continue; //don't update the frequency or the display - - ltoa(globalSettings.cwActiveTimeoutMs, b, 10); - strncat_P(b,(const char*)F(" msec"),sizeof(b) - strlen(b)); - displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); - - } - +//CW Switch Delay +void ssCwSwitchDelayInitialize(long int* start_value_out) +{ + *start_value_out = globalSettings.cwActiveTimeoutMs; +} +void ssCwSwitchDelayValidate(const long int candidate_value_in, long int* validated_value_out) +{ + *validated_value_out = LIMIT(candidate_value_in,100,1000); +} +void ssCwSwitchDelayChange(const long int new_value, char* buff_out, const size_t buff_out_size) +{ + ltoa(new_value,buff_out,10); + strncat_P(buff_out,(const char*)F("ms"),buff_out_size - strlen(buff_out)); +} +void ssCwSwitchDelayFinalize(const long int final_value) +{ + globalSettings.cwActiveTimeoutMs = final_value; SaveSettingsToEeprom(); } +const char SS_CW_SWITCH_T [] PROGMEM = "Set CW Tx/Rx Switch Delay"; +const char SS_CW_SWITCH_A [] PROGMEM = "Select how long the radio\nshould wait before switching\nbetween TX and RX when in\nCW mode"; +const SettingScreen_t ssCwSwitchDelay PROGMEM = { + SS_CW_SWITCH_T, + SS_CW_SWITCH_A, + 1, + 100, + ssCwSwitchDelayInitialize, + ssCwSwitchDelayValidate, + ssCwSwitchDelayChange, + ssCwSwitchDelayFinalize +}; +void runCwSwitchDelaySetting(){runSetting(&ssCwSwitchDelay);} void formatKeyerEnum(char* output, const KeyerMode_e mode) { @@ -461,7 +463,7 @@ const MenuItem_t cwMenu [] PROGMEM { {MT_CW,nullptr},//Title {MI_CW_SPEED,setupCwSpeed}, {MI_CW_TONE,runToneSetting}, - {MI_CW_DELAY,setupCwDelay}, + {MI_CW_DELAY,runCwSwitchDelaySetting}, {MI_CW_KEYER,setupKeyer}, }; void runCwMenu(){RUN_MENU(cwMenu);} From 9af0c1c965b1336b2356393e6e34a5f0074e9869 Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Sun, 19 Jan 2020 19:50:11 -0800 Subject: [PATCH 22/29] Refactor keyer setting into new menu style --- setup.cpp | 69 ++++++++++++++++++++++++++----------------------------- 1 file changed, 32 insertions(+), 37 deletions(-) diff --git a/setup.cpp b/setup.cpp index 994d589..6bfdec8 100644 --- a/setup.cpp +++ b/setup.cpp @@ -294,50 +294,45 @@ const SettingScreen_t ssCwSwitchDelay PROGMEM = { }; void runCwSwitchDelaySetting(){runSetting(&ssCwSwitchDelay);} -void formatKeyerEnum(char* output, const KeyerMode_e mode) +//CW Keyer +void ssKeyerInitialize(long int* start_value_out) { - if(KeyerMode_e::KEYER_STRAIGHT == mode){ - strcpy_P(output,(const char*)F("< Hand Key >")); + *start_value_out = globalSettings.keyerMode; +} +void ssKeyerValidate(const long int candidate_value_in, long int* validated_value_out) +{ + *validated_value_out = LIMIT(candidate_value_in,KeyerMode_e::KEYER_STRAIGHT,KeyerMode_e::KEYER_IAMBIC_B); +} +void ssKeyerChange(const long int new_value, char* buff_out, const size_t buff_out_size) +{ + if(KeyerMode_e::KEYER_STRAIGHT == new_value){ + strncpy_P(buff_out,(const char*)F("< Hand Key >"),buff_out_size); } - else if(KeyerMode_e::KEYER_IAMBIC_A == mode){ - strcpy_P(output,(const char*)F("< Iambic A >")); + else if(KeyerMode_e::KEYER_IAMBIC_A == new_value){ + strncpy_P(buff_out,(const char*)F("< Iambic A >"),buff_out_size); } else{ - strcpy_P(output,(const char*)F("< Iambic B >")); + strncpy_P(buff_out,(const char*)F("< Iambic B >"),buff_out_size); } } - -void setupKeyer(){ - //displayDialog(F("Set CW Keyer"),F("Press tune to Save")); - - int knob = 0; - uint32_t tmp_mode = globalSettings.keyerMode; - formatKeyerEnum(c, tmp_mode); - displayText(c, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); - - while (!btnDown()) - { - knob = enc_read(); - if(knob == 0){ - active_delay(50); - continue; - } - if(knob < 0 && tmp_mode > KeyerMode_e::KEYER_STRAIGHT){ - tmp_mode--; - } - if(knob > 0 && tmp_mode < KeyerMode_e::KEYER_IAMBIC_B){ - tmp_mode++; - } - - formatKeyerEnum(c,tmp_mode); - displayText(c, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); - } - - active_delay(500); - - globalSettings.keyerMode = tmp_mode; +void ssKeyerFinalize(const long int final_value) +{ + globalSettings.keyerMode = final_value; SaveSettingsToEeprom(); } +const char SS_KEYER_T [] PROGMEM = "Set CW Keyer Type"; +const char SS_KEYER_A [] PROGMEM = "Select which type of\nkeyer/paddle is being used"; +const SettingScreen_t ssKeyer PROGMEM = { + SS_KEYER_T, + SS_KEYER_A, + 10, + 1, + ssKeyerInitialize, + ssKeyerValidate, + ssKeyerChange, + ssKeyerFinalize +}; +void runKeyerSetting(){runSetting(&ssKeyer);} void setupCwSpeed() { @@ -464,7 +459,7 @@ const MenuItem_t cwMenu [] PROGMEM { {MI_CW_SPEED,setupCwSpeed}, {MI_CW_TONE,runToneSetting}, {MI_CW_DELAY,runCwSwitchDelaySetting}, - {MI_CW_KEYER,setupKeyer}, + {MI_CW_KEYER,runKeyerSetting}, }; void runCwMenu(){RUN_MENU(cwMenu);} From 4265d56d327337094195c59fa0c8c2fe2d6e90af Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Sun, 19 Jan 2020 19:50:54 -0800 Subject: [PATCH 23/29] Remove unused setupCwTone --- setup.cpp | 26 -------------------------- 1 file changed, 26 deletions(-) diff --git a/setup.cpp b/setup.cpp index 6bfdec8..cea0064 100644 --- a/setup.cpp +++ b/setup.cpp @@ -361,32 +361,6 @@ void setupCwSpeed() SaveSettingsToEeprom(); } -void setupCwTone(){ - //displayDialog(F("Set CW Tone (Hz)"),F("Press tune to Save")); - - tone(CW_TONE, globalSettings.cwSideToneFreq); - itoa(globalSettings.cwSideToneFreq, b, 10); - displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); - - while(!btnDown()){ - int knob = enc_read(); - - if (knob > 0 && globalSettings.cwSideToneFreq < 2000) - globalSettings.cwSideToneFreq += 10; - else if (knob < 0 && globalSettings.cwSideToneFreq > 100 ) - globalSettings.cwSideToneFreq -= 10; - else - continue; //don't update the frequency or the display - - tone(CW_TONE, globalSettings.cwSideToneFreq); - itoa(globalSettings.cwSideToneFreq, b, 10); - displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); - } - noTone(CW_TONE); - - SaveSettingsToEeprom(); -} - void setupResetAll() { //displayDialog(F("Reset all cals and settings?"),F("Press tune to Confirm")); From ece69fff3a6ea59f30d36fb0671c0c5129f30c45 Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Sun, 19 Jan 2020 20:15:01 -0800 Subject: [PATCH 24/29] Let morseText play back at any desired rate, not just the default one --- morse.cpp | 24 ++++++++---------------- morse.h | 4 ++-- 2 files changed, 10 insertions(+), 18 deletions(-) diff --git a/morse.cpp b/morse.cpp index 2afaeaa..20a1020 100644 --- a/morse.cpp +++ b/morse.cpp @@ -56,12 +56,12 @@ static const PROGMEM struct Morse morse_table[] = { {'?', 0x8c}, // 10001100 }; -static void morseLetter(char c){ +static void morseLetter(char c, uint16_t dit_duration_ms){ unsigned char mask = 0x80; //handle space character as three dashes if (c == ' '){ - active_delay(9 * globalSettings.cwDitDurationMs); + active_delay(7 * dit_duration_ms); //Serial.print(' '); return; } @@ -81,36 +81,28 @@ static void morseLetter(char c){ while(mask){ tone(CW_TONE, globalSettings.cwSideToneFreq,10000); if (mask & code){ - delay(3 * globalSettings.cwDitDurationMs); + delay(3 * dit_duration_ms); //Serial.print('-'); } else{ - delay(globalSettings.cwDitDurationMs); + delay(dit_duration_ms); //Serial.print('.'); } //Serial.print('#'); noTone(CW_TONE); - delay(globalSettings.cwDitDurationMs); // space between dots and dashes + delay(dit_duration_ms); // space between dots and dashes mask = mask >> 1; } //Serial.println('@'); - delay(2*globalSettings.cwDitDurationMs); // space between letters is a dash (3 dots), one dot's space has already been sent + delay(2*dit_duration_ms); // space between letters is a dash (3 dots), one dot's space has already been sent break;//We've played the letter, so don't bother checking the rest of the list } } } -void morseText(char *text){ -// while (1){ - noTone(CW_TONE); - delay(1000); - tone(CW_TONE, 600); - delay(1000); -// } - - //Serial.println(globalSettings.cwSideToneFreq); +void morseText(char *text, uint16_t dit_duration_ms){ while(*text){ - morseLetter(*text++); + morseLetter(*text++, dit_duration_ms); } } diff --git a/morse.h b/morse.h index 9c2de08..5e7d76f 100644 --- a/morse.h +++ b/morse.h @@ -1,3 +1,3 @@ +#include "settings.h" //sends out morse code at the speed set by cwSpeed -extern int cwSpeed; //this is actuall the dot period in milliseconds -void morseText(char *text); +void morseText(char *text, uint16_t dit_duration_ms = globalSettings.cwDitDurationMs); From 850cda59e45dfd1fed0ccd7875e2862d9aa5b31a Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Sun, 19 Jan 2020 20:20:03 -0800 Subject: [PATCH 25/29] Refactor cw play speed setting --- setup.cpp | 53 +++++++++++++++++++++++++++++------------------------ 1 file changed, 29 insertions(+), 24 deletions(-) diff --git a/setup.cpp b/setup.cpp index cea0064..747c299 100644 --- a/setup.cpp +++ b/setup.cpp @@ -334,32 +334,37 @@ const SettingScreen_t ssKeyer PROGMEM = { }; void runKeyerSetting(){runSetting(&ssKeyer);} -void setupCwSpeed() +//CW Speed +void ssCwSpeedInitialize(long int* start_value_out) { - //displayDialog(F("Set CW Speed (WPM)"),F("Press tune to Save")); - - unsigned int wpm = 1200/globalSettings.cwDitDurationMs; - - itoa(wpm, b, 10); - displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); - - while (!btnDown()){ - int knob = enc_read(); - - if (knob < 0 && wpm > 1) - --wpm; - else if (knob > 0 && wpm < 100) - ++wpm; - else - continue;//don't update the frequency or the display - - itoa(wpm, b, 10); - displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); - } - - globalSettings.cwDitDurationMs = 1200/wpm; + *start_value_out = 1200L/globalSettings.cwDitDurationMs; +} +void ssCwSpeedValidate(const long int candidate_value_in, long int* validated_value_out) +{ + *validated_value_out = LIMIT(candidate_value_in,1,100); +} +void ssCwSpeedChange(const long int new_value, char* buff_out, const size_t buff_out_size) +{ + ltoa(new_value, buff_out, 10); +} +void ssCwSpeedFinalize(const long int final_value) +{ + globalSettings.cwDitDurationMs = 1200L/final_value; SaveSettingsToEeprom(); } +const char SS_CW_SPEED_T [] PROGMEM = "Set CW Speed"; +const char SS_CW_SPEED_A [] PROGMEM = "Select speed to play CW\ncharacters"; +const SettingScreen_t ssCwSpeed PROGMEM = { + SS_CW_SPEED_T, + SS_CW_SPEED_A, + 5, + 1, + ssCwSpeedInitialize, + ssCwSpeedValidate, + ssCwSpeedChange, + ssCwSpeedFinalize +}; +void runCwSpeedSetting(){runSetting(&ssCwSpeed);} void setupResetAll() { @@ -430,7 +435,7 @@ const char MI_CW_DELAY [] PROGMEM = "Tx/Rx Switching Delay"; const char MI_CW_KEYER [] PROGMEM = "Keyer Type"; const MenuItem_t cwMenu [] PROGMEM { {MT_CW,nullptr},//Title - {MI_CW_SPEED,setupCwSpeed}, + {MI_CW_SPEED,runCwSpeedSetting}, {MI_CW_TONE,runToneSetting}, {MI_CW_DELAY,runCwSwitchDelaySetting}, {MI_CW_KEYER,runKeyerSetting}, From d032664620332bac4e6948df609d7af00bc1e794 Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Sun, 19 Jan 2020 20:21:51 -0800 Subject: [PATCH 26/29] Re-order cw settings to match the menu --- setup.cpp | 64 +++++++++++++++++++++++++++---------------------------- 1 file changed, 32 insertions(+), 32 deletions(-) diff --git a/setup.cpp b/setup.cpp index 747c299..7ffda75 100644 --- a/setup.cpp +++ b/setup.cpp @@ -225,6 +225,38 @@ const SettingScreen_t ssBfo PROGMEM = { }; void runBfoSetting(){runSetting(&ssBfo);} +//CW Speed +void ssCwSpeedInitialize(long int* start_value_out) +{ + *start_value_out = 1200L/globalSettings.cwDitDurationMs; +} +void ssCwSpeedValidate(const long int candidate_value_in, long int* validated_value_out) +{ + *validated_value_out = LIMIT(candidate_value_in,1,100); +} +void ssCwSpeedChange(const long int new_value, char* buff_out, const size_t buff_out_size) +{ + ltoa(new_value, buff_out, 10); +} +void ssCwSpeedFinalize(const long int final_value) +{ + globalSettings.cwDitDurationMs = 1200L/final_value; + SaveSettingsToEeprom(); +} +const char SS_CW_SPEED_T [] PROGMEM = "Set CW Speed"; +const char SS_CW_SPEED_A [] PROGMEM = "Select speed to play CW\ncharacters"; +const SettingScreen_t ssCwSpeed PROGMEM = { + SS_CW_SPEED_T, + SS_CW_SPEED_A, + 5, + 1, + ssCwSpeedInitialize, + ssCwSpeedValidate, + ssCwSpeedChange, + ssCwSpeedFinalize +}; +void runCwSpeedSetting(){runSetting(&ssCwSpeed);} + //CW Tone void ssCwToneInitialize(long int* start_value_out) { @@ -334,38 +366,6 @@ const SettingScreen_t ssKeyer PROGMEM = { }; void runKeyerSetting(){runSetting(&ssKeyer);} -//CW Speed -void ssCwSpeedInitialize(long int* start_value_out) -{ - *start_value_out = 1200L/globalSettings.cwDitDurationMs; -} -void ssCwSpeedValidate(const long int candidate_value_in, long int* validated_value_out) -{ - *validated_value_out = LIMIT(candidate_value_in,1,100); -} -void ssCwSpeedChange(const long int new_value, char* buff_out, const size_t buff_out_size) -{ - ltoa(new_value, buff_out, 10); -} -void ssCwSpeedFinalize(const long int final_value) -{ - globalSettings.cwDitDurationMs = 1200L/final_value; - SaveSettingsToEeprom(); -} -const char SS_CW_SPEED_T [] PROGMEM = "Set CW Speed"; -const char SS_CW_SPEED_A [] PROGMEM = "Select speed to play CW\ncharacters"; -const SettingScreen_t ssCwSpeed PROGMEM = { - SS_CW_SPEED_T, - SS_CW_SPEED_A, - 5, - 1, - ssCwSpeedInitialize, - ssCwSpeedValidate, - ssCwSpeedChange, - ssCwSpeedFinalize -}; -void runCwSpeedSetting(){runSetting(&ssCwSpeed);} - void setupResetAll() { //displayDialog(F("Reset all cals and settings?"),F("Press tune to Confirm")); From 6d5fb97c62fbdd1522d506c915ef57f80cd1a413 Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Sun, 19 Jan 2020 20:35:43 -0800 Subject: [PATCH 27/29] Refactor reset-all setting --- setup.cpp | 67 ++++++++++++++++++++++++++----------------------------- 1 file changed, 32 insertions(+), 35 deletions(-) diff --git a/setup.cpp b/setup.cpp index 7ffda75..7a1b058 100644 --- a/setup.cpp +++ b/setup.cpp @@ -366,47 +366,45 @@ const SettingScreen_t ssKeyer PROGMEM = { }; void runKeyerSetting(){runSetting(&ssKeyer);} -void setupResetAll() +//Reset all settings +void ssResetAllInitialize(long int* start_value_out) { - //displayDialog(F("Reset all cals and settings?"),F("Press tune to Confirm")); - strncpy_P(b,(const char*)F("No"),sizeof(b)); - displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); - - bool reset_all = false; - while(!btnDown()){ - int knob = enc_read(); - - if(knob > 0){ - reset_all = true; - } - else if(knob < 0){ - reset_all = false; - } - else{ - continue; - } - - if(reset_all){ - strncpy_P(b,(const char*)F("Yes"),sizeof(b)); - } - else{ - strncpy_P(b,(const char*)F("No"),sizeof(b)); - } - - displayText(b, LAYOUT_SETTING_VALUE_X, LAYOUT_SETTING_VALUE_Y, LAYOUT_SETTING_VALUE_WIDTH, LAYOUT_SETTING_VALUE_HEIGHT, COLOR_TEXT, COLOR_SETTING_BACKGROUND, COLOR_BACKGROUND); + *start_value_out = 0;//Default to NOT resetting +} +void ssResetAllValidate(const long int candidate_value_in, long int* validated_value_out) +{ + *validated_value_out = LIMIT(candidate_value_in,0,1); +} +void ssResetAllChange(const long int new_value, char* buff_out, const size_t buff_out_size) +{ + if(new_value){ + strncpy_P(buff_out,(const char*)F("Yes"),buff_out_size); } - - while(btnDown()){ - active_delay(50); + else{ + strncpy_P(buff_out,(const char*)F("No"),buff_out_size); } - active_delay(50); - - if(reset_all){ +} +void ssResetAllFinalize(const long int final_value) +{ + if(final_value){ LoadDefaultSettings(); SaveSettingsToEeprom(); setup(); } } +const char SS_RESET_ALL_T [] PROGMEM = "Reset All Cals/Settings"; +const char SS_RESET_ALL_A [] PROGMEM = "WARNING: Selecting \"Yes\"\nwill reset all calibrations and\nsettings to their default\nvalues"; +const SettingScreen_t ssResetAll PROGMEM = { + SS_RESET_ALL_T, + SS_RESET_ALL_A, + 20, + 1, + ssResetAllInitialize, + ssResetAllValidate, + ssResetAllChange, + ssResetAllFinalize +}; +void runResetAllSetting(){runSetting(&ssResetAll);} struct MenuItem_t { const char* const ItemName; @@ -443,12 +441,11 @@ const MenuItem_t cwMenu [] PROGMEM { void runCwMenu(){RUN_MENU(cwMenu);} const char MT_SETTINGS [] PROGMEM = "Settings"; -const char MI_RESET [] PROGMEM = "Reset all Cals/Settings"; const MenuItem_t mainMenu [] PROGMEM { {MT_SETTINGS,nullptr},//Title {MT_CAL,runCalibrationMenu}, {MT_CW,runCwMenu}, - {MI_RESET,setupResetAll}, + {SS_RESET_ALL_T,runResetAllSetting}, }; const char MI_EXIT [] PROGMEM = "Exit"; From dd3a6162afa5647d1e1c49cff05d6a37ae9d7502 Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Sun, 19 Jan 2020 20:42:44 -0800 Subject: [PATCH 28/29] Set global setting so that setFrequency will work correctly --- setup.cpp | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/setup.cpp b/setup.cpp index 7a1b058..c011018 100644 --- a/setup.cpp +++ b/setup.cpp @@ -199,8 +199,9 @@ void ssBfoValidate(const long int candidate_value_in, long int* validated_value_ } void ssBfoChange(const long int new_value, char* buff_out, const size_t buff_out_size) { - si5351bx_setfreq(0, new_value); + globalSettings.usbCarrierFreq = new_value; setFrequency(GetActiveVfoFreq()); + si5351bx_setfreq(0, new_value); formatFreq(new_value,buff_out,buff_out_size); strncat_P(buff_out,(const char*)F("Hz"),buff_out_size - strlen(buff_out)); } From 1f033b6af4e5c56c30f6f0ebfff775ae6781c85f Mon Sep 17 00:00:00 2001 From: Reed Nightingale Date: Sun, 19 Jan 2020 20:51:19 -0800 Subject: [PATCH 29/29] Save 102 progmem bytes by de-duping strings --- setup.cpp | 30 ++++++++++++------------------ 1 file changed, 12 insertions(+), 18 deletions(-) diff --git a/setup.cpp b/setup.cpp index c011018..14272f7 100644 --- a/setup.cpp +++ b/setup.cpp @@ -174,7 +174,7 @@ void ssLocalOscFinalize(const long int final_value) si5351_set_calibration(globalSettings.oscillatorCal); setFrequency(GetActiveVfoFreq()); } -const char SS_LOCAL_OSC_T [] PROGMEM = "Set Local Osc Calibration"; +const char SS_LOCAL_OSC_T [] PROGMEM = "Local Oscillator"; const char SS_LOCAL_OSC_A [] PROGMEM = "Exit menu, tune so that the\ndial displays the desired freq,\nthen tune here until the\nsignal is zerobeat"; const SettingScreen_t ssLocalOsc PROGMEM = { SS_LOCAL_OSC_T, @@ -212,7 +212,7 @@ void ssBfoFinalize(const long int final_value) si5351bx_setfreq(0, globalSettings.usbCarrierFreq); setFrequency(GetActiveVfoFreq()); } -const char SS_BFO_T [] PROGMEM = "Set BFO Calibration"; +const char SS_BFO_T [] PROGMEM = "Beat Frequency Osc (BFO)"; const char SS_BFO_A [] PROGMEM = "Exit menu, tune to an unused\nfrequency, then tune here\nuntil the audio is between\n300-3000Hz"; const SettingScreen_t ssBfo PROGMEM = { SS_BFO_T, @@ -244,7 +244,7 @@ void ssCwSpeedFinalize(const long int final_value) globalSettings.cwDitDurationMs = 1200L/final_value; SaveSettingsToEeprom(); } -const char SS_CW_SPEED_T [] PROGMEM = "Set CW Speed"; +const char SS_CW_SPEED_T [] PROGMEM = "CW Play Speed"; const char SS_CW_SPEED_A [] PROGMEM = "Select speed to play CW\ncharacters"; const SettingScreen_t ssCwSpeed PROGMEM = { SS_CW_SPEED_T, @@ -280,7 +280,7 @@ void ssCwToneFinalize(const long int final_value) globalSettings.cwSideToneFreq = final_value; SaveSettingsToEeprom(); } -const char SS_CW_TONE_T [] PROGMEM = "Set CW Tone"; +const char SS_CW_TONE_T [] PROGMEM = "CW Tone Frequency"; const char SS_CW_TONE_A [] PROGMEM = "Select a frequency that\nCW mode to tune for"; const SettingScreen_t ssTone PROGMEM = { SS_CW_TONE_T, @@ -313,7 +313,7 @@ void ssCwSwitchDelayFinalize(const long int final_value) globalSettings.cwActiveTimeoutMs = final_value; SaveSettingsToEeprom(); } -const char SS_CW_SWITCH_T [] PROGMEM = "Set CW Tx/Rx Switch Delay"; +const char SS_CW_SWITCH_T [] PROGMEM = "CW Tx -> Rx Switch Delay"; const char SS_CW_SWITCH_A [] PROGMEM = "Select how long the radio\nshould wait before switching\nbetween TX and RX when in\nCW mode"; const SettingScreen_t ssCwSwitchDelay PROGMEM = { SS_CW_SWITCH_T, @@ -353,7 +353,7 @@ void ssKeyerFinalize(const long int final_value) globalSettings.keyerMode = final_value; SaveSettingsToEeprom(); } -const char SS_KEYER_T [] PROGMEM = "Set CW Keyer Type"; +const char SS_KEYER_T [] PROGMEM = "CW Keyer/Paddle Type"; const char SS_KEYER_A [] PROGMEM = "Select which type of\nkeyer/paddle is being used"; const SettingScreen_t ssKeyer PROGMEM = { SS_KEYER_T, @@ -416,28 +416,22 @@ void runMenu(const MenuItem_t* const menu_items, const uint16_t num_items); #define RUN_MENU(menu) runMenu(menu,sizeof(menu)/sizeof(menu[0])) const char MT_CAL [] PROGMEM = "Calibrations"; -const char MI_SET_FREQ [] PROGMEM = "Local Oscillator"; -const char MI_SET_BFO [] PROGMEM = "Beat Frequency Osc (BFO)"; const char MI_TOUCH [] PROGMEM = "Touch Screen"; const MenuItem_t calibrationMenu [] PROGMEM { {MT_CAL,nullptr},//Title - {MI_SET_FREQ,runLocalOscSetting}, - {MI_SET_BFO,runBfoSetting}, + {SS_LOCAL_OSC_T,runLocalOscSetting}, + {SS_BFO_T,runBfoSetting}, {MI_TOUCH,setupTouch}, }; void runCalibrationMenu(){RUN_MENU(calibrationMenu);} const char MT_CW [] PROGMEM = "CW/Morse Setup"; -const char MI_CW_SPEED [] PROGMEM = "Play Speed (WPM)"; -const char MI_CW_TONE [] PROGMEM = "Tone Frequency"; -const char MI_CW_DELAY [] PROGMEM = "Tx/Rx Switching Delay"; -const char MI_CW_KEYER [] PROGMEM = "Keyer Type"; const MenuItem_t cwMenu [] PROGMEM { {MT_CW,nullptr},//Title - {MI_CW_SPEED,runCwSpeedSetting}, - {MI_CW_TONE,runToneSetting}, - {MI_CW_DELAY,runCwSwitchDelaySetting}, - {MI_CW_KEYER,runKeyerSetting}, + {SS_CW_SPEED_T,runCwSpeedSetting}, + {SS_CW_TONE_T,runToneSetting}, + {SS_CW_SWITCH_T,runCwSwitchDelaySetting}, + {SS_KEYER_T,runKeyerSetting}, }; void runCwMenu(){RUN_MENU(cwMenu);}