You can use a shortcode or block in order to display the WooCommerce products on sale. However, what if you wanted a proper “product category” called “Sale” – and where you didn’t need to manually assign this category to each product?
Basically, how do we display all the discounted products in a custom category called “Sale”, without doing any manual work?
Here’s a super quick tutorial. Enjoy!
Step 1: Add New Product Category
A one-off manual step is required, actually.
Simply go to WP Dashboard > Products > Categories > Add new category and enter the category name e.g. “Sale” and its slug e.g. “sale”.
The slug is very important as it’s used in the snippet below, so if you decide to rename it, you must change the PHP accordingly.
Step 2: PHP Snippet To Programmatically Populate the “Sale” Product Category With Products On Sale
/**
* @snippet Automatically Populate Sale Product Cat
* @how-to Get CustomizeWoo.com FREE
* @author Rodolfo Melogli
* @compatible WooCommerce 6
* @community https://businessbloomer.com/club/
*/
add_action( 'woocommerce_product_query', 'bbloomer_sale_category' );
function bbloomer_sale_category( $q ) {
if ( "sale" !== $q->get( 'product_cat' ) ) return; // "sale" = slug
$q->set( 'post_type', 'product' );
$q->set( 'product_cat', null );
$product_ids_on_sale = wc_get_product_ids_on_sale() ? wc_get_product_ids_on_sale() : array();
$q->set( 'post__in', $product_ids_on_sale );
}
Hello, works very well, but it doesnt work with the filter. Filter works on every page but not on the sale page I get the following error:
arning: strstr() expects parameter 1 to be string, object given in /home/customer/www/ mydomain.com/public_html/wp-includes/functions.php on line 1147
Warning: stripos() expects parameter 1 to be string, object given in /home/customer/www/ mydomain.com/public_html/wp-includes/functions.php on line 1154
Warning: strpos() expects parameter 1 to be string, object given in /home/customer/www/ mydomain.com/public_html/wp-includes/compat.php on line 473
Warning: explode() expects parameter 2 to be string, object given in /home/customer/www/ mydomain.com/public_html/wp-includes/functions.php on line 1165
When I remove the code it works fine. How can I fix this?
This strictly depends on the filter you’re using, so the function may need some tweaking.
The snippet doesn’t work with php 8.2
Why? What error do you get?
Hello,
Thanks for your code, it works for me.
But it does not put the products in the category: sale.
There is no checkmark in the sales category checkbox on the product page, and if I go to the category sale itself in the back end. The product isn’t in there either.
Can you make it, so the products are placed in and out of the sale category?
Greetings
Just curious – why do you need them there physically?
Yes, as Kevin suggested, this line of code
gives the mentioned error. I also tried your fix to change to:
But it does now tork.
This code snippet does not work with PHP 8.2. It does work on PHP 7.4. Do you have a revision that works with newer versions of PHP?
Hello Brooke, the snippet doesn’t really feature any PHP function apart from the ternary operator, which is properly written. Can you test again please, and if you’re still having problems maybe share the error log so I can have more context about the error please?
I have the same problem as Brooke. On my stage env with 7.4 PHP the snippet works fine. When I change the PHP ver. to 8.x my site gonna be broken.
And what error do you get?
Hmm.. only what I got is this log:
Thank you. That doesn’t seem related to my snippet though. If you remove my snippet does this go away?
yes! When I switched off the snippet it works fine.
PHP version? I’m on 7.4.33
8.2 ๐
And if you temporarily change this line:
to:
or, alternatively, temporarily downgrade to PHP 7.4. does it work?
Can this snippet be edited to apply to different “sale” categories based on the discount percentage the product has?
Say for example:
If the product is 10% off – category 1
If the product is 20% off – category 2
If the product is 30% off – category 3
Hello Evan, thanks so much for your comment! Yes, this is definitely possible – if you’d like to get a quote, feel free to contact me here. Thanks a lot for your understanding!
Hi, how can I show only 1 category on the product archive?
When I put everyone in “sale”, for example it appears like “women, sale”
I want to hide Sale or only show the first category,
Thankss!!!!
Screenshot please?
Hi Rodolfo,
Thanks for the snippet. This works nice!
Just for your info: It is not compatible with Permalink Manager Pro unfortunately. When Permalink Manager Pro is active, the Sale category page shows nothing.
Thank you!
I would love this functionality! However, I can’t get the snippet to work in Kadence Theme and WordPress 6.0.2.
Well, apparently it does populate the Sale Category. I did not see it working immediately because the sale category itself did not appear on Shop Page together with all the available categories. Is there a way for the Sale Category to appear with the rest of the Categories at the shop page?
Great! The Sale Category is a standard WooCommerce product category, so it should show
Thanks for all your hard work, your site has helped me plenty with a starting point. But for this I seen that it works only on the frontend and I needed it to also show up on the backend too. After more research and testing I did finally got a code that works for adding it to the backend and frontend. Hopes this helps someone else.
Thank you Chanel!
syntax error, unexpected ‘&’
here: if ( $product->is_on_sale() ) {
how can i fix that?
Try now
Dont work for variable products ๐
Hello Ivo, please expand on that? The wc_get_product_ids_on_sale function should indeed return variable products IDs as well. Is your DB table “wc_product_meta_lookup” up to date?
And will it work with variable products too?
And will it remove the products automatically when they are not on sale anymore?
And I do not want products from a category to be included, how do I implement this in your code?
Thanks for your feedback!
1) it should
2) it will
3) I’m afraid it’s custom work. If you’d like to get a quote, feel free to contact me here. Thanks a lot for your understanding!
Thanks for your reply.
Your code works but the products do not appear in the category SALE. When I go to Woocommerce -> products -> filter on category: Sale… that product is not there…
Backend or frontend? Because in the backend it won’t show any products, this simply populates the frontend category “Sale”
Hi ๐
It is worth adding a condition if there are no products on sale.
Thx ๐
Cheers