import {Component} from 'react'; import {browserHistory} from 'react-router'; import * as Sentry from '@sentry/react'; import {Location} from 'history'; import {fetchTotalCount} from 'sentry/actionCreators/events'; import {Client} from 'sentry/api'; import OptionSelector from 'sentry/components/charts/optionSelector'; import { ChartControls, InlineContainer, SectionHeading, SectionValue, } from 'sentry/components/charts/styles'; import {t} from 'sentry/locale'; import {Organization} from 'sentry/types'; import {trackAnalyticsEvent} from 'sentry/utils/analytics'; import EventView, {isAPIPayloadSimilar} from 'sentry/utils/discover/eventView'; import {getAxisOptions, TooltipOption} from '../data'; type Props = { api: Client; eventView: EventView; leftAxis: string; location: Location; organization: Organization; rightAxis: string; options?: TooltipOption[]; }; type State = { totalValues: null | number; }; class ChartFooter extends Component { state: State = { totalValues: null, }; componentDidMount() { this.mounted = true; this.fetchTotalCount(); } componentDidUpdate(prevProps: Props) { const orgSlugHasChanged = this.props.organization.slug !== prevProps.organization.slug; const shouldRefetch = this.shouldRefetchData(prevProps); if ((orgSlugHasChanged || shouldRefetch) && this.props.eventView.isValid()) { this.fetchTotalCount(); } } componentWillUnmount() { this.mounted = false; } shouldRefetchData = (prevProps: Props): boolean => { const thisAPIPayload = this.props.eventView.getEventsAPIPayload(this.props.location); const otherAPIPayload = prevProps.eventView.getEventsAPIPayload(prevProps.location); return !isAPIPayloadSimilar(thisAPIPayload, otherAPIPayload); }; handleSelectorChange(key: string, value: string) { const {location, organization} = this.props; trackAnalyticsEvent({ eventKey: 'performance_views.overview.change_chart', eventName: 'Performance Views: Change Overview Chart', organization_id: parseInt(organization.id, 10), metric: value, }); browserHistory.push({ pathname: location.pathname, query: {...location.query, [key]: value}, }); } mounted: boolean = false; async fetchTotalCount() { const {api, organization, location, eventView} = this.props; if (!eventView.isValid() || !this.mounted) { return; } try { const totals = await fetchTotalCount( api, organization.slug, eventView.getEventsAPIPayload(location) ); if (this.mounted) { this.setState({totalValues: totals}); } } catch (err) { Sentry.captureException(err); } } render() { const {leftAxis, organization, rightAxis} = this.props; const {totalValues} = this.state; const value = typeof totalValues === 'number' ? totalValues.toLocaleString() : '-'; const options = this.props.options || getAxisOptions(organization); const leftOptions = options.map(opt => ({ ...opt, disabled: opt.value === rightAxis, })); const rightOptions = options.map(opt => ({ ...opt, disabled: opt.value === leftAxis, })); return ( {t('Total Events')} {value} this.handleSelectorChange('left', val)} /> this.handleSelectorChange('right', val)} /> ); } } export default ChartFooter;