uv-k5-firmware-chinese-lts/ui/menu.c

1181 lines
38 KiB
C
Raw Normal View History

2023-12-07 03:39:09 +00: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>
#include <stdlib.h> // abs()
#include "bitmaps.h"
#include "driver/uart.h"
#include "app/dtmf.h"
#include "app/menu.h"
#include "board.h"
#include "dcs.h"
#include "driver/backlight.h"
#include "driver/bk4819.h"
#include "driver/st7565.h"
#include "external/printf/printf.h"
#include "frequencies.h"
#include "helper/battery.h"
#include "misc.h"
#include "settings.h"
#include "ui/helper.h"
#include "ui/inputbox.h"
#include "ui/menu.h"
#include "ui/ui.h"
#include "chinese.h"
2024-01-05 09:04:33 +00:00
void insertNewline(char a[], int index, int len) {
2023-12-07 03:39:09 +00:00
2024-01-05 09:04:33 +00:00
if (index < 0 || index >= len || len >= 63) {
2023-12-07 03:39:09 +00:00
return;
}
for (int i = len; i >= index; i--) {
a[i + 1] = a[i];
}
a[index] = '\n';
a[len + 1] = '\0'; // Null-terminate the string
}
const t_menu_item MenuList[] =
{
// text, voice ID, menu ID
2024-01-05 09:04:33 +00:00
{/*"Step",*/ VOICE_ID_FREQUENCY_STEP, MENU_STEP, },
{/*"RxDCS",*/ VOICE_ID_DCS, MENU_R_DCS, }, // was "R_DCS"
{/*"RxCTCS",*/ VOICE_ID_CTCSS, MENU_R_CTCS, }, // was "R_CTCS"
{/*"TxDCS",*/ VOICE_ID_DCS, MENU_T_DCS, }, // was "T_DCS"
{/*"TxCTCS",*/ VOICE_ID_CTCSS, MENU_T_CTCS, }, // was "T_CTCS"
{/*"TxODir",*/ VOICE_ID_TX_OFFSET_FREQUENCY_DIRECTION, MENU_SFT_D, }, // was "SFT_D"
{/*"TxOffs",*/ VOICE_ID_TX_OFFSET_FREQUENCY, MENU_OFFSET, }, // was "OFFSET"
{/*"Scramb",*/ VOICE_ID_SCRAMBLER_ON, MENU_SCR, }, // was "SCR"
{/*"BusyCL",*/ VOICE_ID_BUSY_LOCKOUT, MENU_BCL, }, // was "BCL"
{/*"Compnd",*/ VOICE_ID_INVALID, MENU_COMPAND, },
{/*"ChSave",*/ VOICE_ID_MEMORY_CHANNEL, MENU_MEM_CH, }, // was "MEM-CH"
{/*"ChDele",*/ VOICE_ID_DELETE_CHANNEL, MENU_DEL_CH, }, // was "DEL-CH"
{/*"ChName",*/ VOICE_ID_INVALID, MENU_MEM_NAME, },
{/*"SList",*/ VOICE_ID_INVALID, MENU_S_LIST, },
{/*"SList1",*/ VOICE_ID_INVALID, MENU_SLIST1, 1},
{/*"SList2",*/ VOICE_ID_INVALID, MENU_SLIST2, 2},
{/*"ScnRev",*/ VOICE_ID_INVALID, MENU_SC_REV, },
{/*"TxTOut",*/ VOICE_ID_TRANSMIT_OVER_TIME, MENU_TOT, }, // was "TOT"
{/*"BatSav",*/ VOICE_ID_SAVE_MODE, MENU_SAVE, }, // was "SAVE"
{/*"Mic",*/ VOICE_ID_INVALID, MENU_MIC, },
{/*"ChDisp",*/ VOICE_ID_INVALID, MENU_MDF, }, // was "MDF"
2024-01-08 03:46:18 +00:00
#if ENABLE_CHINESE_FULL==4
{/*"POnMsg",*/ VOICE_ID_INVALID, MENU_PONMSG ,},
#endif
2024-01-05 09:04:33 +00:00
{/*"BackLt",*/ VOICE_ID_INVALID, MENU_ABR, }, // was "ABR"
{/*"BLMax",*/ VOICE_ID_INVALID, MENU_ABR_MAX, },
{/*"MDCID",*/ VOICE_ID_INVALID, MENU_MDC_ID, MDC_ID},
{/*"Roger",*/ VOICE_ID_INVALID, MENU_ROGER, },
{/*"STE",*/ VOICE_ID_INVALID, MENU_STE, },
{/*"RP STE",*/ VOICE_ID_INVALID, MENU_RP_STE, },
{/*"1 Call",*/ VOICE_ID_INVALID, MENU_1_CALL, },
2023-12-07 03:39:09 +00:00
#ifdef ENABLE_DTMF_CALLING
{/*"ANI ID",*/ VOICE_ID_ANI_CODE, MENU_ANI_ID ,DTMF_ID},
#endif
2024-01-05 09:04:33 +00:00
{/*"UPCode",*/ VOICE_ID_INVALID, MENU_UPCODE, DTMF上线码},
{/*"DWCode",*/ VOICE_ID_INVALID, MENU_DWCODE, DTMF下线码},
{/*"PTT ID",*/ VOICE_ID_INVALID, MENU_PTT_ID, DTMF发送},
{/*"D ST",*/ VOICE_ID_INVALID, MENU_D_ST, DTMF侧音},
2023-12-07 03:39:09 +00:00
#ifdef ENABLE_DTMF_CALLING
{/*"D Resp",*/ VOICE_ID_INVALID, MENU_D_RSP ,DTMF响应},
{/*"D Hold",*/ VOICE_ID_INVALID, MENU_D_HOLD ,DTMF复位},
#endif
2024-01-05 09:04:33 +00:00
{/*"D Prel",*/ VOICE_ID_INVALID, MENU_D_PRE, DTMF预载波},
2023-12-07 03:39:09 +00:00
#ifdef ENABLE_DTMF_CALLING
{/*"D List",*/ VOICE_ID_INVALID, MENU_D_LIST ,DTMF联系人},
#endif
2024-01-05 09:04:33 +00:00
{/*"D Live",*/ VOICE_ID_INVALID, MENU_D_LIVE_DEC, DTMF显示}, // live DTMF decoder
2023-12-07 03:39:09 +00:00
#ifdef ENABLE_AM_FIX//1
{/*"AM Fix",*/ VOICE_ID_INVALID, MENU_AM_FIX ,AM自动增益},
#endif
#ifdef ENABLE_AM_FIX_TEST1//0
{/*"AM FT1",*/ VOICE_ID_INVALID, MENU_AM_FIX_TEST1 ,""},
#endif
2024-01-05 09:04:33 +00:00
{/*"RxMode",*/ VOICE_ID_DUAL_STANDBY, MENU_TDR, },
{/*"Sql",*/ VOICE_ID_SQUELCH, MENU_SQL, },
2023-12-07 03:39:09 +00:00
// hidden menu items from here on
// enabled if pressing both the PTT and upper side button at power-on
2024-01-05 09:04:33 +00:00
{/*"F Lock",*/ VOICE_ID_INVALID, MENU_F_LOCK, },
2023-12-07 03:39:09 +00:00
// {/*"Tx 200",*/ VOICE_ID_INVALID, MENU_200TX ,两百M发射}, // was "200TX"
// {/*"Tx 350",*/ VOICE_ID_INVALID, MENU_350TX ,三百五十M发射}, // was "350TX"
// {/*"Tx 500",*/ VOICE_ID_INVALID, MENU_500TX ,五百M发射}, // was "500TX"
// {/*"350 En",*/ VOICE_ID_INVALID, MENU_350EN ,三百五十M接收}, // was "350EN"
#ifdef ENABLE_F_CAL_MENU//0
{/*"FrCali",*/ VOICE_ID_INVALID, MENU_F_CALI ,""}, // reference xtal calibration
#endif
2024-01-05 09:04:33 +00:00
{/*"BatCal",*/ VOICE_ID_INVALID, MENU_BATCAL, }, // battery voltage calibration
{/*"BatTyp",*/ VOICE_ID_INVALID, MENU_BATTYP, }, // battery type 1600/2200mAh
{/*"Reset",*/ VOICE_ID_INITIALISATION, MENU_RESET,
}, // might be better to move this to the hidden menu items ?
2023-12-07 03:39:09 +00:00
2024-01-08 04:41:14 +00:00
{/*"",*/ VOICE_ID_INVALID, 0xff ,"\x00"} // end of list - DO NOT delete or move this this
2023-12-07 03:39:09 +00:00
};
2024-01-08 03:46:18 +00:00
#if ENABLE_CHINESE_FULL == 4
const char gSubMenu_PONMSG[][5]={
,
,
};
#endif
2024-01-05 09:04:33 +00:00
#if ENABLE_CHINESE_FULL != 4
2023-12-07 03:39:09 +00:00
const char gSubMenu_SFT_D[][10] =//4
#else
2024-01-08 03:46:18 +00:00
2024-01-05 09:04:33 +00:00
const char gSubMenu_SFT_D[][16] =//4
#endif
2023-12-07 03:39:09 +00:00
{
// "OFF",
// "+",
// "-"
,
,
};
2024-01-05 09:04:33 +00:00
#if ENABLE_CHINESE_FULL != 4
2023-12-07 03:39:09 +00:00
const char gSubMenu_OFF_ON[][3] =//4
#else
2024-01-05 09:04:33 +00:00
const char gSubMenu_OFF_ON[][5] =//4
#endif
2023-12-07 03:39:09 +00:00
{
// "OFF",
// "ON"
,
};
2024-01-05 09:04:33 +00:00
#if ENABLE_CHINESE_FULL != 4
2023-12-07 03:39:09 +00:00
const char gSubMenu_SAVE[][4] =//4
#else
2024-01-05 09:04:33 +00:00
const char gSubMenu_SAVE[][6] =//4
#endif
2023-12-07 03:39:09 +00:00
{
// "OFF",
// "1:1",
// "1:2",
// "1:3",
// "1:4"
,
,
,
,
};
2024-01-05 09:04:33 +00:00
#if ENABLE_CHINESE_FULL != 4
2023-12-07 03:39:09 +00:00
const char gSubMenu_TOT[][5] = //7
#else
2024-01-05 09:04:33 +00:00
const char gSubMenu_TOT[][6] = //7
#endif
2023-12-07 03:39:09 +00:00
{
// "30 sec",
// "1 min",
// "2 min",
// "3 min",
// "4 min",
// "5 min",
// "6 min",
// "7 min",
// "8 min",
// "9 min",
// "15 min"
2024-01-05 09:04:33 +00:00
,
,
,
,
,
,
,
,
,
,
2023-12-07 03:39:09 +00:00
};
2024-01-05 09:04:33 +00:00
const char *const gSubMenu_RXMode[] =
2023-12-07 03:39:09 +00:00
{
// "MAIN\nONLY", // TX and RX on main only
// "DUAL RX\nRESPOND", // Watch both and respond
// "CROSS\nBAND", // TX on main, RX on secondary
// "MAIN TX\nDUAL RX" // always TX on main, but RX on both
, // TX and RX on main only
, // Watch both and respond
, // TX on main, RX on secondary
// always TX on main, but RX on both
};
#ifdef ENABLE_VOICE
const char gSubMenu_VOICE[][4] =
{
"OFF",
"CHI",
"ENG"
};
#endif
2024-01-05 09:04:33 +00:00
#if ENABLE_CHINESE_FULL != 4
2023-12-07 03:39:09 +00:00
const char gSubMenu_SC_REV[][10] =//8
#else
2024-01-05 09:04:33 +00:00
const char gSubMenu_SC_REV[][18] =//8
#endif
2023-12-07 03:39:09 +00:00
{
// "TIMEOUT",
// "CARRIER",
// "STOP"
2024-01-05 09:04:33 +00:00
5,
,
2023-12-07 03:39:09 +00:00
};
2024-01-05 09:04:33 +00:00
const char *const gSubMenu_MDF[] =
2023-12-07 03:39:09 +00:00
{
// "FREQ",
// "CHANNEL\nNUMBER",
// "NAME",
// "NAME\n+\nFREQ"
,
,
,
};
#ifdef ENABLE_ALARM
const char gSubMenu_AL_MOD[][5] =
{
"SITE",
"TONE"
};
#endif
#ifdef ENABLE_DTMF_CALLING
#if ENABLE_CHINESE_FULL!=4
2023-12-07 03:39:09 +00:00
const char gSubMenu_D_RSP[][10] =//11
#else
const char gSubMenu_D_RSP[][18] =//11
#endif
2023-12-07 03:39:09 +00:00
{
// "DO\nNOTHING",
// "RING",
// "REPLY",
// "BOTH"
,
,
,
};
#endif
2024-01-05 09:04:33 +00:00
const char *const gSubMenu_PTT_ID[] =
2023-12-07 03:39:09 +00:00
{
// "OFF",
// "UP CODE",
// "DOWN CODE",
// "UP+DOWN\nCODE",
// "APOLLO\nQUINDAR"
,
线,
线,
线线,
Quindar码
};
2024-01-05 09:04:33 +00:00
#if ENABLE_CHINESE_FULL != 4
2023-12-07 03:39:09 +00:00
const char gSubMenu_ROGER[][13] =
#else
2024-01-05 09:04:33 +00:00
const char gSubMenu_ROGER[][15] =
#endif
2023-12-07 03:39:09 +00:00
{
// "OFF",
// "ROGER",
// "MDC"
,
ROGER尾音,
MDC尾音,
MDC首音,
2024-01-05 09:04:33 +00:00
MDC首尾音,
2023-12-07 03:39:09 +00:00
MDC首音加ROGER
};
2024-01-05 09:04:33 +00:00
#if ENABLE_CHINESE_FULL != 4
2023-12-07 03:39:09 +00:00
const char gSubMenu_RESET[][6] =//4
#else
2024-01-05 09:04:33 +00:00
const char gSubMenu_RESET[][11] =//4
#endif
2023-12-07 03:39:09 +00:00
{
// "VFO",
// "ALL"
,
};
2024-01-05 09:04:33 +00:00
const char *const gSubMenu_F_LOCK[] =
2023-12-07 03:39:09 +00:00
{
"DEFAULT+\n137-174\n400-470",
"FCC HAM\n144-148\n420-450",
"CE HAM\n144-146\n430-440",
"GB HAM\n144-148\n430-440",
"137-174\n400-430",
"137-174\n400-438",
// "DISABLE\nALL",
// "UNLOCK\nALL",
,
,
};
2024-01-05 09:04:33 +00:00
#if ENABLE_CHINESE_FULL != 4
2023-12-07 03:39:09 +00:00
const char gSubMenu_BACKLIGHT[][5] =//7
#else
2024-01-05 09:04:33 +00:00
const char gSubMenu_BACKLIGHT[][6] =//7
#endif
2023-12-07 03:39:09 +00:00
{
// "OFF",
// "5 sec",
// "10 sec",
// "20 sec",
// "1 min",
// "2 min",
// "4 min",
// "ON"
,
,
,
,
,
,
,
};
2024-01-05 09:04:33 +00:00
#if ENABLE_CHINESE_FULL != 4
2023-12-07 03:39:09 +00:00
const char gSubMenu_RX_TX[][7] =//6
#else
2024-01-05 09:04:33 +00:00
const char gSubMenu_RX_TX[][12] =//6
#endif
2023-12-07 03:39:09 +00:00
{
// "OFF",
// "TX",
// "RX",
// "TX/RX"
,
,
,
};
#ifdef ENABLE_AM_FIX_TEST1
const char gSubMenu_AM_fix_test1[][8] =
{
"LNA-S 0",
"LNA-S 1",
"LNA-S 2",
"LNA-S 3"
};
#endif
const char gSubMenu_BATTYP[][8] =
{
"1600mAh",
"2200mAh"
};
const char gSubMenu_SCRAMBLER[][7] =
{
// "OFF",
,
"2600Hz",
"2700Hz",
"2800Hz",
"2900Hz",
"3000Hz",
"3100Hz",
"3200Hz",
"3300Hz",
"3400Hz",
"3500Hz"
};
bool gIsInSubMenu;
uint8_t gMenuCursor;
int UI_MENU_GetCurrentMenuId() {
if (gMenuCursor < ARRAY_SIZE(MenuList))
return MenuList[gMenuCursor].menu_id;
else
return MenuList[ARRAY_SIZE(MenuList) - 1].menu_id;
}
uint8_t UI_MENU_GetMenuIdx(uint8_t id) {
for (uint8_t i = 0; i < ARRAY_SIZE(MenuList); i++)
if (MenuList[i].menu_id == id)
return i;
return 0;
}
int32_t gSubMenuSelection;
// edit box
char edit_original[17]; // a copy of the text before editing so that we can easily test for changes/difference
char edit[17];
int edit_index;
void UI_DisplayMenu(void) {
const unsigned int menu_list_width = 6; // max no. of characters on the menu list (left side)
2024-01-05 09:04:33 +00:00
const unsigned int menu_item_x1 = (8 * menu_list_width);//+ 2;
2023-12-07 03:39:09 +00:00
const unsigned int menu_item_x2 = LCD_WIDTH - 1;
unsigned int i;
char String[64]; // bigger cuz we can now do multi-line in one string (use '\n' char)
#ifdef ENABLE_DTMF_CALLING
char Contact[16];
#endif
// clear the screen buffer
2024-01-05 09:04:33 +00:00
UI_DisplayClear();
2023-12-07 03:39:09 +00:00
#if 1
// original menu layout
// invert the current menu list item pixels反转当前菜单项的像素值
// draw vertical separating dotted line绘制垂直分隔的点线
// for (i = 0; i < 7; i++)
// gFrameBuffer[i][(8 * menu_list_width) + 1] = 0xAA;
// draw the little sub-menu triangle marker绘制子菜单三角标志
//const void *BITMAP_CurrentIndicator = BITMAP_MARKER;
if (gIsInSubMenu)
2023-12-17 04:23:45 +00:00
memmove(gFrameBuffer[2] + 40, BITMAP_VFO_Default, sizeof(BITMAP_VFO_Default));
2023-12-07 03:39:09 +00:00
// draw the menu index number/count绘制菜单索引号/总数
sprintf(String, "%2u/%u", 1 + gMenuCursor, gMenuListCount);
UI_PrintStringSmall(String, 2, 0, 6);
// UI_ShowChineseMenu();
UI_ShowChineseMenu();
#else
{ // new menu layout .. experimental & unfinished
const int menu_index = gMenuCursor; // current selected menu item
i = 1;
if (!gIsInSubMenu)
{
while (i < 2)
{ // leading menu items - small text
const int k = menu_index + i - 2;
if (k < 0)
UI_PrintStringSmall(MenuList[gMenuListCount + k].name, 0, 0, i); // wrap-a-round
else
if (k >= 0 && k < (int)gMenuListCount)
UI_PrintStringSmall(MenuList[k].name, 0, 0, i);
i++;
}
// current menu item - keep big n fat
if (menu_index >= 0 && menu_index < (int)gMenuListCount)
UI_PrintStringSmall(MenuList[menu_index].name, 0, 0, 2);
i++;
while (i < 4)
{ // trailing menu item - small text
const int k = menu_index + i - 2;
if (k >= 0 && k < (int)gMenuListCount)
UI_PrintStringSmall(MenuList[k].name, 0, 0, 1 + i);
else
if (k >= (int)gMenuListCount)
UI_PrintStringSmall(MenuList[gMenuListCount - k].name, 0, 0, 1 + i); // wrap-a-round
i++;
}
// draw the menu index number/count
sprintf(String, "%2u.%u", 1 + gMenuCursor, gMenuListCount);
UI_PrintStringSmall(String, 2, 0, 6);
}
else
if (menu_index >= 0 && menu_index < (int)gMenuListCount)
{ // current menu item
2024-01-05 09:04:33 +00:00
UI_PrintStringSmall(MenuList[menu_index].name, 0, 0, 0);
2023-12-07 03:39:09 +00:00
// UI_PrintStringSmall(String, 0, 0, 0);
}
}
#endif
// **************
memset(String, 0, sizeof(String));
bool already_printed = false;
2023-12-17 04:23:45 +00:00
/* Brightness is set to max in some entries of this menu. Return it to the configured brightness
level the "next" time we enter here.I.e., when we move from one menu to another.
It also has to be set back to max when pressing the Exit key. */
2023-12-07 03:39:09 +00:00
2023-12-17 04:23:45 +00:00
BACKLIGHT_TurnOn();
2023-12-07 03:39:09 +00:00
switch (UI_MENU_GetCurrentMenuId()) {
case MENU_SQL:
sprintf(String, "%d", gSubMenuSelection);
break;
case MENU_MIC: { // display the mic gain in actual dB rather than just an index number
const uint8_t mic = gMicGain_dB2[gSubMenuSelection];
sprintf(String, "+%u.%01udB", mic / 2, mic % 2);
}
break;
2024-01-05 09:04:33 +00:00
//#ifdef ENABLE_AUDIO_BAR
2023-12-07 03:39:09 +00:00
// case MENU_MIC_BAR:
// strcpy(String, gSubMenu_OFF_ON[gSubMenuSelection]);
// break;
2024-01-05 09:04:33 +00:00
//#endif
2023-12-07 03:39:09 +00:00
case MENU_STEP: {
uint16_t step = gStepFrequencyTable[FREQUENCY_GetStepIdxFromSortedIdx(gSubMenuSelection)];
sprintf(String, "%d.%02ukHz", step / 100, step % 100);
break;
}
// case MENU_TXP:
// strncpy(String, gSubMenu_TXP[gSubMenuSelection], sizeof(gSubMenu_TXP[gSubMenuSelection]));
// String[sizeof(gSubMenu_TXP[gSubMenuSelection])] = '\0';
//
//
// break;
case MENU_R_DCS:
case MENU_T_DCS:
if (gSubMenuSelection == 0)
//translate
#ifdef test
2024-01-05 09:04:33 +00:00
strcpy(String, "OFF");
2023-12-07 03:39:09 +00:00
#else
strcpy(String, );
#endif
else if (gSubMenuSelection < 105)
sprintf(String, "D%03oN", DCS_Options[gSubMenuSelection - 1]);
else
sprintf(String, "D%03oI", DCS_Options[gSubMenuSelection - 105]);
break;
case MENU_R_CTCS:
case MENU_T_CTCS: {
if (gSubMenuSelection == 0)
2024-01-05 09:04:33 +00:00
// translate
2023-12-07 03:39:09 +00:00
#ifdef test
2024-01-05 09:04:33 +00:00
strcpy(String, "OFF");
2023-12-07 03:39:09 +00:00
#else
2024-01-05 09:04:33 +00:00
//关闭
2023-12-07 03:39:09 +00:00
strcpy(String, );
#endif
else
sprintf(String, "%u.%uHz", CTCSS_Options[gSubMenuSelection - 1] / 10,
CTCSS_Options[gSubMenuSelection - 1] % 10);
break;
}
case MENU_SFT_D:
2024-01-05 09:04:33 +00:00
strncpy(String, gSubMenu_SFT_D[gSubMenuSelection], sizeof(gSubMenu_SFT_D[gSubMenuSelection]));
String[sizeof(gSubMenu_SFT_D[gSubMenuSelection])] = '\0';
2023-12-07 03:39:09 +00:00
break;
case MENU_OFFSET:
if (!gIsInSubMenu || gInputBoxIndex == 0) {
sprintf(String, "%3d.%05u", gSubMenuSelection / 100000, abs(gSubMenuSelection) % 100000);
UI_PrintStringSmall(String, menu_item_x1, menu_item_x2, 2);
} else {
const char *ascii = INPUTBOX_GetAscii();
sprintf(String, "%.3s.%.3s ", ascii, ascii + 3);
UI_PrintStringSmall(String, menu_item_x1, menu_item_x2, 2);
}
UI_PrintStringSmall("MHz", menu_item_x1, menu_item_x2, 4);
already_printed = true;
break;
// case MENU_W_N:
//
// strcpy(String, gSubMenu_W_N[gSubMenuSelection]);
// break;
2024-01-05 09:04:33 +00:00
2023-12-07 03:39:09 +00:00
case MENU_SCR:
strcpy(String, gSubMenu_SCRAMBLER[gSubMenuSelection]);
2024-01-05 09:04:33 +00:00
2023-12-07 03:39:09 +00:00
#if 1
2024-01-05 09:04:33 +00:00
// if (gSubMenuSelection > 0 && gSetting_ScrambleEnable)
if (gSubMenuSelection > 0)
2023-12-07 03:39:09 +00:00
BK4819_EnableScramble(gSubMenuSelection - 1);
else
BK4819_DisableScramble();
#endif
break;
case MENU_ABR:
strcpy(String, gSubMenu_BACKLIGHT[gSubMenuSelection]);
2024-01-05 09:04:33 +00:00
2023-12-07 03:39:09 +00:00
// BACKLIGHT_SetBrightness(-1);
break;
2024-01-05 09:04:33 +00:00
// case MENU_ABR_MIN:
2023-12-07 03:39:09 +00:00
case MENU_ABR_MAX:
sprintf(String, "%d", gSubMenuSelection);
if (gIsInSubMenu)
BACKLIGHT_SetBrightness(gSubMenuSelection);
// else
// BACKLIGHT_SetBrightness(-1);
break;
// case MENU_AM:
// strcpy(String, gModulationStr[gSubMenuSelection]);
//
// break;
#ifdef ENABLE_AM_FIX_TEST1
case MENU_AM_FIX_TEST1:
strcpy(String, gSubMenu_AM_fix_test1[gSubMenuSelection]);
// gSetting_AM_fix = gSubMenuSelection;
break;
#endif
case MENU_COMPAND:
strcpy(String, gSubMenu_RX_TX[gSubMenuSelection]);
break;
#ifdef ENABLE_AM_FIX
case MENU_AM_FIX:
#endif
case MENU_BCL:
2024-01-05 09:04:33 +00:00
// case MENU_BEEP:
2023-12-07 03:39:09 +00:00
// case MENU_S_ADD1:
// case MENU_S_ADD2:
case MENU_STE:
case MENU_D_ST:
#ifdef ENABLE_DTMF_CALLING
2024-01-05 09:04:33 +00:00
// case MENU_D_DCD:
2023-12-07 03:39:09 +00:00
#endif
case MENU_D_LIVE_DEC:
#ifdef ENABLE_NOAA
case MENU_NOAA_S:
#endif
// case MENU_350TX:
// case MENU_200TX:
// case MENU_500TX:
// case MENU_350EN:
2024-01-05 09:04:33 +00:00
strcpy(String, gSubMenu_OFF_ON[gSubMenuSelection]);
2023-12-07 03:39:09 +00:00
break;
// case MENU_SCREN:
// strcpy(String, gSubMenu_OFF_ON[gSubMenuSelection]);
//
//
// break;
case MENU_MEM_CH:
case MENU_1_CALL:
case MENU_DEL_CH: {
const bool valid = RADIO_CheckValidChannel(gSubMenuSelection, false, 1);
UI_GenerateChannelStringEx(String, valid, gSubMenuSelection);
2024-01-05 09:04:33 +00:00
UI_PrintStringSmall(String, menu_item_x1 - 12, menu_item_x2, 2);
2023-12-07 03:39:09 +00:00
if (valid && !gAskForConfirmation) { // show the frequency so that the user knows the channels frequency
const uint32_t frequency = SETTINGS_FetchChannelFrequency(gSubMenuSelection);
sprintf(String, "%u.%05u", frequency / 100000, frequency % 100000);
2024-01-05 09:04:33 +00:00
UI_PrintStringSmall(String, menu_item_x1 - 12, menu_item_x2, 5);
2023-12-07 03:39:09 +00:00
}
2024-01-05 09:04:33 +00:00
SETTINGS_FetchChannelName(String, gSubMenuSelection);
show_move_flag=1;
2024-01-05 09:04:33 +00:00
UI_PrintStringSmall(String[0] ? String : "--", menu_item_x1 - 12, menu_item_x2, 3);
2023-12-07 03:39:09 +00:00
already_printed = true;
break;
}
#ifdef ENABLE_MDC1200
2024-01-05 09:04:33 +00:00
// case MENU_MDC_ID:
// /// char mdc_id_str[4];
// sprintf(String, "%04X", gEeprom.MDC1200_ID); // %04X确保输出是4个字符长度的十六进制数
// // strcpy(String, id);
// break;
case MENU_MDC_ID:
{
if(gIsInSubMenu){ // show the channel name being edited
UI_PrintStringSmall(edit, menu_item_x1, menu_item_x2, 3);
if (edit_index < 4)
UI_PrintStringSmall("^", menu_item_x1+(((menu_item_x2 - menu_item_x1) - (28)) + 1) / 2 + (7 * edit_index), 0, 4); // show the cursor
}else
{
// sprintf(edit, "%04lX", gEeprom.MDC1200_ID); // %04X确保输出是4个字符长度的十六进制数
sprintf(String, "%04X", gEeprom.MDC1200_ID); // %04X确保输出是4个字符长度的十六进制数
edit_index = -1;
edit[0]=String[0];
edit[1]=String[1];
edit[2]=String[2];
edit[3]=String[3];
UI_PrintStringSmall(String, menu_item_x1, menu_item_x2, 3);//4
}
already_printed = true;
break;
}
2023-12-07 03:39:09 +00:00
#endif
case MENU_MEM_NAME: {
const bool valid = RADIO_CheckValidChannel(gSubMenuSelection, false, 1);
UI_GenerateChannelStringEx(String, valid, gSubMenuSelection);
2024-01-05 09:04:33 +00:00
UI_PrintStringSmall(String, menu_item_x1 - 12, menu_item_x2, 2);
2023-12-07 03:39:09 +00:00
if (valid) {
const uint32_t frequency = SETTINGS_FetchChannelFrequency(gSubMenuSelection);
//bug way
2024-01-05 09:04:33 +00:00
char tmp_name[17] = {0};
2023-12-23 14:44:35 +00:00
SETTINGS_FetchChannelName(tmp_name, gSubMenuSelection);
2023-12-07 03:39:09 +00:00
if (!gIsInSubMenu || edit_index < 0) { // show the channel name
SETTINGS_FetchChannelName(String, gSubMenuSelection);
2024-01-05 09:04:33 +00:00
char *pPrintStr = String[0] ? String : "--";
show_move_flag=1;
2024-01-05 09:04:33 +00:00
UI_PrintStringSmall(pPrintStr, menu_item_x1 - 12, menu_item_x2, 3);
} else if (!CHINESE_JUDGE(tmp_name, strlen(tmp_name))) { // show the channel name being edited
UI_PrintStringSmall(edit, menu_item_x1 - 12, 0, 3);
2023-12-23 14:44:35 +00:00
if (edit_index < MAX_EDIT_INDEX)
2024-01-05 09:04:33 +00:00
UI_PrintStringSmall("^", menu_item_x1 - 12 + (7 * edit_index), 0, 4); // show the cursor
} else if (CHINESE_JUDGE(tmp_name, strlen(tmp_name))) {
edit_index = -1;
2023-12-07 03:39:09 +00:00
}
if (!gAskForConfirmation) { // show the frequency so that the user knows the channels frequency
sprintf(String, "%u.%05u", frequency / 100000, frequency % 100000);
2024-01-05 09:04:33 +00:00
UI_PrintStringSmall(String, menu_item_x1 - 12, menu_item_x2, 5);
2023-12-07 03:39:09 +00:00
}
}
already_printed = true;
break;
}
case MENU_SAVE:
strcpy(String, gSubMenu_SAVE[gSubMenuSelection]);
2024-01-05 09:04:33 +00:00
2023-12-07 03:39:09 +00:00
break;
case MENU_TDR:
strcpy(String, gSubMenu_RXMode[gSubMenuSelection]);
2024-01-05 09:04:33 +00:00
2023-12-07 03:39:09 +00:00
break;
case MENU_TOT:
strcpy(String, gSubMenu_TOT[gSubMenuSelection]);
2024-01-05 09:04:33 +00:00
2023-12-07 03:39:09 +00:00
break;
#ifdef ENABLE_VOICE
case MENU_VOICE:
strcpy(String, gSubMenu_VOICE[gSubMenuSelection]);
break;
#endif
case MENU_SC_REV:
strcpy(String, gSubMenu_SC_REV[gSubMenuSelection]);
break;
case MENU_MDF:
2024-01-05 09:04:33 +00:00
strcpy(String, gSubMenu_MDF[gSubMenuSelection]);
2023-12-07 03:39:09 +00:00
break;
case MENU_RP_STE:
if (gSubMenuSelection == 0)
//translate
#ifdef test
2024-01-05 09:04:33 +00:00
strcpy(String, "OFF");
2023-12-07 03:39:09 +00:00
#else
2024-01-05 09:04:33 +00:00
//关闭
2023-12-07 03:39:09 +00:00
strcpy(String, );
#endif
else
sprintf(String, "%d*100ms", gSubMenuSelection);
break;
case MENU_S_LIST:
if (gSubMenuSelection < 2)
2024-01-05 09:04:33 +00:00
//translate
2023-12-07 03:39:09 +00:00
#ifdef test
2024-01-05 09:04:33 +00:00
sprintf(String, "list %u", 1 + gSubMenuSelection);
2023-12-07 03:39:09 +00:00
#else //!!列表
2023-12-09 03:25:53 +00:00
sprintf(String, " %u", 1 + gSubMenuSelection);
2023-12-07 03:39:09 +00:00
#endif
else
#ifdef test
2024-01-05 09:04:33 +00:00
strcpy(String, "ALL");
2023-12-07 03:39:09 +00:00
#else
2024-01-05 09:04:33 +00:00
//全部
strcpy(String, );
2023-12-07 03:39:09 +00:00
#endif
break;
#ifdef ENABLE_ALARM
case MENU_AL_MOD:
sprintf(String, gSubMenu_AL_MOD[gSubMenuSelection]);
break;
#endif
#ifdef ENABLE_DTMF_CALLING
2024-01-05 09:04:33 +00:00
case MENU_ANI_ID:
2023-12-07 03:39:09 +00:00
2024-01-05 09:04:33 +00:00
strcpy(String, gEeprom.ANI_DTMF_ID);
break;
2023-12-07 03:39:09 +00:00
#endif
case MENU_UPCODE:
sprintf(String, "%.8s\n%.8s", gEeprom.DTMF_UP_CODE, gEeprom.DTMF_UP_CODE + 8);
break;
case MENU_DWCODE:
sprintf(String, "%.8s\n%.8s", gEeprom.DTMF_DOWN_CODE, gEeprom.DTMF_DOWN_CODE + 8);
break;
#ifdef ENABLE_DTMF_CALLING
2024-01-05 09:04:33 +00:00
case MENU_D_RSP:
strcpy(String, gSubMenu_D_RSP[gSubMenuSelection]);
2023-12-07 03:39:09 +00:00
2024-01-05 09:04:33 +00:00
break;
case MENU_D_HOLD:
sprintf(String, "%ds", gSubMenuSelection);
break;
2023-12-07 03:39:09 +00:00
#endif
case MENU_D_PRE:
sprintf(String, "%d*10ms", gSubMenuSelection);
break;
case MENU_PTT_ID:
strcpy(String, gSubMenu_PTT_ID[gSubMenuSelection]);
2024-01-05 09:04:33 +00:00
2023-12-07 03:39:09 +00:00
break;
// case MENU_BAT_TXT:
// strcpy(String, gSubMenu_BAT_TXT[gSubMenuSelection]);
//
//
// break;
#ifdef ENABLE_DTMF_CALLING
2024-01-05 09:04:33 +00:00
case MENU_D_LIST:
gIsDtmfContactValid = DTMF_GetContact((int) gSubMenuSelection - 1, Contact);
if (!gIsDtmfContactValid)
strcpy(String, "NULL");
else
memcpy(String, Contact, 8);
break;
2023-12-07 03:39:09 +00:00
#endif
2024-01-08 03:46:18 +00:00
#if ENABLE_CHINESE_FULL==4
2023-12-07 03:39:09 +00:00
2024-01-08 03:46:18 +00:00
case MENU_PONMSG:
strcpy(String, gSubMenu_PONMSG[gSubMenuSelection]);
break;
#endif
2023-12-07 03:39:09 +00:00
case MENU_ROGER:
strcpy(String, gSubMenu_ROGER[gSubMenuSelection]);
2024-01-05 09:04:33 +00:00
2023-12-07 03:39:09 +00:00
break;
// case MENU_VOL:
// sprintf(String, "%u.%02uV\n%u%%",
// gBatteryVoltageAverage / 100, gBatteryVoltageAverage % 100,
// BATTERY_VoltsToPercent(gBatteryVoltageAverage));
// break;
case MENU_RESET:
strcpy(String, gSubMenu_RESET[gSubMenuSelection]);
2024-01-05 09:04:33 +00:00
2023-12-07 03:39:09 +00:00
break;
case MENU_F_LOCK:
// if (!gIsInSubMenu && gUnlockAllTxConfCnt > 0&& gUnlockAllTxConfCnt < 10)
// strcpy(String, "READ\nMANUAL");
//
// else
2024-01-05 09:04:33 +00:00
strcpy(String, gSubMenu_F_LOCK[gSubMenuSelection]);
2023-12-07 03:39:09 +00:00
break;
#ifdef ENABLE_F_CAL_MENU
case MENU_F_CALI:
{
const uint32_t value = 22656 + gSubMenuSelection;
const uint32_t xtal_Hz = (0x4f0000u + value) * 5;
writeXtalFreqCal(gSubMenuSelection, false);
sprintf(String, "%d\n%u.%06u\nMHz",
gSubMenuSelection,
xtal_Hz / 1000000, xtal_Hz % 1000000);
}
break;
#endif
case MENU_BATCAL: {
const uint16_t vol = (uint32_t) gBatteryVoltageAverage * gBatteryCalibration[3] / gSubMenuSelection;
sprintf(String, "%u.%02uV\n%u", vol / 100, vol % 100, gSubMenuSelection);
break;
}
case MENU_BATTYP:
strcpy(String, gSubMenu_BATTYP[gSubMenuSelection]);
break;
// case MENU_F1SHRT:
// case MENU_F1LONG:
// case MENU_F2SHRT:
// case MENU_F2LONG:
// case MENU_MLONG:
// strcpy(String, gSubMenu_SIDEFUNCTIONS[gSubMenuSelection].name);
// break;
}
if (!already_printed) { // we now do multi-line text in a single string
unsigned int y;
unsigned int lines = 1;
unsigned int len = strlen(String);
bool small = false;
if (len > 0) {
// count number of lines
for (i = 0; i < len; i++) {
if (String[i] == '\n' && i < (len - 1)) { // found new line char
2024-01-05 09:04:33 +00:00
lines += 1;
2023-12-07 03:39:09 +00:00
String[i] = 0; // null terminate the line
}
}
if (lines > 3) { // use small text
small = true;
if (lines > 7)
lines = 7;
}
// center vertically'ish
if (small)
y = 3 - ((lines + 0) / 2); // untested
else
y = 2 - ((lines + 0) / 2);
// draw the text lines
for (i = 0; i < len && lines > 0; lines--) {
if (small)
UI_PrintStringSmall(String + i, menu_item_x1, menu_item_x2, y + 1);
else
UI_PrintStringSmall(String + i, menu_item_x1, menu_item_x2, y + 1);
// look for start of next line
2024-01-05 09:04:33 +00:00
while (i < len && String[i] != 0 && String[i] != '\n')
2023-12-07 03:39:09 +00:00
i++;
// hop over the null term char(s)
2024-01-05 09:04:33 +00:00
while (i < len && (String[i] == 0 || String[i] == '\n'))
2023-12-07 03:39:09 +00:00
i++;
y += small ? 1 : 2;
}
}
}
if (UI_MENU_GetCurrentMenuId() == MENU_SLIST1 || UI_MENU_GetCurrentMenuId() == MENU_SLIST2) {
i = (UI_MENU_GetCurrentMenuId() == MENU_SLIST1) ? 0 : 1;
2024-01-05 09:04:33 +00:00
char *pPrintStr = String;
if (gSubMenuSelection < 0) {
pPrintStr = "NULL";
} else {
2023-12-07 03:39:09 +00:00
UI_GenerateChannelStringEx(String, true, gSubMenuSelection);
2024-01-05 09:04:33 +00:00
pPrintStr = String;
}
2023-12-07 03:39:09 +00:00
2024-01-05 09:04:33 +00:00
// channel number
UI_PrintStringSmall(pPrintStr, menu_item_x1 - 12, menu_item_x2, 2);
2023-12-07 03:39:09 +00:00
show_move_flag=1;
2024-01-05 09:04:33 +00:00
SETTINGS_FetchChannelName(String, gSubMenuSelection);
pPrintStr = String[0] ? String : "--";
// channel name and scan-list
if (gSubMenuSelection < 0 || !gEeprom.SCAN_LIST_ENABLED[i]) {
UI_PrintStringSmall(pPrintStr, menu_item_x1 - 12, menu_item_x2, 4);
2023-12-07 03:39:09 +00:00
} else {
2024-01-05 09:04:33 +00:00
UI_PrintStringSmall(pPrintStr, menu_item_x1 - 12, menu_item_x2, 4);
2023-12-07 03:39:09 +00:00
if (IS_MR_CHANNEL(gEeprom.SCANLIST_PRIORITY_CH1[i])) {
2024-01-05 09:04:33 +00:00
sprintf(String, "PRI%d:%u", 1, gEeprom.SCANLIST_PRIORITY_CH1[i] + 1);
UI_PrintStringSmall(String, menu_item_x1 - 12, menu_item_x2, 3);
2023-12-07 03:39:09 +00:00
}
2024-01-05 09:04:33 +00:00
// if (IS_MR_CHANNEL(gEeprom.SCANLIST_PRIORITY_CH2[i])) {
// sprintf(String, "PRI2:%u", gEeprom.SCANLIST_PRIORITY_CH2[i] + 1);
2023-12-07 03:39:09 +00:00
if (IS_MR_CHANNEL(gEeprom.SCANLIST_PRIORITY_CH2[i])) {
2024-01-05 09:04:33 +00:00
sprintf(String, "PRI%d:%u", 2, gEeprom.SCANLIST_PRIORITY_CH2[i] + 1);
UI_PrintStringSmall(String, menu_item_x1 - 12, menu_item_x2, 6);
2023-12-07 03:39:09 +00:00
}
}
}
if ((UI_MENU_GetCurrentMenuId() == MENU_R_CTCS || UI_MENU_GetCurrentMenuId() == MENU_R_DCS) && gCssBackgroundScan)
//扫描
UI_PrintStringSmall(, menu_item_x1, menu_item_x2, 5);
//
// if (UI_MENU_GetCurrentMenuId() == MENU_UPCODE)
// if (strlen(gEeprom.DTMF_UP_CODE) > 12)
// UI_PrintStringSmall(gEeprom.DTMF_UP_CODE + 12, menu_item_x1, menu_item_x2, 5);
//
// if (UI_MENU_GetCurrentMenuId() == MENU_DWCODE)
// if (strlen(gEeprom.DTMF_DOWN_CODE) > 12)
// UI_PrintStringSmall(gEeprom.DTMF_DOWN_CODE + 12, menu_item_x1, menu_item_x2, 5);
#ifdef ENABLE_DTMF_CALLING
if (UI_MENU_GetCurrentMenuId() == MENU_D_LIST && gIsDtmfContactValid) {
2024-01-05 09:04:33 +00:00
2023-12-07 03:39:09 +00:00
Contact[11] = 0;
2023-12-17 04:23:45 +00:00
memcpy(&gDTMF_ID, Contact + 8, 4);
2024-01-05 09:04:33 +00:00
sprintf(String, "ID:%4s", gDTMF_ID);
2023-12-07 03:39:09 +00:00
UI_PrintStringSmall(String, menu_item_x1, menu_item_x2, 5);
}
#endif
if (UI_MENU_GetCurrentMenuId() == MENU_R_CTCS ||
UI_MENU_GetCurrentMenuId() == MENU_T_CTCS ||
UI_MENU_GetCurrentMenuId() == MENU_R_DCS ||
2024-01-05 09:04:33 +00:00
UI_MENU_GetCurrentMenuId() == MENU_T_DCS
2023-12-07 03:39:09 +00:00
#ifdef ENABLE_DTMF_CALLING
|| UI_MENU_GetCurrentMenuId() == MENU_D_LIST
#endif
2024-01-05 09:04:33 +00:00
) {
2023-12-07 03:39:09 +00:00
sprintf(String, "%2d", gSubMenuSelection);
UI_PrintStringSmall(String, 105, 0, 1);//small
}
if ((UI_MENU_GetCurrentMenuId() == MENU_RESET ||
UI_MENU_GetCurrentMenuId() == MENU_MEM_CH ||
UI_MENU_GetCurrentMenuId() == MENU_MEM_NAME ||
2024-01-05 09:04:33 +00:00
UI_MENU_GetCurrentMenuId() == MENU_MDC_ID ||
2023-12-07 03:39:09 +00:00
UI_MENU_GetCurrentMenuId() == MENU_DEL_CH) && gAskForConfirmation) { // display confirmation
2024-01-05 09:04:33 +00:00
char *pPrintStr = (gAskForConfirmation == 1) ? "SURE?" : "WAIT!";
if ((UI_MENU_GetCurrentMenuId() == MENU_MEM_CH || UI_MENU_GetCurrentMenuId() == MENU_MEM_NAME ||
UI_MENU_GetCurrentMenuId() == MENU_DEL_CH) && gAskForConfirmation)
UI_PrintStringSmall(pPrintStr, menu_item_x1 - 12, menu_item_x2, 5);
else UI_PrintStringSmall(pPrintStr, menu_item_x1, menu_item_x2, 5);
2023-12-23 14:44:35 +00:00
2024-01-05 09:04:33 +00:00
gRequestSaveSettings = 1;
2023-12-07 03:39:09 +00:00
}
ST7565_BlitFullScreen();
}
//
void UI_ShowChineseMenu() {
2024-01-05 09:04:33 +00:00
2023-12-07 03:39:09 +00:00
uint8_t size_menu = 0;
2024-01-05 09:04:33 +00:00
uint8_t cnt_menu = 0;
for (cnt_menu = 0; cnt_menu < 7 && MenuList[gMenuCursor].name[cnt_menu] != 0; cnt_menu++) {
if (is_chn(MenuList[gMenuCursor].name[cnt_menu]) != 255)//中文
{
size_menu += 12;
#if ENABLE_CHINESE_FULL != 0
cnt_menu++;
#endif
} else//英文
{
size_menu += 7;
}
}
2023-12-07 03:39:09 +00:00
show_move_flag = 1;
2023-12-07 03:39:09 +00:00
2024-01-05 09:04:33 +00:00
UI_PrintStringSmall(MenuList[gMenuCursor].name, size_menu < 48 ? (48 - size_menu) / 2 : 0, 0, 0);
2023-12-07 03:39:09 +00:00
2023-11-30 06:38:27 +00:00
}