You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			412 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Python
		
	
			
		
		
	
	
			412 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Python
		
	
#
 | 
						|
# The Python Imaging Library.
 | 
						|
# $Id$
 | 
						|
#
 | 
						|
# macOS icns file decoder, based on icns.py by Bob Ippolito.
 | 
						|
#
 | 
						|
# history:
 | 
						|
# 2004-10-09 fl   Turned into a PIL plugin; removed 2.3 dependencies.
 | 
						|
# 2020-04-04      Allow saving on all operating systems.
 | 
						|
#
 | 
						|
# Copyright (c) 2004 by Bob Ippolito.
 | 
						|
# Copyright (c) 2004 by Secret Labs.
 | 
						|
# Copyright (c) 2004 by Fredrik Lundh.
 | 
						|
# Copyright (c) 2014 by Alastair Houghton.
 | 
						|
# Copyright (c) 2020 by Pan Jing.
 | 
						|
#
 | 
						|
# See the README file for information on usage and redistribution.
 | 
						|
#
 | 
						|
from __future__ import annotations
 | 
						|
 | 
						|
import io
 | 
						|
import os
 | 
						|
import struct
 | 
						|
import sys
 | 
						|
from typing import IO
 | 
						|
 | 
						|
from . import Image, ImageFile, PngImagePlugin, features
 | 
						|
from ._deprecate import deprecate
 | 
						|
 | 
						|
enable_jpeg2k = features.check_codec("jpg_2000")
 | 
						|
if enable_jpeg2k:
 | 
						|
    from . import Jpeg2KImagePlugin
 | 
						|
 | 
						|
MAGIC = b"icns"
 | 
						|
HEADERSIZE = 8
 | 
						|
 | 
						|
 | 
						|
def nextheader(fobj: IO[bytes]) -> tuple[bytes, int]:
 | 
						|
    return struct.unpack(">4sI", fobj.read(HEADERSIZE))
 | 
						|
 | 
						|
 | 
						|
def read_32t(
 | 
						|
    fobj: IO[bytes], start_length: tuple[int, int], size: tuple[int, int, int]
 | 
						|
) -> dict[str, Image.Image]:
 | 
						|
    # The 128x128 icon seems to have an extra header for some reason.
 | 
						|
    (start, length) = start_length
 | 
						|
    fobj.seek(start)
 | 
						|
    sig = fobj.read(4)
 | 
						|
    if sig != b"\x00\x00\x00\x00":
 | 
						|
        msg = "Unknown signature, expecting 0x00000000"
 | 
						|
        raise SyntaxError(msg)
 | 
						|
    return read_32(fobj, (start + 4, length - 4), size)
 | 
						|
 | 
						|
 | 
						|
def read_32(
 | 
						|
    fobj: IO[bytes], start_length: tuple[int, int], size: tuple[int, int, int]
 | 
						|
) -> dict[str, Image.Image]:
 | 
						|
    """
 | 
						|
    Read a 32bit RGB icon resource.  Seems to be either uncompressed or
 | 
						|
    an RLE packbits-like scheme.
 | 
						|
    """
 | 
						|
    (start, length) = start_length
 | 
						|
    fobj.seek(start)
 | 
						|
    pixel_size = (size[0] * size[2], size[1] * size[2])
 | 
						|
    sizesq = pixel_size[0] * pixel_size[1]
 | 
						|
    if length == sizesq * 3:
 | 
						|
        # uncompressed ("RGBRGBGB")
 | 
						|
        indata = fobj.read(length)
 | 
						|
        im = Image.frombuffer("RGB", pixel_size, indata, "raw", "RGB", 0, 1)
 | 
						|
    else:
 | 
						|
        # decode image
 | 
						|
        im = Image.new("RGB", pixel_size, None)
 | 
						|
        for band_ix in range(3):
 | 
						|
            data = []
 | 
						|
            bytesleft = sizesq
 | 
						|
            while bytesleft > 0:
 | 
						|
                byte = fobj.read(1)
 | 
						|
                if not byte:
 | 
						|
                    break
 | 
						|
                byte_int = byte[0]
 | 
						|
                if byte_int & 0x80:
 | 
						|
                    blocksize = byte_int - 125
 | 
						|
                    byte = fobj.read(1)
 | 
						|
                    for i in range(blocksize):
 | 
						|
                        data.append(byte)
 | 
						|
                else:
 | 
						|
                    blocksize = byte_int + 1
 | 
						|
                    data.append(fobj.read(blocksize))
 | 
						|
                bytesleft -= blocksize
 | 
						|
                if bytesleft <= 0:
 | 
						|
                    break
 | 
						|
            if bytesleft != 0:
 | 
						|
                msg = f"Error reading channel [{repr(bytesleft)} left]"
 | 
						|
                raise SyntaxError(msg)
 | 
						|
            band = Image.frombuffer("L", pixel_size, b"".join(data), "raw", "L", 0, 1)
 | 
						|
            im.im.putband(band.im, band_ix)
 | 
						|
    return {"RGB": im}
 | 
						|
 | 
						|
 | 
						|
def read_mk(
 | 
						|
    fobj: IO[bytes], start_length: tuple[int, int], size: tuple[int, int, int]
 | 
						|
) -> dict[str, Image.Image]:
 | 
						|
    # Alpha masks seem to be uncompressed
 | 
						|
    start = start_length[0]
 | 
						|
    fobj.seek(start)
 | 
						|
    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)
 | 
						|
    return {"A": band}
 | 
						|
 | 
						|
 | 
						|
def read_png_or_jpeg2000(
 | 
						|
    fobj: IO[bytes], start_length: tuple[int, int], size: tuple[int, int, int]
 | 
						|
) -> dict[str, Image.Image]:
 | 
						|
    (start, length) = start_length
 | 
						|
    fobj.seek(start)
 | 
						|
    sig = fobj.read(12)
 | 
						|
 | 
						|
    im: Image.Image
 | 
						|
    if sig.startswith(b"\x89PNG\x0d\x0a\x1a\x0a"):
 | 
						|
        fobj.seek(start)
 | 
						|
        im = PngImagePlugin.PngImageFile(fobj)
 | 
						|
        Image._decompression_bomb_check(im.size)
 | 
						|
        return {"RGBA": im}
 | 
						|
    elif (
 | 
						|
        sig.startswith((b"\xff\x4f\xff\x51", b"\x0d\x0a\x87\x0a"))
 | 
						|
        or sig == b"\x00\x00\x00\x0cjP  \x0d\x0a\x87\x0a"
 | 
						|
    ):
 | 
						|
        if not enable_jpeg2k:
 | 
						|
            msg = (
 | 
						|
                "Unsupported icon subimage format (rebuild PIL "
 | 
						|
                "with JPEG 2000 support to fix this)"
 | 
						|
            )
 | 
						|
            raise ValueError(msg)
 | 
						|
        # 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)
 | 
						|
        if im.mode != "RGBA":
 | 
						|
            im = im.convert("RGBA")
 | 
						|
        return {"RGBA": im}
 | 
						|
    else:
 | 
						|
        msg = "Unsupported icon subimage format"
 | 
						|
        raise ValueError(msg)
 | 
						|
 | 
						|
 | 
						|
class IcnsFile:
 | 
						|
    SIZES = {
 | 
						|
        (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),
 | 
						|
            (b"it32", read_32t),
 | 
						|
            (b"t8mk", read_mk),
 | 
						|
        ],
 | 
						|
        (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)],
 | 
						|
        (32, 32, 1): [
 | 
						|
            (b"icp5", read_png_or_jpeg2000),
 | 
						|
            (b"il32", read_32),
 | 
						|
            (b"l8mk", read_mk),
 | 
						|
        ],
 | 
						|
        (16, 16, 2): [(b"ic11", read_png_or_jpeg2000)],
 | 
						|
        (16, 16, 1): [
 | 
						|
            (b"icp4", read_png_or_jpeg2000),
 | 
						|
            (b"is32", read_32),
 | 
						|
            (b"s8mk", read_mk),
 | 
						|
        ],
 | 
						|
    }
 | 
						|
 | 
						|
    def __init__(self, fobj: IO[bytes]) -> None:
 | 
						|
        """
 | 
						|
        fobj is a file-like object as an icns resource
 | 
						|
        """
 | 
						|
        # signature : (start, length)
 | 
						|
        self.dct = {}
 | 
						|
        self.fobj = fobj
 | 
						|
        sig, filesize = nextheader(fobj)
 | 
						|
        if not _accept(sig):
 | 
						|
            msg = "not an icns file"
 | 
						|
            raise SyntaxError(msg)
 | 
						|
        i = HEADERSIZE
 | 
						|
        while i < filesize:
 | 
						|
            sig, blocksize = nextheader(fobj)
 | 
						|
            if blocksize <= 0:
 | 
						|
                msg = "invalid block header"
 | 
						|
                raise SyntaxError(msg)
 | 
						|
            i += HEADERSIZE
 | 
						|
            blocksize -= HEADERSIZE
 | 
						|
            self.dct[sig] = (i, blocksize)
 | 
						|
            fobj.seek(blocksize, io.SEEK_CUR)
 | 
						|
            i += blocksize
 | 
						|
 | 
						|
    def itersizes(self) -> list[tuple[int, int, int]]:
 | 
						|
        sizes = []
 | 
						|
        for size, fmts in self.SIZES.items():
 | 
						|
            for fmt, reader in fmts:
 | 
						|
                if fmt in self.dct:
 | 
						|
                    sizes.append(size)
 | 
						|
                    break
 | 
						|
        return sizes
 | 
						|
 | 
						|
    def bestsize(self) -> tuple[int, int, int]:
 | 
						|
        sizes = self.itersizes()
 | 
						|
        if not sizes:
 | 
						|
            msg = "No 32bit icon resources found"
 | 
						|
            raise SyntaxError(msg)
 | 
						|
        return max(sizes)
 | 
						|
 | 
						|
    def dataforsize(self, size: tuple[int, int, int]) -> dict[str, Image.Image]:
 | 
						|
        """
 | 
						|
        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: tuple[int, int] | tuple[int, int, int] | None = None
 | 
						|
    ) -> Image.Image:
 | 
						|
        if size is None:
 | 
						|
            size = self.bestsize()
 | 
						|
        elif len(size) == 2:
 | 
						|
            size = (size[0], size[1], 1)
 | 
						|
        channels = self.dataforsize(size)
 | 
						|
 | 
						|
        im = channels.get("RGBA")
 | 
						|
        if im:
 | 
						|
            return im
 | 
						|
 | 
						|
        im = channels["RGB"].copy()
 | 
						|
        try:
 | 
						|
            im.putalpha(channels["A"])
 | 
						|
        except KeyError:
 | 
						|
            pass
 | 
						|
        return im
 | 
						|
 | 
						|
 | 
						|
##
 | 
						|
# Image plugin for Mac OS icons.
 | 
						|
 | 
						|
 | 
						|
class IcnsImageFile(ImageFile.ImageFile):
 | 
						|
    """
 | 
						|
    PIL image support for Mac OS .icns files.
 | 
						|
    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) -> None:
 | 
						|
        self.icns = IcnsFile(self.fp)
 | 
						|
        self._mode = "RGBA"
 | 
						|
        self.info["sizes"] = self.icns.itersizes()
 | 
						|
        self.best_size = self.icns.bestsize()
 | 
						|
        self.size = (
 | 
						|
            self.best_size[0] * self.best_size[2],
 | 
						|
            self.best_size[1] * self.best_size[2],
 | 
						|
        )
 | 
						|
 | 
						|
    @property  # type: ignore[override]
 | 
						|
    def size(self) -> tuple[int, int] | tuple[int, int, int]:
 | 
						|
        return self._size
 | 
						|
 | 
						|
    @size.setter
 | 
						|
    def size(self, value: tuple[int, int] | tuple[int, int, int]) -> None:
 | 
						|
        if len(value) == 3:
 | 
						|
            deprecate("Setting size to (width, height, scale)", 12, "load(scale)")
 | 
						|
            if value in self.info["sizes"]:
 | 
						|
                self._size = value  # type: ignore[assignment]
 | 
						|
                return
 | 
						|
        else:
 | 
						|
            # Check that a matching size exists,
 | 
						|
            # or that there is a scale that would create a size that matches
 | 
						|
            for size in self.info["sizes"]:
 | 
						|
                simple_size = size[0] * size[2], size[1] * size[2]
 | 
						|
                scale = simple_size[0] // value[0]
 | 
						|
                if simple_size[1] / value[1] == scale:
 | 
						|
                    self._size = value
 | 
						|
                    return
 | 
						|
        msg = "This is not one of the allowed sizes of this image"
 | 
						|
        raise ValueError(msg)
 | 
						|
 | 
						|
    def load(self, scale: int | None = None) -> Image.core.PixelAccess | None:
 | 
						|
        if scale is not None or len(self.size) == 3:
 | 
						|
            if scale is None and len(self.size) == 3:
 | 
						|
                scale = self.size[2]
 | 
						|
            assert scale is not None
 | 
						|
            width, height = self.size[:2]
 | 
						|
            self.size = width * scale, height * scale
 | 
						|
            self.best_size = width, height, scale
 | 
						|
 | 
						|
        px = Image.Image.load(self)
 | 
						|
        if self._im is not None and self.im.size == self.size:
 | 
						|
            # Already loaded
 | 
						|
            return px
 | 
						|
        self.load_prepare()
 | 
						|
        # This is likely NOT the best way to do it, but whatever.
 | 
						|
        im = self.icns.getimage(self.best_size)
 | 
						|
 | 
						|
        # If this is a PNG or JPEG 2000, it won't be loaded yet
 | 
						|
        px = im.load()
 | 
						|
 | 
						|
        self.im = im.im
 | 
						|
        self._mode = im.mode
 | 
						|
        self.size = im.size
 | 
						|
 | 
						|
        return px
 | 
						|
 | 
						|
 | 
						|
def _save(im: Image.Image, fp: IO[bytes], filename: str | bytes) -> None:
 | 
						|
    """
 | 
						|
    Saves the image as a series of PNG files,
 | 
						|
    that are then combined into a .icns file.
 | 
						|
    """
 | 
						|
    if hasattr(fp, "flush"):
 | 
						|
        fp.flush()
 | 
						|
 | 
						|
    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,
 | 
						|
    }
 | 
						|
    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]
 | 
						|
        entries.append((type, HEADERSIZE + len(stream), stream))
 | 
						|
 | 
						|
    # Header
 | 
						|
    fp.write(MAGIC)
 | 
						|
    file_length = HEADERSIZE  # Header
 | 
						|
    file_length += HEADERSIZE + 8 * len(entries)  # TOC
 | 
						|
    file_length += sum(entry[1] for entry in entries)
 | 
						|
    fp.write(struct.pack(">i", file_length))
 | 
						|
 | 
						|
    # TOC
 | 
						|
    fp.write(b"TOC ")
 | 
						|
    fp.write(struct.pack(">i", HEADERSIZE + len(entries) * HEADERSIZE))
 | 
						|
    for entry in entries:
 | 
						|
        fp.write(entry[0])
 | 
						|
        fp.write(struct.pack(">i", entry[1]))
 | 
						|
 | 
						|
    # Data
 | 
						|
    for entry in entries:
 | 
						|
        fp.write(entry[0])
 | 
						|
        fp.write(struct.pack(">i", entry[1]))
 | 
						|
        fp.write(entry[2])
 | 
						|
 | 
						|
    if hasattr(fp, "flush"):
 | 
						|
        fp.flush()
 | 
						|
 | 
						|
 | 
						|
def _accept(prefix: bytes) -> bool:
 | 
						|
    return prefix.startswith(MAGIC)
 | 
						|
 | 
						|
 | 
						|
Image.register_open(IcnsImageFile.format, IcnsImageFile, _accept)
 | 
						|
Image.register_extension(IcnsImageFile.format, ".icns")
 | 
						|
 | 
						|
Image.register_save(IcnsImageFile.format, _save)
 | 
						|
Image.register_mime(IcnsImageFile.format, "image/icns")
 | 
						|
 | 
						|
if __name__ == "__main__":
 | 
						|
    if len(sys.argv) < 2:
 | 
						|
        print("Syntax: python3 IcnsImagePlugin.py [file]")
 | 
						|
        sys.exit()
 | 
						|
 | 
						|
    with open(sys.argv[1], "rb") as fp:
 | 
						|
        imf = IcnsImageFile(fp)
 | 
						|
        for size in imf.info["sizes"]:
 | 
						|
            width, height, scale = imf.size = size
 | 
						|
            imf.save(f"out-{width}-{height}-{scale}.png")
 | 
						|
        with Image.open(sys.argv[1]) as im:
 | 
						|
            im.save("out.png")
 | 
						|
        if sys.platform == "windows":
 | 
						|
            os.startfile("out.png")
 |