2018-09-03 15:25:38 +00:00
|
|
|
/**
|
|
|
|
* External dependencies
|
|
|
|
*/
|
2019-02-14 03:31:27 +00:00
|
|
|
import { find, forEach, isNull, get, includes } from 'lodash';
|
2019-01-07 10:41:46 +00:00
|
|
|
import moment from 'moment';
|
2018-09-03 15:25:38 +00:00
|
|
|
|
2018-10-30 18:57:48 +00:00
|
|
|
/**
|
|
|
|
* WooCommerce dependencies
|
|
|
|
*/
|
2020-02-14 02:23:21 +00:00
|
|
|
import {
|
|
|
|
appendTimestamp,
|
|
|
|
getCurrentDates,
|
|
|
|
getIntervalForQuery,
|
|
|
|
} from 'lib/date';
|
|
|
|
import {
|
|
|
|
flattenFilters,
|
|
|
|
getActiveFiltersFromQuery,
|
|
|
|
getUrlKey,
|
|
|
|
} from '@woocommerce/navigation';
|
2019-11-21 21:51:52 +00:00
|
|
|
import { formatCurrency } from 'lib/currency-format';
|
2018-10-30 18:57:48 +00:00
|
|
|
|
2018-09-03 15:25:38 +00:00
|
|
|
/**
|
|
|
|
* Internal dependencies
|
|
|
|
*/
|
2019-01-30 07:22:10 +00:00
|
|
|
import { MAX_PER_PAGE, QUERY_DEFAULTS } from 'wc-api/constants';
|
2018-11-22 23:12:12 +00:00
|
|
|
import * as reportsUtils from './utils';
|
2018-10-18 09:34:37 +00:00
|
|
|
|
2019-03-21 03:25:05 +00:00
|
|
|
/**
|
|
|
|
* Add filters and advanced filters values to a query object.
|
|
|
|
*
|
2020-02-14 02:23:21 +00:00
|
|
|
* @param {Object} options arguments
|
|
|
|
* @param {string} options.endpoint Report API Endpoint
|
2019-03-21 03:25:05 +00:00
|
|
|
* @param {Object} options.query Query parameters in the url
|
|
|
|
* @param {Array} options.limitBy Properties used to limit the results. It will be used in the API call to send the IDs.
|
|
|
|
* @param {Array} [options.filters] config filters
|
|
|
|
* @param {Object} [options.advancedFilters] config advanced filters
|
2020-02-14 02:23:21 +00:00
|
|
|
* @return {Object} A query object with the values from filters and advanced fitlters applied.
|
2019-03-21 03:25:05 +00:00
|
|
|
*/
|
|
|
|
export function getFilterQuery( options ) {
|
2020-02-14 02:23:21 +00:00
|
|
|
const {
|
|
|
|
endpoint,
|
|
|
|
query,
|
|
|
|
limitBy,
|
|
|
|
filters = [],
|
|
|
|
advancedFilters = {},
|
|
|
|
} = options;
|
2019-02-01 09:55:19 +00:00
|
|
|
if ( query.search ) {
|
2019-03-07 02:57:22 +00:00
|
|
|
const limitProperties = limitBy || [ endpoint ];
|
|
|
|
return limitProperties.reduce( ( result, limitProperty ) => {
|
|
|
|
result[ limitProperty ] = query[ limitProperty ];
|
|
|
|
return result;
|
|
|
|
}, {} );
|
2019-02-01 09:55:19 +00:00
|
|
|
}
|
|
|
|
|
2019-03-21 03:25:05 +00:00
|
|
|
return filters
|
2020-02-14 02:23:21 +00:00
|
|
|
.map( ( filter ) =>
|
|
|
|
getQueryFromConfig( filter, advancedFilters, query )
|
|
|
|
)
|
|
|
|
.reduce(
|
|
|
|
( result, configQuery ) => Object.assign( result, configQuery ),
|
|
|
|
{}
|
|
|
|
);
|
2018-10-29 01:30:24 +00:00
|
|
|
}
|
2018-10-18 09:34:37 +00:00
|
|
|
|
2019-02-14 03:31:27 +00:00
|
|
|
// Some stats endpoints don't have interval data, so they can ignore after/before params and omit that part of the response.
|
|
|
|
const noIntervalEndpoints = [ 'stock', 'customers' ];
|
|
|
|
|
2019-01-16 20:08:04 +00:00
|
|
|
/**
|
|
|
|
* Add timestamp to advanced filter parameters involving date. The api
|
|
|
|
* expects a timestamp for these values similar to `before` and `after`.
|
|
|
|
*
|
2020-02-14 02:23:21 +00:00
|
|
|
* @param {Object} config - advancedFilters config object.
|
|
|
|
* @param {Object} activeFilter - an active filter.
|
|
|
|
* @return {Object} - an active filter with timestamp added to date values.
|
2019-01-16 20:08:04 +00:00
|
|
|
*/
|
2019-01-15 02:40:12 +00:00
|
|
|
export function timeStampFilterDates( config, activeFilter ) {
|
2019-01-16 20:08:04 +00:00
|
|
|
const advancedFilterConfig = config.filters[ activeFilter.key ];
|
2020-02-14 02:23:21 +00:00
|
|
|
if ( get( advancedFilterConfig, [ 'input', 'component' ] ) !== 'Date' ) {
|
2019-01-17 00:57:11 +00:00
|
|
|
return activeFilter;
|
|
|
|
}
|
2019-01-16 20:37:43 +00:00
|
|
|
|
2019-01-17 00:57:11 +00:00
|
|
|
const { rule, value } = activeFilter;
|
|
|
|
const timeOfDayMap = {
|
|
|
|
after: 'start',
|
|
|
|
before: 'end',
|
|
|
|
};
|
|
|
|
// If the value is an array, it signifies "between" values which must have a timestamp
|
|
|
|
// appended to each value.
|
|
|
|
if ( Array.isArray( value ) ) {
|
|
|
|
const [ after, before ] = value;
|
2019-01-16 20:37:43 +00:00
|
|
|
return Object.assign( {}, activeFilter, {
|
2019-01-17 00:57:11 +00:00
|
|
|
value: [
|
|
|
|
appendTimestamp( moment( after ), timeOfDayMap.after ),
|
|
|
|
appendTimestamp( moment( before ), timeOfDayMap.before ),
|
|
|
|
],
|
2019-01-16 20:37:43 +00:00
|
|
|
} );
|
2019-01-15 02:40:12 +00:00
|
|
|
}
|
2019-01-17 00:57:11 +00:00
|
|
|
|
|
|
|
return Object.assign( {}, activeFilter, {
|
|
|
|
value: appendTimestamp( moment( value ), timeOfDayMap[ rule ] ),
|
|
|
|
} );
|
2019-01-15 02:40:12 +00:00
|
|
|
}
|
2019-01-15 00:02:24 +00:00
|
|
|
|
2018-10-29 01:30:24 +00:00
|
|
|
export function getQueryFromConfig( config, advancedFilters, query ) {
|
|
|
|
const queryValue = query[ config.param ];
|
|
|
|
|
|
|
|
if ( ! queryValue ) {
|
2018-10-18 09:34:37 +00:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2020-02-14 02:23:21 +00:00
|
|
|
if ( queryValue === 'advanced' ) {
|
|
|
|
const activeFilters = getActiveFiltersFromQuery(
|
|
|
|
query,
|
|
|
|
advancedFilters.filters
|
|
|
|
);
|
2018-10-18 09:34:37 +00:00
|
|
|
|
|
|
|
if ( activeFilters.length === 0 ) {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2020-02-14 02:23:21 +00:00
|
|
|
return activeFilters
|
|
|
|
.map( ( filter ) =>
|
|
|
|
timeStampFilterDates( advancedFilters, filter )
|
|
|
|
)
|
|
|
|
.reduce(
|
|
|
|
( result, activeFilter ) => {
|
|
|
|
const { key, rule, value } = activeFilter;
|
|
|
|
result[ getUrlKey( key, rule ) ] = value;
|
|
|
|
return result;
|
|
|
|
},
|
|
|
|
{ match: query.match || 'all' }
|
|
|
|
);
|
2018-10-18 09:34:37 +00:00
|
|
|
}
|
|
|
|
|
2020-02-14 02:23:21 +00:00
|
|
|
const filter = find( flattenFilters( config.filters ), {
|
|
|
|
value: queryValue,
|
|
|
|
} );
|
2018-10-18 09:34:37 +00:00
|
|
|
|
2018-10-29 01:30:24 +00:00
|
|
|
if ( ! filter ) {
|
2018-10-18 09:34:37 +00:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2018-10-29 01:30:24 +00:00
|
|
|
if ( filter.settings && filter.settings.param ) {
|
|
|
|
const { param } = filter.settings;
|
2018-10-18 09:34:37 +00:00
|
|
|
|
|
|
|
if ( query[ param ] ) {
|
|
|
|
return {
|
|
|
|
[ param ]: query[ param ],
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2019-12-05 22:38:26 +00:00
|
|
|
return {
|
|
|
|
[ config.param ]: queryValue,
|
|
|
|
};
|
2018-10-18 09:34:37 +00:00
|
|
|
}
|
2018-09-03 15:25:38 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns true if a report object is empty.
|
|
|
|
*
|
2019-02-14 03:31:27 +00:00
|
|
|
* @param {Object} report Report to check
|
2020-02-14 02:23:21 +00:00
|
|
|
* @param {string} endpoint Endpoint slug
|
|
|
|
* @return {boolean} True if report is data is empty.
|
2018-09-03 15:25:38 +00:00
|
|
|
*/
|
2019-02-14 03:31:27 +00:00
|
|
|
export function isReportDataEmpty( report, endpoint ) {
|
2018-09-03 15:25:38 +00:00
|
|
|
if ( ! report ) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if ( ! report.data ) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if ( ! report.data.totals || isNull( report.data.totals ) ) {
|
|
|
|
return true;
|
|
|
|
}
|
2019-02-14 03:31:27 +00:00
|
|
|
|
|
|
|
const checkIntervals = ! includes( noIntervalEndpoints, endpoint );
|
2020-02-14 02:23:21 +00:00
|
|
|
if (
|
|
|
|
checkIntervals &&
|
|
|
|
( ! report.data.intervals || report.data.intervals.length === 0 )
|
|
|
|
) {
|
2018-09-03 15:25:38 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-10-17 23:06:33 +00:00
|
|
|
/**
|
|
|
|
* Constructs and returns a query associated with a Report data request.
|
2020-02-14 02:23:21 +00:00
|
|
|
*
|
|
|
|
* @param {Object} options arguments
|
|
|
|
* @param {string} options.endpoint Report API Endpoint
|
|
|
|
* @param {string} options.dataType 'primary' or 'secondary'.
|
2019-03-21 03:25:05 +00:00
|
|
|
* @param {Object} options.query Query parameters in the url.
|
|
|
|
* @param {Array} options.limitBy Properties used to limit the results. It will be used in the API call to send the IDs.
|
2020-03-25 03:20:17 +00:00
|
|
|
* @param {string} options.defaultDateRange User specified default date range.
|
2020-02-14 02:23:21 +00:00
|
|
|
* @return {Object} data request query parameters.
|
2018-10-17 23:06:33 +00:00
|
|
|
*/
|
2019-03-21 03:25:05 +00:00
|
|
|
function getRequestQuery( options ) {
|
|
|
|
const { endpoint, dataType, query } = options;
|
2020-03-25 03:20:17 +00:00
|
|
|
const datesFromQuery = getCurrentDates( query, options.defaultDateRange );
|
2018-10-17 23:02:31 +00:00
|
|
|
const interval = getIntervalForQuery( query );
|
2019-03-21 03:25:05 +00:00
|
|
|
const filterQuery = getFilterQuery( options );
|
2019-01-07 10:41:46 +00:00
|
|
|
const end = datesFromQuery[ dataType ].before;
|
2019-02-14 03:31:27 +00:00
|
|
|
|
|
|
|
const noIntervals = includes( noIntervalEndpoints, endpoint );
|
|
|
|
return noIntervals
|
|
|
|
? { ...filterQuery }
|
|
|
|
: {
|
|
|
|
order: 'asc',
|
|
|
|
interval,
|
|
|
|
per_page: MAX_PER_PAGE,
|
2020-02-14 02:23:21 +00:00
|
|
|
after: appendTimestamp(
|
|
|
|
datesFromQuery[ dataType ].after,
|
|
|
|
'start'
|
|
|
|
),
|
2019-02-14 03:31:27 +00:00
|
|
|
before: appendTimestamp( end, 'end' ),
|
|
|
|
segmentby: query.segmentby,
|
|
|
|
...filterQuery,
|
2020-02-14 02:23:21 +00:00
|
|
|
};
|
2018-10-17 23:02:31 +00:00
|
|
|
}
|
|
|
|
|
2018-10-11 18:45:01 +00:00
|
|
|
/**
|
|
|
|
* Returns summary number totals needed to render a report page.
|
|
|
|
*
|
2020-02-14 02:23:21 +00:00
|
|
|
* @param {Object} options arguments
|
|
|
|
* @param {string} options.endpoint Report API Endpoint
|
2019-03-21 03:25:05 +00:00
|
|
|
* @param {Object} options.query Query parameters in the url
|
|
|
|
* @param {Object} options.select Instance of @wordpress/select
|
|
|
|
* @param {Array} options.limitBy Properties used to limit the results. It will be used in the API call to send the IDs.
|
2020-03-25 03:20:17 +00:00
|
|
|
* @param {string} options.defaultDateRange User specified default date range.
|
2019-02-27 16:28:18 +00:00
|
|
|
* @return {Object} Object containing summary number responses.
|
2018-10-11 18:45:01 +00:00
|
|
|
*/
|
2019-03-21 03:25:05 +00:00
|
|
|
export function getSummaryNumbers( options ) {
|
|
|
|
const { endpoint, select } = options;
|
2020-02-14 02:23:21 +00:00
|
|
|
const {
|
|
|
|
getReportStats,
|
|
|
|
getReportStatsError,
|
|
|
|
isReportStatsRequesting,
|
|
|
|
} = select( 'wc-api' );
|
2018-10-11 18:45:01 +00:00
|
|
|
const response = {
|
|
|
|
isRequesting: false,
|
|
|
|
isError: false,
|
|
|
|
totals: {
|
|
|
|
primary: null,
|
|
|
|
secondary: null,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2019-03-21 03:25:05 +00:00
|
|
|
const primaryQuery = getRequestQuery( { ...options, dataType: 'primary' } );
|
2020-02-25 01:18:52 +00:00
|
|
|
|
|
|
|
// Disable eslint rule requiring `getReportStats` to be defined below because the next two statements
|
|
|
|
// depend on `getReportStats` to have been called.
|
|
|
|
// eslint-disable-next-line @wordpress/no-unused-vars-before-return
|
|
|
|
const primary = getReportStats( endpoint, primaryQuery );
|
|
|
|
|
2019-03-21 03:25:05 +00:00
|
|
|
if ( isReportStatsRequesting( endpoint, primaryQuery ) ) {
|
2018-10-11 18:45:01 +00:00
|
|
|
return { ...response, isRequesting: true };
|
2019-03-21 03:25:05 +00:00
|
|
|
} else if ( getReportStatsError( endpoint, primaryQuery ) ) {
|
2018-10-11 18:45:01 +00:00
|
|
|
return { ...response, isError: true };
|
|
|
|
}
|
|
|
|
|
2020-02-14 02:23:21 +00:00
|
|
|
const primaryTotals =
|
|
|
|
( primary && primary.data && primary.data.totals ) || null;
|
2018-10-11 18:45:01 +00:00
|
|
|
|
2020-02-14 02:23:21 +00:00
|
|
|
const secondaryQuery = getRequestQuery( {
|
|
|
|
...options,
|
|
|
|
dataType: 'secondary',
|
|
|
|
} );
|
2020-02-25 01:18:52 +00:00
|
|
|
|
|
|
|
// Disable eslint rule requiring `getReportStats` to be defined below because the next two statements
|
|
|
|
// depend on `getReportStats` to have been called.
|
|
|
|
// eslint-disable-next-line @wordpress/no-unused-vars-before-return
|
|
|
|
const secondary = getReportStats( endpoint, secondaryQuery );
|
|
|
|
|
2019-03-21 03:25:05 +00:00
|
|
|
if ( isReportStatsRequesting( endpoint, secondaryQuery ) ) {
|
2018-10-11 18:45:01 +00:00
|
|
|
return { ...response, isRequesting: true };
|
2019-03-21 03:25:05 +00:00
|
|
|
} else if ( getReportStatsError( endpoint, secondaryQuery ) ) {
|
2018-10-11 18:45:01 +00:00
|
|
|
return { ...response, isError: true };
|
|
|
|
}
|
|
|
|
|
2020-02-14 02:23:21 +00:00
|
|
|
const secondaryTotals =
|
|
|
|
( secondary && secondary.data && secondary.data.totals ) || null;
|
2018-10-11 18:45:01 +00:00
|
|
|
|
2020-02-14 02:23:21 +00:00
|
|
|
return {
|
|
|
|
...response,
|
|
|
|
totals: { primary: primaryTotals, secondary: secondaryTotals },
|
|
|
|
};
|
2018-10-11 18:45:01 +00:00
|
|
|
}
|
|
|
|
|
2018-09-03 15:25:38 +00:00
|
|
|
/**
|
|
|
|
* Returns all of the data needed to render a chart with summary numbers on a report page.
|
2020-02-14 02:23:21 +00:00
|
|
|
*
|
|
|
|
* @param {Object} options arguments
|
|
|
|
* @param {string} options.endpoint Report API Endpoint
|
|
|
|
* @param {string} options.dataType 'primary' or 'secondary'
|
2019-03-21 03:25:05 +00:00
|
|
|
* @param {Object} options.query Query parameters in the url
|
|
|
|
* @param {Object} options.select Instance of @wordpress/select
|
|
|
|
* @param {Array} options.limitBy Properties used to limit the results. It will be used in the API call to send the IDs.
|
2020-03-25 03:20:17 +00:00
|
|
|
* @param {string} options.defaultDateRange User specified default date range.
|
2018-09-03 15:25:38 +00:00
|
|
|
* @return {Object} Object containing API request information (response, fetching, and error details)
|
|
|
|
*/
|
2019-03-21 03:25:05 +00:00
|
|
|
export function getReportChartData( options ) {
|
|
|
|
const { endpoint, select } = options;
|
2020-02-14 02:23:21 +00:00
|
|
|
const {
|
|
|
|
getReportStats,
|
|
|
|
getReportStatsError,
|
|
|
|
isReportStatsRequesting,
|
|
|
|
} = select( 'wc-api' );
|
2018-09-03 15:25:38 +00:00
|
|
|
|
|
|
|
const response = {
|
Add loading indicators, error state, and EmptyContent to the revenue report. (#347, woocommerce/woocommerce-admin#348)
* Add loading indiciators for the revenue report.
* Improve accessibility, and fix up some documentation comments.
* Fix top border on mobile
* Add EmptyContent Component and revenue error/empty states. (https://github.com/woocommerce/woocommerce-admin/pull/348)
* Add EmptyContent Component and revenue error/empty states.
* Move relative image handling to ImageAsset, combine secondary and primary action rendering, add some missing isRequired proptypes, add empty error handling.
* Handle PR Feedback: Clean up button css, set a default for illustration, fix deprecation typo, some code cleanup.
2018-09-05 16:45:49 +00:00
|
|
|
isEmpty: false,
|
2018-09-03 15:25:38 +00:00
|
|
|
isError: false,
|
|
|
|
isRequesting: false,
|
|
|
|
data: {
|
2019-02-05 12:00:37 +00:00
|
|
|
totals: {},
|
2018-09-03 15:25:38 +00:00
|
|
|
intervals: [],
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2019-03-21 03:25:05 +00:00
|
|
|
const requestQuery = getRequestQuery( options );
|
2020-02-14 02:23:21 +00:00
|
|
|
// Disable eslint rule requiring `stats` to be defined below because the next two if statements
|
|
|
|
// depend on `getReportStats` to have been called.
|
|
|
|
// eslint-disable-next-line @wordpress/no-unused-vars-before-return
|
2019-03-21 03:25:05 +00:00
|
|
|
const stats = getReportStats( endpoint, requestQuery );
|
2018-09-03 15:25:38 +00:00
|
|
|
|
2019-03-21 03:25:05 +00:00
|
|
|
if ( isReportStatsRequesting( endpoint, requestQuery ) ) {
|
2018-09-03 15:25:38 +00:00
|
|
|
return { ...response, isRequesting: true };
|
2020-02-14 02:23:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if ( getReportStatsError( endpoint, requestQuery ) ) {
|
2018-09-03 15:25:38 +00:00
|
|
|
return { ...response, isError: true };
|
2020-02-14 02:23:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if ( isReportDataEmpty( stats, endpoint ) ) {
|
Add loading indicators, error state, and EmptyContent to the revenue report. (#347, woocommerce/woocommerce-admin#348)
* Add loading indiciators for the revenue report.
* Improve accessibility, and fix up some documentation comments.
* Fix top border on mobile
* Add EmptyContent Component and revenue error/empty states. (https://github.com/woocommerce/woocommerce-admin/pull/348)
* Add EmptyContent Component and revenue error/empty states.
* Move relative image handling to ImageAsset, combine secondary and primary action rendering, add some missing isRequired proptypes, add empty error handling.
* Handle PR Feedback: Clean up button css, set a default for illustration, fix deprecation typo, some code cleanup.
2018-09-05 16:45:49 +00:00
|
|
|
return { ...response, isEmpty: true };
|
2018-09-03 15:25:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const totals = ( stats && stats.data && stats.data.totals ) || null;
|
|
|
|
let intervals = ( stats && stats.data && stats.data.intervals ) || [];
|
|
|
|
|
|
|
|
// If we have more than 100 results for this time period,
|
|
|
|
// we need to make additional requests to complete the response.
|
|
|
|
if ( stats.totalResults > MAX_PER_PAGE ) {
|
|
|
|
let isFetching = true;
|
|
|
|
let isError = false;
|
|
|
|
const pagedData = [];
|
|
|
|
const totalPages = Math.ceil( stats.totalResults / MAX_PER_PAGE );
|
2019-08-13 22:27:15 +00:00
|
|
|
let pagesFetched = 1;
|
2018-09-03 15:25:38 +00:00
|
|
|
|
|
|
|
for ( let i = 2; i <= totalPages; i++ ) {
|
2018-10-17 23:02:31 +00:00
|
|
|
const nextQuery = { ...requestQuery, page: i };
|
2019-03-21 03:25:05 +00:00
|
|
|
const _data = getReportStats( endpoint, nextQuery );
|
|
|
|
if ( isReportStatsRequesting( endpoint, nextQuery ) ) {
|
2018-09-03 15:25:38 +00:00
|
|
|
continue;
|
|
|
|
}
|
2019-03-21 03:25:05 +00:00
|
|
|
if ( getReportStatsError( endpoint, nextQuery ) ) {
|
2018-09-03 15:25:38 +00:00
|
|
|
isError = true;
|
|
|
|
isFetching = false;
|
|
|
|
break;
|
|
|
|
}
|
2018-11-22 20:49:52 +00:00
|
|
|
|
|
|
|
pagedData.push( _data );
|
2019-08-13 22:27:15 +00:00
|
|
|
pagesFetched++;
|
|
|
|
|
|
|
|
if ( pagesFetched === totalPages ) {
|
2018-11-22 20:49:52 +00:00
|
|
|
isFetching = false;
|
|
|
|
break;
|
2018-09-03 15:25:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( isFetching ) {
|
|
|
|
return { ...response, isRequesting: true };
|
|
|
|
} else if ( isError ) {
|
|
|
|
return { ...response, isError: true };
|
|
|
|
}
|
|
|
|
|
|
|
|
forEach( pagedData, function( _data ) {
|
|
|
|
intervals = intervals.concat( _data.data.intervals );
|
|
|
|
} );
|
|
|
|
}
|
|
|
|
|
|
|
|
return { ...response, data: { totals, intervals } };
|
|
|
|
}
|
2018-11-12 14:34:51 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a formatting function or string to be used by d3-format
|
|
|
|
*
|
2020-02-14 02:23:21 +00:00
|
|
|
* @param {string} type Type of number, 'currency', 'number', 'percent', 'average'
|
|
|
|
* @return {string|Function} returns a number format based on the type or an overriding formatting function
|
2018-11-12 14:34:51 +00:00
|
|
|
*/
|
|
|
|
export function getTooltipValueFormat( type ) {
|
|
|
|
switch ( type ) {
|
|
|
|
case 'currency':
|
|
|
|
return formatCurrency;
|
|
|
|
case 'percent':
|
|
|
|
return '.0%';
|
|
|
|
case 'number':
|
|
|
|
return ',';
|
|
|
|
case 'average':
|
|
|
|
return ',.2r';
|
|
|
|
default:
|
|
|
|
return ',';
|
|
|
|
}
|
|
|
|
}
|
2018-11-22 23:12:12 +00:00
|
|
|
|
2019-03-21 03:25:05 +00:00
|
|
|
/**
|
|
|
|
* Returns query needed for a request to populate a table.
|
|
|
|
*
|
2020-02-14 02:23:21 +00:00
|
|
|
* @param {Object} options arguments
|
2019-03-21 03:25:05 +00:00
|
|
|
* @param {Object} options.query Query parameters in the url
|
|
|
|
* @param {Object} options.tableQuery Query parameters specific for that endpoint
|
2020-03-25 03:20:17 +00:00
|
|
|
* @param {string} options.defaultDateRange User specified default date range.
|
2019-03-21 03:25:05 +00:00
|
|
|
* @return {Object} Object Table data response
|
|
|
|
*/
|
|
|
|
export function getReportTableQuery( options ) {
|
|
|
|
const { query, tableQuery = {} } = options;
|
|
|
|
const filterQuery = getFilterQuery( options );
|
2020-03-25 03:20:17 +00:00
|
|
|
const datesFromQuery = getCurrentDates( query, options.defaultDateRange );
|
2018-11-22 23:12:12 +00:00
|
|
|
|
2019-06-13 18:21:35 +00:00
|
|
|
const noIntervals = includes( noIntervalEndpoints, options.endpoint );
|
|
|
|
|
2018-11-22 23:12:12 +00:00
|
|
|
return {
|
2019-03-21 03:25:05 +00:00
|
|
|
orderby: query.orderby || 'date',
|
|
|
|
order: query.order || 'desc',
|
2020-02-14 02:23:21 +00:00
|
|
|
after: noIntervals
|
|
|
|
? undefined
|
|
|
|
: appendTimestamp( datesFromQuery.primary.after, 'start' ),
|
|
|
|
before: noIntervals
|
|
|
|
? undefined
|
|
|
|
: appendTimestamp( datesFromQuery.primary.before, 'end' ),
|
2019-07-11 17:22:26 +00:00
|
|
|
page: query.paged || 1,
|
2019-03-21 03:25:05 +00:00
|
|
|
per_page: query.per_page || QUERY_DEFAULTS.pageSize,
|
2018-11-22 23:12:12 +00:00
|
|
|
...filterQuery,
|
2019-03-21 03:25:05 +00:00
|
|
|
...tableQuery,
|
2018-11-22 23:12:12 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns table data needed to render a report page.
|
|
|
|
*
|
2019-03-21 03:25:05 +00:00
|
|
|
* @param {Object} options arguments
|
2020-02-14 02:23:21 +00:00
|
|
|
* @param {string} options.endpoint Report API Endpoint
|
2019-03-21 03:25:05 +00:00
|
|
|
* @param {Object} options.query Query parameters in the url
|
|
|
|
* @param {Object} options.select Instance of @wordpress/select
|
|
|
|
* @param {Object} options.tableQuery Query parameters specific for that endpoint
|
2020-03-25 03:20:17 +00:00
|
|
|
* @param {string} options.defaultDateRange User specified default date range.
|
2018-11-22 23:12:12 +00:00
|
|
|
* @return {Object} Object Table data response
|
|
|
|
*/
|
2019-03-21 03:25:05 +00:00
|
|
|
export function getReportTableData( options ) {
|
|
|
|
const { endpoint, select } = options;
|
2020-02-14 02:23:21 +00:00
|
|
|
const {
|
|
|
|
getReportItems,
|
|
|
|
getReportItemsError,
|
|
|
|
isReportItemsRequesting,
|
|
|
|
} = select( 'wc-api' );
|
2018-11-22 23:12:12 +00:00
|
|
|
|
2019-03-21 03:25:05 +00:00
|
|
|
const tableQuery = reportsUtils.getReportTableQuery( options );
|
2018-11-22 23:12:12 +00:00
|
|
|
const response = {
|
|
|
|
query: tableQuery,
|
|
|
|
isRequesting: false,
|
|
|
|
isError: false,
|
2018-12-10 15:51:11 +00:00
|
|
|
items: {
|
|
|
|
data: [],
|
2019-02-05 12:00:37 +00:00
|
|
|
totalResults: 0,
|
2018-12-10 15:51:11 +00:00
|
|
|
},
|
2018-11-22 23:12:12 +00:00
|
|
|
};
|
|
|
|
|
2020-02-17 01:02:32 +00:00
|
|
|
// Disable eslint rule requiring `items` to be defined below because the next two if statements
|
|
|
|
// depend on `getReportItems` to have been called.
|
|
|
|
// eslint-disable-next-line @wordpress/no-unused-vars-before-return
|
|
|
|
const items = getReportItems( endpoint, tableQuery );
|
|
|
|
|
2018-12-05 23:12:34 +00:00
|
|
|
if ( isReportItemsRequesting( endpoint, tableQuery ) ) {
|
2018-11-22 23:12:12 +00:00
|
|
|
return { ...response, isRequesting: true };
|
2018-12-15 12:38:54 +00:00
|
|
|
} else if ( getReportItemsError( endpoint, tableQuery ) ) {
|
2018-11-22 23:12:12 +00:00
|
|
|
return { ...response, isError: true };
|
|
|
|
}
|
|
|
|
|
|
|
|
return { ...response, items };
|
|
|
|
}
|