import './status.css'; import debounce from 'just-debounce-it'; import { Link } from 'preact-router/match'; import { useEffect, useMemo, useRef, useState } from 'preact/hooks'; import { useHotkeys } from 'react-hotkeys-hook'; import { InView } from 'react-intersection-observer'; import { useSnapshot } from 'valtio'; import Icon from '../components/icon'; import Loader from '../components/loader'; import NameText from '../components/name-text'; import RelativeTime from '../components/relative-time'; import Status from '../components/status'; import htmlContentLength from '../utils/html-content-length'; import shortenNumber from '../utils/shorten-number'; import states, { saveStatus } from '../utils/states'; import store from '../utils/store'; import useDebouncedCallback from '../utils/useDebouncedCallback'; import useScroll from '../utils/useScroll'; import useTitle from '../utils/useTitle'; const LIMIT = 40; function StatusPage({ id }) { const snapStates = useSnapshot(states); const [statuses, setStatuses] = useState([]); const [uiState, setUIState] = useState('default'); const heroStatusRef = useRef(); const scrollableRef = useRef(); useEffect(() => { scrollableRef.current?.focus(); }, []); useEffect(() => { const onScroll = debounce(() => { // console.log('onScroll'); if (!scrollableRef.current) return; const { scrollTop } = scrollableRef.current; states.scrollPositions[id] = scrollTop; }, 100); scrollableRef.current.addEventListener('scroll', onScroll, { passive: true, }); onScroll(); return () => { scrollableRef.current?.removeEventListener('scroll', onScroll); }; }, [id]); const scrollOffsets = useRef(); const cachedStatusesMap = useRef({}); const initContext = () => { console.debug('initContext', id); setUIState('loading'); let heroTimer; const cachedStatuses = cachedStatusesMap.current[id]; if (cachedStatuses) { // Case 1: It's cached, let's restore them to make it snappy const reallyCachedStatuses = cachedStatuses.filter( (s) => states.statuses[s.id], // Some are not cached in the global state, so we need to filter them out ); setStatuses(reallyCachedStatuses); } else { // const heroIndex = statuses.findIndex((s) => s.id === id); // if (heroIndex !== -1) { // // Case 2: It's in current statuses. Slice off all descendant statuses after the hero status to be safe // const slicedStatuses = statuses.slice(0, heroIndex + 1); // setStatuses(slicedStatuses); // } else { // Case 3: Not cached and not in statuses, let's start from scratch setStatuses([{ id }]); // } } (async () => { const heroFetch = () => masto.v1.statuses.fetch(id); const contextFetch = masto.v1.statuses.fetchContext(id); const hasStatus = !!snapStates.statuses[id]; let heroStatus = snapStates.statuses[id]; if (hasStatus) { console.debug('Hero status is cached'); } else { try { heroStatus = await heroFetch(); saveStatus(heroStatus); // Give time for context to appear await new Promise((resolve) => { setTimeout(resolve, 100); }); } catch (e) { console.error(e); setUIState('error'); return; } } try { const context = await contextFetch; const { ancestors, descendants } = context; ancestors.forEach((status) => { states.statuses[status.id] = status; }); const nestedDescendants = []; descendants.forEach((status) => { states.statuses[status.id] = status; if (status.inReplyToAccountId === status.account.id) { // If replying to self, it's part of the thread, level 1 nestedDescendants.push(status); } else if (status.inReplyToId === heroStatus.id) { // If replying to the hero status, it's a reply, level 1 nestedDescendants.push(status); } else { // If replying to someone else, it's a reply to a reply, level 2 const parent = descendants.find((s) => s.id === status.inReplyToId); if (parent) { if (!parent.__replies) { parent.__replies = []; } parent.__replies.push(status); } else { // If no parent, it's probably a reply to a reply to a reply, level 3 console.warn('[LEVEL 3] No parent found for', status); } } }); console.log({ ancestors, descendants, nestedDescendants }); const allStatuses = [ ...ancestors.map((s) => ({ id: s.id, ancestor: true, accountID: s.account.id, })), { id, accountID: heroStatus.account.id }, ...nestedDescendants.map((s) => ({ id: s.id, accountID: s.account.id, descendant: true, thread: s.account.id === heroStatus.account.id, replies: s.__replies?.map((r) => ({ id: r.id, repliesCount: r.repliesCount, content: r.content, })), })), ]; setUIState('default'); scrollOffsets.current = { offsetTop: heroStatusRef.current?.offsetTop, scrollTop: scrollableRef.current?.scrollTop, }; console.log({ allStatuses }); setStatuses(allStatuses); cachedStatusesMap.current[id] = allStatuses; } catch (e) { console.error(e); setUIState('error'); } })(); return () => { clearTimeout(heroTimer); }; }; useEffect(initContext, [id]); useEffect(() => { if (!statuses.length) return; console.debug('STATUSES', statuses); const scrollPosition = states.scrollPositions[id]; console.debug('scrollPosition', scrollPosition); if (!!scrollPosition) { console.debug('Case 1', { scrollPosition, }); scrollableRef.current.scrollTop = scrollPosition; } else if (scrollOffsets.current) { const newScrollOffsets = { offsetTop: heroStatusRef.current?.offsetTop, scrollTop: scrollableRef.current?.scrollTop, }; const newScrollTop = newScrollOffsets.offsetTop - scrollOffsets.current.offsetTop; console.debug('Case 2', { scrollOffsets: scrollOffsets.current, newScrollOffsets, newScrollTop, statuses: [...statuses], }); scrollableRef.current.scrollTop = newScrollTop; } // RESET scrollOffsets.current = null; }, [statuses]); useEffect(() => { if (snapStates.reloadStatusPage <= 0) return; // Delete the cache for the context (async () => { try { const accounts = store.local.getJSON('accounts') || []; const currentAccount = store.session.get('currentAccount'); const account = accounts.find((a) => a.info.id === currentAccount) || accounts[0]; const instanceURL = account.instanceURL; const contextURL = `https://${instanceURL}/api/v1/statuses/${id}/context`; console.log('Clear cache', contextURL); const apiCache = await caches.open('api'); await apiCache.delete(contextURL, { ignoreVary: true }); return initContext(); } catch (e) { console.error(e); } })(); }, [snapStates.reloadStatusPage]); useEffect(() => { return () => { // RESET states.scrollPositions = {}; states.reloadStatusPage = 0; cachedStatusesMap.current = {}; }; }, []); const heroStatus = snapStates.statuses[id]; const heroDisplayName = useMemo(() => { // Remove shortcodes from display name if (!heroStatus) return ''; const { account } = heroStatus; const div = document.createElement('div'); div.innerHTML = account.displayName; return div.innerText.trim(); }, [heroStatus]); const heroContentText = useMemo(() => { if (!heroStatus) return ''; const { spoilerText, content } = heroStatus; let text; if (spoilerText) { text = spoilerText; } else { const div = document.createElement('div'); div.innerHTML = content; text = div.innerText.trim(); } if (text.length > 64) { // "The title should ideally be less than 64 characters in length" // https://www.w3.org/Provider/Style/TITLE.html text = text.slice(0, 64) + '…'; } return text; }, [heroStatus]); useTitle( heroDisplayName && heroContentText ? `${heroDisplayName}: "${heroContentText}"` : 'Status', ); const prevRoute = states.history.findLast((h) => { return h === '/' || /notifications/i.test(h); }); const closeLink = `#${prevRoute || '/'}`; const [limit, setLimit] = useState(LIMIT); const showMore = useMemo(() => { // return number of statuses to show return statuses.length - limit; }, [statuses.length, limit]); const hasManyStatuses = statuses.length > LIMIT; const hasDescendants = statuses.some((s) => s.descendant); const ancestors = statuses.filter((s) => s.ancestor); const [heroInView, setHeroInView] = useState(true); const onView = useDebouncedCallback(setHeroInView, 100); const heroPointer = useMemo(() => { // get top offset of heroStatus if (!heroStatusRef.current || heroInView) return null; const { top } = heroStatusRef.current.getBoundingClientRect(); return top > 0 ? 'down' : 'up'; }, [heroInView]); useHotkeys(['esc', 'backspace'], () => { location.hash = closeLink; }); const { nearReachTop } = useScroll({ scrollableElement: scrollableRef.current, distanceFromTop: 0.1, }); return (
1 ? 'padded-bottom' : '' }`} >
{ if ( !/^(a|button)$/i.test(e.target.tagName) && heroStatusRef.current ) { heroStatusRef.current.scrollIntoView({ behavior: 'smooth', block: 'start', }); } }} onDblClick={(e) => { // reload statuses states.reloadStatusPage++; }} > {/*
*/}

{!heroInView && heroStatus && uiState !== 'loading' ? ( {!!heroPointer && ( <> {' '} )} {' '} •{' '} ) : ( <> Status{' '} )}

{!!statuses.length && heroStatus ? ( ) : ( <> {uiState === 'loading' && ( )} {uiState === 'error' && (

Unable to load status

)} )}
); } function SubComments({ hasManyStatuses, replies, onStatusLinkClick = () => {}, }) { // If less than or 2 replies and total number of characters of content from replies is less than 500 let isBrief = false; if (replies.length <= 2) { let totalLength = replies.reduce((acc, reply) => { const { content } = reply; const length = htmlContentLength(content); return acc + length; }, 0); isBrief = totalLength < 500; } const open = isBrief || !hasManyStatuses; return (
); } export default StatusPage;