widgetCard.tsx 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210
  1. import {Fragment, useMemo} from 'react';
  2. import type {InjectedRouter} from 'react-router';
  3. import styled from '@emotion/styled';
  4. import type {Location} from 'history';
  5. import ErrorPanel from 'sentry/components/charts/errorPanel';
  6. import {HeaderTitle} from 'sentry/components/charts/styles';
  7. import EmptyMessage from 'sentry/components/emptyMessage';
  8. import TextOverflow from 'sentry/components/textOverflow';
  9. import {IconSearch, IconWarning} from 'sentry/icons';
  10. import {t} from 'sentry/locale';
  11. import {space} from 'sentry/styles/space';
  12. import type {Organization, PageFilters} from 'sentry/types';
  13. import {getWidgetTitle} from 'sentry/utils/metrics';
  14. import {useMetricsQuery} from 'sentry/utils/metrics/useMetricsQuery';
  15. import {MetricChartContainer} from 'sentry/views/dashboards/metrics/chart';
  16. import {MetricTableContainer} from 'sentry/views/dashboards/metrics/table';
  17. import {
  18. getMetricQueries,
  19. toMetricDisplayType,
  20. } from 'sentry/views/dashboards/metrics/utils';
  21. import type {DashboardFilters, Widget} from 'sentry/views/dashboards/types';
  22. import {DisplayType} from 'sentry/views/dashboards/types';
  23. import {WidgetCardPanel, WidgetTitleRow} from 'sentry/views/dashboards/widgetCard';
  24. import {DashboardsMEPContext} from 'sentry/views/dashboards/widgetCard/dashboardsMEPContext';
  25. import {Toolbar} from 'sentry/views/dashboards/widgetCard/toolbar';
  26. import WidgetCardContextMenu from 'sentry/views/dashboards/widgetCard/widgetCardContextMenu';
  27. type Props = {
  28. isEditingDashboard: boolean;
  29. location: Location;
  30. organization: Organization;
  31. router: InjectedRouter;
  32. selection: PageFilters;
  33. widget: Widget;
  34. dashboardFilters?: DashboardFilters;
  35. index?: string;
  36. isMobile?: boolean;
  37. onDelete?: () => void;
  38. onDuplicate?: () => void;
  39. onEdit?: (index: string) => void;
  40. renderErrorMessage?: (errorMessage?: string) => React.ReactNode;
  41. showContextMenu?: boolean;
  42. };
  43. export function MetricWidgetCard({
  44. organization,
  45. selection,
  46. widget,
  47. isEditingDashboard,
  48. onDelete,
  49. onDuplicate,
  50. location,
  51. router,
  52. dashboardFilters,
  53. renderErrorMessage,
  54. showContextMenu = true,
  55. }: Props) {
  56. const metricQueries = useMemo(
  57. () => getMetricQueries(widget, dashboardFilters),
  58. [widget, dashboardFilters]
  59. );
  60. const widgetMQL = useMemo(() => getWidgetTitle(metricQueries), [metricQueries]);
  61. const isTable = widget.displayType === DisplayType.TABLE;
  62. const {
  63. data: timeseriesData,
  64. isLoading,
  65. isError,
  66. error,
  67. } = useMetricsQuery(metricQueries, selection, {
  68. intervalLadder: widget.displayType === DisplayType.BAR ? 'bar' : 'dashboard',
  69. });
  70. if (isError) {
  71. const errorMessage =
  72. error?.responseJSON?.detail?.toString() || t('Error while fetching metrics data');
  73. return (
  74. <Fragment>
  75. {renderErrorMessage?.(errorMessage)}
  76. <ErrorPanel>
  77. <IconWarning color="gray500" size="lg" />
  78. </ErrorPanel>
  79. </Fragment>
  80. );
  81. }
  82. return (
  83. <DashboardsMEPContext.Provider
  84. value={{
  85. isMetricsData: undefined,
  86. setIsMetricsData: () => {},
  87. }}
  88. >
  89. <WidgetCardPanel isDragging={false}>
  90. <WidgetHeaderWrapper>
  91. <WidgetHeaderDescription>
  92. <WidgetTitleRow>
  93. <WidgetTitle>
  94. <TextOverflow>{widget.title || widgetMQL}</TextOverflow>
  95. </WidgetTitle>
  96. </WidgetTitleRow>
  97. </WidgetHeaderDescription>
  98. <ContextMenuWrapper>
  99. {showContextMenu && !isEditingDashboard && (
  100. <WidgetCardContextMenu
  101. organization={organization}
  102. widget={widget}
  103. selection={selection}
  104. showContextMenu
  105. isPreview={false}
  106. widgetLimitReached={false}
  107. onEdit={() => {
  108. router.push({
  109. pathname: `${location.pathname}${
  110. location.pathname.endsWith('/') ? '' : '/'
  111. }widget/${widget.id}/`,
  112. query: location.query,
  113. });
  114. }}
  115. router={router}
  116. location={location}
  117. onDelete={onDelete}
  118. onDuplicate={onDuplicate}
  119. />
  120. )}
  121. </ContextMenuWrapper>
  122. </WidgetHeaderWrapper>
  123. <WidgetCardBody
  124. isError={isError}
  125. noData={timeseriesData?.data.length === 0}
  126. renderErrorMessage={renderErrorMessage}
  127. error={error}
  128. >
  129. {!isTable ? (
  130. <MetricChartContainer
  131. timeseriesData={timeseriesData}
  132. isLoading={isLoading}
  133. metricQueries={metricQueries}
  134. displayType={toMetricDisplayType(widget.displayType)}
  135. chartHeight={!showContextMenu ? 200 : undefined}
  136. />
  137. ) : (
  138. <MetricTableContainer
  139. metricQueries={metricQueries}
  140. timeseriesData={timeseriesData}
  141. isLoading={isLoading}
  142. />
  143. )}
  144. </WidgetCardBody>
  145. {isEditingDashboard && <Toolbar onDelete={onDelete} onDuplicate={onDuplicate} />}
  146. </WidgetCardPanel>
  147. </DashboardsMEPContext.Provider>
  148. );
  149. }
  150. function WidgetCardBody({children, isError, noData, renderErrorMessage, error}) {
  151. if (isError) {
  152. const errorMessage =
  153. error?.responseJSON?.detail?.toString() || t('Error while fetching metrics data');
  154. return (
  155. <Fragment>
  156. {renderErrorMessage?.(errorMessage)}
  157. <ErrorPanel>
  158. <IconWarning color="gray500" size="lg" />
  159. </ErrorPanel>
  160. </Fragment>
  161. );
  162. }
  163. if (noData) {
  164. return (
  165. <EmptyMessage
  166. icon={<IconSearch size="xxl" />}
  167. title={t('No results')}
  168. description={t('No results found for the given query')}
  169. />
  170. );
  171. }
  172. return children;
  173. }
  174. const WidgetHeaderWrapper = styled('div')`
  175. min-height: 36px;
  176. width: 100%;
  177. display: flex;
  178. align-items: flex-start;
  179. justify-content: space-between;
  180. `;
  181. const ContextMenuWrapper = styled('div')`
  182. padding: ${space(2)} ${space(1)} 0 ${space(3)};
  183. `;
  184. const WidgetHeaderDescription = styled('div')`
  185. ${p => p.theme.overflowEllipsis};
  186. overflow-y: visible;
  187. `;
  188. const WidgetTitle = styled(HeaderTitle)`
  189. padding-left: ${space(3)};
  190. padding-top: ${space(2)};
  191. padding-right: ${space(1)};
  192. ${p => p.theme.overflowEllipsis};
  193. font-weight: normal;
  194. `;