b0y-101 Mini Shell


Current Path : E:/www3/chiangrai/wp-content/plugins/polylang/modules/wpml/
File Upload :
Current File : E:/www3/chiangrai/wp-content/plugins/polylang/modules/wpml/wpml-api.php

<?php
/**
 * @package Polylang
 */

/**
 * A class to handle the WPML API based on hooks, introduced since WPML 3.2.
 * It partly relies on the legacy API.
 *
 * @see https://wpml.org/documentation/support/wpml-coding-api/wpml-hooks-reference/
 *
 * @since 2.0
 */
class PLL_WPML_API {
	/**
	 * Stores the original language when the language is switched.
	 *
	 * @var PLL_Language|null
	 */
	private static $original_language = null;

	/**
	 * Constructor.
	 *
	 * @since 2.0
	 */
	public function __construct() {
		/*
		 * Site Wide Language information.
		 */
		add_filter( 'wpml_active_languages', array( $this, 'wpml_active_languages' ), 10, 2 );
		add_filter( 'wpml_display_language_names', array( $this, 'wpml_display_language_names' ), 10, 2 ); // Because we don't translate language names, 3rd to 5th parameters are not supported.
		// wpml_translated_language_name           => not applicable.
		add_filter( 'wpml_current_language', 'pll_current_language', 10, 0 );
		add_filter( 'wpml_default_language', 'pll_default_language', 10, 0 );
		// wpml_add_language_selector              => not implemented.
		// wpml_footer_language_selector           => not applicable.
		add_action( 'wpml_add_language_form_field', array( $this, 'wpml_add_language_form_field' ) );
		add_filter( 'wpml_language_is_active', array( $this, 'wpml_language_is_active' ), 10, 2 );
		add_filter( 'wpml_is_rtl', array( $this, 'wpml_is_rtl' ) );
		// wpml_language_form_input_field          => See wpml_add_language_form_field
		// wpml_language_has_switched              => See wpml_switch_language
		add_filter( 'wpml_element_trid', array( $this, 'wpml_element_trid' ), 10, 3 );
		add_filter( 'wpml_get_element_translations', array( $this, 'wpml_get_element_translations' ), 10, 3 );
		// wpml_language_switcher                  => not implemented.
		// wpml_browser_redirect_language_params   => not implemented.
		// wpml_enqueue_browser_redirect_language  => not applicable.
		// wpml_enqueued_browser_redirect_language => not applicable.
		// wpml_encode_string                      => not applicable.
		// wpml_decode_string                      => not applicable.

		/*
		 * Retrieving Language Information for Content.
		 */
		add_filter( 'wpml_post_language_details', 'wpml_get_language_information', 10, 2 );
		add_action( 'wpml_switch_language', array( self::class, 'wpml_switch_language' ), 10, 2 );
		add_filter( 'wpml_element_language_code', array( $this, 'wpml_element_language_code' ), 10, 2 );
		// wpml_element_language_details           => not applicable.

		/*
		 * Retrieving Localized Content.
		 */
		add_filter( 'wpml_home_url', 'pll_home_url', 10, 0 );
		add_filter( 'wpml_element_link', 'icl_link_to_element', 10, 7 );
		add_filter( 'wpml_object_id', 'icl_object_id', 10, 4 );
		add_filter( 'wpml_translate_single_string', array( $this, 'wpml_translate_single_string' ), 10, 4 );
		// wpml_translate_string                   => not applicable.
		// wpml_unfiltered_admin_string            => not implemented.
		add_filter( 'wpml_permalink', array( $this, 'wpml_permalink' ), 10, 2 );
		// wpml_elements_without_translations      => not implemented.
		add_filter( 'wpml_get_translated_slug', array( $this, 'wpml_get_translated_slug' ), 10, 3 );

		/*
		 * Finding the Translation State of Content.
		 */
		// wpml_element_translation_type           => not implemented.
		add_filter( 'wpml_element_has_translations', array( $this, 'wpml_element_has_translations' ), 10, 3 );
		// wpml_master_post_from_duplicate         => not applicable.
		// wpml_post_duplicates                    => not applicable.

		/*
		 * Inserting Content.
		 */
		// wpml_admin_make_post_duplicates         => not applicable.
		// wpml_make_post_duplicates               => not applicable.
		add_action( 'wpml_register_single_string', 'icl_register_string', 10, 3 );
		// wpml_register_string                    => not applicable.
		// wpml_register_string_packages           => not applicable.
		// wpml_delete_package_action              => not applicable.
		// wpml_show_package_language_ui           => not applicable.
		// wpml_set_element_language_details       => not implemented.
		// wpml_multilingual_options               => not applicable.

		/*
		 * Miscellaneous
		 */
		// wpml_element_type                       => not applicable.
		// wpml_setting                            => not applicable.
		// wpml_sub_setting                        => not applicable.
		// wpml_editor_cf_to_display               => not applicable.
		// wpml_tm_save_translation_cf             => not implemented.
		// wpml_tm_xliff_export_translated_cf      => not applicable.
		// wpml_tm_xliff_export_original_cf        => not applicable.
		// wpml_duplicate_generic_string           => not applicable.
		// wpml_translatable_user_meta_fields      => not implemented.
		// wpml_cross_domain_language_data         => not applicable.
		// wpml_get_cross_domain_language_data     => not applicable.
		// wpml_loaded                             => not applicable.
		// wpml_st_loaded                          => not applicable.
		// wpml_tm_loaded                          => not applicable.
		// wpml_hide_management_column (3.4.1)     => not applicable.
		// wpml_ls_directories_to_scan             => not applicable.
		// wpml_ls_model_css_classes               => not applicable.
		// wpml_ls_model_language_css_classes      => not applicable.
		// wpml_tf_feedback_open_link              => not applicable.
		// wpml_sync_custom_field                  => not implemented.
		// wpml_sync_all_custom_fields             => not implemented.
		// wpml_is_redirected                      => not implemented.

		/*
		 * Updating Content
		 */
		// wpml_set_translation_mode_for_post_type => not implemented.

		/*
		 * Undocumented
		 */
		add_filter( 'wpml_is_translated_post_type', array( $this, 'wpml_is_translated_post_type' ), 10, 2 );
		add_filter( 'wpml_is_translated_taxonomy', array( $this, 'wpml_is_translated_taxonomy' ), 10, 2 );
	}

	/**
	 * Get a list of the languages enabled for a site.
	 *
	 * @since 2.0
	 *
	 * @param mixed         $null Not used.
	 * @param array| string $args See arguments of icl_get_languages().
	 * @return array Array of arrays per language.
	 */
	public function wpml_active_languages( $null, $args = '' ) {
		return icl_get_languages( $args );
	}

	/**
	 * In WPML, get a language's native and translated name for display in a custom language switcher
	 * Since Polylang does not implement the translated name, always returns only the native name,
	 * so the 3rd, 4th and 5th parameters are not used.
	 *
	 * @since 2.2
	 *
	 * @param mixed  $null        Not used.
	 * @param string $native_name The language native name.
	 * @return string
	 */
	public function wpml_display_language_names( $null, $native_name ) {
		return $native_name;
	}

	/**
	 * Returns an HTML hidden input field with name=”lang” and as value the current language.
	 *
	 * @since 2.0
	 *
	 * @return void
	 */
	public function wpml_add_language_form_field() {
		$lang = pll_current_language();

		if ( empty( $lang ) ) {
			return;
		}

		$field = sprintf( '<input type="hidden" name="lang" value="%s" />', esc_attr( $lang ) );
		$field = apply_filters( 'wpml_language_form_input_field', $field, $lang );
		echo $field; // phpcs:ignore WordPress.Security.EscapeOutput
	}

	/**
	 * Find out if a specific language is enabled for the site.
	 *
	 * @since 2.0
	 *
	 * @param mixed  $null Not used.
	 * @param string $slug Language code.
	 * @return bool
	 */
	public function wpml_language_is_active( $null, $slug ) {
		$language = PLL()->model->get_language( $slug );
		return ! empty( $language ) && $language->active;
	}

	/**
	 * Find out whether the current language text direction is RTL or not.
	 *
	 * @since 2.0
	 *
	 * @return bool
	 */
	public function wpml_is_rtl() {
		return pll_current_language( 'is_rtl' );
	}

	/**
	 * Returns the id of the translation group of a translated element.
	 *
	 * @since 3.4
	 *
	 * @param mixed  $empty_value  Not used.
	 * @param int    $element_id   The id of the item, post id for posts, term_taxonomy_id for terms.
	 * @param string $element_type Optional. The type of an element.
	 * @return int
	 */
	public function wpml_element_trid( $empty_value, $element_id, $element_type = 'post_post' ) {
		if ( 0 === strpos( $element_type, 'tax_' ) ) {
			$element = get_term_by( 'term_taxonomy_id', $element_id );
			if ( $element instanceof WP_Term ) {
				$tr_term = PLL()->model->term->get_object_term( $element->term_id, 'term_translations' );
			}
		}

		if ( 0 === strpos( $element_type, 'post_' ) ) {
			$tr_term = PLL()->model->post->get_object_term( $element_id, 'post_translations' );
		}

		if ( isset( $tr_term ) && $tr_term instanceof WP_Term ) {
			return $tr_term->term_id;
		}

		return 0;
	}

	/**
	 * Returns the element translations info using the ID of the translation group.
	 *
	 * @since 3.4
	 *
	 * @param mixed  $empty_value  Not used.
	 * @param int    $trid         The ID of the translation group.
	 * @param string $element_type Optional. The type of an element.
	 * @return stdClass[]
	 */
	public function wpml_get_element_translations( $empty_value, $trid, $element_type = 'post_post' ) {
		$return = array();

		if ( 0 === strpos( $element_type, 'tax_' ) ) {
			$translations = PLL()->model->term->get_translations_from_term_id( $trid );
			if ( empty( $translations ) ) {
				return array();
			}

			$original    = min( $translations ); // We suppose that the original is the first term created.
			$source_lang = array_search( $original, $translations );

			$args = array(
				'include'    => $translations,
				'hide_empty' => false,
			);
			$_terms = get_terms( $args );

			if ( ! is_array( $_terms ) ) {
				return array();
			}

			$terms = array();
			foreach ( $_terms as $term ) {
				$terms[ $term->term_id ] = $term;
			}

			foreach ( $translations as $lang => $term_id ) {
				if ( empty( $terms[ $term_id ] ) ) {
					continue;
				}

				/*
				 * It seems that WPML fills the `instances` property with the total number of posts
				 * related to this term, while `WP_Term::$count` includes only *published* posts.
				 * We intentionnally accept this difference to avoid extra DB queries.
				 */
				$return[ $lang ] = (object) array(
					'translation_id'       => '0', // We have nothing equivalent.
					'language_code'        => $lang,
					'element_id'           => (string) $terms[ $term_id ]->term_taxonomy_id,
					'source_language_code' => $source_lang === $lang ? null : $source_lang,
					'element_type'         => $element_type,
					'original'             => $original === $term_id ? '1' : '0',
					'name'                 => $terms[ $term_id ]->name,
					'term_id'              => (string) $term_id,
					'instances'            => (string) $terms[ $term_id ]->count,
				);
			}
		}

		if ( 0 === strpos( $element_type, 'post_' ) ) {
			$translations = PLL()->model->post->get_translations_from_term_id( $trid );
			if ( empty( $translations ) ) {
				return array();
			}

			$original    = min( $translations ); // We suppose that the original is the first post created.
			$source_lang = array_search( $original, $translations );

			$args  = array(
				'post__in'               => $translations,
				'no_paging'              => true,
				'posts_per_page'         => -1,
				'update_post_meta_cache' => false,
				'update_post_term_cache' => false,
				'lang'                   => '',
			);
			$_posts = get_posts( $args );

			$posts = array();
			foreach ( $_posts as $post ) {
				$posts[ $post->ID ] = $post;
			}

			foreach ( $translations as $lang => $post_id ) {
				if ( empty( $posts[ $post_id ] ) ) {
					continue;
				}

				$return[ $lang ] = (object) array(
					'translation_id'       => '0', // We have nothing equivalent.
					'language_code'        => $lang,
					'element_id'           => (string) $post_id,
					'source_language_code' => $source_lang === $lang ? null : $source_lang,
					'element_type'         => $element_type,
					'original'             => $original === $post_id ? '1' : '0',
					'post_title'           => $posts[ $post_id ]->post_title,
					'post_status'          => $posts[ $post_id ]->post_status,
				);
			}
		}

		return $return;
	}

	/**
	 * Switches whole site to the given language or restores the language that was set when first calling this function.
	 * Unlike the WPML original action, it is not possible to set the current language and the cookie to different values.
	 *
	 * @since 2.7
	 *
	 * @param null|string $lang   Language code to switch into, restores the original language if null.
	 * @param bool|string $cookie Optionally also switches the cookie.
	 * @return void
	 */
	public static function wpml_switch_language( $lang = null, $cookie = false ) {
		if ( null === self::$original_language ) {
			self::$original_language = PLL()->curlang;
		}

		if ( empty( $lang ) ) {
			PLL()->curlang = self::$original_language;
		} elseif ( 'all' === $lang ) {
			PLL()->curlang = null;
		} elseif ( in_array( $lang, pll_languages_list() ) ) {
			PLL()->curlang = PLL()->model->get_language( $lang );
		}

		if ( $cookie && isset( PLL()->choose_lang ) ) {
			PLL()->choose_lang->maybe_setcookie();
		}

		do_action( 'wpml_language_has_switched', $lang, $cookie, self::$original_language );
	}

	/**
	 * Get the language code for a translatable element.
	 *
	 * @since 2.0
	 *
	 * @param mixed $language_code A 2-letter language code.
	 * @param array $args          An array with two keys element_id => post_id or term_taxonomy_id, element_type => post type or taxonomy
	 * @return string|null
	 */
	public function wpml_element_language_code( $language_code, $args ) {
		$type = $args['element_type'];
		$id   = $args['element_id'];

		if ( 'post' === $type || pll_is_translated_post_type( $type ) ) {
			$language = pll_get_post_language( $id );
			return is_string( $language ) ? $language : null;
		}

		if ( 'term' === $type || pll_is_translated_taxonomy( $type ) ) {
			$term = get_term_by( 'term_taxonomy_id', $id );
			if ( $term instanceof WP_Term ) {
				$id = $term->term_id;
			}
			$language = pll_get_term_language( $id );
			return is_string( $language ) ? $language : null;
		}

		return null;
	}

	/**
	 * Translates a string.
	 *
	 * @since 2.0
	 *
	 * @param string      $string  The string's original value.
	 * @param string      $context The string's registered context.
	 * @param string      $name    The string's registered name.
	 * @param null|string $lang    Optional, return the translation in this language, defaults to current language.
	 * @return string The translated string.
	 */
	public function wpml_translate_single_string( $string, $context, $name, $lang = null ) {
		$has_translation = null; // Passed by reference.
		return icl_translate( $context, $name, $string, false, $has_translation, $lang );
	}

	/**
	 * Converts a permalink to a language specific permalink.
	 *
	 * @since 2.2
	 *
	 * @param string      $url  The url to filter.
	 * @param null|string $lang Language code, optional, defaults to the current language.
	 * @return string
	 */
	public function wpml_permalink( $url, $lang = '' ) {
		$lang = PLL()->model->get_language( $lang );

		if ( empty( $lang ) && ! empty( PLL()->curlang ) ) {
			$lang = PLL()->curlang;
		}

		return empty( $lang ) ? $url : PLL()->links_model->switch_language_in_link( $url, $lang );
	}

	/**
	 * Translates a post type slug.
	 *
	 * @since 2.2
	 *
	 * @param string $slug      Post type slug.
	 * @param string $post_type Post type name.
	 * @param string $lang      Optional language code (defaults to current language).
	 * @return string
	 */
	public function wpml_get_translated_slug( $slug, $post_type, $lang = null ) {
		if ( isset( PLL()->translate_slugs ) ) {
			if ( empty( $lang ) ) {
				$lang = pll_current_language();
			}

			$slug = PLL()->translate_slugs->slugs_model->get_translated_slug( $post_type, $lang );
		}
		return $slug;
	}

	/**
	 * Find out whether a post type or a taxonomy term is translated.
	 *
	 * @since 2.0
	 *
	 * @param mixed  $null Not used.
	 * @param int    $id   The post_id or term_id.
	 * @param string $type The post type or taxonomy.
	 * @return bool
	 */
	public function wpml_element_has_translations( $null, $id, $type ) {
		if ( 'post' === $type || pll_is_translated_post_type( $type ) ) {
			return count( pll_get_post_translations( $id ) ) > 1;
		} elseif ( 'term' === $type || pll_is_translated_taxonomy( $type ) ) {
			return count( pll_get_term_translations( $id ) ) > 1;
		}

		return false;
	}

	/**
	 * Returns true if languages and translations are managed for this post type.
	 *
	 * @since 3.4
	 *
	 * @param mixed  $value     Not used.
	 * @param string $post_type The post type name.
	 * @return bool
	 */
	public function wpml_is_translated_post_type( $value, $post_type ) {
		return pll_is_translated_post_type( $post_type );
	}

	/**
	 * Returns true if languages and translations are managed for this taxonomy.
	 *
	 * @since 3.4
	 *
	 * @param mixed  $value    Not used.
	 * @param string $taxonomy The taxonomy name.
	 * @return bool
	 */
	public function wpml_is_translated_taxonomy( $value, $taxonomy ) {
		return pll_is_translated_taxonomy( $taxonomy );
	}
}

Copyright © 2019 by b0y-101