context.tsx 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269
  1. import {
  2. createContext,
  3. useCallback,
  4. useContext,
  5. useEffect,
  6. useMemo,
  7. useState,
  8. } from 'react';
  9. import {MRI} from 'sentry/types';
  10. import {
  11. defaultMetricDisplayType,
  12. MetricDisplayType,
  13. MetricWidgetQueryParams,
  14. useInstantRef,
  15. useUpdateQuery,
  16. } from 'sentry/utils/metrics';
  17. import {parseMRI} from 'sentry/utils/metrics/mri';
  18. import {useMetricsMeta} from 'sentry/utils/metrics/useMetricsMeta';
  19. import {decodeList} from 'sentry/utils/queryString';
  20. import usePageFilters from 'sentry/utils/usePageFilters';
  21. import useRouter from 'sentry/utils/useRouter';
  22. import {FocusArea} from 'sentry/views/ddm/chartBrush';
  23. import {DEFAULT_SORT_STATE} from 'sentry/views/ddm/constants';
  24. import {useStructuralSharing} from 'sentry/views/ddm/useStructuralSharing';
  25. interface DDMContextValue {
  26. addFocusArea: (area: FocusArea) => void;
  27. addWidget: () => void;
  28. duplicateWidget: (index: number) => void;
  29. focusArea: FocusArea | null;
  30. hasCustomMetrics: boolean;
  31. isLoading: boolean;
  32. metricsMeta: ReturnType<typeof useMetricsMeta>['data'];
  33. removeFocusArea: () => void;
  34. removeWidget: (index: number) => void;
  35. selectedWidgetIndex: number;
  36. setSelectedWidgetIndex: (index: number) => void;
  37. updateWidget: (index: number, data: Partial<MetricWidgetQueryParams>) => void;
  38. widgets: MetricWidgetQueryParams[];
  39. }
  40. export const DDMContext = createContext<DDMContextValue>({
  41. selectedWidgetIndex: 0,
  42. setSelectedWidgetIndex: () => {},
  43. addWidget: () => {},
  44. updateWidget: () => {},
  45. removeWidget: () => {},
  46. addFocusArea: () => {},
  47. removeFocusArea: () => {},
  48. duplicateWidget: () => {},
  49. widgets: [],
  50. metricsMeta: [],
  51. hasCustomMetrics: false,
  52. isLoading: false,
  53. focusArea: null,
  54. });
  55. export function useDDMContext() {
  56. return useContext(DDMContext);
  57. }
  58. const emptyWidget: MetricWidgetQueryParams = {
  59. mri: '' as MRI,
  60. op: undefined,
  61. query: '',
  62. groupBy: [],
  63. sort: DEFAULT_SORT_STATE,
  64. displayType: MetricDisplayType.LINE,
  65. title: undefined,
  66. };
  67. export function useMetricWidgets() {
  68. const router = useRouter();
  69. const updateQuery = useUpdateQuery();
  70. const widgets = useStructuralSharing(
  71. useMemo<MetricWidgetQueryParams[]>(() => {
  72. const currentWidgets = JSON.parse(
  73. router.location.query.widgets ?? JSON.stringify([emptyWidget])
  74. );
  75. return currentWidgets.map((widget: MetricWidgetQueryParams) => {
  76. return {
  77. mri: widget.mri,
  78. op: widget.op,
  79. query: widget.query,
  80. groupBy: decodeList(widget.groupBy),
  81. displayType: widget.displayType ?? defaultMetricDisplayType,
  82. focusedSeries: widget.focusedSeries,
  83. showSummaryTable: widget.showSummaryTable ?? true, // temporary default
  84. powerUserMode: widget.powerUserMode,
  85. sort: widget.sort ?? DEFAULT_SORT_STATE,
  86. title: widget.title,
  87. };
  88. });
  89. }, [router.location.query.widgets])
  90. );
  91. // We want to have it as a ref, so that we can use it in the setWidget callback
  92. // without needing to generate a new callback every time the location changes
  93. const currentWidgetsRef = useInstantRef(widgets);
  94. const setWidgets = useCallback(
  95. (newWidgets: React.SetStateAction<MetricWidgetQueryParams[]>) => {
  96. const currentWidgets = currentWidgetsRef.current;
  97. updateQuery({
  98. widgets: JSON.stringify(
  99. typeof newWidgets === 'function' ? newWidgets(currentWidgets) : newWidgets
  100. ),
  101. });
  102. },
  103. [updateQuery, currentWidgetsRef]
  104. );
  105. const updateWidget = useCallback(
  106. (index: number, data: Partial<MetricWidgetQueryParams>) => {
  107. setWidgets(currentWidgets => {
  108. const newWidgets = [...currentWidgets];
  109. newWidgets[index] = {...currentWidgets[index], ...data};
  110. return newWidgets;
  111. });
  112. },
  113. [setWidgets]
  114. );
  115. const addWidget = useCallback(() => {
  116. setWidgets(currentWidgets => [...currentWidgets, emptyWidget]);
  117. }, [setWidgets]);
  118. const removeWidget = useCallback(
  119. (index: number) => {
  120. setWidgets(currentWidgets => {
  121. const newWidgets = [...currentWidgets];
  122. newWidgets.splice(index, 1);
  123. return newWidgets;
  124. });
  125. },
  126. [setWidgets]
  127. );
  128. const duplicateWidget = useCallback(
  129. (index: number) => {
  130. setWidgets(currentWidgets => {
  131. const newWidgets = [...currentWidgets];
  132. newWidgets.splice(index, 0, currentWidgets[index]);
  133. return newWidgets;
  134. });
  135. },
  136. [setWidgets]
  137. );
  138. return {
  139. widgets,
  140. updateWidget,
  141. addWidget,
  142. removeWidget,
  143. duplicateWidget,
  144. };
  145. }
  146. export function DDMContextProvider({children}: {children: React.ReactNode}) {
  147. const router = useRouter();
  148. const updateQuery = useUpdateQuery();
  149. const [selectedWidgetIndex, setSelectedWidgetIndex] = useState(0);
  150. const [focusArea, setFocusArea] = useState<FocusArea | null>(null);
  151. const {widgets, updateWidget, addWidget, removeWidget, duplicateWidget} =
  152. useMetricWidgets();
  153. const pageFilters = usePageFilters().selection;
  154. const {data: metricsMeta, isLoading} = useMetricsMeta(pageFilters.projects);
  155. // TODO(telemetry-experience): Switch to the logic below once we have the hasCustomMetrics flag on project
  156. // const {projects} = useProjects();
  157. // const selectedProjects = projects.filter(project =>
  158. // pageFilters.projects.includes(parseInt(project.id, 10))
  159. // );
  160. // const hasCustomMetrics = selectedProjects.some(project => project.hasCustomMetrics);
  161. const hasCustomMetrics = !!metricsMeta.find(
  162. meta => parseMRI(meta)?.useCase === 'custom'
  163. );
  164. const handleAddFocusArea = useCallback(
  165. (area: FocusArea) => {
  166. setFocusArea(area);
  167. setSelectedWidgetIndex(area.widgetIndex);
  168. updateQuery({focusArea: JSON.stringify(area)});
  169. },
  170. [updateQuery]
  171. );
  172. const handleRemoveFocusArea = useCallback(() => {
  173. setFocusArea(null);
  174. updateQuery({focusArea: null});
  175. }, [updateQuery]);
  176. // Load focus area from URL
  177. useEffect(() => {
  178. if (focusArea) {
  179. return;
  180. }
  181. const urlFocusArea = router.location.query.focusArea;
  182. if (urlFocusArea) {
  183. handleAddFocusArea(JSON.parse(urlFocusArea));
  184. }
  185. }, [router, handleAddFocusArea, focusArea]);
  186. const handleAddWidget = useCallback(() => {
  187. addWidget();
  188. setSelectedWidgetIndex(widgets.length);
  189. }, [addWidget, widgets.length]);
  190. const handleUpdateWidget = useCallback(
  191. (index: number, data: Partial<MetricWidgetQueryParams>) => {
  192. updateWidget(index, data);
  193. setSelectedWidgetIndex(index);
  194. if (index === focusArea?.widgetIndex) {
  195. handleRemoveFocusArea();
  196. }
  197. },
  198. [updateWidget, handleRemoveFocusArea, focusArea?.widgetIndex]
  199. );
  200. const handleDuplicate = useCallback(
  201. (index: number) => {
  202. duplicateWidget(index);
  203. setSelectedWidgetIndex(index + 1);
  204. },
  205. [duplicateWidget]
  206. );
  207. const contextValue = useMemo<DDMContextValue>(
  208. () => ({
  209. addWidget: handleAddWidget,
  210. selectedWidgetIndex:
  211. selectedWidgetIndex > widgets.length - 1 ? 0 : selectedWidgetIndex,
  212. setSelectedWidgetIndex,
  213. updateWidget: handleUpdateWidget,
  214. removeWidget,
  215. duplicateWidget: handleDuplicate,
  216. widgets,
  217. hasCustomMetrics,
  218. isLoading,
  219. metricsMeta,
  220. focusArea,
  221. addFocusArea: handleAddFocusArea,
  222. removeFocusArea: handleRemoveFocusArea,
  223. }),
  224. [
  225. handleAddWidget,
  226. handleDuplicate,
  227. handleUpdateWidget,
  228. removeWidget,
  229. hasCustomMetrics,
  230. isLoading,
  231. metricsMeta,
  232. selectedWidgetIndex,
  233. widgets,
  234. focusArea,
  235. handleAddFocusArea,
  236. handleRemoveFocusArea,
  237. ]
  238. );
  239. return <DDMContext.Provider value={contextValue}>{children}</DDMContext.Provider>;
  240. }