import { createSelector } from '@reduxjs/toolkit'; import clsx from 'clsx'; import { List as ImmutableList, OrderedSet as ImmutableOrderedSet } from 'immutable'; import React, { useEffect, useRef, useState } from 'react'; import { HotKeys } from 'react-hotkeys'; import { useIntl } from 'react-intl'; import { useHistory } from 'react-router-dom'; import { type VirtuosoHandle } from 'react-virtuoso'; import { mentionCompose, replyCompose } from 'soapbox/actions/compose'; import { favourite, reblog, unfavourite, unreblog } from 'soapbox/actions/interactions'; import { openModal } from 'soapbox/actions/modals'; import { getSettings } from 'soapbox/actions/settings'; import { hideStatus, revealStatus } from 'soapbox/actions/statuses'; import ScrollableList from 'soapbox/components/scrollable-list'; import StatusActionBar from 'soapbox/components/status-action-bar'; import Tombstone from 'soapbox/components/tombstone'; import { Stack } from 'soapbox/components/ui'; import PlaceholderStatus from 'soapbox/features/placeholder/components/placeholder-status'; import PendingStatus from 'soapbox/features/ui/components/pending-status'; import { useAppDispatch, useAppSelector, useOwnAccount, useSettings } from 'soapbox/hooks'; import { RootState } from 'soapbox/store'; import { type Account, type Status } from 'soapbox/types/entities'; import { defaultMediaVisibility, textForScreenReader } from 'soapbox/utils/status'; import DetailedStatus from './detailed-status'; import ThreadLoginCta from './thread-login-cta'; import ThreadStatus from './thread-status'; type DisplayMedia = 'default' | 'hide_all' | 'show_all'; const getAncestorsIds = createSelector([ (_: RootState, statusId: string | undefined) => statusId, (state: RootState) => state.contexts.inReplyTos, ], (statusId, inReplyTos) => { let ancestorsIds = ImmutableOrderedSet(); let id: string | undefined = statusId; while (id && !ancestorsIds.includes(id)) { ancestorsIds = ImmutableOrderedSet([id]).union(ancestorsIds); id = inReplyTos.get(id); } return ancestorsIds; }); export const getDescendantsIds = createSelector([ (_: RootState, statusId: string) => statusId, (state: RootState) => state.contexts.replies, ], (statusId, contextReplies) => { let descendantsIds = ImmutableOrderedSet(); const ids = [statusId]; while (ids.length > 0) { const id = ids.shift(); if (!id) break; const replies = contextReplies.get(id); if (descendantsIds.includes(id)) { break; } if (statusId !== id) { descendantsIds = descendantsIds.union([id]); } if (replies) { replies.reverse().forEach((reply: string) => { ids.unshift(reply); }); } } return descendantsIds; }); interface IThread { status: Status withMedia?: boolean useWindowScroll?: boolean itemClassName?: string next: string | undefined handleLoadMore: () => void } const Thread = (props: IThread) => { const { handleLoadMore, itemClassName, next, status, useWindowScroll = true, withMedia = true, } = props; const dispatch = useAppDispatch(); const history = useHistory(); const intl = useIntl(); const { account: me } = useOwnAccount(); const settings = useSettings(); const displayMedia = settings.get('displayMedia') as DisplayMedia; const isUnderReview = status?.visibility === 'self'; const { ancestorsIds, descendantsIds } = useAppSelector((state) => { let ancestorsIds = ImmutableOrderedSet(); let descendantsIds = ImmutableOrderedSet(); if (status) { const statusId = status.id; ancestorsIds = getAncestorsIds(state, state.contexts.inReplyTos.get(statusId)); descendantsIds = getDescendantsIds(state, statusId); ancestorsIds = ancestorsIds.delete(statusId).subtract(descendantsIds); descendantsIds = descendantsIds.delete(statusId).subtract(ancestorsIds); } return { status, ancestorsIds, descendantsIds, }; }); let initialTopMostItemIndex = ancestorsIds.size; if (!useWindowScroll && initialTopMostItemIndex !== 0) initialTopMostItemIndex = ancestorsIds.size + 1; const [showMedia, setShowMedia] = useState(status?.visibility === 'self' ? false : defaultMediaVisibility(status, displayMedia)); const node = useRef(null); const statusRef = useRef(null); const scroller = useRef(null); const handleToggleMediaVisibility = () => { setShowMedia(!showMedia); }; const handleHotkeyReact = () => { if (statusRef.current) { const firstEmoji: HTMLButtonElement | null = statusRef.current.querySelector('.emoji-react-selector .emoji-react-selector__emoji'); firstEmoji?.focus(); } }; const handleFavouriteClick = (status: Status) => { if (status.favourited) { dispatch(unfavourite(status)); } else { dispatch(favourite(status)); } }; const handleReplyClick = (status: Status) => dispatch(replyCompose(status)); const handleModalReblog = (status: Status) => dispatch(reblog(status)); const handleReblogClick = (status: Status, e?: React.MouseEvent) => { dispatch((_, getState) => { const boostModal = getSettings(getState()).get('boostModal'); if (status.reblogged) { dispatch(unreblog(status)); } else { if ((e && e.shiftKey) || !boostModal) { handleModalReblog(status); } else { dispatch(openModal('BOOST', { status, onReblog: handleModalReblog })); } } }); }; const handleMentionClick = (account: Account) => dispatch(mentionCompose(account)); const handleHotkeyOpenMedia = (e?: KeyboardEvent) => { const media = status?.media_attachments; e?.preventDefault(); if (media && media.size) { const firstAttachment = media.first()!; if (media.size === 1 && firstAttachment.type === 'video') { dispatch(openModal('VIDEO', { media: firstAttachment, status: status })); } else { dispatch(openModal('MEDIA', { media, index: 0, status: status })); } } }; const handleToggleHidden = (status: Status) => { if (status.hidden) { dispatch(revealStatus(status.id)); } else { dispatch(hideStatus(status.id)); } }; const handleHotkeyMoveUp = () => { handleMoveUp(status!.id); }; const handleHotkeyMoveDown = () => { handleMoveDown(status!.id); }; const handleHotkeyReply = (e?: KeyboardEvent) => { e?.preventDefault(); handleReplyClick(status!); }; const handleHotkeyFavourite = () => { handleFavouriteClick(status!); }; const handleHotkeyBoost = () => { handleReblogClick(status!); }; const handleHotkeyMention = (e?: KeyboardEvent) => { e?.preventDefault(); const { account } = status!; if (!account || typeof account !== 'object') return; handleMentionClick(account); }; const handleHotkeyOpenProfile = () => { history.push(`/@${status!.getIn(['account', 'acct'])}`); }; const handleHotkeyToggleHidden = () => { handleToggleHidden(status!); }; const handleHotkeyToggleSensitive = () => { handleToggleMediaVisibility(); }; const handleMoveUp = (id: string) => { if (id === status?.id) { _selectChild(ancestorsIds.size - 1); } else { let index = ImmutableList(ancestorsIds).indexOf(id); if (index === -1) { index = ImmutableList(descendantsIds).indexOf(id); _selectChild(ancestorsIds.size + index); } else { _selectChild(index - 1); } } }; const handleMoveDown = (id: string) => { if (id === status?.id) { _selectChild(ancestorsIds.size + 1); } else { let index = ImmutableList(ancestorsIds).indexOf(id); if (index === -1) { index = ImmutableList(descendantsIds).indexOf(id); _selectChild(ancestorsIds.size + index + 2); } else { _selectChild(index + 1); } } }; const _selectChild = (index: number) => { if (!useWindowScroll) index = index + 1; scroller.current?.scrollIntoView({ index, behavior: 'smooth', done: () => { node.current?.querySelector(`[data-index="${index}"] .focusable`)?.focus(); }, }); }; const renderTombstone = (id: string) => { return (
); }; const renderStatus = (id: string) => { return ( ); }; const renderPendingStatus = (id: string) => { const idempotencyKey = id.replace(/^末pending-/, ''); return ( ); }; const renderChildren = (list: ImmutableOrderedSet) => { return list.map(id => { if (id.endsWith('-tombstone')) { return renderTombstone(id); } else if (id.startsWith('末pending-')) { return renderPendingStatus(id); } else { return renderStatus(id); } }); }; // Reset media visibility if status changes. useEffect(() => { setShowMedia(status?.visibility === 'self' ? false : defaultMediaVisibility(status, displayMedia)); }, [status.id]); // Scroll focused status into view when thread updates. useEffect(() => { scroller.current?.scrollToIndex({ index: ancestorsIds.size, offset: -146, }); setTimeout(() => statusRef.current?.querySelector('.detailed-actualStatus')?.focus(), 0); }, [status.id, ancestorsIds.size]); const handleOpenCompareHistoryModal = (status: Status) => { dispatch(openModal('COMPARE_HISTORY', { statusId: status.id, })); }; const hasAncestors = ancestorsIds.size > 0; const hasDescendants = descendantsIds.size > 0; type HotkeyHandlers = { [key: string]: (keyEvent?: KeyboardEvent) => void }; const handlers: HotkeyHandlers = { moveUp: handleHotkeyMoveUp, moveDown: handleHotkeyMoveDown, reply: handleHotkeyReply, favourite: handleHotkeyFavourite, boost: handleHotkeyBoost, mention: handleHotkeyMention, openProfile: handleHotkeyOpenProfile, toggleHidden: handleHotkeyToggleHidden, toggleSensitive: handleHotkeyToggleSensitive, openMedia: handleHotkeyOpenMedia, react: handleHotkeyReact, }; const focusedStatus = (
{!isUnderReview ? ( <>
) : null}
{hasDescendants && (
)}
); const children: JSX.Element[] = []; if (!useWindowScroll) { // Add padding to the top of the Thread (for Media Modal) children.push(
); } if (hasAncestors) { children.push(...renderChildren(ancestorsIds).toArray()); } children.push(focusedStatus); if (hasDescendants) { children.push(...renderChildren(descendantsIds).toArray()); } return (
} initialTopMostItemIndex={initialTopMostItemIndex} useWindowScroll={useWindowScroll} itemClassName={itemClassName} className={ clsx({ 'h-full': !useWindowScroll, }) } > {children}
{!me && }
); }; export default Thread;