context['singleProduct'] ) && $block_instance->context['singleProduct'] ) { wp_reset_postdata(); } return $block_content; } /** * Update the context by injecting the correct post data * for each one of the Single Product inner blocks. * * @param array $context Block context. * @param array $block Block attributes. * @param WP_Block $parent_block Block instance. * * @return array Updated block context. */ public function update_context( $context, $block, $parent_block ) { if ( 'woocommerce/single-product' === $block['blockName'] && isset( $block['attrs']['productId'] ) ) { $this->product_id = $block['attrs']['productId']; $this->single_product_inner_blocks_names = array_reverse( $this->extract_single_product_inner_block_names( $block ) ); } $this->replace_post_for_single_product_inner_block( $block, $context ); return $context; } /** * Extract the inner block names for the Single Product block. This way it's possible * to map all the inner blocks for a Single Product block and manipulate the data as needed. * * @param array $block The Single Product block or its inner blocks. * @param array $result Array of inner block names. * * @return array Array containing all the inner block names of a Single Product block. */ protected function extract_single_product_inner_block_names( $block, &$result = [] ) { if ( isset( $block['blockName'] ) ) { $result[] = $block['blockName']; } if ( 'woocommerce/product-template' === $block['blockName'] || 'core/post-template' === $block['blockName'] ) { return $result; } if ( isset( $block['innerBlocks'] ) ) { foreach ( $block['innerBlocks'] as $inner_block ) { $this->extract_single_product_inner_block_names( $inner_block, $result ); } } return $result; } /** * Replace the global post for the Single Product inner blocks and reset it after. * * This is needed because some of the inner blocks may use the global post * instead of fetching the product through the `productId` attribute, so even if the * `productId` is passed to the inner block, it will still use the global post. * * @param array $block Block attributes. * @param array $context Block context. */ protected function replace_post_for_single_product_inner_block( $block, &$context ) { if ( $this->single_product_inner_blocks_names ) { $block_name = end( $this->single_product_inner_blocks_names ); if ( $block_name === $block['blockName'] ) { array_pop( $this->single_product_inner_blocks_names ); /** * This is a temporary fix to ensure the Post Title and Excerpt blocks work as expected * until Gutenberg versions 15.2 and 15.6 are included in the core of WordPress. * * Important: the original post data is restored in the restore_global_post method. * * @see https://github.com/WordPress/gutenberg/pull/48001 * @see https://github.com/WordPress/gutenberg/pull/49495 */ if ( 'core/post-excerpt' === $block_name || 'core/post-title' === $block_name ) { global $post; // phpcs:ignore WordPress.WP.GlobalVariablesOverride.Prohibited $post = get_post( $this->product_id ); if ( $post instanceof \WP_Post ) { setup_postdata( $post ); } } $context['postId'] = $this->product_id; $context['singleProduct'] = true; } } } /** * Render the Single Product block. * * @param array $attributes Block attributes. * @param string $content Block content. * @param WP_Block $block Block instance. * * @return string Rendered block type output. */ protected function render( $attributes, $content, $block ) { $product = wc_get_product( $block->context['postId'] ); if ( ! $product instanceof \WC_Product ) { return ''; } $interactivity_context = array( 'productId' => $product->get_id(), 'variationId' => null, ); $html = new \WP_HTML_Tag_Processor( $content ); if ( $html->next_tag( array( 'tag_name' => 'div' ) ) ) { $html->set_attribute( 'data-wp-interactive', $this->get_full_block_name() ); $html->set_attribute( 'data-wp-context', wp_json_encode( $interactivity_context, JSON_HEX_TAG | JSON_HEX_APOS | JSON_HEX_QUOT | JSON_HEX_AMP ) ); } $updated_html = $html->get_updated_html(); return parent::render( $attributes, $updated_html, $block ); } /** * Get the frontend script handle for this block type. * * @param string $key Data to get, or default to everything. * * @return null This block has no frontend script. */ protected function get_block_type_script( $key = null ) { return null; } }