mirror of
				https://github.com/python-pillow/Pillow.git
				synced 2025-10-24 20:51:16 +03:00 
			
		
		
		
	Merge branch 'master' of github.com:python-imaging/Pillow
This commit is contained in:
		
						commit
						ddd5bdedb5
					
				|  | @ -4,9 +4,6 @@ | |||
| # | ||||
| # Windows Icon support for PIL | ||||
| # | ||||
| # Notes: | ||||
| #       uses BmpImagePlugin.py to read the bitmap data. | ||||
| # | ||||
| # History: | ||||
| #       96-05-27 fl     Created | ||||
| # | ||||
|  | @ -16,11 +13,18 @@ | |||
| # See the README file for information on usage and redistribution. | ||||
| # | ||||
| 
 | ||||
| # This plugin is a refactored version of Win32IconImagePlugin by Bryan Davis <casadebender@gmail.com>. | ||||
| # https://code.google.com/p/casadebender/wiki/Win32IconImagePlugin | ||||
| # | ||||
| # Icon format references: | ||||
| #   * http://en.wikipedia.org/wiki/ICO_(file_format) | ||||
| #   * http://msdn.microsoft.com/en-us/library/ms997538.aspx | ||||
| 
 | ||||
| 
 | ||||
| __version__ = "0.1" | ||||
| 
 | ||||
| from PIL import Image, BmpImagePlugin, _binary | ||||
| 
 | ||||
| from PIL import Image, ImageFile, BmpImagePlugin, PngImagePlugin, _binary | ||||
| from math import log, ceil | ||||
| 
 | ||||
| # | ||||
| # -------------------------------------------------------------------- | ||||
|  | @ -29,56 +33,197 @@ i8 = _binary.i8 | |||
| i16 = _binary.i16le | ||||
| i32 = _binary.i32le | ||||
| 
 | ||||
| _MAGIC = b"\0\0\1\0" | ||||
| 
 | ||||
| def _accept(prefix): | ||||
|     return prefix[:4] == b"\0\0\1\0" | ||||
|     return prefix[:4] == _MAGIC | ||||
| 
 | ||||
| 
 | ||||
| class IcoFile: | ||||
|     def __init__(self, buf): | ||||
|         """ | ||||
|         Parse image from file-like object containing ico file data | ||||
|         """ | ||||
|          | ||||
|         # check magic | ||||
|         s = buf.read(6) | ||||
|         if not _accept(s): | ||||
|             raise SyntaxError("not an ICO file") | ||||
|          | ||||
|         self.buf = buf | ||||
|         self.entry = [] | ||||
| 
 | ||||
|         # Number of items in file | ||||
|         self.nb_items = i16(s[4:]) | ||||
| 
 | ||||
|         # Get headers for each item | ||||
|         for i in range(self.nb_items): | ||||
|             s = buf.read(16) | ||||
|              | ||||
|             icon_header = { | ||||
|                 'width': i8(s[0]), | ||||
|                 'height': i8(s[1]), | ||||
|                 'nb_color': i8(s[2]), # Number of colors in image (0 if >=8bpp) | ||||
|                 'reserved': i8(s[3]), | ||||
|                 'planes': i16(s[4:]), | ||||
|                 'bpp': i16(s[6:]), | ||||
|                 'size': i32(s[8:]), | ||||
|                 'offset': i32(s[12:]) | ||||
|             } | ||||
|              | ||||
|             # See Wikipedia | ||||
|             for j in ('width', 'height'): | ||||
|                 if not icon_header[j]: | ||||
|                     icon_header[j] = 256 | ||||
|              | ||||
|             # 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) | ||||
|              | ||||
|             icon_header['dim'] = (icon_header['width'], icon_header['height']) | ||||
|             icon_header['square'] = icon_header['width'] * icon_header['height'] | ||||
|              | ||||
|             self.entry.append(icon_header) | ||||
|          | ||||
|         self.entry = sorted(self.entry, key=lambda x: x['color_depth']) | ||||
|         # ICO images are usually squares | ||||
|         # self.entry = sorted(self.entry, key=lambda x: x['width']) | ||||
|         self.entry = sorted(self.entry, key=lambda x: x['square']) | ||||
|         self.entry.reverse() | ||||
|          | ||||
|     def sizes(self): | ||||
|         """ | ||||
|         Get a list of all available icon sizes and color depths. | ||||
|         """ | ||||
|         return set((h['width'], h['height']) for h in self.entry) | ||||
|      | ||||
|     def getimage(self, size, bpp=False): | ||||
|         """ | ||||
|         Get an image from the icon | ||||
|         """ | ||||
|         for (i, h) in enumerate(self.entry): | ||||
|             if size == h['dim'] and (bpp == False or bpp == h['color_depth']): | ||||
|                 return self.frame(i) | ||||
|         return self.frame(0) | ||||
|      | ||||
|     def frame(self, idx): | ||||
|         """ | ||||
|         Get an image from frame idx | ||||
|         """ | ||||
|          | ||||
|         header = self.entry[idx] | ||||
|          | ||||
|         self.buf.seek(header['offset']) | ||||
|         data = self.buf.read(8) | ||||
|         self.buf.seek(header['offset']) | ||||
|          | ||||
|         if data[:8] == PngImagePlugin._MAGIC: | ||||
|             # png frame | ||||
|             im = PngImagePlugin.PngImageFile(self.buf) | ||||
|         else: | ||||
|             # XOR + AND mask bmp frame | ||||
|             im = BmpImagePlugin.DibImageFile(self.buf) | ||||
|              | ||||
|             # change tile dimension to only encompass XOR image | ||||
|             im.size = (im.size[0], int(im.size[1] / 2)) | ||||
|             d, e, o, a = im.tile[0] | ||||
|             im.tile[0] = d, (0,0) + im.size, o, a | ||||
|              | ||||
|             # figure out where AND mask image starts | ||||
|             mode = a[0] | ||||
|             bpp = 8 | ||||
|             for k in BmpImagePlugin.BIT2MODE.keys(): | ||||
|                 if mode == BmpImagePlugin.BIT2MODE[k][1]: | ||||
|                     bpp = k | ||||
|                     break | ||||
|              | ||||
|             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) | ||||
|                      | ||||
|                 # the total mask data is padded row size * height / bits per char | ||||
| 
 | ||||
|                 and_mask_offset = o + int(im.size[0] * im.size[1] * (bpp / 8.0)) | ||||
|                 total_bytes = int((w * im.size[1]) / 8) | ||||
| 
 | ||||
|                 self.buf.seek(and_mask_offset) | ||||
|                 maskData = self.buf.read(total_bytes) | ||||
| 
 | ||||
|                 # convert raw data to image | ||||
|                 mask = Image.frombuffer( | ||||
|                     '1',            # 1 bpp | ||||
|                     im.size,        # (w, h) | ||||
|                     maskData,       # source chars | ||||
|                     'raw',          # raw decoder | ||||
|                     ('1;I', int(w/8), -1)  # 1bpp inverted, padded, reversed | ||||
|                 ) | ||||
|                  | ||||
|                 # 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) | ||||
|              | ||||
|         return im | ||||
|      | ||||
| ## | ||||
| # Image plugin for Windows Icon files. | ||||
| 
 | ||||
| class IcoImageFile(BmpImagePlugin.BmpImageFile): | ||||
| 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 can  | ||||
|     be changed by altering the 'size' attribute before calling 'load'. | ||||
| 
 | ||||
|     The info dictionary has a key 'sizes' that is a list of the sizes available  | ||||
|     in the icon file. | ||||
| 
 | ||||
|     Handles classic, XP and Vista icon formats. | ||||
|      | ||||
|     This plugin is a refactored version of Win32IconImagePlugin by Bryan Davis <casadebender@gmail.com>. | ||||
|     https://code.google.com/p/casadebender/wiki/Win32IconImagePlugin | ||||
|     """ | ||||
|     format = "ICO" | ||||
|     format_description = "Windows Icon" | ||||
| 
 | ||||
|     def _open(self): | ||||
|         self.ico = IcoFile(self.fp) | ||||
|         self.info['sizes'] = self.ico.sizes() | ||||
|         self.size = self.ico.entry[0]['dim'] | ||||
|         self.load() | ||||
| 
 | ||||
|         # check magic | ||||
|         s = self.fp.read(6) | ||||
|         if not _accept(s): | ||||
|             raise SyntaxError("not an ICO file") | ||||
| 
 | ||||
|         # pick the largest icon in the file | ||||
|         m = b"" | ||||
|         for i in range(i16(s[4:])): | ||||
|             s = self.fp.read(16) | ||||
|             if not m: | ||||
|                 m = s | ||||
|             elif i8(s[0]) > i8(m[0]) and i8(s[1]) > i8(m[1]): | ||||
|                 m = s | ||||
|             #print "width", i8(s[0]) | ||||
|             #print "height", i8(s[1]) | ||||
|             #print "colors", i8(s[2]) | ||||
|             #print "reserved", i8(s[3]) | ||||
|             #print "planes", i16(s[4:]) | ||||
|             #print "bitcount", i16(s[6:]) | ||||
|             #print "bytes", i32(s[8:]) | ||||
|             #print "offset", i32(s[12:]) | ||||
| 
 | ||||
|         # load as bitmap | ||||
|         self._bitmap(i32(m[12:])) | ||||
| 
 | ||||
|         # patch up the bitmap height | ||||
|         self.size = self.size[0], self.size[1]//2 | ||||
|         d, e, o, a = self.tile[0] | ||||
|         self.tile[0] = d, (0,0)+self.size, o, a | ||||
| 
 | ||||
|         return | ||||
| 
 | ||||
|     def load(self): | ||||
|         im = self.ico.getimage(self.size) | ||||
|         # if tile is PNG, it won't really be loaded yet | ||||
|         im.load() | ||||
|         self.im = im.im | ||||
|         self.mode = im.mode | ||||
|         self.size = im.size | ||||
| 
 | ||||
| # | ||||
| # -------------------------------------------------------------------- | ||||
| 
 | ||||
| Image.register_open("ICO", IcoImageFile, _accept) | ||||
| 
 | ||||
| Image.register_extension("ICO", ".ico") | ||||
|  |  | |||
|  | @ -9,6 +9,6 @@ data = open(file, "rb").read() | |||
| def test_sanity(): | ||||
|     im = Image.open(file) | ||||
|     im.load() | ||||
|     assert_equal(im.mode, "P") | ||||
|     assert_equal(im.mode, "RGBA") | ||||
|     assert_equal(im.size, (16, 16)) | ||||
|     assert_equal(im.format, "ICO") | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue
	
	Block a user