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

Add AVIF support #4612

Closed
Closed
Show file tree
Hide file tree
Changes from 14 commits
Commits
Show all changes
90 commits
Select commit Hold shift + click to select a range
b96bb09
Add AVIF support
adamsilverstein Jun 13, 2023
5b622d1
whitespace
adamsilverstein Jun 13, 2023
bb9642f
Add AVIF constants
adamsilverstein Jun 14, 2023
3bd1ae8
update magic numbers for avif detection
adamsilverstein Jun 15, 2023
bc5fc5b
Include AVIF in the displayable image types
adamsilverstein Jun 15, 2023
816dfc4
Merge branch 'trunk' into add/avif-support
adamsilverstein Jun 15, 2023
3e7228d
phpcbf
adamsilverstein Jun 15, 2023
8f64fe8
Apply phpcbf fixes to avif helper
adamsilverstein Jun 16, 2023
bb38f19
more phpcbf
adamsilverstein Jun 16, 2023
f1a90cc
yoda conditionals, other small phpcs fixes
adamsilverstein Jun 16, 2023
41762ff
Restore avif helper original and exclude from phpcs
adamsilverstein Jun 16, 2023
160791f
Merge branch 'trunk' into add/avif-support
adamsilverstein Jun 22, 2023
0259896
Merge branch 'trunk' into add/avif-support
adamsilverstein Jul 7, 2023
b0c4ded
Merge branch 'trunk' into add/avif-support
adamsilverstein Oct 6, 2023
ae6a99f
Merge branch 'trunk' into add/avif-support
adamsilverstein Oct 27, 2023
897ffb5
remove excessive whitespace
adamsilverstein Nov 13, 2023
bbcadec
Merge branch 'trunk' into add/avif-support
adamsilverstein Nov 13, 2023
e22d02a
Merge branch 'trunk' into add/avif-support
adamsilverstein Jan 11, 2024
d3fc325
Add AVIF test images
adamsilverstein Jan 12, 2024
1652946
match other test image names
adamsilverstein Jan 12, 2024
fd93e47
one more test AVIF image
adamsilverstein Jan 12, 2024
345fa5e
Add avif to get_image_mime tests
adamsilverstein Jan 12, 2024
f86c77e
Add AVIF to wp_getimagesize tests
adamsilverstein Jan 12, 2024
fd88a99
Add tests for wp_get_avif_info
adamsilverstein Jan 12, 2024
306870c
Add AVIF to file_is_valid_image_positive tests
adamsilverstein Jan 12, 2024
b50890a
Add AVIF to file_is_displayable_image_positive test
adamsilverstein Jan 12, 2024
8555bfb
Add additional magic file detection numbers for AVIF in wp_get_image_…
adamsilverstein Jan 12, 2024
e25becb
Correctly handle 0x0 sizes returned from `getimagesize`, using shim f…
adamsilverstein Jan 12, 2024
f5b39b7
Add test_resize_avif
adamsilverstein Jan 12, 2024
509ac34
phpcs
adamsilverstein Jan 12, 2024
4c5efd0
remove unused module
adamsilverstein Jan 12, 2024
b5d9624
temporarily log debug info for test runs
adamsilverstein Jan 12, 2024
22f6af1
more debug logging for tests
adamsilverstein Jan 12, 2024
2ec1d34
Imagick editor - fall back to `wp_getimagesize`
adamsilverstein Jan 12, 2024
1b247ec
Improve doc block
adamsilverstein Jan 12, 2024
547b40d
more resize logging
adamsilverstein Jan 12, 2024
07dbc24
remove image size logging
adamsilverstein Jan 12, 2024
74eb3c8
remove error logging
adamsilverstein Jan 12, 2024
82ccce5
log and skip failing test
adamsilverstein Jan 12, 2024
1823866
remove fallback
adamsilverstein Jan 13, 2024
e5ec13c
log mime type check in imagick
adamsilverstein Jan 14, 2024
055859f
better logging
adamsilverstein Jan 14, 2024
117de6d
add logging of failed avif test
adamsilverstein Jan 14, 2024
67291a6
improve logging
adamsilverstein Jan 14, 2024
e43c03e
improve logging
adamsilverstein Jan 14, 2024
13641c0
class name for log
adamsilverstein Jan 14, 2024
3e9745f
log avifinfo results after resize
adamsilverstein Jan 14, 2024
63664bc
skip if either h or w empty
adamsilverstein Jan 14, 2024
b257474
more logging
adamsilverstein Jan 14, 2024
bc38fd4
more logging
adamsilverstein Jan 14, 2024
0195d41
more logging
adamsilverstein Jan 14, 2024
efd37b9
log mime magics
adamsilverstein Jan 14, 2024
c6c3b54
adjust empty logic, log after check
adamsilverstein Jan 14, 2024
7e51218
log a mime check in test
adamsilverstein Jan 14, 2024
569fe27
fix typo
adamsilverstein Jan 14, 2024
b8290a9
add additional magic for AVIF detection
adamsilverstein Jan 14, 2024
6c10a03
remove logging
adamsilverstein Jan 14, 2024
663a424
Update doc blocks, cleanup
adamsilverstein Jan 15, 2024
ee6ad2e
Merge branch 'trunk' into add/avif-support
adamsilverstein Jan 15, 2024
0b54d59
add fallback comment
adamsilverstein Jan 15, 2024
dd14f4a
improve libavifinfo doc block, link to source
adamsilverstein Jan 17, 2024
447a44c
remove lossless AVIF image handling: not supported in PHP
adamsilverstein Jan 17, 2024
6af974c
Simplify wp_get_avif_info and improve doc blocks
adamsilverstein Jan 17, 2024
c5820c3
Add todo for detection change
adamsilverstein Jan 17, 2024
c026771
WP_Image_Editor_GD:stream - improve mime detection and fallback
adamsilverstein Jan 18, 2024
2eb1fc3
Merge branch 'trunk' into add/avif-support
adamsilverstein Jan 18, 2024
ddbd344
WP_Image_Editor_Imagick::update_size - improve logic for avif fallback
adamsilverstein Jan 18, 2024
3ecb682
Update header based AVIF detection
adamsilverstein Jan 18, 2024
b6b78d0
Add commit hash to libavifinfo reference
adamsilverstein Jan 18, 2024
8c51a64
revert creation of additional header variable
adamsilverstein Jan 18, 2024
478e5d5
Doc block cleanup.
adamsilverstein Jan 18, 2024
c773990
replace avif-transparent test image
adamsilverstein Jan 19, 2024
b2ec778
Replace @todo comment with link to specification
adamsilverstein Jan 19, 2024
5905869
Always test `wp_get_avif_info` regardless of server AVIF support
adamsilverstein Jan 19, 2024
2cb8cae
Add additional AVIF test images, remove one unused image
adamsilverstein Jan 19, 2024
79587bd
Always test wp_get_webp_info regardless of server support for webp
adamsilverstein Jan 19, 2024
b7c193c
Update avif-info doc block to indicate it will no longer be required …
adamsilverstein Jan 19, 2024
471f77c
Correct bit_depth detection bug in avif-info class, adjust tests to m…
adamsilverstein Jan 23, 2024
49f4278
Update src/wp-includes/media.php
adamsilverstein Jan 24, 2024
720910b
Update tests/phpunit/tests/image/editor.php
adamsilverstein Jan 24, 2024
609d875
Update doc block for AVIF mime detection
adamsilverstein Jan 24, 2024
b492dbf
Clean up return from `wp_get_avif_info`
adamsilverstein Jan 24, 2024
6fe2847
Correct tests for `wp_get_avif_info`
adamsilverstein Jan 24, 2024
942bfbc
wp_get_avif_info - test fix for default return
adamsilverstein Jan 24, 2024
9dd7c71
Merge branch 'trunk' into add/avif-support
adamsilverstein Jan 29, 2024
036ab53
Update src/wp-includes/class-wp-image-editor-imagick.php
adamsilverstein Jan 29, 2024
93b4cdc
Update src/wp-includes/media.php
adamsilverstein Jan 29, 2024
73c1528
Update src/wp-includes/media.php
adamsilverstein Jan 29, 2024
c2b6df1
Update phpcs.xml.dist
adamsilverstein Jan 29, 2024
9f884c1
Merge branch 'trunk' into add/avif-support
adamsilverstein Feb 2, 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
8 changes: 6 additions & 2 deletions src/wp-includes/class-avif-info.php
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,11 @@
* Media Patent License 1.0 was not distributed with this source code in the
* PATENTS file, you can obtain it at www.aomedia.org/license/patent.
*
* Note: this class is from libavifinfo - https://aomedia.googlesource.com/libavifinfo/+/refs/heads/main/avifinfo.php
* Note: this class is from libavifinfo - https://aomedia.googlesource.com/libavifinfo/+/refs/heads/main/avifinfo.php at f509487.
* It is used as a fallback to parse AVIF files when the server doesn't support AVIF,
* primarily to identify the width and height of the image.
*
* Note PHP 8.2 added native support for AVIF, so this class can be removed when WordPress requires PHP 8.2.
*/

namespace Avifinfo;
Expand Down Expand Up @@ -446,7 +450,7 @@ private function parse_ipco( $num_remaining_bytes ) {
} else if ( $box->type == 'auxC' ) {
// See AV1 Image File Format (AVIF) 4
// at https://aomediacodec.github.io/av1-avif/#auxiliary-images
$kAlphaStr = "urn:mpeg:mpegB:cicp:systems:auxiliary:alpha";
$kAlphaStr = "urn:mpeg:mpegB:cicp:systems:auxiliary:alpha\0";
$kAlphaStrLength = 44; // Includes terminating character.
if ( $box->content_size >= $kAlphaStrLength ) {
if ( !( $data = read( $this->handle, $kAlphaStrLength ) ) ) {
Expand Down
7 changes: 5 additions & 2 deletions src/wp-includes/class-wp-image-editor-gd.php
Original file line number Diff line number Diff line change
Expand Up @@ -577,14 +577,17 @@ public function stream( $mime_type = null ) {
if ( function_exists( 'imagewebp' ) ) {
header( 'Content-Type: image/webp' );
return imagewebp( $this->image, null, $this->get_quality() );
} else {
// Fall back to JPEG.
header( 'Content-Type: image/jpeg' );
return imagejpeg( $this->image, null, $this->get_quality() );
}
// Fall back to JPEG as the default.
case 'image/avif':
if ( function_exists( 'imageavif' ) ) {
header( 'Content-Type: image/avif' );
return imageavif( $this->image, null, $this->get_quality() );
}
// Fall back to JPEG as the default.
// Fall back to JPEG.
default:
header( 'Content-Type: image/jpeg' );
return imagejpeg( $this->image, null, $this->get_quality() );
Expand Down
2 changes: 1 addition & 1 deletion src/wp-includes/class-wp-image-editor-imagick.php
Original file line number Diff line number Diff line change
Expand Up @@ -259,7 +259,7 @@ protected function update_size( $width = null, $height = null ) {

// If we still don't have the image size, fall back to `wp_getimagesize`. This ensures AVIF images
// are properly sized without affecting previous `getImageGeometry` behavior.
if ( ! $width && ! $height && 'image/avif' === $this->mime_type ) {
if ( ( ! $width || ! $height ) && 'image/avif' === $this->mime_type ) {
$size = wp_getimagesize( $this->file );
$width = $size[0];
$height = $size[1];
Expand Down
19 changes: 10 additions & 9 deletions src/wp-includes/functions.php
Original file line number Diff line number Diff line change
Expand Up @@ -3355,17 +3355,18 @@ function wp_get_image_mime( $file ) {
/**
* Add AVIF fallback detection when image library doesn't support AVIF.
*
* @todo check the third 4-byte token "[major_brand]", for "avif" or "avis" .
*
* Detection based on section 4.3.1 File-type box Definition of the ISO/IEC 14496-12
* specification, https://see aomediacodec.github.io/av1-avif/v1.1.0.html#brands.
adamsilverstein marked this conversation as resolved.
Show resolved Hide resolved
*/

// Divide the header string into 4 byte groups.
$magic = str_split( $magic, 8 );

if (
// AVIF: lossless or lossy.
( str_starts_with( $magic, '0000002066' ) ) ||
( str_starts_with( $magic, '0000001866' ) ) || // AVIF created by Imagick.
// AVIF: animated.
( str_starts_with( $magic, '0000002c66' ) ) ||
// AVIF: transparent.
( str_starts_with( $magic, '0000001c66' ) )
isset( $magic[1] ) &&
isset( $magic[2] ) &&
'ftyp' === hex2bin( $magic[1] ) &&
( 'avif' === hex2bin( $magic[2] ) || 'avis' === hex2bin( $magic[2] ) )
) {
$mime = 'image/avif';
}
Expand Down
17 changes: 8 additions & 9 deletions src/wp-includes/media.php
Original file line number Diff line number Diff line change
Expand Up @@ -5521,7 +5521,7 @@ function wp_getimagesize( $filename, array &$image_info = null ) {

if (
! empty( $info ) &&
// Some PHP versions return 0x0 sizes from `getimagesize` for unrecognized image formats, including AVIFs
// Some PHP versions return 0x0 sizes from `getimagesize` for unrecognized image formats, including AVIFs.
! ( empty( $info[0] ) && empty( $info[1] ) )
) {
adamsilverstein marked this conversation as resolved.
Show resolved Hide resolved
return $info;
Expand Down Expand Up @@ -5552,13 +5552,12 @@ function wp_getimagesize( $filename, array &$image_info = null ) {
}
}

// For PHP versions that don't support AVIF images,
// extract the image size info from the file headers.
// For PHP versions that don't support AVIF images, extract the image size info from the file headers.
if ( 'image/avif' === wp_get_image_mime( $filename ) ) {
$avif_info = wp_get_avif_info( $filename );

$width = $avif_info['width'];
$height = $avif_info['height'];
$width = $avif_info['width'];
$height = $avif_info['height'];

// Mimic the native return format.
if ( $width && $height ) {
Expand Down Expand Up @@ -5589,10 +5588,10 @@ function wp_getimagesize( $filename, array &$image_info = null ) {
* @return array {
* An array of AVIF image information.
*
* @type int|false $width Image width on success, false on failure.
* @type int|false $height Image height on success, false on failure.
* @type int|false $bit_depth Image bit depth on success, false on failure.
* @type int|false $num_channels Image number of channels on success, false on failure.
* @type int|false $width Image width on success, false on failure.
* @type int|false $height Image height on success, false on failure.
* @type int|false $bit_depth Image bit depth on success, false on failure.
* @type int|false $num_channels Image number of channels on success, false on failure.
adamsilverstein marked this conversation as resolved.
Show resolved Hide resolved
* }
*/
function wp_get_avif_info( $filename ) {
Expand Down
Binary file removed tests/phpunit/data/images/avif-rotated.avif
Binary file not shown.
Binary file modified tests/phpunit/data/images/avif-transparent.avif
adamsilverstein marked this conversation as resolved.
Show resolved Hide resolved
Binary file not shown.
Binary file not shown.
Binary file added tests/phpunit/data/images/colors_hdr_p3.avif
Binary file not shown.
32 changes: 19 additions & 13 deletions tests/phpunit/tests/image/editor.php
Original file line number Diff line number Diff line change
Expand Up @@ -292,12 +292,6 @@ public function test_get_suffix() {
*
*/
public function test_wp_get_webp_info( $file, $expected ) {
$editor = wp_get_image_editor( $file );

if ( is_wp_error( $editor ) || ! $editor->supports_mime_type( 'image/webp' ) ) {
$this->markTestSkipped( sprintf( 'No WebP support in the editor engine %s on this system.', $this->editor_engine ) );
}

$file_data = wp_get_webp_info( $file );
$this->assertSame( $expected, $file_data );
}
Expand Down Expand Up @@ -372,12 +366,6 @@ public function data_wp_get_webp_info() {
*
*/
public function test_wp_get_avif_info( $file, $expected ) {
$editor = wp_get_image_editor( $file );

if ( is_wp_error( $editor ) || ! $editor->supports_mime_type( 'image/avif' ) ) {
$this->markTestSkipped( sprintf( 'No AVIF support in the editor engine %s on this system.', $this->editor_engine ) );
}

$file_data = wp_get_avif_info( $file );
$this->assertSame( $expected, $file_data );
}
Expand Down Expand Up @@ -412,7 +400,7 @@ public function data_wp_get_avif_info() {
'width' => 150,
'height' => 150,
'bit_depth' => 8,
'num_channels' => 3,
'num_channels' => 4,
),
),
// Lossless AVIF.
Expand Down Expand Up @@ -442,6 +430,24 @@ public function data_wp_get_avif_info() {
'width' => 128,
'height' => 128,
'bit_depth' => 12,
'num_channels' => 4,
),
),
array(
DIR_TESTDATA . '/images/color_grid_alpha_nogrid.avif',
array(
'width' => 80,
'height' => 80,
'bit_depth' => 8,
'num_channels' => 4,
),
),
array(
DIR_TESTDATA . '/images/colors_hdr_p3.avif',
array(
'width' => 200,
'height' => 200,
'bit_depth' => 10,
'num_channels' => 3,
adamsilverstein marked this conversation as resolved.
Show resolved Hide resolved
),
),
Expand Down
1 change: 0 additions & 1 deletion tests/phpunit/tests/image/resize.php
Original file line number Diff line number Diff line change
Expand Up @@ -114,7 +114,6 @@ public function test_resize_avif() {
$this->assertSame( IMAGETYPE_AVIF, $type );
}


public function test_resize_larger() {
// image_resize() should refuse to make an image larger.
$image = $this->resize_helper( DIR_TESTDATA . '/images/test-image.jpg', 100, 100 );
Expand Down
Loading