2014-01-05 22:41:25 +04: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:
|
|
|
|
#
|
2022-11-13 02:16:50 +03:00
|
|
|
# * Implements the pixel access object following Access.c
|
2014-01-05 22:41:25 +04:00
|
|
|
# * Taking only the tuple form, which is used from python.
|
|
|
|
# * Fill.c uses the integer form, but it's still going to use the old
|
|
|
|
# Access.c implementation.
|
|
|
|
#
|
2023-12-21 14:13:31 +03:00
|
|
|
from __future__ import annotations
|
2014-01-05 22:41:25 +04:00
|
|
|
|
2015-02-06 21:58:07 +03:00
|
|
|
import logging
|
2014-01-07 11:15:00 +04:00
|
|
|
import sys
|
2014-01-05 22:41:25 +04:00
|
|
|
|
2023-06-28 13:10:10 +03:00
|
|
|
from ._deprecate import deprecate
|
|
|
|
|
2024-02-10 11:50:45 +03:00
|
|
|
FFI: type
|
2020-06-14 09:35:55 +03:00
|
|
|
try:
|
|
|
|
from cffi import FFI
|
|
|
|
|
|
|
|
defs = """
|
|
|
|
struct Pixel_RGBA {
|
|
|
|
unsigned char r,g,b,a;
|
|
|
|
};
|
|
|
|
struct Pixel_I16 {
|
|
|
|
unsigned char l,r;
|
|
|
|
};
|
|
|
|
"""
|
|
|
|
ffi = FFI()
|
|
|
|
ffi.cdef(defs)
|
|
|
|
except ImportError as ex:
|
|
|
|
# Allow error import for doc purposes, but error out when accessing
|
|
|
|
# anything in core.
|
2022-04-10 21:21:50 +03:00
|
|
|
from ._util import DeferredError
|
2020-06-14 09:35:55 +03:00
|
|
|
|
2023-12-27 02:40:55 +03:00
|
|
|
FFI = ffi = DeferredError.new(ex)
|
2015-02-06 21:58:07 +03:00
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2019-09-30 17:56:31 +03:00
|
|
|
class PyAccess:
|
2014-01-05 22:41:25 +04:00
|
|
|
def __init__(self, img, readonly=False):
|
2023-06-28 13:10:10 +03:00
|
|
|
deprecate("PyAccess", 11)
|
2014-01-05 22:41:25 +04:00
|
|
|
vals = dict(img.im.unsafe_ptrs)
|
|
|
|
self.readonly = readonly
|
2014-01-06 10:00:09 +04:00
|
|
|
self.image8 = ffi.cast("unsigned char **", vals["image8"])
|
|
|
|
self.image32 = ffi.cast("int **", vals["image32"])
|
|
|
|
self.image = ffi.cast("unsigned char **", vals["image"])
|
2017-09-16 21:52:28 +03:00
|
|
|
self.xsize, self.ysize = img.im.size
|
2021-06-23 11:41:46 +03:00
|
|
|
self._img = img
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2016-07-03 05:03:25 +03:00
|
|
|
# Keep pointer to im object to prevent dereferencing.
|
|
|
|
self._im = img.im
|
2022-08-14 07:34:42 +03:00
|
|
|
if self._im.mode in ("P", "PA"):
|
2018-12-31 03:35:15 +03:00
|
|
|
self._palette = img.palette
|
2016-07-03 05:03:25 +03:00
|
|
|
|
2015-09-14 13:38:52 +03:00
|
|
|
# Debugging is polluting test traces, only useful here
|
|
|
|
# when hacking on PyAccess
|
2015-12-10 01:35:35 +03:00
|
|
|
# logger.debug("%s", vals)
|
2014-01-06 10:00:09 +04:00
|
|
|
self._post_init()
|
2014-01-05 22:41:25 +04:00
|
|
|
|
2015-04-01 00:58:10 +03:00
|
|
|
def _post_init(self):
|
2014-01-06 10:00:09 +04:00
|
|
|
pass
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2014-01-05 22:41:25 +04: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
|
|
|
|
|
2018-06-24 07:34:01 +03:00
|
|
|
:param xy: The pixel coordinate, given as (x, y). See
|
|
|
|
:ref:`coordinate-system`.
|
2017-09-29 14:08:49 +03:00
|
|
|
:param color: The pixel value.
|
2014-01-05 22:41:25 +04:00
|
|
|
"""
|
2014-01-07 09:19:58 +04:00
|
|
|
if self.readonly:
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = "Attempt to putpixel a read only image"
|
|
|
|
raise ValueError(msg)
|
2018-10-15 13:51:24 +03:00
|
|
|
(x, y) = xy
|
|
|
|
if x < 0:
|
|
|
|
x = self.xsize + x
|
|
|
|
if y < 0:
|
|
|
|
y = self.ysize + y
|
|
|
|
(x, y) = self.check_xy((x, y))
|
2018-12-31 03:35:15 +03:00
|
|
|
|
|
|
|
if (
|
2022-08-14 07:34:42 +03:00
|
|
|
self._im.mode in ("P", "PA")
|
2018-12-31 05:37:04 +03:00
|
|
|
and isinstance(color, (list, tuple))
|
|
|
|
and len(color) in [3, 4]
|
|
|
|
):
|
2022-08-14 07:34:42 +03:00
|
|
|
# RGB or RGBA value for a P or PA image
|
|
|
|
if self._im.mode == "PA":
|
|
|
|
alpha = color[3] if len(color) == 4 else 255
|
|
|
|
color = color[:3]
|
2021-06-23 11:41:46 +03:00
|
|
|
color = self._palette.getcolor(color, self._img)
|
2022-08-14 07:34:42 +03:00
|
|
|
if self._im.mode == "PA":
|
|
|
|
color = (color, alpha)
|
2018-12-31 03:35:15 +03:00
|
|
|
|
2014-01-05 22:41:25 +04:00
|
|
|
return self.set_pixel(x, y, color)
|
|
|
|
|
|
|
|
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
|
|
|
|
|
2018-06-24 07:34:01 +03:00
|
|
|
:param xy: The pixel coordinate, given as (x, y). See
|
|
|
|
:ref:`coordinate-system`.
|
2014-11-20 01:04:39 +03:00
|
|
|
:returns: a pixel value for single band images, a tuple of
|
|
|
|
pixel values for multiband images.
|
2014-01-05 22:41:25 +04:00
|
|
|
"""
|
2018-10-14 17:12:58 +03:00
|
|
|
(x, y) = xy
|
2018-10-15 13:51:24 +03:00
|
|
|
if x < 0:
|
2018-10-14 17:12:58 +03:00
|
|
|
x = self.xsize + x
|
2018-10-15 13:51:24 +03:00
|
|
|
if y < 0:
|
2018-10-14 17:12:58 +03:00
|
|
|
y = self.ysize + y
|
|
|
|
(x, y) = self.check_xy((x, y))
|
2014-01-05 22:41:25 +04:00
|
|
|
return self.get_pixel(x, y)
|
|
|
|
|
|
|
|
putpixel = __setitem__
|
|
|
|
getpixel = __getitem__
|
|
|
|
|
|
|
|
def check_xy(self, xy):
|
|
|
|
(x, y) = xy
|
2014-01-06 10:00:09 +04:00
|
|
|
if not (0 <= x < self.xsize and 0 <= y < self.ysize):
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = "pixel location out of range"
|
|
|
|
raise ValueError(msg)
|
2014-01-07 09:19:58 +04:00
|
|
|
return xy
|
2014-01-05 22:41:25 +04:00
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2014-01-07 10:51:31 +04:00
|
|
|
class _PyAccess32_2(PyAccess):
|
2021-08-12 14:50:09 +03:00
|
|
|
"""PA, LA, stored in first and last bytes of a 32 bit word"""
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2014-01-07 10:51:31 +04:00
|
|
|
def _post_init(self, *args, **kwargs):
|
|
|
|
self.pixels = ffi.cast("struct Pixel_RGBA **", self.image32)
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2014-01-07 10:51:31 +04:00
|
|
|
def get_pixel(self, x, y):
|
|
|
|
pixel = self.pixels[y][x]
|
2022-04-10 19:25:40 +03:00
|
|
|
return pixel.r, pixel.a
|
2014-01-05 22:41:25 +04:00
|
|
|
|
2014-01-07 10:51:31 +04:00
|
|
|
def set_pixel(self, x, y, color):
|
|
|
|
pixel = self.pixels[y][x]
|
|
|
|
# tuple
|
|
|
|
pixel.r = min(color[0], 255)
|
|
|
|
pixel.a = min(color[1], 255)
|
2014-08-26 17:47:10 +04:00
|
|
|
|
|
|
|
|
2014-01-05 22:41:25 +04:00
|
|
|
class _PyAccess32_3(PyAccess):
|
2021-08-12 14:50:09 +03:00
|
|
|
"""RGB and friends, stored in the first three bytes of a 32 bit word"""
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2014-01-06 10:00:09 +04:00
|
|
|
def _post_init(self, *args, **kwargs):
|
|
|
|
self.pixels = ffi.cast("struct Pixel_RGBA **", self.image32)
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2014-01-05 22:41:25 +04:00
|
|
|
def get_pixel(self, x, y):
|
|
|
|
pixel = self.pixels[y][x]
|
2022-04-10 19:25:40 +03:00
|
|
|
return pixel.r, pixel.g, pixel.b
|
2014-01-05 22:41:25 +04:00
|
|
|
|
|
|
|
def set_pixel(self, x, y, color):
|
|
|
|
pixel = self.pixels[y][x]
|
2014-01-06 10:00:09 +04:00
|
|
|
# tuple
|
2014-01-07 09:20:19 +04:00
|
|
|
pixel.r = min(color[0], 255)
|
|
|
|
pixel.g = min(color[1], 255)
|
|
|
|
pixel.b = min(color[2], 255)
|
2017-08-24 01:45:24 +03:00
|
|
|
pixel.a = 255
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2014-01-05 22:41:25 +04:00
|
|
|
|
|
|
|
class _PyAccess32_4(PyAccess):
|
2021-08-12 14:50:09 +03:00
|
|
|
"""RGBA etc, all 4 bytes of a 32 bit word"""
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2014-01-06 10:00:09 +04:00
|
|
|
def _post_init(self, *args, **kwargs):
|
|
|
|
self.pixels = ffi.cast("struct Pixel_RGBA **", self.image32)
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2014-01-05 22:41:25 +04:00
|
|
|
def get_pixel(self, x, y):
|
|
|
|
pixel = self.pixels[y][x]
|
2022-04-10 19:25:40 +03:00
|
|
|
return pixel.r, pixel.g, pixel.b, pixel.a
|
2014-01-05 22:41:25 +04:00
|
|
|
|
|
|
|
def set_pixel(self, x, y, color):
|
|
|
|
pixel = self.pixels[y][x]
|
2014-01-06 10:00:09 +04:00
|
|
|
# tuple
|
2014-01-07 09:20:19 +04: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-08-26 17:47:10 +04:00
|
|
|
|
2014-01-07 09:20:19 +04:00
|
|
|
|
2014-01-05 22:41:25 +04:00
|
|
|
class _PyAccess8(PyAccess):
|
2021-08-12 14:50:09 +03:00
|
|
|
"""1, L, P, 8 bit images stored as uint8"""
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2014-01-06 10:00:09 +04:00
|
|
|
def _post_init(self, *args, **kwargs):
|
|
|
|
self.pixels = self.image8
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2014-01-05 22:41:25 +04:00
|
|
|
def get_pixel(self, x, y):
|
|
|
|
return self.pixels[y][x]
|
|
|
|
|
|
|
|
def set_pixel(self, x, y, color):
|
|
|
|
try:
|
|
|
|
# integer
|
2014-01-06 10:20:07 +04:00
|
|
|
self.pixels[y][x] = min(color, 255)
|
2017-03-01 12:20:18 +03:00
|
|
|
except TypeError:
|
2014-01-05 22:41:25 +04:00
|
|
|
# tuple
|
2014-01-06 10:20:07 +04:00
|
|
|
self.pixels[y][x] = min(color[0], 255)
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2014-01-05 22:41:25 +04:00
|
|
|
|
2014-01-07 11:15:00 +04:00
|
|
|
class _PyAccessI16_N(PyAccess):
|
2021-08-12 14:50:09 +03:00
|
|
|
"""I;16 access, native bitendian without conversion"""
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2014-01-07 11:15:00 +04:00
|
|
|
def _post_init(self, *args, **kwargs):
|
|
|
|
self.pixels = ffi.cast("unsigned short **", self.image)
|
|
|
|
|
|
|
|
def get_pixel(self, x, y):
|
|
|
|
return self.pixels[y][x]
|
|
|
|
|
|
|
|
def set_pixel(self, x, y, color):
|
|
|
|
try:
|
|
|
|
# integer
|
|
|
|
self.pixels[y][x] = min(color, 65535)
|
2017-03-01 12:20:18 +03:00
|
|
|
except TypeError:
|
2014-01-07 11:15:00 +04:00
|
|
|
# tuple
|
|
|
|
self.pixels[y][x] = min(color[0], 65535)
|
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2014-01-07 11:15:00 +04:00
|
|
|
class _PyAccessI16_L(PyAccess):
|
2021-08-12 14:50:09 +03:00
|
|
|
"""I;16L access, with conversion"""
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2014-01-07 11:15:00 +04:00
|
|
|
def _post_init(self, *args, **kwargs):
|
|
|
|
self.pixels = ffi.cast("struct Pixel_I16 **", self.image)
|
|
|
|
|
|
|
|
def get_pixel(self, x, y):
|
|
|
|
pixel = self.pixels[y][x]
|
|
|
|
return pixel.l + pixel.r * 256
|
|
|
|
|
|
|
|
def set_pixel(self, x, y, color):
|
|
|
|
pixel = self.pixels[y][x]
|
|
|
|
try:
|
|
|
|
color = min(color, 65535)
|
2015-04-24 11:24:52 +03:00
|
|
|
except TypeError:
|
2014-01-07 11:15:00 +04:00
|
|
|
color = min(color[0], 65535)
|
|
|
|
|
2023-02-23 16:45:11 +03:00
|
|
|
pixel.l = color & 0xFF
|
2014-01-07 11:15:00 +04:00
|
|
|
pixel.r = color >> 8
|
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2014-01-07 11:15:00 +04:00
|
|
|
class _PyAccessI16_B(PyAccess):
|
2021-08-12 14:50:09 +03:00
|
|
|
"""I;16B access, with conversion"""
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2014-01-07 11:15:00 +04:00
|
|
|
def _post_init(self, *args, **kwargs):
|
|
|
|
self.pixels = ffi.cast("struct Pixel_I16 **", self.image)
|
|
|
|
|
|
|
|
def get_pixel(self, x, y):
|
|
|
|
pixel = self.pixels[y][x]
|
|
|
|
return pixel.l * 256 + pixel.r
|
|
|
|
|
|
|
|
def set_pixel(self, x, y, color):
|
|
|
|
pixel = self.pixels[y][x]
|
|
|
|
try:
|
|
|
|
color = min(color, 65535)
|
2018-11-17 00:51:52 +03:00
|
|
|
except Exception:
|
2014-01-07 11:15:00 +04:00
|
|
|
color = min(color[0], 65535)
|
|
|
|
|
2023-02-23 16:45:11 +03:00
|
|
|
pixel.l = color >> 8
|
2014-01-07 11:15:00 +04:00
|
|
|
pixel.r = color & 0xFF
|
2014-01-05 22:41:25 +04:00
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2014-01-09 09:18:38 +04:00
|
|
|
class _PyAccessI32_N(PyAccess):
|
2021-08-12 14:50:09 +03:00
|
|
|
"""Signed Int32 access, native endian"""
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2014-01-09 09:18:38 +04:00
|
|
|
def _post_init(self, *args, **kwargs):
|
|
|
|
self.pixels = self.image32
|
|
|
|
|
|
|
|
def get_pixel(self, x, y):
|
|
|
|
return self.pixels[y][x]
|
|
|
|
|
|
|
|
def set_pixel(self, x, y, color):
|
|
|
|
self.pixels[y][x] = color
|
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2014-01-09 09:18:38 +04:00
|
|
|
class _PyAccessI32_Swap(PyAccess):
|
2021-08-12 14:50:09 +03:00
|
|
|
"""I;32L/B access, with byteswapping conversion"""
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2014-01-09 09:18:38 +04:00
|
|
|
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)
|
|
|
|
chars[0], chars[1], chars[2], chars[3] = chars[3], chars[2], chars[1], chars[0]
|
|
|
|
return ffi.cast("int *", chars)[0]
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2014-01-09 09:18:38 +04:00
|
|
|
def get_pixel(self, x, y):
|
|
|
|
return self.reverse(self.pixels[y][x])
|
|
|
|
|
|
|
|
def set_pixel(self, x, y, color):
|
|
|
|
self.pixels[y][x] = self.reverse(color)
|
2014-01-09 08:23:20 +04:00
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2014-01-09 08:23:20 +04:00
|
|
|
class _PyAccessF(PyAccess):
|
2021-08-12 14:50:09 +03:00
|
|
|
"""32 bit float access"""
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2014-01-09 08:23:20 +04:00
|
|
|
def _post_init(self, *args, **kwargs):
|
|
|
|
self.pixels = ffi.cast("float **", self.image32)
|
|
|
|
|
|
|
|
def get_pixel(self, x, y):
|
|
|
|
return self.pixels[y][x]
|
|
|
|
|
|
|
|
def set_pixel(self, x, y, color):
|
|
|
|
try:
|
|
|
|
# not a tuple
|
|
|
|
self.pixels[y][x] = color
|
2017-03-01 12:20:18 +03:00
|
|
|
except TypeError:
|
2014-01-09 08:23:20 +04:00
|
|
|
# tuple
|
|
|
|
self.pixels[y][x] = color[0]
|
|
|
|
|
|
|
|
|
2014-01-05 22:41:25 +04:00
|
|
|
mode_map = {
|
|
|
|
"1": _PyAccess8,
|
|
|
|
"L": _PyAccess8,
|
2014-01-07 10:51:31 +04:00
|
|
|
"P": _PyAccess8,
|
2022-12-28 08:55:59 +03:00
|
|
|
"I;16N": _PyAccessI16_N,
|
2016-05-10 21:46:40 +03:00
|
|
|
"LA": _PyAccess32_2,
|
2014-01-07 10:51:31 +04:00
|
|
|
"La": _PyAccess32_2,
|
|
|
|
"PA": _PyAccess32_2,
|
2014-01-05 22:41:25 +04:00
|
|
|
"RGB": _PyAccess32_3,
|
|
|
|
"LAB": _PyAccess32_3,
|
2014-07-22 09:31:32 +04:00
|
|
|
"HSV": _PyAccess32_3,
|
2014-01-05 22:41:25 +04:00
|
|
|
"YCbCr": _PyAccess32_3,
|
|
|
|
"RGBA": _PyAccess32_4,
|
|
|
|
"RGBa": _PyAccess32_4,
|
|
|
|
"RGBX": _PyAccess32_4,
|
|
|
|
"CMYK": _PyAccess32_4,
|
2014-01-09 08:23:20 +04:00
|
|
|
"F": _PyAccessF,
|
2014-01-09 09:44:18 +04:00
|
|
|
"I": _PyAccessI32_N,
|
2014-01-05 22:41:25 +04:00
|
|
|
}
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2014-01-07 11:15:00 +04:00
|
|
|
if sys.byteorder == "little":
|
|
|
|
mode_map["I;16"] = _PyAccessI16_N
|
|
|
|
mode_map["I;16L"] = _PyAccessI16_N
|
|
|
|
mode_map["I;16B"] = _PyAccessI16_B
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2014-01-09 09:18:38 +04:00
|
|
|
mode_map["I;32L"] = _PyAccessI32_N
|
|
|
|
mode_map["I;32B"] = _PyAccessI32_Swap
|
2014-01-07 11:15:00 +04:00
|
|
|
else:
|
|
|
|
mode_map["I;16"] = _PyAccessI16_L
|
|
|
|
mode_map["I;16L"] = _PyAccessI16_L
|
|
|
|
mode_map["I;16B"] = _PyAccessI16_N
|
2014-01-09 09:44:18 +04:00
|
|
|
|
2014-01-09 09:18:38 +04:00
|
|
|
mode_map["I;32L"] = _PyAccessI32_Swap
|
|
|
|
mode_map["I;32B"] = _PyAccessI32_N
|
2014-01-05 22:41:25 +04:00
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2014-01-05 22:41:25 +04:00
|
|
|
def new(img, readonly=False):
|
|
|
|
access_type = mode_map.get(img.mode, None)
|
|
|
|
if not access_type:
|
2015-02-06 21:58:07 +03:00
|
|
|
logger.debug("PyAccess Not Implemented: %s", img.mode)
|
2014-01-05 22:41:25 +04:00
|
|
|
return None
|
|
|
|
return access_type(img, readonly)
|