HEX
Server: Apache/2
System: Linux vpslll9m.sdns.vn 4.18.0-553.22.1.el8_10.x86_64 #1 SMP Tue Sep 24 05:16:59 EDT 2024 x86_64
User: thuexe247c (1044)
PHP: 7.4.33
Disabled: exec,system,passthru,shell_exec,proc_close,proc_open,dl,popen,show_source,posix_kill,posix_mkfifo,posix_getpwuid,posix_setpgid,posix_setsid,posix_setuid,posix_setgid,posix_seteuid,posix_setegid,posix_uname
Upload Files
File: /home/thuexe247c/public_html/wp-content/plugins/autodescription/inc/compat/plugin-woocommerce.php
<?php
/**
 * @package The_SEO_Framework\Compat\Plugin\WooCommerce
 * @subpackage The_SEO_Framework\Compatibility
 */

namespace The_SEO_Framework;

\defined( 'THE_SEO_FRAMEWORK_PRESENT' ) or die;

use function \The_SEO_Framework\get_query_type_from_args;

use \The_SEO_Framework\{
	Data,
	Data\Filter\Sanitize,
	Helper\Query,
};

\add_action( 'woocommerce_init', __NAMESPACE__ . '\\_init_wc_compat' );
\add_filter( 'the_seo_framework_real_id', __NAMESPACE__ . '\\_set_real_id_wc_shop' );
\add_filter( 'the_seo_framework_is_singular_archive', __NAMESPACE__ . '\\_set_shop_singular_archive', 10, 2 );
\add_filter( 'the_seo_framework_is_shop', __NAMESPACE__ . '\\_set_wc_is_shop', 10, 2 );
\add_filter( 'the_seo_framework_is_product', __NAMESPACE__ . '\\_set_wc_is_product', 10, 2 );
\add_filter( 'the_seo_framework_is_product_admin', __NAMESPACE__ . '\\_set_wc_is_product_admin' );
\add_filter( 'the_seo_framework_robots_meta_array', __NAMESPACE__ . '\\_set_wc_noindex_defaults', 10, 3 );
\add_action( 'the_seo_framework_seo_bar', __NAMESPACE__ . '\\_assert_wc_noindex_defaults_seo_bar', 10, 2 );
\add_filter( 'the_seo_framework_image_generation_params', __NAMESPACE__ . '\\_adjust_wc_image_generation_params', 10, 2 );
\add_filter( 'the_seo_framework_public_post_type_archives', __NAMESPACE__ . '\\_filter_public_wc_post_type_archives' );
\add_filter( 'the_seo_framework_generated_archive_title_items', __NAMESPACE__ . '\\_filter_wc_shop_pta_title_items', 10, 2 );

/**
 * Initializes (late) WooCommerce compatibility.
 *
 * @hook woocommerce_init 10
 * @since 3.1.0
 * @since 4.0.3 Added primary term support to products.
 * @since 4.1.1 Added primary term support to category widgets.
 * @since 4.1.4 1. Now unbinds wc_page_noindex action.
 *              2. Now unbinds wc_page_no_robots filter.
 *              3. Now modifies the SEO Bar.
 * @access private
 */
function _init_wc_compat() {
	// Adjust the product link acknowledging the primary category.
	\add_filter( 'wc_product_post_type_link_product_cat', [ Query\Filter::class, 'filter_post_link_category' ], 10, 3 );

	// Adjust the structured-data breadcrumb primary term. Coincidentally(?), it uses the same filter structure; although, it misses the $post object.
	\add_filter( 'woocommerce_breadcrumb_main_term', [ Query\Filter::class, 'filter_post_link_category' ], 10, 2 );

	// Adjust the widget's tree primary term. Coincidentally(?), it uses the same filter structure; although, it misses the $post object.
	\add_filter( 'woocommerce_product_categories_widget_main_term', [ Query\Filter::class, 'filter_post_link_category' ], 10, 2 );

	\remove_filter( 'wp_robots', 'wc_page_no_robots' );
}

/**
 * Helper function for other methods to determine if there's a shop ID.
 *
 * @since 5.0.0
 * @access private
 *
 * @return int The shop page ID. 0 on failure.
 */
function _get_shop_page_id() {

	static $id;

	if ( isset( $id ) ) return $id;

	$id = \function_exists( 'wc_get_page_id' ) ? (int) \wc_get_page_id( 'shop' ) : 0;

	if ( \get_post( $id ) )
		return $id;

	return $id = 0;
}

/**
 * Helper function for other methods to determine if we're dealing with a shop.
 *
 * @since 4.2.0
 * @access private
 *
 * @param int|WP_Post|null $post Post ID or post object.
 * @return bool
 */
function _is_shop( $post = null ) {

	if ( isset( $post ) ) {
		$id = \is_int( $post )
			? $post
			: ( \get_post( $post )->ID ?? 0 );

		// phpcs:ignore, TSF.Performance.Opcodes -- local funcs
		$is_shop = $id && _get_shop_page_id() === $id;
	} else {
		// phpcs:ignore, TSF.Performance.Opcodes -- local funcs
		$is_shop = ! \is_admin() && \function_exists( 'is_shop' ) && \is_shop();
	}

	return $is_shop;
}

/**
 * Sets the correct shop ID on the shop page, but only if the shop is registered as page.
 *
 * @hook the_seo_framework_real_id 10
 * @since 4.0.5
 * @access private
 *
 * @param int $id The current ID.
 * @return int
 */
function _set_real_id_wc_shop( $id ) {
	// phpcs:ignore, TSF.Performance.Opcodes -- local funcs.
	return _is_shop() ? _get_shop_page_id() : $id;
}

/**
 * Sets singular archives for the WC shop page, but only if the shop is registered as page.
 *
 * @hook the_seo_framework_is_singular_archive 10
 * @since 4.0.5
 * @access private
 *
 * @param bool     $is_singular_archive Whether the post ID is a singular archive.
 * @param int|null $id                  The supplied post ID. Null when in the loop.
 * @return bool
 */
function _set_shop_singular_archive( $is_singular_archive, $id ) {
	// phpcs:ignore, TSF.Performance.Opcodes -- local func
	return $is_singular_archive || ( _get_shop_page_id() && _is_shop( $id ) );
}

/**
 * Sets the is_shop query.
 *
 * @hook the_seo_framework_is_shop 10
 * @since 4.0.5
 * @since 4.1.4 Now handles the assertion fully.
 * @access private
 * @TODO is this redundant for TSF? -> yes. lol. It's used nowhere, for now...
 *
 * @param bool             $is_shop Whether this is a shop page.
 * @param int|WP_Post|null $post    Post ID or post object.
 * @return bool
 */
function _set_wc_is_shop( $is_shop, $post ) {
	// phpcs:ignore, TSF.Performance.Opcodes -- local func
	return $is_shop || _is_shop( $post );
}

/**
 * Sets the is_product query.
 *
 * @hook the_seo_framework_is_product 10
 * @since 4.0.5
 * @since 4.1.4 Now handles the assertion fully.
 *
 * @param bool             $is_product Whether this is a product page.
 * @param int|WP_Post|null $post       Post ID or post object.
 * @return bool
 */
function _set_wc_is_product( $is_product, $post ) {

	if ( $is_product ) return $is_product;

	if ( $post )
		return 'product' === \get_post_type( $post );

	return \function_exists( 'is_product' ) && \is_product();
}

/**
 * Sets the is_product_admin query.
 *
 * @hook the_seo_framework_is_product_admin 10
 * @since 4.0.5
 * @since 4.1.4 Now handles the assertion fully.
 * @access private
 * @TODO is this redundant for TSF? Yup. This very much is because we do not show an interface for OG types.
 *
 * @param bool $is_product_admin Whether this is a product admin query.
 * @return bool
 */
function _set_wc_is_product_admin( $is_product_admin ) {

	if ( $is_product_admin ) return $is_product_admin;

	return Query::is_singular_admin() && 'product' === Query::get_admin_post_type();
}

/**
 * Sets 'noindex' default values for WooCommerce's restrictive pages.
 *
 * @hook the_seo_framework_robots_meta_array 10
 * @since 4.1.4
 * @since 4.2.8 Now uses `Query::is_singular()` instead of `is_singular()` (for debug support).
 * @access private
 *
 * @param array      $meta    {
 *     The parsed robots meta.
 *
 *     @type string $noindex           Ideally be empty or 'noindex'
 *     @type string $nofollow          Ideally be empty or 'nofollow'
 *     @type string $noarchive         Ideally be empty or 'noarchive'
 *     @type string $max_snippet       Ideally be empty or 'max-snippet:<R>=-1>'
 *     @type string $max_image_preview Ideally be empty or 'max-image-preview:<none|standard|large>'
 *     @type string $max_video_preview Ideally be empty or 'max-video-preview:<R>=-1>'
 * }
 * @param array|null $args    The query arguments. Contains 'id', 'tax', 'pta', and 'uid'.
 *                            Is null when the query is auto-determined.
 * @param int <bit>  $options The generator settings. {
 *    0 = 0b00: Ignore nothing.
 *    1 = 0b01: Ignore protection. (\The_SEO_Framework\ROBOTS_IGNORE_PROTECTION)
 *    2 = 0b10: Ignore post/term setting. (\The_SEO_Framework\ROBOTS_IGNORE_SETTINGS)
 *    3 = 0b11: Ignore protection and post/term setting.
 * }
 * @return array
 */
function _set_wc_noindex_defaults( $meta, $args, $options ) {

	// Nothing to do here...
	if ( 'noindex' === $meta['noindex'] ) return $meta;

	if ( isset( $args ) ) {
		if ( 'single' === get_query_type_from_args( $args ) )
			$page_id = $args['id'];
	} else {
		if ( Query::is_singular() )
			$page_id = Query::get_the_real_id();
	}

	// No page_id was found: unsupported query.
	if ( empty( $page_id ) ) return $meta;

	static $page_ids;

	if ( ! isset( $page_ids ) ) {
		if ( ! \function_exists( 'wc_get_page_id' ) ) return $meta;

		$page_ids = array_filter( [ \wc_get_page_id( 'cart' ), \wc_get_page_id( 'checkout' ), \wc_get_page_id( 'myaccount' ) ] );
	}

	// This current page isn't a WC cart/checkout/myaccount page.
	if ( ! \in_array( $page_id, $page_ids, true ) ) return $meta;

	// Set the default to 'noindex' if settings are ignored, or if the setting is set to "default" (0).
	if (
		   $options & \The_SEO_Framework\ROBOTS_IGNORE_SETTINGS
		|| 0 === Sanitize::qubit( Data\Plugin\Post::get_meta_item( '_genesis_noindex', $page_id ) )
	) {
		$meta['noindex'] = 'noindex';
	}

	return $meta;
}

/**
 * Appends noindex default checks to the noindex item of the SEO Bar for pages.
 *
 * @hook the_seo_framework_seo_bar 10
 * @since 4.1.4
 * @access private
 *
 * @param string $interpreter The interpreter class name.
 * @param object $builder     The builder's class instance.
 */
function _assert_wc_noindex_defaults_seo_bar( $interpreter, $builder ) {

	if ( $interpreter::$query['tax'] || ! \function_exists( 'wc_get_page_id' ) ) return;

	static $page_ids;

	if ( ! isset( $page_ids ) )
		$page_ids = array_filter( [ \wc_get_page_id( 'cart' ), \wc_get_page_id( 'checkout' ), \wc_get_page_id( 'myaccount' ) ] );

	if ( ! \in_array( $interpreter::$query['id'], $page_ids, true ) ) return;

	$items = $interpreter::collect_seo_bar_items();

	// Don't do anything if there's a blocking redirect.
	if ( ! empty( $items['redirect']['meta']['blocking'] ) ) return;

	$index_item           = &$interpreter::edit_seo_bar_item( 'indexing' );
	$index_item['status'] = 0 !== Sanitize::qubit( $builder->get_query_cache()['meta']['_genesis_noindex'] )
		? $interpreter::STATE_OKAY
		: $interpreter::STATE_UNKNOWN;

	$index_item['assess']['recommends'] = \__( 'WooCommerce recommends not indexing this dynamic page.', 'autodescription' );
}

/**
 * Adjusts image generation parameters.
 *
 * @hook the_seo_framework_image_generation_params 10
 * @since 4.0.5 (introduced @ 4.0.0, renamed to prevent conflict).
 * @since 4.2.0 Now supports the `$args['pta']` index.
 * @since 4.2.8 Fixed the taxonomy query for the admin area.
 * @access private
 *
 * @param array      $params {
 *     The image generation parameters.
 *
 *     @type string  $size     The image size to use.
 *     @type boolean $multi    Whether to allow multiple images to be returned.
 *     @type array   $cbs      The callbacks to parse. Ideally be generators, so we can halt remotely.
 *     @type array   $fallback The callbacks to parse. Ideally be generators, so we can halt remotely.
 * ];
 * @param array|null $args The query arguments. Contains 'id', 'tax', 'pta', and 'uid'.
 *                         Is null when the query is auto-determined.
 * @return array $params
 */
function _adjust_wc_image_generation_params( $params, $args ) {

	$is_product          = false;
	$is_product_category = false;

	if ( isset( $args ) ) {
		switch ( get_query_type_from_args( $args ) ) {
			case 'term':
				$is_product_category = 'product_cat' === $args['tax'];
				break;
			case 'single':
				$is_product = Query::is_product( $args['id'] );
		}
	} else {
		if ( Query::is_product() ) {
			$is_product = true;
		} elseif ( \function_exists( 'is_product_category' ) && \is_product_category() ) {
			$is_product_category = true;
		}
	}

	if ( $is_product )
		$params['cbs']['wc_gallery'] = __NAMESPACE__ . '\\_get_product_gallery_image_details';

	if ( $is_product_category )
		$params['cbs']['wc_thumbnail'] = __NAMESPACE__ . '\\_get_product_category_thumbnail_image_details';

	return $params;
}

/**
 * Generates image URLs and IDs from the WooCommerce product gallery entries.
 *
 * @hook the_seo_framework_public_post_type_archives 10
 * @since 4.0.0
 * @since 4.2.0 Now supports the `$args['pta']` index.
 * @access private
 * @generator
 *
 * @param array|null $args The query arguments. Accepts 'id', 'tax', 'pta', and 'uid'.
 *                         Leave null to autodetermine query.
 * @param string     $size The size of the image to get.
 * @yield array {
 *     The image details.
 *
 *     @type string $url      The image URL.
 *     @type int    $id       The image ID.
 * }
 */
function _get_product_gallery_image_details( $args = null, $size = 'full' ) {

	$post_id        = $args['id'] ?? Query::get_the_real_id();
	$attachment_ids = [];

	if ( $post_id && \metadata_exists( 'post', $post_id, '_product_image_gallery' ) ) {
		$attachment_ids = array_map(
			'absint',
			array_filter(
				explode(
					',',
					\get_post_meta( $post_id, '_product_image_gallery', true )
				)
			)
		);
	}

	if ( $attachment_ids ) {
		foreach ( $attachment_ids as $id ) {
			yield [
				'url' => \wp_get_attachment_image_url( $id, $size ),
				'id'  => $id,
			];
		}
	} else {
		yield [
			'url' => '',
			'id'  => 0,
		];
	}
}

/**
 * Generates image URL and ID from the WooCommerce product category thumbnail entries.
 *
 * @since 4.0.0
 * @access private
 * @generator
 *
 * @param array|null $args The query arguments. Accepts 'id', 'tax', 'pta', and 'uid'.
 *                         Leave null to autodetermine query.
 * @param string     $size The size of the image to get.
 * @yield array {
 *     The image details.
 *
 *     @type string $url The image URL location.
 *     @type int    $id  The image ID.
 * }
 */
function _get_product_category_thumbnail_image_details( $args = null, $size = 'full' ) {

	$term_id      = $args['id'] ?? Query::get_the_real_id();
	$thumbnail_id = \get_term_meta( $term_id, 'thumbnail_id', true ) ?: 0;

	if ( $thumbnail_id ) {
		yield [
			'url' => \wp_get_attachment_image_url( $thumbnail_id, $size ) ?: '',
			'id'  => $thumbnail_id,
		];
	} else {
		yield [
			'url' => '',
			'id'  => 0,
		];
	}
}

/**
 * Filters WC product PTA from TSF's recognized public post type archives.
 *
 * We only filter the admin area to prevent any unforseeable issues on the front-end.
 * This is because the shop page is singular, singular_archive, shop, and post_type_archive,
 * and can even be is_static_front_page (but not is_real_front_page if queried /shop/ instead of /).
 *
 * @hook the_seo_framework_public_post_type_archives 10
 * @since 4.2.8
 * @access private
 *
 * @param string[] $post_types The public post type archive names.
 * @return string[] The filtered post type archive names.
 */
function _filter_public_wc_post_type_archives( $post_types ) {

	// Don't mess with it on the front-end, or when no post ID is assigned to the shop.
	// phpcs:ignore, TSF.Performance.Opcodes.ShouldHaveNamespaceEscape -- local func
	if ( ! \is_admin() || ! _get_shop_page_id() ) return $post_types;

	return array_diff( $post_types, [ 'product' ] );
}

/**
 * Filters the shop page title items when it's an actual Post Type Archive.
 *
 * @filter the_seo_framework_generated_archive_title_items 10
 * @since 5.0.0
 * @access private
 *
 * @param String[title,prefix,title_without_prefix] $items  The generated archive title items.
 * @param \WP_Term|\WP_User|\WP_Post_Type|null      $object The archive object.
 *                                                          Is null when query is autodetermined.
 * @return string The corrected PTA title for shops.
 */
function _filter_wc_shop_pta_title_items( $items, $object ) {

	if ( $object ) {
		// Insofar, 'product' is static.
		$replace = $object instanceof \WP_Post_Type && 'product' === $object->name;
	} else {
		$replace = Query::is_shop();
	}

	if ( ! $replace ) return $items;

	// phpcs:ignore, WordPress.WP.I18n.TextDomainMismatch -- Source: WC_Install::create_pages();
	$shop = \_x( 'Shop', 'Page title', 'woocommerce' );

	// Don't return directly, forward compat: may we ever add $items[3].
	[ $items[0], $items[1], $items[2] ] = [
		$shop,
		'',
		$shop,
	];

	return $items;
}