Pillow/src/PIL/IcoImagePlugin.py

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

361 lines
12 KiB
Python
Raw Normal View History

2010-07-31 06:52:47 +04:00
#
# The Python Imaging Library.
# $Id$
#
# Windows Icon support for PIL
#
# History:
# 96-05-27 fl Created
#
# Copyright (c) Secret Labs AB 1997.
# Copyright (c) Fredrik Lundh 1996.
#
# See the README file for information on usage and redistribution.
#
2013-06-05 21:01:05 +04:00
# This plugin is a refactored version of Win32IconImagePlugin by Bryan Davis
# <casadebender@gmail.com>.
2016-02-22 10:38:04 +03:00
# https://code.google.com/archive/p/casadebender/wikis/Win32IconImagePlugin.wiki
2013-06-05 21:01:05 +04:00
#
# Icon format references:
2015-12-28 16:04:39 +03:00
# * https://en.wikipedia.org/wiki/ICO_(file_format)
2016-02-10 11:37:16 +03:00
# * https://msdn.microsoft.com/en-us/library/ms997538.aspx
from __future__ import annotations
2010-07-31 06:52:47 +04:00
2019-05-12 13:44:29 +03:00
import warnings
2014-11-08 09:49:50 +03:00
from io import BytesIO
from math import ceil, log
2024-06-05 01:29:28 +03:00
from typing import IO
2014-11-07 22:01:46 +03:00
from . import BmpImagePlugin, Image, ImageFile, PngImagePlugin
from ._binary import i16le as i16
from ._binary import i32le as i32
2022-03-11 12:38:31 +03:00
from ._binary import o8
from ._binary import o16le as o16
2021-05-26 23:21:28 +03:00
from ._binary import o32le as o32
2010-07-31 06:52:47 +04:00
#
# --------------------------------------------------------------------
2013-06-05 21:01:05 +04:00
_MAGIC = b"\0\0\1\0"
2010-07-31 06:52:47 +04:00
2014-08-26 17:47:10 +04:00
2024-06-05 01:29:28 +03:00
def _save(im: Image.Image, fp: IO[bytes], filename: str) -> None:
2014-11-07 22:01:46 +03:00
fp.write(_MAGIC) # (2+2)
bmp = im.encoderinfo.get("bitmap_format") == "bmp"
2014-11-07 22:01:46 +03:00
sizes = im.encoderinfo.get(
"sizes",
[(16, 16), (24, 24), (32, 32), (48, 48), (64, 64), (128, 128), (256, 256)],
2016-11-30 15:50:53 +03:00
)
frames = []
provided_ims = [im] + im.encoderinfo.get("append_images", [])
2014-11-07 22:01:46 +03:00
width, height = im.size
for size in sorted(set(sizes)):
if size[0] > width or size[1] > height or size[0] > 256 or size[1] > 256:
continue
for provided_im in provided_ims:
if provided_im.size != size:
continue
frames.append(provided_im)
if bmp:
bits = BmpImagePlugin.SAVE[provided_im.mode][1]
bits_used = [bits]
for other_im in provided_ims:
if other_im.size != size:
continue
bits = BmpImagePlugin.SAVE[other_im.mode][1]
if bits not in bits_used:
# Another image has been supplied for this size
# with a different bit depth
frames.append(other_im)
bits_used.append(bits)
break
else:
# TODO: invent a more convenient method for proportional scalings
frame = provided_im.copy()
frame.thumbnail(size, Image.Resampling.LANCZOS, reducing_gap=None)
frames.append(frame)
fp.write(o16(len(frames))) # idCount(2)
offset = fp.tell() + len(frames) * 16
for frame in frames:
width, height = frame.size
2016-11-30 15:50:53 +03:00
# 0 means 256
2022-03-11 12:38:31 +03:00
fp.write(o8(width if width < 256 else 0)) # bWidth(1)
fp.write(o8(height if height < 256 else 0)) # bHeight(1)
bits, colors = BmpImagePlugin.SAVE[frame.mode][1:] if bmp else (32, 0)
fp.write(o8(colors)) # bColorCount(1)
2014-11-07 22:01:46 +03:00
fp.write(b"\0") # bReserved(1)
fp.write(b"\0\0") # wPlanes(2)
2022-03-11 12:38:31 +03:00
fp.write(o16(bits)) # wBitCount(2)
2021-05-26 23:21:28 +03:00
image_io = BytesIO()
if bmp:
frame.save(image_io, "dib")
2021-05-26 23:21:28 +03:00
if bits != 32:
and_mask = Image.new("1", size)
2021-05-26 23:21:28 +03:00
ImageFile._save(
and_mask, image_io, [("raw", (0, 0) + size, 0, ("1", 0, -1))]
2021-05-26 23:21:28 +03:00
)
else:
frame.save(image_io, "png")
2014-11-07 22:01:46 +03:00
image_io.seek(0)
image_bytes = image_io.read()
2021-05-26 23:21:28 +03:00
if bmp:
image_bytes = image_bytes[:8] + o32(height * 2) + image_bytes[12:]
2014-11-07 22:01:46 +03:00
bytes_len = len(image_bytes)
2022-03-11 12:38:31 +03:00
fp.write(o32(bytes_len)) # dwBytesInRes(4)
fp.write(o32(offset)) # dwImageOffset(4)
2014-11-07 22:01:46 +03:00
current = fp.tell()
fp.seek(offset)
fp.write(image_bytes)
offset = offset + bytes_len
fp.seek(current)
2024-04-06 05:58:53 +03:00
def _accept(prefix: bytes) -> bool:
2013-06-05 21:01:05 +04:00
return prefix[:4] == _MAGIC
2010-07-31 06:52:47 +04:00
class IcoFile:
2013-06-05 21:01:05 +04:00
def __init__(self, buf):
"""
Parse image from file-like object containing ico file data
"""
2013-06-05 21:01:05 +04:00
# check magic
s = buf.read(6)
if not _accept(s):
msg = "not an ICO file"
raise SyntaxError(msg)
2013-06-05 21:01:05 +04:00
self.buf = buf
self.entry = []
# Number of items in file
self.nb_items = i16(s, 4)
2013-06-05 21:01:05 +04:00
# Get headers for each item
for i in range(self.nb_items):
s = buf.read(16)
2013-06-05 21:01:05 +04:00
icon_header = {
"width": s[0],
"height": s[1],
"nb_color": s[2], # No. of colors in image (0 if >=8bpp)
"reserved": s[3],
"planes": i16(s, 4),
"bpp": i16(s, 6),
"size": i32(s, 8),
"offset": i32(s, 12),
2013-06-05 21:01:05 +04:00
}
2013-06-05 21:01:05 +04:00
# See Wikipedia
for j in ("width", "height"):
if not icon_header[j]:
icon_header[j] = 256
2013-06-05 21:01:05 +04:00
# See Wikipedia notes about color depth.
# We need this just to differ images with equal sizes
icon_header["color_depth"] = (
icon_header["bpp"]
or (
icon_header["nb_color"] != 0
and ceil(log(icon_header["nb_color"], 2))
)
or 256
)
2013-06-05 21:01:05 +04:00
icon_header["dim"] = (icon_header["width"], icon_header["height"])
icon_header["square"] = icon_header["width"] * icon_header["height"]
2013-06-05 21:01:05 +04:00
self.entry.append(icon_header)
2013-06-05 21:01:05 +04:00
self.entry = sorted(self.entry, key=lambda x: x["color_depth"])
# ICO images are usually squares
2023-11-06 22:12:52 +03:00
self.entry = sorted(self.entry, key=lambda x: x["square"], reverse=True)
2013-06-05 21:01:05 +04:00
def sizes(self):
"""
Get a list of all available icon sizes and color depths.
"""
return {(h["width"], h["height"]) for h in self.entry}
2019-05-12 13:44:29 +03:00
def getentryindex(self, size, bpp=False):
for i, h in enumerate(self.entry):
2019-06-11 11:42:05 +03:00
if size == h["dim"] and (bpp is False or bpp == h["color_depth"]):
2019-05-12 13:44:29 +03:00
return i
return 0
2013-06-05 21:01:05 +04:00
def getimage(self, size, bpp=False):
"""
Get an image from the icon
"""
2019-05-12 13:44:29 +03:00
return self.frame(self.getentryindex(size, bpp))
2024-06-05 01:29:28 +03:00
def frame(self, idx: int) -> Image.Image:
2013-06-05 21:01:05 +04:00
"""
Get an image from frame idx
"""
2013-06-05 21:01:05 +04:00
header = self.entry[idx]
2013-06-05 21:01:05 +04:00
self.buf.seek(header["offset"])
data = self.buf.read(8)
self.buf.seek(header["offset"])
2024-06-05 01:29:28 +03:00
im: Image.Image
2013-06-05 21:01:05 +04:00
if data[:8] == PngImagePlugin._MAGIC:
# png frame
im = PngImagePlugin.PngImageFile(self.buf)
Image._decompression_bomb_check(im.size)
2013-06-05 21:01:05 +04:00
else:
# XOR + AND mask bmp frame
im = BmpImagePlugin.DibImageFile(self.buf)
2019-09-29 07:14:38 +03:00
Image._decompression_bomb_check(im.size)
2013-06-05 21:01:05 +04:00
# change tile dimension to only encompass XOR image
im._size = (im.size[0], int(im.size[1] / 2))
2013-06-05 21:01:05 +04:00
d, e, o, a = im.tile[0]
im.tile[0] = d, (0, 0) + im.size, o, a
2013-06-05 21:01:05 +04:00
# figure out where AND mask image starts
2021-04-22 14:18:21 +03:00
bpp = header["bpp"]
2013-06-05 21:01:05 +04:00
if 32 == bpp:
# 32-bit color depth icon image allows semitransparent areas
# PIL's DIB format ignores transparency bits, recover them.
# The DIB is packed in BGRX byte order where X is the alpha
# channel.
# Back up to start of bmp data
self.buf.seek(o)
# extract every 4th byte (eg. 3,7,11,15,...)
alpha_bytes = self.buf.read(im.size[0] * im.size[1] * 4)[3::4]
# convert to an 8bpp grayscale image
mask = Image.frombuffer(
"L", # 8bpp
im.size, # (w, h)
alpha_bytes, # source chars
"raw", # raw decoder
("L", 0, -1), # 8bpp inverted, unpadded, reversed
)
else:
# get AND image from end of bitmap
w = im.size[0]
if (w % 32) > 0:
# bitmap row data is aligned to word boundaries
w += 32 - (im.size[0] % 32)
2013-06-05 21:01:05 +04:00
# the total mask data is
# padded row size * height / bits per char
total_bytes = int((w * im.size[1]) / 8)
2021-08-10 00:04:36 +03:00
and_mask_offset = header["offset"] + header["size"] - total_bytes
2013-06-05 21:01:05 +04:00
self.buf.seek(and_mask_offset)
2016-11-30 15:51:30 +03:00
mask_data = self.buf.read(total_bytes)
2013-06-05 21:01:05 +04:00
# convert raw data to image
mask = Image.frombuffer(
"1", # 1 bpp
im.size, # (w, h)
2016-11-30 15:51:30 +03:00
mask_data, # source chars
2013-06-05 21:01:05 +04:00
"raw", # raw decoder
("1;I", int(w / 8), -1), # 1bpp inverted, padded, reversed
)
2013-06-05 21:01:05 +04:00
# now we have two images, im is XOR image and mask is AND image
# apply mask image as alpha channel
im = im.convert("RGBA")
im.putalpha(mask)
2013-06-05 21:01:05 +04:00
return im
2014-08-26 17:47:10 +04:00
2010-07-31 06:52:47 +04:00
##
# Image plugin for Windows Icon files.
2019-03-21 16:28:20 +03:00
2013-06-05 21:01:05 +04:00
class IcoImageFile(ImageFile.ImageFile):
"""
PIL read-only image support for Microsoft Windows .ico files.
By default the largest resolution image in the file will be loaded. This
2013-06-05 21:01:05 +04:00
can be changed by altering the 'size' attribute before calling 'load'.
2010-07-31 06:52:47 +04:00
The info dictionary has a key 'sizes' that is a list of the sizes available
2013-06-05 21:01:05 +04:00
in the icon file.
Handles classic, XP and Vista icon formats.
When saving, PNG compression is used. Support for this was only added in
Windows Vista. If you are unable to view the icon in Windows, convert the
image to "RGBA" mode before saving.
2013-06-05 21:01:05 +04:00
This plugin is a refactored version of Win32IconImagePlugin by Bryan Davis
<casadebender@gmail.com>.
2016-02-22 10:38:04 +03:00
https://code.google.com/archive/p/casadebender/wikis/Win32IconImagePlugin.wiki
2013-06-05 21:01:05 +04:00
"""
2019-03-21 16:28:20 +03:00
2010-07-31 06:52:47 +04:00
format = "ICO"
format_description = "Windows Icon"
2024-05-04 13:51:54 +03:00
def _open(self) -> None:
2013-06-05 21:01:05 +04:00
self.ico = IcoFile(self.fp)
self.info["sizes"] = self.ico.sizes()
self.size = self.ico.entry[0]["dim"]
self.load()
@property
def size(self):
return self._size
@size.setter
def size(self, value):
if value not in self.info["sizes"]:
msg = "This is not one of the allowed sizes of this image"
raise ValueError(msg)
self._size = value
2013-06-05 21:01:05 +04:00
def 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:05:45 +03:00
# Already loaded
return Image.Image.load(self)
2013-06-05 21:01:05 +04:00
im = self.ico.getimage(self.size)
# if tile is PNG, it won't really be loaded yet
im.load()
self.im = im.im
2022-12-23 15:20:06 +03:00
self.pyaccess = None
self._mode = im.mode
2024-02-14 01:17:22 +03:00
if im.palette:
self.palette = im.palette
2019-05-12 13:44:29 +03:00
if im.size != self.size:
warnings.warn("Image was not the expected size")
index = self.ico.getentryindex(self.size)
2019-06-11 11:42:05 +03:00
sizes = list(self.info["sizes"])
2019-05-12 13:44:29 +03:00
sizes[index] = im.size
2019-06-11 11:42:05 +03:00
self.info["sizes"] = set(sizes)
2019-05-12 13:44:29 +03:00
self.size = im.size
2010-07-31 06:52:47 +04:00
2024-05-15 13:19:09 +03:00
def load_seek(self, pos: int) -> None:
2015-01-29 07:23:15 +03:00
# Flag the ImageFile.Parser so that it
# just does all the decode at the end.
pass
2019-03-21 16:28:20 +03:00
2010-07-31 06:52:47 +04:00
#
# --------------------------------------------------------------------
2016-11-30 15:51:30 +03:00
Image.register_open(IcoImageFile.format, IcoImageFile, _accept)
Image.register_save(IcoImageFile.format, _save)
Image.register_extension(IcoImageFile.format, ".ico")
Image.register_mime(IcoImageFile.format, "image/x-icon")