mirror of
https://github.com/zhenxun-org/zhenxun_bot.git
synced 2025-12-15 06:12:53 +08:00
modified: basic_plugins/admin_bot_manage/admin_config.py modified: basic_plugins/admin_bot_manage/custom_welcome_message.py modified: basic_plugins/admin_bot_manage/timing_task.py modified: basic_plugins/apscheduler/__init__.py modified: basic_plugins/ban/__init__.py modified: basic_plugins/ban/data_source.py modified: basic_plugins/chat_history/chat_message.py modified: basic_plugins/chat_history/chat_message_handle.py modified: basic_plugins/group_handle/__init__.py modified: basic_plugins/hooks/_utils.py modified: basic_plugins/hooks/ban_hook.py modified: basic_plugins/hooks/chkdsk_hook.py modified: basic_plugins/init_plugin_config/__init__.py deleted: basic_plugins/init_plugin_config/init_group_manager.py modified: basic_plugins/invite_manager/__init__.py new file: basic_plugins/invite_manager/utils.py modified: basic_plugins/nickname.py modified: basic_plugins/plugin_shop/__init__.py modified: basic_plugins/plugin_shop/data_source.py modified: basic_plugins/scripts.py modified: basic_plugins/shop/__init__.py modified: basic_plugins/shop/buy.py modified: basic_plugins/shop/gold.py modified: basic_plugins/shop/my_props/__init__.py deleted: basic_plugins/shop/reset_today_gold.py modified: basic_plugins/shop/shop_handle/__init__.py modified: basic_plugins/shop/shop_handle/data_source.py modified: basic_plugins/shop/use/__init__.py modified: basic_plugins/shop/use/data_source.py modified: basic_plugins/super_cmd/__init__.py modified: basic_plugins/super_cmd/bot_friend_group.py modified: basic_plugins/super_cmd/clear_data.py modified: basic_plugins/super_cmd/exec_sql.py modified: basic_plugins/super_cmd/manager_group.py modified: basic_plugins/super_cmd/reload_setting.py modified: basic_plugins/super_cmd/set_admin_permissions.py deleted: basic_plugins/super_cmd/super_task_switch.py modified: basic_plugins/super_cmd/update_friend_group_info.py modified: basic_plugins/super_help/__init__.py modified: basic_plugins/update_info.py modified: configs/config.py modified: configs/utils/__init__.py modified: models/bag_user.py modified: models/ban_user.py modified: models/chat_history.py modified: models/friend_user.py modified: models/goods_info.py modified: models/group_info.py modified: models/group_member_info.py modified: models/level_user.py modified: models/sign_group_user.py modified: models/user_shop_gold_log.py modified: plugins/aconfig/__init__.py modified: plugins/ai/__init__.py modified: plugins/ai/data_source.py modified: plugins/bilibili_sub/__init__.py modified: plugins/bilibili_sub/data_source.py modified: plugins/bilibili_sub/model.py modified: plugins/black_word/__init__.py modified: plugins/black_word/model.py modified: plugins/black_word/utils.py modified: plugins/bt/data_source.py modified: plugins/genshin/almanac/__init__.py modified: plugins/genshin/material_remind/__init__.py modified: plugins/genshin/query_user/_models/__init__.py modified: plugins/genshin/query_user/_utils/__init__.py modified: plugins/genshin/query_user/bind/__init__.py modified: plugins/genshin/query_user/genshin_sign/__init__.py modified: plugins/genshin/query_user/genshin_sign/data_source.py modified: plugins/genshin/query_user/genshin_sign/init_task.py modified: plugins/genshin/query_user/mihoyobbs_sign/__init__.py modified: plugins/genshin/query_user/query_memo/__init__.py modified: plugins/genshin/query_user/query_memo/data_source.py modified: plugins/genshin/query_user/query_role/__init__.py modified: plugins/genshin/query_user/query_role/data_source.py modified: plugins/genshin/query_user/reset_today_query_user_data/__init__.py modified: plugins/genshin/query_user/resin_remind/__init__.py modified: plugins/genshin/query_user/resin_remind/init_task.py modified: plugins/gold_redbag/model.py modified: plugins/image_management/send_image/__init__.py modified: plugins/my_info/__init__.py modified: plugins/open_cases/models/buff_prices.py modified: plugins/open_cases/models/open_cases_user.py modified: plugins/open_cases/open_cases_c.py modified: plugins/open_cases/utils.py modified: plugins/parse_bilibili_json.py modified: plugins/pid_search.py modified: plugins/pix_gallery/__init__.py modified: plugins/pix_gallery/_data_source.py modified: plugins/pix_gallery/_model/omega_pixiv_illusts.py modified: plugins/pix_gallery/_model/pixiv.py modified: plugins/pix_gallery/_model/pixiv_keyword_user.py modified: plugins/pix_gallery/pix_add_keyword.py modified: plugins/pix_gallery/pix_pass_del_keyword.py modified: plugins/pix_gallery/pix_show_info.py modified: plugins/pix_gallery/pix_update.py modified: plugins/pixiv_rank_search/data_source.py modified: plugins/poke/__init__.py modified: plugins/russian/__init__.py modified: plugins/russian/data_source.py modified: plugins/russian/model.py modified: plugins/send_dinggong_voice/__init__.py modified: plugins/send_setu_/_model.py modified: plugins/send_setu_/send_setu/__init__.py modified: plugins/send_setu_/send_setu/data_source.py modified: plugins/send_setu_/update_setu/data_source.py modified: plugins/sign_in/goods_register.py modified: plugins/sign_in/group_user_checkin.py modified: plugins/sign_in/random_event.py modified: plugins/sign_in/utils.py modified: plugins/statistics/_model.py modified: plugins/statistics/statistics_handle.py modified: plugins/statistics/statistics_hook.py modified: plugins/update_picture.py modified: plugins/web_ui/api/request.py modified: plugins/word_bank/_model.py deleted: plugins/word_bank/_old_model.py modified: plugins/word_bank/_rule.py modified: plugins/word_bank/word_handle.py modified: plugins/word_clouds/data_source.py modified: resources/image/sign/sign_res/bar.png modified: resources/image/sign/sign_res/bar_white.png modified: services/db_context.py modified: services/log.py modified: utils/browser.py modified: utils/data_utils.py modified: utils/depends/__init__.py modified: utils/http_utils.py modified: utils/image_utils.py modified: utils/manager/admin_manager.py modified: utils/message_builder.py modified: utils/utils.py
304 lines
12 KiB
Python
Executable File
304 lines
12 KiB
Python
Executable File
from typing import Union
|
||
|
||
import cv2
|
||
import numpy as np
|
||
from nonebot import on_command
|
||
from nonebot.adapters.onebot.v11 import GroupMessageEvent, Message, MessageEvent
|
||
from nonebot.params import Arg, ArgStr, CommandArg, Depends
|
||
from nonebot.rule import to_me
|
||
from nonebot.typing import T_State
|
||
from PIL import Image, ImageFilter
|
||
|
||
from configs.config import NICKNAME
|
||
from configs.path_config import IMAGE_PATH, TEMP_PATH
|
||
from services.log import logger
|
||
from utils.http_utils import AsyncHttpx
|
||
from utils.image_utils import BuildImage, pic2b64
|
||
from utils.message_builder import image
|
||
from utils.utils import get_message_img, is_number
|
||
|
||
__zx_plugin_name__ = "各种图片简易操作"
|
||
__plugin_usage__ = """
|
||
usage:
|
||
简易的基础图片操作,输入 指定操作 或 序号 来进行选择
|
||
指令:
|
||
1.修改尺寸 [宽] [高] [图片]
|
||
2.等比压缩 [比例] [图片]
|
||
3.旋转图片 [角度] [图片]
|
||
4.水平翻转 [图片]
|
||
5.铅笔滤镜 [图片]
|
||
6.模糊效果 [图片]
|
||
7.锐化效果 [图片]
|
||
8.高斯模糊 [图片]
|
||
9.边缘检测 [图片]
|
||
10.底色替换 [红/蓝] [红/蓝/白/绿/黄] [图片]
|
||
示例:图片修改尺寸 100 200 [图片]
|
||
示例:图片 2 0.3 [图片]
|
||
""".strip()
|
||
__plugin_des__ = "10种快捷的图片简易操作"
|
||
__plugin_cmd__ = [
|
||
"改图 修改尺寸 [宽] [高] [图片]",
|
||
"改图 等比压缩 [比例] [图片]",
|
||
"改图 旋转图片 [角度] [图片]",
|
||
"改图 水平翻转 [图片]",
|
||
"改图 铅笔滤镜 [图片]",
|
||
"改图 模糊效果 [图片]",
|
||
"改图 锐化效果 [图片]",
|
||
"改图 高斯模糊 [图片]",
|
||
"改图 边缘检测 [图片]",
|
||
"改图 底色替换 [红/蓝] [红/蓝/白/绿/黄] [图片]",
|
||
]
|
||
__plugin_type__ = ("一些工具", 1)
|
||
__plugin_version__ = 0.1
|
||
__plugin_author__ = "HibiKier"
|
||
__plugin_settings__ = {
|
||
"level": 5,
|
||
"default_status": True,
|
||
"limit_superuser": False,
|
||
"cmd": ["修改图片", "改图", "操作图片"],
|
||
}
|
||
|
||
method_flag = ""
|
||
|
||
update_img = on_command(
|
||
"修改图片", aliases={"操作图片", "改图"}, priority=5, rule=to_me(), block=True
|
||
)
|
||
|
||
method_list = [
|
||
"修改尺寸",
|
||
"等比压缩",
|
||
"旋转图片",
|
||
"水平翻转",
|
||
"铅笔滤镜",
|
||
"模糊效果",
|
||
"锐化效果",
|
||
"高斯模糊",
|
||
"边缘检测",
|
||
"底色替换",
|
||
]
|
||
method_str = ""
|
||
method_oper = []
|
||
for i in range(len(method_list)):
|
||
method_str += f"\n{i + 1}.{method_list[i]}"
|
||
method_oper.append(method_list[i])
|
||
method_oper.append(str(i + 1))
|
||
|
||
update_img_help = BuildImage(960, 700, font_size=24)
|
||
update_img_help.text((10, 10), __plugin_usage__)
|
||
update_img_help.save(IMAGE_PATH / "update_img_help.png")
|
||
|
||
|
||
def parse_key(key: str):
|
||
async def _key_parser(state: T_State, inp: Union[Message, str] = Arg(key)):
|
||
if key != "img_list" and isinstance(inp, Message):
|
||
inp = inp.extract_plain_text().strip()
|
||
if inp in ["取消", "算了"]:
|
||
await update_img.finish("已取消操作..")
|
||
if key == "method":
|
||
if inp not in method_oper:
|
||
await update_img.reject_arg("method", f"操作不正确,请重新输入!{method_str}")
|
||
elif key == "x":
|
||
method = state["method"]
|
||
if method in ["1", "修改尺寸"]:
|
||
if not is_number(inp) or int(inp) < 1:
|
||
await update_img.reject_arg("x", "宽度不正确!请重新输入数字...")
|
||
elif method in ["2", "等比压缩", "3", "旋转图片"]:
|
||
if not is_number(inp):
|
||
await update_img.reject_arg("x", "比率不正确!请重新输入数字...")
|
||
elif method in ["10", "底色替换"]:
|
||
if inp not in ["红色", "蓝色", "红", "蓝"]:
|
||
await update_img.reject_arg("x", "请输入支持的被替换的底色:\n红色 蓝色")
|
||
elif key == "y":
|
||
method = state["method"]
|
||
if method in ["1", "修改尺寸"]:
|
||
if not is_number(inp) or int(inp) < 1:
|
||
await update_img.reject_arg("y", "长度不正确!请重新输入数字...")
|
||
elif method in ["10", "底色替换"]:
|
||
if inp not in [
|
||
"红色",
|
||
"白色",
|
||
"蓝色",
|
||
"绿色",
|
||
"黄色",
|
||
"红",
|
||
"白",
|
||
"蓝",
|
||
"绿",
|
||
"黄",
|
||
]:
|
||
await update_img.reject_arg("y", "请输入支持的替换的底色:\n红色 蓝色 白色 绿色")
|
||
elif key == "img_list":
|
||
if not get_message_img(inp):
|
||
await update_img.reject_arg("img_list", "没图?没图?没图?来图速来!")
|
||
state[key] = inp
|
||
|
||
return _key_parser
|
||
|
||
|
||
@update_img.handle()
|
||
async def _(event: MessageEvent, state: T_State, arg: Message = CommandArg()):
|
||
if str(event.get_message()) in ["帮助"]:
|
||
await update_img.finish(image("update_img_help.png"))
|
||
raw_arg = arg.extract_plain_text().strip()
|
||
img_list = get_message_img(event.json())
|
||
if raw_arg:
|
||
args = raw_arg.split("[")[0].split()
|
||
state["method"] = args[0]
|
||
if len(args) == 2:
|
||
if args[0] in ["等比压缩", "旋转图片"]:
|
||
if is_number(args[1]):
|
||
state["x"] = args[1]
|
||
state["y"] = ""
|
||
elif len(args) > 2:
|
||
if args[0] in ["修改尺寸"]:
|
||
if is_number(args[1]):
|
||
state["x"] = args[1]
|
||
if is_number(args[2]):
|
||
state["y"] = args[2]
|
||
if args[0] in ["底色替换"]:
|
||
if args[1] in ["红色", "蓝色", "蓝", "红"]:
|
||
state["x"] = args[1]
|
||
if args[2] in ["红色", "白色", "蓝色", "绿色", "黄色", "红", "白", "蓝", "绿", "黄"]:
|
||
state["y"] = args[2]
|
||
if args[0] in ["水平翻转", "铅笔滤镜", "模糊效果", "锐化效果", "高斯模糊", "边缘检测"]:
|
||
state["x"] = ""
|
||
state["y"] = ""
|
||
if img_list:
|
||
state["img_list"] = event.message
|
||
|
||
|
||
@update_img.got(
|
||
"method",
|
||
prompt=f"要使用图片的什么操作呢?{method_str}",
|
||
parameterless=[Depends(parse_key("method"))],
|
||
)
|
||
@update_img.got(
|
||
"x", prompt="[宽度? 比率? 旋转角度? 底色?]", parameterless=[Depends(parse_key("x"))]
|
||
)
|
||
@update_img.got("y", prompt="[长度? 0 0 底色?]", parameterless=[Depends(parse_key("y"))])
|
||
@update_img.got(
|
||
"img_list", prompt="图呢图呢图呢图呢?GKD!", parameterless=[Depends(parse_key("img_list"))]
|
||
)
|
||
async def _(
|
||
event: MessageEvent,
|
||
state: T_State,
|
||
method: str = ArgStr("method"),
|
||
x: str = ArgStr("x"),
|
||
y: str = ArgStr("y"),
|
||
img_list: Message = Arg("img_list"),
|
||
):
|
||
x = x or ""
|
||
y = y or ""
|
||
img_list = get_message_img(img_list)
|
||
if is_number(x):
|
||
x = float(x)
|
||
if is_number(y):
|
||
y = int(y)
|
||
index = 0
|
||
result = ""
|
||
for img_url in img_list:
|
||
if await AsyncHttpx.download_file(
|
||
img_url, TEMP_PATH / f"{event.user_id}_{index}_update.png"
|
||
):
|
||
index += 1
|
||
else:
|
||
await update_img.finish("获取图片超时了...", at_sender=True)
|
||
if index == 0:
|
||
return
|
||
if method in ["修改尺寸", "1"]:
|
||
for i in range(index):
|
||
img = Image.open(TEMP_PATH / f"{event.user_id}_{i}_update.png")
|
||
img = img.convert("RGB")
|
||
img = img.resize((int(x), int(y)), Image.ANTIALIAS)
|
||
result += image(b64=pic2b64(img))
|
||
await update_img.finish(result, at_sender=True)
|
||
if method in ["等比压缩", "2"]:
|
||
for i in range(index):
|
||
img = Image.open(TEMP_PATH / f"{event.user_id}_{i}_update.png")
|
||
width, height = img.size
|
||
img = img.convert("RGB")
|
||
if width * x < 8000 and height * x < 8000:
|
||
img = img.resize((int(x * width), int(x * height)))
|
||
result += image(b64=pic2b64(img))
|
||
else:
|
||
await update_img.finish(f"{NICKNAME}不支持图片压缩后宽或高大于8000的存在!!")
|
||
if method in ["旋转图片", "3"]:
|
||
for i in range(index):
|
||
img = Image.open(TEMP_PATH / f"{event.user_id}_{i}_update.png")
|
||
img = img.rotate(x)
|
||
result += image(b64=pic2b64(img))
|
||
if method in ["水平翻转", "4"]:
|
||
for i in range(index):
|
||
img = Image.open(TEMP_PATH / f"{event.user_id}_{i}_update.png")
|
||
img = img.transpose(Image.FLIP_LEFT_RIGHT)
|
||
result += image(b64=pic2b64(img))
|
||
if method in ["铅笔滤镜", "5"]:
|
||
for i in range(index):
|
||
img = Image.open(TEMP_PATH / f"{event.user_id}_{i}_update.png").filter(
|
||
ImageFilter.CONTOUR
|
||
)
|
||
result += image(b64=pic2b64(img))
|
||
if method in ["模糊效果", "6"]:
|
||
for i in range(index):
|
||
img = Image.open(TEMP_PATH / f"{event.user_id}_{i}_update.png").filter(
|
||
ImageFilter.BLUR
|
||
)
|
||
result += image(b64=pic2b64(img))
|
||
if method in ["锐化效果", "7"]:
|
||
for i in range(index):
|
||
img = Image.open(TEMP_PATH / f"{event.user_id}_{i}_update.png").filter(
|
||
ImageFilter.EDGE_ENHANCE
|
||
)
|
||
result += image(b64=pic2b64(img))
|
||
if method in ["高斯模糊", "8"]:
|
||
for i in range(index):
|
||
img = Image.open(TEMP_PATH / f"{event.user_id}_{i}_update.png").filter(
|
||
ImageFilter.GaussianBlur
|
||
)
|
||
result += image(b64=pic2b64(img))
|
||
if method in ["边缘检测", "9"]:
|
||
for i in range(index):
|
||
img = Image.open(TEMP_PATH / f"{event.user_id}_{i}_update.png").filter(
|
||
ImageFilter.FIND_EDGES
|
||
)
|
||
result += image(b64=pic2b64(img))
|
||
if method in ["底色替换", "10"]:
|
||
if x in ["蓝色", "蓝"]:
|
||
lower = np.array([90, 70, 70])
|
||
upper = np.array([110, 255, 255])
|
||
if x in ["红色", "红"]:
|
||
lower = np.array([0, 135, 135])
|
||
upper = np.array([180, 245, 230])
|
||
if y in ["蓝色", "蓝"]:
|
||
color = (255, 0, 0)
|
||
if y in ["红色", "红"]:
|
||
color = (0, 0, 255)
|
||
if y in ["白色", "白"]:
|
||
color = (255, 255, 255)
|
||
if y in ["绿色", "绿"]:
|
||
color = (0, 255, 0)
|
||
if y in ["黄色", "黄"]:
|
||
color = (0, 255, 255)
|
||
for k in range(index):
|
||
img = cv2.imread(TEMP_PATH / f"{event.user_id}_{k}_update.png")
|
||
img = cv2.resize(img, None, fx=0.3, fy=0.3)
|
||
rows, cols, channels = img.shape
|
||
hsv = cv2.cvtColor(img, cv2.COLOR_BGR2HSV)
|
||
mask = cv2.inRange(hsv, lower, upper)
|
||
# erode = cv2.erode(mask, None, iterations=1)
|
||
dilate = cv2.dilate(mask, None, iterations=1)
|
||
for i in range(rows):
|
||
for j in range(cols):
|
||
if dilate[i, j] == 255:
|
||
img[i, j] = color
|
||
cv2.imwrite(TEMP_PATH / f"{event.user_id}_{k}_ok_update.png", img)
|
||
for i in range(index):
|
||
result += image(TEMP_PATH / f"{event.user_id}_{i}_ok_update.png")
|
||
if is_number(method):
|
||
method = method_list[int(method) - 1]
|
||
logger.info(
|
||
f"(USER {event.user_id}, GROUP"
|
||
f" {event.group_id if isinstance(event, GroupMessageEvent) else 'private'}) 使用{method}"
|
||
)
|
||
await update_img.finish(result, at_sender=True)
|