From bbf3c74cb206e427877f65cbb52f318abeb0cda0 Mon Sep 17 00:00:00 2001 From: RuiNtD Date: Fri, 24 Mar 2023 21:00:27 -0600 Subject: [PATCH] Update LastFM plugin (#483) Co-authored-by: Ven Co-authored-by: Sofia Lima --- src/plugins/customRPC.tsx | 3 +- src/plugins/lastfm.tsx | 210 ++++++++++++++++++++++++++------------ src/utils/constants.ts | 4 + 3 files changed, 150 insertions(+), 67 deletions(-) diff --git a/src/plugins/customRPC.tsx b/src/plugins/customRPC.tsx index fe6d574db..ecd77e9b2 100644 --- a/src/plugins/customRPC.tsx +++ b/src/plugins/customRPC.tsx @@ -215,7 +215,8 @@ async function setRpc(disable?: boolean) { FluxDispatcher.dispatch({ type: "LOCAL_ACTIVITY_UPDATE", - activity: !disable ? activity : {} + activity: !disable ? activity : null, + socketId: "CustomRPC", }); } diff --git a/src/plugins/lastfm.tsx b/src/plugins/lastfm.tsx index 9a0647c76..c001a5977 100644 --- a/src/plugins/lastfm.tsx +++ b/src/plugins/lastfm.tsx @@ -16,9 +16,10 @@ * along with this program. If not, see . */ -import { Settings } from "@api/settings"; +import { definePluginSettings } from "@api/settings"; import { Link } from "@components/Link"; import { Devs } from "@utils/constants"; +import Logger from "@utils/Logger"; import definePlugin, { OptionType } from "@utils/types"; import { filters, findByPropsLazy, mapMangledModuleLazy } from "@webpack"; import { FluxDispatcher, Forms } from "@webpack/common"; @@ -30,6 +31,12 @@ interface ActivityAssets { small_text?: string; } + +interface ActivityButton { + label: string; + url: string; +} + interface Activity { state: string; details?: string; @@ -66,6 +73,9 @@ enum ActivityFlag { } const applicationId = "1043533871037284423"; +const placeholderId = "2a96cbd8b46e442fc41c2b86b821562f"; + +const logger = new Logger("LastFMRichPresence"); const presenceStore = findByPropsLazy("getLocalPresence"); const assetManager = mapMangledModuleLazy( @@ -79,14 +89,64 @@ async function getApplicationAsset(key: string): Promise { return (await assetManager.getAsset(applicationId, [key, undefined]))[0]; } -function setActivity(activity?: Activity) { - FluxDispatcher.dispatch({ type: "LOCAL_ACTIVITY_UPDATE", activity: activity }); +function setActivity(activity: Activity | null) { + FluxDispatcher.dispatch({ + type: "LOCAL_ACTIVITY_UPDATE", + activity, + socketId: "LastFM", + }); } +const settings = definePluginSettings({ + username: { + description: "last.fm username", + type: OptionType.STRING, + }, + apiKey: { + description: "last.fm api key", + type: OptionType.STRING, + }, + shareUsername: { + description: "show link to last.fm profile", + type: OptionType.BOOLEAN, + default: false, + }, + hideWithSpotify: { + description: "hide last.fm presence if spotify is running", + type: OptionType.BOOLEAN, + default: true, + }, + statusName: { + description: "text shown in status", + type: OptionType.STRING, + default: "some music", + }, + useListeningStatus: { + description: 'show "Listening to" status instead of "Playing"', + type: OptionType.BOOLEAN, + default: false, + }, + missingArt: { + description: "When album or album art is missing", + type: OptionType.SELECT, + options: [ + { + label: "Use large Last.fm logo", + value: "lastfmLogo", + default: true + }, + { + label: "Use generic placeholder", + value: "placeholder" + } + ], + } +}); + export default definePlugin({ name: "LastFMRichPresence", description: "Little plugin for Last.fm rich presence", - authors: [Devs.dzshn], + authors: [Devs.dzshn, Devs.RuiNtD], settingsAboutComponent: () => ( <> @@ -104,30 +164,9 @@ export default definePlugin({ ), - options: { - username: { - description: "last.fm username", - type: OptionType.STRING, - }, - apiKey: { - description: "last.fm api key", - type: OptionType.STRING, - }, - hideWithSpotify: { - description: "hide last.fm presence if spotify is running", - type: OptionType.BOOLEAN, - default: true, - }, - useListeningStatus: { - description: 'show "Listening to" status instead of "Playing"', - type: OptionType.BOOLEAN, - default: false, - } - }, + settings, start() { - this.settings = Settings.plugins.LastFMRichPresence; - this.updateInterval = setInterval(() => { this.updatePresence(); }, 16000); }, @@ -136,73 +175,112 @@ export default definePlugin({ }, async fetchTrackData(): Promise { - if (!this.settings.username || !this.settings.apiKey) return null; + if (!settings.store.username || !settings.store.apiKey) + return null; - const response = await fetch(`https://ws.audioscrobbler.com/2.0/?method=user.getrecenttracks&api_key=${this.settings.apiKey}&user=${this.settings.username}&limit=1&format=json`); - const trackData = (await response.json()).recenttracks.track[0]; + try { + const params = new URLSearchParams({ + method: "user.getrecenttracks", + api_key: settings.store.apiKey, + user: settings.store.username, + limit: "1", + format: "json" + }); - if (!trackData["@attr"]?.nowplaying) return null; + const res = await fetch(`https://ws.audioscrobbler.com/2.0/?${params}`); + if (!res.ok) throw `${res.status} ${res.statusText}`; - // why does the json api have xml structure - return { - name: trackData.name || "Unknown", - album: trackData.album["#text"], - artist: trackData.artist["#text"] || "Unknown", - url: trackData.url, - imageUrl: (trackData.image || []).filter(x => x.size === "large")[0]?.["#text"] - }; + const json = await res.json(); + if (json.error) { + logger.error("Error from Last.fm API", `${json.error}: ${json.message}`); + return null; + } + + const trackData = json.recenttracks?.track[0]; + + if (!trackData || !trackData["@attr"]?.nowplaying) + return null; + + // why does the json api have xml structure + return { + name: trackData.name || "Unknown", + album: trackData.album["#text"], + artist: trackData.artist["#text"] || "Unknown", + url: trackData.url, + imageUrl: trackData.image?.find((x: any) => x.size === "large")?.["#text"] + }; + } catch (e) { + logger.error("Failed to query Last.fm API", e); + // will clear the rich presence if API fails + return null; + } }, async updatePresence() { - if (this.settings.hideWithSpotify) { + setActivity(await this.getActivity()); + }, + + getLargeImage(track: TrackData): string | undefined { + if (track.imageUrl && !track.imageUrl.includes(placeholderId)) + return track.imageUrl; + + if (settings.store.missingArt === "placeholder") + return "placeholder"; + }, + + async getActivity(): Promise { + if (settings.store.hideWithSpotify) { for (const activity of presenceStore.getActivities()) { if (activity.type === ActivityType.LISTENING && activity.application_id !== applicationId) { - // there is already music status (probably only spotify can do this currently) - setActivity(); - return; + // there is already music status because of Spotify or richerCider (probably more) + return null; } } } const trackData = await this.fetchTrackData(); + if (!trackData) return null; - if (!trackData) { - setActivity(); - return; - } - - const hideAlbumName = !trackData.album || trackData.album === trackData.name; - - let assets: ActivityAssets; - if (trackData.imageUrl) { - assets = { - large_image: await getApplicationAsset(trackData.imageUrl), - large_text: trackData.name, + const largeImage = this.getLargeImage(trackData); + const assets: ActivityAssets = largeImage ? + { + large_image: await getApplicationAsset(largeImage), + large_text: trackData.album || undefined, small_image: await getApplicationAsset("lastfm-small"), small_text: "Last.fm", - }; - } else { - assets = { + } : { large_image: await getApplicationAsset("lastfm-large"), - large_text: "Last.fm", + large_text: trackData.album || undefined, }; - } - setActivity({ + const buttons: ActivityButton[] = [ + { + label: "View Song", + url: trackData.url, + }, + ]; + + if (settings.store.shareUsername) + buttons.push({ + label: "Last.fm Profile", + url: `https://www.last.fm/user/${settings.store.username}`, + }); + + return { application_id: applicationId, - name: "some music", + name: settings.store.statusName, details: trackData.name, - state: hideAlbumName ? trackData.artist : `${trackData.artist} - ${trackData.album}`, + state: trackData.artist, assets, - buttons: ["Open in Last.fm"], + buttons: buttons.map(v => v.label), metadata: { - button_urls: [trackData.url] + button_urls: buttons.map(v => v.url), }, - type: this.settings.useListeningStatus ? ActivityType.LISTENING : ActivityType.PLAYING, + type: settings.store.useListeningStatus ? ActivityType.LISTENING : ActivityType.PLAYING, flags: ActivityFlag.INSTANCE, - }); + }; } }); diff --git a/src/utils/constants.ts b/src/utils/constants.ts index bacf914eb..e7ac09aa7 100644 --- a/src/utils/constants.ts +++ b/src/utils/constants.ts @@ -202,6 +202,10 @@ export const Devs = /* #__PURE__*/ Object.freeze({ name: "lewisakura", id: 96269247411400704n }, + RuiNtD: { + name: "RuiNtD", + id: 157917665162297344n + }, hunt: { name: "hunt-g", id: 222800179697287168n