Pillow/Tests/test_imagechops.py

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

417 lines
11 KiB
Python
Raw Permalink Normal View History

from __future__ import annotations
2024-01-20 14:23:03 +03:00
2024-02-17 07:00:38 +03:00
from typing import Callable
from PIL import Image, ImageChops
2020-02-12 19:29:19 +03:00
from .helper import assert_image_equal, hopper
2018-07-04 11:55:58 +03:00
BLACK = (0, 0, 0)
BROWN = (127, 64, 0)
CYAN = (0, 255, 255)
DARK_GREEN = (0, 128, 0)
GREEN = (0, 255, 0)
ORANGE = (255, 128, 0)
WHITE = (255, 255, 255)
2023-10-19 11:12:01 +03:00
GRAY = 128
2018-07-04 11:55:58 +03:00
2014-06-10 13:10:47 +04:00
def test_sanity() -> None:
2020-02-12 19:29:19 +03:00
im = hopper("L")
2014-06-10 13:10:47 +04:00
2020-02-12 19:29:19 +03:00
ImageChops.constant(im, 128)
ImageChops.duplicate(im)
ImageChops.invert(im)
ImageChops.lighter(im, im)
ImageChops.darker(im, im)
ImageChops.difference(im, im)
ImageChops.multiply(im, im)
ImageChops.screen(im, im)
2014-06-10 13:10:47 +04:00
2020-02-12 19:29:19 +03:00
ImageChops.add(im, im)
ImageChops.add(im, im, 2.0)
ImageChops.add(im, im, 2.0, 128)
ImageChops.subtract(im, im)
ImageChops.subtract(im, im, 2.0)
ImageChops.subtract(im, im, 2.0, 128)
2014-06-10 13:10:47 +04:00
2020-02-12 19:29:19 +03:00
ImageChops.add_modulo(im, im)
ImageChops.subtract_modulo(im, im)
2014-06-10 13:10:47 +04:00
2020-02-12 19:29:19 +03:00
ImageChops.blend(im, im, 0.5)
ImageChops.composite(im, im, im)
2014-06-10 13:10:47 +04:00
2019-12-25 22:23:32 +03:00
ImageChops.soft_light(im, im)
ImageChops.hard_light(im, im)
2019-11-22 16:30:43 +03:00
ImageChops.overlay(im, im)
2019-11-22 16:03:59 +03:00
2020-02-12 19:29:19 +03:00
ImageChops.offset(im, 10)
ImageChops.offset(im, 10, 20)
2014-06-10 13:10:47 +04:00
def test_add() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
with Image.open("Tests/images/imagedraw_ellipse_RGB.png") as im1:
with Image.open("Tests/images/imagedraw_floodfill_RGB.png") as im2:
# Act
new = ImageChops.add(im1, im2)
2018-07-04 11:55:58 +03:00
2020-02-12 19:29:19 +03:00
# Assert
assert new.getbbox() == (25, 25, 76, 76)
assert new.getpixel((50, 50)) == ORANGE
2018-07-04 11:55:58 +03:00
2018-08-25 04:59:27 +03:00
def test_add_scale_offset() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
with Image.open("Tests/images/imagedraw_ellipse_RGB.png") as im1:
with Image.open("Tests/images/imagedraw_floodfill_RGB.png") as im2:
# Act
new = ImageChops.add(im1, im2, scale=2.5, offset=100)
2018-08-25 04:59:27 +03:00
2020-02-12 19:29:19 +03:00
# Assert
assert new.getbbox() == (0, 0, 100, 100)
assert new.getpixel((50, 50)) == (202, 151, 100)
2018-08-25 04:59:27 +03:00
def test_add_clip() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
im = hopper()
2018-08-25 04:59:27 +03:00
2020-02-12 19:29:19 +03:00
# Act
new = ImageChops.add(im, im)
2018-07-04 11:55:58 +03:00
2020-02-12 19:29:19 +03:00
# Assert
assert new.getpixel((50, 50)) == (255, 255, 254)
2018-07-04 11:55:58 +03:00
def test_add_modulo() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
with Image.open("Tests/images/imagedraw_ellipse_RGB.png") as im1:
with Image.open("Tests/images/imagedraw_floodfill_RGB.png") as im2:
# Act
new = ImageChops.add_modulo(im1, im2)
2018-08-25 04:59:27 +03:00
2020-02-12 19:29:19 +03:00
# Assert
assert new.getbbox() == (25, 25, 76, 76)
assert new.getpixel((50, 50)) == ORANGE
2018-08-25 04:59:27 +03:00
2018-07-04 11:55:58 +03:00
def test_add_modulo_no_clip() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
im = hopper()
2018-07-04 11:55:58 +03:00
2020-02-12 19:29:19 +03:00
# Act
new = ImageChops.add_modulo(im, im)
2018-07-04 11:55:58 +03:00
2020-02-12 19:29:19 +03:00
# Assert
assert new.getpixel((50, 50)) == (224, 76, 254)
2018-07-04 11:55:58 +03:00
def test_blend() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
with Image.open("Tests/images/imagedraw_ellipse_RGB.png") as im1:
with Image.open("Tests/images/imagedraw_floodfill_RGB.png") as im2:
# Act
new = ImageChops.blend(im1, im2, 0.5)
# Assert
assert new.getbbox() == (25, 25, 76, 76)
assert new.getpixel((50, 50)) == BROWN
2018-07-04 11:55:58 +03:00
def test_constant() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
im = Image.new("RGB", (20, 10))
2018-07-04 11:55:58 +03:00
2020-02-12 19:29:19 +03:00
# Act
2023-10-19 11:12:01 +03:00
new = ImageChops.constant(im, GRAY)
2018-07-04 11:55:58 +03:00
2020-02-12 19:29:19 +03:00
# Assert
assert new.size == im.size
2023-10-19 11:12:01 +03:00
assert new.getpixel((0, 0)) == GRAY
assert new.getpixel((19, 9)) == GRAY
2020-02-12 19:29:19 +03:00
def test_darker_image() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
with Image.open("Tests/images/imagedraw_chord_RGB.png") as im1:
with Image.open("Tests/images/imagedraw_outline_chord_RGB.png") as im2:
2019-11-25 23:03:23 +03:00
# Act
new = ImageChops.darker(im1, im2)
2018-08-25 04:59:27 +03:00
2020-02-12 19:29:19 +03:00
# Assert
assert_image_equal(new, im2)
2018-08-25 04:59:27 +03:00
2018-07-04 11:55:58 +03:00
def test_darker_pixel() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
im1 = hopper()
with Image.open("Tests/images/imagedraw_chord_RGB.png") as im2:
# Act
new = ImageChops.darker(im1, im2)
2018-07-04 11:55:58 +03:00
2020-02-12 19:29:19 +03:00
# Assert
assert new.getpixel((50, 50)) == (240, 166, 0)
def test_difference() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
with Image.open("Tests/images/imagedraw_arc_end_le_start.png") as im1:
with Image.open("Tests/images/imagedraw_arc_no_loops.png") as im2:
2019-11-25 23:03:23 +03:00
# Act
new = ImageChops.difference(im1, im2)
2018-08-25 04:59:27 +03:00
2020-02-12 19:29:19 +03:00
# Assert
assert new.getbbox() == (25, 25, 76, 76)
2018-08-25 04:59:27 +03:00
def test_difference_pixel() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
im1 = hopper()
with Image.open("Tests/images/imagedraw_polygon_kite_RGB.png") as im2:
2018-07-04 11:55:58 +03:00
# Act
2020-02-12 19:29:19 +03:00
new = ImageChops.difference(im1, im2)
2018-07-04 11:55:58 +03:00
2020-02-12 19:29:19 +03:00
# Assert
assert new.getpixel((50, 50)) == (240, 166, 128)
2018-07-04 11:55:58 +03:00
def test_duplicate() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
im = hopper()
# Act
new = ImageChops.duplicate(im)
2018-07-04 11:55:58 +03:00
2020-02-12 19:29:19 +03:00
# Assert
assert_image_equal(new, im)
2018-07-04 11:55:58 +03:00
def test_invert() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
with Image.open("Tests/images/imagedraw_floodfill_RGB.png") as im:
# Act
new = ImageChops.invert(im)
2018-08-25 04:59:27 +03:00
2020-02-12 19:29:19 +03:00
# Assert
assert new.getbbox() == (0, 0, 100, 100)
assert new.getpixel((0, 0)) == WHITE
assert new.getpixel((50, 50)) == CYAN
def test_lighter_image() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
with Image.open("Tests/images/imagedraw_chord_RGB.png") as im1:
with Image.open("Tests/images/imagedraw_outline_chord_RGB.png") as im2:
2019-11-25 23:03:23 +03:00
# Act
new = ImageChops.lighter(im1, im2)
2018-08-25 04:59:27 +03:00
2020-02-19 11:52:07 +03:00
# Assert
assert_image_equal(new, im1)
2020-02-12 19:29:19 +03:00
2018-07-04 11:55:58 +03:00
def test_lighter_pixel() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
im1 = hopper()
with Image.open("Tests/images/imagedraw_chord_RGB.png") as im2:
2018-07-04 11:55:58 +03:00
# Act
2020-02-12 19:29:19 +03:00
new = ImageChops.lighter(im1, im2)
2018-07-04 11:55:58 +03:00
2020-02-12 19:29:19 +03:00
# Assert
assert new.getpixel((50, 50)) == (255, 255, 127)
2018-07-04 11:55:58 +03:00
def test_multiply_black() -> None:
2020-02-12 19:29:19 +03:00
"""If you multiply an image with a solid black image,
the result is black."""
# Arrange
im1 = hopper()
black = Image.new("RGB", im1.size, "black")
# Act
new = ImageChops.multiply(im1, black)
2018-07-04 11:55:58 +03:00
2020-02-12 19:29:19 +03:00
# Assert
assert_image_equal(new, black)
2018-07-04 11:55:58 +03:00
2020-02-12 19:29:19 +03:00
def test_multiply_green() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
with Image.open("Tests/images/imagedraw_floodfill_RGB.png") as im:
green = Image.new("RGB", im.size, "green")
2018-07-04 11:55:58 +03:00
# Act
2020-02-12 19:29:19 +03:00
new = ImageChops.multiply(im, green)
2018-07-04 11:55:58 +03:00
2020-02-12 19:29:19 +03:00
# Assert
assert new.getbbox() == (25, 25, 76, 76)
assert new.getpixel((25, 25)) == DARK_GREEN
assert new.getpixel((50, 50)) == BLACK
2018-07-04 11:55:58 +03:00
def test_multiply_white() -> None:
2020-02-12 19:29:19 +03:00
"""If you multiply with a solid white image, the image is unaffected."""
# Arrange
im1 = hopper()
white = Image.new("RGB", im1.size, "white")
# Act
new = ImageChops.multiply(im1, white)
2018-07-04 11:55:58 +03:00
2020-02-12 19:29:19 +03:00
# Assert
assert_image_equal(new, im1)
2018-07-04 11:55:58 +03:00
2018-08-25 04:59:27 +03:00
def test_offset() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
xoffset = 45
yoffset = 20
with Image.open("Tests/images/imagedraw_ellipse_RGB.png") as im:
# Act
new = ImageChops.offset(im, xoffset, yoffset)
2018-07-04 11:55:58 +03:00
2020-02-19 11:52:07 +03:00
# Assert
assert new.getbbox() == (0, 45, 100, 96)
assert new.getpixel((50, 50)) == BLACK
assert new.getpixel((50 + xoffset, 50 + yoffset)) == DARK_GREEN
2018-07-04 11:55:58 +03:00
2020-02-19 11:52:07 +03:00
# Test no yoffset
assert ImageChops.offset(im, xoffset) == ImageChops.offset(im, xoffset, xoffset)
2018-07-04 11:55:58 +03:00
def test_screen() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
with Image.open("Tests/images/imagedraw_ellipse_RGB.png") as im1:
with Image.open("Tests/images/imagedraw_floodfill_RGB.png") as im2:
# Act
new = ImageChops.screen(im1, im2)
2018-08-25 04:59:27 +03:00
2020-02-12 19:29:19 +03:00
# Assert
assert new.getbbox() == (25, 25, 76, 76)
assert new.getpixel((50, 50)) == ORANGE
2018-08-25 04:59:27 +03:00
def test_subtract() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
with Image.open("Tests/images/imagedraw_chord_RGB.png") as im1:
with Image.open("Tests/images/imagedraw_outline_chord_RGB.png") as im2:
2019-11-25 23:03:23 +03:00
# Act
new = ImageChops.subtract(im1, im2)
2018-08-25 04:59:27 +03:00
2020-02-12 19:29:19 +03:00
# Assert
assert new.getbbox() == (25, 50, 76, 76)
2020-06-29 23:16:06 +03:00
assert new.getpixel((50, 51)) == GREEN
assert new.getpixel((50, 52)) == BLACK
2020-02-12 19:29:19 +03:00
2018-08-25 04:59:27 +03:00
def test_subtract_scale_offset() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
with Image.open("Tests/images/imagedraw_chord_RGB.png") as im1:
with Image.open("Tests/images/imagedraw_outline_chord_RGB.png") as im2:
# Act
new = ImageChops.subtract(im1, im2, scale=2.5, offset=100)
2020-02-19 11:52:07 +03:00
# Assert
assert new.getbbox() == (0, 0, 100, 100)
assert new.getpixel((50, 50)) == (100, 202, 100)
2020-02-12 19:29:19 +03:00
def test_subtract_clip() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
im1 = hopper()
with Image.open("Tests/images/imagedraw_chord_RGB.png") as im2:
# Act
new = ImageChops.subtract(im1, im2)
2018-07-04 11:55:58 +03:00
2020-02-19 11:52:07 +03:00
# Assert
assert new.getpixel((50, 50)) == (0, 0, 127)
2018-07-04 11:55:58 +03:00
2020-02-12 19:29:19 +03:00
def test_subtract_modulo() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
with Image.open("Tests/images/imagedraw_chord_RGB.png") as im1:
with Image.open("Tests/images/imagedraw_outline_chord_RGB.png") as im2:
2019-11-25 23:03:23 +03:00
# Act
new = ImageChops.subtract_modulo(im1, im2)
2018-08-25 04:59:27 +03:00
2020-02-19 11:52:07 +03:00
# Assert
assert new.getbbox() == (25, 50, 76, 76)
2020-06-29 23:16:06 +03:00
assert new.getpixel((50, 51)) == GREEN
assert new.getpixel((50, 52)) == BLACK
2020-02-12 19:29:19 +03:00
def test_subtract_modulo_no_clip() -> None:
2020-02-12 19:29:19 +03:00
# Arrange
im1 = hopper()
with Image.open("Tests/images/imagedraw_chord_RGB.png") as im2:
# Act
new = ImageChops.subtract_modulo(im1, im2)
2020-02-19 11:52:07 +03:00
# Assert
assert new.getpixel((50, 50)) == (241, 167, 127)
2019-11-22 16:03:59 +03:00
def test_soft_light() -> None:
2019-11-22 16:03:59 +03:00
# Arrange
2021-11-25 15:16:07 +03:00
with Image.open("Tests/images/hopper.png") as im1:
with Image.open("Tests/images/hopper-XYZ.png") as im2:
# Act
new = ImageChops.soft_light(im1, im2)
2019-11-22 16:03:59 +03:00
# Assert
2020-02-19 11:52:07 +03:00
assert new.getpixel((64, 64)) == (163, 54, 32)
assert new.getpixel((15, 100)) == (1, 1, 3)
2019-11-22 16:03:59 +03:00
def test_hard_light() -> None:
2019-11-22 16:03:59 +03:00
# Arrange
2021-11-25 15:16:07 +03:00
with Image.open("Tests/images/hopper.png") as im1:
with Image.open("Tests/images/hopper-XYZ.png") as im2:
# Act
new = ImageChops.hard_light(im1, im2)
2019-11-22 16:03:59 +03:00
2020-02-12 19:29:19 +03:00
# Assert
2020-02-19 11:52:07 +03:00
assert new.getpixel((64, 64)) == (144, 50, 27)
assert new.getpixel((15, 100)) == (1, 1, 2)
2020-02-12 19:29:19 +03:00
def test_overlay() -> None:
2019-11-22 16:30:43 +03:00
# Arrange
2021-11-25 15:16:07 +03:00
with Image.open("Tests/images/hopper.png") as im1:
with Image.open("Tests/images/hopper-XYZ.png") as im2:
# Act
new = ImageChops.overlay(im1, im2)
2019-11-22 16:30:43 +03:00
# Assert
2020-02-19 11:52:07 +03:00
assert new.getpixel((64, 64)) == (159, 50, 27)
assert new.getpixel((15, 100)) == (1, 1, 2)
2019-11-22 16:30:43 +03:00
def test_logical() -> None:
2024-02-17 07:00:38 +03:00
def table(
op: Callable[[Image.Image, Image.Image], Image.Image], a: int, b: int
) -> list[float]:
2020-02-12 19:29:19 +03:00
out = []
for x in (a, b):
imx = Image.new("1", (1, 1), x)
for y in (a, b):
imy = Image.new("1", (1, 1), y)
value = op(imx, imy).getpixel((0, 0))
2024-08-16 14:52:56 +03:00
assert not isinstance(value, tuple)
assert value is not None
out.append(value)
return out
assert table(ImageChops.logical_and, 0, 1) == [0, 0, 0, 255]
assert table(ImageChops.logical_or, 0, 1) == [0, 255, 255, 255]
assert table(ImageChops.logical_xor, 0, 1) == [0, 255, 255, 0]
assert table(ImageChops.logical_and, 0, 128) == [0, 0, 0, 255]
assert table(ImageChops.logical_or, 0, 128) == [0, 255, 255, 255]
assert table(ImageChops.logical_xor, 0, 128) == [0, 255, 255, 0]
assert table(ImageChops.logical_and, 0, 255) == [0, 0, 0, 255]
assert table(ImageChops.logical_or, 0, 255) == [0, 255, 255, 255]
assert table(ImageChops.logical_xor, 0, 255) == [0, 255, 255, 0]