contextMenu.tsx 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154
  1. import {useMemo} from 'react';
  2. import styled from '@emotion/styled';
  3. import {openAddToDashboardModal, openModal} from 'sentry/actionCreators/modal';
  4. import {DropdownMenu, MenuItemProps} from 'sentry/components/dropdownMenu';
  5. import {IconCopy, IconDashboard, IconDelete, IconEllipsis, IconSiren} from 'sentry/icons';
  6. import {t} from 'sentry/locale';
  7. import {space} from 'sentry/styles/space';
  8. import {Organization} from 'sentry/types';
  9. import {isCustomMeasurement, MetricDisplayType, MetricsQuery} from 'sentry/utils/metrics';
  10. import {
  11. convertToDashboardWidget,
  12. encodeWidgetQuery,
  13. getWidgetAsQueryParams,
  14. getWidgetQuery,
  15. } from 'sentry/utils/metrics/dashboard';
  16. import {hasDDMFeature} from 'sentry/utils/metrics/features';
  17. import useOrganization from 'sentry/utils/useOrganization';
  18. import useRouter from 'sentry/utils/useRouter';
  19. import {useDDMContext} from 'sentry/views/ddm/context';
  20. import {CreateAlertModal} from 'sentry/views/ddm/createAlertModal';
  21. import {OrganizationContext} from 'sentry/views/organizationContext';
  22. type ContextMenuProps = {
  23. displayType: MetricDisplayType;
  24. metricsQuery: MetricsQuery;
  25. widgetIndex: number;
  26. };
  27. export function MetricWidgetContextMenu({
  28. metricsQuery,
  29. displayType,
  30. widgetIndex,
  31. }: ContextMenuProps) {
  32. const organization = useOrganization();
  33. const {removeWidget, duplicateWidget} = useDDMContext();
  34. const createAlert = useCreateAlert(organization, metricsQuery);
  35. const createDashboardWidget = useCreateDashboardWidget(
  36. organization,
  37. metricsQuery,
  38. displayType
  39. );
  40. const items = useMemo<MenuItemProps[]>(
  41. () => [
  42. {
  43. leadingItems: [<IconCopy key="icon" />],
  44. key: 'duplicate',
  45. label: t('Duplicate'),
  46. onAction: () => duplicateWidget(widgetIndex),
  47. },
  48. {
  49. leadingItems: [<IconSiren key="icon" />],
  50. key: 'add-alert',
  51. label: t('Create Alert'),
  52. disabled: !createAlert,
  53. onAction: createAlert,
  54. },
  55. {
  56. leadingItems: [<IconDashboard key="icon" />],
  57. key: 'add-dashoard',
  58. label: t('Add to Dashboard'),
  59. disabled: !createDashboardWidget,
  60. onAction: createDashboardWidget,
  61. },
  62. {
  63. leadingItems: [<IconDelete key="icon" />],
  64. key: 'delete',
  65. label: t('Delete'),
  66. onAction: () => removeWidget(widgetIndex),
  67. },
  68. ],
  69. [createAlert, createDashboardWidget, duplicateWidget, removeWidget, widgetIndex]
  70. );
  71. if (!hasDDMFeature(organization)) {
  72. return null;
  73. }
  74. return (
  75. <StyledDropdownMenuControl
  76. items={items}
  77. triggerProps={{
  78. 'aria-label': t('Widget actions'),
  79. size: 'xs',
  80. borderless: true,
  81. showChevron: false,
  82. icon: <IconEllipsis direction="down" size="sm" />,
  83. }}
  84. position="bottom-end"
  85. />
  86. );
  87. }
  88. const StyledDropdownMenuControl = styled(DropdownMenu)`
  89. margin: ${space(1)};
  90. `;
  91. export function useCreateAlert(organization: Organization, metricsQuery: MetricsQuery) {
  92. return useMemo(() => {
  93. if (
  94. !metricsQuery.mri ||
  95. !metricsQuery.op ||
  96. isCustomMeasurement(metricsQuery) ||
  97. !organization.access.includes('alerts:write')
  98. ) {
  99. return undefined;
  100. }
  101. return function () {
  102. return openModal(deps => (
  103. <OrganizationContext.Provider value={organization}>
  104. <CreateAlertModal metricsQuery={metricsQuery} {...deps} />
  105. </OrganizationContext.Provider>
  106. ));
  107. };
  108. }, [metricsQuery, organization]);
  109. }
  110. export function useCreateDashboardWidget(
  111. organization: Organization,
  112. metricsQuery: MetricsQuery,
  113. displayType?: MetricDisplayType
  114. ) {
  115. const router = useRouter();
  116. const {projects, environments, datetime} = metricsQuery;
  117. return useMemo(() => {
  118. if (!metricsQuery.mri || !metricsQuery.op || isCustomMeasurement(metricsQuery)) {
  119. return undefined;
  120. }
  121. const widgetQuery = getWidgetQuery(metricsQuery);
  122. const urlWidgetQuery = encodeWidgetQuery(widgetQuery);
  123. const widgetAsQueryParams = getWidgetAsQueryParams(
  124. metricsQuery,
  125. urlWidgetQuery,
  126. displayType
  127. );
  128. return () =>
  129. openAddToDashboardModal({
  130. organization,
  131. selection: {
  132. projects,
  133. environments,
  134. datetime,
  135. },
  136. widget: convertToDashboardWidget(metricsQuery, displayType),
  137. router,
  138. widgetAsQueryParams,
  139. location: router.location,
  140. });
  141. }, [metricsQuery, datetime, displayType, environments, organization, projects, router]);
  142. }