2022-09-02 19:32:52 +05:00
|
|
|
|
const { SlashCommandBuilder, EmbedBuilder, ActionRowBuilder, ButtonBuilder, ButtonStyle, PermissionsBitField } = require("discord.js"),
|
|
|
|
|
{ Track, Util } = require("discord-player");
|
|
|
|
|
const BaseCommand = require("../../base/BaseCommand"),
|
|
|
|
|
playdl = require("play-dl");
|
2022-08-02 17:18:47 +05:00
|
|
|
|
|
|
|
|
|
class Play extends BaseCommand {
|
|
|
|
|
/**
|
|
|
|
|
*
|
|
|
|
|
* @param {import("../base/JaBa")} client
|
|
|
|
|
*/
|
|
|
|
|
constructor(client) {
|
|
|
|
|
super({
|
|
|
|
|
command: new SlashCommandBuilder()
|
|
|
|
|
.setName("play")
|
|
|
|
|
.setDescription(client.translate("music/play:DESCRIPTION"))
|
2022-08-03 20:57:54 +05:00
|
|
|
|
.addStringOption(option => option.setName("query")
|
|
|
|
|
.setDescription(client.translate("music/play:QUERY"))
|
2022-08-02 17:18:47 +05:00
|
|
|
|
.setRequired(true)),
|
|
|
|
|
aliases: [],
|
|
|
|
|
dirname: __dirname,
|
|
|
|
|
guildOnly: true,
|
|
|
|
|
ownerOnly: false
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
/**
|
|
|
|
|
*
|
|
|
|
|
* @param {import("../../base/JaBa")} client
|
|
|
|
|
*/
|
|
|
|
|
async onLoad() {
|
|
|
|
|
//...
|
|
|
|
|
}
|
|
|
|
|
/**
|
|
|
|
|
*
|
|
|
|
|
* @param {import("../../base/JaBa")} client
|
|
|
|
|
* @param {import("discord.js").ChatInputCommandInteraction} interaction
|
2022-08-09 23:48:33 +05:00
|
|
|
|
* @param {Object} data
|
2022-08-02 17:18:47 +05:00
|
|
|
|
*/
|
|
|
|
|
async execute(client, interaction) {
|
|
|
|
|
await interaction.deferReply();
|
2022-08-29 21:31:36 +05:00
|
|
|
|
|
2022-08-02 17:18:47 +05:00
|
|
|
|
const voice = interaction.member.voice.channel;
|
2022-08-06 22:35:01 +05:00
|
|
|
|
if (!voice) return interaction.editReply({ content: interaction.translate("music/play:NO_VOICE_CHANNEL") });
|
2022-08-03 20:57:54 +05:00
|
|
|
|
const query = interaction.options.getString("query");
|
2022-08-02 17:18:47 +05:00
|
|
|
|
const perms = voice.permissionsFor(client.user);
|
2022-08-06 22:35:01 +05:00
|
|
|
|
if (!perms.has(PermissionsBitField.Flags.Connect) || !perms.has(PermissionsBitField.Flags.Speak)) return interaction.editReply({ content: interaction.translate("music/play:VOICE_CHANNEL_CONNECT") });
|
2022-08-02 17:18:47 +05:00
|
|
|
|
|
2022-08-29 21:31:36 +05:00
|
|
|
|
try {
|
2022-09-02 19:32:52 +05:00
|
|
|
|
var searchResult;
|
|
|
|
|
if (!query.includes("http")) {
|
|
|
|
|
const search = await playdl.search(query, { limit: 10 });
|
|
|
|
|
|
|
|
|
|
if (search) {
|
|
|
|
|
const found = search.map(track => new Track(client.player, {
|
|
|
|
|
title: track.title,
|
|
|
|
|
duration: Util.buildTimeCode(Util.parseMS(track.durationInSec * 1000)),
|
|
|
|
|
thumbnail: track.thumbnails[0].url || "https://cdn.discordapp.com/attachments/708642702602010684/1012418217660121089/noimage.png",
|
|
|
|
|
views: track.views,
|
|
|
|
|
author: track.channel.name,
|
|
|
|
|
description: "search",
|
|
|
|
|
url: track.url,
|
|
|
|
|
requestedBy: interaction.user,
|
|
|
|
|
playlist: null,
|
|
|
|
|
source: "youtube"
|
|
|
|
|
}));
|
|
|
|
|
|
|
|
|
|
searchResult = { playlist: null, tracks: found, searched: true };
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
searchResult = await client.player.search(query, {
|
|
|
|
|
requestedBy: interaction.user
|
|
|
|
|
});
|
|
|
|
|
}
|
2022-08-29 21:31:36 +05:00
|
|
|
|
} catch (error) {
|
|
|
|
|
return interaction.editReply({
|
|
|
|
|
content: interaction.translate("music/play:NO_RESULT", {
|
|
|
|
|
query,
|
|
|
|
|
error
|
|
|
|
|
})
|
|
|
|
|
});
|
|
|
|
|
}
|
2022-08-03 20:57:54 +05:00
|
|
|
|
|
|
|
|
|
const queue = client.player.getQueue(interaction.guildId) || client.player.createQueue(interaction.guild, {
|
2022-08-25 21:25:24 +05:00
|
|
|
|
metadata: { channel: interaction.channel },
|
2022-08-04 22:06:26 +05:00
|
|
|
|
leaveOnEnd: true,
|
|
|
|
|
leaveOnStop: true,
|
|
|
|
|
bufferingTimeout: 1000,
|
|
|
|
|
disableVolume: false,
|
2022-09-02 19:32:52 +05:00
|
|
|
|
spotifyBridge: false,
|
|
|
|
|
/**
|
|
|
|
|
*
|
|
|
|
|
* @param {import("discord-player").Track} track
|
|
|
|
|
* @param {import("discord-player").TrackSource} source
|
|
|
|
|
* @param {import("discord-player").Queue} queue
|
|
|
|
|
* @returns
|
|
|
|
|
*/
|
|
|
|
|
async onBeforeCreateStream(track, source) {
|
|
|
|
|
console.log(track, source);
|
|
|
|
|
if (source === "youtube" || source === "soundcloud")
|
|
|
|
|
return (await playdl.stream(track.url, { discordPlayerCompatibility: true })).stream;
|
|
|
|
|
}
|
2022-08-03 20:57:54 +05:00
|
|
|
|
});
|
|
|
|
|
|
2022-09-02 19:32:52 +05:00
|
|
|
|
if (searchResult.searched) {
|
2022-08-28 16:45:05 +05:00
|
|
|
|
const row1 = new ActionRowBuilder()
|
|
|
|
|
.addComponents(
|
|
|
|
|
new ButtonBuilder()
|
|
|
|
|
.setCustomId("1")
|
2022-09-02 19:37:25 +05:00
|
|
|
|
.setStyle(ButtonStyle.Secondary)
|
2022-08-28 16:45:05 +05:00
|
|
|
|
.setEmoji("1️⃣"),
|
|
|
|
|
new ButtonBuilder()
|
|
|
|
|
.setCustomId("2")
|
2022-09-02 19:37:25 +05:00
|
|
|
|
.setStyle(ButtonStyle.Secondary)
|
2022-08-28 16:45:05 +05:00
|
|
|
|
.setEmoji("2️⃣"),
|
|
|
|
|
new ButtonBuilder()
|
|
|
|
|
.setCustomId("3")
|
2022-09-02 19:37:25 +05:00
|
|
|
|
.setStyle(ButtonStyle.Secondary)
|
2022-08-28 16:45:05 +05:00
|
|
|
|
.setEmoji("3️⃣"),
|
|
|
|
|
new ButtonBuilder()
|
|
|
|
|
.setCustomId("4")
|
2022-09-02 19:37:25 +05:00
|
|
|
|
.setStyle(ButtonStyle.Secondary)
|
2022-08-28 16:45:05 +05:00
|
|
|
|
.setEmoji("4️⃣"),
|
|
|
|
|
new ButtonBuilder()
|
|
|
|
|
.setCustomId("5")
|
2022-09-02 19:37:25 +05:00
|
|
|
|
.setStyle(ButtonStyle.Secondary)
|
2022-08-28 16:45:05 +05:00
|
|
|
|
.setEmoji("5️⃣"),
|
|
|
|
|
);
|
|
|
|
|
const row2 = new ActionRowBuilder()
|
|
|
|
|
.addComponents(
|
|
|
|
|
new ButtonBuilder()
|
|
|
|
|
.setCustomId("6")
|
2022-09-02 19:37:25 +05:00
|
|
|
|
.setStyle(ButtonStyle.Secondary)
|
2022-08-28 16:45:05 +05:00
|
|
|
|
.setEmoji("6️⃣"),
|
|
|
|
|
new ButtonBuilder()
|
|
|
|
|
.setCustomId("7")
|
2022-09-02 19:37:25 +05:00
|
|
|
|
.setStyle(ButtonStyle.Secondary)
|
2022-08-28 16:45:05 +05:00
|
|
|
|
.setEmoji("7️⃣"),
|
|
|
|
|
new ButtonBuilder()
|
|
|
|
|
.setCustomId("8")
|
2022-09-02 19:37:25 +05:00
|
|
|
|
.setStyle(ButtonStyle.Secondary)
|
2022-08-28 16:45:05 +05:00
|
|
|
|
.setEmoji("8️⃣"),
|
|
|
|
|
new ButtonBuilder()
|
|
|
|
|
.setCustomId("9")
|
2022-09-02 19:37:25 +05:00
|
|
|
|
.setStyle(ButtonStyle.Secondary)
|
2022-08-28 16:45:05 +05:00
|
|
|
|
.setEmoji("9️⃣"),
|
|
|
|
|
new ButtonBuilder()
|
|
|
|
|
.setCustomId("10")
|
2022-09-02 19:37:25 +05:00
|
|
|
|
.setStyle(ButtonStyle.Secondary)
|
2022-08-28 16:45:05 +05:00
|
|
|
|
.setEmoji("🔟"),
|
|
|
|
|
);
|
|
|
|
|
const row3 = new ActionRowBuilder()
|
|
|
|
|
.addComponents(
|
|
|
|
|
new ButtonBuilder()
|
|
|
|
|
.setCustomId("search_cancel")
|
2022-09-02 19:38:18 +05:00
|
|
|
|
.setStyle(ButtonStyle.Secondary)
|
2022-08-28 16:45:05 +05:00
|
|
|
|
.setEmoji("❌"),
|
|
|
|
|
);
|
|
|
|
|
const rows = [row1, row2, row3];
|
|
|
|
|
|
|
|
|
|
const embed = new EmbedBuilder()
|
|
|
|
|
.setTitle(interaction.translate("music/play:RESULTS_TITLE", {
|
|
|
|
|
query
|
|
|
|
|
}))
|
|
|
|
|
.setColor(client.config.embed.color)
|
|
|
|
|
.setDescription(searchResult.tracks.map(track => {
|
|
|
|
|
const views = new Intl.NumberFormat(interaction.client.languages.find(language => language.name === interaction.guild.data.language).moment, {
|
|
|
|
|
notation: "compact", compactDisplay: "short"
|
|
|
|
|
}).format(track.views);
|
|
|
|
|
|
2022-08-28 16:56:35 +05:00
|
|
|
|
return `${searchResult.tracks.indexOf(track) + 1}. [${track.title}](${track.url})\n> ${interaction.translate("common:VIEWS")}: **${views}**\n`;
|
2022-08-28 16:45:05 +05:00
|
|
|
|
}).join("\n"))
|
|
|
|
|
.setTimestamp();
|
|
|
|
|
|
|
|
|
|
await interaction.editReply({
|
|
|
|
|
content: interaction.translate("music/play:SEARCH_RESULTS"),
|
|
|
|
|
embeds: [embed],
|
|
|
|
|
components: [row1, row2, row3]
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
const filter = i => i.user.id === interaction.user.id;
|
|
|
|
|
const collector = interaction.channel.createMessageComponentCollector({ filter, idle: (15 * 1000) });
|
|
|
|
|
|
|
|
|
|
collector.on("collect", async i => {
|
|
|
|
|
if (i.isButton()) {
|
|
|
|
|
if (i.customId >= 1 && i.customId <= 10) {
|
|
|
|
|
i.deferUpdate();
|
|
|
|
|
|
|
|
|
|
var selected = searchResult.tracks[i.customId - 1];
|
|
|
|
|
queue.addTrack(selected);
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
if (!queue.connection) await queue.connect(interaction.member.voice.channel);
|
|
|
|
|
if (!queue.playing) await queue.play();
|
|
|
|
|
|
2022-08-29 21:46:12 +05:00
|
|
|
|
interaction.editReply({
|
2022-08-28 16:45:05 +05:00
|
|
|
|
content: interaction.translate("music/play:ADDED_QUEUE", {
|
|
|
|
|
songName: selected.title
|
|
|
|
|
}),
|
2022-08-29 21:31:36 +05:00
|
|
|
|
components: [],
|
|
|
|
|
embeds: []
|
2022-08-28 16:45:05 +05:00
|
|
|
|
});
|
2022-08-29 21:46:12 +05:00
|
|
|
|
|
|
|
|
|
collector.stop();
|
|
|
|
|
return;
|
2022-08-29 21:31:36 +05:00
|
|
|
|
} catch (error) {
|
2022-08-28 16:45:05 +05:00
|
|
|
|
client.player.deleteQueue(interaction.guildId);
|
2022-08-29 21:31:36 +05:00
|
|
|
|
console.log(error);
|
2022-08-28 16:45:05 +05:00
|
|
|
|
return interaction.editReply({
|
|
|
|
|
content: interaction.translate("music/play:ERR_OCCURRED", {
|
2022-08-29 21:31:36 +05:00
|
|
|
|
error
|
2022-08-28 16:45:05 +05:00
|
|
|
|
})
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
} else if (i.customId === "search_cancel") {
|
|
|
|
|
i.deferUpdate();
|
|
|
|
|
|
|
|
|
|
rows.forEach(row => {
|
|
|
|
|
row.components.forEach(component => {
|
|
|
|
|
component.setDisabled(true);
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return interaction.editReply({
|
|
|
|
|
components: [row1, row2, row3]
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
collector.on("end", async (_, reason) => {
|
2022-08-29 21:46:12 +05:00
|
|
|
|
if (reason === "idle") {
|
2022-08-28 16:45:05 +05:00
|
|
|
|
rows.forEach(row => {
|
|
|
|
|
row.components.forEach(component => {
|
|
|
|
|
component.setDisabled(true);
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return interaction.editReply({
|
|
|
|
|
components: [row1, row2, row3]
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-04 22:06:26 +05:00
|
|
|
|
searchResult.playlist ? queue.addTracks(searchResult.tracks) : queue.addTrack(searchResult.tracks[0]);
|
2022-08-03 20:57:54 +05:00
|
|
|
|
|
2022-08-02 17:18:47 +05:00
|
|
|
|
try {
|
2022-08-03 20:57:54 +05:00
|
|
|
|
if (!queue.connection) await queue.connect(interaction.member.voice.channel);
|
|
|
|
|
if (!queue.playing) await queue.play();
|
2022-08-02 17:18:47 +05:00
|
|
|
|
|
|
|
|
|
interaction.editReply({
|
2022-08-03 20:57:54 +05:00
|
|
|
|
content: interaction.translate("music/play:ADDED_QUEUE", {
|
|
|
|
|
songName: searchResult.playlist ? searchResult.playlist.title : searchResult.tracks[0].title
|
|
|
|
|
})
|
2022-08-02 17:18:47 +05:00
|
|
|
|
});
|
2022-08-29 21:31:36 +05:00
|
|
|
|
} catch (error) {
|
2022-08-03 20:57:54 +05:00
|
|
|
|
client.player.deleteQueue(interaction.guildId);
|
2022-08-29 21:31:36 +05:00
|
|
|
|
console.log(error);
|
2022-08-06 22:35:01 +05:00
|
|
|
|
return interaction.editReply({
|
|
|
|
|
content: interaction.translate("music/play:ERR_OCCURRED", {
|
2022-08-29 21:31:36 +05:00
|
|
|
|
error
|
2022-08-06 22:35:01 +05:00
|
|
|
|
})
|
|
|
|
|
});
|
2022-08-02 17:18:47 +05:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
module.exports = Play;
|