import {Fragment, useMemo} from 'react'; import styled from '@emotion/styled'; import Pagination from 'sentry/components/pagination'; import {t, tct} from 'sentry/locale'; import {trackAnalytics} from 'sentry/utils/analytics'; import {browserHistory} from 'sentry/utils/browserHistory'; import {decodeList, decodeScalar, decodeSorts} from 'sentry/utils/queryString'; import useFetchReplayList from 'sentry/utils/replays/hooks/useFetchReplayList'; import {MutableSearch} from 'sentry/utils/tokenizeSearch'; import useLocationQuery from 'sentry/utils/url/useLocationQuery'; import useOrganization from 'sentry/utils/useOrganization'; import usePageFilters from 'sentry/utils/usePageFilters'; import useProjectSdkNeedsUpdate from 'sentry/utils/useProjectSdkNeedsUpdate'; import useAllMobileProj from 'sentry/views/replays/detail/useAllMobileProj'; import ReplayTable from 'sentry/views/replays/replayTable'; import {ReplayColumn} from 'sentry/views/replays/replayTable/types'; const MIN_REPLAY_CLICK_SDK = '7.44.0'; function ReplaysList() { const organization = useOrganization(); const query = useLocationQuery({ fields: { cursor: decodeScalar, end: decodeScalar, environment: decodeList, project: decodeList, query: decodeScalar, sort: value => decodeScalar(value, '-started_at'), start: decodeScalar, statsPeriod: decodeScalar, utc: decodeScalar, }, }); const { data: replays, getResponseHeader, isPending, error, } = useFetchReplayList({ options: {query}, organization, queryReferrer: 'replayList', }); const pageLinks = getResponseHeader?.('Link') ?? null; const { selection: {projects}, } = usePageFilters(); const {allMobileProj} = useAllMobileProj(); const {needsUpdate: allSelectedProjectsNeedUpdates} = useProjectSdkNeedsUpdate({ minVersion: MIN_REPLAY_CLICK_SDK, organization, projectId: projects.map(String), }); const conditions = useMemo(() => new MutableSearch(query.query), [query.query]); const hasReplayClick = conditions.getFilterKeys().some(k => k.startsWith('click.')); // browser isn't applicable for mobile projects // rage and dead clicks not available yet const visibleCols = allMobileProj ? [ ReplayColumn.REPLAY, ReplayColumn.OS, ReplayColumn.DURATION, ReplayColumn.COUNT_ERRORS, ReplayColumn.ACTIVITY, ] : [ ReplayColumn.REPLAY, ReplayColumn.OS, ReplayColumn.BROWSER, ReplayColumn.DURATION, ReplayColumn.COUNT_DEAD_CLICKS, ReplayColumn.COUNT_RAGE_CLICKS, ReplayColumn.COUNT_ERRORS, ReplayColumn.ACTIVITY, ]; return ( {t('Unindexed search field')} {tct('Field [field] requires an [sdkPrompt]', { field: 'click', sdkPrompt: {t('SDK version >= 7.44.0')}, })} ) : undefined } /> { trackAnalytics('replay.list-paginated', { organization, direction: cursor?.endsWith(':1') ? 'prev' : 'next', }); browserHistory.push({ pathname: path, query: {...searchQuery, cursor}, }); }} /> ); } const EmptyStateSubheading = styled('div')` color: ${p => p.theme.subText}; font-size: ${p => p.theme.fontSizeMedium}; `; const ReplayPagination = styled(Pagination)` margin-top: 0; `; export default ReplaysList;