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.
|
2021-06-30 16:30:59 +03:00
|
|
|
# 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.
|
2020-04-04 06:22:11 +03:00
|
|
|
# Copyright (c) 2020 by Pan Jing.
|
2010-07-31 06:52:47 +04:00
|
|
|
#
|
|
|
|
# See the README file for information on usage and redistribution.
|
|
|
|
#
|
2023-12-21 14:13:31 +03:00
|
|
|
from __future__ import annotations
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2014-03-24 22:04:37 +04:00
|
|
|
import io
|
2015-04-23 13:25:45 +03:00
|
|
|
import os
|
2014-03-24 22:04:37 +04:00
|
|
|
import struct
|
2015-04-23 13:25:45 +03:00
|
|
|
import sys
|
2024-06-10 07:15:28 +03:00
|
|
|
from typing import IO
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2023-05-20 10:11:43 +03:00
|
|
|
from . import Image, ImageFile, PngImagePlugin, features
|
2019-07-06 23:40:53 +03:00
|
|
|
|
2019-10-12 16:29:10 +03:00
|
|
|
enable_jpeg2k = features.check_codec("jpg_2000")
|
2014-03-28 13:09:55 +04:00
|
|
|
if enable_jpeg2k:
|
2023-05-20 10:11:43 +03:00
|
|
|
from . import Jpeg2KImagePlugin
|
2014-03-28 13:09:55 +04:00
|
|
|
|
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
|
|
|
|
2024-07-05 20:55:23 +03:00
|
|
|
def nextheader(fobj: IO[bytes]) -> tuple[bytes, int]:
|
2010-07-31 06:52:47 +04:00
|
|
|
return struct.unpack(">4sI", fobj.read(HEADERSIZE))
|
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2024-07-05 20:55:23 +03:00
|
|
|
def read_32t(
|
|
|
|
fobj: IO[bytes], start_length: tuple[int, int], size: tuple[int, int, int]
|
|
|
|
) -> dict[str, Image.Image]:
|
2010-07-31 06:52:47 +04:00
|
|
|
# The 128x128 icon seems to have an extra header for some reason.
|
2012-10-16 06:32:28 +04:00
|
|
|
(start, length) = start_length
|
2010-07-31 06:52:47 +04:00
|
|
|
fobj.seek(start)
|
|
|
|
sig = fobj.read(4)
|
py3k: The big push
There are two main issues fixed with this commit:
* bytes vs. str: All file, image, and palette data are now handled as
bytes. A new _binary module consolidates the hacks needed to do this
across Python versions. tostring/fromstring methods have been renamed to
tobytes/frombytes, but the Python 2.6/2.7 versions alias them to the old
names for compatibility. Users should move to tobytes/frombytes.
One other potentially-breaking change is that text data in image files
(such as tags, comments) are now explicitly handled with a specific
character encoding in mind. This works well with the Unicode str in
Python 3, but may trip up old code expecting a straight byte-for-byte
translation to a Python string. This also required a change to Gohlke's
tags tests (in Tests/test_file_png.py) to expect Unicode strings from
the code.
* True div vs. floor div: Many division operations used the "/" operator
to do floor division, which is now the "//" operator in Python 3. These
were fixed.
As of this commit, on the first pass, I have one failing test (improper
handling of a slice object in a C module, test_imagepath.py) in Python 3,
and three that that I haven't tried running yet (test_imagegl,
test_imagegrab, and test_imageqt). I also haven't tested anything on
Windows. All but the three skipped tests run flawlessly against Pythons
2.6 and 2.7.
2012-10-21 01:01:53 +04:00
|
|
|
if sig != b"\x00\x00\x00\x00":
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = "Unknown signature, expecting 0x00000000"
|
|
|
|
raise SyntaxError(msg)
|
2012-10-16 06:32:28 +04:00
|
|
|
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
|
|
|
|
2024-07-05 20:55:23 +03:00
|
|
|
def read_32(
|
|
|
|
fobj: IO[bytes], start_length: tuple[int, int], size: tuple[int, int, int]
|
|
|
|
) -> dict[str, Image.Image]:
|
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.
|
|
|
|
"""
|
2012-10-16 06:32:28 +04:00
|
|
|
(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
|
2024-07-05 20:55:23 +03:00
|
|
|
byte_int = byte[0]
|
|
|
|
if byte_int & 0x80:
|
|
|
|
blocksize = byte_int - 125
|
2010-07-31 06:52:47 +04:00
|
|
|
byte = fobj.read(1)
|
|
|
|
for i in range(blocksize):
|
|
|
|
data.append(byte)
|
|
|
|
else:
|
2024-07-05 20:55:23 +03:00
|
|
|
blocksize = byte_int + 1
|
2010-07-31 06:52:47 +04:00
|
|
|
data.append(fobj.read(blocksize))
|
2014-05-10 08:36:15 +04:00
|
|
|
bytesleft -= blocksize
|
2010-07-31 06:52:47 +04:00
|
|
|
if bytesleft <= 0:
|
|
|
|
break
|
|
|
|
if bytesleft != 0:
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = f"Error reading channel [{repr(bytesleft)} left]"
|
|
|
|
raise SyntaxError(msg)
|
2014-03-26 14:49:39 +04:00
|
|
|
band = Image.frombuffer("L", pixel_size, b"".join(data), "raw", "L", 0, 1)
|
2024-08-02 16:30:27 +03:00
|
|
|
assert im.im is not None
|
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
|
|
|
|
2024-07-05 20:55:23 +03:00
|
|
|
def read_mk(
|
|
|
|
fobj: IO[bytes], start_length: tuple[int, int], size: tuple[int, int, int]
|
|
|
|
) -> dict[str, Image.Image]:
|
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]
|
|
|
|
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
|
|
|
|
2024-07-05 20:55:23 +03:00
|
|
|
def read_png_or_jpeg2000(
|
|
|
|
fobj: IO[bytes], start_length: tuple[int, int], size: tuple[int, int, int]
|
|
|
|
) -> dict[str, Image.Image]:
|
2014-03-24 20:10:23 +04:00
|
|
|
(start, length) = start_length
|
|
|
|
fobj.seek(start)
|
|
|
|
sig = fobj.read(12)
|
2024-07-05 20:55:23 +03:00
|
|
|
|
|
|
|
im: Image.Image
|
2014-03-24 20:10:23 +04:00
|
|
|
if sig[:8] == b"\x89PNG\x0d\x0a\x1a\x0a":
|
|
|
|
fobj.seek(start)
|
|
|
|
im = PngImagePlugin.PngImageFile(fobj)
|
2021-02-25 01:27:07 +03:00
|
|
|
Image._decompression_bomb_check(im.size)
|
2014-03-24 20:10:23 +04:00
|
|
|
return {"RGBA": im}
|
|
|
|
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"
|
|
|
|
):
|
2014-03-28 13:09:55 +04:00
|
|
|
if not enable_jpeg2k:
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = (
|
2014-03-28 13:09:55 +04:00
|
|
|
"Unsupported icon subimage format (rebuild PIL "
|
|
|
|
"with JPEG 2000 support to fix this)"
|
|
|
|
)
|
2022-12-22 00:51:35 +03:00
|
|
|
raise ValueError(msg)
|
2014-03-24 20:10:23 +04:00
|
|
|
# j2k, jpc or j2c
|
2014-03-24 21:24:49 +04:00
|
|
|
fobj.seek(start)
|
2014-03-24 22:04:37 +04:00
|
|
|
jp2kstream = fobj.read(length)
|
|
|
|
f = io.BytesIO(jp2kstream)
|
|
|
|
im = Jpeg2KImagePlugin.Jpeg2KImageFile(f)
|
2021-02-25 01:27:07 +03:00
|
|
|
Image._decompression_bomb_check(im.size)
|
2014-03-24 21:24:49 +04:00
|
|
|
if im.mode != "RGBA":
|
|
|
|
im = im.convert("RGBA")
|
|
|
|
return {"RGBA": im}
|
2014-03-28 13:09:55 +04:00
|
|
|
else:
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = "Unsupported icon subimage format"
|
|
|
|
raise ValueError(msg)
|
2014-03-24 20:10:23 +04:00
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2019-09-30 17:56:31 +03:00
|
|
|
class IcnsFile:
|
2010-07-31 06:52:47 +04:00
|
|
|
SIZES = {
|
2014-03-24 20:10:23 +04: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)],
|
|
|
|
(128, 128, 1): [
|
|
|
|
(b"ic07", read_png_or_jpeg2000),
|
py3k: The big push
There are two main issues fixed with this commit:
* bytes vs. str: All file, image, and palette data are now handled as
bytes. A new _binary module consolidates the hacks needed to do this
across Python versions. tostring/fromstring methods have been renamed to
tobytes/frombytes, but the Python 2.6/2.7 versions alias them to the old
names for compatibility. Users should move to tobytes/frombytes.
One other potentially-breaking change is that text data in image files
(such as tags, comments) are now explicitly handled with a specific
character encoding in mind. This works well with the Unicode str in
Python 3, but may trip up old code expecting a straight byte-for-byte
translation to a Python string. This also required a change to Gohlke's
tags tests (in Tests/test_file_png.py) to expect Unicode strings from
the code.
* True div vs. floor div: Many division operations used the "/" operator
to do floor division, which is now the "//" operator in Python 3. These
were fixed.
As of this commit, on the first pass, I have one failing test (improper
handling of a slice object in a C module, test_imagepath.py) in Python 3,
and three that that I haven't tried running yet (test_imagegl,
test_imagegrab, and test_imageqt). I also haven't tested anything on
Windows. All but the three skipped tests run flawlessly against Pythons
2.6 and 2.7.
2012-10-21 01:01:53 +04:00
|
|
|
(b"it32", read_32t),
|
|
|
|
(b"t8mk", read_mk),
|
2010-07-31 06:52:47 +04:00
|
|
|
],
|
2014-03-24 20:10:23 +04:00
|
|
|
(64, 64, 1): [(b"icp6", read_png_or_jpeg2000)],
|
|
|
|
(32, 32, 2): [(b"ic12", read_png_or_jpeg2000)],
|
py3k: The big push
There are two main issues fixed with this commit:
* bytes vs. str: All file, image, and palette data are now handled as
bytes. A new _binary module consolidates the hacks needed to do this
across Python versions. tostring/fromstring methods have been renamed to
tobytes/frombytes, but the Python 2.6/2.7 versions alias them to the old
names for compatibility. Users should move to tobytes/frombytes.
One other potentially-breaking change is that text data in image files
(such as tags, comments) are now explicitly handled with a specific
character encoding in mind. This works well with the Unicode str in
Python 3, but may trip up old code expecting a straight byte-for-byte
translation to a Python string. This also required a change to Gohlke's
tags tests (in Tests/test_file_png.py) to expect Unicode strings from
the code.
* True div vs. floor div: Many division operations used the "/" operator
to do floor division, which is now the "//" operator in Python 3. These
were fixed.
As of this commit, on the first pass, I have one failing test (improper
handling of a slice object in a C module, test_imagepath.py) in Python 3,
and three that that I haven't tried running yet (test_imagegl,
test_imagegrab, and test_imageqt). I also haven't tested anything on
Windows. All but the three skipped tests run flawlessly against Pythons
2.6 and 2.7.
2012-10-21 01:01:53 +04:00
|
|
|
(48, 48, 1): [(b"ih32", read_32), (b"h8mk", read_mk)],
|
2014-03-24 20:10:23 +04:00
|
|
|
(32, 32, 1): [
|
|
|
|
(b"icp5", read_png_or_jpeg2000),
|
py3k: The big push
There are two main issues fixed with this commit:
* bytes vs. str: All file, image, and palette data are now handled as
bytes. A new _binary module consolidates the hacks needed to do this
across Python versions. tostring/fromstring methods have been renamed to
tobytes/frombytes, but the Python 2.6/2.7 versions alias them to the old
names for compatibility. Users should move to tobytes/frombytes.
One other potentially-breaking change is that text data in image files
(such as tags, comments) are now explicitly handled with a specific
character encoding in mind. This works well with the Unicode str in
Python 3, but may trip up old code expecting a straight byte-for-byte
translation to a Python string. This also required a change to Gohlke's
tags tests (in Tests/test_file_png.py) to expect Unicode strings from
the code.
* True div vs. floor div: Many division operations used the "/" operator
to do floor division, which is now the "//" operator in Python 3. These
were fixed.
As of this commit, on the first pass, I have one failing test (improper
handling of a slice object in a C module, test_imagepath.py) in Python 3,
and three that that I haven't tried running yet (test_imagegl,
test_imagegrab, and test_imageqt). I also haven't tested anything on
Windows. All but the three skipped tests run flawlessly against Pythons
2.6 and 2.7.
2012-10-21 01:01:53 +04:00
|
|
|
(b"il32", read_32),
|
|
|
|
(b"l8mk", read_mk),
|
2014-03-24 20:10:23 +04:00
|
|
|
],
|
|
|
|
(16, 16, 2): [(b"ic11", read_png_or_jpeg2000)],
|
|
|
|
(16, 16, 1): [
|
|
|
|
(b"icp4", read_png_or_jpeg2000),
|
py3k: The big push
There are two main issues fixed with this commit:
* bytes vs. str: All file, image, and palette data are now handled as
bytes. A new _binary module consolidates the hacks needed to do this
across Python versions. tostring/fromstring methods have been renamed to
tobytes/frombytes, but the Python 2.6/2.7 versions alias them to the old
names for compatibility. Users should move to tobytes/frombytes.
One other potentially-breaking change is that text data in image files
(such as tags, comments) are now explicitly handled with a specific
character encoding in mind. This works well with the Unicode str in
Python 3, but may trip up old code expecting a straight byte-for-byte
translation to a Python string. This also required a change to Gohlke's
tags tests (in Tests/test_file_png.py) to expect Unicode strings from
the code.
* True div vs. floor div: Many division operations used the "/" operator
to do floor division, which is now the "//" operator in Python 3. These
were fixed.
As of this commit, on the first pass, I have one failing test (improper
handling of a slice object in a C module, test_imagepath.py) in Python 3,
and three that that I haven't tried running yet (test_imagegl,
test_imagegrab, and test_imageqt). I also haven't tested anything on
Windows. All but the three skipped tests run flawlessly against Pythons
2.6 and 2.7.
2012-10-21 01:01:53 +04:00
|
|
|
(b"is32", read_32),
|
|
|
|
(b"s8mk", read_mk),
|
2010-07-31 06:52:47 +04:00
|
|
|
],
|
|
|
|
}
|
|
|
|
|
2024-07-05 20:55:23 +03:00
|
|
|
def __init__(self, fobj: IO[bytes]) -> None:
|
2010-07-31 06:52:47 +04:00
|
|
|
"""
|
|
|
|
fobj is a file-like object as an icns resource
|
|
|
|
"""
|
|
|
|
# signature : (start, length)
|
2024-07-05 20:55:23 +03:00
|
|
|
self.dct = {}
|
2010-07-31 06:52:47 +04:00
|
|
|
self.fobj = fobj
|
|
|
|
sig, filesize = nextheader(fobj)
|
2022-02-26 09:53:27 +03:00
|
|
|
if not _accept(sig):
|
2022-12-22 00:51:35 +03:00
|
|
|
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)
|
2014-08-07 03:42:43 +04:00
|
|
|
if blocksize <= 0:
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = "invalid block header"
|
|
|
|
raise SyntaxError(msg)
|
2014-05-10 08:36:15 +04:00
|
|
|
i += HEADERSIZE
|
|
|
|
blocksize -= HEADERSIZE
|
2024-07-05 20:55:23 +03:00
|
|
|
self.dct[sig] = (i, blocksize)
|
2019-01-13 05:05:46 +03:00
|
|
|
fobj.seek(blocksize, io.SEEK_CUR)
|
2014-05-10 08:36:15 +04:00
|
|
|
i += blocksize
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2024-07-05 20:55:23 +03:00
|
|
|
def itersizes(self) -> list[tuple[int, int, int]]:
|
2010-07-31 06:52:47 +04:00
|
|
|
sizes = []
|
|
|
|
for size, fmts in self.SIZES.items():
|
2023-02-06 22:27:15 +03:00
|
|
|
for fmt, reader in fmts:
|
2012-10-16 01:18:27 +04:00
|
|
|
if fmt in self.dct:
|
2010-07-31 06:52:47 +04:00
|
|
|
sizes.append(size)
|
|
|
|
break
|
|
|
|
return sizes
|
|
|
|
|
2024-07-05 20:55:23 +03:00
|
|
|
def bestsize(self) -> tuple[int, int, int]:
|
2010-07-31 06:52:47 +04:00
|
|
|
sizes = self.itersizes()
|
|
|
|
if not sizes:
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = "No 32bit icon resources found"
|
|
|
|
raise SyntaxError(msg)
|
2010-07-31 06:52:47 +04:00
|
|
|
return max(sizes)
|
|
|
|
|
2024-07-05 20:55:23 +03:00
|
|
|
def dataforsize(self, size: tuple[int, int, int]) -> dict[str, Image.Image]:
|
2010-07-31 06:52:47 +04:00
|
|
|
"""
|
|
|
|
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
|
|
|
|
|
2024-07-05 20:55:23 +03:00
|
|
|
def getimage(
|
|
|
|
self, size: tuple[int, int] | tuple[int, int, int] | None = None
|
|
|
|
) -> Image.Image:
|
2010-07-31 06:52:47 +04:00
|
|
|
if size is None:
|
|
|
|
size = self.bestsize()
|
2024-07-05 20:55:23 +03:00
|
|
|
elif len(size) == 2:
|
2014-03-24 20:10:23 +04:00
|
|
|
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
|
|
|
|
2024-07-05 20:55:23 +03:00
|
|
|
im = channels.get("RGBA")
|
2014-03-24 20:10:23 +04:00
|
|
|
if im:
|
|
|
|
return im
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2024-07-05 20:55:23 +03:00
|
|
|
im = channels["RGB"].copy()
|
2010-07-31 06:52:47 +04:00
|
|
|
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"
|
|
|
|
|
2024-05-04 13:51:54 +03:00
|
|
|
def _open(self) -> None:
|
2010-07-31 06:52:47 +04:00
|
|
|
self.icns = IcnsFile(self.fp)
|
2023-07-29 02:28:18 +03:00
|
|
|
self._mode = "RGBA"
|
2018-09-30 05:58:02 +03:00
|
|
|
self.info["sizes"] = self.icns.itersizes()
|
2014-03-24 20:10:23 +04:00
|
|
|
self.best_size = self.icns.bestsize()
|
|
|
|
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
|
|
|
|
2018-09-30 05:58:02 +03:00
|
|
|
@property
|
|
|
|
def size(self):
|
|
|
|
return self._size
|
|
|
|
|
|
|
|
@size.setter
|
2024-07-05 20:55:23 +03:00
|
|
|
def size(self, value) -> None:
|
2018-09-30 05:58:02 +03:00
|
|
|
info_size = value
|
|
|
|
if info_size not in self.info["sizes"] and len(info_size) == 2:
|
|
|
|
info_size = (info_size[0], info_size[1], 1)
|
|
|
|
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"]
|
2018-10-21 09:15:26 +03:00
|
|
|
]
|
2018-09-30 05:58:02 +03:00
|
|
|
if value in simple_sizes:
|
|
|
|
info_size = self.info["sizes"][simple_sizes.index(value)]
|
|
|
|
if info_size not in self.info["sizes"]:
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = "This is not one of the allowed sizes of this image"
|
|
|
|
raise ValueError(msg)
|
2018-09-30 05:58:02 +03:00
|
|
|
self._size = value
|
|
|
|
|
2024-07-05 20:55:23 +03:00
|
|
|
def load(self) -> Image.core.PixelAccess | None:
|
2014-03-24 20:10:23 +04:00
|
|
|
if len(self.size) == 3:
|
|
|
|
self.best_size = self.size
|
|
|
|
self.size = (
|
|
|
|
self.best_size[0] * self.best_size[2],
|
|
|
|
self.best_size[1] * self.best_size[2],
|
|
|
|
)
|
|
|
|
|
2022-02-02 03:49:31 +03: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
|
2022-02-02 03:49:31 +03:00
|
|
|
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
|
2022-02-02 03:49:31 +03:00
|
|
|
px = im.load()
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
self.im = im.im
|
2023-07-29 02:28:18 +03:00
|
|
|
self._mode = im.mode
|
2010-07-31 06:52:47 +04:00
|
|
|
self.size = im.size
|
2022-02-02 03:49:31 +03:00
|
|
|
|
|
|
|
return px
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2015-04-23 13:25:45 +03:00
|
|
|
|
2024-06-10 07:15:28 +03:00
|
|
|
def _save(im: Image.Image, fp: IO[bytes], filename: str | bytes) -> None:
|
2015-04-23 10:00:21 +03:00
|
|
|
"""
|
|
|
|
Saves the image as a series of PNG files,
|
2021-06-30 16:30:59 +03:00
|
|
|
that are then combined into a .icns file.
|
2015-04-23 10:00:21 +03:00
|
|
|
"""
|
2015-09-02 16:48:22 +03:00
|
|
|
if hasattr(fp, "flush"):
|
2015-04-12 05:58:46 +03:00
|
|
|
fp.flush()
|
2015-04-23 13:25:45 +03:00
|
|
|
|
2021-06-30 16:30:59 +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", [])}
|
2021-06-29 13:49:19 +03:00
|
|
|
size_streams = {}
|
2021-06-30 16:30:59 +03:00
|
|
|
for size in set(sizes.values()):
|
|
|
|
image = (
|
|
|
|
provided_images[size]
|
|
|
|
if size in provided_images
|
|
|
|
else im.resize((size, size))
|
|
|
|
)
|
2021-06-29 13:49:19 +03:00
|
|
|
|
|
|
|
temp = io.BytesIO()
|
|
|
|
image.save(temp, "png")
|
2021-06-30 16:30:59 +03:00
|
|
|
size_streams[size] = temp.getvalue()
|
2021-06-29 13:49:19 +03:00
|
|
|
|
|
|
|
entries = []
|
2021-06-30 16:30:59 +03:00
|
|
|
for type, size in sizes.items():
|
2021-06-29 13:49:19 +03:00
|
|
|
stream = size_streams[size]
|
2024-06-10 07:15:28 +03:00
|
|
|
entries.append((type, HEADERSIZE + len(stream), stream))
|
2020-04-04 06:22:11 +03:00
|
|
|
|
|
|
|
# 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
|
2024-06-10 07:15:28 +03:00
|
|
|
file_length += sum(entry[1] for entry in entries)
|
2021-11-20 06:17:42 +03:00
|
|
|
fp.write(struct.pack(">i", file_length))
|
2020-04-04 06:22:11 +03:00
|
|
|
|
|
|
|
# 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:
|
2024-06-10 07:15:28 +03:00
|
|
|
fp.write(entry[0])
|
|
|
|
fp.write(struct.pack(">i", entry[1]))
|
2020-04-04 06:22:11 +03:00
|
|
|
|
|
|
|
# Data
|
2021-06-29 13:18:55 +03:00
|
|
|
for entry in entries:
|
2024-06-10 07:15:28 +03:00
|
|
|
fp.write(entry[0])
|
|
|
|
fp.write(struct.pack(">i", entry[1]))
|
|
|
|
fp.write(entry[2])
|
2020-04-04 06:22:11 +03:00
|
|
|
|
2021-06-28 17:48:06 +03:00
|
|
|
if hasattr(fp, "flush"):
|
|
|
|
fp.flush()
|
2015-04-12 05:58:46 +03:00
|
|
|
|
2018-03-03 12:54:00 +03:00
|
|
|
|
2024-04-06 05:58:53 +03:00
|
|
|
def _accept(prefix: bytes) -> bool:
|
2021-06-29 13:52:35 +03:00
|
|
|
return prefix[:4] == MAGIC
|
2021-04-03 13:42:14 +03:00
|
|
|
|
|
|
|
|
|
|
|
Image.register_open(IcnsImageFile.format, IcnsImageFile, _accept)
|
2015-07-04 16:29:58 +03:00
|
|
|
Image.register_extension(IcnsImageFile.format, ".icns")
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2020-04-04 06:22:11 +03:00
|
|
|
Image.register_save(IcnsImageFile.format, _save)
|
|
|
|
Image.register_mime(IcnsImageFile.format, "image/icns")
|
2015-04-12 05:58:46 +03:00
|
|
|
|
2010-07-31 06:52:47 +04: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"]:
|
2023-11-13 10:28:01 +03:00
|
|
|
width, height, scale = imf.size = size
|
|
|
|
imf.save(f"out-{width}-{height}-{scale}.png")
|
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")
|