2023-12-21 14:13:31 +03:00
|
|
|
from __future__ import annotations
|
2024-01-20 14:23:03 +03:00
|
|
|
|
2022-02-21 05:49:01 +03:00
|
|
|
import warnings
|
2019-07-06 23:40:53 +03:00
|
|
|
from io import BytesIO
|
2024-01-31 12:12:58 +03:00
|
|
|
from pathlib import Path
|
2024-01-31 13:55:32 +03:00
|
|
|
from typing import Generator
|
2012-10-16 00:26:38 +04:00
|
|
|
|
2020-02-03 12:11:32 +03:00
|
|
|
import pytest
|
2020-08-07 13:28:33 +03:00
|
|
|
|
2022-05-03 13:07:47 +03:00
|
|
|
from PIL import GifImagePlugin, Image, ImageDraw, ImagePalette, ImageSequence, features
|
2012-10-16 00:26:38 +04:00
|
|
|
|
2020-01-30 17:56:07 +03:00
|
|
|
from .helper import (
|
|
|
|
assert_image_equal,
|
2021-02-21 14:15:56 +03:00
|
|
|
assert_image_equal_tofile,
|
2020-01-30 17:56:07 +03:00
|
|
|
assert_image_similar,
|
|
|
|
hopper,
|
|
|
|
is_pypy,
|
|
|
|
netpbm_available,
|
|
|
|
)
|
2016-09-27 00:44:40 +03:00
|
|
|
|
2012-10-16 00:26:38 +04:00
|
|
|
# sample gif stream
|
2014-09-04 09:44:46 +04:00
|
|
|
TEST_GIF = "Tests/images/hopper.gif"
|
|
|
|
|
|
|
|
with open(TEST_GIF, "rb") as f:
|
2013-05-23 21:02:19 +04:00
|
|
|
data = f.read()
|
2012-10-16 00:26:38 +04:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_sanity() -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open(TEST_GIF) as im:
|
|
|
|
im.load()
|
|
|
|
assert im.mode == "P"
|
|
|
|
assert im.size == (128, 128)
|
|
|
|
assert im.format == "GIF"
|
|
|
|
assert im.info["version"] == b"GIF89a"
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.skipif(is_pypy(), reason="Requires CPython")
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_unclosed_file() -> None:
|
|
|
|
def open() -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
im = Image.open(TEST_GIF)
|
|
|
|
im.load()
|
2023-03-02 23:50:52 +03:00
|
|
|
|
|
|
|
with pytest.warns(ResourceWarning):
|
|
|
|
open()
|
2020-02-23 00:03:01 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_closed_file() -> None:
|
2022-02-21 05:49:01 +03:00
|
|
|
with warnings.catch_warnings():
|
2020-02-23 00:03:01 +03:00
|
|
|
im = Image.open(TEST_GIF)
|
|
|
|
im.load()
|
|
|
|
im.close()
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_seek_after_close() -> None:
|
2022-04-17 05:14:53 +03:00
|
|
|
im = Image.open("Tests/images/iss634.gif")
|
|
|
|
im.load()
|
|
|
|
im.close()
|
|
|
|
|
|
|
|
with pytest.raises(ValueError):
|
|
|
|
im.is_animated
|
|
|
|
with pytest.raises(ValueError):
|
|
|
|
im.n_frames
|
|
|
|
with pytest.raises(ValueError):
|
|
|
|
im.seek(1)
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_context_manager() -> None:
|
2022-02-21 05:49:01 +03:00
|
|
|
with warnings.catch_warnings():
|
Improve handling of file resources
Follow Python's file object semantics. User code is responsible for
closing resources (usually through a context manager) in a deterministic
way.
To achieve this, remove __del__ functions. These functions used to
closed open file handlers in an attempt to silence Python
ResourceWarnings. However, using __del__ has the following drawbacks:
- __del__ isn't called until the object's reference count reaches 0.
Therefore, resource handlers remain open or in use longer than
necessary.
- The __del__ method isn't guaranteed to execute on system exit. See the
Python documentation:
https://docs.python.org/3/reference/datamodel.html#object.__del__
> It is not guaranteed that __del__() methods are called for objects
> that still exist when the interpreter exits.
- Exceptions that occur inside __del__ are ignored instead of raised.
This has the potential of hiding bugs. This is also in the Python
documentation:
> Warning: Due to the precarious circumstances under which __del__()
> methods are invoked, exceptions that occur during their execution
> are ignored, and a warning is printed to sys.stderr instead.
Instead, always close resource handlers when they are no longer in use.
This will close the file handler at a specified point in the user's code
and not wait until the interpreter chooses to. It is always guaranteed
to run. And, if an exception occurs while closing the file handler, the
bug will not be ignored.
Now, when code receives a ResourceWarning, it will highlight an area
that is mishandling resources. It should not simply be silenced, but
fixed by closing resources with a context manager.
All warnings that were emitted during tests have been cleaned up. To
enable warnings, I passed the `-Wa` CLI option to Python. This exposed
some mishandling of resources in ImageFile.__init__() and
SpiderImagePlugin.loadImageSeries(), they too were fixed.
2019-05-25 19:30:58 +03:00
|
|
|
with Image.open(TEST_GIF) as im:
|
|
|
|
im.load()
|
2019-06-13 18:53:42 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_invalid_file() -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
invalid_file = "Tests/images/flower.jpg"
|
|
|
|
|
|
|
|
with pytest.raises(SyntaxError):
|
|
|
|
GifImagePlugin.GifImageFile(invalid_file)
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_l_mode_transparency() -> None:
|
2022-03-01 12:11:35 +03:00
|
|
|
with Image.open("Tests/images/no_palette_with_transparency.gif") as im:
|
|
|
|
assert im.mode == "L"
|
2022-03-22 14:07:37 +03:00
|
|
|
assert im.load()[0, 0] == 128
|
2022-03-01 12:11:35 +03:00
|
|
|
assert im.info["transparency"] == 255
|
|
|
|
|
|
|
|
im.seek(1)
|
2022-03-22 14:07:37 +03:00
|
|
|
assert im.mode == "L"
|
|
|
|
assert im.load()[0, 0] == 128
|
2022-03-01 12:11:35 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_l_mode_after_rgb() -> None:
|
2022-09-13 17:05:23 +03:00
|
|
|
with Image.open("Tests/images/no_palette_after_rgb.gif") as im:
|
|
|
|
im.seek(1)
|
|
|
|
assert im.mode == "RGB"
|
|
|
|
|
|
|
|
im.seek(2)
|
|
|
|
assert im.mode == "RGB"
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_palette_not_needed_for_second_frame() -> None:
|
2022-10-06 00:46:31 +03:00
|
|
|
with Image.open("Tests/images/palette_not_needed_for_second_frame.gif") as im:
|
|
|
|
im.seek(1)
|
|
|
|
assert_image_similar(im, hopper("L").convert("RGB"), 8)
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_strategy() -> None:
|
2022-09-17 10:56:36 +03:00
|
|
|
with Image.open("Tests/images/iss634.gif") as im:
|
|
|
|
expected_rgb_always = im.convert("RGB")
|
|
|
|
|
2022-03-22 12:28:49 +03:00
|
|
|
with Image.open("Tests/images/chi.gif") as im:
|
2022-09-17 10:56:36 +03:00
|
|
|
expected_rgb_always_rgba = im.convert("RGBA")
|
2022-03-20 08:28:31 +03:00
|
|
|
|
2022-03-22 12:28:49 +03:00
|
|
|
im.seek(1)
|
2022-09-17 10:56:36 +03:00
|
|
|
expected_different = im.convert("RGB")
|
2022-03-22 12:28:49 +03:00
|
|
|
|
|
|
|
try:
|
2022-03-29 13:26:29 +03:00
|
|
|
GifImagePlugin.LOADING_STRATEGY = GifImagePlugin.LoadingStrategy.RGB_ALWAYS
|
2022-09-17 10:56:36 +03:00
|
|
|
with Image.open("Tests/images/iss634.gif") as im:
|
2022-03-22 12:28:49 +03:00
|
|
|
assert im.mode == "RGB"
|
2022-09-17 10:56:36 +03:00
|
|
|
assert_image_equal(im, expected_rgb_always)
|
|
|
|
|
|
|
|
with Image.open("Tests/images/chi.gif") as im:
|
|
|
|
assert im.mode == "RGBA"
|
|
|
|
assert_image_equal(im, expected_rgb_always_rgba)
|
2022-03-22 12:28:49 +03:00
|
|
|
|
2022-03-29 13:26:29 +03:00
|
|
|
GifImagePlugin.LOADING_STRATEGY = (
|
|
|
|
GifImagePlugin.LoadingStrategy.RGB_AFTER_DIFFERENT_PALETTE_ONLY
|
2022-03-22 12:28:49 +03:00
|
|
|
)
|
|
|
|
# Stay in P mode with only a global palette
|
|
|
|
with Image.open("Tests/images/chi.gif") as im:
|
|
|
|
assert im.mode == "P"
|
2022-03-20 08:28:31 +03:00
|
|
|
|
2022-03-22 12:28:49 +03:00
|
|
|
im.seek(1)
|
|
|
|
assert im.mode == "P"
|
2022-09-17 10:56:36 +03:00
|
|
|
assert_image_equal(im.convert("RGB"), expected_different)
|
2022-03-22 12:28:49 +03:00
|
|
|
|
|
|
|
# Change to RGB mode when a frame has an individual palette
|
|
|
|
with Image.open("Tests/images/iss634.gif") as im:
|
|
|
|
assert im.mode == "P"
|
|
|
|
|
|
|
|
im.seek(1)
|
|
|
|
assert im.mode == "RGB"
|
|
|
|
finally:
|
2022-03-29 13:26:29 +03:00
|
|
|
GifImagePlugin.LOADING_STRATEGY = GifImagePlugin.LoadingStrategy.RGB_AFTER_FIRST
|
2022-03-01 12:11:35 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_optimize() -> None:
|
2024-01-31 13:55:32 +03:00
|
|
|
def test_grayscale(optimize: int) -> int:
|
2020-02-23 00:03:01 +03:00
|
|
|
im = Image.new("L", (1, 1), 0)
|
|
|
|
filename = BytesIO()
|
|
|
|
im.save(filename, "GIF", optimize=optimize)
|
|
|
|
return len(filename.getvalue())
|
|
|
|
|
2024-01-31 13:55:32 +03:00
|
|
|
def test_bilevel(optimize: int) -> int:
|
2020-02-23 00:03:01 +03:00
|
|
|
im = Image.new("1", (1, 1), 0)
|
2015-04-24 11:24:52 +03:00
|
|
|
test_file = BytesIO()
|
2020-02-23 00:03:01 +03:00
|
|
|
im.save(test_file, "GIF", optimize=optimize)
|
|
|
|
return len(test_file.getvalue())
|
|
|
|
|
2020-12-26 21:07:16 +03:00
|
|
|
assert test_grayscale(0) == 799
|
|
|
|
assert test_grayscale(1) == 43
|
|
|
|
assert test_bilevel(0) == 799
|
|
|
|
assert test_bilevel(1) == 799
|
2020-02-23 00:03:01 +03:00
|
|
|
|
|
|
|
|
2023-01-08 15:48:56 +03:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"colors, size, expected_palette_length",
|
|
|
|
(
|
|
|
|
# These do optimize the palette
|
|
|
|
(256, 511, 256),
|
|
|
|
(255, 511, 255),
|
|
|
|
(129, 511, 129),
|
|
|
|
(128, 511, 128),
|
|
|
|
(64, 511, 64),
|
|
|
|
(4, 511, 4),
|
|
|
|
# These don't optimize the palette
|
|
|
|
(128, 513, 256),
|
|
|
|
(64, 513, 256),
|
|
|
|
(4, 513, 256),
|
|
|
|
),
|
|
|
|
)
|
2024-01-31 13:55:32 +03:00
|
|
|
def test_optimize_correctness(
|
|
|
|
colors: int, size: int, expected_palette_length: int
|
|
|
|
) -> None:
|
2023-01-08 15:48:56 +03:00
|
|
|
# 256 color Palette image, posterize to > 128 and < 128 levels.
|
|
|
|
# Size bigger and smaller than 512x512.
|
2020-02-23 00:03:01 +03:00
|
|
|
# Check the palette for number of colors allocated.
|
2023-01-08 15:48:56 +03:00
|
|
|
# Check for correctness after conversion back to RGB.
|
|
|
|
|
|
|
|
# make an image with empty colors in the start of the palette range
|
|
|
|
im = Image.frombytes(
|
|
|
|
"P", (colors, colors), bytes(range(256 - colors, 256)) * colors
|
|
|
|
)
|
|
|
|
im = im.resize((size, size))
|
|
|
|
outfile = BytesIO()
|
|
|
|
im.save(outfile, "GIF")
|
|
|
|
outfile.seek(0)
|
|
|
|
with Image.open(outfile) as reloaded:
|
|
|
|
# check palette length
|
|
|
|
palette_length = max(i + 1 for i, v in enumerate(reloaded.histogram()) if v)
|
|
|
|
assert expected_palette_length == palette_length
|
|
|
|
|
|
|
|
assert_image_equal(im.convert("RGB"), reloaded.convert("RGB"))
|
2013-11-08 04:39:57 +04:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_optimize_full_l() -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
im = Image.frombytes("L", (16, 16), bytes(range(256)))
|
|
|
|
test_file = BytesIO()
|
|
|
|
im.save(test_file, "GIF", optimize=True)
|
|
|
|
assert im.mode == "L"
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_optimize_if_palette_can_be_reduced_by_half() -> None:
|
2023-07-17 16:04:43 +03:00
|
|
|
im = Image.new("P", (8, 1))
|
|
|
|
im.palette = ImagePalette.raw("RGB", bytes((0, 0, 0) * 150))
|
|
|
|
for i in range(8):
|
|
|
|
im.putpixel((i, 0), (i + 1, 0, 0))
|
2022-06-19 09:47:50 +03:00
|
|
|
|
2023-02-06 22:27:15 +03:00
|
|
|
for optimize, colors in ((False, 256), (True, 8)):
|
2022-06-19 03:07:58 +03:00
|
|
|
out = BytesIO()
|
2023-07-17 16:04:43 +03:00
|
|
|
im.save(out, "GIF", optimize=optimize)
|
2022-06-19 03:07:58 +03:00
|
|
|
with Image.open(out) as reloaded:
|
2022-06-19 09:47:50 +03:00
|
|
|
assert len(reloaded.palette.palette) // 3 == colors
|
2022-06-19 03:07:58 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_full_palette_second_frame(tmp_path: Path) -> None:
|
2023-11-25 11:16:32 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
im = Image.new("P", (1, 256))
|
|
|
|
|
|
|
|
full_palette_im = Image.new("P", (1, 256))
|
|
|
|
for i in range(256):
|
|
|
|
full_palette_im.putpixel((0, i), i)
|
|
|
|
full_palette_im.palette = ImagePalette.ImagePalette(
|
|
|
|
"RGB", bytearray(i // 3 for i in range(768))
|
|
|
|
)
|
|
|
|
full_palette_im.palette.dirty = 1
|
|
|
|
|
|
|
|
im.save(out, save_all=True, append_images=[full_palette_im])
|
|
|
|
|
|
|
|
with Image.open(out) as reloaded:
|
|
|
|
reloaded.seek(1)
|
|
|
|
|
|
|
|
for i in range(256):
|
|
|
|
reloaded.getpixel((0, i)) == i
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_roundtrip(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
im = hopper()
|
|
|
|
im.save(out)
|
|
|
|
with Image.open(out) as reread:
|
|
|
|
assert_image_similar(reread.convert("RGB"), im, 50)
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_roundtrip2(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
# see https://github.com/python-pillow/Pillow/issues/403
|
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
with Image.open(TEST_GIF) as im:
|
|
|
|
im2 = im.copy()
|
|
|
|
im2.save(out)
|
|
|
|
with Image.open(out) as reread:
|
|
|
|
assert_image_similar(reread.convert("RGB"), hopper(), 50)
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_roundtrip_save_all(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
# Single frame image
|
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
im = hopper()
|
|
|
|
im.save(out, save_all=True)
|
|
|
|
with Image.open(out) as reread:
|
|
|
|
assert_image_similar(reread.convert("RGB"), im, 50)
|
|
|
|
|
|
|
|
# Multiframe image
|
|
|
|
with Image.open("Tests/images/dispose_bgnd.gif") as im:
|
|
|
|
out = str(tmp_path / "temp.gif")
|
2015-06-30 11:07:23 +03:00
|
|
|
im.save(out, save_all=True)
|
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open(out) as reread:
|
|
|
|
assert reread.n_frames == 5
|
2015-06-30 11:07:23 +03:00
|
|
|
|
2015-07-01 02:18:05 +03:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_roundtrip_save_all_1(tmp_path: Path) -> None:
|
2023-05-24 01:55:14 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
im = Image.new("1", (1, 1))
|
|
|
|
im2 = Image.new("1", (1, 1), 1)
|
|
|
|
im.save(out, save_all=True, append_images=[im2])
|
|
|
|
|
|
|
|
with Image.open(out) as reloaded:
|
|
|
|
assert reloaded.getpixel((0, 0)) == 0
|
|
|
|
|
|
|
|
reloaded.seek(1)
|
|
|
|
assert reloaded.getpixel((0, 0)) == 255
|
|
|
|
|
|
|
|
|
2021-11-29 07:12:24 +03:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"path, mode",
|
|
|
|
(
|
|
|
|
("Tests/images/dispose_bgnd.gif", "RGB"),
|
|
|
|
# Hexeditted copy of dispose_bgnd to add transparency
|
|
|
|
("Tests/images/dispose_bgnd_rgba.gif", "RGBA"),
|
|
|
|
),
|
|
|
|
)
|
2024-01-31 13:55:32 +03:00
|
|
|
def test_loading_multiple_palettes(path: str, mode: str) -> None:
|
2021-11-29 07:12:24 +03:00
|
|
|
with Image.open(path) as im:
|
|
|
|
assert im.mode == "P"
|
|
|
|
first_frame_colors = im.palette.colors.keys()
|
|
|
|
original_color = im.convert("RGB").load()[0, 0]
|
|
|
|
|
|
|
|
im.seek(1)
|
|
|
|
assert im.mode == mode
|
|
|
|
if mode == "RGBA":
|
|
|
|
im = im.convert("RGB")
|
|
|
|
|
|
|
|
# Check a color only from the old palette
|
|
|
|
assert im.load()[0, 0] == original_color
|
|
|
|
|
|
|
|
# Check a color from the new palette
|
|
|
|
assert im.load()[24, 24] not in first_frame_colors
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_headers_saving_for_animated_gifs(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
important_headers = ["background", "version", "duration", "loop"]
|
|
|
|
# Multiframe image
|
|
|
|
with Image.open("Tests/images/dispose_bgnd.gif") as im:
|
|
|
|
info = im.info.copy()
|
2015-07-01 02:18:05 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
im.save(out, save_all=True)
|
|
|
|
with Image.open(out) as reread:
|
|
|
|
for header in important_headers:
|
|
|
|
assert info[header] == reread.info[header]
|
2018-09-01 02:28:22 +03:00
|
|
|
|
2015-07-24 12:14:20 +03:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_palette_handling(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
# see https://github.com/python-pillow/Pillow/issues/513
|
2015-07-24 12:14:20 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open(TEST_GIF) as im:
|
|
|
|
im = im.convert("RGB")
|
2014-03-05 10:29:55 +04:00
|
|
|
|
2022-01-15 01:02:31 +03:00
|
|
|
im = im.resize((100, 100), Image.Resampling.LANCZOS)
|
|
|
|
im2 = im.convert("P", palette=Image.Palette.ADAPTIVE, colors=256)
|
2014-03-05 10:02:03 +04:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
f = str(tmp_path / "temp.gif")
|
|
|
|
im2.save(f, optimize=True)
|
2014-06-03 14:02:44 +04:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open(f) as reloaded:
|
|
|
|
assert_image_similar(im, reloaded.convert("RGB"), 10)
|
2014-03-05 10:02:03 +04:00
|
|
|
|
2014-06-03 14:02:44 +04:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_palette_434(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
# see https://github.com/python-pillow/Pillow/issues/434
|
2014-06-03 14:02:44 +04:00
|
|
|
|
2024-01-31 13:55:32 +03:00
|
|
|
def roundtrip(im: Image.Image, **kwargs: bool) -> Image.Image:
|
2020-02-23 00:03:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
2024-01-31 13:55:32 +03:00
|
|
|
im.copy().save(out, **kwargs)
|
2020-02-23 00:03:01 +03:00
|
|
|
reloaded = Image.open(out)
|
2014-03-05 10:29:55 +04:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
return reloaded
|
2014-03-05 10:29:55 +04:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
orig = "Tests/images/test.colors.gif"
|
|
|
|
with Image.open(orig) as im:
|
|
|
|
with roundtrip(im) as reloaded:
|
|
|
|
assert_image_similar(im, reloaded, 1)
|
|
|
|
with roundtrip(im, optimize=True) as reloaded:
|
|
|
|
assert_image_similar(im, reloaded, 1)
|
2014-03-05 10:29:55 +04:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
im = im.convert("RGB")
|
|
|
|
# check automatic P conversion
|
|
|
|
with roundtrip(im) as reloaded:
|
|
|
|
reloaded = reloaded.convert("RGB")
|
|
|
|
assert_image_equal(im, reloaded)
|
2014-06-03 14:02:44 +04:00
|
|
|
|
2014-06-27 07:37:49 +04:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
@pytest.mark.skipif(not netpbm_available(), reason="Netpbm not available")
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_save_netpbm_bmp_mode(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open(TEST_GIF) as img:
|
|
|
|
img = img.convert("RGB")
|
2014-06-27 07:37:49 +04:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
tempfile = str(tmp_path / "temp.gif")
|
|
|
|
GifImagePlugin._save_netpbm(img, 0, tempfile)
|
|
|
|
with Image.open(tempfile) as reloaded:
|
|
|
|
assert_image_similar(img, reloaded.convert("RGB"), 0)
|
2014-06-27 07:37:49 +04:00
|
|
|
|
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
@pytest.mark.skipif(not netpbm_available(), reason="Netpbm not available")
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_save_netpbm_l_mode(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open(TEST_GIF) as img:
|
|
|
|
img = img.convert("L")
|
|
|
|
|
|
|
|
tempfile = str(tmp_path / "temp.gif")
|
|
|
|
GifImagePlugin._save_netpbm(img, 0, tempfile)
|
|
|
|
with Image.open(tempfile) as reloaded:
|
|
|
|
assert_image_similar(img, reloaded.convert("L"), 0)
|
2014-07-07 22:47:18 +04:00
|
|
|
|
2018-09-01 02:28:22 +03:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_seek() -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open("Tests/images/dispose_none.gif") as img:
|
|
|
|
frame_count = 0
|
|
|
|
try:
|
|
|
|
while True:
|
|
|
|
frame_count += 1
|
|
|
|
img.seek(img.tell() + 1)
|
|
|
|
except EOFError:
|
|
|
|
assert frame_count == 5
|
2018-09-01 02:28:22 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_seek_info() -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open("Tests/images/iss634.gif") as im:
|
|
|
|
info = im.info.copy()
|
2019-03-16 12:02:24 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
im.seek(1)
|
|
|
|
im.seek(0)
|
2019-03-16 12:02:24 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
assert im.info == info
|
2017-08-18 13:20:27 +03:00
|
|
|
|
2015-06-18 17:49:18 +03:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_seek_rewind() -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open("Tests/images/iss634.gif") as im:
|
|
|
|
im.seek(2)
|
|
|
|
im.seek(1)
|
2017-09-06 06:19:33 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open("Tests/images/iss634.gif") as expected:
|
|
|
|
expected.seek(1)
|
|
|
|
assert_image_equal(im, expected)
|
2017-09-06 06:19:33 +03:00
|
|
|
|
2015-06-07 18:01:34 +03:00
|
|
|
|
2022-04-16 10:07:39 +03:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"path, n_frames",
|
|
|
|
(
|
|
|
|
(TEST_GIF, 1),
|
|
|
|
("Tests/images/comment_after_last_frame.gif", 2),
|
|
|
|
("Tests/images/iss634.gif", 42),
|
|
|
|
),
|
|
|
|
)
|
2024-01-31 13:55:32 +03:00
|
|
|
def test_n_frames(path: str, n_frames: int) -> None:
|
2022-04-16 10:07:39 +03:00
|
|
|
# Test is_animated before n_frames
|
|
|
|
with Image.open(path) as im:
|
|
|
|
assert im.is_animated == (n_frames != 1)
|
|
|
|
|
|
|
|
# Test is_animated after n_frames
|
|
|
|
with Image.open(path) as im:
|
|
|
|
assert im.n_frames == n_frames
|
|
|
|
assert im.is_animated == (n_frames != 1)
|
2014-07-07 22:47:18 +04:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_no_change() -> None:
|
2022-02-22 00:55:02 +03:00
|
|
|
# Test n_frames does not change the image
|
|
|
|
with Image.open("Tests/images/dispose_bgnd.gif") as im:
|
|
|
|
im.seek(1)
|
|
|
|
expected = im.copy()
|
|
|
|
assert im.n_frames == 5
|
|
|
|
assert_image_equal(im, expected)
|
|
|
|
|
|
|
|
# Test is_animated does not change the image
|
|
|
|
with Image.open("Tests/images/dispose_bgnd.gif") as im:
|
|
|
|
im.seek(3)
|
|
|
|
expected = im.copy()
|
|
|
|
assert im.is_animated
|
|
|
|
assert_image_equal(im, expected)
|
|
|
|
|
2022-07-21 02:05:14 +03:00
|
|
|
with Image.open("Tests/images/comment_after_only_frame.gif") as im:
|
|
|
|
expected = Image.new("P", (1, 1))
|
|
|
|
assert not im.is_animated
|
|
|
|
assert_image_equal(im, expected)
|
|
|
|
|
2022-02-22 00:55:02 +03:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_eoferror() -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open(TEST_GIF) as im:
|
|
|
|
n_frames = im.n_frames
|
|
|
|
|
|
|
|
# Test seeking past the last frame
|
|
|
|
with pytest.raises(EOFError):
|
|
|
|
im.seek(n_frames)
|
|
|
|
assert im.tell() < n_frames
|
|
|
|
|
|
|
|
# Test that seeking to the last frame does not raise an error
|
|
|
|
im.seek(n_frames - 1)
|
2017-12-22 01:26:58 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_first_frame_transparency() -> None:
|
2021-06-25 14:54:21 +03:00
|
|
|
with Image.open("Tests/images/first_frame_transparency.gif") as im:
|
|
|
|
px = im.load()
|
|
|
|
assert px[0, 0] == im.info["transparency"]
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_dispose_none() -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open("Tests/images/dispose_none.gif") as img:
|
|
|
|
try:
|
|
|
|
while True:
|
Improve handling of file resources
Follow Python's file object semantics. User code is responsible for
closing resources (usually through a context manager) in a deterministic
way.
To achieve this, remove __del__ functions. These functions used to
closed open file handlers in an attempt to silence Python
ResourceWarnings. However, using __del__ has the following drawbacks:
- __del__ isn't called until the object's reference count reaches 0.
Therefore, resource handlers remain open or in use longer than
necessary.
- The __del__ method isn't guaranteed to execute on system exit. See the
Python documentation:
https://docs.python.org/3/reference/datamodel.html#object.__del__
> It is not guaranteed that __del__() methods are called for objects
> that still exist when the interpreter exits.
- Exceptions that occur inside __del__ are ignored instead of raised.
This has the potential of hiding bugs. This is also in the Python
documentation:
> Warning: Due to the precarious circumstances under which __del__()
> methods are invoked, exceptions that occur during their execution
> are ignored, and a warning is printed to sys.stderr instead.
Instead, always close resource handlers when they are no longer in use.
This will close the file handler at a specified point in the user's code
and not wait until the interpreter chooses to. It is always guaranteed
to run. And, if an exception occurs while closing the file handler, the
bug will not be ignored.
Now, when code receives a ResourceWarning, it will highlight an area
that is mishandling resources. It should not simply be silenced, but
fixed by closing resources with a context manager.
All warnings that were emitted during tests have been cleaned up. To
enable warnings, I passed the `-Wa` CLI option to Python. This exposed
some mishandling of resources in ImageFile.__init__() and
SpiderImagePlugin.loadImageSeries(), they too were fixed.
2019-05-25 19:30:58 +03:00
|
|
|
img.seek(img.tell() + 1)
|
2020-02-23 00:03:01 +03:00
|
|
|
assert img.disposal_method == 1
|
|
|
|
except EOFError:
|
|
|
|
pass
|
2017-01-31 10:22:54 +03:00
|
|
|
|
2019-03-15 02:29:33 +03:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_dispose_none_load_end() -> None:
|
2020-12-23 05:22:53 +03:00
|
|
|
# Test image created with:
|
|
|
|
#
|
|
|
|
# im = Image.open("transparent.gif")
|
|
|
|
# im_rotated = im.rotate(180)
|
|
|
|
# im.save("dispose_none_load_end.gif",
|
|
|
|
# save_all=True, append_images=[im_rotated], disposal=[1,2])
|
|
|
|
with Image.open("Tests/images/dispose_none_load_end.gif") as img:
|
|
|
|
img.seek(1)
|
|
|
|
|
2021-11-29 09:49:06 +03:00
|
|
|
assert_image_equal_tofile(img, "Tests/images/dispose_none_load_end_second.png")
|
2020-12-23 05:22:53 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_dispose_background() -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open("Tests/images/dispose_bgnd.gif") as img:
|
|
|
|
try:
|
|
|
|
while True:
|
|
|
|
img.seek(img.tell() + 1)
|
|
|
|
assert img.disposal_method == 2
|
|
|
|
except EOFError:
|
|
|
|
pass
|
2019-03-15 02:29:33 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_dispose_background_transparency() -> None:
|
2021-10-12 01:45:52 +03:00
|
|
|
with Image.open("Tests/images/dispose_bgnd_transparency.gif") as img:
|
|
|
|
img.seek(2)
|
2021-11-29 09:49:06 +03:00
|
|
|
px = img.load()
|
2021-10-12 01:45:52 +03:00
|
|
|
assert px[35, 30][3] == 0
|
|
|
|
|
|
|
|
|
2022-03-22 12:28:49 +03:00
|
|
|
@pytest.mark.parametrize(
|
2022-03-29 13:26:29 +03:00
|
|
|
"loading_strategy, expected_colors",
|
2022-03-22 12:28:49 +03:00
|
|
|
(
|
|
|
|
(
|
2022-03-29 13:26:29 +03:00
|
|
|
GifImagePlugin.LoadingStrategy.RGB_AFTER_FIRST,
|
2022-03-22 12:28:49 +03:00
|
|
|
(
|
|
|
|
(2, 1, 2),
|
2022-03-22 14:07:37 +03:00
|
|
|
((0, 255, 24, 255), (0, 0, 255, 255), (0, 255, 24, 255)),
|
|
|
|
((0, 0, 0, 0), (0, 0, 255, 255), (0, 0, 0, 0)),
|
2022-03-22 12:28:49 +03:00
|
|
|
),
|
|
|
|
),
|
|
|
|
(
|
2022-03-29 13:26:29 +03:00
|
|
|
GifImagePlugin.LoadingStrategy.RGB_AFTER_DIFFERENT_PALETTE_ONLY,
|
2022-03-22 12:28:49 +03:00
|
|
|
(
|
|
|
|
(2, 1, 2),
|
2022-03-22 14:07:37 +03:00
|
|
|
(0, 1, 0),
|
|
|
|
(2, 1, 2),
|
2022-03-22 12:28:49 +03:00
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
)
|
2024-01-31 13:55:32 +03:00
|
|
|
def test_transparent_dispose(
|
|
|
|
loading_strategy: GifImagePlugin.LoadingStrategy,
|
|
|
|
expected_colors: tuple[tuple[int | tuple[int, int, int, int], ...]],
|
|
|
|
) -> None:
|
2022-03-29 13:26:29 +03:00
|
|
|
GifImagePlugin.LOADING_STRATEGY = loading_strategy
|
2022-03-22 14:07:37 +03:00
|
|
|
try:
|
|
|
|
with Image.open("Tests/images/transparent_dispose.gif") as img:
|
|
|
|
for frame in range(3):
|
|
|
|
img.seek(frame)
|
|
|
|
for x in range(3):
|
|
|
|
color = img.getpixel((x, 0))
|
|
|
|
assert color == expected_colors[frame][x]
|
|
|
|
finally:
|
2022-03-29 13:26:29 +03:00
|
|
|
GifImagePlugin.LOADING_STRATEGY = GifImagePlugin.LoadingStrategy.RGB_AFTER_FIRST
|
2018-10-24 06:34:29 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_dispose_previous() -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open("Tests/images/dispose_prev.gif") as img:
|
|
|
|
try:
|
|
|
|
while True:
|
|
|
|
img.seek(img.tell() + 1)
|
|
|
|
assert img.disposal_method == 3
|
|
|
|
except EOFError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_dispose_previous_first_frame() -> None:
|
2021-04-15 12:01:12 +03:00
|
|
|
with Image.open("Tests/images/dispose_prev_first_frame.gif") as im:
|
|
|
|
im.seek(1)
|
|
|
|
assert_image_equal_tofile(
|
2021-11-29 09:49:06 +03:00
|
|
|
im, "Tests/images/dispose_prev_first_frame_seeked.png"
|
2021-04-15 12:01:12 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_previous_frame_loaded() -> None:
|
2021-04-06 12:31:51 +03:00
|
|
|
with Image.open("Tests/images/dispose_none.gif") as img:
|
|
|
|
img.load()
|
|
|
|
img.seek(1)
|
|
|
|
img.load()
|
|
|
|
img.seek(2)
|
|
|
|
with Image.open("Tests/images/dispose_none.gif") as img_skipped:
|
|
|
|
img_skipped.seek(2)
|
|
|
|
assert_image_equal(img_skipped, img)
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_save_dispose(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
im_list = [
|
|
|
|
Image.new("L", (100, 100), "#000"),
|
|
|
|
Image.new("L", (100, 100), "#111"),
|
|
|
|
Image.new("L", (100, 100), "#222"),
|
|
|
|
]
|
|
|
|
for method in range(0, 4):
|
|
|
|
im_list[0].save(out, save_all=True, append_images=im_list[1:], disposal=method)
|
|
|
|
with Image.open(out) as img:
|
|
|
|
for _ in range(2):
|
|
|
|
img.seek(img.tell() + 1)
|
|
|
|
assert img.disposal_method == method
|
2019-03-14 23:40:31 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
# Check per frame disposal
|
|
|
|
im_list[0].save(
|
|
|
|
out,
|
|
|
|
save_all=True,
|
|
|
|
append_images=im_list[1:],
|
|
|
|
disposal=tuple(range(len(im_list))),
|
|
|
|
)
|
2019-03-15 02:29:33 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open(out) as img:
|
|
|
|
for i in range(2):
|
|
|
|
img.seek(img.tell() + 1)
|
|
|
|
assert img.disposal_method == i + 1
|
2019-03-14 23:40:31 +03:00
|
|
|
|
2019-03-15 02:29:33 +03:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_dispose2_palette(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
2019-03-22 17:19:01 +03:00
|
|
|
|
2023-10-19 11:12:01 +03:00
|
|
|
# Four colors: white, gray, black, red
|
2020-02-23 00:03:01 +03:00
|
|
|
circles = [(255, 255, 255), (153, 153, 153), (0, 0, 0), (255, 0, 0)]
|
2019-03-22 17:19:01 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
im_list = []
|
|
|
|
for circle in circles:
|
2021-06-28 13:21:46 +03:00
|
|
|
# Red background
|
2020-02-23 00:03:01 +03:00
|
|
|
img = Image.new("RGB", (100, 100), (255, 0, 0))
|
2019-03-22 17:19:01 +03:00
|
|
|
|
2021-06-28 13:21:46 +03:00
|
|
|
# Circle in center of each frame
|
2020-02-23 00:03:01 +03:00
|
|
|
d = ImageDraw.Draw(img)
|
|
|
|
d.ellipse([(40, 40), (60, 60)], fill=circle)
|
2019-03-22 17:19:01 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
im_list.append(img)
|
2019-03-22 17:19:01 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
im_list[0].save(out, save_all=True, append_images=im_list[1:], disposal=2)
|
2019-03-22 17:19:01 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open(out) as img:
|
|
|
|
for i, circle in enumerate(circles):
|
|
|
|
img.seek(i)
|
|
|
|
rgb_img = img.convert("RGB")
|
2019-03-22 17:19:01 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
# Check top left pixel matches background
|
|
|
|
assert rgb_img.getpixel((0, 0)) == (255, 0, 0)
|
2019-03-22 17:19:01 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
# Center remains red every frame
|
|
|
|
assert rgb_img.getpixel((50, 50)) == circle
|
2019-03-22 17:19:01 +03:00
|
|
|
|
2024-02-09 11:47:09 +03:00
|
|
|
# Check that frame transparency wasn't added unnecessarily
|
|
|
|
assert img._frame_transparency is None
|
|
|
|
|
2019-03-22 17:19:01 +03:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_dispose2_diff(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
2019-03-14 23:40:31 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
# 4 frames: red/blue, red/red, blue/blue, red/blue
|
|
|
|
circles = [
|
|
|
|
((255, 0, 0, 255), (0, 0, 255, 255)),
|
|
|
|
((255, 0, 0, 255), (255, 0, 0, 255)),
|
|
|
|
((0, 0, 255, 255), (0, 0, 255, 255)),
|
|
|
|
((255, 0, 0, 255), (0, 0, 255, 255)),
|
|
|
|
]
|
2019-06-29 16:06:45 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
im_list = []
|
|
|
|
for i in range(len(circles)):
|
|
|
|
# Transparent BG
|
|
|
|
img = Image.new("RGBA", (100, 100), (255, 255, 255, 0))
|
2019-06-29 16:06:45 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
# Two circles per frame
|
|
|
|
d = ImageDraw.Draw(img)
|
|
|
|
d.ellipse([(0, 30), (40, 70)], fill=circles[i][0])
|
|
|
|
d.ellipse([(60, 30), (100, 70)], fill=circles[i][1])
|
2019-06-29 16:06:45 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
im_list.append(img)
|
2019-06-29 16:06:45 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
im_list[0].save(
|
|
|
|
out, save_all=True, append_images=im_list[1:], disposal=2, transparency=0
|
|
|
|
)
|
2019-06-29 16:06:45 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open(out) as img:
|
|
|
|
for i, colours in enumerate(circles):
|
|
|
|
img.seek(i)
|
|
|
|
rgb_img = img.convert("RGBA")
|
2019-06-29 16:06:45 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
# Check left circle is correct colour
|
|
|
|
assert rgb_img.getpixel((20, 50)) == colours[0]
|
2014-07-07 22:47:18 +04:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
# Check right circle is correct colour
|
|
|
|
assert rgb_img.getpixel((80, 50)) == colours[1]
|
2015-04-04 03:45:30 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
# Check BG is correct colour
|
|
|
|
assert rgb_img.getpixel((1, 1)) == (255, 255, 255, 0)
|
2018-04-11 01:57:31 +03:00
|
|
|
|
2015-04-04 03:45:30 +03:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_dispose2_background(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
|
|
|
|
im_list = []
|
|
|
|
|
|
|
|
im = Image.new("P", (100, 100))
|
|
|
|
d = ImageDraw.Draw(im)
|
|
|
|
d.rectangle([(50, 0), (100, 100)], fill="#f00")
|
|
|
|
d.rectangle([(0, 0), (50, 100)], fill="#0f0")
|
|
|
|
im_list.append(im)
|
|
|
|
|
|
|
|
im = Image.new("P", (100, 100))
|
|
|
|
d = ImageDraw.Draw(im)
|
|
|
|
d.rectangle([(0, 0), (100, 50)], fill="#f00")
|
|
|
|
d.rectangle([(0, 50), (100, 100)], fill="#0f0")
|
|
|
|
im_list.append(im)
|
|
|
|
|
|
|
|
im_list[0].save(
|
|
|
|
out, save_all=True, append_images=im_list[1:], disposal=[0, 2], background=1
|
|
|
|
)
|
|
|
|
|
|
|
|
with Image.open(out) as im:
|
|
|
|
im.seek(1)
|
2021-11-29 09:49:06 +03:00
|
|
|
assert im.getpixel((0, 0)) == (255, 0, 0)
|
2020-02-23 00:03:01 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_dispose2_background_frame(tmp_path: Path) -> None:
|
2022-12-08 03:35:48 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
|
|
|
|
im_list = [Image.new("RGBA", (1, 20))]
|
|
|
|
|
|
|
|
different_frame = Image.new("RGBA", (1, 20))
|
|
|
|
different_frame.putpixel((0, 10), (255, 0, 0, 255))
|
|
|
|
im_list.append(different_frame)
|
|
|
|
|
|
|
|
# Frame that matches the background
|
|
|
|
im_list.append(Image.new("RGBA", (1, 20)))
|
|
|
|
|
|
|
|
im_list[0].save(out, save_all=True, append_images=im_list[1:], disposal=2)
|
|
|
|
|
|
|
|
with Image.open(out) as im:
|
|
|
|
assert im.n_frames == 3
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_transparency_in_second_frame(tmp_path: Path) -> None:
|
2022-05-21 09:35:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
2021-03-16 16:24:57 +03:00
|
|
|
with Image.open("Tests/images/different_transparency.gif") as im:
|
|
|
|
assert im.info["transparency"] == 0
|
|
|
|
|
|
|
|
# Seek to the second frame
|
|
|
|
im.seek(im.tell() + 1)
|
2021-11-29 09:49:06 +03:00
|
|
|
assert "transparency" not in im.info
|
2021-03-16 16:24:57 +03:00
|
|
|
|
2021-11-29 09:49:06 +03:00
|
|
|
assert_image_equal_tofile(im, "Tests/images/different_transparency_merged.png")
|
2021-03-16 16:24:57 +03:00
|
|
|
|
2022-04-02 13:04:22 +03:00
|
|
|
im.save(out, save_all=True)
|
|
|
|
|
2022-05-21 09:35:01 +03:00
|
|
|
with Image.open(out) as reread:
|
|
|
|
reread.seek(reread.tell() + 1)
|
|
|
|
assert_image_equal_tofile(
|
|
|
|
reread, "Tests/images/different_transparency_merged.png"
|
|
|
|
)
|
2022-04-02 13:04:22 +03:00
|
|
|
|
2021-03-16 16:24:57 +03:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_no_transparency_in_second_frame() -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open("Tests/images/iss634.gif") as img:
|
|
|
|
# Seek to the second frame
|
|
|
|
img.seek(img.tell() + 1)
|
2021-03-16 16:24:57 +03:00
|
|
|
assert "transparency" not in img.info
|
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
# All transparent pixels should be replaced with the color from the first frame
|
2021-03-16 16:24:57 +03:00
|
|
|
assert img.histogram()[255] == 0
|
2020-02-23 00:03:01 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_remapped_transparency(tmp_path: Path) -> None:
|
2022-05-21 09:35:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
|
|
|
|
im = Image.new("P", (1, 2))
|
|
|
|
im2 = im.copy()
|
|
|
|
|
|
|
|
# Add transparency at a higher index
|
|
|
|
# so that it will be optimized to a lower index
|
|
|
|
im.putpixel((0, 1), 5)
|
|
|
|
im.info["transparency"] = 5
|
|
|
|
im.save(out, save_all=True, append_images=[im2])
|
|
|
|
|
|
|
|
with Image.open(out) as reloaded:
|
|
|
|
assert reloaded.info["transparency"] == reloaded.getpixel((0, 1))
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_duration(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
duration = 1000
|
|
|
|
|
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
im = Image.new("L", (100, 100), "#000")
|
|
|
|
|
|
|
|
# Check that the argument has priority over the info settings
|
|
|
|
im.info["duration"] = 100
|
|
|
|
im.save(out, duration=duration)
|
|
|
|
|
|
|
|
with Image.open(out) as reread:
|
|
|
|
assert reread.info["duration"] == duration
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_multiple_duration(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
duration_list = [1000, 2000, 3000]
|
|
|
|
|
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
im_list = [
|
|
|
|
Image.new("L", (100, 100), "#000"),
|
|
|
|
Image.new("L", (100, 100), "#111"),
|
|
|
|
Image.new("L", (100, 100), "#222"),
|
|
|
|
]
|
|
|
|
|
|
|
|
# Duration as list
|
|
|
|
im_list[0].save(
|
|
|
|
out, save_all=True, append_images=im_list[1:], duration=duration_list
|
|
|
|
)
|
|
|
|
with Image.open(out) as reread:
|
|
|
|
for duration in duration_list:
|
2020-02-22 16:06:21 +03:00
|
|
|
assert reread.info["duration"] == duration
|
2020-02-23 00:03:01 +03:00
|
|
|
try:
|
|
|
|
reread.seek(reread.tell() + 1)
|
|
|
|
except EOFError:
|
|
|
|
pass
|
2015-04-04 03:45:30 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
# Duration as tuple
|
|
|
|
im_list[0].save(
|
|
|
|
out, save_all=True, append_images=im_list[1:], duration=tuple(duration_list)
|
|
|
|
)
|
|
|
|
with Image.open(out) as reread:
|
|
|
|
for duration in duration_list:
|
|
|
|
assert reread.info["duration"] == duration
|
|
|
|
try:
|
|
|
|
reread.seek(reread.tell() + 1)
|
|
|
|
except EOFError:
|
|
|
|
pass
|
2016-12-27 14:04:37 +03:00
|
|
|
|
2016-12-27 13:42:58 +03:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_roundtrip_info_duration(tmp_path: Path) -> None:
|
2022-05-03 13:07:47 +03:00
|
|
|
duration_list = [100, 500, 500]
|
|
|
|
|
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
with Image.open("Tests/images/transparent_dispose.gif") as im:
|
|
|
|
assert [
|
|
|
|
frame.info["duration"] for frame in ImageSequence.Iterator(im)
|
|
|
|
] == duration_list
|
|
|
|
|
|
|
|
im.save(out, save_all=True)
|
|
|
|
|
|
|
|
with Image.open(out) as reloaded:
|
|
|
|
assert [
|
|
|
|
frame.info["duration"] for frame in ImageSequence.Iterator(reloaded)
|
|
|
|
] == duration_list
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_roundtrip_info_duration_combined(tmp_path: Path) -> None:
|
2022-12-05 10:53:28 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
with Image.open("Tests/images/duplicate_frame.gif") as im:
|
|
|
|
assert [frame.info["duration"] for frame in ImageSequence.Iterator(im)] == [
|
|
|
|
1000,
|
|
|
|
1000,
|
|
|
|
1000,
|
|
|
|
]
|
|
|
|
im.save(out, save_all=True)
|
|
|
|
|
|
|
|
with Image.open(out) as reloaded:
|
|
|
|
assert [
|
|
|
|
frame.info["duration"] for frame in ImageSequence.Iterator(reloaded)
|
|
|
|
] == [1000, 2000]
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_identical_frames(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
duration_list = [1000, 1500, 2000, 4000]
|
2016-12-27 13:42:58 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
im_list = [
|
|
|
|
Image.new("L", (100, 100), "#000"),
|
|
|
|
Image.new("L", (100, 100), "#000"),
|
|
|
|
Image.new("L", (100, 100), "#000"),
|
|
|
|
Image.new("L", (100, 100), "#111"),
|
|
|
|
]
|
|
|
|
|
|
|
|
# Duration as list
|
|
|
|
im_list[0].save(
|
|
|
|
out, save_all=True, append_images=im_list[1:], duration=duration_list
|
|
|
|
)
|
|
|
|
with Image.open(out) as reread:
|
|
|
|
# Assert that the first three frames were combined
|
|
|
|
assert reread.n_frames == 2
|
2016-12-27 14:04:37 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
# Assert that the new duration is the total of the identical frames
|
|
|
|
assert reread.info["duration"] == 4500
|
2016-12-27 14:04:37 +03:00
|
|
|
|
|
|
|
|
2022-10-03 08:57:42 +03:00
|
|
|
@pytest.mark.parametrize(
|
2023-11-03 13:09:16 +03:00
|
|
|
"duration",
|
|
|
|
(
|
|
|
|
[1000, 1500, 2000],
|
|
|
|
(1000, 1500, 2000),
|
|
|
|
# One more duration than the number of frames
|
|
|
|
[1000, 1500, 2000, 4000],
|
|
|
|
1500,
|
|
|
|
),
|
2022-10-03 08:57:42 +03:00
|
|
|
)
|
2024-01-31 13:55:32 +03:00
|
|
|
def test_identical_frames_to_single_frame(
|
|
|
|
duration: int | list[int], tmp_path: Path
|
|
|
|
) -> None:
|
2022-10-03 08:57:42 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
im_list = [
|
|
|
|
Image.new("L", (100, 100), "#000"),
|
|
|
|
Image.new("L", (100, 100), "#000"),
|
|
|
|
Image.new("L", (100, 100), "#000"),
|
|
|
|
]
|
2016-12-29 03:28:58 +03:00
|
|
|
|
2022-10-03 08:57:42 +03:00
|
|
|
im_list[0].save(out, save_all=True, append_images=im_list[1:], duration=duration)
|
|
|
|
with Image.open(out) as reread:
|
|
|
|
# Assert that all frames were combined
|
|
|
|
assert reread.n_frames == 1
|
|
|
|
|
|
|
|
# Assert that the new duration is the total of the identical frames
|
2023-11-03 13:09:16 +03:00
|
|
|
assert reread.info["duration"] == 4500
|
2015-04-04 03:45:30 +03:00
|
|
|
|
2014-03-05 10:29:55 +04:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_loop_none(tmp_path: Path) -> None:
|
2023-08-09 03:31:34 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
im = Image.new("L", (100, 100), "#000")
|
|
|
|
im.save(out, loop=None)
|
|
|
|
with Image.open(out) as reread:
|
|
|
|
assert "loop" not in reread.info
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_number_of_loops(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
number_of_loops = 2
|
2015-06-11 04:10:05 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
im = Image.new("L", (100, 100), "#000")
|
|
|
|
im.save(out, loop=number_of_loops)
|
|
|
|
with Image.open(out) as reread:
|
|
|
|
assert reread.info["loop"] == number_of_loops
|
2018-10-02 13:52:07 +03:00
|
|
|
|
2022-04-15 09:44:23 +03:00
|
|
|
# Check that even if a subsequent GIF frame has the number of loops specified,
|
|
|
|
# only the value from the first frame is used
|
|
|
|
with Image.open("Tests/images/duplicate_number_of_loops.gif") as im:
|
|
|
|
assert im.info["loop"] == 2
|
|
|
|
|
|
|
|
im.seek(1)
|
|
|
|
assert im.info["loop"] == 2
|
|
|
|
|
2016-05-07 06:57:40 +03:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_background(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
im = Image.new("L", (100, 100), "#000")
|
|
|
|
im.info["background"] = 1
|
|
|
|
im.save(out)
|
|
|
|
with Image.open(out) as reread:
|
|
|
|
assert reread.info["background"] == im.info["background"]
|
|
|
|
|
2022-12-09 02:45:09 +03:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_webp_background(tmp_path: Path) -> None:
|
2022-12-09 02:45:09 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
|
|
|
|
# Test opaque WebP background
|
2020-02-23 00:03:01 +03:00
|
|
|
if features.check("webp") and features.check("webp_anim"):
|
|
|
|
with Image.open("Tests/images/hopper.webp") as im:
|
2022-12-09 02:45:09 +03:00
|
|
|
assert im.info["background"] == (255, 255, 255, 255)
|
2020-02-23 00:03:01 +03:00
|
|
|
im.save(out)
|
|
|
|
|
2022-12-09 02:45:09 +03:00
|
|
|
# Test non-opaque WebP background
|
|
|
|
im = Image.new("L", (100, 100), "#000")
|
|
|
|
im.info["background"] = (0, 0, 0, 0)
|
|
|
|
im.save(out)
|
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_comment(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open(TEST_GIF) as im:
|
|
|
|
assert im.info["comment"] == b"File written by Adobe Photoshop\xa8 4.0"
|
|
|
|
|
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
im = Image.new("L", (100, 100), "#000")
|
|
|
|
im.info["comment"] = b"Test comment text"
|
|
|
|
im.save(out)
|
|
|
|
with Image.open(out) as reread:
|
|
|
|
assert reread.info["comment"] == im.info["comment"]
|
|
|
|
|
|
|
|
im.info["comment"] = "Test comment text"
|
|
|
|
im.save(out)
|
|
|
|
with Image.open(out) as reread:
|
|
|
|
assert reread.info["comment"] == im.info["comment"].encode()
|
|
|
|
|
2022-05-19 13:59:32 +03:00
|
|
|
# Test that GIF89a is used for comments
|
|
|
|
assert reread.info["version"] == b"GIF89a"
|
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_comment_over_255(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
im = Image.new("L", (100, 100), "#000")
|
|
|
|
comment = b"Test comment text"
|
|
|
|
while len(comment) < 256:
|
|
|
|
comment += comment
|
|
|
|
im.info["comment"] = comment
|
|
|
|
im.save(out)
|
|
|
|
with Image.open(out) as reread:
|
|
|
|
assert reread.info["comment"] == comment
|
|
|
|
|
2022-05-19 13:59:32 +03:00
|
|
|
# Test that GIF89a is used for comments
|
2022-05-13 20:33:33 +03:00
|
|
|
assert reread.info["version"] == b"GIF89a"
|
2020-02-23 00:03:01 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_zero_comment_subblocks() -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open("Tests/images/hopper_zero_comment_subblocks.gif") as im:
|
2021-02-21 14:15:56 +03:00
|
|
|
assert_image_equal_tofile(im, TEST_GIF)
|
2020-02-23 00:03:01 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_read_multiple_comment_blocks() -> None:
|
2022-05-13 20:38:39 +03:00
|
|
|
with Image.open("Tests/images/multiple_comments.gif") as im:
|
2022-05-22 07:11:11 +03:00
|
|
|
# Multiple comment blocks in a frame are separated not concatenated
|
|
|
|
assert im.info["comment"] == b"Test comment 1\nTest comment 2"
|
2022-05-13 20:38:39 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_empty_string_comment(tmp_path: Path) -> None:
|
2022-05-13 21:45:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
2022-05-21 16:29:55 +03:00
|
|
|
with Image.open("Tests/images/chi.gif") as im:
|
|
|
|
assert "comment" in im.info
|
|
|
|
|
|
|
|
# Empty string comment should suppress existing comment
|
2022-05-13 21:45:01 +03:00
|
|
|
im.save(out, save_all=True, comment="")
|
2022-05-21 16:29:55 +03:00
|
|
|
|
|
|
|
with Image.open(out) as reread:
|
|
|
|
for frame in ImageSequence.Iterator(reread):
|
|
|
|
assert "comment" not in frame.info
|
2022-05-13 21:45:01 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_retain_comment_in_subsequent_frames(tmp_path: Path) -> None:
|
2022-05-22 08:30:16 +03:00
|
|
|
# Test that a comment block at the beginning is kept
|
|
|
|
with Image.open("Tests/images/chi.gif") as im:
|
|
|
|
for frame in ImageSequence.Iterator(im):
|
|
|
|
assert frame.info["comment"] == b"Created with GIMP"
|
|
|
|
|
|
|
|
with Image.open("Tests/images/second_frame_comment.gif") as im:
|
|
|
|
assert "comment" not in im.info
|
|
|
|
|
|
|
|
# Test that a comment in the middle is read
|
|
|
|
im.seek(1)
|
|
|
|
assert im.info["comment"] == b"Comment in the second frame"
|
|
|
|
|
|
|
|
# Test that it is still present in a later frame
|
|
|
|
im.seek(2)
|
|
|
|
assert im.info["comment"] == b"Comment in the second frame"
|
|
|
|
|
|
|
|
# Test that rewinding removes the comment
|
|
|
|
im.seek(0)
|
|
|
|
assert "comment" not in im.info
|
|
|
|
|
|
|
|
# Test that a saved image keeps the comment
|
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
with Image.open("Tests/images/dispose_prev.gif") as im:
|
|
|
|
im.save(out, save_all=True, comment="Test")
|
|
|
|
|
|
|
|
with Image.open(out) as reread:
|
|
|
|
for frame in ImageSequence.Iterator(reread):
|
|
|
|
assert frame.info["comment"] == b"Test"
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_version(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
|
2024-01-31 13:55:32 +03:00
|
|
|
def assert_version_after_save(im: Image.Image, version: bytes) -> None:
|
2018-11-27 13:05:41 +03:00
|
|
|
im.save(out)
|
Improve handling of file resources
Follow Python's file object semantics. User code is responsible for
closing resources (usually through a context manager) in a deterministic
way.
To achieve this, remove __del__ functions. These functions used to
closed open file handlers in an attempt to silence Python
ResourceWarnings. However, using __del__ has the following drawbacks:
- __del__ isn't called until the object's reference count reaches 0.
Therefore, resource handlers remain open or in use longer than
necessary.
- The __del__ method isn't guaranteed to execute on system exit. See the
Python documentation:
https://docs.python.org/3/reference/datamodel.html#object.__del__
> It is not guaranteed that __del__() methods are called for objects
> that still exist when the interpreter exits.
- Exceptions that occur inside __del__ are ignored instead of raised.
This has the potential of hiding bugs. This is also in the Python
documentation:
> Warning: Due to the precarious circumstances under which __del__()
> methods are invoked, exceptions that occur during their execution
> are ignored, and a warning is printed to sys.stderr instead.
Instead, always close resource handlers when they are no longer in use.
This will close the file handler at a specified point in the user's code
and not wait until the interpreter chooses to. It is always guaranteed
to run. And, if an exception occurs while closing the file handler, the
bug will not be ignored.
Now, when code receives a ResourceWarning, it will highlight an area
that is mishandling resources. It should not simply be silenced, but
fixed by closing resources with a context manager.
All warnings that were emitted during tests have been cleaned up. To
enable warnings, I passed the `-Wa` CLI option to Python. This exposed
some mishandling of resources in ImageFile.__init__() and
SpiderImagePlugin.loadImageSeries(), they too were fixed.
2019-05-25 19:30:58 +03:00
|
|
|
with Image.open(out) as reread:
|
2020-02-23 00:03:01 +03:00
|
|
|
assert reread.info["version"] == version
|
2018-11-27 13:05:41 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
# Test that GIF87a is used by default
|
|
|
|
im = Image.new("L", (100, 100), "#000")
|
2022-04-10 22:17:35 +03:00
|
|
|
assert_version_after_save(im, b"GIF87a")
|
2018-11-27 13:05:41 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
# Test setting the version to 89a
|
|
|
|
im = Image.new("L", (100, 100), "#000")
|
|
|
|
im.info["version"] = b"89a"
|
2022-04-10 22:17:35 +03:00
|
|
|
assert_version_after_save(im, b"GIF89a")
|
2018-11-27 13:09:28 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
# Test that adding a GIF89a feature changes the version
|
|
|
|
im.info["transparency"] = 1
|
2022-04-10 22:17:35 +03:00
|
|
|
assert_version_after_save(im, b"GIF89a")
|
2015-08-21 15:10:13 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
# Test that a GIF87a image is also saved in that format
|
|
|
|
with Image.open("Tests/images/test.colors.gif") as im:
|
2022-04-10 22:17:35 +03:00
|
|
|
assert_version_after_save(im, b"GIF87a")
|
2017-09-01 13:36:51 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
# Test that a GIF89a image is also saved in that format
|
|
|
|
im.info["version"] = b"GIF89a"
|
2022-04-10 22:17:35 +03:00
|
|
|
assert_version_after_save(im, b"GIF87a")
|
2017-09-01 13:36:51 +03:00
|
|
|
|
2015-08-21 15:10:13 +03:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_append_images(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
2015-08-21 15:10:13 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
# Test appending single frame images
|
|
|
|
im = Image.new("RGB", (100, 100), "#f00")
|
|
|
|
ims = [Image.new("RGB", (100, 100), color) for color in ["#0f0", "#00f"]]
|
|
|
|
im.copy().save(out, save_all=True, append_images=ims)
|
2015-08-21 15:10:13 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open(out) as reread:
|
|
|
|
assert reread.n_frames == 3
|
2015-08-21 15:10:13 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
# Tests appending using a generator
|
2024-01-31 13:55:32 +03:00
|
|
|
def im_generator(ims: list[Image.Image]) -> Generator[Image.Image, None, None]:
|
2020-02-23 00:03:01 +03:00
|
|
|
yield from ims
|
2016-09-11 05:04:01 +03:00
|
|
|
|
2022-04-10 22:17:35 +03:00
|
|
|
im.save(out, save_all=True, append_images=im_generator(ims))
|
2017-11-04 02:46:15 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open(out) as reread:
|
|
|
|
assert reread.n_frames == 3
|
2017-11-04 02:46:15 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
# Tests appending single and multiple frame images
|
|
|
|
with Image.open("Tests/images/dispose_none.gif") as im:
|
|
|
|
with Image.open("Tests/images/dispose_prev.gif") as im2:
|
|
|
|
im.save(out, save_all=True, append_images=[im2])
|
2019-06-13 18:53:42 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open(out) as reread:
|
|
|
|
assert reread.n_frames == 10
|
2016-09-11 05:04:01 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_transparent_optimize(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
# From issue #2195, if the transparent color is incorrectly optimized out, GIF loses
|
|
|
|
# transparency.
|
2022-06-19 09:47:50 +03:00
|
|
|
# Need a palette that isn't using the 0 color,
|
|
|
|
# where the transparent color is actually the top palette entry to trigger the bug.
|
2016-09-11 05:04:01 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
data = bytes(range(1, 254))
|
|
|
|
palette = ImagePalette.ImagePalette("RGB", list(range(256)) * 3)
|
2015-08-21 15:10:13 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
im = Image.new("L", (253, 1))
|
|
|
|
im.frombytes(data)
|
|
|
|
im.putpalette(palette)
|
2016-12-27 14:30:47 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
2022-06-19 09:47:50 +03:00
|
|
|
im.save(out, transparency=im.getpixel((252, 0)))
|
2016-12-27 14:30:47 +03:00
|
|
|
|
2022-06-19 09:47:50 +03:00
|
|
|
with Image.open(out) as reloaded:
|
|
|
|
assert reloaded.info["transparency"] == reloaded.getpixel((252, 0))
|
2016-12-27 14:30:47 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_removed_transparency(tmp_path: Path) -> None:
|
2023-07-13 08:20:44 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
im = Image.new("RGB", (256, 1))
|
|
|
|
|
|
|
|
for x in range(256):
|
|
|
|
im.putpixel((x, 0), (x, 0, 0))
|
|
|
|
|
|
|
|
im.info["transparency"] = (255, 255, 255)
|
|
|
|
with pytest.warns(UserWarning):
|
|
|
|
im.save(out)
|
|
|
|
|
|
|
|
with Image.open(out) as reloaded:
|
|
|
|
assert "transparency" not in reloaded.info
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_rgb_transparency(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
2017-03-03 13:31:58 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
# Single frame
|
|
|
|
im = Image.new("RGB", (1, 1))
|
|
|
|
im.info["transparency"] = (255, 0, 0)
|
2021-06-27 08:09:39 +03:00
|
|
|
im.save(out)
|
2018-06-16 12:47:57 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open(out) as reloaded:
|
2021-06-27 08:09:39 +03:00
|
|
|
assert "transparency" in reloaded.info
|
2018-06-16 12:47:57 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
# Multiple frames
|
|
|
|
im = Image.new("RGB", (1, 1))
|
|
|
|
im.info["transparency"] = b""
|
|
|
|
ims = [Image.new("RGB", (1, 1))]
|
2023-02-23 16:30:38 +03:00
|
|
|
with pytest.warns(UserWarning):
|
|
|
|
im.save(out, save_all=True, append_images=ims)
|
2018-06-16 12:47:57 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open(out) as reloaded:
|
|
|
|
assert "transparency" not in reloaded.info
|
2018-06-16 12:47:57 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_rgba_transparency(tmp_path: Path) -> None:
|
2022-03-12 07:14:36 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
|
|
|
|
im = hopper("P")
|
|
|
|
im.save(out, save_all=True, append_images=[Image.new("RGBA", im.size)])
|
|
|
|
|
|
|
|
with Image.open(out) as reloaded:
|
|
|
|
reloaded.seek(1)
|
|
|
|
assert_image_equal(hopper("P").convert("RGB"), reloaded)
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_background_outside_palettte(tmp_path: Path) -> None:
|
2023-11-02 08:05:13 +03:00
|
|
|
with Image.open("Tests/images/background_outside_palette.gif") as im:
|
|
|
|
im.seek(1)
|
|
|
|
assert im.info["background"] == 255
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_bbox(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
2017-01-26 11:39:59 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
im = Image.new("RGB", (100, 100), "#fff")
|
|
|
|
ims = [Image.new("RGB", (100, 100), "#000")]
|
|
|
|
im.save(out, save_all=True, append_images=ims)
|
|
|
|
|
|
|
|
with Image.open(out) as reread:
|
|
|
|
assert reread.n_frames == 2
|
2017-01-26 11:39:59 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_bbox_alpha(tmp_path: Path) -> None:
|
2023-06-14 07:21:07 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
|
|
|
|
im = Image.new("RGBA", (1, 2), (255, 0, 0, 255))
|
|
|
|
im.putpixel((0, 1), (255, 0, 0, 0))
|
|
|
|
im2 = Image.new("RGBA", (1, 2), (255, 0, 0, 0))
|
|
|
|
im.save(out, save_all=True, append_images=[im2])
|
|
|
|
|
|
|
|
with Image.open(out) as reread:
|
|
|
|
assert reread.n_frames == 2
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_palette_save_L(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
# Generate an L mode image with a separate palette
|
2017-04-20 14:14:23 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
im = hopper("P")
|
|
|
|
im_l = Image.frombytes("L", im.size, im.tobytes())
|
|
|
|
palette = bytes(im.getpalette())
|
2017-03-03 19:25:01 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
im_l.save(out, palette=palette)
|
2017-03-03 19:25:01 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open(out) as reloaded:
|
|
|
|
assert_image_equal(reloaded.convert("RGB"), im.convert("RGB"))
|
2017-03-03 19:25:01 +03:00
|
|
|
|
2017-04-20 14:14:23 +03:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_palette_save_P(tmp_path: Path) -> None:
|
2023-07-17 16:04:43 +03:00
|
|
|
im = Image.new("P", (1, 2))
|
|
|
|
im.putpixel((0, 1), 1)
|
2017-03-07 01:46:55 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
2023-07-17 16:04:43 +03:00
|
|
|
im.save(out, palette=bytes((1, 2, 3, 4, 5, 6)))
|
2017-03-07 01:46:55 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open(out) as reloaded:
|
2023-07-17 16:04:43 +03:00
|
|
|
reloaded_rgb = reloaded.convert("RGB")
|
|
|
|
|
|
|
|
assert reloaded_rgb.getpixel((0, 0)) == (1, 2, 3)
|
|
|
|
assert reloaded_rgb.getpixel((0, 1)) == (4, 5, 6)
|
2017-03-07 01:46:55 +03:00
|
|
|
|
2017-03-03 19:25:01 +03:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_palette_save_duplicate_entries(tmp_path: Path) -> None:
|
2022-08-29 16:20:31 +03:00
|
|
|
im = Image.new("P", (1, 2))
|
|
|
|
im.putpixel((0, 1), 1)
|
|
|
|
|
|
|
|
im.putpalette((0, 0, 0, 0, 0, 0))
|
|
|
|
|
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
im.save(out, palette=[0, 0, 0, 0, 0, 0, 1, 1, 1])
|
|
|
|
|
|
|
|
with Image.open(out) as reloaded:
|
|
|
|
assert reloaded.convert("RGB").getpixel((0, 1)) == (0, 0, 0)
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_palette_save_all_P(tmp_path: Path) -> None:
|
2021-07-11 15:52:32 +03:00
|
|
|
frames = []
|
|
|
|
colors = ((255, 0, 0), (0, 255, 0))
|
|
|
|
for color in colors:
|
|
|
|
frame = Image.new("P", (100, 100))
|
|
|
|
frame.putpalette(color)
|
|
|
|
frames.append(frame)
|
|
|
|
|
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
frames[0].save(
|
|
|
|
out, save_all=True, palette=[255, 0, 0, 0, 255, 0], append_images=frames[1:]
|
|
|
|
)
|
|
|
|
|
|
|
|
with Image.open(out) as im:
|
|
|
|
# Assert that the frames are correct, and each frame has the same palette
|
|
|
|
assert_image_equal(im.convert("RGB"), frames[0].convert("RGB"))
|
|
|
|
assert im.palette.palette == im.global_palette.palette
|
|
|
|
|
|
|
|
im.seek(1)
|
|
|
|
assert_image_equal(im.convert("RGB"), frames[1].convert("RGB"))
|
|
|
|
assert im.palette.palette == im.global_palette.palette
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_palette_save_ImagePalette(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
# Pass in a different palette, as an ImagePalette.ImagePalette
|
|
|
|
# effectively the same as test_palette_save_P
|
2017-03-03 19:25:01 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
im = hopper("P")
|
|
|
|
palette = ImagePalette.ImagePalette("RGB", list(range(256))[::-1] * 3)
|
2017-03-03 19:25:01 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
im.save(out, palette=palette)
|
2016-12-27 14:30:47 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open(out) as reloaded:
|
|
|
|
im.putpalette(palette)
|
2021-07-09 17:00:50 +03:00
|
|
|
assert_image_equal(reloaded.convert("RGB"), im.convert("RGB"))
|
2017-03-03 19:38:30 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_save_I(tmp_path: Path) -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
# Test saving something that would trigger the auto-convert to 'L'
|
2017-03-03 19:38:30 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
im = hopper("I")
|
2017-03-03 19:38:30 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
im.save(out)
|
2017-04-20 14:14:23 +03:00
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open(out) as reloaded:
|
|
|
|
assert_image_equal(reloaded.convert("L"), im.convert("L"))
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_getdata() -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
# Test getheader/getdata against legacy values.
|
|
|
|
# Create a 'P' image with holes in the palette.
|
2022-01-15 01:02:31 +03:00
|
|
|
im = Image._wedge().resize((16, 16), Image.Resampling.NEAREST)
|
2020-02-23 00:03:01 +03:00
|
|
|
im.putpalette(ImagePalette.ImagePalette("RGB"))
|
|
|
|
im.info = {"background": 0}
|
|
|
|
|
2021-10-15 13:10:22 +03:00
|
|
|
passed_palette = bytes(255 - i // 3 for i in range(768))
|
2020-02-23 00:03:01 +03:00
|
|
|
|
|
|
|
GifImagePlugin._FORCE_OPTIMIZE = True
|
|
|
|
try:
|
|
|
|
h = GifImagePlugin.getheader(im, passed_palette)
|
|
|
|
d = GifImagePlugin.getdata(im)
|
|
|
|
|
|
|
|
import pickle
|
|
|
|
|
|
|
|
# Enable to get target values on pre-refactor version
|
|
|
|
# with open('Tests/images/gif_header_data.pkl', 'wb') as f:
|
|
|
|
# pickle.dump((h, d), f, 1)
|
|
|
|
with open("Tests/images/gif_header_data.pkl", "rb") as f:
|
|
|
|
(h_target, d_target) = pickle.load(f)
|
|
|
|
|
|
|
|
assert h == h_target
|
|
|
|
assert d == d_target
|
|
|
|
finally:
|
|
|
|
GifImagePlugin._FORCE_OPTIMIZE = False
|
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_lzw_bits() -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
# see https://github.com/python-pillow/Pillow/issues/2811
|
|
|
|
with Image.open("Tests/images/issue_2811.gif") as im:
|
|
|
|
assert im.tile[0][3][0] == 11 # LZW bits
|
|
|
|
# codec error prepatch
|
|
|
|
im.load()
|
2017-03-07 12:52:31 +03:00
|
|
|
|
2019-05-02 12:46:17 +03:00
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_extents() -> None:
|
2020-02-23 00:03:01 +03:00
|
|
|
with Image.open("Tests/images/test_extents.gif") as im:
|
|
|
|
assert im.size == (100, 100)
|
2022-03-21 15:19:26 +03:00
|
|
|
|
|
|
|
# Check that n_frames does not change the size
|
|
|
|
assert im.n_frames == 2
|
|
|
|
assert im.size == (100, 100)
|
|
|
|
|
2020-02-23 00:03:01 +03:00
|
|
|
im.seek(1)
|
|
|
|
assert im.size == (150, 150)
|
2021-04-08 01:04:20 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_missing_background() -> None:
|
2021-04-08 01:04:20 +03:00
|
|
|
# The Global Color Table Flag isn't set, so there is no background color index,
|
|
|
|
# but the disposal method is "Restore to background color"
|
|
|
|
with Image.open("Tests/images/missing_background.gif") as im:
|
|
|
|
im.seek(1)
|
2021-11-29 09:49:06 +03:00
|
|
|
assert_image_equal_tofile(im, "Tests/images/missing_background_first_frame.png")
|
2021-12-06 22:37:01 +03:00
|
|
|
|
|
|
|
|
2024-01-31 12:12:58 +03:00
|
|
|
def test_saving_rgba(tmp_path: Path) -> None:
|
2021-12-06 22:37:01 +03:00
|
|
|
out = str(tmp_path / "temp.gif")
|
|
|
|
with Image.open("Tests/images/transparent.png") as im:
|
|
|
|
im.save(out)
|
|
|
|
|
|
|
|
with Image.open(out) as reloaded:
|
|
|
|
reloaded_rgba = reloaded.convert("RGBA")
|
|
|
|
assert reloaded_rgba.load()[0, 0][3] == 0
|