123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134 |
- from . import Image, ImageFile, JpegImagePlugin
- from ._binary import i16be as i16
- def _accept(prefix):
- return JpegImagePlugin._accept(prefix)
- def _save(im, fp, filename):
-
- return JpegImagePlugin._save(im, fp, filename)
- class MpoImageFile(JpegImagePlugin.JpegImageFile):
- format = "MPO"
- format_description = "MPO (CIPA DC-007)"
- _close_exclusive_fp_after_loading = False
- def _open(self):
- self.fp.seek(0)
- JpegImagePlugin.JpegImageFile._open(self)
- self._after_jpeg_open()
- def _after_jpeg_open(self, mpheader=None):
- self.mpinfo = mpheader if mpheader is not None else self._getmp()
- self.n_frames = self.mpinfo[0xB001]
- self.__mpoffsets = [
- mpent["DataOffset"] + self.info["mpoffset"] for mpent in self.mpinfo[0xB002]
- ]
- self.__mpoffsets[0] = 0
-
-
- assert self.n_frames == len(self.__mpoffsets)
- del self.info["mpoffset"]
- self.is_animated = self.n_frames > 1
- self.__fp = self.fp
- self.__fp.seek(self.__mpoffsets[0])
- self.__frame = 0
- self.offset = 0
-
- self.readonly = 1
- def load_seek(self, pos):
- self.__fp.seek(pos)
- def seek(self, frame):
- if not self._seek_check(frame):
- return
- self.fp = self.__fp
- self.offset = self.__mpoffsets[frame]
- self.fp.seek(self.offset + 2)
- segment = self.fp.read(2)
- if not segment:
- raise ValueError("No data found for frame")
- if i16(segment) == 0xFFE1:
- n = i16(self.fp.read(2)) - 2
- self.info["exif"] = ImageFile._safe_read(self.fp, n)
- exif = self.getexif()
- if 40962 in exif and 40963 in exif:
- self._size = (exif[40962], exif[40963])
- elif "exif" in self.info:
- del self.info["exif"]
- self.tile = [("jpeg", (0, 0) + self.size, self.offset, (self.mode, ""))]
- self.__frame = frame
- def tell(self):
- return self.__frame
- def _close__fp(self):
- try:
- if self.__fp != self.fp:
- self.__fp.close()
- except AttributeError:
- pass
- finally:
- self.__fp = None
- @staticmethod
- def adopt(jpeg_instance, mpheader=None):
- """
- Transform the instance of JpegImageFile into
- an instance of MpoImageFile.
- After the call, the JpegImageFile is extended
- to be an MpoImageFile.
- This is essentially useful when opening a JPEG
- file that reveals itself as an MPO, to avoid
- double call to _open.
- """
- jpeg_instance.__class__ = MpoImageFile
- jpeg_instance._after_jpeg_open(mpheader)
- return jpeg_instance
- Image.register_save(MpoImageFile.format, _save)
- Image.register_extension(MpoImageFile.format, ".mpo")
- Image.register_mime(MpoImageFile.format, "image/mpo")
|