mirror of
https://github.com/zhenxun-org/zhenxun_bot.git
synced 2025-12-15 06:12:53 +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>
471 lines
16 KiB
Python
471 lines
16 KiB
Python
import os
|
||
import random
|
||
from io import BytesIO
|
||
from pathlib import Path
|
||
from datetime import datetime
|
||
|
||
import pytz
|
||
import nonebot
|
||
from nonebot.drivers import Driver
|
||
from nonebot_plugin_htmlrender import template_to_pic
|
||
|
||
from zhenxun.models.sign_log import SignLog
|
||
from zhenxun.models.sign_user import SignUser
|
||
from zhenxun.utils.utils import get_user_avatar
|
||
from zhenxun.utils.image_utils import BuildImage
|
||
from zhenxun.utils.platform import PlatformUtils
|
||
from zhenxun.configs.config import Config, BotConfig
|
||
from zhenxun.configs.path_config import IMAGE_PATH, TEMPLATE_PATH
|
||
|
||
from .config import (
|
||
SIGN_BORDER_PATH,
|
||
SIGN_RESOURCE_PATH,
|
||
SIGN_BACKGROUND_PATH,
|
||
SIGN_TODAY_CARD_PATH,
|
||
lik2level,
|
||
lik2relation,
|
||
level2attitude,
|
||
)
|
||
|
||
assert (
|
||
len(level2attitude) == len(lik2level) == len(lik2relation)
|
||
), "好感度态度、等级、关系长度不匹配!"
|
||
|
||
AVA_URL = "http://q1.qlogo.cn/g?b=qq&nk={}&s=160"
|
||
|
||
driver: Driver = nonebot.get_driver()
|
||
|
||
base_config = Config.get("sign_in")
|
||
|
||
|
||
MORNING_MESSAGE = [
|
||
"早上好,希望今天是美好的一天!",
|
||
"醒了吗,今天也要元气满满哦!",
|
||
"早上好呀,今天也要开心哦!",
|
||
"早安,愿你拥有美好的一天!",
|
||
]
|
||
|
||
LG_MESSAGE = [
|
||
"今天要早点休息哦~",
|
||
"可不要熬夜到太晚呀",
|
||
"请尽早休息吧!",
|
||
"不要熬夜啦!",
|
||
]
|
||
|
||
|
||
@driver.on_startup
|
||
async def init_image():
|
||
SIGN_RESOURCE_PATH.mkdir(parents=True, exist_ok=True)
|
||
SIGN_TODAY_CARD_PATH.mkdir(exist_ok=True, parents=True)
|
||
await generate_progress_bar_pic()
|
||
clear_sign_data_pic()
|
||
|
||
|
||
async def get_card(
|
||
user: SignUser,
|
||
nickname: str,
|
||
add_impression: float,
|
||
gold: int | None,
|
||
gift: str,
|
||
is_double: bool = False,
|
||
is_card_view: bool = False,
|
||
) -> Path:
|
||
"""获取好感度卡片
|
||
|
||
参数:
|
||
user: SignUser
|
||
nickname: 用户昵称
|
||
impression: 新增的好感度
|
||
gold: 金币
|
||
gift: 礼物
|
||
is_double: 是否触发双倍.
|
||
is_card_view: 是否展示好感度卡片.
|
||
|
||
返回:
|
||
Path: 卡片路径
|
||
"""
|
||
user_id = user.user_id
|
||
date = datetime.now().date()
|
||
_type = "view" if is_card_view else "sign"
|
||
file_name = f"{user_id}_{_type}_{date}.png"
|
||
view_name = f"{user_id}_view_{date}.png"
|
||
card_file = Path(SIGN_TODAY_CARD_PATH) / file_name
|
||
if card_file.exists():
|
||
return IMAGE_PATH / "sign" / "today_card" / file_name
|
||
if add_impression == -1:
|
||
card_file = Path(SIGN_TODAY_CARD_PATH) / view_name
|
||
if card_file.exists():
|
||
return card_file
|
||
is_card_view = True
|
||
return (
|
||
await _generate_html_card(
|
||
user, nickname, add_impression, gold, gift, is_double, is_card_view
|
||
)
|
||
if base_config.get("IMAGE_STYLE") == "zhenxun"
|
||
else await _generate_card(
|
||
user, nickname, add_impression, gold, gift, is_double, is_card_view
|
||
)
|
||
)
|
||
|
||
|
||
async def _generate_card(
|
||
user: SignUser,
|
||
nickname: str,
|
||
add_impression: float,
|
||
gold: int | None,
|
||
gift: str,
|
||
is_double: bool = False,
|
||
is_card_view: bool = False,
|
||
) -> Path:
|
||
"""生成签到卡片
|
||
|
||
参数:
|
||
user: SignUser
|
||
nickname: 用户昵称
|
||
add_impression: 新增的好感度
|
||
gold: 金币
|
||
gift: 礼物
|
||
is_double: 是否触发双倍.
|
||
is_card_view: 是否展示好感度卡片.
|
||
|
||
返回:
|
||
Path: 卡片路径
|
||
"""
|
||
ava_bk = BuildImage(140, 140, (255, 255, 255, 0))
|
||
ava_border = BuildImage(
|
||
140,
|
||
140,
|
||
background=SIGN_BORDER_PATH / "ava_border_01.png",
|
||
)
|
||
if user.platform == "qq" and (byt := await get_user_avatar(user.user_id)):
|
||
ava = BuildImage(107, 107, background=BytesIO(byt))
|
||
else:
|
||
ava = BuildImage(107, 107, (0, 0, 0))
|
||
await ava.circle()
|
||
await ava_bk.paste(ava, (19, 18))
|
||
await ava_bk.paste(ava_border, center_type="center")
|
||
impression = float(user.impression)
|
||
info_img = BuildImage(250, 150, color=(255, 255, 255, 0), font_size=15)
|
||
level, next_impression, previous_impression = get_level_and_next_impression(
|
||
impression
|
||
)
|
||
interpolation = next_impression - impression
|
||
await info_img.text((0, 0), f"· 好感度等级:{level} [{lik2relation[level]}]")
|
||
await info_img.text(
|
||
(0, 20), f"· {BotConfig.self_nickname}对你的态度:{level2attitude[level]}"
|
||
)
|
||
await info_img.text((0, 40), f"· 距离升级还差 {interpolation:.2f} 好感度")
|
||
|
||
bar_bk = BuildImage(220, 20, background=SIGN_RESOURCE_PATH / "bar_white.png")
|
||
bar = BuildImage(220, 20, background=SIGN_RESOURCE_PATH / "bar.png")
|
||
ratio = 1 - (next_impression - impression) / (next_impression - previous_impression)
|
||
if next_impression == 0:
|
||
ratio = 0
|
||
await bar.resize(width=int(bar.width * ratio) or 1, height=bar.height)
|
||
await bar_bk.paste(bar)
|
||
font_size = 20 if "好感度双倍加持卡" in gift else 30
|
||
gift_border = BuildImage(
|
||
270,
|
||
100,
|
||
background=SIGN_BORDER_PATH / "gift_border_02.png",
|
||
font_size=font_size,
|
||
)
|
||
await gift_border.text((0, 0), gift, center_type="center")
|
||
|
||
bk = BuildImage(
|
||
876,
|
||
424,
|
||
background=SIGN_BACKGROUND_PATH
|
||
/ random.choice(os.listdir(SIGN_BACKGROUND_PATH)),
|
||
font_size=25,
|
||
)
|
||
A = BuildImage(876, 274, background=SIGN_RESOURCE_PATH / "white.png")
|
||
line = BuildImage(2, 180, color="black")
|
||
await A.transparent(2)
|
||
await A.paste(ava_bk, (25, 80))
|
||
await A.paste(line, (200, 70))
|
||
nickname_img = await BuildImage.build_text_image(
|
||
nickname, size=50, font_color=(255, 255, 255)
|
||
)
|
||
user_console = await user.user_console
|
||
if user_console and user_console.uid is not None:
|
||
uid = f"{user_console.uid}".rjust(12, "0")
|
||
uid = f"{uid[:4]} {uid[4:8]} {uid[8:]}"
|
||
else:
|
||
uid = "XXXX XXXX XXXX"
|
||
uid_img = await BuildImage.build_text_image(
|
||
f"UID: {uid}", size=30, font_color=(255, 255, 255)
|
||
)
|
||
image1 = await bk.build_text_image("Accumulative check-in for", bk.font, size=30)
|
||
image2 = await bk.build_text_image("days", bk.font, size=30)
|
||
sign_day_img = await BuildImage.build_text_image(
|
||
f"{user.sign_count}", size=40, font_color=(211, 64, 33)
|
||
)
|
||
tip_width = image1.width + image2.width + sign_day_img.width + 60
|
||
tip_height = max([image1.height, image2.height, sign_day_img.height])
|
||
tip_image = BuildImage(tip_width, tip_height, (255, 255, 255, 0))
|
||
await tip_image.paste(image1, (0, 7))
|
||
await tip_image.paste(sign_day_img, (image1.width + 7, 0))
|
||
await tip_image.paste(image2, (image1.width + sign_day_img.width + 15, 7))
|
||
|
||
lik_text1_img = await BuildImage.build_text_image("当前", size=20)
|
||
lik_text2_img = await BuildImage.build_text_image(
|
||
f"好感度:{user.impression:.2f}", size=30
|
||
)
|
||
watermark = await BuildImage.build_text_image(
|
||
f"{BotConfig.self_nickname}@{datetime.now().year}",
|
||
size=15,
|
||
font_color=(155, 155, 155),
|
||
)
|
||
today_data = BuildImage(300, 300, color=(255, 255, 255, 0), font_size=20)
|
||
if is_card_view:
|
||
today_sign_text_img = await BuildImage.build_text_image("", size=30)
|
||
value_list = (
|
||
await SignUser.annotate()
|
||
.order_by("-impression")
|
||
.values_list("user_id", flat=True)
|
||
)
|
||
index = value_list.index(user.user_id) + 1 # type: ignore
|
||
rank_img = await BuildImage.build_text_image(
|
||
f"* 好感度排名第 {index} 位", size=30
|
||
)
|
||
await A.paste(rank_img, ((A.width - rank_img.width - 32), 20))
|
||
last_log = (
|
||
await SignLog.filter(user_id=user.user_id).order_by("create_time").first()
|
||
)
|
||
last_date = "从未"
|
||
if last_log:
|
||
last_date = last_log.create_time.astimezone(
|
||
pytz.timezone("Asia/Shanghai")
|
||
).date()
|
||
await today_data.text(
|
||
(0, 0),
|
||
f"上次签到日期:{last_date}",
|
||
)
|
||
await today_data.text((0, 25), f"总金币:{gold}")
|
||
default_setu_prob = (
|
||
Config.get_config("send_setu", "INITIAL_SETU_PROBABILITY") * 100 # type: ignore
|
||
)
|
||
setu_prob = (
|
||
default_setu_prob + float(user.impression) if user.impression < 100 else 100
|
||
)
|
||
await today_data.text(
|
||
(0, 50),
|
||
f"色图概率:{setu_prob:.2f}%",
|
||
)
|
||
await today_data.text((0, 75), f"开箱次数:{(20 + int(user.impression / 3))}")
|
||
_type = "view"
|
||
else:
|
||
await A.paste(gift_border, (570, 140))
|
||
today_sign_text_img = await BuildImage.build_text_image("今日签到", size=30)
|
||
if is_double:
|
||
await today_data.text((0, 0), f"好感度 + {add_impression / 2:.2f} × 2")
|
||
else:
|
||
await today_data.text((0, 0), f"好感度 + {add_impression:.2f}")
|
||
await today_data.text((0, 25), f"金币 + {gold}")
|
||
_type = "sign"
|
||
current_date = datetime.now()
|
||
current_datetime_str = current_date.strftime("%Y-%m-%d %a %H:%M:%S")
|
||
date = current_date.date()
|
||
date_img = await BuildImage.build_text_image(
|
||
f"时间:{current_datetime_str}", size=20
|
||
)
|
||
await bk.paste(nickname_img, (30, 15))
|
||
await bk.paste(uid_img, (30, 85))
|
||
await bk.paste(A, (0, 150))
|
||
# await bk.text((30, 167), "Accumulative check-in for")
|
||
# _x = bk.getsize("Accumulative check-in for")[0] + sign_day_img.width + 45
|
||
# await bk.paste(sign_day_img, (398, 158))
|
||
# await bk.text((_x, 167), "days")
|
||
await bk.paste(tip_image, (10, 167))
|
||
await bk.paste(date_img, (220, 370))
|
||
await bk.paste(lik_text1_img, (220, 240))
|
||
await bk.paste(lik_text2_img, (262, 234))
|
||
await bk.paste(bar_bk, (225, 275))
|
||
await bk.paste(info_img, (220, 305))
|
||
await bk.paste(today_sign_text_img, (550, 180))
|
||
await bk.paste(today_data, (580, 220))
|
||
await bk.paste(watermark, (15, 400))
|
||
await bk.save(SIGN_TODAY_CARD_PATH / f"{user.user_id}_{_type}_{date}.png")
|
||
return IMAGE_PATH / "sign" / "today_card" / f"{user.user_id}_{_type}_{date}.png"
|
||
|
||
|
||
async def generate_progress_bar_pic():
|
||
"""
|
||
初始化进度条图片
|
||
"""
|
||
bg_2 = (254, 1, 254)
|
||
bg_1 = (0, 245, 246)
|
||
|
||
bk = BuildImage(1000, 50)
|
||
img_x = BuildImage(50, 50, color=bg_2)
|
||
await img_x.circle()
|
||
await img_x.crop((25, 0, 50, 50))
|
||
img_y = BuildImage(50, 50, color=bg_1)
|
||
await img_y.circle()
|
||
await img_y.crop((0, 0, 25, 50))
|
||
A = BuildImage(950, 50)
|
||
width, height = A.size
|
||
|
||
step_r = (bg_2[0] - bg_1[0]) / width
|
||
step_g = (bg_2[1] - bg_1[1]) / width
|
||
step_b = (bg_2[2] - bg_1[2]) / width
|
||
|
||
for y in range(width):
|
||
bg_r = round(bg_1[0] + step_r * y)
|
||
bg_g = round(bg_1[1] + step_g * y)
|
||
bg_b = round(bg_1[2] + step_b * y)
|
||
for x in range(height):
|
||
await A.point((y, x), fill=(bg_r, bg_g, bg_b))
|
||
await bk.paste(img_y, (0, 0))
|
||
await bk.paste(A, (25, 0))
|
||
await bk.paste(img_x, (975, 0))
|
||
await bk.save(SIGN_RESOURCE_PATH / "bar.png")
|
||
|
||
A = BuildImage(950, 50)
|
||
bk = BuildImage(1000, 50)
|
||
img_x = BuildImage(50, 50)
|
||
await img_x.circle()
|
||
await img_x.crop((25, 0, 50, 50))
|
||
img_y = BuildImage(50, 50)
|
||
await img_y.circle()
|
||
await img_y.crop((0, 0, 25, 50))
|
||
await bk.paste(img_y, (0, 0))
|
||
await bk.paste(A, (25, 0))
|
||
await bk.paste(img_x, (975, 0))
|
||
await bk.save(SIGN_RESOURCE_PATH / "bar_white.png")
|
||
|
||
|
||
def get_level_and_next_impression(impression: float) -> tuple[str, int | float, int]:
|
||
"""获取当前好感等级与下一等级的差距
|
||
|
||
参数:
|
||
impression: 好感度
|
||
|
||
返回:
|
||
tuple[str, int, int]: 好感度等级,下一等级好感度要求,已达到的好感度要求
|
||
"""
|
||
|
||
keys = list(lik2level.keys())
|
||
level, next_impression, previous_impression = (
|
||
lik2level[keys[-1]],
|
||
keys[-2],
|
||
keys[-1],
|
||
)
|
||
for i in range(len(keys)):
|
||
if impression >= keys[i]:
|
||
level, next_impression, previous_impression = (
|
||
lik2level[keys[i]],
|
||
keys[i - 1],
|
||
keys[i],
|
||
)
|
||
if i == 0:
|
||
next_impression = impression
|
||
break
|
||
return level, next_impression, previous_impression
|
||
|
||
|
||
def clear_sign_data_pic():
|
||
"""
|
||
清空当前签到图片数据
|
||
"""
|
||
date = datetime.now().date()
|
||
for file in os.listdir(SIGN_TODAY_CARD_PATH):
|
||
if str(date) not in file:
|
||
os.remove(SIGN_TODAY_CARD_PATH / file)
|
||
|
||
|
||
async def _generate_html_card(
|
||
user: SignUser,
|
||
nickname: str,
|
||
add_impression: float,
|
||
gold: int | None,
|
||
gift: str,
|
||
is_double: bool = False,
|
||
is_card_view: bool = False,
|
||
) -> Path:
|
||
"""生成签到卡片
|
||
|
||
参数:
|
||
user: SignUser
|
||
nickname: 用户昵称
|
||
add_impression: 新增的好感度
|
||
gold: 金币
|
||
gift: 礼物
|
||
is_double: 是否触发双倍.
|
||
is_card_view: 是否展示好感度卡片.
|
||
|
||
返回:
|
||
Path: 卡片路径
|
||
"""
|
||
impression = float(user.impression)
|
||
user_console = await user.user_console
|
||
if user_console and user_console.uid is not None:
|
||
uid = f"{user_console.uid}".rjust(12, "0")
|
||
uid = f"{uid[:4]} {uid[4:8]} {uid[8:]}"
|
||
else:
|
||
uid = "XXXX XXXX XXXX"
|
||
level, next_impression, previous_impression = get_level_and_next_impression(
|
||
impression
|
||
)
|
||
interpolation = next_impression - impression
|
||
message = f"{BotConfig.self_nickname}希望你开心!"
|
||
hour = datetime.now().hour
|
||
if hour > 6 and hour < 10:
|
||
message = random.choice(MORNING_MESSAGE)
|
||
elif hour >= 0 and hour < 6:
|
||
message = random.choice(LG_MESSAGE)
|
||
_impression = f"{add_impression}(×2)" if is_double else add_impression
|
||
process = 1 - (next_impression - impression) / (
|
||
next_impression - previous_impression
|
||
)
|
||
now = datetime.now()
|
||
ava_url = PlatformUtils.get_user_avatar_url(user.user_id, "qq")
|
||
data = {
|
||
"ava_url": ava_url,
|
||
"name": nickname,
|
||
"uid": uid,
|
||
"sign_count": f"{user.sign_count}",
|
||
"message": f"{BotConfig.self_nickname}说: {message}",
|
||
"cur_impression": f"{impression:.2f}",
|
||
"impression": f"好感度+{_impression}",
|
||
"gold": f"金币+{gold}",
|
||
"gift": gift,
|
||
"level": f"{level} [{lik2relation[level]}]",
|
||
"attitude": f"对你的态度: {level2attitude[level]}",
|
||
"interpolation": f"{interpolation:.2f}",
|
||
"heart2": [1 for _ in range(int(level))],
|
||
"heart1": [1 for _ in range(len(lik2level) - int(level) - 1)],
|
||
"process": process * 100,
|
||
"date": str(now.replace(microsecond=0)),
|
||
"font_size": 45,
|
||
}
|
||
if len(nickname) > 6:
|
||
data["font_size"] = 27
|
||
_type = "sign"
|
||
if is_card_view:
|
||
_type = "view"
|
||
value_list = (
|
||
await SignUser.annotate()
|
||
.order_by("-impression")
|
||
.values_list("user_id", flat=True)
|
||
)
|
||
index = value_list.index(user.user_id) + 1 # type: ignore
|
||
data["impression"] = f"好感度排名第 {index} 位"
|
||
data["gold"] = f"总金币:{gold}"
|
||
data["gift"] = ""
|
||
pic = await template_to_pic(
|
||
template_path=str((TEMPLATE_PATH / "sign").absolute()),
|
||
template_name="main.html",
|
||
templates={"data": data},
|
||
pages={
|
||
"viewport": {"width": 465, "height": 926},
|
||
"base_url": f"file://{TEMPLATE_PATH}",
|
||
},
|
||
wait=2,
|
||
)
|
||
image = BuildImage.open(pic)
|
||
date = now.date()
|
||
await image.save(SIGN_TODAY_CARD_PATH / f"{user.user_id}_{_type}_{date}.png")
|
||
return IMAGE_PATH / "sign" / "today_card" / f"{user.user_id}_{_type}_{date}.png"
|