360 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			360 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php
 | 
						|
if ( ! defined( 'ABSPATH' ) ) {
 | 
						|
	exit; // Exit if accessed directly
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Plugin compatibility for WooCommerce
 | 
						|
 *
 | 
						|
 * @since 3.0.65 (builder version)
 | 
						|
 * @link https://wordpress.org/plugins/woocommerce/
 | 
						|
 */
 | 
						|
class ET_Builder_Plugin_Compat_WooCommerce extends ET_Builder_Plugin_Compat_Base {
 | 
						|
	/**
 | 
						|
	 * Constructor
 | 
						|
	 */
 | 
						|
	function __construct() {
 | 
						|
		$this->plugin_id = 'woocommerce/woocommerce.php';
 | 
						|
		$this->init_hooks();
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Hook methods to WordPress
 | 
						|
	 * Latest plugin version: 3.1.1
 | 
						|
	 *
 | 
						|
	 * @return void
 | 
						|
	 */
 | 
						|
	function init_hooks() {
 | 
						|
		// Bail if there's no version found or needed functions do not exist
 | 
						|
		if (
 | 
						|
			! $this->get_plugin_version() ||
 | 
						|
			! function_exists( 'is_cart' ) ||
 | 
						|
			! function_exists( 'is_account_page' )
 | 
						|
		) {
 | 
						|
			return;
 | 
						|
		}
 | 
						|
 | 
						|
		// Up to: latest theme version
 | 
						|
		add_filter( 'et_grab_image_setting', array( $this, 'disable_et_grab_image_setting' ), 1 );
 | 
						|
 | 
						|
		// Hook before calling comments_template function in module.
 | 
						|
		add_action( 'et_fb_before_comments_template', array( $this, 'remove_filter_comments_number_by_woo' ) );
 | 
						|
		add_action( 'et_builder_before_comments_number', array( $this, 'remove_filter_comments_number_by_woo' ) );
 | 
						|
 | 
						|
		// Hook afer calling comments_template function in module.
 | 
						|
		add_action( 'et_fb_after_comments_template', array( $this, 'restore_filter_comments_number_by_woo' ) );
 | 
						|
		add_action( 'et_builder_after_comments_number', array( $this, 'restore_filter_comments_number_by_woo' ) );
 | 
						|
 | 
						|
		// Prevent malformed html in demo store notice from breaking the VB.
 | 
						|
		add_filter( 'woocommerce_demo_store', 'et_core_fix_unclosed_html_tags' );
 | 
						|
 | 
						|
		// Dynamic Content
 | 
						|
		add_filter( 'et_builder_dynamic_content_display_hidden_meta_keys', array( $this, 'filter_dynamic_content_display_hidden_meta_keys' ), 10, 2 );
 | 
						|
		add_filter( 'et_builder_dynamic_content_custom_field_label', array( $this, 'filter_dynamic_content_custom_field_label' ), 10, 2 );
 | 
						|
		add_filter( 'et_builder_dynamic_content_meta_value', array( $this, 'maybe_filter_dynamic_content_meta_value' ), 10, 3 );
 | 
						|
 | 
						|
		if ( is_object( WC() ) && is_object( WC()->structured_data ) ) {
 | 
						|
			$enabled = array(
 | 
						|
				// phpcs:disable WordPress.Security.NonceVerification.NoNonceVerification
 | 
						|
				'vb'  => et_()->array_get( $_GET, 'et_fb' ),
 | 
						|
				'bfb' => et_()->array_get( $_GET, 'et_bfb' ),
 | 
						|
				// phpcs:enable
 | 
						|
			);
 | 
						|
 | 
						|
			if ( ( $enabled['vb'] || $enabled['bfb'] ) ) {
 | 
						|
				// Hook generates JSON-LD which is used by some search engines but it's not needed in VB/BFB
 | 
						|
				// and it also breaks inline generation of static definitions.
 | 
						|
				remove_action( 'woocommerce_single_product_summary', array( WC()->structured_data, 'generate_product_data' ), 60 );
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		// Theme Builder.
 | 
						|
		add_filter( 'et_theme_builder_template_settings_options', array( $this, 'maybe_filter_theme_builder_template_settings_options' ) );
 | 
						|
		add_action( 'et_theme_builder_after_layout_opening_wrappers', array( $this, 'maybe_trigger_woo_hooks_in_theme_builder_body' ) );
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * When an order is cancelled, WooCommerce cart shortcode changes the order status to prevent
 | 
						|
	 * the 'Your order was cancelled.' notice from being shown multiple times.
 | 
						|
	 * Since grab_image renders shortcodes twice, it must be disabled in the cart page or else the notice
 | 
						|
	 * will not be shown at all.
 | 
						|
	 * My Account Page and Checkout Page is also affected by the same issue.
 | 
						|
	 *
 | 
						|
	 * @return bool
 | 
						|
	 */
 | 
						|
	function disable_et_grab_image_setting( $settings ) {
 | 
						|
		return ( is_cart() || is_checkout() || is_account_page() ) ? false : $settings;
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Remove comments_number filter added by Woo that caused missing comment
 | 
						|
	 * count in Comment module
 | 
						|
	 *
 | 
						|
	 * @return void
 | 
						|
	 */
 | 
						|
	public function remove_filter_comments_number_by_woo() {
 | 
						|
		if ( ! current_theme_supports( 'woocommerce' ) || ( function_exists( 'wc_get_page_id' ) && wc_get_page_id( 'shop' ) < 0 ) ) {
 | 
						|
			remove_filter( 'comments_number', '__return_empty_string' );
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Restore comments_number that removed by remove_filter_comments_number_by_woo
 | 
						|
	 *
 | 
						|
	 * @return void
 | 
						|
	 */
 | 
						|
	public function restore_filter_comments_number_by_woo() {
 | 
						|
		if ( ! current_theme_supports( 'woocommerce' ) || ( function_exists( 'wc_get_page_id' ) && wc_get_page_id( 'shop' ) < 0 ) ) {
 | 
						|
			add_filter( 'comments_number', '__return_empty_string' );
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Allowlist hidden WooCommerce meta keys for dynamic content.
 | 
						|
	 *
 | 
						|
	 * @since 3.17.2
 | 
						|
	 *
 | 
						|
	 * @param string[] $meta_keys
 | 
						|
	 * @param integer  $post_id
 | 
						|
	 *
 | 
						|
	 * @return string[]
 | 
						|
	 */
 | 
						|
	public function filter_dynamic_content_display_hidden_meta_keys( $meta_keys, $post_id ) {
 | 
						|
		return array_merge(
 | 
						|
			$meta_keys,
 | 
						|
			array(
 | 
						|
				'_stock_status',
 | 
						|
				'_regular_price',
 | 
						|
				'_sale_price',
 | 
						|
			)
 | 
						|
		);
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Rename label of known displayed hidden post meta fields in dynamic content.
 | 
						|
	 *
 | 
						|
	 * @since 3.17.2
 | 
						|
	 *
 | 
						|
	 * @param string $label
 | 
						|
	 * @param string $key
 | 
						|
	 *
 | 
						|
	 * @return string
 | 
						|
	 */
 | 
						|
	public function filter_dynamic_content_custom_field_label( $label, $key ) {
 | 
						|
		$custom_labels = array(
 | 
						|
			'total_sales'    => esc_html__( 'Product Total Sales', 'et_builder' ),
 | 
						|
			'_stock_status'  => esc_html__( 'Product Stock Status', 'et_builder' ),
 | 
						|
			'_regular_price' => esc_html__( 'Product Regular Price', 'et_builder' ),
 | 
						|
			'_sale_price'    => esc_html__( 'Product Sale Price', 'et_builder' ),
 | 
						|
		);
 | 
						|
 | 
						|
		if ( isset( $custom_labels[ $key ] ) ) {
 | 
						|
			return $custom_labels[ $key ];
 | 
						|
		}
 | 
						|
 | 
						|
		return $label;
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Format WooCommerce meta values accordingly.
 | 
						|
	 *
 | 
						|
	 * @since 3.17.2
 | 
						|
	 *
 | 
						|
	 * @param string  $meta_value
 | 
						|
	 * @param string  $meta_key
 | 
						|
	 * @param integer $post_id
 | 
						|
	 *
 | 
						|
	 * @return string
 | 
						|
	 */
 | 
						|
	public function maybe_filter_dynamic_content_meta_value( $meta_value, $meta_key, $post_id ) {
 | 
						|
		switch ( $meta_key ) {
 | 
						|
			case '_stock_status':
 | 
						|
				// Check for function existance just in case
 | 
						|
				if ( function_exists( 'wc_get_product_stock_status_options' ) ) {
 | 
						|
					$stock_statuses = wc_get_product_stock_status_options();
 | 
						|
 | 
						|
					// Format meta value into human readable format
 | 
						|
					if ( ! empty( $stock_statuses[ $meta_value ] ) ) {
 | 
						|
						$meta_value = esc_html( $stock_statuses[ $meta_value ] );
 | 
						|
					}
 | 
						|
				}
 | 
						|
 | 
						|
				break;
 | 
						|
		}
 | 
						|
 | 
						|
		return $meta_value;
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Add Theme Builder template settings options.
 | 
						|
	 *
 | 
						|
	 * @since 4.0
 | 
						|
	 *
 | 
						|
	 * @param array $options
 | 
						|
	 *
 | 
						|
	 * @return array
 | 
						|
	 */
 | 
						|
	public function maybe_filter_theme_builder_template_settings_options( $options ) {
 | 
						|
		$woocommerce_options = array(
 | 
						|
			'woocommerce' => array(
 | 
						|
				'label'    => esc_html__( 'WooCommerce Pages', 'et_builder' ),
 | 
						|
				'settings' => array(
 | 
						|
					array(
 | 
						|
						'id'       => 'woocommerce:shop',
 | 
						|
						'label'    => esc_html__( 'Shop', 'et_builder' ),
 | 
						|
						'title'    => trim( str_replace( home_url(), '', get_post_type_archive_link( 'product' ) ), '/' ),
 | 
						|
						'priority' => 120,
 | 
						|
						'validate' => array( $this, 'theme_builder_validate_woocommerce_shop' ),
 | 
						|
					),
 | 
						|
					array(
 | 
						|
						'id'       => 'woocommerce:cart',
 | 
						|
						'label'    => esc_html__( 'Cart', 'et_builder' ),
 | 
						|
						'title'    => get_post_field( 'post_name', wc_get_page_id( 'cart' ) ),
 | 
						|
						'priority' => 120,
 | 
						|
						'validate' => array( $this, 'theme_builder_validate_woocommerce_cart' ),
 | 
						|
					),
 | 
						|
					array(
 | 
						|
						'id'       => 'woocommerce:checkout',
 | 
						|
						'label'    => esc_html__( 'Checkout', 'et_builder' ),
 | 
						|
						'title'    => get_post_field( 'post_name', wc_get_page_id( 'checkout' ) ),
 | 
						|
						'priority' => 120,
 | 
						|
						'validate' => array( $this, 'theme_builder_validate_woocommerce_checkout' ),
 | 
						|
					),
 | 
						|
					array(
 | 
						|
						'id'       => 'woocommerce:my_account',
 | 
						|
						'label'    => esc_html__( 'My Account', 'et_builder' ),
 | 
						|
						'title'    => get_post_field( 'post_name', wc_get_page_id( 'myaccount' ) ),
 | 
						|
						'priority' => 130,
 | 
						|
						'validate' => array( $this, 'theme_builder_validate_woocommerce_my_account' ),
 | 
						|
					),
 | 
						|
				),
 | 
						|
			),
 | 
						|
		);
 | 
						|
 | 
						|
		$archive_index = array_search( 'archive', array_keys( $options ) );
 | 
						|
 | 
						|
		if ( false === $archive_index ) {
 | 
						|
			return array_merge(
 | 
						|
				$options,
 | 
						|
				$woocommerce_options
 | 
						|
			);
 | 
						|
		}
 | 
						|
 | 
						|
		return array_merge(
 | 
						|
			array_slice( $options, 0, $archive_index + 1, true ),
 | 
						|
			$woocommerce_options,
 | 
						|
			array_slice( $options, $archive_index + 1, null, true )
 | 
						|
		);
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Theme Builder: Validate woocommerce:shop.
 | 
						|
	 *
 | 
						|
	 * @since 4.0
 | 
						|
	 *
 | 
						|
	 * @param string   $type
 | 
						|
	 * @param string   $subtype
 | 
						|
	 * @param integer  $id
 | 
						|
	 * @param string[] $setting
 | 
						|
	 *
 | 
						|
	 * @return bool
 | 
						|
	 */
 | 
						|
	public function theme_builder_validate_woocommerce_shop( $type, $subtype, $id, $setting ) {
 | 
						|
		return (
 | 
						|
			( ET_Theme_Builder_Request::TYPE_POST_TYPE_ARCHIVE === $type && $subtype === 'product' )
 | 
						|
			||
 | 
						|
			( ET_Theme_Builder_Request::TYPE_SINGULAR === $type && $id === wc_get_page_id( 'shop' ) )
 | 
						|
		);
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Theme Builder: Validate woocommerce:cart.
 | 
						|
	 *
 | 
						|
	 * @since 4.0
 | 
						|
	 *
 | 
						|
	 * @param string   $type
 | 
						|
	 * @param string   $subtype
 | 
						|
	 * @param integer  $id
 | 
						|
	 * @param string[] $setting
 | 
						|
	 *
 | 
						|
	 * @return bool
 | 
						|
	 */
 | 
						|
	public function theme_builder_validate_woocommerce_cart( $type, $subtype, $id, $setting ) {
 | 
						|
		return ET_Theme_Builder_Request::TYPE_SINGULAR === $type && $id === wc_get_page_id( 'cart' );
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Theme Builder: Validate woocommerce:checkout.
 | 
						|
	 *
 | 
						|
	 * @since 4.0
 | 
						|
	 *
 | 
						|
	 * @param string   $type
 | 
						|
	 * @param string   $subtype
 | 
						|
	 * @param integer  $id
 | 
						|
	 * @param string[] $setting
 | 
						|
	 *
 | 
						|
	 * @return bool
 | 
						|
	 */
 | 
						|
	public function theme_builder_validate_woocommerce_checkout( $type, $subtype, $id, $setting ) {
 | 
						|
		return ET_Theme_Builder_Request::TYPE_SINGULAR === $type && $id === wc_get_page_id( 'checkout' );
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Theme Builder: Validate woocommerce:my_account.
 | 
						|
	 *
 | 
						|
	 * @since 4.0
 | 
						|
	 *
 | 
						|
	 * @param string   $type
 | 
						|
	 * @param string   $subtype
 | 
						|
	 * @param integer  $id
 | 
						|
	 * @param string[] $setting
 | 
						|
	 *
 | 
						|
	 * @return bool
 | 
						|
	 */
 | 
						|
	public function theme_builder_validate_woocommerce_my_account( $type, $subtype, $id, $setting ) {
 | 
						|
		return ET_Theme_Builder_Request::TYPE_SINGULAR === $type && $id === wc_get_page_id( 'myaccount' );
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Trigger Woo hooks before a Theme Builder body layout is rendered
 | 
						|
	 * so stuff like structured data is output.
 | 
						|
	 *
 | 
						|
	 * @since 4.0.10
 | 
						|
	 *
 | 
						|
	 * @param string $layout_type
 | 
						|
	 */
 | 
						|
	public function maybe_trigger_woo_hooks_in_theme_builder_body( $layout_type ) {
 | 
						|
		global $product;
 | 
						|
 | 
						|
		if ( ET_THEME_BUILDER_BODY_LAYOUT_POST_TYPE !== $layout_type || ! is_singular( 'product' ) ) {
 | 
						|
			return;
 | 
						|
		}
 | 
						|
 | 
						|
		if ( $product && ! is_a( $product, 'WC_Product' ) ) {
 | 
						|
			// Required for Woo to setup its $product global.
 | 
						|
			the_post();
 | 
						|
		}
 | 
						|
 | 
						|
		// Make sure builder and non-builder products do not render
 | 
						|
		// anything as this will be taken care of by the
 | 
						|
		// Post Content module in TB, if used.
 | 
						|
		et_builder_wc_disable_default_layout();
 | 
						|
		remove_action(
 | 
						|
			'woocommerce_after_single_product_summary',
 | 
						|
			'et_builder_wc_product_render_layout',
 | 
						|
			5
 | 
						|
		);
 | 
						|
 | 
						|
		// Trigger the usual Woo hooks so functionality like structured data works.
 | 
						|
		do_action( 'woocommerce_before_single_product' );
 | 
						|
 | 
						|
		if ( ! post_password_required() ) {
 | 
						|
			do_action( 'woocommerce_before_single_product_summary' );
 | 
						|
			do_action( 'woocommerce_single_product_summary' );
 | 
						|
			do_action( 'woocommerce_after_single_product_summary' );
 | 
						|
			do_action( 'woocommerce_after_single_product' );
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
new ET_Builder_Plugin_Compat_WooCommerce();
 |