import {useEffect, useState} from 'react'; import styled from '@emotion/styled'; import omit from 'lodash/omit'; import pick from 'lodash/pick'; import * as qs from 'query-string'; import type {SelectOption} from 'sentry/components/compactSelect'; import {CompactSelect} from 'sentry/components/compactSelect'; import {CompositeSelect} from 'sentry/components/compactSelect/composite'; import DropdownButton from 'sentry/components/dropdownButton'; import {IconEllipsis} from 'sentry/icons/iconEllipsis'; import {t} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import type {Organization} from 'sentry/types/organization'; import {trackAnalytics} from 'sentry/utils/analytics'; import {browserHistory} from 'sentry/utils/browserHistory'; import type EventView from 'sentry/utils/discover/eventView'; import type {Field} from 'sentry/utils/discover/fields'; import {DisplayModes, SavedQueryDatasets} from 'sentry/utils/discover/types'; import {useMEPSettingContext} from 'sentry/utils/performance/contexts/metricsEnhancedSetting'; import {usePerformanceDisplayType} from 'sentry/utils/performance/contexts/performanceDisplayContext'; import normalizeUrl from 'sentry/utils/url/normalizeUrl'; import useOrganization from 'sentry/utils/useOrganization'; import withOrganization from 'sentry/utils/withOrganization'; import {hasDatasetSelector} from 'sentry/views/dashboards/utils'; import MobileReleaseComparisonListWidget from 'sentry/views/performance/landing/widgets/widgets/mobileReleaseComparisonListWidget'; import {PerformanceScoreListWidget} from 'sentry/views/performance/landing/widgets/widgets/performanceScoreListWidget'; import {GenericPerformanceWidgetDataType} from '../types'; import {_setChartSetting, filterAllowedChartsMetrics, getChartSetting} from '../utils'; import type {ChartDefinition, PerformanceWidgetSetting} from '../widgetDefinitions'; import {WIDGET_DEFINITIONS} from '../widgetDefinitions'; import {HistogramWidget} from '../widgets/histogramWidget'; import {LineChartListWidget} from '../widgets/lineChartListWidget'; import {PerformanceScoreWidget} from '../widgets/performanceScoreWidget'; import {SingleFieldAreaWidget} from '../widgets/singleFieldAreaWidget'; import {StackedAreaChartListWidget} from '../widgets/stackedAreaChartListWidget'; import {TrendsWidget} from '../widgets/trendsWidget'; import {VitalWidget} from '../widgets/vitalWidget'; import type {ChartRowProps} from './widgetChartRow'; interface Props extends ChartRowProps { allowedCharts: PerformanceWidgetSetting[]; chartHeight: number; defaultChartSetting: PerformanceWidgetSetting; eventView: EventView; index: number; organization: Organization; rowChartSettings: PerformanceWidgetSetting[]; setRowChartSettings: (settings: PerformanceWidgetSetting[]) => void; withStaticFilters: boolean; chartColor?: string; forceDefaultChartSetting?: boolean; } function trackChartSettingChange( previousChartSetting: PerformanceWidgetSetting, chartSetting: PerformanceWidgetSetting, fromDefault: boolean, organization: Organization ) { trackAnalytics('performance_views.landingv3.widget.switch', { organization, from_widget: previousChartSetting, to_widget: chartSetting, from_default: fromDefault, is_new_menu: organization.features.includes('performance-new-widget-designs'), }); } function _WidgetContainer(props: Props) { const { organization, index, chartHeight, rowChartSettings, setRowChartSettings, ...rest } = props; const performanceType = usePerformanceDisplayType(); let _chartSetting = getChartSetting( index, chartHeight, performanceType, rest.defaultChartSetting, rest.forceDefaultChartSetting ); const mepSetting = useMEPSettingContext(); const allowedCharts = filterAllowedChartsMetrics( props.organization, props.allowedCharts, mepSetting ); if (!allowedCharts.includes(_chartSetting)) { _chartSetting = rest.defaultChartSetting; } const [chartSetting, setChartSettingState] = useState(_chartSetting); const setChartSetting = (setting: PerformanceWidgetSetting) => { if (!props.forceDefaultChartSetting) { _setChartSetting(index, chartHeight, performanceType, setting); } setChartSettingState(setting); const newSettings = [...rowChartSettings]; newSettings[index] = setting; setRowChartSettings(newSettings); trackChartSettingChange( chartSetting, setting, rest.defaultChartSetting === chartSetting, organization ); }; useEffect(() => { setChartSettingState(_chartSetting); }, [rest.defaultChartSetting, _chartSetting]); const chartDefinition = WIDGET_DEFINITIONS({organization})[chartSetting]; // Construct an EventView that matches this widget's definition. The // `eventView` from the props is the _landing page_ EventView, which is different const widgetEventView = makeEventViewForWidget(props.eventView, chartDefinition); const showNewWidgetDesign = organization.features.includes( 'performance-new-widget-designs' ); const widgetProps = { ...chartDefinition, chartSetting, chartDefinition, InteractiveTitle: showNewWidgetDesign && allowedCharts.length > 2 ? (containerProps: any) => ( ) : null, ContainerActions: !showNewWidgetDesign ? (containerProps: any) => ( ) : null, }; const passedProps = pick(props, [ 'eventView', 'location', 'organization', 'chartHeight', 'withStaticFilters', ]); const titleTooltip = showNewWidgetDesign ? '' : widgetProps.titleTooltip; switch (widgetProps.dataType) { case GenericPerformanceWidgetDataType.TRENDS: return ( ); case GenericPerformanceWidgetDataType.AREA: return ( ); case GenericPerformanceWidgetDataType.VITALS: return ( ); case GenericPerformanceWidgetDataType.LINE_LIST: return ( ); case GenericPerformanceWidgetDataType.HISTOGRAM: return ( ); case GenericPerformanceWidgetDataType.STACKED_AREA: return ; case GenericPerformanceWidgetDataType.PERFORMANCE_SCORE_LIST: return ; case GenericPerformanceWidgetDataType.PERFORMANCE_SCORE: return ; case GenericPerformanceWidgetDataType.SLOW_SCREENS_BY_TTID: case GenericPerformanceWidgetDataType.SLOW_SCREENS_BY_COLD_START: case GenericPerformanceWidgetDataType.SLOW_SCREENS_BY_WARM_START: return ; default: throw new Error(`Widget type "${widgetProps.dataType}" has no implementation.`); } } export function WidgetInteractiveTitle({ chartSetting, eventView, setChartSetting, allowedCharts, rowChartSettings, }: { allowedCharts: PerformanceWidgetSetting[]; chartSetting: PerformanceWidgetSetting; eventView: EventView; rowChartSettings: PerformanceWidgetSetting[]; setChartSetting: (setting: PerformanceWidgetSetting) => void; }) { const organization = useOrganization(); const menuOptions: SelectOption[] = []; const settingsMap = WIDGET_DEFINITIONS({organization}); for (const setting of allowedCharts) { const options = settingsMap[setting]; menuOptions.push({ value: setting, label: options.title, disabled: setting !== chartSetting && rowChartSettings.includes(setting), }); } const chartDefinition = WIDGET_DEFINITIONS({organization})[chartSetting]; if (chartDefinition.allowsOpenInDiscover) { menuOptions.push({label: t('Open in Discover'), value: 'open_in_discover'}); } const handleChange = (option: {value: string | number}) => { if (option.value === 'open_in_discover') { browserHistory.push( normalizeUrl(getEventViewDiscoverPath(organization, eventView)) ); } else { setChartSetting(option.value as PerformanceWidgetSetting); } }; return ( ); } const StyledCompactSelect = styled(CompactSelect)` /* Reset font-weight set by HeaderTitleLegend, buttons are already bold and * setting this higher up causes it to trickle into the menues */ font-weight: ${p => p.theme.fontWeightNormal}; margin: -${space(0.5)} -${space(1)} -${space(0.25)}; min-width: 0; button { padding: ${space(0.5)} ${space(1)}; font-size: ${p => p.theme.fontSizeLarge}; } `; export function WidgetContainerActions({ chartSetting, eventView, setChartSetting, allowedCharts, rowChartSettings, }: { allowedCharts: PerformanceWidgetSetting[]; chartSetting: PerformanceWidgetSetting; eventView: EventView; rowChartSettings: PerformanceWidgetSetting[]; setChartSetting: (setting: PerformanceWidgetSetting) => void; }) { const organization = useOrganization(); const menuOptions: SelectOption[] = []; const settingsMap = WIDGET_DEFINITIONS({organization}); for (const setting of allowedCharts) { const options = settingsMap[setting]; menuOptions.push({ value: setting, label: options.title, disabled: setting !== chartSetting && rowChartSettings.includes(setting), }); } const chartDefinition = WIDGET_DEFINITIONS({organization})[chartSetting]; function handleWidgetActionChange(value: string) { if (value === 'open_in_discover') { browserHistory.push( normalizeUrl(getEventViewDiscoverPath(organization, eventView)) ); } } return ( ( } /> )} position="bottom-end" > setChartSetting(opt.value)} /> {chartDefinition.allowsOpenInDiscover && ( handleWidgetActionChange(opt.value)} /> )} ); } const getEventViewDiscoverPath = ( organization: Organization, eventView: EventView ): string => { const discoverUrlTarget = eventView.getResultsViewUrlTarget( organization.slug, false, hasDatasetSelector(organization) ? SavedQueryDatasets.TRANSACTIONS : undefined ); // The landing page EventView has some additional conditions, but // `EventView#getResultsViewUrlTarget` omits those! Get them manually discoverUrlTarget.query.query = eventView.getQueryWithAdditionalConditions(); return `${discoverUrlTarget.pathname}?${qs.stringify( omit(discoverUrlTarget.query, ['widths']) // Column widths are not useful in this case )}`; }; /** * Constructs an `EventView` that matches a widget's chart definition. * @param baseEventView Any valid event view. The easiest way to make a new EventView is to clone an existing one, because `EventView#constructor` takes too many abstract arguments * @param chartDefinition */ const makeEventViewForWidget = ( baseEventView: EventView, chartDefinition: ChartDefinition ): EventView => { const widgetEventView = baseEventView.clone(); widgetEventView.name = chartDefinition.title; widgetEventView.yAxis = chartDefinition.fields[0]; // All current widgets only have one field widgetEventView.display = DisplayModes.PREVIOUS; widgetEventView.fields = ['transaction', 'project', ...chartDefinition.fields].map( fieldName => ({field: fieldName}) as Field ); return widgetEventView; }; const WidgetContainer = withOrganization(_WidgetContainer); export default WidgetContainer;