mirror of
https://github.com/SickGear/SickGear.git
synced 2025-01-22 09:33:37 +00:00
980e05cc99
Backported 400 revisions from rev 1de4961-8897c5b (2018-2014). Move core/benchmark, core/cmd_line, core/memory, core/profiler and core/timeout to core/optional/* Remove metadata/qt* PORT: Version 2.0a3 (inline with 3.0a3 @ f80c7d5). Basic Support for XMP Packets. tga: improvements to adhere more closely to the spec. pdf: slightly improved parsing. rar: fix TypeError on unknown block types. Add MacRoman win32 codepage. tiff/exif: support SubIFDs and tiled images. Add method to export metadata in dictionary. mpeg_video: don't attempt to parse Stream past length. mpeg_video: parse ESCR correctly, add SCR value. Change centralise CustomFragments. field: don't set parser class if class is None, to enable autodetect. field: add value/display for CustomFragment. parser: inline warning to enable tracebacks in debug mode. Fix empty bytestrings in makePrintable. Fix contentSize in jpeg.py to account for image_data blocks. Fix the ELF parser. Enhance the AR archive parser. elf parser: fix wrong wrong fields order in parsing little endian section flags. elf parser: add s390 as a machine type. Flesh out mp4 parser. PORT: Version 2.0a1 (inline with 3.0a1). Major refactoring and PEP8. Fix ResourceWarning warnings on files. Add a close() method and support for the context manager protocol ("with obj: ...") to parsers, input and output streams. metadata: get comment from ZIP. Support for InputIOStream.read(0). Fix sizeGe when size is None. Remove unused new_seekable_field_set file. Remove parser Mapsforge .map. Remove parser Parallel Realities Starfighter .pak files. sevenzip: fix for newer archives. java: update access flags and modifiers for Java 1.7 and update description text for most recent Java. Support ustar prefix field in tar archives. Remove file_system* parsers. Remove misc parsers 3d0, 3ds, gnome_keyring, msoffice*, mstask, ole*, word*. Remove program parsers macho, nds, prc. Support non-8bit Character subclasses. Python parser supports Python 3.7. Enhance mpeg_ts parser to support MTS/M2TS. Support for creation date in tiff. Change don't hardcode errno constant. PORT: 1.9.1 Internal Only: The following are legacy reference to upstream commit messages. Relevant changes up to b0a115f8. Use integer division. Replace HACHOIR_ERRORS with Exception. Fix metadata.Data: make it sortable. Import fixes from e7de492. PORT: Version 2.0a1 (inline with 3.0a1 @ e9f8fad). Replace hachoir.core.field with hachoir.field Replace hachoir.core.stream with hachoir.stream Remove the compatibility module for PY1.5 to PY2.5. metadata: support TIFF picture. metadata: fix string normalization. metadata: fix datetime regex Fix hachoir bug #57. FileFromInputStream: fix comparison between None and an int. InputIOStream: open the file in binary mode.
202 lines
6.7 KiB
Python
202 lines
6.7 KiB
Python
"""
|
|
Microsoft Bitmap picture parser.
|
|
- file extension: ".bmp"
|
|
|
|
Author: Victor Stinner
|
|
Creation: 16 december 2005
|
|
"""
|
|
|
|
from hachoir.parser import Parser
|
|
from hachoir.field import (FieldSet,
|
|
UInt8, UInt16, UInt32, Bits,
|
|
String, RawBytes, Enum,
|
|
PaddingBytes, NullBytes, createPaddingField)
|
|
from hachoir.core.endian import LITTLE_ENDIAN
|
|
from hachoir.core.text_handler import textHandler, hexadecimal
|
|
from hachoir.parser.image.common import RGB, PaletteRGBA
|
|
from hachoir.core.tools import alignValue
|
|
|
|
|
|
class Pixel4bit(Bits):
|
|
static_size = 4
|
|
|
|
def __init__(self, parent, name):
|
|
Bits.__init__(self, parent, name, 4)
|
|
|
|
|
|
class ImageLine(FieldSet):
|
|
def __init__(self, parent, name, width, pixel_class):
|
|
FieldSet.__init__(self, parent, name)
|
|
self._pixel = pixel_class
|
|
self._width = width
|
|
self._size = alignValue(self._width * self._pixel.static_size, 32)
|
|
|
|
def createFields(self):
|
|
for x in xrange(self._width):
|
|
yield self._pixel(self, "pixel[]")
|
|
size = self.size - self.current_size
|
|
if size:
|
|
yield createPaddingField(self, size)
|
|
|
|
|
|
class ImagePixels(FieldSet):
|
|
def __init__(self, parent, name, width, height, pixel_class, size=None):
|
|
FieldSet.__init__(self, parent, name, size=size)
|
|
self._width = width
|
|
self._height = height
|
|
self._pixel = pixel_class
|
|
|
|
def createFields(self):
|
|
for y in xrange(self._height - 1, -1, -1):
|
|
yield ImageLine(self, "line[%u]" % y, self._width, self._pixel)
|
|
size = (self.size - self.current_size) // 8
|
|
if size:
|
|
yield NullBytes(self, "padding", size)
|
|
|
|
|
|
class CIEXYZ(FieldSet):
|
|
def createFields(self):
|
|
yield UInt32(self, "x")
|
|
yield UInt32(self, "y")
|
|
yield UInt32(self, "z")
|
|
|
|
|
|
class BmpHeader(FieldSet):
|
|
color_space_name = {
|
|
1: "Business (Saturation)",
|
|
2: "Graphics (Relative)",
|
|
4: "Images (Perceptual)",
|
|
8: "Absolute colormetric (Absolute)",
|
|
}
|
|
|
|
def getFormatVersion(self):
|
|
if "gamma_blue" in self:
|
|
return 4
|
|
if "important_color" in self:
|
|
return 3
|
|
return 2
|
|
|
|
def createFields(self):
|
|
# Version 2 (12 bytes)
|
|
yield UInt32(self, "header_size", "Header size")
|
|
yield UInt32(self, "width", "Width (pixels)")
|
|
yield UInt32(self, "height", "Height (pixels)")
|
|
yield UInt16(self, "nb_plan", "Number of plan (=1)")
|
|
yield UInt16(self, "bpp", "Bits per pixel") # may be zero for PNG/JPEG picture
|
|
|
|
# Version 3 (40 bytes)
|
|
if self["header_size"].value < 40:
|
|
return
|
|
yield Enum(UInt32(self, "compression", "Compression method"), BmpFile.COMPRESSION_NAME)
|
|
yield UInt32(self, "image_size", "Image size (bytes)")
|
|
yield UInt32(self, "horizontal_dpi", "Horizontal DPI")
|
|
yield UInt32(self, "vertical_dpi", "Vertical DPI")
|
|
yield UInt32(self, "used_colors", "Number of color used")
|
|
yield UInt32(self, "important_color", "Number of import colors")
|
|
|
|
# Version 4 (108 bytes)
|
|
if self["header_size"].value < 108:
|
|
return
|
|
yield textHandler(UInt32(self, "red_mask"), hexadecimal)
|
|
yield textHandler(UInt32(self, "green_mask"), hexadecimal)
|
|
yield textHandler(UInt32(self, "blue_mask"), hexadecimal)
|
|
yield textHandler(UInt32(self, "alpha_mask"), hexadecimal)
|
|
yield Enum(UInt32(self, "color_space"), self.color_space_name)
|
|
yield CIEXYZ(self, "red_primary")
|
|
yield CIEXYZ(self, "green_primary")
|
|
yield CIEXYZ(self, "blue_primary")
|
|
yield UInt32(self, "gamma_red")
|
|
yield UInt32(self, "gamma_green")
|
|
yield UInt32(self, "gamma_blue")
|
|
|
|
|
|
def parseImageData(parent, name, size, header):
|
|
if ("compression" not in header) or (header["compression"].value in (0, 3)):
|
|
width = header["width"].value
|
|
height = header["height"].value
|
|
bpp = header["bpp"].value
|
|
if bpp == 32:
|
|
cls = UInt32
|
|
elif bpp == 24:
|
|
cls = RGB
|
|
elif bpp == 8:
|
|
cls = UInt8
|
|
elif bpp == 4:
|
|
cls = Pixel4bit
|
|
else:
|
|
cls = None
|
|
if cls:
|
|
return ImagePixels(parent, name, width, height, cls, size=size * 8)
|
|
return RawBytes(parent, name, size)
|
|
|
|
|
|
class BmpFile(Parser):
|
|
PARSER_TAGS = {
|
|
"id": "bmp",
|
|
"category": "image",
|
|
"file_ext": ("bmp",),
|
|
"mime": (u"image/x-ms-bmp", u"image/x-bmp"),
|
|
"min_size": 30 * 8,
|
|
# "magic": (("BM", 0),),
|
|
"magic_regex": ((
|
|
# "BM", <filesize>, <reserved>, header_size=(12|40|108)
|
|
"BM.{4}.{8}[\x0C\x28\x6C]\0{3}",
|
|
0),),
|
|
"description": "Microsoft bitmap (BMP) picture"
|
|
}
|
|
endian = LITTLE_ENDIAN
|
|
|
|
COMPRESSION_NAME = {
|
|
0: u"Uncompressed",
|
|
1: u"RLE 8-bit",
|
|
2: u"RLE 4-bit",
|
|
3: u"Bitfields",
|
|
4: u"JPEG",
|
|
5: u"PNG",
|
|
}
|
|
|
|
def validate(self):
|
|
if self.stream.readBytes(0, 2) != 'BM':
|
|
return "Wrong file signature"
|
|
if self["header/header_size"].value not in (12, 40, 108):
|
|
return "Unknown header size (%s)" % self["header_size"].value
|
|
if self["header/nb_plan"].value != 1:
|
|
return "Invalid number of planes"
|
|
return True
|
|
|
|
def createFields(self):
|
|
yield String(self, "signature", 2, "Header (\"BM\")", charset="ASCII")
|
|
yield UInt32(self, "file_size", "File size (bytes)")
|
|
yield PaddingBytes(self, "reserved", 4, "Reserved")
|
|
yield UInt32(self, "data_start", "Data start position")
|
|
yield BmpHeader(self, "header")
|
|
|
|
# Compute number of color
|
|
header = self["header"]
|
|
bpp = header["bpp"].value
|
|
if 0 < bpp <= 8:
|
|
if "used_colors" in header and header["used_colors"].value:
|
|
nb_color = header["used_colors"].value
|
|
else:
|
|
nb_color = (1 << bpp)
|
|
else:
|
|
nb_color = 0
|
|
|
|
# Color palette (if any)
|
|
if nb_color:
|
|
yield PaletteRGBA(self, "palette", nb_color)
|
|
|
|
# Seek to data start
|
|
field = self.seekByte(self["data_start"].value)
|
|
if field:
|
|
yield field
|
|
|
|
# Image pixels
|
|
size = min(self["file_size"].value - self["data_start"].value, (self.size - self.current_size) // 8)
|
|
yield parseImageData(self, "pixels", size, header)
|
|
|
|
def createDescription(self):
|
|
return u"Microsoft Bitmap version %s" % self["header"].getFormatVersion()
|
|
|
|
def createContentSize(self):
|
|
return self["file_size"].value * 8
|