2024-02-25 03:18:34 +08:00
|
|
|
import random
|
|
|
|
|
import secrets
|
|
|
|
|
from datetime import datetime
|
|
|
|
|
from pathlib import Path
|
|
|
|
|
|
|
|
|
|
import pytz
|
|
|
|
|
from nonebot_plugin_session import EventSession
|
|
|
|
|
|
|
|
|
|
from zhenxun.configs.path_config import IMAGE_PATH
|
|
|
|
|
from zhenxun.models.friend_user import FriendUser
|
|
|
|
|
from zhenxun.models.group_member_info import GroupInfoUser
|
|
|
|
|
from zhenxun.models.sign_log import SignLog
|
|
|
|
|
from zhenxun.models.sign_user import SignUser
|
|
|
|
|
from zhenxun.models.user_console import UserConsole
|
|
|
|
|
from zhenxun.services.log import logger
|
|
|
|
|
from zhenxun.utils.image_utils import BuildImage, ImageTemplate
|
|
|
|
|
from zhenxun.utils.utils import get_user_avatar
|
|
|
|
|
|
|
|
|
|
from ._random_event import random_event
|
2024-03-05 08:29:46 +08:00
|
|
|
from .goods_register import driver
|
2024-07-31 17:30:59 +08:00
|
|
|
from .utils import get_card
|
2024-02-25 03:18:34 +08:00
|
|
|
|
|
|
|
|
ICON_PATH = IMAGE_PATH / "_icon"
|
|
|
|
|
|
|
|
|
|
PLATFORM_PATH = {
|
|
|
|
|
"dodo": ICON_PATH / "dodo.png",
|
|
|
|
|
"discord": ICON_PATH / "discord.png",
|
|
|
|
|
"kaiheila": ICON_PATH / "kook.png",
|
|
|
|
|
"qq": ICON_PATH / "qq.png",
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class SignManage:
|
|
|
|
|
|
|
|
|
|
@classmethod
|
2024-07-31 17:30:59 +08:00
|
|
|
async def rank(
|
|
|
|
|
cls, user_id: str, num: int, group_id: str | None = None
|
|
|
|
|
) -> BuildImage:
|
|
|
|
|
"""好感度排行
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
user_id: 用户id
|
|
|
|
|
num: 排行榜数量
|
|
|
|
|
group_id: 群组id
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
BuildImage: 构造图片
|
|
|
|
|
"""
|
|
|
|
|
query = SignUser
|
|
|
|
|
if group_id:
|
|
|
|
|
user_list = await GroupInfoUser.filter(group_id=group_id).values_list(
|
|
|
|
|
"user_id", flat=True
|
|
|
|
|
)
|
|
|
|
|
query = query.filter(user_id__in=user_list)
|
2024-02-25 03:18:34 +08:00
|
|
|
all_list = (
|
2024-07-31 17:30:59 +08:00
|
|
|
await query.annotate()
|
2024-07-31 17:14:46 +08:00
|
|
|
.order_by("-impression")
|
2024-02-25 03:18:34 +08:00
|
|
|
.values_list("user_id", flat=True)
|
|
|
|
|
)
|
|
|
|
|
index = all_list.index(user_id) + 1 # type: ignore
|
2024-07-31 17:30:59 +08:00
|
|
|
user_list = await query.annotate().order_by("-impression").limit(num).all()
|
2024-02-25 03:18:34 +08:00
|
|
|
user_id_list = [u.user_id for u in user_list]
|
|
|
|
|
column_name = ["排名", "-", "名称", "好感度", "签到次数", "平台"]
|
|
|
|
|
friend_list = await FriendUser.filter(user_id__in=user_id_list).values_list(
|
|
|
|
|
"user_id", "user_name"
|
|
|
|
|
)
|
|
|
|
|
uid2name = {f[0]: f[1] for f in friend_list}
|
|
|
|
|
group_member_list = await GroupInfoUser.filter(
|
|
|
|
|
user_id__in=user_id_list
|
|
|
|
|
).values_list("user_id", "user_name")
|
|
|
|
|
for gm in group_member_list:
|
|
|
|
|
uid2name[gm[0]] = gm[1]
|
|
|
|
|
data_list = []
|
|
|
|
|
for i, user in enumerate(user_list):
|
|
|
|
|
bytes = await get_user_avatar(user.user_id)
|
|
|
|
|
data_list.append(
|
|
|
|
|
[
|
|
|
|
|
f"{i+1}",
|
|
|
|
|
(bytes, 30, 30) if user.platform == "qq" else "",
|
|
|
|
|
uid2name.get(user.user_id),
|
|
|
|
|
user.impression,
|
2024-07-31 17:14:46 +08:00
|
|
|
user.sign_count,
|
2024-02-25 03:18:34 +08:00
|
|
|
(PLATFORM_PATH.get(user.platform), 30, 30),
|
|
|
|
|
]
|
|
|
|
|
)
|
2024-07-31 17:30:59 +08:00
|
|
|
if group_id:
|
|
|
|
|
title = "好感度群组内排行"
|
|
|
|
|
tip = f"你的排名在本群第 {index} 位哦!"
|
|
|
|
|
else:
|
|
|
|
|
title = "好感度全局排行"
|
|
|
|
|
tip = f"你的排名在全局第 {index} 位哦!"
|
|
|
|
|
return await ImageTemplate.table_page(title, tip, column_name, data_list)
|
2024-02-25 03:18:34 +08:00
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
|
async def sign(
|
2024-07-21 19:06:50 +08:00
|
|
|
cls, session: EventSession, nickname: str, is_card_view: bool = False
|
2024-02-25 03:18:34 +08:00
|
|
|
) -> Path | None:
|
|
|
|
|
"""签到
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
session: Session
|
|
|
|
|
nickname: 用户昵称
|
2024-07-21 19:06:50 +08:00
|
|
|
is_card_view: 是否展示卡片
|
2024-02-25 03:18:34 +08:00
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
Path: 卡片路径
|
|
|
|
|
"""
|
|
|
|
|
if not session.id1:
|
|
|
|
|
return None
|
|
|
|
|
now = datetime.now(pytz.timezone("Asia/Shanghai"))
|
2024-02-26 03:04:32 +08:00
|
|
|
user_console = await UserConsole.get_user(session.id1, session.platform)
|
2024-02-25 03:18:34 +08:00
|
|
|
user, _ = await SignUser.get_or_create(
|
|
|
|
|
user_id=session.id1,
|
|
|
|
|
defaults={"user_console": user_console, "platform": session.platform},
|
|
|
|
|
)
|
2024-07-21 19:06:50 +08:00
|
|
|
new_log = (
|
|
|
|
|
await SignLog.filter(user_id=session.id1).order_by("-create_time").first()
|
|
|
|
|
)
|
2024-07-29 23:31:11 +08:00
|
|
|
log_time = None
|
|
|
|
|
if new_log:
|
|
|
|
|
log_time = new_log.create_time.astimezone(
|
|
|
|
|
pytz.timezone("Asia/Shanghai")
|
|
|
|
|
).date()
|
2024-07-21 19:06:50 +08:00
|
|
|
if not is_card_view:
|
2024-07-29 23:31:11 +08:00
|
|
|
if not new_log or (log_time and log_time != now.date()):
|
2024-07-21 19:06:50 +08:00
|
|
|
return await cls._handle_sign_in(user, nickname, session)
|
|
|
|
|
return await get_card(
|
|
|
|
|
user, nickname, -1, user_console.gold, "", is_card_view=is_card_view
|
|
|
|
|
)
|
2024-02-25 03:18:34 +08:00
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
|
async def _handle_sign_in(
|
|
|
|
|
cls,
|
|
|
|
|
user: SignUser,
|
|
|
|
|
nickname: str,
|
|
|
|
|
session: EventSession,
|
|
|
|
|
) -> Path:
|
|
|
|
|
"""签到处理
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
user: SignUser
|
|
|
|
|
nickname: 用户昵称
|
|
|
|
|
session: Session
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
Path: 卡片路径
|
|
|
|
|
"""
|
|
|
|
|
impression_added = (secrets.randbelow(99) + 1) / 100
|
|
|
|
|
rand = random.random()
|
|
|
|
|
add_probability = float(user.add_probability)
|
|
|
|
|
specify_probability = user.specify_probability
|
|
|
|
|
if rand + add_probability > 0.97:
|
|
|
|
|
impression_added *= 2
|
|
|
|
|
elif rand < specify_probability:
|
|
|
|
|
impression_added *= 2
|
|
|
|
|
await SignUser.sign(user, impression_added, session.bot_id, session.platform)
|
|
|
|
|
gold = random.randint(1, 100)
|
|
|
|
|
gift = random_event(float(user.impression))
|
|
|
|
|
if isinstance(gift, int):
|
|
|
|
|
gold += gift
|
|
|
|
|
await UserConsole.add_gold(
|
|
|
|
|
user.user_id, gold + gift, "sign_in", session.platform
|
|
|
|
|
)
|
|
|
|
|
gift = f"额外金币 +{gift}"
|
|
|
|
|
else:
|
|
|
|
|
await UserConsole.add_gold(user.user_id, gold, "sign_in", session.platform)
|
|
|
|
|
await UserConsole.add_props(user.user_id, gift, 1, session.platform)
|
|
|
|
|
gift += " + 1"
|
|
|
|
|
logger.info(
|
|
|
|
|
f"签到成功. score: {user.impression:.2f} "
|
|
|
|
|
f"(+{impression_added:.2f}).获取金币/道具: {gold}",
|
|
|
|
|
"签到",
|
|
|
|
|
session=session,
|
|
|
|
|
)
|
|
|
|
|
return await get_card(
|
|
|
|
|
user,
|
|
|
|
|
nickname,
|
|
|
|
|
impression_added,
|
|
|
|
|
gold,
|
|
|
|
|
gift,
|
|
|
|
|
rand + add_probability > 0.97 or rand < specify_probability,
|
|
|
|
|
)
|