uv-k5-firmware-chinese-lts/ui/helper.c
2023-12-22 17:48:18 +08:00

328 lines
No EOL
10 KiB
C

/* Copyright 2023 Dual Tachyon
* https://github.com/DualTachyon
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <string.h>
#include "driver/uart.h"
#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"
#include "driver/eeprom.h"
#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) {
*value = *value & ~(1 << bit_position);
} else {
*value = *value | (1 << bit_position);
}
}
uint8_t is_chn(uint8_t num) {
#if ENABLE_CHINESE_FULL==4
if (num>=0x80) return 1;
return 255;
#else
if (num >= 1 && num < 10)return num - 1;
else if (num > 10 && num < 32)return num - 2;
else if (num > 126 && num <= 218)return num - 97;
else return 255;
#endif
}
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';
}
void UI_GenerateChannelStringEx(char *pString, const bool bShowPrefix, const uint8_t ChannelNumber) {
if (gInputBoxIndex > 0) {
unsigned int i;
for (i = 0; i < 3; i++)
pString[i] = (gInputBox[i] == 10) ? '-' : gInputBox[i] + '0';
return;
}
if (bShowPrefix)
sprintf(pString, "CH-%03u", ChannelNumber + 1);
else if (ChannelNumber == 0xFF)
strcpy(pString, "NULL");
else
sprintf(pString, "%03u", ChannelNumber + 1);
}
// 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 > ' ') {
const unsigned int index = (unsigned int) character - ' ' - 1;
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) {
uint8_t Length = strlen(pString);
size_t i;
uint8_t sum_pixel = 0;
uint16_t true_char[Length];
uint8_t cn_flag[Length];
uint8_t char_num=0;
bool flag_move = 0;
for (size_t j = 0; j < Length; j++) {
uint8_t chn_judge = is_chn(pString[j]);
if (chn_judge == 255 && pString[j] != '\n' && pString[j] != '\0') {
true_char[char_num]=pString[j];
cn_flag[char_num]=0;
char_num++;
sum_pixel += 7;
}
else if (chn_judge != 255) {
cn_flag[char_num]=1;
#if ENABLE_CHINESE_FULL==4
true_char[char_num]=(pString[j]<<8)|pString[j+1];
j++;
#else
true_char[char_num]=chn_judge;
#endif
flag_move = 1;
char_num++;
sum_pixel += 13;
}
}
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;
for (i = 0; i < char_num; i++) {
if (cn_flag[i] == 0) {
if (true_char[i] > ' ') {
const unsigned int index = (unsigned int) true_char[i] - ' ' - 1;
if (index < ARRAY_SIZE(gFontSmall)) {
if (flag_move) {
uint8_t gFontSmall_More[12] = {0};
for (int j = 0; j < 12; ++j) {
if (j < 6) {
gFontSmall_More[j] = (gFontSmall[index][j] & 31) << 3;//00011111
} else {
gFontSmall_More[j] = (gFontSmall[index][j - 6] & 224) >> 5;//11100000
}
}
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);
}
now_pixel += 7;
} else if (pString[i] == ' ')
now_pixel += 7;
} else {
uint8_t gFontChinese[22] = {0};
#if ENABLE_CHINESE_FULL==4
uint8_t tmp[17] = {0};
true_char[i]-=0x8000;
true_char[i]=true_char[i]-true_char[i]/256-1;
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;
for (unsigned char i = 0; i < CHN_FONT_WIDTH * 2; ++i) {
unsigned char j_end = 8;
if (i >= CHN_FONT_WIDTH)
j_end = CHN_FONT_HIGH - 8;
for (unsigned char j = 0; j < j_end; ++j) {
if (IS_BIT_SET(tmp[local], local_bit))
set_bit(&gFontChinese[i], j, 1);
local_bit++;
if (local_bit == 8) {
local_bit = 0;
local++;
}
}
}
#else
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;
for (unsigned char i = 0; i < CHN_FONT_WIDTH * 2; ++i) {
unsigned char j_end = 8;
if (i >= CHN_FONT_WIDTH)
j_end = CHN_FONT_HIGH - 8;
for (unsigned char j = 0; j < j_end; ++j) {
if (IS_BIT_SET(gFontChinese_out[local], local_bit))
set_bit(&gFontChinese[i], j, 1);
local_bit++;
if (local_bit == 8) {
local_bit = 0;
local++;
}
}
}
#endif
memcpy(pFb + now_pixel + 1, &gFontChinese[0], 11);
memcpy(pFb1 + now_pixel + 1, &gFontChinese[11], 11);
now_pixel += 13;
}
}
}
#ifdef ENABLE_SMALL_BOLD
void UI_PrintStringSmallBold(const char *pString, uint8_t Start, uint8_t End, uint8_t Line)
{
const size_t Length = strlen(pString);
size_t i;
if (End > Start)
Start += (((End - Start) - (Length * 8)) + 1) / 2;
const unsigned int char_width = ARRAY_SIZE(gFontSmallBold[0]);
const unsigned int char_spacing = char_width + 1;
uint8_t *pFb = gFrameBuffer[Line] + Start;
for (i = 0; i < Length; i++)
{
if (pString[i] > ' ')
{
const unsigned int index = (unsigned int)pString[i] - ' ' - 1;
if (index < ARRAY_SIZE(gFontSmallBold))
memmove(pFb + (i * char_spacing) + 1, &gFontSmallBold[index], char_width);
}
}
}
#endif
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))
memcpy(buffer + (i * (char_width + 1)) + 1, &gFontSmall[index], char_width);
}
}
}
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;
}
}
void UI_DrawPixelBuffer(uint8_t (*buffer)[128], uint8_t x, uint8_t y, bool black) {
const uint8_t pattern = 1 << (y % 8);
if (black)
buffer[y / 8][x] |= pattern;
else
buffer[y / 8][x] &= ~pattern;
}
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键
UI_PrintStringSmall(EXIT键, 9, 118, 5);
}