import {Fragment, useEffect} from 'react'; import {css} from '@emotion/react'; import styled from '@emotion/styled'; import isEqual from 'lodash/isEqual'; import ArchivedBox from 'sentry/components/archivedBox'; import GroupEventDetailsLoadingError from 'sentry/components/errors/groupEventDetailsLoadingError'; import {withMeta} from 'sentry/components/events/meta/metaProxy'; import HookOrDefault from 'sentry/components/hookOrDefault'; import * as Layout from 'sentry/components/layouts/thirds'; import LoadingIndicator from 'sentry/components/loadingIndicator'; import {TransactionProfileIdProvider} from 'sentry/components/profiling/transactionProfileIdProvider'; import ResolutionBox from 'sentry/components/resolutionBox'; import useSentryAppComponentsData from 'sentry/stores/useSentryAppComponentsData'; import {space} from 'sentry/styles/space'; import type {Event} from 'sentry/types/event'; import type {Group, GroupActivityReprocess, GroupReprocessing} from 'sentry/types/group'; import type {RouteComponentProps} from 'sentry/types/legacyReactRouter'; import type {Organization} from 'sentry/types/organization'; import type {Project} from 'sentry/types/project'; import {defined} from 'sentry/utils'; import {browserHistory} from 'sentry/utils/browserHistory'; import {getConfigForIssueType} from 'sentry/utils/issueTypeConfig'; import {VisuallyCompleteWithData} from 'sentry/utils/performanceForSentry'; import normalizeUrl from 'sentry/utils/url/normalizeUrl'; import usePrevious from 'sentry/utils/usePrevious'; import {useSyncedLocalStorageState} from 'sentry/utils/useSyncedLocalStorageState'; import GroupEventDetailsContent from 'sentry/views/issueDetails/groupEventDetails/groupEventDetailsContent'; import GroupEventHeader from 'sentry/views/issueDetails/groupEventHeader'; import GroupSidebar from 'sentry/views/issueDetails/groupSidebar'; import StreamlinedSidebar from 'sentry/views/issueDetails/streamline/sidebar'; import ReprocessingProgress from '../reprocessingProgress'; import { getEventEnvironment, getGroupMostRecentActivity, ReprocessingStatus, useEnvironmentsFromUrl, useHasStreamlinedUI, } from '../utils'; const EscalatingIssuesFeedback = HookOrDefault({ hookName: 'component:escalating-issues-banner-feedback', }); export interface GroupEventDetailsProps extends RouteComponentProps<{groupId: string; eventId?: string}, {}> { eventError: boolean; group: Group; groupReprocessingStatus: ReprocessingStatus; loadingEvent: boolean; onRetry: () => void; organization: Organization; project: Project; event?: Event; } function GroupEventDetails(props: GroupEventDetailsProps) { const { group, project, organization, location, event, groupReprocessingStatus, loadingEvent, onRetry, eventError, params, } = props; const projectId = project.id; const environments = useEnvironmentsFromUrl(); const prevEnvironment = usePrevious(environments); const prevEvent = usePrevious(event); const hasStreamlinedUI = useHasStreamlinedUI(); const [sidebarOpen, _] = useSyncedLocalStorageState('issue-details-sidebar-open', true); // load the data useSentryAppComponentsData({projectId}); // If environments are being actively changed and will no longer contain the // current event's environment, redirect to latest useEffect(() => { const environmentsHaveChanged = !isEqual(prevEnvironment, environments); // If environments are being actively changed and will no longer contain the // current event's environment, redirect to latest if ( environmentsHaveChanged && prevEvent && params.eventId && !['latest', 'oldest'].includes(params.eventId) ) { const shouldRedirect = environments.length > 0 && !environments.find(env => env === getEventEnvironment(prevEvent as Event)); if (shouldRedirect) { browserHistory.replace( normalizeUrl({ pathname: `/organizations/${organization.slug}/issues/${params.groupId}/`, query: location.query, }) ); return; } } }, [ prevEnvironment, environments, location.query, organization.slug, params, prevEvent, ]); const renderGroupStatusBanner = () => { if (group.status === 'ignored') { return ( ); } if (group.status === 'resolved') { return ( ); } return null; }; const renderContent = () => { if (loadingEvent) { return ; } if (eventError) { return ( ); } return ( ); }; const eventWithMeta = withMeta(event); const issueTypeConfig = getConfigForIssueType(group, project); const MainLayoutComponent = hasStreamlinedUI ? GroupContent : StyledLayoutMain; return ( {groupReprocessingStatus === ReprocessingStatus.REPROCESSING ? ( ) : ( {!hasStreamlinedUI && renderGroupStatusBanner()} {eventWithMeta && issueTypeConfig.stats.enabled && !hasStreamlinedUI && ( )} {renderContent()} {hasStreamlinedUI ? ( sidebarOpen ? ( ) : null ) : ( )} )} ); } const StyledLayoutBody = styled(Layout.Body)<{ hasStreamlinedUi: boolean; sidebarOpen: boolean; }>` /* Makes the borders align correctly */ padding: 0 !important; @media (min-width: ${p => p.theme.breakpoints.large}) { align-content: stretch; } ${p => p.hasStreamlinedUi && css` min-height: 100vh; @media (min-width: ${p.theme.breakpoints.large}) { gap: ${space(1.5)}; display: ${p.sidebarOpen ? 'grid' : 'block'}; } `} `; const GroupStatusBannerWrapper = styled('div')` margin-bottom: ${space(2)}; `; const StyledLayoutMain = styled(Layout.Main)` padding-top: ${space(2)}; @media (max-width: ${p => p.theme.breakpoints.medium}) { padding-top: ${space(1)}; } @media (min-width: ${p => p.theme.breakpoints.large}) { border-right: 1px solid ${p => p.theme.border}; padding-right: 0; } `; const GroupContent = styled(Layout.Main)` background: ${p => p.theme.backgroundSecondary}; display: flex; flex-direction: column; padding: ${space(1.5)}; gap: ${space(1.5)}; @media (min-width: ${p => p.theme.breakpoints.large}) { border-right: 1px solid ${p => p.theme.translucentBorder}; } @media (max-width: ${p => p.theme.breakpoints.large}) { border-bottom-width: 1px solid ${p => p.theme.translucentBorder}; } `; const StyledLayoutSide = styled(Layout.Side)<{hasStreamlinedUi: boolean}>` ${p => p.hasStreamlinedUi ? css` padding: ${space(1.5)} ${space(2)} ${space(3)}; ` : css` padding: ${space(3)} ${space(2)} ${space(3)}; @media (min-width: ${p.theme.breakpoints.large}) { padding-right: ${space(4)}; } `} @media (min-width: ${p => p.theme.breakpoints.large}) { padding-left: 0; } `; export default GroupEventDetails;