2021-04-06 17:58:46 +05:00
|
|
|
import { EventEmitter } from 'events';
|
2021-04-09 15:04:36 +05:00
|
|
|
import { Client, Collection, Snowflake, Collector, Message, VoiceChannel, VoiceState } from 'discord.js';
|
2021-04-17 19:11:18 +05:00
|
|
|
import {
|
|
|
|
LyricsData,
|
|
|
|
PlayerOptions as PlayerOptionsType,
|
|
|
|
PlayerProgressbarOptions,
|
|
|
|
PlayerStats,
|
|
|
|
QueueFilters
|
|
|
|
} from './types/types';
|
2021-04-06 17:58:46 +05:00
|
|
|
import Util from './utils/Util';
|
|
|
|
import AudioFilters from './utils/AudioFilters';
|
2021-04-06 20:38:17 +05:00
|
|
|
import { Queue } from './Structures/Queue';
|
|
|
|
import { Track } from './Structures/Track';
|
2021-04-17 19:11:18 +05:00
|
|
|
import { PlayerErrorEventCodes, PlayerEvents, PlayerOptions } from './utils/Constants';
|
2021-04-06 19:08:01 +05:00
|
|
|
import PlayerError from './utils/PlayerError';
|
|
|
|
import ytdl from 'discord-ytdl-core';
|
2021-04-08 19:42:08 +05:00
|
|
|
import { ExtractorModel } from './Structures/ExtractorModel';
|
2021-04-17 09:01:06 +05:00
|
|
|
import os from 'os';
|
2021-04-06 17:55:29 +05:00
|
|
|
|
|
|
|
// @ts-ignore
|
2021-04-06 17:58:46 +05:00
|
|
|
import spotify from 'spotify-url-info';
|
2021-04-06 17:55:29 +05:00
|
|
|
// @ts-ignore
|
2021-04-06 17:58:46 +05:00
|
|
|
import { Client as SoundCloudClient } from 'soundcloud-scraper';
|
2021-04-07 19:05:35 +05:00
|
|
|
import YouTube from 'youtube-sr';
|
2021-04-06 17:55:29 +05:00
|
|
|
|
2021-04-06 17:58:46 +05:00
|
|
|
const SoundCloud = new SoundCloudClient();
|
2021-04-04 22:36:40 +05:00
|
|
|
|
2021-04-06 20:38:17 +05:00
|
|
|
export class Player extends EventEmitter {
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* The discord client that instantiated this player
|
|
|
|
*/
|
2021-04-04 22:36:40 +05:00
|
|
|
public client!: Client;
|
2021-04-17 19:11:18 +05:00
|
|
|
public options: PlayerOptionsType;
|
2021-04-06 17:55:29 +05:00
|
|
|
public filters: typeof AudioFilters;
|
2021-04-09 17:59:14 +05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The collection of queues in this player
|
|
|
|
*/
|
2021-04-09 15:04:36 +05:00
|
|
|
public queues = new Collection<Snowflake, Queue>();
|
|
|
|
private _resultsCollectors = new Collection<string, Collector<Snowflake, Message>>();
|
|
|
|
private _cooldownsTimeout = new Collection<string, NodeJS.Timeout>();
|
2021-04-09 17:59:14 +05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The extractor model collection
|
|
|
|
*/
|
2021-04-08 18:03:04 +05:00
|
|
|
public Extractors = new Collection<string, ExtractorModel>();
|
2021-04-04 22:36:40 +05:00
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Creates new Player instance
|
|
|
|
* @param client The discord.js client
|
|
|
|
* @param options Player options
|
|
|
|
*/
|
2021-04-17 19:11:18 +05:00
|
|
|
constructor(client: Client, options?: PlayerOptionsType) {
|
2021-04-04 22:36:40 +05:00
|
|
|
super();
|
|
|
|
|
2021-04-06 17:58:46 +05:00
|
|
|
Object.defineProperty(this, 'client', {
|
2021-04-04 22:36:40 +05:00
|
|
|
value: client,
|
|
|
|
enumerable: false
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The player options
|
|
|
|
*/
|
2021-04-17 19:11:18 +05:00
|
|
|
this.options = Object.assign({}, PlayerOptions, options ?? {});
|
2021-04-04 22:36:40 +05:00
|
|
|
|
|
|
|
// check FFmpeg
|
|
|
|
void Util.alertFFmpeg();
|
2021-04-06 17:55:29 +05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The audio filters
|
|
|
|
*/
|
|
|
|
this.filters = AudioFilters;
|
|
|
|
|
2021-04-09 15:15:38 +05:00
|
|
|
this.client.on('voiceStateUpdate', (o, n) => void this._handleVoiceStateUpdate(o, n));
|
2021-04-08 19:42:08 +05:00
|
|
|
|
2021-04-19 13:33:34 +05:00
|
|
|
// auto detect @discord-player/extractor
|
|
|
|
let nv: any;
|
|
|
|
if ((nv = Util.require('@discord-player/extractor'))) {
|
|
|
|
['Attachment', 'Facebook', 'Reverbnation', 'Vimeo'].forEach((ext) => void this.use(ext, nv[ext]));
|
|
|
|
}
|
2021-04-06 17:55:29 +05:00
|
|
|
}
|
|
|
|
|
2021-04-19 18:32:10 +05:00
|
|
|
static get AudioFilters(): typeof AudioFilters {
|
2021-04-06 17:55:29 +05:00
|
|
|
return AudioFilters;
|
|
|
|
}
|
|
|
|
|
2021-04-08 18:03:04 +05:00
|
|
|
/**
|
|
|
|
* Define custom extractor in this player
|
|
|
|
* @param extractorName The extractor name
|
|
|
|
* @param extractor The extractor itself
|
|
|
|
*/
|
2021-04-19 18:32:10 +05:00
|
|
|
use(extractorName: string, extractor: any): Player {
|
2021-04-08 19:42:08 +05:00
|
|
|
if (!extractorName) throw new PlayerError('Missing extractor name!', 'PlayerExtractorError');
|
2021-04-08 18:03:04 +05:00
|
|
|
|
2021-04-08 19:42:08 +05:00
|
|
|
const methods = ['validate', 'getInfo'];
|
2021-04-08 18:03:04 +05:00
|
|
|
|
|
|
|
for (const method of methods) {
|
2021-04-08 19:42:08 +05:00
|
|
|
if (typeof extractor[method] !== 'function')
|
|
|
|
throw new PlayerError('Invalid extractor supplied!', 'PlayerExtractorError');
|
2021-04-08 18:03:04 +05:00
|
|
|
}
|
|
|
|
|
|
|
|
this.Extractors.set(extractorName, new ExtractorModel(extractorName, extractor));
|
|
|
|
|
2021-04-08 19:43:33 +05:00
|
|
|
return this;
|
2021-04-08 18:03:04 +05:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove existing extractor from this player
|
|
|
|
* @param extractorName The extractor name
|
|
|
|
*/
|
2021-04-19 18:32:10 +05:00
|
|
|
unuse(extractorName: string): boolean {
|
2021-04-08 19:42:08 +05:00
|
|
|
if (!extractorName) throw new PlayerError('Missing extractor name!', 'PlayerExtractorError');
|
2021-04-08 18:03:04 +05:00
|
|
|
|
|
|
|
return this.Extractors.delete(extractorName);
|
|
|
|
}
|
|
|
|
|
2021-04-06 19:08:01 +05:00
|
|
|
private _searchTracks(message: Message, query: string, firstResult?: boolean): Promise<Track> {
|
2021-04-06 17:55:29 +05:00
|
|
|
return new Promise(async (resolve) => {
|
|
|
|
let tracks: Track[] = [];
|
2021-04-06 17:58:46 +05:00
|
|
|
const queryType = Util.getQueryType(query);
|
|
|
|
|
|
|
|
switch (queryType) {
|
|
|
|
case 'soundcloud_track':
|
|
|
|
{
|
|
|
|
const data = await SoundCloud.getSongInfo(query).catch(() => {});
|
|
|
|
if (data) {
|
|
|
|
const track = new Track(this, {
|
|
|
|
title: data.title,
|
|
|
|
url: data.url,
|
2021-04-10 22:52:48 +05:00
|
|
|
duration: Util.buildTimeCode(Util.parseMS(data.duration / 1000)),
|
2021-04-06 17:58:46 +05:00
|
|
|
description: data.description,
|
|
|
|
thumbnail: data.thumbnail,
|
|
|
|
views: data.playCount,
|
2021-04-09 19:04:39 +05:00
|
|
|
author: data.author.name,
|
2021-04-06 17:58:46 +05:00
|
|
|
requestedBy: message.author,
|
|
|
|
fromPlaylist: false,
|
|
|
|
source: 'soundcloud',
|
2021-04-09 19:04:39 +05:00
|
|
|
engine: data
|
2021-04-06 17:58:46 +05:00
|
|
|
});
|
|
|
|
|
|
|
|
tracks.push(track);
|
|
|
|
}
|
2021-04-06 17:55:29 +05:00
|
|
|
}
|
2021-04-06 17:58:46 +05:00
|
|
|
break;
|
|
|
|
case 'spotify_song':
|
|
|
|
{
|
|
|
|
const matchSpotifyURL = query.match(
|
|
|
|
/https?:\/\/(?:embed\.|open\.)(?:spotify\.com\/)(?:track\/|\?uri=spotify:track:)((\w|-){22})/
|
|
|
|
);
|
|
|
|
if (matchSpotifyURL) {
|
|
|
|
const spotifyData = await spotify.getPreview(query).catch(() => {});
|
|
|
|
if (spotifyData) {
|
|
|
|
tracks = await Util.ytSearch(`${spotifyData.artist} - ${spotifyData.title}`, {
|
|
|
|
user: message.author,
|
2021-04-07 19:05:35 +05:00
|
|
|
player: this,
|
|
|
|
limit: 1
|
2021-04-06 17:58:46 +05:00
|
|
|
});
|
|
|
|
}
|
2021-04-06 17:55:29 +05:00
|
|
|
}
|
|
|
|
}
|
2021-04-06 17:58:46 +05:00
|
|
|
break;
|
2021-04-07 19:05:35 +05:00
|
|
|
|
|
|
|
// todo: make spotify playlist/album load faster
|
|
|
|
case 'spotify_album':
|
|
|
|
case 'spotify_playlist': {
|
|
|
|
this.emit(PlayerEvents.PLAYLIST_PARSE_START, null, message);
|
|
|
|
const playlist = await spotify.getData(query);
|
|
|
|
if (!playlist) return void this.emit(PlayerEvents.NO_RESULTS, message, query);
|
|
|
|
|
|
|
|
// tslint:disable:no-shadowed-variable
|
|
|
|
const tracks = [];
|
|
|
|
|
|
|
|
for (const item of playlist.tracks.items) {
|
|
|
|
const sq =
|
|
|
|
queryType === 'spotify_album'
|
|
|
|
? `${item.artists[0].name} - ${item.name}`
|
|
|
|
: `${item.track.artists[0].name} - ${item.name}`;
|
|
|
|
const data = await Util.ytSearch(sq, {
|
|
|
|
limit: 1,
|
|
|
|
player: this,
|
|
|
|
user: message.author,
|
|
|
|
pl: true
|
|
|
|
});
|
|
|
|
|
|
|
|
if (data[0]) tracks.push(data[0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!tracks.length) return void this.emit(PlayerEvents.NO_RESULTS, message, query);
|
|
|
|
|
|
|
|
const pl = {
|
|
|
|
...playlist,
|
|
|
|
tracks,
|
|
|
|
duration: tracks.reduce((a, c) => a + c.durationMS, 0),
|
|
|
|
thumbnail: playlist.images[0]?.url ?? tracks[0].thumbnail
|
|
|
|
};
|
|
|
|
|
|
|
|
this.emit(PlayerEvents.PLAYLIST_PARSE_END, pl, message);
|
|
|
|
|
|
|
|
if (this.isPlaying(message)) {
|
|
|
|
const queue = this._addTracksToQueue(message, tracks);
|
|
|
|
this.emit(PlayerEvents.PLAYLIST_ADD, message, queue, pl);
|
|
|
|
} else {
|
|
|
|
const track = tracks.shift();
|
|
|
|
const queue = (await this._createQueue(message, track).catch(
|
|
|
|
(e) => void this.emit(PlayerEvents.ERROR, e, message)
|
|
|
|
)) as Queue;
|
|
|
|
this.emit(PlayerEvents.TRACK_START, message, queue.tracks[0], queue);
|
|
|
|
this._addTracksToQueue(message, tracks);
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
case 'youtube_playlist': {
|
|
|
|
this.emit(PlayerEvents.PLAYLIST_PARSE_START, null, message);
|
|
|
|
const playlist = await YouTube.getPlaylist(query);
|
|
|
|
if (!playlist) return void this.emit(PlayerEvents.NO_RESULTS, message, query);
|
|
|
|
|
|
|
|
// @ts-ignore
|
|
|
|
playlist.videos = playlist.videos.map(
|
|
|
|
(data) =>
|
|
|
|
new Track(this, {
|
|
|
|
title: data.title,
|
|
|
|
url: data.url,
|
2021-04-10 22:52:48 +05:00
|
|
|
duration: Util.buildTimeCode(Util.parseMS(data.duration)),
|
2021-04-07 19:05:35 +05:00
|
|
|
description: data.description,
|
|
|
|
thumbnail: data.thumbnail?.displayThumbnailURL(),
|
|
|
|
views: data.views,
|
|
|
|
author: data.channel.name,
|
|
|
|
requestedBy: message.author,
|
|
|
|
fromPlaylist: true,
|
|
|
|
source: 'youtube'
|
|
|
|
})
|
|
|
|
);
|
|
|
|
|
|
|
|
// @ts-ignore
|
|
|
|
playlist.duration = playlist.videos.reduce((a, c) => a + c.durationMS, 0);
|
|
|
|
|
|
|
|
// @ts-ignore
|
|
|
|
playlist.thumbnail = playlist.thumbnail?.url ?? playlist.videos[0].thumbnail;
|
|
|
|
|
|
|
|
// @ts-ignore
|
|
|
|
playlist.requestedBy = message.author;
|
|
|
|
|
|
|
|
this.emit(PlayerEvents.PLAYLIST_PARSE_END, playlist, message);
|
|
|
|
|
|
|
|
// @ts-ignore
|
|
|
|
const tracks = playlist.videos as Track[];
|
|
|
|
|
|
|
|
if (this.isPlaying(message)) {
|
|
|
|
const queue = this._addTracksToQueue(message, tracks);
|
|
|
|
this.emit(PlayerEvents.PLAYLIST_ADD, message, queue, playlist);
|
|
|
|
} else {
|
|
|
|
const track = tracks.shift();
|
|
|
|
const queue = (await this._createQueue(message, track).catch(
|
|
|
|
(e) => void this.emit(PlayerEvents.ERROR, e, message)
|
|
|
|
)) as Queue;
|
|
|
|
this.emit(PlayerEvents.TRACK_START, message, queue.tracks[0], queue);
|
|
|
|
this._addTracksToQueue(message, tracks);
|
|
|
|
}
|
2021-04-09 19:04:39 +05:00
|
|
|
|
|
|
|
return;
|
2021-04-07 19:05:35 +05:00
|
|
|
}
|
|
|
|
case 'soundcloud_playlist': {
|
|
|
|
this.emit(PlayerEvents.PLAYLIST_PARSE_START, null, message);
|
|
|
|
|
|
|
|
const data = await SoundCloud.getPlaylist(query).catch(() => {});
|
|
|
|
if (!data) return void this.emit(PlayerEvents.NO_RESULTS, message, query);
|
|
|
|
|
|
|
|
const res = {
|
|
|
|
id: data.id,
|
|
|
|
title: data.title,
|
|
|
|
tracks: [] as Track[],
|
|
|
|
author: data.author,
|
|
|
|
duration: 0,
|
|
|
|
thumbnail: data.thumbnail,
|
|
|
|
requestedBy: message.author
|
|
|
|
};
|
|
|
|
|
|
|
|
for (const song of data.tracks) {
|
|
|
|
const r = new Track(this, {
|
|
|
|
title: song.title,
|
|
|
|
url: song.url,
|
2021-04-10 22:52:48 +05:00
|
|
|
duration: Util.buildTimeCode(Util.parseMS(song.duration / 1000)),
|
2021-04-07 19:05:35 +05:00
|
|
|
description: song.description,
|
|
|
|
thumbnail: song.thumbnail ?? 'https://soundcloud.com/pwa-icon-192.png',
|
|
|
|
views: song.playCount ?? 0,
|
|
|
|
author: song.author ?? data.author,
|
|
|
|
requestedBy: message.author,
|
|
|
|
fromPlaylist: true,
|
|
|
|
source: 'soundcloud',
|
|
|
|
engine: song
|
|
|
|
});
|
|
|
|
|
|
|
|
res.tracks.push(r);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!res.tracks.length)
|
|
|
|
return this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.PARSE_ERROR, message);
|
|
|
|
res.duration = res.tracks.reduce((a, c) => a + c.durationMS, 0);
|
|
|
|
|
|
|
|
this.emit(PlayerEvents.PLAYLIST_PARSE_END, res, message);
|
|
|
|
|
|
|
|
if (this.isPlaying(message)) {
|
|
|
|
const queue = this._addTracksToQueue(message, res.tracks);
|
|
|
|
this.emit(PlayerEvents.PLAYLIST_ADD, message, queue, res);
|
|
|
|
} else {
|
|
|
|
const track = res.tracks.shift();
|
|
|
|
const queue = (await this._createQueue(message, track).catch(
|
|
|
|
(e) => void this.emit(PlayerEvents.ERROR, e, message)
|
|
|
|
)) as Queue;
|
|
|
|
this.emit(PlayerEvents.TRACK_START, message, queue.tracks[0], queue);
|
|
|
|
this._addTracksToQueue(message, res.tracks);
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
2021-04-06 17:55:29 +05:00
|
|
|
default:
|
|
|
|
tracks = await Util.ytSearch(query, { user: message.author, player: this });
|
|
|
|
}
|
|
|
|
|
2021-04-07 19:05:35 +05:00
|
|
|
if (tracks.length < 1) return void this.emit(PlayerEvents.NO_RESULTS, message, query);
|
|
|
|
if (firstResult || tracks.length === 1) return resolve(tracks[0]);
|
2021-04-06 17:55:29 +05:00
|
|
|
|
|
|
|
const collectorString = `${message.author.id}-${message.channel.id}`;
|
|
|
|
const currentCollector = this._resultsCollectors.get(collectorString);
|
2021-04-06 17:58:46 +05:00
|
|
|
if (currentCollector) currentCollector.stop();
|
2021-04-06 17:55:29 +05:00
|
|
|
|
|
|
|
const collector = message.channel.createMessageCollector((m) => m.author.id === message.author.id, {
|
|
|
|
time: 60000
|
|
|
|
});
|
|
|
|
|
|
|
|
this._resultsCollectors.set(collectorString, collector);
|
|
|
|
|
|
|
|
this.emit(PlayerEvents.SEARCH_RESULTS, message, query, tracks, collector);
|
|
|
|
|
2021-04-06 17:58:46 +05:00
|
|
|
collector.on('collect', ({ content }) => {
|
|
|
|
if (content === 'cancel') {
|
2021-04-06 17:55:29 +05:00
|
|
|
collector.stop();
|
|
|
|
return this.emit(PlayerEvents.SEARCH_CANCEL, message, query, tracks);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isNaN(content) && parseInt(content) >= 1 && parseInt(content) <= tracks.length) {
|
|
|
|
const index = parseInt(content, 10);
|
|
|
|
const track = tracks[index - 1];
|
|
|
|
collector.stop();
|
|
|
|
resolve(track);
|
|
|
|
} else {
|
|
|
|
this.emit(PlayerEvents.SEARCH_INVALID_RESPONSE, message, query, tracks, content, collector);
|
|
|
|
}
|
2021-04-06 17:58:46 +05:00
|
|
|
});
|
2021-04-06 17:55:29 +05:00
|
|
|
|
2021-04-07 19:05:35 +05:00
|
|
|
collector.on('end', (_, reason) => {
|
2021-04-06 17:58:46 +05:00
|
|
|
if (reason === 'time') {
|
2021-04-06 17:55:29 +05:00
|
|
|
this.emit(PlayerEvents.SEARCH_CANCEL, message, query, tracks);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
2021-04-04 22:36:40 +05:00
|
|
|
}
|
2021-04-06 19:06:18 +05:00
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Play a song
|
|
|
|
* @param message The discord.js message object
|
|
|
|
* @param query Search query, can be `Player.Track` instance
|
|
|
|
* @param firstResult If it should play the first result
|
|
|
|
* @example await player.play(message, "never gonna give you up", true)
|
|
|
|
*/
|
2021-04-06 20:38:17 +05:00
|
|
|
async play(message: Message, query: string | Track, firstResult?: boolean): Promise<void> {
|
2021-04-06 19:08:01 +05:00
|
|
|
if (!message) throw new PlayerError('Play function needs message');
|
|
|
|
if (!query) throw new PlayerError('Play function needs search query as a string or Player.Track object');
|
2021-04-06 19:06:18 +05:00
|
|
|
|
|
|
|
if (this._cooldownsTimeout.has(`end_${message.guild.id}`)) {
|
|
|
|
clearTimeout(this._cooldownsTimeout.get(`end_${message.guild.id}`));
|
|
|
|
this._cooldownsTimeout.delete(`end_${message.guild.id}`);
|
|
|
|
}
|
|
|
|
|
2021-04-06 19:08:01 +05:00
|
|
|
if (typeof query === 'string') query = query.replace(/<(.+)>/g, '$1');
|
2021-04-06 19:06:18 +05:00
|
|
|
let track;
|
|
|
|
|
|
|
|
if (query instanceof Track) track = query;
|
|
|
|
else {
|
|
|
|
if (ytdl.validateURL(query)) {
|
|
|
|
const info = await ytdl.getBasicInfo(query).catch(() => {});
|
2021-04-06 20:38:17 +05:00
|
|
|
if (!info) return void this.emit(PlayerEvents.NO_RESULTS, message, query);
|
2021-04-06 19:08:01 +05:00
|
|
|
if (info.videoDetails.isLiveContent && !this.options.enableLive)
|
2021-04-06 20:38:17 +05:00
|
|
|
return void this.emit(
|
2021-04-06 19:08:01 +05:00
|
|
|
PlayerEvents.ERROR,
|
|
|
|
PlayerErrorEventCodes.LIVE_VIDEO,
|
|
|
|
message,
|
|
|
|
new PlayerError('Live video is not enabled!')
|
|
|
|
);
|
2021-04-06 19:06:18 +05:00
|
|
|
const lastThumbnail = info.videoDetails.thumbnails[info.videoDetails.thumbnails.length - 1];
|
|
|
|
|
|
|
|
track = new Track(this, {
|
|
|
|
title: info.videoDetails.title,
|
|
|
|
description: info.videoDetails.description,
|
|
|
|
author: info.videoDetails.author.name,
|
|
|
|
url: info.videoDetails.video_url,
|
|
|
|
thumbnail: lastThumbnail.url,
|
2021-04-10 22:52:48 +05:00
|
|
|
duration: Util.buildTimeCode(Util.parseMS(parseInt(info.videoDetails.lengthSeconds) * 1000)),
|
2021-04-06 19:06:18 +05:00
|
|
|
views: parseInt(info.videoDetails.viewCount),
|
|
|
|
requestedBy: message.author,
|
|
|
|
fromPlaylist: false,
|
2021-04-14 07:59:06 +05:00
|
|
|
source: 'youtube',
|
|
|
|
live: Boolean(info.videoDetails.isLiveContent)
|
2021-04-06 19:06:18 +05:00
|
|
|
});
|
|
|
|
} else {
|
2021-04-08 18:03:04 +05:00
|
|
|
for (const [_, extractor] of this.Extractors) {
|
|
|
|
if (extractor.validate(query)) {
|
|
|
|
const data = await extractor.handle(query);
|
|
|
|
if (data) {
|
|
|
|
track = new Track(this, {
|
|
|
|
title: data.title,
|
|
|
|
description: data.description,
|
2021-04-10 22:52:48 +05:00
|
|
|
duration: Util.buildTimeCode(Util.parseMS(data.duration)),
|
2021-04-08 18:03:04 +05:00
|
|
|
thumbnail: data.thumbnail,
|
|
|
|
author: data.author,
|
|
|
|
views: data.views,
|
|
|
|
engine: data.engine,
|
|
|
|
source: 'arbitrary',
|
|
|
|
fromPlaylist: false,
|
|
|
|
requestedBy: message.author,
|
|
|
|
url: data.url
|
|
|
|
});
|
|
|
|
|
|
|
|
if (extractor.important) break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!track) track = await this._searchTracks(message, query, firstResult);
|
2021-04-06 19:06:18 +05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (track) {
|
|
|
|
if (this.isPlaying(message)) {
|
|
|
|
const queue = this._addTrackToQueue(message, track);
|
|
|
|
this.emit(PlayerEvents.TRACK_ADD, message, queue, queue.tracks[queue.tracks.length - 1]);
|
|
|
|
} else {
|
|
|
|
const queue = await this._createQueue(message, track);
|
|
|
|
if (queue) this.emit(PlayerEvents.TRACK_START, message, queue.tracks[0], queue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Checks if this player is playing in a server
|
|
|
|
* @param message The message object
|
|
|
|
*/
|
2021-04-19 18:32:10 +05:00
|
|
|
isPlaying(message: Message): boolean {
|
2021-04-06 19:08:01 +05:00
|
|
|
return this.queues.some((g) => g.guildID === message.guild.id);
|
2021-04-06 19:06:18 +05:00
|
|
|
}
|
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Returns guild queue object
|
|
|
|
* @param message The message object
|
|
|
|
*/
|
2021-04-19 18:32:10 +05:00
|
|
|
getQueue(message: Message): Queue {
|
2021-04-06 19:08:01 +05:00
|
|
|
return this.queues.find((g) => g.guildID === message.guild.id);
|
2021-04-06 19:06:18 +05:00
|
|
|
}
|
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Sets audio filters in this player
|
|
|
|
* @param message The message object
|
|
|
|
* @param newFilters Audio filters object
|
|
|
|
*/
|
2021-04-06 20:38:17 +05:00
|
|
|
setFilters(message: Message, newFilters: QueueFilters): Promise<void> {
|
|
|
|
return new Promise((resolve) => {
|
|
|
|
const queue = this.queues.find((g) => g.guildID === message.guild.id);
|
|
|
|
if (!queue)
|
|
|
|
this.emit(
|
|
|
|
PlayerEvents.ERROR,
|
|
|
|
PlayerErrorEventCodes.NOT_PLAYING,
|
|
|
|
message,
|
|
|
|
new PlayerError('Not playing')
|
|
|
|
);
|
|
|
|
|
2021-04-14 08:02:02 +05:00
|
|
|
if (queue.playing.raw.live)
|
|
|
|
return void this.emit(
|
|
|
|
PlayerEvents.ERROR,
|
|
|
|
PlayerErrorEventCodes.LIVE_VIDEO,
|
|
|
|
message,
|
|
|
|
new PlayerError('Cannot use setFilters on livestream')
|
|
|
|
);
|
|
|
|
|
2021-04-06 20:38:17 +05:00
|
|
|
Object.keys(newFilters).forEach((filterName) => {
|
|
|
|
// @ts-ignore
|
|
|
|
queue.filters[filterName] = newFilters[filterName];
|
|
|
|
});
|
|
|
|
|
|
|
|
this._playStream(queue, true).then(() => {
|
|
|
|
resolve();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Sets track position
|
|
|
|
* @param message The message object
|
|
|
|
* @param time Time in ms to set
|
|
|
|
* @alias seek
|
|
|
|
*/
|
2021-04-07 19:05:35 +05:00
|
|
|
setPosition(message: Message, time: number): Promise<void> {
|
|
|
|
return new Promise((resolve) => {
|
|
|
|
const queue = this.queues.find((g) => g.guildID === message.guild.id);
|
2021-04-19 19:11:38 +05:00
|
|
|
if (!queue) return this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.NOT_PLAYING, message);
|
2021-04-07 19:05:35 +05:00
|
|
|
|
|
|
|
if (typeof time !== 'number' && !isNaN(time)) time = parseInt(time);
|
|
|
|
if (queue.playing.durationMS >= time) return this.skip(message);
|
|
|
|
if (
|
|
|
|
queue.voiceConnection.dispatcher.streamTime === time ||
|
|
|
|
queue.voiceConnection.dispatcher.streamTime + queue.additionalStreamTime === time
|
|
|
|
)
|
|
|
|
return resolve();
|
|
|
|
if (time < 0) this._playStream(queue, false).then(() => resolve());
|
|
|
|
|
|
|
|
this._playStream(queue, false, time).then(() => resolve());
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Sets track position
|
|
|
|
* @param message The message object
|
|
|
|
* @param time Time in ms to set
|
|
|
|
* @alias setPosition
|
|
|
|
*/
|
2021-04-19 18:32:10 +05:00
|
|
|
seek(message: Message, time: number): Promise<void> {
|
2021-04-07 19:05:35 +05:00
|
|
|
return this.setPosition(message, time);
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Skips current track
|
|
|
|
* @param message The message object
|
|
|
|
*/
|
2021-04-07 19:05:35 +05:00
|
|
|
skip(message: Message): boolean {
|
|
|
|
const queue = this.getQueue(message);
|
|
|
|
if (!queue) {
|
|
|
|
this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.NOT_PLAYING, message);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!queue.voiceConnection || !queue.voiceConnection.dispatcher) {
|
|
|
|
this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.MUSIC_STARTING, message);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
queue.voiceConnection.dispatcher.end();
|
|
|
|
queue.lastSkipped = true;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Moves to a new voice channel
|
|
|
|
* @param message The message object
|
|
|
|
* @param channel New voice channel to move to
|
|
|
|
*/
|
2021-04-19 18:32:10 +05:00
|
|
|
moveTo(message: Message, channel?: VoiceChannel): boolean {
|
2021-04-07 19:05:35 +05:00
|
|
|
if (!channel || channel.type !== 'voice') return;
|
|
|
|
const queue = this.queues.find((g) => g.guildID === message.guild.id);
|
|
|
|
if (!queue) {
|
|
|
|
this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.NOT_PLAYING, message);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!queue.voiceConnection || !queue.voiceConnection.dispatcher) {
|
|
|
|
this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.MUSIC_STARTING, message);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (queue.voiceConnection.channel.id === channel.id) return;
|
|
|
|
|
|
|
|
queue.voiceConnection.dispatcher.pause();
|
|
|
|
channel
|
|
|
|
.join()
|
|
|
|
.then(() => queue.voiceConnection.dispatcher.resume())
|
|
|
|
.catch(() => this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.UNABLE_TO_JOIN, message));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Pause the playback
|
|
|
|
* @param message The message object
|
|
|
|
*/
|
2021-04-19 18:32:10 +05:00
|
|
|
pause(message: Message): boolean {
|
2021-04-09 15:04:36 +05:00
|
|
|
const queue = this.getQueue(message);
|
|
|
|
if (!queue) {
|
|
|
|
this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.NOT_PLAYING, message);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!queue.voiceConnection || !queue.voiceConnection.dispatcher) {
|
|
|
|
this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.MUSIC_STARTING, message);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
queue.voiceConnection.dispatcher.pause();
|
|
|
|
queue.paused = true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Resume the playback
|
|
|
|
* @param message The message object
|
|
|
|
*/
|
2021-04-19 18:32:10 +05:00
|
|
|
resume(message: Message): boolean {
|
2021-04-09 15:04:36 +05:00
|
|
|
const queue = this.getQueue(message);
|
|
|
|
if (!queue) {
|
|
|
|
this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.NOT_PLAYING, message);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!queue.voiceConnection || !queue.voiceConnection.dispatcher) {
|
|
|
|
this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.MUSIC_STARTING, message);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
queue.voiceConnection.dispatcher.resume();
|
|
|
|
queue.paused = false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Stops the player
|
|
|
|
* @param message The message object
|
|
|
|
*/
|
2021-04-19 18:32:10 +05:00
|
|
|
stop(message: Message): boolean {
|
2021-04-09 15:04:36 +05:00
|
|
|
const queue = this.getQueue(message);
|
|
|
|
if (!queue) {
|
|
|
|
this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.NOT_PLAYING, message);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!queue.voiceConnection || !queue.voiceConnection.dispatcher) {
|
|
|
|
this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.MUSIC_STARTING, message);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
queue.stopped = true;
|
|
|
|
queue.tracks = [];
|
|
|
|
if (queue.stream) queue.stream.destroy();
|
|
|
|
queue.voiceConnection.dispatcher.end();
|
|
|
|
if (this.options.leaveOnStop) queue.voiceConnection.channel.leave();
|
|
|
|
this.queues.delete(message.guild.id);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Sets music volume
|
|
|
|
* @param message The message object
|
|
|
|
* @param percent The volume percentage/amount to set
|
|
|
|
*/
|
2021-04-19 18:32:10 +05:00
|
|
|
setVolume(message: Message, percent: number): boolean {
|
2021-04-09 15:04:36 +05:00
|
|
|
const queue = this.getQueue(message);
|
|
|
|
if (!queue) {
|
|
|
|
this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.NOT_PLAYING, message);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!queue.voiceConnection || !queue.voiceConnection.dispatcher) {
|
|
|
|
this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.MUSIC_STARTING, message);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
queue.volume = percent;
|
|
|
|
queue.voiceConnection.dispatcher.setVolumeLogarithmic(queue.calculatedVolume / 200);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Clears the queue
|
|
|
|
* @param message The message object
|
|
|
|
*/
|
2021-04-19 18:32:10 +05:00
|
|
|
clearQueue(message: Message): boolean {
|
2021-04-09 15:04:36 +05:00
|
|
|
const queue = this.getQueue(message);
|
|
|
|
if (!queue) {
|
|
|
|
this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.NOT_PLAYING, message);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
queue.tracks = queue.playing ? [queue.playing] : [];
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Plays previous track
|
|
|
|
* @param message The message object
|
|
|
|
*/
|
2021-04-19 18:32:10 +05:00
|
|
|
back(message: Message): boolean {
|
2021-04-09 15:04:36 +05:00
|
|
|
const queue = this.getQueue(message);
|
|
|
|
if (!queue) {
|
|
|
|
this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.NOT_PLAYING, message);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!queue.voiceConnection || !queue.voiceConnection.dispatcher) {
|
|
|
|
this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.MUSIC_STARTING, message);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
queue.tracks.splice(1, 0, queue.previousTracks.shift());
|
|
|
|
queue.voiceConnection.dispatcher.end();
|
|
|
|
queue.lastSkipped = true;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Sets repeat mode
|
|
|
|
* @param message The message object
|
|
|
|
* @param enabled If it should enable the repeat mode
|
|
|
|
*/
|
2021-04-19 18:32:10 +05:00
|
|
|
setRepeatMode(message: Message, enabled: boolean): boolean {
|
2021-04-09 15:04:36 +05:00
|
|
|
const queue = this.getQueue(message);
|
|
|
|
if (!queue) {
|
|
|
|
this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.NOT_PLAYING, message);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
queue.repeatMode = Boolean(enabled);
|
|
|
|
|
|
|
|
return queue.repeatMode;
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Sets loop mode
|
|
|
|
* @param message The message object
|
|
|
|
* @param enabled If it should enable the loop mode
|
|
|
|
*/
|
2021-04-19 18:32:10 +05:00
|
|
|
setLoopMode(message: Message, enabled: boolean): boolean {
|
2021-04-09 15:04:36 +05:00
|
|
|
const queue = this.getQueue(message);
|
|
|
|
if (!queue) {
|
|
|
|
this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.NOT_PLAYING, message);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
queue.loopMode = Boolean(enabled);
|
|
|
|
|
|
|
|
return queue.loopMode;
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Returns currently playing track
|
|
|
|
* @param message The message object
|
|
|
|
*/
|
2021-04-19 18:32:10 +05:00
|
|
|
nowPlaying(message: Message): Track {
|
2021-04-09 15:04:36 +05:00
|
|
|
const queue = this.getQueue(message);
|
|
|
|
if (!queue) {
|
|
|
|
this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.NOT_PLAYING, message);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
return queue.tracks[0];
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Shuffles the queue
|
|
|
|
* @param message The message object
|
|
|
|
*/
|
2021-04-19 18:32:10 +05:00
|
|
|
shuffle(message: Message): Queue {
|
2021-04-09 15:04:36 +05:00
|
|
|
const queue = this.getQueue(message);
|
|
|
|
if (!queue) {
|
|
|
|
this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.NOT_PLAYING, message);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const currentTrack = queue.tracks.shift();
|
|
|
|
|
|
|
|
for (let i = queue.tracks.length - 1; i > 0; i--) {
|
|
|
|
const j = Math.floor(Math.random() * (i + 1));
|
|
|
|
[queue.tracks[i], queue.tracks[j]] = [queue.tracks[j], queue.tracks[i]];
|
|
|
|
}
|
|
|
|
|
|
|
|
queue.tracks.unshift(currentTrack);
|
|
|
|
|
|
|
|
return queue;
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Removes specified track
|
|
|
|
* @param message The message object
|
|
|
|
* @param track The track object/id to remove
|
|
|
|
*/
|
2021-04-19 18:32:10 +05:00
|
|
|
remove(message: Message, track: Track | number): Track {
|
2021-04-09 15:04:36 +05:00
|
|
|
const queue = this.getQueue(message);
|
|
|
|
if (!queue) {
|
|
|
|
this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.NOT_PLAYING, message);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
let trackFound: Track = null;
|
|
|
|
if (typeof track === 'number') {
|
|
|
|
trackFound = queue.tracks[track];
|
|
|
|
if (trackFound) {
|
|
|
|
queue.tracks = queue.tracks.filter((t) => t !== trackFound);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
trackFound = queue.tracks.find((s) => s === track);
|
|
|
|
if (trackFound) {
|
|
|
|
queue.tracks = queue.tracks.filter((s) => s !== trackFound);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return trackFound;
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Returns time code of currently playing song
|
|
|
|
* @param message The message object
|
|
|
|
* @param queueTime If it should make the time code of the whole queue
|
|
|
|
*/
|
2021-04-19 18:32:10 +05:00
|
|
|
getTimeCode(message: Message, queueTime?: boolean): { current: string; end: string } {
|
2021-04-09 15:04:36 +05:00
|
|
|
const queue = this.getQueue(message);
|
|
|
|
if (!queue) return;
|
|
|
|
|
|
|
|
const previousTracksTime =
|
|
|
|
queue.previousTracks.length > 0 ? queue.previousTracks.reduce((p, c) => p + c.durationMS, 0) : 0;
|
|
|
|
const currentStreamTime = queueTime ? previousTracksTime + queue.currentStreamTime : queue.currentStreamTime;
|
|
|
|
const totalTracksTime = queue.totalTime;
|
|
|
|
const totalTime = queueTime ? previousTracksTime + totalTracksTime : queue.playing.durationMS;
|
|
|
|
|
|
|
|
const currentTimecode = Util.buildTimeCode(Util.parseMS(currentStreamTime));
|
|
|
|
const endTimecode = Util.buildTimeCode(Util.parseMS(totalTime));
|
|
|
|
|
|
|
|
return {
|
|
|
|
current: currentTimecode,
|
|
|
|
end: endTimecode
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:59:14 +05:00
|
|
|
/**
|
|
|
|
* Creates progressbar
|
|
|
|
* @param message The message object
|
|
|
|
* @param options Progressbar options
|
|
|
|
*/
|
2021-04-19 18:32:10 +05:00
|
|
|
createProgressBar(message: Message, options?: PlayerProgressbarOptions): string {
|
2021-04-09 15:04:36 +05:00
|
|
|
const queue = this.getQueue(message);
|
|
|
|
if (!queue) return;
|
|
|
|
|
|
|
|
const previousTracksTime =
|
|
|
|
queue.previousTracks.length > 0 ? queue.previousTracks.reduce((p, c) => p + c.durationMS, 0) : 0;
|
|
|
|
const currentStreamTime = options?.queue
|
|
|
|
? previousTracksTime + queue.currentStreamTime
|
|
|
|
: queue.currentStreamTime;
|
|
|
|
const totalTracksTime = queue.totalTime;
|
|
|
|
const totalTime = options?.queue ? previousTracksTime + totalTracksTime : queue.playing.durationMS;
|
|
|
|
const length =
|
|
|
|
typeof options?.length === 'number'
|
|
|
|
? options?.length <= 0 || options?.length === Infinity
|
|
|
|
? 15
|
|
|
|
: options?.length
|
2021-04-09 19:04:39 +05:00
|
|
|
: 15;
|
2021-04-09 15:04:36 +05:00
|
|
|
|
|
|
|
const index = Math.round((currentStreamTime / totalTime) * length);
|
|
|
|
const indicator = '🔘';
|
|
|
|
const line = '▬';
|
|
|
|
|
|
|
|
if (index >= 1 && index <= length) {
|
|
|
|
const bar = line.repeat(length - 1).split('');
|
|
|
|
bar.splice(index, 0, indicator);
|
|
|
|
if (Boolean(options?.timecodes)) {
|
|
|
|
const currentTimecode = Util.buildTimeCode(Util.parseMS(currentStreamTime));
|
|
|
|
const endTimecode = Util.buildTimeCode(Util.parseMS(totalTime));
|
|
|
|
return `${currentTimecode} ┃ ${bar.join('')} ┃ ${endTimecode}`;
|
|
|
|
} else {
|
|
|
|
return `${bar.join('')}`;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (Boolean(options?.timecodes)) {
|
|
|
|
const currentTimecode = Util.buildTimeCode(Util.parseMS(currentStreamTime));
|
|
|
|
const endTimecode = Util.buildTimeCode(Util.parseMS(totalTime));
|
2021-04-09 15:13:07 +05:00
|
|
|
return `${currentTimecode} ┃ ${indicator}${line.repeat(length - 1)} ┃ ${endTimecode}`;
|
2021-04-09 15:04:36 +05:00
|
|
|
} else {
|
2021-04-09 15:13:07 +05:00
|
|
|
return `${indicator}${line.repeat(length - 1)}`;
|
2021-04-09 15:04:36 +05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-10 10:55:17 +05:00
|
|
|
/**
|
|
|
|
* Gets lyrics of a song
|
|
|
|
* @param query Search query
|
2021-04-10 11:13:23 +05:00
|
|
|
* @example const lyrics = await player.lyrics("alan walker faded")
|
|
|
|
* message.channel.send(lyrics.lyrics);
|
2021-04-10 10:55:17 +05:00
|
|
|
*/
|
2021-04-19 18:32:10 +05:00
|
|
|
async lyrics(query: string): Promise<LyricsData> {
|
2021-04-19 13:33:34 +05:00
|
|
|
const extractor = Util.require('@discord-player/extractor');
|
|
|
|
if (!extractor) throw new PlayerError("Cannot call 'Player.lyrics()' without '@discord-player/extractor'");
|
|
|
|
|
|
|
|
const data = await extractor.Lyrics(query);
|
2021-04-10 10:55:17 +05:00
|
|
|
if (Array.isArray(data)) return null;
|
|
|
|
|
2021-04-19 18:32:10 +05:00
|
|
|
return data;
|
2021-04-10 10:55:17 +05:00
|
|
|
}
|
|
|
|
|
2021-04-11 18:27:48 +05:00
|
|
|
/**
|
|
|
|
* Toggle autoplay for youtube streams
|
|
|
|
* @param message The message object
|
|
|
|
* @param enable Enable/Disable autoplay
|
|
|
|
*/
|
|
|
|
setAutoplay(message: Message, enable: boolean): void {
|
|
|
|
const queue = this.getQueue(message);
|
|
|
|
if (!queue) return void this.emit(PlayerEvents.ERROR, PlayerErrorEventCodes.NOT_PLAYING, message);
|
|
|
|
|
|
|
|
queue.autoPlay = Boolean(enable);
|
|
|
|
}
|
|
|
|
|
2021-04-17 09:01:42 +05:00
|
|
|
/**
|
|
|
|
* Player stats
|
|
|
|
*/
|
2021-04-17 09:01:06 +05:00
|
|
|
getStats(): PlayerStats {
|
|
|
|
return {
|
|
|
|
uptime: this.client.uptime,
|
|
|
|
connections: this.client.voice.connections.size,
|
2021-04-17 19:11:18 +05:00
|
|
|
users: this.client.voice.connections.reduce(
|
|
|
|
(a, c) => a + c.channel.members.filter((a) => a.user.id !== this.client.user.id).size,
|
|
|
|
0
|
|
|
|
),
|
2021-04-17 09:01:06 +05:00
|
|
|
queues: this.queues.size,
|
|
|
|
extractors: this.Extractors.size,
|
|
|
|
versions: {
|
|
|
|
ffmpeg: Util.getFFmpegVersion(),
|
|
|
|
node: process.version,
|
|
|
|
v8: process.versions.v8
|
|
|
|
},
|
|
|
|
system: {
|
|
|
|
arch: process.arch,
|
|
|
|
platform: process.platform,
|
|
|
|
cpu: os.cpus().length,
|
|
|
|
memory: {
|
|
|
|
total: (process.memoryUsage().heapTotal / 1024 / 1024).toFixed(2),
|
|
|
|
usage: (process.memoryUsage().heapUsed / 1024 / 1024).toFixed(2),
|
|
|
|
rss: (process.memoryUsage().rss / 1024 / 1024).toFixed(2),
|
|
|
|
arrayBuffers: (process.memoryUsage().arrayBuffers / 1024 / 1024).toFixed(2)
|
|
|
|
},
|
|
|
|
uptime: process.uptime()
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2021-04-19 18:32:10 +05:00
|
|
|
private _handleVoiceStateUpdate(oldState: VoiceState, newState: VoiceState): void {
|
2021-04-09 15:04:36 +05:00
|
|
|
const queue = this.queues.find((g) => g.guildID === oldState.guild.id);
|
|
|
|
if (!queue) return;
|
|
|
|
|
|
|
|
if (newState.member.id === this.client.user.id && !newState.channelID) {
|
|
|
|
queue.stream.destroy();
|
|
|
|
this.queues.delete(newState.guild.id);
|
|
|
|
this.emit(PlayerEvents.BOT_DISCONNECT, queue.firstMessage);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!queue.voiceConnection || !queue.voiceConnection.channel) return;
|
|
|
|
if (!this.options.leaveOnEmpty) return;
|
|
|
|
|
|
|
|
if (!oldState.channelID || newState.channelID) {
|
|
|
|
const emptyTimeout = this._cooldownsTimeout.get(`empty_${oldState.guild.id}`);
|
|
|
|
const channelEmpty = Util.isVoiceEmpty(queue.voiceConnection.channel);
|
|
|
|
if (!channelEmpty && emptyTimeout) {
|
|
|
|
clearTimeout(emptyTimeout);
|
|
|
|
this._cooldownsTimeout.delete(`empty_${oldState.guild.id}`);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (!Util.isVoiceEmpty(queue.voiceConnection.channel)) return;
|
|
|
|
const timeout = setTimeout(() => {
|
|
|
|
if (!Util.isVoiceEmpty(queue.voiceConnection.channel)) return;
|
|
|
|
if (!this.queues.has(queue.guildID)) return;
|
|
|
|
queue.voiceConnection.channel.leave();
|
|
|
|
this.queues.delete(queue.guildID);
|
|
|
|
this.emit(PlayerEvents.CHANNEL_EMPTY, queue.firstMessage, queue);
|
|
|
|
}, this.options.leaveOnEmptyCooldown || 0);
|
|
|
|
this._cooldownsTimeout.set(`empty_${oldState.guild.id}`, timeout);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-19 18:32:10 +05:00
|
|
|
private _addTrackToQueue(message: Message, track: Track): Queue {
|
2021-04-06 19:06:18 +05:00
|
|
|
const queue = this.getQueue(message);
|
2021-04-06 19:08:01 +05:00
|
|
|
if (!queue)
|
|
|
|
this.emit(
|
|
|
|
PlayerEvents.ERROR,
|
|
|
|
PlayerErrorEventCodes.NOT_PLAYING,
|
|
|
|
message,
|
|
|
|
new PlayerError('Player is not available in this server')
|
|
|
|
);
|
2021-04-06 19:06:18 +05:00
|
|
|
if (!track || !(track instanceof Track)) throw new PlayerError('No track specified to add to the queue');
|
|
|
|
queue.tracks.push(track);
|
|
|
|
return queue;
|
|
|
|
}
|
|
|
|
|
2021-04-19 18:32:10 +05:00
|
|
|
private _addTracksToQueue(message: Message, tracks: Track[]): Queue {
|
2021-04-07 19:05:35 +05:00
|
|
|
const queue = this.getQueue(message);
|
|
|
|
if (!queue)
|
|
|
|
throw new PlayerError(
|
|
|
|
'Cannot add tracks to queue because no song is currently being played on the server.'
|
|
|
|
);
|
|
|
|
queue.tracks.push(...tracks);
|
|
|
|
return queue;
|
|
|
|
}
|
|
|
|
|
2021-04-06 19:06:18 +05:00
|
|
|
private _createQueue(message: Message, track: Track): Promise<Queue> {
|
|
|
|
return new Promise((resolve) => {
|
2021-04-06 19:08:01 +05:00
|
|
|
const channel = message.member.voice ? message.member.voice.channel : null;
|
|
|
|
if (!channel)
|
2021-04-07 19:05:35 +05:00
|
|
|
return void this.emit(
|
2021-04-06 19:08:01 +05:00
|
|
|
PlayerEvents.ERROR,
|
|
|
|
PlayerErrorEventCodes.NOT_CONNECTED,
|
|
|
|
message,
|
|
|
|
new PlayerError('Voice connection is not available in this server!')
|
|
|
|
);
|
2021-04-06 19:06:18 +05:00
|
|
|
|
2021-04-11 17:51:50 +05:00
|
|
|
const queue = new Queue(this, message);
|
2021-04-06 19:06:18 +05:00
|
|
|
this.queues.set(message.guild.id, queue);
|
|
|
|
|
2021-04-06 19:08:01 +05:00
|
|
|
channel
|
|
|
|
.join()
|
|
|
|
.then((connection) => {
|
|
|
|
this.emit(PlayerEvents.CONNECTION_CREATE, message, connection);
|
|
|
|
|
|
|
|
queue.voiceConnection = connection;
|
|
|
|
if (this.options.autoSelfDeaf) connection.voice.setSelfDeaf(true);
|
|
|
|
queue.tracks.push(track);
|
|
|
|
this.emit(PlayerEvents.QUEUE_CREATE, message, queue);
|
|
|
|
resolve(queue);
|
2021-04-06 20:38:17 +05:00
|
|
|
this._playTrack(queue, true);
|
2021-04-06 19:08:01 +05:00
|
|
|
})
|
|
|
|
.catch((err) => {
|
|
|
|
this.queues.delete(message.guild.id);
|
|
|
|
this.emit(
|
|
|
|
PlayerEvents.ERROR,
|
|
|
|
PlayerErrorEventCodes.UNABLE_TO_JOIN,
|
|
|
|
message,
|
|
|
|
new PlayerError(err.message ?? err)
|
|
|
|
);
|
|
|
|
});
|
2021-04-06 19:06:18 +05:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-04-06 20:38:17 +05:00
|
|
|
private async _playTrack(queue: Queue, firstPlay: boolean): Promise<void> {
|
2021-04-06 19:06:18 +05:00
|
|
|
if (queue.stopped) return;
|
|
|
|
|
2021-04-11 18:27:48 +05:00
|
|
|
if (!queue.autoPlay && queue.tracks.length === 1 && !queue.loopMode && !queue.repeatMode && !firstPlay) {
|
2021-04-06 19:06:18 +05:00
|
|
|
if (this.options.leaveOnEnd && !queue.stopped) {
|
2021-04-06 19:08:01 +05:00
|
|
|
this.queues.delete(queue.guildID);
|
2021-04-06 19:06:18 +05:00
|
|
|
const timeout = setTimeout(() => {
|
|
|
|
queue.voiceConnection.channel.leave();
|
|
|
|
}, this.options.leaveOnEndCooldown || 0);
|
|
|
|
this._cooldownsTimeout.set(`end_${queue.guildID}`, timeout);
|
|
|
|
}
|
|
|
|
|
2021-04-06 19:08:01 +05:00
|
|
|
this.queues.delete(queue.guildID);
|
2021-04-06 19:06:18 +05:00
|
|
|
|
|
|
|
if (queue.stopped) {
|
2021-04-06 20:38:17 +05:00
|
|
|
return void this.emit(PlayerEvents.MUSIC_STOP, queue.firstMessage);
|
2021-04-06 19:06:18 +05:00
|
|
|
}
|
|
|
|
|
2021-04-06 20:38:17 +05:00
|
|
|
return void this.emit(PlayerEvents.QUEUE_END, queue.firstMessage, queue);
|
2021-04-06 19:06:18 +05:00
|
|
|
}
|
|
|
|
|
2021-04-11 18:27:48 +05:00
|
|
|
if (queue.autoPlay && !queue.repeatMode && !firstPlay) {
|
|
|
|
const oldTrack = queue.tracks.shift();
|
|
|
|
|
|
|
|
const info = oldTrack.raw.source === 'youtube' ? await ytdl.getInfo(oldTrack.url).catch((e) => {}) : null;
|
|
|
|
if (info) {
|
2021-04-14 07:59:06 +05:00
|
|
|
const res = await Util.ytSearch(info.related_videos[0].title, {
|
|
|
|
player: this,
|
|
|
|
limit: 1,
|
|
|
|
user: oldTrack.requestedBy
|
|
|
|
})
|
2021-04-11 18:27:48 +05:00
|
|
|
.then((v) => v[0])
|
|
|
|
.catch((e) => {});
|
|
|
|
|
|
|
|
if (res) {
|
|
|
|
queue.tracks.push(res);
|
|
|
|
if (queue.loopMode) queue.tracks.push(oldTrack);
|
|
|
|
queue.previousTracks.push(oldTrack);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (!queue.autoPlay && !queue.repeatMode && !firstPlay) {
|
2021-04-06 19:06:18 +05:00
|
|
|
const oldTrack = queue.tracks.shift();
|
|
|
|
if (queue.loopMode) queue.tracks.push(oldTrack);
|
|
|
|
queue.previousTracks.push(oldTrack);
|
|
|
|
}
|
|
|
|
|
|
|
|
const track = queue.playing;
|
|
|
|
|
|
|
|
queue.lastSkipped = false;
|
|
|
|
this._playStream(queue, false).then(() => {
|
|
|
|
if (!firstPlay) this.emit(PlayerEvents.TRACK_START, queue.firstMessage, track, queue);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
private _playStream(queue: Queue, updateFilter: boolean, seek?: number): Promise<void> {
|
|
|
|
return new Promise(async (resolve) => {
|
2021-04-11 18:27:48 +05:00
|
|
|
const ffmpeg = Util.checkFFmpeg();
|
|
|
|
if (!ffmpeg) return;
|
2021-04-06 19:06:18 +05:00
|
|
|
|
2021-04-06 19:08:01 +05:00
|
|
|
const seekTime =
|
|
|
|
typeof seek === 'number'
|
|
|
|
? seek
|
|
|
|
: updateFilter
|
|
|
|
? queue.voiceConnection.dispatcher.streamTime + queue.additionalStreamTime
|
|
|
|
: undefined;
|
2021-04-06 19:06:18 +05:00
|
|
|
const encoderArgsFilters: string[] = [];
|
|
|
|
|
|
|
|
Object.keys(queue.filters).forEach((filterName) => {
|
|
|
|
// @ts-ignore
|
|
|
|
if (queue.filters[filterName]) {
|
|
|
|
// @ts-ignore
|
|
|
|
encoderArgsFilters.push(this.filters[filterName]);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2021-04-06 20:38:17 +05:00
|
|
|
let encoderArgs: string[] = [];
|
2021-04-06 19:06:18 +05:00
|
|
|
if (encoderArgsFilters.length < 1) {
|
2021-04-06 19:08:01 +05:00
|
|
|
encoderArgs = [];
|
2021-04-06 19:06:18 +05:00
|
|
|
} else {
|
2021-04-06 19:08:01 +05:00
|
|
|
encoderArgs = ['-af', encoderArgsFilters.join(',')];
|
2021-04-06 19:06:18 +05:00
|
|
|
}
|
|
|
|
|
|
|
|
let newStream: any;
|
2021-04-06 19:08:01 +05:00
|
|
|
if (queue.playing.raw.source === 'youtube') {
|
2021-04-06 19:06:18 +05:00
|
|
|
newStream = ytdl(queue.playing.url, {
|
2021-04-14 08:00:06 +05:00
|
|
|
filter: queue.playing.raw.live ? undefined : 'audioonly',
|
2021-04-06 19:06:18 +05:00
|
|
|
opusEncoded: true,
|
2021-04-14 07:59:06 +05:00
|
|
|
encoderArgs: queue.playing.raw.live ? [] : encoderArgs,
|
2021-04-06 19:06:18 +05:00
|
|
|
seek: seekTime / 1000,
|
2021-04-06 19:08:01 +05:00
|
|
|
// tslint:disable-next-line:no-bitwise
|
2021-04-06 19:06:18 +05:00
|
|
|
highWaterMark: 1 << 25,
|
|
|
|
...this.options.ytdlDownloadOptions
|
|
|
|
});
|
|
|
|
} else {
|
2021-04-06 19:08:01 +05:00
|
|
|
newStream = ytdl.arbitraryStream(
|
|
|
|
queue.playing.raw.source === 'soundcloud'
|
|
|
|
? await queue.playing.raw.engine.downloadProgressive()
|
|
|
|
: queue.playing.raw.engine,
|
|
|
|
{
|
|
|
|
opusEncoded: true,
|
|
|
|
encoderArgs,
|
|
|
|
seek: seekTime / 1000
|
|
|
|
}
|
|
|
|
);
|
2021-04-06 19:06:18 +05:00
|
|
|
}
|
|
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
if (queue.stream) queue.stream.destroy();
|
|
|
|
queue.stream = newStream;
|
|
|
|
queue.voiceConnection.play(newStream, {
|
|
|
|
type: 'opus',
|
2021-04-07 19:25:45 +05:00
|
|
|
bitrate: 'auto',
|
|
|
|
volume: Util.isRepl() ? false : undefined
|
2021-04-06 19:06:18 +05:00
|
|
|
});
|
|
|
|
|
|
|
|
if (seekTime) {
|
|
|
|
queue.additionalStreamTime = seekTime;
|
|
|
|
}
|
|
|
|
queue.voiceConnection.dispatcher.setVolumeLogarithmic(queue.calculatedVolume / 200);
|
|
|
|
queue.voiceConnection.dispatcher.on('start', () => {
|
|
|
|
resolve();
|
|
|
|
});
|
|
|
|
|
|
|
|
queue.voiceConnection.dispatcher.on('finish', () => {
|
|
|
|
queue.additionalStreamTime = 0;
|
|
|
|
return this._playTrack(queue, false);
|
|
|
|
});
|
|
|
|
|
|
|
|
newStream.on('error', (error: Error) => {
|
|
|
|
if (error.message.toLowerCase().includes('video unavailable')) {
|
2021-04-06 19:08:01 +05:00
|
|
|
this.emit(
|
|
|
|
PlayerEvents.ERROR,
|
|
|
|
PlayerErrorEventCodes.VIDEO_UNAVAILABLE,
|
|
|
|
queue.firstMessage,
|
|
|
|
queue.playing,
|
|
|
|
error
|
|
|
|
);
|
2021-04-06 19:06:18 +05:00
|
|
|
this._playTrack(queue, false);
|
|
|
|
} else {
|
|
|
|
this.emit(PlayerEvents.ERROR, error, queue.firstMessage, error);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}, 1000);
|
|
|
|
});
|
|
|
|
}
|
2021-04-04 22:44:45 +05:00
|
|
|
}
|
2021-04-06 20:38:17 +05:00
|
|
|
|
|
|
|
export default Player;
|
2021-04-19 18:32:10 +05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Emitted when a track starts
|
|
|
|
* @event Player#trackStart
|
|
|
|
* @param {Discord.Message} message
|
|
|
|
* @param {Track} track
|
|
|
|
* @param {Queue} queue
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Emitted when a playlist is started
|
|
|
|
* @event Player#queueCreate
|
|
|
|
* @param {Discord.Message} message
|
|
|
|
* @param {Queue} queue
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Emitted when the bot is awaiting search results
|
|
|
|
* @event Player#searchResults
|
|
|
|
* @param {Discord.Message} message
|
|
|
|
* @param {string} query
|
|
|
|
* @param {Track[]} tracks
|
|
|
|
* @param {Discord.Collector} collector
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Emitted when the user has sent an invalid response for search results
|
|
|
|
* @event Player#searchInvalidResponse
|
|
|
|
* @param {Discord.Message} message
|
|
|
|
* @param {string} query
|
|
|
|
* @param {Track[]} tracks
|
|
|
|
* @param {string} invalidResponse
|
|
|
|
* @param {Discord.MessageCollector} collector
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Emitted when the bot has stopped awaiting search results (timeout)
|
|
|
|
* @event Player#searchCancel
|
|
|
|
* @param {Discord.Message} message
|
|
|
|
* @param {string} query
|
|
|
|
* @param {Track[]} tracks
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Emitted when the bot can't find related results to the query
|
|
|
|
* @event Player#noResults
|
|
|
|
* @param {Discord.Message} message
|
|
|
|
* @param {string} query
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Emitted when the bot is disconnected from the channel
|
|
|
|
* @event Player#botDisconnect
|
|
|
|
* @param {Discord.Message} message
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Emitted when the channel of the bot is empty
|
|
|
|
* @event Player#channelEmpty
|
|
|
|
* @param {Discord.Message} message
|
|
|
|
* @param {Queue} queue
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Emitted when the queue of the server is ended
|
|
|
|
* @event Player#queueEnd
|
|
|
|
* @param {Discord.Message} message
|
|
|
|
* @param {Queue} queue
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Emitted when a track is added to the queue
|
|
|
|
* @event Player#trackAdd
|
|
|
|
* @param {Discord.Message} message
|
|
|
|
* @param {Queue} queue
|
|
|
|
* @param {Track} track
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Emitted when a playlist is added to the queue
|
|
|
|
* @event Player#playlistAdd
|
|
|
|
* @param {Discord.Message} message
|
|
|
|
* @param {Queue} queue
|
|
|
|
* @param {Object} playlist
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Emitted when an error is triggered
|
|
|
|
* @event Player#error
|
|
|
|
* @param {string} error It can be `NotConnected`, `UnableToJoin`, `NotPlaying`, `ParseError`, `LiveVideo` or `VideoUnavailable`.
|
|
|
|
* @param {Discord.Message} message
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Emitted when discord-player attempts to parse playlist contents (mostly soundcloud playlists)
|
|
|
|
* @event Player#playlistParseStart
|
|
|
|
* @param {Object} playlist Raw playlist (unparsed)
|
|
|
|
* @param {Discord.Message} message The message
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Emitted when discord-player finishes parsing playlist contents (mostly soundcloud playlists)
|
|
|
|
* @event Player#playlistParseEnd
|
|
|
|
* @param {Object} playlist The playlist data (parsed)
|
|
|
|
* @param {Discord.Message} message The message
|
2021-04-19 18:32:28 +05:00
|
|
|
*/
|