Base sources for Module 17 UI
This commit is contained in:
parent
76ded06751
commit
d8c4ba4420
|
|
@ -90,7 +90,9 @@ openrtx_ui_default = ['openrtx/src/ui/default/ui.c',
|
|||
'openrtx/src/ui/default/ui_menu.c',
|
||||
'openrtx/src/ui/default/ui_strings.c']
|
||||
|
||||
openrtx_ui_module17 = []
|
||||
openrtx_ui_module17 = ['openrtx/src/ui/module17/ui.c',
|
||||
'openrtx/src/ui/module17/ui_main.c',
|
||||
'openrtx/src/ui/module17/ui_menu.c']
|
||||
|
||||
##
|
||||
## Selection of main entrypoint
|
||||
|
|
|
|||
|
|
@ -0,0 +1,212 @@
|
|||
/***************************************************************************
|
||||
* Copyright (C) 2022 by Federico Amedeo Izzo IU2NUO, *
|
||||
* Niccolò Izzo IU2KIN, *
|
||||
* Silvano Seva IU2KWO *
|
||||
* *
|
||||
* This program is free software; you can redistribute it and/or modify *
|
||||
* it under the terms of the GNU General Public License as published by *
|
||||
* the Free Software Foundation; either version 3 of the License, or *
|
||||
* (at your option) any later version. *
|
||||
* *
|
||||
* This program is distributed in the hope that it will be useful, *
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
|
||||
* GNU General Public License for more details. *
|
||||
* *
|
||||
* You should have received a copy of the GNU General Public License *
|
||||
* along with this program; if not, see <http://www.gnu.org/licenses/> *
|
||||
***************************************************************************/
|
||||
|
||||
#ifndef UI_MOD17_H
|
||||
#define UI_MOD17_H
|
||||
|
||||
#include <stdbool.h>
|
||||
#include <state.h>
|
||||
#include <graphics.h>
|
||||
#include <interfaces/keyboard.h>
|
||||
#include <stdint.h>
|
||||
#include <event.h>
|
||||
#include <hwconfig.h>
|
||||
#include <ui.h>
|
||||
|
||||
// Maximum menu entry length
|
||||
#define MAX_ENTRY_LEN 21
|
||||
// Frequency digits
|
||||
#define FREQ_DIGITS 7
|
||||
// Time & Date digits
|
||||
#define TIMEDATE_DIGITS 10
|
||||
// Max number of UI events
|
||||
#define MAX_NUM_EVENTS 16
|
||||
|
||||
enum uiScreen
|
||||
{
|
||||
MAIN_VFO = 0,
|
||||
MAIN_VFO_INPUT,
|
||||
MAIN_MEM,
|
||||
MODE_VFO,
|
||||
MODE_MEM,
|
||||
MENU_TOP,
|
||||
MENU_BANK,
|
||||
MENU_CHANNEL,
|
||||
MENU_CONTACTS,
|
||||
MENU_GPS,
|
||||
MENU_SETTINGS,
|
||||
MENU_BACKUP_RESTORE,
|
||||
MENU_BACKUP,
|
||||
MENU_RESTORE,
|
||||
MENU_INFO,
|
||||
MENU_ABOUT,
|
||||
SETTINGS_TIMEDATE,
|
||||
SETTINGS_TIMEDATE_SET,
|
||||
SETTINGS_DISPLAY,
|
||||
SETTINGS_GPS,
|
||||
SETTINGS_M17,
|
||||
SETTINGS_RESET2DEFAULTS,
|
||||
LOW_BAT
|
||||
};
|
||||
|
||||
enum SetRxTx
|
||||
{
|
||||
SET_RX = 0,
|
||||
SET_TX
|
||||
};
|
||||
|
||||
// This enum is needed to have item numbers that match
|
||||
// menu elements even if some elements may be missing (GPS)
|
||||
enum menuItems
|
||||
{
|
||||
M_SETTINGS = 0,
|
||||
#ifdef GPS_PRESENT
|
||||
M_GPS,
|
||||
#endif
|
||||
M_INFO,
|
||||
M_ABOUT
|
||||
};
|
||||
|
||||
enum settingsItems
|
||||
{
|
||||
S_DISPLAY = 0
|
||||
#ifdef RTC_PRESENT
|
||||
,S_TIMEDATE
|
||||
#endif
|
||||
#ifdef GPS_PRESENT
|
||||
,S_GPS
|
||||
#endif
|
||||
,S_M17
|
||||
,S_RESET2DEFAULTS
|
||||
};
|
||||
|
||||
enum backupRestoreItems
|
||||
{
|
||||
BR_BACKUP = 0,
|
||||
BR_RESTORE
|
||||
};
|
||||
|
||||
enum displayItems
|
||||
{
|
||||
D_BRIGHTNESS = 0
|
||||
#ifdef SCREEN_CONTRAST
|
||||
,D_CONTRAST
|
||||
#endif
|
||||
,D_TIMER
|
||||
};
|
||||
|
||||
#ifdef GPS_PRESENT
|
||||
enum settingsGPSItems
|
||||
{
|
||||
G_ENABLED = 0,
|
||||
G_SET_TIME,
|
||||
G_TIMEZONE
|
||||
};
|
||||
#endif
|
||||
|
||||
/**
|
||||
* Struct containing a set of positions and sizes that get
|
||||
* calculated for the selected display size.
|
||||
* Using these parameters make the UI automatically adapt
|
||||
* To displays of different sizes
|
||||
*/
|
||||
typedef struct layout_t
|
||||
{
|
||||
uint16_t hline_h;
|
||||
uint16_t top_h;
|
||||
uint16_t line1_h;
|
||||
uint16_t line2_h;
|
||||
uint16_t line3_h;
|
||||
uint16_t menu_h;
|
||||
uint16_t bottom_h;
|
||||
uint16_t bottom_pad;
|
||||
uint16_t status_v_pad;
|
||||
uint16_t horizontal_pad;
|
||||
uint16_t text_v_offset;
|
||||
point_t top_pos;
|
||||
point_t line1_pos;
|
||||
point_t line2_pos;
|
||||
point_t line3_pos;
|
||||
point_t bottom_pos;
|
||||
fontSize_t top_font;
|
||||
fontSize_t line1_font;
|
||||
fontSize_t line2_font;
|
||||
fontSize_t line3_font;
|
||||
fontSize_t bottom_font;
|
||||
fontSize_t input_font;
|
||||
fontSize_t menu_font;
|
||||
fontSize_t mode_font_big;
|
||||
fontSize_t mode_font_small;
|
||||
} layout_t;
|
||||
|
||||
/**
|
||||
* This structs contains state variables internal to the
|
||||
* UI that need to be kept between executions of the UI
|
||||
* This state does not need to be saved on device poweroff
|
||||
*/
|
||||
typedef struct ui_state_t
|
||||
{
|
||||
// Index of the currently selected menu entry
|
||||
uint8_t menu_selected;
|
||||
// If true we can change a menu entry value with UP/DOWN
|
||||
bool edit_mode;
|
||||
// Variables used for VFO input
|
||||
uint8_t input_number;
|
||||
uint8_t input_position;
|
||||
uint8_t input_set;
|
||||
long long last_keypress;
|
||||
freq_t new_rx_frequency;
|
||||
freq_t new_tx_frequency;
|
||||
char new_rx_freq_buf[14];
|
||||
char new_tx_freq_buf[14];
|
||||
#ifdef RTC_PRESENT
|
||||
// Variables used for Time & Date input
|
||||
datetime_t new_timedate;
|
||||
char new_date_buf[9];
|
||||
char new_time_buf[9];
|
||||
#endif
|
||||
char new_callsign[10];
|
||||
// Which state to return to when we exit menu
|
||||
uint8_t last_main_state;
|
||||
}
|
||||
ui_state_t;
|
||||
|
||||
extern layout_t layout;
|
||||
// Copy of the radio state
|
||||
extern state_t last_state;
|
||||
extern const char *menu_items[];
|
||||
extern const char *settings_items[];
|
||||
extern const char *display_items[];
|
||||
extern const char *settings_gps_items[];
|
||||
extern const char *backup_restore_items[];
|
||||
extern const char *info_items[];
|
||||
extern const char *authors[];
|
||||
extern const uint8_t menu_num;
|
||||
extern const uint8_t settings_num;
|
||||
extern const uint8_t display_num;
|
||||
extern const uint8_t settings_gps_num;
|
||||
extern const uint8_t backup_restore_num;
|
||||
extern const uint8_t info_num;
|
||||
extern const uint8_t author_num;
|
||||
extern const color_t color_black;
|
||||
extern const color_t color_grey;
|
||||
extern const color_t color_white;
|
||||
extern const color_t yellow_fab413;
|
||||
|
||||
#endif /* UI_MOD17_H */
|
||||
File diff suppressed because it is too large
Load Diff
|
|
@ -0,0 +1,271 @@
|
|||
/***************************************************************************
|
||||
* Copyright (C) 2020 - 2022 by Federico Amedeo Izzo IU2NUO, *
|
||||
* Niccolò Izzo IU2KIN *
|
||||
* Frederik Saraci IU2NRO *
|
||||
* Silvano Seva IU2KWO *
|
||||
* *
|
||||
* This program is free software; you can redistribute it and/or modify *
|
||||
* it under the terms of the GNU General Public License as published by *
|
||||
* the Free Software Foundation; either version 3 of the License, or *
|
||||
* (at your option) any later version. *
|
||||
* *
|
||||
* This program is distributed in the hope that it will be useful, *
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
|
||||
* GNU General Public License for more details. *
|
||||
* *
|
||||
* You should have received a copy of the GNU General Public License *
|
||||
* along with this program; if not, see <http://www.gnu.org/licenses/> *
|
||||
***************************************************************************/
|
||||
|
||||
#include <interfaces/platform.h>
|
||||
#include <interfaces/cps_io.h>
|
||||
#include <stdio.h>
|
||||
#include <stdint.h>
|
||||
#include <ui/ui_mod17.h>
|
||||
#include <string.h>
|
||||
|
||||
void _ui_drawMainBackground()
|
||||
{
|
||||
// Print top bar line of hline_h pixel height
|
||||
gfx_drawHLine(layout.top_h, layout.hline_h, color_grey);
|
||||
// Print bottom bar line of 1 pixel height
|
||||
gfx_drawHLine(SCREEN_HEIGHT - layout.bottom_h - 1, layout.hline_h, color_grey);
|
||||
}
|
||||
|
||||
void _ui_drawMainTop()
|
||||
{
|
||||
#ifdef RTC_PRESENT
|
||||
// Print clock on top bar
|
||||
datetime_t local_time = utcToLocalTime(last_state.time,
|
||||
last_state.settings.utc_timezone);
|
||||
gfx_print(layout.top_pos, layout.top_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "%02d:%02d:%02d", local_time.hour,
|
||||
local_time.minute, local_time.second);
|
||||
#endif
|
||||
// If the radio has no built-in battery, print input voltage
|
||||
#ifdef BAT_NONE
|
||||
gfx_print(layout.top_pos, layout.top_font, TEXT_ALIGN_RIGHT,
|
||||
color_white,"%.1fV", last_state.v_bat);
|
||||
#else
|
||||
// Otherwise print battery icon on top bar, use 4 px padding
|
||||
uint16_t bat_width = SCREEN_WIDTH / 9;
|
||||
uint16_t bat_height = layout.top_h - (layout.status_v_pad * 2);
|
||||
point_t bat_pos = {SCREEN_WIDTH - bat_width - layout.horizontal_pad,
|
||||
layout.status_v_pad};
|
||||
gfx_drawBattery(bat_pos, bat_width, bat_height, last_state.charge);
|
||||
#endif
|
||||
// Print radio mode on top bar
|
||||
switch(last_state.channel.mode)
|
||||
{
|
||||
case OPMODE_FM:
|
||||
gfx_print(layout.top_pos, layout.top_font, TEXT_ALIGN_LEFT,
|
||||
color_white, "FM");
|
||||
break;
|
||||
case OPMODE_DMR:
|
||||
gfx_print(layout.top_pos, layout.top_font, TEXT_ALIGN_LEFT,
|
||||
color_white, "DMR");
|
||||
break;
|
||||
case OPMODE_M17:
|
||||
gfx_print(layout.top_pos, layout.top_font, TEXT_ALIGN_LEFT,
|
||||
color_white, "M17");
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void _ui_drawBankChannel()
|
||||
{
|
||||
// Print Bank number, channel number and Channel name
|
||||
uint16_t b = (last_state.bank_enabled) ? last_state.bank : 0;
|
||||
gfx_print(layout.line1_pos, layout.line1_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "%01d-%03d: %.12s",
|
||||
b, last_state.channel_index + 1, last_state.channel.name);
|
||||
}
|
||||
|
||||
void _ui_drawModeInfo(ui_state_t* ui_state)
|
||||
{
|
||||
char bw_str[8] = { 0 };
|
||||
char encdec_str[9] = { 0 };
|
||||
|
||||
rtxStatus_t cfg = rtx_getCurrentStatus();
|
||||
|
||||
switch(last_state.channel.mode)
|
||||
{
|
||||
case OPMODE_FM:
|
||||
// Get Bandwidth string
|
||||
if(last_state.channel.bandwidth == BW_12_5)
|
||||
snprintf(bw_str, 8, "12.5");
|
||||
else if(last_state.channel.bandwidth == BW_20)
|
||||
snprintf(bw_str, 8, "20");
|
||||
else if(last_state.channel.bandwidth == BW_25)
|
||||
snprintf(bw_str, 8, "25");
|
||||
// Get encdec string
|
||||
bool tone_tx_enable = last_state.channel.fm.txToneEn;
|
||||
bool tone_rx_enable = last_state.channel.fm.rxToneEn;
|
||||
if (tone_tx_enable && tone_rx_enable)
|
||||
snprintf(encdec_str, 9, "E+D");
|
||||
else if (tone_tx_enable && !tone_rx_enable)
|
||||
snprintf(encdec_str, 9, "E");
|
||||
else if (!tone_tx_enable && tone_rx_enable)
|
||||
snprintf(encdec_str, 9, "D");
|
||||
else
|
||||
snprintf(encdec_str, 9, " ");
|
||||
|
||||
// Print Bandwidth, Tone and encdec info
|
||||
gfx_print(layout.line2_pos, layout.line2_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "B:%s T:%4.1f S:%s",
|
||||
bw_str, ctcss_tone[last_state.channel.fm.txTone]/10.0f,
|
||||
encdec_str);
|
||||
break;
|
||||
case OPMODE_DMR:
|
||||
// Print talkgroup
|
||||
gfx_print(layout.line2_pos, layout.line2_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "TG:%s",
|
||||
"");
|
||||
break;
|
||||
case OPMODE_M17:
|
||||
{
|
||||
// Print M17 Destination ID on line 3 of 3
|
||||
char *dst = NULL;
|
||||
if(ui_state->edit_mode)
|
||||
dst = ui_state->new_callsign;
|
||||
else
|
||||
dst = (!strnlen(cfg.destination_address, 10)) ?
|
||||
"Broadcast" : cfg.destination_address;
|
||||
gfx_print(layout.line2_pos, layout.line2_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "#%s", dst);
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void _ui_drawFrequency()
|
||||
{
|
||||
unsigned long frequency = platform_getPttStatus() ?
|
||||
frequency = last_state.channel.tx_frequency : last_state.channel.rx_frequency;
|
||||
|
||||
// Print big numbers frequency
|
||||
gfx_print(layout.line3_pos, layout.line3_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "%03lu.%05lu",
|
||||
(unsigned long)frequency/1000000,
|
||||
(unsigned long)frequency%1000000/10);
|
||||
}
|
||||
|
||||
void _ui_drawVFOMiddleInput(ui_state_t* ui_state)
|
||||
{
|
||||
// Add inserted number to string, skipping "Rx: "/"Tx: " and "."
|
||||
uint8_t insert_pos = ui_state->input_position + 3;
|
||||
if(ui_state->input_position > 3) insert_pos += 1;
|
||||
char input_char = ui_state->input_number + '0';
|
||||
|
||||
if(ui_state->input_set == SET_RX)
|
||||
{
|
||||
if(ui_state->input_position == 0)
|
||||
{
|
||||
gfx_print(layout.line2_pos, layout.input_font, TEXT_ALIGN_CENTER,
|
||||
color_white, ">Rx:%03lu.%04lu",
|
||||
(unsigned long)ui_state->new_rx_frequency/1000000,
|
||||
(unsigned long)(ui_state->new_rx_frequency%1000000)/100);
|
||||
}
|
||||
else
|
||||
{
|
||||
// Replace Rx frequency with underscorses
|
||||
if(ui_state->input_position == 1)
|
||||
strcpy(ui_state->new_rx_freq_buf, ">Rx:___.____");
|
||||
ui_state->new_rx_freq_buf[insert_pos] = input_char;
|
||||
gfx_print(layout.line2_pos, layout.input_font, TEXT_ALIGN_CENTER,
|
||||
color_white, ui_state->new_rx_freq_buf);
|
||||
}
|
||||
gfx_print(layout.line3_pos, layout.input_font, TEXT_ALIGN_CENTER,
|
||||
color_white, " Tx:%03lu.%04lu",
|
||||
(unsigned long)last_state.channel.tx_frequency/1000000,
|
||||
(unsigned long)(last_state.channel.tx_frequency%1000000)/100);
|
||||
}
|
||||
else if(ui_state->input_set == SET_TX)
|
||||
{
|
||||
gfx_print(layout.line2_pos, layout.input_font, TEXT_ALIGN_CENTER,
|
||||
color_white, " Rx:%03lu.%04lu",
|
||||
(unsigned long)ui_state->new_rx_frequency/1000000,
|
||||
(unsigned long)(ui_state->new_rx_frequency%1000000)/100);
|
||||
// Replace Rx frequency with underscorses
|
||||
if(ui_state->input_position == 0)
|
||||
{
|
||||
gfx_print(layout.line3_pos, layout.input_font, TEXT_ALIGN_CENTER,
|
||||
color_white, ">Tx:%03lu.%04lu",
|
||||
(unsigned long)ui_state->new_rx_frequency/1000000,
|
||||
(unsigned long)(ui_state->new_rx_frequency%1000000)/100);
|
||||
}
|
||||
else
|
||||
{
|
||||
if(ui_state->input_position == 1)
|
||||
strcpy(ui_state->new_tx_freq_buf, ">Tx:___.____");
|
||||
ui_state->new_tx_freq_buf[insert_pos] = input_char;
|
||||
gfx_print(layout.line3_pos, layout.input_font, TEXT_ALIGN_CENTER,
|
||||
color_white, ui_state->new_tx_freq_buf);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void _ui_drawMainBottom()
|
||||
{
|
||||
// Squelch bar
|
||||
float rssi = last_state.rssi;
|
||||
float squelch = last_state.settings.sqlLevel / 16.0f;
|
||||
uint16_t meter_width = SCREEN_WIDTH - 2 * layout.horizontal_pad;
|
||||
uint16_t meter_height = layout.bottom_h;
|
||||
point_t meter_pos = { layout.horizontal_pad,
|
||||
SCREEN_HEIGHT - meter_height - layout.bottom_pad};
|
||||
uint8_t mic_level = platform_getMicLevel();
|
||||
switch(last_state.channel.mode)
|
||||
{
|
||||
case OPMODE_FM:
|
||||
gfx_drawSmeter(meter_pos,
|
||||
meter_width,
|
||||
meter_height,
|
||||
rssi,
|
||||
squelch,
|
||||
yellow_fab413);
|
||||
break;
|
||||
case OPMODE_DMR:
|
||||
gfx_drawSmeterLevel(meter_pos,
|
||||
meter_width,
|
||||
meter_height,
|
||||
rssi,
|
||||
mic_level);
|
||||
break;
|
||||
case OPMODE_M17:
|
||||
gfx_drawSmeterLevel(meter_pos,
|
||||
meter_width,
|
||||
meter_height,
|
||||
rssi,
|
||||
mic_level);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void _ui_drawMainVFO(ui_state_t* ui_state)
|
||||
{
|
||||
gfx_clearScreen();
|
||||
_ui_drawMainTop();
|
||||
_ui_drawModeInfo(ui_state);
|
||||
_ui_drawFrequency();
|
||||
_ui_drawMainBottom();
|
||||
}
|
||||
|
||||
void _ui_drawMainVFOInput(ui_state_t* ui_state)
|
||||
{
|
||||
gfx_clearScreen();
|
||||
_ui_drawMainTop();
|
||||
_ui_drawVFOMiddleInput(ui_state);
|
||||
_ui_drawMainBottom();
|
||||
}
|
||||
|
||||
void _ui_drawMainMEM(ui_state_t* ui_state)
|
||||
{
|
||||
gfx_clearScreen();
|
||||
_ui_drawMainTop();
|
||||
_ui_drawBankChannel();
|
||||
_ui_drawModeInfo(ui_state);
|
||||
_ui_drawFrequency();
|
||||
_ui_drawMainBottom();
|
||||
}
|
||||
|
|
@ -0,0 +1,531 @@
|
|||
/***************************************************************************
|
||||
* Copyright (C) 2020 - 2022 by Federico Amedeo Izzo IU2NUO, *
|
||||
* Niccolò Izzo IU2KIN *
|
||||
* Frederik Saraci IU2NRO *
|
||||
* Silvano Seva IU2KWO *
|
||||
* *
|
||||
* This program is free software; you can redistribute it and/or modify *
|
||||
* it under the terms of the GNU General Public License as published by *
|
||||
* the Free Software Foundation; either version 3 of the License, or *
|
||||
* (at your option) any later version. *
|
||||
* *
|
||||
* This program is distributed in the hope that it will be useful, *
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
|
||||
* GNU General Public License for more details. *
|
||||
* *
|
||||
* You should have received a copy of the GNU General Public License *
|
||||
* along with this program; if not, see <http://www.gnu.org/licenses/> *
|
||||
***************************************************************************/
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdint.h>
|
||||
#include <string.h>
|
||||
#include <utils.h>
|
||||
#include <ui/ui_mod17.h>
|
||||
#include <interfaces/nvmem.h>
|
||||
#include <interfaces/cps_io.h>
|
||||
#include <interfaces/platform.h>
|
||||
#include <interfaces/delays.h>
|
||||
#include <memory_profiling.h>
|
||||
|
||||
/* UI main screen helper functions, their implementation is in "ui_main.c" */
|
||||
extern void _ui_drawMainBottom();
|
||||
|
||||
const char *display_timer_values[] =
|
||||
{
|
||||
"Off",
|
||||
"5 s",
|
||||
"10 s",
|
||||
"15 s",
|
||||
"20 s",
|
||||
"25 s",
|
||||
"30 s",
|
||||
"1 min",
|
||||
"2 min",
|
||||
"3 min",
|
||||
"4 min",
|
||||
"5 min",
|
||||
"15 min",
|
||||
"30 min",
|
||||
"45 min",
|
||||
"1 hour"
|
||||
};
|
||||
|
||||
void _ui_drawMenuList(uint8_t selected, int (*getCurrentEntry)(char *buf, uint8_t max_len, uint8_t index))
|
||||
{
|
||||
point_t pos = layout.line1_pos;
|
||||
// Number of menu entries that fit in the screen height
|
||||
uint8_t entries_in_screen = (SCREEN_HEIGHT - 1 - pos.y) / layout.menu_h + 1;
|
||||
uint8_t scroll = 0;
|
||||
char entry_buf[MAX_ENTRY_LEN] = "";
|
||||
color_t text_color = color_white;
|
||||
for(int item=0, result=0; (result == 0) && (pos.y < SCREEN_HEIGHT); item++)
|
||||
{
|
||||
// If selection is off the screen, scroll screen
|
||||
if(selected >= entries_in_screen)
|
||||
scroll = selected - entries_in_screen + 1;
|
||||
// Call function pointer to get current menu entry string
|
||||
result = (*getCurrentEntry)(entry_buf, sizeof(entry_buf), item+scroll);
|
||||
if(result != -1)
|
||||
{
|
||||
text_color = color_white;
|
||||
if(item + scroll == selected)
|
||||
{
|
||||
text_color = color_black;
|
||||
// Draw rectangle under selected item, compensating for text height
|
||||
point_t rect_pos = {0, pos.y - layout.menu_h + 3};
|
||||
gfx_drawRect(rect_pos, SCREEN_WIDTH, layout.menu_h, color_white, true);
|
||||
}
|
||||
gfx_print(pos, layout.menu_font, TEXT_ALIGN_LEFT, text_color, entry_buf);
|
||||
pos.y += layout.menu_h;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void _ui_drawMenuListValue(ui_state_t* ui_state, uint8_t selected,
|
||||
int (*getCurrentEntry)(char *buf, uint8_t max_len, uint8_t index),
|
||||
int (*getCurrentValue)(char *buf, uint8_t max_len, uint8_t index))
|
||||
{
|
||||
point_t pos = layout.line1_pos;
|
||||
// Number of menu entries that fit in the screen height
|
||||
uint8_t entries_in_screen = (SCREEN_HEIGHT - 1 - pos.y) / layout.menu_h + 1;
|
||||
uint8_t scroll = 0;
|
||||
char entry_buf[MAX_ENTRY_LEN] = "";
|
||||
char value_buf[MAX_ENTRY_LEN] = "";
|
||||
color_t text_color = color_white;
|
||||
for(int item=0, result=0; (result == 0) && (pos.y < SCREEN_HEIGHT); item++)
|
||||
{
|
||||
// If selection is off the screen, scroll screen
|
||||
if(selected >= entries_in_screen)
|
||||
scroll = selected - entries_in_screen + 1;
|
||||
// Call function pointer to get current menu entry string
|
||||
result = (*getCurrentEntry)(entry_buf, sizeof(entry_buf), item+scroll);
|
||||
// Call function pointer to get current entry value string
|
||||
result = (*getCurrentValue)(value_buf, sizeof(value_buf), item+scroll);
|
||||
if(result != -1)
|
||||
{
|
||||
text_color = color_white;
|
||||
if(item + scroll == selected)
|
||||
{
|
||||
// Draw rectangle under selected item, compensating for text height
|
||||
// If we are in edit mode, draw a hollow rectangle
|
||||
text_color = color_black;
|
||||
bool full_rect = true;
|
||||
if(ui_state->edit_mode)
|
||||
{
|
||||
text_color = color_white;
|
||||
full_rect = false;
|
||||
}
|
||||
point_t rect_pos = {0, pos.y - layout.menu_h + 3};
|
||||
gfx_drawRect(rect_pos, SCREEN_WIDTH, layout.menu_h, color_white, full_rect);
|
||||
}
|
||||
gfx_print(pos, layout.menu_font, TEXT_ALIGN_LEFT, text_color, entry_buf);
|
||||
gfx_print(pos, layout.menu_font, TEXT_ALIGN_RIGHT, text_color, value_buf);
|
||||
pos.y += layout.menu_h;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
int _ui_getMenuTopEntryName(char *buf, uint8_t max_len, uint8_t index)
|
||||
{
|
||||
if(index >= menu_num) return -1;
|
||||
snprintf(buf, max_len, "%s", menu_items[index]);
|
||||
return 0;
|
||||
}
|
||||
|
||||
int _ui_getSettingsEntryName(char *buf, uint8_t max_len, uint8_t index)
|
||||
{
|
||||
if(index >= settings_num) return -1;
|
||||
snprintf(buf, max_len, "%s", settings_items[index]);
|
||||
return 0;
|
||||
}
|
||||
|
||||
int _ui_getDisplayEntryName(char *buf, uint8_t max_len, uint8_t index)
|
||||
{
|
||||
if(index >= display_num) return -1;
|
||||
snprintf(buf, max_len, "%s", display_items[index]);
|
||||
return 0;
|
||||
}
|
||||
|
||||
int _ui_getDisplayValueName(char *buf, uint8_t max_len, uint8_t index)
|
||||
{
|
||||
if(index >= display_num) return -1;
|
||||
uint8_t value = 0;
|
||||
switch(index)
|
||||
{
|
||||
#ifdef SCREEN_CONTRAST
|
||||
case D_CONTRAST:
|
||||
value = last_state.settings.contrast;
|
||||
break;
|
||||
#endif
|
||||
case D_TIMER:
|
||||
snprintf(buf, max_len, "%s",
|
||||
display_timer_values[last_state.settings.display_timer]);
|
||||
return 0;
|
||||
}
|
||||
snprintf(buf, max_len, "%d", value);
|
||||
return 0;
|
||||
}
|
||||
|
||||
#ifdef GPS_PRESENT
|
||||
int _ui_getSettingsGPSEntryName(char *buf, uint8_t max_len, uint8_t index)
|
||||
{
|
||||
if(index >= settings_gps_num) return -1;
|
||||
snprintf(buf, max_len, "%s", settings_gps_items[index]);
|
||||
return 0;
|
||||
}
|
||||
|
||||
int _ui_getSettingsGPSValueName(char *buf, uint8_t max_len, uint8_t index)
|
||||
{
|
||||
if(index >= settings_gps_num) return -1;
|
||||
switch(index)
|
||||
{
|
||||
case G_ENABLED:
|
||||
snprintf(buf, max_len, "%s", (last_state.settings.gps_enabled) ? "ON" : "OFF");
|
||||
break;
|
||||
case G_SET_TIME:
|
||||
snprintf(buf, max_len, "%s", (last_state.gps_set_time) ? "ON" : "OFF");
|
||||
break;
|
||||
case G_TIMEZONE:
|
||||
// Add + prefix to positive numbers
|
||||
if(last_state.settings.utc_timezone > 0)
|
||||
snprintf(buf, max_len, "+%d", last_state.settings.utc_timezone);
|
||||
else
|
||||
snprintf(buf, max_len, "%d", last_state.settings.utc_timezone);
|
||||
break;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
#endif
|
||||
|
||||
int _ui_getInfoEntryName(char *buf, uint8_t max_len, uint8_t index)
|
||||
{
|
||||
if(index >= info_num) return -1;
|
||||
snprintf(buf, max_len, "%s", info_items[index]);
|
||||
return 0;
|
||||
}
|
||||
|
||||
int _ui_getInfoValueName(char *buf, uint8_t max_len, uint8_t index)
|
||||
{
|
||||
const hwInfo_t* hwinfo = platform_getHwInfo();
|
||||
if(index >= info_num) return -1;
|
||||
switch(index)
|
||||
{
|
||||
case 0: // Git Version
|
||||
snprintf(buf, max_len, "%s", GIT_VERSION);
|
||||
break;
|
||||
case 1: // Battery voltage
|
||||
{
|
||||
// Compute integer part and mantissa of voltage value, adding 50mV
|
||||
// to mantissa for rounding to nearest integer
|
||||
uint16_t volt = (last_state.v_bat + 50) / 1000;
|
||||
uint16_t mvolt = ((last_state.v_bat - volt * 1000) + 50) / 100;
|
||||
snprintf(buf, max_len, "%d.%dV", volt, mvolt);
|
||||
}
|
||||
break;
|
||||
case 2: // Battery charge
|
||||
snprintf(buf, max_len, "%d%%", last_state.charge);
|
||||
break;
|
||||
case 3: // RSSI
|
||||
snprintf(buf, max_len, "%.1fdBm", last_state.rssi);
|
||||
break;
|
||||
case 4: // Heap usage
|
||||
snprintf(buf, max_len, "%dB", getHeapSize() - getCurrentFreeHeap());
|
||||
break;
|
||||
case 5: // Band
|
||||
snprintf(buf, max_len, "%s %s", hwinfo->vhf_band ? "VHF" : "", hwinfo->uhf_band ? "UHF" : "");
|
||||
break;
|
||||
case 6: // VHF
|
||||
snprintf(buf, max_len, "%d - %d", hwinfo->vhf_minFreq, hwinfo->vhf_maxFreq);
|
||||
break;
|
||||
case 7: // UHF
|
||||
snprintf(buf, max_len, "%d - %d", hwinfo->uhf_minFreq, hwinfo->uhf_maxFreq);
|
||||
break;
|
||||
case 8: // LCD Type
|
||||
snprintf(buf, max_len, "%d", hwinfo->lcd_type);
|
||||
break;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
void _ui_drawMenuTop(ui_state_t* ui_state)
|
||||
{
|
||||
gfx_clearScreen();
|
||||
// Print "Menu" on top bar
|
||||
gfx_print(layout.top_pos, layout.top_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "Menu");
|
||||
// Print menu entries
|
||||
_ui_drawMenuList(ui_state->menu_selected, _ui_getMenuTopEntryName);
|
||||
}
|
||||
|
||||
#ifdef GPS_PRESENT
|
||||
void _ui_drawMenuGPS()
|
||||
{
|
||||
char *fix_buf, *type_buf;
|
||||
gfx_clearScreen();
|
||||
// Print "GPS" on top bar
|
||||
gfx_print(layout.top_pos, layout.top_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "GPS");
|
||||
point_t fix_pos = {layout.line2_pos.x, SCREEN_HEIGHT * 2 / 5};
|
||||
// Print GPS status, if no fix, hide details
|
||||
if(!last_state.settings.gps_enabled)
|
||||
gfx_print(fix_pos, layout.line3_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "GPS OFF");
|
||||
else if (last_state.gps_data.fix_quality == 0)
|
||||
gfx_print(fix_pos, layout.line3_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "No Fix");
|
||||
else if (last_state.gps_data.fix_quality == 6)
|
||||
gfx_print(fix_pos, layout.line3_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "Fix Lost");
|
||||
else
|
||||
{
|
||||
switch(last_state.gps_data.fix_quality)
|
||||
{
|
||||
case 1:
|
||||
fix_buf = "SPS";
|
||||
break;
|
||||
case 2:
|
||||
fix_buf = "DGPS";
|
||||
break;
|
||||
case 3:
|
||||
fix_buf = "PPS";
|
||||
break;
|
||||
default:
|
||||
fix_buf = "ERROR";
|
||||
break;
|
||||
}
|
||||
|
||||
switch(last_state.gps_data.fix_type)
|
||||
{
|
||||
case 1:
|
||||
type_buf = "";
|
||||
break;
|
||||
case 2:
|
||||
type_buf = "2D";
|
||||
break;
|
||||
case 3:
|
||||
type_buf = "3D";
|
||||
break;
|
||||
default:
|
||||
type_buf = "ERROR";
|
||||
break;
|
||||
}
|
||||
gfx_print(layout.line1_pos, layout.top_font, TEXT_ALIGN_LEFT,
|
||||
color_white, fix_buf);
|
||||
gfx_print(layout.line1_pos, layout.top_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "N ");
|
||||
gfx_print(layout.line1_pos, layout.top_font, TEXT_ALIGN_RIGHT,
|
||||
color_white, "%8.6f", last_state.gps_data.latitude);
|
||||
gfx_print(layout.line2_pos, layout.top_font, TEXT_ALIGN_LEFT,
|
||||
color_white, type_buf);
|
||||
// Convert from signed longitude, to unsigned + direction
|
||||
float longitude = last_state.gps_data.longitude;
|
||||
char *direction = (longitude < 0) ? "W " : "E ";
|
||||
longitude = (longitude < 0) ? -longitude : longitude;
|
||||
gfx_print(layout.line2_pos, layout.top_font, TEXT_ALIGN_CENTER,
|
||||
color_white, direction);
|
||||
gfx_print(layout.line2_pos, layout.top_font, TEXT_ALIGN_RIGHT,
|
||||
color_white, "%8.6f", longitude);
|
||||
gfx_print(layout.bottom_pos, layout.bottom_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "S %4.1fkm/h A %4.1fm",
|
||||
last_state.gps_data.speed,
|
||||
last_state.gps_data.altitude);
|
||||
}
|
||||
// Draw compass
|
||||
point_t compass_pos = {layout.horizontal_pad * 2, SCREEN_HEIGHT / 2};
|
||||
gfx_drawGPScompass(compass_pos,
|
||||
SCREEN_WIDTH / 9 + 2,
|
||||
last_state.gps_data.tmg_true,
|
||||
last_state.gps_data.fix_quality != 0 &&
|
||||
last_state.gps_data.fix_quality != 6);
|
||||
// Draw satellites bar graph
|
||||
point_t bar_pos = {layout.line3_pos.x + SCREEN_WIDTH * 1 / 3, SCREEN_HEIGHT / 2};
|
||||
gfx_drawGPSgraph(bar_pos,
|
||||
(SCREEN_WIDTH * 2 / 3) - layout.horizontal_pad,
|
||||
SCREEN_HEIGHT / 3,
|
||||
last_state.gps_data.satellites,
|
||||
last_state.gps_data.active_sats);
|
||||
}
|
||||
#endif
|
||||
|
||||
void _ui_drawMenuSettings(ui_state_t* ui_state)
|
||||
{
|
||||
gfx_clearScreen();
|
||||
// Print "Settings" on top bar
|
||||
gfx_print(layout.top_pos, layout.top_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "Settings");
|
||||
// Print menu entries
|
||||
_ui_drawMenuList(ui_state->menu_selected, _ui_getSettingsEntryName);
|
||||
}
|
||||
|
||||
void _ui_drawMenuInfo(ui_state_t* ui_state)
|
||||
{
|
||||
gfx_clearScreen();
|
||||
// Print "Info" on top bar
|
||||
gfx_print(layout.top_pos, layout.top_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "Info");
|
||||
// Print menu entries
|
||||
_ui_drawMenuListValue(ui_state, ui_state->menu_selected, _ui_getInfoEntryName,
|
||||
_ui_getInfoValueName);
|
||||
}
|
||||
|
||||
void _ui_drawMenuAbout()
|
||||
{
|
||||
gfx_clearScreen();
|
||||
point_t openrtx_pos = {layout.horizontal_pad, layout.line3_h};
|
||||
if(SCREEN_HEIGHT >= 100)
|
||||
ui_drawSplashScreen(false);
|
||||
else
|
||||
gfx_print(openrtx_pos, layout.line3_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "OpenRTX");
|
||||
uint8_t line_h = layout.menu_h;
|
||||
point_t pos = {SCREEN_WIDTH / 7, SCREEN_HEIGHT - (line_h * (author_num - 1)) - 5};
|
||||
for(int author = 0; author < author_num; author++)
|
||||
{
|
||||
gfx_print(pos, layout.top_font, TEXT_ALIGN_LEFT,
|
||||
color_white, "%s", authors[author]);
|
||||
pos.y += line_h;
|
||||
}
|
||||
}
|
||||
|
||||
void _ui_drawSettingsDisplay(ui_state_t* ui_state)
|
||||
{
|
||||
gfx_clearScreen();
|
||||
// Print "Display" on top bar
|
||||
gfx_print(layout.top_pos, layout.top_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "Display");
|
||||
// Print display settings entries
|
||||
_ui_drawMenuListValue(ui_state, ui_state->menu_selected, _ui_getDisplayEntryName,
|
||||
_ui_getDisplayValueName);
|
||||
}
|
||||
|
||||
#ifdef GPS_PRESENT
|
||||
void _ui_drawSettingsGPS(ui_state_t* ui_state)
|
||||
{
|
||||
gfx_clearScreen();
|
||||
// Print "GPS Settings" on top bar
|
||||
gfx_print(layout.top_pos, layout.top_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "GPS Settings");
|
||||
// Print display settings entries
|
||||
_ui_drawMenuListValue(ui_state, ui_state->menu_selected,
|
||||
_ui_getSettingsGPSEntryName,
|
||||
_ui_getSettingsGPSValueName);
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef RTC_PRESENT
|
||||
void _ui_drawSettingsTimeDate()
|
||||
{
|
||||
gfx_clearScreen();
|
||||
datetime_t local_time = utcToLocalTime(last_state.time,
|
||||
last_state.settings.utc_timezone);
|
||||
// Print "Time&Date" on top bar
|
||||
gfx_print(layout.top_pos, layout.top_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "Time&Date");
|
||||
// Print current time and date
|
||||
gfx_print(layout.line2_pos, layout.input_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "%02d/%02d/%02d",
|
||||
local_time.date, local_time.month, local_time.year);
|
||||
gfx_print(layout.line3_pos, layout.input_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "%02d:%02d:%02d",
|
||||
local_time.hour, local_time.minute, local_time.second);
|
||||
}
|
||||
|
||||
void _ui_drawSettingsTimeDateSet(ui_state_t* ui_state)
|
||||
{
|
||||
(void) last_state;
|
||||
|
||||
gfx_clearScreen();
|
||||
// Print "Time&Date" on top bar
|
||||
gfx_print(layout.top_pos, layout.top_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "Time&Date");
|
||||
if(ui_state->input_position <= 0)
|
||||
{
|
||||
strcpy(ui_state->new_date_buf, "__/__/__");
|
||||
strcpy(ui_state->new_time_buf, "__:__:00");
|
||||
}
|
||||
else
|
||||
{
|
||||
char input_char = ui_state->input_number + '0';
|
||||
// Insert date digit
|
||||
if(ui_state->input_position <= 6)
|
||||
{
|
||||
uint8_t pos = ui_state->input_position -1;
|
||||
// Skip "/"
|
||||
if(ui_state->input_position > 2) pos += 1;
|
||||
if(ui_state->input_position > 4) pos += 1;
|
||||
ui_state->new_date_buf[pos] = input_char;
|
||||
}
|
||||
// Insert time digit
|
||||
else
|
||||
{
|
||||
uint8_t pos = ui_state->input_position -7;
|
||||
// Skip ":"
|
||||
if(ui_state->input_position > 8) pos += 1;
|
||||
ui_state->new_time_buf[pos] = input_char;
|
||||
}
|
||||
}
|
||||
gfx_print(layout.line2_pos, layout.input_font, TEXT_ALIGN_CENTER,
|
||||
color_white, ui_state->new_date_buf);
|
||||
gfx_print(layout.line3_pos, layout.input_font, TEXT_ALIGN_CENTER,
|
||||
color_white, ui_state->new_time_buf);
|
||||
}
|
||||
#endif
|
||||
|
||||
void _ui_drawSettingsM17(ui_state_t* ui_state)
|
||||
{
|
||||
gfx_clearScreen();
|
||||
// Print "M17 Settings" on top bar
|
||||
gfx_print(layout.top_pos, layout.top_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "M17 Settings");
|
||||
gfx_printLine(1, 4, layout.top_h, SCREEN_HEIGHT - layout.bottom_h,
|
||||
layout.horizontal_pad, layout.menu_font,
|
||||
TEXT_ALIGN_LEFT, color_white, "Callsign:");
|
||||
if(ui_state->edit_mode)
|
||||
{
|
||||
uint16_t rect_width = SCREEN_WIDTH - (layout.horizontal_pad * 2);
|
||||
uint16_t rect_height = (SCREEN_HEIGHT - (layout.top_h + layout.bottom_h))/2;
|
||||
point_t rect_origin = {(SCREEN_WIDTH - rect_width) / 2,
|
||||
(SCREEN_HEIGHT - rect_height) / 2};
|
||||
gfx_drawRect(rect_origin, rect_width, rect_height, color_white, false);
|
||||
// Print M17 callsign being typed
|
||||
gfx_printLine(1, 1, layout.top_h, SCREEN_HEIGHT - layout.bottom_h,
|
||||
layout.horizontal_pad, layout.input_font,
|
||||
TEXT_ALIGN_CENTER, color_white, ui_state->new_callsign);
|
||||
}
|
||||
else
|
||||
{
|
||||
// Print M17 current callsign
|
||||
gfx_printLine(1, 1, layout.top_h, SCREEN_HEIGHT - layout.bottom_h,
|
||||
layout.horizontal_pad, layout.input_font,
|
||||
TEXT_ALIGN_CENTER, color_white, last_state.settings.callsign);
|
||||
}
|
||||
}
|
||||
|
||||
void _ui_drawSettingsReset2Defaults(ui_state_t* ui_state)
|
||||
{
|
||||
(void) ui_state;
|
||||
|
||||
static int drawcnt = 0;
|
||||
static long long lastDraw = 0;
|
||||
|
||||
gfx_clearScreen();
|
||||
gfx_print(layout.top_pos, layout.top_font, TEXT_ALIGN_CENTER,
|
||||
color_white, "Reset to Defaults");
|
||||
|
||||
// Make text flash yellow once every 1s
|
||||
color_t textcolor = drawcnt % 2 == 0 ? color_white : yellow_fab413;
|
||||
gfx_printLine(1, 4, layout.top_h, SCREEN_HEIGHT - layout.bottom_h,
|
||||
layout.horizontal_pad, layout.top_font,
|
||||
TEXT_ALIGN_CENTER, textcolor, "To reset:");
|
||||
gfx_printLine(2, 4, layout.top_h, SCREEN_HEIGHT - layout.bottom_h,
|
||||
layout.horizontal_pad, layout.top_font,
|
||||
TEXT_ALIGN_CENTER, textcolor, "Press Enter twice");
|
||||
|
||||
if((getTick() - lastDraw) > 1000)
|
||||
{
|
||||
drawcnt++;
|
||||
lastDraw = getTick();
|
||||
}
|
||||
}
|
||||
Loading…
Reference in New Issue