mirror of
https://github.com/python-pillow/Pillow.git
synced 2025-01-26 09:14:27 +03:00
Merge branch 'main' into enum
This commit is contained in:
commit
9a4106c14f
|
@ -5,6 +5,12 @@ Changelog (Pillow)
|
|||
9.1.0 (unreleased)
|
||||
------------------
|
||||
|
||||
- Added get_photoshop_blocks() to parse Photoshop TIFF tag #6030
|
||||
[radarhere]
|
||||
|
||||
- Drop excess values in BITSPERSAMPLE #6041
|
||||
[mikhail-iurkov]
|
||||
|
||||
- Added unpacker from RGBA;15 to RGB #6031
|
||||
[radarhere]
|
||||
|
||||
|
@ -26,7 +32,7 @@ Changelog (Pillow)
|
|||
- Ensure duplicated file pointer is closed #5946
|
||||
[radarhere]
|
||||
|
||||
- Added specific error if ImagePath coordinate type is incorrect #5942
|
||||
- Added specific error if path coordinate type is incorrect #5942
|
||||
[radarhere]
|
||||
|
||||
- Return an empty bytestring from tobytes() for an empty image #5938
|
||||
|
|
BIN
Tests/images/tiff_wrong_bits_per_sample_2.tiff
Normal file
BIN
Tests/images/tiff_wrong_bits_per_sample_2.tiff
Normal file
Binary file not shown.
|
@ -90,11 +90,18 @@ class TestFileTiff:
|
|||
|
||||
assert_image_similar_tofile(im, "Tests/images/pil136.png", 1)
|
||||
|
||||
def test_wrong_bits_per_sample(self):
|
||||
with Image.open("Tests/images/tiff_wrong_bits_per_sample.tiff") as im:
|
||||
assert im.mode == "RGBA"
|
||||
assert im.size == (52, 53)
|
||||
assert im.tile == [("raw", (0, 0, 52, 53), 160, ("RGBA", 0, 1))]
|
||||
@pytest.mark.parametrize(
|
||||
"file_name,mode,size,offset",
|
||||
[
|
||||
("tiff_wrong_bits_per_sample.tiff", "RGBA", (52, 53), 160),
|
||||
("tiff_wrong_bits_per_sample_2.tiff", "RGB", (16, 16), 8),
|
||||
],
|
||||
)
|
||||
def test_wrong_bits_per_sample(self, file_name, mode, size, offset):
|
||||
with Image.open("Tests/images/" + file_name) as im:
|
||||
assert im.mode == mode
|
||||
assert im.size == size
|
||||
assert im.tile == [("raw", (0, 0) + size, offset, (mode, 0, 1))]
|
||||
im.load()
|
||||
|
||||
def test_set_legacy_api(self):
|
||||
|
@ -685,6 +692,32 @@ class TestFileTiff:
|
|||
assert description[0]["format"] == "image/tiff"
|
||||
assert description[3]["BitsPerSample"]["Seq"]["li"] == ["8", "8", "8"]
|
||||
|
||||
def test_get_photoshop_blocks(self):
|
||||
with Image.open("Tests/images/lab.tif") as im:
|
||||
assert list(im.get_photoshop_blocks().keys()) == [
|
||||
1061,
|
||||
1002,
|
||||
1005,
|
||||
1062,
|
||||
1037,
|
||||
1049,
|
||||
1011,
|
||||
1034,
|
||||
10000,
|
||||
1013,
|
||||
1016,
|
||||
1032,
|
||||
1054,
|
||||
1050,
|
||||
1064,
|
||||
1041,
|
||||
1044,
|
||||
1036,
|
||||
1057,
|
||||
4000,
|
||||
4001,
|
||||
]
|
||||
|
||||
def test_close_on_load_exclusive(self, tmp_path):
|
||||
# similar to test_fd_leak, but runs on unixlike os
|
||||
tmpfile = str(tmp_path / "temp.tif")
|
||||
|
|
|
@ -1,8 +1,9 @@
|
|||
import pytest
|
||||
from packaging.version import parse as parse_version
|
||||
|
||||
from PIL import Image
|
||||
from PIL import Image, features
|
||||
|
||||
from .helper import assert_image_similar, hopper, is_ppc64le
|
||||
from .helper import assert_image_similar, hopper, is_ppc64le, skip_unless_feature
|
||||
|
||||
|
||||
def test_sanity():
|
||||
|
@ -17,16 +18,14 @@ def test_sanity():
|
|||
assert_image_similar(converted.convert("RGB"), image, 60)
|
||||
|
||||
|
||||
@pytest.mark.xfail(is_ppc64le(), reason="failing on ppc64le on GHA")
|
||||
@skip_unless_feature("libimagequant")
|
||||
def test_libimagequant_quantize():
|
||||
image = hopper()
|
||||
try:
|
||||
converted = image.quantize(100, Image.Quantize.LIBIMAGEQUANT)
|
||||
except ValueError as ex: # pragma: no cover
|
||||
if "dependency" in str(ex).lower():
|
||||
pytest.skip("libimagequant support not available")
|
||||
else:
|
||||
raise
|
||||
if is_ppc64le():
|
||||
libimagequant = parse_version(features.version_feature("libimagequant"))
|
||||
if libimagequant < parse_version("4"):
|
||||
pytest.skip("Fails with libimagequant earlier than 4.0.0 on ppc64le")
|
||||
converted = image.quantize(100, Image.Quantize.LIBIMAGEQUANT)
|
||||
assert converted.mode == "P"
|
||||
assert_image_similar(converted.convert("RGB"), image, 15)
|
||||
assert len(converted.getcolors()) == 100
|
||||
|
|
56
docs/releasenotes/9.1.0.rst
Normal file
56
docs/releasenotes/9.1.0.rst
Normal file
|
@ -0,0 +1,56 @@
|
|||
9.1.0
|
||||
-----
|
||||
|
||||
API Changes
|
||||
===========
|
||||
|
||||
Raise an error when performing a negative crop
|
||||
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||
|
||||
Performing a negative crop on an image previously just returned a ``(0, 0)`` image. Now
|
||||
it will raise a ``ValueError``, to help reduce confusion if a user has unintentionally
|
||||
provided the wrong arguments.
|
||||
|
||||
Added specific error if path coordinate type is incorrect
|
||||
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||
|
||||
Rather than returning a ``SystemError``, passing the incorrect types of coordinates into
|
||||
a path will now raise a more specific ``ValueError``, with the message "incorrect
|
||||
coordinate type".
|
||||
|
||||
Deprecations
|
||||
^^^^^^^^^^^^
|
||||
|
||||
ImageShow.Viewer.show_file file argument
|
||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
The ``file`` argument in :py:meth:`~PIL.ImageShow.Viewer.show_file()` has been
|
||||
deprecated, replaced by ``path``.
|
||||
|
||||
In effect, ``viewer.show_file("test.jpg")`` will continue to work unchanged.
|
||||
``viewer.show_file(file="test.jpg")`` will raise a deprecation warning, and suggest
|
||||
``viewer.show_file(path="test.jpg")`` instead.
|
||||
|
||||
API Additions
|
||||
=============
|
||||
|
||||
Added get_photoshop_blocks() to parse Photoshop TIFF tag
|
||||
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||
|
||||
:py:meth:`~PIL.TiffImagePlugin.TiffImageFile.get_photoshop_blocks` has been added, to
|
||||
allow users to determine what Photoshop "Image Resource Blocks" are contained within an
|
||||
image. The keys of the returned dictionary are the image resource IDs.
|
||||
|
||||
At present, the information within each block is merely returned as a dictionary with a
|
||||
"data" entry. This will allow more useful information to be added in the future without
|
||||
breaking backwards compatibility.
|
||||
|
||||
Other Changes
|
||||
=============
|
||||
|
||||
Image._repr_pretty_
|
||||
^^^^^^^^^^^^^^^^^^^
|
||||
|
||||
``im._repr_pretty_`` has been added to provide a representation of an image without the
|
||||
identity of the object. This allows Jupyter to describe an image and have that
|
||||
description stay the same on subsequent executions of the same code.
|
|
@ -14,6 +14,7 @@ expected to be backported to earlier versions.
|
|||
.. toctree::
|
||||
:maxdepth: 2
|
||||
|
||||
9.1.0
|
||||
9.0.1
|
||||
9.0.0
|
||||
8.4.0
|
||||
|
|
|
@ -41,6 +41,7 @@
|
|||
import io
|
||||
import itertools
|
||||
import logging
|
||||
import math
|
||||
import os
|
||||
import struct
|
||||
import warnings
|
||||
|
@ -49,6 +50,8 @@ from fractions import Fraction
|
|||
from numbers import Number, Rational
|
||||
|
||||
from . import Image, ImageFile, ImageOps, ImagePalette, TiffTags
|
||||
from ._binary import i16be as i16
|
||||
from ._binary import i32be as i32
|
||||
from ._binary import o8
|
||||
from .TiffTags import TYPES
|
||||
|
||||
|
@ -1129,6 +1132,27 @@ class TiffImageFile(ImageFile.ImageFile):
|
|||
"""
|
||||
return self._getxmp(self.tag_v2[700]) if 700 in self.tag_v2 else {}
|
||||
|
||||
def get_photoshop_blocks(self):
|
||||
"""
|
||||
Returns a dictionary of Photoshop "Image Resource Blocks".
|
||||
The keys are the image resource ID. For more information, see
|
||||
https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#50577409_pgfId-1037727
|
||||
|
||||
:returns: Photoshop "Image Resource Blocks" in a dictionary.
|
||||
"""
|
||||
blocks = {}
|
||||
val = self.tag_v2.get(0x8649)
|
||||
if val:
|
||||
while val[:4] == b"8BIM":
|
||||
id = i16(val[4:6])
|
||||
n = math.ceil((val[6] + 1) / 2) * 2
|
||||
size = i32(val[6 + n : 10 + n])
|
||||
data = val[10 + n : 10 + n + size]
|
||||
blocks[id] = {"data": data}
|
||||
|
||||
val = val[math.ceil((10 + n + size) / 2) * 2 :]
|
||||
return blocks
|
||||
|
||||
def load(self):
|
||||
if self.tile and self.use_load_libtiff:
|
||||
return self._load_libtiff()
|
||||
|
@ -1308,9 +1332,14 @@ class TiffImageFile(ImageFile.ImageFile):
|
|||
else:
|
||||
bps_count = 1
|
||||
bps_count += len(extra_tuple)
|
||||
# Some files have only one value in bps_tuple,
|
||||
# while should have more. Fix it
|
||||
if bps_count > len(bps_tuple) and len(bps_tuple) == 1:
|
||||
bps_actual_count = len(bps_tuple)
|
||||
if bps_count < bps_actual_count:
|
||||
# If a file has more values in bps_tuple than expected,
|
||||
# remove the excess.
|
||||
bps_tuple = bps_tuple[:bps_count]
|
||||
elif bps_count > bps_actual_count and bps_actual_count == 1:
|
||||
# If a file has only one value in bps_tuple, when it should have more,
|
||||
# presume it is the same number of bits for all of the samples.
|
||||
bps_tuple = bps_tuple * bps_count
|
||||
|
||||
samplesPerPixel = self.tag_v2.get(
|
||||
|
|
|
@ -1216,9 +1216,7 @@ frompalette(Imaging imOut, Imaging imIn, const char *mode) {
|
|||
convert = alpha ? pa2f : p2f;
|
||||
} else if (strcmp(mode, "RGB") == 0) {
|
||||
convert = alpha ? pa2rgb : p2rgb;
|
||||
} else if (strcmp(mode, "RGBA") == 0) {
|
||||
convert = alpha ? pa2rgba : p2rgba;
|
||||
} else if (strcmp(mode, "RGBX") == 0) {
|
||||
} else if (strcmp(mode, "RGBA") == 0 || strcmp(mode, "RGBX") == 0) {
|
||||
convert = alpha ? pa2rgba : p2rgba;
|
||||
} else if (strcmp(mode, "CMYK") == 0) {
|
||||
convert = alpha ? pa2cmyk : p2cmyk;
|
||||
|
|
|
@ -280,9 +280,9 @@ deps = {
|
|||
"libs": [r"imagequant.lib"],
|
||||
},
|
||||
"harfbuzz": {
|
||||
"url": "https://github.com/harfbuzz/harfbuzz/archive/3.3.2.zip",
|
||||
"filename": "harfbuzz-3.3.2.zip",
|
||||
"dir": "harfbuzz-3.3.2",
|
||||
"url": "https://github.com/harfbuzz/harfbuzz/archive/3.4.0.zip",
|
||||
"filename": "harfbuzz-3.4.0.zip",
|
||||
"dir": "harfbuzz-3.4.0",
|
||||
"build": [
|
||||
cmd_cmake("-DHB_HAVE_FREETYPE:BOOL=TRUE"),
|
||||
cmd_nmake(target="clean"),
|
||||
|
|
Loading…
Reference in New Issue
Block a user