forked from coracle/flotilla
c4f2f55617
Co-authored-by: Jon Staab <shtaab@gmail.com>
643 lines
17 KiB
TypeScript
643 lines
17 KiB
TypeScript
import type {Unsubscriber, Readable, Subscriber} from "svelte/store"
|
|
import {derived, get} from "svelte/store"
|
|
import {Capacitor} from "@capacitor/core"
|
|
import {Badge} from "@capawesome/capacitor-badge"
|
|
import {PushNotifications} from "@capacitor/push-notifications"
|
|
import type {ActionPerformed, RegistrationError, Token} from "@capacitor/push-notifications"
|
|
import {synced, throttled, withGetter} from "@welshman/store"
|
|
import {load, LOCAL_RELAY_URL} from "@welshman/net"
|
|
import {
|
|
pubkey,
|
|
tracker,
|
|
repository,
|
|
publishThunk,
|
|
loadRelay,
|
|
waitForThunkError,
|
|
userMessagingRelayList,
|
|
} from "@welshman/app"
|
|
import {
|
|
on,
|
|
call,
|
|
assoc,
|
|
poll,
|
|
prop,
|
|
hash,
|
|
spec,
|
|
first,
|
|
identity,
|
|
remove,
|
|
now,
|
|
maybe,
|
|
throttle,
|
|
} from "@welshman/lib"
|
|
import type {TrustedEvent, Filter} from "@welshman/util"
|
|
import {deriveEventsByIdByUrl} from "@welshman/store"
|
|
import {
|
|
DELETE,
|
|
getTagValue,
|
|
getPubkeyTagValues,
|
|
getRelaysFromList,
|
|
matchFilter,
|
|
matchFilters,
|
|
getIdFilters,
|
|
sortEventsDesc,
|
|
makeEvent,
|
|
Address,
|
|
} from "@welshman/util"
|
|
import {buildUrl} from "@lib/util"
|
|
import {makeSpacePath, makeChatPath, getEventPath, goToEvent} from "@app/util/routes"
|
|
import {
|
|
DM_KINDS,
|
|
CONTENT_KINDS,
|
|
MESSAGE_KINDS,
|
|
PUSH_BRIDGE,
|
|
PUSH_SERVER,
|
|
notificationSettings,
|
|
notificationState,
|
|
chatsById,
|
|
userSettingsValues,
|
|
userGroupList,
|
|
getSpaceUrlsFromGroupList,
|
|
makeCommentFilter,
|
|
userSpaceUrls,
|
|
shouldNotify,
|
|
device,
|
|
} from "@app/core/state"
|
|
import {kv} from "@app/core/storage"
|
|
import {goto} from "$app/navigation"
|
|
import {page} from "$app/stores"
|
|
|
|
// Temporarily copied from welshman
|
|
|
|
type Stores = Readable<any> | [Readable<any>, ...Array<Readable<any>>] | Array<Readable<any>>
|
|
|
|
const merged = <S extends Stores>(stores: S) => derived(stores, identity)
|
|
|
|
// 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,
|
|
userGroupList,
|
|
deriveEventsByIdByUrl({
|
|
tracker,
|
|
repository,
|
|
filters: [{kinds: MESSAGE_KINDS}, makeCommentFilter(MESSAGE_KINDS)],
|
|
}),
|
|
],
|
|
identity,
|
|
),
|
|
),
|
|
([$pubkey, $checked, $chatsById, $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 eventsById = eventsByIdByUrl.get(url) || new Map()
|
|
const latestEvent = first(sortEventsDesc(eventsById.values()))
|
|
|
|
if (hasNotification(spacePath, latestEvent)) {
|
|
paths.add(spacePath)
|
|
}
|
|
}
|
|
|
|
return paths
|
|
},
|
|
)
|
|
|
|
export const notifications = derived([page, allNotifications], ([$page, $allNotifications]) => {
|
|
return new Set(remove($page.url.pathname, [...$allNotifications]))
|
|
})
|
|
|
|
export const onNotification = call(() => {
|
|
const allFilters = [{kinds: [...MESSAGE_KINDS, ...DM_KINDS]}, makeCommentFilter(MESSAGE_KINDS)]
|
|
const filters = allFilters.map(assoc("since", now()))
|
|
const subscribers: Subscriber<TrustedEvent>[] = []
|
|
|
|
let unsubscribe: Unsubscriber | undefined
|
|
|
|
return (f: (event: TrustedEvent) => void) => {
|
|
subscribers.push(f)
|
|
|
|
if (!unsubscribe) {
|
|
unsubscribe = on(repository, "update", ({added}) => {
|
|
const $pubkey = pubkey.get()
|
|
|
|
for (const event of added) {
|
|
if (event.pubkey == $pubkey) {
|
|
continue
|
|
}
|
|
|
|
const h = getTagValue("h", event.tags)
|
|
|
|
if (Array.from(tracker.getRelays(event.id)).every(url => !shouldNotify(url, h))) {
|
|
continue
|
|
}
|
|
|
|
if (matchFilters(filters, event)) {
|
|
for (const f of subscribers) {
|
|
f(event)
|
|
}
|
|
}
|
|
}
|
|
})
|
|
}
|
|
|
|
return () => {
|
|
subscribers.splice(subscribers.indexOf(f), 1)
|
|
|
|
if (subscribers.length === 0) {
|
|
unsubscribe?.()
|
|
unsubscribe = undefined
|
|
}
|
|
}
|
|
}
|
|
})
|
|
|
|
// 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
|
|
}
|
|
}
|
|
|
|
// Push notifications
|
|
|
|
interface IPushAdapter {
|
|
request: (prompt?: boolean) => Promise<string>
|
|
disable: () => Promise<void>
|
|
enable: () => Promise<void>
|
|
}
|
|
|
|
class CapacitorNotifications implements IPushAdapter {
|
|
_controller = maybe<AbortController>()
|
|
|
|
async request(prompt = true) {
|
|
let status = await PushNotifications.checkPermissions()
|
|
|
|
if (prompt && status.receive === "prompt") {
|
|
status = await PushNotifications.requestPermissions()
|
|
}
|
|
|
|
if (status.receive !== "granted") {
|
|
return status.receive
|
|
}
|
|
|
|
let {token} = notificationState.get()
|
|
|
|
if (!token) {
|
|
const listeners = [
|
|
PushNotifications.addListener("registration", ({value}: Token) => {
|
|
token = value
|
|
}),
|
|
PushNotifications.addListener("registrationError", (error: RegistrationError) => {
|
|
console.error(error)
|
|
}),
|
|
]
|
|
|
|
await Promise.all([
|
|
PushNotifications.register(),
|
|
poll({
|
|
condition: () => Boolean(token),
|
|
signal: AbortSignal.timeout(5000),
|
|
}),
|
|
])
|
|
|
|
listeners.forEach(p => p.then(listener => listener.remove()))
|
|
notificationState.update(assoc("token", token))
|
|
}
|
|
|
|
return token ? "granted" : "denied"
|
|
}
|
|
|
|
async _syncServer(signal: AbortSignal) {
|
|
const {token, subscription} = notificationState.get()
|
|
|
|
if (!token) {
|
|
throw new Error("Attempted to sync push server without a token")
|
|
}
|
|
|
|
if (!subscription) {
|
|
try {
|
|
const channel = Capacitor.getPlatform() === "ios" ? "apns" : "fcm"
|
|
const url = buildUrl(PUSH_SERVER, "subscription", channel)
|
|
const res = await fetch(url, {
|
|
signal,
|
|
method: "POST",
|
|
body: JSON.stringify({token}),
|
|
headers: {
|
|
Accept: "application/json",
|
|
"Content-Type": "application/json",
|
|
},
|
|
})
|
|
|
|
if (!res.ok) {
|
|
console.warn(`Failed to register with push server (status ${res.status})`)
|
|
} else {
|
|
const json = await res.json()
|
|
|
|
if (json?.callback && json?.key) {
|
|
notificationState.update(assoc("subscription", json))
|
|
} else {
|
|
console.warn("Failed to register with push server (bad response)")
|
|
}
|
|
}
|
|
} catch (e) {
|
|
console.warn("Failed to register with push server:", e)
|
|
}
|
|
}
|
|
}
|
|
|
|
_getSubscriptionIdentifier = (relay: string, key: string) =>
|
|
String(hash(relay + key + device.get()))
|
|
|
|
_getPushUrl = async (url: string) => {
|
|
let relay = await loadRelay(url)
|
|
|
|
if (!relay?.self || !relay?.supported_nips?.map(String)?.includes("9a")) {
|
|
relay = await loadRelay(PUSH_BRIDGE)
|
|
}
|
|
|
|
if (relay?.self) {
|
|
return relay.url
|
|
}
|
|
}
|
|
|
|
_syncRelay = async (relay: string, key: string, filters: Filter[], ignore: Filter[] = []) => {
|
|
const {subscription} = notificationState.get()
|
|
|
|
if (!subscription) {
|
|
console.warn(`Failed to subscribe ${relay} to notifications: no subscription`)
|
|
return
|
|
}
|
|
|
|
const url = await this._getPushUrl(relay)
|
|
|
|
if (!url) {
|
|
console.warn(`Failed to subscribe ${relay} to notifications: unsupported`)
|
|
return
|
|
}
|
|
|
|
const identifier = this._getSubscriptionIdentifier(relay, key)
|
|
|
|
const thunk = publishThunk({
|
|
relays: [url],
|
|
event: makeEvent(30390, {
|
|
tags: [
|
|
["d", identifier],
|
|
["relay", relay],
|
|
["callback", subscription.callback],
|
|
...ignore.map(filter => ["ignore", JSON.stringify(filter)]),
|
|
...filters.map(filter => ["filter", JSON.stringify(filter)]),
|
|
],
|
|
}),
|
|
})
|
|
|
|
const error = await waitForThunkError(thunk)
|
|
|
|
if (error) {
|
|
console.warn(`Failed to subscribe ${relay} to ${key} notifications:`, error)
|
|
}
|
|
}
|
|
|
|
_unsyncRelay = async (relay: string, key: string) => {
|
|
const url = await this._getPushUrl(relay)
|
|
|
|
if (!url) {
|
|
console.warn(`Failed to unsubscribe ${relay} from notifications: unsupported`)
|
|
return
|
|
}
|
|
|
|
const relays = [url]
|
|
const identifier = this._getSubscriptionIdentifier(relay, key)
|
|
const address = new Address(30390, pubkey.get()!, identifier).toString()
|
|
const event = makeEvent(DELETE, {tags: [["a", address]]})
|
|
const error = await waitForThunkError(publishThunk({relays, event}))
|
|
|
|
if (error) {
|
|
console.warn(`Failed to unsubscribe ${relay} from notifications:`, error)
|
|
}
|
|
}
|
|
|
|
async _syncSpaceSubscription(signal: AbortSignal) {
|
|
signal.addEventListener(
|
|
"abort",
|
|
merged([userSpaceUrls, notificationSettings, userSettingsValues]).subscribe(
|
|
throttle(3000, ([$userSpaceUrls, {spaces, mentions}, {alerts}]) => {
|
|
const baseFilters = [{kinds: MESSAGE_KINDS}, makeCommentFilter(CONTENT_KINDS)]
|
|
|
|
for (const url of $userSpaceUrls) {
|
|
const {notify = true, exceptions = []} = alerts.find(spec({url})) || {}
|
|
const filters: Filter[] = []
|
|
const ignore: Filter[] = []
|
|
|
|
// Build filters based on spaces setting
|
|
if (spaces) {
|
|
if (notify) {
|
|
// notify=true: exceptions are opt-out (exclude those rooms)
|
|
if (exceptions.length > 0) {
|
|
ignore.push({"#h": exceptions})
|
|
}
|
|
// Include all other content
|
|
filters.push(...baseFilters)
|
|
} else {
|
|
// notify=false: exceptions are opt-in (only include those rooms)
|
|
if (exceptions.length > 0) {
|
|
filters.push(...baseFilters.map(f => ({...f, "#h": exceptions})))
|
|
}
|
|
}
|
|
}
|
|
|
|
// Build filters for mentions - always notify for p-tagged content
|
|
if (mentions) {
|
|
filters.push(...baseFilters.map(f => ({...f, "#p": [pubkey.get()!]})))
|
|
}
|
|
|
|
// Sync or unsync based on whether we have filters
|
|
if (filters.length > 0) {
|
|
this._syncRelay(url, "spaces", filters, ignore)
|
|
} else {
|
|
this._unsyncRelay(url, "spaces")
|
|
}
|
|
}
|
|
}),
|
|
),
|
|
)
|
|
}
|
|
|
|
async _syncMessageSubscription(signal: AbortSignal) {
|
|
signal.addEventListener(
|
|
"abort",
|
|
merged([userMessagingRelayList, notificationSettings]).subscribe(
|
|
throttle(3000, ([$userMessagingRelayList, {messages}]) => {
|
|
for (const url of getRelaysFromList($userMessagingRelayList)) {
|
|
if (messages) {
|
|
this._syncRelay(url, "messages", [{kinds: DM_KINDS, "#p": [pubkey.get()!]}])
|
|
} else {
|
|
this._unsyncRelay(url, "messages")
|
|
}
|
|
}
|
|
}),
|
|
),
|
|
)
|
|
}
|
|
|
|
async enable() {
|
|
if (!this._controller) {
|
|
this._controller = new AbortController()
|
|
|
|
PushNotifications.addListener(
|
|
"pushNotificationActionPerformed",
|
|
async (action: ActionPerformed) => {
|
|
const {relay, id} = action.notification.data
|
|
|
|
const [event] = await load({
|
|
relays: [relay, LOCAL_RELAY_URL],
|
|
filters: getIdFilters([id]),
|
|
})
|
|
|
|
if (event) {
|
|
goto(await getEventPath(event, [relay]))
|
|
} else {
|
|
goto(makeSpacePath(relay))
|
|
}
|
|
},
|
|
)
|
|
|
|
this._controller.signal.addEventListener("abort", () => {
|
|
PushNotifications.removeAllListeners()
|
|
})
|
|
|
|
try {
|
|
await this._syncServer(this._controller.signal)
|
|
await this._syncSpaceSubscription(this._controller.signal)
|
|
await this._syncMessageSubscription(this._controller.signal)
|
|
} catch (e) {
|
|
console.error(e)
|
|
}
|
|
}
|
|
}
|
|
|
|
async disable() {
|
|
this._controller?.abort()
|
|
this._controller = undefined
|
|
|
|
const {subscription} = notificationState.get()
|
|
|
|
if (subscription) {
|
|
const res = await fetch(buildUrl(PUSH_SERVER, "subscription", subscription.key), {
|
|
method: "delete",
|
|
})
|
|
|
|
if (!res.ok) {
|
|
console.warn("Failed to delete push subscription")
|
|
}
|
|
}
|
|
|
|
notificationState.set({})
|
|
|
|
await Promise.all(get(userSpaceUrls).map(url => this._unsyncRelay(url, "spaces")))
|
|
|
|
await Promise.all(
|
|
getRelaysFromList(get(userMessagingRelayList)).map(url => this._unsyncRelay(url, "messages")),
|
|
)
|
|
}
|
|
}
|
|
|
|
class WebNotifications implements IPushAdapter {
|
|
_unsubscriber = maybe<Unsubscriber>()
|
|
|
|
async request(prompt = true) {
|
|
if (prompt && Notification?.permission === "default") {
|
|
await Notification.requestPermission()
|
|
}
|
|
|
|
return Notification?.permission || "denied"
|
|
}
|
|
|
|
_notify(event: TrustedEvent, title: string, body: string) {
|
|
console.log("notify:", event)
|
|
|
|
const notification = new Notification(title, {
|
|
body,
|
|
tag: event.id,
|
|
icon: "/icon.png",
|
|
badge: "/icon.png",
|
|
})
|
|
|
|
notification.onclick = () => {
|
|
window.focus()
|
|
goToEvent(event)
|
|
notification.close()
|
|
}
|
|
|
|
const onVisibilityChange = () => {
|
|
if (document.visibilityState === "visible") {
|
|
notification.close()
|
|
document.removeEventListener("visibilitychange", onVisibilityChange)
|
|
}
|
|
}
|
|
|
|
document.addEventListener("visibilitychange", onVisibilityChange)
|
|
}
|
|
|
|
async enable() {
|
|
if (!this._unsubscriber) {
|
|
this._unsubscriber = onNotification(event => {
|
|
const {push, messages, mentions, spaces} = notificationSettings.get()
|
|
|
|
if (push && document.hidden && Notification?.permission === "granted") {
|
|
if (messages && matchFilter({kinds: DM_KINDS}, event)) {
|
|
this._notify(event, "New direct message", "Someone sent you a direct message.")
|
|
} else if (
|
|
mentions &&
|
|
event.pubkey !== pubkey.get() &&
|
|
getPubkeyTagValues(event.tags).includes(pubkey.get()!)
|
|
) {
|
|
this._notify(event, "Someone mentioned you", "Someone tagged you in a message.")
|
|
} else if (spaces) {
|
|
this._notify(event, "New activity", "Someone posted a new message.")
|
|
}
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
async disable() {
|
|
this._unsubscriber?.()
|
|
this._unsubscriber = undefined
|
|
}
|
|
}
|
|
|
|
export class Push {
|
|
static _adapter: IPushAdapter | undefined
|
|
|
|
static _getAdapter() {
|
|
if (!Push._adapter) {
|
|
if (Capacitor.isNativePlatform()) {
|
|
Push._adapter = new CapacitorNotifications()
|
|
} else {
|
|
Push._adapter = new WebNotifications()
|
|
}
|
|
}
|
|
|
|
return Push._adapter
|
|
}
|
|
|
|
static request() {
|
|
return Push._getAdapter().request()
|
|
}
|
|
|
|
static disable() {
|
|
return Push._getAdapter().disable()
|
|
}
|
|
|
|
static enable() {
|
|
return Push._getAdapter().enable()
|
|
}
|
|
|
|
static sync() {
|
|
return notificationSettings.subscribe(({push}) => {
|
|
if (push) {
|
|
Push.enable()
|
|
} else {
|
|
Push.disable()
|
|
}
|
|
})
|
|
}
|
|
}
|