laipower/wp-content/plugins/easy-digital-downloads/includes/reports/reports-functions.php

1755 lines
48 KiB
PHP

<?php
/**
* Reports API - Functions
*
* @package EDD
* @subpackage Reports
* @copyright Copyright (c) 2018, Easy Digital Downloads, LLC
* @license http://opensource.org/licenses/gpl-2.0.php GNU Public License
* @since 3.0
*/
namespace EDD\Reports;
//
// Endpoint and report helpers.
//
/**
* Registers a new endpoint to the master registry.
*
* @since 3.0
*
* @see \EDD\Reports\Data\Endpoint_Registry::register_endpoint()
*
* @param string $endpoint_id Reports data endpoint ID.
* @param array $attributes {
* Endpoint attributes. All arguments are required unless otherwise noted.
*
* @type string $label Endpoint label.
* @type int $priority Optional. Priority by which to retrieve the endpoint. Default 10.
* @type array $views {
* Array of view handlers by type.
*
* @type array $view_type {
* View type slug, with array beneath it.
*
* @type callable $data_callback Callback used to retrieve data for the view.
* @type callable $display_callback Callback used to render the view.
* @type array $display_args Optional. Array of arguments to pass to the
* display_callback (if any). Default empty array.
* }
* }
* }
* @return bool True if the endpoint was successfully registered, otherwise false.
*/
function register_endpoint( $endpoint_id, $attributes ) {
/** @var Data\Endpoint_Registry|\WP_Error $registry */
$registry = EDD()->utils->get_registry( 'reports:endpoints' );
if ( empty( $registry ) || is_wp_error( $registry ) ) {
return false;
}
try {
$added = $registry->register_endpoint( $endpoint_id, $attributes );
} catch ( \EDD_Exception $exception ) {
edd_debug_log_exception( $exception );
$added = false;
}
return $added;
}
/**
* Retrieves and builds an endpoint object.
*
* @since 3.0
*
* @see \EDD\Reports\Data\Endpoint_Registry::build_endpoint()
*
* @param string $endpoint_id Endpoint ID.
* @param string $view_type View type to use when building the object.
* @return Data\Endpoint|\WP_Error Endpoint object on success, otherwise a WP_Error object.
*/
function get_endpoint( $endpoint_id, $view_type ) {
/** @var Data\Endpoint_Registry|\WP_Error $registry */
$registry = EDD()->utils->get_registry( 'reports:endpoints' );
if ( empty( $registry ) || is_wp_error( $registry ) ) {
return $registry;
}
return $registry->build_endpoint( $endpoint_id, $view_type );
}
/**
* Registers a new report.
*
* @since 3.0
*
* @see \EDD\Reports\Data\Report_Registry::add_report()
*
* @param string $report_id Report ID.
* @param array $attributes {
* Reports attributes. All arguments are required unless otherwise noted.
*
* @type string $label Report label.
* @type int $priority Optional. Priority by which to register the report. Default 10.
* @type array $filters Filters available to the report.
* @type array $endpoints Endpoints to associate with the report.
* }
* @return bool True if the report was successfully registered, otherwise false.
*/
function add_report( $report_id, $attributes ) {
/** @var Data\Report_Registry|\WP_Error $registry */
$registry = EDD()->utils->get_registry( 'reports' );
if ( empty( $registry ) || is_wp_error( $registry ) ) {
return false;
}
try {
$added = $registry->add_report( $report_id, $attributes );
} catch ( \EDD_Exception $exception ) {
edd_debug_log_exception( $exception );
$added = false;
}
return $added;
}
/**
* Retrieves and builds a report object.
*
* @since 3.0
*
* @see \EDD\Reports\Data\Report_Registry::build_report()
*
* @param string $report_id Report ID.
* @param bool $build_endpoints Optional. Whether to build the endpoints (includes registering
* any endpoint dependencies, such as registering meta boxes).
* Default true.
* @return Data\Report|\WP_Error Report object on success, otherwise a WP_Error object.
*/
function get_report( $report_id = false, $build_endpoints = true ) {
/** @var Data\Report_Registry|\WP_Error $registry */
$registry = EDD()->utils->get_registry( 'reports' );
if ( empty( $registry ) || is_wp_error( $registry ) ) {
return $registry;
}
return $registry->build_report( $report_id, $build_endpoints );
}
/** Sections ******************************************************************/
/**
* Retrieves the list of slug/label report pairs.
*
* @since 3.0
*
* @return array List of reports, otherwise an empty array.
*/
function get_reports() {
/** @var Data\Report_Registry|\WP_Error $registry */
$registry = EDD()->utils->get_registry( 'reports' );
if ( empty( $registry ) || is_wp_error( $registry ) ) {
return array();
} else {
$reports = $registry->get_reports( 'priority', 'core' );
}
// Re-sort by priority.
uasort( $reports, array( $registry, 'priority_sort' ) );
/**
* Filters the list of report slug/label pairs.
*
* @since 3.0
*
* @param array $reports List of slug/label pairs as representative of reports.
*/
return apply_filters( 'edd_get_reports', $reports );
}
/**
* Retrieves the slug for the active report.
*
* @since 3.0
*
* @return string The active report, or the 'overview' report if no view defined
*/
function get_current_report() {
return isset( $_REQUEST['view'] )
? sanitize_key( $_REQUEST['view'] )
: 'overview'; // Hardcoded default
}
/** Endpoints *****************************************************************/
/**
* Retrieves the list of supported endpoint view types and their attributes.
*
* @since 3.0
*
* @return array List of supported endpoint types.
*/
function get_endpoint_views() {
if ( ! did_action( 'edd_reports_init' ) ) {
_doing_it_wrong( __FUNCTION__, 'Endpoint views cannot be retrieved prior to the firing of the edd_reports_init hook.', 'EDD 3.0' );
return array();
}
/** @var Data\Endpoint_View_Registry|\WP_Error $registry */
$registry = EDD()->utils->get_registry( 'reports:endpoints:views' );
if ( empty( $registry ) || is_wp_error( $registry ) ) {
return array();
} else {
$views = $registry->get_endpoint_views();
}
return $views;
}
/**
* Retrieves the name of the handler class for a given endpoint view.
*
* @since 3.0
*
* @param string $view Endpoint view.
* @return string Handler class name if set and the view exists, otherwise an empty string.
*/
function get_endpoint_handler( $view ) {
$views = get_endpoint_views();
return isset( $views[ $view ]['handler'] )
? $views[ $view ]['handler']
: '';
}
/**
* Retrieves the group display callback for a given endpoint view.
*
* @since 3.0
*
* @param string $view Endpoint view.
* @return string Group callback if set, otherwise an empty string.
*/
function get_endpoint_group_callback( $view ) {
$views = get_endpoint_views();
return isset( $views[ $view ]['group_callback'] )
? $views[ $view ]['group_callback']
: '';
}
/**
* Determines whether an endpoint view is valid.
*
* @since 3.0
*
* @param string $view Endpoint view slug.
* @return bool True if the view is valid, otherwise false.
*/
function validate_endpoint_view( $view ) {
return array_key_exists( $view, get_endpoint_views() );
}
/**
* Parses views for an incoming endpoint.
*
* @since 3.0
*
* @see get_endpoint_views()
*
* @param array $views View slugs and attributes as dictated by get_endpoint_views().
*
* @return array (Maybe) adjusted views slugs and attributes array.
*/
function parse_endpoint_views( $views ) {
$valid_views = get_endpoint_views();
foreach ( $views as $view => $attributes ) {
if ( ! empty( $valid_views[ $view ]['fields'] ) ) {
$fields = $valid_views[ $view ]['fields'];
// Merge the incoming args with the field defaults.
$view_args = wp_parse_args( $attributes, $fields );
// Overwrite the view attributes, keeping only the valid fields.
$views[ $view ] = array_intersect_key( $view_args, $fields );
if ( $views[ $view ]['display_callback'] === $fields['display_callback'] ) {
$views[ $view ]['display_args'] = wp_parse_args( $views[ $view ]['display_args'], $fields['display_args'] );
}
}
}
return $views;
}
/** Filters *******************************************************************/
/**
* Retrieves the list of registered reports filters and their attributes.
*
* @since 3.0
*
* @return array List of supported endpoint filters.
*/
function get_filters() {
$filters = array(
'dates' => array(
'label' => __( 'Date', 'easy-digital-downloads' ),
'display_callback' => __NAMESPACE__ . '\\display_dates_filter'
),
'products' => array(
'label' => __( 'Products', 'easy-digital-downloads' ),
'display_callback' => __NAMESPACE__ . '\\display_products_filter'
),
'product_categories' => array(
'label' => __( 'Product Categories', 'easy-digital-downloads' ),
'display_callback' => __NAMESPACE__ . '\\display_product_categories_filter'
),
'taxes' => array(
'label' => __( 'Exclude Taxes', 'easy-digital-downloads' ),
'display_callback' => __NAMESPACE__ . '\\display_taxes_filter'
),
'gateways' => array(
'label' => __( 'Gateways', 'easy-digital-downloads' ),
'display_callback' => __NAMESPACE__ . '\\display_gateways_filter'
),
'discounts' => array(
'label' => __( 'Discounts', 'easy-digital-downloads' ),
'display_callback' => __NAMESPACE__ . '\\display_discounts_filter'
),
'regions' => array(
'label' => __( 'Regions', 'easy-digital-downloads' ),
'display_callback' => __NAMESPACE__ . '\\display_region_filter'
),
'countries' => array(
'label' => __( 'Countries', 'easy-digital-downloads' ),
'display_callback' => __NAMESPACE__ . '\\display_country_filter'
),
'currencies' => array(
'label' => __( 'Currencies', 'easy-digital-downloads' ),
'display_callback' => __NAMESPACE__ . '\\display_currency_filter'
)
);
/**
* Filters the list of available report filters.
*
* @since 3.0
*
* @param array[] $filters
*/
return apply_filters( 'edd_report_filters', $filters );
}
/**
* Determines whether the given filter is valid.
*
* @since 3.0
*
* @param string $filter Filter key.
* @return bool True if the filter is valid, otherwise false.
*/
function validate_filter( $filter ) {
return array_key_exists( $filter, get_filters() );
}
/**
* Retrieves the value of an endpoint filter for the current session and report.
*
* @since 3.0
*
* @param string $filter Filter key to retrieve the value for.
* @return mixed|string Value of the filter if it exists, otherwise an empty string.
*/
function get_filter_value( $filter ) {
$value = '';
// Bail if filter does not validate
if ( ! validate_filter( $filter ) ) {
return $value;
}
switch ( $filter ) {
// Handle dates.
case 'dates':
$default_range = 'this_month';
$default_relative_range = 'previous_period';
if ( ! isset( $_GET['range'] ) ) {
$dates = parse_dates_for_range( $default_range );
$value = array(
'range' => $default_range,
'relative_range' => $default_relative_range,
'from' => $dates['start']->format( 'Y-m-d' ),
'to' => $dates['end']->format( 'Y-m-d' ),
);
} else {
$value = array(
'range' => isset( $_GET['range'] )
? sanitize_text_field( $_GET['range'] )
: $default_range,
'relative_range' => isset( $_GET['relative_range'] )
? sanitize_text_field( $_GET['relative_range'] )
: $default_relative_range,
'from' => isset( $_GET['filter_from'] )
? sanitize_text_field( $_GET['filter_from'] )
: '',
'to' => isset( $_GET['filter_to'] )
? sanitize_text_field( $_GET['filter_to'] )
: ''
);
}
break;
// Handle taxes.
case 'taxes':
$value = array();
if ( isset( $_GET['exclude_taxes'] ) ) {
$value['exclude_taxes'] = true;
}
break;
// Handle default (direct from URL).
default:
$value = isset( $_GET[ $filter ] )
? sanitize_text_field( $_GET[ $filter ] )
: '';
/**
* Filters the value of a report filter.
*
* @since 3.0
*
* @param string $value Report filter value.
* @param string $filter Report filter.
*/
$value = apply_filters( 'edd_reports_get_filter_value', $value, $filter );
}
return $value;
}
/**
* Returns a list of registered report filters that should be persisted across views.
*
* @since 3.0
*
* @return array
*/
function get_persisted_filters() {
$filters = array(
'range',
'relative_range',
'filter_from',
'filter_to',
'exclude_taxes',
);
/**
* Filters registered report filters that should be persisted across views.
*
* @since 3.0
*
* @param array $filters List of registered filters to persist.
*/
$filters = apply_filters( 'edd_reports_get_persisted_filters', $filters );
return $filters;
}
/**
* Retrieves key/label pairs of date filter options for use in a drop-down.
*
* @since 3.0
*
* @return array Key/label pairs of date filter options.
*/
function get_dates_filter_options() {
static $options = null;
if ( is_null( $options ) ) {
$options = array(
'other' => __( 'Custom', 'easy-digital-downloads' ),
'today' => __( 'Today', 'easy-digital-downloads' ),
'yesterday' => __( 'Yesterday', 'easy-digital-downloads' ),
'this_week' => __( 'This Week', 'easy-digital-downloads' ),
'last_week' => __( 'Last Week', 'easy-digital-downloads' ),
'last_30_days' => __( 'Last 30 Days', 'easy-digital-downloads' ),
'this_month' => __( 'Month to Date', 'easy-digital-downloads' ),
'last_month' => __( 'Last Month', 'easy-digital-downloads' ),
'this_quarter' => __( 'Quarter to Date', 'easy-digital-downloads' ),
'last_quarter' => __( 'Last Quarter', 'easy-digital-downloads' ),
'this_year' => __( 'Year to Date', 'easy-digital-downloads' ),
'last_year' => __( 'Last Year', 'easy-digital-downloads' ),
);
}
/**
* Filters the list of key/label pairs of date filter options.
*
* @since 1.3
*
* @param array $date_options Date filter options.
*/
return apply_filters( 'edd_report_date_options', $options );
}
/**
* Retrieves the default relative range key for a specific range.
*
* @since 3.1
*
* @return string Relative date range key.
*/
function get_default_relative_range( $range ) {
switch ( $range ) {
case 'this_month':
case 'last_month':
$relative_range = 'previous_month';
break;
case 'this_quarter':
case 'last_quarter':
$relative_range = 'previous_quarter';
break;
case 'this_year':
case 'last_year':
$relative_range = 'previous_year';
break;
default:
$relative_range = 'previous_period';
break;
}
return $relative_range;
}
/**
* Retrieves key/label pairs of relative date filter options for use in a drop-down.
*
* @since 3.1
*
* @return array Key/label pairs of relative date filter options.
*/
function get_relative_dates_filter_options() {
static $options = null;
if ( is_null( $options ) ) {
$options = array(
'previous_period' => __( 'Previous period', 'easy-digital-downloads' ),
'previous_month' => __( 'Previous month', 'easy-digital-downloads' ),
'previous_quarter' => __( 'Previous quarter', 'easy-digital-downloads' ),
'previous_year' => __( 'Previous year', 'easy-digital-downloads' ),
);
}
return $options;
}
/**
* Retrieves the start and end date filters for use with the Reports API.
*
* @since 3.0
*
* @param string $values Optional. What format to retrieve dates in the resulting array in.
* Accepts 'strings' or 'objects'. Default 'strings'.
* @param string $timezone Optional. Timezone to force for filter dates. Primarily used for
* legacy testing purposes. Default empty.
* @return array|\EDD\Utils\Date[] {
* Query date range for the current graph filter request.
*
* @type string|\EDD\Utils\Date $start Start day and time (based on the beginning of the given day).
* If `$values` is 'objects', a Carbon object, otherwise a date
* time string.
* @type string|\EDD\Utils\Date $end End day and time (based on the end of the given day). If `$values`
* is 'objects', a Carbon object, otherwise a date time string.
* }
*/
function get_dates_filter( $values = 'strings', $timezone = null ) {
$dates = parse_dates_for_range();
if ( 'strings' === $values ) {
if ( ! empty( $dates['start'] ) ) {
$dates['start'] = $dates['start']->toDateTimeString();
}
if ( ! empty( $dates['end'] ) ) {
$dates['end'] = $dates['end']->toDateTimeString();
}
}
/**
* Filters the start and end date filters for use with the Graphs API.
*
* @since 3.0
*
* @param array|\EDD\Utils\Date[] $dates {
* Query date range for the current graph filter request.
*
* @type string|\EDD\Utils\Date $start Start day and time (based on the beginning of the given day).
* If `$values` is 'objects', a Date object, otherwise a date
* time string.
* @type string|\EDD\Utils\Date $end End day and time (based on the end of the given day). If `$values`
* is 'objects', a Date object, otherwise a date time string.
* }
*/
return apply_filters( 'edd_get_dates_filter', $dates );
}
/**
* Parses start and end dates for the given range.
*
* @since 3.0
*
* @param string $range Optional. Range value to generate start and end dates for against `$date`.
* Default is the current range as derived from the session.
* @param string $date Date string converted to `\EDD\Utils\Date` to anchor calculations to.
* @param bool $convert_to_utc Optional. If we should convert the results to UTC for Database Queries
* @return \EDD\Utils\Date[] Array of start and end date objects.
*/
function parse_dates_for_range( $range = null, $date = 'now', $convert_to_utc = true ) {
// Set the time ranges in the user's timezone, so they ultimately see them in their own timezone.
$date = EDD()->utils->date( $date, edd_get_timezone_id(), false );
if ( null === $range || ! array_key_exists( $range, get_dates_filter_options() ) ) {
$range = get_dates_filter_range();
}
switch ( $range ) {
case 'this_month':
$dates = array(
'start' => $date->copy()->startOfMonth(),
'end' => $date->copy()->endOfDay(),
);
break;
case 'last_month':
$dates = array(
'start' => $date->copy()->subMonthNoOverflow( 1 )->startOfMonth(),
'end' => $date->copy()->subMonthNoOverflow( 1 )->endOfMonth(),
);
break;
case 'today':
$dates = array(
'start' => $date->copy()->startOfDay(),
'end' => $date->copy()->endOfDay(),
);
break;
case 'yesterday':
$dates = array(
'start' => $date->copy()->subDay( 1 )->startOfDay(),
'end' => $date->copy()->subDay( 1 )->endOfDay(),
);
break;
case 'this_week':
$dates = array(
'start' => $date->copy()->startOfWeek(),
'end' => $date->copy()->endOfDay(),
);
break;
case 'last_week':
$dates = array(
'start' => $date->copy()->subWeek( 1 )->startOfWeek(),
'end' => $date->copy()->subWeek( 1 )->endOfWeek(),
);
break;
case 'last_30_days':
$dates = array(
'start' => $date->copy()->subDay( 30 )->startOfDay(),
'end' => $date->copy()->endOfDay(),
);
break;
case 'this_quarter':
$dates = array(
'start' => $date->copy()->startOfQuarter(),
'end' => $date->copy()->endOfDay(),
);
break;
case 'last_quarter':
$dates = array(
'start' => $date->copy()->subQuarter( 1 )->startOfQuarter(),
'end' => $date->copy()->subQuarter( 1 )->endOfQuarter(),
);
break;
case 'this_year':
$dates = array(
'start' => $date->copy()->startOfYear(),
'end' => $date->copy()->endOfDay(),
);
break;
case 'last_year':
$dates = array(
'start' => $date->copy()->subYear( 1 )->startOfYear(),
'end' => $date->copy()->subYear( 1 )->endOfYear(),
);
break;
case 'other':
default:
$dates_from_report = get_filter_value( 'dates' );
if ( ! empty( $dates_from_report ) ) {
$start = $dates_from_report['from'];
$end = $dates_from_report['to'];
} else {
$start = $end = 'now';
}
$dates = array(
'start' => EDD()->utils->date( $start, edd_get_timezone_id(), false )->startOfDay(),
'end' => EDD()->utils->date( $end, edd_get_timezone_id(), false )->endOfDay(),
);
break;
}
if ( $convert_to_utc ) {
// Convert the values to the UTC equivalent so that we can query the database using UTC.
$dates['start'] = edd_get_utc_equivalent_date( $dates['start'] );
$dates['end'] = edd_get_utc_equivalent_date( $dates['end'] );
}
$dates['range'] = $range;
return $dates;
}
/**
* Parses relative start and end dates for the given range.
*
* @since 3.1
*
* @param string $range Optional. Range value to generate start and end dates for against `$date`.
* @param string $relative_range Optional. Range value to generate relative start and end dates for against `$date`.
* Default is the current range as derived from the session.
* @param string $date Date string converted to `\EDD\Utils\Date` to anchor calculations to.
* @param bool $convert_to_utc Optional. If we should convert the results to UTC for Database Queries
* @return \EDD\Utils\Date[] Array of start and end date objects.
*/
function parse_relative_dates_for_range( $range = null, $relative_range = null, $date = 'now', $convert_to_utc = true ) {
// Set the time ranges in the user's timezone, so they ultimately see them in their own timezone.
$date = EDD()->utils->date( $date, edd_get_timezone_id(), false );
if ( null === $range || ! array_key_exists( $range, get_dates_filter_options() ) ) {
$range = get_dates_filter_range();
}
if ( null === $relative_range || ! array_key_exists( $relative_range, get_relative_dates_filter_options() ) ) {
$relative_range = get_relative_dates_filter_range();
}
$dates = parse_dates_for_range( $range, $date, false );
switch ( $relative_range ) {
case 'previous_period':
$days_diff = $dates['start']->copy()->diffInDays( $dates['end'], true ) + 1;
$dates = array(
'start' => $dates['start']->copy()->subDays( $days_diff ),
'end' => $dates['end']->copy()->subDays( $days_diff ),
);
break;
case 'previous_month':
$dates = array(
'start' => $dates['start']->copy()->subMonth( 1 ),
'end' => $dates['end']->copy()->subMonth( 1 ),
);
break;
case 'previous_quarter':
$dates = array(
'start' => $dates['start']->copy()->subQuarter( 1 ),
'end' => $dates['end']->copy()->subQuarter( 1 ),
);
break;
case 'previous_year':
$dates = array(
'start' => $dates['start']->copy()->subYear( 1 ),
'end' => $dates['end']->copy()->subYear( 1 ),
);
break;
}
if ( $convert_to_utc ) {
// Convert the values to the UTC equivalent so that we can query the database using UTC.
$dates['start'] = edd_get_utc_equivalent_date( $dates['start'] );
$dates['end'] = edd_get_utc_equivalent_date( $dates['end'] );
}
$dates['range'] = $range;
return $dates;
}
/**
* Retrieves the date filter range.
*
* @since 3.0
*
* @return string Date filter range.
*/
function get_dates_filter_range() {
$dates = get_filter_value( 'dates' );
if ( isset( $dates['range'] ) ) {
$range = sanitize_key( $dates['range'] );
} else {
/**
* Filters the report dates default range.
*
* @since 1.3
*
* @param string $range Date range as derived from the session. Default 'last_30_days'
* @param array $dates Dates filter data array.
*/
$range = apply_filters( 'edd_get_report_dates_default_range', 'this_month', $dates );
}
/**
* Filters the dates filter range.
*
* @since 3.0
*
* @param string $range Dates filter range.
* @param array $dates Dates filter data array.
*/
return apply_filters( 'edd_get_dates_filter_range', $range, $dates );
}
/**
* Retrieves the date filter for relative range.
*
* @since 3.1
*
* @return string Date filter range.
*/
function get_relative_dates_filter_range() {
$dates = get_filter_value( 'dates' );
if ( isset( $dates['relative_range'] ) ) {
$relative_range = sanitize_key( $dates['relative_range'] );
} else {
/**
* Filters the report dates default range.
*
* @since 3.1
*
* @param string $range Relative daate range as derived from the session. Default 'previous_period'
* @param array $dates Dates filter data array.
*/
$relative_range = apply_filters( 'edd_get_report_dates_default_relative_range', 'previous_period', $dates );
}
/**
* Filters the dates filter range.
*
* @since 3.1
*
* @param string $range Dates filter relative range.
* @param array $dates Dates filter data array.
*/
return apply_filters( 'edd_get_dates_filter_relative_range', $relative_range, $dates );
}
/**
* Determines whether results should be displayed hour by hour, or not.
*
* @since 3.0
*
* @return bool True if results should use hour by hour, otherwise false.
*/
function get_dates_filter_hour_by_hour() {
$hour_by_hour = false;
// Retrieve the queried dates.
$dates = get_dates_filter( 'objects' );
// Determine graph options.
switch ( $dates['range'] ) {
case 'today':
case 'yesterday':
$hour_by_hour = true;
break;
case 'this_week':
case 'this_month':
case 'this_quarter':
case 'this_year':
case 'other':
$difference = ( $dates['end']->getTimestamp() - $dates['start']->getTimestamp() );
if ( $difference <= ( DAY_IN_SECONDS * 2 ) ) {
$hour_by_hour = true;
}
break;
default:
$hour_by_hour = false;
break;
}
return $hour_by_hour;
}
/**
* Determines whether results should be displayed day by day or not.
*
* @since 3.0
*
* @return bool True if results should use day by day, otherwise false.
*/
function get_dates_filter_day_by_day() {
// Retrieve the queried dates
$dates = get_dates_filter( 'objects' );
// Determine graph options
switch ( $dates['range'] ) {
case 'today':
case 'yesterday':
case 'this_year':
case 'last_year':
$day_by_day = false;
break;
case 'other':
$difference = ( $dates['end']->getTimestamp() - $dates['start']->getTimestamp() );
if ( $difference >= ( YEAR_IN_SECONDS / 4 ) ) {
$day_by_day = false;
} else {
$day_by_day = true;
}
break;
default:
$day_by_day = true;
break;
}
return $day_by_day;
}
/**
* Given a function and column, make a timezone converted groupby query.
*
* @since 3.0
* @since 3.0.4 If MONTH is passed as the function, always add YEAR and MONTH
* to avoid issues with spanning multiple years.
*
* @param string $function The function to run the value through, like DATE, HOUR, MONTH.
* @param string $column The column to group by.
*
* @return string
*/
function get_groupby_date_string( $function = 'DATE', $column = 'date_created' ) {
$function = strtoupper( $function );
$date = EDD()->utils->date( 'now', edd_get_timezone_id(), false );
$gmt_offset = $date->getOffset();
if ( empty( $gmt_offset ) ) {
switch ( $function ) {
case 'HOUR':
$group_by_string = "DAY({$column}), HOUR({$column})";
break;
case 'MONTH':
$group_by_string = "YEAR({$column}), MONTH({$column})";
break;
default:
$group_by_string = "{$function}({$column})";
break;
}
return $group_by_string;
}
// Output the offset in the proper format.
$hours = abs( floor( $gmt_offset / HOUR_IN_SECONDS ) );
$minutes = abs( floor( ( $gmt_offset / MINUTE_IN_SECONDS ) % MINUTE_IN_SECONDS ) );
$math = ( $gmt_offset >= 0 ) ? '+' : '-';
$formatted_offset = ! empty( $minutes ) ? "{$hours}:{$minutes}" : $hours . ':00';
/**
* There is a limitation here that we cannot get past due to MySQL not having timezone information.
*
* When a requested date group spans the DST change. For instance, a 6 month graph will have slightly
* different results for each month than if you pulled each of those 6 months individually. This is because
* our 'grouping' can only convert the timezone based on the current offset and that can change if the
* range spans the DST break, which would have some dates be in a +/- 1 hour state.
*
* @see https://github.com/awesomemotive/easy-digital-downloads/pull/9449
*/
$column_conversion = "CONVERT_TZ({$column}, '+0:00', '{$math}{$formatted_offset}')";
switch ( $function ) {
case 'HOUR':
$group_by_string = "DAY({$column_conversion}), HOUR({$column_conversion})";
break;
case 'MONTH':
$group_by_string = "YEAR({$column_conversion}), MONTH({$column_conversion})";
break;
default:
$group_by_string = "{$function}({$column_conversion})";
break;
}
return $group_by_string;
}
/**
* Retrieves the tax exclusion filter.
*
* @since 3.0
*
* @return bool True if taxes should be excluded from calculations.
*/
function get_taxes_excluded_filter() {
$taxes = get_filter_value( 'taxes' );
if ( ! isset( $taxes['exclude_taxes'] ) ) {
return false;
}
return (bool) $taxes['exclude_taxes'];
}
/** Display *******************************************************************/
/**
* Handles display of a report.
*
* @since 3.0
*
* @param Data\Report $report Report object.
*/
function default_display_report( $report ) {
// Bail if erroneous report
if ( empty( $report ) || is_wp_error( $report ) ) {
return;
}
// Try to output: tiles, tables, and charts
$report->display_endpoint_group( 'tiles' );
$report->display_endpoint_group( 'tables' );
$report->display_endpoint_group( 'charts' );
}
/**
* Displays the default content for a tile endpoint.
*
* @since 3.0
*
* @param Data\Report $report Report object the tile endpoint is being rendered in.
* Not always set.
* @param array $tile {
* Tile display arguments.
*
* @type Data\Tile_Endpoint $endpoint Endpoint object.
* @type mixed|array $data Date for display. By default, will be an array,
* but can be of other types.
* @type array $display_args Array of any display arguments.
* }
* @return void Meta box display callbacks only echo output.
*/
function default_display_tile( $endpoint, $data, $args ) {
echo '<div class="tile-label">' . esc_html( $endpoint->get_label() ) . '</div>';
if ( empty( $data ) ) {
echo '<div class="tile-no-data tile-value">&mdash;</div>';
} else {
switch ( $args['type'] ) {
case 'number':
echo '<div class="tile-number tile-value">' . edd_format_amount( $data ) . '</div>';
break;
case 'split-number':
printf( '<div class="tile-amount tile-value">%1$d / %2$d</div>',
edd_format_amount( $data['first_value'] ),
edd_format_amount( $data['second_value'] )
);
break;
case 'split-amount':
printf( '<div class="tile-amount tile-value">%1$d / %2$d</div>',
edd_currency_filter( edd_format_amount( $data['first_value'] ) ),
edd_currency_filter( edd_format_amount( $data['second_value'] ) )
);
break;
case 'relative':
$direction = ( ! empty( $data['direction'] ) && in_array( $data['direction'], array( 'up', 'down' ), true ) )
? '-' . sanitize_key( $data['direction'] )
: '';
echo '<div class="tile-change' . esc_attr( $direction ) . ' tile-value">' . edd_format_amount( $data['value'] ) . '</div>';
break;
case 'amount':
echo '<div class="tile-amount tile-value">' . edd_currency_filter( edd_format_amount( $data ) ) . '</div>';
break;
case 'url':
echo '<div class="tile-url tile-value">' . esc_url( $data ) . '</div>';
break;
default:
$tags = wp_kses_allowed_html( 'post' );
echo '<div class="tile-value tile-default">' . wp_kses( $data, $tags ) . '</div>';
break;
}
}
if ( ! empty( $args['comparison_label'] ) ) {
echo '<div class="tile-compare">' . esc_attr( $args['comparison_label'] ) . '</div>';
}
}
/**
* Handles default display of all tile endpoints registered against a report.
*
* @since 3.0
*
* @param Data\Report $report Report object.
*/
function default_display_tiles_group( $report ) {
if ( ! $report->has_endpoints( 'tiles' ) ) {
return;
}
$tiles = $report->get_endpoints( 'tiles' );
?>
<div id="edd-reports-tiles-wrap" class="edd-report-wrap">
<?php
foreach ( $tiles as $endpoint_id => $tile ) :
$tile->display();
endforeach;
?>
</div>
<?php
}
/**
* Handles default display of all table endpoints registered against a report.
*
* @since 3.0
*
* @param Data\Report $report Report object.
*/
function default_display_tables_group( $report ) {
if ( ! $report->has_endpoints( 'tables' ) ) {
return;
}
$tables = $report->get_endpoints( 'tables' ); ?>
<div id="edd-reports-tables-wrap" class="edd-report-wrap"><?php
foreach ( $tables as $endpoint_id => $table ) :
?><div class="edd-reports-table" id="edd-reports-table-<?php echo esc_attr( $endpoint_id ); ?>">
<h3><?php echo esc_html( $table->get_label() ); ?></h3><?php
$table->display();
?></div><?php
endforeach;
?><div class="clear"></div></div><?php
}
/**
* Handles default display of all chart endpoints registered against a report.
*
* @since 3.0
*
* @param Data\Report $report Report object.
*/
function default_display_charts_group( $report ) {
if ( ! $report->has_endpoints( 'charts' ) ) {
return;
}
?>
<div id="edd-reports-charts-wrap" class="edd-report-wrap">
<?php
$charts = $report->get_endpoints( 'charts' );
foreach ( $charts as $endpoint_id => $chart ) {
?>
<div class="edd-reports-chart edd-reports-chart-<?php echo esc_attr( $chart->get_type() ); ?>" id="edd-reports-table-<?php echo esc_attr( $endpoint_id ); ?>">
<h3><?php echo esc_html( $chart->get_label() ); ?></h3>
<?php $chart->display(); ?>
</div>
<?php
}
?>
<div class="chart-timezone">
<?php printf( esc_html__( 'Chart time zone: %s', 'easy-digital-downloads' ), esc_html( edd_get_timezone_id() ) ); ?>
</div>
</div>
<?php
}
/**
* Handles display of the 'Date' filter for reports.
*
* @since 3.0
*/
function display_dates_filter() {
$date_format = get_option( 'date_format' );
$range_options = get_dates_filter_options();
$relative_range_options = get_relative_dates_filter_options();
$dates = get_filter_value( 'dates' );
$selected_range = isset( $dates['range'] )
? $dates['range']
: get_dates_filter_range();
$selected_relative_range = isset( $dates['relative_range'] )
? $dates['relative_range']
: get_relative_dates_filter_range();
$class = ( 'other' !== $selected_range )
? ' screen-reader-text'
: '';
$range_select = EDD()->html->select(
array(
'name' => 'range',
'class' => 'edd-graphs-date-options',
'options' => $range_options,
'variations' => false,
'show_option_all' => false,
'show_option_none' => false,
'selected' => $selected_range,
)
);
$relative_range_select = EDD()->html->select(
array(
'name' => 'relative_range',
'class' => 'edd-graphs-relative-date-options',
'options' => $relative_range_options,
'variations' => false,
'show_option_all' => false,
'show_option_none' => false,
'selected' => $selected_relative_range,
)
);
// From.
$from = EDD()->html->date_field(
array(
'id' => 'filter_from',
'name' => 'filter_from',
'value' => ( empty( $dates['from'] ) || ( 'other' !== $dates['range'] ) ) ? '' : $dates['from'],
'placeholder' => _x( 'From', 'date filter', 'easy-digital-downloads' ),
)
);
// To.
$to = EDD()->html->date_field(
array(
'id' => 'filter_to',
'name' => 'filter_to',
'value' => ( empty( $dates['to'] ) || ( 'other' !== $dates['range'] ) ) ? '' : $dates['to'],
'placeholder' => _x( 'To', 'date filter', 'easy-digital-downloads' ),
)
);
// Output fields
?>
<div class="edd-date-range-picker graph-option-section" data-range="<?php echo esc_attr( $selected_range ); ?>">
<?php echo $range_select; ?>
<!-- DATE RANGES -->
<div class="edd-date-range-dates">
<span class="dashicons dashicons-calendar edd-date-main-icon"></span>
<div class="edd-date-range-selected-date">
<?php
foreach ( $range_options as $range_key => $range_name ) :
$range_dates = \EDD\Reports\parse_dates_for_range( $range_key );
$selected_range_class = ( $selected_range !== $range_key ) ? 'hidden' : '';
$start_date = edd_get_edd_timezone_equivalent_date_from_utc( $range_dates['start'] )->format( $date_format );
$end_date = edd_get_edd_timezone_equivalent_date_from_utc( $range_dates['end'] )->format( $date_format );
$label = $start_date;
if ( $start_date !== $end_date ) {
$label = $start_date . ' - ' . $end_date;
}
?>
<span class="<?php echo esc_attr( $selected_range_class ); ?>" data-range="<?php echo esc_attr( $range_key ); ?>" data-default-relative-range="<?php echo \EDD\Reports\get_default_relative_range( $range_key ); ?>"><?php echo esc_html( $label ); ?></span>
<?php
endforeach;
?>
</div>
</div>
<!-- RELATIVE DATE RANGES -->
<div class="edd-date-range-relative-dates">
<div class="hidden"><?php echo $relative_range_select; ?></div>
<?php echo esc_html__( 'compared to', 'easy-digital-downloads' ); ?>
<div class="edd-date-range-selected-relative-date">
<span class="edd-date-range-selected-relative-range-name"><?php echo esc_html( $relative_range_options[ $selected_relative_range ] ); ?></span>
<img src="<?php echo esc_url( EDD_PLUGIN_URL . '/assets/images/icons/icon-chevron-down.svg' ); ?>" class="arrow-down">
<!-- RELATIVE DATE RANGES DROPDOWN -->
<div class="edd-date-range-relative-dropdown">
<?php echo \EDD\Reports\display_relative_dates_dropdown_options( $selected_range, $selected_relative_range ); ?>
</div>
</div>
</div>
</div>
<span class="edd-date-range-options graph-option-section edd-from-to-wrapper<?php echo esc_attr( $class ); ?>">
<?php echo $from . $to; ?>
</span>
<?php
}
/**
* Handles display of the relative dates dropdown options.
*
* @since 3.1
*/
function display_relative_dates_dropdown_options( $range, $selected_relative_range ) {
$date_format = get_option( 'date_format' );
$relative_range_options = get_relative_dates_filter_options();
?>
<ul data-range="<?php echo esc_attr( $range ); ?>">
<?php
foreach ( $relative_range_options as $relative_range_key => $relative_range_name ) :
$relative_range_dates = \EDD\Reports\parse_relative_dates_for_range( $range, $relative_range_key );
$selected_range_class = ( $selected_relative_range === $relative_range_key ) ? 'active' : '';
?>
<li class="<?php echo esc_attr( $selected_range_class ); ?>" data-range="<?php echo esc_attr( $relative_range_key ); ?>">
<span class="date-range-name"><?php echo esc_html( $relative_range_options[ $relative_range_key ] ); ?></span>
<span class="date-range-dates"><?php echo esc_html( edd_get_edd_timezone_equivalent_date_from_utc( $relative_range_dates['start'] )->format( $date_format ) ); ?> - <?php echo esc_html( edd_get_edd_timezone_equivalent_date_from_utc( $relative_range_dates['end'] )->format( $date_format ) ); ?></span>
</li>
<?php
endforeach;
?>
</ul>
<?php
}
/**
* Handles display of the 'Products' filter for reports.
*
* @since 3.0
*/
function display_products_filter() {
$products = get_filter_value( 'products' );
$select = EDD()->html->product_dropdown( array(
'chosen' => true,
'variations' => true,
'selected' => empty( $products ) ? 0 : $products,
'show_option_none' => false,
'show_option_all' => sprintf( __( 'All %s', 'easy-digital-downloads' ), edd_get_label_plural() ),
) ); ?>
<span class="edd-graph-filter-options graph-option-section"><?php
echo $select;
?></span><?php
}
/**
* Handles display of the 'Products Dropdown' filter for reports.
*
* @since 3.0
*/
function display_product_categories_filter() {
?>
<span class="edd-graph-filter-options graph-option-selection">
<?php echo EDD()->html->category_dropdown( 'product_categories', get_filter_value( 'product_categories' ) ); ?>
</span>
<?php
}
/**
* Handles display of the 'Exclude Taxes' filter for reports.
*
* @since 3.0
*/
function display_taxes_filter() {
if ( false === edd_use_taxes() ) {
return;
}
$taxes = get_filter_value( 'taxes' );
$exclude_taxes = isset( $taxes['exclude_taxes'] ) && true == $taxes['exclude_taxes'];
?>
<span class="edd-graph-filter-options graph-option-section">
<label for="exclude_taxes">
<input type="checkbox" id="exclude_taxes" <?php checked( true, $exclude_taxes, true ); ?> value="1" name="exclude_taxes"/>
<?php esc_html_e( 'Exclude Taxes', 'easy-digital-downloads' ); ?>
</label>
</span>
<?php
}
/**
* Handles display of the 'Discounts' filter for reports.
*
* @since 3.0
*/
function display_discounts_filter() {
$discount = get_filter_value( 'discounts' );
$d = edd_get_discounts( array(
'fields' => array( 'code', 'name' ),
'number' => 100,
) );
$discounts = array();
foreach ( $d as $discount_data ) {
$discounts[ $discount_data->code ] = esc_html( $discount_data->name );
}
// Get the select
$select = EDD()->html->discount_dropdown( array(
'name' => 'discounts',
'chosen' => true,
'selected' => empty( $discount ) ? 0 : $discount,
) ); ?>
<span class="edd-graph-filter-options graph-option-section"><?php
echo $select;
?></span><?php
}
/**
* Handles display of the 'Gateways' filter for reports.
*
* @since 3.0
*/
function display_gateways_filter() {
$gateway = get_filter_value( 'gateways' );
$known_gateways = edd_get_payment_gateways();
$gateways = array();
foreach ( $known_gateways as $id => $data ) {
$gateways[ $id ] = esc_html( $data['admin_label'] );
}
// Get the select
$select = EDD()->html->select( array(
'name' => 'gateways',
'options' => $gateways,
'selected' => empty( $gateway ) ? 0 : $gateway,
'show_option_none' => false,
) ); ?>
<span class="edd-graph-filter-options graph-option-section"><?php
echo $select;
?></span><?php
}
/**
* Handles display of the 'Country' filter for reports.
*
* @since 3.0
*/
function display_region_filter() {
$region = get_filter_value( 'regions' );
$country = get_filter_value( 'countries' );
if ( empty( $region ) ) {
$region = '';
}
if ( empty( $country ) ) {
$country = '';
}
$regions = edd_get_shop_states( $country );
// Remove empty values.
$regions = array_filter( $regions );
// Get the select
$select = EDD()->html->region_select(
array(
'name' => 'regions',
'id' => 'edd_reports_filter_regions',
'options' => $regions,
),
$country,
$region
);
?>
<span class="edd-graph-filter-options graph-option-section"><?php
echo $select;
?></span><?php
}
/**
* Handles display of the 'Country' filter for reports.
*
* @since 3.0
*/
function display_country_filter() {
$country = get_filter_value( 'countries' );
if ( empty( $country ) ) {
$country = '';
}
$countries = edd_get_country_list();
// Remove empty values.
$countries = array_filter( $countries );
// Get the select
$select = EDD()->html->country_select(
array(
'name' => 'countries',
'id' => 'edd_reports_filter_countries',
'options' => $countries,
),
$country
);
?>
<span class="edd-graph-filter-options graph-option-section"><?php
echo $select;
?></span><?php
}
/**
* Handles the display of the 'Currency' filter for reports.
*
* @since 3.0
*/
function display_currency_filter() {
$currency = get_filter_value( 'currencies' );
if ( empty( $currency ) ) {
$currency = 'all';
}
$order_currencies = get_transient( 'edd_distinct_order_currencies' );
if ( false === $order_currencies ) {
global $wpdb;
$order_currencies = $wpdb->get_col(
"SELECT distinct currency FROM {$wpdb->edd_orders}"
);
if ( is_array( $order_currencies ) ) {
$order_currencies = array_filter( $order_currencies );
}
set_transient( 'edd_distinct_order_currencies', $order_currencies, 3 * HOUR_IN_SECONDS );
}
if ( ! is_array( $order_currencies ) || count( $order_currencies ) <= 1 ) {
return;
}
$all_currencies = array_intersect_key( edd_get_currencies(), array_flip( $order_currencies ) );
if ( array_key_exists( edd_get_currency(), $all_currencies ) ) {
$all_currencies = array_merge( array(
'convert' => sprintf( __( '%s - Converted', 'easy-digital-downloads' ), $all_currencies[ edd_get_currency() ] )
), $all_currencies );
}
?>
<span class="edd-graph-filter-options graph-option-section">
<?php
echo EDD()->html->select( array(
'name' => 'currencies',
'id' => 'edd_reports_filter_currencies',
'options' => $all_currencies,
'selected' => $currency,
'show_option_all' => false,
'show_option_none' => false
) );
?>
</span>
<?php
}
/**
* Displays the filters UI for a report.
*
* @since 3.0
*
* @param Data\Report $report Report object.
*/
function display_filters( $report ) {
$action = edd_get_admin_url( array(
'page' => 'edd-reports',
) );
?>
<form action="<?php echo esc_url( $action ); ?>" method="GET">
<?php edd_admin_filter_bar( 'reports', $report ); ?>
</form>
<?php
}
/**
* Output filter items
*
* @since 3.0
*
* @param object $report
*/
function filter_items( $report = false ) {
// Get the report ID
$report_id = $report->get_id();
// Bail if no report
if ( empty( $report_id ) ) {
return;
}
$redirect_url = edd_get_admin_url( array(
'page' => 'edd-reports',
'view' => sanitize_key( $report_id ),
) );
// Bail if no filters
$filters = $report->get_filters();
if ( empty( $filters ) ) {
return;
}
// Bail if no manifest
$manifest = get_filters();
if ( empty( $manifest ) ) {
return;
}
// Setup callables
$callables = array();
// Loop through filters and find the callables
foreach ( $filters as $filter ) {
// Skip if empty
if ( empty( $manifest[ $filter ]['display_callback'] ) ) {
continue;
}
// Skip if not callable
$callback = $manifest[ $filter ]['display_callback'];
if ( ! is_callable( $callback ) ) {
continue;
}
// Add callable to callables
$callables[] = $callback;
}
// Bail if no callables
if ( empty( $callables ) ) {
return;
}
// Start an output buffer
ob_start();
// Call the callables in the buffer
foreach ( $callables as $to_call ) {
call_user_func( $to_call, $report );
} ?>
<span class="edd-graph-filter-submit graph-option-section">
<input type="submit" class="button button-secondary" value="<?php esc_html_e( 'Filter', 'easy-digital-downloads' ); ?>"/>
<input type="hidden" name="edd_action" value="filter_reports">
<input type="hidden" name="edd_redirect" value="<?php echo esc_url_raw( $redirect_url ); ?>">
</span>
<?php
// Output the current buffer
echo ob_get_clean();
}
add_action( 'edd_admin_filter_bar_reports', 'EDD\Reports\filter_items' );
/**
* Renders the mobile link at the bottom of the payment history page
*
* @since 1.8.4
* @since 3.0 Updated filter to display link next to the reports filters.
*/
function mobile_link() {
$url = edd_link_helper(
'https://easydigitaldownloads.com/downloads/ios-app/',
array(
'utm_medium' => 'reports',
'utm_content' => 'ios-app',
)
);
?>
<span class="edd-mobile-link">
<a href="<?php echo $url; ?>" target="_blank">
<?php esc_html_e( 'Try the Sales/Earnings iOS App!', 'easy-digital-downloads' ); ?>
</a>
</span>
<?php
}
add_action( 'edd_after_admin_filter_bar_reports', 'EDD\Reports\mobile_link', 100 );
/** Compat ********************************************************************/
/**
* Private: Injects the value of $_REQUEST['range'] into the Reports\get_dates_filter_range() if set.
*
* To be used only for backward-compatibility with anything relying on the `$_REQUEST['range']` value.
*
* @since 3.0
* @access private
*
* @param string $range Currently resolved dates range.
* @return string (Maybe) modified range based on the value of `$_REQUEST['range']`.
*/
function compat_filter_date_range( $range ) {
return isset( $_REQUEST['range'] )
? sanitize_key( $_REQUEST['range'] )
: $range;
}