本篇讲解的是3D游戏的场景资源打包方式,首先简单的分析一下场景中所包含的资源的类型。

场景资源一般包含:地表模型(或者是Unity Terrain),非实例化物体(摄像机、空气墙、光源、各种逻辑物体之类的)、场景物体(花草树木、房子箱子之类的)。

因为场景物体大多是公用的,所以将场景物体都打成单独的包,将地表模型、非实例化物体打包到场景包中

那么场景TestScene所对应的资源就包括:场景包TestScene.unity3d、场景资源配表TestSceneXML.xml。

打包场景单个物体的具体细节在前面的帖子中已有讲解,此处不再赘述,本帖主要分享场景本身的打包和场景资源的序列化。

(1)打包一个场景

打包当前场景的代码如下所示,使用BuildStreamedSceneAssetBundle和BuildPlayer函数都可以实现此功能。

public class PackScene
{
public static void Execute(UnityEditor.BuildTarget target)
{
string assetPath = SceneAssetProcesser.GetPlatformPath(target); string exportPath = assetPath + "Scene/";
if (Directory.Exists(exportPath) == false)
Directory.CreateDirectory(exportPath); string currentScene = EditorApplication.currentScene;
string currentSceneName = currentScene.Substring(currentScene.LastIndexOf('/') + , currentScene.LastIndexOf('.') - currentScene.LastIndexOf('/') - );
string fileName = exportPath + currentSceneName + ".unity3d";
BuildPipeline.BuildStreamedSceneAssetBundle(new string[] { EditorApplication.currentScene }, fileName, target);
// 另外一种方式
// BuildPipeline.BuildPlayer(new string[1] { EditorApplication.currentScene }, fileName, target, BuildOptions.BuildAdditionalStreamedScenes);
} }

将TestScene打包成TestScene.unity3d的包。

(2)序列化场景物体

对于一个场景物体,主要序列化其下列信息:

(1)Transform信息:位置、朝向、缩放

(2)Mesh信息:Shader名字

        主颜色Color

        光照贴图信息:是否为static对象(static的对象才有光照贴图属性)、LightmapIndex、LightmapTilingOffset

主体代码如下所示:

  public static void ExportXML(string savePath)
{
// 所有的动态加载的物体都挂在ActiveObjectRoot下面
GameObject parent = GameObject.Find("ActiveObjectRoot");
if (parent == null)
{
Debug.LogError("No ActiveObjectRoot Node!");
return;
} XmlDocument XmlDoc = new XmlDocument();
XmlElement XmlRoot = XmlDoc.CreateElement("Root");
XmlRoot.SetAttribute("level", EditorApplication.currentScene);
XmlDoc.AppendChild(XmlRoot); foreach (Transform tranGroup in parent.transform)
{
XmlElement xmlGroupNode = XmlDoc.CreateElement("Group");
XmlRoot.AppendChild(xmlGroupNode); CreateTransformNode(XmlDoc, xmlGroupNode, tranGroup); foreach (Transform tranNode in tranGroup.transform)
{
XmlElement xmlNode = XmlDoc.CreateElement("Node");
xmlGroupNode.AppendChild(xmlNode); CreateTransformNode(XmlDoc, xmlNode, tranNode);
CreateMeshNode(XmlDoc, xmlNode, tranNode);
}
} string path = savePath + "Scene/";
if (Directory.Exists(path) == false)
Directory.CreateDirectory(path);
string levelPath = EditorApplication.currentScene;
string levelName = levelPath.Substring(levelPath.LastIndexOf('/') + , levelPath.LastIndexOf('.') - levelPath.LastIndexOf('/') - );
XmlDoc.Save(path + "Xml" + levelName + ".xml");
XmlDoc = null;
} private static void CreateTransformNode(XmlDocument XmlDoc, XmlElement xmlNode, Transform tran)
{
if (XmlDoc == null || xmlNode == null || tran == null)
return; XmlElement xmlProp = XmlDoc.CreateElement("Transform");
xmlNode.AppendChild(xmlProp); xmlNode.SetAttribute("name", tran.name);
xmlProp.SetAttribute("posX", tran.position.x.ToString());
xmlProp.SetAttribute("posY", tran.position.y.ToString());
xmlProp.SetAttribute("posZ", tran.position.z.ToString());
xmlProp.SetAttribute("rotX", tran.eulerAngles.x.ToString());
xmlProp.SetAttribute("rotY", tran.eulerAngles.y.ToString());
xmlProp.SetAttribute("rotZ", tran.eulerAngles.z.ToString());
xmlProp.SetAttribute("scaleX", tran.localScale.x.ToString());
xmlProp.SetAttribute("scaleY", tran.localScale.y.ToString());
xmlProp.SetAttribute("scaleZ", tran.localScale.z.ToString());
} private static void CreateMeshNode(XmlDocument XmlDoc, XmlElement xmlNode, Transform tran)
{
if (XmlDoc == null || xmlNode == null || tran == null)
return; XmlElement xmlProp = XmlDoc.CreateElement("MeshRenderer");
xmlNode.AppendChild(xmlProp); foreach (MeshRenderer mr in tran.gameObject.GetComponentsInChildren<MeshRenderer>(true))
{
if (mr.material != null)
{
XmlElement xmlMesh = XmlDoc.CreateElement("Mesh");
xmlProp.AppendChild(xmlMesh); // 记录Mesh名字和Shader
xmlMesh.SetAttribute("Mesh", mr.name);
xmlMesh.SetAttribute("Shader", mr.material.shader.name); // 记录主颜色
XmlElement xmlColor = XmlDoc.CreateElement("Color");
xmlMesh.AppendChild(xmlColor);
bool hasColor = mr.material.HasProperty("_Color");
xmlColor.SetAttribute("hasColor", hasColor.ToString());
if (hasColor)
{
xmlColor.SetAttribute("r", mr.material.color.r.ToString());
xmlColor.SetAttribute("g", mr.material.color.g.ToString());
xmlColor.SetAttribute("b", mr.material.color.b.ToString());
xmlColor.SetAttribute("a", mr.material.color.a.ToString());
} // 光照贴图信息
XmlElement xmlLightmap = XmlDoc.CreateElement("Lightmap");
xmlMesh.AppendChild(xmlLightmap);
// 是否为static,static的对象才有lightmap信息
xmlLightmap.SetAttribute("IsStatic", mr.gameObject.isStatic.ToString());
xmlLightmap.SetAttribute("LightmapIndex", mr.lightmapIndex.ToString());
xmlLightmap.SetAttribute("OffsetX", mr.lightmapTilingOffset.x.ToString());
xmlLightmap.SetAttribute("OffsetY", mr.lightmapTilingOffset.y.ToString());
xmlLightmap.SetAttribute("OffsetZ", mr.lightmapTilingOffset.z.ToString());
xmlLightmap.SetAttribute("OffsetW", mr.lightmapTilingOffset.w.ToString());
}
}
}

生成的配表结构如下:

......
<Node name="TestObject">
<Transform posX="-25.13055" posY="12.99786" posZ="26.41202" rotX="" rotY="180.6732" rotZ="" scaleX="1.293338" scaleY="1.293338" scaleZ="1.293338" />
<MeshRenderer>
<Mesh Mesh="TestMesh01" Shader="Diffuse">
<Color hasColor="False" />
<Lightmap IsStatic="True" LightmapIndex="" OffsetX="0.06542969" OffsetY="0.06542969" OffsetZ="0.09154129" OffsetW="0.4391975" />
</Mesh>
<Mesh Mesh="TestMesh02" Shader="AlphaTest">
<Color hasColor="True" r="0.5429823" g="0.8897059" b="0.6888453" a="" />
<Lightmap IsStatic="True" LightmapIndex="" OffsetX="0.1435547" OffsetY="0.1435547" OffsetZ="0.3969002" OffsetW="-0.0005607605" />
</Mesh>
</MeshRenderer>
</Node>
......

下一篇讲解根据上述配表生成场景的具体实现...

AssetBundle系列——场景资源之打包(一)的更多相关文章

  1. AssetBundle系列——场景资源之解包(二)

    本篇接着上一篇继续和大家分享场景资源这一主题,主要包括两个方面: (1)加载场景 场景异步加载的代码比较简单,如下所示: private IEnumerator LoadLevelCoroutine( ...

  2. AssetBundle系列——共享资源打包/依赖资源打包

    有人在之前的博客中问我有关共享资源打包的代码,其实这一块很简单,就两个函数: BuildPipeline.PushAssetDependencies():依赖资源压栈: BuildPipeline.P ...

  3. (转)AssetBundle系列——共享资源打包/依赖资源打包

    有人在之前的博客中问我有关共享资源打包的代码,其实这一块很简单,就两个函数: BuildPipeline.PushAssetDependencies():依赖资源压栈: BuildPipeline.P ...

  4. AssetBundle系列——游戏资源打包(二)

    本篇接着上一篇.上篇中说到的4步的代码分别如下所示: (1)将资源打包成assetbundle,并放到自定目录下 using UnityEditor; using UnityEngine; using ...

  5. AssetBundle系列——游戏资源打包(一)

    将本地资源打包,然后放到资源服务器上供游戏客户端下载或更新.服务器上包含以下资源列表:(1)游戏内容资源assetbundle(2)资源维护列表,包含每个资源的名字(完整路径名)和对应的版本号[资源名 ...

  6. [Unity Asset]AssetBundle系列——游戏资源打包

    转载:http://www.cnblogs.com/sifenkesi/p/3557231.html 将本地资源打包,然后放到资源服务器上供游戏客户端下载或更新.服务器上包含以下资源列表:(1)游戏内 ...

  7. (转)AssetBundle系列——游戏资源打包(二)

    转自:http://www.cnblogs.com/sifenkesi/p/3557290.html 本篇接着上一篇.上篇中说到的4步的代码分别如下所示: (1)将资源打包成assetbundle,并 ...

  8. (转)AssetBundle系列——游戏资源打包(一)

    转自:http://www.cnblogs.com/sifenkesi/p/3557231.html 将本地资源打包,然后放到资源服务器上供游戏客户端下载或更新.服务器上包含以下资源列表:(1)游戏内 ...

  9. Unity 关于AssetBundle读取场景

    一. 1.关于如何打包成ab包,就不多说了,网上很多教程,siki学院也有siki老师的免费视频教程挺详细的,可以看看 http://www.sikiedu.com/my/course/74 2.为了 ...

随机推荐

  1. iOS开发——高级技术&调用地图功能的实现

    调用地图功能的实现 一:苹果自带地图 学习如逆水行舟,不进则退.古人告诉我们要不断的反思和总结,日思则日精,月思则月精,年思则年精.只有不断的尝试和总结,才能让我们的工作和生活更加 轻松愉快和美好.连 ...

  2. javascript基础知识复习一

    JavaScript 一.数据类型 A.String B.Number C.boolean  1.undefined.false.null.0.“”这五个返回的都是false: 2.NAN==NAN返 ...

  3. hive函数总结

    转自:http://www.cnblogs.com/end/archive/2012/06/18/2553682.html 1.内置运算符1.1关系运算符 运算符 类型 说明 A = B 所有原始类型 ...

  4. Why Apache Spark is a Crossover Hit for Data Scientists [FWD]

    Spark is a compelling multi-purpose platform for use cases that span investigative, as well as opera ...

  5. How do I check if a type is a subtype OR the type of an object?

    To check if a type is a subclass of another type in C#, it's easy: typeof (SubClass).IsSubclassOf(ty ...

  6. ASP怎么解除文件上传200kb限制

    第一步:修改IIS设置,允许直接编辑配置数据库.打开,Internet信息服务第二步:先在服务里关闭iis admin service服务,找到windows\system32\inetsrv\下的m ...

  7. 【转】升级Xcode6.3插件失效解决办法

     1.打开终端,输入以下代码获取到DVTPlugInCompatibilityUUID         defaults read /Applications/Xcode.app/Contents/I ...

  8. 特征工程(Feature Enginnering)学习记要

     最近学习特征工程(Feature Enginnering)的相关技术,主要包含两块:特征选取(Feature Selection)和特征抓取(Feature Extraction).这里记录一些要点 ...

  9. java模拟开锁

    java模拟开锁 service qq:928900200 Introduction to Computer Science II: CSCI142Fall 2014Lab #1Instructor: ...

  10. Mesos源码分析

    Mesos源码分析(1): Mesos的启动过程总论 Mesos源码分析(2): Mesos Master的启动之一 Mesos源码分析(3): Mesos Master的启动之二 Mesos源码分析 ...