import {useMemo, useRef, useState} from 'react'; import type {GridCellProps} from 'react-virtualized'; import {AutoSizer, CellMeasurer, MultiGrid} from 'react-virtualized'; import styled from '@emotion/styled'; 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 {t} from 'sentry/locale'; import useCrumbHandlers from 'sentry/utils/replays/hooks/useCrumbHandlers'; import ErrorFilters from 'sentry/views/replays/detail/errorList/errorFilters'; import ErrorHeaderCell, { COLUMN_COUNT, } from 'sentry/views/replays/detail/errorList/errorHeaderCell'; import ErrorTableCell from 'sentry/views/replays/detail/errorList/errorTableCell'; import useErrorFilters from 'sentry/views/replays/detail/errorList/useErrorFilters'; import useSortErrors from 'sentry/views/replays/detail/errorList/useSortErrors'; import FluidHeight from 'sentry/views/replays/detail/layout/fluidHeight'; import NoRowRenderer from 'sentry/views/replays/detail/noRowRenderer'; import useVirtualizedGrid from 'sentry/views/replays/detail/useVirtualizedGrid'; const HEADER_HEIGHT = 25; const BODY_HEIGHT = 25; const cellMeasurer = { defaultHeight: BODY_HEIGHT, defaultWidth: 100, fixedHeight: true, }; function ErrorList() { const {currentTime, currentHoverTime, replay} = useReplayContext(); const {onMouseEnter, onMouseLeave, onClickTimestamp} = useCrumbHandlers(); const errorFrames = replay?.getErrorFrames(); const startTimestampMs = replay?.getReplay().started_at.getTime() ?? 0; const [scrollToRow, setScrollToRow] = useState(undefined); const filterProps = useErrorFilters({errorFrames: errorFrames || []}); const {items: filteredItems, searchTerm, setSearchTerm} = filterProps; const clearSearchTerm = () => setSearchTerm(''); const {handleSort, items, sortConfig} = useSortErrors({items: filteredItems}); const gridRef = useRef(null); const deps = useMemo(() => [items, searchTerm], [items, searchTerm]); const {cache, getColumnWidth, onScrollbarPresenceChange, onWrapperResize} = useVirtualizedGrid({ cellMeasurer, gridRef, columnCount: COLUMN_COUNT, dynamicColumnIndex: 1, deps, }); const { handleClick: onClickToJump, onSectionRendered, showJumpDownButton, showJumpUpButton, } = useJumpButtons({ currentTime, frames: filteredItems, isTable: true, setScrollToRow, }); const cellRenderer = ({columnIndex, rowIndex, key, style, parent}: GridCellProps) => { const error = items[rowIndex - 1]; return ( {({ measure: _, registerChild, }: { measure: () => void; registerChild?: (element?: Element) => void; }) => rowIndex === 0 ? ( e && registerChild?.(e)} handleSort={handleSort} index={columnIndex} sortConfig={sortConfig} style={{...style, height: HEADER_HEIGHT}} /> ) : ( e && registerChild?.(e)} rowIndex={rowIndex} sortConfig={sortConfig} startTimestampMs={startTimestampMs} style={{...style, height: BODY_HEIGHT}} /> ) } ); }; return ( {errorFrames ? ( {({height, width}) => ( ( {t('No errors! Go make some.')} )} onScrollbarPresenceChange={onScrollbarPresenceChange} onScroll={() => { if (scrollToRow !== undefined) { setScrollToRow(undefined); } }} onSectionRendered={onSectionRendered} overscanColumnCount={COLUMN_COUNT} overscanRowCount={5} rowCount={items.length + 1} rowHeight={({index}) => (index === 0 ? HEADER_HEIGHT : BODY_HEIGHT)} scrollToRow={scrollToRow} width={width} /> )} {sortConfig.by === 'timestamp' && items.length ? ( ) : null} ) : ( )} ); } const OverflowHidden = styled('div')` position: relative; height: 100%; overflow: hidden; display: grid; `; const ErrorTable = styled(FluidHeight)` border: 1px solid ${p => p.theme.border}; border-radius: ${p => p.theme.borderRadius}; .beforeHoverTime + .afterHoverTime:before { border-top: 1px solid ${p => p.theme.purple200}; content: ''; left: 0; position: absolute; top: 0; width: 999999999%; } .beforeHoverTime:last-child:before { border-bottom: 1px solid ${p => p.theme.purple200}; content: ''; right: 0; position: absolute; bottom: 0; width: 999999999%; } .beforeCurrentTime + .afterCurrentTime:before { border-top: 1px solid ${p => p.theme.purple300}; content: ''; left: 0; position: absolute; top: 0; width: 999999999%; } .beforeCurrentTime:last-child:before { border-bottom: 1px solid ${p => p.theme.purple300}; content: ''; right: 0; position: absolute; bottom: 0; width: 999999999%; } `; export default ErrorList;