import {useCallback, useState} from 'react'; import {InjectedRouter} from 'react-router'; import styled from '@emotion/styled'; import {Location} from 'history'; import ErrorPanel from 'sentry/components/charts/errorPanel'; import {HeaderTitle} from 'sentry/components/charts/styles'; import TextOverflow from 'sentry/components/textOverflow'; import {IconWarning} from 'sentry/icons'; import {space} from 'sentry/styles/space'; import {MRI, Organization, PageFilters} from 'sentry/types'; import {stringifyMetricWidget} from 'sentry/utils/metrics'; import type {MetricWidgetQueryParams} from 'sentry/utils/metrics/types'; import {WidgetCardPanel, WidgetTitleRow} from 'sentry/views/dashboards/widgetCard'; import {AugmentedEChartDataZoomHandler} from 'sentry/views/dashboards/widgetCard/chart'; import {DashboardsMEPContext} from 'sentry/views/dashboards/widgetCard/dashboardsMEPContext'; import {InlineEditor} from 'sentry/views/dashboards/widgetCard/metricWidgetCard/inlineEditor'; import {Toolbar} from 'sentry/views/dashboards/widgetCard/toolbar'; import WidgetCardContextMenu from 'sentry/views/dashboards/widgetCard/widgetCardContextMenu'; import {MetricWidgetBody} from 'sentry/views/ddm/widget'; import { convertToDashboardWidget, toMetricDisplayType, } from '../../../../utils/metrics/dashboard'; import {parseField} from '../../../../utils/metrics/mri'; import {Widget} from '../../types'; type Props = { isEditingDashboard: boolean; location: Location; organization: Organization; router: InjectedRouter; selection: PageFilters; widget: Widget; index?: string; isEditingWidget?: boolean; isMobile?: boolean; onDelete?: () => void; onDuplicate?: () => void; onEdit?: (index: string) => void; onUpdate?: (widget: Widget | null) => void; onZoom?: AugmentedEChartDataZoomHandler; showSlider?: boolean; tableItemLimit?: number; windowWidth?: number; }; export function MetricWidgetCard({ organization, selection, widget, isEditingWidget, isEditingDashboard, onEdit, onUpdate, onDelete, onDuplicate, location, router, index, }: Props) { const [metricWidgetQueryParams, setMetricWidgetQueryParams] = useState(convertFromWidget(widget)); const [title, setTitle] = useState( widget.title ?? stringifyMetricWidget(metricWidgetQueryParams) ); const handleChange = useCallback( (data: Partial) => { setMetricWidgetQueryParams(curr => ({ ...curr, ...data, })); }, [setMetricWidgetQueryParams] ); const handleSubmit = useCallback(() => { const convertedWidget = convertToDashboardWidget( {...selection, ...metricWidgetQueryParams}, toMetricDisplayType(metricWidgetQueryParams.displayType) ); const updatedWidget = { ...widget, title, queries: convertedWidget.queries, displayType: convertedWidget.displayType, }; onUpdate?.(updatedWidget); }, [title, metricWidgetQueryParams, onUpdate, widget, selection]); const handleCancel = useCallback(() => { onUpdate?.(null); }, [onUpdate]); if (!metricWidgetQueryParams.mri) { return ( ); } return ( {}, }} > {isEditingWidget ? ( ) : ( {title} )} {!isEditingDashboard && ( index && onEdit?.(index)} router={router} location={location} onDelete={onDelete} onDuplicate={onDuplicate} /> )} {isEditingDashboard && } ); } type MetricWidgetChartContainerProps = { selection: PageFilters; widget: Widget; editorParams?: Partial; }; export function MetricWidgetChartContainer({ selection, widget, editorParams = {}, }: MetricWidgetChartContainerProps) { const metricWidgetQueryParams = { ...convertFromWidget(widget), ...editorParams, }; return ( ); } function convertFromWidget(widget: Widget): MetricWidgetQueryParams { const query = widget.queries[0]; const parsed = parseField(query.aggregates[0]) || {mri: '' as MRI, op: ''}; return { mri: parsed.mri, op: parsed.op, query: query.conditions, groupBy: query.columns, title: widget.title, displayType: toMetricDisplayType(widget.displayType), }; } const WidgetHeaderWrapper = styled('div')` min-height: 36px; width: 100%; display: flex; align-items: flex-start; justify-content: space-between; `; const ContextMenuWrapper = styled('div')` padding: ${space(2)} ${space(1)} 0 ${space(3)}; `; const WidgetHeaderDescription = styled('div')` ${p => p.theme.overflowEllipsis}; overflow-y: visible; `; const WidgetTitle = styled(HeaderTitle)` padding-left: ${space(3)}; padding-top: ${space(2)}; padding-right: ${space(1)}; ${p => p.theme.overflowEllipsis}; font-weight: normal; `; const MetricWidgetChartWrapper = styled('div')` height: 100%; width: 100%; padding: ${space(2)}; `;