388 lines
13 KiB
C++
388 lines
13 KiB
C++
#include <Arduino.h>
|
|
#include <EEPROM.h>
|
|
#include "morse.h"
|
|
#include "settings.h"
|
|
#include "ubitx.h"
|
|
#include "nano_gui.h"
|
|
|
|
/** Menus
|
|
* The Radio menus are accessed by tapping on the function button.
|
|
* - The main loop() constantly looks for a button press and calls doMenu() when it detects
|
|
* a function button press.
|
|
* - As the encoder is rotated, at every 10th pulse, the next or the previous menu
|
|
* item is displayed. Each menu item is controlled by it's own function.
|
|
* - Eache menu function may be called to display itself
|
|
* - Each of these menu routines is called with a button parameter.
|
|
* - The btn flag denotes if the menu itme was clicked on or not.
|
|
* - If the menu item is clicked on, then it is selected,
|
|
* - If the menu item is NOT clicked on, then the menu's prompt is to be displayed
|
|
*/
|
|
|
|
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;
|
|
static const unsigned int COLOR_SETTING_BACKGROUND = DISPLAY_NAVY;
|
|
static const unsigned int COLOR_ACTIVE_BORDER = DISPLAY_WHITE;
|
|
static const unsigned int COLOR_INACTIVE_BORDER = COLOR_BACKGROUND;
|
|
|
|
static const unsigned int LAYOUT_OUTER_BORDER_X = 10;
|
|
static const unsigned int LAYOUT_OUTER_BORDER_Y = 10;
|
|
static const unsigned int LAYOUT_OUTER_BORDER_WIDTH = 300;
|
|
static const unsigned int LAYOUT_OUTER_BORDER_HEIGHT = 220;
|
|
|
|
static const unsigned int LAYOUT_INNER_BORDER_X = 12;
|
|
static const unsigned int LAYOUT_INNER_BORDER_Y = 12;
|
|
static const unsigned int LAYOUT_INNER_BORDER_WIDTH = 296;
|
|
static const unsigned int LAYOUT_INNER_BORDER_HEIGHT = 216;
|
|
|
|
static const unsigned int LAYOUT_TITLE_X = LAYOUT_INNER_BORDER_X;
|
|
static const unsigned int LAYOUT_TITLE_Y = LAYOUT_INNER_BORDER_Y;
|
|
static const unsigned int LAYOUT_TITLE_WIDTH = LAYOUT_INNER_BORDER_WIDTH;
|
|
static const unsigned int LAYOUT_TITLE_HEIGHT = 35;
|
|
|
|
static const unsigned int LAYOUT_ITEM_X = 30;
|
|
static const unsigned int LAYOUT_ITEM_Y = LAYOUT_TITLE_Y + LAYOUT_TITLE_HEIGHT + 5;
|
|
static const unsigned int LAYOUT_ITEM_WIDTH = 260;
|
|
static const unsigned int LAYOUT_ITEM_HEIGHT = 30;
|
|
static const unsigned int LAYOUT_ITEM_PITCH_Y = LAYOUT_ITEM_HEIGHT + 1;
|
|
|
|
static const unsigned int LAYOUT_SETTING_VALUE_X = LAYOUT_ITEM_X;
|
|
static const unsigned int LAYOUT_SETTING_VALUE_Y = LAYOUT_ITEM_Y + 3*LAYOUT_ITEM_PITCH_Y;
|
|
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;
|
|
|
|
void displayDialog(const __FlashStringHelper* title, const __FlashStringHelper* instructions){
|
|
strcpy_P(b,(const char*)title);
|
|
strcpy_P(c,(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);
|
|
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);
|
|
}
|
|
|
|
void printCarrierFreq(unsigned long freq)
|
|
{
|
|
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);
|
|
}
|
|
|
|
void setupFreq(){
|
|
displayDialog(F("Set Frequency"),F("Push TUNE to Save"));
|
|
|
|
//round off the the nearest khz
|
|
{
|
|
uint32_t freq = GetActiveVfoFreq();
|
|
freq = (freq/1000l)* 1000l;
|
|
setFrequency(freq);
|
|
}
|
|
|
|
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);
|
|
}
|
|
|
|
SaveSettingsToEeprom();
|
|
initOscillators();
|
|
si5351_set_calibration(globalSettings.oscillatorCal);
|
|
setFrequency(GetActiveVfoFreq());
|
|
|
|
//debounce and delay
|
|
while(btnDown())
|
|
active_delay(50);
|
|
active_delay(100);
|
|
}
|
|
|
|
void setupBFO(){
|
|
displayDialog(F("Set BFO"),F("Press TUNE to Save"));
|
|
|
|
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);
|
|
}
|
|
|
|
SaveSettingsToEeprom();
|
|
si5351bx_setfreq(0, globalSettings.usbCarrierFreq);
|
|
setFrequency(GetActiveVfoFreq());
|
|
}
|
|
|
|
void setupCwDelay(){
|
|
int knob = 0;
|
|
int prev_cw_delay;
|
|
|
|
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);
|
|
strcat_P(b,(const char*)F(" msec"));
|
|
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);
|
|
strcat_P(b,(const char*)F(" msec"));
|
|
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);
|
|
|
|
}
|
|
|
|
SaveSettingsToEeprom();
|
|
active_delay(500);
|
|
setupExit();
|
|
}
|
|
|
|
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);
|
|
}
|
|
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{
|
|
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);
|
|
}
|
|
|
|
int knob = 0;
|
|
uint32_t tmp_mode = globalSettings.keyerMode;
|
|
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++;
|
|
}
|
|
|
|
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);
|
|
}
|
|
}
|
|
|
|
active_delay(500);
|
|
|
|
globalSettings.keyerMode = tmp_mode;
|
|
SaveSettingsToEeprom();
|
|
|
|
setupExit();
|
|
}
|
|
|
|
const char MI_SET_FREQ [] PROGMEM = "Set Freq...";
|
|
const char MI_SET_BFO [] PROGMEM = "Set BFO...";
|
|
const char MI_CW_DELAY [] PROGMEM = "CW Delay...";
|
|
const char MI_CW_KEYER [] PROGMEM = "CW Keyer...";
|
|
const char MI_TOUCH [] PROGMEM = "Touch Screen...";
|
|
const char MI_EXIT [] PROGMEM = "Exit";
|
|
|
|
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(){
|
|
displayClear(COLOR_BACKGROUND);
|
|
strcpy_P(b,(const char*)F("Setup"));
|
|
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);
|
|
}
|
|
}
|
|
|
|
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
|
|
|
|
//Don't update if we're already on the right selection
|
|
if(prevPuck == i){
|
|
return;
|
|
}
|
|
|
|
//Clear old
|
|
displayRect(LAYOUT_ITEM_X, LAYOUT_ITEM_Y + (prevPuck*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;
|
|
|
|
}
|
|
|
|
void doSetup2(){
|
|
static const unsigned int COUNTS_PER_ITEM = 10;
|
|
int select=0, i, btnState;
|
|
|
|
drawSetupMenu();
|
|
movePuck(select);
|
|
|
|
//wait for the button to be raised up
|
|
while(btnDown())
|
|
active_delay(50);
|
|
active_delay(50); //debounce
|
|
|
|
menuOn = 2;
|
|
|
|
while (menuOn){
|
|
i = enc_read();
|
|
|
|
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);
|
|
}
|
|
|
|
if (!btnDown()){
|
|
active_delay(50);
|
|
continue;
|
|
}
|
|
|
|
//wait for the touch to lift off and debounce
|
|
while(btnDown()){
|
|
active_delay(50);
|
|
}
|
|
active_delay(300);
|
|
|
|
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();
|
|
}
|
|
|
|
//debounce the button
|
|
while(btnDown())
|
|
active_delay(50);
|
|
active_delay(50);
|
|
|
|
checkCAT();
|
|
guiUpdate();
|
|
}
|