etectionScripts() { global $wp_version; if ( version_compare( $wp_version, '6.2', '>=' ) ) { remove_action( 'admin_print_scripts', 'print_emoji_detection_script' ); } } /** * Resets the current user if bbPress is active. * We have to do this because our calls to wp_get_current_user() set the current user early and this breaks core functionality in bbPress. * * * @since 4.1.5 * * @return void */ public function resetUserBBPress() { if ( function_exists( 'bbpress' ) ) { global $current_user; $current_user = null; } } /** * Removes the bbPress title filter when adding a new reply with empty title to avoid fatal error. * * * @since 4.3.1 * * @param string $title The post title. * @param int $id The post ID (optional - in order to fix an issue where other plugins/themes don't pass in the second arg). * @return string The post title. */ public function maybeRemoveBBPressReplyFilter( $title, $id = 0 ) { if ( function_exists( 'bbp_get_reply_post_type' ) && get_post_type( $id ) === bbp_get_reply_post_type() && aioseo()->helpers->isScreenBase( 'post' ) ) { remove_filter( 'the_title', 'bbp_get_reply_title_fallback', 2 ); } return $title; } /** * Duplicates the model when duplicate post is triggered. * * @since 4.1.1 * * @param integer $targetPostId The target post ID. * @param \WP_Post $sourcePost The source post object. * @return void */ public function duplicatePost( $targetPostId, $sourcePost = null ) { $sourcePostId = ! empty( $sourcePost->ID ) ? $sourcePost->ID : $sourcePost; $sourceAioseoPost = Models\Post::getPost( $sourcePostId ); $targetPost = Models\Post::getPost( $targetPostId ); $columns = $sourceAioseoPost->getColumns(); foreach ( $columns as $column => $value ) { // Skip the ID column. if ( 'id' === $column ) { continue; } if ( 'post_id' === $column ) { $targetPost->$column = $targetPostId; continue; } $targetPost->$column = $sourceAioseoPost->$column; } $targetPost->save(); } /** * Duplicates the model when rewrite and republish is triggered. * * @since 4.3.4 * * @param integer $postId The post ID. * @param string $metaKey The meta key. * @param mixed $metaValue The meta value. * @return void */ public function rewriteAndRepublish( $postId, $metaKey = '', $metaValue = '' ) { if ( '_dp_has_rewrite_republish_copy' !== $metaKey ) { return; } $originalPost = aioseo()->helpers->getPost( $postId ); if ( ! is_object( $originalPost ) ) { return; } $this->duplicatePost( (int) $metaValue, $originalPost ); } /** * Updates the model when a post is republished. * Yoast Duplicate Post doesn't do this since we store our data in a custom table. * * @since 4.6.7 * * @param int $scheduledPostId The ID of the scheduled post. * @param int $originalPostId The ID of the original post. * @return void */ public function updateRescheduledPostMeta( $scheduledPostId, $originalPostId ) { $this->duplicatePost( $originalPostId, $scheduledPostId ); // Delete the AIOSEO post record for the scheduled post. $scheduledAioseoPost = Models\Post::getPost( $scheduledPostId ); $scheduledAioseoPost->delete(); } /** * Schedules an action to duplicate our meta after the duplicated WooCommerce product has been saved. * * @since 4.1.4 * * @param \WC_Product $newProduct The new, duplicated product. * @param \WC_Product $originalProduct The original product. * @return void */ public function scheduleDuplicateProduct( $newProduct, $originalProduct = null ) { self::$originalProductId = $originalProduct->get_id(); add_action( 'wp_insert_post', [ $this, 'duplicateProduct' ], 10, 2 ); } /** * Duplicates our meta for the new WooCommerce product. * * @since 4.1.4 * * @param integer $postId The new post ID. * @param \WP_Post $post The new post object. * @return void */ public function duplicateProduct( $postId, $post = null ) { if ( ! self::$originalProductId || 'product' !== $post->post_type ) { return; } $this->duplicatePost( $postId, self::$originalProductId ); } /** * Disable SEO inside the Genesis theme if it's running. * * @since 4.0.3 * * @param array $array An array of checks. * @return array An array with our function added. */ public function genesisTheme( $array ) { if ( empty( $array ) || ! isset( $array['functions'] ) ) { return $array; } $array['functions'][] = 'aioseo'; return $array; } /** * Remove XML from the GoDaddy CDN so our urls remain intact. * * @since 4.0.5 * * @param array $extensions The original extensions list. * @return array The extensions list without xml. */ public function goDaddySitemapXml( $extensions ) { $key = array_search( 'xml', $extensions, true ); unset( $extensions[ $key ] ); return $extensions; } /** * Registers our row meta for the plugins page. * * @since 4.0.0 * * @param array $actions List of existing actions. * @param string $pluginFile The plugin file. * @return array List of action links. */ abstract public function pluginRowMeta( $actions, $pluginFile = '' ); /** * Registers our action links for the plugins page. * * @since 4.0.0 * * @param array $actions List of existing actions. * @param string $pluginFile The plugin file. * @return array List of action links. */ abstract public function pluginActionLinks( $actions, $pluginFile = '' ); /** * Parses the action links. * * @since 4.0.0 * * @param array $actions The actions. * @param string $pluginFile The plugin file. * @param array $actionLinks The action links. * @param string $position The position. * @return array The parsed actions. */ protected function parseActionLinks( $actions, $pluginFile, $actionLinks = [], $position = 'after' ) { if ( empty( $this->plugin ) ) { $this->plugin = AIOSEO_PLUGIN_BASENAME; } if ( $this->plugin === $pluginFile && ! empty( $actionLinks ) ) { foreach ( $actionLinks as $key => $value ) { $link = [ $key => sprintf( '%3$s', esc_url( $value['url'] ), isset( $value['title'] ) ? 'title="' . esc_attr( $value['title'] ) . '"' : '', $value['label'] ) ]; $actions = 'after' === $position ? array_merge( $actions, $link ) : array_merge( $link, $actions ); } } return $actions; } /** * Add our routes to this plugins allow list. * * @since 4.1.4 * * @param array $allowList The original list. * @return array The modified list. */ public function allowRestRoutes( $allowList ) { return array_merge( $allowList, [ '/aioseo/' ] ); } /** * Clear the site authors cache when user is updated or registered. * * @since 4.1.8 * * @return void */ public function clearAuthorsCache() { aioseo()->core->cache->delete( 'site_authors' ); } /** * Filters out post types that aren't really public when getPublicPostTypes() is called. * * @since 4.1.9 * * @param array[object]|array[string] $postTypes The post types. * @return array[object]|array[string] The filtered post types. */ public function removeInvalidPublicPostTypes( $postTypes ) { $postTypesToRemove = [ 'fusion_element', // Avada 'elementor_library', 'redirect_rule', // Safe Redirect Manager 'seedprod', 'tcb_lightbox' ]; foreach ( $postTypes as $index => $postType ) { if ( is_string( $postType ) && in_array( $postType, $postTypesToRemove, true ) ) { unset( $postTypes[ $index ] ); continue; } if ( is_array( $postType ) && in_array( $postType['name'], $postTypesToRemove, true ) ) { unset( $postTypes[ $index ] ); } } return array_values( $postTypes ); } /** * Filters out taxonomies that aren't really public when getPublicTaxonomies() is called. * * @since 4.2.4 * * @param array[object]|array[string] $taxonomies The taxonomies. * @return array[object]|array[string] The filtered taxonomies. */ public function removeInvalidPublicTaxonomies( $taxonomies ) { // Check if the Avada Builder plugin is enabled. if ( ! defined( 'FUSION_BUILDER_VERSION' ) ) { return $taxonomies; } $taxonomiesToRemove = [ 'fusion_tb_category', 'element_category', 'template_category' ]; foreach ( $taxonomies as $index => $taxonomy ) { if ( is_string( $taxonomy ) && in_array( $taxonomy, $taxonomiesToRemove, true ) ) { unset( $taxonomies[ $index ] ); continue; } if ( is_array( $taxonomy ) && in_array( $taxonomy['name'], $taxonomiesToRemove, true ) ) { unset( $taxonomies[ $index ] ); } } return array_values( $taxonomies ); } /** * Disable Jetpack sitemaps module. * * @since 4.2.2 */ public function disableJetpackSitemaps( $active ) { unset( $active['sitemaps'] ); return $active; } /** * Dequeues third-party scripts from the other plugins or themes that crashes our menu pages. * * @since 4.1.9 * @version 4.3.1 * * @return void */ public function dequeueThirdPartyAssets() { // TagDiv Opt-in Builder plugin. wp_dequeue_script( 'tds_js_vue_files_last' ); // MyListing theme. if ( function_exists( 'mylisting' ) ) { wp_dequeue_script( 'vuejs' ); wp_dequeue_script( 'theme-script-vendor' ); wp_dequeue_script( 'theme-script-main' ); } // Voxel theme. if ( class_exists( '\Voxel\Controllers\Assets_Controller' ) ) { wp_dequeue_script( 'vue' ); wp_dequeue_script( 'vx:backend.js' ); } // Meta tags for seo plugin. if ( class_exists( '\Pagup\MetaTags\Settings' ) ) { wp_dequeue_script( 'pmt__vuejs' ); wp_dequeue_script( 'pmt__script' ); } // Plugin: Wpbingo Core (By TungHV). if ( strpos( wp_styles()->query( 'bwp-lookbook-css' )->src ?? '', 'wpbingo' ) !== false ) { wp_dequeue_style( 'bwp-lookbook-css' ); } } /** * Dequeues third-party scripts from the other plugins or themes that crashes our menu pages. * * @version 4.3.2 * * @return void */ public function dequeueThirdPartyAssetsEarly() { // Disables scripts for plugins StmMotorsExtends and StmPostType. if ( class_exists( 'STM_Metaboxes' ) ) { remove_action( 'admin_enqueue_scripts', [ 'STM_Metaboxes', 'wpcfto_scripts' ] ); } // Disables scripts for LearnPress plugin. if ( function_exists( 'learn_press_admin_assets' ) ) { remove_action( 'admin_enqueue_scripts', [ learn_press_admin_assets(), 'load_scripts' ] ); } } /** * Removes the duplicate meta description tag from the Hello Elementor theme. * * @since 4.4.3 * * @link https://developers.elementor.com/docs/hello-elementor-theme/hello_elementor_add_description_meta_tag/ * * @return void */ public function removeHelloElementorDescriptionTag() { remove_action( 'wp_head', 'hello_elementor_add_description_meta_tag' ); } /** * Removes the Avada OG tags. * * @since 4.6.5 * * @return void */ public function removeAvadaOgTags() { if ( function_exists( 'Avada' ) ) { $avada = Avada(); if ( is_object( $avada->head ?? null ) ) { remove_action( 'wp_head', [ $avada->head, 'insert_og_meta' ], 5 ); } } } /** * Prevent WP-Optimize from deleting our tables. * * @since 4.4.5 * * @param array $tables List of tables. * @return array Filtered tables. */ public function wpOptimizeAioseoTables( $tables ) { foreach ( $tables as &$table ) { if ( is_object( $table ) && property_exists( $table, 'Name' ) && false !== stripos( $table->Name, 'aioseo_' ) ) { $table->is_using = true; $table->can_be_removed = false; } } return $tables; } }