import styled from '@emotion/styled'; import type {Location} from 'history'; import {useAnalyticsArea} from 'sentry/components/analyticsArea'; import ProjectBadge from 'sentry/components/idBadge/projectBadge'; import Link from 'sentry/components/links/link'; import {generateTraceTarget} from 'sentry/components/quickTrace/utils'; import {t, tn} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import type {Event} from 'sentry/types/event'; import {trackAnalytics} from 'sentry/utils/analytics'; import {useLocation} from 'sentry/utils/useLocation'; import useOrganization from 'sentry/utils/useOrganization'; import useProjects from 'sentry/utils/useProjects'; import type {TimelineEvent} from './useTraceTimelineEvents'; interface TraceTimelineTooltipProps { event: Event; timelineEvents: TimelineEvent[]; } export function TraceTimelineTooltip({event, timelineEvents}: TraceTimelineTooltipProps) { const organization = useOrganization(); const location = useLocation(); const area = useAnalyticsArea(); // TODO: should handling of current event + other events look different if (timelineEvents.length === 1 && timelineEvents[0].id === event.id) { return {t('You are here')}; } const filteredTimelineEvents = timelineEvents.filter( timelineEvent => timelineEvent.id !== event.id ); const displayYouAreHere = filteredTimelineEvents.length !== timelineEvents.length; const hasTitle = filteredTimelineEvents.length > 1 || displayYouAreHere; return ( {displayYouAreHere && {t('You are here')}} {hasTitle && {t('Around the same time')}} {filteredTimelineEvents.slice(0, 3).map(timelineEvent => { return ( ); })} {filteredTimelineEvents.length > 3 && ( { if (area.startsWith('issue_details')) { // Track this event for backwards compatibility. TODO: remove after issues team dashboards/queries are migrated trackAnalytics( 'issue_details.issue_tab.trace_timeline_more_events_clicked', { organization, num_hidden: filteredTimelineEvents.length - 3, } ); } trackAnalytics('trace_timeline_more_events_clicked', { organization, num_hidden: filteredTimelineEvents.length - 3, area, }); }} > {tn( 'View %s more event', 'View %s more events', filteredTimelineEvents.length - 3 )} )} ); } interface EventItemProps { location: Location; timelineEvent: TimelineEvent; } function EventItem({timelineEvent, location}: EventItemProps) { const organization = useOrganization(); const {projects} = useProjects({ slugs: [timelineEvent.project], orgId: organization.slug, }); const project = projects.find(p => p.slug === timelineEvent.project); const area = useAnalyticsArea(); return ( { if (area.includes('issue_details')) { // Track this event for backwards compatibility. TODO: remove after issues team dashboards/queries are migrated trackAnalytics('issue_details.issue_tab.trace_timeline_clicked', { organization, event_id: timelineEvent.id, group_id: `${timelineEvent['issue.id']}`, }); } trackAnalytics('trace_timeline_clicked', { organization, event_id: timelineEvent.id, group_id: `${timelineEvent['issue.id']}`, area, }); }} > {project && } {timelineEvent.title} {timelineEvent.transaction ? timelineEvent.transaction : 'stack.function' in timelineEvent ? timelineEvent['stack.function'].at(-1) : null} ); } const UnstyledUnorderedList = styled('div')` display: flex; flex-direction: column; text-align: left; width: 220px; `; const EventItemsWrapper = styled('div')<{hasTitle: boolean}>` display: flex; flex-direction: column; padding: ${p => space(p.hasTitle ? 1 : 0.5)} ${space(0.5)} ${space(0.5)} ${space(0.5)}; `; const EventItemsTitle = styled('div')` padding-left: ${space(1)}; text-transform: uppercase; font-size: ${p => p.theme.fontSizeExtraSmall}; font-weight: ${p => p.theme.fontWeightBold}; color: ${p => p.theme.subText}; `; const YouAreHere = styled('div')` padding: ${space(1)} ${space(2)}; text-align: center; font-size: ${p => p.theme.fontSizeMedium}; `; const YouAreHereItem = styled('div')` padding: ${space(1)} ${space(2)}; text-align: center; border-bottom: 1px solid ${p => p.theme.innerBorder}; font-size: ${p => p.theme.fontSizeMedium}; `; const EventItemRoot = styled(Link)` display: grid; grid-template-columns: max-content auto; color: ${p => p.theme.textColor}; gap: ${space(1)}; width: 100%; padding: ${space(1)} ${space(1)} ${space(0.5)} ${space(1)}; border-radius: ${p => p.theme.borderRadius}; font-size: ${p => p.theme.fontSizeSmall}; &:hover { background-color: ${p => p.theme.surface200}; color: ${p => p.theme.textColor}; } `; const EventTitleWrapper = styled('div')` width: 100%; overflow: hidden; line-height: 1.2; `; const EventTitle = styled('div')` ${p => p.theme.overflowEllipsis}; font-weight: ${p => p.theme.fontWeightBold}; `; const EventDescription = styled('div')` ${p => p.theme.overflowEllipsis}; direction: rtl; `; const TraceItem = styled('div')` padding: ${space(1)} ${space(1.5)}; border-radius: ${p => p.theme.borderRadius}; border-top: 1px solid ${p => p.theme.innerBorder}; `;