some new functions
[swftools.git] / lib / rfxswf.h
index 95d497c..512eaa9 100644 (file)
 #include <stdio.h>
 #include <stdlib.h>
 #include <string.h>
+#include <unistd.h>
+#include <fcntl.h>
+#include <ctype.h>
 #include "../config.h"
+#include "./bitio.h"
 
 #define DEBUG_RFXSWF
 
+#ifndef TRUE
+#define TRUE (1)
+#endif
+#ifndef FALSE
+#define FALSE (0)
+#endif
+
+
+/* little/big endian stuff */
+
+//#define SWAP16(s) ((U16) ((U8*)&s)[0] | ((U16) ((U8*)&s)[1] << 8))
+//#define SWAP32(s) ((U32) ((U8*)&s)[0] | ((U32) ((U8*)&s)[1] << 8) | ((U32) ((U8*)&s)[2] << 16) | ((U32) ((U8*)&s)[3] << 24))
+
+#define PUT16(ptr,x) {((U8*)(ptr))[0]=(U8)(x);((U8*)(ptr))[1]=(U8)((x)>>8);}
+#define PUT32(ptr,x) {((U8*)(ptr))[0]=(U8)(x);((U8*)(ptr))[1]=(U8)((x)>>8);((U8*)(ptr))[2]=(U8)((x)>>16);((U8*)(ptr))[3]=(U8)((x)>>24);}
+#define GET16(ptr) (((U16)(((U8*)(ptr))[0]))+(((U16)(((U8*)(ptr))[1]))<<8))
+#define GET32(ptr) (((U16)(((U8*)(ptr))[0]))+(((U16)(((U8*)(ptr))[1]))<<8)+(((U16)(((U8*)(ptr))[2]))<<16)+(((U16)(((U8*)(ptr))[3]))<<24))
+
+#ifdef WORDS_BIGENDIAN
+#define SWAP16(s) ((((s)>>8)&0x00ff)|(((s)<<8)&0xff00))
+#define SWAP32(s) (SWAP16(((s)>>16)&0x0000ffff)|((SWAP16(s)<<16)&0xffff0000))
+#define REVERSESWAP16(x) (x)
+#define REVERSESWAP32(x) (x)
+#else
+#define SWAP16(x) (x)
+#define SWAP32(x) (x)
+#define REVERSESWAP16(s) ((((s)>>8)&0x00ff)|(((s)<<8)&0xff00))
+#define REVERSESWAP32(s) (REVERSESWAP16(((s)>>16)&0x0000ffff)|((REVERSESWAP16(s)<<16)&0xffff0000))
+#endif
 // SWF Types
 
 typedef         unsigned long   U32;
@@ -68,36 +101,35 @@ typedef struct _CXFORM
   S16           b0, b1;
 } CXFORM, * LPCXFORM;
 
+typedef struct _GRADIENT
+{
+    int num;
+    U8 ratios[8];
+    RGBA rgba[8];
+} GRADIENT;
+
 typedef struct _TAG             // NEVER access a Tag-Struct directly !
 { U16           id;
-  U32           len;
   U8 *          data;
+  U32           memsize;        // to minimize realloc() calls
+
+  U32         len;            // for Set-Access
+  U32         pos;            // for Get-Access
 
-  int           frame;
+  int           frame;          // not really up-to-date
 
   struct _TAG * next;
   struct _TAG * prev;
 
-  U32           memsize;        // to minimize realloc() calls
-  U32           pos;            // for Get/Set-Access
-  U8            bitmask;        // for Bit-Manipulating Functions [read]
-  U8            bitcount;       // [write]
+  U8            readBit;        // for Bit-Manipulating Functions [read]
+  U8            writeBit;       // [write]
+  
 } TAG, * LPTAG;
 
-typedef struct _ActionTAG 
-{ U8            op;
-  U16           len;
-  U8 *          data;
-
-  struct _ActionTAG * next;
-  struct _ActionTAG * prev;
-
-  TAG* parent; // may be null
-} ActionTAG;
-
 
 typedef struct _SWF
 { U8            fileVersion;
+  U8           compressed;     // SWF or SWC?
   U32           fileSize;       // valid after load and save
   SRECT         movieSize;
   U16           frameRate;
@@ -107,10 +139,28 @@ typedef struct _SWF
 
 // Basic Functions
 
+int  swf_ReadSWF2(struct reader_t*reader, SWF * swf);   // Reads SWF via callback
 int  swf_ReadSWF(int handle,SWF * swf);     // Reads SWF to memory (malloc'ed), returns length or <0 if fails
+int  swf_WriteSWF2(struct writer_t*writer, SWF * swf);     // Writes SWF via callback, returns length or <0 if fails
 int  swf_WriteSWF(int handle,SWF * swf);    // Writes SWF to file, returns length or <0 if fails
+int  swf_WriteSWC(int handle, SWF * swf);   // for convenience, equal to swf->compressed=1;swf_WriteSWF(..)
 int  swf_WriteCGI(SWF * swf);               // Outputs SWF with valid CGI header to stdout
 void swf_FreeTags(SWF * swf);               // Frees all malloc'ed memory for swf
+
+// for streaming:
+int  swf_WriteHeader(int handle,SWF * swf);    // Writes Header of swf to file
+int  swf_WriteHeader2(struct writer_t*writer,SWF * swf);    // Writes Header of swf to file
+int  swf_WriteTag(int handle,TAG * tag);    // Writes TAG to file
+int  swf_WriteTag2(struct writer_t*writer, TAG * t); //Write TAG via callback
+
+int  swf_ReadHeader(struct reader_t*reader, SWF * swf);   // Reads SWF Header via callback
+
+// folding/unfolding:
+
+void swf_FoldAll(SWF*swf);
+void swf_FoldSprite(TAG*tag);
+
+// basic routines:
     
 TAG * swf_InsertTag(TAG * after,U16 id);    // updates frames, if necessary
 int   swf_DeleteTag(TAG * t);
@@ -123,8 +173,8 @@ TAG * swf_PrevTag(TAG * t);
 
 int   swf_GetFrameNo(TAG * t);              // should be renamed to TagGetFrame
 U16   swf_GetTagID(TAG * t);                // ... TagGetID
-U32   swf_GetDataSize(TAG * t);             // ... TagGetDataSize
-U8*   swf_GetDataSizePtr(TAG * t);
+U32   swf_GetTagLen(TAG * t);             // ... TagGetTagLen
+U8*   swf_GetTagLenPtr(TAG * t);
 
 U32   swf_GetBits(TAG * t,int nbits);
 S32   swf_GetSBits(TAG * t,int nbits);
@@ -136,6 +186,10 @@ int   swf_SetBlock(TAG * t,U8 * b,int l);
 U8    swf_GetU8(TAG * t);                   // resets Bitcount
 U16   swf_GetU16(TAG * t);
 U32   swf_GetU32(TAG * t);
+void  swf_GetRGB(TAG * t, RGBA * col);
+void  swf_GetRGBA(TAG * t, RGBA * col);
+void  swf_GetGradient(TAG * t, GRADIENT * gradient, char alpha);
+void  swf_GetMorphGradient(TAG * tag, GRADIENT * gradient1, GRADIENT * gradient2);
 
 int   swf_SetU8(TAG * t,U8 v);              // resets Bitcount
 int   swf_SetU16(TAG * t,U16 v);
@@ -180,6 +234,7 @@ int   swf_SetRGBA(TAG * t,RGBA * col);
 #define ST_PLACEOBJECT          4
 #define ST_REMOVEOBJECT         5
 #define ST_DEFINEBITS           6
+#define ST_DEFINEBITSJPEG       6 
 #define ST_DEFINEBUTTON         7
 #define ST_JPEGTABLES           8
 #define ST_SETBACKGROUNDCOLOR   9
@@ -217,18 +272,19 @@ int   swf_SetRGBA(TAG * t,RGBA * col);
 #define ST_TEMPLATECOMMAND      49
 #define ST_GENERATOR3           51
 #define ST_EXTERNALFONT         52
+#define ST_EXPORTASSETS                56
+#define ST_IMPORTASSETS                57
+#define ST_ENABLEDEBUGGER      58
+#define ST_MX0                 59 /*(?) Components/InitClip */
+#define ST_MX1                 60 /*(?) Sorensen Video*/
+#define ST_MX2                 61 /*(?) Sorensen Video*/
+#define ST_MX3                 62 /*(?) fontinfo2? */
+#define ST_MX4                 63 /*(?) */
 
 #define ST_REFLEX              777 /* to identify generator software */
 
 // Advanced Funtions
 
-// swfdump.c
-
-void swf_DumpHeader(FILE * f,SWF * swf);
-void swf_DumpMatrix(FILE * f,MATRIX * m);
-void swf_DumpTag(FILE * f,TAG * t); 
-char* swf_TagGetName(TAG*tag);
-
 // swfshape.c
 
 typedef struct _LINESTYLE
@@ -249,24 +305,34 @@ typedef struct _SHAPE           // NEVER access a Shape-Struct directly !
   { LINESTYLE * data;
     U16         n;
   } linestyle;
-                  // note: changes of shape structure
-  struct                  // lead to incompatible .efont formats
+                  
+  struct                    
   { FILLSTYLE * data;
     U16         n;
   } fillstyle;
-
-  S32           px;
-  S32           py;
-  
   struct
   { U16         fill;
     U16         line;
   } bits;
-  
+                               // used by Get/SetSimpleShape and glyph handling
   U8 *          data;
   U32           bitlen;         // length of data in bits
 } SHAPE, * LPSHAPE;
 
+/* data/bitlen can be expanded into a SHAPELINE record using
+   parseShapeData: */
+typedef struct _SHAPELINE
+{
+    enum {moveTo, lineTo, splineTo} type;
+    SCOORD x,y;
+    SCOORD sx,sy; //only if type==splineTo
+    int fillstyle0;
+    int fillstyle1;
+    int linestyle;
+    struct _SHAPELINE*next;
+} SHAPELINE;
+
 // Shapes
 
 int   swf_ShapeNew(SHAPE ** s);
@@ -293,44 +359,88 @@ int   swf_ShapeSetCurve(TAG * t,SHAPE * s,S32 x,S32 y,S32 ax,S32 ay);
 int   swf_ShapeSetCircle(TAG * t,SHAPE * s,S32 x,S32 y,S32 rx,S32 ry);
 int   swf_ShapeSetEnd(TAG * t);
 
+SHAPELINE* swf_ParseShapeData(U8*data, int bits, int fillbits, int linebits);
+SRECT     swf_GetBoundingBox(SHAPELINE*shape);
 
 // swffont.c
 
 // does not support wide characters !
 
-#define MAX_CHAR_PER_FONT 256
+#define MAX_CHAR_PER_FONT 512
+
+typedef struct _KERNING
+{
+  U16        char1;
+  U16        char2;
+  U16        adjustment;
+} SWFKERNING;
 
 typedef struct _SWFLAYOUT
-{ S16         ascent;
-  S16         descent;
-  S16         leading;
-  SRECT       bounds[MAX_CHAR_PER_FONT];
-  struct
-  { U16       count;
-    U8 *      data;  // size = count*4 bytes
-  } kerning;
+{ S16          ascent;
+  S16          descent;
+  S16          leading;
+  SRECT      * bounds;
+  U16         kerningcount;
+  SWFKERNING * kerning;
 } SWFLAYOUT, * LPSWFLAYOUT;
 
+typedef struct
+{ S16         advance;
+  SHAPE *     shape;
+} SWFGLYPH;
+
+#define FONT_STYLE_BOLD 1
+#define FONT_STYLE_ITALIC 2
+#define FONT_ENCODING_UNICODE 1
+#define FONT_ENCODING_ANSI 2 
+#define FONT_ENCODING_SHIFTJIS 4
+
 typedef struct _SWFFONT
-{ U16           id;
+{ int          id; // -1 = not set
+  U8           version; // 0 = not set, 1 = definefont, 2 = definefont2
   U8 *          name;
   SWFLAYOUT *   layout;
-
-  U8            flags; // bold/italic/unicode/ansi ...
-
-  U16           codes[MAX_CHAR_PER_FONT];
+  U16          numchars;
+  U16          maxascii; // highest mapped ascii value
   
-  struct
-  { U16         advance;
-    U16         gid;            // Glyph-ID after DefineFont
-    SHAPE *     shape;
-  }             glyph[MAX_CHAR_PER_FONT];
+  U8           style;
+  U8           encoding;
+
+  U16  *       glyph2ascii;
+  int  *       ascii2glyph;
+  SWFGLYPH *   glyph;
 } SWFFONT, * LPSWFFONT;
 
 typedef struct _FONTUSAGE
 { U8 code[MAX_CHAR_PER_FONT];
 } FONTUSAGE, * LPFONTUSAGE;
 
+#define ET_HASTEXT 32768
+#define ET_WORDWRAP 16384
+#define ET_MULTILINE 8192
+#define ET_PASSWORD 4096
+#define ET_READONLY 2048
+#define ET_HASTEXTCOLOR 1024
+#define ET_HASMAXLENGTH 512
+#define ET_HASFONT 256
+#define ET_X3 128
+#define ET_X2 64
+#define ET_HASLAYOUT 32
+#define ET_NOSELECT 16
+#define ET_BORDER 8
+#define ET_X1 4
+#define ET_X0 2
+#define ET_USEOUTLINES 1
+
+typedef struct _EditTextLayout
+{
+    U8 align; // 0=left, 1=right, 2=center, 3=justify
+    U16 leftmargin;
+    U16 rightmargin;
+    U16 indent;
+    U16 leading;
+} EditTextLayout;
+
 int swf_FontEnumerate(SWF * swf,void (*FontCallback) (U16,U8*));
 // -> void fontcallback(U16 id,U8 * name); returns number of defined fonts
 
@@ -348,11 +458,21 @@ int swf_FontInitUsage(FONTUSAGE * use);
 int swf_FontUse(FONTUSAGE * use,U8 * s);
 
 int swf_FontSetDefine(TAG * t,SWFFONT * f);
+int swf_FontSetDefine2(TAG * t,SWFFONT * f);
 int swf_FontSetInfo(TAG * t,SWFFONT * f);
 
+void swf_FontAddLayout(SWFFONT * f, int ascent, int descent, int leading);
+
+int swf_FontExtract_DefineTextCallback(int id,SWFFONT * f,TAG * t,int jobs, 
+       void(*callback)(int*chars, int nr, int id));
+
+// the following two functions are obsolete and will be removed soon
 int swf_FontExport(int handle,SWFFONT * f);
 int swf_FontImport(int handle,SWFFONT * * f);
 
+void swf_WriteFont(SWFFONT* font, char* filename);
+SWFFONT* swf_ReadFont(char* filename);
+
 void swf_FontFree(SWFFONT * f);
 
 U32 swf_TextGetWidth(SWFFONT * font,U8 * s,int scale);
@@ -364,14 +484,17 @@ int swf_TextSetCharRecord(TAG * t,SWFFONT * font,U8 * s,int scale,U8 gbits,U8 ab
 int swf_TextPrintDefineText(TAG * t,SWFFONT * f);
 // Prints text defined in tag t with font f to stdout
 
+/* notice: if you set the fontid, make sure the corresponding font has layout information */
+void swf_SetEditText(TAG*tag, U16 flags, SRECT r, char*text, RGBA*color, 
+       int maxlength, U16 font, U16 height, EditTextLayout*layout, char*variable);
 
-// swfobject.c
-
-// Always use ST_PLACEOBJECT2 !!!
+// swfdump.c
 
-int swf_ObjectPlace(TAG * t,U16 id,U16 depth,MATRIX * m,CXFORM * cx,U8 * name);
-int swf_ObjectPlaceClip(TAG * t,U16 id,U16 depth,MATRIX * m,CXFORM * cx,U8 * name, U16 clipaction);
-int swf_ObjectMove(TAG * t,U16 depth,MATRIX * m,CXFORM * cx);
+void swf_DumpHeader(FILE * f,SWF * swf);
+void swf_DumpMatrix(FILE * f,MATRIX * m);
+void swf_DumpTag(FILE * f,TAG * t); 
+char* swf_TagGetName(TAG*tag);
+void swf_DumpFont(SWFFONT * font);
 
 // swfbutton.c
 
@@ -402,6 +525,7 @@ int swf_ObjectMove(TAG * t,U16 depth,MATRIX * m,CXFORM * cx);
 #define BC_END                  0x0800
 #define BC_INSERT               0x0a00
 #define BC_DELETE               0x0c00
+#define BC_CLEAR               0x0e00
 #define BC_BACKSPACE            0x1000
 #define BC_ENTER                0x1a00
 #define BC_CURSORUP             0x1c00
@@ -409,8 +533,30 @@ int swf_ObjectMove(TAG * t,U16 depth,MATRIX * m,CXFORM * cx);
 #define BC_PAGEUP               0x2000
 #define BC_PAGEDOWN             0x2200
 #define BC_TAB                  0x2400
+#define BC_ESCAPE              0x3600
 #define BC_SPACE                0x4000
 
+/* these are probably only valid with linux:
+   Ctrl-A       0x0200
+   Ctrl-X       0x3000
+   Ctrl-Y       0x3200
+   Ctrl-Z       0x3400
+   Escape/Ctrl-[ 0x3600
+   Ctrl-\       0x3800
+   Ctrl-]       0x3a00
+   Ctrl-^       0x3c00
+   Ctrl-/       0x3e00
+   */
+
+/* everything above 0x4000 is standard ascii:
+   0x4000 ' ' 0x4200 '!' 0x4600 '#' 0x4800 '$' 0x4a00 '%' 0x4c00 '&' ...
+   0x6000 '0' ... 0x7200 '9' 
+   0x8000 '@' 
+   0x8200 'A' ...  0xb400 'Z' 
+   ...
+   0xfc00 '~'
+ */
+
 // Button Flag
 
 #define BF_TRACKMENU            0x01
@@ -429,7 +575,8 @@ int swf_SetJPEGBitsLines(JPEGBITS * jpegbits,U8 ** data,int n);
 int swf_SetJPEGBitsLine(JPEGBITS * jpegbits,U8 * data);
 int swf_SetJPEGBitsFinish(JPEGBITS * jpegbits);
 
-int swf_SetJPEGBits(TAG * t,char * fname,int quality); // paste jpg file into swf stream
+int swf_SetJPEGBits(TAG * t,char * fname,int quality);
+void swf_SetJPEGBits2(TAG * t,U16 width,U16 height,RGBA * bitmap,int quality);
 
 #define BYTES_PER_SCANLINE(width) ((width+3)&0xfffffffc)
 
@@ -449,18 +596,31 @@ int swf_SetLosslessBits(TAG * t,U16 width,U16 height,void * bitmap,U8 bitmap_fla
 int swf_SetLosslessBitsIndexed(TAG * t,U16 width,U16 height,U8 * bitmap,RGBA * palette,U16 ncolors);
 int swf_SetLosslessBitsGrayscale(TAG * t,U16 width,U16 height,U8 * bitmap);
 
+#ifndef RFXSWF_DISABLESOUND
+
+// swfsound.c
+void swf_SetSoundStreamHead(TAG*tag, int avgnumsamples);
+/* expects 2304 samples */
+void swf_SetSoundStreamBlock(TAG*tag, S16*samples, char first);
+
+#endif // RFXSWF_DISABLESOUND
+
 // swftools.c
 
 U8 swf_isDefiningTag(TAG * t);
+U8 swf_isPseudoDefiningTag(TAG * t);
 U8 swf_isAllowedSpriteTag(TAG * t);
 U16 swf_GetDefineID(TAG * t);
+void swf_SetDefineID(TAG * t, U16 newid);
 U16 swf_GetPlaceID(TAG * t); //PLACEOBJECT, PLACEOBJECT2 (sometimes), REMOVEOBJECT
 U16 swf_GetDepth(TAG * t); //PLACEOBJECT,PLACEOBJECT2,REMOVEOBJECT,REMOVEOBJECT2
-char* swf_GetTagName(TAG * t); //PLACEOBJECT2, FRAMELABEL
+char* swf_GetName(TAG * t); //PLACEOBJECT2, FRAMELABEL
 MATRIX * swf_MatrixJoin(MATRIX * d,MATRIX * s1,MATRIX * s2);
 MATRIX * swf_MatrixMapTriangle(MATRIX * m,int dx,int dy,
                     int x0,int y0,int x1,int y1,int x2,int y2);
-
+int swf_GetNumUsedIDs(TAG * t);
+void swf_GetUsedIDs(TAG * t, int * positions);
+void swf_Relocate(SWF*swf, char*bitmap); // bitmap is 65536 bytes, bitmap[a]==0 means id a is free
 
 // swfcgi.c
 
@@ -468,11 +628,153 @@ void swf_uncgi();  // same behaviour as Steven Grimm's uncgi-library
 
 // swfaction.c
 
-ActionTAG* swf_GetActions(TAG*tag);
-void swf_SetActions(TAG*tag, ActionTAG*actions);
+typedef struct _ActionTAG 
+{ U8            op;
+  U16           len;
+  U8 *          data;
+
+  struct _ActionTAG * next;
+  struct _ActionTAG * prev;
+
+  struct _ActionTAG * parent;
+  U8 tmp[4]; // store small operands here.
+} ActionTAG;
+
+typedef struct _ActionMarker
+{
+  ActionTAG* atag;
+} ActionMarker;
+
+ActionTAG* swf_ActionGet(TAG*tag);
+void swf_ActionFree(ActionTAG*tag);
+void swf_ActionSet(TAG*tag, ActionTAG*actions);
 void swf_DumpActions(ActionTAG*atag, char*prefix);
 void swf_ActionEnumerateURLs(ActionTAG*atag, char*(*callback)(char*));
 void swf_ActionEnumerateTargets(ActionTAG*atag, char*(*callback)(char*));
 void swf_ActionEnumerateStrings(ActionTAG*atag, char*(*callback)(char*));
 
+ActionTAG* action_End(ActionTAG*atag);
+ActionTAG* action_NextFrame(ActionTAG*atag);
+ActionTAG* action_PreviousFrame(ActionTAG*atag);
+ActionTAG* action_Play(ActionTAG*atag);
+ActionTAG* action_Stop(ActionTAG*atag);
+ActionTAG* action_ToggleQuality(ActionTAG*atag);
+ActionTAG* action_StopSounds(ActionTAG*atag);
+ActionTAG* action_Add(ActionTAG*atag);
+ActionTAG* action_Subtract(ActionTAG*atag);
+ActionTAG* action_Multiply(ActionTAG*atag);
+ActionTAG* action_Divide(ActionTAG*atag);
+ActionTAG* action_Equals(ActionTAG*atag);
+ActionTAG* action_Less(ActionTAG*atag);
+ActionTAG* action_And(ActionTAG*atag);
+ActionTAG* action_Or(ActionTAG*atag);
+ActionTAG* action_Not(ActionTAG*atag);
+ActionTAG* action_StringEquals(ActionTAG*atag);
+ActionTAG* action_StringLength(ActionTAG*atag);
+ActionTAG* action_StringExtract(ActionTAG*atag);
+ActionTAG* action_Pop(ActionTAG*atag);
+ActionTAG* action_ToInteger(ActionTAG*atag);
+ActionTAG* action_GetVariable(ActionTAG*atag);
+ActionTAG* action_SetVariable(ActionTAG*atag);
+ActionTAG* action_SetTarget2(ActionTAG*atag);
+ActionTAG* action_StringAdd(ActionTAG*atag);
+ActionTAG* action_GetProperty(ActionTAG*atag);
+ActionTAG* action_SetProperty(ActionTAG*atag);
+ActionTAG* action_CloneSprite(ActionTAG*atag);
+ActionTAG* action_RemoveSprite(ActionTAG*atag);
+ActionTAG* action_Trace(ActionTAG*atag);
+ActionTAG* action_StartDrag(ActionTAG*atag);
+ActionTAG* action_EndDrag(ActionTAG*atag);
+ActionTAG* action_StringLess(ActionTAG*atag);
+ActionTAG* action_RandomNumber(ActionTAG*atag);
+ActionTAG* action_MBStringLength(ActionTAG*atag);
+ActionTAG* action_CharToAscii(ActionTAG*atag);
+ActionTAG* action_AsciiToChar(ActionTAG*atag);
+ActionTAG* action_GetTime(ActionTAG*atag);
+ActionTAG* action_MBStringExtract(ActionTAG*atag);
+ActionTAG* action_MBCharToAscii(ActionTAG*atag);
+ActionTAG* action_MBAsciiToChar(ActionTAG*atag);
+ActionTAG* action_Delete(ActionTAG*atag);
+ActionTAG* action_Delete2(ActionTAG*atag);
+ActionTAG* action_DefineLocal(ActionTAG*atag);
+ActionTAG* action_CallFunction(ActionTAG*atag);
+ActionTAG* action_Return(ActionTAG*atag);
+ActionTAG* action_Modulo(ActionTAG*atag);
+ActionTAG* action_NewObject(ActionTAG*atag);
+ActionTAG* action_DefineLocal2(ActionTAG*atag);
+ActionTAG* action_InitArray(ActionTAG*atag);
+ActionTAG* action_Makehash(ActionTAG*atag);
+ActionTAG* action_TypeOf(ActionTAG*atag);
+ActionTAG* action_TargetPath(ActionTAG*atag);
+ActionTAG* action_Enumerate(ActionTAG*atag);
+ActionTAG* action_Add2(ActionTAG*atag);
+ActionTAG* action_Less2(ActionTAG*atag);
+ActionTAG* action_Equals2(ActionTAG*atag);
+ActionTAG* action_ToNumber(ActionTAG*atag);
+ActionTAG* action_ToString(ActionTAG*atag);
+ActionTAG* action_PushDuplicate(ActionTAG*atag);
+ActionTAG* action_StackSwap(ActionTAG*atag);
+ActionTAG* action_GetMember(ActionTAG*atag);
+ActionTAG* action_SetMember(ActionTAG*atag);
+ActionTAG* action_Increment(ActionTAG*atag);
+ActionTAG* action_Decrement(ActionTAG*atag);
+ActionTAG* action_CallMethod(ActionTAG*atag);
+ActionTAG* action_NewMethod(ActionTAG*atag);
+ActionTAG* action_BitAnd(ActionTAG*atag);
+ActionTAG* action_BitOr(ActionTAG*atag);
+ActionTAG* action_BitXor(ActionTAG*atag);
+ActionTAG* action_BitLShift(ActionTAG*atag);
+ActionTAG* action_BitRShift(ActionTAG*atag);
+ActionTAG* action_BitURShift(ActionTAG*atag);
+ActionTAG* action_GotoFrame(ActionTAG*atag, U16 frame);
+ActionTAG* action_GetUrl(ActionTAG*atag, char* url, char* label);
+ActionTAG* action_StoreRegister(ActionTAG*atag, U8 reg);
+ActionTAG* action_Constantpool(ActionTAG*atag, char* constantpool);
+ActionTAG* action_WaitForFrame(ActionTAG*atag, U16 frame, U8 skip);
+ActionTAG* action_SetTarget(ActionTAG*atag, char* target);
+ActionTAG* action_GotoLabel(ActionTAG*atag, char* label);
+ActionTAG* action_WaitForFrame2(ActionTAG*atag, U8 skip);
+ActionTAG* action_With(ActionTAG*atag, char*object);
+ActionTAG* action_PushString(ActionTAG*atag, char*str);
+ActionTAG* action_PushFloat(ActionTAG*atag, float f);
+ActionTAG* action_PushNULL(ActionTAG*atag);
+ActionTAG* action_PushRegister(ActionTAG*atag, U8 reg);
+ActionTAG* action_PushBoolean(ActionTAG*atag, char c);
+ActionTAG* action_PushDouble(ActionTAG*atag, double d);
+ActionTAG* action_PushInt(ActionTAG*atag, int i);
+ActionTAG* action_PushLookup(ActionTAG*atag, U8 index);
+ActionTAG* action_Jump(ActionTAG*atag, U16 branch);
+ActionTAG* action_GetUrl2(ActionTAG*atag, U8 method);
+ActionTAG* action_DefineFunction(ActionTAG*atag, U8*data, int len);
+ActionTAG* action_If(ActionTAG*atag, U16 branch);
+ActionTAG* action_Call(ActionTAG*atag);
+ActionTAG* action_GotoFrame2(ActionTAG*atag, U8 method);
+ActionMarker action_setMarker(ActionTAG*atag);
+void action_fixjump(ActionMarker m1, ActionMarker m2);
+
+// swfobject.c
+
+// The following routines only use placeobject2:
+
+int swf_ObjectPlace(TAG * t,U16 id,U16 depth,MATRIX * m,CXFORM * cx,U8 * name);
+int swf_ObjectPlaceClip(TAG * t,U16 id,U16 depth,MATRIX * m,CXFORM * cx,U8 * name, U16 clipaction);
+int swf_ObjectMove(TAG * t,U16 depth,MATRIX * m,CXFORM * cx);
+
+typedef struct _SWFPLACEOBJECT {
+    U16 depth;
+    U16 id; // may be 0
+    bool move; //true: move/replace character, false: set character
+    MATRIX matrix;
+    CXFORM cxform;
+    U16 ratio;
+    U8*name;
+    U16 clipdepth;
+    ActionTAG* actions;
+} SWFPLACEOBJECT;
+
+void swf_SetPlaceObject(TAG * t,SWFPLACEOBJECT* obj);
+void swf_GetPlaceObject(TAG * t,SWFPLACEOBJECT* obj);
+void swf_PlaceObjectFree(SWFPLACEOBJECT* obj);
+
 #endif
+