diff --git a/src/plugins/webPWA/index.ts b/src/plugins/webPWA/index.ts new file mode 100644 index 00000000000..fcdfecfd356 --- /dev/null +++ b/src/plugins/webPWA/index.ts @@ -0,0 +1,115 @@ +/* + * Vencord, a Discord client mod + * Copyright (c) 2024 Vendicated and contributors + * SPDX-License-Identifier: GPL-3.0-or-later + */ +import "./styles.css"; + +import { Devs } from "@utils/constants"; +import definePlugin from "@utils/types"; +import { filters, waitFor } from "@webpack"; +import { RelationshipStore } from "@webpack/common"; + +const manifest = { + name: "Discord", + short_name: "Discord", + start_url: "https://discord.com/channels/@me", // URL when PWA launches + display: "fullscreen", + display_override: ["window-controls-overlay"], + lang: "en-US", + background_color: "#2a2a2f", + theme_color: "#2a2a2f", + scope: "https://discord.com", // scope of all possible URL"s + description: "Imagine a place...", + orientation: "landscape", + icons: [ + { + src: "", + sizes: "256x256", + type: "image/png" + } + ] +}; + +const isMac = navigator.platform.startsWith("Mac"); + +let link: HTMLLinkElement | undefined; + +let GuildReadStateStore: any; +let NotificationSettingsStore: any; + +function setBadge() { + if (!enabled) return; + try { + const mentionCount = GuildReadStateStore.getTotalMentionCount(); + const pendingRequests = RelationshipStore.getPendingCount(); + const hasUnread = GuildReadStateStore.hasAnyUnread(); + const disableUnreadBadge = NotificationSettingsStore.getDisableUnreadBadge(); + + let totalCount = mentionCount + pendingRequests; + if (!totalCount && hasUnread && !disableUnreadBadge) totalCount = -1; + + navigator.setAppBadge(totalCount); + } catch (e) { + console.error(e); + } +} + +let toFind = 3; + +function waitForAndSubscribeToStore(name: string, cb?: (m: any) => void) { + waitFor(filters.byStoreName(name), store => { + cb?.(store); + store.addChangeListener(setBadge); + + toFind--; + if (toFind === 0) setBadge(); + }); +} + +let enabled = false; + +waitForAndSubscribeToStore("GuildReadStateStore", store => (GuildReadStateStore = store)); +waitForAndSubscribeToStore("NotificationSettingsStore", store => (NotificationSettingsStore = store)); +waitForAndSubscribeToStore("RelationshipStore"); +export default definePlugin({ + name: "WebPWA", + description: "Allows Discord to be installable and usable as a PWA.", + authors: [Devs.ThaUnknown], + manifest: null, + start() { + const url = URL.createObjectURL(new Blob([JSON.stringify(manifest)], { type: "application/json" })); + enabled = true; + link = document.createElement("link"); + link.rel = "manifest"; + link.href = url; + }, + stop() { + link?.remove(); + enabled = false; + navigator.setAppBadge(0); + }, + patches: [ + { + find: "platform-web", + replacement: { + // eslint-disable-next-line no-useless-escape + match: /(?<=" platform-overlay"\):)\i/, + replace: "$self.getPlatformClass()" + } + }, + { + find: "\"NotificationSettingsStore", + replacement: { + // eslint-disable-next-line no-useless-escape + match: /\.isPlatformEmbedded(?=\?\i\.\i\.ALL)/g, + replace: "$&||true" + } + } + ], + + getPlatformClass() { + if (isMac) return "platform-osx"; + return "platform-win"; + } +}); diff --git a/src/plugins/webPWA/styles.css b/src/plugins/webPWA/styles.css new file mode 100644 index 00000000000..e44c20a62c0 --- /dev/null +++ b/src/plugins/webPWA/styles.css @@ -0,0 +1,11 @@ +/* Download Desktop button in guilds list */ +[class^=listItem_]:has([data-list-item-id=guildsnav___app-download-button]), +[class^=listItem_]:has(+ [class^=listItem_] [data-list-item-id=guildsnav___app-download-button]) { + display: none; +} + +/* Workaround for making things in the draggable area clickable again on macOS */ +.platform-osx [class*=topic_], +.platform-osx [class*=notice_] button { + -webkit-app-region: no-drag; +} \ No newline at end of file diff --git a/src/utils/constants.ts b/src/utils/constants.ts index 70eca56fdc9..4dc49505f62 100644 --- a/src/utils/constants.ts +++ b/src/utils/constants.ts @@ -575,10 +575,14 @@ export const Devs = /* #__PURE__*/ Object.freeze({ name: "RamziAH", id: 1279957227612147747n, }, - SomeAspy: { + SomeAspy: { name: "SomeAspy", id: 516750892372852754n, }, + ThaUnknown: { + name: "ThaUnknown_", + id: 252390917665718273n + }, } satisfies Record); // iife so #__PURE__ works correctly