menu = $menu; } /** * Registers all hooks to WordPress. * * @return void */ public function register_hooks() { // Needs the lower than default priority so other plugins can hook underneath it without issue. add_action( 'admin_menu', array( $this, 'register_settings_page' ), 5 ); } /** * Registers the menu item submenus. */ public function register_settings_page() { $can_manage_options = WPSEO_Capability_Utils::current_user_can( $this->get_manage_capability() ); if ( $can_manage_options ) { /* * The current user has the capability to control anything. * This means that all submenus and dashboard can be shown. */ global $admin_page_hooks; add_menu_page( 'Yoast SEO: ' . __( 'Dashboard', 'wordpress-seo' ), __( 'SEO', 'wordpress-seo' ) . ' ' . $this->get_notification_counter(), $this->get_manage_capability(), $this->menu->get_page_identifier(), $this->get_admin_page_callback(), WPSEO_Utils::get_icon_svg(), '99.31337' ); $admin_page_hooks[ $this->menu->get_page_identifier() ] = 'seo'; // Wipe notification bits from hooks. R. } // Get all submenu pages. $submenu_pages = $this->get_submenu_pages(); // Add submenu items to the main menu if possible. if ( $can_manage_options ) { $this->register_submenu_pages( $submenu_pages ); } /* * If the user does not have the general manage options capability, * we need to make sure the desired sub-item can be reached. */ if ( ! $can_manage_options ) { $this->register_menu_pages( $submenu_pages ); } } /** * Registers submenu pages as menu pages. * * @param array $submenu_pages List of submenu pages. */ protected function register_menu_pages( $submenu_pages ) { if ( ! is_array( $submenu_pages ) || $submenu_pages === array() ) { return; } // Loop through submenu pages and add them. foreach ( $submenu_pages as $submenu_page ) { if ( $submenu_page[3] === $this->get_manage_capability() ) { continue; } // Register submenu as menu page. add_menu_page( 'Yoast SEO: ' . $submenu_page[2], $submenu_page[2], $submenu_page[3], $submenu_page[4], $submenu_page[5], WPSEO_Utils::get_icon_svg(), '99.31337' ); } } /** * Returns the list of registered submenu pages. * * @return array List of registered submenu pages. */ protected function get_submenu_pages() { global $wpseo_admin; /** WPSEO_Admin $wpseo_admin */ $admin_features = $wpseo_admin->get_admin_features(); // Submenu pages. $submenu_pages = array( $this->get_submenu_page( __( 'General', 'wordpress-seo' ), $this->menu->get_page_identifier() ), $this->get_submenu_page( __( 'Titles & Metas', 'wordpress-seo' ), 'wpseo_titles' ), $this->get_submenu_page( __( 'Social', 'wordpress-seo' ), 'wpseo_social' ), $this->get_submenu_page( __( 'XML Sitemaps', 'wordpress-seo' ), 'wpseo_xml' ), $this->get_submenu_page( __( 'Advanced', 'wordpress-seo' ), 'wpseo_advanced' ), $this->get_submenu_page( __( 'Tools', 'wordpress-seo' ), 'wpseo_tools' ), $this->get_submenu_page( __( 'Search Console', 'wordpress-seo' ), 'wpseo_search_console', array( $admin_features['google_search_console'], 'display' ), array( array( $admin_features['google_search_console'], 'set_help' ) ) ), $this->get_submenu_page( $this->get_license_page_title(), 'wpseo_licenses' ), ); /** * Filter: 'wpseo_submenu_pages' - Collects all submenus that need to be shown. * * @api array $submenu_pages List with all submenu pages. */ return (array) apply_filters( 'wpseo_submenu_pages', $submenu_pages ); } /** * Creates a submenu formatted array. * * @param string $page_title Page title to use. * @param string $page_slug Page slug to use. * @param callable $callback Optional. Callback which handles the page request. * @param callable[] $hook Optional. Hook to trigger when the page is registered. * * @return array Formatted submenu. */ protected function get_submenu_page( $page_title, $page_slug, $callback = null, $hook = null ) { if ( $callback === null ) { $callback = $this->get_admin_page_callback(); } return array( $this->menu->get_page_identifier(), '', $page_title, $this->get_manage_capability(), $page_slug, $callback, $hook, ); } /** * Registers the submenu pages. * * This is only done when the user has the `wpseo_manage_options` capability, * thus all capabilities can be set to this capability. * * @param array $submenu_pages List of submenu pages to register. * * @return void */ protected function register_submenu_pages( $submenu_pages ) { if ( ! is_array( $submenu_pages ) || $submenu_pages === array() ) { return; } // Loop through submenu pages and add them. foreach ( $submenu_pages as $submenu_page ) { $page_title = $submenu_page[2]; // We cannot use $submenu_page[1] because add-ons define that, so hard-code this value. if ( $submenu_page[4] === 'wpseo_licenses' ) { $page_title = $this->get_license_page_title(); } $page_title .= ' - Yoast SEO'; /* * Add submenu page. * * If we don't register this on `wpseo_manage_options`, admin users with only this capability * will not be able to see the submenus which are configured with something else, * thus all submenu pages are registered with the `wpseo_manage_options` capability here. */ $admin_page = add_submenu_page( $submenu_page[0], $page_title, $submenu_page[2], $this->get_manage_capability(), $submenu_page[4], $submenu_page[5] ); // Check if we need to hook. if ( isset( $submenu_page[6] ) && ( is_array( $submenu_page[6] ) && $submenu_page[6] !== array() ) ) { foreach ( $submenu_page[6] as $submenu_page_action ) { add_action( 'load-' . $admin_page, $submenu_page_action ); } } } // Use WordPress global $submenu to directly access it's properties. global $submenu; if ( isset( $submenu[ $this->menu->get_page_identifier() ] ) && WPSEO_Capability_Utils::current_user_can( $this->get_manage_capability() ) ) { $submenu[ $this->menu->get_page_identifier() ][0][0] = __( 'Dashboard', 'wordpress-seo' ); } } /** * Returns the notification count in HTML format. * * @return string The notification count in HTML format. */ protected function get_notification_counter() { $notification_center = Yoast_Notification_Center::get(); $notification_count = $notification_center->get_notification_count(); // Add main page. /* translators: %s: number of notifications */ $notifications = sprintf( _n( '%s notification', '%s notifications', $notification_count, 'wordpress-seo' ), number_format_i18n( $notification_count ) ); $counter = sprintf( '%2$s', $notification_count, $notifications ); return $counter; } /** * Returns the capability that is required to manage all options. * * @return string Capability to check against. */ protected function get_manage_capability() { /** * Filter: 'wpseo_manage_options_capability' - Allow changing the capability users need to view the settings pages * * @deprecated 5.5 * @api string unsigned The capability */ return apply_filters_deprecated( 'wpseo_manage_options_capability', array( 'wpseo_manage_options' ), 'WPSEO 5.5.0', false, 'Use the introduced wpseo_manage_options capability instead.' ); } /** * Returns the page handler callback. * * @return array Callback page handler. */ protected function get_admin_page_callback() { return array( $this->menu, 'load_page' ); } /** * Returns the page title to use for the licenses page. * * @return string The title for the license page. */ protected function get_license_page_title() { static $title = null; if ( $title === null ) { $title = __( 'Premium', 'wordpress-seo' ); } return $title; } }