LittlePaimon/Paimon_Plugins/paimon_help.py

150 lines
6.4 KiB
Python
Raw Normal View History

2022-06-09 21:57:56 +08:00
from pathlib import Path
from typing import Dict, List
from PIL import Image, ImageDraw, ImageFont
2022-06-09 21:57:56 +08:00
from nonebot import on_command
from nonebot import plugin as nb_plugin
from nonebot.adapters.onebot.v11 import MessageEvent
from nonebot.params import Depends
2022-06-21 16:26:13 +08:00
from nonebot.plugin import PluginMetadata
2022-06-09 21:57:56 +08:00
from ..utils.message_util import MessageBuild
2022-06-09 21:57:56 +08:00
2022-06-21 16:26:13 +08:00
__plugin_meta__ = PluginMetadata(
name="帮助菜单",
description="自动读取插件的信息,生成帮助菜单图片",
usage=(
"help"
),
extra={
'type': '工具',
'range': ['private', 'group', 'guild'],
"author": "惜月 <277073121@qq.com>",
"version": "v1.0.0",
},
)
help_ = on_command('help', aliases={'帮助菜单', '派蒙帮助'}, priority=1, block=True)
2022-06-09 21:57:56 +08:00
help_.__paimon_help__ = {
"usage": "帮助菜单|help",
"introduce": "查看派蒙的帮助信息",
"priority": 99
}
font_path = Path(__file__).parent.parent / 'res'
res_path = Path(__file__).parent.parent / 'res' / 'help'
def get_font(size, font='hywh.ttf') -> ImageFont:
return ImageFont.truetype(str(font_path / font), size=size)
# 绘制带阴影的文字
def draw_shadow_text(draw: ImageDraw, pos: tuple, text: str, font: ImageFont, color: tuple, shadow_color: tuple,
shadow_offset: tuple = (1, 1)):
draw.text(pos, text, font=font, fill=shadow_color)
draw.text((pos[0] - shadow_offset[0], pos[1] - shadow_offset[1]), text, font=font, fill=color)
def draw_table(title: str, funcs: List[dict], total_width: int):
row = len(funcs) // 3 if not len(funcs) % 3 else len(funcs) // 3 + 1
total_height = 120 + row * 160
bg_color = (100, 100, 100, 100)
title_color = (40, 40, 40, 125)
usage_color = (255, 255, 255, 255)
introduce_color = (200, 200, 200, 255)
radius = 50
bg = Image.new('RGBA', (total_width, total_height), (0, 0, 0, 0))
bg_draw = ImageDraw.Draw(bg)
bg_draw.ellipse((0, 0, radius, radius), fill=bg_color)
bg_draw.ellipse((total_width - radius, 0, total_width, radius), fill=bg_color)
bg_draw.ellipse((0, total_height - radius, radius, total_height), fill=bg_color)
bg_draw.ellipse((total_width - radius, total_height - radius, total_width, total_height), fill=bg_color)
bg_draw.rectangle((radius / 2, 0, total_width - radius / 2, total_height), fill=bg_color)
bg_draw.rectangle((0, radius / 2, total_width, total_height - radius / 2), fill=bg_color)
draw_shadow_text(bg_draw, (50, 23), title, get_font(50), (255, 255, 255), (0, 0, 0, 255), (2, 2))
i = 0
for func in funcs:
x = (i % 3) * (total_width / 3) + 4
y = 100 + (i // 3) * 160 + 4
bg_draw.rectangle((x, y, x + total_width / 3 - 8, y + 160 - 8), fill=title_color)
bg_draw.text((x + 7, y + 7), func['usage'], font=get_font(36), fill=usage_color)
# 如果introduce长度大于15 就分多行绘制
if len(func['introduce']) > 15:
bg_draw.text((x + 10, y + 60), func['introduce'][:15], font=get_font(30), fill=introduce_color)
bg_draw.text((x + 10, y + 90), func['introduce'][15:], font=get_font(30), fill=introduce_color)
else:
bg_draw.text((x + 10, y + 60), func['introduce'], font=get_font(30), fill=introduce_color)
i += 1
return row, bg
def draw_help_info(help_info: dict):
total_height = 400 + len(help_info) * 150 - 50
for m in help_info:
total_height += 160 * len(help_info[m]) // 3
if len(help_info[m]) % 3:
total_height += 160
total_height += 150
img = Image.new('RGBA', (1500, total_height), color=(255, 255, 255, 255))
bg_img = Image.open(str(res_path / 'bg.jpg')).convert('RGBA').resize((1500, total_height))
img.paste(bg_img, (0, 0), bg_img)
draw = ImageDraw.Draw(img)
draw_shadow_text(draw, (50, 50), '派蒙帮助', get_font(140), (255, 255, 255), (0, 0, 0, 255), (3, 3))
2022-06-21 16:26:13 +08:00
draw_shadow_text(draw, (610, 140), __plugin_meta__.extra.get('version', '1.0.0'), get_font(50), (255, 255, 255), (0, 0, 0, 255), (3, 3))
2022-06-09 21:57:56 +08:00
draw_shadow_text(draw, (520, 250), '<>内为必须,[]内为可选,()内只需要第一次', get_font(50), (255, 255, 255), (0, 0, 0, 255), (2, 2))
draw_shadow_text(draw, (620, 300), '描述前带*号说明需要绑定私人cookie', get_font(50), (255, 255, 255), (0, 0, 0, 255), (2, 2))
n = 400
for type, func_list in help_info.items():
row, table = draw_table(type, func_list, 1500 - 100)
img.alpha_composite(table, (50, n))
n += 170 + row * 160
draw_shadow_text(draw, (800, n + 15), 'Created by LittlePaimon', get_font(50), (255, 255, 255), (0, 0, 0, 255), (2, 2))
return MessageBuild.Image(img, size=0.7, mode='RGB', quality=80)
async def get_all_plugin(event: MessageEvent) -> dict:
plugin_list = nb_plugin.get_loaded_plugins()
help_info: Dict[str, List[dict]] = {}
for plugin in plugin_list:
try:
2022-06-21 16:26:13 +08:00
plugin_type = plugin.metadata.extra.get('type', '其他')
plugin_range = plugin.metadata.extra.get('range', ['private', 'group', 'guild'])
2022-06-09 21:57:56 +08:00
except AttributeError:
2022-06-21 16:26:13 +08:00
plugin_type = '其他'
plugin_range = ['private', 'group', 'guild']
if event.message_type not in plugin_range:
2022-06-09 21:57:56 +08:00
continue
2022-06-21 16:26:13 +08:00
if plugin_type not in help_info:
help_info[plugin_type] = []
2022-06-09 21:57:56 +08:00
matchers = plugin.matcher
matcher_flag = False
2022-06-09 21:57:56 +08:00
for matcher in matchers:
try:
matchers_info = matcher.__paimon_help__
if 'priority' not in matchers_info:
matchers_info['priority'] = 99
2022-06-21 16:26:13 +08:00
help_info[plugin_type].append(matchers_info)
matcher_flag = True
2022-06-09 21:57:56 +08:00
except AttributeError:
pass
if not matcher_flag:
try:
help_info[plugin_type].append({'usage': plugin.metadata.name, 'introduce': plugin.metadata.description, 'priority': 99})
except AttributeError:
continue
2022-06-09 21:57:56 +08:00
help_info = {k: v for k, v in help_info.items() if v}
if not help_info:
await help_.finish('当前没有已加载的插件哦')
for m in help_info:
help_info[m].sort(key=lambda x: x['priority'])
help_info = dict(sorted(help_info.items(), key=lambda x: len(x[1]), reverse=True))
return help_info
@help_.handle()
async def _(event: MessageEvent, help_info: dict = Depends(get_all_plugin)):
await help_.finish(draw_help_info(help_info))