2010-07-31 06:52:47 +04:00
|
|
|
#
|
|
|
|
# The Python Imaging Library.
|
|
|
|
# $Id$
|
|
|
|
#
|
|
|
|
# FLI/FLC file handling.
|
|
|
|
#
|
|
|
|
# History:
|
|
|
|
# 95-09-01 fl Created
|
|
|
|
# 97-01-03 fl Fixed parser, setup decoder tile
|
|
|
|
# 98-07-15 fl Renamed offset attribute to avoid name clash
|
|
|
|
#
|
|
|
|
# Copyright (c) Secret Labs AB 1997-98.
|
|
|
|
# Copyright (c) Fredrik Lundh 1995-97.
|
|
|
|
#
|
|
|
|
# See the README file for information on usage and redistribution.
|
|
|
|
#
|
|
|
|
|
|
|
|
|
2017-01-17 16:22:18 +03:00
|
|
|
from . import Image, ImageFile, ImagePalette
|
|
|
|
from ._binary import i8, i16le as i16, i32le as i32, o8
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
#
|
|
|
|
# decoder
|
|
|
|
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
def _accept(prefix):
|
2015-06-18 03:12:12 +03:00
|
|
|
return len(prefix) >= 6 and i16(prefix[4:6]) in [0xAF11, 0xAF12]
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2014-08-28 15:44:19 +04:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
##
|
|
|
|
# Image plugin for the FLI/FLC animation format. Use the <b>seek</b>
|
|
|
|
# method to load individual frames.
|
|
|
|
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
class FliImageFile(ImageFile.ImageFile):
|
|
|
|
|
|
|
|
format = "FLI"
|
|
|
|
format_description = "Autodesk FLI/FLC Animation"
|
2017-03-15 02:16:38 +03:00
|
|
|
_close_exclusive_fp_after_loading = False
|
2017-08-12 05:11:36 +03:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
def _open(self):
|
|
|
|
|
|
|
|
# HEAD
|
|
|
|
s = self.fp.read(128)
|
2019-03-21 16:28:20 +03:00
|
|
|
if not (
|
2020-06-23 10:41:13 +03:00
|
|
|
_accept(s)
|
2019-03-21 16:28:20 +03:00
|
|
|
and i16(s[14:16]) in [0, 3] # flags
|
|
|
|
and s[20:22] == b"\x00\x00" # reserved
|
|
|
|
):
|
2012-10-11 07:52:53 +04:00
|
|
|
raise SyntaxError("not an FLI/FLC file")
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2017-08-12 05:11:36 +03:00
|
|
|
# frames
|
2020-04-13 07:37:49 +03:00
|
|
|
self.n_frames = i16(s[6:8])
|
|
|
|
self.is_animated = self.n_frames > 1
|
2017-08-12 05:11:36 +03:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
# image characteristics
|
|
|
|
self.mode = "P"
|
2018-09-30 05:58:02 +03:00
|
|
|
self._size = i16(s[8:10]), i16(s[10:12])
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
# animation speed
|
|
|
|
duration = i32(s[16:20])
|
2020-06-23 10:41:13 +03:00
|
|
|
magic = i16(s[4:6])
|
2010-07-31 06:52:47 +04:00
|
|
|
if magic == 0xAF11:
|
2017-08-04 14:51:02 +03:00
|
|
|
duration = (duration * 1000) // 70
|
2010-07-31 06:52:47 +04:00
|
|
|
self.info["duration"] = duration
|
|
|
|
|
|
|
|
# look for palette
|
2014-08-28 15:44:19 +04:00
|
|
|
palette = [(a, a, a) for a in range(256)]
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
s = self.fp.read(16)
|
|
|
|
|
|
|
|
self.__offset = 128
|
|
|
|
|
|
|
|
if i16(s[4:6]) == 0xF100:
|
|
|
|
# prefix chunk; ignore it
|
|
|
|
self.__offset = self.__offset + i32(s)
|
|
|
|
s = self.fp.read(16)
|
|
|
|
|
|
|
|
if i16(s[4:6]) == 0xF1FA:
|
|
|
|
# look for palette chunk
|
|
|
|
s = self.fp.read(6)
|
|
|
|
if i16(s[4:6]) == 11:
|
|
|
|
self._palette(palette, 2)
|
|
|
|
elif i16(s[4:6]) == 4:
|
|
|
|
self._palette(palette, 0)
|
|
|
|
|
2019-03-21 16:28:20 +03:00
|
|
|
palette = [o8(r) + o8(g) + o8(b) for (r, g, b) in palette]
|
2012-10-24 17:25:00 +04:00
|
|
|
self.palette = ImagePalette.raw("RGB", b"".join(palette))
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
# set things up to decode first frame
|
2015-04-15 03:43:05 +03:00
|
|
|
self.__frame = -1
|
2010-07-31 06:52:47 +04:00
|
|
|
self.__fp = self.fp
|
2015-04-15 03:43:05 +03:00
|
|
|
self.__rewind = self.fp.tell()
|
2010-07-31 06:52:47 +04:00
|
|
|
self.seek(0)
|
|
|
|
|
|
|
|
def _palette(self, palette, shift):
|
|
|
|
# load palette
|
|
|
|
|
|
|
|
i = 0
|
|
|
|
for e in range(i16(self.fp.read(2))):
|
|
|
|
s = self.fp.read(2)
|
py3k: The big push
There are two main issues fixed with this commit:
* bytes vs. str: All file, image, and palette data are now handled as
bytes. A new _binary module consolidates the hacks needed to do this
across Python versions. tostring/fromstring methods have been renamed to
tobytes/frombytes, but the Python 2.6/2.7 versions alias them to the old
names for compatibility. Users should move to tobytes/frombytes.
One other potentially-breaking change is that text data in image files
(such as tags, comments) are now explicitly handled with a specific
character encoding in mind. This works well with the Unicode str in
Python 3, but may trip up old code expecting a straight byte-for-byte
translation to a Python string. This also required a change to Gohlke's
tags tests (in Tests/test_file_png.py) to expect Unicode strings from
the code.
* True div vs. floor div: Many division operations used the "/" operator
to do floor division, which is now the "//" operator in Python 3. These
were fixed.
As of this commit, on the first pass, I have one failing test (improper
handling of a slice object in a C module, test_imagepath.py) in Python 3,
and three that that I haven't tried running yet (test_imagegl,
test_imagegrab, and test_imageqt). I also haven't tested anything on
Windows. All but the three skipped tests run flawlessly against Pythons
2.6 and 2.7.
2012-10-21 01:01:53 +04:00
|
|
|
i = i + i8(s[0])
|
|
|
|
n = i8(s[1])
|
2010-07-31 06:52:47 +04:00
|
|
|
if n == 0:
|
|
|
|
n = 256
|
|
|
|
s = self.fp.read(n * 3)
|
|
|
|
for n in range(0, len(s), 3):
|
py3k: The big push
There are two main issues fixed with this commit:
* bytes vs. str: All file, image, and palette data are now handled as
bytes. A new _binary module consolidates the hacks needed to do this
across Python versions. tostring/fromstring methods have been renamed to
tobytes/frombytes, but the Python 2.6/2.7 versions alias them to the old
names for compatibility. Users should move to tobytes/frombytes.
One other potentially-breaking change is that text data in image files
(such as tags, comments) are now explicitly handled with a specific
character encoding in mind. This works well with the Unicode str in
Python 3, but may trip up old code expecting a straight byte-for-byte
translation to a Python string. This also required a change to Gohlke's
tags tests (in Tests/test_file_png.py) to expect Unicode strings from
the code.
* True div vs. floor div: Many division operations used the "/" operator
to do floor division, which is now the "//" operator in Python 3. These
were fixed.
As of this commit, on the first pass, I have one failing test (improper
handling of a slice object in a C module, test_imagepath.py) in Python 3,
and three that that I haven't tried running yet (test_imagegl,
test_imagegrab, and test_imageqt). I also haven't tested anything on
Windows. All but the three skipped tests run flawlessly against Pythons
2.6 and 2.7.
2012-10-21 01:01:53 +04:00
|
|
|
r = i8(s[n]) << shift
|
2019-03-21 16:28:20 +03:00
|
|
|
g = i8(s[n + 1]) << shift
|
|
|
|
b = i8(s[n + 2]) << shift
|
2010-07-31 06:52:47 +04:00
|
|
|
palette[i] = (r, g, b)
|
2014-05-10 08:36:15 +04:00
|
|
|
i += 1
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
def seek(self, frame):
|
2017-09-30 06:32:43 +03:00
|
|
|
if not self._seek_check(frame):
|
2015-04-15 03:43:05 +03:00
|
|
|
return
|
|
|
|
if frame < self.__frame:
|
|
|
|
self._seek(0)
|
2015-06-18 17:49:18 +03:00
|
|
|
|
2015-04-15 03:43:05 +03:00
|
|
|
for f in range(self.__frame + 1, frame + 1):
|
2017-08-12 05:11:36 +03:00
|
|
|
self._seek(f)
|
2015-04-15 03:43:05 +03:00
|
|
|
|
2015-04-23 21:03:11 +03:00
|
|
|
def _seek(self, frame):
|
2015-04-15 03:43:05 +03:00
|
|
|
if frame == 0:
|
|
|
|
self.__frame = -1
|
|
|
|
self.__fp.seek(self.__rewind)
|
2015-06-07 18:01:50 +03:00
|
|
|
self.__offset = 128
|
2018-11-26 10:52:51 +03:00
|
|
|
else:
|
|
|
|
# ensure that the previous frame was loaded
|
|
|
|
self.load()
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2015-04-15 03:43:05 +03:00
|
|
|
if frame != self.__frame + 1:
|
2012-10-11 07:52:53 +04:00
|
|
|
raise ValueError("cannot seek to frame %d" % frame)
|
2015-04-15 03:43:05 +03:00
|
|
|
self.__frame = frame
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
# move to next frame
|
|
|
|
self.fp = self.__fp
|
|
|
|
self.fp.seek(self.__offset)
|
|
|
|
|
|
|
|
s = self.fp.read(4)
|
|
|
|
if not s:
|
|
|
|
raise EOFError
|
|
|
|
|
|
|
|
framesize = i32(s)
|
|
|
|
|
|
|
|
self.decodermaxblock = framesize
|
2019-03-21 16:28:20 +03:00
|
|
|
self.tile = [("fli", (0, 0) + self.size, self.__offset, None)]
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2015-06-07 18:01:50 +03:00
|
|
|
self.__offset += framesize
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
def tell(self):
|
2015-04-15 03:43:05 +03:00
|
|
|
return self.__frame
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2018-11-17 13:56:06 +03:00
|
|
|
def _close__fp(self):
|
|
|
|
try:
|
2019-01-04 04:29:23 +03:00
|
|
|
if self.__fp != self.fp:
|
|
|
|
self.__fp.close()
|
2018-11-17 13:56:06 +03:00
|
|
|
except AttributeError:
|
|
|
|
pass
|
|
|
|
finally:
|
|
|
|
self.__fp = None
|
|
|
|
|
2017-08-04 14:51:02 +03:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
#
|
|
|
|
# registry
|
|
|
|
|
2015-07-04 16:29:58 +03:00
|
|
|
Image.register_open(FliImageFile.format, FliImageFile, _accept)
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2016-04-25 07:59:02 +03:00
|
|
|
Image.register_extensions(FliImageFile.format, [".fli", ".flc"])
|