import {Fragment} from 'react'; import styled from '@emotion/styled'; import * as qs from 'query-string'; import {openBulkEditMonitorsModal} from 'sentry/actionCreators/modal'; import {Button} from 'sentry/components/button'; import ButtonBar from 'sentry/components/buttonBar'; import FeedbackWidgetButton from 'sentry/components/feedback/widget/feedbackWidgetButton'; import HookOrDefault from 'sentry/components/hookOrDefault'; import * as Layout from 'sentry/components/layouts/thirds'; import LoadingIndicator from 'sentry/components/loadingIndicator'; import {DatePageFilter} from 'sentry/components/organizations/datePageFilter'; import {EnvironmentPageFilter} from 'sentry/components/organizations/environmentPageFilter'; import PageFilterBar from 'sentry/components/organizations/pageFilterBar'; import {normalizeDateTimeParams} from 'sentry/components/organizations/pageFilters/parse'; import {ProjectPageFilter} from 'sentry/components/organizations/projectPageFilter'; import {PageHeadingQuestionTooltip} from 'sentry/components/pageHeadingQuestionTooltip'; import Pagination from 'sentry/components/pagination'; import SearchBar from 'sentry/components/searchBar'; import SentryDocumentTitle from 'sentry/components/sentryDocumentTitle'; import {IconAdd, IconList} from 'sentry/icons'; import {t} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import {useApiQuery} from 'sentry/utils/queryClient'; import {decodeList, decodeScalar} from 'sentry/utils/queryString'; import useRouteAnalyticsEventNames from 'sentry/utils/routeAnalytics/useRouteAnalyticsEventNames'; import useRouteAnalyticsParams from 'sentry/utils/routeAnalytics/useRouteAnalyticsParams'; import useOrganization from 'sentry/utils/useOrganization'; import useRouter from 'sentry/utils/useRouter'; import { CronsLandingPanel, isValidGuide, isValidPlatform, } from './components/cronsLandingPanel'; import {NewMonitorButton} from './components/newMonitorButton'; import {OverviewTimeline} from './components/overviewTimeline'; import {OwnerFilter} from './components/ownerFilter'; import type {Monitor} from './types'; import {makeMonitorListQueryKey} from './utils'; const CronsListPageHeader = HookOrDefault({ hookName: 'component:crons-list-page-header', }); export default function Monitors() { const organization = useOrganization(); const router = useRouter(); const platform = decodeScalar(router.location.query?.platform) ?? null; const guide = decodeScalar(router.location.query?.guide); const queryKey = makeMonitorListQueryKey(organization, router.location.query); const { data: monitorList, getResponseHeader: monitorListHeaders, isLoading, refetch, } = useApiQuery(queryKey, { staleTime: 0, }); useRouteAnalyticsEventNames('monitors.page_viewed', 'Monitors: Page Viewed'); useRouteAnalyticsParams({empty_state: !monitorList || monitorList.length === 0}); const monitorListPageLinks = monitorListHeaders?.('Link'); const handleSearch = (query: string) => { const currentQuery = {...(router.location.query ?? {}), cursor: undefined}; router.push({ pathname: location.pathname, query: normalizeDateTimeParams({...currentQuery, query}), }); }; const showAddMonitor = !isValidPlatform(platform) || !isValidGuide(guide); return ( {t('Cron Monitors')} {showAddMonitor && ( }> {t('Add Monitor')} )} router.replace({ ...router.location, query: {...router.location.query, owner}, }) } /> {isLoading ? ( ) : monitorList?.length ? ( {monitorListPageLinks && } ) : ( )} ); } const Filters = styled('div')` display: flex; gap: ${space(1.5)}; margin-bottom: ${space(2)}; > :last-child { flex-grow: 1; } `;