/** * External dependencies */ import { __ } from '@wordpress/i18n'; import { Component } from '@wordpress/element'; import { compose } from '@wordpress/compose'; import { format as formatDate } from '@wordpress/date'; import { withSelect } from '@wordpress/data'; import { get, isEqual } from 'lodash'; import PropTypes from 'prop-types'; import { Chart } from '@woocommerce/components'; import { getReportChartData, getTooltipValueFormat, SETTINGS_STORE_NAME, REPORTS_STORE_NAME, } from '@woocommerce/data'; import { getAllowedIntervalsForQuery, getCurrentDates, getDateFormatsForInterval, getIntervalForQuery, getChartTypeForQuery, getPreviousDate, } from '@woocommerce/date'; import { CurrencyContext } from '@woocommerce/currency'; /** * Internal dependencies */ import ReportError from '../report-error'; import { getChartMode, getSelectedFilter, createDateFormatter } from './utils'; /** * Component that renders the chart in reports. */ export class ReportChart extends Component { shouldComponentUpdate( nextProps ) { if ( nextProps.isRequesting !== this.props.isRequesting || nextProps.primaryData.isRequesting !== this.props.primaryData.isRequesting || nextProps.secondaryData.isRequesting !== this.props.secondaryData.isRequesting || ! isEqual( nextProps.query, this.props.query ) ) { return true; } return false; } getItemChartData() { const { primaryData, selectedChart } = this.props; const chartData = primaryData.data.intervals.map( function ( interval ) { const intervalData = {}; interval.subtotals.segments.forEach( function ( segment ) { if ( segment.segment_label ) { const label = intervalData[ segment.segment_label ] ? segment.segment_label + ' (#' + segment.segment_id + ')' : segment.segment_label; intervalData[ segment.segment_id ] = { label, value: segment.subtotals[ selectedChart.key ] || 0, }; } } ); return { date: formatDate( 'Y-m-d\\TH:i:s', interval.date_start ), ...intervalData, }; } ); return chartData; } getTimeChartData() { const { query, primaryData, secondaryData, selectedChart, defaultDateRange, } = this.props; const currentInterval = getIntervalForQuery( query, defaultDateRange ); const { primary, secondary } = getCurrentDates( query, defaultDateRange ); const chartData = primaryData.data.intervals.map( function ( interval, index ) { const secondaryDate = getPreviousDate( interval.date_start, primary.after, secondary.after, query.compare, currentInterval ); const secondaryInterval = secondaryData.data.intervals[ index ]; return { date: formatDate( 'Y-m-d\\TH:i:s', interval.date_start ), primary: { label: `${ primary.label } (${ primary.range })`, labelDate: interval.date_start, value: interval.subtotals[ selectedChart.key ] || 0, }, secondary: { label: `${ secondary.label } (${ secondary.range })`, labelDate: secondaryDate.format( 'YYYY-MM-DD HH:mm:ss' ), value: ( secondaryInterval && secondaryInterval.subtotals[ selectedChart.key ] ) || 0, }, }; } ); return chartData; } getTimeChartTotals() { const { primaryData, secondaryData, selectedChart } = this.props; return { primary: get( primaryData, [ 'data', 'totals', selectedChart.key ], null ), secondary: get( secondaryData, [ 'data', 'totals', selectedChart.key ], null ), }; } renderChart( mode, isRequesting, chartData, legendTotals ) { const { emptySearchResults, filterParam, interactiveLegend, itemsLabel, legendPosition, path, query, selectedChart, showHeaderControls, primaryData, defaultDateRange, } = this.props; const currentInterval = getIntervalForQuery( query, defaultDateRange ); const allowedIntervals = getAllowedIntervalsForQuery( query, defaultDateRange ); const formats = getDateFormatsForInterval( currentInterval, primaryData.data.intervals.length, { type: 'php' } ); const emptyMessage = emptySearchResults ? __( 'No data for the current search', 'woocommerce' ) : __( 'No data for the selected date range', 'woocommerce' ); const { formatAmount, getCurrencyConfig } = this.context; return ( ); } renderItemComparison() { const { isRequesting, primaryData } = this.props; if ( primaryData.isError ) { return ; } const isChartRequesting = isRequesting || primaryData.isRequesting; const chartData = this.getItemChartData(); return this.renderChart( 'item-comparison', isChartRequesting, chartData ); } renderTimeComparison() { const { isRequesting, primaryData, secondaryData } = this.props; if ( ! primaryData || primaryData.isError || secondaryData.isError ) { return ; } const isChartRequesting = isRequesting || primaryData.isRequesting || secondaryData.isRequesting; const chartData = this.getTimeChartData(); const legendTotals = this.getTimeChartTotals(); return this.renderChart( 'time-comparison', isChartRequesting, chartData, legendTotals ); } render() { const { mode } = this.props; if ( mode === 'item-comparison' ) { return this.renderItemComparison(); } return this.renderTimeComparison(); } } ReportChart.contextType = CurrencyContext; ReportChart.propTypes = { /** * Filters available for that report. */ filters: PropTypes.array, /** * Whether there is an API call running. */ isRequesting: PropTypes.bool, /** * Label describing the legend items. */ itemsLabel: PropTypes.string, /** * Allows specifying properties different from the `endpoint` that will be used * to limit the items when there is an active search. */ limitProperties: PropTypes.array, /** * `items-comparison` (default) or `time-comparison`, this is used to generate correct * ARIA properties. */ mode: PropTypes.string, /** * Current path */ path: PropTypes.string.isRequired, /** * Primary data to display in the chart. */ primaryData: PropTypes.object, /** * The query string represented in object form. */ query: PropTypes.object.isRequired, /** * Secondary data to display in the chart. */ secondaryData: PropTypes.object, /** * Properties of the selected chart. */ selectedChart: PropTypes.shape( { /** * Key of the selected chart. */ key: PropTypes.string.isRequired, /** * Chart label. */ label: PropTypes.string.isRequired, /** * Order query argument. */ order: PropTypes.oneOf( [ 'asc', 'desc' ] ), /** * Order by query argument. */ orderby: PropTypes.string, /** * Number type for formatting. */ type: PropTypes.oneOf( [ 'average', 'number', 'currency' ] ).isRequired, } ).isRequired, }; ReportChart.defaultProps = { isRequesting: false, primaryData: { data: { intervals: [], }, isError: false, isRequesting: false, }, secondaryData: { data: { intervals: [], }, isError: false, isRequesting: false, }, }; export default compose( withSelect( ( select, props ) => { const { charts, endpoint, filters, isRequesting, limitProperties, query, advancedFilters, } = props; const limitBy = limitProperties || [ endpoint ]; const selectedFilter = getSelectedFilter( filters, query ); const filterParam = get( selectedFilter, [ 'settings', 'param' ] ); const chartMode = props.mode || getChartMode( selectedFilter, query ) || 'time-comparison'; const { woocommerce_default_date_range: defaultDateRange } = select( SETTINGS_STORE_NAME ).getSetting( 'wc_admin', 'wcAdminSettings' ); /* eslint @wordpress/no-unused-vars-before-return: "off" */ const reportStoreSelector = select( REPORTS_STORE_NAME ); const newProps = { mode: chartMode, filterParam, defaultDateRange, }; if ( isRequesting ) { return newProps; } const hasLimitByParam = limitBy.some( ( item ) => query[ item ] && query[ item ].length ); if ( query.search && ! hasLimitByParam ) { return { ...newProps, emptySearchResults: true, }; } const fields = charts && charts.map( ( chart ) => chart.key ); const primaryData = getReportChartData( { endpoint, dataType: 'primary', query, selector: reportStoreSelector, limitBy, filters, advancedFilters, defaultDateRange, fields, } ); if ( chartMode === 'item-comparison' ) { return { ...newProps, primaryData, }; } const secondaryData = getReportChartData( { endpoint, dataType: 'secondary', query, selector: reportStoreSelector, limitBy, filters, advancedFilters, defaultDateRange, fields, } ); return { ...newProps, primaryData, secondaryData, }; } ) )( ReportChart );