added animated cursor tests

This commit is contained in:
jlwoolf 2022-09-27 12:16:52 -06:00
parent 551e2a6afc
commit 5eb3e53502
8 changed files with 145 additions and 10 deletions

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 344 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 331 B

132
Tests/test_file_ani.py Normal file
View File

@ -0,0 +1,132 @@
from io import BytesIO
import pytest
from PIL import Image
def test_aero_busy():
with Image.open("Tests/images/ani/aero_busy.ani") as im:
assert im.size == (64, 64)
assert im.info["frames"] == 18
with Image.open("Tests/images/ani/aero_busy_0.png") as png:
assert png.tobytes() == im.tobytes()
im.seek(8)
with Image.open("Tests/images/ani/aero_busy_8.png") as png:
assert png.tobytes() == im.tobytes()
with pytest.raises(EOFError):
im.seek(-1)
with pytest.raises(EOFError):
im.seek(18)
def test_posy_busy():
with Image.open("Tests/images/ani/posy_busy.ani") as im:
assert im.size == (96, 96)
assert im.info["frames"] == 77
with Image.open("Tests/images/ani/posy_busy_0.png") as png:
assert png.tobytes() == im.tobytes()
im.seek(24)
with Image.open("Tests/images/ani/posy_busy_24.png") as png:
assert png.tobytes() == im.tobytes()
with pytest.raises(EOFError):
im.seek(77)
def test_stopwtch():
with Image.open("Tests/images/ani/stopwtch.ani") as im:
assert im.size == (32, 32)
assert im.info["frames"] == 8
assert im.info["seq"][0] == 0
assert im.info["seq"][2] == 0
for i, r in enumerate(im.info["rate"]):
if i == 1 or i == 2:
assert r == 16
else:
assert r == 8
with Image.open("Tests/images/ani/stopwtch_0.png") as png:
assert png.tobytes() == im.tobytes()
im.seek(5)
with Image.open("Tests/images/ani/stopwtch_5.png") as png:
assert png.tobytes() == im.tobytes()
with pytest.raises(EOFError):
im.seek(8)
def test_save():
directory_path = "Tests/images/ani/"
filenames = ["aero_busy_0.png", "aero_busy_8.png",
"posy_busy_0.png", "posy_busy_24.png",
"stopwtch_0.png", "stopwtch_5.png"]
images = [Image.open(directory_path + filename) for filename in filenames]
with BytesIO() as output:
images[0].save(output, append_images=[images[1]], seq=[
0, 1], rate=[5, 10], format="ANI")
with Image.open(output, formats=["ANI"]) as im:
assert im.tobytes() == images[0].tobytes()
im.seek(1)
assert im.tobytes() == images[1].tobytes()
assert im.info['seq'] == [0, 1]
assert im.info['rate'] == [5, 10]
with BytesIO() as output:
images[2].save(output, append_images=[images[3]], seq=[
1, 0], rate=[2, 2], format="ANI", sizes=[(96, 96)])
with Image.open(output, formats=["ANI"]) as im:
assert im.tobytes() == images[2].tobytes()
im.seek(1)
assert im.tobytes() == images[3].tobytes()
assert im.info['seq'] == [1, 0]
assert im.info['rate'] == [2, 2]
with BytesIO() as output:
images[4].save(output, append_images=[images[5]], seq=[
0, 1], rate=[3, 4], format="ANI")
with Image.open(output, formats=["ANI"]) as im:
assert im.tobytes() == images[4].tobytes()
im.seek(1)
assert im.tobytes() == images[5].tobytes()
assert im.info['seq'] == [0, 1]
assert im.info['rate'] == [3, 4]
with BytesIO() as output:
images[0].save(output, append_images=images[1:],
seq=[0, 2, 4, 1, 3, 5, 0, 1, 0, 1],
rate=[1, 2, 3, 1, 2, 3, 1, 2, 3, 4],
format="ANI", sizes=[(32, 32)])
with Image.open(output, formats=["ANI"]) as im:
assert im.info['frames'] == 6
assert im.info['seq'] == [0, 2, 4, 1, 3, 5, 0, 1, 0, 1]
assert im.info['rate'] == [1, 2, 3, 1, 2, 3, 1, 2, 3, 4]
assert im.size == (32, 32)
im.seek(4)
assert im.tobytes() == images[4].tobytes()
with BytesIO() as output:
with pytest.raises(ValueError):
images[0].save(output, append_images=images[1:], seq=[
0, 1, 8, 1, 2], rate=[1, 1, 1, 1, 1], format="ANI", sizes=[(32, 32)])
with pytest.raises(ValueError):
images[0].save(output, append_images=images[1:], seq=[
0, 1, 1, 1, 2], rate=[1, 1, 1, 1], format="ANI", sizes=[(32, 32)])
with pytest.raises(ValueError):
images[0].save(output, append_images=images[1:], rate=[1, 1, 1, 1], format="ANI", sizes=[(32, 32)])

View File

@ -15,22 +15,21 @@ def _accept(s):
def _save_frame(im: Image.Image, fp: BytesIO, filename: str, info: dict):
fp.write(b"\0\0\2\0")
bmp = True
s = im.encoderinfo.get(
s = info.get(
"sizes",
[(16, 16), (24, 24), (32, 32), (48, 48), (64, 64), (128, 128), (256, 256)],
)
h = im.encoderinfo.get("hotspots", [(0, 0) for i in range(len(s))])
h = info.get("hotspots", [(0, 0) for _ in range(len(s))])
if len(h) != len(s):
raise ValueError("Number of hotspots must be equal to number of cursor sizes")
# sort and remove duplicate sizes
sizes = []
hotspots = []
for size, hotspot in zip(*sorted(zip(s, h), lambda x: x[0])):
sizes, hotspots = [], []
for size, hotspot in sorted(zip(s, h), key=lambda x: x[0]):
if size not in sizes:
sizes.append(size)
hotspots.append(hotspots)
hotspots.append(hotspot)
frames = []
width, height = im.size
@ -72,6 +71,7 @@ def _save_frame(im: Image.Image, fp: BytesIO, filename: str, info: dict):
image_bytes = image_io.read()
if bmp:
image_bytes = image_bytes[:8] + o32(height * 2) + image_bytes[12:]
bytes_len = len(image_bytes)
fp.write(o32(bytes_len)) # dwBytesInRes(4)
fp.write(o32(offset)) # dwImageOffset(4)
@ -121,7 +121,8 @@ def _write_multiple_frames(im: Image.Image, fp: BytesIO, filename: str):
list_offset = fp.tell()
fp.write(b"fram")
frames = [im] + im.encoderinfo.get("append_images", [])
frames = [im]
frames.extend(im.encoderinfo.get("append_images", []))
for frame in frames:
fp.write(b"icon" + o32(0))
icon_offset = fp.tell()
@ -342,7 +343,7 @@ class AniFile:
def frame(self, frame):
if frame > self.anih["nFrames"]:
raise ValueError("Frame index out of animation bounds")
raise EOFError("Frame index out of animation bounds")
offset, size = self.image_data[frame].values()
self.buf.seek(offset)
@ -392,6 +393,7 @@ class AniImageFile(ImageFile.ImageFile):
self.info["frames"] = self.ani.anih["nFrames"]
self.frame = 0
self._min_frame = 0
self.seek(0)
self.size = self.im.size
@ -413,8 +415,9 @@ class AniImageFile(ImageFile.ImageFile):
self.mode = im.mode
def seek(self, frame):
if frame > self.info["frames"]:
raise ValueError("Frame index out of animation bounds")
if frame > self.info["frames"] - 1 or frame < 0:
raise EOFError("Frame index out of animation bounds")
self.frame = frame
self.load()