strpos( $name, 'display_settings_page_' ) === 0 ) { return $this->display_settings_page( $this->substr( $name, 22 ) ); } $error = sprintf( __( "Method %s doesn't exist", 'all-in-one-seo-pack' ), $name ); if ( class_exists( 'BadMethodCallException' ) ) { throw new BadMethodCallException( $error ); } throw new Exception( $error ); } /** * All_in_One_SEO_Pack_Module constructor. */ function __construct() { if ( empty( $this->file ) ) { $this->file = __FILE__; } $this->plugin_name = AIOSEOP_PLUGIN_NAME; $this->plugin_path = array(); // $this->plugin_path['dir'] = plugin_dir_path( $this->file ); $this->plugin_path['basename'] = plugin_basename( $this->file ); $this->plugin_path['dirname'] = dirname( $this->plugin_path['basename'] ); $this->plugin_path['url'] = plugin_dir_url( $this->file ); $this->plugin_path['images_url'] = $this->plugin_path['url'] . 'images'; $this->script_data['plugin_path'] = $this->plugin_path; } /** * Get options for module, stored individually or together. */ function get_class_option() { $option_name = $this->get_option_name(); if ( $this->store_option || $option_name == $this->parent_option ) { return get_option( $option_name ); } else { $option = get_option( $this->parent_option ); if ( isset( $option['modules'] ) && isset( $option['modules'][ $option_name ] ) ) { return $option['modules'][ $option_name ]; } } return false; } /** * Update options for module, stored individually or together. * * @param $option_data * @param bool $option_name * * @return bool */ function update_class_option( $option_data, $option_name = false ) { if ( $option_name == false ) { $option_name = $this->get_option_name(); } if ( $this->store_option || $option_name == $this->parent_option ) { return update_option( $option_name, $option_data ); } else { $option = get_option( $this->parent_option ); if ( ! isset( $option['modules'] ) ) { $option['modules'] = array(); } $option['modules'][ $option_name ] = $option_data; return update_option( $this->parent_option, $option ); } } /** * Delete options for module, stored individually or together. * * @param bool $delete * * @return bool */ function delete_class_option( $delete = false ) { $option_name = $this->get_option_name(); if ( $this->store_option || $delete ) { delete_option( $option_name ); } else { $option = get_option( $this->parent_option ); if ( isset( $option['modules'] ) && isset( $option['modules'][ $option_name ] ) ) { unset( $option['modules'][ $option_name ] ); return update_option( $this->parent_option, $option ); } } return false; } /** * Get the option name with prefix. */ function get_option_name() { if ( ! isset( $this->option_name ) || empty( $this->option_name ) ) { $this->option_name = $this->prefix . 'options'; } return $this->option_name; } /** * Convenience function to see if an option is set. * * @param string $option * * @param null $location * * @return bool */ function option_isset( $option, $location = null ) { $prefix = $this->get_prefix( $location ); $opt = $prefix . $option; return ( isset( $this->options[ $opt ] ) && $this->options[ $opt ] ); } /** * Case conversion; handle non UTF-8 encodings and fallback ** * * @param $str * @param string $mode * * @return string */ function convert_case( $str, $mode = 'upper' ) { static $charset = null; if ( $charset == null ) { $charset = get_bloginfo( 'charset' ); } $str = (string) $str; if ( $mode == 'title' ) { if ( function_exists( 'mb_convert_case' ) ) { return mb_convert_case( $str, MB_CASE_TITLE, $charset ); } else { return ucwords( $str ); } } if ( $charset == 'UTF-8' ) { global $UTF8_TABLES; include_once( AIOSEOP_PLUGIN_DIR . 'inc/aioseop_UTF8.php' ); if ( is_array( $UTF8_TABLES ) ) { if ( $mode == 'upper' ) { return strtr( $str, $UTF8_TABLES['strtoupper'] ); } if ( $mode == 'lower' ) { return strtr( $str, $UTF8_TABLES['strtolower'] ); } } } if ( $mode == 'upper' ) { if ( function_exists( 'mb_strtoupper' ) ) { return mb_strtoupper( $str, $charset ); } else { return strtoupper( $str ); } } if ( $mode == 'lower' ) { if ( function_exists( 'mb_strtolower' ) ) { return mb_strtolower( $str, $charset ); } else { return strtolower( $str ); } } return $str; } /** * Convert a string to lower case * Compatible with mb_strtolower(), an UTF-8 friendly replacement for strtolower() * * @param $str * * @return string */ function strtolower( $str ) { return $this->convert_case( $str, 'lower' ); } /** * Convert a string to upper case * Compatible with mb_strtoupper(), an UTF-8 friendly replacement for strtoupper() * * @param $str * * @return string */ function strtoupper( $str ) { return $this->convert_case( $str, 'upper' ); } /** * Convert a string to title case * Compatible with mb_convert_case(), an UTF-8 friendly replacement for ucwords() * * @param $str * * @return string */ function ucwords( $str ) { return $this->convert_case( $str, 'title' ); } /** * Wrapper for strlen() - uses mb_strlen() if possible. * * @param $string * * @return int */ function strlen( $string ) { if ( function_exists( 'mb_strlen' ) ) { return mb_strlen( $string ); } return strlen( $string ); } /** * Wrapper for substr() - uses mb_substr() if possible. * * @param $string * @param int $start * @param int $length * * @return mixed */ function substr( $string, $start = 0, $length = 2147483647 ) { $args = func_get_args(); if ( function_exists( 'mb_substr' ) ) { return call_user_func_array( 'mb_substr', $args ); } return call_user_func_array( 'substr', $args ); } /** * Wrapper for strpos() - uses mb_strpos() if possible. * * @param $haystack * @param string $needle * * @param int $offset * * @return bool|int */ function strpos( $haystack, $needle, $offset = 0 ) { if ( function_exists( 'mb_strpos' ) ) { return mb_strpos( $haystack, $needle, $offset ); } return strpos( $haystack, $needle, $offset ); } /** * Wrapper for strrpos() - uses mb_strrpos() if possible. * * @param $haystack * @param string $needle * * @param int $offset * * @return bool|int */ function strrpos( $haystack, $needle, $offset = 0 ) { if ( function_exists( 'mb_strrpos' ) ) { return mb_strrpos( $haystack, $needle, $offset ); } return strrpos( $haystack, $needle, $offset ); } /** * Convert html string to php array - useful to get a serializable value. * * @param string $xmlstr * * @return array */ function html_string_to_array( $htmlstr ) { if ( ! class_exists( 'DOMDocument' ) ) { return array(); } else { $doc = new DOMDocument(); $doc->loadXML( $htmlstr ); return $this->domnode_to_array( $doc->documentElement ); } } /** * @param DOMElement $node * * @return array|string */ function domnode_to_array( $node ) { switch ( $node->nodeType ) { case XML_CDATA_SECTION_NODE: case XML_TEXT_NODE: return trim( $node->textContent ); break; case XML_ELEMENT_NODE: $output = array(); for ( $i = 0, $m = $node->childNodes->length; $i < $m; $i ++ ) { $child = $node->childNodes->item( $i ); $v = $this->domnode_to_array( $child ); if ( isset( $child->tagName ) ) { $t = $child->tagName; if ( ! isset( $output[ $t ] ) ) { $output[ $t ] = array(); } if ( is_array( $output ) ) { $output[ $t ][] = $v; } } elseif ( $v || $v === '0' ) { $output = (string) $v; } } if ( $node->attributes->length && ! is_array( $output ) ) { // Has attributes but isn't an array $output = array( '@content' => $output ); } //Change output into an array. if ( is_array( $output ) ) { if ( $node->attributes->length ) { $a = array(); foreach ( $node->attributes as $attrName => $attrNode ) { $a[ $attrName ] = (string) $attrNode->value; } $output['@attributes'] = $a; } foreach ( $output as $t => $v ) { if ( is_array( $v ) && count( $v ) == 1 && $t != '@attributes' ) { $output[ $t ] = $v[0]; } } } } if ( empty( $output ) ) { return ''; } return $output; } /** * adds support for using %cf_(name of field)% for using custom fields / Advanced Custom Fields in titles / descriptions etc. ** * * @param $format * * @return mixed */ function apply_cf_fields( $format ) { return preg_replace_callback( '/%cf_([^%]*?)%/', array( $this, 'cf_field_replace' ), $format ); } /** * @param $matches * * @return bool|mixed|string */ function cf_field_replace( $matches ) { $result = ''; if ( ! empty( $matches ) ) { if ( ! empty( $matches[1] ) ) { if ( function_exists( 'get_field' ) ) { $result = get_field( $matches[1] ); } if ( empty( $result ) ) { global $post; if ( ! empty( $post ) ) { $result = get_post_meta( $post->ID, $matches[1], true ); } } } else { $result = $matches[0]; } } $result = strip_tags( $result ); return $result; } /** * Returns child blogs of parent in a multisite. */ function get_child_blogs() { global $wpdb, $blog_id; $site_id = $wpdb->siteid; if ( is_multisite() ) { if ( $site_id != $blog_id ) { return false; } // @codingStandardsIgnoreStart return $wpdb->get_col( "SELECT blog_id FROM {$wpdb->blogs} WHERE site_id = {$blog_id} AND site_id != blog_id" ); // @codingStandardsIgnoreEnd } return false; } /** * Checks if the plugin is active on a given blog by blogid on a multisite. * * @param bool $bid * * @return bool */ function is_aioseop_active_on_blog( $bid = false ) { global $blog_id; if ( empty( $bid ) || ( $bid == $blog_id ) || ! is_multisite() ) { return true; } if ( ! function_exists( 'is_plugin_active_for_network' ) ) { require_once( ABSPATH . '/wp-admin/includes/plugin.php' ); } if ( is_plugin_active_for_network( AIOSEOP_PLUGIN_BASENAME ) ) { return true; } return in_array( AIOSEOP_PLUGIN_BASENAME, (array) get_blog_option( $bid, 'active_plugins', array() ) ); } /** * @param $list * @param string $quote * * @return string */ function quote_list_for_regex( $list, $quote = '/' ) { $regex = ''; $cont = 0; foreach ( $list as $l ) { $trim_l = trim( $l ); if ( ! empty( $trim_l ) ) { if ( $cont ) { $regex .= '|'; } $cont = 1; $regex .= preg_quote( trim( $l ), $quote ); } } return $regex; } // original code thanks to Sean M. Brown -- http://smbrown.wordpress.com/2009/04/29/verify-googlebot-forward-reverse-dns/ /** * @return bool */ function is_good_bot() { $botlist = array( 'Yahoo! Slurp' => 'crawl.yahoo.net', 'googlebot' => '.googlebot.com', 'msnbot' => 'search.msn.com', ); $botlist = apply_filters( $this->prefix . 'botlist', $botlist ); if ( ! empty( $botlist ) ) { if ( ! isset( $_SERVER['HTTP_USER_AGENT'] ) ) { return false; } $ua = $_SERVER['HTTP_USER_AGENT']; $uas = $this->quote_list_for_regex( $botlist ); if ( preg_match( '/' . $uas . '/i', $ua ) ) { $ip = $_SERVER['REMOTE_ADDR']; $hostname = gethostbyaddr( $ip ); $ip_by_hostname = gethostbyname( $hostname ); if ( $ip_by_hostname == $ip ) { $hosts = array_values( $botlist ); foreach ( $hosts as $k => $h ) { $hosts[ $k ] = preg_quote( $h ) . '$'; } $hosts = join( '|', $hosts ); if ( preg_match( '/' . $hosts . '/i', $hostname ) ) { return true; } } } return false; } } /** * @return array */ function default_bad_bots() { $botlist = array( 'Abonti', 'aggregator', 'AhrefsBot', 'asterias', 'BDCbot', 'BLEXBot', 'BuiltBotTough', 'Bullseye', 'BunnySlippers', 'ca-crawler', 'CCBot', 'Cegbfeieh', 'CheeseBot', 'CherryPicker', 'CopyRightCheck', 'cosmos', 'Crescent', 'discobot', 'DittoSpyder', 'DotBot', 'Download Ninja', 'EasouSpider', 'EmailCollector', 'EmailSiphon', 'EmailWolf', 'EroCrawler', 'Exabot', 'ExtractorPro', 'Fasterfox', 'FeedBooster', 'Foobot', 'Genieo', 'grub-client', 'Harvest', 'hloader', 'httplib', 'HTTrack', 'humanlinks', 'ieautodiscovery', 'InfoNaviRobot', 'IstellaBot', 'Java/1.', 'JennyBot', 'k2spider', 'Kenjin Spider', 'Keyword Density/0.9', 'larbin', 'LexiBot', 'libWeb', 'libwww', 'LinkextractorPro', 'linko', 'LinkScan/8.1a Unix', 'LinkWalker', 'LNSpiderguy', 'lwp-trivial', 'magpie', 'Mata Hari', 'MaxPointCrawler', 'MegaIndex', 'Microsoft URL Control', 'MIIxpc', 'Mippin', 'Missigua Locator', 'Mister PiX', 'MJ12bot', 'moget', 'MSIECrawler', 'NetAnts', 'NICErsPRO', 'Niki-Bot', 'NPBot', 'Nutch', 'Offline Explorer', 'Openfind', 'panscient.com', 'PHP/5.{', 'ProPowerBot/2.14', 'ProWebWalker', 'Python-urllib', 'QueryN Metasearch', 'RepoMonkey', 'SemrushBot', 'SISTRIX', 'sitecheck.Internetseer.com', 'SiteSnagger', 'SnapPreviewBot', 'Sogou', 'SpankBot', 'spanner', 'spbot', 'Spinn3r', 'suzuran', 'Szukacz/1.4', 'Teleport', 'Telesoft', 'The Intraformant', 'TheNomad', 'TightTwatBot', 'Titan', 'toCrawl/UrlDispatcher', 'True_Robot', 'turingos', 'TurnitinBot', 'UbiCrawler', 'UnisterBot', 'URLy Warning', 'VCI', 'WBSearchBot', 'Web Downloader/6.9', 'Web Image Collector', 'WebAuto', 'WebBandit', 'WebCopier', 'WebEnhancer', 'WebmasterWorldForumBot', 'WebReaper', 'WebSauger', 'Website Quester', 'Webster Pro', 'WebStripper', 'WebZip', 'Wotbox', 'wsr-agent', 'WWW-Collector-E', 'Xenu', 'Zao', 'Zeus', 'ZyBORG', 'coccoc', 'Incutio', 'lmspider', 'memoryBot', 'SemrushBot', 'serf', 'Unknown', 'uptime files', ); return $botlist; } /** * @return bool */ function is_bad_bot() { $botlist = $this->default_bad_bots(); $botlist = apply_filters( $this->prefix . 'badbotlist', $botlist ); if ( ! empty( $botlist ) ) { if ( ! isset( $_SERVER['HTTP_USER_AGENT'] ) ) { return false; } $ua = $_SERVER['HTTP_USER_AGENT']; $uas = $this->quote_list_for_regex( $botlist ); if ( preg_match( '/' . $uas . '/i', $ua ) ) { return true; } } return false; } /** * @return array */ function default_bad_referers() { $referlist = array( 'semalt.com', 'kambasoft.com', 'savetubevideo.com', 'buttons-for-website.com', 'sharebutton.net', 'soundfrost.org', 'srecorder.com', 'softomix.com', 'softomix.net', 'myprintscreen.com', 'joinandplay.me', 'fbfreegifts.com', 'openmediasoft.com', 'zazagames.org', 'extener.org', 'openfrost.com', 'openfrost.net', 'googlsucks.com', 'best-seo-offer.com', 'buttons-for-your-website.com', 'www.Get-Free-Traffic-Now.com', 'best-seo-solution.com', 'buy-cheap-online.info', 'site3.free-share-buttons.com', 'webmaster-traffic.com', ); return $referlist; } /** * @return bool */ function is_bad_referer() { $referlist = $this->default_bad_referers(); $referlist = apply_filters( $this->prefix . 'badreferlist', $referlist ); if ( ! empty( $referlist ) && ! empty( $_SERVER ) && ! empty( $_SERVER['HTTP_REFERER'] ) ) { $ref = $_SERVER['HTTP_REFERER']; $regex = $this->quote_list_for_regex( $referlist ); if ( preg_match( '/' . $regex . '/i', $ref ) ) { return true; } } return false; } /** * @return mixed|void */ function allow_bot() { $allow_bot = true; if ( ( ! $this->is_good_bot() ) && $this->is_bad_bot() && ! is_user_logged_in() ) { $allow_bot = false; } return apply_filters( $this->prefix . 'allow_bot', $allow_bot ); } /** * Displays tabs for tabbed locations on a settings page. * * @param $location */ function display_tabs( $location ) { if ( ( $location != null ) && isset( $locations[ $location ]['tabs'] ) ) { $tabs = $locations['location']['tabs']; } else { $tabs = $this->tabs; } if ( ! empty( $tabs ) ) { ?>
label ) ) { $post_types[ $p ] = $post_objs[ $p ]->label; } else { $post_types[ $p ] = $p; } } return $post_types; } /** * @param $post_objs * * @return array */ function get_term_labels( $post_objs ) { $post_types = array(); foreach ( $post_objs as $p ) { if ( ! empty( $p->name ) ) { $post_types[ $p->term_id ] = $p->name; } } return $post_types; } /** * @param array $args * * @return array */ function get_post_type_titles( $args = array() ) { return $this->get_object_labels( get_post_types( $args, 'objects' ) ); } /** * @param array $args * * @return array */ function get_taxonomy_titles( $args = array() ) { return $this->get_object_labels( get_taxonomies( $args, 'objects' ) ); } /** * @param array $args * * @return array */ function get_category_titles( $args = array() ) { return $this->get_term_labels( get_categories( $args ) ); } /** * Helper function for exporting settings on post data. * * @param string $prefix * @param array $query * * @return string */ function post_data_export( $prefix = '_aioseop', $query = array( 'posts_per_page' => - 1 ) ) { $buf = ''; $posts_query = new WP_Query( $query ); while ( $posts_query->have_posts() ) { $posts_query->the_post(); global $post; $guid = $post->guid; $type = $post->post_type; $title = $post->post_title; $date = $post->post_date; $data = ''; $post_custom_fields = get_post_custom( $post->ID ); $has_data = null; if ( is_array( $post_custom_fields ) ) { foreach ( $post_custom_fields as $field_name => $field ) { if ( ( $this->strpos( $field_name, $prefix ) === 0 ) && $field[0] ) { $has_data = true; $data .= $field_name . " = '" . $field[0] . "'\n"; } } } if ( ! empty( $data ) ) { $has_data = true; } if ( $has_data != null ) { $post_info = "\n[post_data]\n\n"; $post_info .= "post_title = '" . $title . "'\n"; $post_info .= "post_guid = '" . $guid . "'\n"; $post_info .= "post_date = '" . $date . "'\n"; $post_info .= "post_type = '" . $type . "'\n"; if ( $data ) { $buf .= $post_info . $data . "\n"; } } } wp_reset_postdata(); return $buf; } /** * Handles exporting settings data for a module. * * @since 2.4.13 Fixed bug on empty options. * * @param $buf * * @return string */ function settings_export( $buf ) { global $aiosp; $post_types = apply_filters( 'aioseop_export_settings_exporter_post_types', null ); $has_data = null; $general_settings = null; $exporter_choices = apply_filters( 'aioseop_export_settings_exporter_choices', '' ); if ( ! empty( $_REQUEST['aiosp_importer_exporter_export_choices'] ) ) { $exporter_choices = $_REQUEST['aiosp_importer_exporter_export_choices']; } if ( ! empty( $exporter_choices ) && is_array( $exporter_choices ) ) { foreach ( $exporter_choices as $ex ) { if ( $ex == 1 ) { $general_settings = true; } if ( $ex == 2 && isset( $_REQUEST['aiosp_importer_exporter_export_post_types'] ) ) { $post_types = $_REQUEST['aiosp_importer_exporter_export_post_types']; } } } if ( ( $post_types != null ) && ( $this === $aiosp ) ) { $buf .= $this->post_data_export( '_aioseop', array( 'posts_per_page' => - 1, 'post_type' => $post_types, 'post_status' => array( 'publish', 'pending', 'draft', 'future', 'private', 'inherit' ), ) ); } /* Add all active settings to settings file */ $name = $this->get_option_name(); $options = $this->get_class_option(); if ( ! empty( $options ) && $general_settings != null ) { $buf .= "\n[$name]\n\n"; foreach ( $options as $key => $value ) { if ( ( $name == $this->parent_option ) && ( $key == 'modules' ) ) { continue; } // don't re-export all module settings -- pdb if ( is_array( $value ) ) { $value = "'" . str_replace( array( "'", "\n", "\r" ), array( "\'", '\n', '\r', ), trim( serialize( $value ) ) ) . "'"; } else { $value = str_replace( array( "\n", "\r" ), array( '\n', '\r', ), trim( var_export( $value, true ) ) ); } $buf .= "$key = $value\n"; } } return $buf; } /** * Order for adding the menus for the aioseop_modules_add_menus hook. */ function menu_order() { return 10; } /** * Print a basic error message. * * @param $error * * @return bool */ function output_error( $error ) { echo "
$error
"; return false; } /** * * Backwards compatibility - see http://php.net/manual/en/function.str-getcsv.php * * @param $input * @param string $delimiter * @param string $enclosure * @param string $escape * * @return array */ function str_getcsv( $input, $delimiter = ',', $enclosure = '"', $escape = '\\' ) { $fp = fopen( 'php://memory', 'r+' ); fputs( $fp, $input ); rewind( $fp ); $data = fgetcsv( $fp, null, $delimiter, $enclosure ); // $escape only got added in 5.3.0 fclose( $fp ); return $data; } /** * * Helper function to convert csv in key/value pair format to an associative array. * * @param $csv * * @return array */ function csv_to_array( $csv ) { $args = array(); if ( ! function_exists( 'str_getcsv' ) ) { $v = $this->str_getcsv( $csv ); } else { $v = str_getcsv( $csv ); } $size = count( $v ); if ( is_array( $v ) && isset( $v[0] ) && $size >= 2 ) { for ( $i = 0; $i < $size; $i += 2 ) { $args[ $v[ $i ] ] = $v[ $i + 1 ]; } } return $args; } /** Allow modules to use WP Filesystem if available and desired, fall back to PHP filesystem access otherwise. * * @param string $method * @param bool $form_fields * @param string $url * @param bool $error * * @return bool */ function use_wp_filesystem( $method = '', $form_fields = false, $url = '', $error = false ) { if ( empty( $method ) ) { $this->credentials = request_filesystem_credentials( $url ); } else { $this->credentials = request_filesystem_credentials( $url, $method, $error, false, $form_fields ); } return $this->credentials; } /** * Wrapper function to get filesystem object. */ function get_filesystem_object() { $cred = get_transient( 'aioseop_fs_credentials' ); if ( ! empty( $cred ) ) { $this->credentials = $cred; } if ( function_exists( 'WP_Filesystem' ) && WP_Filesystem( $this->credentials ) ) { global $wp_filesystem; return $wp_filesystem; } else { require_once( ABSPATH . 'wp-admin/includes/template.php' ); require_once( ABSPATH . 'wp-admin/includes/screen.php' ); require_once( ABSPATH . 'wp-admin/includes/file.php' ); if ( ! WP_Filesystem( $this->credentials ) ) { $this->use_wp_filesystem(); } if ( ! empty( $this->credentials ) ) { set_transient( 'aioseop_fs_credentials', $this->credentials, 10800 ); } global $wp_filesystem; if ( is_object( $wp_filesystem ) ) { return $wp_filesystem; } } return false; } /** * See if a file exists using WP Filesystem. * * @param string $filename * * @return bool */ function file_exists( $filename ) { $wpfs = $this->get_filesystem_object(); if ( is_object( $wpfs ) ) { return $wpfs->exists( $filename ); } return $wpfs; } /** * See if the directory entry is a file using WP Filesystem. * * @param $filename * * @return bool */ function is_file( $filename ) { $wpfs = $this->get_filesystem_object(); if ( is_object( $wpfs ) ) { return $wpfs->is_file( $filename ); } return $wpfs; } /** * List files in a directory using WP Filesystem. * * @param $path * * @return array|bool */ function scandir( $path ) { $wpfs = $this->get_filesystem_object(); if ( is_object( $wpfs ) ) { $dirlist = $wpfs->dirlist( $path ); if ( empty( $dirlist ) ) { return $dirlist; } return array_keys( $dirlist ); } return $wpfs; } /** * Load a file through WP Filesystem; implement basic support for offset and maxlen. * * @param $filename * @param bool $use_include_path * @param null $context * @param int $offset * @param int $maxlen * * @return bool|mixed */ function load_file( $filename, $use_include_path = false, $context = null, $offset = - 1, $maxlen = - 1 ) { $wpfs = $this->get_filesystem_object(); if ( is_object( $wpfs ) ) { if ( ! $wpfs->exists( $filename ) ) { return false; } if ( ( $offset > 0 ) || ( $maxlen >= 0 ) ) { if ( $maxlen === 0 ) { return ''; } if ( $offset < 0 ) { $offset = 0; } $file = $wpfs->get_contents( $filename ); if ( ! is_string( $file ) || empty( $file ) ) { return $file; } if ( $maxlen < 0 ) { return $this->substr( $file, $offset ); } else { return $this->substr( $file, $offset, $maxlen ); } } else { return $wpfs->get_contents( $filename ); } } return false; } /** * Save a file through WP Filesystem. * * @param string $filename * * @param $contents * * @return bool */ function save_file( $filename, $contents ) { $failed_str = sprintf( __( "Failed to write file %s!\n", 'all-in-one-seo-pack' ), $filename ); $readonly_str = sprintf( __( "File %s isn't writable!\n", 'all-in-one-seo-pack' ), $filename ); $wpfs = $this->get_filesystem_object(); if ( is_object( $wpfs ) ) { $file_exists = $wpfs->exists( $filename ); if ( ! $file_exists || $wpfs->is_writable( $filename ) ) { if ( $wpfs->put_contents( $filename, $contents ) === false ) { return $this->output_error( $failed_str ); } } else { return $this->output_error( $readonly_str ); } return true; } return false; } /** * Delete a file through WP Filesystem. * * @param string $filename * * @return bool */ function delete_file( $filename ) { $wpfs = $this->get_filesystem_object(); if ( is_object( $wpfs ) ) { if ( $wpfs->exists( $filename ) ) { if ( $wpfs->delete( $filename ) === false ) { $this->output_error( sprintf( __( "Failed to delete file %s!\n", 'all-in-one-seo-pack' ), $filename ) ); } else { return true; } } else { $this->output_error( sprintf( __( "File %s doesn't exist!\n", 'all-in-one-seo-pack' ), $filename ) ); } } return false; } /** * Rename a file through WP Filesystem. * * @param string $filename * @param string $newname * * @return bool */ function rename_file( $filename, $newname ) { $wpfs = $this->get_filesystem_object(); if ( is_object( $wpfs ) ) { $file_exists = $wpfs->exists( $filename ); $newfile_exists = $wpfs->exists( $newname ); if ( $file_exists && ! $newfile_exists ) { if ( $wpfs->move( $filename, $newname ) === false ) { $this->output_error( sprintf( __( "Failed to rename file %s!\n", 'all-in-one-seo-pack' ), $filename ) ); } else { return true; } } else { if ( ! $file_exists ) { $this->output_error( sprintf( __( "File %s doesn't exist!\n", 'all-in-one-seo-pack' ), $filename ) ); } elseif ( $newfile_exists ) { $this->output_error( sprintf( __( "File %s already exists!\n", 'all-in-one-seo-pack' ), $newname ) ); } } } return false; } /** * Load multiple files. * * @param $options * @param $opts * @param $prefix * * @return mixed */ function load_files( $options, $opts, $prefix ) { foreach ( $opts as $opt => $file ) { $opt = $prefix . $opt; $file = ABSPATH . $file; $contents = $this->load_file( $file ); if ( $contents !== false ) { $options[ $opt ] = $contents; } } return $options; } /** * Save multiple files. * * @param $opts * @param $prefix */ function save_files( $opts, $prefix ) { foreach ( $opts as $opt => $file ) { $opt = $prefix . $opt; if ( isset( $_POST[ $opt ] ) ) { $output = stripslashes_deep( $_POST[ $opt ] ); $file = ABSPATH . $file; $this->save_file( $file, $output ); } } } /** * Delete multiple files. * * @param $opts */ function delete_files( $opts ) { foreach ( $opts as $opt => $file ) { $file = ABSPATH . $file; $this->delete_file( $file ); } } /** * Returns available social seo images. * * @since 2.4 #1079 Fixes array_flip warning on opengraph module. * * @param array $options Plugin/module options. * @param object $p Post. * * @return array */ function get_all_images_by_type( $options = null, $p = null ) { $img = array(); if ( empty( $img ) ) { $size = apply_filters( 'post_thumbnail_size', 'large' ); global $aioseop_options, $wp_query, $aioseop_opengraph; if ( $p === null ) { global $post; } else { $post = $p; } $count = 1; if ( ! empty( $post ) ) { if ( ! is_object( $post ) ) { $post = get_post( $post ); } if ( is_object( $post ) && function_exists( 'get_post_thumbnail_id' ) ) { if ( $post->post_type == 'attachment' ) { $post_thumbnail_id = $post->ID; } else { $post_thumbnail_id = get_post_thumbnail_id( $post->ID ); } if ( ! empty( $post_thumbnail_id ) ) { $image = wp_get_attachment_image_src( $post_thumbnail_id, $size ); if ( is_array( $image ) ) { $img[] = array( 'type' => 'featured', 'id' => $post_thumbnail_id, 'link' => $image[0] ); } } } $post_id = $post->ID; $p = $post; $w = $wp_query; $meta_key = ''; if ( is_array( $options ) && isset( $options['meta_key'] ) ) { $meta_key = $options['meta_key']; } if ( ! empty( $meta_key ) && ! empty( $post ) ) { $image = $this->get_the_image_by_meta_key( array( 'post_id' => $post->ID, 'meta_key' => explode( ',', $meta_key ), ) ); if ( ! empty( $image ) ) { $img[] = array( 'type' => 'meta_key', 'id' => $meta_key, 'link' => $image ); } } if ( ! $post->post_modified_gmt != '' ) { $wp_query = new WP_Query( array( 'p' => $post_id, 'post_type' => $post->post_type ) ); } if ( $post->post_type == 'page' ) { $wp_query->is_page = true; } elseif ( $post->post_type == 'attachment' ) { $wp_query->is_attachment = true; } else { $wp_query->is_single = true; } if ( get_option( 'show_on_front' ) == 'page' && $post->ID == get_option( 'page_for_posts' ) ) { $wp_query->is_home = true; } $args['options']['type'] = 'html'; $args['options']['nowrap'] = false; $args['options']['save'] = false; $wp_query->queried_object = $post; $attachments = get_children( array( 'post_parent' => $post->ID, 'post_status' => 'inherit', 'post_type' => 'attachment', 'post_mime_type' => 'image', 'order' => 'ASC', 'orderby' => 'menu_order ID', ) ); if ( ! empty( $attachments ) ) { foreach ( $attachments as $id => $attachment ) { $image = wp_get_attachment_image_src( $id, $size ); if ( is_array( $image ) ) { $img[] = array( 'type' => 'attachment', 'id' => $id, 'link' => $image[0] ); } } } $matches = array(); preg_match_all( '||i', get_post_field( 'post_content', $post->ID ), $matches ); if ( isset( $matches ) && ! empty( $matches[1] ) && ! empty( $matches[1][0] ) ) { foreach ( $matches[1] as $i => $m ) { $img[] = array( 'type' => 'post_content', 'id' => 'post' . $count ++, 'link' => $m ); } } wp_reset_postdata(); $wp_query = $w; $post = $p; } } return $img; } /** * @param null $options * @param null $p * * @return array */ function get_all_images( $options = null, $p = null ) { $img = $this->get_all_images_by_type( $options, $p ); $legacy = array(); foreach ( $img as $k => $v ) { $v['link'] = set_url_scheme( $v['link'] ); if ( $v['type'] == 'featured' ) { $legacy[ $v['link'] ] = 1; } else { $legacy[ $v['link'] ] = $v['id']; } } return $legacy; } /** * Thanks to Justin Tadlock for the original get-the-image code - http://themehybrid.com/plugins/get-the-image ** * * @param null $options * @param null $p * * @return bool|mixed|string */ function get_the_image( $options = null, $p = null ) { if ( $p === null ) { global $post; } else { $post = $p; } $meta_key = ''; if ( is_array( $options ) && isset( $options['meta_key'] ) ) { $meta_key = $options['meta_key']; } if ( ! empty( $meta_key ) && ! empty( $post ) ) { $meta_key = explode( ',', $meta_key ); $image = $this->get_the_image_by_meta_key( array( 'post_id' => $post->ID, 'meta_key' => $meta_key, ) ); } if ( empty( $image ) ) { $image = $this->get_the_image_by_post_thumbnail( $post ); } if ( empty( $image ) ) { $image = $this->get_the_image_by_attachment( $post ); } if ( empty( $image ) ) { $image = $this->get_the_image_by_scan( $post ); } if ( empty( $image ) ) { $image = $this->get_the_image_by_default( $post ); } return $image; } /** * @param null $p * * @return string */ function get_the_image_by_default( $p = null ) { return ''; } /** * @param array $args * * @return bool|mixed */ function get_the_image_by_meta_key( $args = array() ) { /* If $meta_key is not an array. */ if ( ! is_array( $args['meta_key'] ) ) { $args['meta_key'] = array( $args['meta_key'] ); } /* Loop through each of the given meta keys. */ foreach ( $args['meta_key'] as $meta_key ) { /* Get the image URL by the current meta key in the loop. */ $image = get_post_meta( $args['post_id'], $meta_key, true ); /* If a custom key value has been given for one of the keys, return the image URL. */ if ( ! empty( $image ) ) { return $image; } } return false; } /** * @since 2.4.13 Fixes when content is taxonomy. * * @param null $p * * @return bool */ function get_the_image_by_post_thumbnail( $p = null ) { if ( $p === null ) { global $post; } else { $post = $p; } if ( is_category() || is_tag() || is_tax() ) { return false; } $post_thumbnail_id = null; if ( function_exists( 'get_post_thumbnail_id' ) ) { $post_thumbnail_id = get_post_thumbnail_id( $post->ID ); } if ( empty( $post_thumbnail_id ) ) { return false; } $size = apply_filters( 'post_thumbnail_size', 'large' ); // Check if someone is using built-in WP filter. $size = apply_filters( 'aioseop_thumbnail_size', $size ); $image = wp_get_attachment_image_src( $post_thumbnail_id, $size ); return $image[0]; } /** * @param null $p * * @return bool */ function get_the_image_by_attachment( $p = null ) { if ( $p === null ) { global $post; } else { $post = $p; } $attachments = get_children( array( 'post_parent' => $post->ID, 'post_status' => 'inherit', 'post_type' => 'attachment', 'post_mime_type' => 'image', 'order' => 'ASC', 'orderby' => 'menu_order ID', ) ); if ( empty( $attachments ) && 'attachment' == get_post_type( $post->ID ) ) { $size = 'large'; $size = apply_filters( 'aioseop_attachment_size', $size ); $image = wp_get_attachment_image_src( $post->ID, $size ); } /* If no attachments or image is found, return false. */ if ( empty( $attachments ) && empty( $image ) ) { return false; } /* Set the default iterator to 0. */ $i = 0; /* Loop through each attachment. Once the $order_of_image (default is '1') is reached, break the loop. */ foreach ( $attachments as $id => $attachment ) { if ( ++ $i == 1 ) { $size = 'large'; $size = apply_filters( 'aioseop_attachment_size', $size ); $image = wp_get_attachment_image_src( $id, $size ); $alt = trim( strip_tags( get_post_field( 'post_excerpt', $id ) ) ); break; } } /* Return the image URL. */ return $image[0]; } /** * @param null $p * * @return bool */ function get_the_image_by_scan( $p = null ) { if ( $p === null ) { global $post; } else { $post = $p; } /* Search the post's content for the tag and get its URL. */ preg_match_all( '||i', get_post_field( 'post_content', $post->ID ), $matches ); /* If there is a match for the image, return its URL. */ if ( isset( $matches ) && ! empty( $matches[1][0] ) ) { return $matches[1][0]; } return false; } /** * @param $default_options * @param $options * @param string $help_link */ function help_text_helper( &$default_options, $options, $help_link = '' ) { foreach ( $options as $o ) { $ht = ''; if ( ! empty( $this->help_text[ $o ] ) ) { $ht = $this->help_text[ $o ]; } elseif ( ! empty( $default_options[ $o ]['help_text'] ) ) { $ht = $default_options[ $o ]['help_text']; } if ( $ht && ! is_array( $ht ) ) { $ha = ''; $hl = $help_link; if ( strpos( $o, 'ga_' ) === 0 ) { // special case -- pdb $hl = 'https://semperplugins.com/documentation/advanced-google-analytics-settings/'; } if ( ! empty( $this->help_anchors[ $o ] ) ) { $ha = $this->help_anchors[ $o ]; } if ( ! empty( $ha ) && ( $pos = strrpos( $hl, '#' ) ) ) { $hl = substr( $hl, 0, $pos ); } if ( ! empty( $ha ) && ( $ha[0] == 'h' ) ) { $hl = ''; } if ( ! empty( $ha ) || ! isset( $this->help_anchors[ $o ] ) ) { $ht .= "
" . __( 'Click here for documentation on this setting', 'all-in-one-seo-pack' ) . ''; } $default_options[ $o ]['help_text'] = $ht; } } } function add_help_text_links() { if ( ! empty( $this->help_text ) ) { foreach ( $this->layout as $k => $v ) { $this->help_text_helper( $this->default_options, $v['options'], $v['help_link'] ); } if ( ! empty( $this->locations ) ) { foreach ( $this->locations as $k => $v ) { if ( ! empty( $v['default_options'] ) && ! empty( $v['options'] ) ) { $this->help_text_helper( $this->locations[ $k ]['default_options'], $v['options'], $v['help_link'] ); } } } } } /** * Load scripts and styles for metaboxes. * edit-tags exists only for pre 4.5 support... remove when we drop 4.5 support. * Also, that check and others should be pulled out into their own functions. * * @since 2.4.14 Added term as screen base. */ function enqueue_metabox_scripts() { $screen = ''; if ( function_exists( 'get_current_screen' ) ) { $screen = get_current_screen(); } $bail = false; if ( empty( $screen ) ) { $bail = true; } if ( $bail != true ) { if ( ( $screen->base != 'post' ) && ( $screen->base != 'term' ) && ( $screen->base != 'edit-tags' ) && ( $screen->base != 'toplevel_page_shopp-products' ) ) { $bail = true; } } $prefix = $this->get_prefix(); $bail = apply_filters( $prefix . 'bail_on_enqueue', $bail, $screen ); if ( $bail ) { return; } $this->form = 'post'; if ( $screen->base == 'term' || $screen->base == 'edit-tags' ) { $this->form = 'edittag'; } if ( $screen->base == 'toplevel_page_shopp-products' ) { $this->form = 'product'; } $this->form = apply_filters( $prefix . 'set_form_on_enqueue', $this->form, $screen ); foreach ( $this->locations as $k => $v ) { if ( $v['type'] === 'metabox' && isset( $v['display'] ) && ! empty( $v['display'] ) ) { $enqueue_scripts = false; $enqueue_scripts = ( ( ( $screen->base == 'toplevel_page_shopp-products' ) && in_array( 'shopp_product', $v['display'] ) ) ) || in_array( $screen->post_type, $v['display'] ) || $screen->base == 'edit-category' || $screen->base == 'edit-post_tag' || $screen->base == 'term'; $enqueue_scripts = apply_filters( $prefix . 'enqueue_metabox_scripts', $enqueue_scripts, $screen, $v ); if ( $enqueue_scripts ) { add_filter( 'aioseop_localize_script_data', array( $this, 'localize_script_data' ) ); add_action( 'admin_print_scripts', array( $this, 'enqueue_scripts' ), 20 ); add_action( 'admin_print_scripts', array( $this, 'enqueue_styles' ), 20 ); add_action( 'admin_enqueue_scripts', array( $this, 'admin_enqueue_scripts' ), 20 ); } } } } function admin_enqueue_scripts() { wp_enqueue_media(); // WP 3.5+ Media upload. } /** * Load styles for module. */ function enqueue_styles() { wp_enqueue_style( 'thickbox' ); if ( ! empty( $this->pointers ) ) { wp_enqueue_style( 'wp-pointer' ); } wp_enqueue_style( 'aioseop-module-style', AIOSEOP_PLUGIN_URL . 'css/modules/aioseop_module.css', array(), AIOSEOP_VERSION ); if ( function_exists( 'is_rtl' ) && is_rtl() ) { wp_enqueue_style( 'aioseop-module-style-rtl', AIOSEOP_PLUGIN_URL . 'css/modules/aioseop_module-rtl.css', array( 'aioseop-module-style' ), AIOSEOP_VERSION ); } } /** * Load scripts for module, can pass data to module script. * @since 2.3.12.3 Add missing wp_enqueue_media. */ function enqueue_scripts() { wp_enqueue_script( 'sack' ); wp_enqueue_script( 'jquery' ); wp_enqueue_script( 'media-upload' ); wp_enqueue_script( 'thickbox' ); wp_enqueue_script( 'common' ); wp_enqueue_script( 'wp-lists' ); wp_enqueue_script( 'postbox' ); if ( ! empty( $this->pointers ) ) { wp_enqueue_script( 'wp-pointer', false, array( 'jquery' ) ); } global $post; if ( ! empty( $post->ID ) ) { wp_enqueue_media( array( 'post' => $post->ID ) ); } else { wp_enqueue_media(); } wp_enqueue_script( 'aioseop-module-script', AIOSEOP_PLUGIN_URL . 'js/modules/aioseop_module.js', array(), AIOSEOP_VERSION ); if ( ! empty( $this->script_data ) ) { aioseop_localize_script_data(); } } /** * @param $data * * @return array */ function localize_script_data( $data ) { if ( ! is_array( $data ) ) { $data = array( 0 => $data ); } if ( empty( $this->script_data ) ) { $this->script_data = array(); } if ( ! empty( $this->pointers ) ) { $this->script_data['pointers'] = $this->pointers; } if ( empty( $data[0]['condshow'] ) ) { $data[0]['condshow'] = array(); } if ( empty( $this->script_data['condshow'] ) ) { $this->script_data['condshow'] = array(); } $condshow = $this->script_data['condshow']; $data[0]['condshow'] = array_merge( $data[0]['condshow'], $condshow ); unset( $this->script_data['condshow'] ); $data[0] = array_merge( $this->script_data, $data[0] ); $this->script_data['condshow'] = $condshow; return $data; } /** * Override this to run code at the beginning of the settings page. */ function settings_page_init() { } /** * Filter out admin pointers that have already been clicked. */ function filter_pointers() { if ( ! empty( $this->pointers ) ) { $dismissed = explode( ',', (string) get_user_meta( get_current_user_id(), 'dismissed_wp_pointers', true ) ); foreach ( $dismissed as $d ) { if ( isset( $this->pointers[ $d ] ) ) { unset( $this->pointers[ $d ] ); } } } } /** * Add basic hooks when on the module's page. */ function add_page_hooks() { $hookname = current_filter(); if ( $this->strpos( $hookname, 'load-' ) === 0 ) { $this->pagehook = $this->substr( $hookname, 5 ); } add_action( 'admin_print_scripts', array( $this, 'enqueue_scripts' ) ); add_action( 'admin_print_styles', array( $this, 'enqueue_styles' ) ); add_filter( 'aioseop_localize_script_data', array( $this, 'localize_script_data' ) ); add_action( $this->prefix . 'settings_header', array( $this, 'display_tabs' ) ); } /** * @return array */ function get_admin_links() { if ( ! empty( $this->menu_name ) ) { $name = $this->menu_name; } else { $name = $this->name; } $hookname = plugin_basename( $this->file ); $links = array(); $url = ''; if ( function_exists( 'menu_page_url' ) ) { $url = menu_page_url( $hookname, 0 ); } if ( empty( $url ) ) { $url = esc_url( admin_url( 'admin.php?page=' . $hookname ) ); } if ( $this->locations === null ) { array_unshift( $links, array( 'parent' => AIOSEOP_PLUGIN_DIRNAME, 'title' => $name, 'id' => $hookname, 'href' => $url, 'order' => $this->menu_order(), ) ); } else { foreach ( $this->locations as $k => $v ) { if ( $v['type'] === 'settings' ) { if ( $k === 'default' ) { array_unshift( $links, array( 'parent' => AIOSEOP_PLUGIN_DIRNAME, 'title' => $name, 'id' => $hookname, 'href' => $url, 'order' => $this->menu_order(), ) ); } else { if ( ! empty( $v['menu_name'] ) ) { $name = $v['menu_name']; } else { $name = $v['name']; } array_unshift( $links, array( 'parent' => AIOSEOP_PLUGIN_DIRNAME, 'title' => $name, 'id' => $this->get_prefix( $k ) . $k, 'href' => esc_url( admin_url( 'admin.php?page=' . $this->get_prefix( $k ) . $k ) ), 'order' => $this->menu_order(), ) ); } } } } return $links; } function add_admin_bar_submenu() { global $aioseop_admin_menu, $wp_admin_bar; if ( $aioseop_admin_menu ) { $links = $this->get_admin_links(); if ( ! empty( $links ) ) { foreach ( $links as $l ) { $wp_admin_bar->add_menu( $l ); } } } } /** * Collect metabox data together for tabbed metaboxes. * * @param $args * * @return array */ function filter_return_metaboxes( $args ) { return array_merge( $args, $this->post_metaboxes ); } /** Add submenu for module, call page hooks, set up metaboxes. * * @param $parent_slug * * @return bool */ function add_menu( $parent_slug ) { if ( ! empty( $this->menu_name ) ) { $name = $this->menu_name; } else { $name = $this->name; } if ( $this->locations === null ) { $hookname = add_submenu_page( $parent_slug, $name, $name, apply_filters( 'manage_aiosp', 'aiosp_manage_seo' ), plugin_basename( $this->file ), array( $this, 'display_settings_page', ) ); add_action( "load-{$hookname}", array( $this, 'add_page_hooks' ) ); return true; } foreach ( $this->locations as $k => $v ) { if ( $v['type'] === 'settings' ) { if ( $k === 'default' ) { if ( ! empty( $this->menu_name ) ) { $name = $this->menu_name; } else { $name = $this->name; } $hookname = add_submenu_page( $parent_slug, $name, $name, apply_filters( 'manage_aiosp', 'aiosp_manage_seo' ), plugin_basename( $this->file ), array( $this, 'display_settings_page', ) ); } else { if ( ! empty( $v['menu_name'] ) ) { $name = $v['menu_name']; } else { $name = $v['name']; } $hookname = add_submenu_page( $parent_slug, $name, $name, apply_filters( 'manage_aiosp', 'aiosp_manage_seo' ), $this->get_prefix( $k ) . $k, array( $this, "display_settings_page_$k", ) ); } add_action( "load-{$hookname}", array( $this, 'add_page_hooks' ) ); } elseif ( $v['type'] === 'metabox' ) { $this->setting_options( $k ); // hack -- make sure this runs anyhow, for now -- pdb add_action( 'edit_post', array( $this, 'save_post_data' ) ); add_action( 'publish_post', array( $this, 'save_post_data' ) ); add_action( 'add_attachment', array( $this, 'save_post_data' ) ); add_action( 'edit_attachment', array( $this, 'save_post_data' ) ); add_action( 'save_post', array( $this, 'save_post_data' ) ); add_action( 'edit_page_form', array( $this, 'save_post_data' ) ); if ( isset( $v['display'] ) && ! empty( $v['display'] ) ) { add_action( 'admin_print_scripts', array( $this, 'enqueue_metabox_scripts' ), 5 ); if ( $this->tabbed_metaboxes ) { add_filter( 'aioseop_add_post_metabox', array( $this, 'filter_return_metaboxes' ) ); } foreach ( $v['display'] as $posttype ) { $v['location'] = $k; $v['posttype'] = $posttype; if ( post_type_exists( $posttype ) ) { // Metabox priority/context on edit post screen. $v['context'] = apply_filters( 'aioseop_post_metabox_context', 'normal' ); $v['priority'] = apply_filters( 'aioseop_post_metabox_priority', 'high' ); } if ( false !== strpos( $posttype, 'edit-' ) ) { // Metabox priority/context on edit taxonomy screen. $v['context'] = 'advanced'; $v['priority'] = 'default'; } // Metabox priority for everything else. if ( ! isset( $v['context'] ) ) { $v['context'] = 'advanced'; } if ( ! isset( $v['priority'] ) ) { $v['priority'] = 'default'; } if ( $this->tabbed_metaboxes ) { $this->post_metaboxes[] = array( 'id' => $v['prefix'] . $k, 'title' => $v['name'], 'callback' => array( $this, 'display_metabox' ), 'post_type' => $posttype, 'context' => $v['context'], 'priority' => $v['priority'], 'callback_args' => $v, ); } else { $title = $v['name']; if ( $title != $this->plugin_name ) { $title = $this->plugin_name . ' - ' . $title; } if ( ! empty( $v['help_link'] ) ) { $title .= "" . __( 'Help', 'all-in-one-seo-pack' ) . ''; } add_meta_box( $v['prefix'] . $k, $title, array( $this, 'display_metabox', ), $posttype, $v['context'], $v['priority'], $v ); } } } } } } /** * Update postmeta for metabox. * * @param $post_id */ function save_post_data( $post_id ) { static $update = false; if ( $update ) { return; } if ( $this->locations !== null ) { foreach ( $this->locations as $k => $v ) { if ( isset( $v['type'] ) && ( $v['type'] === 'metabox' ) ) { $opts = $this->default_options( $k ); $options = array(); $update = false; foreach ( $opts as $l => $o ) { if ( isset( $_POST[ $l ] ) ) { $options[ $l ] = stripslashes_deep( $_POST[ $l ] ); $options[ $l ] = esc_attr( $options[ $l ] ); $update = true; } } if ( $update ) { $prefix = $this->get_prefix( $k ); $options = apply_filters( $prefix . 'filter_metabox_options', $options, $k, $post_id ); update_post_meta( $post_id, '_' . $prefix . $k, $options ); } } } } } /** * Outputs radio buttons, checkboxes, selects, multiselects, handles groups. * * @param $args * * @return string */ function do_multi_input( $args ) { $options = $args['options']; $value = $args['value']; $name = $args['name']; $attr = $args['attr']; $buf1 = ''; $type = $options['type']; $strings = array( 'block' => "\n", 'group' => "\t\n%s\t\n", 'item' => "\t\n", 'item_args' => array( 'sel', 'v', 'subopt' ), 'selected' => 'selected ', ); if ( ( $type === 'radio' ) || ( $type === 'checkbox' ) ) { $strings = array( 'block' => "%s\n", 'group' => "\t%s
\n%s\n", 'item' => "\t\n", 'item_args' => array( 'sel', 'name', 'v', 'attr', 'subopt' ), 'selected' => 'checked ', ); } $setsel = $strings['selected']; if ( isset( $options['initial_options'] ) && is_array( $options['initial_options'] ) ) { foreach ( $options['initial_options'] as $l => $option ) { $option_check = strip_tags( is_array( $option ) ? implode( ' ', $option ) : $option ); if ( empty( $l ) && empty( $option_check ) ) { continue; } $is_group = is_array( $option ); if ( ! $is_group ) { $option = array( $l => $option ); } $buf2 = ''; foreach ( $option as $v => $subopt ) { $sel = ''; $is_arr = is_array( $value ); if ( is_string( $v ) || is_string( $value ) ) { if ( is_string( $value ) ) { $cmp = ! strcmp( $v, $value ); } else { $cmp = ! strcmp( $v, '' ); } // $cmp = !strcmp( (string)$v, (string)$value ); } else { $cmp = ( $value == $v ); } if ( ( ! $is_arr && $cmp ) || ( $is_arr && in_array( $v, $value ) ) ) { $sel = $setsel; } $item_arr = array(); foreach ( $strings['item_args'] as $arg ) { $item_arr[] = $$arg; } $buf2 .= vsprintf( $strings['item'], $item_arr ); } if ( $is_group ) { $buf1 .= sprintf( $strings['group'], $l, $buf2 ); } else { $buf1 .= $buf2; } } $buf1 = sprintf( $strings['block'], $buf1 ); } return $buf1; } /** * Outputs a setting item for settings pages and metaboxes. * * @param $args * * @return string */ function get_option_html( $args ) { static $n = 0; $options = $args['options']; $value = $args['value']; $name = $args['name']; $attr = $args['attr']; $prefix = isset( $args['prefix'] ) ? $args['prefix'] : ''; if ( $options['type'] == 'custom' ) { return apply_filters( "{$prefix}output_option", '', $args ); } if ( in_array( $options['type'], array( 'multiselect', 'select', 'multicheckbox', 'radio', 'checkbox', 'textarea', 'text', 'submit', 'hidden', 'date', ) ) && is_string( $value ) ) { $value = esc_attr( $value ); } $buf = ''; $onload = ''; if ( ! empty( $options['count'] ) ) { $n ++; $classes = isset( $options['class'] ) ? $options['class'] : ''; $classes .= ' aioseop_count_chars'; $attr .= " class='{$classes}' data-length-field='{$prefix}length$n'"; } if ( isset( $opts['id'] ) ) { $attr .= " id=\"{$opts['id']}\" "; } if ( isset( $options['required'] ) && true === $options['required'] ) { $attr .= ' required'; } switch ( $options['type'] ) { case 'multiselect': $attr .= ' MULTIPLE'; $args['attr'] = $attr; $args['name'] = $name = "{$name}[]"; // fall through. case 'select': $buf .= $this->do_multi_input( $args ); break; case 'multicheckbox': $args['name'] = $name = "{$name}[]"; $args['options']['type'] = $options['type'] = 'checkbox'; // fall through. case 'radio': $buf .= $this->do_multi_input( $args ); break; case 'checkbox': if ( $value ) { $attr .= ' CHECKED'; } $buf .= "\n"; break; case 'textarea': $buf .= ""; break; case 'image': $buf .= '' . "\n"; break; case 'html': $buf .= $value; break; case 'esc_html': $buf .= '
' . esc_html( $value ) . "
\n"; break; case 'date': // firefox and IE < 11 do not have support for HTML5 date, so we will fall back to the datepicker. wp_enqueue_script( 'jquery-ui-datepicker' ); // fall through. default: $buf .= "\n"; } if ( ! empty( $options['count'] ) ) { $size = 60; if ( isset( $options['size'] ) ) { $size = $options['size']; } elseif ( isset( $options['rows'] ) && isset( $options['cols'] ) ) { $size = $options['rows'] * $options['cols']; } if ( isset( $options['count_desc'] ) ) { $count_desc = $options['count_desc']; } else { $count_desc = __( ' characters. Most search engines use a maximum of %1$s chars for the %2$s.', 'all-in-one-seo-pack' ); } $buf .= "
" . sprintf( $count_desc, $size, trim( $this->strtolower( $options['name'] ), ':' ) ); if ( ! empty( $onload ) ) { $buf .= ""; } } return $buf; } const DISPLAY_HELP_START = ''; const DISPLAY_HELP_END = ''; const DISPLAY_LABEL_FORMAT = '%s'; const DISPLAY_TOP_LABEL = "\n
\n"; const DISPLAY_ROW_TEMPLATE = '
%s
%s
%s

'; /** * Format a row for an option on a settings page. * * @param $name * @param $opts * @param $args * * @return string */ function get_option_row( $name, $opts, $args ) { $label_text = $input_attr = $help_text_2 = $id_attr = ''; $align = 'right'; if ( $opts['label'] == 'top' ) { $align = 'left'; } if ( isset( $opts['id'] ) ) { $id_attr .= " id=\"{$opts['id']}_div\" "; } if ( $opts['label'] != 'none' ) { if ( isset( $opts['help_text'] ) ) { $help_text = sprintf( All_in_One_SEO_Pack_Module::DISPLAY_HELP_START, __( 'Click for Help!', 'all-in-one-seo-pack' ), $name, $opts['name'] ); $help_text_2 = sprintf( All_in_One_SEO_Pack_Module::DISPLAY_HELP_END, $name, $opts['help_text'] ); } else { $help_text = $opts['name']; } $label_text = sprintf( All_in_One_SEO_Pack_Module::DISPLAY_LABEL_FORMAT, $align, $help_text ); } else { $input_attr .= ' aioseop_no_label '; } if ( $opts['label'] == 'top' ) { $label_text .= All_in_One_SEO_Pack_Module::DISPLAY_TOP_LABEL; } $input_attr .= " aioseop_{$opts['type']}_type"; return sprintf( All_in_One_SEO_Pack_Module::DISPLAY_ROW_TEMPLATE, $input_attr, $name, $label_text, $id_attr, $this->get_option_html( $args ), $help_text_2 ); } /** * Display options for settings pages and metaboxes, allows for filtering settings, custom display options. * * @param null $location * @param null $meta_args */ function display_options( $location = null, $meta_args = null ) { static $location_settings = array(); $defaults = null; $prefix = $this->get_prefix( $location ); $help_link = ''; if ( is_array( $meta_args['args'] ) && ! empty( $meta_args['args']['default_options'] ) ) { $defaults = $meta_args['args']['default_options']; } if ( ! empty( $meta_args['callback_args'] ) && ! empty( $meta_args['callback_args']['help_link'] ) ) { $help_link = $meta_args['callback_args']['help_link']; } if ( ! empty( $help_link ) ) { echo "" . __( 'Help', 'all-in-one-seo-pack' ) . ''; } if ( ! isset( $location_settings[ $prefix ] ) ) { $current_options = apply_filters( "{$this->prefix}display_options", $this->get_current_options( array(), $location, $defaults ), $location ); $settings = apply_filters( "{$this->prefix}display_settings", $this->setting_options( $location, $defaults ), $location, $current_options ); $current_options = apply_filters( "{$this->prefix}override_options", $current_options, $location, $settings ); $location_settings[ $prefix ]['current_options'] = $current_options; $location_settings[ $prefix ]['settings'] = $settings; } else { $current_options = $location_settings[ $prefix ]['current_options']; $settings = $location_settings[ $prefix ]['settings']; } // $opts["snippet"]["default"] = sprintf( $opts["snippet"]["default"], "foo", "bar", "moby" ); $container = "
"; if ( is_array( $meta_args['args'] ) && ! empty( $meta_args['args']['options'] ) ) { $args = array(); $arg_keys = array(); foreach ( $meta_args['args']['options'] as $a ) { if ( ! empty( $location ) ) { $key = $prefix . $location . '_' . $a; if ( ! isset( $settings[ $key ] ) ) { $key = $a; } } else { $key = $prefix . $a; } if ( isset( $settings[ $key ] ) ) { $arg_keys[ $key ] = 1; } elseif ( isset( $settings[ $a ] ) ) { $arg_keys[ $a ] = 1; } } $setting_keys = array_keys( $settings ); foreach ( $setting_keys as $s ) { if ( ! empty( $arg_keys[ $s ] ) ) { $args[ $s ] = $settings[ $s ]; } } } else { $args = $settings; } foreach ( $args as $name => $opts ) { $attr_list = array( 'class', 'style', 'readonly', 'disabled', 'size', 'placeholder' ); if ( $opts['type'] == 'textarea' ) { $attr_list = array_merge( $attr_list, array( 'rows', 'cols' ) ); } $attr = ''; foreach ( $attr_list as $a ) { if ( isset( $opts[ $a ] ) ) { $attr .= ' ' . $a . '="' . esc_attr( $opts[ $a ] ) . '" '; } } $opt = ''; if ( isset( $current_options[ $name ] ) ) { $opt = $current_options[ $name ]; } if ( $opts['label'] == 'none' && $opts['type'] == 'submit' && $opts['save'] == false ) { $opt = $opts['name']; } if ( $opts['type'] == 'html' && empty( $opt ) && $opts['save'] == false ) { $opt = $opts['default']; } $args = array( 'name' => $name, 'options' => $opts, 'attr' => $attr, 'value' => $opt, 'prefix' => $prefix, ); if ( ! empty( $opts['nowrap'] ) ) { echo $this->get_option_html( $args ); } else { if ( $container ) { echo $container; $container = ''; } echo $this->get_option_row( $name, $opts, $args ); } } if ( ! $container ) { echo '
'; } } /** * @param $domain * * @return mixed|string */ function sanitize_domain( $domain ) { $domain = trim( $domain ); $domain = $this->strtolower( $domain ); if ( $this->strpos( $domain, 'http://' ) === 0 ) { $domain = $this->substr( $domain, 7 ); } elseif ( $this->strpos( $domain, 'https://' ) === 0 ) { $domain = $this->substr( $domain, 8 ); } $domain = untrailingslashit( $domain ); return $domain; } /** Sanitize options * * @param null $location */ function sanitize_options( $location = null ) { foreach ( $this->setting_options( $location ) as $k => $v ) { if ( isset( $this->options[ $k ] ) ) { if ( ! empty( $v['sanitize'] ) ) { $type = $v['sanitize']; } else { $type = $v['type']; } switch ( $type ) { case 'multiselect': // fall through. case 'multicheckbox': $this->options[ $k ] = urlencode_deep( $this->options[ $k ] ); break; case 'textarea': $this->options[ $k ] = wp_kses_post( $this->options[ $k ] ); $this->options[ $k ] = htmlspecialchars( $this->options[ $k ], ENT_QUOTES ); break; case 'filename': $this->options[ $k ] = sanitize_file_name( $this->options[ $k ] ); break; case 'text': $this->options[ $k ] = wp_kses_post( $this->options[ $k ] ); // fall through. case 'checkbox': // fall through. case 'radio': // fall through. case 'select': // fall through. default: if ( ! is_array( $this->options[ $k ] ) ) { $this->options[ $k ] = esc_attr( $this->options[ $k ] ); } } } } } /** * Display metaboxes with display_options() * * @param $post * @param $metabox */ function display_metabox( $post, $metabox ) { $this->display_options( $metabox['args']['location'], $metabox ); } /** * Handle resetting options to defaults. * * @param null $location * @param bool $delete */ function reset_options( $location = null, $delete = false ) { if ( $delete === true ) { $this->delete_class_option( $delete ); $this->options = array(); } $default_options = $this->default_options( $location ); foreach ( $default_options as $k => $v ) { $this->options[ $k ] = $v; } $this->update_class_option( $this->options ); } /** handle option resetting and updating * * @param null $location * * @return mixed|string|void */ function handle_settings_updates( $location = null ) { $message = ''; if ( ( isset( $_POST['action'] ) && $_POST['action'] == 'aiosp_update_module' && ( isset( $_POST['Submit_Default'] ) || isset( $_POST['Submit_All_Default'] ) || ! empty( $_POST['Submit'] ) ) ) ) { $nonce = $_POST['nonce-aioseop']; if ( ! wp_verify_nonce( $nonce, 'aioseop-nonce' ) ) { die( __( 'Security Check - If you receive this in error, log out and back in to WordPress', 'all-in-one-seo-pack' ) ); } if ( isset( $_POST['Submit_Default'] ) || isset( $_POST['Submit_All_Default'] ) ) { $message = __( 'Options Reset.', 'all-in-one-seo-pack' ); if ( isset( $_POST['Submit_All_Default'] ) ) { $this->reset_options( $location, true ); do_action( 'aioseop_options_reset' ); } else { $this->reset_options( $location ); } } if ( ! empty( $_POST['Submit'] ) ) { $message = __( 'All in One SEO Options Updated.', 'all-in-one-seo-pack' ); $default_options = $this->default_options( $location ); foreach ( $default_options as $k => $v ) { if ( isset( $_POST[ $k ] ) ) { $this->options[ $k ] = stripslashes_deep( $_POST[ $k ] ); } else { $this->options[ $k ] = ''; } } $this->sanitize_options( $location ); $this->options = apply_filters( $this->prefix . 'update_options', $this->options, $location ); $this->update_class_option( $this->options ); wp_cache_flush(); } do_action( $this->prefix . 'settings_update', $this->options, $location ); } return $message; } /** Update / reset settings, printing options, sanitizing, posting back * * @param null $location */ function display_settings_page( $location = null ) { if ( $location != null ) { $location_info = $this->locations[ $location ]; } $name = null; if ( $location && isset( $location_info['name'] ) ) { $name = $location_info['name']; } if ( ! $name ) { $name = $this->name; } $message = $this->handle_settings_updates( $location ); $this->settings_page_init(); ?>
prefix . 'settings_header_errors', $location ); $errors = ob_get_clean(); echo $errors; ?>

$message

"; } ?>

prefix . 'settings_header', $location ); ?>
array( 'type' => 'hidden', 'value' => 'aiosp_update_module' ), 'module' => array( 'type' => 'hidden', 'value' => get_class( $this ) ), 'location' => array( 'type' => 'hidden', 'value' => $location ), 'nonce-aioseop' => array( 'type' => 'hidden', 'value' => wp_create_nonce( 'aioseop-nonce' ), ), 'page_options' => array( 'type' => 'hidden', 'value' => 'aiosp_home_description' ), 'Submit' => array( 'type' => 'submit', 'class' => 'button-primary', 'value' => __( 'Update Options', 'all-in-one-seo-pack' ) . ' »', ), 'Submit_Default' => array( 'type' => 'submit', 'class' => 'button-secondary', 'value' => sprintf( __( 'Reset %s Settings to Defaults', 'all-in-one-seo-pack' ), $name ) . ' »', ), ); $submit_options = apply_filters( "{$this->prefix}submit_options", $submit_options, $location ); foreach ( $submit_options as $k => $s ) { if ( $s['type'] == 'submit' && $k != 'Submit' ) { continue; } $class = ''; if ( isset( $s['class'] ) ) { $class = " class='{$s['class']}' "; } echo $this->get_option_html( array( 'name' => $k, 'options' => $s, 'attr' => $class, 'value' => $s['value'], ) ); } ?>
get_class_option(); if ( $opts !== false ) { $this->options = $opts; } if ( is_array( $this->layout ) ) { foreach ( $this->layout as $l => $lopts ) { if ( ! isset( $lopts['tab'] ) || ( $this->current_tab == $lopts['tab'] ) ) { $title = $lopts['name']; if ( ! empty( $lopts['help_link'] ) ) { $title .= "" . __( 'Help', 'all-in-one-seo-pack' ) . ''; } add_meta_box( $this->get_prefix( $location ) . $l . '_metabox', $title, array( $this, 'display_options', ), "{$this->prefix}settings", 'advanced', 'default', $lopts ); } } } else { add_meta_box( $this->get_prefix( $location ) . 'metabox', $name, array( $this, 'display_options', ), "{$this->prefix}settings", 'advanced' ); } do_meta_boxes( "{$this->prefix}settings", 'advanced', $location ); ?>

$s ) { $class = ''; if ( isset( $s['class'] ) ) { $class = " class='{$s['class']}' "; } echo $this->get_option_html( array( 'name' => $k, 'options' => $s, 'attr' => $class, 'value' => $s['value'], ) ); } ?>

prefix . 'settings_footer', $location ); do_action( 'aioseop_global_settings_footer', $location ); ?>
locations[ $location ]['prefix'] ) ) { return $this->locations[ $location ]['prefix']; } return $this->prefix; } /** Sets up initial settings * * @param null $location * @param null $defaults * * @return array */ function setting_options( $location = null, $defaults = null ) { if ( $defaults === null ) { $defaults = $this->default_options; } $prefix = $this->get_prefix( $location ); $opts = array(); if ( $location == null || $this->locations[ $location ]['options'] === null ) { $options = $defaults; } else { $options = array(); $prefix = "{$prefix}{$location}_"; if ( ! empty( $this->locations[ $location ]['default_options'] ) ) { $options = $this->locations[ $location ]['default_options']; } foreach ( $this->locations[ $location ]['options'] as $opt ) { if ( isset( $defaults[ $opt ] ) ) { $options[ $opt ] = $defaults[ $opt ]; } } } if ( ! $prefix ) { $prefix = $this->prefix; } if ( ! empty( $options ) ) { foreach ( $options as $k => $v ) { if ( ! isset( $v['name'] ) ) { $v['name'] = $this->ucwords( strtr( $k, '_', ' ' ) ); } if ( ! isset( $v['type'] ) ) { $v['type'] = 'checkbox'; } if ( ! isset( $v['default'] ) ) { $v['default'] = null; } if ( ! isset( $v['initial_options'] ) ) { $v['initial_options'] = $v['default']; } if ( $v['type'] == 'custom' && ( ! isset( $v['nowrap'] ) ) ) { $v['nowrap'] = true; } elseif ( ! isset( $v['nowrap'] ) ) { $v['nowrap'] = null; } if ( isset( $v['condshow'] ) ) { if ( ! is_array( $this->script_data ) ) { $this->script_data = array(); } if ( ! isset( $this->script_data['condshow'] ) ) { $this->script_data['condshow'] = array(); } $this->script_data['condshow'][ $prefix . $k ] = $v['condshow']; } if ( $v['type'] == 'submit' ) { if ( ! isset( $v['save'] ) ) { $v['save'] = false; } if ( ! isset( $v['label'] ) ) { $v['label'] = 'none'; } if ( ! isset( $v['prefix'] ) ) { $v['prefix'] = false; } } else { if ( ! isset( $v['label'] ) ) { $v['label'] = null; } } if ( $v['type'] == 'hidden' ) { if ( ! isset( $v['label'] ) ) { $v['label'] = 'none'; } if ( ! isset( $v['prefix'] ) ) { $v['prefix'] = false; } } if ( ( $v['type'] == 'text' ) && ( ! isset( $v['size'] ) ) ) { $v['size'] = 57; } if ( $v['type'] == 'textarea' ) { if ( ! isset( $v['cols'] ) ) { $v['cols'] = 57; } if ( ! isset( $v['rows'] ) ) { $v['rows'] = 2; } } if ( ! isset( $v['save'] ) ) { $v['save'] = true; } if ( ! isset( $v['prefix'] ) ) { $v['prefix'] = true; } if ( $v['prefix'] ) { $opts[ $prefix . $k ] = $v; } else { $opts[ $k ] = $v; } } } return $opts; } /** * Generates just the default option names and values * * @since 2.4.13 Applies filter before final return. * * @param null $location * @param null $defaults * * @return array */ function default_options( $location = null, $defaults = null ) { $prefix = $this->get_prefix( $location ); $options = $this->setting_options( $location, $defaults ); $opts = array(); foreach ( $options as $k => $v ) { if ( $v['save'] ) { $opts[ $k ] = $v['default']; } } return apply_filters( $prefix . 'default_options', $opts, $location ); } /** * Gets the current options stored for a given location. * * @since 2.4.14 Added taxonomy options. * * @param array $opts * @param null $location * @param null $defaults * @param null $post * * @return array */ function get_current_options( $opts = array(), $location = null, $defaults = null, $post = null ) { $prefix = $this->get_prefix( $location ); $get_opts = ''; if ( empty( $location ) ) { $type = 'settings'; } else { $type = $this->locations[ $location ]['type']; } if ( $type === 'settings' ) { $get_opts = $this->get_class_option(); } elseif ( $type == 'metabox' ) { if ( $post == null ) { global $post; } if ( ( isset( $_GET['taxonomy'] ) && isset( $_GET['tag_ID'] ) ) || is_category() || is_tag() || is_tax() ) { $term_id = isset( $_GET['tag_ID'] ) ? (int) $_GET['tag_ID'] : 0; $term_id = $term_id ? $term_id : get_queried_object()->term_id; if ( AIOSEOPPRO ) { $get_opts = AIO_ProGeneral::getprotax( $get_opts ); $get_opts = get_term_meta( $term_id, '_' . $prefix . $location, true ); } } elseif ( isset( $post ) ) { $get_opts = get_post_meta( $post->ID, '_' . $prefix . $location, true ); } } if ( is_home() && ! is_front_page() ) { // If we're on the non-front page blog page, WP doesn't really know its post meta data so we need to get that manually for social meta. $get_opts = get_post_meta( get_option( 'page_for_posts' ), '_' . $prefix . $location, true ); } $defs = $this->default_options( $location, $defaults ); if ( empty( $get_opts ) ) { $get_opts = $defs; } else { $get_opts = wp_parse_args( $get_opts, $defs ); } $opts = wp_parse_args( $opts, $get_opts ); return $opts; } /** Updates the options array in the module; loads saved settings with get_option() or uses defaults * * @param array $opts * @param null $location * @param null $defaults */ function update_options( $opts = array(), $location = null, $defaults = null ) { if ( $location === null ) { $type = 'settings'; } else { $type = $this->locations[ $location ][ $type ]; } if ( $type === 'settings' ) { $get_opts = $this->get_class_option(); } if ( $get_opts === false ) { $get_opts = $this->default_options( $location, $defaults ); } else { $this->setting_options( $location, $defaults ); } // hack -- make sure this runs anyhow, for now -- pdb $this->options = wp_parse_args( $opts, $get_opts ); } } }