Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Leverage URL metrics to reserve space for embeds to reduce CLS #1373

Merged
merged 59 commits into from
Oct 14, 2024
Merged
Show file tree
Hide file tree
Changes from 43 commits
Commits
Show all changes
59 commits
Select commit Hold shift + click to select a range
0a376c1
Introduce methods to get minumum height of element
westonruter Jul 17, 2024
8aa7e63
Set the min-height of an embed prior to it loading
westonruter Jul 29, 2024
11f98f4
Set min-height on embed-wrapper instead of figure container
westonruter Jul 30, 2024
d7cc7cf
Use 500px as a better representation of an element that could be LCP
westonruter Jul 30, 2024
48e57e8
Add test for existing style manipulation
westonruter Jul 30, 2024
0d285f2
Add helper generator method to get all elements
westonruter Jul 31, 2024
4778a3d
Try using MutationObserve to watch for embed height changes
westonruter Aug 14, 2024
1dbd4a1
Use the more appropriate ResizeObserver instead of MutationObserver
westonruter Aug 14, 2024
a4bab7e
Remove condition that breaks monitoring resizes of post embeds
westonruter Aug 14, 2024
5f0cdbe
Introduce client-side Optimization Detective extensions and move Embe…
westonruter Aug 17, 2024
0ba2d6e
Override clientBoundingRect once embed has loaded
westonruter Aug 17, 2024
5f4189d
Move jsdoc types to types.d.ts for reuse
westonruter Aug 18, 2024
bf2b3c5
Send URL metric when leaving the page
westonruter Aug 18, 2024
b9bad0d
Use boundingClientRect instead of intersectionRect in get_all_element…
westonruter Aug 18, 2024
c6b02ec
Eliminate timeout for disconneccting ResizeObsever
westonruter Aug 18, 2024
52a2260
Move extension initialization after idle callback
westonruter Aug 18, 2024
edc52fa
Fix warning when prematurely applying buffered text replacements, esp…
westonruter Aug 18, 2024
820d66d
Prepend min-height to style attribute instead of appending
westonruter Aug 19, 2024
def2aab
Use object spread
westonruter Aug 22, 2024
cd9a618
Merge branch 'trunk' of https://github.com/WordPress/performance into…
westonruter Aug 22, 2024
02c4fd9
Merge branch 'trunk' into add/embed-optimizer-min-height-reservation
westonruter Sep 13, 2024
1da219f
Use get_json_params() instead of get_params() so _wpnonce query param…
westonruter Sep 17, 2024
e34d9fe
Implement resizedBoundingClientRect extended property in schema
westonruter Sep 17, 2024
5db6f54
Fix testing JSON request
westonruter Sep 18, 2024
0fa263a
Go back to get_params() by ignoring _wpnonce
westonruter Sep 18, 2024
72b285d
Merge branch 'trunk' into add/embed-optimizer-min-height-reservation
westonruter Sep 19, 2024
2a723f7
Fix jsdoc
westonruter Sep 29, 2024
71dd914
Merge branch 'trunk' of https://github.com/WordPress/performance into…
westonruter Oct 3, 2024
29d4383
Eliminate use of deprecated property
westonruter Oct 4, 2024
a529218
Add breakpoint-specific min-heights to account for responsive embeds
westonruter Oct 4, 2024
fa8a34e
Add od_generate_media_query() helper
westonruter Oct 4, 2024
1e40f84
Break up embed tag visitor into separate methods
westonruter Oct 4, 2024
5d4d5b2
Bump alpha versions
westonruter Oct 8, 2024
5f1c2ac
Add missing short-circuit in case EMBED_OPTIMIZER_VERSION is defined
westonruter Oct 8, 2024
915e1e7
Rework bootstrap logic to wait until init priority 9 and add od_init …
westonruter Oct 8, 2024
26ae396
Add test for when resizedBoundingClientRect data not available
westonruter Oct 8, 2024
cd80ed1
Remove obsolete short-circuiting now that OD dependency version is ch…
westonruter Oct 8, 2024
bd008c5
Evolve get_all_url_metrics_groups_elements into get_all_denormalized_…
westonruter Oct 8, 2024
1b5cf13
Add Embed Optimizer tests
westonruter Oct 8, 2024
a70df28
Account for error when passing single-item array to min() or max()
westonruter Oct 8, 2024
4e48d3d
Add test for Image Prioritizer
westonruter Oct 8, 2024
d17cace
Remove now-unused method to get element minimum hights
westonruter Oct 8, 2024
5574081
Improve handling of get_updated_html
westonruter Oct 8, 2024
19c0425
Add test for get_all_denormalized_elements
westonruter Oct 9, 2024
ea36bac
Add tests for new OD code
westonruter Oct 9, 2024
01c083d
Clarify purpose of overridden get_updated_html method
westonruter Oct 9, 2024
c5d6991
Add missing since tags
westonruter Oct 9, 2024
455ef4f
Clarify handling of embed block tags and embed wrapper tags
westonruter Oct 9, 2024
a390e15
Replace tuple with assoc array
westonruter Oct 9, 2024
a760705
Add doc block for detect.js
westonruter Oct 9, 2024
7ca1fbc
Add API functions to pass to finalize callbacks to avoid direct mutat…
westonruter Oct 11, 2024
f66445f
Improve error handling
westonruter Oct 11, 2024
6e0aa8e
Harden types and disallow setting core properties
westonruter Oct 11, 2024
9e99e0d
Reuse sets for reserved property keys
westonruter Oct 11, 2024
46ba7e3
Move functions to root of module
westonruter Oct 11, 2024
0bc521e
Fix TypeScript error related to embedWrapper
westonruter Oct 11, 2024
477cc33
Add missing period to return doc
westonruter Oct 14, 2024
dca43e9
Eliminate needless ternary
westonruter Oct 14, 2024
73d2252
Rename amend to extend
westonruter Oct 14, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
127 changes: 120 additions & 7 deletions plugins/embed-optimizer/class-embed-optimizer-tag-visitor.php
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,38 @@ final class Embed_Optimizer_Tag_Visitor {
*/
protected $added_lazy_script = false;

/**
* Determines whether the processor is currently at a figure.wp-block-embed tag.
*
* @since n.e.x.t
*
* @param OD_HTML_Tag_Processor $processor Processor.
* @return bool Whether at the tag.
*/
private function is_embed_figure( OD_HTML_Tag_Processor $processor ): bool {
return (
'FIGURE' === $processor->get_tag()
&&
true === $processor->has_class( 'wp-block-embed' )
);
}

/**
* Determines whether the processor is currently at a div.wp-block-embed__wrapper tag.
*
* @since n.e.x.t
*
* @param OD_HTML_Tag_Processor $processor Processor.
* @return bool Whether the tag should be measured and stored in URL metrics
westonruter marked this conversation as resolved.
Show resolved Hide resolved
*/
private function is_embed_wrapper( OD_HTML_Tag_Processor $processor ): bool {
return (
'DIV' === $processor->get_tag()
&&
true === $processor->has_class( 'wp-block-embed__wrapper' )
);
}

/**
* Visits a tag.
*
Expand All @@ -36,16 +68,23 @@ final class Embed_Optimizer_Tag_Visitor {
*/
public function __invoke( OD_Tag_Visitor_Context $context ): bool {
$processor = $context->processor;
if ( ! (
'FIGURE' === $processor->get_tag()
&&
true === $processor->has_class( 'wp-block-embed' )
) ) {

/*
* The only thing we need to do if it is a div.wp-block-embed__wrapper tag is return true so that the tag
* will get measured and stored in the URL Metrics.
*/
if ( $this->is_embed_wrapper( $processor ) ) {
return true;
}

// Short-circuit if not a figure.wp-block-embed tag.
if ( ! $this->is_embed_figure( $processor ) ) {
return false;
}

$max_intersection_ratio = $context->url_metric_group_collection->get_element_max_intersection_ratio( $processor->get_xpath() );
$this->reduce_layout_shifts( $context );

$max_intersection_ratio = $context->url_metric_group_collection->get_element_max_intersection_ratio( $this->get_embed_wrapper_xpath( $context ) );
if ( $max_intersection_ratio > 0 ) {
/*
* The following embeds have been chosen for optimization due to their relative popularity among all embed types.
Expand Down Expand Up @@ -119,6 +158,80 @@ public function __invoke( OD_Tag_Visitor_Context $context ): bool {
$this->added_lazy_script = true;
}

return true;
/*
* At this point the tag is a figure.wp-block-embed, and we can return false because this does not need to be
* measured and stored in URL Metrics. Only the child div.wp-block-embed__wrapper tag is measured and stored
* so that this visitor can look up the height to set as a min-height on the figure.wp-block-embed. For more
* information on what the return values mean for tag visitors, see <https://github.com/WordPress/performance/issues/1342>.
*/
return false;
}

/**
* Gets the XPath for the embed wrapper.
*
* @since n.e.x.t
*
* @param OD_Tag_Visitor_Context $context Tag visitor context.
* @return string XPath.
*/
private function get_embed_wrapper_xpath( OD_Tag_Visitor_Context $context ): string {
return $context->processor->get_xpath() . '/*[1][self::DIV]';
felixarntz marked this conversation as resolved.
Show resolved Hide resolved
}

/**
* Reduces layout shifts.
*
* @since n.e.x.t
*
* @param OD_Tag_Visitor_Context $context Tag visitor context.
*/
private function reduce_layout_shifts( OD_Tag_Visitor_Context $context ): void {
$processor = $context->processor;
$embed_wrapper_xpath = $this->get_embed_wrapper_xpath( $context );

/**
* Array of tuples of groups and their minimum heights keyed by the minimum viewport width.
*
* @var array<int, array{OD_URL_Metric_Group, int}> $group_minimum_heights
*/
$group_minimum_heights = array();

$denormalized_elements = $context->url_metric_group_collection->get_all_denormalized_elements()[ $embed_wrapper_xpath ] ?? array();
foreach ( $denormalized_elements as list( $group, $url_metric, $element ) ) {
Comment on lines +217 to +218
Copy link
Member Author

@westonruter westonruter Oct 9, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

See note below on the get_all_denormalized_elements() method that I'd like to instead have $element be an OD_Element class instance which has a $url_metric property to access the OD_URL_Metric instance if is a part of. In the same way, the OD_URL_Metric class can have a $group property to indicate the OD_URL_Metric_Group it is a part of. This would eliminate the need to pass back an array of tuples and instead. So instead, to get the $group this code could do $element->url_metric->group.

This I want to do in a follow-up PR.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

See #1585

if ( ! isset( $element['resizedBoundingClientRect'] ) ) {
continue;
}
$group_min_width = $group->get_minimum_viewport_width();
if ( ! isset( $group_minimum_heights[ $group_min_width ] ) ) {
$group_minimum_heights[ $group_min_width ] = array( $group, $element['resizedBoundingClientRect']['height'] );
} else {
$group_minimum_heights[ $group_min_width ][1] = min(
$group_minimum_heights[ $group_min_width ][1],
$element['resizedBoundingClientRect']['height']
);
}
felixarntz marked this conversation as resolved.
Show resolved Hide resolved
}

// Add style rules to set the min-height for each viewport group.
if ( count( $group_minimum_heights ) > 0 ) {
$element_id = $processor->get_attribute( 'id' );
if ( ! is_string( $element_id ) ) {
$element_id = 'embed-optimizer-' . md5( $processor->get_xpath() );
$processor->set_attribute( 'id', $element_id );
}

$style_rules = array();
foreach ( $group_minimum_heights as list( $group, $minimum_height ) ) {
$style_rules[] = sprintf(
'@media %s { #%s { min-height: %dpx; } }',
od_generate_media_query( $group->get_minimum_viewport_width(), $group->get_maximum_viewport_width() ),
$element_id,
$minimum_height
);
}

$processor->append_head_html( sprintf( "<style>\n%s\n</style>\n", join( "\n", $style_rules ) ) );
}
}
}
90 changes: 90 additions & 0 deletions plugins/embed-optimizer/detect.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,90 @@
const consoleLogPrefix = '[Embed Optimizer]';
felixarntz marked this conversation as resolved.
Show resolved Hide resolved

/**
* @typedef {import("../optimization-detective/types.d.ts").ElementMetrics} ElementMetrics
* @typedef {import("../optimization-detective/types.d.ts").URLMetric} URLMetric
*/

/**
* Log a message.
*
* @param {...*} message
*/
function log( ...message ) {
// eslint-disable-next-line no-console
console.log( consoleLogPrefix, ...message );
}

/**
* Embed element heights.
*
* @type {Map<string, DOMRectReadOnly>}
*/
const loadedElementContentRects = new Map();

/**
* Initialize.
*
* @param {Object} args Args.
* @param {boolean} args.isDebug Whether to show debug messages.
*/
export async function initialize( { isDebug } ) {
const embedWrappers =
/** @type NodeListOf<HTMLDivElement> */ document.querySelectorAll(
'.wp-block-embed > .wp-block-embed__wrapper[data-od-xpath]'
);

for ( const embedWrapper of embedWrappers ) {
monitorEmbedWrapperForResizes( embedWrapper );
}

if ( isDebug ) {
log( 'Loaded embed content rects:', loadedElementContentRects );
}
}

/**
* Finalize.
*
* @param {Object} args Args.
* @param {boolean} args.isDebug Whether to show debug messages.
* @param {URLMetric} args.urlMetric Pending URL metric.
*/
export async function finalize( { urlMetric, isDebug } ) {
if ( isDebug ) {
log( 'URL metric to be sent:', urlMetric );
}

for ( const element of urlMetric.elements ) {
if ( loadedElementContentRects.has( element.xpath ) ) {
if ( isDebug ) {
log(
`boundingClientRect for ${ element.xpath } resized:`,
element.boundingClientRect,
'=>',
loadedElementContentRects.get( element.xpath )
);
}
element.resizedBoundingClientRect = loadedElementContentRects.get(
element.xpath
);
}
}
}

/**
* Monitors embed wrapper for resizes.
*
* @param {HTMLDivElement} embedWrapper Embed wrapper DIV.
*/
function monitorEmbedWrapperForResizes( embedWrapper ) {
if ( ! ( 'odXpath' in embedWrapper.dataset ) ) {
throw new Error( 'Embed wrapper missing data-od-xpath attribute.' );
}
const xpath = embedWrapper.dataset.odXpath;
const observer = new ResizeObserver( ( entries ) => {
const [ entry ] = entries;
loadedElementContentRects.set( xpath, entry.contentRect );
} );
observer.observe( embedWrapper, { box: 'content-box' } );
}
123 changes: 115 additions & 8 deletions plugins/embed-optimizer/hooks.php
Original file line number Diff line number Diff line change
Expand Up @@ -18,14 +18,53 @@
function embed_optimizer_add_hooks(): void {
add_action( 'wp_head', 'embed_optimizer_render_generator' );

if ( defined( 'OPTIMIZATION_DETECTIVE_VERSION' ) ) {
add_action( 'od_register_tag_visitors', 'embed_optimizer_register_tag_visitors' );
} else {
add_filter( 'embed_oembed_html', 'embed_optimizer_filter_oembed_html' );
}
add_action( 'od_init', 'embed_optimizer_init_optimization_detective' );
add_action( 'wp_loaded', 'embed_optimizer_add_non_optimization_detective_hooks' );
}
add_action( 'init', 'embed_optimizer_add_hooks' );

/**
* Adds hooks for when the Optimization Detective logic is not running.
*
* @since n.e.x.t
*/
function embed_optimizer_add_non_optimization_detective_hooks(): void {
if ( false === has_action( 'od_register_tag_visitors', 'embed_optimizer_register_tag_visitors' ) ) {
add_filter( 'embed_oembed_html', 'embed_optimizer_filter_oembed_html_to_lazy_load' );
}
}

/**
* Initializes Embed Optimizer when Optimization Detective has loaded.
*
* @since n.e.x.t
*
* @param string $optimization_detective_version Current version of the optimization detective plugin.
*/
function embed_optimizer_init_optimization_detective( string $optimization_detective_version ): void {
$required_od_version = '0.7.0';
if ( version_compare( (string) strtok( $optimization_detective_version, '-' ), $required_od_version, '<' ) ) {
add_action(
'admin_notices',
static function (): void {
global $pagenow;
if ( ! in_array( $pagenow, array( 'index.php', 'plugins.php' ), true ) ) {
return;
}
wp_admin_notice(
esc_html__( 'The Embed Optimizer plugin requires a newer version of the Optimization Detective plugin. Please update your plugins.', 'embed-optimizer' ),
array( 'type' => 'warning' )
);
}
);
return;
}

add_action( 'od_register_tag_visitors', 'embed_optimizer_register_tag_visitors' );
add_filter( 'embed_oembed_html', 'embed_optimizer_filter_oembed_html_to_detect_embed_presence' );
add_filter( 'od_url_metric_schema_element_item_additional_properties', 'embed_optimizer_add_element_item_schema_properties' );
}

/**
* Registers the tag visitor for embeds.
*
Expand All @@ -40,17 +79,85 @@ function embed_optimizer_register_tag_visitors( OD_Tag_Visitor_Registry $registr
}

/**
* Filter the oEmbed HTML.
* Filters additional properties for the element item schema for Optimization Detective.
*
* @since n.e.x.t
*
* @param array<string, array{type: string}> $additional_properties Additional properties.
* @return array<string, array{type: string}> Additional properties.
*/
function embed_optimizer_add_element_item_schema_properties( array $additional_properties ): array {
$additional_properties['resizedBoundingClientRect'] = array(
'type' => 'object',
'properties' => array_fill_keys(
array(
'width',
'height',
'x',
'y',
'top',
'right',
'bottom',
'left',
),
array(
'type' => 'number',
'required' => true,
)
),
);
return $additional_properties;
}

/**
* Filters the list of Optimization Detective extension module URLs to include the extension for Embed Optimizer.
*
* @since n.e.x.t
*
* @param string[]|mixed $extension_module_urls Extension module URLs.
* @return string[] Extension module URLs.
*/
function embed_optimizer_filter_extension_module_urls( $extension_module_urls ): array {
if ( ! is_array( $extension_module_urls ) ) {
$extension_module_urls = array();
}
$extension_module_urls[] = add_query_arg( 'ver', EMBED_OPTIMIZER_VERSION, plugin_dir_url( __FILE__ ) . 'detect.js' );
return $extension_module_urls;
}

/**
* Filter the oEmbed HTML to detect when an embed is present so that the Optimization Detective extension module can be enqueued.
*
* This ensures that the module for handling embeds is only loaded when there is an embed on the page.
*
* @since n.e.x.t
*
* @param string|mixed $html The oEmbed HTML.
* @return string Unchanged oEmbed HTML.
*/
function embed_optimizer_filter_oembed_html_to_detect_embed_presence( $html ): string {
if ( ! is_string( $html ) ) {
$html = '';
}
add_filter( 'od_extension_module_urls', 'embed_optimizer_filter_extension_module_urls' );
return $html;
}

/**
* Filter the oEmbed HTML to lazy load the embed.
*
* Add loading="lazy" to any iframe tags.
* Lazy load any script tags.
*
* @since 0.1.0
*
* @param string $html The oEmbed HTML.
* @param string|mixed $html The oEmbed HTML.
* @return string Filtered oEmbed HTML.
*/
function embed_optimizer_filter_oembed_html( string $html ): string {
function embed_optimizer_filter_oembed_html_to_lazy_load( $html ): string {
if ( ! is_string( $html ) ) {
$html = '';
}
$html_processor = new WP_HTML_Tag_Processor( $html );
if ( embed_optimizer_update_markup( $html_processor, true ) ) {
add_action( 'wp_footer', 'embed_optimizer_lazy_load_scripts' );
Expand Down
Loading