• File: Chart.php
  • Full Path: /home/matthif/www/wp-content/plugins/wpforms-lite/src/Admin/Helpers/Chart.php
  • Date Modified: 02/16/2024 11:45 AM
  • File size: 4.37 KB
  • MIME-type: text/x-php
  • Charset: utf-8
<?php

namespace WPForms\Admin\Helpers;

use 
DateInterval;
use 
DatePeriod;
use 
DateTimeImmutable;

/**
 * Chart dataset processing helper methods.
 *
 * @since 1.8.2
 */
class Chart {

    
/**
     * Default date format.
     *
     * @since 1.8.2
     */
    
const DATE_FORMAT 'Y-m-d';

    
/**
     * Default date-time format.
     *
     * @since 1.8.2
     */
    
const DATETIME_FORMAT 'Y-m-d H:i:s';

    
/**
     * Processes the provided dataset to make sure the formatting needed for the "Chart.js" instance is provided.
     *
     * @since 1.8.2
     *
     * @param array             $query      Dataset retrieved from the database.
     * @param DateTimeImmutable $start_date Start date for the timespan.
     * @param DateTimeImmutable $end_date   End date for the timespan.
     *
     * @return array
     */
    
public static function process_chart_dataset_data$query$start_date$end_date ) { // phpcs:ignore Generic.Metrics.CyclomaticComplexity.TooHigh

        // Bail early if the given query contains no records to iterate.
        
if ( ! is_array$query ) || empty( $query ) ) {
            return [ 
0, [] ];
        }

        
$dataset        = [];
        
$timezone       wpforms_get_timezone(); // Retrieve the timezone object for the site.
        
$mysql_timezone timezone_open'UTC' ); // In the database, all datetime are stored in UTC.

        
foreach ( $query as $row ) {

            
$row_day   = isset( $row['day'] ) ? sanitize_text_field$row['day'] ) : '';
            
$row_count = isset( $row['count'] ) ? abs( (float) $row['count'] ) : 0;

            
// Skip the rest of the current iteration if the date (day) is unavailable.
            
if ( empty( $row_day ) ) {
                continue;
            }

            
// Since we won’t need the initial datetime instances after the query,
            // there is no need to create immutable date objects.
            
$row_datetime date_create_from_formatself::DATETIME_FORMAT$row_day$mysql_timezone );

            
// Skip the rest of the current iteration if the date creation function fails.
            
if ( ! $row_datetime ) {
                continue;
            }

            
$row_datetime->setTimezone$timezone );

            
$row_date_formatted $row_datetime->formatself::DATE_FORMAT );

            
// We must take into account entries submitted at different hours of the day,
            // because it is possible that more than one entry could be submitted on a given day.
            
if ( ! isset( $dataset$row_date_formatted ] ) ) {
                
$dataset$row_date_formatted ] = $row_count;

                continue;
            }

            
$dataset_count                  $dataset$row_date_formatted ];
            
$dataset$row_date_formatted ] = $dataset_count $row_count;
        }

        return 
self::format_chart_dataset_data$dataset$start_date$end_date );
    }

    
/**
     * Format given forms dataset to ensure correct data structure is parsed for serving the "chart.js" instance.
     * i.e., [ '2023-02-11' => [ 'day' => '2023-02-11', 'count' => 12 ] ].
     *
     * @since 1.8.2
     *
     * @param array             $dataset    Dataset for the chart.
     * @param DateTimeImmutable $start_date Start date for the timespan.
     * @param DateTimeImmutable $end_date   End date for the timespan.
     *
     * @return array
     */
    
private static function format_chart_dataset_data$dataset$start_date$end_date ) {

        
// In the event that there is no dataset to process, leave early.
        
if ( empty( $dataset ) ) {
            return [ 
0, [] ];
        }

        
$interval           = new DateInterval'P1D' ); // Variable that store the date interval of period 1 day.
        
$period             = new DatePeriod$start_date$interval$end_date ); // Used for iteration between start and end date period.
        
$data               = []; // Placeholder for the actual chart dataset data.
        
$total_entries      0;
        
$has_non_zero_count false;

        
// Use loop to store date into array.
        
foreach ( $period as $date ) {

            
$date_formatted          $date->formatself::DATE_FORMAT );
            
$count                   = isset( $dataset$date_formatted ] ) ? (float) $dataset$date_formatted ] : 0;
            
$total_entries          += $count;
            
$data$date_formatted ] = [
                
'day'   => $date_formatted,
                
'count' => $count,
            ];

            
// This check helps determine whether there is at least one non-zero count value in the dataset being processed.
            // It's used to optimize the function's behavior and to decide whether to include certain data in the returned result.
            
if ( $count && ! $has_non_zero_count ) {
                
$has_non_zero_count true;
            }
        }

        return [
            
$total_entries,
            
$has_non_zero_count $data : [], // We will return an empty array to indicate that there is no data to display.
        
];
    }
}