forked from mirrors/Vencord
99 lines
3.2 KiB
TypeScript
99 lines
3.2 KiB
TypeScript
/*
|
|
* Vencord, a modification for Discord's desktop app
|
|
* Copyright (c) 2023 Vendicated and contributors
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
import { Settings } from "@api/settings";
|
|
import { Queue } from "@utils/Queue";
|
|
import { ReactDOM } from "@webpack/common";
|
|
import type { ReactNode } from "react";
|
|
import type { Root } from "react-dom/client";
|
|
|
|
import NotificationComponent from "./NotificationComponent";
|
|
|
|
const NotificationQueue = new Queue();
|
|
|
|
let reactRoot: Root;
|
|
let id = 42;
|
|
|
|
function getRoot() {
|
|
if (!reactRoot) {
|
|
const container = document.createElement("div");
|
|
container.id = "vc-notification-container";
|
|
document.body.append(container);
|
|
reactRoot = ReactDOM.createRoot(container);
|
|
}
|
|
return reactRoot;
|
|
}
|
|
|
|
export interface NotificationData {
|
|
title: string;
|
|
body: string;
|
|
/**
|
|
* Same as body but can be a custom component.
|
|
* Will be used over body if present.
|
|
* Not supported on desktop notifications, those will fall back to body */
|
|
richBody?: ReactNode;
|
|
/** Small icon. This is for things like profile pictures and should be square */
|
|
icon?: string;
|
|
/** Large image. Optimally, this should be around 16x9 but it doesn't matter much. Desktop Notifications might not support this */
|
|
image?: string;
|
|
onClick?(): void;
|
|
onClose?(): void;
|
|
color?: string;
|
|
}
|
|
|
|
function _showNotification(notification: NotificationData, id: number) {
|
|
const root = getRoot();
|
|
return new Promise<void>(resolve => {
|
|
root.render(
|
|
<NotificationComponent key={id} {...notification} onClose={() => {
|
|
notification.onClose?.();
|
|
root.render(null);
|
|
resolve();
|
|
}} />,
|
|
);
|
|
});
|
|
}
|
|
|
|
function shouldBeNative() {
|
|
const { useNative } = Settings.notifications;
|
|
if (useNative === "always") return true;
|
|
if (useNative === "not-focused") return !document.hasFocus();
|
|
return false;
|
|
}
|
|
|
|
export async function requestPermission() {
|
|
return (
|
|
Notification.permission === "granted" ||
|
|
(Notification.permission !== "denied" && (await Notification.requestPermission()) === "granted")
|
|
);
|
|
}
|
|
|
|
export async function showNotification(data: NotificationData) {
|
|
if (shouldBeNative() && await requestPermission()) {
|
|
const { title, body, icon, image, onClick = null, onClose = null } = data;
|
|
const n = new Notification(title, {
|
|
body,
|
|
icon,
|
|
image
|
|
});
|
|
n.onclick = onClick;
|
|
n.onclose = onClose;
|
|
} else {
|
|
NotificationQueue.push(() => _showNotification(data, id++));
|
|
}
|
|
}
|