From 5732aee6ca1ef00ccecee44faf1705604f2efb45 Mon Sep 17 00:00:00 2001 From: Dennis Snell Date: Wed, 2 Aug 2023 15:55:20 -0700 Subject: [PATCH 1/5] HTML API: Add functions to read inner and outer HTML. --- .../html-api/class-wp-html-open-elements.php | 45 ++++ .../html-api/class-wp-html-processor.php | 192 +++++++++++++++++- .../wpHtmlProcessorGetInnerMarkup.php | 126 ++++++++++++ .../wpHtmlProcessorGetOuterMarkup.php | 128 ++++++++++++ .../html-api/wpHtmlProcessorSetOuterHtml.php | 0 5 files changed, 483 insertions(+), 8 deletions(-) create mode 100644 tests/phpunit/tests/html-api/wpHtmlProcessorGetInnerMarkup.php create mode 100644 tests/phpunit/tests/html-api/wpHtmlProcessorGetOuterMarkup.php create mode 100644 tests/phpunit/tests/html-api/wpHtmlProcessorSetOuterHtml.php diff --git a/src/wp-includes/html-api/class-wp-html-open-elements.php b/src/wp-includes/html-api/class-wp-html-open-elements.php index fe5625545b0ac..4fb1cc2e8de40 100644 --- a/src/wp-includes/html-api/class-wp-html-open-elements.php +++ b/src/wp-includes/html-api/class-wp-html-open-elements.php @@ -37,6 +37,17 @@ class WP_HTML_Open_Elements { */ public $stack = array(); + /** + * Holds functions added to be called after popping an element off the stack. + * + * Listeners are passed the WP_HTML_Token for the item that was removed. + * + * @since 6.4.0 + * + * @var array + */ + private $after_pop_listeners = array(); + /** * Whether a P element is in button scope currently. * @@ -428,5 +439,39 @@ public function after_element_pop( $item ) { $this->has_p_in_button_scope = $this->has_element_in_button_scope( 'P' ); break; } + + // Call any listeners that are registered. + foreach ( $this->after_pop_listeners as $listener ) { + call_user_func( $listener, $item ); + } + } + + /** + * Creates a context in which a given listener is called after + * popping an element off of the stack of open elements. + * + * It's unlikely that you will need this function. It exists + * to aid an optimization in the `WP_HTML_Processor` and the + * strange form of calling a generator inside a `foreach` + * loop ensures that proper cleanup of the listener occurs. + * + * Example: + * + * $did_close = false; + * $closed_a_p = function ( $item ) use ( &$did_close ) { $did_close = 'P' === $item->node_name; }; + * foreach ( $stack_of_open_elements->with_pop_listener( $closed_a_p ) ) { + * while ( ! $did_close && $processor->next_tag() ) { + * // This loop executes until _any_ P element is closed. + * } + * } + * + * @since 6.4.0 + * + * @param callable $listener Called with the WP_HTML_Token for the item that was popped off of the stack. + */ + public function with_pop_listener( $listener ) { + $this->after_pop_listeners[] = $listener; + yield; + array_pop( $this->after_pop_listeners ); } } diff --git a/src/wp-includes/html-api/class-wp-html-processor.php b/src/wp-includes/html-api/class-wp-html-processor.php index b8e1093054726..3a2439b5341b8 100644 --- a/src/wp-includes/html-api/class-wp-html-processor.php +++ b/src/wp-includes/html-api/class-wp-html-processor.php @@ -322,7 +322,7 @@ public function __construct( $html, $use_the_static_create_methods_instead = nul * be found or if content in the document caused the processor * to give up and abort processing. * - * Example + * Example: * * $processor = WP_HTML_Processor::create_fragment( '