2012-08-10 09:35:25 +00:00
< ? php
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
*
2018-03-09 20:28:08 +00:00
* @ package WooCommerce / Widgets
* @ version 2.3 . 0
2018-03-09 20:26:40 +00:00
*/
defined ( 'ABSPATH' ) || exit ;
/**
* Widget price filter class .
2011-09-04 00:02:44 +00:00
*/
2013-05-24 15:51:58 +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
*/
2013-05-24 15:51:58 +00:00
public function __construct () {
$this -> widget_cssclass = 'woocommerce widget_price_filter' ;
2017-08-25 11:07:17 +00:00
$this -> widget_description = __ ( 'Display a slider to filter products in your store by price.' , 'woocommerce' );
2013-05-24 15:51:58 +00:00
$this -> widget_id = 'woocommerce_price_filter' ;
2017-08-25 11:07:17 +00:00
$this -> widget_name = __ ( 'Filter Products by Price' , 'woocommerce' );
2013-05-24 15:51:58 +00:00
$this -> settings = array (
2018-03-09 20:26:40 +00:00
'title' => array (
2013-05-24 15:51:58 +00:00
'type' => 'text' ,
'std' => __ ( 'Filter by price' , 'woocommerce' ),
2016-08-27 01:46:45 +00:00
'label' => __ ( 'Title' , 'woocommerce' ),
),
2013-05-24 15:51:58 +00:00
);
2018-03-09 20:26:40 +00:00
$suffix = defined ( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG ? '' : '.min' ;
2019-01-10 15:13:50 +00:00
wp_register_script ( 'accounting' , WC () -> plugin_url () . '/assets/js/accounting/accounting' . $suffix . '.js' , array ( 'jquery' ), '0.4.2' , true );
2016-02-09 21:14:55 +00:00
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 );
2016-12-20 13:11:09 +00:00
wp_register_script ( 'wc-price-slider' , WC () -> plugin_url () . '/assets/js/frontend/price-slider' . $suffix . '.js' , array ( 'jquery-ui-slider' , 'wc-jquery-ui-touchpunch' , 'accounting' ), WC_VERSION , true );
2018-03-09 20:26:40 +00:00
wp_localize_script (
2019-01-10 15:13:50 +00:00
'wc-price-slider' ,
'woocommerce_price_slider_params' ,
array (
2018-03-09 20:26:40 +00:00
'currency_format_num_decimals' => 0 ,
'currency_format_symbol' => get_woocommerce_currency_symbol (),
'currency_format_decimal_sep' => esc_attr ( wc_get_price_decimal_separator () ),
'currency_format_thousand_sep' => esc_attr ( wc_get_price_thousand_separator () ),
'currency_format' => esc_attr ( str_replace ( array ( '%1$s' , '%2$s' ), array ( '%s' , '%v' ), get_woocommerce_price_format () ) ),
)
);
2017-09-16 19:37:29 +00:00
if ( is_customize_preview () ) {
wp_enqueue_script ( 'wc-price-slider' );
}
2013-05-24 15:51:58 +00:00
parent :: __construct ();
2011-08-10 17:11:11 +00:00
}
2012-08-14 17:37:50 +00:00
/**
2016-01-06 18:58:38 +00:00
* Output widget .
2012-08-14 17:37:50 +00:00
*
* @ see WP_Widget
2014-11-15 01:12:59 +00:00
*
2018-03-09 20:26:40 +00:00
* @ param array $args Arguments .
* @ param array $instance Widget instance .
2012-08-14 17:37:50 +00:00
*/
2013-05-24 15:51:58 +00:00
public function widget ( $args , $instance ) {
2017-11-09 12:01:27 +00:00
global $wp ;
2012-08-10 09:35:25 +00:00
2017-11-09 11:16:47 +00:00
if ( ! is_shop () && ! is_product_taxonomy () ) {
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
2019-01-23 18:29:33 +00:00
// Current active min and max price is retrieved from the query string.
$current_min_price = isset ( $_GET [ 'min_price' ] ) ? wc_clean ( wp_unslash ( $_GET [ 'min_price' ] ) ) : null ; // WPCS: input var ok, CSRF ok.
$current_max_price = isset ( $_GET [ 'max_price' ] ) ? wc_clean ( wp_unslash ( $_GET [ 'max_price' ] ) ) : null ; // WPCS: input var ok, CSRF ok.
2019-01-23 17:34:51 +00:00
// If there are not posts and we're not filtering, hide the widget.
2019-01-23 18:29:33 +00:00
if ( ! WC () -> query -> get_main_query () -> post_count && null === $current_min_price && null === $current_max_price ) {
2016-02-09 21:14:55 +00:00
return ;
2014-10-14 11:54:10 +00:00
}
2012-08-10 09:35:25 +00:00
wp_enqueue_script ( 'wc-price-slider' );
2019-01-23 18:29:33 +00:00
// Find min and max price in current result set.
2018-11-16 09:26:49 +00:00
$prices = $this -> get_filtered_price ();
$min_price = $prices -> min_price ;
$max_price = $prices -> max_price ;
// Check to see if we should add taxes to the prices if store are excl tax but display incl.
$tax_display_mode = get_option ( 'woocommerce_tax_display_shop' );
if ( wc_tax_enabled () && ! wc_prices_include_tax () && 'incl' === $tax_display_mode ) {
$class_min = $min_price ;
$class_max = $max_price ;
$tax_classes = array_merge ( array ( '' ), WC_Tax :: get_tax_classes () );
foreach ( $tax_classes as $tax_class ) {
$tax_rates = WC_Tax :: get_rates ( $tax_class );
if ( $tax_rates ) {
$class_min = $min_price + WC_Tax :: get_tax_total ( WC_Tax :: calc_exclusive_tax ( $min_price , $tax_rates ) );
$class_max = $max_price + WC_Tax :: get_tax_total ( WC_Tax :: calc_exclusive_tax ( $max_price , $tax_rates ) );
}
}
$min_price = $class_min ;
$max_price = $class_max ;
}
$min_price = apply_filters ( 'woocommerce_price_filter_widget_min_amount' , floor ( $min_price / 10 ) * 10 );
$max_price = apply_filters ( 'woocommerce_price_filter_widget_max_amount' , ceil ( $max_price / 10 ) * 10 );
2016-02-09 21:14:55 +00:00
2019-01-23 17:34:51 +00:00
// If both min and max are equal, we don't need a slider.
2019-01-23 18:29:33 +00:00
if ( $min_price === $max_price ) {
2013-05-24 15:51:58 +00:00
return ;
2014-10-14 11:54:10 +00:00
}
2012-08-10 09:35:25 +00:00
2018-11-16 09:26:49 +00:00
$current_min_price = is_null ( $current_min_price ) ? $min_price : floor ( $current_min_price / 10 ) * 10 ;
$current_max_price = is_null ( $current_max_price ) ? $max_price : ceil ( $current_max_price / 10 ) * 10 ;
$step = apply_filters ( 'woocommerce_price_filter_widget_step' , 10 );
2014-11-15 01:12:59 +00:00
$this -> widget_start ( $args , $instance );
2012-08-10 09:35:25 +00:00
2016-02-22 13:23:22 +00:00
if ( '' === get_option ( 'permalink_structure' ) ) {
2018-02-01 11:15:46 +00:00
$form_action = remove_query_arg ( array ( 'page' , 'paged' , 'product-page' ), add_query_arg ( $wp -> query_string , '' , home_url ( $wp -> request ) ) );
2014-10-14 11:54:10 +00:00
} else {
2015-04-10 13:34:09 +00:00
$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
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 " >
2011-12-02 11:43:15 +00:00
< div class = " price_slider " style = " display:none; " ></ div >
2019-01-23 18:29:33 +00:00
< div class = " price_slider_amount " data - step = " ' . esc_attr( $step ) . ' " >
< input type = " text " id = " min_price " name = " min_price " value = " ' . esc_attr( $current_min_price ) . ' " data - min = " ' . esc_attr( $min_price ) . ' " placeholder = " ' . esc_attr__( 'Min price', 'woocommerce' ) . ' " />
< input type = " text " id = " max_price " name = " max_price " value = " ' . esc_attr( $current_max_price ) . ' " data - max = " ' . esc_attr( $max_price ) . ' " placeholder = " ' . esc_attr__( 'Max price', 'woocommerce' ) . ' " />
2017-03-13 05:39:46 +00:00
< button type = " submit " class = " button " > ' . esc_html__( ' Filter ', ' woocommerce ' ) . ' </ button >
2011-12-02 11:43:15 +00:00
< div class = " price_label " style = " display:none; " >
2017-03-13 05:39:46 +00:00
' . esc_html__( ' Price : ', ' woocommerce ' ) . ' < span class = " from " ></ span > & mdash ; < span class = " to " ></ span >
2011-12-02 11:43:15 +00:00
</ div >
2019-01-23 15:31:26 +00:00
' . wc_query_string_form_fields( null, array( ' min_price ', ' max_price ', ' paged ' ), ' ', true ) . '
2011-12-02 11:43:15 +00:00
< div class = " clear " ></ div >
2011-08-10 17:11:11 +00:00
</ div >
</ div >
2018-03-09 20:26:40 +00:00
</ form > ' ; // WPCS: XSS ok.
2012-08-10 09:35:25 +00:00
2014-11-15 01:12:59 +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 .
2018-03-09 20:26:40 +00:00
*
2016-02-09 21:14:55 +00:00
* @ return int
*/
2016-02-10 10:36:06 +00:00
protected function get_filtered_price () {
2017-11-09 12:01:27 +00:00
global $wpdb ;
2016-02-09 21:14:55 +00:00
2018-11-08 10:59:07 +00:00
$args = WC () -> query -> get_main_query () -> query_vars ;
2016-02-09 21:14:55 +00:00
$tax_query = isset ( $args [ 'tax_query' ] ) ? $args [ 'tax_query' ] : array ();
$meta_query = isset ( $args [ 'meta_query' ] ) ? $args [ 'meta_query' ] : array ();
2017-05-19 10:19:30 +00:00
if ( ! is_post_type_archive ( 'product' ) && ! empty ( $args [ 'taxonomy' ] ) && ! empty ( $args [ 'term' ] ) ) {
2016-02-09 21:14:55 +00:00
$tax_query [] = array (
'taxonomy' => $args [ 'taxonomy' ],
'terms' => array ( $args [ 'term' ] ),
'field' => 'slug' ,
);
}
2016-12-09 15:43:25 +00:00
foreach ( $meta_query + $tax_query as $key => $query ) {
2016-02-09 21:14:55 +00:00
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-10-21 07:58:57 +00:00
$sql = " SELECT min( FLOOR( price_meta.meta_value ) ) as min_price, max( CEILING( price_meta.meta_value ) ) as max_price 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' ];
2016-09-15 11:05:59 +00:00
$sql .= " WHERE { $wpdb -> posts } .post_type IN (' " . implode ( " ',' " , array_map ( 'esc_sql' , apply_filters ( 'woocommerce_price_filter_post_type' , array ( 'product' ) ) ) ) . " ')
2018-03-09 20:26:40 +00:00
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 ' ) ) ) ) . "' )
AND price_meta . meta_value > '' " ;
2016-02-09 21:14:55 +00:00
$sql .= $tax_query_sql [ 'where' ] . $meta_query_sql [ 'where' ];
2018-03-09 20:26:40 +00:00
$search = WC_Query :: get_main_search_query_sql ();
if ( $search ) {
2016-12-23 14:45:58 +00:00
$sql .= ' AND ' . $search ;
}
2018-03-29 14:52:49 +00:00
$sql = apply_filters ( 'woocommerce_price_filter_sql' , $sql , $meta_query_sql , $tax_query_sql );
2018-03-09 20:26:40 +00:00
return $wpdb -> get_row ( $sql ); // WPCS: unprepared SQL ok.
2016-02-09 21:14:55 +00:00
}
2013-05-24 15:51:58 +00:00
}