FliImagePlugin.py 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  1. #
  2. # The Python Imaging Library.
  3. # $Id$
  4. #
  5. # FLI/FLC file handling.
  6. #
  7. # History:
  8. # 95-09-01 fl Created
  9. # 97-01-03 fl Fixed parser, setup decoder tile
  10. # 98-07-15 fl Renamed offset attribute to avoid name clash
  11. #
  12. # Copyright (c) Secret Labs AB 1997-98.
  13. # Copyright (c) Fredrik Lundh 1995-97.
  14. #
  15. # See the README file for information on usage and redistribution.
  16. #
  17. from __future__ import annotations
  18. import os
  19. from . import Image, ImageFile, ImagePalette
  20. from ._binary import i16le as i16
  21. from ._binary import i32le as i32
  22. from ._binary import o8
  23. from ._util import DeferredError
  24. #
  25. # decoder
  26. def _accept(prefix: bytes) -> bool:
  27. return (
  28. len(prefix) >= 16
  29. and i16(prefix, 4) in [0xAF11, 0xAF12]
  30. and i16(prefix, 14) in [0, 3] # flags
  31. )
  32. ##
  33. # Image plugin for the FLI/FLC animation format. Use the <b>seek</b>
  34. # method to load individual frames.
  35. class FliImageFile(ImageFile.ImageFile):
  36. format = "FLI"
  37. format_description = "Autodesk FLI/FLC Animation"
  38. _close_exclusive_fp_after_loading = False
  39. def _open(self) -> None:
  40. # HEAD
  41. assert self.fp is not None
  42. s = self.fp.read(128)
  43. if not (
  44. _accept(s)
  45. and s[20:22] == b"\x00" * 2
  46. and s[42:80] == b"\x00" * 38
  47. and s[88:] == b"\x00" * 40
  48. ):
  49. msg = "not an FLI/FLC file"
  50. raise SyntaxError(msg)
  51. # frames
  52. self.n_frames = i16(s, 6)
  53. self.is_animated = self.n_frames > 1
  54. # image characteristics
  55. self._mode = "P"
  56. self._size = i16(s, 8), i16(s, 10)
  57. # animation speed
  58. duration = i32(s, 16)
  59. magic = i16(s, 4)
  60. if magic == 0xAF11:
  61. duration = (duration * 1000) // 70
  62. self.info["duration"] = duration
  63. # look for palette
  64. palette = [(a, a, a) for a in range(256)]
  65. s = self.fp.read(16)
  66. self.__offset = 128
  67. if i16(s, 4) == 0xF100:
  68. # prefix chunk; ignore it
  69. self.fp.seek(self.__offset + i32(s))
  70. s = self.fp.read(16)
  71. if i16(s, 4) == 0xF1FA:
  72. # look for palette chunk
  73. number_of_subchunks = i16(s, 6)
  74. chunk_size: int | None = None
  75. for _ in range(number_of_subchunks):
  76. if chunk_size is not None:
  77. self.fp.seek(chunk_size - 6, os.SEEK_CUR)
  78. s = self.fp.read(6)
  79. chunk_type = i16(s, 4)
  80. if chunk_type in (4, 11):
  81. self._palette(palette, 2 if chunk_type == 11 else 0)
  82. break
  83. chunk_size = i32(s)
  84. if not chunk_size:
  85. break
  86. self.palette = ImagePalette.raw(
  87. "RGB", b"".join(o8(r) + o8(g) + o8(b) for (r, g, b) in palette)
  88. )
  89. # set things up to decode first frame
  90. self.__frame = -1
  91. self._fp = self.fp
  92. self.__rewind = self.fp.tell()
  93. self.seek(0)
  94. def _palette(self, palette: list[tuple[int, int, int]], shift: int) -> None:
  95. # load palette
  96. i = 0
  97. assert self.fp is not None
  98. for e in range(i16(self.fp.read(2))):
  99. s = self.fp.read(2)
  100. i = i + s[0]
  101. n = s[1]
  102. if n == 0:
  103. n = 256
  104. s = self.fp.read(n * 3)
  105. for n in range(0, len(s), 3):
  106. r = s[n] << shift
  107. g = s[n + 1] << shift
  108. b = s[n + 2] << shift
  109. palette[i] = (r, g, b)
  110. i += 1
  111. def seek(self, frame: int) -> None:
  112. if not self._seek_check(frame):
  113. return
  114. if frame < self.__frame:
  115. self._seek(0)
  116. for f in range(self.__frame + 1, frame + 1):
  117. self._seek(f)
  118. def _seek(self, frame: int) -> None:
  119. if isinstance(self._fp, DeferredError):
  120. raise self._fp.ex
  121. if frame == 0:
  122. self.__frame = -1
  123. self._fp.seek(self.__rewind)
  124. self.__offset = 128
  125. else:
  126. # ensure that the previous frame was loaded
  127. self.load()
  128. if frame != self.__frame + 1:
  129. msg = f"cannot seek to frame {frame}"
  130. raise ValueError(msg)
  131. self.__frame = frame
  132. # move to next frame
  133. self.fp = self._fp
  134. self.fp.seek(self.__offset)
  135. s = self.fp.read(4)
  136. if not s:
  137. msg = "missing frame size"
  138. raise EOFError(msg)
  139. framesize = i32(s)
  140. self.decodermaxblock = framesize
  141. self.tile = [ImageFile._Tile("fli", (0, 0) + self.size, self.__offset)]
  142. self.__offset += framesize
  143. def tell(self) -> int:
  144. return self.__frame
  145. #
  146. # registry
  147. Image.register_open(FliImageFile.format, FliImageFile, _accept)
  148. Image.register_extensions(FliImageFile.format, [".fli", ".flc"])