home *** CD-ROM | disk | FTP | other *** search
/ Enter 2004 April / enter-2004-04.iso / files / EVE_1424_100181.exe / JpegImagePlugin.py < prev    next >
Encoding:
Text File  |  2004-04-20  |  11.3 KB  |  351 lines

  1. #
  2. # The Python Imaging Library.
  3. # $Id: //modules/pil/PIL/JpegImagePlugin.py#2 $
  4. #
  5. # JPEG (JFIF) file handling
  6. #
  7. # See "Digital Compression and Coding of Continous-Tone Still Images,
  8. # Part 1, Requirements and Guidelines" (CCITT T.81 / ISO 10918-1)
  9. #
  10. # History:
  11. # 1995-09-09 fl   Created
  12. # 1995-09-13 fl   Added full parser
  13. # 1996-03-25 fl   Added hack to use the IJG command line utilities
  14. # 1996-05-05 fl   Workaround Photoshop 2.5 CMYK polarity bug
  15. # 1996-05-28 fl   Added draft support, JFIF version (0.1)
  16. # 1996-12-30 fl   Added encoder options, added progression property (0.2)
  17. # 1997-08-27 fl   Save mode 1 images as BW (0.3)
  18. # 1998-07-12 fl   Added YCbCr to draft and save methods (0.4)
  19. # 1998-10-19 fl   Don't hang on files using 16-bit DQT's (0.4.1)
  20. # 2001-04-16 fl   Extract DPI settings from JFIF files (0.4.2)
  21. #
  22. # Copyright (c) 1997-2001 by Secret Labs AB.
  23. # Copyright (c) 1995-1996 by Fredrik Lundh.
  24. #
  25. # See the README file for information on usage and redistribution.
  26. #
  27.  
  28. __version__ = "0.4.2"
  29.  
  30. import array, string
  31. import Image, ImageFile
  32.  
  33. def i16(c,o=0):
  34.     return ord(c[o+1]) + (ord(c[o])<<8)
  35.  
  36. def i32(c,o=0):
  37.     return ord(c[o+3]) + (ord(c[o+2])<<8) + (ord(c[o+1])<<16) + (ord(c[o])<<24)
  38.  
  39. #
  40. # Parser
  41.  
  42. def Skip(self, marker):
  43.     self.fp.read(i16(self.fp.read(2))-2)
  44.  
  45. def APP(self, marker):
  46.     #
  47.     # Application marker.  Store these in the APP dictionary.
  48.     # Also look for well-known application markers.
  49.  
  50.     s = self.fp.read(i16(self.fp.read(2))-2)
  51.  
  52.     self.app["APP%d" % (marker&15)] = s
  53.  
  54.     if marker == 0xFFE0 and s[:4] == "JFIF":
  55.         self.info["jfif"] = version = i16(s, 5) # version
  56.         self.info["jfif_version"] = divmod(version, 256)
  57.         # extract JFIF properties
  58.         try:
  59.             jfif_unit = ord(s[7])
  60.             jfif_density = i16(s, 8), i16(s, 10)
  61.         except:
  62.             pass
  63.         else:
  64.             if jfif_unit == 1:
  65.                 self.info["dpi"] = jfif_density
  66.             self.info["jfif_unit"] = jfif_unit
  67.             self.info["jfif_density"] = jfif_density
  68.     elif marker == 0xFFEE and s[:5] == "Adobe":
  69.         self.info["adobe"] = i16(s, 5)
  70.         # extract Adobe custom properties
  71.         try:
  72.             adobe_transform = ord(s[1])
  73.         except:
  74.             pass
  75.         else:
  76.             self.info["adobe_transform"] = adobe_transform
  77.  
  78. def SOF(self, marker):
  79.     #
  80.     # Start of frame marker.  Defines the size and mode of the
  81.     # image.  JPEG is colour blind, so we use some simple
  82.     # heuristics to map the number of layers to an appropriate
  83.     # mode.  Note that this could be made a bit brighter, by
  84.     # looking for JFIF and Adobe APP markers.
  85.  
  86.     s = self.fp.read(i16(self.fp.read(2))-2)
  87.     self.size = i16(s[3:]), i16(s[1:])
  88.  
  89.     self.bits = ord(s[0])
  90.     if self.bits != 8:
  91.         raise SyntaxError, "cannot handle %d-bit layers" % self.bits
  92.  
  93.     self.layers = ord(s[5])
  94.     if self.layers == 1:
  95.         self.mode = "L"
  96.     elif self.layers == 3:
  97.         self.mode = "RGB"
  98.     elif self.layers == 4:
  99.         self.mode = "CMYK"
  100.     else:
  101.         raise SyntaxError, "cannot handle %d-layer images" % self.layers
  102.  
  103.     if marker in [0xFFC2, 0xFFC6, 0xFFCA, 0xFFCE]:
  104.         self.info["progression"] = 1
  105.  
  106.     for i in range(6, len(s), 3):
  107.         t = s[i:i+3]
  108.         # 4-tuples: id, vsamp, hsamp, qtable
  109.         self.layer.append((t[0], ord(t[1])/16, ord(t[1])&15, ord(t[2])))
  110.  
  111. def DQT(self, marker):
  112.     #
  113.     # Define quantization table.  Support baseline 8-bit tables
  114.     # only.  Note that there might be more than one table in
  115.     # each marker.
  116.  
  117.     # FIXME: The quantization tables can be used to estimate the
  118.     # compression quality.
  119.  
  120.     s = self.fp.read(i16(self.fp.read(2))-2)
  121.     while len(s):
  122.         if len(s) < 65:
  123.             raise SyntaxError, "bad quantization table marker"
  124.         v = ord(s[0])
  125.         if v/16 == 0:
  126.             self.quantization[v&15] = array.array("b", s[1:65])
  127.             s = s[65:]
  128.         else:
  129.             return # FIXME: add code to read 16-bit tables!
  130.             # raise SyntaxError, "bad quantization table element size"
  131.  
  132.  
  133. #
  134. # JPEG marker table
  135.  
  136. MARKER = {
  137.     0xFFC0: ("SOF0", "Baseline DCT", SOF),
  138.     0xFFC1: ("SOF1", "Extended Sequential DCT", SOF),
  139.     0xFFC2: ("SOF2", "Progressive DCT", SOF),
  140.     0xFFC3: ("SOF3", "Spatial lossless", SOF),
  141.     0xFFC4: ("DHT", "Define Huffman table", Skip),
  142.     0xFFC5: ("SOF5", "Differential sequential DCT", SOF),
  143.     0xFFC6: ("SOF6", "Differential progressive DCT", SOF),
  144.     0xFFC7: ("SOF7", "Differential spatial", SOF),
  145.     0xFFC8: ("JPG", "Extension", None),
  146.     0xFFC9: ("SOF9", "Extended sequential DCT (AC)", SOF),
  147.     0xFFCA: ("SOF10", "Progressive DCT (AC)", SOF),
  148.     0xFFCB: ("SOF11", "Spatial lossless DCT (AC)", SOF),
  149.     0xFFCC: ("DAC", "Define arithmetic coding conditioning", Skip),
  150.     0xFFCD: ("SOF13", "Differential sequential DCT (AC)", SOF),
  151.     0xFFCE: ("SOF14", "Differential progressive DCT (AC)", SOF),
  152.     0xFFCF: ("SOF15", "Differential spatial (AC)", SOF),
  153.     0xFFD0: ("RST0", "Restart 0", None),
  154.     0xFFD1: ("RST1", "Restart 1", None),
  155.     0xFFD2: ("RST2", "Restart 2", None),
  156.     0xFFD3: ("RST3", "Restart 3", None),
  157.     0xFFD4: ("RST4", "Restart 4", None),
  158.     0xFFD5: ("RST5", "Restart 5", None),
  159.     0xFFD6: ("RST6", "Restart 6", None),
  160.     0xFFD7: ("RST7", "Restart 7", None),
  161.     0xFFD8: ("SOI", "Start of image", None),
  162.     0xFFD9: ("EOI", "End of image", None),
  163.     0xFFDA: ("SOS", "Start of scan", Skip),
  164.     0xFFDB: ("DQT", "Define quantization table", DQT),
  165.     0xFFDC: ("DNL", "Define number of lines", Skip),
  166.     0xFFDD: ("DRI", "Define restart interval", Skip),
  167.     0xFFDE: ("DHP", "Define hierarchical progression", SOF),
  168.     0xFFDF: ("EXP", "Expand reference component", Skip),
  169.     0xFFE0: ("APP0", "Application segment 0", APP),
  170.     0xFFE1: ("APP1", "Application segment 1", APP),
  171.     0xFFE2: ("APP2", "Application segment 2", APP),
  172.     0xFFE3: ("APP3", "Application segment 3", APP),
  173.     0xFFE4: ("APP4", "Application segment 4", APP),
  174.     0xFFE5: ("APP5", "Application segment 5", APP),
  175.     0xFFE6: ("APP6", "Application segment 6", APP),
  176.     0xFFE7: ("APP7", "Application segment 7", APP),
  177.     0xFFE8: ("APP8", "Application segment 8", APP),
  178.     0xFFE9: ("APP9", "Application segment 9", APP),
  179.     0xFFEA: ("APP10", "Application segment 10", APP),
  180.     0xFFEB: ("APP11", "Application segment 11", APP),
  181.     0xFFEC: ("APP12", "Application segment 12", APP),
  182.     0xFFED: ("APP13", "Application segment 13", APP),
  183.     0xFFEE: ("APP14", "Application segment 14", APP),
  184.     0xFFEF: ("APP15", "Application segment 15", APP),
  185.     0xFFF0: ("JPG0", "Extension 0", None),
  186.     0xFFF1: ("JPG1", "Extension 1", None),
  187.     0xFFF2: ("JPG2", "Extension 2", None),
  188.     0xFFF3: ("JPG3", "Extension 3", None),
  189.     0xFFF4: ("JPG4", "Extension 4", None),
  190.     0xFFF5: ("JPG5", "Extension 5", None),
  191.     0xFFF6: ("JPG6", "Extension 6", None),
  192.     0xFFF7: ("JPG7", "Extension 7", None),
  193.     0xFFF8: ("JPG8", "Extension 8", None),
  194.     0xFFF9: ("JPG9", "Extension 9", None),
  195.     0xFFFA: ("JPG10", "Extension 10", None),
  196.     0xFFFB: ("JPG11", "Extension 11", None),
  197.     0xFFFC: ("JPG12", "Extension 12", None),
  198.     0xFFFD: ("JPG13", "Extension 13", None),
  199.     0xFFFE: ("COM", "Comment", Skip)
  200. }
  201.  
  202.  
  203. def _accept(prefix):
  204.     return prefix[0] == "\377"
  205.  
  206. class JpegImageFile(ImageFile.ImageFile):
  207.  
  208.     format = "JPEG"
  209.     format_description = "JPEG (ISO 10918)"
  210.  
  211.     def _open(self):
  212.  
  213.         s = self.fp.read(1)
  214.  
  215.         if ord(s[0]) != 255:
  216.             raise SyntaxError, "not an JPEG file"
  217.  
  218.         # Create attributes
  219.         self.bits = self.layers = 0
  220.  
  221.         # JPEG specifics (internal)
  222.         self.layer = []
  223.         self.huffman_dc = {}
  224.         self.huffman_ac = {}
  225.         self.quantization = {}
  226.         self.app = {}
  227.  
  228.         while 1:
  229.  
  230.             s = s + self.fp.read(1)
  231.  
  232.             i = i16(s)
  233.  
  234.             if MARKER.has_key(i):
  235.                 name, description, handler = MARKER[i]
  236.                 # print hex(i), name, description
  237.                 if handler is not None:
  238.                     handler(self, i)
  239.                 if i == 0xFFDA: # start of scan
  240.                     rawmode = self.mode
  241.                     if self.mode == "CMYK" and self.info.has_key("adobe"):
  242.                         rawmode = "CMYK;I" # Photoshop 2.5 is broken!
  243.                     self.tile = [("jpeg", (0,0) + self.size, 0, (rawmode, ""))]
  244.                     # self.__offset = self.fp.tell()
  245.                     break
  246.                 s = self.fp.read(1)
  247.             else:
  248.                 raise SyntaxError, "no marker found"
  249.  
  250.     def draft(self, mode, size):
  251.  
  252.         if len(self.tile) != 1:
  253.             return
  254.  
  255.         d, e, o, a = self.tile[0]
  256.         scale = 0
  257.  
  258.         if a[0] == "RGB" and mode in ["L", "YCbCr"]:
  259.             self.mode = mode
  260.             a = mode, ""
  261.  
  262.         if size:
  263.             scale = max(self.size[0] / size[0], self.size[1] / size[1])
  264.             for s in [8, 4, 2, 1]:
  265.                 if scale >= s:
  266.                     break
  267.             e = e[0], e[1], (e[2]-e[0]+s-1)/s+e[0], (e[3]-e[1]+s-1)/s+e[1]
  268.             self.size = ((self.size[0]+s-1)/s, (self.size[1]+s-1)/s)
  269.             scale = s
  270.  
  271.         self.tile = [(d, e, o, a)]
  272.         self.decoderconfig = (scale, 1)
  273.  
  274.         return self
  275.  
  276.     def load_djpeg(self):
  277.  
  278.         # ALTERNATIVE: handle JPEGs via the IJG command line utilities
  279.  
  280.         import tempfile, os
  281.         file = tempfile.mktemp()
  282.         os.system("djpeg %s >%s" % (self.filename, file))
  283.  
  284.         try:
  285.             self.im = Image.core.open_ppm(file)
  286.         finally:
  287.             try: os.unlink(file)
  288.             except: pass
  289.  
  290.         self.mode = self.im.mode
  291.         self.size = self.im.size
  292.  
  293.         self.tile = []
  294.  
  295.  
  296. def _fetch(dict, key, default = 0):
  297.     try:
  298.         return dict[key]
  299.     except KeyError:
  300.         return default
  301.  
  302. RAWMODE = {
  303.     "1": "L",
  304.     "L": "L",
  305.     "RGB": "RGB",
  306.     "RGBA": "RGB",
  307.     "RGBX": "RGB",
  308.     "CMYK": "CMYK",
  309.     "YCbCr": "YCbCr",
  310. }
  311.  
  312. def _save(im, fp, filename):
  313.  
  314.     try:
  315.         rawmode = RAWMODE[im.mode]
  316.     except KeyError:
  317.         raise IOError, "cannot write mode %s as JPEG" % im.mode
  318.  
  319.     dpi = _fetch(im.encoderinfo, "dpi", (0, 0))
  320.  
  321.     # get keyword arguments
  322.     im.encoderconfig = (_fetch(im.encoderinfo, "quality", 0),
  323.                         im.encoderinfo.has_key("progressive"),
  324.                         _fetch(im.encoderinfo, "smooth", 0),
  325.                         im.encoderinfo.has_key("optimize"),
  326.                         _fetch(im.encoderinfo, "streamtype", 0),
  327.                         dpi[0], dpi[1])
  328.  
  329.     ImageFile._save(im, fp, [("jpeg", (0,0)+im.size, 0, rawmode)])
  330.  
  331. def _save_cjpeg(im, fp, filename):
  332.     # ALTERNATIVE: handle JPEGs via the IJG command line utilities.
  333.     import os
  334.     file = im._dump()
  335.     os.system("cjpeg %s >%s" % (file, filename))
  336.     try: os.unlink(file)
  337.     except: pass
  338.  
  339. # -------------------------------------------------------------------q-
  340. # Registry stuff
  341.  
  342. Image.register_open("JPEG", JpegImageFile, _accept)
  343. Image.register_save("JPEG", _save)
  344.  
  345. Image.register_extension("JPEG", ".jfif")
  346. Image.register_extension("JPEG", ".jpe")
  347. Image.register_extension("JPEG", ".jpg")
  348. Image.register_extension("JPEG", ".jpeg")
  349.  
  350. Image.register_mime("JPEG", "image/jpeg")
  351.