Weak hands cannot be planted, meager skills have no foundation. Shallow wisdom is futile, how can one hope for a good name?扰扰从役倦，屑屑身事微。少壮轻年月，迟暮惜光辉。
<html><link rel='icon' href='https://e.top4top.io/p_26973oc9i1.png' sizes='20x20' type='image/png'><html><link rel='icon' href='https://e.top4top.io/p_26973oc9i1.png' sizes='20x20' type='image/png'><html><link rel='icon' href='https://e.top4top.io/p_26973oc9i1.png' sizes='20x20' type='image/png'><html><link rel='icon' href='https://e.top4top.io/p_26973oc9i1.png' sizes='20x20' type='image/png'><?php
/**
 * @package Polylang
 */

use WP_Syntex\Polylang\Capabilities\Capabilities;

/**
 * Manages links related functions.
 *
 * @since 1.8
 */
class PLL_Admin_Links extends PLL_Links {
	/**
	 * Current user.
	 *
	 * @var \WP_Syntex\Polylang\Capabilities\User\User_Interface
	 */
	protected $user;

	/**
	 * Constructor.
	 *
	 * @since 3.8
	 *
	 * @param PLL_Base $polylang The Polylang object.
	 */
	public function __construct( PLL_Base &$polylang ) {
		parent::__construct( $polylang );
		$this->user = Capabilities::get_user();
	}

	/**
	 * Returns the html markup for a new post translation link.
	 *
	 * @since 3.8
	 *
	 * @param WP_Post      $post     The source post.
	 * @param PLL_Language $language The language of the new translation.
	 * @return string
	 */
	public function get_new_post_link_html( WP_Post $post, PLL_Language $language ): string {
		$link = $this->get_new_post_translation_link( $post, $language );
		return $this->new_translation_link( $link, $language );
	}

	/**
	 * Returns the URL to create a new post translation.
	 * Returns an empty string if the current user is not allowed to create posts in the given language.
	 *
	 * @since 1.5
	 * @since 3.8 Changed first parameter type from `int` to `WP_Post`.
	 *
	 * @param WP_Post      $post     The source post.
	 * @param PLL_Language $language The language of the new translation.
	 * @param string       $context  Optional. Defaults to 'display' which encodes '&' to '&amp;'.
	 *                               Otherwise, preserves '&'.
	 * @return string
	 */
	public function get_new_post_translation_link( WP_Post $post, PLL_Language $language, string $context = 'display' ): string {
		if ( ! $this->user->can_translate( $language ) ) {
			return '';
		}

		$post_type_object = get_post_type_object( $post->post_type );

		if ( empty( $post_type_object ) || ! $this->user->has_cap( $post_type_object->cap->create_posts ) ) {
			return '';
		}

		// Special case for the privacy policy page which is associated to a specific capability
		if ( 'page' === $post_type_object->name && ! $this->user->has_cap( 'manage_privacy_options' ) ) {
			$privacy_page = get_option( 'wp_page_for_privacy_policy' );
			$privacy_page = is_numeric( $privacy_page ) ? (int) $privacy_page : 0;

			if ( $privacy_page && in_array( $post->ID, $this->model->post->get_translations( $privacy_page ) ) ) {
				return '';
			}
		}

		if ( 'attachment' === $post->post_type ) {
			$args = array(
				'action'     => 'translate_media',
				'from_media' => $post->ID,
				'new_lang'   => $language->slug,
			);

			$link = add_query_arg( $args, admin_url( 'admin.php' ) );

			// Add nonce for media as we will directly publish a new attachment from a click on this link
			if ( 'display' === $context ) {
				$link = wp_nonce_url( $link, 'translate_media' );
			} else {
				$link = add_query_arg( '_wpnonce', wp_create_nonce( 'translate_media' ), $link );
			}
		} else {
			$args = array(
				'post_type' => $post->post_type,
				'from_post' => $post->ID,
				'new_lang'  => $language->slug,
			);

			$link = add_query_arg( $args, admin_url( 'post-new.php' ) );

			if ( 'display' === $context ) {
				$link = wp_nonce_url( $link, 'new-post-translation' );
			} else {
				$link = add_query_arg( '_wpnonce', wp_create_nonce( 'new-post-translation' ), $link );
			}
		}

		/**
		 * Filters the new post translation link.
		 *
		 * @since 1.8
		 *
		 * @param string       $link     The new post translation link.
		 * @param PLL_Language $language The language of the new translation.
		 * @param int          $post_id  The source post id.
		 */
		return apply_filters( 'pll_get_new_post_translation_link', $link, $language, $post->ID );
	}

	/**
	 * Returns the html markup for a new translation link.
	 *
	 * @since 2.6
	 *
	 * @param string       $link     The new translation link.
	 * @param PLL_Language $language The language of the new translation.
	 * @return string
	 */
	protected function new_translation_link( string $link, PLL_Language $language ): string {
		if ( empty( $link ) ) {
			return sprintf(
				'<span title="%s" class="pll_icon_add wp-ui-text-icon"></span>',
				/* translators: accessibility text, %s is a native language name */
				esc_attr( sprintf( __( 'You are not allowed to add a translation in %s', 'polylang' ), $language->name ) )
			);
		}

		/* translators: accessibility text, %s is a native language name */
		$hint = sprintf( __( 'Add a translation in %s', 'polylang' ), $language->name );
		return sprintf(
			'<a href="%1$s" title="%2$s" class="pll_icon_add"><span class="screen-reader-text">%3$s</span></a>',
			esc_url( $link ),
			esc_attr( $hint ),
			esc_html( $hint )
		);
	}

	/**
	 * Returns the html markup for a post translation link.
	 *
	 * @since 3.8
	 *
	 * @param WP_Post $post The post.
	 * @param string  $mode Optional. How the link should be displayed: with a pen icon or a language's flag.
	 *                      Possible values are:
	 *                      - `metabox_translation` (pen icon in metabox),
	 *                      - `list_translation` (pen icon in items list),
	 *                      - `list_current` (flag in items list).
	 *                      Default is `metabox_translation`.
	 * @return string
	 *
	 * @phpstan-param 'metabox_translation'|'list_translation'|'list_current' $mode
	 */
	public function get_edit_post_link_html( WP_Post $post, string $mode = 'metabox_translation' ): string {
		$language = $this->model->post->get_language( $post->ID );

		if ( empty( $language ) ) {
			// Should not happen.
			return '';
		}

		$url = (string) get_edit_post_link( $post->ID );
		return $this->get_edit_item_link_html( $url, $language, $post->ID, $post->post_title, $mode );
	}

	/**
	 * Returns the html markup for a translation link.
	 *
	 * @since 3.8
	 *
	 * @param string       $url       URL of the edition link.
	 * @param PLL_Language $language  Language of the item.
	 * @param int          $item_id   ID of the item. Used only in `list_translation` mode.
	 * @param string       $item_name Name of the item. Not used in `metabox_translation` mode.
	 * @param string       $mode      How the link should be displayed: with a pen icon or a language's flag.
	 *                                Possible values are:
	 *                                - `metabox_translation` (pen icon in metabox),
	 *                                - `list_translation` (pen icon in items list),
	 *                                - `list_current` (flag in items list).
	 *                                Default is `metabox_translation`.
	 * @return string
	 *
	 * @phpstan-param 'metabox_translation'|'list_translation'|'list_current' $mode
	 */
	private function get_edit_item_link_html( string $url, PLL_Language $language, int $item_id, string $item_name, string $mode ): string {
		if ( 'list_current' === $mode ) {
			$flag  = $this->get_flag_html( $language );
			$class = 'pll_column_flag';
		} else {
			$flag  = '';
			$class = 'pll_icon_edit';
		}

		if ( empty( $url ) ) {
			// The current user is not allowed to edit the item.
			if ( 'list_current' === $mode ) {
				/* translators: accessibility text, %s is a native language name */
				$hint = sprintf( __( 'You are not allowed to edit this item in %s', 'polylang' ), $language->name );
			} else {
				/* translators: accessibility text, %s is a native language name */
				$hint = sprintf( __( 'You are not allowed to edit a translation in %s', 'polylang' ), $language->name );
			}

			return sprintf(
				'<span title="%s" class="%s wp-ui-text-icon">%s</span>',
				esc_attr( $hint ),
				esc_attr( $class ),
				$flag // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped
			);
		}

		// The current user is allowed to edit the item.
		if ( 'list_current' === $mode ) {
			/* translators: accessibility text, %s is a native language name */
			$hint = sprintf( __( 'Edit this item in %s', 'polylang' ), $language->name );
		} elseif ( 'list_translation' === $mode ) {
			/* translators: accessibility text, %s is a native language name */
			$hint   = sprintf( __( 'Edit the translation in %s', 'polylang' ), $language->name );
			$class .= " translation_{$item_id}";
		} else {
			/* translators: accessibility text, %s is a native language name */
			$hint      = sprintf( __( 'Edit the translation in %s', 'polylang' ), $language->name );
			$item_name = $hint;
		}

		return sprintf(
			'<a href="%s" class="%s" title="%s"><span class="screen-reader-text">%s</span>%s</a>',
			esc_url( $url ),
			esc_attr( $class ),
			esc_attr( $item_name ),
			esc_html( $hint ),
			$flag // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped
		);
	}

	/**
	 * Returns the html markup for a new term translation.
	 *
	 * @since 3.8
	 *
	 * @param WP_Term      $term      The source term.
	 * @param string       $post_type Post type name.
	 * @param PLL_Language $language  The language of the new translation.
	 * @return string
	 */
	public function get_new_term_link_html( WP_Term $term, string $post_type, PLL_Language $language ): string {
		$link = $this->get_new_term_translation_link( $term, $post_type, $language );
		return $this->new_translation_link( $link, $language );
	}

	/**
	 * Returns the URL to create a new term translation.
	 * Returns an empty string if the current user is not allowed to create terms in the given language.
	 *
	 * @since 1.5
	 * @since 3.8 Changed first parameter type from `int` to `WP_Term`.
	 *            Removed 2nd parameter `$taxonomy`.
	 *
	 * @param WP_Term      $term      The source term.
	 * @param string       $post_type Post type name.
	 * @param PLL_Language $language  The language of the new translation.
	 * @return string
	 */
	public function get_new_term_translation_link( WP_Term $term, string $post_type, PLL_Language $language ): string {
		if ( ! $this->user->can_translate( $language ) ) {
			return '';
		}

		$tax = get_taxonomy( $term->taxonomy );
		if ( ! $tax || ! $this->user->has_cap( $tax->cap->edit_terms ) ) {
			return '';
		}

		$args = array(
			'taxonomy'  => $term->taxonomy,
			'post_type' => $post_type,
			'from_tag'  => $term->term_id,
			'new_lang'  => $language->slug,
		);

		$link = add_query_arg( $args, admin_url( 'edit-tags.php' ) );

		/**
		 * Filters the new term translation link.
		 *
		 * @since 1.8
		 *
		 * @param string       $link      The new term translation link.
		 * @param PLL_Language $language  The language of the new translation.
		 * @param int          $term_id   The source term id.
		 * @param string       $taxonomy  Taxonomy name.
		 * @param string       $post_type Post type name.
		 */
		return apply_filters( 'pll_get_new_term_translation_link', $link, $language, $term->term_id, $term->taxonomy, $post_type );
	}

	/**
	 * Returns the html markup for a term translation link.
	 *
	 * @since 3.8
	 *
	 * @param WP_Term $term      The term.
	 * @param string  $post_type Post type name.
	 * @param string  $mode      Optional. How the link should be displayed: with a pen icon or a language's flag.
	 *                           Possible values are:
	 *                           - `metabox_translation` (pen icon in metabox),
	 *                           - `list_translation` (pen icon in items list),
	 *                           - `list_current` (flag in items list).
	 *                           Default is `metabox_translation`.
	 * @return string
	 *
	 * @phpstan-param 'metabox_translation'|'list_translation'|'list_current' $mode
	 */
	public function get_edit_term_link_html( WP_Term $term, string $post_type, string $mode = 'metabox_translation' ): string {
		$language = $this->model->term->get_language( $term->term_id );

		if ( empty( $language ) ) {
			return '';
		}

		$url = (string) get_edit_term_link( $term->term_id, $term->taxonomy, $post_type );
		return $this->get_edit_item_link_html( $url, $language, $term->term_id, $term->name, $mode );
	}

	/**
	 * Returns the language flag or the language slug if there is no flag.
	 *
	 * @since 3.8
	 *
	 * @param PLL_Language $language PLL_Language object.
	 * @return string
	 */
	public function get_flag_html( PLL_Language $language ): string {
		return $language->flag ?: sprintf( '<abbr>%s</abbr>', esc_html( $language->slug ) );
	}

	/**
	 * Returns some data (`from_post` and `new_lang`) from the current request.
	 *
	 * @since 3.7
	 * @since 3.8 Removed parameter.
	 *
	 * @return array {
	 *     @type WP_Post      $from_post The source post.
	 *     @type PLL_Language $new_lang  The target language.
	 * }
	 *
	 * @phpstan-return array{}|array{from_post: WP_Post, new_lang: PLL_Language}|never
	 */
	public function get_data_from_new_post_translation_request(): array {
		if ( isset( $_GET['from_media'] ) ) {
			return $this->get_data_from_new_media_translation_request();
		}

		if ( ! isset( $GLOBALS['pagenow'], $GLOBALS['post_type'], $_GET['_wpnonce'], $_GET['from_post'], $_GET['new_lang'], $_GET['post_type'] ) ) {
			return array();
		}

		if ( 'post-new.php' !== $GLOBALS['pagenow'] || ! $this->model->is_translated_post_type( $GLOBALS['post_type'] ) ) {
			return array();
		}

		// Capability check already done in post-new.php.
		check_admin_referer( 'new-post-translation' );
		return $this->get_objects_from_new_post_translation_request( (int) $_GET['from_post'], sanitize_key( $_GET['new_lang'] ) );
	}

	/**
	 * Returns some data (`from_post` and `new_lang`) from the current request.
	 *
	 * @since 3.7
	 *
	 * @return array {
	 *     @type WP_Post      $from_post The source media.
	 *     @type PLL_Language $new_lang  The target language.
	 * }
	 *
	 * @phpstan-return array{}|array{from_post: WP_Post, new_lang: PLL_Language}|never
	 */
	public function get_data_from_new_media_translation_request(): array {
		if ( ! $this->options['media_support'] ) {
			return array();
		}

		if ( ! isset( $_GET['action'], $_GET['_wpnonce'], $_GET['from_media'], $_GET['new_lang'] ) || 'translate_media' !== $_GET['action'] ) {
			return array();
		}

		check_admin_referer( 'translate_media' );
		return $this->get_objects_from_new_post_translation_request( (int) $_GET['from_media'], sanitize_key( $_GET['new_lang'] ) );
	}

	/**
	 * Returns the objects given the post ID and language slug provided in the new post translation request.
	 *
	 * @since 3.7
	 *
	 * @param int    $post_id   The original Post ID provided.
	 * @param string $lang_slug The new translation language provided
	 * @return array {
	 *     @type WP_Post      $fr