import { Client, Presence } from "discord-rpc"; import { app, ipcMain } from "electron"; import { globalEvents } from "../constants/globalEvents"; import { settings } from "../constants/settings"; import { Logger } from "../features/logger"; import { convertDurationToSeconds } from "../features/time/parse"; import { MediaStatus } from "../models/mediaStatus"; import { mediaInfo } from "./mediaInfo"; import { settingsStore } from "./settings"; const clientId = "833617820704440341"; export let rpc: Client; const observer = () => { if (rpc) { const showIdle = settingsStore.get(settings.discord.showIdle) ?? true; if (mediaInfo.status === MediaStatus.paused && !showIdle) { rpc.clearActivity(); } else rpc.setActivity(getActivity()); } }; const defaultPresence = { largeImageKey: "tidal-hifi-icon", largeImageText: `TIDAL Hi-Fi ${app.getVersion()}`, instance: false, }; const getActivity = (): Presence => { const presence: Presence = { ...defaultPresence }; if (mediaInfo.status === MediaStatus.paused) { presence.details = settingsStore.get(settings.discord.idleText) ?? "Browsing Tidal"; } else { const showSong = settingsStore.get(settings.discord.showSong) ?? false; if (showSong) { const { includeTimestamps, detailsPrefix, buttonText } = getFromStore(); includeTimeStamps(includeTimestamps); setPresenceFromMediaInfo(detailsPrefix, buttonText); } else { presence.details = settingsStore.get(settings.discord.usingText) ?? "Playing media on TIDAL"; } } return presence; function getFromStore() { const includeTimestamps = settingsStore.get(settings.discord.includeTimestamps) ?? true; const detailsPrefix = settingsStore.get(settings.discord.detailsPrefix) ?? "Listening to "; const buttonText = settingsStore.get(settings.discord.buttonText) ?? "Play on TIDAL"; return { includeTimestamps, detailsPrefix, buttonText }; } function setPresenceFromMediaInfo(detailsPrefix: string, buttonText: string) { if (mediaInfo.url) { presence.details = `${detailsPrefix}${mediaInfo.title}`; presence.state = mediaInfo.artists ? mediaInfo.artists : "unknown artist(s)"; presence.largeImageKey = mediaInfo.image; if (mediaInfo.album) { presence.largeImageText = mediaInfo.album; } presence.buttons = [{ label: buttonText, url: mediaInfo.url }]; } else { presence.details = `Watching ${mediaInfo.title}`; presence.state = mediaInfo.artists; } } function includeTimeStamps(includeTimestamps: boolean) { if (includeTimestamps) { const currentSeconds = convertDurationToSeconds(mediaInfo.current); const durationSeconds = convertDurationToSeconds(mediaInfo.duration); const date = new Date(); const now = (date.getTime() / 1000) | 0; const remaining = date.setSeconds(date.getSeconds() + (durationSeconds - currentSeconds)); presence.startTimestamp = now; presence.endTimestamp = remaining; } } }; /** * Set up the discord rpc and listen on globalEvents.updateInfo */ export const initRPC = () => { rpc = new Client({ transport: "ipc" }); rpc.login({ clientId }).then( () => { rpc.on("ready", () => { const showIdle = settingsStore.get(settings.discord.showIdle) ?? true; if (mediaInfo.status === MediaStatus.paused && !showIdle) { rpc.clearActivity(); } else rpc.setActivity(getActivity()); }); ipcMain.on(globalEvents.updateInfo, observer); }, () => { Logger.log("Can't connect to Discord, is it running?"); } ); }; /** * Remove any RPC connection with discord and remove the event listener on globalEvents.updateInfo */ export const unRPC = () => { if (rpc) { rpc.clearActivity(); rpc.destroy(); rpc = null; ipcMain.removeListener(globalEvents.updateInfo, observer); } };