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

283 lines
9.1 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 "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-17 04:23:45 +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))
uint8_t menu_set_flag = 0;
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) {
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;
}
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 > ' ') {
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) {
uint8_t Length = strlen(pString);
2023-12-17 04:23:45 +00:00
if (menu_set_flag == 1) {
2023-12-07 03:39:09 +00:00
Length = Length > 7 ? 7 : Length;
menu_set_flag = 0;
}
size_t i;
uint8_t sum_pixel = 0;
uint8_t chn_flag[Length];
2023-12-17 10:27:47 +00:00
bool flag_move = 0;
2023-12-07 03:39:09 +00:00
for (size_t j = 0; j < Length; j++) {
chn_flag[j] = is_chn(pString[j]);
2023-12-17 04:23:45 +00:00
if (chn_flag[j] == 255 && pString[j] != '\n' && pString[j] != '\0')sum_pixel += 7;
2023-12-07 03:39:09 +00:00
2023-12-17 10:27:47 +00:00
else if (chn_flag[j] != 255) {
flag_move = 1;
sum_pixel += 12;
}
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;
for (i = 0; i < Length; i++) {
if (chn_flag[i] == 255) {
if (pString[i] > ' ') {
const unsigned int index = (unsigned int) pString[i] - ' ' - 1;
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) {
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);
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-17 04:23:45 +00:00
unsigned int local = CHN_FONT_HIGH * CHN_FONT_WIDTH * chn_flag[i] / 8;
unsigned int local_bit = (CHN_FONT_HIGH * CHN_FONT_WIDTH * chn_flag[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++;
}
2023-12-07 03:39:09 +00:00
}
}
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-07 03:39:09 +00:00
now_pixel += 12;
}
}
}
#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))
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键
UI_PrintStringSmall(EXIT键, 9, 118, 6);
2023-11-30 06:38:27 +00:00
}