mirror of
https://github.com/silenty4ng/uv-k5-firmware-chinese-lts
synced 2025-01-26 04:04:24 +00:00
804 lines
22 KiB
C
804 lines
22 KiB
C
/* Copyright 2023 Dual Tachyon
|
|
* https://github.com/DualTachyon
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#include <string.h>
|
|
#include <stdlib.h> // abs()
|
|
#include "driver/uart.h"
|
|
#include "app/dtmf.h"
|
|
#include "app/chFrScanner.h"
|
|
#ifdef ENABLE_AM_FIX_SHOW_DATA
|
|
#include "am_fix.h"
|
|
#endif
|
|
#include "bitmaps.h"
|
|
#include "board.h"
|
|
#include "driver/bk4819.h"
|
|
#include "driver/st7565.h"
|
|
#include "external/printf/printf.h"
|
|
#include "functions.h"
|
|
#include "helper/battery.h"
|
|
#include "misc.h"
|
|
#include "radio.h"
|
|
#include "settings.h"
|
|
#include "ui/helper.h"
|
|
#include "ui/inputbox.h"
|
|
#include "ui/main.h"
|
|
#include "ui/ui.h"
|
|
|
|
center_line_t center_line = CENTER_LINE_NONE;
|
|
|
|
// ***************************************************************************
|
|
|
|
static void DrawSmallAntennaAndBars(uint8_t *p, unsigned int level)
|
|
{
|
|
if(level>6)
|
|
level = 6;
|
|
|
|
memcpy(p, BITMAP_Antenna, ARRAY_SIZE(BITMAP_Antenna));
|
|
|
|
for(uint8_t i = 1; i <= level; i++) {
|
|
char bar = (0xff << (6-i)) & 0x7F;
|
|
memset(p + 2 + i*3, bar, 2);
|
|
}
|
|
}
|
|
#if defined ENABLE_AUDIO_BAR || defined ENABLE_RSSI_BAR
|
|
|
|
static void DrawLevelBar(uint8_t xpos, uint8_t line, uint8_t level)
|
|
{
|
|
const char hollowBar[] = {
|
|
0b01111111,
|
|
0b01000001,
|
|
0b01000001,
|
|
0b01111111
|
|
};
|
|
|
|
uint8_t *p_line = gFrameBuffer[line];
|
|
level = MIN(level, 13);
|
|
|
|
for(uint8_t i = 0; i < level; i++) {
|
|
if(i < 9) {
|
|
for(uint8_t j = 0; j < 4; j++)
|
|
p_line[xpos + i * 5 + j] = (~(0x7F >> (i+1))) & 0x7F;
|
|
}
|
|
else {
|
|
memcpy(p_line + (xpos + i * 5), &hollowBar, ARRAY_SIZE(hollowBar));
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#ifdef ENABLE_AUDIO_BAR
|
|
|
|
unsigned int sqrt16(unsigned int value)
|
|
{ // return square root of 'value'
|
|
unsigned int shift = 16; // number of bits supplied in 'value' .. 2 ~ 32
|
|
unsigned int bit = 1u << --shift;
|
|
unsigned int sqrti = 0;
|
|
while (bit)
|
|
{
|
|
const unsigned int temp = ((sqrti << 1) | bit) << shift--;
|
|
if (value >= temp) {
|
|
value -= temp;
|
|
sqrti |= bit;
|
|
}
|
|
bit >>= 1;
|
|
}
|
|
return sqrti;
|
|
}
|
|
|
|
void UI_DisplayAudioBar(void)
|
|
{
|
|
|
|
if(gLowBattery && !gLowBatteryConfirmed)
|
|
return;
|
|
|
|
const unsigned int line = 3;
|
|
|
|
if (gCurrentFunction != FUNCTION_TRANSMIT ||
|
|
gScreenToDisplay != DISPLAY_MAIN
|
|
#ifdef ENABLE_DTMF_CALLING
|
|
|| gDTMF_CallState != DTMF_CALL_STATE_NONE
|
|
#endif
|
|
)
|
|
{
|
|
return; // screen is in use
|
|
}
|
|
|
|
#if defined(ENABLE_ALARM) || defined(ENABLE_TX1750)
|
|
if (gAlarmState != ALARM_STATE_OFF)
|
|
return;
|
|
#endif
|
|
const unsigned int voice_amp = BK4819_GetVoiceAmplitudeOut(); // 15:0
|
|
|
|
// make non-linear to make more sensitive at low values
|
|
const unsigned int level = MIN(voice_amp * 8, 65535u);
|
|
const unsigned int sqrt_level = MIN(sqrt16(level), 124u);
|
|
uint8_t bars = 13 * sqrt_level / 124;
|
|
|
|
uint8_t *p_line = gFrameBuffer[line];
|
|
memset(p_line, 0, LCD_WIDTH);
|
|
|
|
DrawLevelBar(62, line, bars);
|
|
|
|
if (gCurrentFunction == FUNCTION_TRANSMIT)
|
|
ST7565_BlitFullScreen();
|
|
|
|
}
|
|
#endif
|
|
|
|
|
|
static void DisplayRSSIBar(const int16_t rssi, const bool now)
|
|
{
|
|
#if defined(ENABLE_RSSI_BAR)
|
|
|
|
if (center_line == CENTER_LINE_RSSI) {
|
|
const unsigned int txt_width = 7 * 8; // 8 text chars
|
|
const unsigned int bar_x = 2 + txt_width + 4; // X coord of bar graph
|
|
|
|
const unsigned int line = 3;
|
|
uint8_t *p_line = gFrameBuffer[line];
|
|
char str[16];
|
|
|
|
const char plus[] = {
|
|
0b00011000,
|
|
0b00011000,
|
|
0b01111110,
|
|
0b01111110,
|
|
0b01111110,
|
|
0b00011000,
|
|
0b00011000,
|
|
};
|
|
|
|
if (gEeprom.KEY_LOCK && gKeypadLocked > 0)
|
|
return; // display is in use
|
|
|
|
if (gCurrentFunction == FUNCTION_TRANSMIT ||
|
|
gScreenToDisplay != DISPLAY_MAIN
|
|
#ifdef ENABLE_DTMF_CALLING
|
|
|| gDTMF_CallState != DTMF_CALL_STATE_NONE
|
|
#endif
|
|
)
|
|
return; // display is in use
|
|
|
|
if (now)
|
|
memset(p_line, 0, LCD_WIDTH);
|
|
|
|
const int8_t dBmCorrTable[7] = {
|
|
-15, // band 1
|
|
-25, // band 2
|
|
-20, // band 3
|
|
-4, // band 4
|
|
-7, // band 5
|
|
-6, // band 6
|
|
-1 // band 7
|
|
};
|
|
const int16_t s0_dBm = -130; // S0 .. base level
|
|
// const int16_t rssi_dBm = (rssi / 2) - 160;
|
|
const int16_t rssi_dBm = (rssi / 2) - 160 + dBmCorrTable[gRxVfo->Band];
|
|
const uint8_t s_level = MIN(MAX((rssi_dBm - s0_dBm) / 6, 0), 9); // S0 - S9
|
|
uint8_t overS9dBm = MIN(MAX(rssi_dBm - (s0_dBm + 9*6), 0), 99);
|
|
uint8_t overS9Bars = MIN(overS9dBm/10, 4);
|
|
|
|
if(overS9Bars == 0) {
|
|
sprintf(str, "% 4d S%d", rssi_dBm, s_level);
|
|
}
|
|
else {
|
|
sprintf(str, "% 4d %2d", rssi_dBm, overS9dBm);
|
|
memcpy(p_line + 2 + 7*5, &plus, ARRAY_SIZE(plus));
|
|
}
|
|
|
|
UI_PrintStringSmall(str, 2, 0, line);
|
|
|
|
DrawLevelBar(bar_x, line, s_level + overS9Bars);
|
|
}
|
|
#else
|
|
|
|
uint8_t Level;
|
|
|
|
if (rssi >= gEEPROM_RSSI_CALIB[gRxVfo->Band][3]) {
|
|
Level = 6;
|
|
} else if (rssi >= gEEPROM_RSSI_CALIB[gRxVfo->Band][2]) {
|
|
Level = 4;
|
|
} else if (rssi >= gEEPROM_RSSI_CALIB[gRxVfo->Band][1]) {
|
|
Level = 2;
|
|
} else if (rssi >= gEEPROM_RSSI_CALIB[gRxVfo->Band][0]) {
|
|
Level = 1;
|
|
} else {
|
|
Level = 0;
|
|
}
|
|
|
|
uint8_t *pLine = (gEeprom.RX_VFO == 0)? gFrameBuffer[2] : gFrameBuffer[6];
|
|
if (now)
|
|
memset(pLine, 0, 23);
|
|
DrawSmallAntennaAndBars(pLine, Level);
|
|
#endif
|
|
|
|
if (now)
|
|
ST7565_BlitFullScreen();
|
|
}
|
|
|
|
|
|
void UI_UpdateRSSI(const int16_t rssi, const int vfo)
|
|
{
|
|
(void)vfo; // unused
|
|
|
|
// optional larger RSSI dBm, S-point and bar level
|
|
|
|
if (gCurrentFunction == FUNCTION_RECEIVE ||
|
|
gCurrentFunction == FUNCTION_MONITOR ||
|
|
gCurrentFunction == FUNCTION_INCOMING)
|
|
{
|
|
|
|
DisplayRSSIBar(rssi, true);
|
|
}
|
|
|
|
}
|
|
|
|
// ***************************************************************************
|
|
|
|
void UI_DisplayMain(void)
|
|
{
|
|
const unsigned int line0 = 0; // text screen line
|
|
const unsigned int line1 = 4;
|
|
char String[22];
|
|
unsigned int vfo_num;
|
|
|
|
center_line = CENTER_LINE_NONE;
|
|
|
|
// clear the screen
|
|
memset(gFrameBuffer, 0, sizeof(gFrameBuffer));
|
|
|
|
if(gLowBattery && !gLowBatteryConfirmed) {
|
|
//低电压
|
|
UI_DisplayPopup("\xD5\x9F\x1F");
|
|
ST7565_BlitFullScreen();
|
|
return;
|
|
}
|
|
|
|
if (gEeprom.KEY_LOCK && gKeypadLocked > 0)
|
|
{ // tell user how to unlock the keyboard
|
|
//translate
|
|
|
|
|
|
|
|
#ifdef test
|
|
UI_PrintStringSmall("press # long", 0, LCD_WIDTH, 1);
|
|
|
|
|
|
UI_PrintStringSmall("to unlock", 0, LCD_WIDTH, 3);
|
|
#else
|
|
//长按 # 键解锁
|
|
UI_PrintStringSmall("\x97\x94 # \x96\xC9\x9A", 0, LCD_WIDTH, 1);
|
|
|
|
#endif
|
|
|
|
|
|
ST7565_BlitFullScreen();
|
|
return;
|
|
}
|
|
|
|
unsigned int activeTxVFO = gRxVfoIsActive ? gEeprom.RX_VFO : gEeprom.TX_VFO;
|
|
|
|
for (vfo_num = 0; vfo_num < 2; vfo_num++)
|
|
{
|
|
|
|
|
|
|
|
const unsigned int line = (vfo_num == 0) ? line0 : line1;
|
|
const bool isMainVFO = (vfo_num == gEeprom.TX_VFO);
|
|
uint8_t *p_line0 = gFrameBuffer[line + 0];
|
|
uint8_t *p_line1 = gFrameBuffer[line + 1];
|
|
unsigned int mode = 0;
|
|
|
|
if (activeTxVFO != vfo_num) // this is not active TX VFO
|
|
{
|
|
#ifdef ENABLE_SCAN_RANGES
|
|
if(gScanRangeStart) {
|
|
UI_PrintStringSmall("ScnRng", 5, 0, line);
|
|
sprintf(String, "%3u.%05u", gScanRangeStart / 100000, gScanRangeStart % 100000);
|
|
UI_PrintStringSmall(String, 56, 0, line);
|
|
uint32_t frq = gEeprom.VfoInfo[vfo_num].pRX->Frequency;
|
|
sprintf(String, "%3u.%05u", frq / 100000, frq % 100000);
|
|
UI_PrintStringSmall(String, 56, 0, line + 1);
|
|
continue;
|
|
}
|
|
#endif
|
|
|
|
|
|
if (
|
|
#ifdef ENABLE_DTMF_CALLING
|
|
gDTMF_CallState != DTMF_CALL_STATE_NONE || gDTMF_IsTx ||
|
|
#endif
|
|
gDTMF_InputMode)
|
|
{ // show DTMF stuff
|
|
#ifdef ENABLE_DTMF_CALLING
|
|
char Contact[16];
|
|
|
|
if (!gDTMF_InputMode)
|
|
{
|
|
memset(Contact, 0, sizeof(Contact));
|
|
if (gDTMF_CallState == DTMF_CALL_STATE_CALL_OUT)
|
|
strcpy(String, (gDTMF_State == DTMF_STATE_CALL_OUT_RSP) ? "CALL OUT(RSP)" : "CALL OUT");
|
|
else
|
|
if (gDTMF_CallState == DTMF_CALL_STATE_RECEIVED || gDTMF_CallState == DTMF_CALL_STATE_RECEIVED_STAY)
|
|
sprintf(String, "CALL FRM:%s", (DTMF_FindContact(gDTMF_Caller, Contact)) ? Contact : gDTMF_Caller);
|
|
else
|
|
if (gDTMF_IsTx)
|
|
strcpy(String, (gDTMF_State == DTMF_STATE_TX_SUCC) ? "DTMF TX(SUCC)" : "DTMF TX");
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
sprintf(String, ">%s", gDTMF_InputBox);
|
|
}
|
|
UI_PrintStringSmall(String, 2, 0, 0 + (vfo_num * 3));
|
|
#ifdef ENABLE_DTMF_CALLING
|
|
memset(String, 0, sizeof(String));
|
|
if (!gDTMF_InputMode) {
|
|
|
|
memset(Contact, 0, sizeof(Contact));
|
|
if (gDTMF_CallState == DTMF_CALL_STATE_CALL_OUT)
|
|
sprintf(String, ">%s", (DTMF_FindContact(gDTMF_String, Contact)) ? Contact : gDTMF_String);
|
|
else
|
|
if (gDTMF_CallState == DTMF_CALL_STATE_RECEIVED || gDTMF_CallState == DTMF_CALL_STATE_RECEIVED_STAY)
|
|
sprintf(String, ">%s", (DTMF_FindContact(gDTMF_Callee, Contact)) ? Contact : gDTMF_Callee);
|
|
else
|
|
if (gDTMF_IsTx)
|
|
sprintf(String, ">%s", gDTMF_String);
|
|
}
|
|
UI_PrintStringSmall(String, 2, 0, 2 + (vfo_num * 3));
|
|
#endif
|
|
center_line = CENTER_LINE_IN_USE;
|
|
|
|
continue;
|
|
}
|
|
|
|
// highlight the selected/used VFO with a marker
|
|
if (isMainVFO)
|
|
memmove(p_line0 + 0, BITMAP_VFO_Default, sizeof(BITMAP_VFO_Default));
|
|
}
|
|
else // active TX VFO
|
|
{ // highlight the selected/used VFO with a marker
|
|
if (isMainVFO)
|
|
memmove(p_line0 + 0, BITMAP_VFO_Default, sizeof(BITMAP_VFO_Default));
|
|
else
|
|
memmove(p_line0 + 0, BITMAP_VFO_NotDefault, sizeof(BITMAP_VFO_NotDefault));
|
|
}
|
|
|
|
if (gCurrentFunction == FUNCTION_TRANSMIT)
|
|
{ // transmitting
|
|
|
|
#ifdef ENABLE_ALARM
|
|
if (gAlarmState == ALARM_STATE_ALARM)
|
|
mode = 2;
|
|
else
|
|
#endif
|
|
{
|
|
if (activeTxVFO == vfo_num)
|
|
{ // show the TX symbol
|
|
mode = 1;
|
|
#ifdef ENABLE_SMALL_BOLD
|
|
UI_PrintStringSmallBold("TX", 14, 0, line);
|
|
#else
|
|
UI_PrintStringSmall("TX", 14, 0, line);
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{ // receiving .. show the RX symbol
|
|
mode = 2;
|
|
if ((gCurrentFunction == FUNCTION_RECEIVE ||
|
|
gCurrentFunction == FUNCTION_MONITOR ||
|
|
gCurrentFunction == FUNCTION_INCOMING) &&
|
|
gEeprom.RX_VFO == vfo_num)
|
|
{
|
|
#ifdef ENABLE_SMALL_BOLD
|
|
UI_PrintStringSmallBold("RX", 14, 0, line);
|
|
#else
|
|
UI_PrintStringSmall("RX", 14, 0, line);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
if (IS_MR_CHANNEL(gEeprom.ScreenChannel[vfo_num]))
|
|
{ // channel mode
|
|
const unsigned int x = 2;
|
|
const bool inputting = (gInputBoxIndex == 0 || gEeprom.TX_VFO != vfo_num) ? false : true;
|
|
if (!inputting)
|
|
sprintf(String, "M%u", gEeprom.ScreenChannel[vfo_num] + 1);
|
|
else
|
|
sprintf(String, "M%.3s", INPUTBOX_GetAscii()); // show the input text
|
|
UI_PrintStringSmall(String, x, 0, line + 1);
|
|
}
|
|
else if (IS_FREQ_CHANNEL(gEeprom.ScreenChannel[vfo_num]))
|
|
{ // frequency mode
|
|
// show the frequency band number
|
|
const unsigned int x = 2;
|
|
char * buf = gEeprom.VfoInfo[vfo_num].pRX->Frequency < 100000000 ? "" : "+";
|
|
sprintf(String, "F%u%s", 1 + gEeprom.ScreenChannel[vfo_num] - FREQ_CHANNEL_FIRST, buf);
|
|
UI_PrintStringSmall(String, x, 0, line + 1);
|
|
}
|
|
#ifdef ENABLE_NOAA
|
|
else
|
|
{
|
|
if (gInputBoxIndex == 0 || gEeprom.TX_VFO != vfo_num)
|
|
{ // channel number
|
|
sprintf(String, "N%u", 1 + gEeprom.ScreenChannel[vfo_num] - NOAA_CHANNEL_FIRST);
|
|
}
|
|
else
|
|
{ // user entering channel number
|
|
sprintf(String, "N%u%u", '0' + gInputBox[0], '0' + gInputBox[1]);
|
|
}
|
|
UI_PrintStringSmall(String, 7, 0, line + 1);
|
|
}
|
|
#endif
|
|
|
|
// ************
|
|
|
|
unsigned int state = VfoState[vfo_num];
|
|
|
|
#ifdef ENABLE_ALARM
|
|
if (gCurrentFunction == FUNCTION_TRANSMIT && gAlarmState == ALARM_STATE_ALARM) {
|
|
if (activeTxVFO == vfo_num)
|
|
state = VFO_STATE_ALARM;
|
|
}
|
|
#endif
|
|
|
|
uint32_t frequency = gEeprom.VfoInfo[vfo_num].pRX->Frequency;
|
|
|
|
if (state != VFO_STATE_NORMAL)
|
|
{
|
|
//遇忙,低电压,禁止发射,发送超时,高电压
|
|
//const char *state_list[] = {"", "BUSY", "BAT LOW", "TX DISABLE", "TIMEOUT", "ALARM", "VOLT HIGH"};
|
|
const char *state_list[] = {"", "\x1C\x1D", "\xD5\x9F\x1F", "\x1E\xE2\x05\x06", "\x05\x11\x9C\x9D", "ALARM", "\xD6\x9F\x1F"};
|
|
if (state < ARRAY_SIZE(state_list))
|
|
UI_PrintStringSmall(state_list[state], 31, 0, line);
|
|
}
|
|
else if (gInputBoxIndex > 0 && IS_FREQ_CHANNEL(gEeprom.ScreenChannel[vfo_num]) && gEeprom.TX_VFO == vfo_num)
|
|
{ // user entering a frequency
|
|
|
|
|
|
const char * ascii = INPUTBOX_GetAscii();
|
|
bool isGigaF = frequency>=100000000;
|
|
sprintf(String, "%.*s.%.3s", 3 + isGigaF, ascii, ascii + 3 + isGigaF);
|
|
#ifdef ENABLE_BIG_FREQ
|
|
if(!isGigaF) {
|
|
// show the remaining 2 small frequency digits
|
|
UI_PrintStringSmall(String + 7, 113, 0, line + 1);
|
|
String[7] = 0;
|
|
// show the main large frequency digits
|
|
|
|
UI_DisplayFrequency(String, 32, line, false);
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
// show the frequency in the main font
|
|
UI_PrintStringSmall(String, 32, 0, line);
|
|
}
|
|
continue;
|
|
// if(vfo_num==1) break;
|
|
}
|
|
else
|
|
{
|
|
|
|
if (gCurrentFunction == FUNCTION_TRANSMIT)
|
|
{ // transmitting
|
|
if (activeTxVFO == vfo_num)
|
|
frequency = gEeprom.VfoInfo[vfo_num].pTX->Frequency;
|
|
}
|
|
|
|
if (IS_MR_CHANNEL(gEeprom.ScreenChannel[vfo_num]))
|
|
{ // it's a channel
|
|
|
|
// show the scan list assigment symbols
|
|
|
|
const ChannelAttributes_t att = gMR_ChannelAttributes[gEeprom.ScreenChannel[vfo_num]];
|
|
if (att.scanlist1)
|
|
|
|
memmove(p_line0 + 113, BITMAP_ScanList1, sizeof(BITMAP_ScanList1));
|
|
if (att.scanlist2)
|
|
|
|
memmove(p_line0 + 120, BITMAP_ScanList2, sizeof(BITMAP_ScanList2));
|
|
|
|
// compander symbol
|
|
#ifndef ENABLE_BIG_FREQ
|
|
if (att.compander)
|
|
memmove(p_line0 + 120 + LCD_WIDTH, BITMAP_compand, sizeof(BITMAP_compand));
|
|
#else
|
|
// TODO: // find somewhere else to put the symbol
|
|
#endif
|
|
|
|
switch (gEeprom.CHANNEL_DISPLAY_MODE)
|
|
{
|
|
case MDF_FREQUENCY: // show the channel frequency
|
|
sprintf(String, "%3u.%05u", frequency / 100000, frequency % 100000);
|
|
#ifdef ENABLE_BIG_FREQ
|
|
if(frequency < 100000000) {
|
|
// show the remaining 2 small frequency digits
|
|
UI_PrintStringSmall(String + 7, 113, 0, line + 1);
|
|
String[7] = 0;
|
|
// show the main large frequency digits
|
|
|
|
|
|
UI_DisplayFrequency(String, 32, line, false);
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
// show the frequency in the main font
|
|
UI_PrintStringSmall(String, 32, 0, line);
|
|
}
|
|
|
|
break;
|
|
|
|
case MDF_CHANNEL: // show the channel number
|
|
sprintf(String, "CH-%03u", gEeprom.ScreenChannel[vfo_num] + 1);
|
|
UI_PrintStringSmall(String, 32, 0, line);
|
|
break;
|
|
|
|
case MDF_NAME: // show the channel name
|
|
case MDF_NAME_FREQ: // show the channel name and frequency
|
|
|
|
BOARD_fetchChannelName(String, gEeprom.ScreenChannel[vfo_num]);
|
|
if (String[0] == 0)
|
|
{ // no channel name, show the channel number instead
|
|
sprintf(String, "CH-%03u", gEeprom.ScreenChannel[vfo_num] + 1);
|
|
}
|
|
|
|
if (gEeprom.CHANNEL_DISPLAY_MODE == MDF_NAME) {
|
|
UI_PrintStringSmall(String, 32, 0, line);
|
|
}
|
|
else {
|
|
#ifdef ENABLE_SMALL_BOLD
|
|
UI_PrintStringSmallBold(String, 32 + 4, 0, line);
|
|
#else
|
|
UI_PrintStringSmall(String, 32 + 4, 0, line);
|
|
#endif
|
|
// show the channel frequency below the channel number/name
|
|
sprintf(String, "%03u.%05u", frequency / 100000, frequency % 100000);
|
|
UI_PrintStringSmall(String, 32 + 4, 0, line + 1);
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{ // frequency mode
|
|
|
|
sprintf(String, "%3u.%05u", frequency / 100000, frequency % 100000);
|
|
|
|
#ifdef ENABLE_BIG_FREQ
|
|
if(frequency < 100000000) {
|
|
// show the remaining 2 small frequency digits
|
|
UI_PrintStringSmall(String + 7, 113, 0, line + 1);
|
|
String[7] = 0;
|
|
// show the main large frequency digits
|
|
UI_DisplayFrequency(String, 32, line, false);
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
// show the frequency in the main font
|
|
UI_PrintStringSmall(String, 32, 0, line);
|
|
}
|
|
|
|
// show the channel symbols
|
|
const ChannelAttributes_t att = gMR_ChannelAttributes[gEeprom.ScreenChannel[vfo_num]];
|
|
if (att.compander)
|
|
|
|
#ifdef ENABLE_BIG_FREQ
|
|
memmove(p_line0 + 120, BITMAP_compand, sizeof(BITMAP_compand));
|
|
#else
|
|
memmove(p_line0 + 120 + LCD_WIDTH, BITMAP_compand, sizeof(BITMAP_compand));
|
|
#endif
|
|
}
|
|
}
|
|
|
|
// ************
|
|
|
|
{ // show the TX/RX level
|
|
uint8_t Level = 0;
|
|
|
|
if (mode == 1)
|
|
{ // TX power level
|
|
switch (gRxVfo->OUTPUT_POWER)
|
|
{
|
|
case OUTPUT_POWER_LOW: Level = 2; break;
|
|
case OUTPUT_POWER_MID: Level = 4; break;
|
|
case OUTPUT_POWER_HIGH: Level = 6; break;
|
|
}
|
|
}
|
|
else
|
|
if (mode == 2)
|
|
{ // RX signal level
|
|
#ifndef ENABLE_RSSI_BAR
|
|
// bar graph
|
|
if (gVFO_RSSI_bar_level[vfo_num] > 0)
|
|
Level = gVFO_RSSI_bar_level[vfo_num];
|
|
#endif
|
|
}
|
|
if(Level)
|
|
DrawSmallAntennaAndBars(p_line1 + LCD_WIDTH, Level);
|
|
}
|
|
|
|
// ************
|
|
|
|
String[0] = '\0';
|
|
|
|
// show the modulation symbol
|
|
const char * s = "";
|
|
const ModulationMode_t mod = gEeprom.VfoInfo[vfo_num].Modulation;
|
|
switch (mod){
|
|
case MODULATION_FM: {
|
|
const FREQ_Config_t *pConfig = (mode == 1) ? gEeprom.VfoInfo[vfo_num].pTX : gEeprom.VfoInfo[vfo_num].pRX;
|
|
const unsigned int code_type = pConfig->CodeType;
|
|
const char *code_list[] = {"", "CT", "DCS", "DCR"};
|
|
// UART_Send((uint8_t*)&code_type,1);
|
|
if (code_type < ARRAY_SIZE(code_list))
|
|
s = code_list[code_type];
|
|
break;
|
|
}
|
|
default:
|
|
s = gModulationStr[mod];
|
|
break;
|
|
}
|
|
UI_PrintStringSmall(s, LCD_WIDTH + 24, 0, line + 1);
|
|
|
|
if (state == VFO_STATE_NORMAL || state == VFO_STATE_ALARM)
|
|
{ // show the TX power
|
|
const char pwr_list[] = "LMH";
|
|
const unsigned int i = gEeprom.VfoInfo[vfo_num].OUTPUT_POWER;
|
|
String[0] = (i < ARRAY_SIZE(pwr_list)) ? pwr_list[i] : '\0';
|
|
String[1] = '\0';
|
|
UI_PrintStringSmall(String, LCD_WIDTH + 46, 0, line + 1);
|
|
}
|
|
|
|
if (gEeprom.VfoInfo[vfo_num].freq_config_RX.Frequency != gEeprom.VfoInfo[vfo_num].freq_config_TX.Frequency)
|
|
{ // show the TX offset symbol
|
|
const char dir_list[] = "\0+-";
|
|
const unsigned int i = gEeprom.VfoInfo[vfo_num].TX_OFFSET_FREQUENCY_DIRECTION;
|
|
String[0] = (i < sizeof(dir_list)) ? dir_list[i] : '?';
|
|
String[1] = '\0';
|
|
UI_PrintStringSmall(String, LCD_WIDTH + 54, 0, line + 1);
|
|
}
|
|
|
|
// show the TX/RX reverse symbol
|
|
if (gEeprom.VfoInfo[vfo_num].FrequencyReverse)
|
|
UI_PrintStringSmall("R", LCD_WIDTH + 62, 0, line + 1);
|
|
|
|
{ // show the narrow band symbol
|
|
String[0] = '\0';
|
|
if (gEeprom.VfoInfo[vfo_num].CHANNEL_BANDWIDTH == BANDWIDTH_NARROW)
|
|
{
|
|
String[0] = 'N';
|
|
String[1] = '\0';
|
|
}
|
|
UI_PrintStringSmall(String, LCD_WIDTH + 70, 0, line + 1);
|
|
}
|
|
#ifdef ENABLE_DTMF_CALLING
|
|
// show the DTMF decoding symbol
|
|
if (gEeprom.VfoInfo[vfo_num].DTMF_DECODING_ENABLE || gSetting_KILLED)
|
|
UI_PrintStringSmall("DTMF", LCD_WIDTH + 78, 0, line + 1);
|
|
#endif
|
|
// show the audio scramble symbol
|
|
if (gEeprom.VfoInfo[vfo_num].SCRAMBLING_TYPE > 0/* && gSetting_ScrambleEnable*/)
|
|
|
|
// if (gEeprom.VfoInfo[vfo_num].SCRAMBLING_TYPE > 0 && gSetting_ScrambleEnable)
|
|
UI_PrintStringSmall("SCR", LCD_WIDTH + 106, 0, line + 1);
|
|
|
|
}
|
|
|
|
if (center_line == CENTER_LINE_NONE)
|
|
{ // we're free to use the middle line
|
|
|
|
const bool rx = (gCurrentFunction == FUNCTION_RECEIVE ||
|
|
gCurrentFunction == FUNCTION_MONITOR ||
|
|
gCurrentFunction == FUNCTION_INCOMING);
|
|
|
|
#ifdef ENABLE_AUDIO_BAR
|
|
if (gCurrentFunction == FUNCTION_TRANSMIT) {
|
|
center_line = CENTER_LINE_AUDIO_BAR;
|
|
UI_DisplayAudioBar();
|
|
}
|
|
else
|
|
#endif
|
|
|
|
#if defined(ENABLE_AM_FIX) && defined(ENABLE_AM_FIX_SHOW_DATA)
|
|
if (rx && gEeprom.VfoInfo[gEeprom.RX_VFO].Modulation == MODULATION_AM && gSetting_AM_fix)
|
|
{
|
|
if (gScreenToDisplay != DISPLAY_MAIN
|
|
#ifdef ENABLE_DTMF_CALLING
|
|
|| gDTMF_CallState != DTMF_CALL_STATE_NONE
|
|
#endif
|
|
)
|
|
return;
|
|
|
|
center_line = CENTER_LINE_AM_FIX_DATA;
|
|
AM_fix_print_data(gEeprom.RX_VFO, String);
|
|
UI_PrintStringSmall(String, 2, 0, 3);
|
|
}
|
|
else
|
|
#endif
|
|
|
|
#ifdef ENABLE_RSSI_BAR
|
|
if (rx) {
|
|
center_line = CENTER_LINE_RSSI;
|
|
DisplayRSSIBar(gCurrentRSSI[gEeprom.RX_VFO], false);
|
|
}
|
|
else
|
|
#endif
|
|
if (rx || gCurrentFunction == FUNCTION_FOREGROUND || gCurrentFunction == FUNCTION_POWER_SAVE)
|
|
{
|
|
#if 1
|
|
if (gSetting_live_DTMF_decoder && gDTMF_RX_live[0] != 0)
|
|
{ // show live DTMF decode
|
|
const unsigned int len = strlen(gDTMF_RX_live);
|
|
const unsigned int idx = (len > (17 - 5)) ? len - (17 - 5) : 0; // limit to last 'n' chars
|
|
|
|
if (gScreenToDisplay != DISPLAY_MAIN ||
|
|
gDTMF_CallState != DTMF_CALL_STATE_NONE)
|
|
return;
|
|
|
|
center_line = CENTER_LINE_DTMF_DEC;
|
|
|
|
strcpy(String, "DTMF ");
|
|
strcat(String, gDTMF_RX_live + idx);
|
|
UI_PrintStringSmall(String, 2, 0, 3);
|
|
}
|
|
#else
|
|
if (gSetting_live_DTMF_decoder && gDTMF_RX_index > 0)
|
|
{ // show live DTMF decode
|
|
const unsigned int len = gDTMF_RX_index;
|
|
const unsigned int idx = (len > (17 - 5)) ? len - (17 - 5) : 0; // limit to last 'n' chars
|
|
|
|
if (gScreenToDisplay != DISPLAY_MAIN ||
|
|
gDTMF_CallState != DTMF_CALL_STATE_NONE)
|
|
return;
|
|
|
|
center_line = CENTER_LINE_DTMF_DEC;
|
|
|
|
strcpy(String, "DTMF ");
|
|
strcat(String, gDTMF_RX + idx);
|
|
UI_PrintStringSmall(String, 2, 0, 3);
|
|
}
|
|
#endif
|
|
|
|
#ifdef ENABLE_SHOW_CHARGE_LEVEL
|
|
else if (gChargingWithTypeC)
|
|
{ // charging .. show the battery state
|
|
if (gScreenToDisplay != DISPLAY_MAIN
|
|
#ifdef ENABLE_DTMF_CALLING
|
|
|| gDTMF_CallState != DTMF_CALL_STATE_NONE
|
|
#endif
|
|
)
|
|
return;
|
|
|
|
center_line = CENTER_LINE_CHARGE_DATA;
|
|
|
|
sprintf(String, "Charge %u.%02uV %u%%",
|
|
gBatteryVoltageAverage / 100, gBatteryVoltageAverage % 100,
|
|
BATTERY_VoltsToPercent(gBatteryVoltageAverage));
|
|
UI_PrintStringSmall(String, 2, 0, 3);
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
|
|
ST7565_BlitFullScreen();
|
|
}
|
|
|
|
// ***************************************************************************
|