import {memo, useMemo, useRef} from 'react'; import { AutoSizer, CellMeasurer, List as ReactVirtualizedList, ListRowProps, } from 'react-virtualized'; import {useQuery} from '@tanstack/react-query'; import Placeholder from 'sentry/components/placeholder'; import JumpButtons from 'sentry/components/replays/jumpButtons'; import {useReplayContext} from 'sentry/components/replays/replayContext'; import {t} from 'sentry/locale'; import extractDomNodes from 'sentry/utils/replays/extractDomNodes'; import useCrumbHandlers from 'sentry/utils/replays/hooks/useCrumbHandlers'; import type ReplayReader from 'sentry/utils/replays/replayReader'; import DomFilters from 'sentry/views/replays/detail/domMutations/domFilters'; import DomMutationRow from 'sentry/views/replays/detail/domMutations/domMutationRow'; import useDomFilters from 'sentry/views/replays/detail/domMutations/useDomFilters'; import FilterLoadingIndicator from 'sentry/views/replays/detail/filterLoadingIndicator'; import FluidHeight from 'sentry/views/replays/detail/layout/fluidHeight'; import NoRowRenderer from 'sentry/views/replays/detail/noRowRenderer'; import TabItemContainer from 'sentry/views/replays/detail/tabItemContainer'; import useVirtualizedList from 'sentry/views/replays/detail/useVirtualizedList'; // Ensure this object is created once as it is an input to // `useVirtualizedList`'s memoization const cellMeasurer = { fixedWidth: true, minHeight: 82, }; function useExtractedDomNodes({replay}: {replay: null | ReplayReader}) { return useQuery( ['getDomNodes', replay], () => extractDomNodes({ frames: replay?.getDOMFrames(), rrwebEvents: replay?.getRRWebFrames(), startTimestampMs: replay?.getReplay().started_at.getTime() ?? 0, }), {enabled: Boolean(replay), cacheTime: Infinity} ); } function DomMutations() { const {currentTime, currentHoverTime, replay} = useReplayContext(); const {onMouseEnter, onMouseLeave, onClickTimestamp} = useCrumbHandlers(); const {data: frameToExtraction, isFetching} = useExtractedDomNodes({replay}); const actions = useMemo( () => Array.from(frameToExtraction?.values() || []), [frameToExtraction] ); const startTimestampMs = replay?.getReplay()?.started_at?.getTime() ?? 0; const filterProps = useDomFilters({actions}); const {items, setSearchTerm} = filterProps; const clearSearchTerm = () => setSearchTerm(''); const listRef = useRef(null); const deps = useMemo(() => [items], [items]); const {cache, updateList} = useVirtualizedList({ cellMeasurer, ref: listRef, deps, }); const renderRow = ({index, key, style, parent}: ListRowProps) => { const mutation = items[index]; return ( ); }; const showJumpUpButton = false; const showJumpDownButton = false; return ( {isFetching || !actions ? ( ) : ( {({width, height}) => ( ( {t('No DOM events recorded')} )} overscanRowCount={5} ref={listRef} rowCount={items.length} rowHeight={cache.rowHeight} rowRenderer={renderRow} width={width} /> )} )} {}} tableHeaderHeight={0} /> ); } export default memo(DomMutations);