import {Component} from 'react'; import styled from '@emotion/styled'; import {Location} from 'history'; import moment from 'moment-timezone'; import {Button} from 'sentry/components/button'; import DateTime from 'sentry/components/dateTime'; import {DataSection} from 'sentry/components/events/styles'; import FileSize from 'sentry/components/fileSize'; import GlobalAppStoreConnectUpdateAlert from 'sentry/components/globalAppStoreConnectUpdateAlert'; import ExternalLink from 'sentry/components/links/externalLink'; import Link from 'sentry/components/links/link'; import NavigationButtonGroup from 'sentry/components/navigationButtonGroup'; import {Tooltip} from 'sentry/components/tooltip'; import {IconPlay, IconWarning} from 'sentry/icons'; import {t} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import {Group, Organization, Project} from 'sentry/types'; import {Event} from 'sentry/types/event'; import trackAdvancedAnalyticsEvent from 'sentry/utils/analytics/trackAdvancedAnalyticsEvent'; import {shouldUse24Hours} from 'sentry/utils/dates'; import getDynamicText from 'sentry/utils/getDynamicText'; import {TraceLink} from 'sentry/views/issueDetails/quickTrace/traceLink'; import EventCreatedTooltip from './eventCreatedTooltip'; import QuickTrace from './quickTrace'; type Props = { event: Event; group: Group; location: Location; organization: Organization; project: Project; hasReplay?: boolean; }; class GroupEventToolbar extends Component { shouldComponentUpdate(nextProps: Props) { return this.props.event.id !== nextProps.event.id; } handleNavigationClick(button: string) { trackAdvancedAnalyticsEvent('issue_details.event_navigation_clicked', { organization: this.props.organization, project_id: parseInt(this.props.project.id, 10), button, }); } render() { const is24Hours = shouldUse24Hours(); const evt = this.props.event; const {group, organization, location, project, hasReplay} = this.props; const groupId = group.id; const isReplayEnabled = organization.features.includes('session-replay-ui'); const baseEventsPath = `/organizations/${organization.slug}/issues/${groupId}/events/`; // TODO: possible to define this as a route in react-router, but without a corresponding // React component? const jsonUrl = `/organizations/${organization.slug}/issues/${groupId}/events/${evt.id}/json/`; const latencyThreshold = 30 * 60 * 1000; // 30 minutes const isOverLatencyThreshold = evt.dateReceived && Math.abs(+moment(evt.dateReceived) - +moment(evt.dateCreated)) > latencyThreshold; return (
{t('Event ID')}{' '} {evt.eventID} trackAdvancedAnalyticsEvent('issue_details.event_json_clicked', { organization, group_id: parseInt(`${evt.groupID}`, 10), }) } > {'JSON'} () } showUnderline disableForVisualTest > {isOverLatencyThreshold && }
{hasReplay && isReplayEnabled ? ( ) : null} this.handleNavigationClick('oldest')} onOlderClick={() => this.handleNavigationClick('older')} onNewerClick={() => this.handleNavigationClick('newer')} onNewestClick={() => this.handleNavigationClick('newest')} size="sm" />
); } } const Wrapper = styled(DataSection)` @media (max-width: 767px) { display: none; } `; const HeadingAndNavWrapper = styled('div')` display: flex; position: relative; flex-direction: row; justify-content: space-between; align-items: flex-start; gap: ${space(3)}; `; const EventIdLink = styled(Link)` font-weight: normal; `; const Heading = styled('h4')` line-height: 1.3; margin: 0; font-size: ${p => p.theme.fontSizeLarge}; `; const StyledIconWarning = styled(IconWarning)` margin-left: ${space(0.5)}; position: relative; top: ${space(0.25)}; `; const StyledDateTime = styled(DateTime)` color: ${p => p.theme.subText}; `; const StyledGlobalAppStoreConnectUpdateAlert = styled(GlobalAppStoreConnectUpdateAlert)` margin: ${space(0.5)} 0; `; const LinkContainer = styled('span')` margin-left: ${space(1)}; padding-left: ${space(1)}; position: relative; font-weight: normal; &:before { display: block; position: absolute; content: ''; left: 0; top: 2px; height: 14px; border-left: 1px solid ${p => p.theme.border}; } `; const NavigationContainer = styled('div')` display: flex; align-items: center; justify-content: flex-end; gap: 0 ${space(1)}; `; export default GroupEventToolbar;