/* 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 #include // abs() #include "app/dtmf.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" bool center_line_is_free = true; // *************************************************************************** #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 (gSetting_mic_bar) { const unsigned int line = 3; const unsigned int bar_x = 2; const unsigned int bar_width = LCD_WIDTH - 2 - bar_x; unsigned int i; #if 1 // TX audio level if (gCurrentFunction != FUNCTION_TRANSMIT) return; const unsigned int voice_amp = BK4819_GetVoiceAmplitudeOut(); // 15:0 // const unsigned int max = 65535; // const unsigned int level = ((voice_amp * bar_width) + (max / 2)) / max; // with rounding // const unsigned int len = (level <= bar_width) ? level : bar_width; // 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); const unsigned int len = (sqrt_level <= bar_width) ? sqrt_level : bar_width; #else // TX/RX AF input level (dB) const uint8_t af_tx_rx = BK4819_GetAfTxRx(); // 6:0 const unsigned int max = 63; 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; #endif uint8_t *pLine = gFrameBuffer[line]; memset(pLine, 0, LCD_WIDTH); #if 1 // solid bar for (i = 0; i < bar_width; i++) pLine[bar_x + i] = (i > len) ? ((i & 1) == 0) ? 0x41 : 0x00 : ((i & 1) == 0) ? 0x7f : 0x3e; #else // knuled bar for (i = 0; i < bar_width; i += 2) pLine[bar_x + i] = (i <= len) ? 0x7f : 0x41; #endif if (gCurrentFunction == FUNCTION_TRANSMIT) ST7565_BlitFullScreen(); } } #endif #if defined(ENABLE_RSSI_BAR) void UI_DisplayRSSIBar(const int16_t rssi, const bool now) { const int16_t s0_dBm = -127; // S0 .. base level // const int16_t s0_dBm = -147; // S0 .. base level const int16_t s9_dBm = s0_dBm + (6 * 9); // S9 .. 6dB/S-Point const int16_t bar_max_dBm = s9_dBm + 30; // S9+30dB const int16_t bar_min_dBm = s0_dBm; // S0 // ************ 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; 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; 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; const unsigned int line = 3; uint8_t *pLine = gFrameBuffer[line]; char s[16]; unsigned int i; if (gEeprom.KEY_LOCK && gKeypadLocked > 0) return; // display is in use if (gCurrentFunction == FUNCTION_TRANSMIT || gScreenToDisplay != DISPLAY_MAIN) return; // display is in use if (now) memset(pLine, 0, LCD_WIDTH); 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); } else { // 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); } UI_PrintStringSmall(s, 2, 0, line); #if 1 // solid bar for (i = 0; i < bar_width; i++) pLine[bar_x + i] = (i > len) ? ((i & 1) == 0) ? 0x41 : 0x00 : ((i & 1) == 0) ? 0x7f : 0x3e; #else // knuled bar for (i = 0; i < bar_width; i += 2) pLine[bar_x + i] = (i <= len) ? 0x7f : 0x41; #endif if (now) ST7565_BlitFullScreen(); } #endif void UI_UpdateRSSI(const int16_t rssi, const int vfo) { #ifdef ENABLE_RSSI_BAR if (!center_line_is_free) return; const bool rx = (gCurrentFunction == FUNCTION_RECEIVE || gCurrentFunction == FUNCTION_MONITOR || gCurrentFunction == FUNCTION_INCOMING); #if defined(ENABLE_AM_FIX) && defined(ENABLE_AM_FIX_SHOW_DATA) if (gEeprom.VfoInfo[gEeprom.RX_CHANNEL].AM_mode && gSetting_AM_fix) { // AM test data is currently being shown } else #endif if (rx) UI_DisplayRSSIBar(rssi, true); #else // const int16_t dBm = (rssi / 2) - 160; const uint8_t Line = (vfo == 0) ? 3 : 7; uint8_t *pLine = gFrameBuffer[Line - 1]; // TODO: sort out all 8 values from the eeprom #if 0 // 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; 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; gVFO_RSSI[vfo] = rssi; uint8_t rssi_level = 0; 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; if (gVFO_RSSI_bar_level[vfo] == rssi_level) return; gVFO_RSSI_bar_level[vfo] = rssi_level; // ********************************************************** if (gEeprom.KEY_LOCK && gKeypadLocked > 0) return; // display is in use if (gCurrentFunction == FUNCTION_TRANSMIT || gScreenToDisplay != DISPLAY_MAIN) return; // display is in use pLine = gFrameBuffer[Line - 1]; memset(pLine, 0, 23); if (rssi_level > 0) { //if (rssi_level >= 1) memmove(pLine, BITMAP_Antenna, 5); if (rssi_level >= 2) memmove(pLine + 5, BITMAP_AntennaLevel1, sizeof(BITMAP_AntennaLevel1)); if (rssi_level >= 3) memmove(pLine + 8, BITMAP_AntennaLevel2, sizeof(BITMAP_AntennaLevel2)); if (rssi_level >= 4) memmove(pLine + 11, BITMAP_AntennaLevel3, sizeof(BITMAP_AntennaLevel3)); if (rssi_level >= 5) memmove(pLine + 14, BITMAP_AntennaLevel4, sizeof(BITMAP_AntennaLevel4)); if (rssi_level >= 6) memmove(pLine + 17, BITMAP_AntennaLevel5, sizeof(BITMAP_AntennaLevel5)); if (rssi_level >= 7) memmove(pLine + 20, BITMAP_AntennaLevel6, sizeof(BITMAP_AntennaLevel6)); } else pLine = NULL; ST7565_DrawLine(0, Line, 23, pLine); #endif } // *************************************************************************** void UI_DisplayMain(void) { char String[16]; unsigned int vfo_num; center_line_is_free = true; // #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 memset(gFrameBuffer, 0, sizeof(gFrameBuffer)); if (gEeprom.KEY_LOCK && gKeypadLocked > 0) { // tell user how to unlock the keyboard UI_PrintString("Long press #", 0, LCD_WIDTH, 1, 8); UI_PrintString("to unlock", 0, LCD_WIDTH, 3, 8); ST7565_BlitFullScreen(); return; } for (vfo_num = 0; vfo_num < 2; vfo_num++) { uint8_t Channel = gEeprom.TX_CHANNEL; bool bIsSameVfo = !!(Channel == vfo_num); uint8_t Line = (vfo_num == 0) ? 0 : 4; uint8_t *pLine0 = gFrameBuffer[Line + 0]; uint8_t *pLine1 = gFrameBuffer[Line + 1]; if (single_vfo) { // we're in single VFO mode - screen is dedicated to just one VFO if (!bIsSameVfo) continue; // skip the unused vfo } if (gEeprom.DUAL_WATCH != DUAL_WATCH_OFF && gRxVfoIsActive) Channel = gEeprom.RX_CHANNEL; // we're currently monitoring the other VFO if (Channel != vfo_num) { if (gDTMF_CallState != DTMF_CALL_STATE_NONE || gDTMF_IsTx || gDTMF_InputMode) { // show DTMF stuff 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) 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 { sprintf(String, ">%s", gDTMF_InputBox); center_line_is_free = false; } UI_PrintString(String, 2, 0, vfo_num * 3, 8); 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) sprintf(String, ">%s", (DTMF_FindContact(gDTMF_Callee, Contact)) ? Contact : gDTMF_Callee); else if (gDTMF_IsTx) sprintf(String, ">%s", gDTMF_String); } else { center_line_is_free = false; } UI_PrintString(String, 2, 0, 2 + (vfo_num * 3), 8); center_line_is_free = false; continue; } // highlight the selected/used VFO with a marker if (!single_vfo && bIsSameVfo) memmove(pLine0 + 0, BITMAP_VFO_Default, sizeof(BITMAP_VFO_Default)); else if (gEeprom.CROSS_BAND_RX_TX != CROSS_BAND_OFF) memmove(pLine0 + 0, BITMAP_VFO_NotDefault, sizeof(BITMAP_VFO_NotDefault)); } else if (!single_vfo) { // highlight the selected/used VFO with a marker if (bIsSameVfo) memmove(pLine0 + 0, BITMAP_VFO_Default, sizeof(BITMAP_VFO_Default)); else //if (gEeprom.CROSS_BAND_RX_TX != CROSS_BAND_OFF) memmove(pLine0 + 0, BITMAP_VFO_NotDefault, sizeof(BITMAP_VFO_NotDefault)); } uint32_t duff_beer = 0; if (gCurrentFunction == FUNCTION_TRANSMIT) { // transmitting #ifdef ENABLE_ALARM if (gAlarmState == ALARM_STATE_ALARM) duff_beer = 2; else #endif { Channel = (gEeprom.CROSS_BAND_RX_TX == CROSS_BAND_OFF) ? gEeprom.RX_CHANNEL : gEeprom.TX_CHANNEL; if (Channel == vfo_num) { // show the TX symbol duff_beer = 1; UI_PrintStringSmall("TX", 14, 0, Line); } } } else { // receiving .. show the RX symbol duff_beer = 2; if ((gCurrentFunction == FUNCTION_RECEIVE || gCurrentFunction == FUNCTION_MONITOR) && gEeprom.RX_CHANNEL == vfo_num) UI_PrintStringSmall("RX", 14, 0, Line); } if (IS_MR_CHANNEL(gEeprom.ScreenChannel[vfo_num])) { // channel mode const unsigned int x = 2; const bool inputting = (gInputBoxIndex == 0 || gEeprom.TX_CHANNEL != vfo_num) ? false : true; if (!inputting) NUMBER_ToDigits(gEeprom.ScreenChannel[vfo_num] + 1, String); // show the memory channel number else 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); } else if (IS_FREQ_CHANNEL(gEeprom.ScreenChannel[vfo_num])) { // frequency mode // show the frequency band number const unsigned int x = 2; // was 14 // sprintf(String, "FB%u", 1 + gEeprom.ScreenChannel[vfo_num] - FREQ_CHANNEL_FIRST); sprintf(String, "VFO%u", 1 + gEeprom.ScreenChannel[vfo_num] - FREQ_CHANNEL_FIRST); UI_PrintStringSmall(String, x, 0, Line + 1); } #ifdef ENABLE_NOAA else { 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]); } UI_PrintStringSmall(String, 7, 0, Line + 1); } #endif // ************ uint8_t State = VfoState[vfo_num]; #ifdef ENABLE_ALARM if (gCurrentFunction == FUNCTION_TRANSMIT && gAlarmState == ALARM_STATE_ALARM) { Channel = (gEeprom.CROSS_BAND_RX_TX == CROSS_BAND_OFF) ? gEeprom.RX_CHANNEL : gEeprom.TX_CHANNEL; if (Channel == vfo_num) State = VFO_STATE_ALARM; } #endif if (State != VFO_STATE_NORMAL) { const char *state_list[] = {"", "BUSY", "BAT LOW", "TX DISABLE", "TIMEOUT", "ALARM", "VOLT HIGH"}; if (State >= 0 && State < ARRAY_SIZE(state_list)) { UI_PrintString(state_list[State], 31, 0, Line, 8); } //else //{ // sprintf(String, "State %u ?", State); // UI_PrintString(String, 31, 0, Line, 8); //} } else if (gInputBoxIndex > 0 && IS_FREQ_CHANNEL(gEeprom.ScreenChannel[vfo_num]) && gEeprom.TX_CHANNEL == vfo_num) { // user entering a frequency UI_DisplayFrequency(gInputBox, 32, Line, true, false); center_line_is_free = false; } else { uint32_t frequency = gEeprom.VfoInfo[vfo_num].pRX->Frequency; if (gCurrentFunction == FUNCTION_TRANSMIT) { // transmitting Channel = (gEeprom.CROSS_BAND_RX_TX == CROSS_BAND_OFF) ? gEeprom.RX_CHANNEL : gEeprom.TX_CHANNEL; if (Channel == vfo_num) frequency = gEeprom.VfoInfo[vfo_num].pTX->Frequency; } if (IS_MR_CHANNEL(gEeprom.ScreenChannel[vfo_num])) { // channel mode // show the channel symbols const uint8_t attributes = gMR_ChannelAttributes[gEeprom.ScreenChannel[vfo_num]]; if (attributes & MR_CH_SCANLIST1) memmove(pLine0 + 113, BITMAP_ScanList1, sizeof(BITMAP_ScanList1)); if (attributes & MR_CH_SCANLIST2) memmove(pLine0 + 120, BITMAP_ScanList2, sizeof(BITMAP_ScanList2)); #ifndef ENABLE_BIG_FREQ #ifdef ENABLE_COMPANDER if ((attributes & MR_CH_COMPAND) > 0) memmove(pLine0 + 120 + LCD_WIDTH, BITMAP_compand, sizeof(BITMAP_compand)); #endif #endif switch (gEeprom.CHANNEL_DISPLAY_MODE) { case MDF_FREQUENCY: // show the channel frequency #ifdef ENABLE_BIG_FREQ NUMBER_ToDigits(frequency, String); // show the main large frequency digits UI_DisplayFrequency(String, 32, Line, false, false); // show the remaining 2 small frequency digits UI_DisplaySmallDigits(2, String + 6, 113, Line + 1, true); #else // show the frequency in the main font sprintf(String, "%03u.%05u", frequency / 100000, frequency % 100000); UI_PrintString(String, 32, 0, Line, 8); #endif break; case MDF_CHANNEL: // show the channel number sprintf(String, "CH-%03u", gEeprom.ScreenChannel[vfo_num] + 1); UI_PrintString(String, 32, 0, Line, 8); 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_PrintString(String, 32, 0, Line, 8); } 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 #ifdef ENABLE_BIG_FREQ NUMBER_ToDigits(frequency, String); // 8 digits // show the main large frequency digits UI_DisplayFrequency(String, 32, Line, false, false); // show the remaining 2 small frequency digits UI_DisplaySmallDigits(2, String + 6, 113, Line + 1, true); #else // show the frequency in the main font sprintf(String, "%03u.%05u", frequency / 100000, frequency % 100000); UI_PrintString(String, 32, 0, Line, 8); #endif #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 memmove(pLine0 + 120, BITMAP_compand, sizeof(BITMAP_compand)); #else memmove(pLine0 + 120 + LCD_WIDTH, BITMAP_compand, sizeof(BITMAP_compand)); #endif #endif } } // ************ { // show the TX/RX level uint8_t Level = 0; if (duff_beer == 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 (duff_beer == 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 >= 1) { uint8_t *pLine = pLine1 + LCD_WIDTH; memmove(pLine + 0, BITMAP_Antenna, sizeof(BITMAP_Antenna)); if (Level >= 2) memmove(pLine + 5, BITMAP_AntennaLevel1, sizeof(BITMAP_AntennaLevel1)); if (Level >= 3) memmove(pLine + 8, BITMAP_AntennaLevel2, sizeof(BITMAP_AntennaLevel2)); if (Level >= 4) memmove(pLine + 11, BITMAP_AntennaLevel3, sizeof(BITMAP_AntennaLevel3)); if (Level >= 5) memmove(pLine + 14, BITMAP_AntennaLevel4, sizeof(BITMAP_AntennaLevel4)); if (Level >= 6) memmove(pLine + 17, BITMAP_AntennaLevel5, sizeof(BITMAP_AntennaLevel5)); if (Level >= 7) memmove(pLine + 20, BITMAP_AntennaLevel6, sizeof(BITMAP_AntennaLevel6)); } } // ************ String[0] = '\0'; if (gEeprom.VfoInfo[vfo_num].AM_mode) { // show the AM symbol strcpy(String, "AM"); } else { // or show the CTCSS/DCS symbol const FREQ_Config_t *pConfig = (duff_beer == 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"}; if (code_type >= 0 && code_type < ARRAY_SIZE(code_list)) strcpy(String, code_list[code_type]); } UI_PrintStringSmall(String, LCD_WIDTH + 24, 0, Line + 1); if (State != VFO_STATE_TX_DISABLE) { // 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'; 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); } // show the DTMF decoding symbol if (gEeprom.VfoInfo[vfo_num].DTMF_DECODING_ENABLE || gSetting_KILLED) UI_PrintStringSmall("DTMF", LCD_WIDTH + 78, 0, Line + 1); // show the audio scramble symbol if (gEeprom.VfoInfo[vfo_num].SCRAMBLING_TYPE > 0 && gSetting_ScrambleEnable) UI_PrintStringSmall("SCR", LCD_WIDTH + 106, 0, Line + 1); } if (center_line_is_free) { // 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 (gSetting_mic_bar && gCurrentFunction == FUNCTION_TRANSMIT) UI_DisplayAudioBar(); else #endif #if defined(ENABLE_AM_FIX) && defined(ENABLE_AM_FIX_SHOW_DATA) if (gEeprom.VfoInfo[gEeprom.RX_CHANNEL].AM_mode && gSetting_AM_fix) { if (rx) { AM_fix_print_data(gEeprom.RX_CHANNEL, String); UI_PrintStringSmall(String, 2, 0, 3); } } else #endif #ifdef ENABLE_RSSI_BAR if (rx) UI_DisplayRSSIBar(gCurrentRSSI[gEeprom.RX_CHANNEL], false); else #endif if (rx || gCurrentFunction == FUNCTION_FOREGROUND) { #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 #ifdef ENABLE_SHOW_CHARGE_LEVEL else if (gChargingWithTypeC) { // charging .. show the battery state 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); } #endif } } ST7565_BlitFullScreen(); } // ***************************************************************************