BetterDiscordAddons/Plugins/SpotifyControls/SpotifyControls.plugin.js

1110 lines
62 KiB
JavaScript
Raw Normal View History

2020-10-20 23:25:34 +02:00
/**
* @name SpotifyControls
2021-03-05 13:26:41 +01:00
* @author DevilBro
2020-10-20 23:25:34 +02:00
* @authorId 278543574059057154
2023-11-01 11:23:33 +01:00
* @version 1.3.8
2021-03-05 13:26:41 +01:00
* @description Adds a Control Panel while listening to Spotify on a connected Account
2020-10-20 23:25:34 +02:00
* @invite Jx3TjNS
* @donate https://www.paypal.me/MircoWittrien
* @patreon https://www.patreon.com/MircoWittrien
2021-03-09 15:10:55 +01:00
* @website https://mwittrien.github.io/
* @source https://github.com/mwittrien/BetterDiscordAddons/tree/master/Plugins/SpotifyControls/
2021-03-10 09:17:37 +01:00
* @updateUrl https://mwittrien.github.io/BetterDiscordAddons/Plugins/SpotifyControls/SpotifyControls.plugin.js
2020-10-20 23:25:34 +02:00
*/
2020-07-19 16:59:08 +02:00
2020-09-19 20:49:33 +02:00
module.exports = (_ => {
2022-09-01 14:40:11 +02:00
const changeLog = {
2022-09-02 12:37:10 +02:00
2020-09-19 20:49:33 +02:00
};
2020-11-13 19:47:44 +01:00
2022-02-05 21:14:17 +01:00
return !window.BDFDB_Global || (!window.BDFDB_Global.loaded && !window.BDFDB_Global.started) ? class {
2022-09-01 14:55:22 +02:00
constructor (meta) {for (let key in meta) this[key] = meta[key];}
getName () {return this.name;}
getAuthor () {return this.author;}
getVersion () {return this.version;}
getDescription () {return `The Library Plugin needed for ${this.name} is missing. Open the Plugin Settings to download it. \n\n${this.description}`;}
2021-02-01 17:13:13 +01:00
downloadLibrary () {
2023-11-18 18:31:04 +01:00
BdApi.Net.fetch("https://mwittrien.github.io/BetterDiscordAddons/Library/0BDFDB.plugin.js").then(r => {
if (!r || r.status != 200) throw new Error();
else return r.text();
}).then(b => {
if (!b) throw new Error();
else return require("fs").writeFile(require("path").join(BdApi.Plugins.folder, "0BDFDB.plugin.js"), b, _ => BdApi.showToast("Finished downloading BDFDB Library", {type: "success"}));
}).catch(error => {
BdApi.alert("Error", "Could not download BDFDB Library Plugin. Try again later or download it manually from GitHub: https://mwittrien.github.io/downloader/?library");
2021-02-01 17:13:13 +01:00
});
}
2020-09-19 20:49:33 +02:00
2021-01-06 12:38:36 +01:00
load () {
2020-11-19 16:51:14 +01:00
if (!window.BDFDB_Global || !Array.isArray(window.BDFDB_Global.pluginQueue)) window.BDFDB_Global = Object.assign({}, window.BDFDB_Global, {pluginQueue: []});
2020-09-19 20:49:33 +02:00
if (!window.BDFDB_Global.downloadModal) {
window.BDFDB_Global.downloadModal = true;
2022-09-01 14:55:22 +02:00
BdApi.showConfirmationModal("Library Missing", `The Library Plugin needed for ${this.name} is missing. Please click "Download Now" to install it.`, {
2020-09-19 20:49:33 +02:00
confirmText: "Download Now",
cancelText: "Cancel",
onCancel: _ => {delete window.BDFDB_Global.downloadModal;},
2020-09-20 08:15:13 +02:00
onConfirm: _ => {
delete window.BDFDB_Global.downloadModal;
2021-02-01 17:13:13 +01:00
this.downloadLibrary();
2020-09-20 08:15:13 +02:00
}
2020-07-23 21:51:49 +02:00
});
2020-07-19 17:21:45 +02:00
}
2022-09-01 14:55:22 +02:00
if (!window.BDFDB_Global.pluginQueue.includes(this.name)) window.BDFDB_Global.pluginQueue.push(this.name);
2020-10-09 21:09:35 +02:00
}
2021-01-06 12:38:36 +01:00
start () {this.load();}
stop () {}
getSettingsPanel () {
2020-11-28 23:12:09 +01:00
let template = document.createElement("template");
2022-09-01 14:55:22 +02:00
template.innerHTML = `<div style="color: var(--header-primary); font-size: 16px; font-weight: 300; white-space: pre; line-height: 22px;">The Library Plugin needed for ${this.name} is missing.\nPlease click <a style="font-weight: 500;">Download Now</a> to install it.</div>`;
2021-02-01 17:13:13 +01:00
template.content.firstElementChild.querySelector("a").addEventListener("click", this.downloadLibrary);
2020-11-28 23:12:09 +01:00
return template.content.firstElementChild;
}
2020-10-09 21:09:35 +02:00
} : (([Plugin, BDFDB]) => {
2020-09-19 20:49:33 +02:00
var _this;
2021-05-03 21:22:44 +02:00
var controls;
2021-05-18 19:50:28 +02:00
var starting, lastSong, showActivity, currentVolume, lastVolume, stopTime, previousIsClicked, previousDoubleTimeout;
2021-05-03 21:22:44 +02:00
var timelineTimeout, timelineDragging, updateInterval;
2020-09-19 20:49:33 +02:00
var playbackState = {};
const repeatStates = [
"off",
"context",
"track"
];
const SpotifyControlsComponent = class SpotifyControls extends BdApi.React.Component {
2021-01-20 16:00:29 +01:00
componentDidMount() {
2020-09-19 20:49:33 +02:00
controls = this;
}
request(socket, device, type, data) {
return new Promise(callback => {
2022-10-13 13:15:23 +02:00
let method = "put";
2020-09-19 20:49:33 +02:00
switch (type) {
case "next":
case "previous":
2022-10-13 13:15:23 +02:00
method = "post";
2020-09-19 20:49:33 +02:00
break;
case "get":
type = "";
2022-10-13 13:15:23 +02:00
method = "get";
2020-09-19 20:49:33 +02:00
break;
};
2022-11-03 12:27:27 +01:00
BDFDB.LibraryRequires.request(`https://api.spotify.com/v1/me/player${type ? "/" + type : ""}`, {method: method, form: Object.assign({device_id: device.id}, data), headers: {authorization: `Bearer ${socket.accessToken}`}}, (error, response, result) => {
2020-11-16 10:06:21 +01:00
if (response && response.statusCode == 401) {
2020-09-19 20:49:33 +02:00
BDFDB.LibraryModules.SpotifyUtils.getAccessToken(socket.accountId).then(promiseResult => {
2022-09-27 16:53:10 +02:00
let newSocketDevice = BDFDB.LibraryStores.SpotifyStore.getActiveSocketAndDevice();
2020-09-19 20:49:33 +02:00
this.request(newSocketDevice.socket, newSocketDevice.device, type, data).then(_ => {
try {callback(JSON.parse(result));}
catch (err) {callback({});}
});
});
}
2022-06-17 20:38:21 +02:00
else if (response && response.statusCode == 404) {
this.props.noDevice = true;
BDFDB.NotificationUtils.toast(_this.labels.nodevice_text, {type: "danger", timeout: 60000});
2022-06-29 17:37:27 +02:00
lastSong = null;
2022-06-17 20:38:21 +02:00
callback({});
}
2020-09-19 20:49:33 +02:00
else {
try {callback(JSON.parse(result));}
catch (err) {callback({});}
}
});
});
2020-07-23 16:12:08 +02:00
}
2021-01-20 16:00:29 +01:00
render() {
2022-09-27 16:53:10 +02:00
let socketDevice = BDFDB.LibraryStores.SpotifyStore.getActiveSocketAndDevice();
2022-06-29 17:37:27 +02:00
if (this.props.song) this.props.noDevice = false;
2022-06-17 20:38:21 +02:00
if (!socketDevice || this.props.noDevice) return null;
2020-09-19 20:49:33 +02:00
if (this.props.song) {
playbackState.is_playing = true;
let fetchState = !BDFDB.equals(this.props.song, lastSong);
lastSong = this.props.song;
stopTime = null;
if (fetchState) this.request(socketDevice.socket, socketDevice.device, "get").then(response => {
playbackState = Object.assign({}, response);
BDFDB.ReactUtils.forceUpdate(this);
});
}
else if (!stopTime && lastSong) {
playbackState.is_playing = false;
stopTime = new Date();
}
if (!lastSong) return null;
2021-05-18 19:50:28 +02:00
2022-09-27 19:46:53 +02:00
let coverSrc = BDFDB.LibraryModules.ApplicationAssetUtils.getAssetImage(lastSong.application_id, lastSong.assets.large_image);
2022-09-27 16:53:10 +02:00
let connection = (BDFDB.LibraryStores.ConnectedAccountsStore.getAccounts().find(n => n.type == "spotify") || {});
2021-08-18 17:17:30 +02:00
showActivity = showActivity != undefined ? showActivity : (connection.show_activity || connection.showActivity);
2021-05-18 19:50:28 +02:00
currentVolume = this.props.draggingVolume ? currentVolume : socketDevice.device.volume_percent;
2020-09-19 20:49:33 +02:00
return BDFDB.ReactUtils.createElement("div", {
2023-07-18 23:40:08 +02:00
className: BDFDB.DOMUtils.formatClassName(BDFDB.disCN._spotifycontrolscontainer, !this.props.song && BDFDB.disCN._spotifycontrolscontainerpaused, this.props.maximized && BDFDB.disCN._spotifycontrolscontainermaximized, this.props.timeline && BDFDB.disCN._spotifycontrolscontainerwithtimeline),
2020-09-19 20:49:33 +02:00
children: [
BDFDB.ReactUtils.createElement("div", {
className: BDFDB.disCN._spotifycontrolscontainerinner,
children: [
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Clickable, {
className: BDFDB.disCN._spotifycontrolscoverwrapper,
onClick: _ => {
this.props.maximized = !this.props.maximized;
BDFDB.DataUtils.save(this.props.maximized, _this, "playerState", "maximized");
if (this.props.maximized) this.request(socketDevice.socket, socketDevice.device, "get").then(response => {
playbackState = Object.assign({}, response);
BDFDB.ReactUtils.forceUpdate(this);
});
else BDFDB.ReactUtils.forceUpdate(this);
},
2020-07-19 16:59:08 +02:00
children: [
2021-04-09 17:15:09 +02:00
coverSrc ? BDFDB.ReactUtils.createElement("img", {
2020-09-19 20:49:33 +02:00
className: BDFDB.disCN._spotifycontrolscover,
2021-04-09 17:15:09 +02:00
src: coverSrc
}) : BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SvgIcon, {
className: BDFDB.disCN._spotifycontrolscover,
width: "100%",
height: "100%",
name: BDFDB.LibraryComponents.SvgIcon.Names.QUESTIONMARK_ACTIVITY
2020-07-23 20:53:46 +02:00
}),
2020-09-19 20:49:33 +02:00
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SvgIcon, {
className: BDFDB.disCN._spotifycontrolscovermaximizer,
name: BDFDB.LibraryComponents.SvgIcon.Names.LEFT_CARET
2021-05-18 19:50:28 +02:00
}),
this.props.maximized && this.props.activityToggle && BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TooltipContainer, {
text: _ => `${BDFDB.LanguageUtils.LanguageStringsFormat("DISPLAY_ACTIVITY", "Spotify")} (${showActivity ? BDFDB.LanguageUtils.LanguageStrings.REPLY_MENTION_ON : BDFDB.LanguageUtils.LanguageStrings.REPLY_MENTION_OFF})`,
children: BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SvgIcon, {
className: BDFDB.disCN._spotifycontrolsactivitybutton,
width: 20,
height: 20,
foreground: BDFDB.disCN.accountinfobuttonstrikethrough,
name: showActivity ? BDFDB.LibraryComponents.SvgIcon.Names.ACTIVITY : BDFDB.LibraryComponents.SvgIcon.Names.ACTIVITY_DISABLED,
onClick: event => {
BDFDB.ListenerUtils.stopEvent(event);
showActivity = !showActivity;
2022-09-27 16:53:10 +02:00
let account = BDFDB.LibraryStores.ConnectedAccountsStore.getAccounts().find(n => n.type == "spotify");
2021-05-18 20:32:00 +02:00
account && BDFDB.LibraryModules.ConnectionUtils.setShowActivity("spotify", account.id, showActivity);
2021-05-18 19:50:28 +02:00
}
})
2020-09-19 20:49:33 +02:00
})
]
}),
BDFDB.ReactUtils.createElement("div", {
className: BDFDB.disCN._spotifycontrolsdetails,
children: [
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TextElement, {
className: BDFDB.disCN._spotifycontrolssong,
children: BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TextScroller, {
children: lastSong.details
})
2020-07-23 16:12:08 +02:00
}),
2020-09-19 20:49:33 +02:00
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TextElement, {
className: BDFDB.disCNS.subtext + BDFDB.disCN._spotifycontrolsinterpret,
2023-11-01 11:23:33 +01:00
color: BDFDB.LibraryComponents.TextElement.Colors.CUSTOM,
2020-09-19 20:49:33 +02:00
size: BDFDB.LibraryComponents.TextElement.Sizes.SIZE_12,
children: BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TextScroller, {
2022-04-18 07:11:02 +02:00
children: _this.settings.general.addBy ? BDFDB.LanguageUtils.LanguageStringsFormat("USER_ACTIVITY_LISTENING_ARTISTS", lastSong.state) : lastSong.state
2020-09-19 20:49:33 +02:00
})
})
]
}),
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TooltipContainer, {
2021-01-23 21:47:04 +01:00
text: socketDevice.device.is_restricted ? _this.labels.restricted_device : null,
2020-09-19 20:49:33 +02:00
tooltipConfig: {color: "red"},
children: BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex, {
grow: 0,
children: [
BDFDB.ReactUtils.createElement(SpotifyControlsButtonComponent, {
type: "share",
2021-08-27 13:42:55 +02:00
player: this,
2020-09-19 20:49:33 +02:00
style: this.props.maximized ? {marginRight: 4} : {},
onClick: _ => {
let url = BDFDB.ObjectUtils.get(playbackState, "item.external_urls.spotify") || BDFDB.ObjectUtils.get(playbackState, "context.external_urls.spotify");
if (url) {
2022-10-14 17:11:45 +02:00
BDFDB.LibraryModules.WindowUtils.copy(url);
2021-01-29 07:55:45 +01:00
BDFDB.NotificationUtils.toast(_this.labels.toast_copyurl_success, {type: "success"});
2020-09-19 20:49:33 +02:00
}
2021-01-29 07:55:45 +01:00
else BDFDB.NotificationUtils.toast(_this.labels.toast_copyurl_fail, {type: "danger"});
2020-07-23 16:12:08 +02:00
}
2020-09-19 20:49:33 +02:00
}),
BDFDB.ReactUtils.createElement(SpotifyControlsButtonComponent, {
type: "shuffle",
2021-08-27 13:42:55 +02:00
player: this,
2020-09-19 20:49:33 +02:00
active: playbackState.shuffle_state,
disabled: socketDevice.device.is_restricted,
onClick: _ => {
playbackState.shuffle_state = !playbackState.shuffle_state;
this.request(socketDevice.socket, socketDevice.device, "shuffle", {
state: playbackState.shuffle_state
});
BDFDB.ReactUtils.forceUpdate(this);
2020-07-23 16:12:08 +02:00
}
2020-09-19 20:49:33 +02:00
}),
BDFDB.ReactUtils.createElement(SpotifyControlsButtonComponent, {
type: "previous",
2021-08-27 13:42:55 +02:00
player: this,
2022-06-29 17:37:27 +02:00
disabled: socketDevice.device.is_restricted || !socketDevice.socket.isPremium,
2020-09-19 20:49:33 +02:00
onClick: _ => {
2021-05-03 21:22:44 +02:00
if (previousIsClicked || !_this.settings.general.doubleBack) {
2020-09-19 20:49:33 +02:00
previousIsClicked = false;
2021-09-06 18:54:06 +02:00
BDFDB.TimeUtils.clear(previousDoubleTimeout);
2020-09-19 20:49:33 +02:00
this.request(socketDevice.socket, socketDevice.device, "previous");
}
else {
previousIsClicked = true;
previousDoubleTimeout = BDFDB.TimeUtils.timeout(_ => {
previousIsClicked = false;
this.request(socketDevice.socket, socketDevice.device, "seek", {
position_ms: 0
});
}, 300);
}
2020-07-23 16:12:08 +02:00
}
2020-09-19 20:49:33 +02:00
}),
BDFDB.ReactUtils.createElement(SpotifyControlsButtonComponent, {
type: "pauseplay",
2021-08-27 13:42:55 +02:00
player: this,
2020-09-19 20:49:33 +02:00
icon: this.props.song ? 0 : 1,
disabled: socketDevice.device.is_restricted,
onClick: _ => {
2022-06-29 17:37:27 +02:00
if (this.props.song) this.request(socketDevice.socket, socketDevice.device, "pause");
else this.request(socketDevice.socket, socketDevice.device, "play");
2020-07-23 16:12:08 +02:00
}
2020-09-19 20:49:33 +02:00
}),
BDFDB.ReactUtils.createElement(SpotifyControlsButtonComponent, {
type: "next",
2021-08-27 13:42:55 +02:00
player: this,
2020-09-19 20:49:33 +02:00
disabled: socketDevice.device.is_restricted,
2022-06-17 20:38:21 +02:00
onClick: _ => this.request(socketDevice.socket, socketDevice.device, "next")
2020-09-19 20:49:33 +02:00
}),
BDFDB.ReactUtils.createElement(SpotifyControlsButtonComponent, {
type: "repeat",
2021-08-27 13:42:55 +02:00
player: this,
2020-09-19 20:49:33 +02:00
icon: playbackState.repeat_state != repeatStates[2] ? 0 : 1,
active: playbackState.repeat_state != repeatStates[0],
disabled: socketDevice.device.is_restricted,
onClick: _ => {
playbackState.repeat_state = repeatStates[repeatStates.indexOf(playbackState.repeat_state) + 1] || repeatStates[0];
this.request(socketDevice.socket, socketDevice.device, "repeat", {
state: playbackState.repeat_state
2020-08-31 16:38:37 +02:00
});
2020-09-19 20:49:33 +02:00
BDFDB.ReactUtils.forceUpdate(this);
2020-08-31 16:38:37 +02:00
}
2020-09-19 20:49:33 +02:00
}),
BDFDB.ReactUtils.createElement(SpotifyControlsButtonComponent, {
type: "volume",
2021-08-27 13:42:55 +02:00
player: this,
2020-09-19 20:49:33 +02:00
icon: Math.ceil(currentVolume/34),
disabled: socketDevice.device.is_restricted,
style: this.props.maximized ? {marginLeft: 4} : {},
onContextMenu: _ => {
if (currentVolume == 0) {
if (lastVolume) this.request(socketDevice.socket, socketDevice.device, "volume", {
volume_percent: lastVolume
});
}
else {
lastVolume = currentVolume;
2020-07-23 20:23:23 +02:00
this.request(socketDevice.socket, socketDevice.device, "volume", {
2020-09-19 20:49:33 +02:00
volume_percent: 0
2020-07-23 20:23:23 +02:00
});
}
2020-09-19 20:49:33 +02:00
},
renderPopout: instance => {
2021-05-12 11:16:22 +02:00
let changeTimeout;
2020-09-19 20:49:33 +02:00
return BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Slider, {
className: BDFDB.disCN._spotifycontrolsvolumeslider,
defaultValue: currentVolume,
digits: 0,
barStyles: {height: 6, top: 3},
2021-08-15 20:02:03 +02:00
fillStyles: {backgroundColor: "var(--SC-spotify-green)"},
2020-09-19 20:49:33 +02:00
onValueRender: value => {
2023-07-18 23:40:08 +02:00
if (currentVolume == value) return;
2021-05-12 11:16:22 +02:00
this.props.draggingVolume = true;
currentVolume = value;
BDFDB.TimeUtils.clear(changeTimeout);
changeTimeout = BDFDB.TimeUtils.timeout(_ => this.props.draggingVolume && this.request(socketDevice.socket, socketDevice.device, "volume", {
volume_percent: currentVolume
}), 500);
2020-09-19 20:49:33 +02:00
return value + "%";
},
onValueChange: value => {
2023-07-18 23:40:08 +02:00
if (currentVolume == value) return;
2021-05-12 11:16:22 +02:00
this.props.draggingVolume = false;
2020-09-19 20:49:33 +02:00
currentVolume = value;
this.request(socketDevice.socket, socketDevice.device, "volume", {
volume_percent: currentVolume
});
}
});
}
})
].filter(n => n)
})
2020-07-19 16:59:08 +02:00
})
2020-09-19 20:49:33 +02:00
]
}),
this.props.timeline && BDFDB.ReactUtils.createElement(SpotifyControlsTimelineComponent, {
song: lastSong,
socket: socketDevice.socket,
device: socketDevice.device,
controls: this
})
].filter(n => n)
});
}
};
const SpotifyControlsButtonComponent = class SpotifyControlsButton extends BdApi.React.Component {
2021-01-20 16:00:29 +01:00
render() {
2021-08-27 13:42:55 +02:00
let playerSize = this.props.player.props.maximized ? "big" : "small";
if (!playerSize || !_this.settings.buttons[this.props.type] || !_this.settings.buttons[this.props.type][playerSize]) return null;
2020-09-19 20:49:33 +02:00
let button = BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Button, BDFDB.ObjectUtils.exclude(Object.assign({}, this.props, {
className: BDFDB.DOMUtils.formatClassName(BDFDB.disCN.accountinfobutton, this.props.disabled ? BDFDB.disCN.accountinfobuttondisabled : BDFDB.disCN.accountinfobuttonenabled, this.props.active && BDFDB.disCN._spotifycontrolsbuttonactive),
look: BDFDB.LibraryComponents.Button.Looks.BLANK,
size: BDFDB.LibraryComponents.Button.Sizes.NONE,
2021-05-03 21:22:44 +02:00
children: _this.defaults.buttons[this.props.type] && _this.defaults.buttons[this.props.type].icons ? (_this.defaults.buttons[this.props.type].icons[this.props.icon] || _this.defaults.buttons[this.props.type].icons[0]) : "?",
2020-09-19 20:49:33 +02:00
onClick: this.props.disabled ? _ => {} : this.props.onClick,
onContextMenu: this.props.disabled ? _ => {} : this.props.onContextMenu,
2022-12-08 18:26:45 +01:00
}), "active", "disabled", "renderPopout", "icon", "type", "player"));
2020-09-19 20:49:33 +02:00
return !this.props.disabled && typeof this.props.renderPopout == "function" ? BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.PopoutContainer, {
children: button,
animation: BDFDB.LibraryComponents.PopoutContainer.Animation.SCALE,
position: BDFDB.LibraryComponents.PopoutContainer.Positions.TOP,
align: BDFDB.LibraryComponents.PopoutContainer.Align.CENTER,
arrow: true,
2021-08-27 13:42:55 +02:00
open: this.props.player.props.buttonStates.indexOf(this.props.type) > -1,
onOpen: _ => this.props.player.props.buttonStates.push(this.props.type),
onClose: _ => BDFDB.ArrayUtils.remove(this.props.player.props.buttonStates, this.props.type, true),
2020-09-19 20:49:33 +02:00
renderPopout: this.props.renderPopout
}) : button;
2022-04-18 07:11:02 +02:00
}
2020-09-19 20:49:33 +02:00
};
const SpotifyControlsTimelineComponent = class SpotifyControlsTimeline extends BdApi.React.Component {
2021-01-20 16:00:29 +01:00
componentDidMount() {
2020-09-19 20:49:33 +02:00
BDFDB.TimeUtils.clear(updateInterval);
updateInterval = BDFDB.TimeUtils.interval(_ => {
if (!this.updater || typeof this.updater.isMounted != "function" || !this.updater.isMounted(this)) BDFDB.TimeUtils.clear(updateInterval);
else if (playbackState.is_playing) {
2022-09-27 16:53:10 +02:00
let song = BDFDB.LibraryStores.SpotifyStore.getActivity(false);
2020-09-19 20:49:33 +02:00
if (!song) BDFDB.ReactUtils.forceUpdate(controls);
else if (playbackState.is_playing) BDFDB.ReactUtils.forceUpdate(this);
}
}, 1000);
}
formatTime(time) {
let seconds = Math.floor((time / 1000) % 60);
let minutes = Math.floor((time / (1000 * 60)) % 60);
let hours = Math.floor((time / (1000 * 60 * 60)) % 24);
return `${hours > 0 ? hours + ":" : ""}${hours > 0 && minutes < 10 ? "0" + minutes : minutes}:${seconds < 10 ? "0" + seconds : seconds}`
}
2021-01-20 16:00:29 +01:00
render() {
2022-06-29 17:37:27 +02:00
let maxTime = !this.props.song ? 0 : this.props.song.timestamps.end - this.props.song.timestamps.start;
let currentTime = !this.props.song ? 0 : (!playbackState.is_playing && stopTime ? stopTime : new Date()) - this.props.song.timestamps.start;
2020-09-19 20:49:33 +02:00
currentTime = currentTime > maxTime ? maxTime : currentTime;
return BDFDB.ReactUtils.createElement("div", {
className: BDFDB.disCN._spotifycontrolstimeline,
children: [
2022-06-29 17:37:27 +02:00
BDFDB.ReactUtils.createElement(this.props.socket.isPremium ? BDFDB.LibraryComponents.Clickable : "div", {
2020-09-19 20:49:33 +02:00
className: BDFDB.disCN._spotifycontrolsbar,
children: [
BDFDB.ReactUtils.createElement("div", {
className: BDFDB.disCN._spotifycontrolsbarfill,
style: {width: `${currentTime / maxTime * 100}%`}
}),
BDFDB.ReactUtils.createElement("div", {
className: BDFDB.disCN._spotifycontrolsbargrabber,
style: {left: `${currentTime / maxTime * 100}%`}
})
],
2022-06-29 17:37:27 +02:00
onClick: !this.props.socket.isPremium ? (_ => {}) : event => {
2020-09-19 20:49:33 +02:00
let rects = BDFDB.DOMUtils.getRects(BDFDB.DOMUtils.getParent(BDFDB.dotCN._spotifycontrolsbar, event.target));
this.props.controls.request(this.props.socket, this.props.device, "seek", {
position_ms: Math.round(BDFDB.NumberUtils.mapRange([rects.left, rects.left + rects.width], [0, maxTime], event.clientX))
});
}
}),
BDFDB.ReactUtils.createElement("div", {
className: BDFDB.disCN._spotifycontrolsbartext,
children: [
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TextElement, {
size: BDFDB.LibraryComponents.TextElement.Sizes.SIZE_12,
children: this.formatTime(currentTime)
}),
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TextElement, {
size: BDFDB.LibraryComponents.TextElement.Sizes.SIZE_12,
children: this.formatTime(maxTime)
})
]
})
]
});
}
};
2020-07-19 16:59:08 +02:00
2020-10-09 21:09:35 +02:00
return class SpotifyControls extends Plugin {
2021-01-06 12:38:36 +01:00
onLoad () {
2020-09-19 20:49:33 +02:00
_this = this;
this.defaults = {
2021-05-03 21:22:44 +02:00
general: {
2023-11-18 18:31:04 +01:00
addBy: {value: true, description: "Adds the Word 'by' infront of the Author Name"},
2021-05-18 19:50:28 +02:00
addTimeline: {value: true, description: "Shows the Song Timeline in the Controls"},
addActivityButton: {value: true, description: "Shows the Activity Status Toggle Button in the Controls"},
2022-04-18 07:11:02 +02:00
doubleBack: {value: true, description: "Requires the User to press the Back Button twice to go to previous Track"}
2020-09-19 20:49:33 +02:00
},
2021-05-03 21:22:44 +02:00
buttons: {
2023-11-18 18:31:04 +01:00
share: {value: {small: false, big: true}, icons: [""], description: "Share"},
shuffle: {value: {small: false, big: true}, icons: [""], description: "Shuffle"},
previous: {value: {small: true, big: true}, icons: [""], description: "Previous"},
pauseplay: {value: {small: true, big: true}, icons: ["", ""], description: "Pause/Play"},
next: {value: {small: true, big: true}, icons: [""], description: "Next"},
repeat: {value: {small: false, big: true}, icons: ["", ""], description: "Repeat"},
volume: {value: {small: false, big: true}, icons: ["", "", "", ""], description: "Volume"}
2020-09-19 20:49:33 +02:00
}
};
2022-10-13 13:15:23 +02:00
this.modulePatches = {
after: [
"ChannelSidebar"
]
2020-09-19 20:49:33 +02:00
};
this.css = `
@font-face {
font-family: glue1-spoticon;
src: url("https://mwittrien.github.io/BetterDiscordAddons/Plugins/SpotifyControls/_res/spoticon.ttf") format("truetype");
font-weight: 400;
font-style: normal
}
2021-08-15 20:02:03 +02:00
:root {
2023-02-19 04:56:43 +01:00
--SC-spotify-green: ${BDFDB.DiscordConstants.Colors.SPOTIFY || "#1db954"};
2021-08-15 20:02:03 +02:00
}
2021-08-24 15:00:34 +02:00
${BDFDB.dotCN.channelpanels} {
2022-04-18 07:11:02 +02:00
display: flex;
flex-direction: column;
2021-08-24 15:00:34 +02:00
}
2020-09-19 20:49:33 +02:00
${BDFDB.dotCN._spotifycontrolscontainer} {
display: flex;
flex-direction: column;
justify-content: center;
min-height: 52px;
margin-bottom: 1px;
border-bottom: 1px solid var(--background-modifier-accent);
padding: 0 8px;
box-sizing: border-box;
2021-08-24 15:00:34 +02:00
order: -1;
2020-09-19 20:49:33 +02:00
}
2023-04-14 17:40:23 +02:00
${BDFDB.dotCN.themelight + BDFDB.dotCNS.themecustombackground + BDFDB.dotCN._spotifycontrolscontainer} {
2023-04-14 19:38:58 +02:00
background: var(--bg-overlay-3);
2023-04-14 17:40:23 +02:00
}
${BDFDB.dotCN.themedark + BDFDB.dotCNS.themecustombackground + BDFDB.dotCN._spotifycontrolscontainer} {
2023-04-14 19:38:58 +02:00
background: var(--bg-overlay-1);
2023-04-14 17:40:23 +02:00
}
2020-09-19 20:49:33 +02:00
${BDFDB.dotCN._spotifycontrolscontainer + BDFDB.dotCN._spotifycontrolscontainerwithtimeline} {
padding-top: 8px;
}
${BDFDB.dotCN._spotifycontrolscontainerinner} {
display: flex;
align-items: center;
font-size: 14px;
width: 100%;
}
${BDFDB.dotCN._spotifycontrolstimeline} {
margin: 6px 0 4px 0;
}
${BDFDB.dotCN._spotifycontrolsbar} {
position: relative;
border-radius: 2px;
background-color: rgba(79, 84, 92, 0.16);
height: 4px;
margin-bottom: 4px;
}
${BDFDB.dotCN._spotifycontrolsbarfill} {
border-radius: 2px;
height: 100%;
min-width: 4px;
border-radius: 2px;
background: var(--text-normal);
}
${BDFDB.dotCN._spotifycontrolstimeline}:hover ${BDFDB.dotCN._spotifycontrolsbarfill} {
2021-08-15 20:02:03 +02:00
background: var(--SC-spotify-green);
2020-09-19 20:49:33 +02:00
}
${BDFDB.dotCN._spotifycontrolsbargrabber} {
display: none;
position: absolute;
top: 0;
left: 0;
width: 8px;
height: 8px;
margin-top: -2px;
margin-left: -2px;
background: var(--text-normal);
border-radius: 50%;
}
${BDFDB.dotCN._spotifycontrolstimeline}:hover ${BDFDB.dotCN._spotifycontrolsbargrabber} {
display: block;
}
${BDFDB.dotCN._spotifycontrolsbartext} {
display: flex;
align-items: center;
justify-content: space-between;
}
${BDFDB.dotCN._spotifycontrolscoverwrapper} {
position: relative;
width: 32px;
min-width: 32px;
height: 32px;
min-height: 32px;
margin-right: 8px;
border-radius: 4px;
overflow: hidden;
2021-05-17 12:57:34 +02:00
transition: border-radius .3s ease, margin .3s ease, width .3s ease, height .3s ease;
2020-09-19 20:49:33 +02:00
}
${BDFDB.dotCN._spotifycontrolscover} {
display: block;
width: 100%;
height: 100%;
2021-05-03 21:22:44 +02:00
color: var(--header-primary);
2020-09-19 20:49:33 +02:00
object-fit: cover;
}
${BDFDB.dotCN._spotifycontrolscovermaximizer} {
visibility: hidden;
position: absolute;
background-color: rgba(0, 0, 0, 0.5);
color: rgba(255, 255, 255, 0.5);
top: 0;
right: 0;
border-radius: 50%;
width: 12px;
height: 12px;
padding: 3px;
transform: rotate(90deg);
transition: width .3s ease, height .3s ease, transform .3s ease;
pointer-events: none;
}
${BDFDB.dotCN._spotifycontrolscoverwrapper}:hover ${BDFDB.dotCN._spotifycontrolscovermaximizer} {
visibility: visible;
}
2021-05-18 19:50:28 +02:00
${BDFDB.dotCN._spotifycontrolsactivitybutton} {
visibility: hidden;
position: absolute;
background-color: rgba(0, 0, 0, 0.5);
color: rgba(255, 255, 255, 0.5);
border-radius: 50%;
bottom: 4px;
right: 4px;
width: 22px;
height: 22px;
padding: 5px;
}
${BDFDB.dotCN._spotifycontrolsactivitybutton}:hover {
color: rgb(255, 255, 255);
}
${BDFDB.dotCN._spotifycontrolscoverwrapper}:hover ${BDFDB.dotCN._spotifycontrolsactivitybutton} {
visibility: visible;
}
2020-09-19 20:49:33 +02:00
${BDFDB.dotCN._spotifycontrolsdetails} {
flex-grow: 1;
margin-right: 4px;
min-width: 0;
2021-05-18 19:50:28 +02:00
user-select: text;
2020-09-19 20:49:33 +02:00
}
${BDFDB.dotCN._spotifycontrolssong} {
font-weight: 500;
}
${BDFDB.dotCN._spotifycontrolsinterpret} {
font-weight: 300;
}
${BDFDB.dotCN._spotifycontrolsvolumeslider} {
height: 12px;
width: 140px;
margin: 5px;
}
${BDFDB.dotCNS._spotifycontrolsvolumeslider + BDFDB.dotCN.slidergrabber} {
height: 10px;
margin-top: -6px;
border-radius: 50%;
}
${BDFDB.dotCNS._spotifycontrolscontainer + BDFDB.dotCN.accountinfobuttondisabled} {
cursor: no-drop;
}
${BDFDB.dotCNS._spotifycontrolscontainer + BDFDB.dotCNS.accountinfobutton + BDFDB.dotCN.buttoncontents} {
font-family: glue1-spoticon !important;
}
${BDFDB.dotCNS._spotifycontrolscontainer + BDFDB.dotCN.accountinfobutton + BDFDB.dotCN._spotifycontrolsbuttonactive} {
2023-10-19 12:23:12 +02:00
color: var(--SC-spotify-green) !important;
2020-09-19 20:49:33 +02:00
}
${BDFDB.dotCN._spotifycontrolscontainer + BDFDB.dotCN._spotifycontrolscontainermaximized} {
padding-top: 0;
}
${BDFDB.dotCN._spotifycontrolscontainer + BDFDB.dotCNS._spotifycontrolscontainermaximized + BDFDB.dotCN._spotifycontrolscontainerinner} {
flex-direction: column;
}
${BDFDB.dotCN._spotifycontrolscontainer + BDFDB.dotCNS._spotifycontrolscontainermaximized + BDFDB.dotCN._spotifycontrolsdetails} {
margin: 0 0 4px 0;
width: 100%;
text-align: center;
}
${BDFDB.dotCN._spotifycontrolscontainer + BDFDB.dotCNS._spotifycontrolscontainermaximized + BDFDB.dotCN._spotifycontrolscoverwrapper} {
width: calc(100% + 16px);
height: 100%;
margin: 0 0 8px 0;
border-radius: 0;
}
${BDFDB.dotCN._spotifycontrolscontainer + BDFDB.dotCNS._spotifycontrolscontainermaximized + BDFDB.dotCN._spotifycontrolscovermaximizer} {
top: 4px;
right: 4px;
width: 22px;
height: 22px;
padding: 5px;
transform: rotate(-90deg);
}
${BDFDB.dotCN._spotifycontrolssettingsicon} {
margin: 4px;
font-size: 16px;
font-family: glue1-spoticon !important;
}
${BDFDB.dotCN._spotifycontrolssettingslabel} {
margin-left: 10px;
}
2021-04-09 17:33:04 +02:00
${BDFDB.dotCNS._bdminimalmode + BDFDB.dotCN._spotifycontrolsbar} {
height: 3px;
}
${BDFDB.dotCNS._bdminimalmode + BDFDB.dotCNS._spotifycontrolscontainer + BDFDB.dotCN.accountinfobutton} {
width: 26px;
height: 26px;
}
${BDFDB.dotCNS._bdminimalmode + BDFDB.dotCNS._spotifycontrolscontainer + BDFDB.dotCN.size14} {
font-size: 13px;
line-height: 13px;
}
${BDFDB.dotCNS._bdminimalmode + BDFDB.dotCNS._spotifycontrolscontainer + BDFDB.dotCN.size12} {
font-size: 11px;
line-height: 11px;
}
2020-09-19 20:49:33 +02:00
`;
2020-07-19 16:59:08 +02:00
}
2020-09-19 20:49:33 +02:00
2021-01-06 12:38:36 +01:00
onStart () {
2022-09-27 16:53:10 +02:00
BDFDB.PatchUtils.patch(this, BDFDB.LibraryStores.SpotifyStore, "getActivity", {after: e => {
2020-07-21 15:12:43 +02:00
if (e.methodArguments[0] !== false) {
if (e.returnValue && e.returnValue.name == "Spotify") this.updatePlayer(e.returnValue);
else if (!e.returnValue) this.updatePlayer(null);
}
2020-07-19 16:59:08 +02:00
}});
2022-09-27 16:53:10 +02:00
BDFDB.PatchUtils.patch(this, BDFDB.LibraryStores.SpotifyStore, "wasAutoPaused", {instead: e => {
2020-07-19 16:59:08 +02:00
return false;
}});
2020-09-11 19:31:36 +02:00
BDFDB.PatchUtils.patch(this, BDFDB.LibraryModules.SpotifyUtils, "pause", {instead: e => {
2020-07-19 16:59:08 +02:00
return false;
}});
2022-10-13 13:15:23 +02:00
BDFDB.DiscordUtils.rerenderAll();
2020-07-19 16:59:08 +02:00
}
2020-09-19 20:49:33 +02:00
2021-01-06 12:38:36 +01:00
onStop () {
2022-10-13 13:15:23 +02:00
BDFDB.DiscordUtils.rerenderAll();
2020-09-19 20:49:33 +02:00
}
2020-07-19 16:59:08 +02:00
2021-04-27 14:03:05 +02:00
getSettingsPanel (collapseStates = {}) {
let settingsPanel;
return settingsPanel = BDFDB.PluginUtils.createSettingsPanel(this, {
2020-09-19 20:49:33 +02:00
collapseStates: collapseStates,
2021-04-27 14:03:05 +02:00
children: _ => {
let settingsItems = [];
2022-09-27 16:53:10 +02:00
if (!BDFDB.LibraryStores.SpotifyStore.hasConnectedAccount()) BDFDB.ModalUtils.open(this, {
2021-04-27 14:03:05 +02:00
size: "SMALL",
header: `${this.name}: ${this.labels.noaccount_header}...`,
subHeader: this.labels.noaccount_subheader,
text: this.labels.noaccount_text,
buttons: [{
contents: BDFDB.LanguageUtils.LanguageStrings.CONNECT,
color: "BRAND",
close: true,
onClick: _ => BDFDB.LibraryModules.UserSettingsUtils.open(BDFDB.DiscordConstants.UserSettingsSections.CONNECTIONS)
}]
});
settingsItems.push(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.CollapseContainer, {
title: "Settings",
collapseStates: collapseStates,
2021-05-03 21:22:44 +02:00
children: Object.keys(this.defaults.general).map(key => BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SettingsSaveItem, {
2021-04-27 14:03:05 +02:00
type: "Switch",
plugin: this,
2021-05-03 21:22:44 +02:00
keys: ["general", key],
label: this.defaults.general[key].description,
value: this.settings.general[key]
2021-04-27 14:03:05 +02:00
}))
}));
settingsItems.push(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.CollapseContainer, {
title: "Button Settings",
collapseStates: collapseStates,
children: [BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.FormComponents.FormTitle, {
className: BDFDB.disCN.marginbottom4,
2022-11-22 01:30:30 +01:00
tag: BDFDB.LibraryComponents.FormComponents.FormTags.H3,
2021-05-18 19:50:28 +02:00
children: "Add Control Buttons in small and/or big Player Version: "
2021-04-27 14:03:05 +02:00
})].concat(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SettingsList, {
2021-05-03 21:22:44 +02:00
settings: Object.keys(this.defaults.buttons[Object.keys(this.defaults.buttons)[0]].value),
data: Object.keys(this.defaults.buttons).map(key => Object.assign({}, this.settings.buttons[key], {
2021-04-27 14:03:05 +02:00
key: key,
2021-05-03 21:22:44 +02:00
label: this.defaults.buttons[key].description,
icons: this.defaults.buttons[key].icons
2021-04-27 14:03:05 +02:00
})),
noRemove: true,
renderLabel: data => BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex, {
align: BDFDB.LibraryComponents.Flex.Align.CENTER,
children: [
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex, {
justify: BDFDB.LibraryComponents.Flex.Justify.CENTER,
wrap: BDFDB.LibraryComponents.Flex.Wrap.WRAP,
basis: 50,
grow: 0,
children: data.icons.map(icon => BDFDB.ReactUtils.createElement("div", {
className: BDFDB.disCN._spotifycontrolssettingsicon,
children: icon
}))
}),
BDFDB.ReactUtils.createElement("div", {
className: BDFDB.disCN._spotifycontrolssettingslabel,
children: data.label
})
]
2020-09-19 20:49:33 +02:00
}),
2021-04-27 14:03:05 +02:00
onCheckboxChange: (value, instance) => {
2021-05-03 21:22:44 +02:00
this.settings.buttons[instance.props.cardId][instance.props.settingId] = value;
BDFDB.DataUtils.save(this.settings.buttons, this, "buttons");
2021-04-27 14:03:05 +02:00
this.SettingsUpdated = true;
}
}))
}));
return settingsItems;
}
});
2020-07-19 16:59:08 +02:00
}
2021-01-06 12:38:36 +01:00
onSettingsClosed () {
2020-09-19 20:49:33 +02:00
if (this.SettingsUpdated) {
delete this.SettingsUpdated;
2022-10-13 13:15:23 +02:00
BDFDB.DiscordUtils.rerenderAll();
2020-09-19 20:49:33 +02:00
}
}
2020-07-21 15:08:40 +02:00
2022-10-13 13:15:23 +02:00
processChannelSidebar (e) {
let panels = BDFDB.ReactUtils.findChild(e.returnvalue, {props: [["className", BDFDB.disCN.channelpanels]]});
if (panels) panels.props.children.unshift(BDFDB.ReactUtils.createElement(SpotifyControlsComponent, {
key: "SPOTIFY_CONTROLS",
song: BDFDB.LibraryStores.SpotifyStore.getActivity(false),
maximized: BDFDB.DataUtils.load(this, "playerState", "maximized"),
buttonStates: [],
timeline: this.settings.general.addTimeline,
activityToggle: this.settings.general.addActivityButton
}, true));
2020-07-19 16:59:08 +02:00
}
2020-07-21 15:08:40 +02:00
2020-09-19 20:49:33 +02:00
updatePlayer (song) {
if (controls) {
controls.props.song = song;
BDFDB.ReactUtils.forceUpdate(controls);
}
}
2021-01-23 21:47:04 +01:00
setLabelsByLanguage () {
switch (BDFDB.LanguageUtils.getLanguage().id) {
case "bg": // Bulgarian
return {
noaccount_header: "Нещо липсва",
noaccount_subheader: "Трябва да свържете акаунт в Spotify",
noaccount_text: "Липсва ви свързан акаунт в Spotify, без акаунт няма да можете да използвате Spotify Controls. За да свържете акаунт в Spotify с вашия акаунт в Discord, кликнете върху бутона по-долу.",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord загуби връзката с последното устройство, което е възпроизвеждало Spotify, отворете Spotify на устройството отново и ръчно възобновете песента",
2021-01-23 21:47:04 +01:00
restricted_device: "Не може да контролира Spotify, докато възпроизвежда музика на ограничено устройство",
toast_copyurl_fail: "URL адресът на песента не може да бъде копиран в клипборда",
toast_copyurl_success: "URL адресът на песента беше копиран в клипборда"
};
2022-06-17 20:38:21 +02:00
case "cs": // Czech
return {
noaccount_header: "Něco chybí",
noaccount_subheader: "Musíte si připojit účet Spotify",
noaccount_text: "Chybí vám připojený účet Spotify, bez účtu nebudete moci používat ovládání Spotify. Chcete-li propojit účet Spotify s účtem Discord, klikněte na tlačítko níže.",
nodevice_text: "Discord ztratil spojení s posledním zařízením, které přehrávalo Spotify, otevřete Spotify na zařízení znovu a ručně obnovte skladbu",
restricted_device: "Nelze ovládat Spotify při přehrávání hudby na omezeném zařízení",
toast_copyurl_fail: "URL skladby se nepodařilo zkopírovat do schránky",
toast_copyurl_success: "Adresa URL skladby byla zkopírována do schránky"
};
2021-01-23 21:47:04 +01:00
case "da": // Danish
return {
noaccount_header: "Noget mangler",
noaccount_subheader: "Du skal oprette forbindelse til en Spotify-konto",
noaccount_text: "Du mangler en tilsluttet Spotify-konto, uden en konto kan du ikke bruge Spotify Controls. For at forbinde en Spotify-konto med din Discord-konto skal du klikke på knappen nedenfor.",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord mistede forbindelsen til den sidste enhed, der spillede Spotify, åbn Spotify på enheden igen og genoptag sangen manuelt",
2021-01-23 21:47:04 +01:00
restricted_device: "Kan ikke kontrollere Spotify, mens du spiller musik på en begrænset enhed",
toast_copyurl_fail: "Sang-URL kunne ikke kopieres til udklipsholderen",
toast_copyurl_success: "Sang-URL blev kopieret til udklipsholderen"
};
case "de": // German
return {
noaccount_header: "Etwas fehlt",
2022-06-17 20:38:21 +02:00
noaccount_subheader: "Es muss ein Spotify-Konto verbunden werden",
noaccount_text: "Ihnen fehlt ein verbundenes Spotify-Konto. Ohne ein Konto kann Spotify Controls nicht verwenden. Um ein Spotify-Konto mit Ihrem Discord-Konto zu verbinden, klicke auf die Schaltfläche unten.",
nodevice_text: "Discord hat die Verbindung zum letzten Gerät verloren, auf dem Spotify gespielt wurde, öffne Spotify erneut auf dem Gerät und setze den Song manuell fort",
2021-01-23 21:47:04 +01:00
restricted_device: "Spotify kann nicht gesteuert werden, während Musik auf einem eingeschränkten Gerät abgespielt wird",
toast_copyurl_fail: "Die Song-URL konnte nicht in die Zwischenablage kopiert werden",
toast_copyurl_success: "Die Song-URL wurde in die Zwischenablage kopiert"
};
case "el": // Greek
return {
noaccount_header: "Κάτι λείπει",
noaccount_subheader: "Πρέπει να συνδέσετε έναν λογαριασμό Spotify",
noaccount_text: "Λείπει ένας συνδεδεμένος λογαριασμός Spotify, χωρίς λογαριασμό δεν θα μπορείτε να χρησιμοποιήσετε το Spotify Controls. Για να συνδέσετε έναν λογαριασμό Spotify με τον λογαριασμό σας Discord κάντε κλικ στο παρακάτω κουμπί.",
2022-06-17 20:38:21 +02:00
nodevice_text: "Το Discord έχασε τη σύνδεση με την τελευταία συσκευή που έπαιζε Spotify, ανοίξτε ξανά το Spotify στη συσκευή και συνεχίστε μη αυτόματα το τραγούδι",
2021-01-23 21:47:04 +01:00
restricted_device: "Δεν είναι δυνατή ο έλεγχος του Spotify κατά την αναπαραγωγή μουσικής σε περιορισμένη συσκευή",
toast_copyurl_fail: "Δεν ήταν δυνατή η αντιγραφή του URL τραγουδιού στο πρόχειρο",
toast_copyurl_success: "Η διεύθυνση URL του τραγουδιού αντιγράφηκε στο πρόχειρο"
};
case "es": // Spanish
return {
noaccount_header: "Algo falta",
noaccount_subheader: "Necesitas conectar una cuenta de Spotify",
noaccount_text: "Falta una cuenta de Spotify conectada, sin una cuenta no podrá usar Spotify Controls. Para conectar una cuenta de Spotify con su cuenta de Discord, haga clic en el botón de abajo.",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord perdió la conexión con el último dispositivo que estaba reproduciendo Spotify, abra Spotify en el dispositivo nuevamente y reanude manualmente la canción",
2021-01-23 21:47:04 +01:00
restricted_device: "No se puede controlar Spotify mientras se reproduce música en un dispositivo restringido",
toast_copyurl_fail: "No se pudo copiar la URL de la canción al portapapeles",
toast_copyurl_success: "La URL de la canción se copió al portapapeles"
};
case "fi": // Finnish
return {
noaccount_header: "Jotain puuttuu",
noaccount_subheader: "Sinun on yhdistettävä Spotify-tili",
2022-02-03 21:27:18 +01:00
noaccount_text: "Sinulta puuttuu yhdistetty Spotify-tili. Ilman tiliä et voi käyttää Spotify Controlsia. Yhdistä Spotify-tili Discord-tiliisi napsauttamalla alla olevaa painiketta.",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord menetti yhteyden viimeksi Spotifyta toistavaan laitteeseen, avaa Spotify uudelleen laitteessa ja jatka kappaleen toistoa manuaalisesti",
2022-02-03 21:27:18 +01:00
restricted_device: "Spotify ei voi hallita musiikkia toistettaessa rajoitetulla laitteella",
2021-01-23 21:47:04 +01:00
toast_copyurl_fail: "Kappaleen URL-osoitetta ei voitu kopioida leikepöydälle",
toast_copyurl_success: "Kappaleen URL-osoite kopioitiin leikepöydälle"
};
case "fr": // French
return {
noaccount_header: "Quelque chose manque",
noaccount_subheader: "Vous devez connecter un compte Spotify",
noaccount_text: "Il vous manque un compte Spotify connecté, sans compte, vous ne pourrez pas utiliser Spotify Controls. Pour connecter un compte Spotify à votre compte Discord, cliquez sur le bouton ci-dessous.",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord a perdu la connexion au dernier appareil qui lisait Spotify, ouvrez à nouveau Spotify sur l'appareil et reprenez manuellement la chanson",
2021-01-23 21:47:04 +01:00
restricted_device: "Impossible de contrôler Spotify lors de la lecture de musique sur un appareil restreint",
toast_copyurl_fail: "L'URL de la chanson n'a pas pu être copiée dans le presse-papiers",
toast_copyurl_success: "L'URL de la chanson a été copiée dans le presse-papiers"
};
2022-06-17 20:38:21 +02:00
case "hi": // Hindi
return {
noaccount_header: "कुछ याद आ रही है",
noaccount_subheader: "आपको एक Spotify खाता कनेक्ट करने की आवश्यकता है",
noaccount_text: "आप एक कनेक्टेड Spotify खाता खो रहे हैं, बिना किसी खाते के आप Spotify नियंत्रणों का उपयोग नहीं कर पाएंगे। Spotify खाते को अपने Discord खाते से जोड़ने के लिए नीचे दिए गए बटन पर क्लिक करें।",
nodevice_text: "डिस्कॉर्ड ने पिछले डिवाइस से कनेक्शन खो दिया जो Spotify चला रहा था, डिवाइस पर फिर से Spotify खोलें और गाने को मैन्युअल रूप से फिर से शुरू करें",
restricted_device: "प्रतिबंधित डिवाइस पर संगीत चलाते समय Spotify को नियंत्रित नहीं कर सकता",
toast_copyurl_fail: "गाने के यूआरएल को क्लिपबोर्ड पर कॉपी नहीं किया जा सका",
toast_copyurl_success: "गाने के यूआरएल को क्लिपबोर्ड पर कॉपी किया गया था"
};
2021-01-23 21:47:04 +01:00
case "hr": // Croatian
return {
noaccount_header: "Nešto nedostaje",
noaccount_subheader: "Morate povezati Spotify račun",
noaccount_text: "Nedostaje vam povezani Spotify račun, bez računa nećete moći koristiti Spotify Controls. Da biste povezali Spotify račun sa svojim Discord računom, kliknite gumb u nastavku.",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord je izgubio vezu s posljednjim uređajem koji je reproducirao Spotify, ponovno otvorite Spotify na uređaju i ručno nastavite pjesmu",
2021-01-23 21:47:04 +01:00
restricted_device: "Ne može kontrolirati Spotify tijekom reprodukcije glazbe na ograničenom uređaju",
toast_copyurl_fail: "URL pjesme nije se mogao kopirati u međuspremnik",
toast_copyurl_success: "URL pjesme kopiran je u međuspremnik"
};
case "hu": // Hungarian
return {
noaccount_header: "Valami hiányzik",
noaccount_subheader: "Csatlakoztatnia kell egy Spotify-fiókot",
2021-01-28 14:18:17 +01:00
noaccount_text: "Hiányzik egy csatlakoztatott Spotify-fiók, fiók nélkül nem fogja tudni használni a Spotify Controls szolgáltatást. Ha Spotify-fiókot szeretne összekapcsolni Discord-fiókot, kattintson az alábbi gombra.",
2022-06-17 20:38:21 +02:00
nodevice_text: "A Discord elvesztette a kapcsolatot az utolsó Spotify-t lejátszott eszközzel, nyissa meg újra a Spotify-t az eszközön, és folytassa manuálisan a dalt",
2021-01-23 21:47:04 +01:00
restricted_device: "Nem lehet irányítani a Spotify szolgáltatást, miközben zenét játszik le korlátozott eszközön",
toast_copyurl_fail: "A dal URL-jét nem sikerült átmásolni a vágólapra",
toast_copyurl_success: "A dal URL-jét a vágólapra másolta"
};
case "it": // Italian
return {
noaccount_header: "Manca qualcosa",
noaccount_subheader: "Devi collegare un account Spotify",
noaccount_text: "Ti manca un account Spotify collegato, senza un account non sarai in grado di utilizzare Spotify Controls. Per collegare un account Spotify al tuo account Discord, fai clic sul pulsante in basso.",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord ha perso la connessione all'ultimo dispositivo che stava riproducendo Spotify, apri nuovamente Spotify sul dispositivo e riprendi manualmente il brano",
2021-01-23 21:47:04 +01:00
restricted_device: "Non è possibile controllare Spotify durante la riproduzione di musica su un dispositivo limitato",
toast_copyurl_fail: "L'URL del brano non può essere copiato negli appunti",
toast_copyurl_success: "L'URL del brano è stato copiato negli appunti"
};
case "ja": // Japanese
return {
noaccount_header: "何かが欠けています",
noaccount_subheader: "Spotify アカウントを接続する必要があります",
noaccount_text: "接続されている Spotify アカウントがありません。アカウントがないと、 Spotify Controls を使用できません。 Spotify アカウントを Discord アカウントに接続するには、下のボタンをクリックしてください。",
2022-06-17 20:38:21 +02:00
nodevice_text: "DiscordはSpotifyを再生していた最後のデバイスへの接続を失い、デバイスでSpotifyを再度開き、手動で曲を再開します",
2021-01-23 21:47:04 +01:00
restricted_device: "制限されたデバイスで音楽を再生している間は Spotify を制御できません",
toast_copyurl_fail: "曲のURLをクリップボードにコピーできませんでした",
toast_copyurl_success: "曲のURLがクリップボードにコピーされました"
};
case "ko": // Korean
return {
noaccount_header: "무언가가 빠졌어",
noaccount_subheader: "Spotify 계정을 연결해야합니다",
noaccount_text: "연결된 Spotify 계정이 없습니다. 계정이 없으면 Spotify Controls 을 사용할 수 없습니다. Spotify 계정을 Discord 계정과 연결하려면 아래 버튼을 클릭하세요.",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord는 Spotify를 재생하던 마지막 장치와의 연결이 끊어졌습니다. 장치에서 Spotify를 다시 열고 수동으로 노래를 다시 시작합니다.",
2021-01-23 21:47:04 +01:00
restricted_device: "제한된 장치에서 음악을 재생하는 동안 Spotify 를 제어 할 수 없습니다.",
toast_copyurl_fail: "노래 URL을 클립 보드에 복사 할 수 없습니다.",
toast_copyurl_success: "노래 URL이 클립 보드에 복사되었습니다. "
};
case "lt": // Lithuanian
return {
noaccount_header: "Kažko trūksta",
noaccount_subheader: "Turite prijungti „ Spotify “ paskyrą",
noaccount_text: "Trūksta prijungtos „ Spotify “ paskyros, be paskyros negalėsite naudoti Spotify Controls. Norėdami susieti „ Spotify “ paskyrą su Discord paskyra, spustelėkite toliau pateiktą mygtuką.",
2022-06-17 20:38:21 +02:00
nodevice_text: "„Discord“ prarado ryšį su paskutiniu įrenginiu, kuriame grojo „Spotify“, dar kartą atidarykite „Spotify“ įrenginyje ir rankiniu būdu atnaujinkite dainą",
2021-01-23 21:47:04 +01:00
restricted_device: "Nepavyksta valdyti „ Spotify “ grojant muziką ribotame įrenginyje",
toast_copyurl_fail: "Dainos URL nepavyko nukopijuoti į iškarpinę",
toast_copyurl_success: "Dainos URL buvo nukopijuotas į iškarpinę"
};
case "nl": // Dutch
return {
noaccount_header: "Er mist iets",
noaccount_subheader: "U moet een Spotify-account verbinden",
noaccount_text: "U mist een verbonden Spotify-account. Zonder account kunt u Spotify Controls niet gebruiken. Om een Spotify-account aan uw Discord-account te koppelen, klikt u op de onderstaande knop.",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord heeft de verbinding verbroken met het laatste apparaat dat Spotify afspeelde, open Spotify opnieuw op het apparaat en hervat het nummer handmatig",
2021-01-23 21:47:04 +01:00
restricted_device: "Kan Spotify niet bedienen tijdens het afspelen van muziek op een beperkt apparaat",
toast_copyurl_fail: "Nummer-URL kan niet naar klembord worden gekopieerd",
toast_copyurl_success: "Nummer-URL is naar klembord gekopieerd"
};
case "no": // Norwegian
return {
noaccount_header: "Noe mangler",
noaccount_subheader: "Du må koble til en Spotify-konto",
noaccount_text: "Du mangler en tilkoblet Spotify-konto, uten en konto kan du ikke bruke Spotify Controls. For å koble en Spotify-konto til Discord-konto din, klikk på knappen nedenfor.",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord mistet forbindelsen til den siste enheten som spilte Spotify, åpne Spotify på enheten igjen og gjenoppta sangen manuelt",
2021-01-23 21:47:04 +01:00
restricted_device: "Kan ikke kontrollere Spotify mens du spiller musikk på begrenset enhet",
toast_copyurl_fail: "Sangens URL kunne ikke kopieres til utklippstavlen",
toast_copyurl_success: "Sang-URL ble kopiert til utklippstavlen"
};
case "pl": // Polish
return {
noaccount_header: "Czegoś brakuje",
noaccount_subheader: "Musisz połączyć konto Spotify",
noaccount_text: "Brakuje połączonego konta Spotify, bez konta nie będziesz mógł korzystać z Spotify Controls. Aby połączyć konto Spotify z kontem Discord, kliknij przycisk poniżej.",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord stracił połączenie z ostatnim urządzeniem, które odtwarzało Spotify, ponownie otwórz Spotify na urządzeniu i ręcznie wznów piosenkę",
2021-01-23 21:47:04 +01:00
restricted_device: "Nie można sterować Spotify podczas odtwarzania muzyki na urządzeniu z ograniczeniami",
toast_copyurl_fail: "Nie udało się skopiować adresu URL utworu do schowka",
toast_copyurl_success: "URL utworu został skopiowany do schowka"
};
case "pt-BR": // Portuguese (Brazil)
return {
noaccount_header: "Algo está faltando",
noaccount_subheader: "Você precisa conectar uma conta Spotify",
noaccount_text: "Está faltando uma conta Spotify conectada, sem uma conta você não poderá usar Spotify Controls. Para conectar uma conta Spotify à sua conta Discord, clique no botão abaixo.",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord perdeu a conexão com o último dispositivo que estava tocando Spotify, abra o Spotify no dispositivo novamente e retome manualmente a música",
2021-01-23 21:47:04 +01:00
restricted_device: "Não é possível controlar o Spotify enquanto reproduz música em dispositivo restrito",
toast_copyurl_fail: "O URL da música não pôde ser copiado para a área de transferência",
toast_copyurl_success: "O URL da música foi copiado para a área de transferência"
};
case "ro": // Romanian
return {
noaccount_header: "Ceva lipseste",
noaccount_subheader: "Trebuie să vă conectați un cont Spotify",
noaccount_text: "Vă lipsește un cont Spotify conectat, fără un cont pe care nu îl veți putea folosi Spotify Controls. Pentru a conecta un cont Spotify la contul dvs. Discord faceți clic pe butonul de mai jos.",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord a pierdut conexiunea la ultimul dispozitiv care redă Spotify, deschideți Spotify din nou pe dispozitiv și reluați manual melodia",
2021-01-23 21:47:04 +01:00
restricted_device: "Nu pot controla Spotify în timp ce redați muzică pe dispozitiv restricționat",
toast_copyurl_fail: "Adresa URL a melodiei nu a putut fi copiată în clipboard",
toast_copyurl_success: "Adresa URL a melodiei a fost copiată în clipboard"
};
case "ru": // Russian
return {
noaccount_header: "Что-то пропало",
noaccount_subheader: "Вам необходимо подключить учетную запись Spotify",
noaccount_text: "У вас отсутствует подключенная учетная запись Spotify, без нее вы не сможете использовать Spotify Controls. Чтобы связать учетную запись Spotify со своей учетной записью Discord, нажмите кнопку ниже.",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord потерял соединение с последним устройством, на котором воспроизводился Spotify, снова откройте Spotify на устройстве и вручную возобновите воспроизведение песни.",
2021-01-23 21:47:04 +01:00
restricted_device: "Невозможно управлять Spotify во время воспроизведения музыки на ограниченном устройстве",
toast_copyurl_fail: "URL-адрес песни не может быть скопирован в буфер обмена",
toast_copyurl_success: "URL песни скопирован в буфер обмена"
};
case "sv": // Swedish
return {
noaccount_header: "Något saknas",
noaccount_subheader: "Du måste ansluta ett Spotify-konto",
noaccount_text: "Du saknar ett anslutet Spotify-konto utan ett konto kan du inte använda Spotify Controls. För att ansluta ett Spotify-konto till ditt Discord-konto, klicka på knappen nedan.",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord tappade anslutningen till den senaste enheten som spelade Spotify, öppna Spotify på enheten igen och återuppta låten manuellt",
2021-01-23 21:47:04 +01:00
restricted_device: "Kan inte styra Spotify när du spelar musik på en begränsad enhet",
toast_copyurl_fail: "Låtens URL kunde inte kopieras till Urklipp",
toast_copyurl_success: "Låtens URL kopierades till Urklipp"
};
case "th": // Thai
return {
noaccount_header: "มีบางอย่างหายไป",
noaccount_subheader: "คุณต้องเชื่อมต่อบัญชี Spotify",
noaccount_text: "คุณไม่มีบัญชี Spotify ที่เชื่อมต่อหากไม่มีบัญชีคุณจะไม่สามารถใช้ Spotify Controls ได้หากต้องการเชื่อมต่อบัญชี Spotify กับบัญชี Discord ของคุณให้คลิกปุ่มด้านล่าง",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord ขาดการเชื่อมต่อกับอุปกรณ์ล่าสุดที่กำลังเล่น Spotify เปิด Spotify บนอุปกรณ์อีกครั้งและเล่นเพลงต่อด้วยตนเอง",
2021-01-23 21:47:04 +01:00
restricted_device: "ไม่สามารถควบคุม Spotify ขณะเล่นเพลงบนอุปกรณ์ที่ จำกัด",
toast_copyurl_fail: "ไม่สามารถคัดลอก URL ของเพลงไปยังคลิปบอร์ด",
toast_copyurl_success: "คัดลอก URL ของเพลงไปยังคลิปบอร์ดแล้ว"
};
case "tr": // Turkish
return {
noaccount_header: "Bir şey eksik",
noaccount_subheader: "Spotify Hesabı bağlamanız gerekiyor",
noaccount_text: "Bağlı bir Spotify Hesabınız yok, bir Hesap olmadan Spotify Controls kullanamazsınız. Bir Spotify Hesabını Discord Hesabınıza bağlamak için aşağıdaki düğmeyi tıklayın.",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord, Spotify'ı çalan son cihazla olan bağlantısını kaybetti, cihazda Spotify'ı tekrar açın ve şarkıyı manuel olarak devam ettirin",
2021-01-23 21:47:04 +01:00
restricted_device: "Kısıtlı Cihazda Müzik çalarken Spotify 'ı kontrol edemez",
toast_copyurl_fail: "Şarkı URL'si panoya kopyalanamadı",
toast_copyurl_success: "Şarkı URL'si panoya kopyalandı"
};
case "uk": // Ukrainian
return {
noaccount_header: "Щось не вистачає",
noaccount_subheader: "Вам потрібно підключити акаунт Spotify",
noaccount_text: "У вас відсутній підключений обліковий запис Spotify, без якого ви не зможете використовувати Spotify Controls. Щоб підключити обліковий запис Spotify до свого облікового запису Discord, натисніть кнопку нижче.",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord втратив з’єднання з останнім пристроєм, на якому відтворювався Spotify, знову відкрийте Spotify на пристрої та вручну відновіть пісню",
2021-01-23 21:47:04 +01:00
restricted_device: "Не вдається керувати Spotify під час відтворення музики на обмеженому пристрої",
toast_copyurl_fail: "URL-адресу пісні не вдалося скопіювати в буфер обміну",
toast_copyurl_success: "URL-адресу пісні скопійовано в буфер обміну"
};
case "vi": // Vietnamese
return {
noaccount_header: "Thiêu một thư gi đo",
noaccount_subheader: "Bạn cần kết nối Tài khoản Spotify",
noaccount_text: "Bạn đang thiếu Tài khoản Spotify được kết nối, nếu không có Tài khoản, bạn sẽ không thể sử dụng Spotify Controls. Để kết nối Tài khoản Spotify với Tài khoản Discord của bạn, hãy nhấp vào nút bên dưới.",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord mất kết nối với thiết bị cuối cùng đang phát Spotify, hãy mở lại Spotify trên thiết bị và tiếp tục bài hát theo cách thủ công",
2021-01-23 21:47:04 +01:00
restricted_device: "Không thể điều khiển Spotify khi phát Nhạc trên Thiết bị bị hạn chế",
toast_copyurl_fail: "Không thể sao chép URL bài hát vào khay nhớ tạm",
toast_copyurl_success: "URL bài hát đã được sao chép vào khay nhớ tạm"
};
case "zh-CN": // Chinese (China)
return {
noaccount_header: "缺了点什么",
noaccount_subheader: "您需要连接一个 Spotify 帐户",
noaccount_text: "您缺少关联的 Spotify 帐户,如果没有帐户,将无法使用 Spotify Controls。要将 Spotify 帐户与您的 Discord 帐户关联,请单击下面的按钮。",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord 失去了与上一个播放 Spotify 的设备的连接,再次在设备上打开 Spotify 并手动恢复歌曲",
2021-01-23 21:47:04 +01:00
restricted_device: "在受限设备上播放音乐时无法控制 Spotify",
toast_copyurl_fail: "歌曲网址无法复制到剪贴板",
toast_copyurl_success: "歌曲网址已复制到剪贴板"
};
case "zh-TW": // Chinese (Taiwan)
return {
noaccount_header: "缺了點什麼",
noaccount_subheader: "您需要連接一個 Spotify 帳戶",
noaccount_text: "您缺少關聯的 Spotify 帳戶,如果沒有帳戶,將無法使用 Spotify Controls。要將 Spotify 帳戶與您的 Discord 帳戶關聯,請單擊下面的按鈕。",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord 失去了與上一個播放 Spotify 的設備的連接,再次在設備上打開 Spotify 並手動恢復歌曲",
2021-01-23 21:47:04 +01:00
restricted_device: "在受限設備上播放音樂時無法控制 Spotify",
toast_copyurl_fail: "歌曲網址無法複製到剪貼板",
toast_copyurl_success: "歌曲網址已復製到剪貼板"
};
default: // English
return {
noaccount_header: "Something is missing",
noaccount_subheader: "You need to connect a Spotify Account",
noaccount_text: "You are missing a connected Spotify Account, without an Account you won't be able to use Spotify Controls. To connect a Spotify Account with your Discord Account click the button below.",
2022-06-17 20:38:21 +02:00
nodevice_text: "Discord lost the connection to the last device that was playing Spotify, open Spotify on the device again and manually resume the song",
2021-01-23 21:47:04 +01:00
restricted_device: "Can not control Spotify while playing Music on restricted Device",
toast_copyurl_fail: "Song URL could not be copied to clipboard",
toast_copyurl_success: "Song URL was copied to clipboard"
};
}
}
2020-09-19 20:49:33 +02:00
};
2022-09-01 14:40:11 +02:00
})(window.BDFDB_Global.PluginUtils.buildPlugin(changeLog));
2021-01-29 07:55:45 +01:00
})();