| |
|
| |
|
| |
|
| |
|
| |
|
| |
|
| |
|
| |
|
| |
|
| |
|
| |
|
| |
|
| |
|
| |
|
| |
|
| |
|
| | from __future__ import annotations
|
| |
|
| | import os
|
| |
|
| | from . import Image, ImageFile, ImagePalette
|
| | from ._binary import i16le as i16
|
| | from ._binary import i32le as i32
|
| | from ._binary import o8
|
| | from ._util import DeferredError
|
| |
|
| |
|
| |
|
| |
|
| |
|
| | def _accept(prefix: bytes) -> bool:
|
| | return (
|
| | len(prefix) >= 6
|
| | and i16(prefix, 4) in [0xAF11, 0xAF12]
|
| | and i16(prefix, 14) in [0, 3]
|
| | )
|
| |
|
| |
|
| |
|
| |
|
| |
|
| |
|
| |
|
| | class FliImageFile(ImageFile.ImageFile):
|
| | format = "FLI"
|
| | format_description = "Autodesk FLI/FLC Animation"
|
| | _close_exclusive_fp_after_loading = False
|
| |
|
| | def _open(self) -> None:
|
| |
|
| | s = self.fp.read(128)
|
| | if not (_accept(s) and s[20:22] == b"\x00\x00"):
|
| | msg = "not an FLI/FLC file"
|
| | raise SyntaxError(msg)
|
| |
|
| |
|
| | self.n_frames = i16(s, 6)
|
| | self.is_animated = self.n_frames > 1
|
| |
|
| |
|
| | self._mode = "P"
|
| | self._size = i16(s, 8), i16(s, 10)
|
| |
|
| |
|
| | duration = i32(s, 16)
|
| | magic = i16(s, 4)
|
| | if magic == 0xAF11:
|
| | duration = (duration * 1000) // 70
|
| | self.info["duration"] = duration
|
| |
|
| |
|
| | palette = [(a, a, a) for a in range(256)]
|
| |
|
| | s = self.fp.read(16)
|
| |
|
| | self.__offset = 128
|
| |
|
| | if i16(s, 4) == 0xF100:
|
| |
|
| | self.__offset = self.__offset + i32(s)
|
| | self.fp.seek(self.__offset)
|
| | s = self.fp.read(16)
|
| |
|
| | if i16(s, 4) == 0xF1FA:
|
| |
|
| | number_of_subchunks = i16(s, 6)
|
| | chunk_size: int | None = None
|
| | for _ in range(number_of_subchunks):
|
| | if chunk_size is not None:
|
| | self.fp.seek(chunk_size - 6, os.SEEK_CUR)
|
| | s = self.fp.read(6)
|
| | chunk_type = i16(s, 4)
|
| | if chunk_type in (4, 11):
|
| | self._palette(palette, 2 if chunk_type == 11 else 0)
|
| | break
|
| | chunk_size = i32(s)
|
| | if not chunk_size:
|
| | break
|
| |
|
| | self.palette = ImagePalette.raw(
|
| | "RGB", b"".join(o8(r) + o8(g) + o8(b) for (r, g, b) in palette)
|
| | )
|
| |
|
| |
|
| | self.__frame = -1
|
| | self._fp = self.fp
|
| | self.__rewind = self.fp.tell()
|
| | self.seek(0)
|
| |
|
| | def _palette(self, palette: list[tuple[int, int, int]], shift: int) -> None:
|
| |
|
| |
|
| | i = 0
|
| | for e in range(i16(self.fp.read(2))):
|
| | s = self.fp.read(2)
|
| | i = i + s[0]
|
| | n = s[1]
|
| | if n == 0:
|
| | n = 256
|
| | s = self.fp.read(n * 3)
|
| | for n in range(0, len(s), 3):
|
| | r = s[n] << shift
|
| | g = s[n + 1] << shift
|
| | b = s[n + 2] << shift
|
| | palette[i] = (r, g, b)
|
| | i += 1
|
| |
|
| | def seek(self, frame: int) -> None:
|
| | if not self._seek_check(frame):
|
| | return
|
| | if frame < self.__frame:
|
| | self._seek(0)
|
| |
|
| | for f in range(self.__frame + 1, frame + 1):
|
| | self._seek(f)
|
| |
|
| | def _seek(self, frame: int) -> None:
|
| | if isinstance(self._fp, DeferredError):
|
| | raise self._fp.ex
|
| | if frame == 0:
|
| | self.__frame = -1
|
| | self._fp.seek(self.__rewind)
|
| | self.__offset = 128
|
| | else:
|
| |
|
| | self.load()
|
| |
|
| | if frame != self.__frame + 1:
|
| | msg = f"cannot seek to frame {frame}"
|
| | raise ValueError(msg)
|
| | self.__frame = frame
|
| |
|
| |
|
| | self.fp = self._fp
|
| | self.fp.seek(self.__offset)
|
| |
|
| | s = self.fp.read(4)
|
| | if not s:
|
| | msg = "missing frame size"
|
| | raise EOFError(msg)
|
| |
|
| | framesize = i32(s)
|
| |
|
| | self.decodermaxblock = framesize
|
| | self.tile = [ImageFile._Tile("fli", (0, 0) + self.size, self.__offset)]
|
| |
|
| | self.__offset += framesize
|
| |
|
| | def tell(self) -> int:
|
| | return self.__frame
|
| |
|
| |
|
| |
|
| |
|
| |
|
| | Image.register_open(FliImageFile.format, FliImageFile, _accept)
|
| |
|
| | Image.register_extensions(FliImageFile.format, [".fli", ".flc"])
|
| |
|