2012-08-10 09:35:25 +00:00
< ? php
2011-08-10 17:11:11 +00:00
/**
2011-09-04 00:02:44 +00:00
* Price Filter Widget and related functions
2012-08-10 09:35:25 +00:00
*
2011-08-10 17:11:11 +00:00
* Generates a range slider to filter products by price
*
* @ package WooCommerce
* @ category Widgets
* @ author WooThemes
*/
2011-12-09 21:47:12 +00:00
2011-09-04 00:02:44 +00:00
/**
* Price filter Init
*/
2012-06-25 21:53:15 +00:00
add_action ( 'init' , 'woocommerce_price_filter_init' );
2011-08-10 17:11:11 +00:00
function woocommerce_price_filter_init () {
2012-07-17 14:09:18 +00:00
global $woocommerce ;
2012-08-10 09:35:25 +00:00
2012-06-25 21:53:15 +00:00
if ( is_active_widget ( false , false , 'price_filter' , true ) && ! is_admin () ) {
2012-08-10 09:35:25 +00:00
2012-07-17 14:09:18 +00:00
$suffix = defined ( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG ? '' : '.min' ;
2012-08-10 09:35:25 +00:00
2012-07-17 14:09:18 +00:00
wp_register_script ( 'wc-price-slider' , $woocommerce -> plugin_url () . '/assets/js/frontend/price-slider' . $suffix . '.js' , array ( 'jquery-ui' ), '1.6' , true );
2012-08-10 09:35:25 +00:00
2012-06-25 21:53:15 +00:00
unset ( $_SESSION [ 'min_price' ] );
unset ( $_SESSION [ 'max_price' ] );
2012-08-10 09:35:25 +00:00
if ( isset ( $_GET [ 'min_price' ] ) )
2012-06-25 21:53:15 +00:00
$_SESSION [ 'min_price' ] = $_GET [ 'min_price' ];
if ( isset ( $_GET [ 'max_price' ] ) )
$_SESSION [ 'max_price' ] = $_GET [ 'max_price' ];
2012-08-10 09:35:25 +00:00
2012-06-25 21:53:15 +00:00
add_filter ( 'loop_shop_post_in' , 'woocommerce_price_filter' );
}
2011-09-04 00:02:44 +00:00
}
/**
* Price Filter post filter
*/
2012-03-21 11:55:34 +00:00
function woocommerce_price_filter ( $filtered_posts ) {
global $wpdb ;
2012-08-10 09:35:25 +00:00
2012-03-23 20:54:45 +00:00
if ( isset ( $_GET [ 'max_price' ] ) && isset ( $_GET [ 'min_price' ] ) ) {
2011-09-04 00:02:44 +00:00
2012-03-21 11:55:34 +00:00
$matched_products = array ();
2012-03-23 20:54:45 +00:00
$min = floatval ( $_GET [ 'min_price' ] );
$max = floatval ( $_GET [ 'max_price' ] );
2012-08-10 09:35:25 +00:00
2012-03-23 20:54:45 +00:00
$matched_products_query = $wpdb -> get_results ( $wpdb -> prepare ( "
SELECT DISTINCT ID , post_parent , post_type FROM $wpdb -> posts
INNER JOIN $wpdb -> postmeta ON ID = post_id
WHERE post_type IN ( 'product' , 'product_variation' ) AND post_status = 'publish' AND meta_key = % s AND meta_value BETWEEN % d AND % d
" , '_price', $min , $max ), OBJECT_K );
2011-09-04 00:02:44 +00:00
2012-03-23 20:54:45 +00:00
if ( $matched_products_query ) {
foreach ( $matched_products_query as $product ) {
if ( $product -> post_type == 'product' )
2012-03-21 11:55:34 +00:00
$matched_products [] = $product -> ID ;
2012-03-23 20:54:45 +00:00
if ( $product -> post_parent > 0 && ! in_array ( $product -> post_parent , $matched_products ) )
2012-03-21 11:55:34 +00:00
$matched_products [] = $product -> post_parent ;
2012-03-23 20:54:45 +00:00
}
}
2012-03-21 11:55:34 +00:00
// Filter the id's
2012-03-23 20:54:45 +00:00
if ( sizeof ( $filtered_posts ) == 0 ) {
2012-03-21 11:55:34 +00:00
$filtered_posts = $matched_products ;
$filtered_posts [] = 0 ;
2012-03-23 20:54:45 +00:00
} else {
$filtered_posts = array_intersect ( $filtered_posts , $matched_products );
2012-03-21 11:55:34 +00:00
$filtered_posts [] = 0 ;
2012-03-23 20:54:45 +00:00
}
2012-03-21 11:55:34 +00:00
2012-03-23 20:54:45 +00:00
}
2012-03-21 11:55:34 +00:00
return ( array ) $filtered_posts ;
2011-08-10 17:11:11 +00:00
}
2011-09-04 00:02:44 +00:00
/**
* Price Filter post Widget
*/
2011-08-10 17:11:11 +00:00
class WooCommerce_Widget_Price_Filter extends WP_Widget {
2011-08-26 21:28:55 +00:00
/** Variables to setup the widget. */
var $woo_widget_cssclass ;
var $woo_widget_description ;
var $woo_widget_idbase ;
var $woo_widget_name ;
2012-08-10 09:35:25 +00:00
2011-08-10 17:11:11 +00:00
/** constructor */
function WooCommerce_Widget_Price_Filter () {
2012-08-10 09:35:25 +00:00
2011-08-26 21:28:55 +00:00
/* Widget variable settings. */
$this -> woo_widget_cssclass = 'widget_price_filter' ;
2012-01-05 11:31:22 +00:00
$this -> woo_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' );
2011-08-26 21:28:55 +00:00
$this -> woo_widget_idbase = 'woocommerce_price_filter' ;
2012-01-05 11:31:22 +00:00
$this -> woo_widget_name = __ ( 'WooCommerce Price Filter' , 'woocommerce' );
2012-08-10 09:35:25 +00:00
2011-08-26 21:28:55 +00:00
/* Widget settings. */
$widget_ops = array ( 'classname' => $this -> woo_widget_cssclass , 'description' => $this -> woo_widget_description );
2012-08-10 09:35:25 +00:00
2011-08-26 21:28:55 +00:00
/* Create the widget. */
$this -> WP_Widget ( 'price_filter' , $this -> woo_widget_name , $widget_ops );
2011-08-10 17:11:11 +00:00
}
2011-09-06 11:11:22 +00:00
/** @see WP_Widget */
2011-08-10 17:11:11 +00:00
function widget ( $args , $instance ) {
extract ( $args );
2012-08-10 09:35:25 +00:00
2012-06-29 20:18:22 +00:00
global $_chosen_attributes , $wpdb , $woocommerce , $wp_query , $wp ;
2012-08-10 09:35:25 +00:00
2012-03-26 14:34:23 +00:00
if ( ! is_tax ( 'product_cat' ) && ! is_post_type_archive ( 'product' ) && ! is_tax ( 'product_tag' )) return ; // Not on product page - return
2012-08-10 09:35:25 +00:00
2012-03-26 14:34:23 +00:00
if ( sizeof ( $woocommerce -> query -> unfiltered_product_ids ) == 0 ) return ; // None shown - return
2012-08-10 09:35:25 +00:00
wp_enqueue_script ( 'wc-price-slider' );
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 ( $_SESSION [ 'min_price' ] ) ? $_SESSION [ 'min_price' ] : '' ,
'max_price' => isset ( $_SESSION [ 'max_price' ] ) ? $_SESSION [ 'max_price' ] : ''
) );
2011-11-15 15:15:12 +00:00
2011-08-10 17:11:11 +00:00
$title = $instance [ 'title' ];
$title = apply_filters ( 'widget_title' , $title , $instance , $this -> id_base );
2012-08-10 09:35:25 +00:00
2011-08-10 17:11:11 +00:00
// Remember current filters/search
$fields = '' ;
2012-08-10 09:35:25 +00:00
2011-08-10 17:11:11 +00:00
if ( get_search_query ()) $fields = '<input type="hidden" name="s" value="' . get_search_query () . '" />' ;
2011-09-30 19:14:12 +00:00
if ( isset ( $_GET [ 'post_type' ])) $fields .= '<input type="hidden" name="post_type" value="' . esc_attr ( $_GET [ 'post_type' ] ) . '" />' ;
2012-01-13 11:28:23 +00:00
if ( isset ( $_GET [ 'product_cat' ])) $fields .= '<input type="hidden" name="product_cat" value="' . esc_attr ( $_GET [ 'product_cat' ] ) . '" />' ;
if ( isset ( $_GET [ 'product_tag' ])) $fields .= '<input type="hidden" name="product_tag" value="' . esc_attr ( $_GET [ 'product_tag' ] ) . '" />' ;
2012-08-10 09:35:25 +00:00
2011-11-15 15:15:12 +00:00
if ( $_chosen_attributes ) foreach ( $_chosen_attributes as $attribute => $data ) :
2012-08-10 09:35:25 +00:00
2011-11-15 15:15:12 +00:00
$fields .= '<input type="hidden" name="' . esc_attr ( str_replace ( 'pa_' , 'filter_' , $attribute ) ) . '" value="' . esc_attr ( implode ( ',' , $data [ 'terms' ]) ) . '" />' ;
if ( $data [ 'query_type' ] == 'or' ) $fields .= '<input type="hidden" name="' . esc_attr ( str_replace ( 'pa_' , 'query_type_' , $attribute ) ) . '" value="or" />' ;
2012-08-10 09:35:25 +00:00
2011-08-10 17:11:11 +00:00
endforeach ;
2012-08-10 09:35:25 +00:00
2011-12-02 11:43:15 +00:00
$min = $max = 0 ;
$post_min = $post_max = '' ;
2012-08-10 09:35:25 +00:00
2012-03-26 14:34:23 +00:00
if ( sizeof ( $woocommerce -> query -> layered_nav_product_ids ) == 0 ) :
2011-11-14 01:20:07 +00:00
2012-08-10 09:35:25 +00:00
$max = ceil ( $wpdb -> get_var ( " SELECT max(meta_value + 0)
2011-11-15 15:15:12 +00:00
FROM $wpdb -> posts
LEFT JOIN $wpdb -> postmeta ON $wpdb -> posts . ID = $wpdb -> postmeta . post_id
2011-12-24 17:05:51 +00:00
WHERE meta_key = '_price' " ));
2011-11-15 15:15:12 +00:00
else :
2012-08-10 09:35:25 +00:00
$max = ceil ( $wpdb -> get_var ( " SELECT max(meta_value + 0)
2011-11-15 15:15:12 +00:00
FROM $wpdb -> posts
LEFT JOIN $wpdb -> postmeta ON $wpdb -> posts . ID = $wpdb -> postmeta . post_id
2011-12-24 17:05:51 +00:00
WHERE meta_key = '_price' AND (
2012-08-10 09:35:25 +00:00
$wpdb -> posts . ID IN ( " .implode(',', $woocommerce->query ->layered_nav_product_ids). " )
2011-11-15 15:15:12 +00:00
OR (
$wpdb -> posts . post_parent IN ( " .implode(',', $woocommerce->query ->layered_nav_product_ids). " )
AND $wpdb -> posts . post_parent != 0
)
) " ));
2012-08-10 09:35:25 +00:00
2011-11-15 15:15:12 +00:00
endif ;
2012-08-10 09:35:25 +00:00
2012-03-26 09:34:04 +00:00
if ( $min == $max ) return ;
2012-08-10 09:35:25 +00:00
2011-12-02 11:43:15 +00:00
if ( isset ( $_SESSION [ 'min_price' ])) $post_min = $_SESSION [ 'min_price' ];
if ( isset ( $_SESSION [ 'max_price' ])) $post_max = $_SESSION [ 'max_price' ];
2012-03-29 14:50:20 +00:00
echo $before_widget . $before_title . $title . $after_title ;
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 ) ) );
else
$form_action = preg_replace ( '%\/page/[0-9]+%' , '' , home_url ( $wp -> request ) );
2012-08-10 09:35:25 +00:00
2012-06-29 20:18:22 +00:00
echo '<form method="get" action="' . $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 >
2011-08-10 17:11:11 +00:00
< div class = " price_slider_amount " >
2012-01-05 11:31:22 +00:00
< input type = " text " id = " min_price " name = " min_price " value = " '.esc_attr( $post_min ).' " data - min = " '.esc_attr( $min ).' " placeholder = " '.__('Min price', 'woocommerce').' " />
< input type = " text " id = " max_price " name = " max_price " value = " '.esc_attr( $post_max ).' " data - max = " '.esc_attr( $max ).' " placeholder = " '.__('Max price', 'woocommerce').' " />
< button type = " submit " class = " button " > '.__(' Filter ', ' woocommerce ').' </ button >
2011-12-02 11:43:15 +00:00
< div class = " price_label " style = " display:none; " >
2012-02-16 17:16:27 +00:00
'.__(' Price : ', ' woocommerce ').' < span class = " from " ></ span > & mdash ; < span class = " to " ></ span >
2011-12-02 11:43:15 +00:00
</ div >
2011-08-10 17:11:11 +00:00
'.$fields.'
2011-12-02 11:43:15 +00:00
< div class = " clear " ></ div >
2011-08-10 17:11:11 +00:00
</ div >
</ div >
</ form > ' ;
2012-08-10 09:35:25 +00:00
2011-08-10 17:11:11 +00:00
echo $after_widget ;
}
2011-09-06 11:11:22 +00:00
/** @see WP_Widget->update */
2011-08-10 17:11:11 +00:00
function update ( $new_instance , $old_instance ) {
2012-01-05 11:31:22 +00:00
if ( ! isset ( $new_instance [ 'title' ]) || empty ( $new_instance [ 'title' ])) $new_instance [ 'title' ] = __ ( 'Filter by price' , 'woocommerce' );
2011-08-10 17:11:11 +00:00
$instance [ 'title' ] = strip_tags ( stripslashes ( $new_instance [ 'title' ]));
return $instance ;
}
2011-09-06 11:11:22 +00:00
/** @see WP_Widget->form */
2011-08-10 17:11:11 +00:00
function form ( $instance ) {
global $wpdb ;
?>
2012-01-05 11:31:22 +00:00
< p >< label for = " <?php echo $this->get_field_id ('title'); ?> " >< ? php _e ( 'Title:' , 'woocommerce' ) ?> </label>
2011-09-19 06:01:26 +00:00
< input type = " text " class = " widefat " id = " <?php echo esc_attr( $this->get_field_id ('title') ); ?> " name = " <?php echo esc_attr( $this->get_field_name ('title') ); ?> " value = " <?php if (isset ( $instance['title'] )) { echo esc_attr( $instance['title'] );} ?> " /></ p >
2011-08-10 17:11:11 +00:00
< ? php
}
} // class WooCommerce_Widget_Price_Filter