- WordPress core y plugins - Tema Twenty Twenty-Four configurado - Plugin allow-unfiltered-html.php simplificado - .gitignore configurado para excluir wp-config.php y uploads 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
516 lines
15 KiB
PHP
Executable File
516 lines
15 KiB
PHP
Executable File
<?php // phpcs:ignore WordPress.Files.FileName
|
||
|
||
use AdvancedAds\Utilities\WordPress;
|
||
|
||
/**
|
||
* Visitor conditions under which to (not) show an ad
|
||
*
|
||
* @since 1.5.4
|
||
*/
|
||
class Advanced_Ads_Visitor_Conditions {
|
||
|
||
/**
|
||
* Instance of Advanced_Ads_Visitor_Conditions
|
||
*
|
||
* @var Advanced_Ads_Visitor_Conditions
|
||
*/
|
||
protected static $instance;
|
||
|
||
/**
|
||
* Registered visitor conditions
|
||
*
|
||
* @var array $conditions
|
||
*/
|
||
public $conditions;
|
||
|
||
/**
|
||
* Start of name in form elements
|
||
*/
|
||
const FORM_NAME = 'advanced_ad[visitors]';
|
||
|
||
/**
|
||
* Advanced_Ads_Visitor_Conditions constructor
|
||
*/
|
||
public function __construct() {
|
||
|
||
// register conditions.
|
||
$this->conditions = apply_filters(
|
||
'advanced-ads-visitor-conditions',
|
||
[
|
||
'mobile' => [
|
||
// type of the condition.
|
||
'label' => __( 'Device', 'advanced-ads' ),
|
||
'metabox' => [ 'Advanced_Ads_Visitor_Conditions', 'mobile_is_or_not' ], // callback to generate the metabox.
|
||
'check' => [ 'Advanced_Ads_Visitor_Conditions', 'check_device' ], // callback for frontend check.
|
||
'helplink' => 'https://wpadvancedads.com/manual/display-ads-either-on-mobile-or-desktop/?utm_source=advanced-ads&utm_medium=link&utm_campaign=condition-device',
|
||
'device_types' => [
|
||
'mobile' => [
|
||
'id' => 'mobile',
|
||
'label' => _x( 'Mobile', 'Device condition', 'advanced-ads' ),
|
||
],
|
||
'tablet' => [
|
||
'id' => 'tablet',
|
||
'label' => _x( 'Tablet', 'Device condition', 'advanced-ads' ),
|
||
],
|
||
'desktop' => [
|
||
'id' => 'desktop',
|
||
'label' => _x( 'Desktop', 'Device condition', 'advanced-ads' ),
|
||
],
|
||
],
|
||
],
|
||
'loggedin' => [
|
||
'label' => __( 'logged-in visitor', 'advanced-ads' ),
|
||
'description' => __( 'Whether the visitor has to be logged in or not in order to see the ad.', 'advanced-ads' ),
|
||
'metabox' => [ 'Advanced_Ads_Visitor_Conditions', 'metabox_is_or_not' ], // callback to generate the metabox.
|
||
'check' => [ 'Advanced_Ads_Visitor_Conditions', 'check_logged_in' ], // callback for frontend check.
|
||
'helplink' => 'https://wpadvancedads.com/manual/logged-in-visitors/?utm_source=advanced-ads&utm_medium=link&utm_campaign=condition-logged-in-visitors',
|
||
'passive_info' => [
|
||
'hash_fields' => null,
|
||
'remove' => 'login',
|
||
'function' => 'is_user_logged_in',
|
||
],
|
||
],
|
||
]
|
||
);
|
||
}
|
||
|
||
/**
|
||
* Load instance of Advanced_Ads_Visitor_Conditions
|
||
*
|
||
* @return Advanced_Ads_Visitor_Conditions
|
||
*/
|
||
public static function get_instance() {
|
||
// If the single instance hasn't been set, set it now.
|
||
if ( null === self::$instance ) {
|
||
self::$instance = new self();
|
||
}
|
||
|
||
return self::$instance;
|
||
}
|
||
|
||
|
||
/**
|
||
* Get the conditions array alphabetically by label
|
||
*
|
||
* @since 1.8.12
|
||
*/
|
||
public function get_conditions() {
|
||
uasort( $this->conditions, [ WordPress::class, 'sort_array_by_label' ] );
|
||
|
||
return $this->conditions;
|
||
}
|
||
|
||
/**
|
||
* Callback to render the mobile condition using the "is not" condition
|
||
*
|
||
* @param array $options options of the condition.
|
||
* @param int $index index of the condition.
|
||
* @param string $form_name name of the form, falls back to class constant.
|
||
*/
|
||
public static function mobile_is_or_not( $options, $index = 0, $form_name = '' ) {
|
||
|
||
if ( ! isset( $options['type'] ) || '' === $options['type'] ) {
|
||
return;
|
||
}
|
||
|
||
$type_options = self::get_instance()->conditions;
|
||
|
||
if ( ! isset( $type_options[ $options['type'] ] ) ) {
|
||
return;
|
||
}
|
||
|
||
// options.
|
||
$operator = isset( $options['operator'] ) ? $options['operator'] : 'is';
|
||
|
||
// convert previous binary option to device selector.
|
||
if ( ! array_key_exists( 'value', $options ) ) {
|
||
$options['value'] = 'is' === $operator ? [ 'tablet', 'mobile' ] : [ 'desktop' ];
|
||
$operator = 'is';
|
||
}
|
||
|
||
$type_options[ $options['type'] ]['device_types'] = array_map(
|
||
function ( $device_type ) use ( $options ) {
|
||
$device_type['checked'] = in_array( $device_type['id'], $options['value'], true );
|
||
|
||
return $device_type;
|
||
},
|
||
$type_options[ $options['type'] ]['device_types']
|
||
);
|
||
|
||
// form name basis.
|
||
$name = self::get_form_name_with_index( $form_name, $index );
|
||
|
||
include ADVADS_ABSPATH . 'admin/views/conditions/condition-device.php';
|
||
}
|
||
|
||
/**
|
||
* Callback to display the "is not" condition
|
||
*
|
||
* @param array $options options of the condition.
|
||
* @param int $index index of the condition.
|
||
* @param string $form_name name of the form, falls back to class constant.
|
||
*/
|
||
public static function metabox_is_or_not( $options, $index = 0, $form_name = '' ) {
|
||
if ( ! isset( $options['type'] ) || '' === $options['type'] ) {
|
||
return;
|
||
}
|
||
|
||
$type_options = self::get_instance()->conditions;
|
||
if ( ! isset( $type_options[ $options['type'] ] ) ) {
|
||
return;
|
||
}
|
||
|
||
// form name basis.
|
||
$name = self::get_form_name_with_index( $form_name, $index );
|
||
$operator = isset( $options['operator'] ) ? $options['operator'] : 'is';
|
||
|
||
include ADVADS_ABSPATH . 'admin/views/conditions/condition-is-or-not.php';
|
||
}
|
||
|
||
/**
|
||
* Callback to display the any condition based on a number
|
||
*
|
||
* @param array $options options of the condition.
|
||
* @param int $index index of the condition.
|
||
* @param string $form_name name of the form, falls back to class constant.
|
||
*/
|
||
public static function metabox_number( $options, $index = 0, $form_name = '' ) {
|
||
|
||
if ( ! isset( $options['type'] ) || '' === $options['type'] ) {
|
||
return;
|
||
}
|
||
|
||
$type_options = self::get_instance()->conditions;
|
||
|
||
if ( ! isset( $type_options[ $options['type'] ] ) ) {
|
||
return;
|
||
}
|
||
|
||
// form name basis.
|
||
$name = self::get_form_name_with_index( $form_name, $index );
|
||
|
||
// options.
|
||
$value = isset( $options['value'] ) ? $options['value'] : 0;
|
||
$operator = isset( $options['operator'] ) ? $options['operator'] : 'is_equal';
|
||
|
||
include ADVADS_ABSPATH . 'admin/views/conditions/condition-number.php';
|
||
}
|
||
|
||
/**
|
||
* Callback to display the any condition based on a number
|
||
*
|
||
* @param array $options options of the condition.
|
||
* @param int $index index of the condition.
|
||
* @param string $form_name name of the form, falls back to class constant.
|
||
*/
|
||
public static function metabox_string( $options, $index = 0, $form_name = '' ) {
|
||
|
||
if ( ! isset( $options['type'] ) || '' === $options['type'] ) {
|
||
return;
|
||
}
|
||
|
||
$type_options = self::get_instance()->conditions;
|
||
|
||
if ( ! isset( $type_options[ $options['type'] ] ) ) {
|
||
return;
|
||
}
|
||
|
||
// form name basis.
|
||
$name = self::get_form_name_with_index( $form_name, $index );
|
||
|
||
// options.
|
||
$value = isset( $options['value'] ) ? $options['value'] : '';
|
||
$operator = isset( $options['operator'] ) ? $options['operator'] : 'contains';
|
||
|
||
include ADVADS_ABSPATH . 'admin/views/conditions/condition-string.php';
|
||
}
|
||
|
||
/**
|
||
* Controls frontend checks for conditions
|
||
*
|
||
* @param array $options Options of the condition.
|
||
* @param bool|Ad $ad Ad instance.
|
||
*
|
||
* @return bool false, if ad can’t be delivered
|
||
*/
|
||
public static function frontend_check( $options = [], $ad = false ) {
|
||
$visitor_conditions = self::get_instance()->conditions;
|
||
|
||
if ( is_array( $options ) && isset( $visitor_conditions[ $options['type'] ]['check'] ) ) {
|
||
$check = $visitor_conditions[ $options['type'] ]['check'];
|
||
} else {
|
||
return true;
|
||
}
|
||
|
||
// call frontend check callback.
|
||
if ( method_exists( $check[0], $check[1] ) ) {
|
||
return call_user_func( [ $check[0], $check[1] ], $options, $ad );
|
||
}
|
||
|
||
return true;
|
||
}
|
||
|
||
/**
|
||
* Render the list of set visisor conditions
|
||
*
|
||
* @param array $set_conditions array of existing conditions.
|
||
* @param string $list_target ID of the list with the conditions.
|
||
* @param string $form_name prefix of the form.
|
||
*/
|
||
public static function render_condition_list( array $set_conditions, $list_target = '', $form_name = '' ) {
|
||
|
||
$conditions = self::get_instance()->get_conditions();
|
||
|
||
// use default form name if not given explicitly.
|
||
// TODO: create a random form name, in case we have more than one form per page and the parameter is not given.
|
||
$form_name = ! $form_name ? self::FORM_NAME : $form_name;
|
||
|
||
include ADVADS_ABSPATH . 'admin/views/conditions/visitor-conditions-list.php';
|
||
|
||
/**
|
||
* Prepare condition form
|
||
*
|
||
* @todo if needed, allow to disable the form to add new conditions
|
||
*/
|
||
|
||
// add mockup conditions if add-ons are missing.
|
||
$pro_conditions = [];
|
||
if ( ! defined( 'AAP_VERSION' ) ) {
|
||
$pro_conditions[] = __( 'browser language', 'advanced-ads' );
|
||
$pro_conditions[] = __( 'cookie', 'advanced-ads' );
|
||
$pro_conditions[] = __( 'max. ad clicks', 'advanced-ads' );
|
||
$pro_conditions[] = __( 'max. ad impressions', 'advanced-ads' );
|
||
$pro_conditions[] = __( 'new visitor', 'advanced-ads' );
|
||
$pro_conditions[] = __( 'page impressions', 'advanced-ads' );
|
||
$pro_conditions[] = __( 'geo location', 'advanced-ads' );
|
||
$pro_conditions[] = __( 'referrer url', 'advanced-ads' );
|
||
$pro_conditions[] = __( 'user agent', 'advanced-ads' );
|
||
$pro_conditions[] = __( 'user can (capabilities)', 'advanced-ads' );
|
||
$pro_conditions[] = __( 'user role', 'advanced-ads' );
|
||
$pro_conditions[] = __( 'browser width', 'advanced-ads' );
|
||
}
|
||
|
||
asort( $pro_conditions );
|
||
|
||
// the action to call using AJAX.
|
||
$action = 'load_visitor_conditions_metabox';
|
||
$connector_default = 'and';
|
||
|
||
$empty_options = ! is_array( $set_conditions ) || ! count( $set_conditions );
|
||
|
||
include ADVADS_ABSPATH . 'admin/views/conditions/conditions-form.php';
|
||
}
|
||
|
||
/**
|
||
* Render connector option
|
||
*
|
||
* @param int $index incremental index of the options.
|
||
* @param string $value connector value.
|
||
* @param string $form_name name of the form, falls back to class constant.
|
||
*
|
||
* @return string HTML of the connector option
|
||
* @todo combine this with the same function used for Display Conditions
|
||
*
|
||
* @since 1.7.0.4
|
||
*/
|
||
public static function render_connector_option( $index, $value, $form_name ) {
|
||
|
||
$label = ( 'or' === $value ) ? __( 'or', 'advanced-ads' ) : __( 'and', 'advanced-ads' );
|
||
|
||
$name = self::get_form_name_with_index( $form_name, $index );
|
||
|
||
// create random value to identify the form field.
|
||
$rand = uniqid();
|
||
|
||
return '<input type="checkbox" name="' . $name . '[connector]' . '" value="or" id="advads-conditions-' . // phpcs:ignore
|
||
$index . '-connector-' . $rand . '"' .
|
||
checked( 'or', $value, false )
|
||
. '><label for="advads-conditions-' . $index . '-connector-' . $rand . '">' . $label . '</label>';
|
||
}
|
||
|
||
/**
|
||
* Helper function to the name of a form field.
|
||
* falls back to default
|
||
*
|
||
* @param string $form_name form name if submitted.
|
||
* @param int $index index of the condition.
|
||
*
|
||
* @return string
|
||
*/
|
||
public static function get_form_name_with_index( $form_name = '', $index = 0 ) {
|
||
return ! empty( $form_name ) ? $form_name . '[' . $index . ']' : self::FORM_NAME . '[' . $index . ']';
|
||
}
|
||
|
||
/**
|
||
* Check whether device visitor condition in frontend is true.
|
||
*
|
||
* @param array $options options of the condition.
|
||
*
|
||
* @return bool
|
||
*/
|
||
public static function check_device( $options = [] ) {
|
||
if ( ! array_key_exists( 'value', $options ) ) {
|
||
return self::check_mobile( $options );
|
||
}
|
||
|
||
$mobile_detect = new \Detection\MobileDetect();
|
||
// register callbacks to decide whether device "is".
|
||
$callbacks = array_intersect_key(
|
||
[
|
||
'mobile' => function () use ( $mobile_detect ) {
|
||
return $mobile_detect->isMobile() && ! $mobile_detect->isTablet();
|
||
},
|
||
'tablet' => function () use ( $mobile_detect ) {
|
||
return $mobile_detect->isTablet();
|
||
},
|
||
'desktop' => function () use ( $mobile_detect ) {
|
||
return ! $mobile_detect->isTablet() && ! $mobile_detect->isMobile();
|
||
},
|
||
],
|
||
array_flip( $options['value'] )
|
||
);
|
||
|
||
// Only call devices that are part of the condition.
|
||
array_walk(
|
||
$callbacks,
|
||
function ( callable &$value ) {
|
||
$value = $value();
|
||
}
|
||
);
|
||
|
||
return array_filter( $callbacks ) !== [];
|
||
}
|
||
|
||
/**
|
||
* Check mobile visitor condition in frontend
|
||
*
|
||
* @param array $options options of the condition.
|
||
* @deprecated -- Only used if new options hasn't been saved
|
||
*
|
||
* @return bool
|
||
*/
|
||
private static function check_mobile( $options ) {
|
||
if ( ! isset( $options['operator'] ) ) {
|
||
return true;
|
||
}
|
||
|
||
switch ( $options['operator'] ) {
|
||
case 'is':
|
||
if ( ! wp_is_mobile() ) {
|
||
return false;
|
||
}
|
||
break;
|
||
case 'is_not':
|
||
if ( wp_is_mobile() ) {
|
||
return false;
|
||
}
|
||
break;
|
||
}
|
||
|
||
return true;
|
||
}
|
||
|
||
/**
|
||
* Check mobile visitor condition in frontend
|
||
*
|
||
* @param array $options options of the condition.
|
||
*
|
||
* @return bool true if can be displayed
|
||
* @since 1.6.3
|
||
*/
|
||
public static function check_logged_in( $options = [] ) {
|
||
|
||
if ( ! isset( $options['operator'] ) ) {
|
||
return true;
|
||
}
|
||
|
||
switch ( $options['operator'] ) {
|
||
case 'is':
|
||
if ( ! is_user_logged_in() ) {
|
||
return false;
|
||
}
|
||
break;
|
||
case 'is_not':
|
||
if ( is_user_logged_in() ) {
|
||
return false;
|
||
}
|
||
break;
|
||
}
|
||
|
||
return true;
|
||
}
|
||
|
||
/**
|
||
* Helper for check with strings
|
||
*
|
||
* @param string $string string that is going to be checked.
|
||
* @param array $options options of this condition.
|
||
*
|
||
* @return bool true if ad can be displayed
|
||
* @since 1.6.3
|
||
*/
|
||
public static function helper_check_string( $string = '', $options = [] ) { // phpcs:ignore
|
||
if ( ! isset( $options['operator'] ) || empty( $options['value'] ) ) {
|
||
return true;
|
||
}
|
||
|
||
$operator = $options['operator'];
|
||
$string = (string) maybe_serialize( $string );
|
||
$value = (string) maybe_serialize( $options['value'] );
|
||
|
||
// check the condition by mode and bool.
|
||
$condition = true;
|
||
switch ( $operator ) {
|
||
// referrer contains string on any position.
|
||
case 'contain':
|
||
$condition = stripos( $string, $value ) !== false;
|
||
break;
|
||
// referrer does not contain string on any position.
|
||
case 'contain_not':
|
||
$condition = stripos( $string, $value ) === false;
|
||
break;
|
||
// referrer starts with the string.
|
||
case 'start':
|
||
$condition = stripos( $string, $value ) === 0;
|
||
break;
|
||
// referrer does not start with the string.
|
||
case 'start_not':
|
||
$condition = stripos( $string, $value ) !== 0;
|
||
break;
|
||
// referrer ends with the string.
|
||
case 'end':
|
||
$condition = substr( $string, - strlen( $value ) ) === $value;
|
||
break;
|
||
// referrer does not end with the string.
|
||
case 'end_not':
|
||
$condition = substr( $string, - strlen( $value ) ) !== $value;
|
||
break;
|
||
// referrer is equal to the string.
|
||
case 'match':
|
||
// strings do match, but should not or not match but should.
|
||
$condition = strcasecmp( $value, $string ) === 0;
|
||
break;
|
||
// referrer is not equal to the string.
|
||
case 'match_not':
|
||
// strings do match, but should not or not match but should.
|
||
$condition = strcasecmp( $value, $string ) !== 0;
|
||
break;
|
||
case 'regex':
|
||
case 'regex_not':
|
||
$condition = @preg_match( sprintf( '/%s/', $value ), $string ); // phpcs:ignore WordPress.PHP.NoSilencedErrors.Discouraged
|
||
// If the return value is `false`, the regex is incorrect.
|
||
if ( false === $condition ) {
|
||
Advanced_Ads::log( "Advanced Ads: regular expression '$value' in visitor condition is broken." );
|
||
break;
|
||
}
|
||
|
||
if ( 'regex_not' === $operator ) {
|
||
$condition = ! $condition;
|
||
}
|
||
break;
|
||
}
|
||
|
||
return $condition;
|
||
}
|
||
}
|