2014-07-16 19:36:56 +04:00
|
|
|
#
|
|
|
|
# The Python Imaging Library.
|
|
|
|
# $Id$
|
|
|
|
#
|
|
|
|
# MPO file handling
|
|
|
|
#
|
|
|
|
# See "Multi-Picture Format" (CIPA DC-007-Translation 2009, Standard of the
|
|
|
|
# Camera & Imaging Products Association)
|
|
|
|
#
|
|
|
|
# The multi-picture object combines multiple JPEG images (with a modified EXIF
|
|
|
|
# data format) into a single file. While it can theoretically be used much like
|
|
|
|
# a GIF animation, it is commonly used to represent 3D photographs and is (as
|
|
|
|
# of this writing) the most commonly used format by 3D cameras.
|
|
|
|
#
|
|
|
|
# History:
|
|
|
|
# 2014-03-13 Feneric Created
|
|
|
|
#
|
|
|
|
# See the README file for information on usage and redistribution.
|
|
|
|
#
|
2023-12-21 14:13:31 +03:00
|
|
|
from __future__ import annotations
|
2014-07-16 19:36:56 +04:00
|
|
|
|
2022-07-16 13:02:58 +03:00
|
|
|
import itertools
|
|
|
|
import os
|
|
|
|
import struct
|
2024-06-05 01:29:28 +03:00
|
|
|
from typing import IO
|
2022-07-16 13:02:58 +03:00
|
|
|
|
2022-11-28 00:39:56 +03:00
|
|
|
from . import (
|
|
|
|
Image,
|
|
|
|
ImageSequence,
|
|
|
|
JpegImagePlugin,
|
|
|
|
TiffImagePlugin,
|
|
|
|
)
|
2022-07-16 13:02:58 +03:00
|
|
|
from ._binary import o32le
|
2014-07-16 19:36:56 +04:00
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2024-06-10 07:15:28 +03:00
|
|
|
def _save(im: Image.Image, fp: IO[bytes], filename: str | bytes) -> None:
|
2022-07-16 13:02:58 +03:00
|
|
|
JpegImagePlugin._save(im, fp, filename)
|
|
|
|
|
|
|
|
|
2024-06-11 16:26:00 +03:00
|
|
|
def _save_all(im: Image.Image, fp: IO[bytes], filename: str | bytes) -> None:
|
2022-07-16 13:02:58 +03:00
|
|
|
append_images = im.encoderinfo.get("append_images", [])
|
2024-06-11 16:26:00 +03:00
|
|
|
if not append_images and not getattr(im, "is_animated", False):
|
|
|
|
_save(im, fp, filename)
|
|
|
|
return
|
2022-07-16 13:02:58 +03:00
|
|
|
|
2022-12-22 07:31:36 +03:00
|
|
|
mpf_offset = 28
|
2024-06-11 16:26:00 +03:00
|
|
|
offsets: list[int] = []
|
2022-07-16 13:02:58 +03:00
|
|
|
for imSequence in itertools.chain([im], append_images):
|
|
|
|
for im_frame in ImageSequence.Iterator(imSequence):
|
|
|
|
if not offsets:
|
|
|
|
# APP2 marker
|
2022-12-22 07:31:36 +03:00
|
|
|
im_frame.encoderinfo["extra"] = (
|
2022-11-13 00:00:20 +03:00
|
|
|
b"\xFF\xE2" + struct.pack(">H", 6 + 82) + b"MPF\0" + b" " * 82
|
2022-07-16 13:02:58 +03:00
|
|
|
)
|
2022-12-22 07:31:36 +03:00
|
|
|
exif = im_frame.encoderinfo.get("exif")
|
|
|
|
if isinstance(exif, Image.Exif):
|
|
|
|
exif = exif.tobytes()
|
|
|
|
im_frame.encoderinfo["exif"] = exif
|
|
|
|
if exif:
|
|
|
|
mpf_offset += 4 + len(exif)
|
|
|
|
|
2022-07-16 13:02:58 +03:00
|
|
|
JpegImagePlugin._save(im_frame, fp, filename)
|
|
|
|
offsets.append(fp.tell())
|
|
|
|
else:
|
|
|
|
im_frame.save(fp, "JPEG")
|
|
|
|
offsets.append(fp.tell() - offsets[-1])
|
|
|
|
|
|
|
|
ifd = TiffImagePlugin.ImageFileDirectory_v2()
|
2022-11-13 00:00:20 +03:00
|
|
|
ifd[0xB000] = b"0100"
|
2022-07-16 13:02:58 +03:00
|
|
|
ifd[0xB001] = len(offsets)
|
|
|
|
|
|
|
|
mpentries = b""
|
|
|
|
data_offset = 0
|
|
|
|
for i, size in enumerate(offsets):
|
|
|
|
if i == 0:
|
|
|
|
mptype = 0x030000 # Baseline MP Primary Image
|
|
|
|
else:
|
|
|
|
mptype = 0x000000 # Undefined
|
|
|
|
mpentries += struct.pack("<LLLHH", mptype, size, data_offset, 0, 0)
|
|
|
|
if i == 0:
|
2022-12-22 07:31:36 +03:00
|
|
|
data_offset -= mpf_offset
|
2022-07-16 13:02:58 +03:00
|
|
|
data_offset += size
|
|
|
|
ifd[0xB002] = mpentries
|
|
|
|
|
2022-12-22 07:31:36 +03:00
|
|
|
fp.seek(mpf_offset)
|
2022-07-16 13:02:58 +03:00
|
|
|
fp.write(b"II\x2A\x00" + o32le(8) + ifd.tobytes(8))
|
|
|
|
fp.seek(0, os.SEEK_END)
|
2014-07-16 19:36:56 +04:00
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2014-07-16 19:36:56 +04:00
|
|
|
##
|
|
|
|
# Image plugin for MPO images.
|
|
|
|
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2014-07-16 19:36:56 +04:00
|
|
|
class MpoImageFile(JpegImagePlugin.JpegImageFile):
|
|
|
|
format = "MPO"
|
|
|
|
format_description = "MPO (CIPA DC-007)"
|
2017-03-15 02:16:38 +03:00
|
|
|
_close_exclusive_fp_after_loading = False
|
2018-01-27 09:02:56 +03:00
|
|
|
|
2024-05-11 03:48:09 +03:00
|
|
|
def _open(self) -> None:
|
2014-07-24 19:16:12 +04:00
|
|
|
self.fp.seek(0) # prep the fp in order to pass the JPEG test
|
2014-07-16 19:36:56 +04:00
|
|
|
JpegImagePlugin.JpegImageFile._open(self)
|
2019-03-24 11:48:23 +03:00
|
|
|
self._after_jpeg_open()
|
2019-02-16 00:07:11 +03:00
|
|
|
|
2019-03-24 11:48:23 +03:00
|
|
|
def _after_jpeg_open(self, mpheader=None):
|
|
|
|
self.mpinfo = mpheader if mpheader is not None else self._getmp()
|
2020-04-13 07:37:49 +03:00
|
|
|
self.n_frames = self.mpinfo[0xB001]
|
2014-07-22 21:31:51 +04:00
|
|
|
self.__mpoffsets = [
|
|
|
|
mpent["DataOffset"] + self.info["mpoffset"] for mpent in self.mpinfo[0xB002]
|
2014-08-28 15:44:19 +04:00
|
|
|
]
|
2014-07-23 02:23:45 +04:00
|
|
|
self.__mpoffsets[0] = 0
|
2014-07-28 19:14:38 +04:00
|
|
|
# Note that the following assertion will only be invalid if something
|
|
|
|
# gets broken within JpegImagePlugin.
|
2020-04-13 07:37:49 +03:00
|
|
|
assert self.n_frames == len(self.__mpoffsets)
|
2014-07-22 21:31:51 +04:00
|
|
|
del self.info["mpoffset"] # no longer needed
|
2020-04-13 07:37:49 +03:00
|
|
|
self.is_animated = self.n_frames > 1
|
2022-04-13 02:54:17 +03:00
|
|
|
self._fp = self.fp # FIXME: hack
|
|
|
|
self._fp.seek(self.__mpoffsets[0]) # get ready to read first frame
|
2014-07-22 21:31:51 +04:00
|
|
|
self.__frame = 0
|
|
|
|
self.offset = 0
|
2014-07-23 19:27:46 +04:00
|
|
|
# for now we can only handle reading and individual frame extraction
|
|
|
|
self.readonly = 1
|
|
|
|
|
2024-05-15 13:19:09 +03:00
|
|
|
def load_seek(self, pos: int) -> None:
|
2022-04-13 02:54:17 +03:00
|
|
|
self._fp.seek(pos)
|
2014-07-16 19:36:56 +04:00
|
|
|
|
2024-05-04 13:51:54 +03:00
|
|
|
def seek(self, frame: int) -> None:
|
2017-09-30 06:32:43 +03:00
|
|
|
if not self._seek_check(frame):
|
|
|
|
return
|
2022-04-13 02:54:17 +03:00
|
|
|
self.fp = self._fp
|
2017-09-30 06:32:43 +03:00
|
|
|
self.offset = self.__mpoffsets[frame]
|
2019-03-28 00:32:33 +03:00
|
|
|
|
2024-03-16 10:40:16 +03:00
|
|
|
original_exif = self.info.get("exif")
|
|
|
|
if "exif" in self.info:
|
|
|
|
del self.info["exif"]
|
|
|
|
|
2019-03-28 00:32:33 +03:00
|
|
|
self.fp.seek(self.offset + 2) # skip SOI marker
|
2024-03-16 10:40:16 +03:00
|
|
|
if not self.fp.read(2):
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = "No data found for frame"
|
|
|
|
raise ValueError(msg)
|
2024-03-16 10:40:16 +03:00
|
|
|
self.fp.seek(self.offset)
|
|
|
|
JpegImagePlugin.JpegImageFile._open(self)
|
|
|
|
if self.info.get("exif") != original_exif:
|
2022-05-27 00:54:54 +03:00
|
|
|
self._reload_exif()
|
2019-03-28 00:32:33 +03:00
|
|
|
|
2024-03-16 10:40:16 +03:00
|
|
|
self.tile = [("jpeg", (0, 0) + self.size, self.offset, self.tile[0][-1])]
|
2014-07-16 19:36:56 +04:00
|
|
|
self.__frame = frame
|
|
|
|
|
2024-05-04 13:51:54 +03:00
|
|
|
def tell(self) -> int:
|
2014-07-16 19:36:56 +04:00
|
|
|
return self.__frame
|
|
|
|
|
2019-02-16 00:08:14 +03:00
|
|
|
@staticmethod
|
|
|
|
def adopt(jpeg_instance, mpheader=None):
|
2019-03-07 15:22:14 +03:00
|
|
|
"""
|
2019-02-16 00:08:14 +03:00
|
|
|
Transform the instance of JpegImageFile into
|
2019-03-24 00:11:32 +03:00
|
|
|
an instance of MpoImageFile.
|
2019-02-16 00:08:14 +03:00
|
|
|
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.
|
2019-03-07 15:22:20 +03:00
|
|
|
"""
|
2019-02-16 00:08:14 +03:00
|
|
|
jpeg_instance.__class__ = MpoImageFile
|
|
|
|
jpeg_instance._after_jpeg_open(mpheader)
|
|
|
|
return jpeg_instance
|
|
|
|
|
2014-07-16 19:36:56 +04:00
|
|
|
|
2018-08-09 13:54:16 +03:00
|
|
|
# ---------------------------------------------------------------------
|
2014-07-16 19:36:56 +04:00
|
|
|
# Registry stuff
|
|
|
|
|
2014-07-28 19:14:38 +04:00
|
|
|
# Note that since MPO shares a factory with JPEG, we do not need to do a
|
|
|
|
# separate registration for it here.
|
2015-07-04 16:29:58 +03:00
|
|
|
# Image.register_open(MpoImageFile.format,
|
|
|
|
# JpegImagePlugin.jpeg_factory, _accept)
|
|
|
|
Image.register_save(MpoImageFile.format, _save)
|
2022-07-16 13:02:58 +03:00
|
|
|
Image.register_save_all(MpoImageFile.format, _save_all)
|
2014-07-16 19:36:56 +04:00
|
|
|
|
2015-07-04 16:29:58 +03:00
|
|
|
Image.register_extension(MpoImageFile.format, ".mpo")
|
2014-07-16 19:36:56 +04:00
|
|
|
|
2015-07-04 16:29:58 +03:00
|
|
|
Image.register_mime(MpoImageFile.format, "image/mpo")
|