2014-03-12 20:25:59 +04:00
|
|
|
#
|
|
|
|
# The Python Imaging Library
|
|
|
|
# $Id$
|
|
|
|
#
|
|
|
|
# JPEG2000 file handling
|
|
|
|
#
|
|
|
|
# History:
|
|
|
|
# 2014-03-12 ajh Created
|
2021-06-30 15:53:11 +03:00
|
|
|
# 2021-06-30 rogermb Extract dpi information from the 'resc' header box
|
2014-03-12 20:25:59 +04:00
|
|
|
#
|
|
|
|
# Copyright (c) 2014 Coriolis Systems Limited
|
|
|
|
# Copyright (c) 2014 Alastair Houghton
|
|
|
|
#
|
|
|
|
# See the README file for information on usage and redistribution.
|
|
|
|
#
|
2023-12-21 14:13:31 +03:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2014-03-12 20:25:59 +04:00
|
|
|
import io
|
2019-07-06 23:40:53 +03:00
|
|
|
import os
|
|
|
|
import struct
|
2024-06-09 00:12:05 +03:00
|
|
|
from typing import IO
|
2019-07-06 23:40:53 +03:00
|
|
|
|
2024-03-12 11:30:23 +03:00
|
|
|
from . import Image, ImageFile, ImagePalette, _binary
|
2014-03-12 20:25:59 +04:00
|
|
|
|
2014-04-14 13:51:12 +04:00
|
|
|
|
2021-06-30 04:07:29 +03:00
|
|
|
class BoxReader:
|
|
|
|
"""
|
|
|
|
A small helper class to read fields stored in JPEG2000 header boxes
|
|
|
|
and to easily step into and read sub-boxes.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, fp, length=-1):
|
|
|
|
self.fp = fp
|
|
|
|
self.has_length = length >= 0
|
|
|
|
self.length = length
|
|
|
|
self.remaining_in_box = -1
|
|
|
|
|
2024-06-04 13:37:09 +03:00
|
|
|
def _can_read(self, num_bytes: int) -> bool:
|
2021-07-25 13:36:54 +03:00
|
|
|
if self.has_length and self.fp.tell() + num_bytes > self.length:
|
|
|
|
# Outside box: ensure we don't read past the known file length
|
|
|
|
return False
|
2021-06-30 04:07:29 +03:00
|
|
|
if self.remaining_in_box >= 0:
|
|
|
|
# Inside box contents: ensure read does not go past box boundaries
|
|
|
|
return num_bytes <= self.remaining_in_box
|
|
|
|
else:
|
|
|
|
return True # No length known, just read
|
|
|
|
|
2024-06-04 13:37:09 +03:00
|
|
|
def _read_bytes(self, num_bytes: int) -> bytes:
|
2021-06-30 04:07:29 +03:00
|
|
|
if not self._can_read(num_bytes):
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = "Not enough data in header"
|
|
|
|
raise SyntaxError(msg)
|
2021-06-30 04:07:29 +03:00
|
|
|
|
|
|
|
data = self.fp.read(num_bytes)
|
|
|
|
if len(data) < num_bytes:
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = f"Expected to read {num_bytes} bytes but only got {len(data)}."
|
|
|
|
raise OSError(msg)
|
2021-06-30 04:07:29 +03:00
|
|
|
|
|
|
|
if self.remaining_in_box > 0:
|
|
|
|
self.remaining_in_box -= num_bytes
|
|
|
|
return data
|
|
|
|
|
|
|
|
def read_fields(self, field_format):
|
|
|
|
size = struct.calcsize(field_format)
|
|
|
|
data = self._read_bytes(size)
|
|
|
|
return struct.unpack(field_format, data)
|
|
|
|
|
2024-05-13 11:47:51 +03:00
|
|
|
def read_boxes(self) -> BoxReader:
|
2021-06-30 04:07:29 +03:00
|
|
|
size = self.remaining_in_box
|
|
|
|
data = self._read_bytes(size)
|
|
|
|
return BoxReader(io.BytesIO(data), size)
|
|
|
|
|
2024-05-13 11:47:51 +03:00
|
|
|
def has_next_box(self) -> bool:
|
2021-06-30 04:07:29 +03:00
|
|
|
if self.has_length:
|
|
|
|
return self.fp.tell() + self.remaining_in_box < self.length
|
|
|
|
else:
|
|
|
|
return True
|
|
|
|
|
2024-06-04 13:37:09 +03:00
|
|
|
def next_box_type(self) -> bytes:
|
2021-06-30 04:07:29 +03:00
|
|
|
# Skip the rest of the box if it has not been read
|
|
|
|
if self.remaining_in_box > 0:
|
|
|
|
self.fp.seek(self.remaining_in_box, os.SEEK_CUR)
|
|
|
|
self.remaining_in_box = -1
|
|
|
|
|
|
|
|
# Read the length and type of the next box
|
|
|
|
lbox, tbox = self.read_fields(">I4s")
|
|
|
|
if lbox == 1:
|
|
|
|
lbox = self.read_fields(">Q")[0]
|
|
|
|
hlen = 16
|
|
|
|
else:
|
|
|
|
hlen = 8
|
|
|
|
|
|
|
|
if lbox < hlen or not self._can_read(lbox - hlen):
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = "Invalid header length"
|
|
|
|
raise SyntaxError(msg)
|
2021-06-30 04:07:29 +03:00
|
|
|
|
|
|
|
self.remaining_in_box = lbox - hlen
|
|
|
|
return tbox
|
|
|
|
|
|
|
|
|
2014-03-12 20:25:59 +04:00
|
|
|
def _parse_codestream(fp):
|
|
|
|
"""Parse the JPEG 2000 codestream to extract the size and component
|
|
|
|
count from the SIZ marker segment, returning a PIL (size, mode) tuple."""
|
2014-04-14 13:51:12 +04:00
|
|
|
|
2014-03-12 20:25:59 +04:00
|
|
|
hdr = fp.read(2)
|
2023-03-29 12:14:29 +03:00
|
|
|
lsiz = _binary.i16be(hdr)
|
2014-03-12 20:25:59 +04:00
|
|
|
siz = hdr + fp.read(lsiz - 2)
|
|
|
|
lsiz, rsiz, xsiz, ysiz, xosiz, yosiz, _, _, _, _, csiz = struct.unpack_from(
|
2018-06-24 20:00:22 +03:00
|
|
|
">HHIIIIIIIIH", siz
|
|
|
|
)
|
2014-03-12 20:25:59 +04:00
|
|
|
|
|
|
|
size = (xsiz - xosiz, ysiz - yosiz)
|
|
|
|
if csiz == 1:
|
2024-03-25 10:23:52 +03:00
|
|
|
ssiz = struct.unpack_from(">B", siz, 38)
|
2024-03-25 09:39:21 +03:00
|
|
|
if (ssiz[0] & 0x7F) + 1 > 8:
|
2014-06-25 22:06:56 +04:00
|
|
|
mode = "I;16"
|
2014-06-25 19:13:33 +04:00
|
|
|
else:
|
|
|
|
mode = "L"
|
2014-03-12 20:25:59 +04:00
|
|
|
elif csiz == 2:
|
|
|
|
mode = "LA"
|
|
|
|
elif csiz == 3:
|
|
|
|
mode = "RGB"
|
|
|
|
elif csiz == 4:
|
2014-04-16 17:07:43 +04:00
|
|
|
mode = "RGBA"
|
2014-03-12 20:25:59 +04:00
|
|
|
else:
|
|
|
|
mode = None
|
2014-04-14 13:51:12 +04:00
|
|
|
|
2022-04-10 19:25:40 +03:00
|
|
|
return size, mode
|
2014-03-12 20:25:59 +04:00
|
|
|
|
2014-04-14 13:51:12 +04:00
|
|
|
|
2021-06-30 15:53:11 +03:00
|
|
|
def _res_to_dpi(num, denom, exp):
|
|
|
|
"""Convert JPEG2000's (numerator, denominator, exponent-base-10) resolution,
|
|
|
|
calculated as (num / denom) * 10^exp and stored in dots per meter,
|
|
|
|
to floating-point dots per inch."""
|
2021-08-01 11:38:56 +03:00
|
|
|
if denom != 0:
|
2022-03-04 08:42:24 +03:00
|
|
|
return (254 * num * (10**exp)) / (10000 * denom)
|
2021-06-30 15:53:11 +03:00
|
|
|
|
|
|
|
|
2014-03-12 20:25:59 +04:00
|
|
|
def _parse_jp2_header(fp):
|
2021-06-30 15:53:11 +03:00
|
|
|
"""Parse the JP2 header box to extract size, component count,
|
|
|
|
color space information, and optionally DPI information,
|
|
|
|
returning a (size, mode, mimetype, dpi) tuple."""
|
2014-04-14 13:51:12 +04:00
|
|
|
|
2014-03-12 20:25:59 +04:00
|
|
|
# Find the JP2 header box
|
2021-06-30 04:07:29 +03:00
|
|
|
reader = BoxReader(fp)
|
2014-03-12 20:25:59 +04:00
|
|
|
header = None
|
2019-01-02 07:39:39 +03:00
|
|
|
mimetype = None
|
2021-06-30 04:07:29 +03:00
|
|
|
while reader.has_next_box():
|
|
|
|
tbox = reader.next_box_type()
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2014-03-12 20:25:59 +04:00
|
|
|
if tbox == b"jp2h":
|
2021-06-30 04:07:29 +03:00
|
|
|
header = reader.read_boxes()
|
2014-03-12 20:25:59 +04:00
|
|
|
break
|
2019-01-02 07:39:39 +03:00
|
|
|
elif tbox == b"ftyp":
|
2021-06-30 04:07:29 +03:00
|
|
|
if reader.read_fields(">4s")[0] == b"jpx ":
|
2019-01-02 07:39:39 +03:00
|
|
|
mimetype = "image/jpx"
|
2014-03-12 20:25:59 +04:00
|
|
|
|
|
|
|
size = None
|
|
|
|
mode = None
|
2014-06-25 19:13:33 +04:00
|
|
|
bpc = None
|
2016-03-26 23:41:00 +03:00
|
|
|
nc = None
|
2021-06-30 15:53:11 +03:00
|
|
|
dpi = None # 2-tuple of DPI info, or None
|
2024-03-12 11:30:23 +03:00
|
|
|
palette = None
|
2016-09-03 05:23:42 +03:00
|
|
|
|
2021-06-30 04:07:29 +03:00
|
|
|
while header.has_next_box():
|
|
|
|
tbox = header.next_box_type()
|
2014-03-12 20:25:59 +04:00
|
|
|
|
|
|
|
if tbox == b"ihdr":
|
2021-08-04 18:06:01 +03:00
|
|
|
height, width, nc, bpc = header.read_fields(">IIHB")
|
2014-03-12 20:25:59 +04:00
|
|
|
size = (width, height)
|
2021-08-04 18:06:01 +03:00
|
|
|
if nc == 1 and (bpc & 0x7F) > 8:
|
|
|
|
mode = "I;16"
|
|
|
|
elif nc == 1:
|
|
|
|
mode = "L"
|
|
|
|
elif nc == 2:
|
|
|
|
mode = "LA"
|
|
|
|
elif nc == 3:
|
|
|
|
mode = "RGB"
|
|
|
|
elif nc == 4:
|
|
|
|
mode = "RGBA"
|
2024-04-06 05:40:39 +03:00
|
|
|
elif tbox == b"colr" and nc == 4:
|
|
|
|
meth, _, _, enumcs = header.read_fields(">BBBI")
|
|
|
|
if meth == 1 and enumcs == 12:
|
|
|
|
mode = "CMYK"
|
2024-03-12 11:30:23 +03:00
|
|
|
elif tbox == b"pclr" and mode in ("L", "LA"):
|
|
|
|
ne, npc = header.read_fields(">HB")
|
|
|
|
bitdepths = header.read_fields(">" + ("B" * npc))
|
|
|
|
if max(bitdepths) <= 8:
|
|
|
|
palette = ImagePalette.ImagePalette()
|
|
|
|
for i in range(ne):
|
|
|
|
palette.getcolor(header.read_fields(">" + ("B" * npc)))
|
|
|
|
mode = "P" if mode == "L" else "PA"
|
2021-06-30 15:53:11 +03:00
|
|
|
elif tbox == b"res ":
|
|
|
|
res = header.read_boxes()
|
|
|
|
while res.has_next_box():
|
|
|
|
tres = res.next_box_type()
|
|
|
|
if tres == b"resc":
|
|
|
|
vrcn, vrcd, hrcn, hrcd, vrce, hrce = res.read_fields(">HHHHBB")
|
|
|
|
hres = _res_to_dpi(hrcn, hrcd, hrce)
|
|
|
|
vres = _res_to_dpi(vrcn, vrcd, vrce)
|
2021-08-01 11:38:56 +03:00
|
|
|
if hres is not None and vres is not None:
|
|
|
|
dpi = (hres, vres)
|
2021-08-01 11:39:35 +03:00
|
|
|
break
|
2014-03-12 20:25:59 +04:00
|
|
|
|
2016-03-26 23:41:00 +03:00
|
|
|
if size is None or mode is None:
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = "Malformed JP2 header"
|
|
|
|
raise SyntaxError(msg)
|
2016-09-03 05:23:42 +03:00
|
|
|
|
2024-03-12 11:30:23 +03:00
|
|
|
return size, mode, mimetype, dpi, palette
|
2014-03-12 20:25:59 +04:00
|
|
|
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2014-03-12 20:25:59 +04:00
|
|
|
##
|
|
|
|
# Image plugin for JPEG2000 images.
|
|
|
|
|
2014-06-25 19:13:33 +04:00
|
|
|
|
2014-03-12 20:25:59 +04:00
|
|
|
class Jpeg2KImageFile(ImageFile.ImageFile):
|
|
|
|
format = "JPEG2000"
|
|
|
|
format_description = "JPEG 2000 (ISO 15444)"
|
|
|
|
|
2024-05-11 03:48:09 +03:00
|
|
|
def _open(self) -> None:
|
2014-03-12 20:25:59 +04:00
|
|
|
sig = self.fp.read(4)
|
|
|
|
if sig == b"\xff\x4f\xff\x51":
|
|
|
|
self.codec = "j2k"
|
2023-07-29 02:28:18 +03:00
|
|
|
self._size, self._mode = _parse_codestream(self.fp)
|
2014-03-12 20:25:59 +04:00
|
|
|
else:
|
|
|
|
sig = sig + self.fp.read(8)
|
2014-04-14 13:51:12 +04:00
|
|
|
|
2014-03-12 20:25:59 +04:00
|
|
|
if sig == b"\x00\x00\x00\x0cjP \x0d\x0a\x87\x0a":
|
|
|
|
self.codec = "jp2"
|
2019-01-02 07:39:39 +03:00
|
|
|
header = _parse_jp2_header(self.fp)
|
2024-03-12 11:30:23 +03:00
|
|
|
self._size, self._mode, self.custom_mimetype, dpi, self.palette = header
|
2021-06-30 15:53:11 +03:00
|
|
|
if dpi is not None:
|
|
|
|
self.info["dpi"] = dpi
|
2023-03-09 05:36:22 +03:00
|
|
|
if self.fp.read(12).endswith(b"jp2c\xff\x4f\xff\x51"):
|
|
|
|
self._parse_comment()
|
2014-03-12 20:25:59 +04:00
|
|
|
else:
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = "not a JPEG 2000 file"
|
|
|
|
raise SyntaxError(msg)
|
2014-04-14 13:51:12 +04:00
|
|
|
|
2014-03-12 20:25:59 +04:00
|
|
|
if self.size is None or self.mode is None:
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = "unable to determine size/mode"
|
|
|
|
raise SyntaxError(msg)
|
2014-04-14 13:51:12 +04:00
|
|
|
|
2020-03-24 11:19:46 +03:00
|
|
|
self._reduce = 0
|
2014-03-12 20:25:59 +04:00
|
|
|
self.layers = 0
|
2014-03-19 16:16:14 +04:00
|
|
|
|
2014-03-14 15:21:08 +04:00
|
|
|
fd = -1
|
2014-05-27 15:43:54 +04:00
|
|
|
length = -1
|
2014-03-12 20:25:59 +04:00
|
|
|
|
2014-07-01 22:09:20 +04:00
|
|
|
try:
|
|
|
|
fd = self.fp.fileno()
|
|
|
|
length = os.fstat(fd).st_size
|
2018-11-17 00:51:52 +03:00
|
|
|
except Exception:
|
2014-07-01 22:09:20 +04:00
|
|
|
fd = -1
|
2014-03-19 16:16:14 +04:00
|
|
|
try:
|
2014-07-01 22:09:20 +04:00
|
|
|
pos = self.fp.tell()
|
2019-01-13 05:05:46 +03:00
|
|
|
self.fp.seek(0, io.SEEK_END)
|
2014-07-01 22:09:20 +04:00
|
|
|
length = self.fp.tell()
|
2019-01-13 05:05:46 +03:00
|
|
|
self.fp.seek(pos)
|
2018-11-17 00:51:52 +03:00
|
|
|
except Exception:
|
2014-05-27 15:43:54 +04:00
|
|
|
length = -1
|
2014-06-25 19:13:33 +04:00
|
|
|
|
2014-03-12 20:25:59 +04:00
|
|
|
self.tile = [
|
2019-03-21 16:28:20 +03:00
|
|
|
(
|
2014-03-12 20:25:59 +04:00
|
|
|
"jpeg2k",
|
|
|
|
(0, 0) + self.size,
|
|
|
|
0,
|
2020-03-24 11:19:46 +03:00
|
|
|
(self.codec, self._reduce, self.layers, fd, length),
|
2019-03-21 16:28:20 +03:00
|
|
|
)
|
2018-03-15 16:27:10 +03:00
|
|
|
]
|
2014-03-12 20:25:59 +04:00
|
|
|
|
2024-05-11 03:48:09 +03:00
|
|
|
def _parse_comment(self) -> None:
|
2023-03-09 05:36:22 +03:00
|
|
|
hdr = self.fp.read(2)
|
2023-03-29 12:14:29 +03:00
|
|
|
length = _binary.i16be(hdr)
|
2023-03-09 05:36:22 +03:00
|
|
|
self.fp.seek(length - 2, os.SEEK_CUR)
|
|
|
|
|
|
|
|
while True:
|
|
|
|
marker = self.fp.read(2)
|
|
|
|
if not marker:
|
|
|
|
break
|
|
|
|
typ = marker[1]
|
|
|
|
if typ in (0x90, 0xD9):
|
|
|
|
# Start of tile or end of codestream
|
|
|
|
break
|
|
|
|
hdr = self.fp.read(2)
|
2023-03-29 12:14:29 +03:00
|
|
|
length = _binary.i16be(hdr)
|
2023-03-09 05:36:22 +03:00
|
|
|
if typ == 0x64:
|
|
|
|
# Comment
|
|
|
|
self.info["comment"] = self.fp.read(length - 2)[2:]
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
self.fp.seek(length - 2, os.SEEK_CUR)
|
|
|
|
|
2020-03-24 11:19:46 +03:00
|
|
|
@property
|
|
|
|
def reduce(self):
|
2020-03-20 09:49:36 +03:00
|
|
|
# https://github.com/python-pillow/Pillow/issues/4343 found that the
|
|
|
|
# new Image 'reduce' method was shadowed by this plugin's 'reduce'
|
|
|
|
# property. This attempts to allow for both scenarios
|
2020-03-24 11:19:46 +03:00
|
|
|
return self._reduce or super().reduce
|
|
|
|
|
|
|
|
@reduce.setter
|
|
|
|
def reduce(self, value):
|
|
|
|
self._reduce = value
|
|
|
|
|
2014-03-12 20:25:59 +04:00
|
|
|
def load(self):
|
2020-03-20 10:14:08 +03:00
|
|
|
if self.tile and self._reduce:
|
2020-03-24 11:19:46 +03:00
|
|
|
power = 1 << self._reduce
|
2014-03-13 15:57:47 +04:00
|
|
|
adjust = power >> 1
|
2018-09-30 05:58:02 +03:00
|
|
|
self._size = (
|
|
|
|
int((self.size[0] + adjust) / power),
|
|
|
|
int((self.size[1] + adjust) / power),
|
|
|
|
)
|
2014-03-19 16:16:14 +04:00
|
|
|
|
|
|
|
# Update the reduce and layers settings
|
|
|
|
t = self.tile[0]
|
2020-03-24 11:19:46 +03:00
|
|
|
t3 = (t[3][0], self._reduce, self.layers, t[3][3], t[3][4])
|
2014-03-27 12:35:38 +04:00
|
|
|
self.tile = [(t[0], (0, 0) + self.size, t[2], t3)]
|
2014-04-14 13:51:12 +04:00
|
|
|
|
2016-08-08 17:36:34 +03:00
|
|
|
return ImageFile.ImageFile.load(self)
|
2014-04-14 13:51:12 +04:00
|
|
|
|
|
|
|
|
2024-04-06 05:58:53 +03:00
|
|
|
def _accept(prefix: bytes) -> bool:
|
2015-04-01 16:47:01 +03:00
|
|
|
return (
|
|
|
|
prefix[:4] == b"\xff\x4f\xff\x51"
|
|
|
|
or prefix[:12] == b"\x00\x00\x00\x0cjP \x0d\x0a\x87\x0a"
|
|
|
|
)
|
2014-03-12 20:25:59 +04:00
|
|
|
|
2014-04-14 13:51:12 +04:00
|
|
|
|
2014-03-13 22:27:16 +04:00
|
|
|
# ------------------------------------------------------------
|
|
|
|
# Save support
|
|
|
|
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2024-06-09 00:12:05 +03:00
|
|
|
def _save(im: Image.Image, fp: IO[bytes], filename: str) -> None:
|
2021-05-11 13:45:48 +03:00
|
|
|
# Get the keyword arguments
|
|
|
|
info = im.encoderinfo
|
|
|
|
|
2021-05-16 03:46:43 +03:00
|
|
|
if filename.endswith(".j2k") or info.get("no_jp2", False):
|
2014-03-13 22:27:16 +04:00
|
|
|
kind = "j2k"
|
|
|
|
else:
|
2021-05-16 03:46:43 +03:00
|
|
|
kind = "jp2"
|
2014-03-14 15:21:08 +04:00
|
|
|
|
|
|
|
offset = info.get("offset", None)
|
|
|
|
tile_offset = info.get("tile_offset", None)
|
|
|
|
tile_size = info.get("tile_size", None)
|
|
|
|
quality_mode = info.get("quality_mode", "rates")
|
|
|
|
quality_layers = info.get("quality_layers", None)
|
2018-11-16 15:31:42 +03:00
|
|
|
if quality_layers is not None and not (
|
|
|
|
isinstance(quality_layers, (list, tuple))
|
|
|
|
and all(
|
2023-11-06 16:24:39 +03:00
|
|
|
isinstance(quality_layer, (int, float)) for quality_layer in quality_layers
|
2018-11-16 15:31:42 +03:00
|
|
|
)
|
|
|
|
):
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = "quality_layers must be a sequence of numbers"
|
|
|
|
raise ValueError(msg)
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2014-03-14 15:21:08 +04:00
|
|
|
num_resolutions = info.get("num_resolutions", 0)
|
|
|
|
cblk_size = info.get("codeblock_size", None)
|
2014-03-14 19:40:30 +04:00
|
|
|
precinct_size = info.get("precinct_size", None)
|
2014-03-14 15:21:08 +04:00
|
|
|
irreversible = info.get("irreversible", False)
|
|
|
|
progression = info.get("progression", "LRCP")
|
|
|
|
cinema_mode = info.get("cinema_mode", "no")
|
2021-05-11 13:01:35 +03:00
|
|
|
mct = info.get("mct", 0)
|
2022-11-03 10:26:31 +03:00
|
|
|
signed = info.get("signed", False)
|
2023-01-31 13:42:25 +03:00
|
|
|
comment = info.get("comment")
|
2023-03-24 01:45:51 +03:00
|
|
|
if isinstance(comment, str):
|
|
|
|
comment = comment.encode()
|
2023-03-27 14:38:29 +03:00
|
|
|
plt = info.get("plt", False)
|
2014-03-14 15:21:08 +04:00
|
|
|
|
2023-03-24 01:45:51 +03:00
|
|
|
fd = -1
|
2014-03-14 15:21:08 +04:00
|
|
|
if hasattr(fp, "fileno"):
|
2014-03-19 16:16:14 +04:00
|
|
|
try:
|
|
|
|
fd = fp.fileno()
|
2018-11-17 00:51:52 +03:00
|
|
|
except Exception:
|
2014-03-19 16:16:14 +04:00
|
|
|
fd = -1
|
2014-04-14 13:51:12 +04:00
|
|
|
|
2014-03-14 15:21:08 +04:00
|
|
|
im.encoderconfig = (
|
|
|
|
offset,
|
|
|
|
tile_offset,
|
|
|
|
tile_size,
|
|
|
|
quality_mode,
|
|
|
|
quality_layers,
|
|
|
|
num_resolutions,
|
|
|
|
cblk_size,
|
2014-03-14 19:40:30 +04:00
|
|
|
precinct_size,
|
2014-03-14 15:21:08 +04:00
|
|
|
irreversible,
|
|
|
|
progression,
|
|
|
|
cinema_mode,
|
2021-05-11 13:01:35 +03:00
|
|
|
mct,
|
2022-11-03 10:26:31 +03:00
|
|
|
signed,
|
2014-03-14 15:21:08 +04:00
|
|
|
fd,
|
2023-01-19 03:37:14 +03:00
|
|
|
comment,
|
2023-03-27 14:38:29 +03:00
|
|
|
plt,
|
2014-04-14 13:51:12 +04:00
|
|
|
)
|
|
|
|
|
2014-03-13 22:27:16 +04:00
|
|
|
ImageFile._save(im, fp, [("jpeg2k", (0, 0) + im.size, 0, kind)])
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2014-04-14 13:51:12 +04:00
|
|
|
|
2014-03-12 20:25:59 +04:00
|
|
|
# ------------------------------------------------------------
|
|
|
|
# Registry stuff
|
|
|
|
|
2018-03-03 12:54:00 +03:00
|
|
|
|
2015-07-04 16:29:58 +03:00
|
|
|
Image.register_open(Jpeg2KImageFile.format, Jpeg2KImageFile, _accept)
|
|
|
|
Image.register_save(Jpeg2KImageFile.format, _save)
|
|
|
|
|
2018-06-24 15:32:25 +03:00
|
|
|
Image.register_extensions(
|
|
|
|
Jpeg2KImageFile.format, [".jp2", ".j2k", ".jpc", ".jpf", ".jpx", ".j2c"]
|
|
|
|
)
|
2015-07-04 16:29:58 +03:00
|
|
|
|
|
|
|
Image.register_mime(Jpeg2KImageFile.format, "image/jp2")
|