mirror of
https://github.com/python-pillow/Pillow.git
synced 2025-01-28 10:14:30 +03:00
Merge branch 'main' into ppm
This commit is contained in:
commit
673eb9a845
|
@ -65,10 +65,10 @@ As of 2019, Pillow development is
|
||||||
<a href="https://tidelift.com/subscription/pkg/pypi-pillow?utm_source=pypi-pillow&utm_medium=badge"><img
|
<a href="https://tidelift.com/subscription/pkg/pypi-pillow?utm_source=pypi-pillow&utm_medium=badge"><img
|
||||||
alt="Tidelift"
|
alt="Tidelift"
|
||||||
src="https://tidelift.com/badges/package/pypi/Pillow?style=flat"></a>
|
src="https://tidelift.com/badges/package/pypi/Pillow?style=flat"></a>
|
||||||
<a href="https://pypi.org/project/Pillow/"><img
|
<a href="https://pypi.org/project/pillow/"><img
|
||||||
alt="Newest PyPI version"
|
alt="Newest PyPI version"
|
||||||
src="https://img.shields.io/pypi/v/pillow.svg"></a>
|
src="https://img.shields.io/pypi/v/pillow.svg"></a>
|
||||||
<a href="https://pypi.org/project/Pillow/"><img
|
<a href="https://pypi.org/project/pillow/"><img
|
||||||
alt="Number of PyPI downloads"
|
alt="Number of PyPI downloads"
|
||||||
src="https://img.shields.io/pypi/dm/pillow.svg"></a>
|
src="https://img.shields.io/pypi/dm/pillow.svg"></a>
|
||||||
<a href="https://www.bestpractices.dev/projects/6331"><img
|
<a href="https://www.bestpractices.dev/projects/6331"><img
|
||||||
|
|
BIN
Tests/images/hopper.pfm
Normal file
BIN
Tests/images/hopper.pfm
Normal file
Binary file not shown.
BIN
Tests/images/hopper_be.pfm
Normal file
BIN
Tests/images/hopper_be.pfm
Normal file
Binary file not shown.
|
@ -6,7 +6,12 @@ import pytest
|
||||||
|
|
||||||
from PIL import Image, PpmImagePlugin
|
from PIL import Image, PpmImagePlugin
|
||||||
|
|
||||||
from .helper import assert_image_equal_tofile, assert_image_similar, hopper
|
from .helper import (
|
||||||
|
assert_image_equal,
|
||||||
|
assert_image_equal_tofile,
|
||||||
|
assert_image_similar,
|
||||||
|
hopper,
|
||||||
|
)
|
||||||
|
|
||||||
# sample ppm stream
|
# sample ppm stream
|
||||||
TEST_FILE = "Tests/images/hopper.ppm"
|
TEST_FILE = "Tests/images/hopper.ppm"
|
||||||
|
@ -84,20 +89,58 @@ def test_16bit_pgm():
|
||||||
|
|
||||||
def test_16bit_pgm_write(tmp_path):
|
def test_16bit_pgm_write(tmp_path):
|
||||||
with Image.open("Tests/images/16_bit_binary.pgm") as im:
|
with Image.open("Tests/images/16_bit_binary.pgm") as im:
|
||||||
f = str(tmp_path / "temp.pgm")
|
filename = str(tmp_path / "temp.pgm")
|
||||||
im.save(f, "PPM")
|
im.save(filename, "PPM")
|
||||||
|
|
||||||
assert_image_equal_tofile(im, f)
|
assert_image_equal_tofile(im, filename)
|
||||||
|
|
||||||
|
|
||||||
def test_pnm(tmp_path):
|
def test_pnm(tmp_path):
|
||||||
with Image.open("Tests/images/hopper.pnm") as im:
|
with Image.open("Tests/images/hopper.pnm") as im:
|
||||||
assert_image_similar(im, hopper(), 0.0001)
|
assert_image_similar(im, hopper(), 0.0001)
|
||||||
|
|
||||||
f = str(tmp_path / "temp.pnm")
|
filename = str(tmp_path / "temp.pnm")
|
||||||
im.save(f)
|
im.save(filename)
|
||||||
|
|
||||||
assert_image_equal_tofile(im, f)
|
assert_image_equal_tofile(im, filename)
|
||||||
|
|
||||||
|
|
||||||
|
def test_pfm(tmp_path):
|
||||||
|
with Image.open("Tests/images/hopper.pfm") as im:
|
||||||
|
assert im.info["scale"] == 1.0
|
||||||
|
assert_image_equal(im, hopper("F"))
|
||||||
|
|
||||||
|
filename = str(tmp_path / "tmp.pfm")
|
||||||
|
im.save(filename)
|
||||||
|
|
||||||
|
assert_image_equal_tofile(im, filename)
|
||||||
|
|
||||||
|
|
||||||
|
def test_pfm_big_endian(tmp_path):
|
||||||
|
with Image.open("Tests/images/hopper_be.pfm") as im:
|
||||||
|
assert im.info["scale"] == 2.5
|
||||||
|
assert_image_equal(im, hopper("F"))
|
||||||
|
|
||||||
|
filename = str(tmp_path / "tmp.pfm")
|
||||||
|
im.save(filename)
|
||||||
|
|
||||||
|
assert_image_equal_tofile(im, filename)
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize(
|
||||||
|
"data",
|
||||||
|
[
|
||||||
|
b"Pf 1 1 NaN \0\0\0\0",
|
||||||
|
b"Pf 1 1 inf \0\0\0\0",
|
||||||
|
b"Pf 1 1 -inf \0\0\0\0",
|
||||||
|
b"Pf 1 1 0.0 \0\0\0\0",
|
||||||
|
b"Pf 1 1 -0.0 \0\0\0\0",
|
||||||
|
],
|
||||||
|
)
|
||||||
|
def test_pfm_invalid(data):
|
||||||
|
with pytest.raises(ValueError):
|
||||||
|
with Image.open(BytesIO(data)):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize(
|
@pytest.mark.parametrize(
|
||||||
|
|
|
@ -13,7 +13,7 @@ The fork author's goal is to foster and support active development of PIL throug
|
||||||
.. _GitHub Actions: https://github.com/python-pillow/Pillow/actions
|
.. _GitHub Actions: https://github.com/python-pillow/Pillow/actions
|
||||||
.. _AppVeyor: https://ci.appveyor.com/project/Python-pillow/pillow
|
.. _AppVeyor: https://ci.appveyor.com/project/Python-pillow/pillow
|
||||||
.. _GitHub: https://github.com/python-pillow/Pillow
|
.. _GitHub: https://github.com/python-pillow/Pillow
|
||||||
.. _Python Package Index: https://pypi.org/project/Pillow/
|
.. _Python Package Index: https://pypi.org/project/pillow/
|
||||||
|
|
||||||
License
|
License
|
||||||
-------
|
-------
|
||||||
|
|
|
@ -696,6 +696,25 @@ PCX
|
||||||
|
|
||||||
Pillow reads and writes PCX files containing ``1``, ``L``, ``P``, or ``RGB`` data.
|
Pillow reads and writes PCX files containing ``1``, ``L``, ``P``, or ``RGB`` data.
|
||||||
|
|
||||||
|
PFM
|
||||||
|
^^^
|
||||||
|
|
||||||
|
.. versionadded:: 10.3.0
|
||||||
|
|
||||||
|
Pillow reads and writes grayscale (Pf format) Portable FloatMap (PFM) files
|
||||||
|
containing ``F`` data.
|
||||||
|
|
||||||
|
Color (PF format) PFM files are not supported.
|
||||||
|
|
||||||
|
Opening
|
||||||
|
~~~~~~~
|
||||||
|
|
||||||
|
The :py:func:`~PIL.Image.open` function sets the following
|
||||||
|
:py:attr:`~PIL.Image.Image.info` properties:
|
||||||
|
|
||||||
|
**scale**
|
||||||
|
The absolute value of the number stored in the *Scale Factor / Endianness* line.
|
||||||
|
|
||||||
PNG
|
PNG
|
||||||
^^^
|
^^^
|
||||||
|
|
||||||
|
|
|
@ -58,11 +58,11 @@ Pillow for enterprise is available via the Tidelift Subscription. `Learn more <h
|
||||||
:alt: Fuzzing Status
|
:alt: Fuzzing Status
|
||||||
|
|
||||||
.. image:: https://img.shields.io/pypi/v/pillow.svg
|
.. image:: https://img.shields.io/pypi/v/pillow.svg
|
||||||
:target: https://pypi.org/project/Pillow/
|
:target: https://pypi.org/project/pillow/
|
||||||
:alt: Latest PyPI version
|
:alt: Latest PyPI version
|
||||||
|
|
||||||
.. image:: https://img.shields.io/pypi/dm/pillow.svg
|
.. image:: https://img.shields.io/pypi/dm/pillow.svg
|
||||||
:target: https://pypi.org/project/Pillow/
|
:target: https://pypi.org/project/pillow/
|
||||||
:alt: Number of PyPI downloads
|
:alt: Number of PyPI downloads
|
||||||
|
|
||||||
.. image:: https://www.bestpractices.dev/projects/6331/badge
|
.. image:: https://www.bestpractices.dev/projects/6331/badge
|
||||||
|
|
|
@ -385,7 +385,7 @@ After navigating to the Pillow directory, run::
|
||||||
python3 -m pip install --upgrade pip
|
python3 -m pip install --upgrade pip
|
||||||
python3 -m pip install .
|
python3 -m pip install .
|
||||||
|
|
||||||
.. _compressed archive from PyPI: https://pypi.org/project/Pillow/#files
|
.. _compressed archive from PyPI: https://pypi.org/project/pillow/#files
|
||||||
|
|
||||||
Build Options
|
Build Options
|
||||||
"""""""""""""
|
"""""""""""""
|
||||||
|
@ -602,5 +602,5 @@ Old Versions
|
||||||
------------
|
------------
|
||||||
|
|
||||||
You can download old distributions from the `release history at PyPI
|
You can download old distributions from the `release history at PyPI
|
||||||
<https://pypi.org/project/Pillow/#history>`_ and by direct URL access
|
<https://pypi.org/project/pillow/#history>`_ and by direct URL access
|
||||||
eg. https://pypi.org/project/Pillow/1.0/.
|
eg. https://pypi.org/project/pillow/1.0/.
|
||||||
|
|
49
docs/releasenotes/10.3.0.rst
Normal file
49
docs/releasenotes/10.3.0.rst
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
10.3.0
|
||||||
|
------
|
||||||
|
|
||||||
|
Backwards Incompatible Changes
|
||||||
|
==============================
|
||||||
|
|
||||||
|
TODO
|
||||||
|
^^^^
|
||||||
|
|
||||||
|
Deprecations
|
||||||
|
============
|
||||||
|
|
||||||
|
TODO
|
||||||
|
^^^^
|
||||||
|
|
||||||
|
TODO
|
||||||
|
|
||||||
|
API Changes
|
||||||
|
===========
|
||||||
|
|
||||||
|
TODO
|
||||||
|
^^^^
|
||||||
|
|
||||||
|
TODO
|
||||||
|
|
||||||
|
API Additions
|
||||||
|
=============
|
||||||
|
|
||||||
|
TODO
|
||||||
|
^^^^
|
||||||
|
|
||||||
|
TODO
|
||||||
|
|
||||||
|
Security
|
||||||
|
========
|
||||||
|
|
||||||
|
TODO
|
||||||
|
^^^^
|
||||||
|
|
||||||
|
TODO
|
||||||
|
|
||||||
|
Other Changes
|
||||||
|
=============
|
||||||
|
|
||||||
|
Portable FloatMap (PFM) images
|
||||||
|
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||||
|
|
||||||
|
Support has been added for reading and writing grayscale (Pf format)
|
||||||
|
Portable FloatMap (PFM) files containing ``F`` data.
|
|
@ -14,6 +14,7 @@ expected to be backported to earlier versions.
|
||||||
.. toctree::
|
.. toctree::
|
||||||
:maxdepth: 2
|
:maxdepth: 2
|
||||||
|
|
||||||
|
10.3.0
|
||||||
10.2.0
|
10.2.0
|
||||||
10.1.0
|
10.1.0
|
||||||
10.0.1
|
10.0.1
|
||||||
|
|
|
@ -15,6 +15,8 @@
|
||||||
#
|
#
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import math
|
||||||
|
|
||||||
from . import Image, ImageFile
|
from . import Image, ImageFile
|
||||||
from ._binary import i16be as i16
|
from ._binary import i16be as i16
|
||||||
from ._binary import o8
|
from ._binary import o8
|
||||||
|
@ -35,6 +37,7 @@ MODES = {
|
||||||
b"P6": "RGB",
|
b"P6": "RGB",
|
||||||
# extensions
|
# extensions
|
||||||
b"P0CMYK": "CMYK",
|
b"P0CMYK": "CMYK",
|
||||||
|
b"Pf": "F",
|
||||||
# PIL extensions (for test purposes only)
|
# PIL extensions (for test purposes only)
|
||||||
b"PyP": "P",
|
b"PyP": "P",
|
||||||
b"PyRGBA": "RGBA",
|
b"PyRGBA": "RGBA",
|
||||||
|
@ -43,7 +46,7 @@ MODES = {
|
||||||
|
|
||||||
|
|
||||||
def _accept(prefix):
|
def _accept(prefix):
|
||||||
return prefix[0:1] == b"P" and prefix[1] in b"0123456y"
|
return prefix[0:1] == b"P" and prefix[1] in b"0123456fy"
|
||||||
|
|
||||||
|
|
||||||
##
|
##
|
||||||
|
@ -97,6 +100,7 @@ class PpmImageFile(ImageFile.ImageFile):
|
||||||
except KeyError:
|
except KeyError:
|
||||||
msg = "not a PPM file"
|
msg = "not a PPM file"
|
||||||
raise SyntaxError(msg)
|
raise SyntaxError(msg)
|
||||||
|
self._mode = mode
|
||||||
|
|
||||||
if magic_number in (b"P1", b"P4"):
|
if magic_number in (b"P1", b"P4"):
|
||||||
self.custom_mimetype = "image/x-portable-bitmap"
|
self.custom_mimetype = "image/x-portable-bitmap"
|
||||||
|
@ -111,14 +115,23 @@ class PpmImageFile(ImageFile.ImageFile):
|
||||||
if magic_number in (b"P1", b"P2", b"P3"):
|
if magic_number in (b"P1", b"P2", b"P3"):
|
||||||
decoder_name = "ppm_plain"
|
decoder_name = "ppm_plain"
|
||||||
if mode == "1":
|
if mode == "1":
|
||||||
self._mode = "1"
|
|
||||||
args = "1;I"
|
args = "1;I"
|
||||||
|
elif mode == "F":
|
||||||
|
scale = float(self._read_token())
|
||||||
|
if scale == 0.0 or not math.isfinite(scale):
|
||||||
|
msg = "scale must be finite and non-zero"
|
||||||
|
raise ValueError(msg)
|
||||||
|
self.info["scale"] = abs(scale)
|
||||||
|
|
||||||
|
rawmode = "F;32F" if scale < 0 else "F;32BF"
|
||||||
|
args = (rawmode, 0, -1)
|
||||||
else:
|
else:
|
||||||
maxval = int(self._read_token())
|
maxval = int(self._read_token())
|
||||||
if not 0 < maxval < 65536:
|
if not 0 < maxval < 65536:
|
||||||
msg = "maxval must be greater than 0 and less than 65536"
|
msg = "maxval must be greater than 0 and less than 65536"
|
||||||
raise ValueError(msg)
|
raise ValueError(msg)
|
||||||
self._mode = "I" if maxval > 255 and mode == "L" else mode
|
if maxval > 255 and mode == "L":
|
||||||
|
self._mode = "I"
|
||||||
|
|
||||||
rawmode = mode
|
rawmode = mode
|
||||||
if decoder_name != "ppm_plain":
|
if decoder_name != "ppm_plain":
|
||||||
|
@ -128,7 +141,7 @@ class PpmImageFile(ImageFile.ImageFile):
|
||||||
elif maxval != 255:
|
elif maxval != 255:
|
||||||
decoder_name = "ppm"
|
decoder_name = "ppm"
|
||||||
|
|
||||||
args = (rawmode, 0, 1) if decoder_name == "raw" else (rawmode, maxval)
|
args = rawmode if decoder_name == "raw" else (rawmode, maxval)
|
||||||
self.tile = [(decoder_name, (0, 0) + self.size, self.fp.tell(), args)]
|
self.tile = [(decoder_name, (0, 0) + self.size, self.fp.tell(), args)]
|
||||||
|
|
||||||
|
|
||||||
|
@ -298,6 +311,7 @@ class PpmDecoder(ImageFile.PyDecoder):
|
||||||
|
|
||||||
|
|
||||||
def _save(im, fp, filename):
|
def _save(im, fp, filename):
|
||||||
|
row_order = 1
|
||||||
if im.mode == "1":
|
if im.mode == "1":
|
||||||
rawmode, head = "1;I", b"P4"
|
rawmode, head = "1;I", b"P4"
|
||||||
elif im.mode == "L":
|
elif im.mode == "L":
|
||||||
|
@ -306,6 +320,9 @@ def _save(im, fp, filename):
|
||||||
rawmode, head = "I;16B", b"P5"
|
rawmode, head = "I;16B", b"P5"
|
||||||
elif im.mode in ("RGB", "RGBA"):
|
elif im.mode in ("RGB", "RGBA"):
|
||||||
rawmode, head = "RGB", b"P6"
|
rawmode, head = "RGB", b"P6"
|
||||||
|
elif im.mode == "F":
|
||||||
|
rawmode, head = "F;32F", b"Pf"
|
||||||
|
row_order = -1
|
||||||
else:
|
else:
|
||||||
msg = f"cannot write mode {im.mode} as PPM"
|
msg = f"cannot write mode {im.mode} as PPM"
|
||||||
raise OSError(msg)
|
raise OSError(msg)
|
||||||
|
@ -317,7 +334,9 @@ def _save(im, fp, filename):
|
||||||
fp.write(b"255\n")
|
fp.write(b"255\n")
|
||||||
else:
|
else:
|
||||||
fp.write(b"65535\n")
|
fp.write(b"65535\n")
|
||||||
ImageFile._save(im, fp, [("raw", (0, 0) + im.size, 0, (rawmode, 0, 1))])
|
elif head == b"Pf":
|
||||||
|
fp.write(b"-1.0\n")
|
||||||
|
ImageFile._save(im, fp, [("raw", (0, 0) + im.size, 0, (rawmode, 0, row_order))])
|
||||||
|
|
||||||
|
|
||||||
#
|
#
|
||||||
|
@ -330,6 +349,6 @@ Image.register_save(PpmImageFile.format, _save)
|
||||||
Image.register_decoder("ppm", PpmDecoder)
|
Image.register_decoder("ppm", PpmDecoder)
|
||||||
Image.register_decoder("ppm_plain", PpmPlainDecoder)
|
Image.register_decoder("ppm_plain", PpmPlainDecoder)
|
||||||
|
|
||||||
Image.register_extensions(PpmImageFile.format, [".pbm", ".pgm", ".ppm", ".pnm"])
|
Image.register_extensions(PpmImageFile.format, [".pbm", ".pgm", ".ppm", ".pnm", ".pfm"])
|
||||||
|
|
||||||
Image.register_mime(PpmImageFile.format, "image/x-portable-anymap")
|
Image.register_mime(PpmImageFile.format, "image/x-portable-anymap")
|
||||||
|
|
Loading…
Reference in New Issue
Block a user