Skip to content

Commit

Permalink
using encoderinfo for saving also for the non-primary images
Browse files Browse the repository at this point in the history
Signed-off-by: Alexander Piskun <[email protected]>
  • Loading branch information
bigcat88 committed Nov 28, 2024
1 parent 4c24a2e commit c9b6a7c
Show file tree
Hide file tree
Showing 2 changed files with 40 additions and 2 deletions.
3 changes: 1 addition & 2 deletions pillow_heif/as_plugin.py
Original file line number Diff line number Diff line change
Expand Up @@ -272,8 +272,7 @@ def _pil_encode_image(ctx: CtxEncode, img: Image.Image, primary: bool, **kwargs)
_info = img.info.copy()
_info["exif"] = _exif_from_pillow(img)
_info["xmp"] = _xmp_from_pillow(img)
if primary:
_info.update(**kwargs)
_info.update(**kwargs)
_info["primary"] = primary
if img.mode == "YCbCr":
ctx.add_image_ycbcr(img, image_orientation=_get_orientation_for_encoder(_info), **_info)
Expand Down
39 changes: 39 additions & 0 deletions tests/write_test.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
import builtins
import math
import os
import zlib
from gc import collect
from io import SEEK_END, BytesIO
from pathlib import Path
Expand Down Expand Up @@ -583,6 +585,43 @@ def test_lossless_encoding_rgba(save_format):
helpers.assert_image_equal(im_rgb, Image.open(buf))


@pytest.mark.parametrize("save_format", ("HEIF", "AVIF"))
def test_lossless_encoding_non_primary_image(save_format):
def encode_to_image(data):
binary_data = zlib.compress(data)
side_length = math.ceil(math.sqrt((len(binary_data) + 3) // 4))
total_pixels = side_length * side_length
padded_binary_data = binary_data + b"\x00" * (total_pixels * 4 - len(binary_data))
binary_array = [tuple(padded_binary_data[i : i + 4]) for i in range(0, len(padded_binary_data), 4)]
image_data = [binary_array[i : i + side_length] for i in range(0, len(binary_array), side_length)]
im = Image.new("RGBA", (side_length, side_length))
im.putdata([pixel for row in image_data for pixel in row])
return im

im_rgb = helpers.gradient_rgb()
im_rgb.encoderinfo = {"matrix_coefficients": 0}
binary_data_to_encode = b"Some binary data to encode"
second_image = encode_to_image(binary_data_to_encode)
second_image.encoderinfo = {"matrix_coefficients": 0}
buf = BytesIO()
im_rgb.save(
buf,
format=save_format,
quality=-1,
chroma=444,
matrix_coefficients=0,
save_all=True,
append_images=[second_image],
)
im_out = Image.open(buf)
helpers.assert_image_equal(im_rgb, im_out)
bin_data = ImageSequence.Iterator(im_out)[1].copy()
flattened_binary_array = list(bin_data.getdata())
binary_data_with_padding = b"".join(bytes(pixel) for pixel in flattened_binary_array)
z = binary_data_with_padding.rstrip(b"\x00")
assert zlib.decompress(z) == binary_data_to_encode


def test_input_chroma_value():
im = Image.open(Path("images/heif_other/RGB_8_chroma444.heif"))
assert im.info["chroma"] == 444
Expand Down

0 comments on commit c9b6a7c

Please sign in to comment.