Ticket #40854: 40854-widget-shortcodes.3.diff
File 40854-widget-shortcodes.3.diff, 12.6 KB (added by , 8 years ago) |
---|
-
src/wp-includes/default-filters.php
diff --git src/wp-includes/default-filters.php src/wp-includes/default-filters.php index a196f4ff19..9fb06cb083 100644
add_filter( 'widget_text_content', 'capital_P_dangit', 11 ); 169 169 add_filter( 'widget_text_content', 'wptexturize' ); 170 170 add_filter( 'widget_text_content', 'convert_smilies', 20 ); 171 171 add_filter( 'widget_text_content', 'wpautop' ); 172 add_filter( 'widget_text_content', 'shortcode_unautop' ); 173 add_filter( 'widget_text_content', 'do_shortcode', 11 ); // Runs after wpautop() but note that $post global will be null when shortcodes run. 172 174 173 175 add_filter( 'date_i18n', 'wp_maybe_decline_date' ); 174 176 -
src/wp-includes/widgets/class-wp-widget-text.php
diff --git src/wp-includes/widgets/class-wp-widget-text.php src/wp-includes/widgets/class-wp-widget-text.php index a79e0daa6c..13a737ab6e 100644
class WP_Widget_Text extends WP_Widget { 183 183 * 184 184 * @since 2.8.0 185 185 * 186 * @global WP_Post $post 187 * 186 188 * @param array $args Display arguments including 'before_title', 'after_title', 187 189 * 'before_widget', and 'after_widget'. 188 190 * @param array $instance Settings for the current Text widget instance. 189 191 */ 190 192 public function widget( $args, $instance ) { 193 global $post; 191 194 192 195 /** This filter is documented in wp-includes/widgets/class-wp-widget-pages.php */ 193 196 $title = apply_filters( 'widget_title', empty( $instance['title'] ) ? '' : $instance['title'], $instance, $this->id_base ); … … class WP_Widget_Text extends WP_Widget { 205 208 } 206 209 207 210 /* 208 * Just-in-time temporarily upgrade Visual Text widget shortcode handling209 * (with support added by plugin) from the widget_text filter to210 * widget_text_content:11 to prevent wpautop from corrupting HTML output211 * added by the shortcode.211 * Suspend legacy plugin-supplied do_shortcode() for 'widget_text' filter for the visual Text widget to prevent 212 * shortcodes being processed twice. Now do_shortcode() is added to the 'widget_text_content' filter in core itself 213 * and it applies after wpautop() to prevent corrupting HTML output added by the shortcode. When do_shortcode() is 214 * added to 'widget_text_content' then do_shortcode() will be manually called when in legacy mode as well. 212 215 */ 213 216 $widget_text_do_shortcode_priority = has_filter( 'widget_text', 'do_shortcode' ); 214 $should_ upgrade_shortcode_handling= ( $is_visual_text_widget && false !== $widget_text_do_shortcode_priority );215 if ( $should_ upgrade_shortcode_handling) {217 $should_suspend_legacy_shortcode_support = ( $is_visual_text_widget && false !== $widget_text_do_shortcode_priority ); 218 if ( $should_suspend_legacy_shortcode_support ) { 216 219 remove_filter( 'widget_text', 'do_shortcode', $widget_text_do_shortcode_priority ); 217 add_filter( 'widget_text_content', 'do_shortcode', 11 ); 220 } 221 222 // Nullify the $post global during widget rendering to prevent shortcodes from running with the unexpected context. 223 $suspended_post = null; 224 if ( isset( $post ) ) { 225 $suspended_post = $post; 226 $post = null; 218 227 } 219 228 220 229 /** … … class WP_Widget_Text extends WP_Widget { 244 253 * @param WP_Widget_Text $this Current Text widget instance. 245 254 */ 246 255 $text = apply_filters( 'widget_text_content', $text, $instance, $this ); 256 } else { 257 if ( ! empty( $instance['filter'] ) ) { 258 $text = wpautop( $text ); // Back-compat for instances prior to 4.8. 259 } 260 261 /* 262 * Manually do shortcodes on the content when the core-added filter is present. It is added by default 263 * in core by adding do_shortcode() to the 'widget_text_content' filter to apply after wpautop(). 264 * Since the legacy Text widget runs wpautop() after 'widget_text' filters are applied, the widget in 265 * legacy mode here manually applies do_shortcode() on the content unless the default 266 * core filter for 'widget_text_content' has been removed, or if do_shortcode() has already 267 * been applied via a plugin adding do_shortcode() to 'widget_text' filters. 268 */ 269 if ( has_filter( 'widget_text_content', 'do_shortcode' ) && ! $widget_text_do_shortcode_priority ) { 270 271 if ( ! empty( $instance['filter'] ) ) { 272 $text = shortcode_unautop( $text ); 273 } 274 275 $text = do_shortcode( $text ); 276 } 277 } 247 278 248 } elseif ( ! empty( $instance['filter'] ) ) { 249 $text = wpautop( $text ); // Back-compat for instances prior to 4.8. 279 // Restore post global. 280 if ( isset( $suspended_post ) ) { 281 $post = $suspended_post; 250 282 } 251 283 252 // Undo temporary upgrade of the plugin-supplied shortcode handling. 253 if ( $should_upgrade_shortcode_handling ) { 254 remove_filter( 'widget_text_content', 'do_shortcode', 11 ); 284 // Undo suspension of legacy plugin-supplied shortcode handling. 285 if ( $should_suspend_legacy_shortcode_support ) { 255 286 add_filter( 'widget_text', 'do_shortcode', $widget_text_do_shortcode_priority ); 256 287 } 257 288 -
tests/phpunit/tests/widgets/text-widget.php
diff --git tests/phpunit/tests/widgets/text-widget.php tests/phpunit/tests/widgets/text-widget.php index f7e810fd04..a3e348f8a6 100644
class Test_WP_Widget_Text extends WP_UnitTestCase { 222 222 * 223 223 * @var string 224 224 */ 225 protected $example_shortcode_content = "<p>One\nTwo\n\nThree</p>\n<script>\ndocument.write('Test1');\n\ndocument.write('Test2');\n</script>"; 225 protected $example_shortcode_content = "<p class='sortcodep'>One\nTwo\n\nThree\n\nThis is testing the <code>[example note='This will not get processed since it is part of shortcode output itself.']</code> shortcode.</p>\n<script>\ndocument.write('Test1');\n\ndocument.write('Test2');\n</script>"; 226 227 /** 228 * The captured global post during shortcode rendering. 229 * 230 * @var WP_Post|null 231 */ 232 protected $post_during_shortcode = null; 233 234 /** 235 * Number of times the shortcode was rendered. 236 * 237 * @var int 238 */ 239 protected $shortcode_render_count = 0; 226 240 227 241 /** 228 242 * Do example shortcode. … … class Test_WP_Widget_Text extends WP_UnitTestCase { 230 244 * @return string Shortcode content. 231 245 */ 232 246 function do_example_shortcode() { 247 $this->post_during_shortcode = get_post(); 248 $this->shortcode_render_count++; 233 249 return $this->example_shortcode_content; 234 250 } 235 251 236 252 /** 237 * Test widget method w hen a plugin has added shortcode support.253 * Test widget method with shortcodes. 238 254 * 239 255 * @covers WP_Widget_Text::widget 240 256 */ 241 257 function test_widget_shortcodes() { 258 global $post; 259 $post_id = $this->factory()->post->create(); 260 $post = get_post( $post_id ); 261 242 262 $args = array( 243 263 'before_title' => '<h2>', 244 264 'after_title' => "</h2>\n", … … class Test_WP_Widget_Text extends WP_UnitTestCase { 246 266 'after_widget' => "</section>\n", 247 267 ); 248 268 $widget = new WP_Widget_Text(); 249 add_filter( 'widget_text', 'do_shortcode' );250 269 add_shortcode( 'example', array( $this, 'do_example_shortcode' ) ); 251 270 252 271 $base_instance = array( 253 272 'title' => 'Example', 254 'text' => "This is an example:\n\n[example] ",273 'text' => "This is an example:\n\n[example]\n\nHello.", 255 274 'filter' => false, 256 275 ); 257 276 258 // Legacy Text Widget .277 // Legacy Text Widget without wpautop. 259 278 $instance = array_merge( $base_instance, array( 260 279 'filter' => false, 261 280 ) ); 281 $this->shortcode_render_count = 0; 262 282 ob_start(); 263 283 $widget->widget( $args, $instance ); 264 284 $output = ob_get_clean(); 285 $this->assertEquals( 1, $this->shortcode_render_count ); 286 $this->assertNotContains( '[example]', $output, 'Expected shortcode to be processed in legacy widget with plugin adding filter' ); 265 287 $this->assertContains( $this->example_shortcode_content, $output, 'Shortcode was applied without wpautop corrupting it.' ); 266 $this->assertEquals( 10, has_filter( 'widget_text', 'do_shortcode' ), 'Filter was restored.' ); 288 $this->assertNotContains( '<p>' . $this->example_shortcode_content . '</p>', $output, 'Expected shortcode_unautop() to have run.' ); 289 $this->assertNull( $this->post_during_shortcode ); 267 290 268 // Visual Text Widget.291 // Legacy Text Widget with wpautop. 269 292 $instance = array_merge( $base_instance, array( 270 'filter' => 'content', 293 'filter' => true, 294 'visual' => false, 271 295 ) ); 296 $this->shortcode_render_count = 0; 272 297 ob_start(); 273 298 $widget->widget( $args, $instance ); 274 299 $output = ob_get_clean(); 300 $this->assertEquals( 1, $this->shortcode_render_count ); 301 $this->assertNotContains( '[example]', $output, 'Expected shortcode to be processed in legacy widget with plugin adding filter' ); 275 302 $this->assertContains( $this->example_shortcode_content, $output, 'Shortcode was applied without wpautop corrupting it.' ); 276 $this->assert Equals( 10, has_filter( 'widget_text', 'do_shortcode' ), 'Filter was restored.' );277 $this->assert False( has_filter( 'widget_text_content', 'do_shortcode' ), 'Filter was removed.');303 $this->assertNotContains( '<p>' . $this->example_shortcode_content . '</p>', $output, 'Expected shortcode_unautop() to have run.' ); 304 $this->assertNull( $this->post_during_shortcode ); 278 305 279 // Visual Text Widget with properly-used widget_text_content filter. 306 // Legacy text widget with plugin adding shortcode support as well. 307 add_filter( 'widget_text', 'do_shortcode' ); 308 $this->shortcode_render_count = 0; 309 ob_start(); 310 $widget->widget( $args, $instance ); 311 $output = ob_get_clean(); 312 $this->assertEquals( 1, $this->shortcode_render_count ); 313 $this->assertNotContains( '[example]', $output, 'Expected shortcode to be processed in legacy widget with plugin adding filter' ); 314 $this->assertContains( wpautop( $this->example_shortcode_content ), $output, 'Shortcode was applied *with* wpautop() applying to shortcode output since plugin used legacy filter.' ); 315 $this->assertNull( $this->post_during_shortcode ); 280 316 remove_filter( 'widget_text', 'do_shortcode' ); 281 add_filter( 'widget_text_content', 'do_shortcode', 11 ); 317 282 318 $instance = array_merge( $base_instance, array( 283 'filter' => 'content', 319 'filter' => true, 320 'visual' => true, 284 321 ) ); 322 323 // Visual Text Widget with only core-added widget_text_content filter for do_shortcode. 324 $this->assertFalse( has_filter( 'widget_text', 'do_shortcode' ) ); 325 $this->assertEquals( 11, has_filter( 'widget_text_content', 'do_shortcode' ), 'Expected core to have set do_shortcode as widget_text_content filter.' ); 326 $this->shortcode_render_count = 0; 285 327 ob_start(); 286 328 $widget->widget( $args, $instance ); 287 329 $output = ob_get_clean(); 330 $this->assertEquals( 1, $this->shortcode_render_count ); 288 331 $this->assertContains( $this->example_shortcode_content, $output, 'Shortcode was applied without wpautop corrupting it.' ); 289 $this->assertFalse( has_filter( 'widget_text', 'do_shortcode' ), 'Filter was not erroneously restored.' ); 332 $this->assertNotContains( '<p>' . $this->example_shortcode_content . '</p>', $output, 'Expected shortcode_unautop() to have run.' ); 333 $this->assertFalse( has_filter( 'widget_text', 'do_shortcode' ), 'The widget_text filter still lacks do_shortcode handler.' ); 334 $this->assertEquals( 11, has_filter( 'widget_text_content', 'do_shortcode' ), 'The widget_text_content filter still has do_shortcode handler.' ); 335 $this->assertNull( $this->post_during_shortcode ); 336 337 // Visual Text Widget with both filters applied added, one from core and another via plugin. 338 add_filter( 'widget_text', 'do_shortcode' ); 339 $this->shortcode_render_count = 0; 340 ob_start(); 341 $widget->widget( $args, $instance ); 342 $output = ob_get_clean(); 343 $this->assertEquals( 1, $this->shortcode_render_count ); 344 $this->assertContains( $this->example_shortcode_content, $output, 'Shortcode was applied without wpautop corrupting it.' ); 345 $this->assertNotContains( '<p>' . $this->example_shortcode_content . '</p>', $output, 'Expected shortcode_unautop() to have run.' ); 346 $this->assertEquals( 10, has_filter( 'widget_text', 'do_shortcode' ), 'Expected do_shortcode to be restored to widget_text.' ); 347 $this->assertNull( $this->post_during_shortcode ); 348 $this->assertNull( $this->post_during_shortcode ); 349 remove_filter( 'widget_text', 'do_shortcode' ); 350 351 // Visual Text Widget with shortcode handling disabled via plugin removing filter. 352 remove_filter( 'widget_text_content', 'do_shortcode', 11 ); 353 remove_filter( 'widget_text', 'do_shortcode' ); 354 $this->shortcode_render_count = 0; 355 ob_start(); 356 $widget->widget( $args, $instance ); 357 $output = ob_get_clean(); 358 $this->assertEquals( 0, $this->shortcode_render_count ); 359 $this->assertContains( '[example]', $output ); 360 $this->assertNotContains( $this->example_shortcode_content, $output ); 361 $this->assertFalse( has_filter( 'widget_text', 'do_shortcode' ) ); 362 $this->assertFalse( has_filter( 'widget_text_content', 'do_shortcode' ) ); 290 363 } 291 364 292 365 /**