woocommerce/includes/widgets/class-wc-widget-price-filte...

246 lines
9.2 KiB
PHP
Raw Normal View History

2012-08-10 09:35:25 +00:00
<?php
if ( ! defined( 'ABSPATH' ) ) {
exit;
}
2011-08-10 17:11:11 +00:00
/**
2015-11-03 13:31:20 +00:00
* Price Filter Widget and related functions.
2012-08-10 09:35:25 +00:00
*
2012-08-14 17:37:50 +00:00
* Generates a range slider to filter products by price.
2011-08-10 17:11:11 +00:00
*
* @author WooThemes
* @category Widgets
* @package WooCommerce/Widgets
* @version 2.3.0
* @extends WC_Widget
2011-09-04 00:02:44 +00:00
*/
class WC_Widget_Price_Filter extends WC_Widget {
2012-08-10 09:35:25 +00:00
2012-08-14 17:37:50 +00:00
/**
2015-11-03 13:31:20 +00:00
* Constructor.
2012-08-14 17:37:50 +00:00
*/
public function __construct() {
$this->widget_cssclass = 'woocommerce widget_price_filter';
$this->widget_description = __( 'Shows a price filter slider in a widget which lets you narrow down the list of shown products when viewing product categories.', 'woocommerce' );
$this->widget_id = 'woocommerce_price_filter';
$this->widget_name = __( 'WooCommerce Price Filter', 'woocommerce' );
$this->settings = array(
'title' => array(
'type' => 'text',
'std' => __( 'Filter by price', 'woocommerce' ),
'label' => __( 'Title', 'woocommerce' )
)
);
2016-02-09 21:14:55 +00:00
$suffix = defined( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG ? '' : '.min';
wp_register_script( 'wc-jquery-ui-touchpunch', WC()->plugin_url() . '/assets/js/jquery-ui-touch-punch/jquery-ui-touch-punch' . $suffix . '.js', array( 'jquery-ui-slider' ), WC_VERSION, true );
wp_register_script( 'wc-price-slider', WC()->plugin_url() . '/assets/js/frontend/price-slider' . $suffix . '.js', array( 'jquery-ui-slider', 'wc-jquery-ui-touchpunch' ), WC_VERSION, true );
wp_localize_script( 'wc-price-slider', 'woocommerce_price_slider_params', array(
'currency_symbol' => get_woocommerce_currency_symbol(),
'currency_pos' => get_option( 'woocommerce_currency_pos' ),
'min_price' => isset( $_GET['min_price'] ) ? esc_attr( $_GET['min_price'] ) : '',
'max_price' => isset( $_GET['max_price'] ) ? esc_attr( $_GET['max_price'] ) : ''
) );
parent::__construct();
2011-08-10 17:11:11 +00:00
}
2012-08-14 17:37:50 +00:00
/**
* Output widget.
2012-08-14 17:37:50 +00:00
*
* @see WP_Widget
*
2012-08-14 17:37:50 +00:00
* @param array $args
* @param array $instance
*/
public function widget( $args, $instance ) {
2016-02-09 21:14:55 +00:00
global $_chosen_attributes, $wpdb, $wp, $wp_the_query;
2012-08-10 09:35:25 +00:00
2014-10-14 11:54:10 +00:00
if ( ! is_post_type_archive( 'product' ) && ! is_tax( get_object_taxonomies( 'product' ) ) ) {
2013-06-20 11:05:07 +00:00
return;
2014-10-14 11:54:10 +00:00
}
2012-08-10 09:35:25 +00:00
2016-02-09 21:14:55 +00:00
if ( ! $wp_the_query->post_count ) {
return;
2014-10-14 11:54:10 +00:00
}
2012-08-10 09:35:25 +00:00
2013-01-12 13:03:19 +00:00
$min_price = isset( $_GET['min_price'] ) ? esc_attr( $_GET['min_price'] ) : '';
$max_price = isset( $_GET['max_price'] ) ? esc_attr( $_GET['max_price'] ) : '';
2012-08-10 09:35:25 +00:00
wp_enqueue_script( 'wc-price-slider' );
2011-08-10 17:11:11 +00:00
// Remember current filters/search
$fields = '';
2012-08-10 09:35:25 +00:00
2014-10-14 11:54:10 +00:00
if ( get_search_query() ) {
$fields .= '<input type="hidden" name="s" value="' . get_search_query() . '" />';
2014-10-14 11:54:10 +00:00
}
2012-08-10 09:35:25 +00:00
2014-10-14 11:54:10 +00:00
if ( ! empty( $_GET['post_type'] ) ) {
$fields .= '<input type="hidden" name="post_type" value="' . esc_attr( $_GET['post_type'] ) . '" />';
2014-10-14 11:54:10 +00:00
}
2012-08-10 09:35:25 +00:00
2014-10-14 11:54:10 +00:00
if ( ! empty ( $_GET['product_cat'] ) ) {
$fields .= '<input type="hidden" name="product_cat" value="' . esc_attr( $_GET['product_cat'] ) . '" />';
2014-10-14 11:54:10 +00:00
}
2012-08-10 09:35:25 +00:00
2014-10-14 11:54:10 +00:00
if ( ! empty( $_GET['product_tag'] ) ) {
$fields .= '<input type="hidden" name="product_tag" value="' . esc_attr( $_GET['product_tag'] ) . '" />';
2014-10-14 11:54:10 +00:00
}
2014-10-14 11:54:10 +00:00
if ( ! empty( $_GET['orderby'] ) ) {
$fields .= '<input type="hidden" name="orderby" value="' . esc_attr( $_GET['orderby'] ) . '" />';
2014-10-14 11:54:10 +00:00
}
if ( ! empty( $_GET['min_rating'] ) ) {
$fields .= '<input type="hidden" name="min_rating" value="' . esc_attr( $_GET['min_rating'] ) . '" />';
}
if ( $_chosen_attributes ) {
foreach ( $_chosen_attributes as $attribute => $data ) {
$taxonomy_filter = 'filter_' . str_replace( 'pa_', '', $attribute );
$fields .= '<input type="hidden" name="' . esc_attr( $taxonomy_filter ) . '" value="' . esc_attr( implode( ',', $data['terms'] ) ) . '" />';
if ( 'or' == $data['query_type'] ) {
$fields .= '<input type="hidden" name="' . esc_attr( str_replace( 'pa_', 'query_type_', $attribute ) ) . '" value="or" />';
}
2014-10-14 11:54:10 +00:00
}
}
2012-08-10 09:35:25 +00:00
2016-02-09 21:14:55 +00:00
// Find min and max price in current result set
$min = $this->get_filtered_min_price();
$max = $this->get_filtered_max_price();
if ( $min === $max ) {
return;
2014-10-14 11:54:10 +00:00
}
2012-08-10 09:35:25 +00:00
$this->widget_start( $args, $instance );
2012-08-10 09:35:25 +00:00
if ( '' == get_option( 'permalink_structure' ) ) {
2012-06-29 20:18:22 +00:00
$form_action = remove_query_arg( array( 'page', 'paged' ), add_query_arg( $wp->query_string, '', home_url( $wp->request ) ) );
2014-10-14 11:54:10 +00:00
} else {
$form_action = preg_replace( '%\/page/[0-9]+%', '', home_url( trailingslashit( $wp->request ) ) );
2014-10-14 11:54:10 +00:00
}
2012-08-10 09:35:25 +00:00
2016-02-09 21:14:55 +00:00
// Adjust min and max if the store taxes are not displayed how they are stored
2015-06-09 13:58:27 +00:00
if ( wc_tax_enabled() && 'incl' === get_option( 'woocommerce_tax_display_shop' ) && ! wc_prices_include_tax() ) {
$tax_classes = array_merge( array( '' ), WC_Tax::get_tax_classes() );
2015-05-29 13:01:43 +00:00
2015-06-09 13:58:27 +00:00
foreach ( $tax_classes as $tax_class ) {
$tax_rates = WC_Tax::get_rates( $tax_class );
$class_min = $min + WC_Tax::get_tax_total( WC_Tax::calc_exclusive_tax( $min, $tax_rates ) );
$class_max = $max + WC_Tax::get_tax_total( WC_Tax::calc_exclusive_tax( $max, $tax_rates ) );
2016-02-09 21:14:55 +00:00
if ( $class_min < $min ) {
2015-06-09 13:58:27 +00:00
$min = $class_min;
}
if ( $class_max > $max ) {
$max = $class_max;
2015-05-29 13:01:43 +00:00
}
}
}
2012-08-10 09:35:25 +00:00
2015-01-29 14:33:06 +00:00
echo '<form method="get" action="' . esc_url( $form_action ) . '">
2011-08-10 17:11:11 +00:00
<div class="price_slider_wrapper">
<div class="price_slider" style="display:none;"></div>
2011-08-10 17:11:11 +00:00
<div class="price_slider_amount">
2016-02-09 21:14:55 +00:00
<input type="text" id="min_price" name="min_price" value="' . esc_attr( $min_price ) . '" data-min="' . esc_attr( apply_filters( 'woocommerce_price_filter_widget_min_amount', $min ) ) . '" placeholder="' . esc_attr__( 'Min price', 'woocommerce' ) . '" />
2015-08-05 19:17:52 +00:00
<input type="text" id="max_price" name="max_price" value="' . esc_attr( $max_price ) . '" data-max="' . esc_attr( apply_filters( 'woocommerce_price_filter_widget_max_amount', $max ) ) . '" placeholder="' . esc_attr__( 'Max price', 'woocommerce' ) . '" />
2015-01-29 14:33:06 +00:00
<button type="submit" class="button">' . __( 'Filter', 'woocommerce' ) . '</button>
<div class="price_label" style="display:none;">
2015-01-29 14:33:06 +00:00
' . __( 'Price:', 'woocommerce' ) . ' <span class="from"></span> &mdash; <span class="to"></span>
</div>
' . $fields . '
<div class="clear"></div>
2011-08-10 17:11:11 +00:00
</div>
</div>
</form>';
2012-08-10 09:35:25 +00:00
$this->widget_end( $args );
2011-08-10 17:11:11 +00:00
}
2016-02-09 21:14:55 +00:00
/**
* Get filtered min price for current products.
* @return int
*/
protected function get_filtered_min_price() {
global $wpdb, $wp_the_query;
$args = $wp_the_query->query_vars;
$tax_query = isset( $args['tax_query'] ) ? $args['tax_query'] : array();
$meta_query = isset( $args['meta_query'] ) ? $args['meta_query'] : array();
if ( ! empty( $args['taxonomy'] ) && ! empty( $args['term'] ) ) {
$tax_query[] = array(
'taxonomy' => $args['taxonomy'],
'terms' => array( $args['term'] ),
'field' => 'slug',
);
}
foreach ( $meta_query as $key => $query ) {
if ( ! empty( $query['price_filter'] ) || ! empty( $query['rating_filter'] ) ) {
unset( $meta_query[ $key ] );
}
}
$meta_query = new WP_Meta_Query( $meta_query );
$tax_query = new WP_Tax_Query( $tax_query );
$meta_query_sql = $meta_query->get_sql( 'post', $wpdb->posts, 'ID' );
$tax_query_sql = $tax_query->get_sql( $wpdb->posts, 'ID' );
2016-02-10 00:01:55 +00:00
$sql = "SELECT min( CAST( price_meta.meta_value AS UNSIGNED ) ) FROM {$wpdb->posts} ";
2016-02-09 21:14:55 +00:00
$sql .= " LEFT JOIN {$wpdb->postmeta} as price_meta ON {$wpdb->posts}.ID = price_meta.post_id " . $tax_query_sql['join'] . $meta_query_sql['join'];
$sql .= " WHERE {$wpdb->posts}.post_type = 'product'
AND {$wpdb->posts}.post_status = 'publish'
AND price_meta.meta_key IN ('" . implode( "','", array_map( 'esc_sql', apply_filters( 'woocommerce_price_filter_meta_keys', array( '_price' ) ) ) ) . "')
2016-02-10 00:04:01 +00:00
AND price_meta.meta_value > '' ";
2016-02-09 21:14:55 +00:00
$sql .= $tax_query_sql['where'] . $meta_query_sql['where'];
return floor( $wpdb->get_var( $sql ) );
}
/**
* Get filtered min price for current products.
* @return int
*/
protected function get_filtered_max_price() {
global $wpdb, $wp_the_query;
$args = $wp_the_query->query_vars;
$tax_query = isset( $args['tax_query'] ) ? $args['tax_query'] : array();
$meta_query = isset( $args['meta_query'] ) ? $args['meta_query'] : array();
if ( ! empty( $args['taxonomy'] ) && ! empty( $args['term'] ) ) {
$tax_query[] = array(
'taxonomy' => $args['taxonomy'],
'terms' => array( $args['term'] ),
'field' => 'slug',
);
}
foreach ( $meta_query as $key => $query ) {
if ( ! empty( $query['price_filter'] ) || ! empty( $query['rating_filter'] ) ) {
unset( $meta_query[ $key ] );
}
}
$meta_query = new WP_Meta_Query( $meta_query );
$tax_query = new WP_Tax_Query( $tax_query );
$meta_query_sql = $meta_query->get_sql( 'post', $wpdb->posts, 'ID' );
$tax_query_sql = $tax_query->get_sql( $wpdb->posts, 'ID' );
2016-02-10 00:01:55 +00:00
$sql = "SELECT MAX( CAST( price_meta.meta_value AS UNSIGNED ) ) FROM {$wpdb->posts} ";
2016-02-09 21:14:55 +00:00
$sql .= " LEFT JOIN {$wpdb->postmeta} as price_meta ON {$wpdb->posts}.ID = price_meta.post_id " . $tax_query_sql['join'] . $meta_query_sql['join'];
$sql .= " WHERE {$wpdb->posts}.post_type = 'product'
AND {$wpdb->posts}.post_status = 'publish'
AND price_meta.meta_key IN ('" . implode( "','", array_map( 'esc_sql', apply_filters( 'woocommerce_price_filter_meta_keys', array( '_price' ) ) ) ) . "')
2016-02-10 00:04:01 +00:00
AND price_meta.meta_value > '' ";
2016-02-09 21:14:55 +00:00
$sql .= $tax_query_sql['where'] . $meta_query_sql['where'];
return ceil( $wpdb->get_var( $sql ) );
}
}