import { decodeTime } from "ulid"; import { AppContext } from "./RevoltClient"; import { useTranslation } from "../../lib/i18n"; import { Users } from "revolt.js/dist/api/objects"; import { useContext, useEffect } from "preact/hooks"; import { connectState } from "../../redux/connector"; import { playSound } from "../../assets/sounds/Audio"; import { Message, SYSTEM_USER_ID, User } from "revolt.js"; import { NotificationOptions } from "../../redux/reducers/settings"; import { Route, Switch, useHistory, useParams } from "react-router-dom"; interface Props { options?: NotificationOptions; } const notifications: { [key: string]: Notification } = {}; async function createNotification(title: string, options: globalThis.NotificationOptions) { try { return new Notification(title, options); } catch (err) { let sw = await navigator.serviceWorker.getRegistration(); sw?.showNotification(title, options); } } function Notifier(props: Props) { const translate = useTranslation(); const showNotification = props.options?.desktopEnabled ?? false; // const playIncoming = props.options?.soundEnabled ?? true; // const playOutgoing = props.options?.outgoingSoundEnabled ?? true; const client = useContext(AppContext); const { guild: guild_id, channel: channel_id } = useParams<{ guild: string; channel: string; }>(); const history = useHistory(); async function message(msg: Message) { if (msg.author === client.user!._id) return; if (msg.channel === channel_id && document.hasFocus()) return; if (client.user?.status?.presence === Users.Presence.Busy) return; // Sounds.playInbound(); playSound('message'); if (!showNotification) return; const channel = client.channels.get(msg.channel); const author = client.users.get(msg.author); if (author?.relationship === Users.Relationship.Blocked) return; let title; switch (channel?.channel_type) { case "SavedMessages": return; case "DirectMessage": title = `@${author?.username}`; break; case "Group": if (author?._id === SYSTEM_USER_ID) { title = channel.name; } else { title = `@${author?.username} - ${channel.name}`; } break; case "TextChannel": const server = client.servers.get(channel.server); title = `@${author?.username} (#${channel.name}, ${server?.name})`; break; default: title = msg.channel; break; } let image; if (msg.attachments) { let imageAttachment = msg.attachments.find(x => x.metadata.type === 'Image'); if (imageAttachment) { image = client.generateFileURL(imageAttachment, { max_side: 720 }); } } let body, icon; if (typeof msg.content === "string") { body = client.markdownToText(msg.content); icon = client.users.getAvatarURL(msg.author, { max_side: 256 }); } else { let users = client.users; switch (msg.content.type) { case "user_added": case "user_remove": body = translate( `app.main.channel.system.${msg.content.type === 'user_added' ? 'added_by' : 'removed_by'}`, { user: users.get(msg.content.id)?.username, other_user: users.get(msg.content.by)?.username } ); icon = client.users.getAvatarURL(msg.content.id, { max_side: 256 }); break; case "user_joined": case "user_left": case "user_kicked": case "user_banned": body = translate( `app.main.channel.system.${msg.content.type}`, { user: users.get(msg.content.id)?.username } ); icon = client.users.getAvatarURL(msg.content.id, { max_side: 256 }); break; case "channel_renamed": body = translate( `app.main.channel.system.channel_renamed`, { user: users.get(msg.content.by)?.username, name: msg.content.name } ); icon = client.users.getAvatarURL(msg.content.by, { max_side: 256 }); break; case "channel_description_changed": case "channel_icon_changed": body = translate( `app.main.channel.system.${msg.content.type}`, { user: users.get(msg.content.by)?.username } ); icon = client.users.getAvatarURL(msg.content.by, { max_side: 256 }); break; } } let notif = await createNotification(title, { icon, image, body, timestamp: decodeTime(msg._id), tag: msg.channel, badge: '/assets/icons/android-chrome-512x512.png', silent: true }); if (notif) { notif.addEventListener("click", () => { window.focus(); const id = msg.channel; if (id !== channel_id) { let channel = client.channels.get(id); if (channel) { if (channel.channel_type === 'TextChannel') { history.push(`/server/${channel.server}/channel/${id}`); } else { history.push(`/channel/${id}`); } } } }); notifications[msg.channel] = notif; notif.addEventListener( "close", () => delete notifications[msg.channel] ); } } async function relationship(user: User, property: string) { if (client.user?.status?.presence === Users.Presence.Busy) return; if (property !== "relationship") return; if (!showNotification) return; let event; switch (user.relationship) { case Users.Relationship.Incoming: event = translate("notifications.sent_request", { person: user.username }); break; case Users.Relationship.Friend: event = translate("notifications.now_friends", { person: user.username }); break; default: return; } let notif = await createNotification(event, { icon: client.users.getAvatarURL(user._id, { max_side: 256 }), badge: '/assets/icons/android-chrome-512x512.png', timestamp: +new Date() }); notif?.addEventListener("click", () => { history.push(`/friends`); }); } useEffect(() => { client.addListener("message", message); client.users.addListener("mutation", relationship); return () => { client.removeListener("message", message); client.users.removeListener("mutation", relationship); }; }, [client, guild_id, channel_id, showNotification]); useEffect(() => { function visChange() { if (document.visibilityState === "visible") { if (notifications[channel_id]) { notifications[channel_id].close(); } } } visChange(); document.addEventListener("visibilitychange", visChange); return () => document.removeEventListener("visibilitychange", visChange); }, [guild_id, channel_id]); return <>; } const NotifierComponent = connectState( Notifier, state => { return { options: state.settings.notification }; }, true ); export default function Notifications() { return ( ); }