* * @copyright (c) 2016, Incsub (http://incsub.com) */ if ( ! class_exists( 'WpSmushNextGenStats' ) ) { class WpSmushNextGenStats extends WpSmushNextGen { /** * * @var array Contains the total Stats, for displaying it on bulk page */ var $stats = array(); private $is_pro_user; function __construct() { global $wp_smush; $this->is_pro_user = $wp_smush->validate_install(); // Update Total Image count add_action( 'ngg_added_new_image', array( $this, 'image_count' ), 10 ); // Update images list in cache add_action( 'wp_smush_nextgen_image_stats', array( $this, 'update_cache' ) ); // Add the resizing stats to Global stats add_action( 'wp_smush_image_nextgen_resized', array( $this, 'update_stats' ), '', 2 ); // Get the stats for single image, update the global stats add_action( 'wp_smush_nextgen_image_stats', array( $this, 'update_stats' ), '', 2 ); } /** * Refreshes the total image count when a new image is added to nextgen gallery * Should be called only if image count need to be updated, use total_count(), otherwise */ function image_count() { // Force the cache refresh for top-commented posts. $this->total_count( $force_refresh = true ); } /** * Get the images id for nextgen gallery * * @param bool $force_refresh Optional. Whether to force the cache to be refreshed. * Default false. * * @param bool $return_ids Whether to return the ids array, set to false by default * * @return int|mixed|void Returns the images ids or the count */ function total_count( $force_refresh = false, $return_ids = false ) { // Check for the wp_smush_images in the 'nextgen' group. $attachment_ids = wp_cache_get( 'wp_smush_images', 'nextgen' ); // If nothing is found, build the object. if ( true === $force_refresh || false === $attachment_ids ) { // Get the nextgen image ids $attachment_ids = $this->get_nextgen_attachments(); if ( ! is_wp_error( $attachment_ids ) ) { // In this case we don't need a timed cache expiration. wp_cache_set( 'wp_smush_images', $attachment_ids, 'nextgen' ); } } return $return_ids ? $attachment_ids : count( $attachment_ids ); } /** * Returns the ngg images list(id and meta ) or count * * @param string $type Whether to return smushed images or unsmushed images * @param bool|false $count Return count only * @param bool|false $force_update true/false to update the cache or not * * @return bool|mixed Returns assoc array of image ids and meta or Image count */ function get_ngg_images( $type = 'smushed', $count = false, $force_update = false ) { global $wpdb, $wpsmushit_admin; $limit = $wpsmushit_admin->nextgen_query_limit(); $offset = 0; // Check type of images being queried if ( ! in_array( $type, array( 'smushed', 'unsmushed' ) ) ) { return false; } // Check for the wp_smush_images_smushed in the 'nextgen' group. $images = wp_cache_get( 'wp_smush_images_' . $type, 'nextgen' ); // If nothing is found, build the object. if ( ! $images || $force_update ) { // Query Attachments for meta key while ( $attachments = $wpdb->get_results( "SELECT pid, meta_data FROM $wpdb->nggpictures LIMIT $offset, $limit" ) ) { foreach ( $attachments as $attachment ) { // Check if it has `wp_smush` key if ( class_exists( 'Ngg_Serializable' ) ) { $serializer = new Ngg_Serializable(); $meta = $serializer->unserialize( $attachment->meta_data ); } else { $meta = unserialize( $attachment->meta_data ); } // Store pid in image meta if ( is_array( $meta ) && empty( $meta['pid'] ) ) { $meta['pid'] = $attachment->pid; } elseif ( is_object( $meta ) && empty( $meta->pid ) ) { $meta->pid = $attachment->pid; } // Check meta for wp_smush if ( ! is_array( $meta ) || empty( $meta['wp_smush'] ) ) { $unsmushed_images[ $attachment->pid ] = $meta; continue; } $smushed_images[ $attachment->pid ] = $meta; } // Set the offset $offset += $limit; } if ( ! empty( $smushed_images ) ) { wp_cache_set( 'wp_smush_images_smushed', $smushed_images, 'nextgen', 300 ); } if ( ! empty( $unsmushed_images ) ) { wp_cache_set( 'wp_smush_images_unsmushed', $unsmushed_images, 'nextgen', 300 ); } } if ( $type == 'smushed' ) { $smushed_images = ! empty( $smushed_images ) ? $smushed_images : $images; if ( ! $smushed_images ) { return 0; } else { return $count ? count( $smushed_images ) : $smushed_images; } } else { $unsmushed_images = ! empty( $unsmushed_images ) ? $unsmushed_images : $images; if ( ! $unsmushed_images ) { return 0; } else { return $count ? count( $unsmushed_images ) : $unsmushed_images; } } } /** * Display the smush stats for the image * * @param int $pid Image Id stored in nextgen table * @param bool $wp_smush_data Stats, stored after smushing the image * @param string $image_type Used for determining if not gif, to show the Super Smush button * @param bool $text_only Return only text instead of button (Useful for Ajax) * @param bool $echo Whether to echo the stats or not * * @uses WpSmushNextGenAdmin::column_html(), WP_Smush::get_restore_link(), WP_Smush::get_resmush_link() * * @return bool|null|string|void */ function show_stats( $pid, $wp_smush_data = false, $image_type = '', $text_only = false, $echo = true ) { global $wp_smush, $wpsmushnextgenadmin, $wpsmush_settings; if ( empty( $wp_smush_data ) ) { return false; } $button_txt = $stats = ''; $show_button = $show_resmush = $show_restore = false; $bytes = isset( $wp_smush_data['stats']['bytes'] ) ? $wp_smush_data['stats']['bytes'] : 0; $bytes_readable = ! empty( $bytes ) ? size_format( $bytes, 1 ) : ''; $percent = isset( $wp_smush_data['stats']['percent'] ) ? $wp_smush_data['stats']['percent'] : 0; $percent = $percent < 0 ? 0 : $percent; if ( isset( $wp_smush_data['stats']['size_before'] ) && $wp_smush_data['stats']['size_before'] == 0 && ! empty( $wp_smush_data['sizes'] ) ) { $status_txt = __( 'Already Optimized', 'wp-smushit' ); } else { if ( $bytes == 0 || $percent == 0 ) { $status_txt = __( 'Already Optimized', 'wp-smushit' ); // Add resmush option if needed $show_resmush = $this->show_resmush( $show_resmush, $wp_smush_data ); if ( $show_resmush ) { $status_txt .= '
' . $wp_smush->get_resmsuh_link( $pid, 'nextgen' ); } } elseif ( ! empty( $percent ) && ! empty( $bytes_readable ) ) { $status_txt = sprintf( __( 'Reduced by %1$s ( %2$01.1f%% )', 'wp-smushit' ), $bytes_readable, number_format_i18n( $percent, 2, '.', '' ) ); $show_resmush = $this->show_resmush( $show_resmush, $wp_smush_data ); if ( $show_resmush ) { $status_txt .= '
' . $wp_smush->get_resmsuh_link( $pid, 'nextgen' ); } // Restore Image: Check if we need to show the restore image option $show_restore = $this->show_restore_option( $pid, $wp_smush_data ); if ( $show_restore ) { if ( $show_resmush ) { // Show Separator $status_txt .= ' | '; } else { // Show the link in next line $status_txt .= '
'; } $status_txt .= $wp_smush->get_restore_link( $pid, 'nextgen' ); } // Show detailed stats if available if ( ! empty( $wp_smush_data['sizes'] ) ) { if ( $show_resmush || $show_restore ) { // Show Separator $status_txt .= ' | '; } else { // Show the link in next line $status_txt .= '
'; } // Detailed Stats Link $status_txt .= '' . esc_html__( 'Smush stats', 'wp-smushit' ) . ' [+]'; // Get metadata For the image // Registry Object for NextGen Gallery $registry = C_Component_Registry::get_instance(); // Gallery Storage Object $storage = $registry->get_utility( 'I_Gallery_Storage' ); // get an array of sizes available for the $image $sizes = $storage->get_image_sizes(); $image = $storage->object->_image_mapper->find( $pid ); $full_image = $storage->get_image_abspath( $image, 'full' ); // Stats $stats = $this->get_detailed_stats( $pid, $wp_smush_data, array( 'sizes' => $sizes ), $full_image ); if ( ! $text_only ) { $status_txt .= $stats; } } } } // IF current compression is lossy if ( ! empty( $wp_smush_data ) && ! empty( $wp_smush_data['stats'] ) ) { $lossy = ! empty( $wp_smush_data['stats']['lossy'] ) ? $wp_smush_data['stats']['lossy'] : ''; $is_lossy = $lossy == 1 ? true : false; } // Check if Lossy enabled $opt_lossy_val = $wpsmush_settings->settings['lossy']; // Check if premium user, compression was lossless, and lossy compression is enabled if ( ! $show_resmush && $this->is_pro_user && ! $is_lossy && $opt_lossy_val && ! empty( $image_type ) && $image_type != 'image/gif' ) { // the button text $button_txt = __( 'Super-Smush', 'wp-smushit' ); $show_button = true; } if ( $text_only ) { // For ajax response return array( 'status' => $status_txt, 'stats' => $stats, ); } // If show button is true for some reason, column html can print out the button for us $text = $wpsmushnextgenadmin->column_html( $pid, $status_txt, $button_txt, $show_button, true, $echo ); if ( ! $echo ) { return $text; } } /** * Updated the global smush stats for NextGen gallery * * @param $stats Compression stats fo respective image */ function update_stats( $image_id, $stats ) { $stats = ! empty( $stats['stats'] ) ? $stats['stats'] : ''; $smush_stats = get_option( 'wp_smush_stats_nextgen', array() ); if ( ! empty( $stats ) ) { // Human Readable $smush_stats['human'] = ! empty( $smush_stats['bytes'] ) ? size_format( $smush_stats['bytes'], 1 ) : ''; // Size of images before the compression $smush_stats['size_before'] = ! empty( $smush_stats['size_before'] ) ? ( $smush_stats['size_before'] + $stats['size_before'] ) : $stats['size_before']; // Size of image after compression $smush_stats['size_after'] = ! empty( $smush_stats['size_after'] ) ? ( $smush_stats['size_after'] + $stats['size_after'] ) : $stats['size_after']; $smush_stats['bytes'] = ! empty( $smush_stats['size_before'] ) && ! empty( $smush_stats['size_after'] ) ? ( $smush_stats['size_before'] - $smush_stats['size_after'] ) : 0; // Compression Percentage $smush_stats['percent'] = ! empty( $smush_stats['size_before'] ) && ! empty( $smush_stats['size_after'] ) && $smush_stats['size_before'] > 0 ? ( $smush_stats['bytes'] / $smush_stats['size_before'] ) * 100 : $stats['percent']; } update_option( 'wp_smush_stats_nextgen', $smush_stats, false ); } /** * Updated the global smush stats for NextGen gallery * * @param $stats Compression stats fo respective image */ function update_resize_stats( $image_id, $stats ) { global $wp_smush; $stats = ! empty( $stats['stats'] ) ? $stats['stats'] : ''; $smush_stats = get_option( 'wp_smush_stats_nextgen', array() ); if ( ! empty( $stats ) ) { // Compression Bytes $smush_stats['bytes'] = ! empty( $smush_stats['bytes'] ) ? ( $smush_stats['bytes'] + $stats['bytes'] ) : $stats['bytes']; // Human Readable $smush_stats['human'] = ! empty( $smush_stats['bytes'] ) ? size_format( $smush_stats['bytes'], 1 ) : ''; // Size of images before the compression $smush_stats['size_before'] = ! empty( $smush_stats['size_before'] ) ? ( $smush_stats['size_before'] + $stats['size_before'] ) : $stats['size_before']; // Size of image after compression $smush_stats['size_after'] = ! empty( $smush_stats['size_after'] ) ? ( $smush_stats['size_after'] + $stats['size_after'] ) : $stats['size_after']; // Compression Percentage $smush_stats['percent'] = ! empty( $smush_stats['size_before'] ) && ! empty( $smush_stats['size_after'] ) && $smush_stats['size_before'] > 0 ? ( $smush_stats['bytes'] / $smush_stats['size_before'] ) * 100 : $stats['percent']; } update_option( 'wp_smush_stats_nextgen', $smush_stats, false ); } /** * Get the attachment stats for a image * * @param $id * * @return null */ function get_attachment_stats( $id ) { // We'll get the image object in $id itself, else fetch it using Gallery Storage if ( is_object( $id ) || is_array( $id ) ) { $image = $id; } else { // Registry Object for NextGen Gallery $registry = C_Component_Registry::get_instance(); // Gallery Storage Object $storage = $registry->get_utility( 'I_Gallery_Storage' ); // get an image object $image = $storage->object->_image_mapper->find( $id ); } // Check if we've smush stats, return it if ( is_object( $image ) ) { if ( ! empty( $image->meta_data ) && ! empty( $image->meta_data['wp_smush'] ) ) { return $image->meta_data['wp_smush']; } } elseif ( is_array( $image ) ) { if ( ! empty( $image['wp_smush'] ) ) { return $image['wp_smush']; } elseif ( ! empty( $image['meta_data'] ) && ! empty( $image['meta_data']['wp_smush'] ) ) { return $image['meta_data']['wp_smush']; } } return null; } /** * Get the Nextgen Smush stats * * @return bool|mixed|void */ function get_smush_stats() { global $wpsmushnextgenadmin; $smushed_stats = array( 'savings_bytes' => 0, 'size_before' => 0, 'size_after' => 0, 'savings_percent' => 0, ); // Clear up the stats if ( 0 == $this->total_count() ) { delete_option( 'wp_smush_stats_nextgen' ); } // Check for the wp_smush_images in the 'nextgen' group. $stats = get_option( 'wp_smush_stats_nextgen', array() ); if ( empty( $stats['bytes'] ) || $stats['bytes'] < 0 ) { $stats['bytes'] = 0; } if ( ! empty( $stats['size_before'] ) && $stats['size_before'] > 0 ) { $stats['percent'] = ( $stats['bytes'] / $stats['size_before'] ) * 100; } // Round off precentage $stats['percent'] = ! empty( $stats['percent'] ) ? round( $stats['percent'], 1 ) : 0; $stats['human'] = size_format( $stats['bytes'], 1 ); $smushed_stats = array_merge( $smushed_stats, $stats ); // Gotta remove the stats for re-smush ids if ( is_array( $wpsmushnextgenadmin->resmush_ids ) && ! empty( $wpsmushnextgenadmin->resmush_ids ) ) { $resmush_stats = $this->get_stats_for_ids( $wpsmushnextgenadmin->resmush_ids ); // Recalculate stats, Remove stats for resmush ids $smushed_stats = $this->recalculate_stats( 'sub', $smushed_stats, $resmush_stats ); } return $smushed_stats; } /** * Updates the cache for Smushed and Unsmushed images */ function update_cache() { $this->get_ngg_images( 'smushed', '', true ); } /** * Returns the Stats for a image formatted into a nice table * * @param $image_id * @param $wp_smush_data * @param $attachment_metadata * @param $full_image * * @return string */ function get_detailed_stats( $image_id, $wp_smush_data, $attachment_metadata, $full_image ) { global $wp_smush; $stats = ''; return $stats; } /** * Compare Values * * @param $a * @param $b * * @return int */ function cmp( $a, $b ) { if ( is_object( $a ) ) { // Check and typecast $b if required $b = is_object( $b ) ? $b : (object) $b; return $a->bytes < $b->bytes; } elseif ( is_array( $a ) ) { $b = is_array( $b ) ? $b : (array) $b; return $a['bytes'] < $b['bytes']; } } /** * Return a list of images not smushed and reason * * @param $image_id * @param $size_stats * @param $attachment_metadata * * @return array */ function get_skipped_images( $size_stats, $full_image ) { $skipped = array(); // If full image was not smushed, reason 1. Large Size logic, 2. Free and greater than 1Mb if ( ! array_key_exists( 'full', $size_stats ) ) { // For free version, Check the image size if ( ! $this->is_pro_user ) { // For free version, check if full size is greater than 1 Mb, show the skipped status $file_size = file_exists( $full_image ) ? filesize( $full_image ) : ''; if ( ! empty( $file_size ) && ( $file_size / WP_SMUSH_MAX_BYTES ) > 1 ) { $skipped[] = array( 'size' => 'full', 'reason' => 'size_limit', ); } else { $skipped[] = array( 'size' => 'full', 'reason' => 'large_size', ); } } else { // Paid version, Check if we have large size if ( array_key_exists( 'large', $size_stats ) ) { $skipped[] = array( 'size' => 'full', 'reason' => 'large_size', ); } } } return $skipped; } /** * Check if image can be resmushed * * @param $status_txt * * @return string */ function show_resmush( $show_resmush, $wp_smush_data ) { global $wp_smush; // Resmush: Show resmush link, Check if user have enabled smushing the original and full image was skipped if ( $wp_smush->smush_original ) { // IF full image was not smushed if ( ! empty( $wp_smush_data ) && empty( $wp_smush_data['sizes']['full'] ) ) { $show_resmush = true; } } if ( ! $wp_smush->keep_exif ) { // If Keep Exif was set to tru initially, and since it is set to false now if ( ! empty( $wp_smush_data['stats']['keep_exif'] ) && $wp_smush_data['stats']['keep_exif'] == 1 ) { $show_resmush = true; } } return $show_resmush; } /** * Get the combined stats for given Ids * * @param $ids * * @return array Array of Stats for the given ids */ function get_stats_for_ids( $ids = array() ) { // Return if we don't have an array or no ids if ( ! is_array( $ids ) || empty( $ids ) ) { return false; } // Initialize the Stats array $stats = array( 'size_before' => 0, 'size_after' => 0, ); // Calculate the stats, Expensive Operation foreach ( $ids as $id ) { $image_stats = $this->get_attachment_stats( $id ); // Add the stats to $stats foreach ( $stats as $k => $value ) { if ( empty( $image_stats['stats'] ) || empty( $image_stats['stats'][ $k ] ) ) { continue; } $stats[ $k ] += $image_stats['stats'][ $k ]; } } // Calculate savings if ( ! empty( $stats['size_before'] ) && ! empty( $stats['size_after'] ) ) { $stats['bytes'] = $stats['size_before'] - $stats['size_after']; } return $stats; } /** * Add/Subtract the values from 2nd array to First array * This function is very specific to current requirement of stats re-calculation * * @param string $op 'add', 'sub' Add or Subtract the values * @param array $a1 * @param array $a2 * * @return array Return $a1 */ function recalculate_stats( $op = 'add', $a1 = array(), $a2 = array() ) { // If the first array itself is not set, return if ( empty( $a1 ) ) { return $a1; } // Iterate over keys in first array, and add/subtract the values foreach ( $a1 as $k => $v ) { // If the key is not set in 2nd array, skip if ( empty( $a2[ $k ] ) ) { continue; } // Else perform the operation, Considers the value to be integer, doesn't performs a check if ( 'sub' == $op ) { // Subtract the value $a1[ $k ] -= $a2[ $k ]; } elseif ( 'add' == $op ) { // add the value $a1[ $k ] += $a2[ $k ]; } } // Recalculate percentage and human savings $a1['percent'] = ! empty( $a1['size_before'] ) ? ( ( $a1['bytes'] / $a1['size_before'] ) * 100 ) : 0; $a1['human'] = ! empty( $a1['bytes'] ) ? size_format( $a1['bytes'], 1 ) : 0; return $a1; } /** * Get Super smushed images from the given images array * * @param array $images Array of images containing metadata * * @return array Array containing ids of supersmushed images */ function get_super_smushed_images( $images = array() ) { if ( empty( $images ) ) { return array(); } $super_smushed = array(); // Iterate Over all the images foreach ( $images as $image_k => $image ) { if ( empty( $image ) || ! is_array( $image ) || ! isset( $image['wp_smush'] ) ) { continue; } // Check for lossy compression if ( ! empty( $image['wp_smush']['stats'] ) && ! empty( $image['wp_smush']['stats']['lossy'] ) ) { $super_smushed[] = $image_k; } } return $super_smushed; } /** * Recalculate stats for the given smushed ids and update the cache * Update Super smushed image ids */ function update_stats_cache() { global $wpsmushnextgenadmin; // Get the Image ids $smushed_images = $this->get_ngg_images( 'smushed' ); $super_smushed = array( 'ids' => array(), 'timestamp' => '', ); $stats = $this->get_stats_for_ids( $smushed_images ); $lossy = $this->get_super_smushed_images( $smushed_images ); if ( empty( $stats['bytes'] ) && ! empty( $stats['size_before'] ) ) { $stats['bytes'] = $stats['size_before'] - $stats['size_after']; } $stats['human'] = size_format( $stats['bytes'] ); if ( ! empty( $stats['size_before'] ) ) { $stats['percent'] = ( $stats['bytes'] / $stats['size_before'] ) * 100; $stats['percent'] = round( $stats['percent'], 2 ); } $super_smushed['ids'] = $lossy; $super_smushed['timestamp'] = current_time( 'timestamp' ); // Update Re-smush list if ( is_array( $wpsmushnextgenadmin->resmush_ids ) && is_array( $smushed_images ) ) { $resmush_ids = array_intersect( $wpsmushnextgenadmin->resmush_ids, array_keys( $smushed_images ) ); } // If we have resmush ids, add it to db if ( ! empty( $resmush_ids ) ) { // Update re-smush images to db update_option( 'wp-smush-nextgen-resmush-list', $resmush_ids, false ); } // Update Super smushed images in db update_option( 'wp-smush-super_smushed_nextgen', $super_smushed, false ); // Update Stats Cache update_option( 'wp_smush_stats_nextgen', $stats, false ); } }//end class }//End Of if class not exists