openmedialibrary_platform/Darwin/lib/python3.5/site-packages/PIL/PyAccess.py

318 lines
8.5 KiB
Python
Raw Normal View History

2014-09-30 16:15:32 +00:00
#
# The Python Imaging Library
# Pillow fork
#
# Python implementation of the PixelAccess Object
#
# Copyright (c) 1997-2009 by Secret Labs AB. All rights reserved.
# Copyright (c) 1995-2009 by Fredrik Lundh.
# Copyright (c) 2013 Eric Soroos
#
# See the README file for information on usage and redistribution
#
# Notes:
#
# * Implements the pixel access object following Access.
# * Does not implement the line functions, as they don't appear to be used
# * Taking only the tuple form, which is used from python.
2015-11-25 01:25:01 +00:00
# * Fill.c uses the integer form, but it's still going to use the old
# Access.c implementation.
2014-09-30 16:15:32 +00:00
#
from __future__ import print_function
2015-11-25 01:25:01 +00:00
import logging
2014-09-30 16:15:32 +00:00
import sys
2015-11-25 01:25:01 +00:00
from cffi import FFI
logger = logging.getLogger(__name__)
2014-09-30 16:15:32 +00:00
defs = """
struct Pixel_RGBA {
unsigned char r,g,b,a;
};
struct Pixel_I16 {
unsigned char l,r;
};
"""
ffi = FFI()
ffi.cdef(defs)
class PyAccess(object):
2015-11-25 01:25:01 +00:00
def __init__(self, img, readonly=False):
2014-09-30 16:15:32 +00:00
vals = dict(img.im.unsafe_ptrs)
self.readonly = readonly
self.image8 = ffi.cast('unsigned char **', vals['image8'])
self.image32 = ffi.cast('int **', vals['image32'])
self.image = ffi.cast('unsigned char **', vals['image'])
self.xsize = vals['xsize']
self.ysize = vals['ysize']
2015-11-25 01:25:01 +00:00
# Debugging is polluting test traces, only useful here
# when hacking on PyAccess
#logger.debug("%s", vals)
2014-09-30 16:15:32 +00:00
self._post_init()
2015-11-25 01:25:01 +00:00
def _post_init(self):
pass
2014-09-30 16:15:32 +00:00
def __setitem__(self, xy, color):
"""
Modifies the pixel at x,y. The color is given as a single
numerical value for single band images, and a tuple for
multi-band images
:param xy: The pixel coordinate, given as (x, y).
2015-11-25 01:25:01 +00:00
:param value: The pixel value.
2014-09-30 16:15:32 +00:00
"""
2015-11-25 01:25:01 +00:00
if self.readonly:
raise ValueError('Attempt to putpixel a read only image')
(x, y) = self.check_xy(xy)
return self.set_pixel(x, y, color)
2014-09-30 16:15:32 +00:00
def __getitem__(self, xy):
"""
Returns the pixel at x,y. The pixel is returned as a single
value for single band images or a tuple for multiple band
images
:param xy: The pixel coordinate, given as (x, y).
2015-11-25 01:25:01 +00:00
:returns: a pixel value for single band images, a tuple of
pixel values for multiband images.
2014-09-30 16:15:32 +00:00
"""
2015-11-25 01:25:01 +00:00
(x, y) = self.check_xy(xy)
return self.get_pixel(x, y)
2014-09-30 16:15:32 +00:00
putpixel = __setitem__
getpixel = __getitem__
def check_xy(self, xy):
2015-11-25 01:25:01 +00:00
(x, y) = xy
2014-09-30 16:15:32 +00:00
if not (0 <= x < self.xsize and 0 <= y < self.ysize):
raise ValueError('pixel location out of range')
return xy
2015-11-25 01:25:01 +00:00
2014-09-30 16:15:32 +00:00
class _PyAccess32_2(PyAccess):
""" PA, LA, stored in first and last bytes of a 32 bit word """
def _post_init(self, *args, **kwargs):
self.pixels = ffi.cast("struct Pixel_RGBA **", self.image32)
2015-11-25 01:25:01 +00:00
def get_pixel(self, x, y):
2014-09-30 16:15:32 +00:00
pixel = self.pixels[y][x]
return (pixel.r, pixel.a)
2015-11-25 01:25:01 +00:00
def set_pixel(self, x, y, color):
2014-09-30 16:15:32 +00:00
pixel = self.pixels[y][x]
# tuple
2015-11-25 01:25:01 +00:00
pixel.r = min(color[0], 255)
pixel.a = min(color[1], 255)
2014-09-30 16:15:32 +00:00
class _PyAccess32_3(PyAccess):
""" RGB and friends, stored in the first three bytes of a 32 bit word """
2015-11-25 01:25:01 +00:00
2014-09-30 16:15:32 +00:00
def _post_init(self, *args, **kwargs):
self.pixels = ffi.cast("struct Pixel_RGBA **", self.image32)
2015-11-25 01:25:01 +00:00
def get_pixel(self, x, y):
2014-09-30 16:15:32 +00:00
pixel = self.pixels[y][x]
return (pixel.r, pixel.g, pixel.b)
2015-11-25 01:25:01 +00:00
def set_pixel(self, x, y, color):
2014-09-30 16:15:32 +00:00
pixel = self.pixels[y][x]
# tuple
2015-11-25 01:25:01 +00:00
pixel.r = min(color[0], 255)
pixel.g = min(color[1], 255)
pixel.b = min(color[2], 255)
2014-09-30 16:15:32 +00:00
class _PyAccess32_4(PyAccess):
""" RGBA etc, all 4 bytes of a 32 bit word """
def _post_init(self, *args, **kwargs):
self.pixels = ffi.cast("struct Pixel_RGBA **", self.image32)
2015-11-25 01:25:01 +00:00
def get_pixel(self, x, y):
2014-09-30 16:15:32 +00:00
pixel = self.pixels[y][x]
return (pixel.r, pixel.g, pixel.b, pixel.a)
2015-11-25 01:25:01 +00:00
def set_pixel(self, x, y, color):
2014-09-30 16:15:32 +00:00
pixel = self.pixels[y][x]
# tuple
2015-11-25 01:25:01 +00:00
pixel.r = min(color[0], 255)
pixel.g = min(color[1], 255)
pixel.b = min(color[2], 255)
pixel.a = min(color[3], 255)
2014-09-30 16:15:32 +00:00
class _PyAccess8(PyAccess):
""" 1, L, P, 8 bit images stored as uint8 """
def _post_init(self, *args, **kwargs):
self.pixels = self.image8
2015-11-25 01:25:01 +00:00
def get_pixel(self, x, y):
2014-09-30 16:15:32 +00:00
return self.pixels[y][x]
2015-11-25 01:25:01 +00:00
def set_pixel(self, x, y, color):
2014-09-30 16:15:32 +00:00
try:
# integer
2015-11-25 01:25:01 +00:00
self.pixels[y][x] = min(color, 255)
2014-09-30 16:15:32 +00:00
except:
# tuple
2015-11-25 01:25:01 +00:00
self.pixels[y][x] = min(color[0], 255)
2014-09-30 16:15:32 +00:00
class _PyAccessI16_N(PyAccess):
""" I;16 access, native bitendian without conversion """
def _post_init(self, *args, **kwargs):
self.pixels = ffi.cast('unsigned short **', self.image)
2015-11-25 01:25:01 +00:00
def get_pixel(self, x, y):
2014-09-30 16:15:32 +00:00
return self.pixels[y][x]
2015-11-25 01:25:01 +00:00
def set_pixel(self, x, y, color):
2014-09-30 16:15:32 +00:00
try:
# integer
self.pixels[y][x] = min(color, 65535)
except:
# tuple
self.pixels[y][x] = min(color[0], 65535)
2015-11-25 01:25:01 +00:00
2014-09-30 16:15:32 +00:00
class _PyAccessI16_L(PyAccess):
""" I;16L access, with conversion """
def _post_init(self, *args, **kwargs):
self.pixels = ffi.cast('struct Pixel_I16 **', self.image)
2015-11-25 01:25:01 +00:00
def get_pixel(self, x, y):
2014-09-30 16:15:32 +00:00
pixel = self.pixels[y][x]
return pixel.l + pixel.r * 256
2015-11-25 01:25:01 +00:00
def set_pixel(self, x, y, color):
2014-09-30 16:15:32 +00:00
pixel = self.pixels[y][x]
try:
color = min(color, 65535)
2015-11-25 01:25:01 +00:00
except TypeError:
2014-09-30 16:15:32 +00:00
color = min(color[0], 65535)
pixel.l = color & 0xFF
pixel.r = color >> 8
2015-11-25 01:25:01 +00:00
2014-09-30 16:15:32 +00:00
class _PyAccessI16_B(PyAccess):
""" I;16B access, with conversion """
def _post_init(self, *args, **kwargs):
self.pixels = ffi.cast('struct Pixel_I16 **', self.image)
2015-11-25 01:25:01 +00:00
def get_pixel(self, x, y):
2014-09-30 16:15:32 +00:00
pixel = self.pixels[y][x]
2015-11-25 01:25:01 +00:00
return pixel.l * 256 + pixel.r
2014-09-30 16:15:32 +00:00
2015-11-25 01:25:01 +00:00
def set_pixel(self, x, y, color):
2014-09-30 16:15:32 +00:00
pixel = self.pixels[y][x]
try:
color = min(color, 65535)
except:
color = min(color[0], 65535)
pixel.l = color >> 8
pixel.r = color & 0xFF
2015-11-25 01:25:01 +00:00
2014-09-30 16:15:32 +00:00
class _PyAccessI32_N(PyAccess):
""" Signed Int32 access, native endian """
def _post_init(self, *args, **kwargs):
self.pixels = self.image32
2015-11-25 01:25:01 +00:00
def get_pixel(self, x, y):
2014-09-30 16:15:32 +00:00
return self.pixels[y][x]
2015-11-25 01:25:01 +00:00
def set_pixel(self, x, y, color):
2014-09-30 16:15:32 +00:00
self.pixels[y][x] = color
2015-11-25 01:25:01 +00:00
2014-09-30 16:15:32 +00:00
class _PyAccessI32_Swap(PyAccess):
""" I;32L/B access, with byteswapping conversion """
def _post_init(self, *args, **kwargs):
self.pixels = self.image32
def reverse(self, i):
orig = ffi.new('int *', i)
chars = ffi.cast('unsigned char *', orig)
2015-11-25 01:25:01 +00:00
chars[0], chars[1], chars[2], chars[3] = chars[3], chars[2], \
chars[1], chars[0]
2014-09-30 16:15:32 +00:00
return ffi.cast('int *', chars)[0]
2015-11-25 01:25:01 +00:00
def get_pixel(self, x, y):
2014-09-30 16:15:32 +00:00
return self.reverse(self.pixels[y][x])
2015-11-25 01:25:01 +00:00
def set_pixel(self, x, y, color):
2014-09-30 16:15:32 +00:00
self.pixels[y][x] = self.reverse(color)
2015-11-25 01:25:01 +00:00
2014-09-30 16:15:32 +00:00
class _PyAccessF(PyAccess):
""" 32 bit float access """
def _post_init(self, *args, **kwargs):
self.pixels = ffi.cast('float **', self.image32)
2015-11-25 01:25:01 +00:00
def get_pixel(self, x, y):
2014-09-30 16:15:32 +00:00
return self.pixels[y][x]
2015-11-25 01:25:01 +00:00
def set_pixel(self, x, y, color):
2014-09-30 16:15:32 +00:00
try:
# not a tuple
self.pixels[y][x] = color
except:
# tuple
self.pixels[y][x] = color[0]
mode_map = {'1': _PyAccess8,
'L': _PyAccess8,
'P': _PyAccess8,
'LA': _PyAccess32_2,
'PA': _PyAccess32_2,
'RGB': _PyAccess32_3,
'LAB': _PyAccess32_3,
2015-11-25 01:25:01 +00:00
'HSV': _PyAccess32_3,
2014-09-30 16:15:32 +00:00
'YCbCr': _PyAccess32_3,
'RGBA': _PyAccess32_4,
'RGBa': _PyAccess32_4,
'RGBX': _PyAccess32_4,
'CMYK': _PyAccess32_4,
'F': _PyAccessF,
'I': _PyAccessI32_N,
}
if sys.byteorder == 'little':
mode_map['I;16'] = _PyAccessI16_N
mode_map['I;16L'] = _PyAccessI16_N
mode_map['I;16B'] = _PyAccessI16_B
2015-11-25 01:25:01 +00:00
2014-09-30 16:15:32 +00:00
mode_map['I;32L'] = _PyAccessI32_N
mode_map['I;32B'] = _PyAccessI32_Swap
else:
mode_map['I;16'] = _PyAccessI16_L
mode_map['I;16L'] = _PyAccessI16_L
mode_map['I;16B'] = _PyAccessI16_N
mode_map['I;32L'] = _PyAccessI32_Swap
mode_map['I;32B'] = _PyAccessI32_N
2015-11-25 01:25:01 +00:00
def new(img, readonly=False):
2014-09-30 16:15:32 +00:00
access_type = mode_map.get(img.mode, None)
if not access_type:
2015-11-25 01:25:01 +00:00
logger.debug("PyAccess Not Implemented: %s", img.mode)
2014-09-30 16:15:32 +00:00
return None
return access_type(img, readonly)
2015-11-25 01:25:01 +00:00
# End of file