mirror of
https://github.com/python-pillow/Pillow.git
synced 2025-07-28 08:59:57 +03:00
Merge pull request #2 from radarhere/qoi_write
Removed qoi_ prefix from save argument
This commit is contained in:
commit
5acf16b00b
|
@ -41,13 +41,13 @@ def test_op_index() -> None:
|
||||||
def test_save(tmp_path: Path) -> None:
|
def test_save(tmp_path: Path) -> None:
|
||||||
f = tmp_path / "temp.qoi"
|
f = tmp_path / "temp.qoi"
|
||||||
|
|
||||||
im = hopper("RGB")
|
im = hopper()
|
||||||
im.save(f, qoi_colorspace="sRGB")
|
im.save(f, colorspace="sRGB")
|
||||||
|
|
||||||
assert_image_equal_tofile(im, f)
|
assert_image_equal_tofile(im, f)
|
||||||
|
|
||||||
for image in ["Tests/images/default_font.png", "Tests/images/pil123rgba.png"]:
|
for path in ("Tests/images/default_font.png", "Tests/images/pil123rgba.png"):
|
||||||
with Image.open(image) as im:
|
with Image.open(path) as im:
|
||||||
im.save(f)
|
im.save(f)
|
||||||
|
|
||||||
assert_image_equal_tofile(im, f)
|
assert_image_equal_tofile(im, f)
|
||||||
|
|
|
@ -1082,6 +1082,26 @@ Pillow reads and writes PBM, PGM, PPM and PNM files containing ``1``, ``L``, ``I
|
||||||
|
|
||||||
Since Pillow 9.2.0, "plain" (P1 to P3) formats can be read as well.
|
Since Pillow 9.2.0, "plain" (P1 to P3) formats can be read as well.
|
||||||
|
|
||||||
|
QOI
|
||||||
|
^^^
|
||||||
|
|
||||||
|
.. versionadded:: 9.5.0
|
||||||
|
|
||||||
|
Pillow reads and writes images in Quite OK Image format using a Python decoder. If you
|
||||||
|
wish to write code specifically for this format, :pypi:`qoi` is an alternative library
|
||||||
|
that uses C to decode the image and interfaces with NumPy.
|
||||||
|
|
||||||
|
.. _qoi-saving:
|
||||||
|
|
||||||
|
Saving
|
||||||
|
~~~~~~
|
||||||
|
|
||||||
|
The :py:meth:`~PIL.Image.Image.save` method can take the following keyword arguments:
|
||||||
|
|
||||||
|
**colorspace**
|
||||||
|
If set to "sRGB", the colorspace will be written as sRGB with linear alpha, instead
|
||||||
|
of all channels being linear.
|
||||||
|
|
||||||
SGI
|
SGI
|
||||||
^^^
|
^^^
|
||||||
|
|
||||||
|
@ -1578,15 +1598,6 @@ PSD
|
||||||
|
|
||||||
Pillow identifies and reads PSD files written by Adobe Photoshop 2.5 and 3.0.
|
Pillow identifies and reads PSD files written by Adobe Photoshop 2.5 and 3.0.
|
||||||
|
|
||||||
QOI
|
|
||||||
^^^
|
|
||||||
|
|
||||||
.. versionadded:: 9.5.0
|
|
||||||
|
|
||||||
Pillow reads images in Quite OK Image format using a Python decoder. If you wish to
|
|
||||||
write code specifically for this format, :pypi:`qoi` is an alternative library that
|
|
||||||
uses C to decode the image and interfaces with NumPy.
|
|
||||||
|
|
||||||
SUN
|
SUN
|
||||||
^^^
|
^^^
|
||||||
|
|
||||||
|
|
|
@ -122,10 +122,7 @@ def _save(im: Image.Image, fp: IO[bytes], filename: str | bytes) -> None:
|
||||||
msg = "Unsupported QOI image mode"
|
msg = "Unsupported QOI image mode"
|
||||||
raise ValueError(msg)
|
raise ValueError(msg)
|
||||||
|
|
||||||
if im.encoderinfo.get("qoi_colorspace") == "sRGB":
|
colorspace = 0 if im.encoderinfo.get("colorspace") == "sRGB" else 1
|
||||||
colorspace = 0
|
|
||||||
else:
|
|
||||||
colorspace = 1
|
|
||||||
|
|
||||||
fp.write(b"qoif")
|
fp.write(b"qoif")
|
||||||
fp.write(o32(im.size[0]))
|
fp.write(o32(im.size[0]))
|
||||||
|
@ -140,14 +137,17 @@ class QoiEncoder(ImageFile.PyEncoder):
|
||||||
_pushes_fd = True
|
_pushes_fd = True
|
||||||
_previous_pixel: tuple[int, int, int, int] | None = None
|
_previous_pixel: tuple[int, int, int, int] | None = None
|
||||||
_previously_seen_pixels: dict[int, tuple[int, int, int, int]] = {}
|
_previously_seen_pixels: dict[int, tuple[int, int, int, int]] = {}
|
||||||
|
_run = 0
|
||||||
|
|
||||||
def _write_run(self, run: int) -> bytes:
|
def _write_run(self) -> bytes:
|
||||||
return o8(0xC0 | (run - 1)) # QOI_OP_RUN
|
data = o8(0b11000000 | (self._run - 1)) # QOI_OP_RUN
|
||||||
|
self._run = 0
|
||||||
|
return data
|
||||||
|
|
||||||
def _delta(self, left: int, right: int) -> int:
|
def _delta(self, left: int, right: int) -> int:
|
||||||
result = (left - right) & 0xFF
|
result = (left - right) & 255
|
||||||
if result >= 0x80:
|
if result >= 128:
|
||||||
result -= 0x100
|
result -= 256
|
||||||
return result
|
return result
|
||||||
|
|
||||||
def encode(self, bufsize: int) -> tuple[int, int, bytes]:
|
def encode(self, bufsize: int) -> tuple[int, int, bytes]:
|
||||||
|
@ -158,8 +158,7 @@ class QoiEncoder(ImageFile.PyEncoder):
|
||||||
|
|
||||||
data = bytearray()
|
data = bytearray()
|
||||||
w, h = self.im.size
|
w, h = self.im.size
|
||||||
run = 0
|
bands = Image.getmodebands(self.mode)
|
||||||
bands = Image.getmodebands(self.im.mode)
|
|
||||||
|
|
||||||
for y in range(h):
|
for y in range(h):
|
||||||
for x in range(w):
|
for x in range(w):
|
||||||
|
@ -168,14 +167,12 @@ class QoiEncoder(ImageFile.PyEncoder):
|
||||||
pixel = (*pixel, 255)
|
pixel = (*pixel, 255)
|
||||||
|
|
||||||
if pixel == self._previous_pixel:
|
if pixel == self._previous_pixel:
|
||||||
run += 1
|
self._run += 1
|
||||||
if run == 62:
|
if self._run == 62:
|
||||||
data += self._write_run(run)
|
data += self._write_run()
|
||||||
run = 0
|
|
||||||
else:
|
else:
|
||||||
if run > 0:
|
if self._run:
|
||||||
data += self._write_run(run)
|
data += self._write_run()
|
||||||
run = 0
|
|
||||||
|
|
||||||
r, g, b, a = pixel
|
r, g, b, a = pixel
|
||||||
hash_value = (r * 3 + g * 5 + b * 7 + a * 11) % 64
|
hash_value = (r * 3 + g * 5 + b * 7 + a * 11) % 64
|
||||||
|
@ -184,32 +181,46 @@ class QoiEncoder(ImageFile.PyEncoder):
|
||||||
elif self._previous_pixel:
|
elif self._previous_pixel:
|
||||||
self._previously_seen_pixels[hash_value] = pixel
|
self._previously_seen_pixels[hash_value] = pixel
|
||||||
|
|
||||||
pr, pg, pb, pa = self._previous_pixel
|
prev_r, prev_g, prev_b, prev_a = self._previous_pixel
|
||||||
if a == pa:
|
if prev_a == a:
|
||||||
dr = self._delta(r, pr)
|
delta_r = self._delta(r, prev_r)
|
||||||
dg = self._delta(g, pg)
|
delta_g = self._delta(g, prev_g)
|
||||||
db = self._delta(b, pb)
|
delta_b = self._delta(b, prev_b)
|
||||||
dgr = self._delta(dr, dg)
|
|
||||||
dgb = self._delta(db, dg)
|
|
||||||
|
|
||||||
if -2 <= dr < 2 and -2 <= dg < 2 and -2 <= db < 2:
|
if (
|
||||||
|
-2 <= delta_r < 2
|
||||||
|
and -2 <= delta_g < 2
|
||||||
|
and -2 <= delta_b < 2
|
||||||
|
):
|
||||||
data += o8(
|
data += o8(
|
||||||
0x40 | (dr + 2) << 4 | (dg + 2) << 2 | (db + 2)
|
0b01000000
|
||||||
|
| (delta_r + 2) << 4
|
||||||
|
| (delta_g + 2) << 2
|
||||||
|
| (delta_b + 2)
|
||||||
) # QOI_OP_DIFF
|
) # QOI_OP_DIFF
|
||||||
elif -8 <= dgr < 8 and -32 <= dg < 32 and -8 <= dgb < 8:
|
|
||||||
data += o8(0x80 | (dg + 32)) # QOI_OP_LUMA
|
|
||||||
data += o8((dgr + 8) << 4 | (dgb + 8))
|
|
||||||
else:
|
else:
|
||||||
data += o8(0xFE) # QOI_OP_RGB
|
delta_gr = self._delta(delta_r, delta_g)
|
||||||
data += bytes(pixel[:3])
|
delta_gb = self._delta(delta_b, delta_g)
|
||||||
|
if (
|
||||||
|
-8 <= delta_gr < 8
|
||||||
|
and -32 <= delta_g < 32
|
||||||
|
and -8 <= delta_gb < 8
|
||||||
|
):
|
||||||
|
data += o8(
|
||||||
|
0b10000000 | (delta_g + 32)
|
||||||
|
) # QOI_OP_LUMA
|
||||||
|
data += o8((delta_gr + 8) << 4 | (delta_gb + 8))
|
||||||
|
else:
|
||||||
|
data += o8(0b11111110) # QOI_OP_RGB
|
||||||
|
data += bytes(pixel[:3])
|
||||||
else:
|
else:
|
||||||
data += o8(0xFF) # QOI_OP_RGBA
|
data += o8(0b11111111) # QOI_OP_RGBA
|
||||||
data += bytes(pixel)
|
data += bytes(pixel)
|
||||||
|
|
||||||
self._previous_pixel = pixel
|
self._previous_pixel = pixel
|
||||||
|
|
||||||
if run > 0:
|
if self._run:
|
||||||
data += self._write_run(run)
|
data += self._write_run()
|
||||||
data += bytes((0, 0, 0, 0, 0, 0, 0, 1)) # padding
|
data += bytes((0, 0, 0, 0, 0, 0, 0, 1)) # padding
|
||||||
|
|
||||||
return len(data), 0, data
|
return len(data), 0, data
|
||||||
|
|
Loading…
Reference in New Issue
Block a user