2023-09-09 08:03:56 +01:00
|
|
|
/* 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>
|
2023-09-27 17:29:49 +01:00
|
|
|
#include <stdlib.h> // abs()
|
2023-09-09 08:03:56 +01:00
|
|
|
|
|
|
|
#include "app/dtmf.h"
|
2023-09-23 12:07:46 +01:00
|
|
|
#ifdef ENABLE_AM_FIX_SHOW_DATA
|
|
|
|
#include "am_fix.h"
|
|
|
|
#endif
|
2023-09-09 08:03:56 +01:00
|
|
|
#include "bitmaps.h"
|
2023-09-20 08:29:11 +01:00
|
|
|
#include "board.h"
|
2023-09-21 07:31:29 +01:00
|
|
|
#include "driver/bk4819.h"
|
2023-09-09 08:03:56 +01:00
|
|
|
#include "driver/st7565.h"
|
|
|
|
#include "external/printf/printf.h"
|
|
|
|
#include "functions.h"
|
2023-09-18 14:31:14 +01:00
|
|
|
#include "helper/battery.h"
|
2023-09-09 08:03:56 +01:00
|
|
|
#include "misc.h"
|
|
|
|
#include "radio.h"
|
|
|
|
#include "settings.h"
|
|
|
|
#include "ui/helper.h"
|
|
|
|
#include "ui/inputbox.h"
|
|
|
|
#include "ui/main.h"
|
2023-09-23 12:25:06 +01:00
|
|
|
#include "ui/ui.h"
|
2023-09-09 08:03:56 +01:00
|
|
|
|
2023-09-30 11:22:19 +01:00
|
|
|
bool center_line_is_free = true;
|
|
|
|
|
2023-09-29 08:43:05 +01:00
|
|
|
// ***************************************************************************
|
|
|
|
|
2023-09-21 07:31:29 +01:00
|
|
|
#ifdef ENABLE_AUDIO_BAR
|
2023-09-27 00:26:58 +01:00
|
|
|
|
|
|
|
unsigned int sqrt16(unsigned int value)
|
|
|
|
{ // return square root of 'value'
|
2023-09-29 08:43:05 +01:00
|
|
|
unsigned int shift = 16; // number of bits supplied in 'value' .. 2 ~ 32
|
|
|
|
unsigned int bit = 1u << --shift;
|
2023-09-27 00:26:58 +01:00
|
|
|
unsigned int sqrti = 0;
|
2023-09-29 08:43:05 +01:00
|
|
|
while (bit)
|
|
|
|
{
|
2023-09-27 00:26:58 +01:00
|
|
|
const unsigned int temp = ((sqrti << 1) | bit) << shift--;
|
|
|
|
if (value >= temp)
|
|
|
|
{
|
|
|
|
value -= temp;
|
|
|
|
sqrti |= bit;
|
|
|
|
}
|
2023-09-29 08:43:05 +01:00
|
|
|
bit >>= 1;
|
|
|
|
}
|
2023-09-27 00:26:58 +01:00
|
|
|
return sqrti;
|
|
|
|
}
|
|
|
|
|
2023-09-21 07:31:29 +01:00
|
|
|
void UI_DisplayAudioBar(void)
|
|
|
|
{
|
|
|
|
if (gSetting_mic_bar)
|
|
|
|
{
|
2023-09-29 09:58:32 +01:00
|
|
|
const unsigned int line = 3;
|
|
|
|
const unsigned int bar_x = 2;
|
|
|
|
const unsigned int bar_width = LCD_WIDTH - 2 - bar_x;
|
|
|
|
unsigned int i;
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-09-21 07:31:29 +01:00
|
|
|
#if 1
|
|
|
|
// TX audio level
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-09-25 13:27:52 +01:00
|
|
|
if (gCurrentFunction != FUNCTION_TRANSMIT)
|
|
|
|
return;
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-09-27 00:26:58 +01:00
|
|
|
const unsigned int voice_amp = BK4819_GetVoiceAmplitudeOut(); // 15:0
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-09-27 00:26:58 +01:00
|
|
|
// const unsigned int max = 65535;
|
2023-09-29 09:58:32 +01:00
|
|
|
// const unsigned int level = ((voice_amp * bar_width) + (max / 2)) / max; // with rounding
|
|
|
|
// const unsigned int len = (level <= bar_width) ? level : bar_width;
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-09-27 00:26:58 +01:00
|
|
|
// make non-linear to make more sensitive at low values
|
|
|
|
const unsigned int level = voice_amp * 8;
|
|
|
|
const unsigned int sqrt_level = sqrt16((level < 65535) ? level : 65535);
|
2023-09-29 09:58:32 +01:00
|
|
|
const unsigned int len = (sqrt_level <= bar_width) ? sqrt_level : bar_width;
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-09-21 07:31:29 +01:00
|
|
|
#else
|
|
|
|
// TX/RX AF input level (dB)
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-09-27 00:26:58 +01:00
|
|
|
const uint8_t af_tx_rx = BK4819_GetAfTxRx(); // 6:0
|
|
|
|
const unsigned int max = 63;
|
2023-09-29 09:58:32 +01:00
|
|
|
const unsigned int level = (((uint16_t)af_tx_rx * bar_width) + (max / 2)) / max; // with rounding
|
|
|
|
const unsigned int len = (level <= bar_width) ? level : bar_width;
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-09-21 07:31:29 +01:00
|
|
|
#endif
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-10-01 10:15:24 +01:00
|
|
|
uint8_t *p_line = gFrameBuffer[line];
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-10-01 10:15:24 +01:00
|
|
|
memset(p_line, 0, LCD_WIDTH);
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-09-29 09:58:32 +01:00
|
|
|
#if 1
|
2023-09-21 07:31:29 +01:00
|
|
|
// solid bar
|
2023-09-29 09:58:32 +01:00
|
|
|
for (i = 0; i < bar_width; i++)
|
2023-10-01 10:15:24 +01:00
|
|
|
p_line[bar_x + i] = (i > len) ? ((i & 1) == 0) ? 0x41 : 0x00 : ((i & 1) == 0) ? 0x7f : 0x3e;
|
2023-09-21 07:31:29 +01:00
|
|
|
#else
|
2023-09-29 09:58:32 +01:00
|
|
|
// knuled bar
|
|
|
|
for (i = 0; i < bar_width; i += 2)
|
2023-10-01 10:15:24 +01:00
|
|
|
p_line[bar_x + i] = (i <= len) ? 0x7f : 0x41;
|
2023-09-21 07:31:29 +01:00
|
|
|
#endif
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-09-25 13:27:52 +01:00
|
|
|
if (gCurrentFunction == FUNCTION_TRANSMIT)
|
|
|
|
ST7565_BlitFullScreen();
|
2023-09-21 07:31:29 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2023-09-29 08:43:05 +01:00
|
|
|
#if defined(ENABLE_RSSI_BAR)
|
|
|
|
void UI_DisplayRSSIBar(const int16_t rssi, const bool now)
|
|
|
|
{
|
2023-10-01 00:04:51 +01:00
|
|
|
// const int16_t s0_dBm = -127; // S0 .. base level
|
|
|
|
const int16_t s0_dBm = -147; // S0 .. base level
|
2023-09-29 08:43:05 +01:00
|
|
|
|
2023-09-30 17:14:48 +01:00
|
|
|
const int16_t s9_dBm = s0_dBm + (6 * 9); // S9 .. 6dB/S-Point
|
2023-09-29 20:48:23 +01:00
|
|
|
const int16_t bar_max_dBm = s9_dBm + 30; // S9+30dB
|
2023-10-01 09:14:19 +01:00
|
|
|
// const int16_t bar_min_dBm = s0_dBm + (6 * 0); // S0
|
2023-10-01 00:10:45 +01:00
|
|
|
const int16_t bar_min_dBm = s0_dBm + (6 * 4); // S4
|
2023-09-29 08:43:05 +01:00
|
|
|
|
2023-09-29 20:48:23 +01:00
|
|
|
// ************
|
2023-09-29 08:43:05 +01:00
|
|
|
|
2023-09-29 20:48:23 +01:00
|
|
|
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 bar_width = LCD_WIDTH - 1 - bar_x;
|
2023-09-29 08:43:05 +01:00
|
|
|
|
2023-09-30 20:03:09 +01:00
|
|
|
const int16_t rssi_dBm = (rssi / 2) - 160;
|
|
|
|
const int16_t clamped_dBm = (rssi_dBm <= bar_min_dBm) ? bar_min_dBm : (rssi_dBm >= bar_max_dBm) ? bar_max_dBm : rssi_dBm;
|
2023-09-29 20:48:23 +01:00
|
|
|
const unsigned int bar_range_dB = bar_max_dBm - bar_min_dBm;
|
|
|
|
const unsigned int len = ((clamped_dBm - bar_min_dBm) * bar_width) / bar_range_dB;
|
2023-09-29 08:43:05 +01:00
|
|
|
|
2023-09-29 20:48:23 +01:00
|
|
|
const unsigned int line = 3;
|
2023-10-01 10:15:24 +01:00
|
|
|
uint8_t *p_line = gFrameBuffer[line];
|
2023-09-29 20:48:23 +01:00
|
|
|
|
|
|
|
char s[16];
|
|
|
|
unsigned int i;
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-09-29 08:43:05 +01:00
|
|
|
if (gEeprom.KEY_LOCK && gKeypadLocked > 0)
|
|
|
|
return; // display is in use
|
|
|
|
if (gCurrentFunction == FUNCTION_TRANSMIT || gScreenToDisplay != DISPLAY_MAIN)
|
|
|
|
return; // display is in use
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-09-29 08:43:05 +01:00
|
|
|
if (now)
|
2023-10-01 10:15:24 +01:00
|
|
|
memset(p_line, 0, LCD_WIDTH);
|
2023-09-29 08:43:05 +01:00
|
|
|
|
2023-09-30 20:03:09 +01:00
|
|
|
if (rssi_dBm >= (s9_dBm + 6))
|
|
|
|
{ // S9+XXdB, 1dB increment
|
|
|
|
const char *fmt[] = {"%-4d +%u ", "%-4d +%u "};
|
|
|
|
const unsigned int dB = rssi_dBm - s9_dBm;
|
|
|
|
sprintf(s, (dB < 10) ? fmt[0] : fmt[1], rssi_dBm, dB);
|
|
|
|
}
|
2023-09-29 08:43:05 +01:00
|
|
|
else
|
2023-09-30 20:03:09 +01:00
|
|
|
{ // S0 ~ S9, 6dB per S-point
|
|
|
|
const unsigned int s_level = (rssi_dBm >= s0_dBm) ? (rssi_dBm - s0_dBm) / 6 : 0;
|
|
|
|
sprintf(s, "%-4d S%u ", rssi_dBm, s_level);
|
|
|
|
}
|
2023-09-29 08:43:05 +01:00
|
|
|
UI_PrintStringSmall(s, 2, 0, line);
|
|
|
|
|
2023-09-29 09:58:32 +01:00
|
|
|
#if 1
|
|
|
|
// solid bar
|
|
|
|
for (i = 0; i < bar_width; i++)
|
2023-10-01 10:15:24 +01:00
|
|
|
p_line[bar_x + i] = (i > len) ? ((i & 1) == 0) ? 0x41 : 0x00 : ((i & 1) == 0) ? 0x7f : 0x3e;
|
2023-09-29 09:58:32 +01:00
|
|
|
#else
|
|
|
|
// knuled bar
|
|
|
|
for (i = 0; i < bar_width; i += 2)
|
2023-10-01 10:15:24 +01:00
|
|
|
p_line[bar_x + i] = (i <= len) ? 0x7f : 0x41;
|
2023-09-29 09:58:32 +01:00
|
|
|
#endif
|
2023-09-29 08:43:05 +01:00
|
|
|
|
|
|
|
if (now)
|
|
|
|
ST7565_BlitFullScreen();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void UI_UpdateRSSI(const int16_t rssi, const int vfo)
|
|
|
|
{
|
|
|
|
#ifdef ENABLE_RSSI_BAR
|
|
|
|
|
2023-09-30 11:22:19 +01:00
|
|
|
if (!center_line_is_free)
|
|
|
|
return;
|
2023-09-30 20:03:09 +01:00
|
|
|
|
2023-09-29 08:43:05 +01:00
|
|
|
const bool rx = (gCurrentFunction == FUNCTION_RECEIVE ||
|
|
|
|
gCurrentFunction == FUNCTION_MONITOR ||
|
|
|
|
gCurrentFunction == FUNCTION_INCOMING);
|
|
|
|
|
2023-09-29 10:25:33 +01:00
|
|
|
#if defined(ENABLE_AM_FIX) && defined(ENABLE_AM_FIX_SHOW_DATA)
|
|
|
|
if (gEeprom.VfoInfo[gEeprom.RX_CHANNEL].AM_mode && gSetting_AM_fix)
|
2023-09-29 20:48:23 +01:00
|
|
|
{ // AM test data is currently being shown
|
2023-09-29 10:25:33 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
if (rx)
|
|
|
|
UI_DisplayRSSIBar(rssi, true);
|
2023-09-29 08:43:05 +01:00
|
|
|
|
|
|
|
#else
|
|
|
|
|
2023-09-29 15:22:41 +01:00
|
|
|
// const int16_t dBm = (rssi / 2) - 160;
|
2023-09-29 08:43:05 +01:00
|
|
|
const uint8_t Line = (vfo == 0) ? 3 : 7;
|
2023-10-01 10:15:24 +01:00
|
|
|
uint8_t *p_line = gFrameBuffer[Line - 1];
|
2023-09-29 15:22:41 +01:00
|
|
|
|
|
|
|
// TODO: sort out all 8 values from the eeprom
|
2023-09-29 08:43:05 +01:00
|
|
|
|
|
|
|
#if 0
|
2023-09-29 15:22:41 +01:00
|
|
|
// dBm -105 -100 -95 -90 -70 -65 -60 -55
|
|
|
|
// RSSI 110 120 130 140 180 190 200 210
|
|
|
|
// 0000C0 6E 00 78 00 82 00 8C 00 B4 00 BE 00 C8 00 D2 00
|
|
|
|
//
|
|
|
|
const unsigned int band = 1;
|
2023-09-29 08:43:05 +01:00
|
|
|
const int16_t level0 = gEEPROM_RSSI_CALIB[band][0];
|
|
|
|
const int16_t level1 = gEEPROM_RSSI_CALIB[band][1];
|
|
|
|
const int16_t level2 = gEEPROM_RSSI_CALIB[band][2];
|
|
|
|
const int16_t level3 = gEEPROM_RSSI_CALIB[band][3];
|
|
|
|
#else
|
|
|
|
const int16_t level0 = (-115 + 160) * 2; // dB
|
|
|
|
const int16_t level1 = ( -89 + 160) * 2; // dB
|
|
|
|
const int16_t level2 = ( -64 + 160) * 2; // dB
|
|
|
|
const int16_t level3 = ( -39 + 160) * 2; // dB
|
|
|
|
#endif
|
|
|
|
const int16_t level01 = (level0 + level1) / 2;
|
|
|
|
const int16_t level12 = (level1 + level2) / 2;
|
|
|
|
const int16_t level23 = (level2 + level3) / 2;
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-09-29 08:43:05 +01:00
|
|
|
gVFO_RSSI[vfo] = rssi;
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-09-29 08:43:05 +01:00
|
|
|
uint8_t rssi_level = 0;
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-09-29 08:43:05 +01:00
|
|
|
if (rssi >= level3) rssi_level = 7;
|
|
|
|
else
|
|
|
|
if (rssi >= level23) rssi_level = 6;
|
|
|
|
else
|
|
|
|
if (rssi >= level2) rssi_level = 5;
|
|
|
|
else
|
|
|
|
if (rssi >= level12) rssi_level = 4;
|
|
|
|
else
|
|
|
|
if (rssi >= level1) rssi_level = 3;
|
|
|
|
else
|
|
|
|
if (rssi >= level01) rssi_level = 2;
|
|
|
|
else
|
|
|
|
if (rssi >= level0) rssi_level = 1;
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-09-29 08:43:05 +01:00
|
|
|
if (gVFO_RSSI_bar_level[vfo] == rssi_level)
|
|
|
|
return;
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-09-29 08:43:05 +01:00
|
|
|
gVFO_RSSI_bar_level[vfo] = rssi_level;
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-09-29 08:43:05 +01:00
|
|
|
// **********************************************************
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-09-29 08:43:05 +01:00
|
|
|
if (gEeprom.KEY_LOCK && gKeypadLocked > 0)
|
|
|
|
return; // display is in use
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-09-29 08:43:05 +01:00
|
|
|
if (gCurrentFunction == FUNCTION_TRANSMIT || gScreenToDisplay != DISPLAY_MAIN)
|
|
|
|
return; // display is in use
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-10-01 10:15:24 +01:00
|
|
|
p_line = gFrameBuffer[Line - 1];
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-10-01 10:15:24 +01:00
|
|
|
memset(p_line, 0, 23);
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-09-29 08:43:05 +01:00
|
|
|
if (rssi_level > 0)
|
|
|
|
{
|
|
|
|
//if (rssi_level >= 1)
|
2023-10-01 10:15:24 +01:00
|
|
|
memmove(p_line, BITMAP_Antenna, 5);
|
2023-09-29 08:43:05 +01:00
|
|
|
if (rssi_level >= 2)
|
2023-10-01 10:15:24 +01:00
|
|
|
memmove(p_line + 5, BITMAP_AntennaLevel1, sizeof(BITMAP_AntennaLevel1));
|
2023-09-29 08:43:05 +01:00
|
|
|
if (rssi_level >= 3)
|
2023-10-01 10:15:24 +01:00
|
|
|
memmove(p_line + 8, BITMAP_AntennaLevel2, sizeof(BITMAP_AntennaLevel2));
|
2023-09-29 08:43:05 +01:00
|
|
|
if (rssi_level >= 4)
|
2023-10-01 10:15:24 +01:00
|
|
|
memmove(p_line + 11, BITMAP_AntennaLevel3, sizeof(BITMAP_AntennaLevel3));
|
2023-09-29 08:43:05 +01:00
|
|
|
if (rssi_level >= 5)
|
2023-10-01 10:15:24 +01:00
|
|
|
memmove(p_line + 14, BITMAP_AntennaLevel4, sizeof(BITMAP_AntennaLevel4));
|
2023-09-29 08:43:05 +01:00
|
|
|
if (rssi_level >= 6)
|
2023-10-01 10:15:24 +01:00
|
|
|
memmove(p_line + 17, BITMAP_AntennaLevel5, sizeof(BITMAP_AntennaLevel5));
|
2023-09-29 08:43:05 +01:00
|
|
|
if (rssi_level >= 7)
|
2023-10-01 10:15:24 +01:00
|
|
|
memmove(p_line + 20, BITMAP_AntennaLevel6, sizeof(BITMAP_AntennaLevel6));
|
2023-09-29 08:43:05 +01:00
|
|
|
}
|
|
|
|
else
|
2023-10-01 10:15:24 +01:00
|
|
|
p_line = NULL;
|
2023-09-29 10:25:33 +01:00
|
|
|
|
2023-10-01 10:15:24 +01:00
|
|
|
ST7565_DrawLine(0, Line, 23, p_line);
|
2023-09-29 08:43:05 +01:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
// ***************************************************************************
|
|
|
|
|
2023-09-09 08:03:56 +01:00
|
|
|
void UI_DisplayMain(void)
|
|
|
|
{
|
2023-10-01 10:15:24 +01:00
|
|
|
const unsigned int line0 = 0; // text screen line
|
|
|
|
const unsigned int line1 = 4;
|
|
|
|
char String[16];
|
|
|
|
unsigned int vfo_num;
|
2023-09-28 22:22:06 +01:00
|
|
|
|
2023-10-01 10:15:24 +01:00
|
|
|
// true is the center screen line is not in use
|
2023-09-30 11:22:19 +01:00
|
|
|
center_line_is_free = true;
|
2023-09-30 20:03:09 +01:00
|
|
|
|
2023-09-28 22:22:06 +01:00
|
|
|
// #ifdef SINGLE_VFO_CHAN
|
|
|
|
// const bool single_vfo = (gEeprom.DUAL_WATCH == DUAL_WATCH_OFF && gEeprom.CROSS_BAND_RX_TX == CROSS_BAND_OFF) ? true : false;
|
|
|
|
// #else
|
|
|
|
const bool single_vfo = false;
|
|
|
|
// #endif
|
2023-09-09 08:03:56 +01:00
|
|
|
|
2023-10-01 10:15:24 +01:00
|
|
|
// clear the screen
|
2023-09-09 08:03:56 +01:00
|
|
|
memset(gFrameBuffer, 0, sizeof(gFrameBuffer));
|
|
|
|
|
2023-09-13 18:45:52 +01:00
|
|
|
if (gEeprom.KEY_LOCK && gKeypadLocked > 0)
|
2023-09-19 07:16:15 +01:00
|
|
|
{ // tell user how to unlock the keyboard
|
2023-09-19 15:35:32 +01:00
|
|
|
UI_PrintString("Long press #", 0, LCD_WIDTH, 1, 8);
|
|
|
|
UI_PrintString("to unlock", 0, LCD_WIDTH, 3, 8);
|
2023-09-09 08:03:56 +01:00
|
|
|
ST7565_BlitFullScreen();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-09-10 05:49:39 +01:00
|
|
|
for (vfo_num = 0; vfo_num < 2; vfo_num++)
|
2023-09-09 08:03:56 +01:00
|
|
|
{
|
2023-10-01 19:17:51 +01:00
|
|
|
const unsigned int line = (vfo_num == 0) ? line0 : line1;
|
|
|
|
uint8_t channel = gEeprom.TX_CHANNEL;
|
|
|
|
// uint8_t tx_channel = (gEeprom.CROSS_BAND_RX_TX == CROSS_BAND_OFF) ? gEeprom.RX_CHANNEL : gEeprom.TX_CHANNEL;
|
|
|
|
const bool same_vfo = (channel == vfo_num) ? true : false;
|
|
|
|
uint8_t *p_line0 = gFrameBuffer[line + 0];
|
|
|
|
uint8_t *p_line1 = gFrameBuffer[line + 1];
|
|
|
|
uint32_t duff_beer = 0;
|
|
|
|
uint8_t state;
|
2023-09-16 22:29:02 +01:00
|
|
|
|
2023-09-10 13:52:41 +01:00
|
|
|
if (single_vfo)
|
2023-09-12 15:31:37 +01:00
|
|
|
{ // we're in single VFO mode - screen is dedicated to just one VFO
|
2023-09-11 00:02:57 +01:00
|
|
|
|
2023-10-01 10:15:24 +01:00
|
|
|
if (!same_vfo)
|
2023-09-12 15:31:37 +01:00
|
|
|
continue; // skip the unused vfo
|
2023-09-09 08:03:56 +01:00
|
|
|
|
2023-09-16 17:16:36 +01:00
|
|
|
|
2023-09-10 13:52:41 +01:00
|
|
|
}
|
2023-09-16 22:29:02 +01:00
|
|
|
|
2023-09-09 08:03:56 +01:00
|
|
|
if (gEeprom.DUAL_WATCH != DUAL_WATCH_OFF && gRxVfoIsActive)
|
2023-10-01 10:15:24 +01:00
|
|
|
channel = gEeprom.RX_CHANNEL; // we're currently monitoring the other VFO
|
2023-09-09 08:03:56 +01:00
|
|
|
|
2023-10-01 10:15:24 +01:00
|
|
|
if (channel != vfo_num)
|
2023-09-09 08:03:56 +01:00
|
|
|
{
|
|
|
|
if (gDTMF_CallState != DTMF_CALL_STATE_NONE || gDTMF_IsTx || gDTMF_InputMode)
|
2023-09-11 00:02:57 +01:00
|
|
|
{ // show DTMF stuff
|
2023-09-16 22:29:02 +01:00
|
|
|
|
2023-09-09 08:03:56 +01:00
|
|
|
char Contact[16];
|
|
|
|
|
|
|
|
if (!gDTMF_InputMode)
|
|
|
|
{
|
2023-09-10 13:52:41 +01:00
|
|
|
memset(Contact, 0, sizeof(Contact));
|
2023-09-09 08:03:56 +01:00
|
|
|
if (gDTMF_CallState == DTMF_CALL_STATE_CALL_OUT)
|
2023-09-10 13:52:41 +01:00
|
|
|
strcpy(String, (gDTMF_State == DTMF_STATE_CALL_OUT_RSP) ? "CALL OUT(RSP)" : "CALL OUT");
|
2023-09-09 08:03:56 +01:00
|
|
|
else
|
|
|
|
if (gDTMF_CallState == DTMF_CALL_STATE_RECEIVED)
|
2023-09-30 11:22:19 +01:00
|
|
|
sprintf(String, "CALL FRM:%s", (DTMF_FindContact(gDTMF_Caller, Contact)) ? Contact : gDTMF_Caller);
|
2023-09-09 08:03:56 +01:00
|
|
|
else
|
|
|
|
if (gDTMF_IsTx)
|
2023-09-10 13:52:41 +01:00
|
|
|
strcpy(String, (gDTMF_State == DTMF_STATE_TX_SUCC) ? "DTMF TX(SUCC)" : "DTMF TX");
|
2023-09-09 08:03:56 +01:00
|
|
|
}
|
|
|
|
else
|
2023-09-17 10:13:55 +01:00
|
|
|
{
|
2023-09-09 08:03:56 +01:00
|
|
|
sprintf(String, ">%s", gDTMF_InputBox);
|
2023-09-17 10:13:55 +01:00
|
|
|
|
2023-09-19 10:41:31 +01:00
|
|
|
center_line_is_free = false;
|
2023-09-17 10:13:55 +01:00
|
|
|
}
|
2023-09-12 15:31:37 +01:00
|
|
|
UI_PrintString(String, 2, 0, vfo_num * 3, 8);
|
2023-09-09 08:03:56 +01:00
|
|
|
|
|
|
|
memset(String, 0, sizeof(String));
|
|
|
|
if (!gDTMF_InputMode)
|
|
|
|
{
|
2023-09-10 13:52:41 +01:00
|
|
|
memset(Contact, 0, sizeof(Contact));
|
2023-09-09 08:03:56 +01:00
|
|
|
if (gDTMF_CallState == DTMF_CALL_STATE_CALL_OUT)
|
2023-09-10 13:52:41 +01:00
|
|
|
sprintf(String, ">%s", (DTMF_FindContact(gDTMF_String, Contact)) ? Contact : gDTMF_String);
|
2023-09-09 08:03:56 +01:00
|
|
|
else
|
|
|
|
if (gDTMF_CallState == DTMF_CALL_STATE_RECEIVED)
|
2023-09-10 13:52:41 +01:00
|
|
|
sprintf(String, ">%s", (DTMF_FindContact(gDTMF_Callee, Contact)) ? Contact : gDTMF_Callee);
|
2023-09-09 08:03:56 +01:00
|
|
|
else
|
|
|
|
if (gDTMF_IsTx)
|
|
|
|
sprintf(String, ">%s", gDTMF_String);
|
|
|
|
}
|
2023-09-17 10:13:55 +01:00
|
|
|
else
|
|
|
|
{
|
2023-09-19 10:41:31 +01:00
|
|
|
center_line_is_free = false;
|
2023-09-17 10:13:55 +01:00
|
|
|
}
|
2023-09-12 15:31:37 +01:00
|
|
|
UI_PrintString(String, 2, 0, 2 + (vfo_num * 3), 8);
|
2023-09-10 13:52:41 +01:00
|
|
|
|
2023-09-19 10:41:31 +01:00
|
|
|
center_line_is_free = false;
|
2023-09-09 08:03:56 +01:00
|
|
|
continue;
|
|
|
|
}
|
2023-09-09 13:58:21 +01:00
|
|
|
|
2023-09-10 13:52:41 +01:00
|
|
|
// highlight the selected/used VFO with a marker
|
2023-10-01 10:15:24 +01:00
|
|
|
if (!single_vfo && same_vfo)
|
|
|
|
memmove(p_line0 + 0, BITMAP_VFO_Default, sizeof(BITMAP_VFO_Default));
|
2023-09-12 16:53:47 +01:00
|
|
|
else
|
|
|
|
if (gEeprom.CROSS_BAND_RX_TX != CROSS_BAND_OFF)
|
2023-10-01 10:15:24 +01:00
|
|
|
memmove(p_line0 + 0, BITMAP_VFO_NotDefault, sizeof(BITMAP_VFO_NotDefault));
|
2023-09-09 08:03:56 +01:00
|
|
|
}
|
|
|
|
else
|
2023-09-10 13:52:41 +01:00
|
|
|
if (!single_vfo)
|
|
|
|
{ // highlight the selected/used VFO with a marker
|
2023-10-01 10:15:24 +01:00
|
|
|
if (same_vfo)
|
|
|
|
memmove(p_line0 + 0, BITMAP_VFO_Default, sizeof(BITMAP_VFO_Default));
|
2023-09-12 15:31:37 +01:00
|
|
|
else
|
2023-09-12 19:53:54 +01:00
|
|
|
//if (gEeprom.CROSS_BAND_RX_TX != CROSS_BAND_OFF)
|
2023-10-01 10:15:24 +01:00
|
|
|
memmove(p_line0 + 0, BITMAP_VFO_NotDefault, sizeof(BITMAP_VFO_NotDefault));
|
2023-09-09 08:03:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (gCurrentFunction == FUNCTION_TRANSMIT)
|
2023-09-11 00:02:57 +01:00
|
|
|
{ // transmitting
|
2023-09-16 22:29:02 +01:00
|
|
|
|
2023-09-14 09:56:30 +01:00
|
|
|
#ifdef ENABLE_ALARM
|
2023-09-09 09:01:52 +01:00
|
|
|
if (gAlarmState == ALARM_STATE_ALARM)
|
2023-09-19 14:48:09 +01:00
|
|
|
duff_beer = 2;
|
2023-09-09 09:01:52 +01:00
|
|
|
else
|
|
|
|
#endif
|
2023-09-09 08:03:56 +01:00
|
|
|
{
|
2023-10-01 10:15:24 +01:00
|
|
|
channel = (gEeprom.CROSS_BAND_RX_TX == CROSS_BAND_OFF) ? gEeprom.RX_CHANNEL : gEeprom.TX_CHANNEL;
|
|
|
|
if (channel == vfo_num)
|
2023-09-10 13:52:41 +01:00
|
|
|
{ // show the TX symbol
|
2023-09-19 14:48:09 +01:00
|
|
|
duff_beer = 1;
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_PrintStringSmall("TX", 14, 0, line);
|
2023-09-09 08:03:56 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2023-09-12 15:31:37 +01:00
|
|
|
{ // receiving .. show the RX symbol
|
2023-09-19 14:48:09 +01:00
|
|
|
duff_beer = 2;
|
2023-09-10 05:49:39 +01:00
|
|
|
if ((gCurrentFunction == FUNCTION_RECEIVE || gCurrentFunction == FUNCTION_MONITOR) && gEeprom.RX_CHANNEL == vfo_num)
|
2023-10-01 19:17:51 +01:00
|
|
|
UI_PrintStringSmall("RX", 14, 0, line);
|
2023-09-09 08:03:56 +01:00
|
|
|
}
|
|
|
|
|
2023-09-10 05:49:39 +01:00
|
|
|
if (IS_MR_CHANNEL(gEeprom.ScreenChannel[vfo_num]))
|
2023-09-11 00:02:57 +01:00
|
|
|
{ // channel mode
|
2023-09-12 15:31:37 +01:00
|
|
|
const unsigned int x = 2;
|
2023-09-16 22:29:02 +01:00
|
|
|
const bool inputting = (gInputBoxIndex == 0 || gEeprom.TX_CHANNEL != vfo_num) ? false : true;
|
|
|
|
if (!inputting)
|
2023-09-14 19:38:28 +01:00
|
|
|
NUMBER_ToDigits(gEeprom.ScreenChannel[vfo_num] + 1, String); // show the memory channel number
|
2023-09-09 08:03:56 +01:00
|
|
|
else
|
2023-10-01 12:44:22 +01:00
|
|
|
memmove(String + 5, gInputBox, 3); // show the input text
|
|
|
|
UI_PrintStringSmall("M", x, 0, line + 1);
|
|
|
|
UI_DisplaySmallDigits(3, String + 5, x + 7, line + 1, inputting);
|
2023-09-09 08:03:56 +01:00
|
|
|
}
|
|
|
|
else
|
2023-09-10 05:49:39 +01:00
|
|
|
if (IS_FREQ_CHANNEL(gEeprom.ScreenChannel[vfo_num]))
|
2023-09-16 22:29:02 +01:00
|
|
|
{ // frequency mode
|
2023-09-11 00:02:57 +01:00
|
|
|
// show the frequency band number
|
2023-09-16 22:29:02 +01:00
|
|
|
const unsigned int x = 2; // was 14
|
2023-09-20 16:55:50 +01:00
|
|
|
// sprintf(String, "FB%u", 1 + gEeprom.ScreenChannel[vfo_num] - FREQ_CHANNEL_FIRST);
|
|
|
|
sprintf(String, "VFO%u", 1 + gEeprom.ScreenChannel[vfo_num] - FREQ_CHANNEL_FIRST);
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_PrintStringSmall(String, x, 0, line + 1);
|
2023-09-09 08:03:56 +01:00
|
|
|
}
|
2023-09-16 22:29:02 +01:00
|
|
|
#ifdef ENABLE_NOAA
|
2023-09-09 08:03:56 +01:00
|
|
|
else
|
|
|
|
{
|
2023-09-16 22:29:02 +01:00
|
|
|
if (gInputBoxIndex == 0 || gEeprom.TX_CHANNEL != 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]);
|
|
|
|
}
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_PrintStringSmall(String, 7, 0, line + 1);
|
2023-09-09 08:03:56 +01:00
|
|
|
}
|
2023-09-16 22:29:02 +01:00
|
|
|
#endif
|
2023-09-09 08:03:56 +01:00
|
|
|
|
2023-09-12 15:31:37 +01:00
|
|
|
// ************
|
2023-09-16 22:29:02 +01:00
|
|
|
|
2023-10-01 19:17:51 +01:00
|
|
|
state = VfoState[vfo_num];
|
2023-09-09 08:03:56 +01:00
|
|
|
|
2023-09-14 09:56:30 +01:00
|
|
|
#ifdef ENABLE_ALARM
|
2023-09-09 09:01:52 +01:00
|
|
|
if (gCurrentFunction == FUNCTION_TRANSMIT && gAlarmState == ALARM_STATE_ALARM)
|
|
|
|
{
|
2023-10-01 10:15:24 +01:00
|
|
|
channel = (gEeprom.CROSS_BAND_RX_TX == CROSS_BAND_OFF) ? gEeprom.RX_CHANNEL : gEeprom.TX_CHANNEL;
|
|
|
|
if (channel == vfo_num)
|
|
|
|
state = VFO_STATE_ALARM;
|
2023-09-09 09:01:52 +01:00
|
|
|
}
|
|
|
|
#endif
|
2023-09-09 13:58:21 +01:00
|
|
|
|
2023-10-01 10:15:24 +01:00
|
|
|
if (state != VFO_STATE_NORMAL)
|
2023-09-09 08:03:56 +01:00
|
|
|
{
|
2023-09-16 22:29:02 +01:00
|
|
|
const char *state_list[] = {"", "BUSY", "BAT LOW", "TX DISABLE", "TIMEOUT", "ALARM", "VOLT HIGH"};
|
2023-10-01 10:15:24 +01:00
|
|
|
if (state >= 0 && state < ARRAY_SIZE(state_list))
|
2023-10-01 17:19:38 +01:00
|
|
|
UI_PrintString(state_list[state], 31, 0, line, 8);
|
2023-09-09 08:03:56 +01:00
|
|
|
}
|
|
|
|
else
|
2023-09-16 22:29:02 +01:00
|
|
|
if (gInputBoxIndex > 0 && IS_FREQ_CHANNEL(gEeprom.ScreenChannel[vfo_num]) && gEeprom.TX_CHANNEL == vfo_num)
|
|
|
|
{ // user entering a frequency
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_DisplayFrequency(gInputBox, 32, line, true, false);
|
2023-09-17 10:13:55 +01:00
|
|
|
|
2023-09-19 10:41:31 +01:00
|
|
|
center_line_is_free = false;
|
2023-09-16 22:29:02 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-09-18 18:26:10 +01:00
|
|
|
uint32_t frequency = gEeprom.VfoInfo[vfo_num].pRX->Frequency;
|
2023-09-16 22:29:02 +01:00
|
|
|
if (gCurrentFunction == FUNCTION_TRANSMIT)
|
|
|
|
{ // transmitting
|
2023-10-01 10:15:24 +01:00
|
|
|
channel = (gEeprom.CROSS_BAND_RX_TX == CROSS_BAND_OFF) ? gEeprom.RX_CHANNEL : gEeprom.TX_CHANNEL;
|
2023-10-01 12:44:22 +01:00
|
|
|
if (channel == vfo_num)
|
2023-09-18 18:26:10 +01:00
|
|
|
frequency = gEeprom.VfoInfo[vfo_num].pTX->Frequency;
|
2023-09-09 08:03:56 +01:00
|
|
|
}
|
2023-09-16 22:29:02 +01:00
|
|
|
|
|
|
|
if (IS_MR_CHANNEL(gEeprom.ScreenChannel[vfo_num]))
|
|
|
|
{ // channel mode
|
|
|
|
|
2023-09-19 17:30:12 +01:00
|
|
|
// show the channel symbols
|
|
|
|
const uint8_t attributes = gMR_ChannelAttributes[gEeprom.ScreenChannel[vfo_num]];
|
|
|
|
if (attributes & MR_CH_SCANLIST1)
|
2023-10-01 10:15:24 +01:00
|
|
|
memmove(p_line0 + 113, BITMAP_ScanList1, sizeof(BITMAP_ScanList1));
|
2023-09-19 17:30:12 +01:00
|
|
|
if (attributes & MR_CH_SCANLIST2)
|
2023-10-01 10:15:24 +01:00
|
|
|
memmove(p_line0 + 120, BITMAP_ScanList2, sizeof(BITMAP_ScanList2));
|
2023-09-19 17:30:12 +01:00
|
|
|
#ifndef ENABLE_BIG_FREQ
|
|
|
|
#ifdef ENABLE_COMPANDER
|
|
|
|
if ((attributes & MR_CH_COMPAND) > 0)
|
2023-10-01 10:15:24 +01:00
|
|
|
memmove(p_line0 + 120 + LCD_WIDTH, BITMAP_compand, sizeof(BITMAP_compand));
|
2023-09-19 17:30:12 +01:00
|
|
|
#endif
|
|
|
|
#endif
|
2023-09-21 07:31:29 +01:00
|
|
|
|
2023-09-16 22:29:02 +01:00
|
|
|
switch (gEeprom.CHANNEL_DISPLAY_MODE)
|
|
|
|
{
|
|
|
|
case MDF_FREQUENCY: // show the channel frequency
|
|
|
|
#ifdef ENABLE_BIG_FREQ
|
2023-09-18 18:26:10 +01:00
|
|
|
NUMBER_ToDigits(frequency, String);
|
2023-09-16 22:29:02 +01:00
|
|
|
// show the main large frequency digits
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_DisplayFrequency(String, 32, line, false, false);
|
2023-09-16 22:29:02 +01:00
|
|
|
// show the remaining 2 small frequency digits
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_DisplaySmallDigits(2, String + 6, 113, line + 1, true);
|
2023-09-16 22:29:02 +01:00
|
|
|
#else
|
|
|
|
// show the frequency in the main font
|
2023-09-18 18:26:10 +01:00
|
|
|
sprintf(String, "%03u.%05u", frequency / 100000, frequency % 100000);
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_PrintString(String, 32, 0, line, 8);
|
2023-09-16 22:29:02 +01:00
|
|
|
#endif
|
|
|
|
break;
|
2023-09-09 13:58:21 +01:00
|
|
|
|
2023-09-16 22:29:02 +01:00
|
|
|
case MDF_CHANNEL: // show the channel number
|
|
|
|
sprintf(String, "CH-%03u", gEeprom.ScreenChannel[vfo_num] + 1);
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_PrintString(String, 32, 0, line, 8);
|
2023-09-16 22:29:02 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case MDF_NAME: // show the channel name
|
2023-09-20 08:29:11 +01:00
|
|
|
case MDF_NAME_FREQ: // show the channel name and frequency
|
2023-09-22 13:52:17 +01:00
|
|
|
|
2023-09-20 08:29:11 +01:00
|
|
|
BOARD_fetchChannelName(String, gEeprom.ScreenChannel[vfo_num]);
|
|
|
|
if (String[0] == 0)
|
2023-09-16 22:29:02 +01:00
|
|
|
{ // no channel name, show the channel number instead
|
2023-09-13 02:01:35 +01:00
|
|
|
sprintf(String, "CH-%03u", gEeprom.ScreenChannel[vfo_num] + 1);
|
2023-09-16 22:29:02 +01:00
|
|
|
}
|
2023-09-22 13:52:17 +01:00
|
|
|
|
2023-09-20 08:29:11 +01:00
|
|
|
if (gEeprom.CHANNEL_DISPLAY_MODE == MDF_NAME)
|
|
|
|
{
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_PrintString(String, 32, 0, line, 8);
|
2023-09-19 15:35:32 +01:00
|
|
|
}
|
|
|
|
else
|
2023-09-20 08:29:11 +01:00
|
|
|
{
|
2023-09-24 01:36:43 +01:00
|
|
|
#ifdef ENABLE_SMALL_BOLD
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_PrintStringSmallBold(String, 32 + 4, 0, line);
|
2023-09-24 01:36:43 +01:00
|
|
|
#else
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_PrintStringSmall(String, 32 + 4, 0, line);
|
2023-09-24 01:36:43 +01:00
|
|
|
#endif
|
2023-09-27 00:26:58 +01:00
|
|
|
|
2023-09-20 08:29:11 +01:00
|
|
|
// show the channel frequency below the channel number/name
|
|
|
|
sprintf(String, "%03u.%05u", frequency / 100000, frequency % 100000);
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_PrintStringSmall(String, 32 + 4, 0, line + 1);
|
2023-09-19 15:35:32 +01:00
|
|
|
}
|
2023-09-22 13:52:17 +01:00
|
|
|
|
2023-09-19 15:35:32 +01:00
|
|
|
break;
|
2023-09-09 08:03:56 +01:00
|
|
|
}
|
|
|
|
}
|
2023-09-16 22:29:02 +01:00
|
|
|
else
|
|
|
|
{ // frequency mode
|
|
|
|
#ifdef ENABLE_BIG_FREQ
|
2023-09-18 18:26:10 +01:00
|
|
|
NUMBER_ToDigits(frequency, String); // 8 digits
|
2023-09-16 22:29:02 +01:00
|
|
|
// show the main large frequency digits
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_DisplayFrequency(String, 32, line, false, false);
|
2023-09-16 22:29:02 +01:00
|
|
|
// show the remaining 2 small frequency digits
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_DisplaySmallDigits(2, String + 6, 113, line + 1, true);
|
2023-09-16 22:29:02 +01:00
|
|
|
#else
|
|
|
|
// show the frequency in the main font
|
2023-09-18 18:26:10 +01:00
|
|
|
sprintf(String, "%03u.%05u", frequency / 100000, frequency % 100000);
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_PrintString(String, 32, 0, line, 8);
|
2023-09-16 22:29:02 +01:00
|
|
|
#endif
|
2023-09-19 17:30:12 +01:00
|
|
|
|
2023-09-20 08:29:11 +01:00
|
|
|
#ifdef ENABLE_COMPANDER
|
|
|
|
// show the channel symbols
|
|
|
|
const uint8_t attributes = gMR_ChannelAttributes[gEeprom.ScreenChannel[vfo_num]];
|
|
|
|
if ((attributes & MR_CH_COMPAND) > 0)
|
|
|
|
#ifdef ENABLE_BIG_FREQ
|
2023-10-01 10:15:24 +01:00
|
|
|
memmove(p_line0 + 120, BITMAP_compand, sizeof(BITMAP_compand));
|
2023-09-20 08:29:11 +01:00
|
|
|
#else
|
2023-10-01 10:15:24 +01:00
|
|
|
memmove(p_line0 + 120 + LCD_WIDTH, BITMAP_compand, sizeof(BITMAP_compand));
|
2023-09-20 08:29:11 +01:00
|
|
|
#endif
|
2023-09-19 17:30:12 +01:00
|
|
|
#endif
|
2023-09-16 22:29:02 +01:00
|
|
|
}
|
2023-09-09 08:03:56 +01:00
|
|
|
}
|
|
|
|
|
2023-09-12 15:31:37 +01:00
|
|
|
// ************
|
|
|
|
|
2023-09-10 13:52:41 +01:00
|
|
|
{ // show the TX/RX level
|
|
|
|
uint8_t Level = 0;
|
2023-09-09 08:03:56 +01:00
|
|
|
|
2023-09-19 14:48:09 +01:00
|
|
|
if (duff_beer == 1)
|
2023-09-10 13:52:41 +01:00
|
|
|
{ // TX power level
|
|
|
|
switch (gRxVfo->OUTPUT_POWER)
|
|
|
|
{
|
2023-09-16 22:29:02 +01:00
|
|
|
case OUTPUT_POWER_LOW: Level = 2; break;
|
|
|
|
case OUTPUT_POWER_MID: Level = 4; break;
|
|
|
|
case OUTPUT_POWER_HIGH: Level = 6; break;
|
2023-09-10 13:52:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2023-09-19 14:48:09 +01:00
|
|
|
if (duff_beer == 2)
|
2023-09-10 13:52:41 +01:00
|
|
|
{ // RX signal level
|
2023-09-29 08:43:05 +01:00
|
|
|
#ifndef ENABLE_RSSI_BAR
|
2023-09-17 15:36:23 +01:00
|
|
|
// bar graph
|
2023-09-23 03:28:21 +01:00
|
|
|
if (gVFO_RSSI_bar_level[vfo_num] > 0)
|
|
|
|
Level = gVFO_RSSI_bar_level[vfo_num];
|
2023-09-17 15:36:23 +01:00
|
|
|
#endif
|
2023-09-10 13:52:41 +01:00
|
|
|
}
|
2023-09-16 22:29:02 +01:00
|
|
|
|
2023-09-10 13:52:41 +01:00
|
|
|
if (Level >= 1)
|
|
|
|
{
|
2023-10-01 10:15:24 +01:00
|
|
|
uint8_t *p_line = p_line1 + LCD_WIDTH;
|
|
|
|
memmove(p_line + 0, BITMAP_Antenna, sizeof(BITMAP_Antenna));
|
2023-09-10 13:52:41 +01:00
|
|
|
if (Level >= 2)
|
2023-10-01 10:15:24 +01:00
|
|
|
memmove(p_line + 5, BITMAP_AntennaLevel1, sizeof(BITMAP_AntennaLevel1));
|
2023-09-10 13:52:41 +01:00
|
|
|
if (Level >= 3)
|
2023-10-01 10:15:24 +01:00
|
|
|
memmove(p_line + 8, BITMAP_AntennaLevel2, sizeof(BITMAP_AntennaLevel2));
|
2023-09-10 13:52:41 +01:00
|
|
|
if (Level >= 4)
|
2023-10-01 10:15:24 +01:00
|
|
|
memmove(p_line + 11, BITMAP_AntennaLevel3, sizeof(BITMAP_AntennaLevel3));
|
2023-09-10 13:52:41 +01:00
|
|
|
if (Level >= 5)
|
2023-10-01 10:15:24 +01:00
|
|
|
memmove(p_line + 14, BITMAP_AntennaLevel4, sizeof(BITMAP_AntennaLevel4));
|
2023-09-10 13:52:41 +01:00
|
|
|
if (Level >= 6)
|
2023-10-01 10:15:24 +01:00
|
|
|
memmove(p_line + 17, BITMAP_AntennaLevel5, sizeof(BITMAP_AntennaLevel5));
|
2023-09-17 15:36:23 +01:00
|
|
|
if (Level >= 7)
|
2023-10-01 10:15:24 +01:00
|
|
|
memmove(p_line + 20, BITMAP_AntennaLevel6, sizeof(BITMAP_AntennaLevel6));
|
2023-09-10 13:52:41 +01:00
|
|
|
}
|
2023-09-09 08:03:56 +01:00
|
|
|
}
|
2023-09-16 22:29:02 +01:00
|
|
|
|
2023-09-12 15:31:37 +01:00
|
|
|
// ************
|
|
|
|
|
2023-09-17 15:36:23 +01:00
|
|
|
String[0] = '\0';
|
2023-09-28 23:32:08 +01:00
|
|
|
if (gEeprom.VfoInfo[vfo_num].AM_mode)
|
2023-09-10 13:52:41 +01:00
|
|
|
{ // show the AM symbol
|
2023-09-17 15:36:23 +01:00
|
|
|
strcpy(String, "AM");
|
2023-09-09 08:03:56 +01:00
|
|
|
}
|
|
|
|
else
|
2023-09-19 07:16:15 +01:00
|
|
|
{ // or show the CTCSS/DCS symbol
|
2023-09-19 14:48:09 +01:00
|
|
|
const FREQ_Config_t *pConfig = (duff_beer == 1) ? gEeprom.VfoInfo[vfo_num].pTX : gEeprom.VfoInfo[vfo_num].pRX;
|
2023-09-16 22:29:02 +01:00
|
|
|
const unsigned int code_type = pConfig->CodeType;
|
|
|
|
const char *code_list[] = {"", "CT", "DCS", "DCR"};
|
|
|
|
if (code_type >= 0 && code_type < ARRAY_SIZE(code_list))
|
2023-09-17 15:36:23 +01:00
|
|
|
strcpy(String, code_list[code_type]);
|
2023-09-09 08:03:56 +01:00
|
|
|
}
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_PrintStringSmall(String, LCD_WIDTH + 24, 0, line + 1);
|
2023-09-09 08:03:56 +01:00
|
|
|
|
2023-10-01 10:15:24 +01:00
|
|
|
if (state == VFO_STATE_NORMAL || state == VFO_STATE_ALARM)
|
2023-09-16 22:29:02 +01:00
|
|
|
{ // show the TX power
|
|
|
|
const char pwr_list[] = "LMH";
|
|
|
|
const unsigned int i = gEeprom.VfoInfo[vfo_num].OUTPUT_POWER;
|
|
|
|
String[0] = (i >= 0 && i < ARRAY_SIZE(pwr_list)) ? pwr_list[i] : '\0';
|
|
|
|
String[1] = '\0';
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_PrintStringSmall(String, LCD_WIDTH + 46, 0, line + 1);
|
2023-09-09 08:03:56 +01:00
|
|
|
}
|
|
|
|
|
2023-09-25 18:08:21 +01:00
|
|
|
if (gEeprom.VfoInfo[vfo_num].freq_config_RX.Frequency != gEeprom.VfoInfo[vfo_num].freq_config_TX.Frequency)
|
2023-09-10 13:52:41 +01:00
|
|
|
{ // show the TX offset symbol
|
2023-09-20 08:29:11 +01:00
|
|
|
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] : '?';
|
2023-09-16 17:16:36 +01:00
|
|
|
String[1] = '\0';
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_PrintStringSmall(String, LCD_WIDTH + 54, 0, line + 1);
|
2023-09-09 08:03:56 +01:00
|
|
|
}
|
|
|
|
|
2023-09-10 13:52:41 +01:00
|
|
|
// show the TX/RX reverse symbol
|
2023-09-10 05:49:39 +01:00
|
|
|
if (gEeprom.VfoInfo[vfo_num].FrequencyReverse)
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_PrintStringSmall("R", LCD_WIDTH + 62, 0, line + 1);
|
2023-09-09 08:03:56 +01:00
|
|
|
|
2023-09-18 00:48:40 +01:00
|
|
|
{ // show the narrow band symbol
|
2023-09-19 07:16:15 +01:00
|
|
|
String[0] = '\0';
|
|
|
|
if (gEeprom.VfoInfo[vfo_num].CHANNEL_BANDWIDTH == BANDWIDTH_NARROW)
|
2023-09-18 00:48:40 +01:00
|
|
|
{
|
2023-09-19 07:16:15 +01:00
|
|
|
String[0] = 'N';
|
|
|
|
String[1] = '\0';
|
2023-09-18 00:48:40 +01:00
|
|
|
}
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_PrintStringSmall(String, LCD_WIDTH + 70, 0, line + 1);
|
2023-09-18 00:48:40 +01:00
|
|
|
}
|
2023-09-21 07:31:29 +01:00
|
|
|
|
2023-09-30 11:22:19 +01:00
|
|
|
// show the DTMF decoding symbol
|
2023-09-10 05:49:39 +01:00
|
|
|
if (gEeprom.VfoInfo[vfo_num].DTMF_DECODING_ENABLE || gSetting_KILLED)
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_PrintStringSmall("DTMF", LCD_WIDTH + 78, 0, line + 1);
|
2023-09-09 08:03:56 +01:00
|
|
|
|
2023-09-10 13:52:41 +01:00
|
|
|
// show the audio scramble symbol
|
2023-09-17 15:36:23 +01:00
|
|
|
if (gEeprom.VfoInfo[vfo_num].SCRAMBLING_TYPE > 0 && gSetting_ScrambleEnable)
|
2023-10-01 12:44:22 +01:00
|
|
|
UI_PrintStringSmall("SCR", LCD_WIDTH + 106, 0, line + 1);
|
2023-09-09 08:03:56 +01:00
|
|
|
}
|
|
|
|
|
2023-09-19 10:23:50 +01:00
|
|
|
if (center_line_is_free)
|
2023-09-29 08:43:05 +01:00
|
|
|
{ // we're free to use the middle line
|
|
|
|
|
|
|
|
const bool rx = (gCurrentFunction == FUNCTION_RECEIVE ||
|
|
|
|
gCurrentFunction == FUNCTION_MONITOR ||
|
|
|
|
gCurrentFunction == FUNCTION_INCOMING);
|
2023-09-22 13:52:17 +01:00
|
|
|
|
2023-09-29 20:48:23 +01:00
|
|
|
#ifdef ENABLE_AUDIO_BAR
|
|
|
|
if (gSetting_mic_bar && gCurrentFunction == FUNCTION_TRANSMIT)
|
|
|
|
UI_DisplayAudioBar();
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
|
2023-09-24 00:06:10 +01:00
|
|
|
#if defined(ENABLE_AM_FIX) && defined(ENABLE_AM_FIX_SHOW_DATA)
|
2023-09-29 09:58:32 +01:00
|
|
|
if (gEeprom.VfoInfo[gEeprom.RX_CHANNEL].AM_mode && gSetting_AM_fix)
|
2023-09-23 12:25:06 +01:00
|
|
|
{
|
2023-09-29 09:58:32 +01:00
|
|
|
if (rx)
|
|
|
|
{
|
|
|
|
AM_fix_print_data(gEeprom.RX_CHANNEL, String);
|
2023-09-29 20:48:23 +01:00
|
|
|
UI_PrintStringSmall(String, 2, 0, 3);
|
2023-09-29 09:58:32 +01:00
|
|
|
}
|
2023-09-23 12:25:06 +01:00
|
|
|
}
|
2023-09-24 01:58:03 +01:00
|
|
|
else
|
|
|
|
#endif
|
2023-09-29 08:43:05 +01:00
|
|
|
|
|
|
|
#ifdef ENABLE_RSSI_BAR
|
|
|
|
if (rx)
|
|
|
|
UI_DisplayRSSIBar(gCurrentRSSI[gEeprom.RX_CHANNEL], false);
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
|
2023-09-29 09:58:32 +01:00
|
|
|
if (rx || gCurrentFunction == FUNCTION_FOREGROUND)
|
|
|
|
{
|
2023-09-30 11:22:19 +01:00
|
|
|
#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
|
|
|
|
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
|
|
|
|
strcpy(String, "DTMF ");
|
|
|
|
strcat(String, gDTMF_RX + idx);
|
|
|
|
UI_PrintStringSmall(String, 2, 0, 3);
|
|
|
|
}
|
|
|
|
#endif
|
2023-09-30 20:03:09 +01:00
|
|
|
|
2023-09-29 09:58:32 +01:00
|
|
|
#ifdef ENABLE_SHOW_CHARGE_LEVEL
|
2023-09-23 12:07:46 +01:00
|
|
|
else
|
|
|
|
if (gChargingWithTypeC)
|
|
|
|
{ // charging .. show the battery state
|
2023-09-29 09:58:32 +01:00
|
|
|
const uint16_t volts = (gBatteryVoltageAverage < gMin_bat_v) ? gMin_bat_v : gBatteryVoltageAverage;
|
|
|
|
const uint16_t percent = (100 * (volts - gMin_bat_v)) / (gMax_bat_v - gMin_bat_v);
|
|
|
|
sprintf(String, "Charge %u.%02uV %u%%", gBatteryVoltageAverage / 100, gBatteryVoltageAverage % 100, percent);
|
|
|
|
UI_PrintStringSmall(String, 2, 0, 3);
|
2023-09-23 12:07:46 +01:00
|
|
|
}
|
2023-09-29 09:58:32 +01:00
|
|
|
#endif
|
2023-09-24 01:58:03 +01:00
|
|
|
}
|
2023-09-19 10:23:50 +01:00
|
|
|
}
|
2023-09-17 15:36:23 +01:00
|
|
|
|
2023-09-09 08:03:56 +01:00
|
|
|
ST7565_BlitFullScreen();
|
|
|
|
}
|
2023-09-29 08:43:05 +01:00
|
|
|
|
|
|
|
// ***************************************************************************
|