2021-05-24 17:38:56 +00:00
from typing import Any , Dict , Generator , List , Set , Tuple , Optional , Union
2023-02-17 03:32:27 +00:00
from PIL import Image
2021-04-15 23:18:33 +00:00
2021-08-03 17:04:20 +00:00
from . blend import affine_composite , perspective_composite
2021-05-25 02:01:17 +00:00
from . swf import (
SWF ,
Frame ,
Tag ,
AP2ShapeTag ,
AP2DefineSpriteTag ,
AP2PlaceObjectTag ,
AP2RemoveObjectTag ,
AP2DoActionTag ,
AP2DefineFontTag ,
AP2DefineEditTextTag ,
2021-08-11 18:31:37 +00:00
AP2DefineMorphShapeTag ,
2021-05-25 02:01:17 +00:00
AP2PlaceCameraTag ,
AP2ImageTag ,
)
2021-05-23 20:32:21 +00:00
from . decompile import ByteCode
2021-05-29 22:11:15 +00:00
from . types import (
Color ,
2022-10-16 19:30:02 +00:00
HSL ,
2021-05-29 22:11:15 +00:00
Matrix ,
Point ,
Rectangle ,
2021-08-09 19:09:00 +00:00
AAMode ,
2021-05-29 22:11:15 +00:00
AP2Trigger ,
AP2Action ,
PushAction ,
StoreRegisterAction ,
StringConstant ,
Register ,
NULL ,
UNDEFINED ,
GLOBAL ,
ROOT ,
PARENT ,
THIS ,
CLIP ,
)
2021-05-10 22:26:26 +00:00
from . geo import Shape , DrawParams
2021-04-15 23:18:33 +00:00
from . util import VerboseOutput
2021-05-10 22:26:26 +00:00
class RegisteredClip :
2021-04-17 23:30:46 +00:00
# A movie clip that we are rendering, frame by frame. These are manifest by the root
2021-04-20 21:41:28 +00:00
# SWF as well as AP2DefineSpriteTags which are essentially embedded movie clips. The
# tag_id is the AP2DefineSpriteTag that created us, or None if this is the clip for
# the root of the movie.
2022-10-15 18:56:30 +00:00
def __init__ (
self ,
tag_id : Optional [ int ] ,
frames : List [ Frame ] ,
tags : List [ Tag ] ,
labels : Dict [ str , int ] ,
) - > None :
2021-04-15 23:18:33 +00:00
self . tag_id = tag_id
self . frames = frames
self . tags = tags
2021-05-25 02:01:17 +00:00
self . labels = labels
2021-04-15 23:18:33 +00:00
2021-05-10 22:26:26 +00:00
def __repr__ ( self ) - > str :
return f " RegisteredClip(tag_id= { self . tag_id } ) "
2021-04-15 23:18:33 +00:00
2022-11-20 23:01:26 +00:00
@property
def reference ( self ) - > str :
return " anonymous sprite "
2021-04-15 23:18:33 +00:00
2021-05-10 22:26:26 +00:00
class RegisteredShape :
# A shape that we are rendering, as placed by some placed clip somewhere.
2022-10-15 18:56:30 +00:00
def __init__ (
self ,
tag_id : int ,
2022-11-20 23:01:26 +00:00
reference : str ,
2022-10-15 18:56:30 +00:00
vertex_points : List [ Point ] ,
tex_points : List [ Point ] ,
tex_colors : List [ Color ] ,
draw_params : List [ DrawParams ] ,
) - > None :
2021-05-10 22:26:26 +00:00
self . tag_id = tag_id
2022-11-20 23:01:26 +00:00
self . __reference = reference
2021-05-10 22:26:26 +00:00
self . vertex_points : List [ Point ] = vertex_points
self . tex_points : List [ Point ] = tex_points
self . tex_colors : List [ Color ] = tex_colors
self . draw_params : List [ DrawParams ] = draw_params
2023-02-17 03:32:27 +00:00
self . rectangle : Optional [ Image . Image ] = None
2021-04-17 23:31:08 +00:00
2022-11-20 23:01:26 +00:00
@property
def reference ( self ) - > str :
textures = { dp . region for dp in self . draw_params if dp . region is not None }
if textures :
vals = " , " . join ( textures )
return f " { self . __reference } , { vals } "
else :
return f " { self . __reference } , untextured "
2021-05-10 22:26:26 +00:00
def __repr__ ( self ) - > str :
2022-11-20 23:01:26 +00:00
return f " RegisteredShape(tag_id= { self . tag_id } , reference= { self . reference } vertex_points= { self . vertex_points } , tex_points= { self . tex_points } , tex_colors= { self . tex_colors } , draw_params= { self . draw_params } ) "
2021-05-10 22:26:26 +00:00
2021-05-25 02:01:17 +00:00
class RegisteredImage :
# An image that we should draw directly.
def __init__ ( self , tag_id : int , reference : str ) - > None :
self . tag_id = tag_id
self . reference = reference
def __repr__ ( self ) - > str :
return f " RegisteredImage(tag_id= { self . tag_id } , reference= { self . reference } ) "
2021-05-21 16:58:01 +00:00
class RegisteredDummy :
# An imported tag that we could not find.
def __init__ ( self , tag_id : int ) - > None :
self . tag_id = tag_id
def __repr__ ( self ) - > str :
return f " RegisteredDummy(tag_id= { self . tag_id } ) "
2022-11-20 23:01:26 +00:00
@property
def reference ( self ) - > str :
return " anonymous dummy "
2021-05-21 16:58:01 +00:00
2021-05-23 20:37:18 +00:00
class Mask :
def __init__ ( self , bounds : Rectangle ) - > None :
self . bounds = bounds
self . rectangle : Optional [ Image . Image ] = None
2021-05-10 22:26:26 +00:00
class PlacedObject :
# An object that occupies the screen at some depth.
2021-08-05 17:31:28 +00:00
def __init__ (
self ,
object_id : int ,
depth : int ,
rotation_origin : Point ,
transform : Matrix ,
projection : int ,
mult_color : Color ,
add_color : Color ,
2022-10-16 19:30:02 +00:00
hsl_shift : HSL ,
2021-08-05 17:31:28 +00:00
blend : int ,
mask : Optional [ Mask ] ,
) - > None :
2021-05-10 22:26:26 +00:00
self . __object_id = object_id
self . __depth = depth
2021-07-06 21:58:32 +00:00
self . rotation_origin = rotation_origin
2021-05-10 22:26:26 +00:00
self . transform = transform
2021-08-05 17:31:28 +00:00
self . projection = projection
2021-05-10 22:26:26 +00:00
self . mult_color = mult_color
self . add_color = add_color
2022-10-16 19:30:02 +00:00
self . hsl_shift = hsl_shift
2021-05-10 22:26:26 +00:00
self . blend = blend
2021-05-23 20:32:21 +00:00
self . mask = mask
2021-07-29 22:02:10 +00:00
self . visible : bool = True
2021-04-20 21:41:28 +00:00
2021-04-17 23:30:46 +00:00
@property
2022-10-15 18:56:30 +00:00
def source (
self ,
) - > Union [ RegisteredClip , RegisteredShape , RegisteredImage , RegisteredDummy ] :
2021-05-10 22:26:26 +00:00
raise NotImplementedError ( " Only implemented in subclass! " )
2021-04-15 23:18:33 +00:00
2021-04-17 23:30:46 +00:00
@property
2021-05-10 22:26:26 +00:00
def depth ( self ) - > int :
return self . __depth
2021-04-17 23:31:08 +00:00
@property
2021-05-10 22:26:26 +00:00
def object_id ( self ) - > int :
return self . __object_id
2021-04-17 23:30:46 +00:00
def __repr__ ( self ) - > str :
2021-05-10 22:26:26 +00:00
return f " PlacedObject(object_id= { self . object_id } , depth= { self . depth } ) "
2021-04-15 23:18:33 +00:00
2021-05-10 22:26:26 +00:00
class PlacedShape ( PlacedObject ) :
# A shape that occupies its parent clip at some depth. Placed by an AP2PlaceObjectTag
# referencing an AP2ShapeTag.
2021-05-23 20:32:21 +00:00
def __init__ (
self ,
object_id : int ,
depth : int ,
2021-07-06 21:58:32 +00:00
rotation_origin : Point ,
2021-05-23 20:32:21 +00:00
transform : Matrix ,
2021-08-05 17:31:28 +00:00
projection : int ,
2021-05-23 20:32:21 +00:00
mult_color : Color ,
add_color : Color ,
2022-10-16 19:30:02 +00:00
hsl_shift : HSL ,
2021-05-23 20:32:21 +00:00
blend : int ,
2021-05-23 20:37:18 +00:00
mask : Optional [ Mask ] ,
2021-05-23 20:32:21 +00:00
source : RegisteredShape ,
) - > None :
2022-10-15 18:56:30 +00:00
super ( ) . __init__ (
object_id ,
depth ,
rotation_origin ,
transform ,
projection ,
mult_color ,
add_color ,
2022-10-16 19:30:02 +00:00
hsl_shift ,
2022-10-15 18:56:30 +00:00
blend ,
mask ,
)
2021-05-10 22:26:26 +00:00
self . __source = source
2021-04-15 23:18:33 +00:00
2021-04-17 23:30:46 +00:00
@property
2021-05-10 22:26:26 +00:00
def source ( self ) - > RegisteredShape :
return self . __source
def __repr__ ( self ) - > str :
return f " PlacedShape(object_id= { self . object_id } , depth= { self . depth } , source= { self . source } ) "
class PlacedClip ( PlacedObject ) :
# A movieclip that occupies its parent clip at some depth. Placed by an AP2PlaceObjectTag
# referencing an AP2DefineSpriteTag. Essentially an embedded movie clip.
2021-05-23 20:32:21 +00:00
def __init__ (
self ,
object_id : int ,
depth : int ,
2021-07-06 21:58:32 +00:00
rotation_origin : Point ,
2021-05-23 20:32:21 +00:00
transform : Matrix ,
2021-08-05 17:31:28 +00:00
projection : int ,
2021-05-23 20:32:21 +00:00
mult_color : Color ,
add_color : Color ,
2022-10-16 19:30:02 +00:00
hsl_shift : HSL ,
2021-05-23 20:32:21 +00:00
blend : int ,
2021-05-23 20:37:18 +00:00
mask : Optional [ Mask ] ,
2021-05-23 20:32:21 +00:00
source : RegisteredClip ,
) - > None :
2022-10-15 18:56:30 +00:00
super ( ) . __init__ (
object_id ,
depth ,
rotation_origin ,
transform ,
projection ,
mult_color ,
add_color ,
2022-10-16 19:30:02 +00:00
hsl_shift ,
2022-10-15 18:56:30 +00:00
blend ,
mask ,
)
2021-05-10 22:26:26 +00:00
self . placed_objects : List [ PlacedObject ] = [ ]
self . frame : int = 0
2021-05-24 18:35:43 +00:00
self . unplayed_tags : List [ int ] = [ i for i in range ( len ( source . tags ) ) ]
2021-05-10 22:26:26 +00:00
self . __source = source
2021-05-24 01:24:26 +00:00
# Dynamic properties that are adjustable by SWF bytecode.
2021-05-23 20:32:21 +00:00
self . playing : bool = True
self . requested_frame : Optional [ int ] = None
2021-07-29 22:02:10 +00:00
self . visible_frame : int = - 1
2021-04-17 23:30:46 +00:00
2022-07-26 23:25:35 +00:00
# Root clip resizing, which we don't really support.
self . __width = 0
self . __height = 0
2021-04-17 23:30:46 +00:00
@property
2021-05-10 22:26:26 +00:00
def source ( self ) - > RegisteredClip :
return self . __source
2021-07-31 01:15:15 +00:00
def __check_visible ( self ) - > None :
if self . visible_frame > = 0 and self . frame > = self . visible_frame :
self . visible = True
self . visible_frame = - 1
2021-05-10 22:26:26 +00:00
def advance ( self ) - > None :
if self . frame < len ( self . source . frames ) :
self . frame + = 1
2021-07-31 01:15:15 +00:00
self . __check_visible ( )
2021-05-10 22:26:26 +00:00
2021-05-24 18:35:43 +00:00
def rewind ( self ) - > None :
self . frame = 0
self . unplayed_tags = [ i for i in range ( len ( self . __source . tags ) ) ]
self . placed_objects = [ ]
2021-07-31 01:15:15 +00:00
self . __check_visible ( )
2021-05-24 18:35:43 +00:00
2021-05-10 22:26:26 +00:00
@property
def finished ( self ) - > bool :
return self . frame == len ( self . source . frames )
2021-04-17 23:30:46 +00:00
def __repr__ ( self ) - > str :
2021-06-12 17:17:02 +00:00
return (
2022-10-15 18:56:30 +00:00
f " PlacedClip(object_id= { self . object_id } , depth= { self . depth } , source= { self . source } , frame= { self . frame } , "
+ f " requested_frame= { self . requested_frame } , total_frames= { len ( self . source . frames ) } , playing= { self . playing } , "
+ f " finished= { self . finished } ) "
2021-06-12 17:17:02 +00:00
)
2021-04-17 23:30:46 +00:00
2021-05-25 02:01:17 +00:00
def __resolve_frame ( self , frame : Any ) - > Optional [ int ] :
if isinstance ( frame , int ) :
return frame
if isinstance ( frame , str ) :
if frame in self . __source . labels :
return self . __source . labels [ frame ]
return None
2021-05-23 20:32:21 +00:00
# The following are attributes and functions necessary to support some simple bytecode.
2021-05-23 20:33:05 +00:00
def gotoAndStop ( self , frame : Any ) - > None :
2021-05-25 02:01:17 +00:00
actual_frame = self . __resolve_frame ( frame )
if actual_frame is None :
print ( f " WARNING: Unrecognized frame { frame } to gotoAndStop function! " )
2021-05-23 20:33:05 +00:00
return
2021-08-10 23:36:39 +00:00
if actual_frame < = 0 :
actual_frame = 1
if actual_frame > len ( self . source . frames ) :
actual_frame = len ( self . source . frames )
2021-05-25 02:01:17 +00:00
self . requested_frame = actual_frame
2021-05-23 20:33:05 +00:00
self . playing = False
2021-05-23 20:32:21 +00:00
def gotoAndPlay ( self , frame : Any ) - > None :
2021-05-25 02:01:17 +00:00
actual_frame = self . __resolve_frame ( frame )
if actual_frame is None :
2021-05-23 20:32:21 +00:00
print ( f " WARNING: Non-integer frame { frame } to gotoAndPlay function! " )
return
2021-08-10 23:36:39 +00:00
if actual_frame < = 0 :
actual_frame = 1
if actual_frame > len ( self . source . frames ) :
actual_frame = len ( self . source . frames )
2021-05-25 02:01:17 +00:00
self . requested_frame = actual_frame
2021-05-23 20:32:21 +00:00
self . playing = True
2021-05-23 20:33:05 +00:00
def stop ( self ) - > None :
self . playing = False
def play ( self ) - > None :
self . playing = True
2021-07-29 22:02:10 +00:00
def setInvisibleUntil ( self , frame : Any ) - > None :
actual_frame = self . __resolve_frame ( frame )
if actual_frame is None :
print ( f " WARNING: Non-integer frame { frame } to setInvisibleUntil function! " )
return
2021-09-20 05:05:14 +00:00
actual_frame + = self . frameOffset - 1
2021-07-29 22:02:10 +00:00
self . visible = False
2021-08-10 23:36:39 +00:00
if actual_frame < = 0 :
actual_frame = 1
if actual_frame > len ( self . source . frames ) :
actual_frame = len ( self . source . frames )
2021-07-29 22:02:10 +00:00
self . visible_frame = actual_frame
2021-09-20 05:04:33 +00:00
self . __check_visible ( )
2021-07-29 22:02:10 +00:00
2021-05-23 20:32:21 +00:00
@property
def frameOffset ( self ) - > int :
return self . requested_frame or self . frame
@frameOffset.setter
def frameOffset ( self , val : Any ) - > None :
2021-06-12 17:17:02 +00:00
actual_frame = self . __resolve_frame ( val )
if actual_frame is None :
2021-05-23 20:32:21 +00:00
print ( f " WARNING: Non-integer frameOffset { val } to frameOffset attribute! " )
return
2021-08-10 23:36:39 +00:00
if actual_frame < 0 :
actual_frame = 0
if actual_frame > = len ( self . source . frames ) :
actual_frame = len ( self . source . frames ) - 1
2021-06-12 17:17:02 +00:00
self . requested_frame = actual_frame + 1
2021-05-23 20:32:21 +00:00
2022-07-26 22:35:46 +00:00
@property
def _visible ( self ) - > int :
return 1 if self . visible else 0
@_visible.setter
def _visible ( self , val : Any ) - > None :
self . visible = val != 0
2022-07-26 23:25:35 +00:00
@property
def _width ( self ) - > int :
calculated_width = self . __width
for obj in self . placed_objects :
if isinstance ( obj , PlacedClip ) :
calculated_width = max ( calculated_width , obj . _width )
return calculated_width
@_width.setter
def _width ( self , val : Any ) - > None :
self . __width = val
@property
def _height ( self ) - > int :
calculated_height = self . __height
for obj in self . placed_objects :
if isinstance ( obj , PlacedClip ) :
calculated_height = max ( calculated_height , obj . _height )
return calculated_height
@_height.setter
def _height ( self , val : Any ) - > None :
self . __height = val
2021-04-15 23:18:33 +00:00
2021-05-25 02:01:17 +00:00
class PlacedImage ( PlacedObject ) :
# An image that occupies its parent clip at some depth. Placed by an AP2PlaceObjectTag
# referencing an AP2ImageTag.
def __init__ (
self ,
object_id : int ,
depth : int ,
2021-07-06 21:58:32 +00:00
rotation_origin : Point ,
2021-05-25 02:01:17 +00:00
transform : Matrix ,
2021-08-05 17:31:28 +00:00
projection : int ,
2021-05-25 02:01:17 +00:00
mult_color : Color ,
add_color : Color ,
2022-10-16 19:30:02 +00:00
hsl_shift : HSL ,
2021-05-25 02:01:17 +00:00
blend : int ,
mask : Optional [ Mask ] ,
source : RegisteredImage ,
) - > None :
2022-10-15 18:56:30 +00:00
super ( ) . __init__ (
object_id ,
depth ,
rotation_origin ,
transform ,
projection ,
mult_color ,
add_color ,
2022-10-16 19:30:02 +00:00
hsl_shift ,
2022-10-15 18:56:30 +00:00
blend ,
mask ,
)
2021-05-25 02:01:17 +00:00
self . __source = source
@property
def source ( self ) - > RegisteredImage :
return self . __source
def __repr__ ( self ) - > str :
return f " PlacedImage(object_id= { self . object_id } , depth= { self . depth } , source= { self . source } ) "
2021-05-21 16:58:01 +00:00
class PlacedDummy ( PlacedObject ) :
# A reference to an object we can't find because we're missing the import.
2021-05-23 20:32:21 +00:00
def __init__ (
self ,
object_id : int ,
depth : int ,
2021-07-06 21:58:32 +00:00
rotation_origin : Point ,
2021-05-23 20:32:21 +00:00
transform : Matrix ,
2021-08-05 17:31:28 +00:00
projection : int ,
2021-05-23 20:32:21 +00:00
mult_color : Color ,
add_color : Color ,
2022-10-16 19:30:02 +00:00
hsl_shift : HSL ,
2021-05-23 20:32:21 +00:00
blend : int ,
2021-05-23 20:37:18 +00:00
mask : Optional [ Mask ] ,
2021-05-23 20:32:21 +00:00
source : RegisteredDummy ,
) - > None :
2022-10-15 18:56:30 +00:00
super ( ) . __init__ (
object_id ,
depth ,
rotation_origin ,
transform ,
projection ,
mult_color ,
add_color ,
2022-10-16 19:30:02 +00:00
hsl_shift ,
2022-10-15 18:56:30 +00:00
blend ,
mask ,
)
2021-05-21 16:58:01 +00:00
self . __source = source
@property
def source ( self ) - > RegisteredDummy :
return self . __source
2021-08-05 17:35:10 +00:00
class PlacedCamera :
def __init__ ( self , center : Point , focal_length : float ) - > None :
self . center = center
self . focal_length = focal_length
self . adjusted = False
2021-05-24 17:36:57 +00:00
class Global :
def __init__ ( self , root : PlacedClip , clip : PlacedClip ) - > None :
2021-05-23 20:37:18 +00:00
self . root = root
2021-05-24 17:36:57 +00:00
self . clip = clip
2021-05-23 20:32:21 +00:00
def getInstanceAtDepth ( self , depth : Any ) - > Any :
if not isinstance ( depth , int ) :
return UNDEFINED
# For some reason, it looks like internally the depth of all objects is
# stored added to -0x4000, so let's reverse that.
depth = depth + 0x4000
2021-05-24 17:36:57 +00:00
for obj in self . clip . placed_objects :
2021-05-23 20:32:21 +00:00
if obj . depth == depth :
return obj
print ( f " WARNING: Could not find object at depth { depth } ! " )
return UNDEFINED
2022-07-26 22:27:27 +00:00
def deepGotoAndPlay ( self , frame : Any ) - > Any :
# This is identical to regular gotoAndPlay, however it also recursively
# goes through and sets all child clips playing as well.
try :
2022-10-15 18:56:30 +00:00
meth = getattr ( self . clip , " gotoAndPlay " )
2022-07-26 22:27:27 +00:00
# Call it, set the return on the stack.
retval = meth ( frame )
# Recursively go through any children of "clip" and call play
# on them as well.
def play_children ( obj : Any ) - > None :
if isinstance ( obj , PlacedClip ) :
obj . play ( )
for child in obj . placed_objects :
play_children ( child )
play_children ( self . clip )
return retval
except AttributeError :
# Function does not exist!
2022-10-15 18:56:30 +00:00
print (
f " WARNING: Tried to call gotoAndPlay( { frame } ) on { self . clip } but that method doesn ' t exist! "
)
2022-07-26 22:27:27 +00:00
return UNDEFINED
2022-10-15 18:56:30 +00:00
def __find_parent (
self , parent : PlacedClip , child : PlacedClip
) - > Optional [ PlacedClip ] :
2021-05-24 17:36:57 +00:00
for obj in parent . placed_objects :
if obj is child :
# This is us, so the parent is our parent.
return parent
if isinstance ( obj , PlacedClip ) :
maybe_parent = self . __find_parent ( obj , child )
if maybe_parent is not None :
return maybe_parent
return None
def find_parent ( self , child : PlacedClip ) - > Optional [ PlacedClip ] :
return self . __find_parent ( self . root , child )
2021-05-23 20:32:21 +00:00
class AEPLib :
2022-10-15 18:56:30 +00:00
def aep_set_rect_mask (
self , thisptr : Any , left : Any , right : Any , top : Any , bottom : Any
) - > None :
if (
not isinstance ( left , ( int , float ) )
or not isinstance ( right , ( int , float ) )
or not isinstance ( top , ( int , float ) )
or not isinstance ( bottom , ( int , float ) )
) :
print (
f " WARNING: Ignoring aeplib.aep_set_rect_mask call with invalid parameters { left } , { right } , { top } , { bottom } ! "
)
2021-05-23 20:32:21 +00:00
return
2021-05-23 20:37:18 +00:00
if isinstance ( thisptr , PlacedObject ) :
thisptr . mask = Mask (
Rectangle (
left = float ( left ) ,
right = float ( right ) ,
top = float ( top ) ,
bottom = float ( bottom ) ,
) ,
2021-05-23 20:32:21 +00:00
)
else :
2022-10-15 18:56:30 +00:00
print (
f " WARNING: Ignoring aeplib.aep_set_rect_mask call with unrecognized target { thisptr } ! "
)
2021-05-23 20:32:21 +00:00
def aep_set_set_frame ( self , thisptr : Any , frame : Any ) - > None :
2021-06-12 17:17:02 +00:00
# This appears to be some sort of callback that the game or other animations can use to figure out
# what frame of animation is currently happening. Whenever I've seen it, it is with the 'frame' set
# to an integer value that matches the currently rendering frame in the render loop. I think its
# safe to ignore this, but if we ever create animations it might be necessary to add calls to this.
2021-05-23 20:32:21 +00:00
pass
2021-06-12 17:17:02 +00:00
def aep_set_frame_control ( self , thisptr : Any , depth : Any , frame : Any ) - > None :
if not isinstance ( thisptr , PlacedClip ) :
2022-10-15 18:56:30 +00:00
print (
f " WARNING: Ignoring aeplib.aep_set_frame_control with unrecognized current object { thisptr } ! "
)
2021-06-12 17:17:02 +00:00
return
for obj in thisptr . placed_objects :
if obj . depth == depth :
if not isinstance ( obj , PlacedClip ) :
2022-10-15 18:56:30 +00:00
print (
f " WARNING: Ignoring aeplib.aep_set_frame_control called on object { obj } at depth { depth } ! "
)
2021-06-12 17:17:02 +00:00
return
2021-07-29 22:02:10 +00:00
obj . setInvisibleUntil ( frame )
2021-06-12 17:17:02 +00:00
return
2022-10-15 18:56:30 +00:00
print (
f " WARNING: Ignoring aeplib.aep_set_frame_control called on nonexistent object at depth { depth } ! "
)
2021-06-12 17:17:02 +00:00
2021-05-25 02:01:17 +00:00
def gotoAndPlay ( self , thisptr : Any , frame : Any ) - > Any :
# This appears to be a wrapper to allow calling gotoAndPlay on clips.
try :
2022-10-15 18:56:30 +00:00
meth = getattr ( thisptr , " gotoAndPlay " )
2021-05-25 02:01:17 +00:00
# Call it, set the return on the stack.
return meth ( frame )
except AttributeError :
# Function does not exist!
2022-10-15 18:56:30 +00:00
print (
f " WARNING: Tried to call gotoAndPlay( { frame } ) on { thisptr } but that method doesn ' t exist! "
)
2021-05-29 00:21:42 +00:00
return UNDEFINED
2021-09-20 05:04:33 +00:00
def gotoAndStop ( self , thisptr : Any , frame : Any ) - > Any :
# This appears to be a wrapper to allow calling gotoAndStop on clips.
try :
2022-10-15 18:56:30 +00:00
meth = getattr ( thisptr , " gotoAndStop " )
2021-09-20 05:04:33 +00:00
# Call it, set the return on the stack.
return meth ( frame )
except AttributeError :
# Function does not exist!
2022-10-15 18:56:30 +00:00
print (
f " WARNING: Tried to call gotoAndStop( { frame } ) on { thisptr } but that method doesn ' t exist! "
)
2021-09-20 05:04:33 +00:00
return UNDEFINED
2021-05-29 00:21:42 +00:00
def deepGotoAndPlay ( self , thisptr : Any , frame : Any ) - > Any :
2021-09-20 05:04:33 +00:00
# This is identical to regular gotoAndPlay, however it also recursively
# goes through and sets all child clips playing as well.
2021-05-29 00:21:42 +00:00
try :
2022-10-15 18:56:30 +00:00
meth = getattr ( thisptr , " gotoAndPlay " )
2021-05-29 00:21:42 +00:00
# Call it, set the return on the stack.
2021-09-20 05:04:33 +00:00
retval = meth ( frame )
# Recursively go through any children of "thisptr" and call play
# on them as well.
def play_children ( obj : Any ) - > None :
if isinstance ( obj , PlacedClip ) :
obj . play ( )
for child in obj . placed_objects :
play_children ( child )
play_children ( thisptr )
return retval
2021-05-29 00:21:42 +00:00
except AttributeError :
# Function does not exist!
2022-10-15 18:56:30 +00:00
print (
f " WARNING: Tried to call gotoAndPlay( { frame } ) on { thisptr } but that method doesn ' t exist! "
)
2021-05-29 00:21:42 +00:00
return UNDEFINED
2021-09-20 05:04:33 +00:00
def deepGotoAndStop ( self , thisptr : Any , frame : Any ) - > Any :
# This is identical to regular gotoAndStop, however it also recursively
# goes through and sets all child clips stopped as well.
try :
2022-10-15 18:56:30 +00:00
meth = getattr ( thisptr , " gotoAndStop " )
2021-09-20 05:04:33 +00:00
# Call it, set the return on the stack.
retval = meth ( frame )
# Recursively go through any children of "thisptr" and call stop
# on them as well.
def stop_children ( obj : Any ) - > None :
if isinstance ( obj , PlacedClip ) :
obj . stop ( )
for child in obj . placed_objects :
stop_children ( child )
stop_children ( thisptr )
return retval
except AttributeError :
# Function does not exist!
2022-10-15 18:56:30 +00:00
print (
f " WARNING: Tried to call gotoAndStop( { frame } ) on { thisptr } but that method doesn ' t exist! "
)
2021-09-20 05:04:33 +00:00
return UNDEFINED
def play ( self , thisptr : Any ) - > Any :
# This appears to be a wrapper to allow calling play on clips.
try :
2022-10-15 18:56:30 +00:00
meth = getattr ( thisptr , " play " )
2021-09-20 05:04:33 +00:00
# Call it, set the return on the stack.
return meth ( )
except AttributeError :
# Function does not exist!
2022-10-15 18:56:30 +00:00
print (
f " WARNING: Tried to call play() on { thisptr } but that method doesn ' t exist! "
)
2021-09-20 05:04:33 +00:00
return UNDEFINED
2021-05-29 00:21:42 +00:00
def stop ( self , thisptr : Any ) - > Any :
# This appears to be a wrapper to allow calling stop on clips.
try :
2022-10-15 18:56:30 +00:00
meth = getattr ( thisptr , " stop " )
2021-05-29 00:21:42 +00:00
# Call it, set the return on the stack.
return meth ( )
except AttributeError :
# Function does not exist!
2022-10-15 18:56:30 +00:00
print (
f " WARNING: Tried to call stop() on { thisptr } but that method doesn ' t exist! "
)
2021-05-25 02:01:17 +00:00
return UNDEFINED
2021-05-23 20:32:21 +00:00
2021-05-24 17:37:13 +00:00
class ASDLib :
def sound_play ( self , sound : Any ) - > None :
if not isinstance ( sound , str ) :
2022-10-15 18:56:30 +00:00
print (
f " WARNING: Ignoring asdlib.sound_play call with invalid parameters { sound } ! "
)
print (
f " WARNING: Requested sound { sound } be played but we don ' t support sound yet! "
)
2021-05-24 17:37:13 +00:00
2021-05-23 20:37:18 +00:00
MissingThis = object ( )
2021-04-15 23:18:33 +00:00
class AFPRenderer ( VerboseOutput ) :
2022-10-15 18:56:30 +00:00
def __init__ (
self ,
shapes : Dict [ str , Shape ] = { } ,
textures : Dict [ str , Image . Image ] = { } ,
swfs : Dict [ str , SWF ] = { } ,
single_threaded : bool = False ,
enable_aa : bool = False ,
) - > None :
2021-04-15 23:18:33 +00:00
super ( ) . __init__ ( )
2021-05-16 15:15:06 +00:00
# Options for rendering
self . __single_threaded = single_threaded
2021-05-30 04:16:25 +00:00
self . __enable_aa = enable_aa
2021-05-16 15:15:06 +00:00
2021-05-21 16:58:01 +00:00
# Library of shapes (draw instructions), textures (actual images) and swfs (us and other files for imports).
2021-04-15 23:18:33 +00:00
self . shapes : Dict [ str , Shape ] = shapes
2021-04-17 23:30:46 +00:00
self . textures : Dict [ str , Image . Image ] = textures
2021-04-15 23:18:33 +00:00
self . swfs : Dict [ str , SWF ] = swfs
2021-05-10 22:26:26 +00:00
# Internal render parameters.
2022-10-15 18:56:30 +00:00
self . __registered_objects : Dict [
int ,
Union [ RegisteredShape , RegisteredClip , RegisteredImage , RegisteredDummy ] ,
] = { }
2021-05-24 17:36:57 +00:00
self . __root : Optional [ PlacedClip ] = None
2021-08-05 17:35:10 +00:00
self . __camera : Optional [ PlacedCamera ] = None
2021-04-15 23:18:33 +00:00
2021-05-24 17:38:56 +00:00
# List of imports that we provide stub implementations for.
self . __stubbed_swfs : Set [ str ] = {
2022-10-15 18:56:30 +00:00
" aeplib.aeplib " ,
" aeplib.__Packages.aeplib " ,
2021-05-24 17:38:56 +00:00
}
2021-04-15 23:18:33 +00:00
def add_shape ( self , name : str , data : Shape ) - > None :
2021-04-17 23:30:46 +00:00
# Register a named shape with the renderer.
2021-04-15 23:18:33 +00:00
if not data . parsed :
data . parse ( )
self . shapes [ name ] = data
2021-04-17 23:30:46 +00:00
def add_texture ( self , name : str , data : Image . Image ) - > None :
# Register a named texture (already loaded PIL image) with the renderer.
2021-04-21 00:01:35 +00:00
self . textures [ name ] = data . convert ( " RGBA " )
2021-04-15 23:18:33 +00:00
def add_swf ( self , name : str , data : SWF ) - > None :
2021-04-17 23:30:46 +00:00
# Register a named SWF with the renderer.
2021-04-15 23:18:33 +00:00
if not data . parsed :
data . parse ( )
self . swfs [ name ] = data
2021-05-21 21:31:13 +00:00
def render_path (
self ,
path : str ,
background_color : Optional [ Color ] = None ,
2021-08-11 21:40:01 +00:00
background_image : Optional [ List [ Image . Image ] ] = None ,
2021-05-21 21:31:13 +00:00
only_depths : Optional [ List [ int ] ] = None ,
2021-05-25 02:01:36 +00:00
only_frames : Optional [ List [ int ] ] = None ,
2021-05-21 21:31:13 +00:00
movie_transform : Matrix = Matrix . identity ( ) ,
2022-07-26 23:25:35 +00:00
overridden_width : Optional [ float ] = None ,
overridden_height : Optional [ float ] = None ,
2021-05-21 21:31:13 +00:00
verbose : bool = False ,
2021-05-24 17:36:34 +00:00
) - > Generator [ Image . Image , None , None ] :
2021-05-16 22:19:07 +00:00
# Given a path to a SWF root animation, attempt to render it to a list of frames.
2021-07-06 22:29:41 +00:00
for _name , swf in self . swfs . items ( ) :
2021-05-16 22:19:07 +00:00
if swf . exported_name == path :
2021-04-15 23:18:33 +00:00
# This is the SWF we care about.
with self . debugging ( verbose ) :
2021-04-21 01:06:48 +00:00
swf . color = background_color or swf . color
2022-10-15 18:56:30 +00:00
yield from self . __render (
swf ,
only_depths ,
only_frames ,
movie_transform ,
background_image ,
overridden_width ,
overridden_height ,
)
2021-05-24 17:36:34 +00:00
return
2021-05-21 21:31:13 +00:00
2022-10-15 18:56:30 +00:00
raise Exception ( f " { path } not found in registered SWFs! " )
2021-05-21 21:31:13 +00:00
def compute_path_location (
self ,
path : str ,
) - > Rectangle :
# Given a path to a SWF root animation, find its bounding rectangle.
2021-07-06 22:29:41 +00:00
for _name , swf in self . swfs . items ( ) :
2021-05-21 21:31:13 +00:00
if swf . exported_name == path :
# This is the SWF we care about.
return swf . location
2021-04-15 23:18:33 +00:00
2022-10-15 18:56:30 +00:00
raise Exception ( f " { path } not found in registered SWFs! " )
2021-04-15 23:18:33 +00:00
2021-05-24 17:36:34 +00:00
def compute_path_frames (
self ,
path : str ,
) - > int :
# Given a path to a SWF root animation, figure out how many frames are
# in that root path with no regard to bytecode 'stop()' commands.
2021-07-06 22:29:41 +00:00
for _name , swf in self . swfs . items ( ) :
2021-05-24 17:36:34 +00:00
if swf . exported_name == path :
# This is the SWF we care about.
return len ( swf . frames )
2022-10-15 18:56:30 +00:00
raise Exception ( f " { path } not found in registered SWFs! " )
2021-04-16 21:08:41 +00:00
2021-05-24 17:36:34 +00:00
def compute_path_frame_duration (
self ,
path : str ,
) - > int :
# Given a path to a SWF root animation, figure out how many milliseconds are
# occupied by each frame.
2021-07-06 22:29:41 +00:00
for _name , swf in self . swfs . items ( ) :
2021-05-24 17:36:34 +00:00
if swf . exported_name == path :
# This is the SWF we care about.
spf = 1.0 / swf . fps
return int ( spf * 1000.0 )
2022-10-15 18:56:30 +00:00
raise Exception ( f " { path } not found in registered SWFs! " )
2021-04-16 21:08:41 +00:00
2021-05-29 03:41:25 +00:00
def compute_path_size (
self ,
path : str ,
) - > Rectangle :
# Given a path to a SWF root animation, figure out what the dimensions
# of the SWF are.
2021-07-06 22:29:41 +00:00
for _name , swf in self . swfs . items ( ) :
2021-05-29 03:41:25 +00:00
if swf . exported_name == path :
return swf . location
2022-10-15 18:56:30 +00:00
raise Exception ( f " { path } not found in registered SWFs! " )
2021-05-29 03:41:25 +00:00
2021-05-24 17:36:34 +00:00
def list_paths ( self , verbose : bool = False ) - > Generator [ str , None , None ] :
# Given the loaded animations, return a list of possible paths to render.
2021-07-06 22:29:41 +00:00
for _name , swf in self . swfs . items ( ) :
2021-05-24 17:36:34 +00:00
yield swf . exported_name
2021-04-16 21:08:41 +00:00
2022-10-15 18:56:30 +00:00
def __execute_bytecode (
self ,
bytecode : ByteCode ,
clip : PlacedClip ,
thisptr : Optional [ Any ] = MissingThis ,
prefix : str = " " ,
) - > None :
2021-05-24 17:36:57 +00:00
if self . __root is None :
2022-10-15 18:56:30 +00:00
raise Exception (
" Logic error, executing bytecode outside of a rendering movie clip! "
)
2021-05-23 20:32:21 +00:00
2021-05-24 17:36:57 +00:00
thisobj = clip if ( thisptr is MissingThis ) else thisptr
globalobj = Global ( self . __root , clip )
2021-05-23 20:32:21 +00:00
location : int = 0
stack : List [ Any ] = [ ]
variables : Dict [ str , Any ] = {
2022-10-15 18:56:30 +00:00
" aeplib " : AEPLib ( ) ,
" asdlib " : ASDLib ( ) ,
2021-05-23 20:32:21 +00:00
}
registers : List [ Any ] = [ UNDEFINED ] * 256
2021-09-20 01:16:40 +00:00
self . vprint ( f " { prefix } Bytecode engine starting. " , component = " bytecode " )
2021-05-24 01:24:26 +00:00
2021-05-23 20:32:21 +00:00
while location < len ( bytecode . actions ) :
action = bytecode . actions [ location ]
if action . opcode == AP2Action . END :
# End the execution.
2022-10-15 18:56:30 +00:00
self . vprint (
f " { prefix } Ending bytecode execution. " , component = " bytecode "
)
2021-05-23 20:32:21 +00:00
break
elif action . opcode == AP2Action . GET_VARIABLE :
varname = stack . pop ( )
# Look up the variable, put it on the stack.
if varname in variables :
stack . append ( variables [ varname ] )
else :
stack . append ( UNDEFINED )
elif action . opcode == AP2Action . SET_MEMBER :
# Grab what we're about to do.
set_value = stack . pop ( )
attribute = stack . pop ( )
obj = stack . pop ( )
if not hasattr ( obj , attribute ) :
2022-10-15 18:56:30 +00:00
print (
f " WARNING: Tried to set attribute { attribute } on { obj } to { set_value } but that attribute doesn ' t exist! "
)
2021-05-23 20:32:21 +00:00
else :
2022-10-15 18:56:30 +00:00
self . vprint (
f " { prefix } Setting attribute { attribute } on { obj } to { set_value } " ,
component = " bytecode " ,
)
2021-05-23 20:32:21 +00:00
setattr ( obj , attribute , set_value )
elif action . opcode == AP2Action . CALL_METHOD :
# Grab the method name.
methname = stack . pop ( )
# Grab the object to perform the call on.
obj = stack . pop ( )
# Grab the parameters to pass to the function.
num_params = stack . pop ( )
if not isinstance ( num_params , int ) :
2022-10-15 18:56:30 +00:00
raise Exception (
" Logic error, cannot get number of parameters to method call! "
)
2021-05-23 20:32:21 +00:00
params = [ ]
for _ in range ( num_params ) :
params . append ( stack . pop ( ) )
# Look up the python function we're calling.
try :
2022-10-15 18:56:30 +00:00
self . vprint (
f " { prefix } Calling method { methname } ( { ' , ' . join ( repr ( s ) for s in params ) } ) on { obj } " ,
component = " bytecode " ,
)
2021-05-23 20:32:21 +00:00
meth = getattr ( obj , methname )
# Call it, set the return on the stack.
stack . append ( meth ( * params ) )
except AttributeError :
# Function does not exist!
2022-10-15 18:56:30 +00:00
print (
f " WARNING: Tried to call { methname } ( { ' , ' . join ( repr ( s ) for s in params ) } ) on { obj } but that method doesn ' t exist! "
)
2021-05-23 20:32:21 +00:00
stack . append ( UNDEFINED )
elif action . opcode == AP2Action . CALL_FUNCTION :
# Grab the method name.
funcname = stack . pop ( )
# Grab the parameters to pass to the function.
num_params = stack . pop ( )
if not isinstance ( num_params , int ) :
2022-10-15 18:56:30 +00:00
raise Exception (
" Logic error, cannot get number of parameters to function call! "
)
2021-05-23 20:32:21 +00:00
params = [ ]
for _ in range ( num_params ) :
params . append ( stack . pop ( ) )
# Look up the python function we're calling.
try :
2022-10-15 18:56:30 +00:00
self . vprint (
f " { prefix } Calling global function { funcname } ( { ' , ' . join ( repr ( s ) for s in params ) } ) " ,
component = " bytecode " ,
)
2021-05-24 17:36:57 +00:00
func = getattr ( globalobj , funcname )
2021-05-23 20:32:21 +00:00
# Call it, set the return on the stack.
stack . append ( func ( * params ) )
except AttributeError :
# Function does not exist!
2022-10-15 18:56:30 +00:00
print (
f " WARNING: Tried to call { funcname } ( { ' , ' . join ( repr ( s ) for s in params ) } ) on { globalobj } but that function doesn ' t exist! "
)
2021-05-23 20:32:21 +00:00
stack . append ( UNDEFINED )
elif isinstance ( action , PushAction ) :
for obj in action . objects :
if isinstance ( obj , Register ) :
stack . append ( registers [ obj . no ] )
elif isinstance ( obj , StringConstant ) :
if obj . alias :
stack . append ( obj . alias )
else :
stack . append ( StringConstant . property_to_name ( obj . const ) )
2021-05-23 20:37:18 +00:00
elif obj is NULL :
stack . append ( None )
2021-05-23 20:33:05 +00:00
elif obj is THIS :
2021-05-24 17:36:57 +00:00
stack . append ( thisobj )
2021-05-23 20:33:05 +00:00
elif obj is GLOBAL :
2021-05-24 17:36:57 +00:00
stack . append ( globalobj )
2021-05-23 20:37:18 +00:00
elif obj is ROOT :
2021-05-24 17:36:57 +00:00
stack . append ( self . __root )
2021-05-23 20:37:18 +00:00
elif obj is CLIP :
# I am not sure this is correct? Maybe it works out
# in circumstances where "THIS" is pointed at something
# else, such as defined function calls maybe?
stack . append ( clip )
elif obj is PARENT :
# Find the parent of this clip.
2021-05-24 17:36:57 +00:00
stack . append ( globalobj . find_parent ( clip ) or UNDEFINED )
2021-05-23 20:32:21 +00:00
else :
stack . append ( obj )
elif isinstance ( action , StoreRegisterAction ) :
set_value = stack . pop ( )
if action . preserve_stack :
stack . append ( set_value )
for reg in action . registers :
registers [ reg . no ] = set_value
elif action . opcode == AP2Action . POP :
stack . pop ( )
else :
print ( f " WARNING: Unhandled opcode { action } with stack { stack } " )
# Next opcode!
location + = 1
2021-09-20 01:16:40 +00:00
self . vprint ( f " { prefix } Bytecode engine finished. " , component = " bytecode " )
2021-05-24 01:24:26 +00:00
2022-10-15 18:56:30 +00:00
def __place (
self , tag : Tag , operating_clip : PlacedClip , prefix : str = " "
) - > Tuple [ Optional [ PlacedClip ] , bool ] :
2021-04-17 23:30:46 +00:00
# "Place" a tag on the screen. Most of the time, this means performing the action of the tag,
# such as defining a shape (registering it with our shape list) or adding/removing an object.
2021-04-15 23:18:33 +00:00
if isinstance ( tag , AP2ShapeTag ) :
2022-10-15 18:56:30 +00:00
self . vprint (
2022-11-20 23:01:26 +00:00
f " { prefix } Loading { tag . reference } shape into object slot { tag . id } " ,
2022-10-15 18:56:30 +00:00
component = " tags " ,
)
2021-04-15 23:18:33 +00:00
if tag . reference not in self . shapes :
raise Exception ( f " Cannot find shape reference { tag . reference } ! " )
2021-05-10 22:26:26 +00:00
self . __registered_objects [ tag . id ] = RegisteredShape (
tag . id ,
2022-11-20 23:01:26 +00:00
tag . reference ,
2021-05-10 22:26:26 +00:00
self . shapes [ tag . reference ] . vertex_points ,
self . shapes [ tag . reference ] . tex_points ,
self . shapes [ tag . reference ] . tex_colors ,
self . shapes [ tag . reference ] . draw_params ,
)
2021-04-15 23:18:33 +00:00
2021-05-10 22:26:26 +00:00
# Didn't place a new clip, didn't change anything.
return None , False
2021-04-17 23:30:46 +00:00
2021-05-25 02:01:17 +00:00
elif isinstance ( tag , AP2ImageTag ) :
2022-10-15 18:56:30 +00:00
self . vprint (
2022-11-20 23:01:26 +00:00
f " { prefix } Loading { tag . reference } image into object slot { tag . id } " ,
2022-10-15 18:56:30 +00:00
component = " tags " ,
)
2021-05-25 02:01:17 +00:00
if tag . reference not in self . textures :
raise Exception ( f " Cannot find texture reference { tag . reference } ! " )
self . __registered_objects [ tag . id ] = RegisteredImage (
tag . id ,
tag . reference ,
)
# Didn't place a new clip, didn't change anything.
return None , False
2021-04-15 23:18:33 +00:00
elif isinstance ( tag , AP2DefineSpriteTag ) :
2022-10-15 18:56:30 +00:00
self . vprint (
2022-11-20 23:01:26 +00:00
f " { prefix } Loading anonymous sprite into object slot { tag . id } " ,
2022-10-15 18:56:30 +00:00
component = " tags " ,
)
2021-04-20 21:41:28 +00:00
2021-04-17 23:31:08 +00:00
# Register a new clip that we might reference to execute.
2022-10-15 18:56:30 +00:00
self . __registered_objects [ tag . id ] = RegisteredClip (
tag . id , tag . frames , tag . tags , tag . labels
)
2021-05-10 22:26:26 +00:00
# Didn't place a new clip, didn't change anything.
return None , False
2021-04-17 23:31:08 +00:00
2021-04-20 21:41:28 +00:00
elif isinstance ( tag , AP2PlaceObjectTag ) :
2021-07-29 22:02:10 +00:00
if tag . unrecognized_options :
if tag . source_tag_id is not None :
2022-10-15 18:56:30 +00:00
print (
f " WARNING: Place object tag referencing { tag . source_tag_id } includes unparsed options and might not display properly! "
)
2021-07-29 22:02:10 +00:00
else :
2022-10-15 18:56:30 +00:00
print (
f " WARNING: Place object tag on depth { tag . depth } includes unparsed options and might not display properly! "
)
2021-07-29 22:02:10 +00:00
2021-04-15 23:18:33 +00:00
if tag . update :
2021-05-15 04:22:43 +00:00
for i in range ( len ( operating_clip . placed_objects ) - 1 , - 1 , - 1 ) :
obj = operating_clip . placed_objects [ i ]
2021-04-15 23:19:56 +00:00
2021-04-17 23:30:46 +00:00
if obj . object_id == tag . object_id and obj . depth == tag . depth :
2021-05-15 04:22:43 +00:00
new_mult_color = tag . mult_color or obj . mult_color
new_add_color = tag . add_color or obj . add_color
2022-10-16 19:30:02 +00:00
new_hsl_shift = tag . hsl_shift or obj . hsl_shift
2022-10-15 18:56:30 +00:00
new_transform = (
2022-11-21 02:04:55 +00:00
obj . transform . update (
tag . transform ,
tag . projection
== AP2PlaceObjectTag . PROJECTION_PERSPECTIVE ,
)
if (
tag . transform is not None
and tag . projection != AP2PlaceObjectTag . PROJECTION_NONE
)
2022-10-15 18:56:30 +00:00
else obj . transform
)
2021-07-06 21:58:32 +00:00
new_rotation_origin = tag . rotation_origin or obj . rotation_origin
2021-05-15 04:22:43 +00:00
new_blend = tag . blend or obj . blend
2022-10-15 18:56:30 +00:00
new_projection = (
tag . projection
if tag . projection != AP2PlaceObjectTag . PROJECTION_NONE
else obj . projection
)
2021-05-15 04:22:43 +00:00
2022-10-15 18:56:30 +00:00
if (
tag . source_tag_id is not None
and tag . source_tag_id != obj . source . tag_id
) :
2021-05-15 04:22:43 +00:00
# This completely updates the pointed-at object.
2022-11-20 23:01:26 +00:00
newobj = self . __registered_objects [ tag . source_tag_id ]
2022-10-15 18:56:30 +00:00
self . vprint (
2022-11-20 23:01:26 +00:00
f " { prefix } Replacing Object source { obj . source . tag_id } ( { obj . source . reference } ) with { tag . source_tag_id } ( { newobj . reference } ) on object with Object ID { tag . object_id } onto Depth { tag . depth } " ,
2022-10-15 18:56:30 +00:00
component = " tags " ,
)
2021-05-15 04:22:43 +00:00
if isinstance ( newobj , RegisteredShape ) :
operating_clip . placed_objects [ i ] = PlacedShape (
obj . object_id ,
obj . depth ,
2021-07-06 21:58:32 +00:00
new_rotation_origin ,
2021-05-15 04:22:43 +00:00
new_transform ,
2021-08-05 17:31:28 +00:00
new_projection ,
2021-05-15 04:22:43 +00:00
new_mult_color ,
new_add_color ,
2022-10-16 19:30:02 +00:00
new_hsl_shift ,
2021-05-15 04:22:43 +00:00
new_blend ,
2021-05-23 20:32:21 +00:00
obj . mask ,
2021-05-15 04:22:43 +00:00
newobj ,
)
2021-05-25 02:01:17 +00:00
# Didn't place a new clip, changed the parent clip.
return None , True
elif isinstance ( newobj , RegisteredImage ) :
operating_clip . placed_objects [ i ] = PlacedImage (
obj . object_id ,
obj . depth ,
2021-07-06 21:58:32 +00:00
new_rotation_origin ,
2021-05-25 02:01:17 +00:00
new_transform ,
2021-08-05 17:31:28 +00:00
new_projection ,
2021-05-25 02:01:17 +00:00
new_mult_color ,
new_add_color ,
2022-10-16 19:30:02 +00:00
new_hsl_shift ,
2021-05-25 02:01:17 +00:00
new_blend ,
obj . mask ,
newobj ,
)
2021-05-15 04:22:43 +00:00
# Didn't place a new clip, changed the parent clip.
return None , True
elif isinstance ( newobj , RegisteredClip ) :
new_clip = PlacedClip (
tag . object_id ,
tag . depth ,
2021-07-06 21:58:32 +00:00
new_rotation_origin ,
2021-05-15 04:22:43 +00:00
new_transform ,
2021-08-05 17:31:28 +00:00
new_projection ,
2021-05-15 04:22:43 +00:00
new_mult_color ,
new_add_color ,
2022-10-16 19:30:02 +00:00
new_hsl_shift ,
2021-05-15 04:22:43 +00:00
new_blend ,
2021-05-23 20:32:21 +00:00
obj . mask ,
2021-05-15 04:22:43 +00:00
newobj ,
)
operating_clip . placed_objects [ i ] = new_clip
# Placed a new clip, changed the parent.
return new_clip , True
2021-05-21 16:58:01 +00:00
elif isinstance ( newobj , RegisteredDummy ) :
operating_clip . placed_objects [ i ] = PlacedDummy (
obj . object_id ,
obj . depth ,
2021-07-06 21:58:32 +00:00
new_rotation_origin ,
2021-05-21 16:58:01 +00:00
new_transform ,
2021-08-05 17:31:28 +00:00
new_projection ,
2021-05-21 16:58:01 +00:00
new_mult_color ,
new_add_color ,
2022-10-16 19:30:02 +00:00
new_hsl_shift ,
2021-05-21 16:58:01 +00:00
new_blend ,
2021-05-23 20:32:21 +00:00
obj . mask ,
2021-05-21 16:58:01 +00:00
newobj ,
)
# Didn't place a new clip, changed the parent clip.
return None , True
2021-05-15 04:22:43 +00:00
else :
2022-10-15 18:56:30 +00:00
raise Exception (
f " Unrecognized object with Tag ID { tag . source_tag_id } ! "
)
2021-05-15 04:22:43 +00:00
else :
# As far as I can tell, pretty much only color and matrix stuff can be updated.
2022-10-15 18:56:30 +00:00
self . vprint (
2022-11-20 23:01:26 +00:00
f " { prefix } Updating Object ID { tag . object_id } ( { obj . source . reference } ) on Depth { tag . depth } " ,
2022-10-15 18:56:30 +00:00
component = " tags " ,
)
2021-05-15 04:22:43 +00:00
obj . mult_color = new_mult_color
obj . add_color = new_add_color
2022-10-16 19:30:02 +00:00
obj . hsl_shift = new_hsl_shift
2021-05-15 04:22:43 +00:00
obj . transform = new_transform
2021-07-06 21:58:32 +00:00
obj . rotation_origin = new_rotation_origin
2021-08-05 17:31:28 +00:00
obj . projection = new_projection
2021-05-15 04:22:43 +00:00
obj . blend = new_blend
return None , True
2021-04-20 21:41:28 +00:00
2021-05-10 22:26:26 +00:00
# Didn't place a new clip, did change something.
2022-10-15 18:56:30 +00:00
print (
f " WARNING: Couldn ' t find tag { tag . object_id } on depth { tag . depth } to update! "
)
2021-05-15 04:22:43 +00:00
return None , False
2021-04-15 23:18:33 +00:00
else :
2021-04-20 21:41:28 +00:00
if tag . source_tag_id is None :
2022-10-15 18:56:30 +00:00
raise Exception (
" Cannot place a tag with no source ID and no update flags! "
)
2021-04-15 23:18:33 +00:00
2021-05-10 22:26:26 +00:00
if tag . source_tag_id in self . __registered_objects :
2022-11-20 23:01:26 +00:00
newobj = self . __registered_objects [ tag . source_tag_id ]
2022-10-15 18:56:30 +00:00
self . vprint (
2022-11-20 23:01:26 +00:00
f " { prefix } Placing Object { tag . source_tag_id } ( { newobj . reference } ) with Object ID { tag . object_id } onto Depth { tag . depth } " ,
2022-10-15 18:56:30 +00:00
component = " tags " ,
)
2021-05-10 22:26:26 +00:00
if isinstance ( newobj , RegisteredShape ) :
operating_clip . placed_objects . append (
PlacedShape (
tag . object_id ,
tag . depth ,
2021-07-06 21:58:32 +00:00
tag . rotation_origin or Point . identity ( ) ,
2021-05-10 22:26:26 +00:00
tag . transform or Matrix . identity ( ) ,
2021-08-05 17:31:28 +00:00
tag . projection ,
2021-05-10 22:26:26 +00:00
tag . mult_color or Color ( 1.0 , 1.0 , 1.0 , 1.0 ) ,
tag . add_color or Color ( 0.0 , 0.0 , 0.0 , 0.0 ) ,
2022-10-16 19:30:02 +00:00
tag . hsl_shift or HSL ( 0.0 , 0.0 , 0.0 ) ,
2021-05-10 22:26:26 +00:00
tag . blend or 0 ,
2021-05-23 20:32:21 +00:00
None ,
2021-05-10 22:26:26 +00:00
newobj ,
)
)
2021-05-25 02:01:17 +00:00
# Didn't place a new clip, changed the parent clip.
return None , True
elif isinstance ( newobj , RegisteredImage ) :
operating_clip . placed_objects . append (
PlacedImage (
tag . object_id ,
tag . depth ,
2021-07-06 21:58:32 +00:00
tag . rotation_origin or Point . identity ( ) ,
2021-05-25 02:01:17 +00:00
tag . transform or Matrix . identity ( ) ,
2021-08-05 17:31:28 +00:00
tag . projection ,
2021-05-25 02:01:17 +00:00
tag . mult_color or Color ( 1.0 , 1.0 , 1.0 , 1.0 ) ,
tag . add_color or Color ( 0.0 , 0.0 , 0.0 , 0.0 ) ,
2022-10-16 19:30:02 +00:00
tag . hsl_shift or HSL ( 0.0 , 0.0 , 0.0 ) ,
2021-05-25 02:01:17 +00:00
tag . blend or 0 ,
None ,
newobj ,
)
)
2021-05-10 22:26:26 +00:00
# Didn't place a new clip, changed the parent clip.
return None , True
elif isinstance ( newobj , RegisteredClip ) :
placed_clip = PlacedClip (
tag . object_id ,
tag . depth ,
2021-07-06 21:58:32 +00:00
tag . rotation_origin or Point . identity ( ) ,
2021-05-10 22:26:26 +00:00
tag . transform or Matrix . identity ( ) ,
2021-08-05 17:31:28 +00:00
tag . projection ,
2021-05-10 22:26:26 +00:00
tag . mult_color or Color ( 1.0 , 1.0 , 1.0 , 1.0 ) ,
tag . add_color or Color ( 0.0 , 0.0 , 0.0 , 0.0 ) ,
2022-10-16 19:30:02 +00:00
tag . hsl_shift or HSL ( 0.0 , 0.0 , 0.0 ) ,
2021-05-10 22:26:26 +00:00
tag . blend or 0 ,
2021-05-23 20:32:21 +00:00
None ,
2021-05-10 22:26:26 +00:00
newobj ,
)
operating_clip . placed_objects . append ( placed_clip )
2021-05-23 20:32:21 +00:00
for flags , code in tag . triggers . items ( ) :
if flags & AP2Trigger . ON_LOAD :
for bytecode in code :
2022-10-15 18:56:30 +00:00
self . __execute_bytecode (
bytecode , placed_clip , prefix = prefix + " "
)
2021-05-23 20:32:21 +00:00
else :
2022-10-15 18:56:30 +00:00
print (
" WARNING: Unhandled PLACE_OBJECT trigger with flags {flags} ! "
)
2021-05-23 20:32:21 +00:00
2021-05-10 22:26:26 +00:00
# Placed a new clip, changed the parent.
return placed_clip , True
2021-05-21 16:58:01 +00:00
elif isinstance ( newobj , RegisteredDummy ) :
operating_clip . placed_objects . append (
PlacedDummy (
tag . object_id ,
tag . depth ,
2021-07-06 21:58:32 +00:00
tag . rotation_origin or Point . identity ( ) ,
2021-05-21 16:58:01 +00:00
tag . transform or Matrix . identity ( ) ,
2021-08-05 17:31:28 +00:00
tag . projection ,
2021-05-21 16:58:01 +00:00
tag . mult_color or Color ( 1.0 , 1.0 , 1.0 , 1.0 ) ,
tag . add_color or Color ( 0.0 , 0.0 , 0.0 , 0.0 ) ,
2022-10-16 19:30:02 +00:00
tag . hsl_shift or HSL ( 0.0 , 0.0 , 0.0 ) ,
2021-05-21 16:58:01 +00:00
tag . blend or 0 ,
2021-05-23 20:32:21 +00:00
None ,
2021-05-21 16:58:01 +00:00
newobj ,
)
)
# Didn't place a new clip, changed the parent clip.
return None , True
2021-05-10 22:26:26 +00:00
else :
2022-10-15 18:56:30 +00:00
raise Exception (
f " Unrecognized object with Tag ID { tag . source_tag_id } ! "
)
2021-04-20 23:19:14 +00:00
2022-10-15 18:56:30 +00:00
raise Exception (
f " Cannot find a shape or sprite with Tag ID { tag . source_tag_id } ! "
)
2021-05-10 22:26:26 +00:00
2021-04-15 23:18:33 +00:00
elif isinstance ( tag , AP2RemoveObjectTag ) :
2022-10-15 18:56:30 +00:00
self . vprint (
f " { prefix } Removing Object ID { tag . object_id } from Depth { tag . depth } " ,
component = " tags " ,
)
2021-04-15 23:18:33 +00:00
if tag . object_id != 0 :
# Remove the identified object by object ID and depth.
2021-04-17 23:31:08 +00:00
# Remember removed objects so we can stop any clips.
removed_objects = [
2022-10-15 18:56:30 +00:00
obj
for obj in operating_clip . placed_objects
2021-04-17 23:31:08 +00:00
if obj . object_id == tag . object_id and obj . depth == tag . depth
]
2021-04-17 23:30:46 +00:00
2021-04-17 23:31:08 +00:00
# Get rid of the objects that we're removing from the master list.
2021-05-10 22:26:26 +00:00
operating_clip . placed_objects = [
2022-10-15 18:56:30 +00:00
obj
for obj in operating_clip . placed_objects
if not ( obj . object_id == tag . object_id and obj . depth == tag . depth )
2021-04-15 23:18:33 +00:00
]
else :
2021-04-17 23:30:46 +00:00
# Remove the last placed object at this depth. The placed objects list isn't
# ordered so much as apppending to the list means the last placed object at a
# depth comes last.
2021-04-20 23:19:14 +00:00
removed_objects = [ ]
2021-05-10 22:26:26 +00:00
for i in range ( len ( operating_clip . placed_objects ) ) :
real_index = len ( operating_clip . placed_objects ) - ( i + 1 )
2021-04-15 23:18:33 +00:00
2021-05-10 22:26:26 +00:00
if operating_clip . placed_objects [ real_index ] . depth == tag . depth :
2022-10-15 18:56:30 +00:00
removed_objects = operating_clip . placed_objects [
real_index : ( real_index + 1 )
]
operating_clip . placed_objects = (
operating_clip . placed_objects [ : real_index ]
+ operating_clip . placed_objects [ ( real_index + 1 ) : ]
)
2021-04-15 23:18:33 +00:00
break
2021-04-17 23:31:08 +00:00
2021-04-20 23:19:14 +00:00
if not removed_objects :
2022-10-15 18:56:30 +00:00
print (
f " WARNING: Couldn ' t find object to remove by ID { tag . object_id } and depth { tag . depth } ! "
)
2021-04-20 23:19:14 +00:00
2021-05-23 20:32:21 +00:00
# TODO: Handle ON_UNLOAD triggers for this object. I don't think I've ever seen one
# on any object so this might be a pedantic request.
2021-05-10 22:26:26 +00:00
# Didn't place a new clip, changed parent clip.
return None , True
2021-04-15 23:20:27 +00:00
elif isinstance ( tag , AP2DoActionTag ) :
2021-09-20 01:16:40 +00:00
self . vprint ( f " { prefix } Execution action tag. " , component = " tags " )
2022-10-15 18:56:30 +00:00
self . __execute_bytecode (
tag . bytecode , operating_clip , prefix = prefix + " "
)
2021-05-10 22:26:26 +00:00
# Didn't place a new clip.
return None , False
2021-04-16 21:28:53 +00:00
elif isinstance ( tag , AP2DefineFontTag ) :
print ( " WARNING: Unhandled DEFINE_FONT tag! " )
2021-05-10 22:26:26 +00:00
# Didn't place a new clip.
return None , False
2021-04-16 21:28:53 +00:00
elif isinstance ( tag , AP2DefineEditTextTag ) :
print ( " WARNING: Unhandled DEFINE_EDIT_TEXT tag! " )
2021-05-10 22:26:26 +00:00
# Didn't place a new clip.
return None , False
2021-08-11 18:31:37 +00:00
elif isinstance ( tag , AP2DefineMorphShapeTag ) :
print ( " WARNING: Unhandled DEFINE_MORPH_SHAPE tag! " )
self . __registered_objects [ tag . id ] = RegisteredDummy (
tag . id ,
)
# Didn't place a new clip.
return None , False
2021-05-17 03:55:41 +00:00
elif isinstance ( tag , AP2PlaceCameraTag ) :
2021-09-20 01:16:40 +00:00
self . vprint ( f " { prefix } Place camera tag. " , component = " tags " )
2021-08-05 17:35:10 +00:00
self . __camera = PlacedCamera (
tag . center ,
tag . focal_length ,
)
2021-05-17 03:55:41 +00:00
# Didn't place a new clip.
return None , False
2021-04-15 23:18:33 +00:00
else :
raise Exception ( f " Failed to process tag: { tag } " )
2021-05-23 20:37:18 +00:00
def __apply_mask (
self ,
parent_mask : Image . Image ,
transform : Matrix ,
2021-08-09 17:33:21 +00:00
projection : int ,
2021-05-23 20:37:18 +00:00
mask : Mask ,
) - > Image . Image :
if mask . rectangle is None :
# Calculate the new mask rectangle.
2021-08-10 23:37:08 +00:00
mask . rectangle = affine_composite (
2022-10-15 18:56:30 +00:00
Image . new (
" RGBA " ,
( int ( mask . bounds . right ) , int ( mask . bounds . bottom ) ) ,
( 0 , 0 , 0 , 0 ) ,
) ,
2021-08-10 23:37:08 +00:00
Color ( 0.0 , 0.0 , 0.0 , 0.0 ) ,
Color ( 1.0 , 1.0 , 1.0 , 1.0 ) ,
2022-10-16 19:30:02 +00:00
HSL ( 0.0 , 0.0 , 0.0 ) ,
2021-08-10 23:37:08 +00:00
Matrix . identity ( ) . translate ( Point ( mask . bounds . left , mask . bounds . top ) ) ,
None ,
0 ,
2022-10-15 18:56:30 +00:00
Image . new (
" RGBA " ,
( int ( mask . bounds . width ) , int ( mask . bounds . height ) ) ,
( 255 , 0 , 0 , 255 ) ,
) ,
2021-08-10 23:37:08 +00:00
single_threaded = self . __single_threaded ,
aa_mode = AAMode . NONE ,
)
2021-05-23 20:37:18 +00:00
# Draw the mask onto a new image.
2021-08-09 17:33:21 +00:00
if projection == AP2PlaceObjectTag . PROJECTION_AFFINE :
calculated_mask = affine_composite (
2022-10-15 18:56:30 +00:00
Image . new (
" RGBA " , ( parent_mask . width , parent_mask . height ) , ( 0 , 0 , 0 , 0 )
) ,
2021-08-09 17:33:21 +00:00
Color ( 0.0 , 0.0 , 0.0 , 0.0 ) ,
Color ( 1.0 , 1.0 , 1.0 , 1.0 ) ,
2022-10-16 19:30:02 +00:00
HSL ( 0.0 , 0.0 , 0.0 ) ,
2021-08-09 17:33:21 +00:00
transform ,
None ,
257 ,
mask . rectangle ,
single_threaded = self . __single_threaded ,
2021-08-09 19:09:00 +00:00
aa_mode = AAMode . NONE ,
2021-08-09 17:33:21 +00:00
)
elif projection == AP2PlaceObjectTag . PROJECTION_PERSPECTIVE :
if self . __camera is None :
2022-10-15 18:56:30 +00:00
print (
" WARNING: Element requests perspective projection but no camera exists! "
)
2021-08-09 17:33:21 +00:00
calculated_mask = affine_composite (
2022-10-15 18:56:30 +00:00
Image . new (
" RGBA " , ( parent_mask . width , parent_mask . height ) , ( 0 , 0 , 0 , 0 )
) ,
2021-08-09 17:33:21 +00:00
Color ( 0.0 , 0.0 , 0.0 , 0.0 ) ,
Color ( 1.0 , 1.0 , 1.0 , 1.0 ) ,
2022-10-16 19:30:02 +00:00
HSL ( 0.0 , 0.0 , 0.0 ) ,
2021-08-09 17:33:21 +00:00
transform ,
None ,
257 ,
mask . rectangle ,
single_threaded = self . __single_threaded ,
2021-08-09 19:09:00 +00:00
aa_mode = AAMode . NONE ,
2021-08-09 17:33:21 +00:00
)
else :
calculated_mask = perspective_composite (
2022-10-15 18:56:30 +00:00
Image . new (
" RGBA " , ( parent_mask . width , parent_mask . height ) , ( 0 , 0 , 0 , 0 )
) ,
2021-08-09 17:33:21 +00:00
Color ( 0.0 , 0.0 , 0.0 , 0.0 ) ,
Color ( 1.0 , 1.0 , 1.0 , 1.0 ) ,
2022-10-16 19:30:02 +00:00
HSL ( 0.0 , 0.0 , 0.0 ) ,
2021-08-09 17:33:21 +00:00
transform ,
self . __camera . center ,
self . __camera . focal_length ,
None ,
257 ,
mask . rectangle ,
single_threaded = self . __single_threaded ,
2021-08-09 19:09:00 +00:00
aa_mode = AAMode . NONE ,
2021-08-09 17:33:21 +00:00
)
2021-05-23 20:37:18 +00:00
# Composite it onto the current mask.
return affine_composite (
parent_mask . copy ( ) ,
Color ( 0.0 , 0.0 , 0.0 , 0.0 ) ,
Color ( 1.0 , 1.0 , 1.0 , 1.0 ) ,
2022-10-16 19:30:02 +00:00
HSL ( 0.0 , 0.0 , 0.0 ) ,
2021-05-23 20:37:18 +00:00
Matrix . identity ( ) ,
None ,
256 ,
calculated_mask ,
single_threaded = self . __single_threaded ,
2021-08-09 19:09:00 +00:00
aa_mode = AAMode . NONE ,
2021-05-23 20:37:18 +00:00
)
2021-05-16 19:40:06 +00:00
def __render_object (
self ,
img : Image . Image ,
renderable : PlacedObject ,
parent_transform : Matrix ,
2021-08-05 17:31:28 +00:00
parent_projection : int ,
2021-05-23 20:37:18 +00:00
parent_mask : Image . Image ,
2021-05-22 21:52:17 +00:00
parent_mult_color : Color ,
parent_add_color : Color ,
2022-10-16 19:30:02 +00:00
parent_hsl_shift : HSL ,
2021-05-22 21:56:28 +00:00
parent_blend : int ,
2021-05-16 19:40:06 +00:00
only_depths : Optional [ List [ int ] ] = None ,
2022-10-15 18:56:30 +00:00
prefix : str = " " ,
2021-05-16 19:40:06 +00:00
) - > Image . Image :
2021-07-29 22:02:10 +00:00
if not renderable . visible :
2022-10-15 18:56:30 +00:00
self . vprint (
2022-11-20 23:01:26 +00:00
f " { prefix } Ignoring invisible placed object ID { renderable . object_id } from sprite { renderable . source . tag_id } ( { renderable . source . reference } ) on Depth { renderable . depth } " ,
2022-10-15 18:56:30 +00:00
component = " render " ,
)
2021-07-29 22:02:10 +00:00
return img
2022-10-15 18:56:30 +00:00
self . vprint (
2022-11-20 23:01:26 +00:00
f " { prefix } Rendering placed object ID { renderable . object_id } from sprite { renderable . source . tag_id } ( { renderable . source . reference } ) onto Depth { renderable . depth } " ,
2022-10-15 18:56:30 +00:00
component = " render " ,
)
2021-05-16 19:39:25 +00:00
2021-05-10 22:26:26 +00:00
# Compute the affine transformation matrix for this object.
2022-10-15 18:56:30 +00:00
transform = renderable . transform . multiply ( parent_transform ) . translate (
Point . identity ( ) . subtract ( renderable . rotation_origin )
)
projection = (
AP2PlaceObjectTag . PROJECTION_PERSPECTIVE
if parent_projection == AP2PlaceObjectTag . PROJECTION_PERSPECTIVE
else renderable . projection
)
2021-04-15 23:18:33 +00:00
2021-05-22 21:54:22 +00:00
# Calculate blending and blend color if it is present.
2022-10-15 18:56:30 +00:00
mult_color = ( renderable . mult_color or Color ( 1.0 , 1.0 , 1.0 , 1.0 ) ) . multiply (
parent_mult_color
)
add_color = (
( renderable . add_color or Color ( 0.0 , 0.0 , 0.0 , 0.0 ) )
. multiply ( parent_mult_color )
. add ( parent_add_color )
)
2022-11-20 23:26:27 +00:00
hsl_shift = ( renderable . hsl_shift or HSL ( 0.0 , 0.0 , 0.0 ) ) . add ( parent_hsl_shift )
2021-05-22 21:54:22 +00:00
blend = renderable . blend or 0
2021-05-22 21:56:28 +00:00
if parent_blend not in { 0 , 1 , 2 } and blend in { 0 , 1 , 2 } :
blend = parent_blend
2021-05-22 21:52:17 +00:00
2021-05-23 20:32:21 +00:00
if renderable . mask :
2022-10-15 18:56:30 +00:00
mask = self . __apply_mask (
parent_mask , transform , projection , renderable . mask
)
2021-05-23 20:37:18 +00:00
else :
mask = parent_mask
2021-05-23 20:32:21 +00:00
2022-11-20 23:26:27 +00:00
if projection == AP2PlaceObjectTag . PROJECTION_AFFINE :
projection_string = " affine projection "
elif projection == AP2PlaceObjectTag . PROJECTION_PERSPECTIVE :
projection_string = " perspective projection "
else :
projection_string = " no projection "
if blend == 3 :
blend_string = " multiply "
elif blend == 8 :
blend_string = " addition "
elif blend == 9 or blend == 70 :
blend_string = " subtraction "
elif blend == 13 :
blend_string = " overlay "
else :
blend_string = " normal "
2021-04-20 21:41:28 +00:00
# Render individual shapes if this is a sprite.
2021-05-10 22:26:26 +00:00
if isinstance ( renderable , PlacedClip ) :
2021-05-31 18:13:43 +00:00
new_only_depths : Optional [ List [ int ] ] = None
2021-05-22 01:30:58 +00:00
if only_depths is not None :
if renderable . depth not in only_depths :
if renderable . depth != - 1 :
# Not on the correct depth plane.
return img
new_only_depths = only_depths
2022-11-20 23:26:27 +00:00
self . vprint (
f " { prefix } Rendered object uses { projection_string } with transform [ { transform } ] " ,
component = " render " ,
)
self . vprint (
f " { prefix } Rendered object uses { blend_string } with { mult_color } and { add_color } colors " ,
component = " render " ,
)
self . vprint (
f " { prefix } Rendered object applies a HSL shift of { hsl_shift } " ,
component = " render " ,
)
2021-05-22 01:30:28 +00:00
# This is a sprite placement reference. Make sure that we render lower depths
# first, but preserved placed order as well.
depths = set ( obj . depth for obj in renderable . placed_objects )
for depth in sorted ( depths ) :
for obj in renderable . placed_objects :
if obj . depth != depth :
continue
2022-10-15 18:56:30 +00:00
img = self . __render_object (
img ,
obj ,
transform ,
projection ,
mask ,
mult_color ,
add_color ,
2022-10-16 19:30:02 +00:00
hsl_shift ,
2022-10-15 18:56:30 +00:00
blend ,
only_depths = new_only_depths ,
2022-11-20 23:26:27 +00:00
prefix = prefix + " " ,
2022-10-15 18:56:30 +00:00
)
2021-05-10 22:26:26 +00:00
elif isinstance ( renderable , PlacedShape ) :
2021-05-22 01:30:58 +00:00
if only_depths is not None and renderable . depth not in only_depths :
# Not on the correct depth plane.
return img
2022-11-20 23:26:27 +00:00
self . vprint (
f " { prefix } Rendered object uses { projection_string } with transform [ { transform } ] " ,
component = " render " ,
)
self . vprint (
f " { prefix } Rendered object uses { blend_string } with { mult_color } and { add_color } colors " ,
component = " render " ,
)
self . vprint (
f " { prefix } Rendered object applies a HSL shift of { hsl_shift } " ,
component = " render " ,
)
2021-05-10 22:26:26 +00:00
# This is a shape draw reference.
shape = renderable . source
# Now, render out shapes.
for params in shape . draw_params :
if not ( params . flags & 0x1 ) :
# Not instantiable, don't render.
2021-05-16 15:15:06 +00:00
return img
2021-05-10 22:26:26 +00:00
if params . flags & 0x4 :
# TODO: Need to support blending and UV coordinate colors here.
print ( " WARNING: Unhandled UV coordinate color! " )
texture = None
2021-08-09 19:09:00 +00:00
rectangle = False
2021-05-10 22:26:26 +00:00
if params . flags & 0x2 :
# We need to look up the texture for this.
if params . region not in self . textures :
2022-10-15 18:56:30 +00:00
raise Exception (
f " Cannot find texture reference { params . region } ! "
)
2021-05-10 22:26:26 +00:00
texture = self . textures [ params . region ]
2021-05-21 16:58:39 +00:00
if params . flags & 0x8 :
# TODO: This texture gets further blended somehow? Not sure this is ever used.
2021-05-22 21:54:22 +00:00
print ( f " WARNING: Unhandled texture blend color { params . blend } ! " )
2021-05-21 16:58:39 +00:00
elif params . flags & 0x8 :
if shape . rectangle is None :
# This is a raw rectangle. Its possible that the number of vertex points is
# not 4, or that the four points in the vertex_points aren't the four corners
# of a rectangle, but let's assume that doesn't happen for now.
2021-05-22 21:54:22 +00:00
if len ( shape . vertex_points ) != 4 :
print ( " WARNING: Unsupported non-rectangle shape! " )
2021-05-31 18:13:43 +00:00
if params . blend is None :
2022-10-15 18:56:30 +00:00
raise Exception (
" Logic error, rectangle without a blend color! "
)
2021-05-22 21:54:22 +00:00
2021-05-21 16:58:39 +00:00
x_points = set ( p . x for p in shape . vertex_points )
y_points = set ( p . y for p in shape . vertex_points )
left = min ( x_points )
right = max ( x_points )
top = min ( y_points )
bottom = max ( y_points )
2021-05-22 21:54:22 +00:00
# Make sure that the four corners are aligned.
bad = False
for point in x_points :
if point not in { left , right } :
bad = True
break
for point in y_points :
if point not in { top , bottom } :
bad = True
break
if bad :
print ( " WARNING: Unsupported non-rectangle shape! " )
2022-10-15 18:56:30 +00:00
shape . rectangle = Image . new (
" RGBA " ,
( int ( right - left ) , int ( bottom - top ) ) ,
( params . blend . as_tuple ( ) ) ,
)
2021-05-21 16:58:39 +00:00
texture = shape . rectangle
2021-08-09 19:09:00 +00:00
rectangle = True
2021-05-10 22:26:26 +00:00
2021-05-21 16:58:39 +00:00
if texture is not None :
2021-08-05 17:31:28 +00:00
if projection == AP2PlaceObjectTag . PROJECTION_AFFINE :
2021-08-09 19:09:00 +00:00
if self . __enable_aa :
2022-10-15 18:56:30 +00:00
aamode = (
AAMode . UNSCALED_SSAA_ONLY
if rectangle
else AAMode . SSAA_OR_BILINEAR
)
2021-08-09 19:09:00 +00:00
else :
aamode = AAMode . NONE
2021-08-03 17:04:20 +00:00
img = affine_composite (
img ,
add_color ,
mult_color ,
2022-10-16 19:30:02 +00:00
hsl_shift ,
2021-08-03 17:04:20 +00:00
transform ,
mask ,
blend ,
texture ,
single_threaded = self . __single_threaded ,
2021-08-09 19:09:00 +00:00
aa_mode = aamode ,
2021-08-03 17:04:20 +00:00
)
2021-08-05 17:31:28 +00:00
elif projection == AP2PlaceObjectTag . PROJECTION_PERSPECTIVE :
if self . __camera is None :
2021-08-09 19:09:00 +00:00
if self . __enable_aa :
2022-10-15 18:56:30 +00:00
aamode = (
AAMode . UNSCALED_SSAA_ONLY
if rectangle
else AAMode . SSAA_OR_BILINEAR
)
2021-08-09 19:09:00 +00:00
else :
aamode = AAMode . NONE
2022-10-15 18:56:30 +00:00
print (
" WARNING: Element requests perspective projection but no camera exists! "
)
2021-08-05 17:31:28 +00:00
img = affine_composite (
img ,
add_color ,
mult_color ,
2022-10-16 19:30:02 +00:00
hsl_shift ,
2021-08-05 17:31:28 +00:00
transform ,
mask ,
blend ,
texture ,
single_threaded = self . __single_threaded ,
2021-08-09 19:09:00 +00:00
aa_mode = aamode ,
2021-08-05 17:31:28 +00:00
)
else :
2021-08-09 19:09:00 +00:00
if self . __enable_aa :
2022-10-15 18:56:30 +00:00
aamode = (
AAMode . UNSCALED_SSAA_ONLY
if rectangle
else AAMode . SSAA_ONLY
)
2021-08-09 19:09:00 +00:00
else :
aamode = AAMode . NONE
2021-08-05 17:31:28 +00:00
img = perspective_composite (
img ,
add_color ,
mult_color ,
2022-10-16 19:30:02 +00:00
hsl_shift ,
2021-08-05 17:31:28 +00:00
transform ,
self . __camera . center ,
self . __camera . focal_length ,
mask ,
blend ,
texture ,
single_threaded = self . __single_threaded ,
2021-08-09 19:09:00 +00:00
aa_mode = aamode ,
2021-08-05 17:31:28 +00:00
)
2021-05-25 02:01:17 +00:00
elif isinstance ( renderable , PlacedImage ) :
if only_depths is not None and renderable . depth not in only_depths :
# Not on the correct depth plane.
return img
2022-11-20 23:26:27 +00:00
self . vprint (
f " { prefix } Rendered object uses { projection_string } with transform [ { transform } ] " ,
component = " render " ,
)
self . vprint (
f " { prefix } Rendered object uses { blend_string } with { mult_color } and { add_color } colors " ,
component = " render " ,
)
self . vprint (
f " { prefix } Rendered object applies a HSL shift of { hsl_shift } " ,
component = " render " ,
)
2021-05-25 02:01:17 +00:00
# This is a shape draw reference.
texture = self . textures [ renderable . source . reference ]
2021-08-05 17:31:28 +00:00
if projection == AP2PlaceObjectTag . PROJECTION_AFFINE :
2021-08-03 17:04:20 +00:00
img = affine_composite (
img ,
add_color ,
mult_color ,
2022-10-16 19:30:02 +00:00
hsl_shift ,
2021-08-03 17:04:20 +00:00
transform ,
mask ,
blend ,
texture ,
single_threaded = self . __single_threaded ,
2022-10-15 18:56:30 +00:00
aa_mode = AAMode . SSAA_OR_BILINEAR
if self . __enable_aa
else AAMode . NONE ,
2021-08-03 17:04:20 +00:00
)
2021-08-05 17:31:28 +00:00
elif projection == AP2PlaceObjectTag . PROJECTION_PERSPECTIVE :
if self . __camera is None :
2022-10-15 18:56:30 +00:00
print (
" WARNING: Element requests perspective projection but no camera exists! "
)
2021-08-05 17:31:28 +00:00
img = affine_composite (
img ,
add_color ,
mult_color ,
2022-10-16 19:30:02 +00:00
hsl_shift ,
2021-08-05 17:31:28 +00:00
transform ,
mask ,
blend ,
texture ,
single_threaded = self . __single_threaded ,
2022-10-15 18:56:30 +00:00
aa_mode = AAMode . SSAA_OR_BILINEAR
if self . __enable_aa
else AAMode . NONE ,
2021-08-05 17:31:28 +00:00
)
else :
img = perspective_composite (
img ,
add_color ,
mult_color ,
2022-10-16 19:30:02 +00:00
hsl_shift ,
2021-08-05 17:31:28 +00:00
transform ,
self . __camera . center ,
self . __camera . focal_length ,
mask ,
blend ,
texture ,
single_threaded = self . __single_threaded ,
2021-08-09 19:09:00 +00:00
aa_mode = AAMode . SSAA_ONLY if self . __enable_aa else AAMode . NONE ,
2021-08-05 17:31:28 +00:00
)
2021-05-21 16:58:01 +00:00
elif isinstance ( renderable , PlacedDummy ) :
# Nothing to do!
pass
2021-05-10 22:26:26 +00:00
else :
raise Exception ( f " Unknown placed object type to render { renderable } ! " )
2021-04-17 23:31:36 +00:00
2021-05-16 15:15:06 +00:00
return img
2021-05-24 01:24:26 +00:00
def __is_dirty ( self , clip : PlacedClip ) - > bool :
# If we are dirty ourselves, then the clip is definitely dirty.
if clip . requested_frame is not None :
return True
# If one of our children is dirty, then we are dirty.
for child in clip . placed_objects :
if isinstance ( child , PlacedClip ) :
if self . __is_dirty ( child ) :
return True
# None of our children (or their children, etc...) or ourselves is dirty.
return False
2022-10-15 18:56:30 +00:00
def __process_tags (
self , clip : PlacedClip , only_dirty : bool , prefix : str = " "
) - > bool :
self . vprint (
f " { prefix } Handling { ' dirty updates on ' if only_dirty else ' ' } placed clip { clip . object_id } at depth { clip . depth } " ,
component = " tags " ,
)
2021-05-10 22:26:26 +00:00
# Track whether anything in ourselves or our children changes during this processing.
changed = False
2021-05-24 01:24:26 +00:00
# Make sure to set the requested frame if it isn't set by an external force.
if clip . requested_frame is None :
2022-10-15 18:56:30 +00:00
if (
not clip . playing
or only_dirty
or ( clip . finished and clip is self . __root )
) :
2021-05-24 01:24:26 +00:00
# We aren't playing this clip because its either paused or finished,
# or it isn't dirty and we're doing dirty updates only. So, we don't
# need to advance to any frame.
clip . requested_frame = clip . frame
2021-09-20 05:04:52 +00:00
elif clip . finished :
# Rewind the clip to the beginning, loop it.
clip . rewind ( )
clip . requested_frame = clip . frame + 1
2021-05-24 01:24:26 +00:00
else :
# We need to do as many things as we need to get to the next frame.
clip . requested_frame = clip . frame + 1
2021-05-23 20:32:21 +00:00
while True :
2021-05-24 01:24:26 +00:00
# First, see if we need to rewind the clip if we were requested to go backwards
# during some bytecode update in this loop.
if clip . frame > clip . requested_frame :
# Rewind this clip to the beginning so we can replay until the requested frame.
2021-05-25 02:01:36 +00:00
if clip is self . __root :
2022-10-15 18:56:30 +00:00
print (
" WARNING: Root clip was rewound, its possible this animation plays forever! "
)
2021-05-24 18:35:43 +00:00
clip . rewind ( )
2021-05-23 20:32:21 +00:00
2022-10-15 18:56:30 +00:00
self . vprint (
f " { prefix } Processing frame { clip . frame } on our way to frame { clip . requested_frame } " ,
component = " tags " ,
)
2021-05-23 20:32:21 +00:00
# Clips that are part of our own placed objects which we should handle.
child_clips = [ c for c in clip . placed_objects if isinstance ( c , PlacedClip ) ]
2021-05-24 01:24:26 +00:00
# Execute each tag in the frame if we need to move forward to a new frame.
if clip . frame != clip . requested_frame :
2021-05-23 20:32:21 +00:00
frame = clip . source . frames [ clip . frame ]
2021-05-24 18:35:43 +00:00
orphans : List [ Tag ] = [ ]
played_tags : Set [ int ] = set ( )
# See if we have any orphans that need to be placed before this frame will work.
for unplayed_tag in clip . unplayed_tags :
if unplayed_tag < frame . start_tag_offset :
2022-10-15 18:56:30 +00:00
self . vprint (
f " { prefix } Including orphaned tag { unplayed_tag } in frame evaluation " ,
component = " tags " ,
)
2021-05-24 18:35:43 +00:00
played_tags . add ( unplayed_tag )
orphans . append ( clip . source . tags [ unplayed_tag ] )
2022-10-15 18:56:30 +00:00
for tagno in range (
frame . start_tag_offset , frame . start_tag_offset + frame . num_tags
) :
2021-05-24 18:35:43 +00:00
played_tags . add ( tagno )
# Check these off our future todo list.
2022-10-15 18:56:30 +00:00
clip . unplayed_tags = [
t for t in clip . unplayed_tags if t not in played_tags
]
2021-05-23 20:32:21 +00:00
2021-05-24 18:35:43 +00:00
# Grab the normal list of tags, add to the orphans.
2022-10-15 18:56:30 +00:00
tags = (
orphans
+ clip . source . tags [
frame . start_tag_offset : (
frame . start_tag_offset + frame . num_tags
)
]
)
2021-05-23 20:32:21 +00:00
for tagno , tag in enumerate ( tags ) :
# Perform the action of this tag.
2022-10-15 18:56:30 +00:00
self . vprint (
2022-11-20 23:01:26 +00:00
f " { prefix } Sprite Tag ID: { clip . source . tag_id } ( { clip . source . reference } ), Current Tag: { frame . start_tag_offset + tagno } , Num Tags: { frame . num_tags } " ,
2022-10-15 18:56:30 +00:00
component = " tags " ,
)
2021-05-23 20:32:21 +00:00
new_clip , clip_changed = self . __place ( tag , clip , prefix = prefix )
changed = changed or clip_changed
# If we create a new movie clip, process it as well for this frame.
if new_clip :
2021-05-24 01:24:26 +00:00
# These are never dirty-only updates as they're fresh-placed.
2022-10-15 18:56:30 +00:00
changed = (
self . __process_tags ( new_clip , False , prefix = prefix + " " )
or changed
)
2021-05-24 01:24:26 +00:00
# Now, advance the frame for this clip since we processed the frame.
clip . advance ( )
2021-05-23 20:32:21 +00:00
# Now, handle each of the existing clips.
for child in child_clips :
2022-10-15 18:56:30 +00:00
changed = (
self . __process_tags ( child , only_dirty , prefix = prefix + " " )
or changed
)
2021-05-23 20:32:21 +00:00
2021-05-24 01:24:26 +00:00
# See if we're done with this clip.
if clip . frame == clip . requested_frame :
clip . requested_frame = None
2021-05-23 20:32:21 +00:00
break
2021-05-10 22:26:26 +00:00
2022-10-15 18:56:30 +00:00
self . vprint (
f " { prefix } Finished handling { ' dirty updates on ' if only_dirty else ' ' } placed clip { clip . object_id } at depth { clip . depth } " ,
component = " tags " ,
)
2021-05-10 22:26:26 +00:00
# Return if anything was modified.
return changed
2022-10-15 18:56:30 +00:00
def __handle_imports (
self , swf : SWF
) - > Dict [
int , Union [ RegisteredShape , RegisteredClip , RegisteredImage , RegisteredDummy ]
] :
external_objects : Dict [
int ,
Union [ RegisteredShape , RegisteredClip , RegisteredImage , RegisteredDummy ] ,
] = { }
2021-05-21 16:58:01 +00:00
# Go through, recursively resolve imports for all SWF files.
for tag_id , imp in swf . imported_tags . items ( ) :
2021-07-06 22:29:41 +00:00
for _name , other in self . swfs . items ( ) :
2021-05-21 16:58:01 +00:00
if other . exported_name == imp . swf :
# This SWF should have the tag reference.
if imp . tag not in other . exported_tags :
2022-10-15 18:56:30 +00:00
print (
f " WARNING: { swf . exported_name } imports { imp } but that import is not in { other . exported_name } ! "
)
2021-05-21 16:58:01 +00:00
external_objects [ tag_id ] = RegisteredDummy ( tag_id )
break
else :
2022-10-15 18:56:30 +00:00
external_objects [ tag_id ] = self . __find_import (
other , other . exported_tags [ imp . tag ]
)
2021-05-21 16:58:01 +00:00
break
else :
2021-05-24 17:38:56 +00:00
# Only display a warning if we don't have our own stub implementation of this SWF.
if repr ( imp ) not in self . __stubbed_swfs :
2022-10-15 18:56:30 +00:00
print (
f " WARNING: { swf . exported_name } imports { imp } but that SWF is not in our library! "
)
2021-05-21 16:58:01 +00:00
external_objects [ tag_id ] = RegisteredDummy ( tag_id )
# Fix up tag IDs to point at our local definition of them.
for tid in external_objects :
external_objects [ tid ] . tag_id = tid
# Return our newly populated registered object table containing all imports!
return external_objects
2022-10-15 18:56:30 +00:00
def __find_import (
self , swf : SWF , tag_id : int
) - > Union [ RegisteredShape , RegisteredClip , RegisteredImage , RegisteredDummy ] :
2021-05-21 16:58:01 +00:00
if tag_id in swf . imported_tags :
external_objects = self . __handle_imports ( swf )
if tag_id not in external_objects :
2022-10-15 18:56:30 +00:00
raise Exception (
f " Logic error, tag ID { tag_id } is an export for { swf . exported_name } but we didn ' t populate it! "
)
2021-05-21 16:58:01 +00:00
return external_objects [ tag_id ]
# We need to do a basic placement to find the registered object so we can return it.
root_clip = RegisteredClip (
None ,
swf . frames ,
swf . tags ,
2021-05-25 02:01:17 +00:00
swf . labels ,
2021-05-21 16:58:01 +00:00
)
tag = self . __find_tag ( root_clip , tag_id )
if tag is None :
2022-10-15 18:56:30 +00:00
print (
f " WARNING: { swf . exported_name } exports { swf . imported_tags [ tag_id ] } but does not manifest an object! "
)
2021-05-21 16:58:01 +00:00
return RegisteredDummy ( tag_id )
return tag
2022-10-15 18:56:30 +00:00
def __find_tag (
self , clip : RegisteredClip , tag_id : int
) - > Optional [
Union [ RegisteredShape , RegisteredClip , RegisteredImage , RegisteredDummy ]
] :
2021-05-21 16:58:01 +00:00
# Fake-execute this clip to find the tag we need to manifest.
for frame in clip . frames :
2022-10-15 18:56:30 +00:00
tags = clip . tags [
frame . start_tag_offset : ( frame . start_tag_offset + frame . num_tags )
]
2021-05-21 16:58:01 +00:00
2021-07-06 22:29:41 +00:00
for tag in tags :
2021-05-21 16:58:01 +00:00
# Attempt to place any tags.
if isinstance ( tag , AP2ShapeTag ) :
if tag . id == tag_id :
# We need to be able to see this shape to place it.
if tag . reference not in self . shapes :
2022-10-15 18:56:30 +00:00
raise Exception (
f " Cannot find shape reference { tag . reference } ! "
)
2021-05-21 16:58:01 +00:00
# This matched, so this is the import.
return RegisteredShape (
tag . id ,
2022-11-20 23:01:26 +00:00
tag . reference ,
2021-05-21 16:58:01 +00:00
self . shapes [ tag . reference ] . vertex_points ,
self . shapes [ tag . reference ] . tex_points ,
self . shapes [ tag . reference ] . tex_colors ,
self . shapes [ tag . reference ] . draw_params ,
)
2021-05-25 02:01:17 +00:00
elif isinstance ( tag , AP2ImageTag ) :
if tag . id == tag_id :
# We need to be able to see this shape to place it.
if tag . reference not in self . textures :
2022-10-15 18:56:30 +00:00
raise Exception (
f " Cannot find texture reference { tag . reference } ! "
)
2021-05-25 02:01:17 +00:00
# This matched, so this is the import.
return RegisteredImage (
tag . id ,
tag . reference ,
)
2021-05-21 16:58:01 +00:00
elif isinstance ( tag , AP2DefineSpriteTag ) :
2021-05-25 02:01:17 +00:00
new_clip = RegisteredClip ( tag . id , tag . frames , tag . tags , tag . labels )
2021-05-21 16:58:01 +00:00
if tag . id == tag_id :
# This matched, so it is the clip that we want to export.
return new_clip
# Recursively look in this as well.
maybe_tag = self . __find_tag ( new_clip , tag_id )
if maybe_tag is not None :
return maybe_tag
# We didn't find the tag we were after.
return None
2021-05-23 23:37:05 +00:00
def __render (
self ,
swf : SWF ,
only_depths : Optional [ List [ int ] ] ,
2021-05-25 02:01:36 +00:00
only_frames : Optional [ List [ int ] ] ,
2021-05-23 23:37:05 +00:00
movie_transform : Matrix ,
2021-08-11 21:40:01 +00:00
background_image : Optional [ List [ Image . Image ] ] ,
2022-07-26 23:25:35 +00:00
overridden_width : Optional [ float ] ,
overridden_height : Optional [ float ] ,
2021-05-24 17:36:34 +00:00
) - > Generator [ Image . Image , None , None ] :
2021-05-21 16:58:01 +00:00
# First, let's attempt to resolve imports.
self . __registered_objects = self . __handle_imports ( swf )
2021-05-21 21:31:13 +00:00
# Initialize overall frame advancement stuff.
2021-05-24 17:36:34 +00:00
last_rendered_frame : Optional [ Image . Image ] = None
2021-04-15 23:18:33 +00:00
frameno : int = 0
2021-04-17 23:31:08 +00:00
2021-05-21 21:31:13 +00:00
# Calculate actual size based on given movie transform.
2022-07-26 23:25:35 +00:00
actual_width = overridden_width or swf . location . width
actual_height = overridden_height or swf . location . height
2022-10-15 18:56:30 +00:00
resized_width , resized_height , _ = movie_transform . multiply_point (
Point ( actual_width , actual_height )
) . as_tuple ( )
2021-05-21 21:31:13 +00:00
2022-07-26 23:25:35 +00:00
if round ( swf . location . top , 2 ) != 0.0 or round ( swf . location . left , 2 ) != 0.0 :
# TODO: If the location top/left is nonzero, we need move the root transform
# so that the correct viewport is rendered.
print ( " WARNING: Root clip requested to play not in top-left corner! " )
2021-05-29 03:41:25 +00:00
2021-05-10 22:26:26 +00:00
# Create a root clip for the movie to play.
root_clip = PlacedClip (
- 1 ,
- 1 ,
Point . identity ( ) ,
2021-05-21 21:32:19 +00:00
Matrix . identity ( ) ,
2021-08-05 17:31:28 +00:00
AP2PlaceObjectTag . PROJECTION_AFFINE ,
2021-05-10 22:26:26 +00:00
Color ( 1.0 , 1.0 , 1.0 , 1.0 ) ,
Color ( 0.0 , 0.0 , 0.0 , 0.0 ) ,
2022-10-16 19:30:02 +00:00
HSL ( 0.0 , 0.0 , 0.0 ) ,
2021-05-10 22:26:26 +00:00
0 ,
2021-05-23 20:32:21 +00:00
None ,
2021-05-10 22:26:26 +00:00
RegisteredClip (
None ,
swf . frames ,
swf . tags ,
2021-05-25 02:01:17 +00:00
swf . labels ,
2021-05-10 22:26:26 +00:00
) ,
)
2022-07-26 23:25:35 +00:00
root_clip . _width = int ( actual_width )
root_clip . _height = int ( actual_height )
last_width = actual_width
last_height = actual_height
2021-05-24 17:36:57 +00:00
self . __root = root_clip
2021-04-15 23:18:33 +00:00
2021-05-23 23:37:05 +00:00
# If we have a background image, add it to the root clip.
2021-08-11 21:40:01 +00:00
background_object = RegisteredImage ( - 1 , " INVALID_REFERENCE_NAME " )
2021-10-24 16:16:48 +00:00
background_container : Optional [ PlacedImage ] = None
2021-08-11 21:40:01 +00:00
background_frames = 0
2021-05-23 23:37:05 +00:00
if background_image :
2021-08-11 21:40:01 +00:00
# Stretch the images to make sure they fit the entire frame.
imgwidth = background_image [ 0 ] . width
imgheight = background_image [ 0 ] . height
2021-07-06 21:58:32 +00:00
background_matrix = Matrix . affine (
2022-07-26 23:25:35 +00:00
a = actual_width / imgwidth ,
2021-05-23 23:37:05 +00:00
b = 0 ,
c = 0 ,
2022-07-26 23:25:35 +00:00
d = actual_height / imgheight ,
2021-05-23 23:37:05 +00:00
tx = 0 ,
ty = 0 ,
)
2021-08-11 21:40:01 +00:00
background_frames = len ( background_image )
2021-05-23 23:37:05 +00:00
2021-08-11 21:40:01 +00:00
# Register the background images with the texture library.
for background_frame in range ( background_frames ) :
2022-10-15 18:56:30 +00:00
if (
background_image [ background_frame ] . width != imgwidth
or background_image [ background_frame ] . height != imgheight
) :
raise Exception (
f " Frame { background_frame + 1 } of background image sequence has different dimensions than others! "
)
2021-08-11 21:40:01 +00:00
name = f " { swf . exported_name } _inserted_background_ { background_frame } "
self . textures [ name ] = background_image [ background_frame ] . convert ( " RGBA " )
2021-05-23 23:37:05 +00:00
# Place an instance of this background on the root clip.
2021-10-24 16:16:48 +00:00
background_container = PlacedImage (
- 1 ,
- 1 ,
Point . identity ( ) ,
background_matrix ,
AP2PlaceObjectTag . PROJECTION_AFFINE ,
Color ( 1.0 , 1.0 , 1.0 , 1.0 ) ,
Color ( 0.0 , 0.0 , 0.0 , 0.0 ) ,
2022-10-16 19:30:02 +00:00
HSL ( 0.0 , 0.0 , 0.0 ) ,
2021-10-24 16:16:48 +00:00
0 ,
None ,
2022-10-15 18:56:30 +00:00
background_object ,
2021-05-23 23:37:05 +00:00
)
2021-10-24 16:16:48 +00:00
root_clip . placed_objects . append ( background_container )
2021-05-23 23:37:05 +00:00
2021-05-23 20:37:18 +00:00
# Create the root mask for where to draw the root clip.
2022-10-15 18:56:30 +00:00
movie_mask = Image . new (
" RGBA " , ( resized_width , resized_height ) , color = ( 255 , 0 , 0 , 255 )
)
2021-05-23 20:37:18 +00:00
2021-05-22 21:52:17 +00:00
# These could possibly be overwritten from an external source of we wanted.
actual_mult_color = Color ( 1.0 , 1.0 , 1.0 , 1.0 )
actual_add_color = Color ( 0.0 , 0.0 , 0.0 , 0.0 )
2022-10-16 19:30:02 +00:00
actual_hsl_shift = HSL ( 0.0 , 0.0 , 0.0 )
2021-05-22 21:56:28 +00:00
actual_blend = 0
2021-05-22 21:52:17 +00:00
2021-05-31 18:13:43 +00:00
max_frame : Optional [ int ] = None
2021-05-25 02:01:36 +00:00
if only_frames :
max_frame = max ( only_frames )
2021-05-21 16:58:01 +00:00
# Now play the frames of the root clip.
2021-04-21 00:01:35 +00:00
try :
2021-05-23 20:33:05 +00:00
while root_clip . playing and not root_clip . finished :
2021-04-21 00:01:35 +00:00
# Create a new image to render into.
2022-10-15 18:56:30 +00:00
self . vprint (
f " Rendering frame { frameno + 1 } / { len ( root_clip . source . frames ) } " ,
component = " core " ,
)
2021-04-21 00:01:35 +00:00
# Go through all registered clips, place all needed tags.
2021-05-24 01:24:26 +00:00
changed = self . __process_tags ( root_clip , False )
while self . __is_dirty ( root_clip ) :
changed = self . __process_tags ( root_clip , True ) or changed
2021-04-21 00:01:35 +00:00
2021-08-11 21:40:01 +00:00
# Calculate a new background frame if needed.
2021-10-24 16:16:48 +00:00
if background_container is not None and background_frames > 0 :
# First, make sure we're still placed in the root clip, which can be undone
# if it is rewound.
for obj in root_clip . placed_objects :
if obj is background_container :
break
else :
2022-10-15 18:56:30 +00:00
self . vprint (
" Root clip was rewound, re-placing background image on clip. "
)
2021-10-24 16:16:48 +00:00
root_clip . placed_objects . append ( background_container )
# Now, update the background image if we need to.
2021-08-11 21:40:01 +00:00
background_frame = frameno % background_frames
name = f " { swf . exported_name } _inserted_background_ { background_frame } "
if background_object . reference != name :
background_object . reference = name
changed = True
2021-08-05 17:35:10 +00:00
# Adjust camera based on the movie's scaling.
if self . __camera is not None and not self . __camera . adjusted :
2022-10-15 18:56:30 +00:00
self . __camera . center = movie_transform . multiply_point (
self . __camera . center
)
2021-08-05 17:35:10 +00:00
self . __camera . adjusted = True
2021-05-25 02:01:36 +00:00
# If we're only rendering some frames, don't bother to do the draw operations
# if we aren't going to return the frame.
if only_frames and ( frameno + 1 ) not in only_frames :
2022-10-15 18:56:30 +00:00
self . vprint (
f " Skipped rendering frame { frameno + 1 } / { len ( root_clip . source . frames ) } " ,
component = " core " ,
)
2021-05-25 02:01:36 +00:00
last_rendered_frame = None
frameno + = 1
continue
2021-05-24 17:36:34 +00:00
if changed or last_rendered_frame is None :
2022-10-15 18:56:30 +00:00
if (
last_width != root_clip . _width
or last_height != root_clip . _height
) :
2022-07-26 23:25:35 +00:00
last_width = root_clip . _width
last_height = root_clip . _height
2022-10-15 18:56:30 +00:00
if (
root_clip . _width > actual_width
or root_clip . _height > actual_height
) :
print (
f " WARNING: Root clip requested to resize to { last_width } x { last_height } which overflows root canvas! "
)
2022-07-26 23:25:35 +00:00
2021-05-22 01:30:28 +00:00
# Now, render out the placed objects.
2021-05-25 02:01:36 +00:00
color = swf . color or Color ( 0.0 , 0.0 , 0.0 , 0.0 )
2022-10-15 18:56:30 +00:00
curimage = Image . new (
" RGBA " , ( resized_width , resized_height ) , color = color . as_tuple ( )
)
2021-08-03 17:04:20 +00:00
curimage = self . __render_object (
curimage ,
root_clip ,
movie_transform ,
2021-08-05 17:31:28 +00:00
AP2PlaceObjectTag . PROJECTION_AFFINE ,
2021-08-03 17:04:20 +00:00
movie_mask ,
actual_mult_color ,
actual_add_color ,
2022-10-16 19:30:02 +00:00
actual_hsl_shift ,
2021-08-03 17:04:20 +00:00
actual_blend ,
only_depths = only_depths ,
)
2021-04-21 00:01:35 +00:00
else :
# Nothing changed, make a copy of the previous render.
2021-09-20 01:16:40 +00:00
self . vprint ( " Using previous frame render " , component = " core " )
2021-05-24 17:36:34 +00:00
curimage = last_rendered_frame . copy ( )
2021-04-15 23:18:33 +00:00
2021-05-24 17:36:34 +00:00
# Return that frame, advance our bookkeeping.
2022-10-15 18:56:30 +00:00
self . vprint (
f " Finished rendering frame { frameno + 1 } / { len ( root_clip . source . frames ) } " ,
component = " core " ,
)
2021-05-24 17:36:34 +00:00
last_rendered_frame = curimage
2021-04-21 00:01:35 +00:00
frameno + = 1
2021-05-24 17:36:34 +00:00
yield curimage
2021-05-25 02:01:36 +00:00
# See if we should bail because we passed the last requested frame.
if max_frame is not None and frameno == max_frame :
break
2021-04-21 00:01:35 +00:00
except KeyboardInterrupt :
# Allow ctrl-c to end early and render a partial animation.
2022-10-15 18:56:30 +00:00
print (
f " WARNING: Interrupted early, will render only { frameno } / { len ( root_clip . source . frames ) } frames of animation! "
)
2021-04-17 23:30:46 +00:00
2021-05-23 20:32:21 +00:00
# Clean up
2021-05-25 02:01:36 +00:00
self . __root = None