attributes = $this->parse_attributes( $attributes ); $this->content = $content; $this->query_args = $this->parse_query_args(); } /** * Get the block's attributes. * * @param array $attributes Block attributes. Default empty array. * @return array Block attributes merged with defaults. */ protected function parse_attributes( $attributes ) { // These should match what's set in JS `registerBlockType`. $defaults = array( 'columns' => wc_get_theme_support( 'product_blocks::default_columns', 3 ), 'rows' => wc_get_theme_support( 'product_blocks::default_rows', 1 ), 'categories' => array(), 'catOperator' => 'any', 'contentVisibility' => array( 'title' => true, 'price' => true, 'rating' => true, 'button' => true, ), ); return wp_parse_args( $attributes, $defaults ); } /** * Parse query args. * * @return array */ protected function parse_query_args() { $query_args = array( 'post_type' => 'product', 'post_status' => 'publish', 'ignore_sticky_posts' => true, 'no_found_rows' => false, 'orderby' => '', 'order' => '', ); if ( isset( $this->attributes['orderby'] ) ) { if ( 'price_desc' === $this->attributes['orderby'] ) { $query_args['orderby'] = 'price'; $query_args['order'] = 'DESC'; } elseif ( 'price_asc' === $this->attributes['orderby'] ) { $query_args['orderby'] = 'price'; $query_args['order'] = 'ASC'; } elseif ( 'date' === $this->attributes['orderby'] ) { $query_args['orderby'] = 'date'; $query_args['order'] = 'DESC'; } else { $query_args['orderby'] = $this->attributes['orderby']; } } if ( ! empty( $this->attributes['rows'] ) ) { $this->attributes['limit'] = intval( $this->attributes['columns'] ) * intval( $this->attributes['rows'] ); } $query_args['posts_per_page'] = intval( $this->attributes['limit'] ); $query_args['meta_query'] = WC()->query->get_meta_query(); // phpcs:ignore WordPress.DB.SlowDBQuery $query_args['tax_query'] = array(); // phpcs:ignore WordPress.DB.SlowDBQuery $this->set_block_query_args( $query_args ); $ordering_args = WC()->query->get_catalog_ordering_args( $query_args['orderby'], $query_args['order'] ); $query_args['orderby'] = $ordering_args['orderby']; $query_args['order'] = $ordering_args['order']; if ( $ordering_args['meta_key'] ) { $query_args['meta_key'] = $ordering_args['meta_key']; // phpcs:ignore WordPress.DB.SlowDBQuery } // Categories. $this->set_categories_query_args( $query_args ); // Always query only IDs. $query_args['fields'] = 'ids'; return $query_args; } /** * Set args specific to this block * * @param array $query_args Query args. */ abstract protected function set_block_query_args( &$query_args ); /** * Set categories query args. * * @param array $query_args Query args. */ protected function set_categories_query_args( &$query_args ) { if ( ! empty( $this->attributes['categories'] ) ) { $categories = array_map( 'absint', $this->attributes['categories'] ); $query_args['tax_query'][] = array( 'taxonomy' => 'product_cat', 'terms' => $categories, 'field' => 'term_id', 'operator' => 'all' === $this->attributes['catOperator'] ? 'AND' : 'IN', /* * When cat_operator is AND, the children categories should be excluded, * as only products belonging to all the children categories would be selected. */ 'include_children' => 'all' === $this->attributes['catOperator'] ? false : true, ); } } /** * Run the query and return an array of product IDs * * @return array List of product IDs */ protected function get_products() { if ( 'product-top-rated' === $this->block_name ) { add_filter( 'posts_clauses', array( WC()->query, 'order_by_rating_post_clauses' ) ); $query = new WP_Query( $this->query_args ); remove_filter( 'posts_clauses', array( WC()->query, 'order_by_rating_post_clauses' ) ); } else { $query = new WP_Query( $this->query_args ); } $results = wp_parse_id_list( $query->posts ); // Prime caches to reduce future queries. if ( is_callable( '_prime_post_caches' ) ) { _prime_post_caches( $results ); } // Remove ordering query arguments which may have been added by get_catalog_ordering_args. WC()->query->remove_ordering_args(); return $results; } /** * Render the Products block. * * @return string Rendered block type output. */ public function render() { $products = $this->get_products(); $classes = $this->get_container_classes(); $output = implode( '', array_map( array( $this, 'render_product' ), $products ) ); return sprintf( '
', esc_attr( $classes ), $output ); } /** * Get the list of classes to apply to this block. * * @return string space-separated list of classes. */ protected function get_container_classes() { $classes = array( 'wc-block-grid', "wp-block-{$this->block_name}", "wc-block-{$this->block_name}", "has-{$this->attributes['columns']}-columns", 'woocommerce', ); if ( $this->attributes['rows'] > 1 ) { $classes[] = 'has-multiple-rows'; } if ( isset( $this->attributes['align'] ) ) { $classes[] = "align{$this->attributes['align']}"; } return implode( ' ', $classes ); } /** * Render a single products. * * @param int $id Product ID. * @return string Rendered product output. */ public function render_product( $id ) { $product = wc_get_product( $id ); if ( ! $product ) { return ''; } $data = (object) array( 'permalink' => $product->get_permalink(), 'image' => $this->get_image_html( $product ), 'title' => $this->get_title_html( $product ), 'rating' => $this->get_rating_html( $product ), 'price' => $this->get_price_html( $product ), 'button' => $this->get_button_html( $product ), ); return "
  • permalink}\" class=\"wc-block-grid__product-link\"> {$data->image} {$data->title} {$data->rating} {$data->price} {$data->button}
  • "; } /** * Get the product image. * * @param WC_Product $product Product. * @return string */ protected function get_image_html( $product ) { return '
    ' . $product->get_image( 'woocommerce_thumbnail' ) . '
    '; } /** * Get the product title. * * @param WC_Product $product Product. * @return string */ protected function get_title_html( $product ) { if ( empty( $this->attributes['contentVisibility']['title'] ) ) { return ''; } return '
    ' . $product->get_title() . '
    '; } /** * Render the rating icons. * * @param WC_Product $product Product. * @return string Rendered product output. */ protected function get_rating_html( $product ) { if ( empty( $this->attributes['contentVisibility']['rating'] ) ) { return ''; } $rating_count = $product->get_rating_count(); $review_count = $product->get_review_count(); $average = $product->get_average_rating(); if ( $rating_count > 0 ) { return sprintf( '
    %s
    ', wc_get_rating_html( $average, $rating_count ) ); } return ''; } /** * Get the price. * * @param WC_Product $product Product. * @return string Rendered product output. */ protected function get_price_html( $product ) { if ( empty( $this->attributes['contentVisibility']['price'] ) ) { return ''; } $badge = ''; if ( $product->is_on_sale() ) { $badge = '' . esc_html__( 'Sale!', 'woo-gutenberg-products-block' ) . ''; } return sprintf( '
    %s%s
    ', $product->get_price_html(), $badge ); } /** * Get the button. * * @param WC_Product $product Product. * @return string Rendered product output. */ protected function get_button_html( $product ) { if ( empty( $this->attributes['contentVisibility']['button'] ) ) { return ''; } return '
    ' . $this->get_add_to_cart( $product ) . '
    '; } /** * Get the "add to cart" button. * * @param WC_Product $product Product. * @return string Rendered product output. */ protected function get_add_to_cart( $product ) { if ( $product->supports( 'ajax_add_to_cart' ) ) { return sprintf( '%5$s', esc_url( $product->add_to_cart_url() ), esc_attr( $product->get_id() ), esc_attr( $product->get_sku() ), esc_attr( $product->add_to_cart_description() ), esc_html( $product->add_to_cart_text() ) ); } return sprintf( '%3$s', esc_url( $product->add_to_cart_url() ), esc_attr( $product->add_to_cart_description() ), esc_html( $product->add_to_cart_text() ) ); } }