IcnsImagePlugin.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399
  1. #
  2. # The Python Imaging Library.
  3. # $Id$
  4. #
  5. # macOS icns file decoder, based on icns.py by Bob Ippolito.
  6. #
  7. # history:
  8. # 2004-10-09 fl Turned into a PIL plugin; removed 2.3 dependencies.
  9. # 2020-04-04 Allow saving on all operating systems.
  10. #
  11. # Copyright (c) 2004 by Bob Ippolito.
  12. # Copyright (c) 2004 by Secret Labs.
  13. # Copyright (c) 2004 by Fredrik Lundh.
  14. # Copyright (c) 2014 by Alastair Houghton.
  15. # Copyright (c) 2020 by Pan Jing.
  16. #
  17. # See the README file for information on usage and redistribution.
  18. #
  19. from __future__ import annotations
  20. import io
  21. import os
  22. import struct
  23. import sys
  24. from typing import IO
  25. from . import Image, ImageFile, PngImagePlugin, features
  26. enable_jpeg2k = features.check_codec("jpg_2000")
  27. if enable_jpeg2k:
  28. from . import Jpeg2KImagePlugin
  29. MAGIC = b"icns"
  30. HEADERSIZE = 8
  31. def nextheader(fobj):
  32. return struct.unpack(">4sI", fobj.read(HEADERSIZE))
  33. def read_32t(fobj, start_length, size):
  34. # The 128x128 icon seems to have an extra header for some reason.
  35. (start, length) = start_length
  36. fobj.seek(start)
  37. sig = fobj.read(4)
  38. if sig != b"\x00\x00\x00\x00":
  39. msg = "Unknown signature, expecting 0x00000000"
  40. raise SyntaxError(msg)
  41. return read_32(fobj, (start + 4, length - 4), size)
  42. def read_32(fobj, start_length, size):
  43. """
  44. Read a 32bit RGB icon resource. Seems to be either uncompressed or
  45. an RLE packbits-like scheme.
  46. """
  47. (start, length) = start_length
  48. fobj.seek(start)
  49. pixel_size = (size[0] * size[2], size[1] * size[2])
  50. sizesq = pixel_size[0] * pixel_size[1]
  51. if length == sizesq * 3:
  52. # uncompressed ("RGBRGBGB")
  53. indata = fobj.read(length)
  54. im = Image.frombuffer("RGB", pixel_size, indata, "raw", "RGB", 0, 1)
  55. else:
  56. # decode image
  57. im = Image.new("RGB", pixel_size, None)
  58. for band_ix in range(3):
  59. data = []
  60. bytesleft = sizesq
  61. while bytesleft > 0:
  62. byte = fobj.read(1)
  63. if not byte:
  64. break
  65. byte = byte[0]
  66. if byte & 0x80:
  67. blocksize = byte - 125
  68. byte = fobj.read(1)
  69. for i in range(blocksize):
  70. data.append(byte)
  71. else:
  72. blocksize = byte + 1
  73. data.append(fobj.read(blocksize))
  74. bytesleft -= blocksize
  75. if bytesleft <= 0:
  76. break
  77. if bytesleft != 0:
  78. msg = f"Error reading channel [{repr(bytesleft)} left]"
  79. raise SyntaxError(msg)
  80. band = Image.frombuffer("L", pixel_size, b"".join(data), "raw", "L", 0, 1)
  81. im.im.putband(band.im, band_ix)
  82. return {"RGB": im}
  83. def read_mk(fobj, start_length, size):
  84. # Alpha masks seem to be uncompressed
  85. start = start_length[0]
  86. fobj.seek(start)
  87. pixel_size = (size[0] * size[2], size[1] * size[2])
  88. sizesq = pixel_size[0] * pixel_size[1]
  89. band = Image.frombuffer("L", pixel_size, fobj.read(sizesq), "raw", "L", 0, 1)
  90. return {"A": band}
  91. def read_png_or_jpeg2000(fobj, start_length, size):
  92. (start, length) = start_length
  93. fobj.seek(start)
  94. sig = fobj.read(12)
  95. if sig[:8] == b"\x89PNG\x0d\x0a\x1a\x0a":
  96. fobj.seek(start)
  97. im = PngImagePlugin.PngImageFile(fobj)
  98. Image._decompression_bomb_check(im.size)
  99. return {"RGBA": im}
  100. elif (
  101. sig[:4] == b"\xff\x4f\xff\x51"
  102. or sig[:4] == b"\x0d\x0a\x87\x0a"
  103. or sig == b"\x00\x00\x00\x0cjP \x0d\x0a\x87\x0a"
  104. ):
  105. if not enable_jpeg2k:
  106. msg = (
  107. "Unsupported icon subimage format (rebuild PIL "
  108. "with JPEG 2000 support to fix this)"
  109. )
  110. raise ValueError(msg)
  111. # j2k, jpc or j2c
  112. fobj.seek(start)
  113. jp2kstream = fobj.read(length)
  114. f = io.BytesIO(jp2kstream)
  115. im = Jpeg2KImagePlugin.Jpeg2KImageFile(f)
  116. Image._decompression_bomb_check(im.size)
  117. if im.mode != "RGBA":
  118. im = im.convert("RGBA")
  119. return {"RGBA": im}
  120. else:
  121. msg = "Unsupported icon subimage format"
  122. raise ValueError(msg)
  123. class IcnsFile:
  124. SIZES = {
  125. (512, 512, 2): [(b"ic10", read_png_or_jpeg2000)],
  126. (512, 512, 1): [(b"ic09", read_png_or_jpeg2000)],
  127. (256, 256, 2): [(b"ic14", read_png_or_jpeg2000)],
  128. (256, 256, 1): [(b"ic08", read_png_or_jpeg2000)],
  129. (128, 128, 2): [(b"ic13", read_png_or_jpeg2000)],
  130. (128, 128, 1): [
  131. (b"ic07", read_png_or_jpeg2000),
  132. (b"it32", read_32t),
  133. (b"t8mk", read_mk),
  134. ],
  135. (64, 64, 1): [(b"icp6", read_png_or_jpeg2000)],
  136. (32, 32, 2): [(b"ic12", read_png_or_jpeg2000)],
  137. (48, 48, 1): [(b"ih32", read_32), (b"h8mk", read_mk)],
  138. (32, 32, 1): [
  139. (b"icp5", read_png_or_jpeg2000),
  140. (b"il32", read_32),
  141. (b"l8mk", read_mk),
  142. ],
  143. (16, 16, 2): [(b"ic11", read_png_or_jpeg2000)],
  144. (16, 16, 1): [
  145. (b"icp4", read_png_or_jpeg2000),
  146. (b"is32", read_32),
  147. (b"s8mk", read_mk),
  148. ],
  149. }
  150. def __init__(self, fobj):
  151. """
  152. fobj is a file-like object as an icns resource
  153. """
  154. # signature : (start, length)
  155. self.dct = dct = {}
  156. self.fobj = fobj
  157. sig, filesize = nextheader(fobj)
  158. if not _accept(sig):
  159. msg = "not an icns file"
  160. raise SyntaxError(msg)
  161. i = HEADERSIZE
  162. while i < filesize:
  163. sig, blocksize = nextheader(fobj)
  164. if blocksize <= 0:
  165. msg = "invalid block header"
  166. raise SyntaxError(msg)
  167. i += HEADERSIZE
  168. blocksize -= HEADERSIZE
  169. dct[sig] = (i, blocksize)
  170. fobj.seek(blocksize, io.SEEK_CUR)
  171. i += blocksize
  172. def itersizes(self):
  173. sizes = []
  174. for size, fmts in self.SIZES.items():
  175. for fmt, reader in fmts:
  176. if fmt in self.dct:
  177. sizes.append(size)
  178. break
  179. return sizes
  180. def bestsize(self):
  181. sizes = self.itersizes()
  182. if not sizes:
  183. msg = "No 32bit icon resources found"
  184. raise SyntaxError(msg)
  185. return max(sizes)
  186. def dataforsize(self, size):
  187. """
  188. Get an icon resource as {channel: array}. Note that
  189. the arrays are bottom-up like windows bitmaps and will likely
  190. need to be flipped or transposed in some way.
  191. """
  192. dct = {}
  193. for code, reader in self.SIZES[size]:
  194. desc = self.dct.get(code)
  195. if desc is not None:
  196. dct.update(reader(self.fobj, desc, size))
  197. return dct
  198. def getimage(self, size=None):
  199. if size is None:
  200. size = self.bestsize()
  201. if len(size) == 2:
  202. size = (size[0], size[1], 1)
  203. channels = self.dataforsize(size)
  204. im = channels.get("RGBA", None)
  205. if im:
  206. return im
  207. im = channels.get("RGB").copy()
  208. try:
  209. im.putalpha(channels["A"])
  210. except KeyError:
  211. pass
  212. return im
  213. ##
  214. # Image plugin for Mac OS icons.
  215. class IcnsImageFile(ImageFile.ImageFile):
  216. """
  217. PIL image support for Mac OS .icns files.
  218. Chooses the best resolution, but will possibly load
  219. a different size image if you mutate the size attribute
  220. before calling 'load'.
  221. The info dictionary has a key 'sizes' that is a list
  222. of sizes that the icns file has.
  223. """
  224. format = "ICNS"
  225. format_description = "Mac OS icns resource"
  226. def _open(self) -> None:
  227. self.icns = IcnsFile(self.fp)
  228. self._mode = "RGBA"
  229. self.info["sizes"] = self.icns.itersizes()
  230. self.best_size = self.icns.bestsize()
  231. self.size = (
  232. self.best_size[0] * self.best_size[2],
  233. self.best_size[1] * self.best_size[2],
  234. )
  235. @property
  236. def size(self):
  237. return self._size
  238. @size.setter
  239. def size(self, value):
  240. info_size = value
  241. if info_size not in self.info["sizes"] and len(info_size) == 2:
  242. info_size = (info_size[0], info_size[1], 1)
  243. if (
  244. info_size not in self.info["sizes"]
  245. and len(info_size) == 3
  246. and info_size[2] == 1
  247. ):
  248. simple_sizes = [
  249. (size[0] * size[2], size[1] * size[2]) for size in self.info["sizes"]
  250. ]
  251. if value in simple_sizes:
  252. info_size = self.info["sizes"][simple_sizes.index(value)]
  253. if info_size not in self.info["sizes"]:
  254. msg = "This is not one of the allowed sizes of this image"
  255. raise ValueError(msg)
  256. self._size = value
  257. def load(self):
  258. if len(self.size) == 3:
  259. self.best_size = self.size
  260. self.size = (
  261. self.best_size[0] * self.best_size[2],
  262. self.best_size[1] * self.best_size[2],
  263. )
  264. px = Image.Image.load(self)
  265. if self.im is not None and self.im.size == self.size:
  266. # Already loaded
  267. return px
  268. self.load_prepare()
  269. # This is likely NOT the best way to do it, but whatever.
  270. im = self.icns.getimage(self.best_size)
  271. # If this is a PNG or JPEG 2000, it won't be loaded yet
  272. px = im.load()
  273. self.im = im.im
  274. self._mode = im.mode
  275. self.size = im.size
  276. return px
  277. def _save(im: Image.Image, fp: IO[bytes], filename: str | bytes) -> None:
  278. """
  279. Saves the image as a series of PNG files,
  280. that are then combined into a .icns file.
  281. """
  282. if hasattr(fp, "flush"):
  283. fp.flush()
  284. sizes = {
  285. b"ic07": 128,
  286. b"ic08": 256,
  287. b"ic09": 512,
  288. b"ic10": 1024,
  289. b"ic11": 32,
  290. b"ic12": 64,
  291. b"ic13": 256,
  292. b"ic14": 512,
  293. }
  294. provided_images = {im.width: im for im in im.encoderinfo.get("append_images", [])}
  295. size_streams = {}
  296. for size in set(sizes.values()):
  297. image = (
  298. provided_images[size]
  299. if size in provided_images
  300. else im.resize((size, size))
  301. )
  302. temp = io.BytesIO()
  303. image.save(temp, "png")
  304. size_streams[size] = temp.getvalue()
  305. entries = []
  306. for type, size in sizes.items():
  307. stream = size_streams[size]
  308. entries.append((type, HEADERSIZE + len(stream), stream))
  309. # Header
  310. fp.write(MAGIC)
  311. file_length = HEADERSIZE # Header
  312. file_length += HEADERSIZE + 8 * len(entries) # TOC
  313. file_length += sum(entry[1] for entry in entries)
  314. fp.write(struct.pack(">i", file_length))
  315. # TOC
  316. fp.write(b"TOC ")
  317. fp.write(struct.pack(">i", HEADERSIZE + len(entries) * HEADERSIZE))
  318. for entry in entries:
  319. fp.write(entry[0])
  320. fp.write(struct.pack(">i", entry[1]))
  321. # Data
  322. for entry in entries:
  323. fp.write(entry[0])
  324. fp.write(struct.pack(">i", entry[1]))
  325. fp.write(entry[2])
  326. if hasattr(fp, "flush"):
  327. fp.flush()
  328. def _accept(prefix: bytes) -> bool:
  329. return prefix[:4] == MAGIC
  330. Image.register_open(IcnsImageFile.format, IcnsImageFile, _accept)
  331. Image.register_extension(IcnsImageFile.format, ".icns")
  332. Image.register_save(IcnsImageFile.format, _save)
  333. Image.register_mime(IcnsImageFile.format, "image/icns")
  334. if __name__ == "__main__":
  335. if len(sys.argv) < 2:
  336. print("Syntax: python3 IcnsImagePlugin.py [file]")
  337. sys.exit()
  338. with open(sys.argv[1], "rb") as fp:
  339. imf = IcnsImageFile(fp)
  340. for size in imf.info["sizes"]:
  341. width, height, scale = imf.size = size
  342. imf.save(f"out-{width}-{height}-{scale}.png")
  343. with Image.open(sys.argv[1]) as im:
  344. im.save("out.png")
  345. if sys.platform == "windows":
  346. os.startfile("out.png")