You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
245 lines
9.1 KiB
245 lines
9.1 KiB
import storage from '@/services/local-storage.service' |
|
import { LIVE_ACTIVITIES_SLIDE_INTERVAL_MS } from '@/lib/live-activities' |
|
import { toNote } from '@/lib/link' |
|
import { cn } from '@/lib/utils' |
|
import { useSmartNoteNavigation } from '@/PageManager' |
|
import { useLiveActivitiesOptional } from '@/providers/useLiveActivities' |
|
import { useUserPreferencesOptional } from '@/providers/UserPreferencesProvider' |
|
import { ExternalLink } from 'lucide-react' |
|
import { useCallback, useEffect, useId, useLayoutEffect, useRef, useState } from 'react' |
|
import { useTranslation } from 'react-i18next' |
|
|
|
type TPlacement = 'sidebar' | 'mobile' |
|
|
|
const SWIPE_MIN_PX = 44 |
|
const SWIPE_DOMINANCE_RATIO = 1.2 |
|
const SWIPE_NOTE_OPEN_SUPPRESS_MS = 400 |
|
|
|
export default function LiveActivitiesStrip({ placement }: { placement: TPlacement }) { |
|
const { t } = useTranslation() |
|
const { navigateToNote } = useSmartNoteNavigation() |
|
const userPrefs = useUserPreferencesOptional() |
|
const showLiveActivitiesBanner = |
|
userPrefs?.showLiveActivitiesBanner ?? storage.getShowLiveActivitiesBanner() |
|
const live = useLiveActivitiesOptional() |
|
const items = live?.items ?? [] |
|
|
|
const [reduceMotion, setReduceMotion] = useState(false) |
|
useEffect(() => { |
|
const mq = window.matchMedia('(prefers-reduced-motion: reduce)') |
|
const apply = () => setReduceMotion(mq.matches) |
|
apply() |
|
mq.addEventListener('change', apply) |
|
return () => mq.removeEventListener('change', apply) |
|
}, []) |
|
|
|
const [slide, setSlide] = useState(0) |
|
|
|
useEffect(() => { |
|
setSlide(0) |
|
}, [items]) |
|
|
|
useEffect(() => { |
|
if (items.length <= 1 || reduceMotion) return |
|
const id = window.setInterval(() => { |
|
setSlide((s) => (s + 1) % items.length) |
|
}, LIVE_ACTIVITIES_SLIDE_INTERVAL_MS) |
|
return () => window.clearInterval(id) |
|
}, [items.length, reduceMotion]) |
|
|
|
useLayoutEffect(() => { |
|
if (items.length === 0) return |
|
setSlide((s) => Math.min(s, items.length - 1)) |
|
}, [items.length]) |
|
|
|
const swipeHintId = useId() |
|
const swipeGrabRef = useRef<{ x: number; y: number; pointerId: number } | null>(null) |
|
const swipeNavAtRef = useRef(0) |
|
|
|
const releasePointerIfCaptured = useCallback((el: HTMLDivElement, pointerId: number) => { |
|
if (typeof el.hasPointerCapture === 'function' && el.hasPointerCapture(pointerId)) { |
|
el.releasePointerCapture(pointerId) |
|
} |
|
}, []) |
|
|
|
const onSwipePointerDown = useCallback( |
|
(e: React.PointerEvent<HTMLDivElement>) => { |
|
if (placement !== 'mobile' || items.length <= 1) return |
|
swipeGrabRef.current = { x: e.clientX, y: e.clientY, pointerId: e.pointerId } |
|
e.currentTarget.setPointerCapture(e.pointerId) |
|
}, |
|
[placement, items.length] |
|
) |
|
|
|
const onSwipePointerUp = useCallback( |
|
(e: React.PointerEvent<HTMLDivElement>) => { |
|
if (placement !== 'mobile' || items.length <= 1) return |
|
const grab = swipeGrabRef.current |
|
swipeGrabRef.current = null |
|
releasePointerIfCaptured(e.currentTarget, e.pointerId) |
|
if (!grab || grab.pointerId !== e.pointerId) return |
|
const dx = e.clientX - grab.x |
|
const dy = e.clientY - grab.y |
|
const ax = Math.abs(dx) |
|
const ay = Math.abs(dy) |
|
if (ax < SWIPE_MIN_PX || ax < ay * SWIPE_DOMINANCE_RATIO) return |
|
swipeNavAtRef.current = Date.now() |
|
if (dx < 0) { |
|
setSlide((s) => (s + 1) % items.length) |
|
} else { |
|
setSlide((s) => (s - 1 + items.length) % items.length) |
|
} |
|
}, |
|
[placement, items.length, releasePointerIfCaptured] |
|
) |
|
|
|
const onSwipePointerCancel = useCallback( |
|
(e: React.PointerEvent<HTMLDivElement>) => { |
|
swipeGrabRef.current = null |
|
releasePointerIfCaptured(e.currentTarget, e.pointerId) |
|
}, |
|
[releasePointerIfCaptured] |
|
) |
|
|
|
// `items` can shrink without a new array identity; `slide` may then be out of range until effects run. |
|
const displayIndex = items.length === 0 ? 0 : Math.min(slide, items.length - 1) |
|
const itemAtSlide = items[displayIndex] |
|
|
|
const openLiveNote = useCallback(() => { |
|
if (Date.now() - swipeNavAtRef.current < SWIPE_NOTE_OPEN_SUPPRESS_MS) return |
|
const ev = itemAtSlide?.event |
|
if (!ev) return |
|
// Same bech32 as {@link getNoteBech32Id} + pass event so {@link navigationEventStore} / cache match the URL. |
|
navigateToNote(toNote(ev), ev) |
|
}, [navigateToNote, itemAtSlide]) |
|
|
|
if (!showLiveActivitiesBanner || items.length === 0) { |
|
return null |
|
} |
|
|
|
const current = items[displayIndex] |
|
if (!current) { |
|
return null |
|
} |
|
|
|
const mobileSwipe = placement === 'mobile' && items.length > 1 |
|
|
|
return ( |
|
<div |
|
className={cn( |
|
'min-w-0 max-w-full overflow-hidden', |
|
placement === 'sidebar' && |
|
'mb-2 rounded-lg border border-border/80 bg-muted/50 p-2 shadow-sm dark:bg-muted/30', |
|
placement === 'mobile' && 'w-full shrink-0 border-b border-border/80 bg-muted/50 px-2 py-2 dark:bg-muted/30' |
|
)} |
|
role="region" |
|
aria-label={t('liveActivities.regionLabel')} |
|
aria-describedby={mobileSwipe ? swipeHintId : undefined} |
|
> |
|
{mobileSwipe ? ( |
|
<span id={swipeHintId} className="sr-only"> |
|
{t('liveActivities.swipeToBrowse')} |
|
</span> |
|
) : null} |
|
<div className="mb-1 text-[10px] font-semibold uppercase tracking-wide text-muted-foreground xl:text-xs"> |
|
{t('liveActivities.heading')} |
|
</div> |
|
<div |
|
className={cn( |
|
'flex min-w-0 gap-1.5 rounded-md', |
|
placement === 'sidebar' && 'flex-col xl:flex-row xl:items-stretch', |
|
placement === 'mobile' && 'items-stretch touch-pan-y', |
|
mobileSwipe && 'cursor-grab active:cursor-grabbing' |
|
)} |
|
onPointerDown={mobileSwipe ? onSwipePointerDown : undefined} |
|
onPointerUp={mobileSwipe ? onSwipePointerUp : undefined} |
|
onPointerCancel={mobileSwipe ? onSwipePointerCancel : undefined} |
|
> |
|
<button |
|
type="button" |
|
onClick={openLiveNote} |
|
className={cn( |
|
'flex min-w-0 flex-1 gap-2 rounded-md text-left transition-colors hover:bg-muted/80 focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring', |
|
placement === 'sidebar' && 'flex-col xl:flex-row xl:items-start', |
|
placement === 'mobile' && 'items-center' |
|
)} |
|
title={t('liveActivities.viewNoteTitle')} |
|
> |
|
{current.imageUrl ? ( |
|
<img |
|
src={current.imageUrl} |
|
alt="" |
|
className={cn( |
|
'shrink-0 rounded object-cover', |
|
placement === 'sidebar' ? 'h-14 w-full xl:h-12 xl:w-12' : 'h-12 w-12' |
|
)} |
|
/> |
|
) : null} |
|
<div className="min-w-0 flex-1"> |
|
<div className="line-clamp-2 text-xs font-medium leading-snug xl:text-sm">{current.title}</div> |
|
{current.summary ? ( |
|
<p className="mt-0.5 line-clamp-2 text-[11px] text-muted-foreground xl:text-xs">{current.summary}</p> |
|
) : null} |
|
{current.fromFollowedHost ? ( |
|
<p className="mt-1 text-[10px] text-green-600 dark:text-green-500">{t('liveActivities.fromFollow')}</p> |
|
) : null} |
|
</div> |
|
</button> |
|
<a |
|
href={current.joinUrl} |
|
target="_blank" |
|
rel="noopener noreferrer" |
|
className={cn( |
|
'flex shrink-0 items-center justify-center rounded-md text-muted-foreground transition-colors hover:bg-muted/80 hover:text-foreground', |
|
placement === 'sidebar' ? 'h-9 w-full xl:h-auto xl:w-9 xl:self-start' : 'h-12 w-10' |
|
)} |
|
title={t('liveActivities.openJoinPageTitle')} |
|
aria-label={t('liveActivities.openJoinPageTitle')} |
|
onClick={(e) => e.stopPropagation()} |
|
> |
|
<ExternalLink className="size-4 shrink-0" aria-hidden /> |
|
</a> |
|
</div> |
|
{items.length > 1 ? ( |
|
placement === 'mobile' ? ( |
|
<div className="mt-2 flex justify-center gap-1.5" aria-hidden> |
|
{items.map((item, i) => ( |
|
<span |
|
key={item.address} |
|
className={cn( |
|
'block size-1.5 rounded-full', |
|
i === displayIndex ? 'bg-primary' : 'bg-muted-foreground/45' |
|
)} |
|
/> |
|
))} |
|
</div> |
|
) : ( |
|
<div className="mt-2 flex justify-center gap-1.5" role="tablist"> |
|
{items.map((item, i) => ( |
|
<button |
|
key={item.address} |
|
type="button" |
|
role="tab" |
|
aria-selected={i === displayIndex} |
|
aria-label={t('liveActivities.goToSlide', { n: i + 1 })} |
|
className="flex shrink-0 items-center justify-center rounded-full p-0.5 active:opacity-80" |
|
onClick={(e) => { |
|
e.preventDefault() |
|
setSlide(i) |
|
}} |
|
> |
|
<span |
|
className={cn( |
|
'block size-1.5 rounded-full transition-colors', |
|
i === displayIndex ? 'bg-primary' : 'bg-muted-foreground/45 hover:bg-muted-foreground/70' |
|
)} |
|
aria-hidden |
|
/> |
|
</button> |
|
))} |
|
</div> |
|
) |
|
) : null} |
|
</div> |
|
) |
|
}
|
|
|