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>
|
2023-12-20 15:20:23 +00:00
|
|
|
#include "driver/uart.h"
|
2023-12-07 03:39:09 +00:00
|
|
|
#include "driver/st7565.h"
|
|
|
|
#include "external/printf/printf.h"
|
|
|
|
#include "font.h"
|
|
|
|
#include "ui/helper.h"
|
|
|
|
#include "ui/inputbox.h"
|
|
|
|
#include "misc.h"
|
|
|
|
#include "chinese.h"
|
2023-12-20 15:20:23 +00:00
|
|
|
#include "driver/eeprom.h"
|
2024-01-05 09:04:33 +00:00
|
|
|
|
2023-12-07 03:39:09 +00:00
|
|
|
#ifndef ARRAY_SIZE
|
|
|
|
#define ARRAY_SIZE(arr) (sizeof(arr)/sizeof((arr)[0]))
|
|
|
|
#endif
|
|
|
|
#define IS_BIT_SET(byte, bit) ((byte>>bit) & (1))
|
|
|
|
|
|
|
|
void set_bit(uint8_t *value, uint8_t bit_position, uint8_t bit_value) {
|
|
|
|
if (bit_value == 0) {
|
2023-12-17 04:23:45 +00:00
|
|
|
*value = *value & ~(1 << bit_position);
|
2023-12-07 03:39:09 +00:00
|
|
|
} else {
|
2023-12-17 04:23:45 +00:00
|
|
|
*value = *value | (1 << bit_position);
|
2023-12-07 03:39:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t is_chn(uint8_t num) {
|
2024-01-05 09:04:33 +00:00
|
|
|
#if ENABLE_CHINESE_FULL != 0
|
2023-12-20 15:20:23 +00:00
|
|
|
if (num>=0x80) return 1;
|
|
|
|
return 255;
|
|
|
|
#else
|
2023-12-07 03:39:09 +00:00
|
|
|
if (num >= 1 && num < 10)return num - 1;
|
|
|
|
else if (num > 10 && num < 32)return num - 2;
|
2024-01-13 12:00:44 +00:00
|
|
|
else if (num > 126 && num <= 233)return num - 97;
|
2023-12-07 03:39:09 +00:00
|
|
|
else return 255;
|
2023-12-20 15:20:23 +00:00
|
|
|
#endif
|
2023-12-07 03:39:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void UI_GenerateChannelString(char *pString, const uint8_t Channel) {
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
if (gInputBoxIndex == 0) {
|
|
|
|
sprintf(pString, "CH-%02u", Channel + 1);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
pString[0] = 'C';
|
|
|
|
pString[1] = 'H';
|
|
|
|
pString[2] = '-';
|
|
|
|
for (i = 0; i < 2; i++)
|
|
|
|
pString[i + 3] = (gInputBox[i] == 10) ? '-' : gInputBox[i] + '0';
|
|
|
|
}
|
2024-01-05 09:04:33 +00:00
|
|
|
|
|
|
|
bool CHINESE_JUDGE(char *name, uint8_t len) {
|
|
|
|
for (int i = 0; i < len; i++)
|
|
|
|
if (name[i] >= 0x80 && name[i] <= 0x9b && i != len - 1 && name[i + 1] != 0)return 1;
|
2023-12-07 03:39:09 +00:00
|
|
|
|
2023-12-23 14:44:35 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2024-01-05 09:04:33 +00:00
|
|
|
|
2023-12-07 03:39:09 +00:00
|
|
|
void UI_GenerateChannelStringEx(char *pString, const bool bShowPrefix, const uint8_t ChannelNumber) {
|
|
|
|
if (gInputBoxIndex > 0) {
|
2024-01-05 09:04:33 +00:00
|
|
|
for (unsigned int i = 0; i < 3; i++) {
|
2023-12-07 03:39:09 +00:00
|
|
|
pString[i] = (gInputBox[i] == 10) ? '-' : gInputBox[i] + '0';
|
2024-01-05 09:04:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pString[3] = 0;
|
2023-12-07 03:39:09 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-01-05 09:04:33 +00:00
|
|
|
if (bShowPrefix) {
|
|
|
|
// BUG here? Prefixed NULLs are allowed
|
2023-12-07 03:39:09 +00:00
|
|
|
sprintf(pString, "CH-%03u", ChannelNumber + 1);
|
2024-01-05 09:04:33 +00:00
|
|
|
} else if (ChannelNumber == 0xFF) {
|
2023-12-07 03:39:09 +00:00
|
|
|
strcpy(pString, "NULL");
|
2024-01-05 09:04:33 +00:00
|
|
|
} else {
|
2023-12-07 03:39:09 +00:00
|
|
|
sprintf(pString, "%03u", ChannelNumber + 1);
|
2024-01-05 09:04:33 +00:00
|
|
|
}
|
2023-12-07 03:39:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Example usage:
|
|
|
|
// UI_PrintChar('A', 0, 0, 16);
|
|
|
|
|
|
|
|
// Example usage:
|
|
|
|
// UI_PrintChar('A', 0, 0, 8);
|
|
|
|
|
|
|
|
void UI_PrintCharSmall(char character, uint8_t Start, uint8_t Line) {
|
|
|
|
const uint8_t char_width = ARRAY_SIZE(gFontSmall[0]);
|
|
|
|
|
|
|
|
// Calculate the position for the character
|
|
|
|
uint8_t *pFb = gFrameBuffer[Line] + Start + (char_width + 1) / 2;
|
|
|
|
|
|
|
|
// Display the character if it's a printable character
|
|
|
|
if (character > ' ') {
|
2023-12-17 04:23:45 +00:00
|
|
|
const unsigned int index = (unsigned int) character - ' ' - 1;
|
2023-12-07 03:39:09 +00:00
|
|
|
if (index < ARRAY_SIZE(gFontSmall)) {
|
|
|
|
memmove(pFb, &gFontSmall[index], char_width);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void UI_PrintStringSmall(const char *pString, uint8_t Start, uint8_t End, uint8_t Line) {
|
2024-01-05 09:04:33 +00:00
|
|
|
bool flag_move = 0;
|
2023-12-22 09:48:18 +00:00
|
|
|
|
2023-12-07 03:39:09 +00:00
|
|
|
uint8_t Length = strlen(pString);
|
2024-01-05 09:04:33 +00:00
|
|
|
|
2024-01-09 09:17:26 +00:00
|
|
|
if (show_move_flag) {
|
2023-12-24 10:15:10 +00:00
|
|
|
#if ENABLE_CHINESE_FULL == 0
|
|
|
|
Length = Length > 7 ? 7 : Length;
|
2024-01-05 09:04:33 +00:00
|
|
|
#else
|
|
|
|
Length = Length > 14 ? 14 : Length;
|
|
|
|
|
|
|
|
#endif
|
|
|
|
flag_move = 1;
|
2024-01-09 09:17:26 +00:00
|
|
|
show_move_flag = 0;
|
2024-01-05 09:04:33 +00:00
|
|
|
|
2023-12-24 10:15:10 +00:00
|
|
|
}
|
2023-12-07 03:39:09 +00:00
|
|
|
uint8_t sum_pixel = 0;
|
2023-12-20 15:20:23 +00:00
|
|
|
uint16_t true_char[Length];
|
|
|
|
uint8_t cn_flag[Length];
|
2024-01-05 09:04:33 +00:00
|
|
|
uint8_t char_num = 0;
|
2023-12-07 03:39:09 +00:00
|
|
|
for (size_t j = 0; j < Length; j++) {
|
|
|
|
|
2024-01-05 09:04:33 +00:00
|
|
|
uint8_t chn_judge = is_chn(pString[j]);
|
2023-12-20 15:20:23 +00:00
|
|
|
if (chn_judge == 255 && pString[j] != '\n' && pString[j] != '\0') {
|
2024-01-05 09:04:33 +00:00
|
|
|
true_char[char_num] = pString[j];
|
|
|
|
cn_flag[char_num] = 0;
|
2023-12-20 15:20:23 +00:00
|
|
|
|
|
|
|
char_num++;
|
|
|
|
sum_pixel += 7;
|
2024-01-05 09:04:33 +00:00
|
|
|
} else if (chn_judge != 255) {
|
|
|
|
cn_flag[char_num] = 1;
|
2023-12-20 15:20:23 +00:00
|
|
|
|
2024-01-05 09:04:33 +00:00
|
|
|
#if ENABLE_CHINESE_FULL != 0
|
2023-12-20 15:20:23 +00:00
|
|
|
|
|
|
|
true_char[char_num]=(pString[j]<<8)|pString[j+1];
|
|
|
|
j++;
|
|
|
|
#else
|
2024-01-05 09:04:33 +00:00
|
|
|
true_char[char_num] = chn_judge;
|
2023-12-20 15:20:23 +00:00
|
|
|
#endif
|
|
|
|
|
2023-12-17 10:27:47 +00:00
|
|
|
flag_move = 1;
|
2023-12-20 15:20:23 +00:00
|
|
|
char_num++;
|
2023-12-19 03:10:48 +00:00
|
|
|
sum_pixel += 13;
|
2023-12-17 10:27:47 +00:00
|
|
|
}
|
2023-12-07 03:39:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (End > Start)
|
|
|
|
Start += (((End - Start) - (sum_pixel)) + 1) / 2;
|
|
|
|
// if(Start+sum_pixel>=128)Start=128-sum_pixel;
|
|
|
|
uint8_t *pFb = gFrameBuffer[Line] + Start;
|
|
|
|
uint8_t *pFb1 = gFrameBuffer[Line + 1] + Start;
|
|
|
|
uint8_t now_pixel = 0;
|
2024-01-05 09:04:33 +00:00
|
|
|
for (unsigned short i = 0; i < char_num; i++) {
|
2023-12-20 15:20:23 +00:00
|
|
|
if (cn_flag[i] == 0) {
|
|
|
|
if (true_char[i] > ' ') {
|
|
|
|
const unsigned int index = (unsigned int) true_char[i] - ' ' - 1;
|
2023-12-07 03:39:09 +00:00
|
|
|
if (index < ARRAY_SIZE(gFontSmall)) {
|
2023-12-17 10:27:47 +00:00
|
|
|
if (flag_move) {
|
|
|
|
uint8_t gFontSmall_More[12] = {0};
|
|
|
|
for (int j = 0; j < 12; ++j) {
|
|
|
|
if (j < 6) {
|
2024-01-11 13:38:54 +00:00
|
|
|
gFontSmall_More[j] = (gFontSmall[index][j] & 0x1F) << 3;//00011111
|
2023-12-17 10:27:47 +00:00
|
|
|
} else {
|
2024-01-11 13:38:54 +00:00
|
|
|
gFontSmall_More[j] = (gFontSmall[index][j - 6] & 0XE0) >> 5|(0xFB& *(pFb1+ now_pixel + 1+j-6));//11100000
|
|
|
|
//
|
2023-12-17 10:27:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
memcpy(pFb + now_pixel + 1, &gFontSmall_More[0], 6);
|
|
|
|
memcpy(pFb1 + now_pixel + 1, &gFontSmall_More[6], 6);
|
|
|
|
} else
|
|
|
|
memcpy(pFb + now_pixel + 1, &gFontSmall[index], 6);
|
2023-12-07 03:39:09 +00:00
|
|
|
}
|
|
|
|
now_pixel += 7;
|
2023-12-17 04:23:45 +00:00
|
|
|
} else if (pString[i] == ' ')
|
2023-12-07 03:39:09 +00:00
|
|
|
now_pixel += 7;
|
|
|
|
} else {
|
|
|
|
uint8_t gFontChinese[22] = {0};
|
2023-12-20 15:20:23 +00:00
|
|
|
|
2024-01-05 09:04:33 +00:00
|
|
|
#if ENABLE_CHINESE_FULL != 0
|
2023-12-20 15:20:23 +00:00
|
|
|
uint8_t tmp[17] = {0};
|
|
|
|
true_char[i]-=0x8000;
|
2023-12-22 09:48:18 +00:00
|
|
|
true_char[i]=true_char[i]-true_char[i]/256-1;
|
2023-12-20 15:20:23 +00:00
|
|
|
unsigned int local = (CHN_FONT_HIGH * CHN_FONT_WIDTH * true_char[i]) / 8;
|
|
|
|
unsigned int local_bit =(CHN_FONT_HIGH * CHN_FONT_WIDTH * true_char[i]) % 8;
|
|
|
|
EEPROM_ReadBuffer(local+0x2000,tmp,17);
|
|
|
|
local=0;
|
2024-01-05 09:04:33 +00:00
|
|
|
for (unsigned char k = 0; k < CHN_FONT_WIDTH * 2; ++k) {
|
2023-12-20 15:20:23 +00:00
|
|
|
unsigned char j_end = 8;
|
2024-01-05 09:04:33 +00:00
|
|
|
if (k >= CHN_FONT_WIDTH)
|
2023-12-20 15:20:23 +00:00
|
|
|
j_end = CHN_FONT_HIGH - 8;
|
|
|
|
for (unsigned char j = 0; j < j_end; ++j) {
|
|
|
|
if (IS_BIT_SET(tmp[local], local_bit))
|
2024-01-05 09:04:33 +00:00
|
|
|
set_bit(&gFontChinese[k], j, 1);
|
2023-12-20 15:20:23 +00:00
|
|
|
local_bit++;
|
|
|
|
if (local_bit == 8) {
|
|
|
|
local_bit = 0;
|
|
|
|
local++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
unsigned int local = (CHN_FONT_HIGH * CHN_FONT_WIDTH * true_char[i]) / 8;
|
2024-01-05 09:04:33 +00:00
|
|
|
unsigned int local_bit = (CHN_FONT_HIGH * CHN_FONT_WIDTH * true_char[i]) % 8;
|
|
|
|
for (unsigned char k = 0; k < CHN_FONT_WIDTH * 2; ++k) {
|
2023-12-17 04:23:45 +00:00
|
|
|
unsigned char j_end = 8;
|
2024-01-05 09:04:33 +00:00
|
|
|
if (k >= CHN_FONT_WIDTH)
|
2023-12-17 04:23:45 +00:00
|
|
|
j_end = CHN_FONT_HIGH - 8;
|
|
|
|
for (unsigned char j = 0; j < j_end; ++j) {
|
|
|
|
if (IS_BIT_SET(gFontChinese_out[local], local_bit))
|
2024-01-05 09:04:33 +00:00
|
|
|
set_bit(&gFontChinese[k], j, 1);
|
2023-12-17 04:23:45 +00:00
|
|
|
local_bit++;
|
|
|
|
if (local_bit == 8) {
|
|
|
|
local_bit = 0;
|
|
|
|
local++;
|
|
|
|
}
|
2023-12-07 03:39:09 +00:00
|
|
|
}
|
|
|
|
}
|
2023-12-20 15:20:23 +00:00
|
|
|
#endif
|
2023-12-17 04:23:45 +00:00
|
|
|
memcpy(pFb + now_pixel + 1, &gFontChinese[0], 11);
|
|
|
|
memcpy(pFb1 + now_pixel + 1, &gFontChinese[11], 11);
|
2023-12-19 03:10:48 +00:00
|
|
|
now_pixel += 13;
|
2023-12-07 03:39:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void UI_PrintStringSmallBuffer(const char *pString, uint8_t *buffer) {
|
|
|
|
size_t i;
|
|
|
|
const unsigned int char_width = ARRAY_SIZE(gFontSmall[0]);
|
|
|
|
for (i = 0; i < strlen(pString); i++) {
|
|
|
|
if (pString[i] > ' ') {
|
|
|
|
const unsigned int index = (unsigned int) pString[i] - ' ' - 1;
|
|
|
|
if (index < ARRAY_SIZE(gFontSmall))
|
2023-12-17 04:23:45 +00:00
|
|
|
memcpy(buffer + (i * (char_width + 1)) + 1, &gFontSmall[index], char_width);
|
2023-12-07 03:39:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void UI_DisplayFrequency(const char *string, uint8_t X, uint8_t Y, bool center) {
|
|
|
|
const unsigned int char_width = 13;
|
|
|
|
uint8_t *pFb0 = gFrameBuffer[Y] + X;
|
|
|
|
uint8_t *pFb1 = pFb0 + 128;
|
|
|
|
bool bCanDisplay = false;
|
|
|
|
|
|
|
|
uint8_t len = strlen(string);
|
|
|
|
for (int i = 0; i < len; i++) {
|
|
|
|
char c = string[i];
|
|
|
|
if (c == '-') c = '9' + 1;
|
|
|
|
if (bCanDisplay || c != ' ') {
|
|
|
|
bCanDisplay = true;
|
|
|
|
if (c >= '0' && c <= '9' + 1) {
|
|
|
|
memcpy(pFb0 + 2, gFontBigDigits[c - '0'], char_width - 3);
|
|
|
|
memcpy(pFb1 + 2, gFontBigDigits[c - '0'] + char_width - 3, char_width - 3);
|
|
|
|
} else if (c == '.') {
|
|
|
|
*pFb1 = 0x60;
|
|
|
|
pFb0++;
|
|
|
|
pFb1++;
|
|
|
|
*pFb1 = 0x60;
|
|
|
|
pFb0++;
|
|
|
|
pFb1++;
|
|
|
|
*pFb1 = 0x60;
|
|
|
|
pFb0++;
|
|
|
|
pFb1++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
} else if (center) {
|
|
|
|
pFb0 -= 6;
|
|
|
|
pFb1 -= 6;
|
|
|
|
}
|
|
|
|
pFb0 += char_width;
|
|
|
|
pFb1 += char_width;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-12-17 04:23:45 +00:00
|
|
|
void UI_DrawPixelBuffer(uint8_t (*buffer)[128], uint8_t x, uint8_t y, bool black) {
|
2023-12-10 02:32:39 +00:00
|
|
|
const uint8_t pattern = 1 << (y % 8);
|
2023-12-17 04:23:45 +00:00
|
|
|
if (black)
|
|
|
|
buffer[y / 8][x] |= pattern;
|
2023-12-07 03:39:09 +00:00
|
|
|
else
|
2023-12-17 04:23:45 +00:00
|
|
|
buffer[y / 8][x] &= ~pattern;
|
2023-12-07 03:39:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void UI_DisplayPopup(const char *string) {
|
|
|
|
for (uint8_t i = 0; i < 7; i++) {
|
|
|
|
memset(gFrameBuffer[i], 0x00, 128);
|
|
|
|
}
|
|
|
|
|
|
|
|
// for(uint8_t i = 1; i < 5; i++) {
|
|
|
|
// memset(gFrameBuffer[i]+8, 0x00, 111);
|
|
|
|
// }
|
|
|
|
|
|
|
|
// for(uint8_t x = 10; x < 118; x++) {
|
|
|
|
// UI_DrawPixel(x, 10, true);
|
|
|
|
// UI_DrawPixel(x, 46-9, true);
|
|
|
|
// }
|
|
|
|
|
|
|
|
// for(uint8_t y = 11; y < 37; y++) {
|
|
|
|
// UI_DrawPixel(10, y, true);
|
|
|
|
// UI_DrawPixel(117, y, true);
|
|
|
|
// }
|
|
|
|
// DrawRectangle(9,9, 118,38, true);
|
|
|
|
|
|
|
|
UI_PrintStringSmall(string, 9, 118, 2);
|
|
|
|
//按EXIT键
|
2023-12-20 15:20:23 +00:00
|
|
|
UI_PrintStringSmall(按EXIT键, 9, 118, 5);
|
2024-01-05 09:04:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void UI_DisplayClear() {
|
|
|
|
memset(gFrameBuffer, 0, sizeof(gFrameBuffer));
|
2023-11-30 06:38:27 +00:00
|
|
|
}
|