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
|
|
|
|
2019-02-28 14:24:05 +00:00
|
|
|
// Requires lookup table added in 3.6.
|
|
|
|
if ( version_compare( get_option( 'woocommerce_db_version', null ), '3.6', '<' ) ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
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 17:34:51 +00:00
|
|
|
// If there are not posts and we're not filtering, hide the widget.
|
2019-01-23 19:39:59 +00:00
|
|
|
if ( ! WC()->query->get_main_query()->post_count && ! isset( $_GET['min_price'] ) && ! isset( $_GET['max_price'] ) ) { // WPCS: input var ok, CSRF ok.
|
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 19:39:59 +00:00
|
|
|
// Round values to nearest 10 by default.
|
|
|
|
$step = max( apply_filters( 'woocommerce_price_filter_widget_step', 10 ), 1 );
|
|
|
|
|
2017-09-16 19:37:29 +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 ) {
|
2019-01-23 19:39:59 +00:00
|
|
|
$tax_class = apply_filters( 'woocommerce_price_filter_widget_tax_class', '' ); // Uses standard tax class.
|
|
|
|
$tax_rates = WC_Tax::get_rates( $tax_class );
|
|
|
|
|
|
|
|
if ( $tax_rates ) {
|
|
|
|
$min_price += WC_Tax::get_tax_total( WC_Tax::calc_exclusive_tax( $min_price, $tax_rates ) );
|
|
|
|
$max_price += WC_Tax::get_tax_total( WC_Tax::calc_exclusive_tax( $max_price, $tax_rates ) );
|
2018-11-16 09:26:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-23 19:39:59 +00:00
|
|
|
$min_price = apply_filters( 'woocommerce_price_filter_widget_min_amount', floor( $min_price / $step ) * $step );
|
|
|
|
$max_price = apply_filters( 'woocommerce_price_filter_widget_max_amount', ceil( $max_price / $step ) * $step );
|
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
|
|
|
|
2019-01-23 19:39:59 +00:00
|
|
|
$current_min_price = isset( $_GET['min_price'] ) ? floor( floatval( wp_unslash( $_GET['min_price'] ) ) / $step ) * $step : $min_price; // WPCS: input var ok, CSRF ok.
|
|
|
|
$current_max_price = isset( $_GET['max_price'] ) ? ceil( floatval( wp_unslash( $_GET['max_price'] ) ) / $step ) * $step : $max_price; // WPCS: input var ok, CSRF ok.
|
2018-11-16 09:26:49 +00:00
|
|
|
|
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> — <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 );
|
2019-02-11 14:42:54 +00:00
|
|
|
$search = WC_Query::get_main_search_query_sql();
|
|
|
|
|
|
|
|
$meta_query_sql = $meta_query->get_sql( 'post', $wpdb->posts, 'ID' );
|
|
|
|
$tax_query_sql = $tax_query->get_sql( $wpdb->posts, 'ID' );
|
|
|
|
$search_query_sql = $search ? ' AND ' . $search : '';
|
|
|
|
|
|
|
|
$sql = "
|
|
|
|
SELECT min( min_price ) as min_price, MAX( max_price ) as max_price
|
2019-02-15 12:37:45 +00:00
|
|
|
FROM {$wpdb->wc_product_meta_lookup}
|
2019-02-11 14:42:54 +00:00
|
|
|
WHERE product_id IN (
|
|
|
|
SELECT ID FROM {$wpdb->posts}
|
|
|
|
" . $tax_query_sql['join'] . $meta_query_sql['join'] . "
|
|
|
|
WHERE {$wpdb->posts}.post_type IN ('" . implode( "','", array_map( 'esc_sql', apply_filters( 'woocommerce_price_filter_post_type', array( 'product' ) ) ) ) . "')
|
|
|
|
AND {$wpdb->posts}.post_status = 'publish'
|
|
|
|
" . $tax_query_sql['where'] . $meta_query_sql['where'] . $search_query_sql . '
|
|
|
|
)';
|
2016-12-23 14:45:58 +00:00
|
|
|
|
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
|
|
|
}
|