thumbnail: fix embedded ICC profile conversion (#3027)

This commit is contained in:
Kleis Auke Wolthuizen 2022-10-19 18:32:04 +02:00 committed by GitHub
parent 1fc01c05e8
commit e6198361e5
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 47 additions and 16 deletions

View File

@ -859,9 +859,25 @@ vips_thumbnail_build( VipsObject *object )
in = t[7]; in = t[7];
} }
else if( thumbnail->export_profile ) { else if( thumbnail->export_profile ) {
/* We are in one of the resize space (sRGB, scRGB, B_W, GREY16, /* If there's some kind of import profile, we can transform to
* etc.) and we have an export profile. Go to PCS, then export. * the output. Otherwise, we are in one of the resize space
* (sRGB, scRGB, B_W, GREY16, etc.) and need to go to PCS,
* then export.
*/ */
if( thumbnail->import_profile ||
vips_image_get_typeof( in, VIPS_META_ICC_NAME ) ) {
g_info( "transforming with supplied profiles" );
if( vips_icc_transform( in, &t[7],
thumbnail->export_profile,
"input_profile", thumbnail->import_profile,
"intent", thumbnail->intent,
"embedded", TRUE,
NULL ) )
return( -1 );
in = t[7];
}
else {
g_info( "exporting with %s", g_info( "exporting with %s",
thumbnail->export_profile ); thumbnail->export_profile );
if( vips_colourspace( in, &t[7], if( vips_colourspace( in, &t[7],
@ -874,6 +890,7 @@ vips_thumbnail_build( VipsObject *object )
return( -1 ); return( -1 );
in = t[10]; in = t[10];
} }
}
else { else {
/* We are in one of the resize spaces and there's no export /* We are in one of the resize spaces and there's no export
* profile. Output to sRGB or B_W * profile. Output to sRGB or B_W

View File

@ -9,6 +9,7 @@ import pyvips
IMAGES = os.path.join(os.path.dirname(__file__), os.pardir, 'images') IMAGES = os.path.join(os.path.dirname(__file__), os.pardir, 'images')
JPEG_FILE = os.path.join(IMAGES, "sample.jpg") JPEG_FILE = os.path.join(IMAGES, "sample.jpg")
JPEG_FILE_XYB = os.path.join(IMAGES, "sample-xyb.jpg")
TRUNCATED_FILE = os.path.join(IMAGES, "truncated.jpg") TRUNCATED_FILE = os.path.join(IMAGES, "truncated.jpg")
SRGB_FILE = os.path.join(IMAGES, "sRGB.icm") SRGB_FILE = os.path.join(IMAGES, "sRGB.icm")
MATLAB_FILE = os.path.join(IMAGES, "sample.mat") MATLAB_FILE = os.path.join(IMAGES, "sample.mat")

Binary file not shown.

After

Width:  |  Height:  |  Size: 59 KiB

View File

@ -1395,14 +1395,12 @@ class TestForeign:
# scrgb mode # scrgb mode
scrgb = self.colour.colourspace("scrgb") scrgb = self.colour.colourspace("scrgb")
no_profile.remove("icc-profile-data")
self.save_load_buffer("jxlsave_buffer", "jxlload_buffer", self.save_load_buffer("jxlsave_buffer", "jxlload_buffer",
scrgb, 120) scrgb, 120)
# scrgb mode, no profile # scrgb mode, no profile
scrgb_no_profile = scrgb.copy() scrgb_no_profile = scrgb.copy()
scrgb_no_profile.remove("icc-profile-data") scrgb_no_profile.remove("icc-profile-data")
no_profile.remove("icc-profile-data")
self.save_load_buffer("jxlsave_buffer", "jxlload_buffer", self.save_load_buffer("jxlsave_buffer", "jxlload_buffer",
scrgb_no_profile, 120) scrgb_no_profile, 120)

View File

@ -2,8 +2,8 @@
import pytest import pytest
import pyvips import pyvips
from helpers import JPEG_FILE, OME_FILE, HEIC_FILE, TIF_FILE, all_formats, \ from helpers import JPEG_FILE, JPEG_FILE_XYB, OME_FILE, HEIC_FILE, TIF_FILE, \
have, RGBA_FILE, RGBA_CORRECT_FILE, AVIF_FILE all_formats, have, RGBA_FILE, RGBA_CORRECT_FILE, AVIF_FILE
# Run a function expecting a complex image on a two-band image # Run a function expecting a complex image on a two-band image
@ -229,6 +229,21 @@ class TestResample:
assert thumb.width < thumb.height assert thumb.width < thumb.height
assert thumb.height == 100 assert thumb.height == 100
@pytest.mark.skipif(not pyvips.at_least_libvips(8, 5),
reason="requires libvips >= 8.5")
def test_thumbnail_icc(self):
im = pyvips.Image.thumbnail(JPEG_FILE_XYB, 442, export_profile="srgb", intent="perceptual")
assert im.width == 290
assert im.height == 442
assert im.bands == 3
assert im.bands == 3
# the colour distance should not deviate too much
# (i.e. the embedded profile should not be ignored)
im_orig = pyvips.Image.new_from_file(JPEG_FILE)
assert im_orig.de00(im).max() < 10
def test_similarity(self): def test_similarity(self):
im = pyvips.Image.new_from_file(JPEG_FILE) im = pyvips.Image.new_from_file(JPEG_FILE)
im2 = im.similarity(angle=90) im2 = im.similarity(angle=90)