queries.tsx 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284
  1. import {Fragment, useCallback, useLayoutEffect, useMemo} from 'react';
  2. import styled from '@emotion/styled';
  3. import * as echarts from 'echarts/core';
  4. import {Button} from 'sentry/components/button';
  5. import SwitchButton from 'sentry/components/switchButton';
  6. import {IconAdd} from 'sentry/icons';
  7. import {t} from 'sentry/locale';
  8. import {space} from 'sentry/styles/space';
  9. import {
  10. type MetricFormulaWidgetParams,
  11. MetricQueryType,
  12. type MetricQueryWidgetParams,
  13. type MetricWidgetQueryParams,
  14. } from 'sentry/utils/metrics/types';
  15. import useOrganization from 'sentry/utils/useOrganization';
  16. import usePageFilters from 'sentry/utils/usePageFilters';
  17. import {DDM_CHART_GROUP} from 'sentry/views/ddm/constants';
  18. import {useDDMContext} from 'sentry/views/ddm/context';
  19. import {FormulaInput} from 'sentry/views/ddm/formulaInput';
  20. import {MetricFormulaContextMenu} from 'sentry/views/ddm/metricFormulaContextMenu';
  21. import {MetricQueryContextMenu} from 'sentry/views/ddm/metricQueryContextMenu';
  22. import {QueryBuilder} from 'sentry/views/ddm/queryBuilder';
  23. import {getQuerySymbol, QuerySymbol} from 'sentry/views/ddm/querySymbol';
  24. export function Queries() {
  25. const {
  26. widgets,
  27. updateWidget,
  28. setSelectedWidgetIndex,
  29. showQuerySymbols,
  30. selectedWidgetIndex,
  31. isMultiChartMode,
  32. setIsMultiChartMode,
  33. addWidget,
  34. } = useDDMContext();
  35. const {selection} = usePageFilters();
  36. const organization = useOrganization();
  37. // Make sure all charts are connected to the same group whenever the widgets definition changes
  38. useLayoutEffect(() => {
  39. echarts.connect(DDM_CHART_GROUP);
  40. }, [widgets]);
  41. const handleChange = useCallback(
  42. (index: number, widget: Partial<MetricWidgetQueryParams>) => {
  43. updateWidget(index, widget);
  44. },
  45. [updateWidget]
  46. );
  47. const [querySymbols, formulaSymbols] = useMemo(() => {
  48. const querySymbolSet = new Set<string>();
  49. const formulaSymbolSet = new Set<string>();
  50. for (const widget of widgets) {
  51. const symbol = getQuerySymbol(widget.id);
  52. if (widget.type === MetricQueryType.QUERY) {
  53. querySymbolSet.add(symbol);
  54. } else {
  55. formulaSymbolSet.add(symbol);
  56. }
  57. }
  58. return [querySymbolSet, formulaSymbolSet];
  59. }, [widgets]);
  60. return (
  61. <Fragment>
  62. <Wrapper showQuerySymbols={showQuerySymbols}>
  63. {widgets.map((widget, index) => (
  64. <Row key={widget.id} onFocusCapture={() => setSelectedWidgetIndex(index)}>
  65. {widget.type === MetricQueryType.QUERY ? (
  66. <Query
  67. widget={widget}
  68. onChange={handleChange}
  69. index={index}
  70. projects={selection.projects}
  71. symbol={
  72. showQuerySymbols && (
  73. <StyledQuerySymbol
  74. queryId={widget.id}
  75. isClickable={isMultiChartMode}
  76. isSelected={index === selectedWidgetIndex}
  77. onClick={() => setSelectedWidgetIndex(index)}
  78. role={isMultiChartMode ? 'button' : undefined}
  79. aria-label={t('Select query')}
  80. />
  81. )
  82. }
  83. contextMenu={
  84. <MetricQueryContextMenu
  85. displayType={widget.displayType}
  86. widgetIndex={index}
  87. metricsQuery={{
  88. mri: widget.mri,
  89. query: widget.query,
  90. op: widget.op,
  91. groupBy: widget.groupBy,
  92. }}
  93. />
  94. }
  95. />
  96. ) : (
  97. <Formula
  98. availableVariables={querySymbols}
  99. formulaVariables={formulaSymbols}
  100. onChange={handleChange}
  101. index={index}
  102. widget={widget}
  103. symbol={
  104. showQuerySymbols && (
  105. <StyledQuerySymbol
  106. queryId={widget.id}
  107. isClickable={isMultiChartMode}
  108. isSelected={index === selectedWidgetIndex}
  109. onClick={() => setSelectedWidgetIndex(index)}
  110. role={isMultiChartMode ? 'button' : undefined}
  111. aria-label={t('Select query')}
  112. />
  113. )
  114. }
  115. contextMenu={<MetricFormulaContextMenu widgetIndex={index} />}
  116. />
  117. )}
  118. </Row>
  119. ))}
  120. </Wrapper>
  121. <ButtonBar addQuerySymbolSpacing={showQuerySymbols}>
  122. <Button
  123. size="sm"
  124. icon={<IconAdd isCircled />}
  125. onClick={() => addWidget(MetricQueryType.QUERY)}
  126. >
  127. Add query
  128. </Button>
  129. {organization.features.includes('ddm-formulas') && (
  130. <Button
  131. size="sm"
  132. icon={<IconAdd isCircled />}
  133. onClick={() => addWidget(MetricQueryType.FORMULA)}
  134. >
  135. Add formula
  136. </Button>
  137. )}
  138. <SwitchWrapper>
  139. {t('One chart per query')}
  140. <SwitchButton
  141. isActive={isMultiChartMode}
  142. toggle={() => setIsMultiChartMode(!isMultiChartMode)}
  143. />
  144. </SwitchWrapper>
  145. </ButtonBar>
  146. </Fragment>
  147. );
  148. }
  149. interface QueryProps {
  150. index: number;
  151. onChange: (index: number, data: Partial<MetricWidgetQueryParams>) => void;
  152. projects: number[];
  153. widget: MetricQueryWidgetParams;
  154. contextMenu?: React.ReactNode;
  155. symbol?: React.ReactNode;
  156. }
  157. export function Query({
  158. widget,
  159. projects,
  160. onChange,
  161. contextMenu,
  162. symbol,
  163. index,
  164. }: QueryProps) {
  165. const metricsQuery = useMemo(
  166. () => ({
  167. mri: widget.mri,
  168. op: widget.op,
  169. groupBy: widget.groupBy,
  170. query: widget.query,
  171. }),
  172. [widget.groupBy, widget.mri, widget.op, widget.query]
  173. );
  174. const handleChange = useCallback(
  175. (data: Partial<MetricWidgetQueryParams>) => {
  176. onChange(index, data);
  177. },
  178. [index, onChange]
  179. );
  180. return (
  181. <QueryWrapper hasSymbol={!!symbol}>
  182. {symbol}
  183. <QueryBuilder
  184. onChange={handleChange}
  185. metricsQuery={metricsQuery}
  186. displayType={widget.displayType}
  187. isEdit
  188. projects={projects}
  189. />
  190. {contextMenu}
  191. </QueryWrapper>
  192. );
  193. }
  194. interface FormulaProps {
  195. availableVariables: Set<string>;
  196. formulaVariables: Set<string>;
  197. index: number;
  198. onChange: (index: number, data: Partial<MetricWidgetQueryParams>) => void;
  199. widget: MetricFormulaWidgetParams;
  200. contextMenu?: React.ReactNode;
  201. symbol?: React.ReactNode;
  202. }
  203. export function Formula({
  204. availableVariables,
  205. formulaVariables,
  206. index,
  207. widget,
  208. onChange,
  209. contextMenu,
  210. symbol,
  211. }: FormulaProps) {
  212. const handleChange = useCallback(
  213. (formula: string) => {
  214. onChange(index, {formula});
  215. },
  216. [index, onChange]
  217. );
  218. return (
  219. <QueryWrapper hasSymbol={!!symbol}>
  220. {symbol}
  221. <FormulaInput
  222. availableVariables={availableVariables}
  223. formulaVariables={formulaVariables}
  224. value={widget.formula}
  225. onChange={handleChange}
  226. />
  227. {contextMenu}
  228. </QueryWrapper>
  229. );
  230. }
  231. const QueryWrapper = styled('div')<{hasSymbol: boolean}>`
  232. display: grid;
  233. gap: ${space(1)};
  234. padding-bottom: ${space(1)};
  235. grid-template-columns: 1fr max-content;
  236. ${p => p.hasSymbol && `grid-template-columns: min-content 1fr max-content;`}
  237. `;
  238. const StyledQuerySymbol = styled(QuerySymbol)<{isClickable: boolean}>`
  239. margin-top: 10px;
  240. ${p => p.isClickable && `cursor: pointer;`}
  241. `;
  242. const Wrapper = styled('div')<{showQuerySymbols: boolean}>``;
  243. const Row = styled('div')`
  244. display: contents;
  245. `;
  246. const ButtonBar = styled('div')<{addQuerySymbolSpacing: boolean}>`
  247. align-items: center;
  248. display: flex;
  249. padding-bottom: ${space(2)};
  250. padding-top: ${space(1)};
  251. gap: ${space(2)};
  252. ${p =>
  253. p.addQuerySymbolSpacing &&
  254. `
  255. padding-left: ${space(1)};
  256. margin-left: ${space(2)};
  257. `}
  258. `;
  259. const SwitchWrapper = styled('label')`
  260. display: flex;
  261. margin: 0;
  262. align-items: center;
  263. gap: ${space(1)};
  264. `;