feat: show number of new notifications

This commit is contained in:
codytseng
2025-04-12 23:04:52 +08:00
parent 68f4b1e909
commit 4a143d1814
2 changed files with 42 additions and 21 deletions

View File

@@ -2,12 +2,12 @@ import { Separator } from '@/components/ui/separator'
import { Skeleton } from '@/components/ui/skeleton' import { Skeleton } from '@/components/ui/skeleton'
import { BIG_RELAY_URLS, ExtendedKind } from '@/constants' import { BIG_RELAY_URLS, ExtendedKind } from '@/constants'
import { cn } from '@/lib/utils' import { cn } from '@/lib/utils'
import { usePrimaryPage } from '@/PageManager'
import { useDeepBrowsing } from '@/providers/DeepBrowsingProvider' import { useDeepBrowsing } from '@/providers/DeepBrowsingProvider'
import { useNostr } from '@/providers/NostrProvider' import { useNostr } from '@/providers/NostrProvider'
import { useNoteStats } from '@/providers/NoteStatsProvider' import { useNoteStats } from '@/providers/NoteStatsProvider'
import { useNotification } from '@/providers/NotificationProvider' import { useNotification } from '@/providers/NotificationProvider'
import client from '@/services/client.service' import client from '@/services/client.service'
import storage from '@/services/local-storage.service'
import { TNotificationType } from '@/types' import { TNotificationType } from '@/types'
import dayjs from 'dayjs' import dayjs from 'dayjs'
import { Event, kinds } from 'nostr-tools' import { Event, kinds } from 'nostr-tools'
@@ -21,8 +21,9 @@ const SHOW_COUNT = 30
const NotificationList = forwardRef((_, ref) => { const NotificationList = forwardRef((_, ref) => {
const { t } = useTranslation() const { t } = useTranslation()
const { current } = usePrimaryPage()
const { pubkey } = useNostr() const { pubkey } = useNostr()
const { clearNewNotifications } = useNotification() const { clearNewNotifications, getNotificationsSeenAt } = useNotification()
const { updateNoteStatsByEvents } = useNoteStats() const { updateNoteStatsByEvents } = useNoteStats()
const [notificationType, setNotificationType] = useState<TNotificationType>('all') const [notificationType, setNotificationType] = useState<TNotificationType>('all')
const [lastReadTime, setLastReadTime] = useState(0) const [lastReadTime, setLastReadTime] = useState(0)
@@ -59,6 +60,8 @@ const NotificationList = forwardRef((_, ref) => {
) )
useEffect(() => { useEffect(() => {
if (current !== 'notifications') return
if (!pubkey) { if (!pubkey) {
setUntil(undefined) setUntil(undefined)
return return
@@ -68,7 +71,7 @@ const NotificationList = forwardRef((_, ref) => {
setLoading(true) setLoading(true)
setNotifications([]) setNotifications([])
setShowCount(SHOW_COUNT) setShowCount(SHOW_COUNT)
setLastReadTime(storage.getLastReadNotificationTime(pubkey)) setLastReadTime(getNotificationsSeenAt())
clearNewNotifications() clearNewNotifications()
const relayList = await client.fetchRelayList(pubkey) const relayList = await client.fetchRelayList(pubkey)
@@ -113,7 +116,7 @@ const NotificationList = forwardRef((_, ref) => {
return () => { return () => {
promise.then((closer) => closer?.()) promise.then((closer) => closer?.())
} }
}, [pubkey, refreshCount, filterKinds]) }, [pubkey, refreshCount, filterKinds, current])
useEffect(() => { useEffect(() => {
const visibleNotifications = notifications.slice(0, showCount) const visibleNotifications = notifications.slice(0, showCount)

View File

@@ -2,12 +2,13 @@ import { BIG_RELAY_URLS, ExtendedKind } from '@/constants'
import client from '@/services/client.service' import client from '@/services/client.service'
import { kinds } from 'nostr-tools' import { kinds } from 'nostr-tools'
import { SubCloser } from 'nostr-tools/abstract-pool' import { SubCloser } from 'nostr-tools/abstract-pool'
import { createContext, useContext, useEffect, useState } from 'react' import { createContext, useContext, useEffect, useRef, useState } from 'react'
import { useMuteList } from './MuteListProvider' import { useMuteList } from './MuteListProvider'
import { useNostr } from './NostrProvider' import { useNostr } from './NostrProvider'
type TNotificationContext = { type TNotificationContext = {
hasNewNotification: boolean hasNewNotification: boolean
getNotificationsSeenAt: () => number
clearNewNotifications: () => Promise<void> clearNewNotifications: () => Promise<void>
} }
@@ -24,16 +25,16 @@ export const useNotification = () => {
export function NotificationProvider({ children }: { children: React.ReactNode }) { export function NotificationProvider({ children }: { children: React.ReactNode }) {
const { pubkey, notificationsSeenAt, updateNotificationsSeenAt } = useNostr() const { pubkey, notificationsSeenAt, updateNotificationsSeenAt } = useNostr()
const { mutePubkeys } = useMuteList() const { mutePubkeys } = useMuteList()
const [hasNewNotification, setHasNewNotification] = useState(false) const [newNotificationIds, setNewNotificationIds] = useState(new Set<string>())
const subCloserRef = useRef<SubCloser | null>(null)
useEffect(() => { useEffect(() => {
if (!pubkey || notificationsSeenAt < 0) return if (!pubkey || notificationsSeenAt < 0) return
setHasNewNotification(false) setNewNotificationIds(new Set())
// Track if component is mounted // Track if component is mounted
const isMountedRef = { current: true } const isMountedRef = { current: true }
let currentSubCloser: SubCloser | null = null
const subscribe = async () => { const subscribe = async () => {
if (!isMountedRef.current) return null if (!isMountedRef.current) return null
@@ -54,15 +55,14 @@ export function NotificationProvider({ children }: { children: React.ReactNode }
], ],
'#p': [pubkey], '#p': [pubkey],
since: notificationsSeenAt, since: notificationsSeenAt,
limit: 10 limit: 20
} }
], ],
{ {
onevent: (evt) => { onevent: (evt) => {
// Only show notification if not from self and not muted // Only show notification if not from self and not muted
if (evt.pubkey !== pubkey && !mutePubkeys.includes(evt.pubkey)) { if (evt.pubkey !== pubkey && !mutePubkeys.includes(evt.pubkey)) {
setHasNewNotification(true) setNewNotificationIds((prev) => new Set([...prev, evt.id]))
subCloser.close()
} }
}, },
onclose: (reasons) => { onclose: (reasons) => {
@@ -71,7 +71,7 @@ export function NotificationProvider({ children }: { children: React.ReactNode }
} }
// Only reconnect if still mounted and not a manual close // Only reconnect if still mounted and not a manual close
if (isMountedRef.current && currentSubCloser) { if (isMountedRef.current && subCloserRef.current) {
setTimeout(() => { setTimeout(() => {
if (isMountedRef.current) { if (isMountedRef.current) {
subscribe() subscribe()
@@ -82,7 +82,7 @@ export function NotificationProvider({ children }: { children: React.ReactNode }
} }
) )
currentSubCloser = subCloser subCloserRef.current = subCloser
return subCloser return subCloser
} catch (error) { } catch (error) {
console.error('Subscription error:', error) console.error('Subscription error:', error)
@@ -105,30 +105,48 @@ export function NotificationProvider({ children }: { children: React.ReactNode }
// Cleanup function // Cleanup function
return () => { return () => {
isMountedRef.current = false isMountedRef.current = false
if (currentSubCloser) { if (subCloserRef.current) {
currentSubCloser.close() subCloserRef.current.close()
currentSubCloser = null subCloserRef.current = null
} }
} }
}, [notificationsSeenAt, pubkey]) }, [notificationsSeenAt, pubkey])
useEffect(() => { useEffect(() => {
if (hasNewNotification) { if (newNotificationIds.size >= 10 && subCloserRef.current) {
document.title = '📥 Jumble' subCloserRef.current.close()
subCloserRef.current = null
}
}, [newNotificationIds])
useEffect(() => {
const newNotificationCount = newNotificationIds.size
if (newNotificationCount > 0) {
document.title = `(${newNotificationCount >= 10 ? '9+' : newNotificationCount}) Jumble`
} else { } else {
document.title = 'Jumble' document.title = 'Jumble'
} }
}, [hasNewNotification]) }, [newNotificationIds])
const getNotificationsSeenAt = () => {
return notificationsSeenAt
}
const clearNewNotifications = async () => { const clearNewNotifications = async () => {
if (!pubkey) return if (!pubkey) return
setHasNewNotification(false) setNewNotificationIds(new Set())
await updateNotificationsSeenAt() await updateNotificationsSeenAt()
} }
return ( return (
<NotificationContext.Provider value={{ hasNewNotification, clearNewNotifications }}> <NotificationContext.Provider
value={{
hasNewNotification: newNotificationIds.size > 0,
clearNewNotifications,
getNotificationsSeenAt
}}
>
{children} {children}
</NotificationContext.Provider> </NotificationContext.Provider>
) )