import {Component, Fragment} from 'react';
import type {Theme} from '@emotion/react';
import {useTheme} from '@emotion/react';
import type {LegendComponentOption, LineSeriesOption} from 'echarts';
import isEqual from 'lodash/isEqual';
import type {Client} from 'sentry/api';
import {BarChart} from 'sentry/components/charts/barChart';
import type {ZoomRenderProps} from 'sentry/components/charts/chartZoom';
import ChartZoom from 'sentry/components/charts/chartZoom';
import ErrorPanel from 'sentry/components/charts/errorPanel';
import type {LineChartProps} from 'sentry/components/charts/lineChart';
import {LineChart} from 'sentry/components/charts/lineChart';
import ReleaseSeries from 'sentry/components/charts/releaseSeries';
import StackedAreaChart from 'sentry/components/charts/stackedAreaChart';
import {HeaderTitleLegend} from 'sentry/components/charts/styles';
import TransitionChart from 'sentry/components/charts/transitionChart';
import TransparentLoadingMask from 'sentry/components/charts/transparentLoadingMask';
import {RELEASE_LINES_THRESHOLD} from 'sentry/components/charts/utils';
import QuestionTooltip from 'sentry/components/questionTooltip';
import {IconWarning} from 'sentry/icons';
import {t} from 'sentry/locale';
import type {PageFilters} from 'sentry/types/core';
import type {EChartEventHandler, Series} from 'sentry/types/echarts';
import type {Organization} from 'sentry/types/organization';
import getDynamicText from 'sentry/utils/getDynamicText';
import {MINUTES_THRESHOLD_TO_DISPLAY_SECONDS} from 'sentry/utils/sessions';
import withPageFilters from 'sentry/utils/withPageFilters';
import {displayCrashFreePercent} from 'sentry/views/releases/utils';
import {sessionTerm} from 'sentry/views/releases/utils/sessionTerm';
import {DisplayModes} from '../projectCharts';
import ProjectSessionsAnrRequest from './projectSessionsAnrRequest';
import ProjectSessionsChartRequest from './projectSessionsChartRequest';
type Props = {
api: Client;
displayMode:
| DisplayModes.SESSIONS
| DisplayModes.STABILITY_USERS
| DisplayModes.ANR_RATE
| DisplayModes.FOREGROUND_ANR_RATE
| DisplayModes.STABILITY;
onTotalValuesChange: (value: number | null) => void;
organization: Organization;
selection: PageFilters;
title: string;
disablePrevious?: boolean;
help?: string;
query?: string;
};
function ProjectBaseSessionsChart({
title,
organization,
selection,
api,
onTotalValuesChange,
displayMode,
help,
disablePrevious,
query,
}: Props) {
const theme = useTheme();
const {projects, environments, datetime} = selection;
const {start, end, period, utc} = datetime;
const Request = [DisplayModes.ANR_RATE, DisplayModes.FOREGROUND_ANR_RATE].includes(
displayMode
)
? ProjectSessionsAnrRequest
: ProjectSessionsChartRequest;
return (
{getDynamicText({
value: (
{zoomRenderProps => (
{({
errored,
loading,
reloading,
timeseriesData,
previousTimeseriesData,
additionalSeries,
}) => (
{({releaseSeries}) => {
if (errored) {
return (
);
}
return (
{title}
{help && (
)}
);
}}
)}
)}
),
fixed: `${title} Chart`,
})}
);
}
type ChartProps = {
displayMode:
| DisplayModes.SESSIONS
| DisplayModes.STABILITY
| DisplayModes.STABILITY_USERS
| DisplayModes.ANR_RATE
| DisplayModes.FOREGROUND_ANR_RATE;
releaseSeries: Series[];
reloading: boolean;
theme: Theme;
timeSeries: Series[];
zoomRenderProps: ZoomRenderProps;
additionalSeries?: LineSeriesOption[];
previousTimeSeries?: Series[];
};
type ChartState = {
forceUpdate: boolean;
seriesSelection: Record;
};
class Chart extends Component {
state: ChartState = {
seriesSelection: {},
forceUpdate: false,
};
shouldComponentUpdate(nextProps: ChartProps, nextState: ChartState) {
if (nextState.forceUpdate) {
return true;
}
if (!isEqual(this.state.seriesSelection, nextState.seriesSelection)) {
return true;
}
if (
nextProps.releaseSeries !== this.props.releaseSeries &&
!nextProps.reloading &&
!this.props.reloading
) {
return true;
}
if (this.props.reloading && !nextProps.reloading) {
return true;
}
if (nextProps.timeSeries !== this.props.timeSeries) {
return true;
}
return false;
}
// inspired by app/components/charts/eventsChart.tsx@handleLegendSelectChanged
handleLegendSelectChanged: EChartEventHandler<{
name: string;
selected: Record;
type: 'legendselectchanged';
}> = ({selected}) => {
const seriesSelection = Object.keys(selected).reduce((state, key) => {
state[key] = selected[key];
return state;
}, {});
// we have to force an update here otherwise ECharts will
// update its internal state and disable the series
this.setState({seriesSelection, forceUpdate: true}, () =>
this.setState({forceUpdate: false})
);
};
get isCrashFree() {
const {displayMode} = this.props;
return [DisplayModes.STABILITY, DisplayModes.STABILITY_USERS].includes(displayMode);
}
get isAnr() {
const {displayMode} = this.props;
return [DisplayModes.ANR_RATE, DisplayModes.FOREGROUND_ANR_RATE].includes(
displayMode
);
}
get legend(): LegendComponentOption {
const {theme, timeSeries, previousTimeSeries, releaseSeries, additionalSeries} =
this.props;
const {seriesSelection} = this.state;
const hideReleasesByDefault =
(releaseSeries[0] as any)?.markLine?.data.length >= RELEASE_LINES_THRESHOLD;
const hideHealthyByDefault = timeSeries
.filter(s => sessionTerm.healthy !== s.seriesName)
.some(s => s.data.some(d => d.value > 0));
const selected =
Object.keys(seriesSelection).length === 0 &&
(hideReleasesByDefault || hideHealthyByDefault)
? {
[t('Releases')]: !hideReleasesByDefault,
[sessionTerm.healthy]: !hideHealthyByDefault,
}
: seriesSelection;
return {
right: 10,
top: 0,
icon: 'circle',
itemHeight: 8,
itemWidth: 8,
itemGap: 12,
align: 'left' as const,
textStyle: {
color: theme.textColor,
verticalAlign: 'top',
fontSize: 11,
fontFamily: theme.text.family,
},
data: [
...timeSeries.map(s => s.seriesName),
...(previousTimeSeries ?? []).map(s => s.seriesName),
...(additionalSeries ?? []).map(s => s.name?.toString() ?? ''),
...releaseSeries.map(s => s.seriesName),
],
selected,
};
}
get chartOptions(): Omit {
return {
grid: {left: '10px', right: '10px', top: '40px', bottom: '0px'},
seriesOptions: {
showSymbol: false,
},
tooltip: {
trigger: 'axis',
truncate: 80,
valueFormatter: (value: number | null) => {
if (value === null) {
return '\u2014';
}
if (this.isCrashFree) {
return displayCrashFreePercent(value, 0, 3);
}
if (this.isAnr) {
return displayCrashFreePercent(value, 0, 3, false);
}
return typeof value === 'number' ? value.toLocaleString() : value;
},
},
yAxis: this.isCrashFree
? {
axisLabel: {
formatter: (value: number) => displayCrashFreePercent(value),
},
scale: true,
max: 100,
}
: this.isAnr
? {
axisLabel: {
formatter: (value: number) => displayCrashFreePercent(value, 0, 3, false),
},
scale: true,
}
: {min: 0},
};
}
render() {
const {
zoomRenderProps,
timeSeries,
previousTimeSeries,
releaseSeries,
additionalSeries,
} = this.props;
const ChartComponent = this.isCrashFree
? LineChart
: this.isAnr
? BarChart
: StackedAreaChart;
return (
);
}
}
export default withPageFilters(ProjectBaseSessionsChart);