123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176 |
- import {useMemo} from 'react';
- import styled from '@emotion/styled';
- import {openAddToDashboardModal} from 'sentry/actionCreators/modal';
- import {DropdownMenu} from 'sentry/components/dropdownMenu';
- import {IconEllipsis} from 'sentry/icons';
- import {t} from 'sentry/locale';
- import {space} from 'sentry/styles/space';
- import {Organization} from 'sentry/types';
- import {MetricDisplayType, MetricsQuery, mriToField} from 'sentry/utils/metrics';
- import useOrganization from 'sentry/utils/useOrganization';
- import usePageFilters from 'sentry/utils/usePageFilters';
- import useProjects from 'sentry/utils/useProjects';
- import useRouter from 'sentry/utils/useRouter';
- import {Dataset, EventTypes} from 'sentry/views/alerts/rules/metric/types';
- import {DashboardWidgetSource} from 'sentry/views/dashboards/types';
- type ContextMenuProps = {
- displayType: MetricDisplayType;
- metricsQuery: MetricsQuery;
- };
- export function MetricWidgetContextMenu({metricsQuery, displayType}: ContextMenuProps) {
- const organization = useOrganization();
- const createAlertUrl = useCreateAlertUrl(organization, metricsQuery);
- const handleAddQueryToDashboard = useHandleAddQueryToDashboard(
- organization,
- metricsQuery,
- displayType
- );
- if (!organization.features.includes('ddm-experimental')) {
- return null;
- }
- return (
- <StyledDropdownMenuControl
- items={[
- {
- key: 'add-alert',
- label: t('Create Alert'),
- disabled: !createAlertUrl,
- to: createAlertUrl,
- },
- {
- key: 'add-dashoard',
- label: t('Add to Dashboard'),
- disabled: !handleAddQueryToDashboard,
- onAction: handleAddQueryToDashboard,
- },
- ]}
- triggerProps={{
- 'aria-label': t('Widget actions'),
- size: 'xs',
- borderless: true,
- showChevron: false,
- icon: <IconEllipsis direction="down" size="sm" />,
- }}
- position="bottom-end"
- />
- );
- }
- function useHandleAddQueryToDashboard(
- organization: Organization,
- {projects, environments, datetime, op, mri, groupBy, query}: MetricsQuery,
- displayType?: MetricDisplayType
- ) {
- const router = useRouter();
- const {start, end, period} = datetime;
- return useMemo(() => {
- if (!mri || !op) {
- return undefined;
- }
- const field = mriToField(mri, op);
- const limit = !groupBy?.length ? 1 : 10;
- const widgetAsQueryParams = {
- ...router.location?.query,
- source: DashboardWidgetSource.DDM,
- start,
- end,
- statsPeriod: period,
- defaultWidgetQuery: field,
- defaultTableColumns: [],
- defaultTitle: 'DDM Widget',
- displayType,
- };
- return () =>
- openAddToDashboardModal({
- organization,
- selection: {
- projects,
- environments,
- datetime,
- },
- widget: {
- title: 'DDM Widget',
- displayType,
- widgetType: 'custom-metrics',
- limit,
- queries: [
- {
- name: '',
- aggregates: [field],
- columns: groupBy ?? [],
- fields: [field],
- conditions: query,
- },
- ],
- },
- router,
- widgetAsQueryParams,
- location: router.location,
- });
- }, [
- datetime,
- displayType,
- end,
- environments,
- groupBy,
- mri,
- op,
- organization,
- period,
- projects,
- query,
- router,
- start,
- ]);
- }
- function useCreateAlertUrl(organization: Organization, metricsQuery: MetricsQuery) {
- const projects = useProjects();
- const pageFilters = usePageFilters();
- const selectedProjects = pageFilters.selection.projects;
- const firstProjectSlug =
- selectedProjects.length > 0 &&
- projects.projects.find(p => p.id === selectedProjects[0].toString())?.slug;
- return useMemo(() => {
- if (!firstProjectSlug || !metricsQuery.mri || !metricsQuery.op) {
- return undefined;
- }
- return {
- pathname: `/organizations/${organization.slug}/alerts/new/metric/`,
- query: {
- // Needed, so alerts-create also collects environment via event view
- createFromDiscover: true,
- dataset: Dataset.GENERIC_METRICS,
- eventTypes: EventTypes.TRANSACTION,
- aggregate: mriToField(metricsQuery.mri, metricsQuery.op as string),
- referrer: 'ddm',
- // Event type also needs to be added to the query
- query: `${metricsQuery.query} event.type:transaction`.trim(),
- environment: metricsQuery.environments,
- project: firstProjectSlug,
- },
- };
- }, [
- firstProjectSlug,
- metricsQuery.environments,
- metricsQuery.mri,
- metricsQuery.op,
- metricsQuery.query,
- organization.slug,
- ]);
- }
- const StyledDropdownMenuControl = styled(DropdownMenu)`
- margin: ${space(1)};
- `;
|