swf文件的数据结构以及转为exe或从exe中剥离出swf的源代码
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
swf文件的数据结构以及转为exe或从exe中剥离出swf的源代码
SWF文件是由一个文件头,以及一系列的标签组成的。
标签类型有两种:定义型标签和控制型标签。
定义型标签把所有物体定义成一个个角色,这些角色存在字典里面。
控制型标签操作这些角色,并且控制影片的流程。
swf 的长度单位使用twips(缇),1像素=20缇,这样,分数的像素可以使用整数的缇表示。
swf 的字符串是以00结束的byte序列,采用utf-8编码。
下面介绍swf文件的数据结构。
一、文件头
文件头的结构见表1。
表1:SWF文件头
---------------------------------------------------------------------------
字段字节数说明
---------------------------------------------------------------------------
标识符 3 “FWS”或“CWS”的ascii码
版本号 1 版本,例如:1表示swf1,6表示swf6
文件长度 4 长整形,文件的字节数,低位在前
帧大小RECT结构SWF场景的大小,单位为twip(缇,1缇=1/20像素,与VB不同)
帧速度 2 以8.8表示的浮点数(也有文章说这是一个整形数),默认=000C
帧数 2 整形,影片总帧数,低位在前
---------------------------------------------------------------------------
说明:
1.文件头以标识符开始,不是“FWS”就是“CWS”。
FWS表示该文件未压缩;CWS 表示文件从 RECT 字段开始(第 9 个字节开始)至文件结尾的所有内容,都是使用开放标准 ZLIB 压缩过的(CWS 是在 SWF6 以后才被使用的。
对其解压缩可以使用 zlib1.dll 中提供的uncompress)。
本文的所有数据结构分析都是针对“FWS”的,因为“CWS”压缩过的数据看起来面目全非,没法进行分析。
2.文件长度字段,如果是 FWS,该字段表示包括文件头在内的整个文件的准确大小;如果是CWS,该字段表示解压后文件的大小(不包括前 8 个字节),而不是实际文件的大小,。
这样做的目的是让解压后的大小可见,可以使解压过程更加有效。
3.帧大小字段表示影片的宽度和高度.它是一个 RECT 结构,根据 4 个点的坐标可计算出宽高,其结构见表2。
4.帧速率表示理想的每秒播放帧数,如果 SWF 文件包含声音流数据,或者 Flas3 播放器运行在一个慢速 CPU 上,这个速率是不能保证的。
这 2 个字节中,前字节表示小数,后字节表示整数,不过一般极少有小数位的帧速率,所以一般我们只计算整数就可以了,如:00 0C,表示每秒 12 帧。
表2:RECT结构
---------------------------------------------------
字段类型说明
---------------------------------------------------
Nbits UB(5)指定后面每个字段各占二进制的几位
Xmin SB(Nbits)x最小值,一般=0
Xmax SB(Nbits)x最大值,即影片宽度
Ymin SB(Nbits)y最小值,一般=0
Ymax SB(Nbits)y最大值,即影片高度
---------------------------------------------------
说明:
1.RECT字段是采用 swf 文件格式规范中定义的“位值”(bit_value)进行存储的,其特点是不按字节区分,而按最小位数(bits)将值连续存储,在末字节中空位补0。
比如两个用 9 位表示的无符号值7、8 将占用 3 个字节,表示为二进制就是:00000011 10000010 00000000,将 3 个字节按位连在一起,前 9 位值为7,再 9 位值为8,在第 3 个字节的其他空位全部补0。
2.Nbits字段使用整个结构的前5位来指定后面4个字段各占二进制的几位。
例如:
78 00 05 5F 00 00 0F A0 00
换成二进制:
01111000 00000000 00000101 01011111 00000000 00000000 00001111 10100000 00000000 先取前5位转换为10进制:01111=15,这就是说,后面的4个字段,各占用15位,共15×4+5=65位,
需要9个字节,72-65=7,后面有7个空闲位。
划分如下:
01111 000000000000000 010101011111000 000000000000000 001111101000000 0000000 然后我们可以算出
Xmin = (000000000000000)2=(0)10
Xmax = (010101011111000)2=(11000)10
Ymin = (000000000000000)2=(0)10
Ymax = (001111101000000)2=(8000)10
所以,影片的宽高是11000×8000缇,换算成像素是:550×400
关于影片宽、高的计算,笔者编写了一段代码,见文后的“附一”。
二、标签
在文件头后面的是一些标签化的数据块。
所有的标签都是用一种通用格式。
所以任何程序在解析一个 SWF文件时,都可以跳过那些还不明确的标签。
在每个标签中的数据可以指向这个标签中的偏移量,但绝不能指向另外一个标签的偏移量。
这样,在用工具处理 SWF 文件的时候就任意可以删除、插入和修改,而 SWF 文件不会被破坏。
这种方式保证了版本的兼容性,即使出现了新的标签类型,老版本的播放器还是能够解析完整个 swf 文件而不出现错误,大不了就是不能提供新的功能而已。
每个标签都由标签头和数据体两部分组成。
标签头包括标签类型和标签长度。
标签头格式有两种:短型和长型。
短型由 2 字节构成,长型由 6 字节构成。
短型标签头用于数据体不超过 62 字节的标签;长型标签头则用于数据体在 4G 之内的任何标签中。
标签头格式见表3和表4。
表3:标签头(短型)
----------------------
字段占位说明
----------------------
标签类型10 高10位
标签长度 6 低6位
数据体≤62字节
----------------------
说明:
1.标签头是一个 2 字节的整形数(低位在前高位在后),例如标签头数据为44 11,那么:“44 11”→“11 44”=“00010001 01000100”,高10位是(0001000101)2=(69)10,低6位是(000100)2=(4)10。
查询swf文件中的tag值和action值,可以知道标签类型码=69的是FileAttribute标签。
低6位的值4表示标签长度为4个字节。
2.标签长度是指数据体的长度,不包括标签头的 2 字节。
表4:标签头(长型)
-----------------------------------------------
字段占位说明
-----------------------------------------------
标签类型10 高10位,意义同短型标签头
6 低6位,数值固定为3F(二进制6个1)
标签长度32
数据体≥63字节,<4GB字节
-----------------------------------------------
说明:
区分短型标签头和长型标签头的关键,就是看标签头的低6位是否全1。
三、标签类型
SWF有两种类型的标签:
1.定义型标签(definition tag):
这类标签定义 SWF影片的内容,如各种形状,文字,位图,声音等等。
每个定义型标签都被分配了一个唯一的编号(标识号),以供其它标签引用,这叫做角色标识(character ID),编号的原则是从1开始顺序编号。
flash播放器把这些角色放到一个存储空间里面,这个存储空间我们一般叫它字典。
定义型标签是不会绘制任何图形的,也就是说不会产生任何动画。
2.控制型标签(control tag):
这类标签用来产生和操作字典中的角色实例的渲染,并且控制影片的流程。
这类标签如有character ID,并不是用来标识自己的,而是用来调用的。
3.字典
字典是已经定义好的所有角色的仓库,并且可以通过控制型标签来使用它。
建立和使用字典的过程是以下这样的:
①一个定义型标签定义了一些内容,如形体,字体,位图或者声音。
②定义型标签给该内容赋上一个唯一的角色标识(CharacterID),这是一个从1开始的数字标识,如果中间出现缺漏,从缺漏开始的所有character ID都被忽略,而重复的话,后出现的将覆盖先出现的标签。
③依据角色标识把内容存到字典中。
④一个控制型标签根据角色标识从字典中找出相应的内容,然后给这个内容执行一些动作,比如显示一个形体,或者播放一个声音。
每个控制型标签都只指定一个唯一的标识。
相同的标识是不允许的。
举个象征性的例子,第一个角色的标识是1,第二个角色的标识是2,依次类推。
角色标识为0的是一个特殊的标识,被看作是空角色。
控制型标签并不是唯一指向字典的标签。
定义型标签也可以指向多个角色来定义一些更复杂的角色。
例如,定义按钮(DefineButton)和定义精灵(DefineSprite)标签都是根据其它角色来定义它们的内容的。
定义文字(DefineText)标签可以指向字体角色来为文字选择不同的字体。
四、标签的顺序
除了FileAttribute要放在最前(这是一些全局的东西),结束标签要放在最末,标签之间一般不需要顺序,但是要保证依存关系,比如标签B要引用标签A的标志,就不能在A的前面出现。
Stream标签需要按照顺序。
一个定义了角色的定义型标签必须在引用这个角色的控制型标签之前。
五、swf必有的几个标签介绍
1.SetBackgroundColor标签
该标签是 Flash7 及以前版本的 SWF 文件里紧跟在文件头后面的第一个标签:
------------------------------------------------
名称占位说明
------------------------------------------------
标签头16 43 02,短型,类型码=9
背景色24 RGB类型,3个字节分别表示红、绿、蓝
------------------------------------------------
说明:
标签头的2字节数据是16进制数据,类型码“=”号后面的是10进制数据,以下同。
2.FileAttributes标签
该标签是 Flash8 及以后版本的 SWF 文件里紧跟在文件头后面的第一个标签:
--------------------------------------------------------------------------------- 名称占位说明
--------------------------------------------------------------------------------- 标签头16 44 11,短型,标签类型=69
Reserved 3 总是0
hasMetaData 1 =1表示含有MetaData标签,=0表示不含
Reserved 3 总是0
UseNetWork 1 =1表示在本地加载时有网络权限,=0表示在本地加载时只有本地权限。
Reserved 24 总是0
---------------------------------------------------------------------------------
3.ShowFrame标签
这是文件倒数第2个标签,它是必然存在的。
结构如下:
------------------------------------
名称占位说明
------------------------------------
标签头16 40 00,短型,类型码=1
------------------------------------
说明:
标签长度=0表示后面没有数据体。
该标签在文件中会多次出现,它表示当前帧显示完毕。
4.end标签
结束标签,必然是文件的最后一个标签。
------------------------------------
名称占位说明
------------------------------------
标签头16 00 00,短型,类型码=0
------------------------------------
六、其它重要标签介绍
以下有的标签在文件中会多次出现。
1.DefineFont2
这个标签的作用是定义一个字体,或者一组静态轮廓字,用以给DefineEdit Text使用。
关于文字的几乎所有信息,都可以在这个标签中进行设置,它的结构如下:
------------------------------------------------------------------------------------------
占位名称备注
------------------------------------------------------------------------------------------
16 Header 标签头, 类型码=48
16 FontID 字符,唯一的标号
1 FontFlagsHasLayout 根据字面解释,判断是否有变型的标记
1 FontFlagsShiftJIS 是否使用ShiftJIS编码
1 FontFlagsSmallText 是否使用小字体显示
1 FontFlagsANSI 是否使用ANSI编码
1 FontFlagsWideOffsets 是否使用32位偏移量
1 FontFlagsWideCodes 是否使用16位文字编码
1 FontFlagsItalic 文字是否是斜体
1 FontFlagsBold 文字是否是粗体
8 LanguageCode 语言编码有相应编码表对应
8 FontNameLen 字体文件名长度
* FontName 字体文件名(utf-8编码)字段长度=FontNameLen×8
16 NumGlyphs 轮廓字个数
32/16 OffsetTable 根据FontFlagsWideOffsets
32/16 CodeTableOffset 同上
* GlyphShapeTable 轮廓字信息,为shape结构(又是一个复杂结构),字段长度=不定×NumGlyphs
16/8 CodeTable 根据FontFlagsWideCodes,编码表为固定值UCS-2
16/0 FontAscent 根据FontFlagsHasLayout,否则无该字段
16/0 FontDescent 根据FontFlagsHasLayout,否则无该字段
16/0 FontLeading 根据FontFlagsHasLayout,否则无该字段
* FontAdvanceTable 根据FontFlagsHasLayout,否则无该字段,如有,字段长度=16×NumGlyphs
* FontBoundsTable 根据FontFlagsHasLayout,否则无该字段,如有,字段长度=RECT ×NumGlyphs
16/0 KerningCount 根据FontFlagsHasLayout,否则无该字段
*16/0 FontKerningTable 根据FontFlagsHasLayout,否则无该字段,如有,字段长度
=KERNINGRECORD×KerningCount
------------------------------------------------------------------------------------------
说明:
1.*标记说明该字段的占位是不确定的,看后面的备注说明。
2.占位是分数的,表示如果有这个字段,占位为分子值,否则为分母值。
3.如果单纯分析动态文本的这个标签的信息,只需要分析到上面的fontName部分就足够了,其他信息只对轮廓字,也就是静态文字有效。
动态文字在信息上,关键的只有一个字体名,而静态文字却包含了他的轮廓信息(包含在shape里),这是动态文字和静态文字最大的不同。
2.DoAction标签
----------------------------------------------------------------------------
标签头 2 3F 03,长型,类型码=12
标签长度 4 02 00 00 00,表示后面的数据体占2个字节
数据体 2 07 00,07表示Action代码,查Action代码可以知道它表示ActionStop,也就是写在第一帧的代码AS Code;00表示DoAction标签结束。
----------------------------------------------------------------------------
3.FrameLabel标签(帧标签)
----------------------------------------------------------------------------
标签头 2 FF 0A,长型,类型码=43
标签长度 4 06 00 00 00,表示后面的数据体占6个字节
数据体 6 73 74 61 72 74 00,字符串“start”,00是字符串结束标志。
----------------------------------------------------------------------------
七、播放器对SWF文件的处理
Flash播放器在一个显示帧标签到来之前会处理显示帧标签之前SWF文件的所有标签。
在这个时候,播放列表被复制到屏幕上面,与此同时Flash播放器在处理下一帧之前是空闲的。
第一
帧所显示的内容,是在第一个显示帧标签之前的所有控制型标签操作产生的累积效果。
第二帧所显示的内容,是从文件开始到第二个显示帧标签所有控制型标签操作产生的累积效果。
以此类推。
八、由swf文件转换为的exe格式的文件
exe 格式的 swf 文件不过是一个 Flash 播放器程序后面跟着一个 swf 文件,两个文件合成一个文件,就这么简单!由于它自带了播放器,所以比较方便,但缺点是文件体积大,每个这种文件内都含有一个播放器,有这个必要吗?
转换步骤:
1.以二进制方式读入你选定的 Flash 播放器,并原封不动地写入新文件。
这个播放器如果是自己编写的,那么还要对代码做一点修改(详见文后的附二),否则无法播放exe格式的swf。
2.以二进制方式读入一个 swf 文件内容,并原封不动地写入新文件。
3.写入四个字节的文件标识符“56 34 12 FA”。
4.写入 swf 文件长度。
九、由exe格式的文件中剥离出swf文件
剥离步骤:
1.以二进制方式打开 exe 格式的 Flash 文件,并获取内含的 swf 文件的长度(最后 4 个字节)。
2.原文件总长度减去 swf 文件长度,得到 swf 文件在原文件中的位址,从这个位址处开始读入。
3.把读入的数据写盘。
十、相互转换的代码
新建一个工程,在窗体上添加 3 个文本框、2 个按纽、1 个公用对话框。
Text1 输入全路径源文件名,Text2输入转换后的全路径文件名,Text3 输入全路径播放器文件名。
按纽1的标题是“转为exe”,按纽2的标题是“转为swf”。
代码如下:
Option Explicit
Private Sub Text1_DblClick() '选择源文件
On Error GoTo 100
CD.Filter = "*.exe;*.swf|*.exe;*.swf"
CD.ShowOpen
If Len(Dir(CD.FileName)) = 0 Then Exit Sub
Text1 = CD.FileName
Text2 = Left(Text1, Len(Text1) - 3) & IIf(LCase(Right(Text1, 3)) = "swf", "exe", "swf") 100
End Sub
Private Sub Text3_DblClick() '选择播放器
On Error GoTo 100
CD.Filter = "播放器文件(*.exe)|*.exe"
CD.ShowOpen
If Len(Dir(CD.FileName)) Then Text3 = CD.FileName
100
End Sub
Private Sub Command1_Click() 'SWF转EXE
If Len(Text1) = 0 Or LCase(Right(Text1, 3)) <> "swf" Then MsgBox "请选择要转换的 SWF 文件": Exit Sub
If Len(Text3) = 0 Or LCase(Right(Text3, 3)) <> "exe" Then MsgBox "请输入播放器文件名": Exit Sub
If Len(Text2) = 0 Then MsgBox "请输入要保存的 EXE 文件名": Exit Sub
If SWFtoEXE(Text1, Text2, Text3) Then
MsgBox "转换成功!"
Else
MsgBox "转换失败!"
End If
End Sub
Private Sub Command2_Click() 'EXE转SWF
If Len(Text1) = 0 Or LCase(Right(Text1, 3)) <> "exe" Then MsgBox "请选择要转换的 EXE 文件": Exit Sub
If Len(Text2) = 0 Then MsgBox "请输入要保存的 SWF 文件名": Exit Sub
If EXEtoSWF(Text1, Text2) Then
MsgBox "转换成功!"
Else
MsgBox "转换失败!"
End If
End Sub
Private Function SWFtoEXE(swfFile As String, exeFile As String, PlayreFile As String) As Boolean
On Error GoTo 100
Dim fLen As Long
Dim Dat() As Byte
Open swfFile For Binary As #1 '打开待转换的SWF文件
Open exeFile For Binary As #2 '创建将生成的EXE文件
Open PlayreFile For Binary As #3 '打开播放器EXE文件
fLen = LOF(3)
ReDim Dat(fLen - 1)
Get #3, , Dat
Put #2, , Dat '将播放器写入文件2
fLen = LOF(1)
ReDim Dat(fLen - 1)
Get #1, , Dat
Put #2, , Dat '将SWF文件写入文件2
ReDim Dat(3)
Dat(0) = Val(&H56): Dat(1) = Val(&H34): Dat(2) = Val(&H12): Dat(3) = Val(&HFA) '56 34 12 FA是EXE形式的标识符
Put #2, , Dat '写入标识符
Put #2, , fLen '写入SWF文件长度
SWFtoEXE = True
100
Close
End Function
Private Function EXEtoSWF(exeFile As String, swfFile As String) As Boolean
On Error GoTo 100
Dim Dat() As Byte
Dim fLen As Long
Dim n As Long, i As Long, st As String
Open exeFile For Binary As #1
fLen = LOF(1) '获取exe文件长度
ReDim Dat(3)
Get #1, fLen - 7, Dat '获取exe格式的Flash文件标识
For i = 0 To 3: st = st & Hex(Dat(i)): Next
If st <> "563412FA" Then GoTo 100 '如果不是EXE格式的Flash文件退出
Get #1, fLen - 3, n '获取exe文件中的swf部分长度
ReDim Dat(n - 1)
fLen = fLen - n - 7
Get #1, fLen, Dat '读取exe文件中的swf部分
Close #1
st = ""
For i = 0 To 2: st = st & Chr(Dat(i)): Next
If InStr("FWS,CWS", st) = 0 Then Exit Function '如果不是EXE格式的Flash文件退出
Open swfFile For Binary As #2
Put #2, , Dat
Close #2
EXEtoSWF = True
Exit Function
100
Close
End Function
对 EXEtoSWF 函数的代码稍加改动,还可以由exe格式的文件中剥离出播放器来。
附一:计算 swf 影片宽、高的代码
Private Sub Command1_Click()
Dim swfFile As String, swfMark As String * 3
Dim mem As Byte, w As Integer, h As Integer
Dim z As String, i As Integer, k1 As Integer, k2 As Integer
swfFile = "I:\swf动画\丁香花.swf" '全路径swf文件名
Open swfFile For Binary As #1 '打开swf文件
Get #1, , swfMark
If swfMark <> "FWS" Then GoTo 100 '如果不是"FWS"退出
Get #1, 9, mem
z = HexToBin(Hex(mem))
k1 = BinToDec(Left(z, 5)) '计算Nbits字段的二进制位数
i = k1 * 4 + 5
k2 = i \ 8: If i Mod 8 Then k2 = k2 + 1 '计算RECT结构有多少字节
z = Right(z, 3)
For i = 1 To k2 - 1
Get #1, , mem
z = z & HexToBin(Hex(mem)) '把RECT结构数据化为二进制字串
Next
w = BinToDec(Mid(z, k1 + 1, k1)) / 20
h = BinToDec(Mid(z, k1 * 3 + 1, k1)) / 20
MsgBox "影片宽:" & w & "像素,高:" & h & "像素"
100
Close #1
End Sub
Private Function BinToDec(Dat As String) As Integer
Dim A As Integer, B As Integer, D As String, i As Integer, L As Integer
L = Len(Dat)
For i = 0 To L - 1: D = Mid(Dat, L - i, 1): A = Val(D): B = B + A * 2 ^ i: Next BinToDec = B
End Function
Private Function HexToBin(Dat As String) As String
Dim A As Integer, B As Integer, C As Integer, D As String
C = Val("&H" & Dat)
Do: A = Int(C / 2): B = C Mod 2: C = A: D = B & D: Loop While C > 0
HexToBin = Right("0000000" & D, 8)
End Function
附二、使自编的播放器也能够播放exe格式的swf文件
先简单介绍一下App对象的 2 个属性,要想播放exe格式的swf文件,这 2 个属性是关键。
1.EXEName 属性
EXEName 属性返回当前正在运行的可执行文件名(无路径也无扩展名),这个名称一般就是我们用播放器工程生成 exe 文件时所键入的名称,但如果是运行 exe 格式的 swf 文件,那么这个名称就是 exe 格式的 swf 文件名。
比如说,我的播放器名是“王牌播放器”,swf文件名是“丁香花.swf”,转换后是“丁香花.exe”,那么,当我们点击“王牌播放器.exe”时(或者点击用王牌播放器关联的媒体文件时),EXEName 属性返回的是“王牌播放器”,而当我们点击“丁香花.exe”时,EXEName 属性返回的是“丁香花”。
2.ProductName 属性
ProductName 属性返回运行中的应用程序的产品名,这个产品名是我们用播放器工程生成exe 文件时,在“生成工程”对话框的“选项”中的“类型”下拉框里的第一项“产品名”键入的名称。
所以,我们要想播放 exe 格式的 swf 文件,就必须使这个产品名与播放器名相同,例如,都是“王牌播放器”。
我下面介绍的代码都写在窗体的 Activate 事件中,当 Activate 事件发生时,窗体已经成形了。
要注意一点:代码中的 fName 必须是模块级的字符变量,而在你的播放器代码的“播放”过程中,fName 就是要打开的媒体文件名。
编程思路是:
1.当点击 exe 格式的 swf 文件时,程序判断上述的两个属性值是否相同,如果不同,才接着往下执行。
2.从 exe 格式的 swf 文件中剥离出swf,保存到系统的临时文件夹。
3.程序从系统文件夹中打开 swf 文件开始播放。
代码如下:
Dim fName As String '媒体文件名
Private Sub Form_Activate()
If App.EXEName = App.ProductName Then Exit Sub '如果可执行文件名与产品名相同退出
Dim Dat() As Byte, fLen As Long, n As Long, i As Long, st As String
fName = App.Path & "\" & App.EXEName
Open fName & ".exe" For Binary As #1
fLen = LOF(1) '获取exe文件长度
ReDim Dat(3)
Get #1, fLen - 7, Dat '获取exe格式的Flash文件标识
For i = 0 To 3: st = st & Hex(Dat(i)): Next
If st <> "563412FA" Then Close #1: Exit Sub '如果不是EXE格式的Flash文件退出
Get #1, fLen - 3, n '获取exe文件中的swf部分长度
ReDim Dat(n - 1)
fLen = fLen - n - 7
Get #1, fLen, Dat '读取exe文件中的swf部分
Close #1
st = ""
For i = 0 To 2: st = st & Chr(Dat(i)): Next
If InStr("FWS,CWS", st) = 0 Then Exit Sub '如果不是EXE格式的Flash文件退出
fName = Environ("TEMP") & "\" & App.EXEName & ".swf" '把剥离出的swf保存到临时文件夹Open fName For Binary As #2
Put #2, , Dat
Close #2
播放 '调用你写的播放过程
End Sub
[此帖子已被 wzc2011 在 2011-7-20 21:58:43 编辑过]。