Pillow/src/PIL/IcnsImagePlugin.py

400 lines
12 KiB
Python
Raw Normal View History

2010-07-31 06:52:47 +04:00
#
# The Python Imaging Library.
# $Id$
#
2016-09-23 14:12:03 +03:00
# macOS icns file decoder, based on icns.py by Bob Ippolito.
2010-07-31 06:52:47 +04:00
#
# history:
# 2004-10-09 fl Turned into a PIL plugin; removed 2.3 dependencies.
# 2020-04-04 Allow saving on all operating systems.
2010-07-31 06:52:47 +04:00
#
# Copyright (c) 2004 by Bob Ippolito.
# Copyright (c) 2004 by Secret Labs.
# Copyright (c) 2004 by Fredrik Lundh.
2014-03-24 20:10:23 +04:00
# Copyright (c) 2014 by Alastair Houghton.
# Copyright (c) 2020 by Pan Jing.
2010-07-31 06:52:47 +04:00
#
# See the README file for information on usage and redistribution.
#
2014-08-26 17:47:10 +04:00
import io
2015-04-23 13:25:45 +03:00
import os
2014-08-26 17:47:10 +04:00
import struct
2015-04-23 13:25:45 +03:00
import sys
2010-07-31 06:52:47 +04:00
2019-10-12 16:29:10 +03:00
from PIL import Image, ImageFile, PngImagePlugin, features
2019-10-12 16:29:10 +03:00
enable_jpeg2k = features.check_codec("jpg_2000")
if enable_jpeg2k:
2014-03-28 13:30:16 +04:00
from PIL import Jpeg2KImagePlugin
2021-06-29 14:08:26 +03:00
MAGIC = b"icns"
2010-07-31 06:52:47 +04:00
HEADERSIZE = 8
2014-08-26 17:47:10 +04:00
2010-07-31 06:52:47 +04:00
def nextheader(fobj):
2019-03-21 16:28:20 +03:00
return struct.unpack(">4sI", fobj.read(HEADERSIZE))
2010-07-31 06:52:47 +04:00
2014-08-26 17:47:10 +04:00
def read_32t(fobj, start_length, size):
2010-07-31 06:52:47 +04:00
# The 128x128 icon seems to have an extra header for some reason.
(start, length) = start_length
2010-07-31 06:52:47 +04:00
fobj.seek(start)
sig = fobj.read(4)
2019-03-21 16:28:20 +03:00
if sig != b"\x00\x00\x00\x00":
msg = "Unknown signature, expecting 0x00000000"
raise SyntaxError(msg)
return read_32(fobj, (start + 4, length - 4), size)
2010-07-31 06:52:47 +04:00
2014-08-26 17:47:10 +04:00
def read_32(fobj, start_length, size):
2010-07-31 06:52:47 +04:00
"""
Read a 32bit RGB icon resource. Seems to be either uncompressed or
an RLE packbits-like scheme.
"""
(start, length) = start_length
2010-07-31 06:52:47 +04:00
fobj.seek(start)
2014-03-24 20:10:23 +04:00
pixel_size = (size[0] * size[2], size[1] * size[2])
sizesq = pixel_size[0] * pixel_size[1]
2010-07-31 06:52:47 +04:00
if length == sizesq * 3:
# uncompressed ("RGBRGBGB")
indata = fobj.read(length)
2014-03-24 20:10:23 +04:00
im = Image.frombuffer("RGB", pixel_size, indata, "raw", "RGB", 0, 1)
2010-07-31 06:52:47 +04:00
else:
# decode image
2014-03-24 20:10:23 +04:00
im = Image.new("RGB", pixel_size, None)
2010-07-31 06:52:47 +04:00
for band_ix in range(3):
data = []
bytesleft = sizesq
while bytesleft > 0:
byte = fobj.read(1)
if not byte:
break
byte = byte[0]
2010-07-31 06:52:47 +04:00
if byte & 0x80:
blocksize = byte - 125
byte = fobj.read(1)
for i in range(blocksize):
data.append(byte)
else:
blocksize = byte + 1
data.append(fobj.read(blocksize))
bytesleft -= blocksize
2010-07-31 06:52:47 +04:00
if bytesleft <= 0:
break
if bytesleft != 0:
msg = f"Error reading channel [{repr(bytesleft)} left]"
raise SyntaxError(msg)
2019-03-21 16:28:20 +03:00
band = Image.frombuffer("L", pixel_size, b"".join(data), "raw", "L", 0, 1)
2010-07-31 06:52:47 +04:00
im.im.putband(band.im, band_ix)
return {"RGB": im}
2014-08-26 17:47:10 +04:00
def read_mk(fobj, start_length, size):
2010-07-31 06:52:47 +04:00
# Alpha masks seem to be uncompressed
2015-04-24 11:24:52 +03:00
start = start_length[0]
2010-07-31 06:52:47 +04:00
fobj.seek(start)
2014-03-24 20:10:23 +04:00
pixel_size = (size[0] * size[2], size[1] * size[2])
sizesq = pixel_size[0] * pixel_size[1]
2019-03-21 16:28:20 +03:00
band = Image.frombuffer("L", pixel_size, fobj.read(sizesq), "raw", "L", 0, 1)
2010-07-31 06:52:47 +04:00
return {"A": band}
2014-08-26 17:47:10 +04:00
2014-03-24 20:10:23 +04:00
def read_png_or_jpeg2000(fobj, start_length, size):
(start, length) = start_length
fobj.seek(start)
sig = fobj.read(12)
2019-03-21 16:28:20 +03:00
if sig[:8] == b"\x89PNG\x0d\x0a\x1a\x0a":
2014-03-24 20:10:23 +04:00
fobj.seek(start)
im = PngImagePlugin.PngImageFile(fobj)
Image._decompression_bomb_check(im.size)
2014-03-24 20:10:23 +04:00
return {"RGBA": im}
2019-03-21 16:28:20 +03:00
elif (
sig[:4] == b"\xff\x4f\xff\x51"
or sig[:4] == b"\x0d\x0a\x87\x0a"
or sig == b"\x00\x00\x00\x0cjP \x0d\x0a\x87\x0a"
):
if not enable_jpeg2k:
msg = (
2019-03-21 16:28:20 +03:00
"Unsupported icon subimage format (rebuild PIL "
"with JPEG 2000 support to fix this)"
)
raise ValueError(msg)
2014-03-24 20:10:23 +04:00
# j2k, jpc or j2c
fobj.seek(start)
jp2kstream = fobj.read(length)
f = io.BytesIO(jp2kstream)
im = Jpeg2KImagePlugin.Jpeg2KImageFile(f)
Image._decompression_bomb_check(im.size)
2019-03-21 16:28:20 +03:00
if im.mode != "RGBA":
im = im.convert("RGBA")
return {"RGBA": im}
else:
msg = "Unsupported icon subimage format"
raise ValueError(msg)
2014-03-24 20:10:23 +04:00
2014-08-26 17:47:10 +04:00
class IcnsFile:
2010-07-31 06:52:47 +04:00
SIZES = {
2019-03-21 16:28:20 +03:00
(512, 512, 2): [(b"ic10", read_png_or_jpeg2000)],
(512, 512, 1): [(b"ic09", read_png_or_jpeg2000)],
(256, 256, 2): [(b"ic14", read_png_or_jpeg2000)],
(256, 256, 1): [(b"ic08", read_png_or_jpeg2000)],
(128, 128, 2): [(b"ic13", read_png_or_jpeg2000)],
2014-03-24 20:10:23 +04:00
(128, 128, 1): [
2019-03-21 16:28:20 +03:00
(b"ic07", read_png_or_jpeg2000),
(b"it32", read_32t),
(b"t8mk", read_mk),
2010-07-31 06:52:47 +04:00
],
2019-03-21 16:28:20 +03:00
(64, 64, 1): [(b"icp6", read_png_or_jpeg2000)],
(32, 32, 2): [(b"ic12", read_png_or_jpeg2000)],
(48, 48, 1): [(b"ih32", read_32), (b"h8mk", read_mk)],
2014-03-24 20:10:23 +04:00
(32, 32, 1): [
2019-03-21 16:28:20 +03:00
(b"icp5", read_png_or_jpeg2000),
(b"il32", read_32),
(b"l8mk", read_mk),
2014-03-24 20:10:23 +04:00
],
2019-03-21 16:28:20 +03:00
(16, 16, 2): [(b"ic11", read_png_or_jpeg2000)],
2014-03-24 20:10:23 +04:00
(16, 16, 1): [
2019-03-21 16:28:20 +03:00
(b"icp4", read_png_or_jpeg2000),
(b"is32", read_32),
(b"s8mk", read_mk),
2010-07-31 06:52:47 +04:00
],
}
def __init__(self, fobj):
"""
fobj is a file-like object as an icns resource
"""
# signature : (start, length)
self.dct = dct = {}
self.fobj = fobj
sig, filesize = nextheader(fobj)
2022-02-26 09:53:27 +03:00
if not _accept(sig):
msg = "not an icns file"
raise SyntaxError(msg)
2010-07-31 06:52:47 +04:00
i = HEADERSIZE
while i < filesize:
sig, blocksize = nextheader(fobj)
if blocksize <= 0:
msg = "invalid block header"
raise SyntaxError(msg)
i += HEADERSIZE
blocksize -= HEADERSIZE
2010-07-31 06:52:47 +04:00
dct[sig] = (i, blocksize)
fobj.seek(blocksize, io.SEEK_CUR)
i += blocksize
2010-07-31 06:52:47 +04:00
def itersizes(self):
sizes = []
for size, fmts in self.SIZES.items():
for fmt, reader in fmts:
if fmt in self.dct:
2010-07-31 06:52:47 +04:00
sizes.append(size)
break
return sizes
def bestsize(self):
sizes = self.itersizes()
if not sizes:
msg = "No 32bit icon resources found"
raise SyntaxError(msg)
2010-07-31 06:52:47 +04:00
return max(sizes)
def dataforsize(self, size):
"""
Get an icon resource as {channel: array}. Note that
the arrays are bottom-up like windows bitmaps and will likely
need to be flipped or transposed in some way.
"""
dct = {}
for code, reader in self.SIZES[size]:
desc = self.dct.get(code)
if desc is not None:
dct.update(reader(self.fobj, desc, size))
return dct
def getimage(self, size=None):
if size is None:
size = self.bestsize()
2014-03-24 20:10:23 +04:00
if len(size) == 2:
size = (size[0], size[1], 1)
2010-07-31 06:52:47 +04:00
channels = self.dataforsize(size)
2014-03-24 20:10:23 +04:00
2019-03-21 16:28:20 +03:00
im = channels.get("RGBA", None)
2014-03-24 20:10:23 +04:00
if im:
return im
2014-08-26 17:47:10 +04:00
2010-07-31 06:52:47 +04:00
im = channels.get("RGB").copy()
try:
im.putalpha(channels["A"])
except KeyError:
pass
return im
2014-08-26 17:47:10 +04:00
2010-07-31 06:52:47 +04:00
##
# Image plugin for Mac OS icons.
2019-03-21 16:28:20 +03:00
2010-07-31 06:52:47 +04:00
class IcnsImageFile(ImageFile.ImageFile):
"""
2015-04-23 10:00:21 +03:00
PIL image support for Mac OS .icns files.
2010-07-31 06:52:47 +04:00
Chooses the best resolution, but will possibly load
a different size image if you mutate the size attribute
before calling 'load'.
The info dictionary has a key 'sizes' that is a list
of sizes that the icns file has.
"""
format = "ICNS"
format_description = "Mac OS icns resource"
def _open(self):
self.icns = IcnsFile(self.fp)
2019-03-21 16:28:20 +03:00
self.mode = "RGBA"
self.info["sizes"] = self.icns.itersizes()
2014-03-24 20:10:23 +04:00
self.best_size = self.icns.bestsize()
2019-03-21 16:28:20 +03:00
self.size = (
self.best_size[0] * self.best_size[2],
self.best_size[1] * self.best_size[2],
)
2010-07-31 06:52:47 +04:00
@property
def size(self):
return self._size
@size.setter
def size(self, value):
info_size = value
2019-03-21 16:28:20 +03:00
if info_size not in self.info["sizes"] and len(info_size) == 2:
info_size = (info_size[0], info_size[1], 1)
2019-03-21 16:28:20 +03:00
if (
info_size not in self.info["sizes"]
and len(info_size) == 3
and info_size[2] == 1
):
simple_sizes = [
(size[0] * size[2], size[1] * size[2]) for size in self.info["sizes"]
]
if value in simple_sizes:
2019-03-21 16:28:20 +03:00
info_size = self.info["sizes"][simple_sizes.index(value)]
if info_size not in self.info["sizes"]:
msg = "This is not one of the allowed sizes of this image"
raise ValueError(msg)
self._size = value
2010-07-31 06:52:47 +04:00
def load(self):
2014-03-24 20:10:23 +04:00
if len(self.size) == 3:
self.best_size = self.size
2019-03-21 16:28:20 +03:00
self.size = (
self.best_size[0] * self.best_size[2],
self.best_size[1] * self.best_size[2],
)
2014-03-24 20:10:23 +04:00
px = Image.Image.load(self)
2022-03-03 14:10:19 +03:00
if self.im is not None and self.im.size == self.size:
2019-05-29 14:14:18 +03:00
# Already loaded
return px
2010-07-31 06:52:47 +04:00
self.load_prepare()
# This is likely NOT the best way to do it, but whatever.
2014-03-24 20:10:23 +04:00
im = self.icns.getimage(self.best_size)
# If this is a PNG or JPEG 2000, it won't be loaded yet
px = im.load()
2014-08-26 17:47:10 +04:00
2010-07-31 06:52:47 +04:00
self.im = im.im
self.mode = im.mode
self.size = im.size
return px
2010-07-31 06:52:47 +04:00
2015-04-23 13:25:45 +03:00
2015-04-12 05:58:46 +03:00
def _save(im, fp, filename):
2015-04-23 10:00:21 +03:00
"""
Saves the image as a series of PNG files,
that are then combined into a .icns file.
2015-04-23 10:00:21 +03:00
"""
if hasattr(fp, "flush"):
2015-04-12 05:58:46 +03:00
fp.flush()
2015-04-23 13:25:45 +03:00
sizes = {
b"ic07": 128,
b"ic08": 256,
b"ic09": 512,
b"ic10": 1024,
b"ic11": 32,
b"ic12": 64,
b"ic13": 256,
b"ic14": 512,
}
2020-04-04 06:54:49 +03:00
provided_images = {im.width: im for im in im.encoderinfo.get("append_images", [])}
size_streams = {}
for size in set(sizes.values()):
image = (
provided_images[size]
if size in provided_images
else im.resize((size, size))
)
temp = io.BytesIO()
image.save(temp, "png")
size_streams[size] = temp.getvalue()
entries = []
for type, size in sizes.items():
stream = size_streams[size]
2021-11-20 06:17:42 +03:00
entries.append(
{"type": type, "size": HEADERSIZE + len(stream), "stream": stream}
)
# Header
2021-06-29 14:08:26 +03:00
fp.write(MAGIC)
2021-11-20 06:17:42 +03:00
file_length = HEADERSIZE # Header
file_length += HEADERSIZE + 8 * len(entries) # TOC
file_length += sum(entry["size"] for entry in entries)
fp.write(struct.pack(">i", file_length))
# TOC
2021-06-29 14:08:26 +03:00
fp.write(b"TOC ")
2021-06-29 13:20:34 +03:00
fp.write(struct.pack(">i", HEADERSIZE + len(entries) * HEADERSIZE))
2021-06-29 13:18:55 +03:00
for entry in entries:
2021-06-29 14:08:26 +03:00
fp.write(entry["type"])
2021-11-20 06:17:42 +03:00
fp.write(struct.pack(">i", entry["size"]))
# Data
2021-06-29 13:18:55 +03:00
for entry in entries:
2021-06-29 14:08:26 +03:00
fp.write(entry["type"])
2021-11-20 06:17:42 +03:00
fp.write(struct.pack(">i", entry["size"]))
2021-06-29 13:18:55 +03:00
fp.write(entry["stream"])
if hasattr(fp, "flush"):
fp.flush()
2015-04-12 05:58:46 +03:00
2018-03-03 12:54:00 +03:00
def _accept(prefix):
2021-06-29 13:52:35 +03:00
return prefix[:4] == MAGIC
Image.register_open(IcnsImageFile.format, IcnsImageFile, _accept)
2019-03-21 16:28:20 +03:00
Image.register_extension(IcnsImageFile.format, ".icns")
2010-07-31 06:52:47 +04:00
Image.register_save(IcnsImageFile.format, _save)
Image.register_mime(IcnsImageFile.format, "image/icns")
2015-04-12 05:58:46 +03:00
2019-03-21 16:28:20 +03:00
if __name__ == "__main__":
2018-01-06 13:51:45 +03:00
if len(sys.argv) < 2:
2021-05-08 05:37:06 +03:00
print("Syntax: python3 IcnsImagePlugin.py [file]")
2018-01-06 13:51:45 +03:00
sys.exit()
2020-02-17 14:12:46 +03:00
with open(sys.argv[1], "rb") as fp:
imf = IcnsImageFile(fp)
for size in imf.info["sizes"]:
imf.size = size
imf.save("out-%s-%s-%s.png" % size)
2020-02-18 12:49:05 +03:00
with Image.open(sys.argv[1]) as im:
im.save("out.png")
2020-02-17 14:12:46 +03:00
if sys.platform == "windows":
os.startfile("out.png")