CSharpGL(5)解析3DS文件并用CSharpGL渲染
我曾經寫過一個簡單的*.3ds文件的解析器,但是只能解析最基本的頂點、索引信息,且此解析器是仿照別人的C++代碼改寫的,設計的也不好,不方便擴展。
現在我重新設計實現了一個*.3ds文件的解析器,它能解析的Chunk類型更多,且容易擴展。以后需要解析更多類型的Chunk時比較簡單。
下載
這個3DS解析器是CSharpGL的一部分,CSharpGL已在GitHub開源,歡迎對OpenGL有興趣的同學加入(https://github.com/bitzhuwei/CSharpGL)
本文所用的3ds文件您可以在此(http://www.cgrealm.org/d/downpage.php?n=2&id=15764::1326768548)下載,由于文件比較大我就不上傳了。
3DS文件格式
樹
3ds文件是二進制的。3ds格式的基本單元叫塊(chunk)。我們就是讀這樣一塊一塊的信息。目錄樹如下,縮進風格體現了塊的父子關系。可見3ds模型文件和XML文件類似,都是只有1個根結點的樹狀結構。

1 0x4D4D // Main Chunk 2 ├─ 0x0002 // M3D Version 3 ├─ 0x3D3D // 3D Editor Chunk 4 │ ├─ 0x4000 // Object Block 5 │ │ ├─ 0x4100 // Triangular Mesh 6 │ │ │ ├─ 0x4110 // Vertices List 7 │ │ │ ├─ 0x4120 // Faces Description 8 │ │ │ │ ├─ 0x4130 // Faces Material 9 │ │ │ │ └─ 0x4150 // Smoothing Group List 10 │ │ │ ├─ 0x4140 // Mapping Coordinates List 11 │ │ │ └─ 0x4160 // Local Coordinates System 12 │ │ ├─ 0x4600 // Light 13 │ │ │ └─ 0x4610 // Spotlight 14 │ │ └─ 0x4700 // Camera 15 │ └─ 0xAFFF // Material Block 16 │ ├─ 0xA000 // Material Name 17 │ ├─ 0xA010 // Ambient Color 18 │ ├─ 0xA020 // Diffuse Color 19 │ ├─ 0xA030 // Specular Color 20 │ ├─ 0xA200 // Texture Map 1 21 │ ├─ 0xA230 // Bump Map 22 │ └─ 0xA220 // Reflection Map 23 │ │ // Sub Chunks For Each Map 24 │ ├─ 0xA300 // Mapping Filename 25 │ └─ 0xA351 // Mapping Parameters 26 └─ 0xB000 // Keyframer Chunk 27 ├─ 0xB002 // Mesh Information Block 28 ├─ 0xB007 // Spot Light Information Block 29 └─ 0xB008 // Frames (Start and End) 30 ├─ 0xB010 // Object Name 31 ├─ 0xB013 // Object Pivot Point 32 ├─ 0xB020 // Position Track 33 ├─ 0xB021 // Rotation Track 34 ├─ 0xB022 // Scale Track 35 └─ 0xB030 // Hierarchy Position
實際上完整的chunk列表有上千種類型,我們只需解析其中的頂點列表、面列表和紋理UV列表就行了。
以類型標識為0x4D4D的MAIN CHUNK為例,整個3ds文件的前兩個byte必須是0x4D4D,否則就說明這個文件不是3ds模型文件。然后從第3到第6個byte是一個Uint32型的數值,表示整個MAIN CHUNK的長度。由于MAIN CHUNK是整個3ds文件的根結點,它的長度也即整個3ds文件的長度。
塊(Chunk)的結構
每一個“chunk”的結構如下所示:
偏移量 |
長度 |
|
0 |
2 |
塊標識符 |
2 |
4 |
塊長: 塊數據 + 子塊內容 |
6 |
n |
塊數據 |
6+n |
m |
S子塊 |
文件內容
一個3DS文件,其中包含若干材質對象,材質對象里有材質參數和貼圖文件名;還有若干子模型,每個子模型都由頂點位置、UV位置、三角形索引和分組索引構成。分組索引是這么一個東西:它由若干三角形索引的編號和一個材質對象名組成。這個分組索引似乎暗示著:渲染過程應根據分組索引描繪的順序進行,即取出一個分組索引,綁定它指定的材質和貼圖,渲染它指定的三角形,然后取出下一個分組索引繼續上述渲染操作。我們將在后文進行驗證。
2016-01-21
今天發現有的3ds文件是沒有分組索引這個玩意的。所以要特殊處理一下。
解析器設計思路
在之前寫的解析器中使用的思路是:首先根據偏移量和長度找到一個塊的標識符,然后據此來判斷它是什么塊,遇到我們需要的塊,就進一步讀取,如果不需要,直接跳過這一塊,讀取下面的塊。這沒有用到面向對象的思想,只有面向過程編程。如果需要添加一個新的Chunk類型,修改起來是比較困難的。
我重新設計的解析器的思路如下:
遞歸讀取各個塊
讀取一個塊,然后依次讀取它的各個子塊。鑒于各個塊之間的樹狀關系,這是一個遞歸的過程。
各個類型的塊都應該繼承自同一基類型ChunkBase。對于具體的Chunk類型,只需override掉Process方法即可實現自己的解析過程。
1 public abstract class ChunkBase 2 { 3 public ChunkBase Parent; 4 public List<ChunkBase> Childern; 5 6 public uint Length; 7 public uint BytesRead; 8 9 public ChunkBase() 10 { 11 this.Childern = new List<ChunkBase>(); 12 } 13 14 internal virtual void Process(ParsingContext context) 15 { 16 var chunk = this; 17 var reader = context.reader; 18 19 while (chunk.BytesRead < chunk.Length) 20 { 21 ChunkBase child = reader.ReadChunk(); 22 child.Parent = this; 23 this.Childern.Add(child); 24 25 child.Process(context); 26 27 chunk.BytesRead += child.BytesRead; 28 } 29 } 30 }
數據字典
各個類型的Chunk都用一個具體的class類型表達,為了方便這些class類型與用ushort表達的的Chunk類型相互轉換,我們需要2個字典。

1 public static partial class ChunkBaseHelper 2 { 3 4 private static readonly Dictionary<Type, ushort> chunkTypeDict = new Dictionary<Type, ushort>(); 5 private static readonly Dictionary<ushort, Type> chunkIDDict = new Dictionary<ushort, Type>(); 6 7 /// <summary> 8 /// 開發者必須了解的東西。 9 /// </summary> 10 static ChunkBaseHelper() 11 { 12 chunkTypeDict.Add(typeof(MainChunk), 0x4D4D); 13 { 14 chunkTypeDict.Add(typeof(VersionChunk), 0x0002); 15 chunkTypeDict.Add(typeof(_3DEditorChunk), 0x3D3D); 16 { 17 chunkTypeDict.Add(typeof(ObjectBlockChunk), 0x4000); 18 { 19 chunkTypeDict.Add(typeof(TriangularMeshChunk), 0x4100); 20 { 21 chunkTypeDict.Add(typeof(VerticesListChunk), 0x4110); 22 chunkTypeDict.Add(typeof(FacesDescriptionChunk), 0x4120); 23 { 24 chunkTypeDict.Add(typeof(FacesMaterialChunk), 0x4130); 25 chunkTypeDict.Add(typeof(SmoothingGroupListChunk), 0x4150); 26 } 27 chunkTypeDict.Add(typeof(MappingCoordinatesListChunk), 0x4140); 28 chunkTypeDict.Add(typeof(LocalCoordinatesSystemChunk), 0x4160); 29 } 30 chunkTypeDict.Add(typeof(LightChunk), 0x4600); 31 { 32 chunkTypeDict.Add(typeof(SpotlightChunk), 0x4610); 33 } 34 chunkTypeDict.Add(typeof(CameraChunk), 0x4700); 35 } 36 chunkTypeDict.Add(typeof(MaterialBlockChunk), 0xAFFF); 37 { 38 chunkTypeDict.Add(typeof(MaterialNameChunk), 0xA000); 39 chunkTypeDict.Add(typeof(AmbientColorChunk), 0xA010); 40 chunkTypeDict.Add(typeof(DiffuseColorChunk), 0xA020); 41 chunkTypeDict.Add(typeof(SpecularColorChunk), 0xA030); 42 chunkTypeDict.Add(typeof(MatShininessChunk), 0xA040); 43 chunkTypeDict.Add(typeof(TextureMapChunk), 0xA200); 44 chunkTypeDict.Add(typeof(BumpMapChunk), 0xA230); 45 chunkTypeDict.Add(typeof(ReflectionMapChunk), 0xA220); 46 { 47 chunkTypeDict.Add(typeof(MappingFilenameChunk), 0xA300); 48 chunkTypeDict.Add(typeof(MappingParametersChunk), 0xA351); 49 } 50 } 51 } 52 chunkTypeDict.Add(typeof(KeyframeChunk), 0xB000); 53 { 54 chunkTypeDict.Add(typeof(MeshInformationBlockChunk), 0xB002); 55 chunkTypeDict.Add(typeof(SpotLightInformationBlockChunk), 0xB007); 56 chunkTypeDict.Add(typeof(FramesChunk), 0xB008); 57 { 58 chunkTypeDict.Add(typeof(ObjectNameChunk), 0xB010); 59 chunkTypeDict.Add(typeof(ObjectPivotPointChunk), 0xB013); 60 chunkTypeDict.Add(typeof(PositionTrackChunk), 0xB020); 61 chunkTypeDict.Add(typeof(RotationTrackChunk), 0xB021); 62 chunkTypeDict.Add(typeof(ScaleTrackChunk), 0xB022); 63 chunkTypeDict.Add(typeof(HierarchyPositionChunk), 0xB030); 64 } 65 } 66 } 67 68 chunkIDDict.Add(0x4D4D, typeof(MainChunk)); 69 { 70 chunkIDDict.Add(0x0002, typeof(VersionChunk)); 71 chunkIDDict.Add(0x3D3D, typeof(_3DEditorChunk)); 72 { 73 chunkIDDict.Add(0x4000, typeof(ObjectBlockChunk)); 74 { 75 chunkIDDict.Add(0x4100, typeof(TriangularMeshChunk)); 76 { 77 chunkIDDict.Add(0x4110, typeof(VerticesListChunk)); 78 chunkIDDict.Add(0x4120, typeof(FacesDescriptionChunk)); 79 { 80 chunkIDDict.Add(0x4130, typeof(FacesMaterialChunk)); 81 chunkIDDict.Add(0x4150, typeof(SmoothingGroupListChunk)); 82 } 83 chunkIDDict.Add(0x4140, typeof(MappingCoordinatesListChunk)); 84 chunkIDDict.Add(0x4160, typeof(LocalCoordinatesSystemChunk)); 85 } 86 chunkIDDict.Add(0x4600, typeof(LightChunk)); 87 { 88 chunkIDDict.Add(0x4610, typeof(SpotlightChunk)); 89 } 90 chunkIDDict.Add(0x4700, typeof(CameraChunk)); 91 } 92 chunkIDDict.Add(0xAFFF, typeof(MaterialBlockChunk)); 93 { 94 chunkIDDict.Add(0xA000, typeof(MaterialNameChunk)); 95 chunkIDDict.Add(0xA010, typeof(AmbientColorChunk)); 96 chunkIDDict.Add(0xA020, typeof(DiffuseColorChunk)); 97 chunkIDDict.Add(0xA030, typeof(SpecularColorChunk)); 98 chunkIDDict.Add(0xA040, typeof(MatShininessChunk)); 99 chunkIDDict.Add(0xA200, typeof(TextureMapChunk)); 100 chunkIDDict.Add(0xA230, typeof(BumpMapChunk)); 101 chunkIDDict.Add(0xA220, typeof(ReflectionMapChunk)); 102 { 103 chunkIDDict.Add(0xA300, typeof(MappingFilenameChunk)); 104 chunkIDDict.Add(0xA351, typeof(MappingParametersChunk)); 105 } 106 } 107 } 108 chunkIDDict.Add(0xB000, typeof(KeyframeChunk)); 109 { 110 chunkIDDict.Add(0xB002, typeof(MeshInformationBlockChunk)); 111 chunkIDDict.Add(0xB007, typeof(SpotLightInformationBlockChunk)); 112 chunkIDDict.Add(0xB008, typeof(FramesChunk)); 113 { 114 chunkIDDict.Add(0xB010, typeof(ObjectNameChunk)); 115 chunkIDDict.Add(0xB013, typeof(ObjectPivotPointChunk)); 116 chunkIDDict.Add(0xB020, typeof(PositionTrackChunk)); 117 chunkIDDict.Add(0xB021, typeof(RotationTrackChunk)); 118 chunkIDDict.Add(0xB022, typeof(ScaleTrackChunk)); 119 chunkIDDict.Add(0xB030, typeof(HierarchyPositionChunk)); 120 } 121 } 122 } 123 } 124 }
未定義的Chunk
3ds文件有上千種Chunk,我們暫時不會都解析出來(也沒必要全解析出來)。所以我們用一個“未定義的Chunk”類型來代表那些我們不想解析的Chunk類型。
1 /// <summary> 2 /// 3ds文件有上千種Chunk,我們暫時不會都解析出來(也沒必要全解析出來)。所以我們用一個“未定義的Chunk”類型來代表那些我們不想解析的Chunk類型。 3 /// </summary> 4 public class UndefinedChunk : ChunkBase 5 { 6 public ushort ID; 7 public bool IsChunk { get; private set; } 8 9 public UndefinedChunk() 10 { 11 this.IsChunk = true; 12 } 13 14 public override string ToString() 15 { 16 return string.Format("{0}(0x{1:X4}), position: {2}, length: {3}, read bytes: {4}", 17 this.IsChunk ? "Unknown Chunk" : "Fake Chunk", ID, Position, Length, BytesRead); 18 } 19 20 internal override void Process(ParsingContext context) 21 { 22 var chunk = this; 23 var reader = context.reader; 24 var parent = this.Parent; 25 26 uint length = this.Length - this.BytesRead; 27 28 if ((parent != null)) 29 { 30 var another = parent.Length - parent.BytesRead - this.BytesRead; 31 length = Math.Min(length, another); 32 } 33 34 reader.BaseStream.Position += length; 35 chunk.BytesRead += length; 36 if (chunk.Length != chunk.BytesRead) 37 { 38 chunk.Length = chunk.BytesRead; 39 this.IsChunk = false; 40 } 41 } 42 }
注意:這里獲取到的UndefinedChunk對象,不一定代表真的有這樣一個未被解析的Chunk,它也可能是其父Chunk的一部分數據內容。所以,我們要結合這里的another值來判斷到底應該繼續讀取多少字節,并且修補好可能出錯的chunk.Length。
讀出一個Chunk的擴展方法
每次獲取一個Chunk對象時,都是借助BinaryReader得到Chunk類型和長度的,所以我們給它一個擴展方法,用于“讀出一個Chunk”。
1 public static partial class ChunkBaseHelper 2 { 3 public static ChunkBase ReadChunk(this BinaryReader reader) 4 { 5 // 2 byte ID 6 ushort id = reader.ReadUInt16(); 7 // 4 byte length 8 uint length = reader.ReadUInt32(); 9 // 2 + 4 = 6 10 uint bytesRead = 6; 11 12 Type type; 13 if (chunkIDDict.TryGetValue(id, out type)) 14 { 15 object obj = Activator.CreateInstance(type); 16 ChunkBase result = obj as ChunkBase; 17 //result.ID = id;//不再需要記錄ID,此對象的類型就指明了它的ID。 18 result.Length = length; 19 result.BytesRead = bytesRead; 20 return result; 21 } 22 else 23 { 24 return new UndefinedChunk() { ID = id, Length = length, BytesRead = bytesRead, }; 25 } 26 } 27 }
獲取Chunk類型的ushort值
得到一個Chunk對象后,可能會需要獲取此對象代表的Chunk類型。
1 public static partial class ChunkBaseHelper 2 { 3 public static ushort GetID(this ChunkBase chunk) 4 { 5 ushort value; 6 7 if (chunk is UndefinedChunk) 8 { 9 value = (chunk as UndefinedChunk).ID; 10 } 11 else 12 { 13 Type type = chunk.GetType(); 14 value = chunkTypeDict[type];//如果此處不存在此type的key,說明static構造函數需要添加此類型的字典信息。 15 } 16 17 return value; 18 } 19 }
解析器輸出:Chunk樹
我們用TreeView控件來展示解析出來的Chunk樹。
如果不想看那些未定義的Chunk類型,可以隱藏之。
如果需要,你可以將此Chunk樹導出為文本格式:
從Chunk樹到legacy OpenGL
Dumper
已經得到了Chunk樹,下面需要得到可用于OpenGL渲染的模型。這實際上是一個語義分析和生成中間代碼的過程。以根結點MainChunk為例:
1 public static partial class ChunkDumper 2 { 3 public static void Dump(this MainChunk chunk, out ThreeDSModel4LegacyOpenGL model) 4 { 5 model = new ThreeDSModel4LegacyOpenGL(); 6 7 foreach (var item in chunk.Children) 8 { 9 if(item is VersionChunk) 10 { 11 (item as VersionChunk).Dump(model); 12 } 13 else if(item is _3DEditorChunk) 14 { 15 (item as _3DEditorChunk).Dump(model); 16 } 17 else if (item is KeyframeChunk) 18 { 19 (item as KeyframeChunk).Dump(model); 20 } 21 else if(!(item is UndefinedChunk)) 22 { 23 throw new NotImplementedException(string.Format( 24 "not dumper implemented for {0}", item.GetType())); 25 } 26 } 27 } 28 }
我們為每個Chunk類型都編寫一個Dumper,在各個Dump過程中收集需要的信息(頂點位置、UV、貼圖文件名、材質、光照等),匯總到一個ThreeDSModel4LegacyOpenGL對象,這個對象就可以用來渲染圖形了。
渲染
根據上文對分組索引的推測,我給出如下的渲染過程。

1 public class ThreeDSModel4LegacyOpenGL 2 { 3 public List<ThreeDSMesh4LegacyOpenGL> Entities = new List<ThreeDSMesh4LegacyOpenGL>(); 4 public Dictionary<string, ThreeDSMaterial4LegacyOpenGL> MaterialDict = new Dictionary<string, ThreeDSMaterial4LegacyOpenGL>(); 5 6 public void Render() 7 { 8 foreach (ThreeDSMesh4LegacyOpenGL mesh in Entities) 9 { 10 mesh.Render(this); 11 } 12 } 13 } 14 public class ThreeDSMesh4LegacyOpenGL 15 { 16 public List<Tuple<string, ushort[]>> usingMaterialIndexesList = new List<Tuple<string, ushort[]>>(); 17 // TODO: OO this 18 // fields should be private 19 // constructor with verts and faces 20 // normalize in ctor 21 22 //public ThreeDSMaterial material = new ThreeDSMaterial(); 23 //public string UsesMaterial; 24 25 // The stored vertices 26 public Vector[] Vertexes; 27 28 // The calculated normals 29 public Vector[] normals; 30 31 // The indices of the triangles which point to vertices 32 public Triangle[] TriangleIndexes; 33 34 // The coordinates which map the texture onto the entity 35 public TexCoord[] TexCoords; 36 37 bool normalized = false; 38 public ushort[] UsesIndexes; 39 40 public void Render(ThreeDSModel4LegacyOpenGL model) 41 { 42 if (TriangleIndexes == null) return; 43 44 // Draw every triangle in the entity 45 foreach (var item in this.usingMaterialIndexesList) 46 { 47 var material = model.MaterialDict[item.Item1]; 48 49 GL.Materialfv(GL.GL_FRONT_AND_BACK, GL.GL_AMBIENT, material.Ambient); 50 GL.Materialfv(GL.GL_FRONT_AND_BACK, GL.GL_DIFFUSE, material.Diffuse); 51 GL.Materialfv(GL.GL_FRONT_AND_BACK, GL.GL_SPECULAR, material.Specular); 52 GL.Materialf(GL.GL_FRONT_AND_BACK, GL.GL_SHININESS, material.Shininess); 53 54 Texture2D[] textures = new Texture2D[] { material.GetTexture(), material.GetBumpTexture(), material.GetReflectionTexture(), }; 55 bool drawn = false; 56 foreach (var texture in textures) 57 { 58 if (!(drawn && texture == null)) // 如果沒有貼圖,就只畫一次。 59 { 60 if (texture != null) 61 { 62 GL.Enable(GL.GL_TEXTURE_2D); 63 texture.Bind(); 64 } 65 66 DrawTriangles(item, texture); 67 68 if (texture != null) 69 { 70 texture.Unbind(); 71 GL.Disable(GL.GL_TEXTURE_2D); 72 } 73 } 74 75 drawn = true; 76 } 77 } 78 } 79 80 private void DrawTriangles(Tuple<string, ushort[]> usingMaterialIndexes, Texture2D texture) 81 { 82 GL.Begin(GL.GL_TRIANGLES); 83 foreach (var usingIndex in usingMaterialIndexes.Item2) 84 { 85 Triangle tri = this.TriangleIndexes[usingIndex]; 86 // Vertex 1 87 if (normalized) 88 { 89 var normal = this.normals[tri.vertex1]; 90 GL.Normal3d(normal.X, normal.Y, normal.Z); 91 } 92 if (texture != null) 93 { 94 var texCoord = this.TexCoords[tri.vertex1]; 95 GL.TexCoord2f(texCoord.U, texCoord.V); 96 } 97 { 98 var vertex = this.Vertexes[tri.vertex1]; 99 GL.Vertex3d(vertex.X, vertex.Y, vertex.Z); 100 } 101 102 // Vertex 2 103 if (normalized) 104 { 105 var normal = this.normals[tri.vertex2]; 106 GL.Normal3d(normal.X, normal.Y, normal.Z); 107 } 108 if (texture != null) 109 { 110 var texCoord = this.TexCoords[tri.vertex2]; 111 GL.TexCoord2f(texCoord.U, texCoord.V); 112 } 113 { 114 var vertex = this.Vertexes[tri.vertex2]; 115 GL.Vertex3d(vertex.X, vertex.Y, vertex.Z); 116 } 117 118 // Vertex 3 119 if (normalized) 120 { 121 var normal = this.normals[tri.vertex3]; 122 GL.Normal3d(normal.X, normal.Y, normal.Z); 123 } 124 if (texture != null) 125 { 126 var texCoord = this.TexCoords[tri.vertex3]; 127 GL.TexCoord2f(texCoord.U, texCoord.V); 128 } 129 { 130 var vertex = this.Vertexes[tri.vertex3]; 131 GL.Vertex3d(vertex.X, vertex.Y, vertex.Z); 132 } 133 } 134 GL.End(); 135 } 136 }
驗證分組索引的功能
上文中我們發現了分組索引的存在,根據它的內容推測了它的功能,現在來驗證一下。我找到一個3ds文件,用A3dsViewer打開是這樣的:
這個3ds文件附帶多個貼圖:
這個是樹皮。
這是花盆里的石頭。
這是花盆里的苔蘚(某種綠色植物?)
這是盆景的紅葉。
現在再用我制作的3DSViewer渲染看看:
整體上是對了,分組索引成功地將各個貼圖附到了對應的三角形上。
但是花盆不應該是白的,這是某些光照沒有解析的原因。
從Chunk樹到modern OpenGL
有了legacy OpenGL探路,modern OpenGL的渲染就容易多了,這里暫時不詳述。
總結
目前這個3ds解析器算是可用了,以后需要擴展時也很容易。如果能找到更多的3ds文件來測試,就能知道還需要解析哪些類型的Chunk了。
文章列表