mirror of
https://github.com/zhenxun-org/zhenxun_bot.git
synced 2025-12-15 14:22:55 +08:00
* ✨ 父级插件加载 * ✅ 添加测试:更新与添加插件 (#1594) * ✅ 测试更新与添加插件 * ✅ Sourcery建议 * 👷 添加pytest * 🎨 优化代码 * 🐛 bug修复 * 🐛修复添加插件返回403的问题 (#1595) * 完善测试方法 * vscode测试配置 * 重构插件安装过程 * 🎨 修改readme * Update README.md * 🐛 修改bug与版本锁定 * 🐛 修复超级用户对群组功能开关 * 🐛 修复插件商店检查插件更新问题 (#1597) * 🐛 修复插件商店检查插件更新问题 * 🐛 恶意命令检测问题 * 🐛 增加插件状态检查 (#1598) * ✅ 优化测试用例 * 🐛 更改插件更新与安装逻辑 * 🐛 修复更新群组成员信息 * 🎨 代码优化 * 🚀 更新Dockerfile (#1599) * 🎨 更新requirements * ➕ 添加依赖aiocache * ⚡ 添加github镜像 * ✨ 添加仓库目录多获取渠道 * 🐛 修复测试用例 * ✨ 添加API缓存 * 🎨 采取Sourcery建议 * 🐛 文件下载逻辑修改 * 🎨 优化代码 * 🐛 修复插件开关有时出现错误 * ✨ 重构自检ui * 🐛 自检html修正 * 修复签到逻辑bug,并使代码更灵活以适应签到好感度等级配置 (#1606) * 修复签到功能已知问题 * 修复签到功能已知问题 * 修改参数名称 * 修改uid判断 --------- Co-authored-by: HibiKier <45528451+HibiKier@users.noreply.github.com> * 🎨 代码结构优化 * 🐛 私聊时修改插件时删除私聊帮助 * 🐛 过滤父插件 * 🐛 修复自检在ARM上的问题 (#1607) * 🐛 修复自检在ARM上的问题 * ✅ 优化测试 * ✨ 支持mysql,psql,sqlite随机函数 * 🔧 VSCode配置修改 * 🔧 VSCode配置修改 * ✨ 添加金币排行 Co-Authored-By: HibiKier <45528451+HibiKier@users.noreply.github.com> * 📝 修改README Co-Authored-By: HibiKier <45528451+HibiKier@users.noreply.github.com> * 🔨 提取GitHub相关操作 (#1609) * 🔨 提取GitHub相关操作 * 🔨 重构API策略 * ✨ 签到/金币排行限制最大数量 (#1616) * ✨ 签到/金币排行限制最大数量 * 🐛 修复超级用户id获取问题 * 🐛 修复路径解压与挂载 (#1619) * 🐛 修复功能少时zhenxun帮助图片排序问题 (#1620) * 🐛 签到文本适应 (#1622) * 🐛 好感度排行提供默认值 (#1624) * 🎈 优先使用github api (#1625) * ✨ 重构帮助,限制普通用户查询管理插件 (#1626) * 🐛 修复群权限与插件等级匹配 (#1627) * ✨ 当管理员尝试ban真寻时将被反杀 (#1628) * ✨ 群组发言时间检测提供开关配置 (#1630) * 🐳 chore: 支持自动修改版本号 (#1629) * 🎈 perf(github_utils): 支持github url下载遍历 (#1632) * 🎈 perf(github_utils): 支持github url下载遍历 * 🐞 fix(http_utils): 修复一些下载问题 * 🦄 refactor(http_utils): 部分重构 * chore(version): Update version to v0.2.2-e6f17c4 --------- Co-authored-by: AkashiCoin <AkashiCoin@users.noreply.github.com> * 🧪 test(auto_update): 修复测试用例 (#1633) * 🐛 修复商店商品为空时报错 (#1634) * 🐛 修复群权限与插件等级匹配 (#1635) * ✨ message_build支持AtAll (#1639) * 🎈 perf: 使用commit号下载插件 (#1641) * 🎈 perf: 使用commit号下载插件 * chore(version): Update version to v0.2.2-f9c7360 --------- Co-authored-by: AkashiCoin <AkashiCoin@users.noreply.github.com> * 🐳 chore: 修改运行检查触发路径 (#1642) * 🐳 chore: 修改运行检查触发路径 * 🐳 chore: 添加tests目录 * ✨ 重构qq群事件处理 (#1643) * 🐛 签到名称自适应 (#1644) * 🎨 更新README (#1645) * 🐛 fix(http_utils): 流式下载Content-Length错误 (#1647) * 🐛 修复群组中帮助功能状态显示问题 (#1650) * 🐛 修复群欢迎消息设置 (#1651) * 🐛 修复webui下载后首次启动错误 (#1652) * 🐛 修复webui下载后首次启动错误 * chore(version): Update version to v0.2.2-4a8ef85 --------- Co-authored-by: HibiKier <HibiKier@users.noreply.github.com> * ✨ 移除默认图片文件夹:爬 (#1653) * ✨ 安装/移除插件提供插件安装/卸载方法用于插件初始化 (#1654) * ✨ 新增超级用户与管理员帮助模板 (#1655) * ✨ 新增个人信息命令 (#1657) * ✨ 修改个人信息菜单名称 (#1658) * ✨ 新增插件商店api (#1659) * ✨ 新增插件商店api * chore(version): Update version to v0.2.2-7e15f20 --------- Co-authored-by: HibiKier <HibiKier@users.noreply.github.com> * ✨ 将cd,block,count限制复原配置文件 (#1662) * 🎨 修改README (#1663) * 🎨 修改版本号 (#1664) * 🎨 修改requirements (#1665) --------- Co-authored-by: AkashiCoin <l1040186796@gmail.com> Co-authored-by: fanyinrumeng <42991257+fanyinrumeng@users.noreply.github.com> Co-authored-by: AkashiCoin <i@loli.vet> Co-authored-by: Elaga <1728903318@qq.com> Co-authored-by: AkashiCoin <AkashiCoin@users.noreply.github.com> Co-authored-by: HibiKier <HibiKier@users.noreply.github.com>
288 lines
10 KiB
Python
288 lines
10 KiB
Python
import random
|
|
from io import BytesIO
|
|
from pathlib import Path
|
|
from collections.abc import Callable
|
|
|
|
from pydantic import BaseModel
|
|
from PIL.ImageFont import FreeTypeFont
|
|
|
|
from ._build_image import BuildImage
|
|
|
|
|
|
class RowStyle(BaseModel):
|
|
|
|
font: FreeTypeFont | str | Path | None = "HYWenHei-85W.ttf"
|
|
"""字体"""
|
|
font_size: int = 20
|
|
"""字体大小"""
|
|
font_color: str | tuple[int, int, int] = (0, 0, 0)
|
|
"""字体颜色"""
|
|
|
|
class Config:
|
|
arbitrary_types_allowed = True
|
|
|
|
|
|
class ImageTemplate:
|
|
|
|
color_list = ["#C2CEFE", "#FFA94C", "#3FE6A0", "#D1D4F5"] # noqa: RUF012
|
|
|
|
@classmethod
|
|
async def hl_page(
|
|
cls,
|
|
head_text: str,
|
|
items: dict[str, str],
|
|
row_space: int = 10,
|
|
padding: int = 30,
|
|
) -> BuildImage:
|
|
"""列文档 (如插件帮助)
|
|
|
|
参数:
|
|
head_text: 头标签文本
|
|
items: 列内容
|
|
row_space: 列间距.
|
|
padding: 间距.
|
|
|
|
返回:
|
|
BuildImage: 图片
|
|
"""
|
|
font = BuildImage.load_font("HYWenHei-85W.ttf", 20)
|
|
width, height = BuildImage.get_text_size(head_text, font)
|
|
for title, item in items.items():
|
|
title_width, title_height = await cls.__get_text_size(title, font)
|
|
it_width, it_height = await cls.__get_text_size(item, font)
|
|
width = max([width, title_width, it_width])
|
|
height += title_height + it_height
|
|
width = max([width + padding * 2 + 100, 300])
|
|
height = max([height + padding * 2 + 150, 100])
|
|
A = BuildImage(width + padding * 2, height + padding * 2, color="#FAF9FE")
|
|
top_head = BuildImage(width, 100, color="#FFFFFF", font_size=40)
|
|
await top_head.line((0, 1, width, 1), "#C2CEFE", 2)
|
|
await top_head.text((15, 20), head_text, "#9FA3B2", "center")
|
|
await top_head.circle_corner()
|
|
await A.paste(top_head, (0, 20), "width")
|
|
_min_width = top_head.width - 60
|
|
cur_h = top_head.height + 35 + row_space * len(items)
|
|
for title, item in items.items():
|
|
title_width, title_height = BuildImage.get_text_size(title, font)
|
|
title_background = BuildImage(
|
|
title_width + 6, title_height + 10, font=font, color="#C1CDFF"
|
|
)
|
|
await title_background.text((3, 5), title)
|
|
await title_background.circle_corner(5)
|
|
_text_width, _text_height = await cls.__get_text_size(item, font)
|
|
_width = max([title_background.width, _text_width, _min_width])
|
|
text_image = await cls.__build_text_image(
|
|
item, _width, _text_height, font, color="#FDFCFA"
|
|
)
|
|
B = BuildImage(_width + 20, title_height + text_image.height + 40)
|
|
await B.paste(title_background, (10, 10))
|
|
await B.paste(text_image, (10, 20 + title_background.height))
|
|
await B.line((0, 0, 0, B.height), random.choice(cls.color_list))
|
|
await A.paste(B, (0, cur_h), "width")
|
|
cur_h += B.height + row_space
|
|
return A
|
|
|
|
@classmethod
|
|
async def table_page(
|
|
cls,
|
|
head_text: str,
|
|
tip_text: str | None,
|
|
column_name: list[str],
|
|
data_list: list[list[str | tuple[Path | BuildImage, int, int]]],
|
|
row_space: int = 35,
|
|
column_space: int = 30,
|
|
padding: int = 5,
|
|
text_style: Callable[[str, str], RowStyle] | None = None,
|
|
) -> BuildImage:
|
|
"""表格页
|
|
|
|
参数:
|
|
head_text: 标题文本.
|
|
tip_text: 标题注释.
|
|
column_name: 表头列表.
|
|
data_list: 数据列表.
|
|
row_space: 行间距.
|
|
column_space: 列间距.
|
|
padding: 文本内间距.
|
|
text_style: 文本样式.
|
|
|
|
返回:
|
|
BuildImage: 表格图片
|
|
"""
|
|
font = BuildImage.load_font(font_size=50)
|
|
min_width, _ = BuildImage.get_text_size(head_text, font)
|
|
table = await cls.table(
|
|
column_name,
|
|
data_list,
|
|
row_space,
|
|
column_space,
|
|
padding,
|
|
text_style,
|
|
)
|
|
await table.circle_corner()
|
|
table_bk = BuildImage(
|
|
max(table.width, min_width) + 100, table.height + 50, "#EAEDF2"
|
|
)
|
|
await table_bk.paste(table, center_type="center")
|
|
height = table_bk.height + 200
|
|
background = BuildImage(table_bk.width, height, (255, 255, 255), font_size=50)
|
|
await background.paste(table_bk, (0, 200))
|
|
await background.text((0, 50), head_text, "#334762", center_type="width")
|
|
if tip_text:
|
|
text_image = await BuildImage.build_text_image(tip_text, size=22)
|
|
await background.paste(text_image, (0, 110), center_type="width")
|
|
return background
|
|
|
|
@classmethod
|
|
async def table(
|
|
cls,
|
|
column_name: list[str],
|
|
data_list: list[list[str | tuple[Path | BuildImage, int, int]]],
|
|
row_space: int = 25,
|
|
column_space: int = 10,
|
|
padding: int = 5,
|
|
text_style: Callable[[str, str], RowStyle] | None = None,
|
|
) -> BuildImage:
|
|
"""表格
|
|
|
|
参数:
|
|
column_name: 表头列表
|
|
data_list: 数据列表
|
|
row_space: 行间距.
|
|
column_space: 列间距.
|
|
padding: 文本内间距.
|
|
text_style: 文本样式.
|
|
min_width: 最低宽度
|
|
|
|
返回:
|
|
BuildImage: 表格图片
|
|
"""
|
|
font = BuildImage.load_font("HYWenHei-85W.ttf", 20)
|
|
column_data = []
|
|
for i in range(len(column_name)):
|
|
c = []
|
|
for lst in data_list:
|
|
if len(lst) > i:
|
|
c.append(lst[i])
|
|
else:
|
|
c.append("")
|
|
column_data.append(c)
|
|
build_data_list = []
|
|
_, base_h = BuildImage.get_text_size("A", font)
|
|
for i, column_list in enumerate(column_data):
|
|
name_width, _ = BuildImage.get_text_size(column_name[i], font)
|
|
_temp = {"width": name_width, "data": column_list}
|
|
for s in column_list:
|
|
if isinstance(s, tuple):
|
|
w = s[1]
|
|
else:
|
|
w, _ = BuildImage.get_text_size(s, font)
|
|
if w > _temp["width"]:
|
|
_temp["width"] = w
|
|
build_data_list.append(_temp)
|
|
column_image_list = []
|
|
column_name_image_list: list[BuildImage] = []
|
|
for i, data in enumerate(build_data_list):
|
|
column_name_image = await BuildImage.build_text_image(
|
|
column_name[i], font, 12, "#C8CCCF"
|
|
)
|
|
column_name_image_list.append(column_name_image)
|
|
max_h = max(c.height for c in column_name_image_list)
|
|
for i, data in enumerate(build_data_list):
|
|
width = data["width"] + padding * 2
|
|
height = (base_h + row_space) * (len(data["data"]) + 1) + padding * 2
|
|
background = BuildImage(width, height, (255, 255, 255))
|
|
column_name_image = column_name_image_list[i]
|
|
await background.paste(column_name_image, (0, 20), center_type="width")
|
|
cur_h = max_h + row_space + 20
|
|
for item in data["data"]:
|
|
style = RowStyle(font=font)
|
|
if text_style:
|
|
style = text_style(column_name[i], item)
|
|
if isinstance(item, tuple):
|
|
"""图片"""
|
|
data, width, height = item
|
|
image_ = None
|
|
if isinstance(data, Path):
|
|
image_ = BuildImage(width, height, background=data)
|
|
elif isinstance(data, bytes):
|
|
image_ = BuildImage(width, height, background=BytesIO(data))
|
|
elif isinstance(data, BuildImage):
|
|
image_ = data
|
|
if image_:
|
|
await background.paste(image_, (padding, cur_h))
|
|
else:
|
|
await background.text(
|
|
(padding, cur_h),
|
|
item if item is not None else "",
|
|
style.font_color,
|
|
font=style.font,
|
|
font_size=style.font_size,
|
|
)
|
|
cur_h += base_h + row_space
|
|
column_image_list.append(background)
|
|
# height = max([bk.height for bk in column_image_list])
|
|
# width = sum([bk.width for bk in column_image_list])
|
|
return await BuildImage.auto_paste(
|
|
column_image_list, len(column_image_list), column_space
|
|
)
|
|
|
|
@classmethod
|
|
async def __build_text_image(
|
|
cls,
|
|
text: str,
|
|
width: int,
|
|
height: int,
|
|
font: FreeTypeFont,
|
|
font_color: str | tuple[int, int, int] = (0, 0, 0),
|
|
color: str | tuple[int, int, int] = (255, 255, 255),
|
|
) -> BuildImage:
|
|
"""文本转图片
|
|
|
|
参数:
|
|
text: 文本
|
|
width: 宽度
|
|
height: 长度
|
|
font: 字体
|
|
font_color: 文本颜色
|
|
color: 背景颜色
|
|
|
|
返回:
|
|
BuildImage: 文本转图片
|
|
"""
|
|
_, h = BuildImage.get_text_size("A", font)
|
|
A = BuildImage(width, height, color=color)
|
|
cur_h = 0
|
|
for s in text.split("\n"):
|
|
text_image = await BuildImage.build_text_image(
|
|
s, font, font_color=font_color
|
|
)
|
|
await A.paste(text_image, (0, cur_h))
|
|
cur_h += h
|
|
return A
|
|
|
|
@classmethod
|
|
async def __get_text_size(
|
|
cls,
|
|
text: str,
|
|
font: FreeTypeFont,
|
|
) -> tuple[int, int]:
|
|
"""获取文本所占大小
|
|
|
|
参数:
|
|
text: 文本
|
|
font: 字体
|
|
|
|
返回:
|
|
tuple[int, int]: 宽, 高
|
|
"""
|
|
width = 0
|
|
height = 0
|
|
_, h = BuildImage.get_text_size("A", font)
|
|
for s in text.split("\n"):
|
|
s = s.strip() or "A"
|
|
w, _ = BuildImage.get_text_size(s, font)
|
|
width = max(width, w)
|
|
height += h
|
|
return width, height
|