
本教程将指导您如何在 discord.js v14 中实现语音机器人播放完mp3文件后自动离开语音频道。通过利用 `@discordjs/voice` 库中的 `audioplayer` 状态变化事件,特别是监听 `idle` 状态,我们可以确保在音频播放结束时,机器人能够优雅地断开与语音频道的连接,从而优化资源管理并提升用户体验。
在开发 Discord 语音机器人时,一个常见的需求是让机器人在完成特定任务(例如播放完一段音频)后自动离开语音频道。这不仅能避免机器人不必要地占用语音频道资源,还能提升用户体验。在 discord.js v14 及其配套的 @discordjs/voice 库中,实现这一功能需要理解音频播放器的状态管理。
在 discord.js v14 中,与语音相关的操作主要由 @discordjs/voice 库处理。以下是实现语音播放和连接管理的关键组件:
以下是一个基本的斜杠命令结构,用于连接机器人到语音频道并播放音频:
const { SlashCommandBuilder, ChannelType } = require('discord.js');
const { getVoiceConnection, entersState, joinVoiceChannel, createAudioPlayer, createAudioResource, VoiceConnectionStatus } = require('@discordjs/voice');
const { join } = require('node:path');
module.exports = {
data: new SlashCommandBuilder()
.setName('playaudio')
.setDescription('播放指定音频并自动离开。')
.addChannelOption(option =>
option.setName('channel')
.setDescription('选择一个语音频道。')
.setRequired(true)
.addChannelTypes(ChannelType.GuildVoice)),
async execute(interaction) {
if (!interaction.isChatInputCommand()) return;
const voiceChannel = interaction.options.getChannel('channel');
if (!voiceChannel || voiceChannel.type !== ChannelType.GuildVoice) {
return interaction.reply({ content: '请选择一个有效的语音频道。', ephemeral: true });
}
let voiceConnection;
try {
voiceConnection = joinVoiceChannel({
channelId: voiceChannel.id,
guildId: interaction.guild.id,
adapterCreator: interaction.guild.voiceAdapterCreator,
selfDeaf: false, // 根据需求设置是否自闭麦
});
await entersState(voiceConnection, VoiceConnectionStatus.Ready, 5000);
console.log(`成功连接到语音频道: ${voiceChannel.name}`);
const player = createAudioPlayer();
// 假设音频文件位于项目的 medias/sound_effect 目录下
const resource = createAudioResource(join(__dirname, '../../medias/sound_effect/padorupadoru.mp3'));
voiceConnection.subscribe(player); // 将播放器连接到语音连接
player.play(resource); // 开始播放音频
await interaction.reply({ content: `正在 ${voiceChannel.name} 播放音频...`, ephemeral: false });
// 错误:此处不应直接尝试断开连接,因为音频尚未播放完毕
// if (player.pause(), () => voiceConnection.disconnect());
} catch (error) {
console.error("连接语音频道或播放音频时发生错误:", error);
if (voiceConnection && voiceConnection.state.status !== VoiceConnectionStatus.Destroyed) {
voiceConnection.destroy(); // 确保在出错时销毁连接
}
await interaction.reply({ content: '无法连接到语音频道或播放音频。请稍后再试。', ephemeral: true });
}
}
};在上述代码中,直接在 player.play(resource); 之后尝试断开连接是无效的,因为它会在音频开始播放时立即执行,而不是等待播放结束。我们需要一种机制来监听音频播放的状态。
@discordjs/voice 库中的 AudioPlayer 实例会触发 stateChange 事件,该事件在播放器状态发生变化时被调用。AudioPlayer 具有多种状态,其中:
当 AudioPlayer 的状态从 playing 变为 idle 时,就意味着音频文件已经播放完毕。我们可以监听这个状态变化来触发机器人离开语音频道的逻辑。
修改后的代码片段:
在 player.play(resource); 之后,添加一个 stateChange 事件监听器:
// ... 接上文代码 ...
voiceConnection.subscribe(player);
player.play(resource);
// 监听播放器状态变化,当播放完成(变为 idle 状态)时断开连接
player.on('stateChange', (oldState, newState) => {
console.log(`AudioPlayer state changed from ${oldState.status} to ${newState.status}`);
if (newState.status === 'idle') {
console.log('音频播放完毕,机器人正在离开语音频道。');
voiceConnection.destroy(); // 销毁连接,释放资源
// 可以选择在此处发送一个交互回复,告知用户机器人已离开
// interaction.followUp({ content: '音频播放完毕,我已离开频道。', ephemeral: true });
}
});
// 错误处理:如果播放器遇到错误
player.on('error', error => {
console.error(`AudioPlayer 发生错误: ${error.message}`);
voiceConnection.destroy(); // 遇到错误也销毁连接
interaction.followUp({ content: '播放音频时发生错误,我已离开频道。', ephemeral: true });
});
// ... 其余代码 ...通过这种方式,机器人会等待音频播放器进入 idle 状态,然后才执行 voiceConnection.destroy() 操作,从而实现音频播放完毕后自动离开语音频道。
以下是整合了上述逻辑的完整斜杠命令示例:
const { SlashCommandBuilder, ChannelType } = require('discord.js');
const { getVoiceConnection, entersState, joinVoiceChannel, createAudioPlayer, createAudioResource, VoiceConnectionStatus } = require('@discordjs/voice');
const { join } = require('node:path');
module.exports = {
data: new SlashCommandBuilder()
.setName('playaudio')
.setDescription('播放指定音频并自动离开。')
.addChannelOption(option =>
option.setName('channel')
.setDescription('选择一个语音频道。')
.setRequired(true)
.addChannelTypes(ChannelType.GuildVoice)),
async execute(interaction) {
if (!interaction.isChatInputCommand()) return;
const voiceChannel = interaction.options.getChannel('channel');
// 验证选择的频道是否为有效的语音频道
if (!voiceChannel || voiceChannel.type !== ChannelType.GuildVoice) {
return interaction.reply({ content: '请选择一个有效的语音频道。', ephemeral: true });
}
let voiceConnection;
try {
// 尝试连接到语音频道
voiceConnection = joinVoiceChannel({
channelId: voiceChannel.id,
guildId: interaction.guild.id,
adapterCreator: interaction.guild.voiceAdapterCreator,
selfDeaf: false, // 机器人连接时是否自闭麦,根据需求设置
});
// 等待语音连接准备就绪,最多等待5秒
await entersState(voiceConnection, VoiceConnectionStatus.Ready, 5000);
console.log(`成功连接到语音频道: ${voiceChannel.name} (${voiceChannel.guild.name})`);
const player = createAudioPlayer();
// 假设音频文件位于项目的 medias/sound_effect 目录下
const resource = createAudioResource(join(__dirname, '../../medias/sound_effect/padorupadoru.mp3'));
voiceConnection.subscribe(player); // 将播放器连接到语音连接
player.play(resource); // 开始播放音频
// 监听播放器状态变化
player.on('stateChange', (oldState, newState) => {
console.log(`AudioPlayer state changed from ${oldState.status} to ${newState.status}`);
// 当播放器状态变为 'idle' 时,表示音频播放完毕
if (newState.status === 'idle') {
console.log('音频播放完毕,机器人正在离开语音频道。');
voiceConnection.destroy(); // 销毁连接,释放所有相关资源
// 可以在此处发送一个后续回复,告知用户机器人已离开
// interaction.followUp({ content: '音频播放完毕,我已离开频道。', ephemeral: true });
}
});
// 监听播放器错误事件,确保即使播放出错也能断开连接
player.on('error', error => {
console.error(`AudioPlayer 发生错误: ${error.message}`);
voiceConnection.destroy(); // 遇到错误时也销毁连接
interaction.followUp({ content: '播放音频时发生错误,我已离开频道。', ephemeral: true });
});
// 向用户发送初始回复
await interaction.reply({ content: `正在 ${voiceChannel.name} 播放音频...`, ephemeral: false });
} catch (error) {
console.error("连接语音频道或播放音频时发生错误:", error);
// 如果在连接或播放过程中发生错误,确保销毁任何已建立的连接
if (voiceConnection && voiceConnection.state.status !== VoiceConnectionStatus.Destroyed) {
voiceConnection.destroy();
}
await interaction.reply({ content: '无法连接到语音频道或播放音频。请稍后再试。', ephemeral: true });
}
}
};通过监听 AudioPlayer 实例的 stateChange 事件,并特别关注其状态变为 idle 的时机,我们可以精确地判断音频播放何时结束。结合 voiceConnection.destroy() 方法,可以实现在 discord.js v14 语音机器人中,音频播放完毕后自动、优雅地离开语音频道。这种方法不仅保证了播放逻辑的完整性,也有效管理了机器人资源,提升了整体的用户体验。
以上就是discord.js v14 教程:控制语音机器人播放音频后自动断开连接的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号