import {Fragment, useCallback, useMemo, useState} from 'react'; import styled from '@emotion/styled'; import omit from 'lodash/omit'; import Feature from 'sentry/components/acl/feature'; import {Button} from 'sentry/components/button'; import HookOrDefault from 'sentry/components/hookOrDefault'; import { type Field, MetricSamplesTable, SearchableMetricSamplesTable, } from 'sentry/components/metrics/metricSamplesTable'; import {TabList, TabPanels, Tabs} from 'sentry/components/tabs'; import {Tooltip} from 'sentry/components/tooltip'; import {t} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import type {MRI} from 'sentry/types/metrics'; import {trackAnalytics} from 'sentry/utils/analytics'; import {isCustomMetric} from 'sentry/utils/metrics'; import type { FocusedMetricsSeries, MetricsQueryWidget, MetricsWidget, } from 'sentry/utils/metrics/types'; import {MetricExpressionType} from 'sentry/utils/metrics/types'; import type {MetricsSamplesResults} from 'sentry/utils/metrics/useMetricsSamples'; import {useLocation} from 'sentry/utils/useLocation'; import useOrganization from 'sentry/utils/useOrganization'; import {CodeLocations} from 'sentry/views/metrics/codeLocations'; import type {FocusAreaProps} from 'sentry/views/metrics/context'; import {useMetricsContext} from 'sentry/views/metrics/context'; import {extendQueryWithGroupBys} from 'sentry/views/metrics/utils'; import {generateTracesRouteWithQuery} from 'sentry/views/performance/traces/utils'; enum Tab { SAMPLES = 'samples', CODE_LOCATIONS = 'codeLocations', } export function WidgetDetails() { const { selectedWidgetIndex, widgets, focusArea, setHighlightedSampleId, setMetricsSamples, } = useMetricsContext(); const selectedWidget = widgets[selectedWidgetIndex] as MetricsWidget | undefined; const handleSampleRowHover = useCallback( (sampleId?: string) => { setHighlightedSampleId(sampleId); }, [setHighlightedSampleId] ); // TODO(aknaus): better fallback if (selectedWidget?.type === MetricExpressionType.EQUATION) { ; } const {mri, op, query, focusedSeries} = selectedWidget as MetricsQueryWidget; return ( ); } interface MetricDetailsProps { focusArea?: FocusAreaProps; focusedSeries?: FocusedMetricsSeries[]; mri?: MRI; onRowHover?: (sampleId?: string) => void; op?: string; query?: string; setMetricsSamples?: React.Dispatch< React.SetStateAction['data'] | undefined> >; } export function MetricDetails({ mri, op, query, focusedSeries, onRowHover, focusArea, setMetricsSamples, }: MetricDetailsProps) { const location = useLocation(); const organization = useOrganization(); const [selectedTab, setSelectedTab] = useState(Tab.SAMPLES); const isCodeLocationsDisabled = mri && !isCustomMetric({mri}); if (isCodeLocationsDisabled && selectedTab === Tab.CODE_LOCATIONS) { setSelectedTab(Tab.SAMPLES); } const queryWithFocusedSeries = useMemo( () => focusedSeries && extendQueryWithGroupBys( query || '', focusedSeries.map(s => s.groupBy) ), [focusedSeries, query] ); const handleTabChange = useCallback( (tab: Tab) => { if (tab === Tab.CODE_LOCATIONS) { trackAnalytics('ddm.code-locations', { organization, }); } setSelectedTab(tab); }, [organization] ); const tracesTarget = generateTracesRouteWithQuery({ orgSlug: organization.slug, metric: op && mri ? { metricsOp: op, mri, metricsQuery: queryWithFocusedSeries, } : undefined, query: omit(location.query, ['widgets', 'interval']), }); return ( {t('Sampled Events')} {t('Code Location')} {organization.features.includes('metrics-samples-list-search') ? ( ) : ( )} ); } const MetricSampleTableWrapper = HookOrDefault({ hookName: 'component:ddm-metrics-samples-list', defaultComponent: ({children}) => {children}, }); const TrayWrapper = styled('div')` padding-top: ${space(4)}; display: grid; grid-template-rows: auto auto 1fr; `; const ContentWrapper = styled('div')` position: relative; padding-top: ${space(2)}; `; const OpenInTracesWrapper = styled('div')` display: flex; justify-content: flex-end; `;