abstracts
1 day ago
admin
1 day ago
ads
1 day ago
compatibility
1 day ago
crons
1 day ago
frontend
1 day ago
groups
1 day ago
importers
1 day ago
installation
1 year ago
interfaces
4 months ago
license
3 months ago
placements
1 day ago
rest
1 year ago
traits
1 day ago
utilities
1 day ago
cap_map.php
3 years ago
class-assets-registry.php
3 weeks ago
class-autoloader.php
1 day ago
class-cache-invalidator.php
1 day ago
class-constants.php
1 year ago
class-content-injector.php
3 weeks ago
class-entities.php
3 months ago
class-modal.php
1 year ago
class-modules.php
1 year ago
class-options.php
1 year ago
class-plugin.php
1 day ago
class-post-data.php
10 months ago
class-shortcodes.php
1 day ago
class-upgrades.php
1 year ago
class-widget.php
11 months ago
default-hooks.php
4 months ago
functions-ad.php
1 day ago
functions-components.php
3 months ago
functions-conditional.php
1 year ago
functions-core.php
1 year ago
functions-group.php
1 day ago
functions-placement.php
1 day ago
functions.php
1 day ago
index.php
2 years ago
load_modules.php
2 years ago
class-widget.php
304 lines
| 1 | <?php |
| 2 | /** |
| 3 | * Advanced Ads Widget |
| 4 | * |
| 5 | * @package AdvancedAds |
| 6 | * @author Advanced Ads <info@wpadvancedads.com> |
| 7 | * @since 1.47.0 |
| 8 | */ |
| 9 | |
| 10 | namespace AdvancedAds; |
| 11 | |
| 12 | use AdvancedAds\Utilities\Data; |
| 13 | |
| 14 | defined( 'ABSPATH' ) || exit; |
| 15 | |
| 16 | /** |
| 17 | * Ad widget |
| 18 | */ |
| 19 | class Widget extends \WP_Widget { |
| 20 | |
| 21 | /** |
| 22 | * The constructor. |
| 23 | */ |
| 24 | public function __construct() { |
| 25 | $prefix = wp_advads()->get_frontend_prefix(); |
| 26 | $classname = $prefix . 'widget'; |
| 27 | |
| 28 | $widget_ops = [ |
| 29 | 'classname' => $classname, |
| 30 | 'show_instance_in_rest' => true, |
| 31 | 'description' => __( 'Display Ads and Ad Groups.', 'advanced-ads' ), |
| 32 | ]; |
| 33 | $control_ops = []; |
| 34 | $base_id = self::get_base_id(); |
| 35 | |
| 36 | parent::__construct( $base_id, 'Advanced Ads', $widget_ops, $control_ops ); |
| 37 | |
| 38 | add_filter( 'q2w3-fixed-widgets', [ $this, 'q2w3_replace_frontend_id' ] ); |
| 39 | } |
| 40 | |
| 41 | /** |
| 42 | * Echoes the widget content. |
| 43 | * |
| 44 | * @param array $args Display arguments including 'before_title', 'after_title', 'before_widget', and 'after_widget'. |
| 45 | * @param array $instance The settings for the particular instance of the widget. |
| 46 | * |
| 47 | * @return void |
| 48 | */ |
| 49 | public function widget( $args, $instance ): void { |
| 50 | $item_id = empty( $instance['item_id'] ) ? '' : $instance['item_id']; |
| 51 | $output = self::output( $item_id ); |
| 52 | if ( ! $output ) { |
| 53 | return; |
| 54 | } |
| 55 | |
| 56 | $title = apply_filters( 'widget_title', empty( $instance['title'] ) ? '' : $instance['title'], $instance, $this->id_base ); |
| 57 | $before_widget = isset( $args['before_widget'] ) ? $args['before_widget'] : ''; |
| 58 | $after_widget = isset( $args['after_widget'] ) ? $args['after_widget'] : ''; |
| 59 | |
| 60 | $before_widget = $this->maybe_replace_frontend_id( $before_widget, $instance ); |
| 61 | |
| 62 | // phpcs:disable WordPress.Security.EscapeOutput.OutputNotEscaped |
| 63 | echo $before_widget; |
| 64 | if ( ! empty( $title ) ) { |
| 65 | echo $args['before_title'] . $title . $args['after_title']; |
| 66 | } |
| 67 | echo $output; |
| 68 | echo $after_widget; |
| 69 | // phpcs:enable WordPress.Security.EscapeOutput.OutputNotEscaped |
| 70 | } |
| 71 | |
| 72 | /** |
| 73 | * Updates a particular instance of a widget. |
| 74 | * |
| 75 | * This function should check that `$new_instance` is set correctly. The newly-calculated |
| 76 | * value of `$instance` should be returned. If false is returned, the instance won't be |
| 77 | * saved/updated. |
| 78 | * |
| 79 | * @param array $new_instance New settings for this instance as input by the user via WP_Widget::form(). |
| 80 | * @param array $old_instance Old settings for this instance. |
| 81 | * @return array Settings to save or bool false to cancel saving. |
| 82 | */ |
| 83 | public function update( $new_instance, $old_instance ) { |
| 84 | $instance = $old_instance; |
| 85 | $instance['title'] = $new_instance['title']; |
| 86 | $instance['item_id'] = $new_instance['item_id']; |
| 87 | |
| 88 | // Allow to remove/replace id for new widgets and if it was allowed earlier. |
| 89 | if ( [] === $old_instance || ! empty( $old_instance['remove-widget-id'] ) ) { |
| 90 | $instance['remove-widget-id'] = true; |
| 91 | } |
| 92 | |
| 93 | return $instance; |
| 94 | } |
| 95 | |
| 96 | /** |
| 97 | * Outputs the settings update form. |
| 98 | * |
| 99 | * @param array $instance Current settings. |
| 100 | */ |
| 101 | public function form( $instance ) { |
| 102 | $instance = wp_parse_args( |
| 103 | (array) $instance, |
| 104 | [ |
| 105 | 'title' => '', |
| 106 | 'item_id' => '', |
| 107 | ] |
| 108 | ); |
| 109 | |
| 110 | $elementid = $instance['item_id']; |
| 111 | $title = wp_strip_all_tags( $instance['title'] ); |
| 112 | $items = array_merge( self::items_for_select(), self::widget_placements_for_select() ); |
| 113 | ?> |
| 114 | <p> |
| 115 | <label for="<?php echo esc_attr( $this->get_field_id( 'title' ) ); ?>"><?php esc_html_e( 'Title:', 'advanced-ads' ); ?></label> |
| 116 | <input class="widefat" id="<?php echo esc_attr( $this->get_field_id( 'title' ) ); ?>" name="<?php echo esc_attr( $this->get_field_name( 'title' ) ); ?>" type="text" value="<?php echo esc_attr( $title ); ?>" /> |
| 117 | </p> |
| 118 | <select id="<?php echo esc_attr( $this->get_field_id( 'item_id' ) ); ?>" name="<?php echo esc_attr( $this->get_field_name( 'item_id' ) ); ?>"> |
| 119 | <option value=""><?php esc_html_e( '--empty--', 'advanced-ads' ); ?></option> |
| 120 | <?php if ( isset( $items['placements'] ) ) : ?> |
| 121 | <optgroup label="<?php esc_html_e( 'Placements', 'advanced-ads' ); ?>"> |
| 122 | <?php foreach ( $items['placements'] as $_item_id => $_item_title ) : ?> |
| 123 | <option value="<?php echo esc_attr( $_item_id ); ?>" <?php selected( $_item_id, $elementid ); ?>><?php echo esc_attr( $_item_title ); ?></option> |
| 124 | <?php endforeach; ?> |
| 125 | </optgroup> |
| 126 | <?php endif; ?> |
| 127 | <?php if ( isset( $items['groups'] ) ) : ?> |
| 128 | <optgroup label="<?php esc_html_e( 'Ad Groups', 'advanced-ads' ); ?>"> |
| 129 | <?php foreach ( $items['groups'] as $_item_id => $_item_title ) : ?> |
| 130 | <option value="<?php echo esc_attr( $_item_id ); ?>" <?php selected( $_item_id, $elementid ); ?>><?php echo esc_html( $_item_title ); ?></option> |
| 131 | <?php endforeach; ?> |
| 132 | </optgroup> |
| 133 | <?php endif; ?> |
| 134 | <?php if ( isset( $items['ads'] ) ) : ?> |
| 135 | <optgroup label="<?php esc_html_e( 'Ads', 'advanced-ads' ); ?>"> |
| 136 | <?php foreach ( $items['ads'] as $_item_id => $_item_title ) : ?> |
| 137 | <option value="<?php echo esc_attr( $_item_id ); ?>" <?php selected( $_item_id, $elementid ); ?>><?php echo esc_html( $_item_title ); ?></option> |
| 138 | <?php endforeach; ?> |
| 139 | </optgroup> |
| 140 | <?php endif; ?> |
| 141 | </select> |
| 142 | |
| 143 | <?php |
| 144 | $this->display_hints( $elementid ); |
| 145 | } |
| 146 | |
| 147 | /** |
| 148 | * Display hints related to the currently selected item in the dropdown. |
| 149 | * |
| 150 | * @param string $elementid Currently selected item. |
| 151 | */ |
| 152 | private function display_hints( $elementid ) { |
| 153 | $elementid_parts = explode( '_', $elementid ); |
| 154 | if ( ! isset( $elementid_parts[1] ) || 'group' !== $elementid_parts[0] ) { |
| 155 | return; |
| 156 | } |
| 157 | |
| 158 | $group = wp_advads_get_group( absint( $elementid_parts[1] ) ); |
| 159 | if ( ! $group ) { |
| 160 | return; |
| 161 | } |
| 162 | |
| 163 | echo $group->get_hints_html(); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped |
| 164 | } |
| 165 | |
| 166 | /** |
| 167 | * Get items for widget select field |
| 168 | * |
| 169 | * @return array $select items for select field. |
| 170 | */ |
| 171 | public static function items_for_select() { |
| 172 | return Data::items_for_select( [ 'placements' => false ] ); |
| 173 | } |
| 174 | |
| 175 | /** |
| 176 | * Get widget placements for select field |
| 177 | * |
| 178 | * @return array $items for select field. |
| 179 | */ |
| 180 | public static function widget_placements_for_select() { |
| 181 | $select = []; |
| 182 | $placements = wp_advads_get_placements(); |
| 183 | |
| 184 | foreach ( $placements as $key => $placement ) { |
| 185 | if ( $placement->is_type( [ 'sidebar_widget', 'default' ] ) ) { |
| 186 | $select['placements'][ 'placement_' . $key ] = $placement->get_title(); |
| 187 | } |
| 188 | } |
| 189 | |
| 190 | if ( ! empty( $select['placements'] ) ) { |
| 191 | asort( $select['placements'] ); |
| 192 | } |
| 193 | |
| 194 | return $select; |
| 195 | } |
| 196 | |
| 197 | /** |
| 198 | * Return content of an in a widget |
| 199 | * |
| 200 | * @param string $id slug of the display. |
| 201 | * @return bool|string |
| 202 | */ |
| 203 | public static function output( $id = '' ) { |
| 204 | // Early bail!! |
| 205 | if ( empty( $id ) ) { |
| 206 | return; |
| 207 | } |
| 208 | |
| 209 | $item = explode( '_', $id, 2 ); |
| 210 | if ( isset( $item[1] ) ) { |
| 211 | $item_id = $item[1]; |
| 212 | } |
| 213 | |
| 214 | if ( empty( $item_id ) ) { |
| 215 | return; |
| 216 | } |
| 217 | |
| 218 | $func = 'get_the_' . $item[0]; |
| 219 | return $func( absint( $item_id ) ); |
| 220 | } |
| 221 | |
| 222 | /** |
| 223 | * Get the base id of the widget |
| 224 | * |
| 225 | * @return string |
| 226 | */ |
| 227 | public static function get_base_id() { |
| 228 | $options = \Advanced_Ads::get_instance()->options(); |
| 229 | |
| 230 | // deprecated to keep previously changed prefixed working. |
| 231 | $prefix2 = ( isset( $options['id-prefix'] ) && '' !== $options['id-prefix'] ) ? $options['id-prefix'] : 'advads_ad_'; |
| 232 | return $prefix2 . 'widget'; |
| 233 | } |
| 234 | |
| 235 | /** |
| 236 | * Get frontend widget id. |
| 237 | * |
| 238 | * @param int $number Unique ID number of the current widget instance. |
| 239 | * @return string |
| 240 | */ |
| 241 | private function get_frontend_id( $number ) { |
| 242 | $prefix = wp_advads()->get_frontend_prefix(); |
| 243 | return $prefix . 'widget-' . $number; |
| 244 | } |
| 245 | |
| 246 | /** |
| 247 | * Make it harder for ad blockers to block the widget. |
| 248 | * removes the pre-defined widget ID (e.g., advads_ad_widget-20) and replaces it with one that uses the individual frontend prefix |
| 249 | * |
| 250 | * @param string $before_widget content before the widget. |
| 251 | * @param array $instance Settings for the current widget instance. |
| 252 | * @return string $before_widget |
| 253 | */ |
| 254 | private function maybe_replace_frontend_id( $before_widget, $instance ) { |
| 255 | if ( |
| 256 | ! empty( $instance['remove-widget-id'] ) |
| 257 | || defined( 'JNEWS_THEME_ID' ) // the JNews theme overrides the widget ID and resets it, so we target this specifically. |
| 258 | ) { |
| 259 | $pattern = '#\sid=("|\')[^"\']+["\']#'; |
| 260 | if ( |
| 261 | ( defined( 'ADVANCED_ADS_SHOW_WIDGET_ID' ) && ADVANCED_ADS_SHOW_WIDGET_ID ) |
| 262 | || ! empty( $instance['q2w3_fixed_widget'] ) |
| 263 | ) { |
| 264 | // Replace id. |
| 265 | $number = ! empty( $this->number ) ? $this->number : ''; |
| 266 | $before_widget = preg_replace( $pattern, ' id=$01' . $this->get_frontend_id( $number ) . '$01', $before_widget ); |
| 267 | } else { |
| 268 | // Remove id. |
| 269 | $before_widget = preg_replace( $pattern, '', $before_widget ); |
| 270 | } |
| 271 | } |
| 272 | |
| 273 | return $before_widget; |
| 274 | } |
| 275 | |
| 276 | /** |
| 277 | * Provide the 'Q2W3 Fixed Widget' plugin with the new frontend widget id. |
| 278 | * |
| 279 | * @param array $sidebars_widgets existing sidebar widgets. |
| 280 | * @return array $sidebars_widgets |
| 281 | */ |
| 282 | public function q2w3_replace_frontend_id( $sidebars_widgets ) { |
| 283 | foreach ( $sidebars_widgets as $sidebar => $widgets ) { |
| 284 | foreach ( $widgets as $k => $widget ) { |
| 285 | // after Fixed Widget 5.3.0, the widget option includes '#'. It didn’t before. We store the information since we need it later again. |
| 286 | $has_hash = strpos( $widget, '#' ) !== false; |
| 287 | $widget = str_replace( '#', '', $widget ); |
| 288 | $pos = strrpos( $widget, '-' ); |
| 289 | $option_name = substr( $widget, 0, $pos ); |
| 290 | $number = substr( $widget, $pos + 1 ); |
| 291 | |
| 292 | if ( self::get_base_id() === $option_name ) { |
| 293 | $widget_options = get_option( 'widget_' . $option_name ); |
| 294 | if ( ! empty( $widget_options[ $number ]['remove-widget-id'] ) ) { |
| 295 | // Add a hash if the widget had one before. See comment above. |
| 296 | $sidebars_widgets[ $sidebar ][ $k ] = $has_hash ? ( '#' . $this->get_frontend_id( $number ) ) : $this->get_frontend_id( $number ); |
| 297 | } |
| 298 | } |
| 299 | } |
| 300 | } |
| 301 | return $sidebars_widgets; |
| 302 | } |
| 303 | } |
| 304 |