import {useState} from 'react'; import {Link} from 'react-router'; import {css} from '@emotion/react'; import styled from '@emotion/styled'; import {Button} from 'sentry/components/button'; import {openConfirmModal} from 'sentry/components/confirm'; import {DropdownMenu} from 'sentry/components/dropdownMenu'; import ProjectBadge from 'sentry/components/idBadge/projectBadge'; import Tag from 'sentry/components/tag'; import {IconEllipsis} from 'sentry/icons'; import {t, tct} from 'sentry/locale'; import {fadeIn} from 'sentry/styles/animations'; import {space} from 'sentry/styles/space'; import type {ObjectStatus} from 'sentry/types'; import {trimSlug} from 'sentry/utils/trimSlug'; import useOrganization from 'sentry/utils/useOrganization'; import {normalizeUrl} from 'sentry/utils/withDomainRequired'; import MonitorEnvironmentLabel from 'sentry/views/monitors/components/overviewTimeline/monitorEnvironmentLabel'; import {StatusToggleButton} from 'sentry/views/monitors/components/statusToggleButton'; import type {Monitor} from 'sentry/views/monitors/types'; import {scheduleAsText} from 'sentry/views/monitors/utils'; import type {CheckInTimelineProps} from './checkInTimeline'; import {CheckInTimeline} from './checkInTimeline'; import {TimelinePlaceholder} from './timelinePlaceholder'; import type {MonitorBucket} from './types'; interface Props extends Omit { monitor: Monitor; bucketedData?: MonitorBucket[]; onDeleteEnvironment?: (env: string) => Promise; onToggleMuteEnvironment?: (env: string, isMuted: boolean) => Promise; onToggleStatus?: (monitor: Monitor, status: ObjectStatus) => Promise; /** * Whether only one monitor is being rendered in a larger view with this component * turns off things like zebra striping, hover effect, and showing monitor name */ singleMonitorView?: boolean; } const MAX_SHOWN_ENVIRONMENTS = 4; export function TimelineTableRow({ monitor, bucketedData, singleMonitorView, onDeleteEnvironment, onToggleMuteEnvironment, onToggleStatus, ...timelineProps }: Props) { const organization = useOrganization(); const [isExpanded, setExpanded] = useState( monitor.environments.length <= MAX_SHOWN_ENVIRONMENTS ); const environments = isExpanded ? monitor.environments : monitor.environments.slice(0, MAX_SHOWN_ENVIRONMENTS); const isDisabled = monitor.status === 'disabled'; const monitorDetails = singleMonitorView ? null : ( {monitor.name} {isDisabled && {t('Disabled')}} {scheduleAsText(monitor.config)} {trimSlug(monitor.project.slug)} {onToggleStatus && ( onToggleStatus(monitor, status)} /> )} ); const environmentActionCreators = [ (env: string) => ({ label: t('View Environment'), key: 'view', to: normalizeUrl( `/organizations/${organization.slug}/crons/${monitor.project.slug}/${monitor.slug}/?environment=${env}` ), }), ...(onToggleMuteEnvironment ? [ (env: string, isMuted: boolean) => ({ label: isMuted ? t('Unmute Environment') : t('Mute Environment'), key: 'mute', onAction: () => onToggleMuteEnvironment(env, !isMuted), }), ] : []), ...(onDeleteEnvironment ? [ (env: string) => ({ label: t('Delete Environment'), key: 'delete', onAction: () => { openConfirmModal({ onConfirm: () => onDeleteEnvironment(env), header: t('Delete Environment?'), message: tct( 'Are you sure you want to permanently delete the "[envName]" environment?', {envName: env} ), confirmText: t('Delete'), priority: 'danger', }); }, }), ] : []), ]; return ( {monitorDetails} {environments.map(env => { const {name, isMuted} = env; return ( ( } /> )} items={environmentActionCreators.map(actionCreator => actionCreator(name, isMuted) )} /> ); })} {!isExpanded && ( )} {environments.map(({name}) => { return ( {!bucketedData ? ( ) : ( )} ); })} ); } const DetailsLink = styled(Link)` display: block; padding: ${space(3)}; color: ${p => p.theme.textColor}; `; const DetailsArea = styled('div')` border-right: 1px solid ${p => p.theme.border}; border-radius: 0; position: relative; `; const DetailsActions = styled('div')` position: absolute; top: 0; right: 0; display: none; align-items: center; /* Align to the center of the heading text */ height: calc(${p => p.theme.fontSizeLarge} * ${p => p.theme.text.lineHeightHeading}); margin: ${space(3)}; `; const DetailsHeadline = styled('div')` display: grid; gap: ${space(1)}; /* We always leave at least enough room for the status toggle button */ grid-template-columns: 1fr minmax(30px, max-content); `; const ProjectScheduleDetails = styled('div')` display: flex; gap: ${space(1)}; flex-wrap: wrap; `; const ProjectDetails = styled('div')` display: flex; gap: ${space(0.5)}; `; const Name = styled('h3')` font-size: ${p => p.theme.fontSizeLarge}; margin-bottom: ${space(0.25)}; word-break: break-word; `; const DetailsText = styled('small')` color: ${p => p.theme.subText}; font-size: ${p => p.theme.fontSizeSmall}; `; interface TimelineRowProps { isDisabled?: boolean; singleMonitorView?: boolean; } const TimelineRow = styled('div')` grid-column: 1/-1; display: grid; grid-template-columns: subgrid; ${p => !p.singleMonitorView && css` transition: background 50ms ease-in-out; &:nth-child(odd) { background: ${p.theme.backgroundSecondary}; } &:hover { background: ${p.theme.backgroundTertiary}; } `} /* Show detail actions on hover */ &:hover ${DetailsActions} { display: flex; } /* Hide trailing items on hover */ &:hover ${DetailsHeadline} ${Tag} { visibility: hidden; } /* Disabled monitors become more opaque */ --disabled-opacity: ${p => (p.isDisabled ? '0.6' : 'unset')}; &:last-child { border-bottom-left-radius: ${p => p.theme.borderRadius}; border-bottom-right-radius: ${p => p.theme.borderRadius}; } `; const MonitorEnvContainer = styled('div')` display: flex; padding: ${space(3)} ${space(2)}; gap: ${space(4)}; flex-direction: column; border-right: 1px solid ${p => p.theme.innerBorder}; text-align: right; `; const EnvActionButton = styled(Button)` padding: ${space(0.5)} ${space(1)}; display: none; `; const EnvRow = styled('div')` &:hover ${EnvActionButton} { display: block; } display: flex; gap: ${space(0.5)}; justify-content: space-between; align-items: center; height: calc(${p => p.theme.fontSizeLarge} * ${p => p.theme.text.lineHeightHeading}); `; const TimelineContainer = styled('div')` display: flex; padding: ${space(3)} 0; flex-direction: column; gap: ${space(4)}; contain: content; `; const TimelineEnvOuterContainer = styled('div')` position: relative; height: calc(${p => p.theme.fontSizeLarge} * ${p => p.theme.text.lineHeightHeading}); opacity: var(--disabled-opacity); `; const TimelineEnvContainer = styled('div')` position: absolute; inset: 0; opacity: 0; animation: ${fadeIn} 1.5s ease-out forwards; contain: content; `;