123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305 |
- <?php
- /**
- * WPSEO plugin file.
- *
- * @package WPSEO\Admin
- */
- /**
- * Adds the UI to change the primary term for a post.
- */
- class WPSEO_Primary_Term_Admin implements WPSEO_WordPress_Integration {
- /**
- * Constructor.
- */
- public function register_hooks() {
- add_filter( 'wpseo_content_meta_section_content', [ $this, 'add_input_fields' ] );
- add_action( 'admin_footer', [ $this, 'wp_footer' ], 10 );
- add_action( 'admin_enqueue_scripts', [ $this, 'enqueue_assets' ] );
- add_action( 'save_post', [ $this, 'save_primary_terms' ] );
- $primary_term = new WPSEO_Frontend_Primary_Category();
- $primary_term->register_hooks();
- }
- /**
- * Gets the current post ID.
- *
- * @return integer The post ID.
- */
- protected function get_current_id() {
- $post_id = filter_input( INPUT_GET, 'post', FILTER_SANITIZE_NUMBER_INT );
- if ( empty( $post_id ) && isset( $GLOBALS['post_ID'] ) ) {
- $post_id = filter_var( $GLOBALS['post_ID'], FILTER_SANITIZE_NUMBER_INT );
- }
- return $post_id;
- }
- /**
- * Adds hidden fields for primary taxonomies.
- *
- * @param string $content The metabox content.
- *
- * @return string The HTML content.
- */
- public function add_input_fields( $content ) {
- $taxonomies = $this->get_primary_term_taxonomies();
- foreach ( $taxonomies as $taxonomy ) {
- $content .= $this->primary_term_field( $taxonomy->name );
- $content .= wp_nonce_field( 'save-primary-term', WPSEO_Meta::$form_prefix . 'primary_' . $taxonomy->name . '_nonce', false, false );
- }
- return $content;
- }
- /**
- * Generates the HTML for a hidden field for a primary taxonomy.
- *
- * @param string $taxonomy_name The taxonomy's slug.
- *
- * @return string The HTML for a hidden primary taxonomy field.
- */
- protected function primary_term_field( $taxonomy_name ) {
- return sprintf(
- '<input class="yoast-wpseo-primary-term" type="hidden" id="%1$s" name="%2$s" value="%3$s" />',
- esc_attr( $this->generate_field_id( $taxonomy_name ) ),
- esc_attr( $this->generate_field_name( $taxonomy_name ) ),
- esc_attr( $this->get_primary_term( $taxonomy_name ) )
- );
- }
- /**
- * Generates an id for a primary taxonomy's hidden field.
- *
- * @param string $taxonomy_name The taxonomy's slug.
- *
- * @return string The field id.
- */
- protected function generate_field_id( $taxonomy_name ) {
- return 'yoast-wpseo-primary-' . $taxonomy_name;
- }
- /**
- * Generates a name for a primary taxonomy's hidden field.
- *
- * @param string $taxonomy_name The taxonomy's slug.
- *
- * @return string The field id.
- */
- protected function generate_field_name( $taxonomy_name ) {
- return WPSEO_Meta::$form_prefix . 'primary_' . $taxonomy_name . '_term';
- }
- /**
- * Adds primary term templates.
- */
- public function wp_footer() {
- $taxonomies = $this->get_primary_term_taxonomies();
- if ( ! empty( $taxonomies ) ) {
- $this->include_js_templates();
- }
- }
- /**
- * Enqueues all the assets needed for the primary term interface.
- *
- * @return void
- */
- public function enqueue_assets() {
- global $pagenow;
- if ( ! WPSEO_Metabox::is_post_edit( $pagenow ) ) {
- return;
- }
- $taxonomies = $this->get_primary_term_taxonomies();
- // Only enqueue if there are taxonomies that need a primary term.
- if ( empty( $taxonomies ) ) {
- return;
- }
- $asset_manager = new WPSEO_Admin_Asset_Manager();
- $asset_manager->enqueue_style( 'primary-category' );
- $asset_manager->enqueue_script( 'primary-category' );
- $mapped_taxonomies = $this->get_mapped_taxonomies_for_js( $taxonomies );
- $data = [
- 'taxonomies' => $mapped_taxonomies,
- ];
- wp_localize_script( WPSEO_Admin_Asset_Manager::PREFIX . 'primary-category', 'wpseoPrimaryCategoryL10n', $data );
- }
- /**
- * Saves all selected primary terms.
- *
- * @param int $post_id Post ID to save primary terms for.
- */
- public function save_primary_terms( $post_id ) {
- // Bail if this is a multisite installation and the site has been switched.
- if ( is_multisite() && ms_is_switched() ) {
- return;
- }
- $taxonomies = $this->get_primary_term_taxonomies( $post_id );
- foreach ( $taxonomies as $taxonomy ) {
- $this->save_primary_term( $post_id, $taxonomy );
- }
- }
- /**
- * Gets the id of the primary term.
- *
- * @param string $taxonomy_name Taxonomy name for the term.
- *
- * @return int primary term id
- */
- protected function get_primary_term( $taxonomy_name ) {
- $primary_term = new WPSEO_Primary_Term( $taxonomy_name, $this->get_current_id() );
- return $primary_term->get_primary_term();
- }
- /**
- * Returns all the taxonomies for which the primary term selection is enabled.
- *
- * @param int $post_id Default current post ID.
- * @return array
- */
- protected function get_primary_term_taxonomies( $post_id = null ) {
- if ( null === $post_id ) {
- $post_id = $this->get_current_id();
- }
- $taxonomies = wp_cache_get( 'primary_term_taxonomies_' . $post_id, 'wpseo' );
- if ( false !== $taxonomies ) {
- return $taxonomies;
- }
- $taxonomies = $this->generate_primary_term_taxonomies( $post_id );
- wp_cache_set( 'primary_term_taxonomies_' . $post_id, $taxonomies, 'wpseo' );
- return $taxonomies;
- }
- /**
- * Includes templates file.
- */
- protected function include_js_templates() {
- include_once WPSEO_PATH . 'admin/views/js-templates-primary-term.php';
- }
- /**
- * Saves the primary term for a specific taxonomy.
- *
- * @param int $post_id Post ID to save primary term for.
- * @param WP_Term $taxonomy Taxonomy to save primary term for.
- */
- protected function save_primary_term( $post_id, $taxonomy ) {
- $primary_term = filter_input( INPUT_POST, WPSEO_Meta::$form_prefix . 'primary_' . $taxonomy->name . '_term', FILTER_SANITIZE_NUMBER_INT );
- // We accept an empty string here because we need to save that if no terms are selected.
- if ( null !== $primary_term && check_admin_referer( 'save-primary-term', WPSEO_Meta::$form_prefix . 'primary_' . $taxonomy->name . '_nonce' ) ) {
- $primary_term_object = new WPSEO_Primary_Term( $taxonomy->name, $post_id );
- $primary_term_object->set_primary_term( $primary_term );
- }
- }
- /**
- * Generates the primary term taxonomies.
- *
- * @param int $post_id ID of the post.
- *
- * @return array
- */
- protected function generate_primary_term_taxonomies( $post_id ) {
- $post_type = get_post_type( $post_id );
- $all_taxonomies = get_object_taxonomies( $post_type, 'objects' );
- $all_taxonomies = array_filter( $all_taxonomies, [ $this, 'filter_hierarchical_taxonomies' ] );
- /**
- * Filters which taxonomies for which the user can choose the primary term.
- *
- * @api array $taxonomies An array of taxonomy objects that are primary_term enabled.
- *
- * @param string $post_type The post type for which to filter the taxonomies.
- * @param array $all_taxonomies All taxonomies for this post types, even ones that don't have primary term
- * enabled.
- */
- $taxonomies = (array) apply_filters( 'wpseo_primary_term_taxonomies', $all_taxonomies, $post_type, $all_taxonomies );
- return $taxonomies;
- }
- /**
- * Creates a map of taxonomies for localization.
- *
- * @param array $taxonomies The taxononmies that should be mapped.
- *
- * @return array The mapped taxonomies.
- */
- protected function get_mapped_taxonomies_for_js( $taxonomies ) {
- return array_map( [ $this, 'map_taxonomies_for_js' ], $taxonomies );
- }
- /**
- * Returns an array suitable for use in the javascript.
- *
- * @param stdClass $taxonomy The taxonomy to map.
- *
- * @return array The mapped taxonomy.
- */
- private function map_taxonomies_for_js( $taxonomy ) {
- $primary_term = $this->get_primary_term( $taxonomy->name );
- if ( empty( $primary_term ) ) {
- $primary_term = '';
- }
- $terms = get_terms( $taxonomy->name );
- return [
- 'title' => $taxonomy->labels->singular_name,
- 'name' => $taxonomy->name,
- 'primary' => $primary_term,
- 'singularLabel' => $taxonomy->labels->singular_name,
- 'fieldId' => $this->generate_field_id( $taxonomy->name ),
- 'restBase' => ( $taxonomy->rest_base ) ? $taxonomy->rest_base : $taxonomy->name,
- 'terms' => array_map( [ $this, 'map_terms_for_js' ], $terms ),
- ];
- }
- /**
- * Returns an array suitable for use in the javascript.
- *
- * @param stdClass $term The term to map.
- *
- * @return array The mapped terms.
- */
- private function map_terms_for_js( $term ) {
- return [
- 'id' => $term->term_id,
- 'name' => $term->name,
- ];
- }
- /**
- * Returns whether or not a taxonomy is hierarchical.
- *
- * @param stdClass $taxonomy Taxonomy object.
- *
- * @return bool
- */
- private function filter_hierarchical_taxonomies( $taxonomy ) {
- return (bool) $taxonomy->hierarchical;
- }
- }
|