Former-commit-id:2b462d8665
[formerly133dc97f67
] [formerlya02aeb236c
] [formerlya02aeb236c
[formerly9f19e3f712
]] [formerly2b462d8665
[formerly133dc97f67
] [formerlya02aeb236c
] [formerlya02aeb236c
[formerly9f19e3f712
]] [formerly06a8b51d6d
[formerlya02aeb236c
[formerly9f19e3f712
] [formerly06a8b51d6d
[formerly 64fa9254b946eae7e61bbc3f513b7c3696c4f54f]]]]] Former-commit-id:06a8b51d6d
Former-commit-id:2c3569dd39
[formerly9bb8decbcf
] [formerly8e80217e59
] [formerlye2ecdcfe33
[formerly377dcd10b9
] [formerly8e80217e59
[formerly3360eb6c5f
]]] Former-commit-id:e2ecdcfe33
[formerly377dcd10b9
] Former-commit-id:e2ecdcfe33
Former-commit-id:7dbd17a5aa
142 lines
3.4 KiB
Python
142 lines
3.4 KiB
Python
#
|
|
# The Python Imaging Library.
|
|
# $Id: FliImagePlugin.py 2134 2004-10-06 08:55:20Z fredrik $
|
|
#
|
|
# 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.
|
|
#
|
|
|
|
|
|
__version__ = "0.2"
|
|
|
|
import Image, ImageFile, ImagePalette
|
|
import string
|
|
|
|
|
|
def i16(c):
|
|
return ord(c[0]) + (ord(c[1])<<8)
|
|
|
|
def i32(c):
|
|
return ord(c[0]) + (ord(c[1])<<8) + (ord(c[2])<<16) + (ord(c[3])<<24)
|
|
|
|
#
|
|
# decoder
|
|
|
|
def _accept(prefix):
|
|
return i16(prefix[4:6]) in [0xAF11, 0xAF12]
|
|
|
|
##
|
|
# Image plugin for the FLI/FLC animation format. Use the <b>seek</b>
|
|
# method to load individual frames.
|
|
|
|
class FliImageFile(ImageFile.ImageFile):
|
|
|
|
format = "FLI"
|
|
format_description = "Autodesk FLI/FLC Animation"
|
|
|
|
def _open(self):
|
|
|
|
# HEAD
|
|
s = self.fp.read(128)
|
|
magic = i16(s[4:6])
|
|
if magic not in [0xAF11, 0xAF12]:
|
|
raise SyntaxError, "not an FLI/FLC file"
|
|
|
|
# image characteristics
|
|
self.mode = "P"
|
|
self.size = i16(s[8:10]), i16(s[10:12])
|
|
|
|
# animation speed
|
|
duration = i32(s[16:20])
|
|
if magic == 0xAF11:
|
|
duration = (duration * 1000) / 70
|
|
self.info["duration"] = duration
|
|
|
|
# look for palette
|
|
palette = map(lambda a: (a,a,a), range(256))
|
|
|
|
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)
|
|
|
|
palette = map(lambda (r,g,b): chr(r)+chr(g)+chr(b), palette)
|
|
self.palette = ImagePalette.raw("RGB", string.join(palette, ""))
|
|
|
|
# set things up to decode first frame
|
|
self.frame = -1
|
|
self.__fp = self.fp
|
|
|
|
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)
|
|
i = i + ord(s[0])
|
|
n = ord(s[1])
|
|
if n == 0:
|
|
n = 256
|
|
s = self.fp.read(n * 3)
|
|
for n in range(0, len(s), 3):
|
|
r = ord(s[n]) << shift
|
|
g = ord(s[n+1]) << shift
|
|
b = ord(s[n+2]) << shift
|
|
palette[i] = (r, g, b)
|
|
i = i + 1
|
|
|
|
def seek(self, frame):
|
|
|
|
if frame != self.frame + 1:
|
|
raise ValueError, "cannot seek to frame %d" % frame
|
|
self.frame = frame
|
|
|
|
# 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
|
|
self.tile = [("fli", (0,0)+self.size, self.__offset, None)]
|
|
|
|
self.__offset = self.__offset + framesize
|
|
|
|
def tell(self):
|
|
|
|
return self.frame
|
|
|
|
#
|
|
# registry
|
|
|
|
Image.register_open("FLI", FliImageFile, _accept)
|
|
|
|
Image.register_extension("FLI", ".fli")
|
|
Image.register_extension("FLI", ".flc")
|