diff --git a/README.md b/README.md index 2c4bd78..fb3996c 100644 --- a/README.md +++ b/README.md @@ -4,16 +4,62 @@ Make your experience better with spotify links. Like Playing directly or adding in queue. (Also opening directly in app) -Direct Download Link: [dev.tharki.Distify.asar](https://github.com/YofukashiNo/Distify/releases/latest/download/dev.tharki.Distify.asar) +Direct Download Link: +[![dev.yofukashino.Distify.asar](https://img.shields.io/github/downloads/YofukashiNo/MessageLinkEmbeds/total.svg?style=social&label=Direct%20Download)](https://github.com/YofukashiNo/Distify/releases/latest/download/dev.yofukashino.Distify.asar) Install Link: +[![Safe](https://img.shields.io/github/downloads/YofukashiNo/MessageLinkEmbeds/total.svg?style=social&label=Install%20in%20Replugged&logo=)](https://replugged.dev/install?identifier=YofukashiNo/Distify&source=github) +![distify](https://i.imgur.com/i7r0j9L.png) -[![Install in Replugged](https://img.shields.io/badge/-Install%20in%20Replugged-blue?style=for-the-badge&logo=none)](https://replugged.dev/install?identifier=YofukashiNo/Distify&source=github) +--- -![image](https://i.imgur.com/i7r0j9L.png) +# FAQ +### How to install a plugin? +
+Automatic +- Simply click on install link and thats all <3 +
+
+Manual -> For Contributing: [Make a pr thats all.] +- Click on Direct Download Link and save the file +- Open Replugged Plugin Settings +- Click on "Open Plugins Folder" button +- Paste downloaded file in folder that opened +
+### How Do I contribute? +- Just make a pull request + +### How Do I support without pull request? +- You Can Donate on my [ko-fi](https://ko-fi.com/yofukashino) or UPI at `yofukashi.no.singh@fam` + +[![Buy Me a Coffee at ko-fi.com](https://storage.ko-fi.com/cdn/kofi3.png?v=3)](https://ko-fi.com/yofukashino) + +### Where can I find other plugins by you? + +You can find them [here](https://github.com/YofukashiNo/RepluggedPlugins) + +### This plugin isn't listed in the all plugin list? + +PRs are appreciated or just make an issue on the [repo](https://github.com/YofukashiNo/RepluggedPlugins) + + +### Where can I find the support? + +~~You can make post in support channel of replugged server and ping me.~~ + +OR + +There is support server for all the plugins. You can join it here: + +[![Support Server](https://discordapp.com/api/guilds/919649417005506600/widget.png?style=banner3)](https://discord.gg/SgKSKyh9gY) + + + +# Who is the author of these plugins? + +[![Discord Presence](https://lanyard.cnrad.dev/api/1121961711080050780?hideDiscrim=true&idleMessage=Leave%20the%20kid%20alone...)](https://discordapp.com/users/1121961711080050780) diff --git a/manifest.json b/manifest.json index 0634cf9..df6b87d 100644 --- a/manifest.json +++ b/manifest.json @@ -1,16 +1,16 @@ { - "id": "dev.tharki.Distify", + "id": "dev.yofukashino.Distify", "name": "Distify", "description": "Make your experience better with spotify links.", "author": { - "name": "Ahlawat", + "name": "Nanakusa", "discordID": "1121961711080050780", "github": "YofukashiNo" }, "version": "1.0.3", "updater": { "type": "store", - "id": "dev.tharki.Distify" + "id": "dev.yofukashino.Distify" }, "license": "MIT", "type": "replugged-plugin", diff --git a/src/Components/MenuItems.tsx b/src/Components/MenuItems.tsx index 3b148a8..4dbcbf2 100644 --- a/src/Components/MenuItems.tsx +++ b/src/Components/MenuItems.tsx @@ -3,7 +3,6 @@ import { ContextMenu } from "replugged/components"; import { PluginLogger } from "../index"; import Utils from "../lib/utils"; import Icons from "./Icons"; -import Types from "../types"; export const noAccounts = (): React.ReactElement => { return ( @@ -17,10 +16,7 @@ export const noAccounts = (): React.ReactElement => { ); }; -export const addToQueue = ( - SpotifyLinks: string[][], - SpotifyAccount: Types.SpotifyAccounts, -): React.ReactElement => { +export const addToQueue = (SpotifyLinks: string[][]): React.ReactElement => { if (SpotifyLinks.length === 1) { const [, type, id, name] = SpotifyLinks[0]; return ( @@ -32,7 +28,7 @@ export const addToQueue = ( icon={() => } action={async () => { try { - await Utils.queue(type, id, SpotifyAccount.accessToken); + await Utils.queue(type, id); ToastUtils.toast("Successfully Queued on Spotify", ToastUtils.Kind.SUCCESS, { duration: 5000, }); @@ -54,8 +50,7 @@ export const addToQueue = ( icon={() => } action={async () => { try { - for (const [, type, id] of SpotifyLinks) - await Utils.queue(type, id, SpotifyAccount.accessToken); + for (const [, type, id] of SpotifyLinks) await Utils.queue(type, id); ToastUtils.toast("Successfully Queued on Spotify", ToastUtils.Kind.SUCCESS, { duration: 5000, }); @@ -75,7 +70,7 @@ export const addToQueue = ( icon={() => } action={async () => { try { - await Utils.queue(type, id, SpotifyAccount.accessToken); + await Utils.queue(type, id); ToastUtils.toast("Successfully Queued on Spotify", ToastUtils.Kind.SUCCESS, { duration: 5000, }); @@ -93,10 +88,7 @@ export const addToQueue = ( ); }; -export const play = ( - SpotifyLinks: string[][], - SpotifyAccount: Types.SpotifyAccounts, -): React.ReactElement => { +export const play = (SpotifyLinks: string[][]): React.ReactElement => { if (SpotifyLinks.length === 1) { const [, type, id, name] = SpotifyLinks[0]; return ( @@ -108,7 +100,7 @@ export const play = ( icon={() => } action={async () => { try { - await Utils.play(type, id, SpotifyAccount.accessToken); + await Utils.play(type, id); ToastUtils.toast("Successfully Played on Spotify", ToastUtils.Kind.SUCCESS, { duration: 5000, }); @@ -130,8 +122,7 @@ export const play = ( icon={() => } action={async () => { try { - for (const [, type, id] of SpotifyLinks) - await Utils.play(type, id, SpotifyAccount.accessToken); + for (const [, type, id] of SpotifyLinks) await Utils.play(type, id); ToastUtils.toast("Successfully Played on Spotify", ToastUtils.Kind.SUCCESS, { duration: 5000, }); @@ -151,7 +142,7 @@ export const play = ( icon={() => } action={async () => { try { - await Utils.play(type, id, SpotifyAccount.accessToken); + await Utils.play(type, id); ToastUtils.toast("Successfully Played on Spotify", ToastUtils.Kind.SUCCESS, { duration: 5000, }); diff --git a/src/lib/requiredModules.ts b/src/lib/requiredModules.ts index 7a1c4b1..1be20ba 100644 --- a/src/lib/requiredModules.ts +++ b/src/lib/requiredModules.ts @@ -5,3 +5,5 @@ export const ConnectedAccountsStore = webpack.getByStoreName("ConnectedAccountsStore"); export const ElementParser = webpack.getByProps("sanitizeUrl", "sanitizeText"); + +export const SpotifyStore = webpack.getByStoreName("SpotifyStore"); diff --git a/src/lib/utils.tsx b/src/lib/utils.tsx index c0a915a..3c2b448 100644 --- a/src/lib/utils.tsx +++ b/src/lib/utils.tsx @@ -2,10 +2,44 @@ import { contextMenu as ContextMenuUtils, React } from "replugged/common"; import { ContextMenu } from "replugged/components"; import { PluginLogger } from "../index"; import { BASE_URL, BASE_URL_PLAYER } from "./consts"; -import { ConnectedAccountsStore } from "./requiredModules"; +import { ConnectedAccountsStore, SpotifyStore } from "./requiredModules"; import MenuItems from "../Components/MenuItems"; import Types from "../types"; export const customCacheSpotifyMeta = new Map(); +export const ensureSpotifyPlayer = (): Promise<{ + socket?: Types.SpotifySocket; + device?: Types.SpotifyDevice; +}> => { + const activePlayer = SpotifyStore.getActiveSocketAndDevice(); + if (activePlayer) return Promise.resolve(activePlayer); + const playableDevices = SpotifyStore.getPlayableComputerDevices(); + if (playableDevices.length > 0) { + const [{ socket, device }] = playableDevices; + return Promise.resolve({ + socket, + device, + }); + } + return new Promise((res) => { + const timer = { timeout: null }; + const changeListerner = () => { + const playableDevices = SpotifyStore.getPlayableComputerDevices(); + const [{ socket, device }] = playableDevices; + clearTimeout(timer?.timeout); + SpotifyStore.removeChangeListener(changeListerner); + res({ + socket, + device, + }); + }; + SpotifyStore.addChangeListener(changeListerner); + timer.timeout = setTimeout(() => { + SpotifyStore.removeChangeListener(changeListerner); + res({}); + }, 2500); + open("spotify:"); + }); +}; export const error = async (res): Promise => { switch (res.status) { case 401: @@ -21,13 +55,14 @@ export const error = async (res): Promise => { return new Error("Unknown Error, Check the console and report the dev"); } }; -export const play = async (type: string, id: string, accessToken: string): Promise => { - if (!accessToken) { +export const play = async (type: string, id: string): Promise => { + const { socket, device } = await ensureSpotifyPlayer(); + if (!socket?.accessToken) { PluginLogger.error("Please link your Spotify to Discord in Settings > Connections"); return; } - const SpotifyResponse = await fetch(`${BASE_URL_PLAYER}/play`, { + const SpotifyResponse = await fetch(`${BASE_URL_PLAYER}/play?device_id=${device.id}`, { method: "PUT", body: JSON.stringify( type === "track" @@ -36,7 +71,7 @@ export const play = async (type: string, id: string, accessToken: string): Promi ), headers: { "Content-Type": "application/json", - Authorization: `Bearer ${accessToken}`, + Authorization: `Bearer ${socket?.accessToken}`, }, }); if (SpotifyResponse.ok) { @@ -45,19 +80,21 @@ export const play = async (type: string, id: string, accessToken: string): Promi throw await error(SpotifyResponse); }; -export const queue = async (type: string, id: string, accessToken: string): Promise => { - if (!accessToken) { +export const queue = async (type: string, id: string): Promise => { + const { socket, device } = await ensureSpotifyPlayer(); + if (!socket?.accessToken) { PluginLogger.error("Please link your Spotify to Discord in Settings > Connections"); return; } - const SpotifyResponse = await fetch( - `${BASE_URL_PLAYER}/queue?uri=${encodeURIComponent(`spotify:${type}:${id}`)}`, + `${BASE_URL_PLAYER}/queue?uri=${encodeURIComponent(`spotify:${type}:${id}`)}&device_id=${ + device.id + }`, { method: "POST", headers: { "Content-Type": "application/json", - Authorization: `Bearer ${accessToken}`, + Authorization: `Bearer ${socket?.accessToken}`, }, }, ); @@ -104,7 +141,6 @@ export const mapMenuItems = ( }) .then((res) => res.json()) .catch(() => ({ name: "Error Fetching Name" })); - console.log(SpotifyResponse); if (SpotifyResponse?.name) customCacheSpotifyMeta.set(id, [_, type, id, SpotifyResponse?.name]); return [_, type, id, SpotifyResponse?.name ?? "Error Fetching Name"]; @@ -131,24 +167,10 @@ export const mapMenuItems = ( if ((type as { data: boolean }).data) { return SpotifyMeta; } - if (SpotifyAccounts.length === 1) { - return [ - type.play && MenuItems.play(SpotifyMeta, SpotifyAccounts[0]), - type.queue && MenuItems.addToQueue(SpotifyMeta, SpotifyAccounts[0]), - ]; - } - return SpotifyAccounts.map((SpotifyAccount) => { - return ( - - {...[ - type.play && MenuItems.play(SpotifyMeta, SpotifyAccount), - type.queue && MenuItems.addToQueue(SpotifyMeta, SpotifyAccount), - ]} - - ); - }); + return [ + type.play && MenuItems.play(SpotifyMeta), + type.queue && MenuItems.addToQueue(SpotifyMeta), + ]; } catch { return []; } @@ -179,13 +201,12 @@ export const manipulateMenu = ( message: Types.Message, menu: { children: React.ReactElement[] }, ): React.ReactElement | void => { - console.log(menu); const MenuGroup = menu?.children?.find?.((c) => c?.props?.id === "distify") ?? ( ); MenuGroup.props.id = "distify"; if (!menu?.children?.some?.((c) => c?.props?.id === "distify")) - menu?.children.splice(1, 0, MenuGroup); + menu?.children.splice(-1, 0, MenuGroup); const SpotifyLinks = Array.from( message.content.matchAll(/open.spotify.com\/(album|track|playlist)\/([^?]+)/g) as string[][] & IterableIterator, @@ -198,22 +219,8 @@ export const manipulateMenu = ( data: true, }) as string[][]; if (SpotifyLinks.length <= 0) return; - if (SpotifyMeta && SpotifyAccounts.length === 1) { - MenuGroup.props.children = [ - MenuItems.play(SpotifyMeta, SpotifyAccounts[0]), - MenuItems.addToQueue(SpotifyMeta, SpotifyAccounts[0]), - ]; - } else if (SpotifyMeta) { - MenuGroup.props.children = SpotifyAccounts.map((SpotifyAccount) => ( - - {...[ - MenuItems.play(SpotifyMeta, SpotifyAccount), - MenuItems.addToQueue(SpotifyMeta, SpotifyAccount), - ]} - - )); + if (SpotifyMeta) { + MenuGroup.props.children = [MenuItems.play(SpotifyMeta), MenuItems.addToQueue(SpotifyMeta)]; } }; diff --git a/src/patches/Popover.tsx b/src/patches/Popover.tsx index 76f188f..1fc561d 100644 --- a/src/patches/Popover.tsx +++ b/src/patches/Popover.tsx @@ -26,10 +26,10 @@ export default (): void => { }); } else { try { - const [, type, id] = SpotifyLinks.pop(); - await Utils.play(type, id, SpotifyAccounts[0].accessToken); - for (const [, type, id] of SpotifyLinks) - await Utils.queue(type, id, SpotifyAccounts[0].accessToken); + const [play, ...queue] = SpotifyLinks; + const [, type, id] = play; + await Utils.play(type, id); + for (const [, type, id] of queue ?? []) await Utils.queue(type, id); ToastUtils.toast("Successfully Played on Spotify", ToastUtils.Kind.SUCCESS, { duration: 5000, }); @@ -42,8 +42,7 @@ export default (): void => { } }, onContextMenu: (e) => { - if (SpotifyAccounts.length > 1) - Utils.openContextMenu(e, SpotifyLinks, SpotifyAccounts, { play: true, queue: false }); + Utils.openContextMenu(e, SpotifyLinks, SpotifyAccounts, { play: true, queue: false }); }, }; }); @@ -68,8 +67,7 @@ export default (): void => { }); } else { try { - for (const [, type, id] of SpotifyLinks) - await Utils.queue(type, id, SpotifyAccounts[0].accessToken); + for (const [, type, id] of SpotifyLinks) await Utils.queue(type, id); ToastUtils.toast("Successfully Queued on Spotify", ToastUtils.Kind.SUCCESS, { duration: 5000, }); @@ -82,8 +80,7 @@ export default (): void => { } }, onContextMenu: (e) => { - if (SpotifyAccounts.length > 1) - Utils.openContextMenu(e, SpotifyLinks, SpotifyAccounts, { play: false, queue: true }); + Utils.openContextMenu(e, SpotifyLinks, SpotifyAccounts, { play: false, queue: true }); }, }; }); diff --git a/src/types.ts b/src/types.ts index 380fae5..172b767 100644 --- a/src/types.ts +++ b/src/types.ts @@ -38,7 +38,49 @@ export namespace Types { isJoining: DefaultTypes.AnyFunction; isSuggestedAccountType: DefaultTypes.AnyFunction; } - + export interface SpotifyStore extends Store { + canPlay: DefaultTypes.AnyFunction; + getActiveSocketAndDevice: () => { socket: SpotifySocket; device: SpotifyDevice }; + getActivity: DefaultTypes.AnyFunction; + getLastPlayedTrackId: DefaultTypes.AnyFunction; + getPlayableComputerDevices: () => Array<{ socket: SpotifySocket; device: SpotifyDevice }>; + getPlayerState: DefaultTypes.AnyFunction; + getSyncingWith: DefaultTypes.AnyFunction; + getTrack: DefaultTypes.AnyFunction; + hasConnectedAccount: DefaultTypes.AnyFunction; + initialize: DefaultTypes.AnyFunction; + shouldShowActivity: DefaultTypes.AnyFunction; + wasAutoPaused: DefaultTypes.AnyFunction; + } + export interface SpotifySocket { + accessToken: string; + accountId: string; + connectionId: string; + handleDeviceStateChange: DefaultTypes.AnyFunction; + isPremium: boolean; + pingInterval: { _ref: number }; + socket: WebSocket; + _requestedConnect: boolean; + _requestedDisconnect: boolean; + connect: DefaultTypes.AnyFunction; + connected: boolean; + disconnect: DefaultTypes.AnyFunction; + handleClose: DefaultTypes.AnyFunction; + handleEvent: DefaultTypes.AnyFunction; + handleMessage: DefaultTypes.AnyFunction; + handleOpen: DefaultTypes.AnyFunction; + ping: DefaultTypes.AnyFunction; + } + export interface SpotifyDevice { + id: string; + is_active: boolean; + is_private_session: boolean; + is_restricted: boolean; + name: string; + supports_volume: boolean; + type: string; + volume_percent: number; + } export interface ElementParser { ReactMarkdown: DefaultTypes.AnyFunction; anyScopeRegex: DefaultTypes.AnyFunction;