import {useEffect, useMemo, useRef, useState} from 'react'; import type {ListRowProps} from 'react-virtualized'; import {AutoSizer, CellMeasurer, List as ReactVirtualizedList} from 'react-virtualized'; import styled from '@emotion/styled'; import Alert from 'sentry/components/alert'; import {Button} from 'sentry/components/button'; import ExternalLink from 'sentry/components/links/externalLink'; import Placeholder from 'sentry/components/placeholder'; import JumpButtons from 'sentry/components/replays/jumpButtons'; import {useReplayContext} from 'sentry/components/replays/replayContext'; import useJumpButtons from 'sentry/components/replays/useJumpButtons'; import {IconClose} from 'sentry/icons'; import {t, tct} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import {trackAnalytics} from 'sentry/utils/analytics'; import useCrumbHandlers from 'sentry/utils/replays/hooks/useCrumbHandlers'; import useExtractedDomNodes from 'sentry/utils/replays/hooks/useExtractedDomNodes'; import useDismissAlert from 'sentry/utils/useDismissAlert'; import useOrganization from 'sentry/utils/useOrganization'; import useVirtualizedInspector from 'sentry/views/replays/detail//useVirtualizedInspector'; import BreadcrumbFilters from 'sentry/views/replays/detail/breadcrumbs/breadcrumbFilters'; import BreadcrumbRow from 'sentry/views/replays/detail/breadcrumbs/breadcrumbRow'; import useBreadcrumbFilters from 'sentry/views/replays/detail/breadcrumbs/useBreadcrumbFilters'; import useScrollToCurrentItem from 'sentry/views/replays/detail/breadcrumbs/useScrollToCurrentItem'; 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 useReplayPerfData from 'sentry/views/replays/detail/perfTable/useReplayPerfData'; import TabItemContainer from 'sentry/views/replays/detail/tabItemContainer'; import useVirtualizedList from 'sentry/views/replays/detail/useVirtualizedList'; import useVirtualListDimentionChange from 'sentry/views/replays/detail/useVirtualListDimentionChange'; const LOCAL_STORAGE_KEY = 'replay-details-mask-config-instructions-dismissed'; // Ensure this object is created once as it is an input to // `useVirtualizedList`'s memoization const cellMeasurer = { fixedWidth: true, minHeight: 53, }; function Breadcrumbs() { const {dismiss, isDismissed} = useDismissAlert({key: LOCAL_STORAGE_KEY}); const {currentTime, replay, startTimeOffsetMs, durationMs} = useReplayContext(); const organization = useOrganization(); const hasPerfTab = organization.features.includes('session-replay-trace-table'); const {onClickTimestamp} = useCrumbHandlers(); const {data: frameToExtraction, isFetching: isFetchingExtractions} = useExtractedDomNodes({replay}); const {data: frameToTrace, isFetching: isFetchingTraces} = useReplayPerfData({replay}); const startTimestampMs = replay?.getReplay()?.started_at?.getTime() ?? 0 + startTimeOffsetMs; const allFrames = replay?.getChapterFrames(); const frames = useMemo( () => allFrames?.filter( frame => frame.offsetMs >= startTimeOffsetMs && frame.offsetMs <= startTimeOffsetMs + durationMs ), [allFrames, durationMs, startTimeOffsetMs] ); const [scrollToRow, setScrollToRow] = useState(undefined); const filterProps = useBreadcrumbFilters({frames: frames || []}); const {expandPathsRef, items, searchTerm, setSearchTerm} = filterProps; const clearSearchTerm = () => setSearchTerm(''); const listRef = useRef(null); const deps = useMemo(() => [items, searchTerm], [items, searchTerm]); const {cache, updateList} = useVirtualizedList({ cellMeasurer, ref: listRef, deps, }); const {handleDimensionChange} = useVirtualListDimentionChange({cache, listRef}); const {handleDimensionChange: handleInspectorExpanded} = useVirtualizedInspector({ cache, listRef, expandPathsRef, }); const { handleClick: onClickToJump, onRowsRendered, showJumpDownButton, showJumpUpButton, } = useJumpButtons({ currentTime, frames: items, isTable: false, setScrollToRow, }); useScrollToCurrentItem({ frames, ref: listRef, }); // Need to refresh the item dimensions as DOM & Trace data gets loaded useEffect(() => { if (!isFetchingExtractions || !isFetchingTraces) { updateList(); } }, [isFetchingExtractions, isFetchingTraces, updateList]); const renderRow = ({index, key, style, parent}: ListRowProps) => { const item = (items || [])[index]; return ( { onClickTimestamp(item); }} onDimensionChange={handleDimensionChange} onInspectorExpanded={handleInspectorExpanded} /> ); }; return ( {isDismissed ? null : ( } onClick={dismiss} size="zero" borderless /> } > {tct('Learn how to unmask text (****) and unblock media [link:here].', { link: ( { trackAnalytics('replay.details-mask-banner-link-clicked', { organization, }); }} /> ), })} )} {frames ? ( {({height, width}) => ( ( {t('No breadcrumbs recorded')} )} onRowsRendered={onRowsRendered} onScroll={() => { if (scrollToRow !== undefined) { setScrollToRow(undefined); } }} overscanRowCount={5} ref={listRef} rowCount={items.length} rowHeight={cache.rowHeight} rowRenderer={renderRow} scrollToIndex={scrollToRow} width={width} /> )} ) : ( )} {items?.length ? ( ) : null} ); } const StyledAlert = styled(Alert)` margin-bottom: ${space(1)}; `; export default Breadcrumbs;