discord-player-play-dl/src/Player.js

374 lines
13 KiB
JavaScript
Raw Normal View History

2020-01-12 00:01:18 +05:00
const ytdl = require('ytdl-core');
const SYA = require('simple-youtube-api');
const mergeOptions = require('merge-options');
2020-02-21 15:32:16 +05:00
const { VoiceChannel, version } = require("discord.js");
if(version.split('.')[0] !== '12') throw new Error("Only the master branch of discord.js library is supported for now. Install it using 'npm install discordjs/discord.js'.");
2020-01-12 00:01:18 +05:00
const Queue = require('./Queue');
const Util = require('./Util');
const Song = require('./Song');
/**
* Player options.
* @typedef {PlayerOptions}
*
* @property {Boolean} leaveOnEnd Whether the bot should leave the current voice channel when the queue ends.
* @property {Boolean} leaveOnStop Whether the bot should leave the current voice channel when the stop() function is used.
* @property {Boolean} leaveOnEmpty Whether the bot should leave the voice channel if there is no more member in it.
2020-01-12 00:01:18 +05:00
*/
const PlayerOptions = {
leaveOnEnd: true,
leaveOnStop: true,
leaveOnEmpty: true
2020-01-12 00:01:18 +05:00
};
class Player {
/**
* @param {Client} client Your Discord Client instance.
* @param {string} youtubeToken Your Youtube Data v3 API key.
* @param {PlayerOptions} options The PlayerOptions object.
*/
constructor(client, youtubeToken, options = {}){
if(!client) throw new SyntaxError('Invalid Discord client');
if(!youtubeToken) throw new SyntaxError('Invalid Token: Token must be a String');
/**
* Your Discord Client instance.
* @type {Client}
*/
this.client = client;
/**
* Your Youtube Data v3 API key.
* @type {string}
*/
this.youtubeToken = youtubeToken;
/**
* The Simple Youtube API Client.
* @type {Youtube}
*/
this.SYA = new SYA(this.youtubeToken);
/**
* The guilds data.
* @type {Queue[]}
*/
this.queues = [];
/**
* Player options.
* @type {PlayerOptions}
*/
this.options = mergeOptions(PlayerOptions, options);
// Listener to check if the channel is empty
2020-01-12 23:06:44 +05:00
client.on('voiceStateUpdate', (oldState, newState) => {
if(!this.options.leaveOnEmpty) return;
// If the member leaves a voice channel
2020-01-13 00:14:26 +05:00
if(!oldState.channelID || newState.channelID) return;
// Search for a queue for this channel
2020-01-13 00:14:26 +05:00
let queue = this.queues.find((g) => g.connection.channel.id === oldState.channelID);
if(queue){
// If the channel is not empty
if(queue.connection.channel.members.size > 1) return;
// Disconnect from the voice channel
queue.connection.channel.leave();
// Delete the queue
this.queues = this.queues.filter((g) => g.guildID !== queue.guildID);
// Emit end event
2020-01-13 00:14:26 +05:00
queue.emit('channelEmpty');
}
});
2020-01-12 00:01:18 +05:00
}
/**
* Whether a guild is currently playing songs
* @param {string} guildID The guild ID to check
* @returns {Boolean} Whether the guild is currently playing songs
*/
isPlaying(guildID) {
return this.queues.some((g) => g.guildID === guildID);
}
/**
* Plays a song in a voice channel.
* @param {voiceChannel} voiceChannel The voice channel in which the song will be played.
* @param {string} songName The name of the song to play.
* @returns {Promise<Song>}
*/
play(voiceChannel, songName) {
this.queues = this.queues.filter((g) => g.guildID !== voiceChannel.id);
return new Promise(async (resolve, reject) => {
if(!voiceChannel || typeof voiceChannel !== "object") return reject("voiceChannel must be type of VoiceChannel. value="+voiceChannel);
2020-02-21 15:32:16 +05:00
if(typeof songName !== "string") return reject("songName must be type of string. value="+songName);
2020-01-12 00:01:18 +05:00
// Searches the song
2020-01-12 21:13:57 +05:00
let video = await Util.getFirstYoutubeResult(songName, this.SYA);
2020-02-21 15:32:16 +05:00
if(!video) return reject('Song not found');
// Joins the voice channel
let connection = await voiceChannel.join();
2020-01-12 00:01:18 +05:00
// Creates a new guild with data
let queue = new Queue(voiceChannel.guild.id);
queue.connection = connection;
2020-01-12 00:01:18 +05:00
let song = new Song(video, queue);
queue.songs.push(song);
// Add the queue to the list
this.queues.push(queue);
// Plays the song
this._playSong(queue.guildID, true);
// Resolves the song.
resolve(song);
});
}
/**
* Pauses the current song.
* @param {string} guildID
* @returns {Promise<Song>}
*/
pause(guildID){
return new Promise(async(resolve, reject) => {
// Gets guild queue
let queue = this.queues.find((g) => g.guildID === guildID);
2020-02-21 15:32:16 +05:00
if(!queue) return reject('Not playing');
2020-01-12 00:01:18 +05:00
// Pauses the dispatcher
queue.dispatcher.pause();
queue.playing = false;
// Resolves the guild queue
resolve(queue.songs[0]);
});
}
/**
* Resumes the current song.
* @param {string} guildID
* @returns {Promise<Song>}
*/
resume(guildID){
return new Promise(async(resolve, reject) => {
// Gets guild queue
let queue = this.queues.find((g) => g.guildID === guildID);
2020-02-21 15:32:16 +05:00
if(!queue) return reject('Not playing');
2020-01-12 00:01:18 +05:00
// Pauses the dispatcher
queue.dispatcher.resume();
queue.playing = true;
// Resolves the guild queue
resolve(queue.songs[0]);
});
}
/**
* Stops playing music.
* @param {string} guildID
* @returns {Promise<void>}
*/
stop(guildID){
return new Promise(async(resolve, reject) => {
// Gets guild queue
let queue = this.queues.find((g) => g.guildID === guildID);
2020-02-21 15:32:16 +05:00
if(!queue) return reject('Not playing');
2020-01-12 00:01:18 +05:00
// Stops the dispatcher
queue.stopped = true;
queue.songs = [];
queue.dispatcher.end();
// Resolves
resolve();
});
}
/**
* Updates the volume.
* @param {string} guildID
* @param {number} percent
* @returns {Promise<void>}
*/
setVolume(guildID, percent) {
return new Promise(async(resolve, reject) => {
// Gets guild queue
let queue = this.queues.find((g) => g.guildID === guildID);
2020-02-21 15:32:16 +05:00
if(!queue) return reject('Not playing');
2020-01-12 00:01:18 +05:00
// Updates volume
queue.dispatcher.setVolumeLogarithmic(percent / 200);
// Resolves guild queue
resolve(queue);
});
}
/**
* Gets the guild queue.
* @param {string} guildID
2020-01-18 22:29:53 +05:00
* @returns {?Queue}
2020-01-12 00:01:18 +05:00
*/
getQueue(guildID) {
2020-01-18 22:29:53 +05:00
// Gets guild queue
let queue = this.queues.find((g) => g.guildID === guildID);
return queue;
2020-01-12 00:01:18 +05:00
}
/**
* Adds a song to the guild queue.
* @param {string} guildID
* @param {string} songName The name of the song to add to the queue.
* @returns {Promise<Song>}
*/
addToQueue(guildID, songName){
return new Promise(async(resolve, reject) => {
// Gets guild queue
let queue = this.queues.find((g) => g.guildID === guildID);
2020-02-21 15:32:16 +05:00
if(!queue) return reject('Not playing');
2020-01-12 00:01:18 +05:00
// Searches the song
let video = await Util.getFirstYoutubeResult(songName, this.SYA).catch(() => {});
2020-02-21 15:32:16 +05:00
if(!video) return reject('Song not found');
2020-01-12 00:01:18 +05:00
let song = new Song(video, queue);
// Updates queue
queue.songs.push(song);
// Resolves the song
resolve(song);
});
}
2020-01-18 17:04:05 +05:00
/**
* Sets the queue for a guild.
* @param {string} guildID
* @param {Array<Song>} songs The songs list
* @returns {Promise<Queue>}
*/
setQueue(guildID, songs){
return new Promise(async(resolve, reject) => {
// Gets guild queue
let queue = this.queues.find((g) => g.guildID === guildID);
2020-02-21 15:32:16 +05:00
if(!queue) return reject('Not playing');
2020-01-18 17:04:05 +05:00
// Updates queue
queue.songs = songs;
// Resolves the queue
resolve(queue);
});
}
2020-01-12 00:01:18 +05:00
/**
* Clears the guild queue, but not the current song.
* @param {string} guildID
* @returns {Promise<Queue>}
*/
clearQueue(guildID){
return new Promise(async(resolve, reject) => {
// Gets guild queue
let queue = this.queues.find((g) => g.guildID === guildID);
2020-02-21 15:32:16 +05:00
if(!queue) return reject('Not playing');
2020-01-12 00:01:18 +05:00
// Clears queue
let currentlyPlaying = queue.songs.shift();
queue.songs = [ currentlyPlaying ];
// Resolves guild queue
resolve(queue);
});
}
/**
* Skips a song.
* @param {string} guildID
* @returns {Promise<Song>}
2020-01-12 00:01:18 +05:00
*/
skip(guildID){
return new Promise(async(resolve, reject) => {
// Gets guild queue
let queue = this.queues.find((g) => g.guildID === guildID);
2020-02-21 15:32:16 +05:00
if(!queue) return reject('Not playing');
let currentSong = queue.songs[0];
2020-01-12 00:01:18 +05:00
// Ends the dispatcher
queue.dispatcher.end();
queue.skipped = true;
// Resolves the current song
resolve(currentSong);
2020-01-12 00:01:18 +05:00
});
}
2020-01-18 15:15:10 +05:00
/**
* Gets the currently playing song.
* @param {string} guildID
* @returns {Promise<Song>}
*/
nowPlaying(guildID){
return new Promise(async(resolve, reject) => {
// Gets guild queue
let queue = this.queues.find((g) => g.guildID === guildID);
2020-02-21 15:32:16 +05:00
if(!queue) return reject('Not playing');
2020-01-18 15:15:10 +05:00
let currentSong = queue.songs[0];
// Resolves the current song
resolve(currentSong);
});
}
2020-01-18 15:03:22 +05:00
/**
* Enable or disable the repeat mode
2020-02-02 15:20:46 +05:00
* @param {string} guildID
2020-01-18 15:03:22 +05:00
* @param {Boolean} enabled Whether the repeat mode should be enabled
* @returns {Promise<Void>}
*/
2020-02-02 15:20:46 +05:00
setRepeatMode(guildID, enabled) {
2020-01-18 15:03:22 +05:00
return new Promise(async(resolve, reject) => {
// Gets guild queue
let queue = this.queues.find((g) => g.guildID === guildID);
2020-02-21 15:32:16 +05:00
if(!queue) return reject('Not playing');
2020-01-18 15:03:22 +05:00
// Enable/Disable repeat mode
queue.repeatMode = enabled;
// Resolve
resolve();
});
}
2020-04-24 20:14:34 +05:00
/**
* Shuffles the guild queue.
* @param {string} guildID
* @returns {Promise<Void>}
*/
shuffle(guildID){
return new Promise(async(resolve, reject) => {
// Gets guild queue
let queue = this.queues.find((g) => g.guildID === guildID);
if(!queue) return reject('Not playing');
// Shuffle the queue (except the first song)
let currentSong = queue.songs.shift();
queue.songs = queue.songs.sort(() => Math.random() - 0.5);
queue.songs.unshift(currentSong);
// Resolve
resolve();
});
}
2020-01-12 00:01:18 +05:00
/**
* Start playing songs in a guild.
* @ignore
* @param {string} guildID
* @param {Boolean} firstPlay Whether the function was called from the play() one
*/
async _playSong(guildID, firstPlay) {
// Gets guild queue
let queue = this.queues.find((g) => g.guildID === guildID);
// If there isn't any music in the queue
if(queue.songs.length < 2 && !firstPlay && !queue.repeatMode){
2020-01-12 00:01:18 +05:00
// Leaves the voice channel
2020-01-12 22:56:11 +05:00
if(this.options.leaveOnEnd && !queue.stopped) queue.connection.channel.leave();
2020-01-12 00:01:18 +05:00
// Remoces the guild from the guilds list
this.queues = this.queues.filter((g) => g.guildID !== guildID);
// Emits stop event
2020-01-12 22:56:11 +05:00
if(queue.stopped){
if(this.options.leaveOnStop) queue.connection.channel.leave();
return queue.emit('stop');
}
2020-01-12 00:01:18 +05:00
// Emits end event
return queue.emit('end');
}
// Emit songChanged event
2020-01-18 15:03:22 +05:00
if(!firstPlay) queue.emit('songChanged', (!queue.repeatMode ? queue.songs.shift() : queue.songs[0]), queue.songs[0], queue.skipped, queue.repeatMode);
2020-01-12 00:01:18 +05:00
queue.skipped = false;
let song = queue.songs[0];
// Download the song
let dispatcher = queue.connection.play(ytdl(song.url, { filter: "audioonly" }));
queue.dispatcher = dispatcher;
// Set volume
dispatcher.setVolumeLogarithmic(queue.volume / 200);
// When the song ends
dispatcher.on('finish', () => {
// Play the next song
return this._playSong(guildID, false);
});
}
};
module.exports = Player;