backup codes

This commit is contained in:
2026-01-25 16:36:08 +08:00
parent 388e0e3657
commit 54d17fba30
3 changed files with 384 additions and 149 deletions

View File

@@ -11,12 +11,14 @@ ASSETS_DIR="$PROJECT_ROOT/utils/assets"
OUTPUT_DIR="$PROJECT_ROOT/utils/output"
# 创建输出目录
mkdir -p "$OUTPUT_DIR"
mkdir -p "$OUTPUT_DIR/py"
mkdir -p "$OUTPUT_DIR/bin"
# 颜色输出
RED='\033[0;31m'
GREEN='\033[0;32m'
YELLOW='\033[1;33m'
BLUE='\033[0;34m'
NC='\033[0m' # No Color
# 打印带颜色的信息
@@ -32,6 +34,55 @@ print_error() {
echo -e "${RED}[ERROR]${NC} $1"
}
print_blue() {
echo -e "${BLUE}[CONFIG]${NC} $1"
}
# ==============================================
# 字体配置表 - 调整这里来自定义字体设置
# 格式: "字体名称|字符集|字体文件|字号|输出名称"
# ==============================================
# 小号字体配置
SMALL_FONT_CHARS="你好世界温度湿度1234567890.,?!℃°%:"
SMALL_FONT_FILE="$ASSETS_DIR/NotoSansSC-Regular.otf"
SMALL_FONT_SIZE=14
SMALL_FONT_OUTPUT="small_font"
# 中号字体配置
MEDIUM_FONT_CHARS="你好今天天气温度湿度空气质量良一般差零一二三四五六七八九十百千万亿点°%℃ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789.-:"
MEDIUM_FONT_FILE="$ASSETS_DIR/NotoSansSC-Regular.otf"
MEDIUM_FONT_SIZE=18
MEDIUM_FONT_OUTPUT="medium_font"
# 大号字体配置
LARGE_FONT_CHARS="你好今天温度湿度一二三四五六七八九十°%℃:"
LARGE_FONT_FILE="$ASSETS_DIR/NotoSansSC-Regular.otf"
LARGE_FONT_SIZE=24
LARGE_FONT_OUTPUT="large_font"
# 天气字体配置
WEATHER_FONT_CHARS="晴多云阴雨雪雷风雾霾霜露冰雹"
WEATHER_FONT_FILE="$ASSETS_DIR/NotoSansSC-Regular.otf"
WEATHER_FONT_SIZE=28
WEATHER_FONT_OUTPUT="weather_font"
# ==============================================
# 图片转换配置表 - 调整这里来自定义图片设置
# ==============================================
# 默认图片颜色深度
DEFAULT_IMAGE_COLORS=8
# 支持的图片文件扩展名
IMAGE_EXTENSIONS=("png" "jpg" "jpeg" "bmp" "gif")
# ==============================================
# 函数定义
# ==============================================
# 检查依赖
check_dependencies() {
print_info "检查依赖..."
@@ -57,18 +108,88 @@ check_dependencies() {
print_info "依赖检查完成"
}
# 显示字体配置
show_font_configs() {
print_blue "字体配置:"
echo "----------------------------------------"
printf "%-12s %-8s %-15s %-30s %s\n" "字体名称" "字号" "输出名称" "描述" "字符集预览"
echo "----------------------------------------"
# 小字体
local preview="${SMALL_FONT_CHARS:0:25}"
if [[ ${#SMALL_FONT_CHARS} -gt 25 ]]; then
preview="$preview..."
fi
printf "%-12s %-8s %-15s %-30s %s\n" "小字体" "$SMALL_FONT_SIZE" "$SMALL_FONT_OUTPUT" "基本UI元素" "$preview"
# 中字体
preview="${MEDIUM_FONT_CHARS:0:25}"
if [[ ${#MEDIUM_FONT_CHARS} -gt 25 ]]; then
preview="$preview..."
fi
printf "%-12s %-8s %-15s %-30s %s\n" "中字体" "$MEDIUM_FONT_SIZE" "$MEDIUM_FONT_OUTPUT" "主要文本显示" "$preview"
# 大字体
preview="${LARGE_FONT_CHARS:0:25}"
if [[ ${#LARGE_FONT_CHARS} -gt 25 ]]; then
preview="$preview..."
fi
printf "%-12s %-8s %-15s %-30s %s\n" "大字体" "$LARGE_FONT_SIZE" "$LARGE_FONT_OUTPUT" "标题和重要信息" "$preview"
# 天气字体
preview="${WEATHER_FONT_CHARS:0:25}"
if [[ ${#WEATHER_FONT_CHARS} -gt 25 ]]; then
preview="$preview..."
fi
printf "%-12s %-30s %-8s %-15s %s\n" "天气字体" "$WEATHER_FONT_SIZE" "$WEATHER_FONT_OUTPUT" "天气图标" "$preview"
echo
}
# 显示图片配置
show_image_configs() {
print_blue "图片转换:"
echo "----------------------------------------"
echo "支持格式: ${IMAGE_EXTENSIONS[*]}"
echo "默认颜色深度: $DEFAULT_IMAGE_COLORS"
echo
echo "所有图片将使用默认颜色深度: $DEFAULT_IMAGE_COLORS"
echo
}
# 检查图片文件
is_image_file() {
local file_name="$1"
local ext="${file_name##*.}"
for supported_ext in "${IMAGE_EXTENSIONS[@]}"; do
if [[ "$ext" == "$supported_ext" ]]; then
return 0
fi
done
return 1
}
# 获取图片颜色深度
get_image_colors() {
# 统一使用默认颜色深度
echo "$DEFAULT_IMAGE_COLORS"
}
# 转换字体
convert_font() {
local font_file="$1"
local font_size="$2"
local characters="$3"
local output_name="$4"
local output_py_dir="$5"
local output_bin_dir="$6"
print_info "转换字体: $font_file (大小: $font_size)"
print_info "转换字体: $(basename "$font_file") (字号: $font_size)"
local font_name=$(basename "$font_file" | cut -d'.' -f1)
local temp_py="$OUTPUT_DIR/${font_name}_${font_size}.py"
local bin_file="$OUTPUT_DIR/${output_name}.bin"
local temp_py="$output_py_dir/${output_name}.py"
local bin_file="$output_bin_dir/${output_name}.bin"
# 步骤1: 转换为Python模块
print_info " 步骤1: 将字体转换为Python模块..."
@@ -86,11 +207,13 @@ convert_image() {
local image_file="$1"
local colors="$2"
local output_name="$3"
local output_py_dir="$4"
local output_bin_dir="$5"
print_info "转换图片: $image_file (颜色深度: $colors)"
print_info "转换图片: $(basename "$image_file") (颜色深度: $colors)"
local image_name=$(basename "$image_file" | cut -d'.' -f1)
local temp_py="$OUTPUT_DIR/${image_name}.py"
local temp_py="$output_py_dir/${output_name}.py"
local bin_file="$output_bin_dir/${output_name}.bin"
# 检查imgtobitmap.py是否存在
if [ ! -f "$PROJECT_ROOT/utils/imgtobitmap.py" ]; then
@@ -104,9 +227,9 @@ convert_image() {
# 步骤2: 转换为二进制文件
print_info " 步骤2: 将Python模块转换为二进制文件..."
python3 "$PROJECT_ROOT/utils/image2bin.py" "$temp_py" "$OUTPUT_DIR/${output_name}.bin"
python3 "$PROJECT_ROOT/utils/image2bin.py" "$temp_py" "$bin_file"
print_info "图片转换完成: $OUTPUT_DIR/${output_name}.bin"
print_info " 图片转换完成: $bin_file"
}
# 主转换函数
@@ -114,52 +237,101 @@ main() {
print_info "开始转换资源..."
print_info "输出目录: $OUTPUT_DIR"
# 定义常用字符集
CHINESE_CHARS="你好世界温度湿度天气空气质量良一般差零一二三四五六七八九十百千万亿点℃°%‰"
ENGLISH_CHARS="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789.,!?-+%:="
WEATHER_CHARS="晴多云阴雨雪雷风雾霾霜露冰雹"
# 显示配置
show_font_configs
show_image_configs
# 转换中文字体
if [ -f "$ASSETS_DIR/NotoSansSC-Regular.otf" ]; then
convert_font "$ASSETS_DIR/NotoSansSC-Regular.otf" "20" "$CHINESE_CHARS" "ch_font_20"
convert_font "$ASSETS_DIR/NotoSansSC-Regular.otf" "16" "$CHINESE_CHARS" "ch_font_16"
else
print_warn "中文字体文件不存在: $ASSETS_DIR/NotoSansSC-Regular.otf"
# 确认字体文件存在
local font_files_exist=false
if [ -f "$SMALL_FONT_FILE" ]; then
font_files_exist=true
elif [ -f "$MEDIUM_FONT_FILE" ]; then
font_files_exist=true
elif [ -f "$LARGE_FONT_FILE" ]; then
font_files_exist=true
elif [ -f "$WEATHER_FONT_FILE" ]; then
font_files_exist=true
fi
# 转换英文字体
if [ -f "$ASSETS_DIR/NotoSansSC-Regular.otf" ]; then
convert_font "$ASSETS_DIR/NotoSansSC-Regular.otf" "18" "$ENGLISH_CHARS" "en_font_18"
convert_font "$ASSETS_DIR/NotoSansSC-Regular.otf" "14" "$ENGLISH_CHARS" "en_font_14"
if [ "$font_files_exist" = false ]; then
print_warn "没有找到任何配置的字体文件,跳过字体转换"
else
# 转换小字体
if [ -f "$SMALL_FONT_FILE" ]; then
convert_font "$SMALL_FONT_FILE" "$SMALL_FONT_SIZE" "$SMALL_FONT_CHARS" "$SMALL_FONT_OUTPUT" "$OUTPUT_DIR/py" "$OUTPUT_DIR/bin"
else
print_warn "字体文件不存在: $SMALL_FONT_FILE"
fi
# 转换中字体
if [ -f "$MEDIUM_FONT_FILE" ]; then
convert_font "$MEDIUM_FONT_FILE" "$MEDIUM_FONT_SIZE" "$MEDIUM_FONT_CHARS" "$MEDIUM_FONT_OUTPUT" "$OUTPUT_DIR/py" "$OUTPUT_DIR/bin"
else
print_warn "字体文件不存在: $MEDIUM_FONT_FILE"
fi
# 转换大字体
if [ -f "$LARGE_FONT_FILE" ]; then
convert_font "$LARGE_FONT_FILE" "$LARGE_FONT_SIZE" "$LARGE_FONT_CHARS" "$LARGE_FONT_OUTPUT" "$OUTPUT_DIR/py" "$OUTPUT_DIR/bin"
else
print_warn "字体文件不存在: $LARGE_FONT_FILE"
fi
# 转换天气字体
if [ -f "$ASSETS_DIR/NotoSansSC-Regular.otf" ]; then
convert_font "$ASSETS_DIR/NotoSansSC-Regular.otf" "24" "$WEATHER_CHARS$CHINESE_CHARS$ENGLISH_CHARS" "weather_font_24"
if [ -f "$WEATHER_FONT_FILE" ]; then
convert_font "$WEATHER_FONT_FILE" "$WEATHER_FONT_SIZE" "$WEATHER_FONT_CHARS" "$WEATHER_FONT_OUTPUT" "$OUTPUT_DIR/py" "$OUTPUT_DIR/bin"
else
print_warn "字体文件不存在: $WEATHER_FONT_FILE"
fi
fi
# 转换图片
if [ -f "$ASSETS_DIR/python-logo.png" ]; then
convert_image "$ASSETS_DIR/python-logo.png" "4" "python_logo"
else
print_warn "Python Logo图片不存在: $ASSETS_DIR/python-logo.png"
fi
# 检查并转换图片
local image_files_exist=false
if [ -f "$ASSETS_DIR/boot.jpg" ]; then
convert_image "$ASSETS_DIR/boot.jpg" "4" "boot_image"
# 遍历assets目录下的所有图片文件
for image_file in "$ASSETS_DIR"/*; do
if [ -f "$image_file" ]; then
local file_name=$(basename "$image_file")
if is_image_file "$file_name"; then
image_files_exist=true
break
fi
fi
done
if [ "$image_files_exist" = false ]; then
print_warn "assets目录下没有找到支持的图片文件跳过图片转换"
else
print_warn "启动图片不存在: $ASSETS_DIR/boot.jpg"
print_info "开始转换图片文件..."
# 遍历assets目录下的所有图片文件
for image_file in "$ASSETS_DIR"/*; do
if [ -f "$image_file" ]; then
local file_name=$(basename "$image_file")
if is_image_file "$file_name"; then
local image_name="${file_name%.*}"
local colors=$(get_image_colors "$file_name")
convert_image "$image_file" "$colors" "$image_name" "$OUTPUT_DIR/py" "$OUTPUT_DIR/bin"
fi
fi
done
fi
# 显示输出文件信息
print_info "转换完成! 生成的二进制文件:"
ls -lh "$OUTPUT_DIR"/*.bin 2>/dev/null || print_warn "没有生成二进制文件"
print_info "转换完成! 生成的文件:"
print_info "信息文件:"
ls -lh "$OUTPUT_DIR"/*.info 2>/dev/null || print_warn "没有生成信息文件"
echo -e "\n${GREEN}=== 二进制文件 ===${NC}"
ls -lh "$OUTPUT_DIR/bin"/*.bin 2>/dev/null || print_warn "没有生成二进制文件"
print_info "Python模块:"
ls -lh "$OUTPUT_DIR"/*.py 2>/dev/null || print_warn "没有生成Python模块"
echo -e "\n${GREEN}=== 信息文件 ===${NC}"
ls -lh "$OUTPUT_DIR/bin"/*.info 2>/dev/null || print_warn "没有生成信息文件"
echo -e "\n${GREEN}=== Python模块 ===${NC}"
ls -lh "$OUTPUT_DIR/py"/*.py 2>/dev/null || print_warn "没有生成Python模块"
print_info "转换脚本执行完成!"
}
@@ -172,13 +344,15 @@ show_help() {
echo " -h, --help 显示此帮助信息"
echo ""
echo "此脚本将转换以下资源:"
echo " 1. 中文字体 - 转换为不同大小的二进制字体文件"
echo " 2. 英文字体 - 转换为不同大小的二进制字体文件"
echo " 3. 天气图标字体 - 转换为二进制字体文件"
echo " 4. 图片 - 转换为二进制图片文件"
echo " 1. 字体 - 根据配置表转换为不同大小的二进制字体文件"
echo " 2. 图片 - 遍历assets目录下所有支持的图片文件转换为二进制图片文件"
echo ""
echo "输入文件应位于: $ASSETS_DIR"
echo "输出文件将保存到: $OUTPUT_DIR"
echo "输出文件将保存到:"
echo " - Python模块: $OUTPUT_DIR/py"
echo " - 二进制文件: $OUTPUT_DIR/bin"
echo ""
echo "字体和图片配置可在脚本顶部的配置表中调整"
}
# 解析命令行参数

View File

@@ -29,13 +29,21 @@ def read_image_module(module_path):
spec.loader.exec_module(image_module)
# Extract image metadata and data
bitmap_obj = image_module._bitmap
try:
bitmap_bytes = bytes(bitmap_obj)
except Exception as e:
print(f"Error converting bitmap to bytes: {e}", file=sys.stderr)
bitmap_bytes = b"\x00" # Fallback
image_data = {
"width": image_module.WIDTH,
"height": image_module.HEIGHT,
"colors": image_module.COLORS,
"bpp": image_module.BPP,
"palette": getattr(image_module, "PALETTE", []),
"bitmap": bytes(image_module._bitmap),
"bitmap": bitmap_bytes,
}
return image_data
@@ -52,14 +60,21 @@ def write_binary_file(image_data, output_path):
f.write(b"IMG ") # Magic number ("IMG " with trailing space)
f.write(struct.pack("H", image_data["width"])) # Image width
f.write(struct.pack("H", image_data["height"])) # Image height
f.write(struct.pack("B", image_data["colors"])) # Number of colors
# Limit colors to ubyte range (0-255)
colors_val = min(image_data["colors"], 255)
f.write(struct.pack("B", colors_val)) # Number of colors
f.write(struct.pack("B", image_data["bpp"])) # Bits per pixel
# Write palette
if image_data["palette"]:
f.write(struct.pack("B", len(image_data["palette"]))) # Palette length
for color in image_data["palette"]:
f.write(struct.pack("H", color)) # RGB565 color value
palette_length = min(
len(image_data["palette"]), 255
) # Limit to ubyte range
f.write(struct.pack("B", palette_length)) # Palette length
for i in range(palette_length):
f.write(
struct.pack("H", image_data["palette"][i])
) # RGB565 color value
else:
f.write(struct.pack("B", 0)) # Zero palette length

View File

@@ -1,110 +1,156 @@
#!python3
'''
Convert image file to python module for use with blit_bitmap.
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
"""
imgtobitmap.py - Convert image files to Python bitmap modules for st7789_mpy
Usage imgtobitmap image_file bits_per_pixel >image.py
'''
This script converts image files (PNG, JPG, etc.) to Python bitmap modules
compatible with the st7789_mpy library and the image2bin.py conversion script.
The output modules can be used with st7789.bitmap() method.
Usage:
python imgtobitmap.py <image_file> <colors>
Example:
python imgtobitmap.py logo.png 4 > logo.py
"""
import sys
from PIL import Image
import argparse
import sys
from PIL import Image
def main():
def convert_image_to_bitmap(image_path, colors):
"""
Convert an image file to a Python bitmap module
parser = argparse.ArgumentParser(
prog='imgtobitmap',
description='Convert image file to python module for use with bitmap method.')
Args:
image_path (str): Path to the input image file
colors (int): Number of colors in the output (must be a power of 2, max 256)
"""
try:
# Validate colors
if colors <= 0 or (colors & (colors - 1)) != 0 or colors > 256:
print(f"Error: Colors must be a power of 2 between 1 and 256, got {colors}")
return False
parser.add_argument(
'image_file',
help='Name of file containing image to convert')
# Calculate bits per pixel
bpp = 0
while (1 << bpp) < colors:
bpp += 1
parser.add_argument(
'bits_per_pixel',
type=int,
choices=range(1, 9),
default=1,
metavar='bits_per_pixel',
help='The number of bits to use per pixel (1..8)')
# Load image
img = Image.open(image_path)
args = parser.parse_args()
bits = args.bits_per_pixel
colors_requested = 1 << bits
img = Image.open(args.image_file)
img = img.convert("P", palette=Image.Palette.ADAPTIVE, colors=colors_requested)
palette = img.getpalette() # Make copy of palette colors
# Convert to palette image with specified number of colors
img = img.convert("P", palette=Image.Palette.ADAPTIVE, colors=colors)
palette = img.getpalette()
# Get actual number of colors
palette_colors = len(palette) // 3
bits_required = palette_colors.bit_length()
if (bits_required < bits):
print(f'\nNOTE: Quantization reduced colors to {palette_colors} from the {colors_requested} '
f'requested, reconverting using {bits_required} bit per pixel could save memory.\n''', file=sys.stderr)
if bits_required < bpp:
print(
f"\nNOTE: Quantization reduced colors to {palette_colors} from the {colors} "
f"requested, reconverting using {bits_required} bit per pixel could save memory.\n",
file=sys.stderr,
)
# For all the colors in the palette
colors = []
colors_hex = []
for color in range(palette_colors):
# get rgb values and convert to 565
# Get RGB values and convert to 565
color565 = (
((palette[color * 3] & 0xF8) << 8)
| ((palette[color * 3 + 1] & 0xFC) << 3)
| ((palette[color*3+2] & 0xF8) >> 3))
| (palette[color * 3 + 2] & 0xF8) >> 3
)
# swap bytes in 565
color = ((color565 & 0xff) << 8) + ((color565 & 0xff00) >> 8)
# Swap bytes in 565
color_val = ((color565 & 0xFF) << 8) + ((color565 & 0xFF00) >> 8)
# append byte swapped 565 color to colors
colors.append(f'{color:04x}')
# Append byte swapped 565 color to colors
colors_hex.append(f"{color_val:04x}")
image_bitstring = ''
max_colors = 1 << bits
# Generate bitmap data
image_bitstring = ""
# Run through the image and create a string with the ascii binary
# representation of the color of each pixel.
for y in range(img.height):
for x in range(img.width):
pixel = img.getpixel((x, y))
color = pixel
bstring = ''.join(
'1' if (color & (1 << bit - 1)) else '0'
for bit in range(bits, 0, -1)
bstring = "".join(
"1" if (pixel & (1 << bit - 1)) else "0"
for bit in range(bpp, 0, -1)
)
image_bitstring += bstring
bitmap_bits = len(image_bitstring)
max_colors = 1 << bpp
# Create python source with image parameters
print(f'HEIGHT = {img.height}')
print(f'WIDTH = {img.width}')
print(f'COLORS = {max_colors}')
print(f'BITS = {bitmap_bits}')
print(f'BPP = {bits}')
print('PALETTE = [', sep='', end='')
print(f"HEIGHT = {img.height}")
print(f"WIDTH = {img.width}")
print(f"COLORS = {max_colors}")
print(f"BPP = {bpp}")
print("PALETTE = [", sep="", end="")
for color, rgb in enumerate(colors):
for color, rgb in enumerate(colors_hex):
if color:
print(',', sep='', end='')
print(f'0x{rgb}', sep='', end='')
print(",", sep="", end="")
print(f"0x{rgb}", sep="", end="")
print("]")
# Run though image bit string 8 bits at a time
# and create python array source for memoryview
print("_bitmap =\\", sep='')
print("b'", sep='', end='')
print("_bitmap =\\", sep="")
print("b'", sep="", end="")
for i in range(0, bitmap_bits, 8):
# Limit line length for readability
if i and i % (16 * 8) == 0:
print("'\\\nb'", end='', sep='')
print("'\\\nb'", end="", sep="")
value = image_bitstring[i : i + 8]
color = int(value, 2)
print(f'\\x{color:02x}', sep='', end='')
print(f"\\x{color:02x}", sep="", end="")
print("'\nBITMAP = memoryview(_bitmap)")
print("'")
return True
except Exception as e:
print(f"Error processing image: {e}", file=sys.stderr)
return False
main()
def main():
parser = argparse.ArgumentParser(
prog="imgtobitmap",
description="Convert image file to python module for use with bitmap method.",
)
parser.add_argument("image_file", help="Name of file containing image to convert")
parser.add_argument(
"bits_per_pixel",
type=int,
choices=range(1, 9),
default=1,
metavar="bits_per_pixel",
help="The number of bits to use per pixel (1..8)",
)
args = parser.parse_args()
bits = args.bits_per_pixel
colors = 1 << bits
return 0 if convert_image_to_bitmap(args.image_file, colors) else 1
if __name__ == "__main__":
sys.exit(main())