revite/src/context/revoltjs/Notifications.tsx

318 lines
11 KiB
TypeScript
Raw Normal View History

2021-07-05 06:23:23 -04:00
import { Route, Switch, useHistory, useParams } from "react-router-dom";
import { Presence, RelationshipStatus } from "revolt-api/types/Users";
import { Message } from "revolt.js/dist/maps/Messages";
import { User } from "revolt.js/dist/maps/Users";
2021-07-05 06:23:23 -04:00
import { decodeTime } from "ulid";
2021-08-05 09:47:00 -04:00
import { useCallback, useContext, useEffect } from "preact/hooks";
2021-07-05 06:23:23 -04:00
import { useTranslation } from "../../lib/i18n";
import { connectState } from "../../redux/connector";
2021-07-05 06:23:23 -04:00
import {
2021-07-05 06:25:20 -04:00
getNotificationState,
Notifications,
shouldNotify,
2021-07-05 06:23:23 -04:00
} from "../../redux/reducers/notifications";
import { NotificationOptions } from "../../redux/reducers/settings";
2021-07-05 06:23:23 -04:00
import { SoundContext } from "../Settings";
import { AppContext } from "./RevoltClient";
interface Props {
2021-07-05 06:25:20 -04:00
options?: NotificationOptions;
notifs: Notifications;
}
const notifications: { [key: string]: Notification } = {};
2021-07-05 06:23:23 -04:00
async function createNotification(
2021-07-05 06:25:20 -04:00
title: string,
options: globalThis.NotificationOptions,
2021-07-05 06:23:23 -04:00
) {
2021-07-05 06:25:20 -04:00
try {
return new Notification(title, options);
} catch (err) {
const sw = await navigator.serviceWorker.getRegistration();
2021-07-05 06:25:20 -04:00
sw?.showNotification(title, options);
}
}
function Notifier({ options, notifs }: Props) {
2021-07-05 06:25:20 -04:00
const translate = useTranslation();
const showNotification = options?.desktopEnabled ?? false;
2021-07-05 06:23:23 -04:00
2021-07-05 06:25:20 -04:00
const client = useContext(AppContext);
const { guild: guild_id, channel: channel_id } = useParams<{
guild: string;
channel: string;
}>();
const history = useHistory();
const playSound = useContext(SoundContext);
2021-07-05 06:23:23 -04:00
2021-08-05 09:47:00 -04:00
const message = useCallback(
async (msg: Message) => {
if (msg.author_id === client.user!._id) return;
if (msg.channel_id === channel_id && document.hasFocus()) return;
if (client.user!.status?.presence === Presence.Busy) return;
if (msg.author?.relationship === RelationshipStatus.Blocked) return;
const notifState = getNotificationState(notifs, msg.channel!);
if (!shouldNotify(notifState, msg, client.user!._id)) return;
playSound("message");
if (!showNotification) return;
let title;
switch (msg.channel?.channel_type) {
case "SavedMessages":
return;
case "DirectMessage":
title = `@${msg.author?.username}`;
2021-07-05 06:25:20 -04:00
break;
2021-08-05 09:47:00 -04:00
case "Group":
if (msg.author?._id === "00000000000000000000000000") {
2021-08-05 09:47:00 -04:00
title = msg.channel.name;
} else {
title = `@${msg.author?.username} - ${msg.channel.name}`;
}
2021-07-05 06:25:20 -04:00
break;
2021-08-05 09:47:00 -04:00
case "TextChannel":
title = `@${msg.author?.username} (#${msg.channel.name}, ${msg.channel.server?.name})`;
2021-07-05 06:25:20 -04:00
break;
2021-08-05 09:47:00 -04:00
default:
title = msg.channel?._id;
2021-07-05 06:25:20 -04:00
break;
}
2021-08-05 09:47:00 -04:00
let image;
if (msg.attachments) {
const 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 = msg.author?.generateAvatarURL({ max_side: 256 });
} else {
const users = client.users;
switch (msg.content.type) {
case "user_added":
case "user_remove":
{
const user = users.get(msg.content.id);
body = translate(
`app.main.channel.system.${
msg.content.type === "user_added"
? "added_by"
: "removed_by"
}`,
{
user: user?.username,
other_user: users.get(msg.content.by)
?.username,
},
2021-07-05 06:25:20 -04:00
);
2021-08-05 09:47:00 -04:00
icon = user?.generateAvatarURL({
max_side: 256,
});
2021-07-05 06:25:20 -04:00
}
2021-08-05 09:47:00 -04:00
break;
case "user_joined":
case "user_left":
case "user_kicked":
case "user_banned":
{
const user = users.get(msg.content.id);
body = translate(
`app.main.channel.system.${msg.content.type}`,
{ user: user?.username },
);
icon = user?.generateAvatarURL({
max_side: 256,
});
}
break;
case "channel_renamed":
{
const user = users.get(msg.content.by);
body = translate(
`app.main.channel.system.channel_renamed`,
{
user: users.get(msg.content.by)?.username,
name: msg.content.name,
},
);
icon = user?.generateAvatarURL({
max_side: 256,
});
}
break;
case "channel_description_changed":
case "channel_icon_changed":
{
const user = users.get(msg.content.by);
body = translate(
`app.main.channel.system.${msg.content.type}`,
{ user: users.get(msg.content.by)?.username },
);
icon = user?.generateAvatarURL({
max_side: 256,
});
}
break;
2021-07-05 06:25:20 -04:00
}
2021-08-05 09:47:00 -04:00
}
const notif = await createNotification(title!, {
icon,
image,
body,
timestamp: decodeTime(msg._id),
tag: msg.channel?._id,
badge: "/assets/icons/android-chrome-512x512.png",
silent: true,
2021-07-05 06:25:20 -04:00
});
2021-07-05 06:23:23 -04:00
2021-08-05 09:47:00 -04:00
if (notif) {
notif.addEventListener("click", () => {
window.focus();
const id = msg.channel_id;
if (id !== channel_id) {
const channel = client.channels.get(id);
if (channel) {
if (channel.channel_type === "TextChannel") {
history.push(
`/server/${channel.server_id}/channel/${id}`,
);
} else {
history.push(`/channel/${id}`);
}
}
}
});
2021-07-05 06:23:23 -04:00
2021-08-05 09:47:00 -04:00
notifications[msg.channel_id] = notif;
notif.addEventListener(
"close",
() => delete notifications[msg.channel_id],
);
}
},
[
history,
showNotification,
translate,
channel_id,
client,
notifs,
playSound,
],
);
2021-07-05 06:23:23 -04:00
2021-08-05 09:47:00 -04:00
const relationship = useCallback(
async (user: User) => {
if (client.user?.status?.presence === Presence.Busy) return;
if (!showNotification) return;
let event;
switch (user.relationship) {
case RelationshipStatus.Incoming:
event = translate("notifications.sent_request", {
person: user.username,
});
break;
case RelationshipStatus.Friend:
event = translate("notifications.now_friends", {
person: user.username,
});
break;
default:
return;
}
const notif = await createNotification(event, {
icon: user.generateAvatarURL({ max_side: 256 }),
badge: "/assets/icons/android-chrome-512x512.png",
timestamp: +new Date(),
});
notif?.addEventListener("click", () => {
history.push(`/friends`);
});
},
[client.user?.status?.presence, history, showNotification, translate],
);
2021-07-05 06:23:23 -04:00
2021-07-05 06:25:20 -04:00
useEffect(() => {
client.addListener("message", message);
client.addListener("user/relationship", relationship);
2021-07-05 06:23:23 -04:00
2021-07-05 06:25:20 -04:00
return () => {
client.removeListener("message", message);
client.removeListener("user/relationship", relationship);
2021-07-05 06:25:20 -04:00
};
2021-08-05 09:47:00 -04:00
}, [
client,
playSound,
guild_id,
channel_id,
showNotification,
notifs,
message,
relationship,
]);
2021-07-05 06:23:23 -04:00
2021-07-05 06:25:20 -04:00
useEffect(() => {
function visChange() {
if (document.visibilityState === "visible") {
if (notifications[channel_id]) {
notifications[channel_id].close();
}
}
}
2021-07-05 06:23:23 -04:00
2021-07-05 06:25:20 -04:00
visChange();
2021-07-05 06:23:23 -04:00
2021-07-05 06:25:20 -04:00
document.addEventListener("visibilitychange", visChange);
return () =>
document.removeEventListener("visibilitychange", visChange);
}, [guild_id, channel_id]);
2021-07-05 06:23:23 -04:00
2021-07-05 06:25:20 -04:00
return null;
}
const NotifierComponent = connectState(
2021-07-05 06:25:20 -04:00
Notifier,
(state) => {
return {
options: state.settings.notification,
notifs: state.notifications,
};
},
true,
);
2021-07-05 06:23:23 -04:00
export default function NotificationsComponent() {
2021-07-05 06:25:20 -04:00
return (
<Switch>
<Route path="/server/:server/channel/:channel">
<NotifierComponent />
</Route>
<Route path="/channel/:channel">
<NotifierComponent />
</Route>
<Route path="/">
<NotifierComponent />
</Route>
</Switch>
);
}