123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320 |
- import {
- createContext,
- useCallback,
- useContext,
- useEffect,
- useMemo,
- useState,
- } from 'react';
- import * as Sentry from '@sentry/react';
- import isEqual from 'lodash/isEqual';
- import {MRI} from 'sentry/types';
- import {
- getAbsoluteDateTimeRange,
- getDefaultMetricDisplayType,
- MetricDisplayType,
- MetricWidgetQueryParams,
- useInstantRef,
- useUpdateQuery,
- } from 'sentry/utils/metrics';
- import {useMetricsMeta} from 'sentry/utils/metrics/useMetricsMeta';
- import {decodeList} from 'sentry/utils/queryString';
- import {useLocalStorageState} from 'sentry/utils/useLocalStorageState';
- import usePageFilters from 'sentry/utils/usePageFilters';
- import useRouter from 'sentry/utils/useRouter';
- import {FocusArea} from 'sentry/views/ddm/chartBrush';
- import {DEFAULT_SORT_STATE} from 'sentry/views/ddm/constants';
- import {useStructuralSharing} from 'sentry/views/ddm/useStructuralSharing';
- interface DDMContextValue {
- addFocusArea: (area: FocusArea) => void;
- addWidget: () => void;
- addWidgets: (widgets: Partial<MetricWidgetQueryParams>[]) => void;
- duplicateWidget: (index: number) => void;
- focusArea: FocusArea | null;
- isDefaultQuery: boolean;
- isLoading: boolean;
- metricsMeta: ReturnType<typeof useMetricsMeta>['data'];
- removeFocusArea: () => void;
- removeWidget: (index: number) => void;
- selectedWidgetIndex: number;
- setDefaultQuery: (query: Record<string, any> | null) => void;
- setSelectedWidgetIndex: (index: number) => void;
- updateWidget: (index: number, data: Partial<MetricWidgetQueryParams>) => void;
- widgets: MetricWidgetQueryParams[];
- }
- export const DDMContext = createContext<DDMContextValue>({
- addFocusArea: () => {},
- addWidget: () => {},
- addWidgets: () => {},
- duplicateWidget: () => {},
- focusArea: null,
- isDefaultQuery: false,
- isLoading: false,
- metricsMeta: [],
- removeFocusArea: () => {},
- removeWidget: () => {},
- selectedWidgetIndex: 0,
- setDefaultQuery: () => {},
- setSelectedWidgetIndex: () => {},
- updateWidget: () => {},
- widgets: [],
- });
- export function useDDMContext() {
- return useContext(DDMContext);
- }
- const emptyWidget: MetricWidgetQueryParams = {
- mri: 'd:transactions/duration@millisecond' satisfies MRI,
- op: 'avg',
- query: '',
- groupBy: [],
- sort: DEFAULT_SORT_STATE,
- displayType: MetricDisplayType.LINE,
- title: undefined,
- };
- export function useMetricWidgets() {
- const router = useRouter();
- const updateQuery = useUpdateQuery();
- const widgets = useStructuralSharing(
- useMemo<MetricWidgetQueryParams[]>(() => {
- const currentWidgets = JSON.parse(
- router.location.query.widgets ?? JSON.stringify([emptyWidget])
- );
- return currentWidgets.map((widget: MetricWidgetQueryParams) => {
- return {
- mri: widget.mri,
- op: widget.op,
- query: widget.query,
- groupBy: decodeList(widget.groupBy),
- displayType:
- widget.displayType ?? getDefaultMetricDisplayType(widget.mri, widget.op),
- focusedSeries: widget.focusedSeries,
- showSummaryTable: widget.showSummaryTable ?? true, // temporary default
- powerUserMode: widget.powerUserMode,
- sort: widget.sort ?? DEFAULT_SORT_STATE,
- title: widget.title,
- };
- });
- }, [router.location.query.widgets])
- );
- // 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<MetricWidgetQueryParams[]>) => {
- const currentWidgets = currentWidgetsRef.current;
- updateQuery({
- widgets: JSON.stringify(
- typeof newWidgets === 'function' ? newWidgets(currentWidgets) : newWidgets
- ),
- });
- },
- [updateQuery, currentWidgetsRef]
- );
- const updateWidget = useCallback(
- (index: number, data: Partial<MetricWidgetQueryParams>) => {
- setWidgets(currentWidgets => {
- const newWidgets = [...currentWidgets];
- newWidgets[index] = {...currentWidgets[index], ...data};
- return newWidgets;
- });
- },
- [setWidgets]
- );
- const addWidget = useCallback(() => {
- setWidgets(currentWidgets => [...currentWidgets, emptyWidget]);
- }, [setWidgets]);
- const addWidgets = useCallback(
- (newWidgets: Partial<MetricWidgetQueryParams>[]) => {
- const widgetsCopy = [...widgets].filter(widget => !!widget.mri);
- widgetsCopy.push(...newWidgets.map(widget => ({...emptyWidget, ...widget})));
- setWidgets(widgetsCopy);
- },
- [widgets, setWidgets]
- );
- const removeWidget = useCallback(
- (index: number) => {
- setWidgets(currentWidgets => {
- const newWidgets = [...currentWidgets];
- newWidgets.splice(index, 1);
- return newWidgets;
- });
- },
- [setWidgets]
- );
- const duplicateWidget = useCallback(
- (index: number) => {
- setWidgets(currentWidgets => {
- const newWidgets = [...currentWidgets];
- newWidgets.splice(index, 0, currentWidgets[index]);
- return newWidgets;
- });
- },
- [setWidgets]
- );
- return {
- widgets,
- updateWidget,
- addWidget,
- addWidgets,
- removeWidget,
- duplicateWidget,
- };
- }
- const useDefaultQuery = () => {
- const router = useRouter();
- const [defaultQuery, setDefaultQuery] = useLocalStorageState<Record<
- string,
- any
- > | null>('ddm:default-query', null);
- useEffect(() => {
- if (defaultQuery) {
- 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]
- );
- };
- export function DDMContextProvider({children}: {children: React.ReactNode}) {
- const router = useRouter();
- const updateQuery = useUpdateQuery();
- const {setDefaultQuery, isDefaultQuery} = useDefaultQuery();
- const [selectedWidgetIndex, setSelectedWidgetIndex] = useState(0);
- const {widgets, updateWidget, addWidget, addWidgets, removeWidget, duplicateWidget} =
- useMetricWidgets();
- const [focusArea, setFocusArea] = useState<FocusArea | null>(null);
- const pageFilters = usePageFilters().selection;
- const {data: metricsMeta, isLoading} = useMetricsMeta(pageFilters.projects);
- const handleAddFocusArea = useCallback(
- (area: FocusArea) => {
- const dateRange = getAbsoluteDateTimeRange(pageFilters.datetime);
- if (!area.range.start || !area.range.end) {
- Sentry.metrics.increment('ddm.enhance.range-undefined');
- return;
- }
- if (area.range.start < dateRange.start || area.range.end > dateRange.end) {
- Sentry.metrics.increment('ddm.enhance.range-overflow');
- return;
- }
- Sentry.metrics.increment('ddm.enhance.add');
- setFocusArea(area);
- setSelectedWidgetIndex(area.widgetIndex);
- updateQuery({focusArea: JSON.stringify(area)});
- },
- [updateQuery, pageFilters.datetime]
- );
- const handleRemoveFocusArea = useCallback(() => {
- Sentry.metrics.increment('ddm.enhance.remove');
- setFocusArea(null);
- updateQuery({focusArea: null});
- }, [updateQuery]);
- // Load focus area from URL
- useEffect(() => {
- if (focusArea) {
- return;
- }
- const urlFocusArea = router.location.query.focusArea;
- if (urlFocusArea) {
- handleAddFocusArea(JSON.parse(urlFocusArea));
- }
- }, [router, handleAddFocusArea, focusArea]);
- const handleAddWidget = useCallback(() => {
- addWidget();
- setSelectedWidgetIndex(widgets.length);
- }, [addWidget, widgets.length]);
- const handleUpdateWidget = useCallback(
- (index: number, data: Partial<MetricWidgetQueryParams>) => {
- updateWidget(index, data);
- setSelectedWidgetIndex(index);
- if (index === focusArea?.widgetIndex) {
- handleRemoveFocusArea();
- }
- },
- [updateWidget, handleRemoveFocusArea, focusArea?.widgetIndex]
- );
- const handleDuplicate = useCallback(
- (index: number) => {
- duplicateWidget(index);
- setSelectedWidgetIndex(index + 1);
- },
- [duplicateWidget]
- );
- const contextValue = useMemo<DDMContextValue>(
- () => ({
- addWidget: handleAddWidget,
- addWidgets,
- selectedWidgetIndex:
- selectedWidgetIndex > widgets.length - 1 ? 0 : selectedWidgetIndex,
- setSelectedWidgetIndex,
- updateWidget: handleUpdateWidget,
- removeWidget,
- duplicateWidget: handleDuplicate,
- widgets,
- isLoading,
- metricsMeta,
- focusArea,
- addFocusArea: handleAddFocusArea,
- removeFocusArea: handleRemoveFocusArea,
- setDefaultQuery,
- isDefaultQuery,
- }),
- [
- handleAddWidget,
- addWidgets,
- selectedWidgetIndex,
- widgets,
- handleUpdateWidget,
- removeWidget,
- handleDuplicate,
- isLoading,
- metricsMeta,
- focusArea,
- handleAddFocusArea,
- handleRemoveFocusArea,
- setDefaultQuery,
- isDefaultQuery,
- ]
- );
- return <DDMContext.Provider value={contextValue}>{children}</DDMContext.Provider>;
- }
|