123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404 |
- import {
- createContext,
- useCallback,
- useContext,
- useEffect,
- useMemo,
- useState,
- } from 'react';
- import * as Sentry from '@sentry/react';
- import isEqual from 'lodash/isEqual';
- import type {Field} from 'sentry/components/metrics/metricSamplesTable';
- import {useInstantRef, useUpdateQuery} from 'sentry/utils/metrics';
- import {
- emptyMetricsFormulaWidget,
- emptyMetricsQueryWidget,
- NO_QUERY_ID,
- } from 'sentry/utils/metrics/constants';
- import {MetricExpressionType, type MetricsWidget} from 'sentry/utils/metrics/types';
- import type {MetricsSamplesResults} from 'sentry/utils/metrics/useMetricsSamples';
- import {decodeInteger, decodeScalar} from 'sentry/utils/queryString';
- import useLocationQuery from 'sentry/utils/url/useLocationQuery';
- import {useLocalStorageState} from 'sentry/utils/useLocalStorageState';
- import usePageFilters from 'sentry/utils/usePageFilters';
- import useProjects from 'sentry/utils/useProjects';
- import useRouter from 'sentry/utils/useRouter';
- import type {FocusAreaSelection} from 'sentry/views/metrics/chart/types';
- import {parseMetricWidgetsQueryParam} from 'sentry/views/metrics/utils/parseMetricWidgetsQueryParam';
- import {useStructuralSharing} from 'sentry/views/metrics/utils/useStructuralSharing';
- export type FocusAreaProps = {
- onAdd?: (area: FocusAreaSelection) => void;
- onDraw?: () => void;
- onRemove?: () => void;
- selection?: FocusAreaSelection;
- };
- interface MetricsContextValue {
- addWidget: (type?: MetricExpressionType) => void;
- duplicateWidget: (index: number) => void;
- focusArea: FocusAreaProps;
- hasMetrics: boolean;
- isDefaultQuery: boolean;
- isMultiChartMode: boolean;
- removeWidget: (index: number) => void;
- selectedWidgetIndex: number;
- setDefaultQuery: (query: Record<string, any> | null) => void;
- setHighlightedSampleId: (sample?: string) => void;
- setIsMultiChartMode: (value: boolean) => void;
- setMetricsSamples: React.Dispatch<
- React.SetStateAction<MetricsSamplesResults<Field>['data'] | undefined>
- >;
- setSelectedWidgetIndex: (index: number) => void;
- showQuerySymbols: boolean;
- toggleWidgetVisibility: (index: number) => void;
- updateWidget: (index: number, data: Partial<Omit<MetricsWidget, 'type'>>) => void;
- widgets: MetricsWidget[];
- highlightedSampleId?: string;
- metricsSamples?: MetricsSamplesResults<Field>['data'];
- }
- export const MetricsContext = createContext<MetricsContextValue>({
- addWidget: () => {},
- duplicateWidget: () => {},
- focusArea: {},
- hasMetrics: false,
- highlightedSampleId: undefined,
- isDefaultQuery: false,
- isMultiChartMode: false,
- metricsSamples: [],
- removeWidget: () => {},
- selectedWidgetIndex: 0,
- setDefaultQuery: () => {},
- setHighlightedSampleId: () => {},
- setIsMultiChartMode: () => {},
- setMetricsSamples: () => {},
- setSelectedWidgetIndex: () => {},
- showQuerySymbols: false,
- updateWidget: () => {},
- widgets: [],
- toggleWidgetVisibility: () => {},
- });
- export function useMetricsContext() {
- return useContext(MetricsContext);
- }
- export function useMetricWidgets() {
- const {widgets: urlWidgets} = useLocationQuery({fields: {widgets: decodeScalar}});
- const updateQuery = useUpdateQuery();
- const widgets = useStructuralSharing(
- useMemo<MetricsWidget[]>(() => parseMetricWidgetsQueryParam(urlWidgets), [urlWidgets])
- );
- // We want to have it as a ref, so that we can use it in the setWidget callback
- // without needing to generate a new callback every time the location changes
- const currentWidgetsRef = useInstantRef(widgets);
- const setWidgets = useCallback(
- (newWidgets: React.SetStateAction<MetricsWidget[]>) => {
- const currentWidgets = currentWidgetsRef.current;
- const newData =
- typeof newWidgets === 'function' ? newWidgets(currentWidgets) : newWidgets;
- updateQuery({widgets: JSON.stringify(newData)});
- // We need to update the ref so that the next call to setWidgets in the same render cycle will have the updated value
- currentWidgetsRef.current = newData;
- },
- [updateQuery, currentWidgetsRef]
- );
- const updateWidget = useCallback(
- (index: number, data: Partial<Omit<MetricsWidget, 'type'>>) => {
- setWidgets(currentWidgets => {
- const newWidgets = [...currentWidgets];
- newWidgets[index] = {
- ...currentWidgets[index],
- ...data,
- };
- return newWidgets;
- });
- },
- [setWidgets]
- );
- const duplicateWidget = useCallback(
- (index: number) => {
- setWidgets(currentWidgets => {
- const newWidgets = [...currentWidgets];
- const newWidget = {...currentWidgets[index]};
- newWidget.id = NO_QUERY_ID;
- newWidgets.splice(index + 1, 0, newWidget);
- return newWidgets;
- });
- },
- [setWidgets]
- );
- const addWidget = useCallback(
- (type: MetricExpressionType = MetricExpressionType.QUERY) => {
- const lastIndexOfSameType = currentWidgetsRef.current.findLastIndex(
- w => w.type === type
- );
- if (lastIndexOfSameType > -1) {
- duplicateWidget(lastIndexOfSameType);
- } else {
- setWidgets(currentWidgets => [
- ...currentWidgets,
- type === MetricExpressionType.QUERY
- ? emptyMetricsQueryWidget
- : emptyMetricsFormulaWidget,
- ]);
- }
- },
- [currentWidgetsRef, duplicateWidget, setWidgets]
- );
- const removeWidget = useCallback(
- (index: number) => {
- setWidgets(currentWidgets => {
- let newWidgets = [...currentWidgets];
- newWidgets.splice(index, 1);
- // Ensure that a visible widget remains
- if (!newWidgets.find(w => !w.isHidden)) {
- newWidgets = newWidgets.map(w => ({...w, isHidden: false}));
- }
- return newWidgets;
- });
- },
- [setWidgets]
- );
- return {
- widgets,
- updateWidget,
- addWidget,
- removeWidget,
- duplicateWidget,
- setWidgets,
- };
- }
- const useDefaultQuery = () => {
- const router = useRouter();
- const [defaultQuery, setDefaultQuery] = useLocalStorageState<Record<
- string,
- any
- > | null>('ddm:default-query', null);
- useEffect(() => {
- if (defaultQuery && router.location.query.widgets === undefined) {
- router.replace({...router.location, query: defaultQuery});
- }
- // Only call on page load
- // eslint-disable-next-line react-hooks/exhaustive-deps
- }, []);
- return useMemo(
- () => ({
- defaultQuery,
- setDefaultQuery,
- isDefaultQuery: !!defaultQuery && isEqual(defaultQuery, router.location.query),
- }),
- [defaultQuery, router.location.query, setDefaultQuery]
- );
- };
- function useSelectedProjects() {
- const {selection} = usePageFilters();
- const {projects} = useProjects();
- return useMemo(() => {
- if (selection.projects.length === 0) {
- return projects.filter(project => project.isMember);
- }
- if (selection.projects.includes(-1)) {
- return projects;
- }
- return projects.filter(project => selection.projects.includes(Number(project.id)));
- }, [selection.projects, projects]);
- }
- export function DDMContextProvider({children}: {children: React.ReactNode}) {
- const router = useRouter();
- const updateQuery = useUpdateQuery();
- const {multiChartMode} = useLocationQuery({fields: {multiChartMode: decodeInteger}});
- const isMultiChartMode = multiChartMode === 1;
- const {setDefaultQuery, isDefaultQuery} = useDefaultQuery();
- const [selectedWidgetIndex, setSelectedWidgetIndex] = useState(0);
- const {widgets, updateWidget, addWidget, removeWidget, duplicateWidget, setWidgets} =
- useMetricWidgets();
- const [metricsSamples, setMetricsSamples] = useState<
- MetricsSamplesResults<Field>['data'] | undefined
- >();
- const [highlightedSampleId, setHighlightedSampleId] = useState<string | undefined>();
- const selectedProjects = useSelectedProjects();
- const hasMetrics = useMemo(
- () =>
- selectedProjects.some(
- project =>
- project.hasCustomMetrics || project.hasSessions || project.firstTransactionEvent
- ),
- [selectedProjects]
- );
- const handleSetSelectedWidgetIndex = useCallback(
- (value: number) => {
- if (!isMultiChartMode) {
- return;
- }
- setSelectedWidgetIndex(value);
- },
- [isMultiChartMode]
- );
- const focusAreaSelection = useMemo<FocusAreaSelection | undefined>(
- () => router.location.query.focusArea && JSON.parse(router.location.query.focusArea),
- [router.location.query.focusArea]
- );
- const handleAddFocusArea = useCallback(
- (area: FocusAreaSelection) => {
- if (!area.range.start || !area.range.end) {
- Sentry.metrics.increment('ddm.enhance.range-undefined');
- return;
- }
- Sentry.metrics.increment('ddm.enhance.add');
- handleSetSelectedWidgetIndex(area.widgetIndex);
- updateQuery({focusArea: JSON.stringify(area)}, {replace: true});
- },
- [handleSetSelectedWidgetIndex, updateQuery]
- );
- const handleRemoveFocusArea = useCallback(() => {
- Sentry.metrics.increment('ddm.enhance.remove');
- updateQuery({focusArea: undefined}, {replace: true});
- }, [updateQuery]);
- const focusArea = useMemo<FocusAreaProps>(() => {
- return {
- selection: focusAreaSelection,
- onAdd: handleAddFocusArea,
- onRemove: handleRemoveFocusArea,
- };
- }, [focusAreaSelection, handleAddFocusArea, handleRemoveFocusArea]);
- const handleAddWidget = useCallback(
- (type?: MetricExpressionType) => {
- addWidget(type);
- handleSetSelectedWidgetIndex(widgets.length);
- },
- [addWidget, handleSetSelectedWidgetIndex, widgets.length]
- );
- const handleUpdateWidget = useCallback(
- (index: number, data: Partial<MetricsWidget>) => {
- updateWidget(index, data);
- handleSetSelectedWidgetIndex(index);
- if (index === focusAreaSelection?.widgetIndex) {
- handleRemoveFocusArea();
- }
- },
- [
- updateWidget,
- handleSetSelectedWidgetIndex,
- focusAreaSelection?.widgetIndex,
- handleRemoveFocusArea,
- ]
- );
- const handleDuplicate = useCallback(
- (index: number) => {
- duplicateWidget(index);
- handleSetSelectedWidgetIndex(index + 1);
- },
- [duplicateWidget, handleSetSelectedWidgetIndex]
- );
- const handleSetIsMultiChartMode = useCallback(
- (value: boolean) => {
- updateQuery({multiChartMode: value ? 1 : 0}, {replace: true});
- updateWidget(0, {focusedSeries: undefined});
- const firstVisibleWidgetIndex = widgets.findIndex(w => !w.isHidden);
- setSelectedWidgetIndex(firstVisibleWidgetIndex);
- },
- [updateQuery, updateWidget, widgets]
- );
- const toggleWidgetVisibility = useCallback(
- (index: number) => {
- if (index === selectedWidgetIndex) {
- const firstVisibleWidgetIndex = widgets.findIndex(w => !w.isHidden);
- setSelectedWidgetIndex(firstVisibleWidgetIndex);
- }
- if (!isMultiChartMode) {
- // Reset the focused series when hiding a widget
- setWidgets(currentWidgets => {
- return currentWidgets.map(w => ({...w, focusedSeries: undefined}));
- });
- }
- updateWidget(index, {isHidden: !widgets[index].isHidden});
- },
- [isMultiChartMode, selectedWidgetIndex, setWidgets, updateWidget, widgets]
- );
- const selectedWidget = widgets[selectedWidgetIndex];
- const isSelectionValid = selectedWidget && !selectedWidget.isHidden;
- const contextValue = useMemo<MetricsContextValue>(
- () => ({
- addWidget: handleAddWidget,
- selectedWidgetIndex: isSelectionValid
- ? selectedWidgetIndex
- : widgets.findIndex(w => !w.isHidden),
- setSelectedWidgetIndex: handleSetSelectedWidgetIndex,
- updateWidget: handleUpdateWidget,
- removeWidget,
- duplicateWidget: handleDuplicate,
- widgets,
- hasMetrics,
- focusArea,
- setDefaultQuery,
- isDefaultQuery,
- showQuerySymbols: widgets.length > 1,
- highlightedSampleId,
- setHighlightedSampleId,
- isMultiChartMode: isMultiChartMode,
- setIsMultiChartMode: handleSetIsMultiChartMode,
- metricsSamples,
- setMetricsSamples,
- toggleWidgetVisibility,
- }),
- [
- handleAddWidget,
- isSelectionValid,
- selectedWidgetIndex,
- widgets,
- handleSetSelectedWidgetIndex,
- handleUpdateWidget,
- removeWidget,
- handleDuplicate,
- hasMetrics,
- focusArea,
- setDefaultQuery,
- isDefaultQuery,
- highlightedSampleId,
- isMultiChartMode,
- handleSetIsMultiChartMode,
- metricsSamples,
- toggleWidgetVisibility,
- ]
- );
- return (
- <MetricsContext.Provider value={contextValue}>{children}</MetricsContext.Provider>
- );
- }
|