Skip to content

Commit 9e59c8b

Browse files
committed
Change minimum viewport width to be exclusive whereas the maximum width remains inclusive
This will allow for creating media queries that avoid any 1px gaps. The key code change is in `OD_URL_Metric_Group::is_viewport_width_in_range()` (and `isViewportNeeded` in `detect.js`). This also avoid the need to add the number 1 to the minimum viewport width which can be confusing given the known breakpoints. This instead moves the addition of 1 at the moment to `od_generate_media_query()` but this will be removed with resolution of #1696. Require that a viewport width and height for a URL Metric be at least 1px.
1 parent d94977f commit 9e59c8b

14 files changed

+92
-77
lines changed

plugins/optimization-detective/class-od-link-collection.php

+5-5
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@
1818
* @phpstan-type Link array{
1919
* attributes: LinkAttributes,
2020
* minimum_viewport_width: int<0, max>|null,
21-
* maximum_viewport_width: positive-int|null
21+
* maximum_viewport_width: int<1, max>|null
2222
* }
2323
*
2424
* @phpstan-type LinkAttributes array{
@@ -53,9 +53,9 @@ final class OD_Link_Collection implements Countable {
5353
*
5454
* @phpstan-param LinkAttributes $attributes
5555
*
56-
* @param array $attributes Attributes.
57-
* @param int<0, max>|null $minimum_viewport_width Minimum width or null if not bounded or relevant.
58-
* @param positive-int|null $maximum_viewport_width Maximum width or null if not bounded (i.e. infinity) or relevant.
56+
* @param array $attributes Attributes.
57+
* @param int<0, max>|null $minimum_viewport_width Minimum width (exclusive) or null if not bounded or relevant.
58+
* @param int<1, max>|null $maximum_viewport_width Maximum width (inclusive) or null if not bounded (i.e. infinity) or relevant.
5959
*
6060
* @throws InvalidArgumentException When invalid arguments are provided.
6161
*/
@@ -194,7 +194,7 @@ static function ( array $carry, array $link ): array {
194194
&&
195195
is_int( $last_link['maximum_viewport_width'] )
196196
&&
197-
$last_link['maximum_viewport_width'] + 1 === $link['minimum_viewport_width']
197+
$last_link['maximum_viewport_width'] === $link['minimum_viewport_width']
198198
) {
199199
$last_link['maximum_viewport_width'] = max( $last_link['maximum_viewport_width'], $link['maximum_viewport_width'] );
200200

plugins/optimization-detective/class-od-url-metric-group-collection.php

+7-7
Original file line numberDiff line numberDiff line change
@@ -244,13 +244,13 @@ public function clear_cache(): void {
244244
* @return OD_URL_Metric_Group[] Groups.
245245
*/
246246
private function create_groups(): array {
247-
$groups = array();
248-
$min_width = 0;
249-
foreach ( $this->breakpoints as $max_width ) {
250-
$groups[] = new OD_URL_Metric_Group( array(), $min_width, $max_width, $this->sample_size, $this->freshness_ttl, $this );
251-
$min_width = $max_width + 1;
247+
$groups = array();
248+
$min_width_exclusive = 0;
249+
foreach ( $this->breakpoints as $max_width_inclusive ) {
250+
$groups[] = new OD_URL_Metric_Group( array(), $min_width_exclusive, $max_width_inclusive, $this->sample_size, $this->freshness_ttl, $this );
251+
$min_width_exclusive = $max_width_inclusive;
252252
}
253-
$groups[] = new OD_URL_Metric_Group( array(), $min_width, PHP_INT_MAX, $this->sample_size, $this->freshness_ttl, $this );
253+
$groups[] = new OD_URL_Metric_Group( array(), $min_width_exclusive, PHP_INT_MAX, $this->sample_size, $this->freshness_ttl, $this );
254254
return $groups;
255255
}
256256

@@ -285,7 +285,7 @@ public function add_url_metric( OD_URL_Metric $new_url_metric ): void {
285285
* @since 0.1.0
286286
* @throws InvalidArgumentException When there is no group for the provided viewport width. This would only happen if a negative width is provided.
287287
*
288-
* @param int $viewport_width Viewport width.
288+
* @param positive-int $viewport_width Viewport width.
289289
* @return OD_URL_Metric_Group URL Metric group for the viewport width.
290290
*/
291291
public function get_group_for_viewport_width( int $viewport_width ): OD_URL_Metric_Group {

plugins/optimization-detective/class-od-url-metric-group.php

+12-12
Original file line numberDiff line numberDiff line change
@@ -32,12 +32,11 @@ final class OD_URL_Metric_Group implements IteratorAggregate, Countable, JsonSer
3232
private $url_metrics;
3333

3434
/**
35-
* Minimum possible viewport width for the group (inclusive).
35+
* Minimum possible viewport width for the group (exclusive).
3636
*
3737
* @since 0.1.0
3838
*
39-
* @var int
40-
* @phpstan-var 0|positive-int
39+
* @var int<0, max>
4140
*/
4241
private $minimum_viewport_width;
4342

@@ -46,8 +45,7 @@ final class OD_URL_Metric_Group implements IteratorAggregate, Countable, JsonSer
4645
*
4746
* @since 0.1.0
4847
*
49-
* @var int
50-
* @phpstan-var positive-int
48+
* @var int<1, max>
5149
*/
5250
private $maximum_viewport_width;
5351

@@ -103,8 +101,8 @@ final class OD_URL_Metric_Group implements IteratorAggregate, Countable, JsonSer
103101
* @throws InvalidArgumentException If arguments are invalid.
104102
*
105103
* @param OD_URL_Metric[] $url_metrics URL Metrics to add to the group.
106-
* @param int $minimum_viewport_width Minimum possible viewport width for the group. Must be zero or greater.
107-
* @param int $maximum_viewport_width Maximum possible viewport width for the group. Must be greater than zero and the minimum viewport width.
104+
* @param int $minimum_viewport_width Minimum possible viewport width (exclusive) for the group. Must be zero or greater.
105+
* @param int $maximum_viewport_width Maximum possible viewport width (inclusive) for the group. Must be greater than zero and the minimum viewport width.
108106
* @param int $sample_size Sample size for the maximum number of viewports in a group between breakpoints.
109107
* @param int $freshness_ttl Freshness age (TTL) for a given URL Metric.
110108
* @param OD_URL_Metric_Group_Collection $collection Collection that this instance belongs to.
@@ -158,12 +156,12 @@ public function __construct( array $url_metrics, int $minimum_viewport_width, in
158156
}
159157

160158
/**
161-
* Gets the minimum possible viewport width (inclusive).
159+
* Gets the minimum possible viewport width (exclusive).
162160
*
163161
* @since 0.1.0
164162
*
165163
* @todo Eliminate in favor of readonly public property.
166-
* @return int<0, max> Minimum viewport width.
164+
* @return int<0, max> Minimum viewport width (exclusive).
167165
*/
168166
public function get_minimum_viewport_width(): int {
169167
return $this->minimum_viewport_width;
@@ -175,7 +173,7 @@ public function get_minimum_viewport_width(): int {
175173
* @since 0.1.0
176174
*
177175
* @todo Eliminate in favor of readonly public property.
178-
* @return int<1, max> Minimum viewport width.
176+
* @return int<1, max> Minimum viewport width (inclusive).
179177
*/
180178
public function get_maximum_viewport_width(): int {
181179
return $this->maximum_viewport_width;
@@ -208,16 +206,18 @@ public function get_freshness_ttl(): int {
208206
}
209207

210208
/**
211-
* Checks whether the provided viewport width is within the minimum/maximum range for.
209+
* Checks whether the provided viewport width is between the minimum (exclusive) and maximum (inclusive).
212210
*
213211
* @since 0.1.0
214212
*
213+
* @phpstan-param int<1, max> $viewport_width
214+
*
215215
* @param int $viewport_width Viewport width.
216216
* @return bool Whether the viewport width is in range.
217217
*/
218218
public function is_viewport_width_in_range( int $viewport_width ): bool {
219219
return (
220-
$viewport_width >= $this->minimum_viewport_width &&
220+
$viewport_width > $this->minimum_viewport_width &&
221221
$viewport_width <= $this->maximum_viewport_width
222222
);
223223
}

plugins/optimization-detective/class-od-url-metric.php

+5-5
Original file line numberDiff line numberDiff line change
@@ -16,8 +16,8 @@
1616
* Representation of the measurements taken from a single client's visit to a specific URL.
1717
*
1818
* @phpstan-type ViewportRect array{
19-
* width: int,
20-
* height: int
19+
* width: positive-int,
20+
* height: positive-int
2121
* }
2222
* @phpstan-type DOMRect array{
2323
* width: float,
@@ -249,12 +249,12 @@ public static function get_json_schema(): array {
249249
'width' => array(
250250
'type' => 'integer',
251251
'required' => true,
252-
'minimum' => 0,
252+
'minimum' => 1,
253253
),
254254
'height' => array(
255255
'type' => 'integer',
256256
'required' => true,
257-
'minimum' => 0,
257+
'minimum' => 1,
258258
),
259259
),
260260
'additionalProperties' => false,
@@ -482,7 +482,7 @@ public function get_viewport(): array {
482482
*
483483
* @since 0.1.0
484484
*
485-
* @return int Viewport width.
485+
* @return positive-int Viewport width.
486486
*/
487487
public function get_viewport_width(): int {
488488
return $this->data['viewport']['width'];

plugins/optimization-detective/detect.js

+17-7
Original file line numberDiff line numberDiff line change
@@ -98,20 +98,30 @@ function error( ...message ) {
9898
/**
9999
* Checks whether the URL Metric(s) for the provided viewport width is needed.
100100
*
101+
* The comparison logic here corresponds with the PHP logic in `OD_URL_Metric_Group::is_viewport_width_in_range()`.
102+
*
101103
* @param {number} viewportWidth - Current viewport width.
102104
* @param {URLMetricGroupStatus[]} urlMetricGroupStatuses - Viewport group statuses.
103105
* @return {boolean} Whether URL Metrics are needed.
104106
*/
105107
function isViewportNeeded( viewportWidth, urlMetricGroupStatuses ) {
106-
let lastWasLacking = false;
107-
for ( const { minimumViewportWidth, complete } of urlMetricGroupStatuses ) {
108-
if ( viewportWidth >= minimumViewportWidth ) {
109-
lastWasLacking = ! complete;
110-
} else {
111-
break;
108+
if ( viewportWidth === 0 ) {
109+
return false;
110+
}
111+
112+
for ( const {
113+
minimumViewportWidth,
114+
maximumViewportWidth,
115+
complete,
116+
} of urlMetricGroupStatuses ) {
117+
if (
118+
viewportWidth > minimumViewportWidth &&
119+
viewportWidth <= maximumViewportWidth
120+
) {
121+
return complete;
112122
}
113123
}
114-
return lastWasLacking;
124+
return false;
115125
}
116126

117127
/**

plugins/optimization-detective/detection.php

+2-1
Original file line numberDiff line numberDiff line change
@@ -128,7 +128,8 @@ function od_get_detection_script( string $slug, OD_URL_Metric_Group_Collection $
128128
'urlMetricGroupStatuses' => array_map(
129129
static function ( OD_URL_Metric_Group $group ): array {
130130
return array(
131-
'minimumViewportWidth' => $group->get_minimum_viewport_width(),
131+
'minimumViewportWidth' => $group->get_minimum_viewport_width(), // Exclusive.
132+
'maximumViewportWidth' => $group->get_maximum_viewport_width(), // Inclusive.
132133
'complete' => $group->is_complete(),
133134
);
134135
},

plugins/optimization-detective/helper.php

+5-5
Original file line numberDiff line numberDiff line change
@@ -37,18 +37,18 @@ function od_initialize_extensions(): void {
3737
*
3838
* @since 0.7.0
3939
*
40-
* @param int|null $minimum_viewport_width Minimum viewport width.
41-
* @param int|null $maximum_viewport_width Maximum viewport width.
40+
* @param int<0, max>|null $minimum_viewport_width Minimum viewport width (exclusive).
41+
* @param int<1, max>|null $maximum_viewport_width Maximum viewport width (inclusive).
4242
* @return non-empty-string|null Media query, or null if the min/max were both unspecified or invalid.
4343
*/
4444
function od_generate_media_query( ?int $minimum_viewport_width, ?int $maximum_viewport_width ): ?string {
45-
if ( is_int( $minimum_viewport_width ) && is_int( $maximum_viewport_width ) && $minimum_viewport_width > $maximum_viewport_width ) {
46-
_doing_it_wrong( __FUNCTION__, esc_html__( 'The minimum width cannot be greater than the maximum width.', 'optimization-detective' ), 'Optimization Detective 0.7.0' );
45+
if ( is_int( $minimum_viewport_width ) && is_int( $maximum_viewport_width ) && $minimum_viewport_width >= $maximum_viewport_width ) {
46+
_doing_it_wrong( __FUNCTION__, esc_html__( 'The minimum width cannot be greater than or equal to the maximum width.', 'optimization-detective' ), 'Optimization Detective 0.7.0' );
4747
return null;
4848
}
4949
$media_attributes = array();
5050
if ( null !== $minimum_viewport_width && $minimum_viewport_width > 0 ) {
51-
$media_attributes[] = sprintf( '(min-width: %dpx)', $minimum_viewport_width );
51+
$media_attributes[] = sprintf( '(min-width: %dpx)', $minimum_viewport_width + 1 );
5252
}
5353
if ( null !== $maximum_viewport_width && PHP_INT_MAX !== $maximum_viewport_width ) {
5454
$media_attributes[] = sprintf( '(max-width: %dpx)', $maximum_viewport_width );

plugins/optimization-detective/tests/storage/test-rest-api.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -573,7 +573,7 @@ static function () use ( $breakpoint_width ): array {
573573
);
574574
$url_metric_groups = iterator_to_array( $group_collection );
575575
$this->assertSame(
576-
array( 0, $breakpoint_width + 1 ),
576+
array( 0, $breakpoint_width ),
577577
array_map(
578578
static function ( OD_URL_Metric_Group $group ) {
579579
return $group->get_minimum_viewport_width();

plugins/optimization-detective/tests/test-class-od-link-collection.php

+7-7
Original file line numberDiff line numberDiff line change
@@ -79,9 +79,9 @@ public function data_provider_to_test_add_link(): array {
7979
),
8080
),
8181
'expected_html' => '
82-
<link data-od-added-tag rel="preload" href="https://example.com/foo.jpg" crossorigin="anonymous" fetchpriority="high" as="image" media="screen and (min-width: 100px) and (max-width: 200px)">
82+
<link data-od-added-tag rel="preload" href="https://example.com/foo.jpg" crossorigin="anonymous" fetchpriority="high" as="image" media="screen and (min-width: 101px) and (max-width: 200px)">
8383
',
84-
'expected_header' => 'Link: <https://example.com/foo.jpg>; rel="preload"; crossorigin="anonymous"; fetchpriority="high"; as="image"; media="screen and (min-width: 100px) and (max-width: 200px)"',
84+
'expected_header' => 'Link: <https://example.com/foo.jpg>; rel="preload"; crossorigin="anonymous"; fetchpriority="high"; as="image"; media="screen and (min-width: 101px) and (max-width: 200px)"',
8585
'expected_count' => 1,
8686
'error' => '',
8787
),
@@ -116,15 +116,15 @@ public function data_provider_to_test_add_link(): array {
116116
'as' => 'image',
117117
'media' => 'screen',
118118
),
119-
201,
119+
200,
120120
300,
121121
),
122122
),
123123
'expected_html' => '
124124
<link data-od-added-tag rel="preload" href="https://example.com/bar.jpg" as="image" media="screen">
125-
<link data-od-added-tag rel="preload" href="https://example.com/foo.jpg" crossorigin="anonymous" fetchpriority="high" as="image" media="screen and (min-width: 100px) and (max-width: 300px)">
125+
<link data-od-added-tag rel="preload" href="https://example.com/foo.jpg" crossorigin="anonymous" fetchpriority="high" as="image" media="screen and (min-width: 101px) and (max-width: 300px)">
126126
',
127-
'expected_header' => 'Link: <https://example.com/bar.jpg>; rel="preload"; as="image"; media="screen", <https://example.com/foo.jpg>; rel="preload"; crossorigin="anonymous"; fetchpriority="high"; as="image"; media="screen and (min-width: 100px) and (max-width: 300px)"',
127+
'expected_header' => 'Link: <https://example.com/bar.jpg>; rel="preload"; as="image"; media="screen", <https://example.com/foo.jpg>; rel="preload"; crossorigin="anonymous"; fetchpriority="high"; as="image"; media="screen and (min-width: 101px) and (max-width: 300px)"',
128128
'expected_count' => 3,
129129
'error' => '',
130130
),
@@ -135,7 +135,7 @@ public function data_provider_to_test_add_link(): array {
135135
'rel' => 'preconnect',
136136
'href' => 'https://youtube.com/',
137137
),
138-
201,
138+
200,
139139
300,
140140
),
141141
),
@@ -154,7 +154,7 @@ public function data_provider_to_test_add_link(): array {
154154
'href' => 'https://youtube.com/',
155155
'media' => 'tty',
156156
),
157-
201,
157+
200,
158158
300,
159159
),
160160
),

0 commit comments

Comments
 (0)