add_skip_paragraph_filters(); // Check if ads can be displayed by post type. add_filter( 'advanced-ads-can-display', array( $this, 'can_display_by_post_type' ), 10, 2 ); // Check if Verification code & Auto ads ads can be displayed by post type. add_filter( 'advanced-ads-can-display-ads-in-header', array( $this, 'can_display_in_header_by_post_type' ), 10 ); add_action( 'advanced-ads-body-classes', array( $this, 'body_class' ) ); } /** * add new placement types * * @since 1.0.0 * @param array $types * * @return array $types */ public function add_placement_types($types) { // ad injection on random position $types['post_content_random'] = array( 'title' => __( 'Random Paragraph', 'advanced-ads-pro' ), 'description' => __( 'After a random paragraph in the main content.', 'advanced-ads-pro' ), 'image' => AAP_BASE_URL . 'modules/inject-content/assets/img/content-random.png', 'order' => 22, 'options' => array( 'show_position' => true, 'uses_the_content' => true, 'amp' => true ) ); // ad injection above the post headline $types['post_above_headline'] = array( 'title' => __( 'Above Headline', 'advanced-ads-pro' ), 'description' => __( 'Above the main headline on the page (<h1>).', 'advanced-ads-pro' ), 'image' => AAP_BASE_URL . 'modules/inject-content/assets/img/content-above-headline.png', 'order' => 7, 'options' => array( 'show_position' => true, 'uses_the_content' => true ) ); // ad injection in the middle of a post $types['post_content_middle'] = array( 'title' => __( 'Content Middle', 'advanced-ads-pro' ), 'description' => __( 'In the middle of the main content based on the number of paragraphs.', 'advanced-ads-pro' ), 'image' => AAP_BASE_URL . 'modules/inject-content/assets/img/content-middle.png', 'order' => 23, 'options' => array( 'show_position' => true, 'uses_the_content' => true, 'amp' => true ), ); // ad injection at a hand selected element in the frontend $types['custom_position'] = array( 'title' => __( 'Custom Position', 'advanced-ads-pro' ), 'description' => __( 'Attach the ad to any element in the frontend.', 'advanced-ads-pro' ), 'image' => AAP_BASE_URL . 'modules/inject-content/assets/img/custom-position.png', 'order' => 60, 'options' => array( 'show_position' => true ) ); // ad injection at a hand selected element in the frontend $types['archive_pages'] = array( 'title' => __( 'Post Lists', 'advanced-ads-pro' ), 'description' => __( 'Display the ad between posts on post lists, e.g. home, archives, search etc.', 'advanced-ads-pro' ), 'image' => AAP_BASE_URL . 'modules/inject-content/assets/img/post-list.png', 'order' => 40, 'options' => array( 'show_position' => true, 'show_lazy_load' => true ) ); return $types; } /** * injected ad randomly into post content * * @since 1.0.0 * @param str $content post content */ public function inject_content( $content = '' ) { global $post; $options = Advanced_Ads::get_instance()->options(); // do not inject in content when on a BuddyPress profile upload page (avatar & cover image). if ( ( function_exists( 'bp_is_user_change_avatar' ) && bp_is_user_change_avatar() ) || ( function_exists( 'bp_is_user_change_cover_image' ) && bp_is_user_change_cover_image() ) ) { return $content; } if ( $this->has_many_the_content() ) { return $content; } // Check if ads are disabled in secondary queries. if ( ! empty( $options['disabled-ads']['secondary'] ) ) { if ( defined( 'DOING_AJAX' ) && DOING_AJAX ) { // This function was called by ajax (in secondary query). return $content; } // get out of wp_router_page post type if ads are disabled in secondary queries. if ( 'wp_router_page' === get_post_type() ) { return $content; } } // No need to inject ads because all tags are stripped from excepts. if ( doing_filter( 'get_the_excerpt' ) ) { return $content; } // run only within the loop on single pages of public post types $public_post_types = get_post_types( array( 'public' => true, 'publicly_queryable' => true ), 'names', 'or' ); // make sure that no ad is injected into another ad if ( get_post_type() == Advanced_Ads::POST_TYPE_SLUG ){ return $content; } // Do not inject on admin pages. if ( is_admin() && ! defined( 'DOING_AJAX' ) ) { return $content; } // check if admin allows injection in all places if( ! isset( $options['content-injection-everywhere'] ) ){ // check if this is a singular page within the loop or an amp page $is_amp = function_exists( 'advads_is_amp' ) && advads_is_amp(); if ( ( ! is_singular( $public_post_types ) && ! is_feed() ) || ( ! $is_amp && ! in_the_loop() ) ) { return $content; } } $placements = get_option( 'advads-ads-placements', array() ); if( ! apply_filters( 'advanced-ads-can-inject-into-content', true, $content, $placements )){ return $content; } if( is_array( $placements ) ){ foreach ( $placements as $_placement_id => $_placement ){ if ( empty( $_placement['item'] ) ) { continue; } if ( isset($_placement['type']) && in_array( $_placement['type'], array('post_content_random', 'post_above_headline', 'post_content_middle')) ){ // don’t inject above headline on non-singular pages if( 'post_above_headline' === $_placement['type'] && ! is_singular( $public_post_types ) ){ continue; } // check if injection is ok for a specific placement id if( ! apply_filters( 'advanced-ads-can-inject-into-content-' . $_placement_id, true, $content, $_placement_id )){ continue; } $_options = isset( $_placement['options'] ) ? $_placement['options'] : array(); $_options['placement']['type'] = $_placement['type']; switch ( $_placement['type'] ) { case 'post_above_headline' : $content .= Advanced_Ads_Select::get_instance()->get_ad_by_method( $_placement_id, Advanced_Ads_Select::PLACEMENT, $_options ); break; case 'post_content_middle' : $content = Advanced_Ads_Placements::inject_in_content( $_placement_id, $_options, $content ); break; case 'post_content_random' : if ( $this->content_random_use_js( $_options ) ) { $content .= Advanced_Ads_Select::get_instance()->get_ad_by_method( $_placement_id, Advanced_Ads_Select::PLACEMENT, $_options ); } else { $content = Advanced_Ads_Placements::inject_in_content( $_placement_id, $_options, $content ); } break; } } } } return $content; } /** * inject ad into footer * * @since 1.1.2 */ public function inject_footer(){ $placements = get_option( 'advads-ads-placements', array() ); if( is_array( $placements ) ){ foreach ( $placements as $_placement_id => $_placement ){ if ( isset($_placement['type']) && 'custom_position' == $_placement['type'] ){ // Do not inject on AMP pages. if ( function_exists( 'advads_is_amp' ) && advads_is_amp() ) { continue; } $_options = isset( $_placement['options'] ) ? $_placement['options'] : array(); $_options['placement']['type'] = $_placement['type']; echo Advanced_Ads_Select::get_instance()->get_ad_by_method( $_placement_id, Advanced_Ads_Select::PLACEMENT, $_options ); } } } } /** * inject ad output and js code * * @since 1.1 * @param str $content ad content * @param obj $ad ad object */ public function after_ad_output( $content = '', Advanced_Ads_Ad $ad ) { if ( isset( $ad->args['previous_method'] ) && Advanced_Ads_Select::GROUP === $ad->args['previous_method'] ) { return $content; } if ( ! isset( $ad->args['cache_busting_elementid'] ) && isset( $ad->wrapper['id'] ) ) { $content .= $this->get_output_js( $ad->wrapper['id'], $ad->args ); } return $content; } /** * inject js code after group output * * @param str $output_string final group output * @param obj $group Advanced_Ads_Group */ public function after_group_output( $output_string, Advanced_Ads_Group $group ) { if ( $output_string ) { if ( ! isset( $group->ad_args['cache_busting_elementid'] ) ) { $wrapper_id = Advanced_Ads_Pro_Utils::generate_wrapper_id(); if ( $js_output = $this->get_output_js( $wrapper_id, $group->ad_args ) ) { $output_string = '
' . $output_string . '
' . $js_output; } } } return $output_string; } /** * get js to append after ad/group output * * @return string */ private function get_output_js( $wrapper_id, $args ) { $content = ''; // Do not inject js on AMP pages. if ( function_exists( 'advads_is_amp' ) && advads_is_amp() ) { return $content; } // Group refresh: do not move if the top level wrapper was moved earlier. if ( isset( $args['group_refresh'] ) && ! $args['group_refresh']['is_top_level'] ) { return $content; } // Move only the most outer group wrapper. $top_level = ! isset( $args['previous_method'] ) || 'placement' === $args['previous_method']; if ( ! $top_level ) { return $content; } if ( isset ( $args['placement']['type'] ) ) { switch( $args['placement']['type'] ){ case 'post_content_random' : if ( ! $this->content_random_use_js( $args ) ) { return ''; } $paragraphs_selector = $this->get_paragraph_selector( $args ); $content .= 'var advads_content_p = jQuery("#'. $wrapper_id .'")' . $paragraphs_selector . ';' . 'var advads_content_random_p = advads_content_p.eq( Math.round(Math.random() * ( advads_content_p.length - 1) ) );' . 'if( advads_content_random_p.length ) { advads.move("#'. $wrapper_id .'", advads_content_random_p, { method: "insertAfter" }); }'; break; case 'post_above_headline' : $content .= 'advads.move("#'. $wrapper_id .'", "h1", { method: "insertBefore" });'; break; case 'custom_position' : // By element Selector. if ( ! isset( $args['inject_by'] ) || $args['inject_by'] === 'pro_custom_element' ) { $target = isset( $args['pro_custom_element'] ) ? $args['pro_custom_element'] : ''; $position = isset( $args['pro_custom_position'] ) ? $args['pro_custom_position'] : 'insertBefore'; // By HTML container. } else { $target = isset( $args['container_id'] ) ? $args['container_id'] : ''; $position = 'appendTo'; } $options[] = 'method: "'. $position . '"'; // check if can be moved into hidden elements if( defined( 'ADVANCED_ADS_PRO_CUSTOM_POSITION_MOVE_INTO_HIDDEN') ){ $options[] = 'moveintohidden: "true"'; } $content .= 'advads.move("#'. $wrapper_id .'", "'. $target .'", { '. implode( ', ', $options ) .' });'; break; } if ( $content ) { if ( ! empty( $args['cache_busting_elementid'] ) ) { // Document is ready. Do not use another 'ready' block so that the wrapper is moved before executing js in ad content. $content = ''; } else { $content = ''; } } } return $content; } /** * get paragraph selector for js depending on cache busting settings * * @return str $paragraph_selector * @since 1.2.3 */ private function get_paragraph_selector( $args ) { // check if level limitation is disabled $plugin_options = Advanced_Ads_Plugin::get_instance()->options(); $content_injection_level_disabled = isset( $plugin_options['content-injection-level-disabled'] ); /** * find paragraphs * which are not within tables * which are not within blockquotes * which are not empty * which are not within an image caption * * depending on "Disable injection limitation" setting, * either inject into all p tags, including subordinated * or only direct and preceding siblings */ if( $content_injection_level_disabled ){ $paragraphs_selector = '.parent().find("p:not(table p):not(blockquote p):not(div.wp-caption p)").filter(function(){return jQuery.trim(this.innerHTML)!==""})'; } else { // Do not use 'prevAll' because it returns elements in reverse order. $paragraphs_selector = '.parent().children("p:not(table p):not(blockquote p):not(div.wp-caption p)").filter(function(){return jQuery.trim(this.innerHTML)!==""})'; } // TODO: Deprecated. if ( defined( 'DOING_AJAX' ) && DOING_AJAX && ! isset( $args['cache_busting_elementid'] ) ) { $paragraphs_selector = '.parent()' . $paragraphs_selector; } return apply_filters( 'advanced-ads-pro-inject-content-selector', $paragraphs_selector ); } /** * Check content length for injecting ads into the post content * * @param bool $inject whether to inject or not * @param string $content post content * @param array $placements array with all placements * * @return bool true, if injection is ok */ public function check_content_length( $inject = true, $content = '', $placements = array() ){ if ( ! $inject ) { return false; } if ( defined( 'ADVADS_CURRENT_CONTENT_LENGTH' ) ) { return $inject; } // content injection placements $cj_placements = array( 'post_top', 'post_bottom', 'post_content', 'post_content_random', 'post_content_middle' ); // find out of content injection placements are defined at all $has_content_placements = false; foreach( $placements as $_placement_id => $_placement ){ if( isset( $_placement['type'] ) && in_array( $_placement['type'], $cj_placements ) ){ $has_content_placements = true; // register filter for placement specific length check add_filter( 'advanced-ads-can-inject-into-content-' . $_placement_id, array( $this, 'check_placement_minimum_length' ), 10, 3 ); } } if ( $has_content_placements ) { // Remove all HTML tags and comments and count spaces in content. $length = (int) preg_match_all( '/\s+/', wp_strip_all_tags( $content ) ); define( 'ADVADS_CURRENT_CONTENT_LENGTH', $length ); } return $inject; } /** * Allow to prevent injections inside `the_content`. * * @param bool $inject whether to inject or not * @param str $content post content * @param arr $placements array with all placements * @return bool true, if injection is ok */ public function prevent_injection_the_content( $inject = true, $content = '', $placements = array() ) { if ( ! $inject ) { return false; } global $post; if( empty( $post->ID ) ){ return true; } $post_ad_options = get_post_meta( $post->ID, '_advads_ad_settings', true ); return empty( $post_ad_options['disable_the_content'] ); } /** * check content length setting of content injection placements * * @since 1.2.3 * @param bol $return whether to inject or not * @param str $content post content * @param str $placement_id id of the placement * @return bool false if placement should not show up in current article */ public function check_placement_minimum_length( $return, $content = '', $placement_id ){ if ( ! $return ) { return false; } // get all placements $placements = Advanced_Ads::get_ad_placements_array(); if( ! isset( $placements[ $placement_id ]['options']['pro_minimum_length'] ) || ! $placements[ $placement_id ]['options']['pro_minimum_length'] ){ return $return; } if( defined('ADVADS_CURRENT_CONTENT_LENGTH') && ADVADS_CURRENT_CONTENT_LENGTH < absint( $placements[ $placement_id ]['options']['pro_minimum_length'] ) ){ return false; } return $return; } /** * echo ad before/after posts in loops on archive pages * * @since 1.2.1 * @param arr $post post object * @param WP_Query $wp_query query object */ public function inject_loop_post( $post, $wp_query = null ) { $is_ajax = defined( 'DOING_AJAX' ) && DOING_AJAX; if ( ! $wp_query instanceof WP_Query || is_feed() || ( is_admin() && ! $is_ajax ) ) { return; } $plugin_options = Advanced_Ads_Plugin::get_instance()->options(); // only inject on AJAX requests when Secondary Query option is enabled if ( ! empty( $plugin_options['disabled-ads']['secondary'] ) && $is_ajax ) { return; } if( ! isset( $wp_query->current_post )) { return; }; // don’t inject into main query on single pages. if( $wp_query->is_main_query() && is_single() ){ return; } $curr_index = $wp_query->current_post + 1; // normalize index // 'wp_reset_postdata()' does 'the_post' action. // handle the situation when wp_reset_postdata() is used after secondary query inside main query. static $handled_indexes = array(); if ( $wp_query->is_main_query() ) { if ( in_array( $curr_index, $handled_indexes ) ) { return; } $handled_indexes[] = $curr_index; } $placements = get_option( 'advads-ads-placements', array() ); if( is_array( $placements ) ){ foreach ( $placements as $_placement_id => $_placement ){ if ( empty($_placement['item']) ) { continue; } if ( isset($_placement['type']) && 'archive_pages' === $_placement['type'] ){ $_options = isset( $_placement['options'] ) ? $_placement['options'] : array(); if ( empty( $_options['in_any_loop'] ) && ( $wp_query->is_singular() || ! $wp_query->in_the_loop || ! $wp_query->is_main_query() ) ) { continue; } // check if the loop is outside of wp_head, but only on non-AJAX calls. if ( ! is_admin() && ! did_action( 'wp_head' ) ) { continue; } // don’t attach if not container attachment selected /*if( ! isset( $_options['pro_archive_pages_type'] ) || 'container' !== $_options['pro_archive_pages_type'] ){ continue; }*/ if( isset( $_options['pro_archive_pages_index'] ) ){ $ad_index = absint( $_options['pro_archive_pages_index'] ); if( $ad_index === $curr_index ){ // todo: leave a comment about the use of the next line. Might be needed to submit placement information to options. $_options['placement']['type'] = $_placement['type']; echo Advanced_Ads_Select::get_instance()->get_ad_by_method( $_placement_id, Advanced_Ads_Select::PLACEMENT, $_options ); } } } } } } /** * Insert an ad in the loop for archive pages created by AMP for WP (https://wordpress.org/plugins/accelerated-mobile-pages/) * * We can ommit the checks in inject_loop_post() here because the ampforwp_between_loop hook should provide the right position. * * @param int $count index of the current position in the loop. */ public function inject_loop_post_amp_for_wp( $count ) { $placements = get_option( 'advads-ads-placements', array() ); if ( is_array( $placements ) ) { foreach ( $placements as $_placement_id => $_placement ) { if ( empty( $_placement['item'] ) ) { continue; } if ( isset( $_placement['type'] ) && 'archive_pages' === $_placement['type'] ) { $_options = isset( $_placement['options'] ) ? $_placement['options'] : array(); if ( isset( $_options['pro_archive_pages_index'] ) ) { $ad_index = absint( $_options['pro_archive_pages_index'] ); // We need to reduce our index by one to match how AMP for WP counts the index. if ( ( $ad_index - 1 ) === $count ) { // Todo: leave a comment about the use of the next line. Might be needed to submit placement information to options. $_options['placement']['type'] = $_placement['type']; // phpcs:ignore echo Advanced_Ads_Select::get_instance()->get_ad_by_method( $_placement_id, Advanced_Ads_Select::PLACEMENT, $_options ); } } } } } } /** * Find the calls to `the_content` inside functions hooked to `the_content`. * * @return bool */ public function has_many_the_content() { global $wp_current_filter; if ( count( array_keys( $wp_current_filter, 'the_content', true ) ) > 1 ) { // More then one `the_content` in the stack. return true; } return false; } /** * Check whether or not to use JS to position the placement. * * @return bool */ private function content_random_use_js( $placement_options ) { if ( function_exists( 'advads_is_amp' ) && advads_is_amp() ) { return false; } if ( ! Advanced_Ads_Pro_Module_Cache_Busting::is_enabled() ) { return false; } if ( isset( $placement_options['cache-busting'] ) && 'off' === $placement_options['cache-busting'] // Check if we did not switch to `off` from `auto`. && ! isset( $placement_options['cache-busting-orig'] ) ) { return false; } return true; } /** * Inject a script to output before cache busting output. * The script moves the empty wrapper because some ad networks do not allow to move ads inserted to the DOM. * * @param array $r Cache busting item. * @param array $request Request info. * @return array $r Cache busting item. */ function inject_js_before_cache_busting_output( $r, $request ) { if ( ! isset( $request['method'] ) || 'placement' !== $request['method'] || empty( $request['args']['cache_busting_elementid'] ) ) { return $r; } $el_id = $request['args']['cache_busting_elementid']; $r['inject_before'][] = $this->get_output_js( $el_id, $request['args'] ); return $r; } /** * Add filters to skip paragraph. */ private function add_skip_paragraph_filters() { $placements = Advanced_Ads::get_ad_placements_array(); foreach ( $placements as $placement_id => $placement ) { if ( ! empty( $placement['options']['words_between_repeats'] ) ) { add_filter( 'advanced-ads-can-inject-into-content-' . $placement_id, array( $this, 'maybe_skip_content_placement' ), 10, 3 ); } } } /** * Check if the "Before/After Content" placement has enough words before. * * @param bool $return Whether to inject or not. * @param str $content Post content. * @param str $placement_id Placement id. * @return bool */ public function maybe_skip_content_placement( $return, $content = '', $placement_id ) { if ( ! $return ) { return false; } $placements = Advanced_Ads::get_ad_placements_array(); if ( empty( $placements[ $placement_id ]['type'] ) || ! in_array( $placements[ $placement_id ]['type'], array( 'post_top', 'post_bottom' ) ) ) { return $return; } if ( ! empty( $placements[ $placement_id ]['options']['words_between_repeats'] ) ) { $options['words_between_repeats'] = absint( $placements[ $placement_id ]['options']['words_between_repeats'] ); $offset_shifter = Advanced_Ads_Pro_Offset_Shifter::from_html( $content, $options ); if ( 'post_top' === $placements[ $placement_id ]['type'] ) { return $offset_shifter->can_inject_before_content_placement(); } else { return $offset_shifter->can_inject_after_content_placement(); } } return $return; } /** * Check if the ad should be displayed based on post type. * * @param bool $can_display True if the ad should be displayed, false otherwise. * @param Advanced_Ads_Ad $ad Advanced_Ads_Ad object. * @return bool True if the ad should be displayed, false otherwise. */ public function can_display_by_post_type( $can_display, Advanced_Ads_Ad $ad ) { if ( ! $can_display ) { return false; } $ad_options = $ad->options(); if ( ! empty( $ad_options['post']['post_type'] ) && $this->post_type_disabled( $ad_options['post']['post_type'] ) ) { return false; } return true; } /** * Return true if the ad can be displayed in the header of the page depending on the current post type. * * @param bool $can_display if the ad can already be displayed. * * @return bool */ public function can_display_in_header_by_post_type( $can_display ) { if ( ! $can_display ) { return false; } $post_type = $this->get_current_post_type(); if ( $this->post_type_disabled( $post_type ) ) { return false; } return true; } /** * Get current post type. * * @return bool|string False on failure or post type on success. */ private function get_current_post_type() { global $wp_the_query, $post; // If currently on a single site, use the main query information just in case a custom query is broken. if ( isset( $wp_the_query->post->post_type ) && $wp_the_query->is_single() ) { return $wp_the_query->post->post_type; } elseif ( isset( $post->post_type ) ) { return $post->post_type; } return false; } /** * Check if post type disabled. * * @param string $post_type Post type. * @return bool */ private function post_type_disabled( $post_type ) { $options = Advanced_Ads_Pro::get_instance()->get_options(); if ( ! empty( $options['general']['disable-by-post-types'] ) && is_array( $options['general']['disable-by-post-types'] ) && in_array( $post_type, $options['general']['disable-by-post-types'], true ) ) { return true; } return false; } /** * Add classes to the `body` tag. * * @param string[] $aa_classes Array of existing class names. * @return string[] $aa_classes Array of existing and new class names. */ public function body_class( $aa_classes ) { $post_type = $this->get_current_post_type(); if ( $this->post_type_disabled( $post_type ) ) { $aa_classes[] = 'aa-disabled-post-type'; } return $aa_classes; } }