2020-01-29 20:37:25 +01:00
|
|
|
<template>
|
2023-04-08 02:01:42 +02:00
|
|
|
<MkPagination ref="pagingComponent" :pagination="pagination">
|
|
|
|
<template #empty>
|
|
|
|
<div class="_fullinfo">
|
|
|
|
<img
|
2023-09-26 05:57:14 +02:00
|
|
|
src="/static-assets/badges/info.webp"
|
2023-04-08 02:01:42 +02:00
|
|
|
class="_ghost"
|
|
|
|
alt="Info"
|
|
|
|
/>
|
|
|
|
<div>{{ i18n.ts.noNotifications }}</div>
|
|
|
|
</div>
|
|
|
|
</template>
|
2022-01-09 13:35:35 +01:00
|
|
|
|
2023-04-08 02:01:42 +02:00
|
|
|
<template #default="{ items: notifications }">
|
|
|
|
<XList
|
|
|
|
v-slot="{ item: notification }"
|
|
|
|
class="elsfgstc"
|
|
|
|
:items="notifications"
|
|
|
|
:no-gap="true"
|
|
|
|
>
|
|
|
|
<XNote
|
|
|
|
v-if="
|
|
|
|
['reply', 'quote', 'mention'].includes(
|
2023-07-06 03:28:27 +02:00
|
|
|
notification.type,
|
2023-04-08 02:01:42 +02:00
|
|
|
)
|
|
|
|
"
|
|
|
|
:key="notification.id"
|
|
|
|
:note="notification.note"
|
2023-09-02 01:27:33 +02:00
|
|
|
:collapsed-reply="
|
2023-06-23 01:46:18 +02:00
|
|
|
notification.type === 'reply' ||
|
|
|
|
(notification.type === 'mention' &&
|
|
|
|
notification.note.replyId != null)
|
|
|
|
"
|
2023-04-08 02:01:42 +02:00
|
|
|
/>
|
|
|
|
<XNotification
|
|
|
|
v-else
|
|
|
|
:key="notification.id"
|
|
|
|
:notification="notification"
|
|
|
|
:with-time="true"
|
|
|
|
:full="true"
|
|
|
|
class="_panel notification"
|
|
|
|
/>
|
|
|
|
</XList>
|
|
|
|
</template>
|
|
|
|
</MkPagination>
|
2020-01-29 20:37:25 +01:00
|
|
|
</template>
|
|
|
|
|
2022-01-09 13:35:35 +01:00
|
|
|
<script lang="ts" setup>
|
2023-09-02 01:27:33 +02:00
|
|
|
import { computed, onMounted, onUnmounted, ref } from "vue";
|
|
|
|
import type { notificationTypes } from "firefish-js";
|
|
|
|
import type { Paging } from "@/components/MkPagination.vue";
|
|
|
|
import MkPagination from "@/components/MkPagination.vue";
|
2023-04-08 02:01:42 +02:00
|
|
|
import XNotification from "@/components/MkNotification.vue";
|
|
|
|
import XList from "@/components/MkDateSeparatedList.vue";
|
|
|
|
import XNote from "@/components/MkNote.vue";
|
2024-02-29 14:21:19 +01:00
|
|
|
import { useStream } from "@/stream";
|
2024-03-07 03:06:45 +01:00
|
|
|
import { me } from "@/me";
|
2023-04-08 02:01:42 +02:00
|
|
|
import { i18n } from "@/i18n";
|
2022-01-09 13:35:35 +01:00
|
|
|
|
|
|
|
const props = defineProps<{
|
2023-04-08 02:01:42 +02:00
|
|
|
includeTypes?: (typeof notificationTypes)[number][];
|
2022-01-09 13:35:35 +01:00
|
|
|
unreadOnly?: boolean;
|
|
|
|
}>();
|
|
|
|
|
2024-02-29 14:21:19 +01:00
|
|
|
const stream = useStream();
|
|
|
|
|
2022-01-09 13:35:35 +01:00
|
|
|
const pagingComponent = ref<InstanceType<typeof MkPagination>>();
|
|
|
|
|
|
|
|
const pagination: Paging = {
|
2023-04-08 02:01:42 +02:00
|
|
|
endpoint: "i/notifications" as const,
|
2022-01-09 13:35:35 +01:00
|
|
|
limit: 10,
|
|
|
|
params: computed(() => ({
|
2022-03-06 08:06:27 +01:00
|
|
|
includeTypes: props.includeTypes ?? undefined,
|
2024-03-28 17:26:52 +01:00
|
|
|
excludeTypes: props.includeTypes ? undefined : me.mutingNotificationTypes,
|
2022-01-09 13:35:35 +01:00
|
|
|
unreadOnly: props.unreadOnly,
|
|
|
|
})),
|
|
|
|
};
|
|
|
|
|
|
|
|
const onNotification = (notification) => {
|
2023-04-08 02:01:42 +02:00
|
|
|
const isMuted = props.includeTypes
|
|
|
|
? !props.includeTypes.includes(notification.type)
|
2024-03-07 03:06:45 +01:00
|
|
|
: me.mutingNotificationTypes.includes(notification.type);
|
2023-04-08 02:01:42 +02:00
|
|
|
if (isMuted || document.visibilityState === "visible") {
|
|
|
|
stream.send("readNotification", {
|
2022-06-05 05:26:36 +02:00
|
|
|
id: notification.id,
|
2022-01-09 13:35:35 +01:00
|
|
|
});
|
|
|
|
}
|
2020-07-29 16:03:08 +02:00
|
|
|
|
2022-01-09 13:35:35 +01:00
|
|
|
if (!isMuted) {
|
2024-02-29 14:21:19 +01:00
|
|
|
pagingComponent.value?.prepend({
|
2022-01-09 13:35:35 +01:00
|
|
|
...notification,
|
2023-04-08 02:01:42 +02:00
|
|
|
isRead: document.visibilityState === "visible",
|
2022-01-09 13:35:35 +01:00
|
|
|
});
|
2020-01-29 20:37:25 +01:00
|
|
|
}
|
2022-01-09 13:35:35 +01:00
|
|
|
};
|
|
|
|
|
2022-06-25 20:12:58 +02:00
|
|
|
let connection;
|
|
|
|
|
2022-01-09 13:35:35 +01:00
|
|
|
onMounted(() => {
|
2023-04-08 02:01:42 +02:00
|
|
|
connection = stream.useChannel("main");
|
|
|
|
connection.on("notification", onNotification);
|
|
|
|
connection.on("readAllNotifications", () => {
|
2022-04-30 14:52:07 +02:00
|
|
|
if (pagingComponent.value) {
|
|
|
|
for (const item of pagingComponent.value.queue) {
|
|
|
|
item.isRead = true;
|
|
|
|
}
|
|
|
|
for (const item of pagingComponent.value.items) {
|
|
|
|
item.isRead = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
2023-04-08 02:01:42 +02:00
|
|
|
connection.on("readNotifications", (notificationIds) => {
|
2022-04-30 14:52:07 +02:00
|
|
|
if (pagingComponent.value) {
|
|
|
|
for (let i = 0; i < pagingComponent.value.queue.length; i++) {
|
2024-03-28 17:26:52 +01:00
|
|
|
if (notificationIds.includes(pagingComponent.value.queue[i].id)) {
|
2022-04-30 14:52:07 +02:00
|
|
|
pagingComponent.value.queue[i].isRead = true;
|
|
|
|
}
|
|
|
|
}
|
2024-03-28 17:26:52 +01:00
|
|
|
for (let i = 0; i < (pagingComponent.value.items || []).length; i++) {
|
|
|
|
if (notificationIds.includes(pagingComponent.value.items[i].id)) {
|
2022-04-30 14:52:07 +02:00
|
|
|
pagingComponent.value.items[i].isRead = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
2022-06-25 20:12:58 +02:00
|
|
|
});
|
2022-04-30 14:52:07 +02:00
|
|
|
|
2022-06-25 20:12:58 +02:00
|
|
|
onUnmounted(() => {
|
|
|
|
if (connection) connection.dispose();
|
2020-01-29 20:37:25 +01:00
|
|
|
});
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<style lang="scss" scoped>
|
2021-08-21 10:40:15 +02:00
|
|
|
.elsfgstc {
|
|
|
|
background: var(--panel);
|
2023-03-26 05:33:44 +02:00
|
|
|
border-radius: var(--radius);
|
2021-08-21 10:40:15 +02:00
|
|
|
}
|
2020-01-29 20:37:25 +01:00
|
|
|
</style>
|