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;
`;