Skip to content
Open
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
172 changes: 135 additions & 37 deletions YTOpen-Ai/functions_tools/PokeTool.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ import { AbstractTool } from './AbstractTool.js';

/**
* 戳一戳工具类,用于对群聊中的用户进行戳一戳操作
* 适配napcat-onebotv11版本
*/
export class PokeTool extends AbstractTool {
constructor() {
Expand Down Expand Up @@ -33,42 +34,128 @@ export class PokeTool extends AbstractTool {
};
}

/**
* 调用OneBotv11 API
* @param {string} action - API动作名称
* @param {Object} params - 参数对象
* @returns {Promise<Object>} - API响应结果
*/
async callApi(action, params = {}) {
try {
// 假设有全局的bot实例,根据实际情况调整
if (typeof Bot !== 'undefined' && Bot.sendApi) {
return await Bot.sendApi(action, params);
} else if (typeof global.bot !== 'undefined' && global.bot.sendApi) {
return await global.bot.sendApi(action, params);
} else {
throw new Error('找不到OneBotv11 API调用接口');
}
} catch (error) {
console.error(`调用API ${action} 失败:`, error);
throw error;
}
}

/**
* 获取群成员列表
* @param {number} groupId - 群号
* @returns {Promise<Array>} - 群成员列表
*/
async getGroupMemberList(groupId) {
try {
const response = await this.callApi('get_group_member_list', {
group_id: groupId
});

if (response.status === 'ok' && response.data) {
return response.data;
} else {
throw new Error('获取群成员列表失败');
}
} catch (error) {
console.error('获取群成员列表失败:', error);
throw error;
}
}

/**
* 发送戳一戳
* @param {number} groupId - 群号
* @param {number} userId - 用户QQ号
* @returns {Promise<boolean>} - 是否成功
*/
async sendGroupPoke(groupId, userId) {
try {
// 尝试使用标准的戳一戳API
let response;
try {
response = await this.callApi('send_group_poke', {
group_id: groupId,
user_id: userId
});
} catch (e) {
// 如果标准API不支持,尝试其他可能的API名称
try {
response = await this.callApi('group_poke', {
group_id: groupId,
user_id: userId
});
} catch (e2) {
// 如果都不支持,使用发送特殊消息的方式
response = await this.callApi('send_group_msg', {
group_id: groupId,
message: [
{
type: 'poke',
data: {
qq: userId
}
}
]
});
}
}

return response.status === 'ok';
} catch (error) {
console.error('发送戳一戳失败:', error);
return false;
}
}

/**
* 查找群成员
* @param {string} target - 目标用户的QQ号或名称
* @param {Map} members - 群成员Map
* @param {Array} members - 群成员数组
* @returns {Object|null} - 找到的成员信息或null
*/
findMember(target, members) {
// 首先尝试作为QQ号查找
if (/^\d+$/.test(target)) {
const member = members.get(Number(target));
if (member) return { qq: Number(target), info: member };
const member = members.find(m => m.user_id === Number(target));
if (member) return member;
}

// 按群名片或昵称查找
for (const [qq, info] of members.entries()) {
const card = info.card?.toLowerCase();
const nickname = info.nickname?.toLowerCase();
const searchTarget = target.toLowerCase();
const searchTarget = target.toLowerCase();
return members.find(member => {
const card = member.card?.toLowerCase() || '';
const nickname = member.nickname?.toLowerCase() || '';

if (card === searchTarget || nickname === searchTarget ||
card?.includes(searchTarget) || nickname?.includes(searchTarget)) {
return { qq, info };
}
}
return null;
return card === searchTarget || nickname === searchTarget ||
card.includes(searchTarget) || nickname.includes(searchTarget);
});
}

/**
* 对单个用户执行戳一戳操作
* @param {Object} group - 群对象
* @param {number} groupId - 群号
* @param {string} target - 目标用户标识(QQ号或名称)
* @param {Map} members - 群成员Map
* @param {Array} members - 群成员数组
* @param {number} pokeCount - 戳一戳次数
* @returns {Promise<Object>} - 操作结果
*/
async pokeSingleUser(group, target, members, pokeCount) {
async pokeSingleUser(groupId, target, members, pokeCount) {
const foundMember = this.findMember(target, members);

if (!foundMember) {
Expand All @@ -78,12 +165,19 @@ export class PokeTool extends AbstractTool {
};
}

const { qq: targetQQ, info: targetMember } = foundMember;
const targetQQ = foundMember.user_id;
const targetName = foundMember.card || foundMember.nickname;

try {
let successCount = 0;

// 执行指定次数的戳一戳
for (let i = 0; i < pokeCount; i++) {
await group.pokeMember(targetQQ);
const success = await this.sendGroupPoke(groupId, targetQQ);
if (success) {
successCount++;
}

// 多次戳一戳时添加延迟
if (i < pokeCount - 1) {
await new Promise(resolve => setTimeout(resolve, 1500));
Expand All @@ -93,15 +187,16 @@ export class PokeTool extends AbstractTool {
return {
target,
targetQQ,
targetName: targetMember.card || targetMember.nickname,
times: pokeCount,
success: true
targetName,
times: successCount,
requestedTimes: pokeCount,
success: successCount > 0
};
} catch (error) {
return {
target,
targetQQ,
targetName: targetMember.card || targetMember.nickname,
targetName,
error: error.message
};
}
Expand All @@ -122,18 +217,18 @@ export class PokeTool extends AbstractTool {
} = opts;

const groupId = e.group_id;

// 获取群对象
let group;
try {
group = await Bot.pickGroup(groupId);
} catch {
return `未找到群 ${groupId}`;

if (!groupId) {
return '此功能仅支持群聊使用';
}

try {
// 获取群成员列表
const members = await group.getMemberMap();
const members = await this.getGroupMemberList(groupId);

if (!members || members.length === 0) {
return '获取群成员列表失败';
}

// 限制戳一戳次数在1-10次之间
const pokeCount = Math.min(Math.max(1, times), 10);
Expand All @@ -142,18 +237,20 @@ export class PokeTool extends AbstractTool {
// 处理目标用户列表
if (!target || random) {
// 随机选择模式:过滤掉机器人自己和发送者
const availableMembers = Array.from(members.entries())
.filter(([id, _]) => id !== Bot.uin && id !== e.sender.user_id);
const selfId = e.self_id || Bot?.uin;
const availableMembers = members.filter(member =>
member.user_id !== selfId && member.user_id !== e.user_id
);

if (availableMembers.length === 0) {
return '群内没有可戳的成员';
}

// 随机选择一个目标
const [randomQQ, randomMember] = availableMembers[
const randomMember = availableMembers[
Math.floor(Math.random() * availableMembers.length)
];
targets = [String(randomQQ)];
targets = [String(randomMember.user_id)];
} else {
// 将输入转换为数组形式
targets = Array.isArray(target) ? target : [target];
Expand All @@ -162,7 +259,7 @@ export class PokeTool extends AbstractTool {
// 并行执行所有戳一戳操作
const results = await Promise.all(
targets.map(async target => {
return await this.pokeSingleUser(group, target, members, pokeCount);
return await this.pokeSingleUser(groupId, target, members, pokeCount);
})
);

Expand All @@ -179,14 +276,15 @@ export class PokeTool extends AbstractTool {
target: r.target,
qq: r.targetQQ,
name: r.targetName,
times: r.times
times: r.times,
requestedTimes: r.requestedTimes
}))
},
errors: errorResults.map(r => ({
target: r.target,
reason: r.error
})),
groupName: group.name || groupId,
groupId: groupId,
isRandom: !target || random
};

Expand Down