175 lines
4.5 KiB
TypeScript
175 lines
4.5 KiB
TypeScript
import {derived} from "svelte/store"
|
|
import {Badge} from "@capawesome/capacitor-badge"
|
|
import {synced, throttled, withGetter} from "@welshman/store"
|
|
import {pubkey, tracker, repository, relaysByUrl} from "@welshman/app"
|
|
import {assoc, prop, spec, first, identity, now} from "@welshman/lib"
|
|
import type {TrustedEvent} from "@welshman/util"
|
|
import {deriveEventsByIdByUrl} from "@welshman/store"
|
|
import {sortEventsDesc} from "@welshman/util"
|
|
import {makeSpacePath, makeRoomPath, makeSpaceChatPath, makeChatPath} from "@app/util/routes"
|
|
import {
|
|
MESSAGE_KINDS,
|
|
notificationSettings,
|
|
chatsById,
|
|
userGroupList,
|
|
getSpaceUrlsFromGroupList,
|
|
getSpaceRoomsFromGroupList,
|
|
makeCommentFilter,
|
|
hasNip29,
|
|
} from "@app/core/state"
|
|
import {kv} from "@app/core/storage"
|
|
import {page} from "$app/stores"
|
|
export {Push} from "@app/util/push"
|
|
|
|
// Checked state
|
|
|
|
export const checked = withGetter(
|
|
synced<Record<string, number>>({
|
|
key: "checked",
|
|
defaultValue: {},
|
|
storage: kv,
|
|
}),
|
|
)
|
|
|
|
export const getChecked = (key: string) => checked.get()[key]
|
|
|
|
export const deriveChecked = (key: string) => derived(checked, prop<number>(key))
|
|
|
|
export const setChecked = (key: string) => checked.update(assoc(key, now()))
|
|
|
|
export const syncChecked = () => {
|
|
let prev = ""
|
|
|
|
const getPaths = (path: string) =>
|
|
path
|
|
.split("/")
|
|
.map((_, i, segments) => segments.slice(0, i + 1).join("/"))
|
|
.slice(1)
|
|
|
|
// Set checked when we enter and when we leave a given page
|
|
return page.subscribe($page => {
|
|
checked.update($checked => {
|
|
for (const path of getPaths($page.url.pathname)) {
|
|
$checked[path] = now()
|
|
}
|
|
|
|
for (const path of getPaths(prev)) {
|
|
$checked[path] = now()
|
|
}
|
|
|
|
return $checked
|
|
})
|
|
|
|
prev = $page.url.pathname
|
|
})
|
|
}
|
|
|
|
// Derived notifications state
|
|
|
|
export const allNotifications = derived(
|
|
throttled(
|
|
2000,
|
|
derived(
|
|
[
|
|
pubkey,
|
|
checked,
|
|
chatsById,
|
|
relaysByUrl,
|
|
userGroupList,
|
|
deriveEventsByIdByUrl({
|
|
tracker,
|
|
repository,
|
|
filters: [{kinds: MESSAGE_KINDS}, makeCommentFilter(MESSAGE_KINDS)],
|
|
}),
|
|
],
|
|
identity,
|
|
),
|
|
),
|
|
([$pubkey, $checked, $chatsById, $relaysByUrl, $userGroupList, eventsByIdByUrl]) => {
|
|
const hasNotification = (path: string, latestEvent?: TrustedEvent) => {
|
|
if (!latestEvent || latestEvent.pubkey === $pubkey) {
|
|
return false
|
|
}
|
|
|
|
for (const [entryPath, ts] of Object.entries($checked)) {
|
|
const isMatch =
|
|
entryPath === "*" ||
|
|
entryPath.startsWith(path) ||
|
|
(entryPath === "/chat/*" && path.startsWith("/chat/"))
|
|
|
|
if (isMatch && ts > latestEvent.created_at) {
|
|
return false
|
|
}
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
const paths = new Set<string>()
|
|
|
|
for (const {pubkeys, messages} of $chatsById.values()) {
|
|
const chatPath = makeChatPath(pubkeys)
|
|
|
|
if (hasNotification(chatPath, messages[0])) {
|
|
paths.add("/chat")
|
|
paths.add(chatPath)
|
|
}
|
|
}
|
|
|
|
for (const url of getSpaceUrlsFromGroupList($userGroupList)) {
|
|
const spacePath = makeSpacePath(url)
|
|
const events = sortEventsDesc((eventsByIdByUrl.get(url) || new Map()).values())
|
|
|
|
if (hasNip29($relaysByUrl.get(url))) {
|
|
for (const h of getSpaceRoomsFromGroupList(url, $userGroupList)) {
|
|
const roomPath = makeRoomPath(url, h)
|
|
const latestEvent = events.find(e => e.tags.some(spec(["h", h])))
|
|
|
|
if (hasNotification(roomPath, latestEvent)) {
|
|
paths.add(spacePath)
|
|
paths.add(roomPath)
|
|
}
|
|
}
|
|
} else {
|
|
const messagesPath = makeSpaceChatPath(url)
|
|
|
|
if (hasNotification(messagesPath, first(events))) {
|
|
paths.add(spacePath)
|
|
paths.add(messagesPath)
|
|
}
|
|
}
|
|
}
|
|
|
|
return paths
|
|
},
|
|
)
|
|
|
|
export const notifications = derived([page, allNotifications], ([$page, $allNotifications]) => {
|
|
return new Set([...$allNotifications].filter(p => !$page.url.pathname.startsWith(p)))
|
|
})
|
|
|
|
// Badges
|
|
|
|
export const syncBadges = () =>
|
|
derived([notifications, notificationSettings], identity).subscribe(
|
|
async ([$notifications, $notificationSettings]) => {
|
|
if ($notificationSettings.badge) {
|
|
try {
|
|
await Badge.set({count: $notifications.size})
|
|
} catch (err) {
|
|
// pass - firefox doesn't support badges
|
|
}
|
|
} else {
|
|
await clearBadges()
|
|
}
|
|
},
|
|
)
|
|
|
|
export const clearBadges = async () => {
|
|
try {
|
|
await Badge.clear()
|
|
} catch (e) {
|
|
// pass - firefox doesn't support this
|
|
}
|
|
}
|