mirror of
https://github.com/python-pillow/Pillow.git
synced 2025-01-12 10:16:17 +03:00
Merge branch 'master' into winbuild
This commit is contained in:
commit
9c29819941
|
@ -1,9 +1,15 @@
|
||||||
Changelog (Pillow)
|
Changelog (Pillow)
|
||||||
==================
|
==================
|
||||||
|
|
||||||
2.4.0 (unreleased)
|
2.4.0 (2014-04-01)
|
||||||
------------------
|
------------------
|
||||||
|
|
||||||
|
- Indexed Transparency handled for conversions between L, RGB, and P modes. Fixes #510
|
||||||
|
[wiredfool]
|
||||||
|
|
||||||
|
- Conversions enabled from RGBA->P, Fixes #544
|
||||||
|
[wiredfool]
|
||||||
|
|
||||||
- Improved icns support
|
- Improved icns support
|
||||||
[al45tair]
|
[al45tair]
|
||||||
|
|
||||||
|
|
|
@ -9,6 +9,7 @@ include tox.ini
|
||||||
recursive-include Images *.bdf
|
recursive-include Images *.bdf
|
||||||
recursive-include Images *.fli
|
recursive-include Images *.fli
|
||||||
recursive-include Images *.gif
|
recursive-include Images *.gif
|
||||||
|
recursive-include Images *.icns
|
||||||
recursive-include Images *.ico
|
recursive-include Images *.ico
|
||||||
recursive-include Images *.jpg
|
recursive-include Images *.jpg
|
||||||
recursive-include Images *.pbm
|
recursive-include Images *.pbm
|
||||||
|
@ -34,7 +35,9 @@ recursive-include Tests *.gif
|
||||||
recursive-include Tests *.gnuplot
|
recursive-include Tests *.gnuplot
|
||||||
recursive-include Tests *.html
|
recursive-include Tests *.html
|
||||||
recursive-include Tests *.icm
|
recursive-include Tests *.icm
|
||||||
|
recursive-include Tests *.icns
|
||||||
recursive-include Tests *.ico
|
recursive-include Tests *.ico
|
||||||
|
recursive-include Tests *.jp2
|
||||||
recursive-include Tests *.jpg
|
recursive-include Tests *.jpg
|
||||||
recursive-include Tests *.pcf
|
recursive-include Tests *.pcf
|
||||||
recursive-include Tests *.pcx
|
recursive-include Tests *.pcx
|
||||||
|
@ -47,6 +50,7 @@ recursive-include Tests *.ttf
|
||||||
recursive-include Tests *.txt
|
recursive-include Tests *.txt
|
||||||
recursive-include Tk *.c
|
recursive-include Tk *.c
|
||||||
recursive-include Tk *.txt
|
recursive-include Tk *.txt
|
||||||
|
recursive-include depends *.sh
|
||||||
recursive-include docs *.bat
|
recursive-include docs *.bat
|
||||||
recursive-include docs *.gitignore
|
recursive-include docs *.gitignore
|
||||||
recursive-include docs *.html
|
recursive-include docs *.html
|
||||||
|
|
80
PIL/Image.py
80
PIL/Image.py
|
@ -735,21 +735,65 @@ class Image:
|
||||||
im = self.im.convert_matrix(mode, matrix)
|
im = self.im.convert_matrix(mode, matrix)
|
||||||
return self._new(im)
|
return self._new(im)
|
||||||
|
|
||||||
|
if mode == "P" and self.mode == "RGBA":
|
||||||
|
return self.quantize(colors)
|
||||||
|
|
||||||
|
trns = None
|
||||||
|
delete_trns = False
|
||||||
|
# transparency handling
|
||||||
|
if "transparency" in self.info and self.info['transparency'] is not None:
|
||||||
|
if self.mode in ('L', 'RGB') and mode == 'RGBA':
|
||||||
|
# Use transparent conversion to promote from transparent
|
||||||
|
# color to an alpha channel.
|
||||||
|
return self._new(self.im.convert_transparent(
|
||||||
|
mode, self.info['transparency']))
|
||||||
|
elif self.mode in ('L', 'RGB', 'P') and mode in ('L', 'RGB', 'P'):
|
||||||
|
t = self.info['transparency']
|
||||||
|
if isinstance(t, bytes):
|
||||||
|
# Dragons. This can't be represented by a single color
|
||||||
|
warnings.warn('Palette images with Transparency expressed '+
|
||||||
|
' in bytes should be converted to RGBA images')
|
||||||
|
delete_trns = True
|
||||||
|
else:
|
||||||
|
# get the new transparency color.
|
||||||
|
# use existing conversions
|
||||||
|
trns_im = Image()._new(core.new(self.mode, (1,1)))
|
||||||
|
if self.mode == 'P':
|
||||||
|
trns_im.putpalette(self.palette)
|
||||||
|
trns_im.putpixel((0,0), t)
|
||||||
|
|
||||||
|
if mode in ('L','RGB'):
|
||||||
|
trns_im = trns_im.convert(mode)
|
||||||
|
else:
|
||||||
|
# can't just retrieve the palette number, got to do it
|
||||||
|
# after quantization.
|
||||||
|
trns_im = trns_im.convert('RGB')
|
||||||
|
trns = trns_im.getpixel((0,0))
|
||||||
|
|
||||||
|
|
||||||
if mode == "P" and palette == ADAPTIVE:
|
if mode == "P" and palette == ADAPTIVE:
|
||||||
im = self.im.quantize(colors)
|
im = self.im.quantize(colors)
|
||||||
new = self._new(im)
|
new = self._new(im)
|
||||||
from PIL import ImagePalette
|
from PIL import ImagePalette
|
||||||
new.palette = ImagePalette.raw("RGB", new.im.getpalette("RGB"))
|
new.palette = ImagePalette.raw("RGB", new.im.getpalette("RGB"))
|
||||||
|
if delete_trns:
|
||||||
|
# This could possibly happen if we requantize to fewer colors.
|
||||||
|
# The transparency would be totally off in that case.
|
||||||
|
del(new.info['transparency'])
|
||||||
|
if trns is not None:
|
||||||
|
try:
|
||||||
|
new.info['transparency'] = new.palette.getcolor(trns)
|
||||||
|
except:
|
||||||
|
# if we can't make a transparent color, don't leave the old
|
||||||
|
# transparency hanging around to mess us up.
|
||||||
|
del(new.info['transparency'])
|
||||||
|
warnings.warn("Couldn't allocate palette entry for transparency")
|
||||||
return new
|
return new
|
||||||
|
|
||||||
# colorspace conversion
|
# colorspace conversion
|
||||||
if dither is None:
|
if dither is None:
|
||||||
dither = FLOYDSTEINBERG
|
dither = FLOYDSTEINBERG
|
||||||
|
|
||||||
# Use transparent conversion to promote from transparent color to an alpha channel.
|
|
||||||
if self.mode in ("L", "RGB") and mode == "RGBA" and "transparency" in self.info:
|
|
||||||
return self._new(self.im.convert_transparent(mode, self.info['transparency']))
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
im = self.im.convert(mode, dither)
|
im = self.im.convert(mode, dither)
|
||||||
except ValueError:
|
except ValueError:
|
||||||
|
@ -760,9 +804,22 @@ class Image:
|
||||||
except KeyError:
|
except KeyError:
|
||||||
raise ValueError("illegal conversion")
|
raise ValueError("illegal conversion")
|
||||||
|
|
||||||
return self._new(im)
|
new_im = self._new(im)
|
||||||
|
if delete_trns:
|
||||||
|
#crash fail if we leave a bytes transparency in an rgb/l mode.
|
||||||
|
del(new.info['transparency'])
|
||||||
|
if trns is not None:
|
||||||
|
if new_im.mode == 'P':
|
||||||
|
try:
|
||||||
|
new_im.info['transparency'] = new_im.palette.getcolor(trns)
|
||||||
|
except:
|
||||||
|
del(new_im.info['transparency'])
|
||||||
|
warnings.warn("Couldn't allocate palette entry for transparency")
|
||||||
|
else:
|
||||||
|
new_im.info['transparency'] = trns
|
||||||
|
return new_im
|
||||||
|
|
||||||
def quantize(self, colors=256, method=0, kmeans=0, palette=None):
|
def quantize(self, colors=256, method=None, kmeans=0, palette=None):
|
||||||
|
|
||||||
# methods:
|
# methods:
|
||||||
# 0 = median cut
|
# 0 = median cut
|
||||||
|
@ -774,6 +831,17 @@ class Image:
|
||||||
|
|
||||||
self.load()
|
self.load()
|
||||||
|
|
||||||
|
if method is None:
|
||||||
|
# defaults:
|
||||||
|
method = 0
|
||||||
|
if self.mode == 'RGBA':
|
||||||
|
method = 2
|
||||||
|
|
||||||
|
if self.mode == 'RGBA' and method != 2:
|
||||||
|
# Caller specified an invalid mode.
|
||||||
|
raise ValueError('Fast Octree (method == 2) is the ' +
|
||||||
|
' only valid method for quantizing RGBA images')
|
||||||
|
|
||||||
if palette:
|
if palette:
|
||||||
# use palette from reference image
|
# use palette from reference image
|
||||||
palette.load()
|
palette.load()
|
||||||
|
|
875
PIL/ImageCms.py
875
PIL/ImageCms.py
File diff suppressed because it is too large
Load Diff
|
@ -12,7 +12,7 @@
|
||||||
# ;-)
|
# ;-)
|
||||||
|
|
||||||
VERSION = '1.1.7' # PIL version
|
VERSION = '1.1.7' # PIL version
|
||||||
PILLOW_VERSION = '2.3.0' # Pillow
|
PILLOW_VERSION = '2.4.0' # Pillow
|
||||||
|
|
||||||
_plugins = ['ArgImagePlugin',
|
_plugins = ['ArgImagePlugin',
|
||||||
'BmpImagePlugin',
|
'BmpImagePlugin',
|
||||||
|
|
|
@ -14,3 +14,9 @@ else:
|
||||||
# Checks if an object is a string, and that it points to a directory.
|
# Checks if an object is a string, and that it points to a directory.
|
||||||
def isDirectory(f):
|
def isDirectory(f):
|
||||||
return isPath(f) and os.path.isdir(f)
|
return isPath(f) and os.path.isdir(f)
|
||||||
|
|
||||||
|
class import_err(object):
|
||||||
|
def __init__(self, ex):
|
||||||
|
self.ex = ex
|
||||||
|
def __getattr__(self, elt):
|
||||||
|
raise self.ex
|
||||||
|
|
|
@ -46,5 +46,67 @@ def test_16bit_workaround():
|
||||||
im = Image.open('Tests/images/16bit.cropped.tif')
|
im = Image.open('Tests/images/16bit.cropped.tif')
|
||||||
_test_float_conversion(im.convert('I'))
|
_test_float_conversion(im.convert('I'))
|
||||||
|
|
||||||
|
def test_rgba_p():
|
||||||
|
im = lena('RGBA')
|
||||||
|
im.putalpha(lena('L'))
|
||||||
|
|
||||||
|
converted = im.convert('P')
|
||||||
|
comparable = converted.convert('RGBA')
|
||||||
|
|
||||||
|
assert_image_similar(im, comparable, 20)
|
||||||
|
|
||||||
|
def test_trns_p():
|
||||||
|
im = lena('P')
|
||||||
|
im.info['transparency']=0
|
||||||
|
|
||||||
|
f = tempfile('temp.png')
|
||||||
|
|
||||||
|
l = im.convert('L')
|
||||||
|
assert_equal(l.info['transparency'], 0) # undone
|
||||||
|
assert_no_exception(lambda: l.save(f))
|
||||||
|
|
||||||
|
|
||||||
|
rgb = im.convert('RGB')
|
||||||
|
assert_equal(rgb.info['transparency'], (0,0,0)) # undone
|
||||||
|
assert_no_exception(lambda: rgb.save(f))
|
||||||
|
|
||||||
|
def test_trns_l():
|
||||||
|
im = lena('L')
|
||||||
|
im.info['transparency'] = 128
|
||||||
|
|
||||||
|
f = tempfile('temp.png')
|
||||||
|
|
||||||
|
rgb = im.convert('RGB')
|
||||||
|
assert_equal(rgb.info['transparency'], (128,128,128)) # undone
|
||||||
|
assert_no_exception(lambda: rgb.save(f))
|
||||||
|
|
||||||
|
p = im.convert('P')
|
||||||
|
assert_true('transparency' in p.info)
|
||||||
|
assert_no_exception(lambda: p.save(f))
|
||||||
|
|
||||||
|
p = assert_warning(UserWarning,
|
||||||
|
lambda: im.convert('P', palette = Image.ADAPTIVE))
|
||||||
|
assert_false('transparency' in p.info)
|
||||||
|
assert_no_exception(lambda: p.save(f))
|
||||||
|
|
||||||
|
|
||||||
|
def test_trns_RGB():
|
||||||
|
im = lena('RGB')
|
||||||
|
im.info['transparency'] = im.getpixel((0,0))
|
||||||
|
|
||||||
|
f = tempfile('temp.png')
|
||||||
|
|
||||||
|
l = im.convert('L')
|
||||||
|
assert_equal(l.info['transparency'], l.getpixel((0,0))) # undone
|
||||||
|
assert_no_exception(lambda: l.save(f))
|
||||||
|
|
||||||
|
p = im.convert('P')
|
||||||
|
assert_true('transparency' in p.info)
|
||||||
|
assert_no_exception(lambda: p.save(f))
|
||||||
|
|
||||||
|
p = assert_warning(UserWarning,
|
||||||
|
lambda: im.convert('P', palette = Image.ADAPTIVE))
|
||||||
|
assert_false('transparency' in p.info)
|
||||||
|
assert_no_exception(lambda: p.save(f))
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -20,3 +20,8 @@ def test_octree_quantize():
|
||||||
assert_image(im, "P", im.size)
|
assert_image(im, "P", im.size)
|
||||||
|
|
||||||
assert len(im.getcolors()) == 100
|
assert len(im.getcolors()) == 100
|
||||||
|
|
||||||
|
def test_rgba_quantize():
|
||||||
|
im = lena('RGBA')
|
||||||
|
assert_no_exception(lambda: im.quantize())
|
||||||
|
assert_exception(Exception, lambda: im.quantize(method=0))
|
||||||
|
|
|
@ -3,6 +3,7 @@ from tester import *
|
||||||
from PIL import Image
|
from PIL import Image
|
||||||
try:
|
try:
|
||||||
from PIL import ImageCms
|
from PIL import ImageCms
|
||||||
|
ImageCms.core.profile_open
|
||||||
except ImportError:
|
except ImportError:
|
||||||
skip()
|
skip()
|
||||||
|
|
||||||
|
|
|
@ -121,9 +121,9 @@ def assert_no_exception(func):
|
||||||
def assert_warning(warn_class, func):
|
def assert_warning(warn_class, func):
|
||||||
# note: this assert calls func three times!
|
# note: this assert calls func three times!
|
||||||
import warnings
|
import warnings
|
||||||
def warn_error(message, category, **options):
|
def warn_error(message, category=UserWarning, **options):
|
||||||
raise category(message)
|
raise category(message)
|
||||||
def warn_ignore(message, category, **options):
|
def warn_ignore(message, category=UserWarning, **options):
|
||||||
pass
|
pass
|
||||||
warn = warnings.warn
|
warn = warnings.warn
|
||||||
result = None
|
result = None
|
||||||
|
|
|
@ -71,7 +71,7 @@
|
||||||
* See the README file for information on usage and redistribution.
|
* See the README file for information on usage and redistribution.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#define PILLOW_VERSION "2.3.0"
|
#define PILLOW_VERSION "2.4.0"
|
||||||
|
|
||||||
#include "Python.h"
|
#include "Python.h"
|
||||||
|
|
||||||
|
|
|
@ -156,6 +156,8 @@ The :py:meth:`~PIL.Image.Image.save` method supports the following options:
|
||||||
JPEG 2000
|
JPEG 2000
|
||||||
^^^^^^^^^
|
^^^^^^^^^
|
||||||
|
|
||||||
|
.. versionadded:: 2.4.0
|
||||||
|
|
||||||
PIL reads and writes JPEG 2000 files containing ``L``, ``LA``, ``RGB`` or
|
PIL reads and writes JPEG 2000 files containing ``L``, ``LA``, ``RGB`` or
|
||||||
``RGBA`` data. It can also read files containing ``YCbCr`` data, which it
|
``RGBA`` data. It can also read files containing ``YCbCr`` data, which it
|
||||||
converts on read into ``RGB`` or ``RGBA`` depending on whether or not there is
|
converts on read into ``RGB`` or ``RGBA`` depending on whether or not there is
|
||||||
|
|
|
@ -18,12 +18,9 @@ Python Imaging Library by Fredrik Lundh and Contributors.
|
||||||
To start using Pillow, please read the :doc:`installation
|
To start using Pillow, please read the :doc:`installation
|
||||||
instructions <installation>`.
|
instructions <installation>`.
|
||||||
|
|
||||||
If you can't find the information you need, try the old `PIL Handbook`_, but be
|
You can get the source and contribute at
|
||||||
aware that it was last updated for PIL 1.1.5. You can download archives and old
|
https://github.com/python-imaging/Pillow. You can download archives
|
||||||
versions from `PyPI <https://pypi.python.org/pypi/Pillow>`_. You can get the
|
and old versions from `PyPI <https://pypi.python.org/pypi/Pillow>`_.
|
||||||
source and contribute at https://github.com/python-imaging/Pillow.
|
|
||||||
|
|
||||||
.. _PIL Handbook: http://effbot.org/imagingbook/pil-index.htm
|
|
||||||
|
|
||||||
.. toctree::
|
.. toctree::
|
||||||
:maxdepth: 2
|
:maxdepth: 2
|
||||||
|
|
|
@ -66,10 +66,15 @@ Many of Pillow's features require external libraries:
|
||||||
* **libwebp** provides the Webp format.
|
* **libwebp** provides the Webp format.
|
||||||
|
|
||||||
* Pillow has been tested with version **0.1.3**, which does not read
|
* Pillow has been tested with version **0.1.3**, which does not read
|
||||||
transparent webp files. Version **0.3.0** supports transparency.
|
transparent webp files. Versions **0.3.0** and **0.4.0** support
|
||||||
|
transparency.
|
||||||
|
|
||||||
* **tcl/tk** provides support for tkinter bitmap and photo images.
|
* **tcl/tk** provides support for tkinter bitmap and photo images.
|
||||||
|
|
||||||
|
* **openjpeg** provides JPEG 2000 functionality.
|
||||||
|
|
||||||
|
* Pillow has been tested with openjpeg **2.0.0**.
|
||||||
|
|
||||||
If the prerequisites are installed in the standard library locations for your
|
If the prerequisites are installed in the standard library locations for your
|
||||||
machine (e.g. :file:`/usr` or :file:`/usr/local`), no additional configuration
|
machine (e.g. :file:`/usr` or :file:`/usr/local`), no additional configuration
|
||||||
should be required. If they are installed in a non-standard location, you may
|
should be required. If they are installed in a non-standard location, you may
|
||||||
|
@ -225,6 +230,6 @@ current versions of Linux, OS X, and Windows.
|
||||||
+----------------------------------+-------------+------------------------------+------------------------------+-----------------------+
|
+----------------------------------+-------------+------------------------------+------------------------------+-----------------------+
|
||||||
| Windows 8 Pro |Yes | 2.6,2.7,3.2,3.3,3.4a3 | 2.2.0 |x86,x86-64 |
|
| Windows 8 Pro |Yes | 2.6,2.7,3.2,3.3,3.4a3 | 2.2.0 |x86,x86-64 |
|
||||||
+----------------------------------+-------------+------------------------------+------------------------------+-----------------------+
|
+----------------------------------+-------------+------------------------------+------------------------------+-----------------------+
|
||||||
| Windows 8.1 Pro |Yes | 2.6,2.7,3.2,3.3,3.4 | 2.3.0 |x86,x86-64 |
|
| Windows 8.1 Pro |Yes | 2.6,2.7,3.2,3.3,3.4 | 2.3.0, 2.4.0 |x86,x86-64 |
|
||||||
+----------------------------------+-------------+------------------------------+------------------------------+-----------------------+
|
+----------------------------------+-------------+------------------------------+------------------------------+-----------------------+
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
# requirements for working on docs
|
# requirements for working on docs
|
||||||
|
|
||||||
# install pillow from master if you're into that, but RtD needs this
|
# install pillow from master if you're into that, but RtD needs this
|
||||||
pillow>=2.2.1
|
pillow>=2.4.0
|
||||||
|
|
||||||
Jinja2==2.7.1
|
Jinja2==2.7.1
|
||||||
MarkupSafe==0.18
|
MarkupSafe==0.18
|
||||||
|
|
9
setup.py
9
setup.py
|
@ -87,7 +87,7 @@ except (ImportError, OSError):
|
||||||
|
|
||||||
|
|
||||||
NAME = 'Pillow'
|
NAME = 'Pillow'
|
||||||
VERSION = '2.3.0'
|
VERSION = '2.4.0'
|
||||||
TCL_ROOT = None
|
TCL_ROOT = None
|
||||||
JPEG_ROOT = None
|
JPEG_ROOT = None
|
||||||
JPEG2K_ROOT = None
|
JPEG2K_ROOT = None
|
||||||
|
@ -226,7 +226,12 @@ class pil_build_ext(build_ext):
|
||||||
_add_directory(include_dirs, "/usr/X11/include")
|
_add_directory(include_dirs, "/usr/X11/include")
|
||||||
|
|
||||||
elif sys.platform.startswith("linux"):
|
elif sys.platform.startswith("linux"):
|
||||||
for platform_ in (plat.architecture()[0], plat.processor()):
|
arch_tp = (plat.processor(), plat.architecture()[0])
|
||||||
|
if arch_tp == ("x86_64","32bit"):
|
||||||
|
# 32 bit build on 64 bit machine.
|
||||||
|
_add_directory(library_dirs, "/usr/lib/i386-linux-gnu")
|
||||||
|
else:
|
||||||
|
for platform_ in arch_tp:
|
||||||
|
|
||||||
if not platform_:
|
if not platform_:
|
||||||
continue
|
continue
|
||||||
|
|
Loading…
Reference in New Issue
Block a user