本篇讲解的是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. 64-bit Tips

    终究还是来了.Apple下发了支持64位的最后通牒: As we announced in October, beginning February 1, 2015 new iOS apps submi ...

  2. Liferay7 BPM门户开发之30: 通用帮助类Validator、ArrayUtil、StringUtil等使用

    废话不多说,直接上代码. 验证类Validator 主要是空验证.数字.格式验证 调用的例子: protected void validateEmailFrom(ActionRequest actio ...

  3. Alcatraz的安装和使用

    一.简单说明 Alcatraz 是一款 Xcode的插件管理工具,可以用来管理XCode的 插件.模版以及颜色配置的工具. 二.如何安装 1.github地址:https://github.com/a ...

  4. C#中服务端接受前端JSON字符串转换成字典集合

    我们是否可以把从前端接受的JSON字符串转换成字典集合呢? 比如从前端接收:{'size':'10', 'weight':'10kg'} 在服务端转换成:[{size:"10"}, ...

  5. 机器学习编程语言之争,Python 夺魁【转载+整理】

    原文地址 en cn 本文内容 表现平平的 MATLAB 貌似强大的 Julia 本身无错的 R 语言 逐渐没落的 Perl 老而弥坚的 Python 我个人很喜欢 Python~ 随着科技的发展,拥 ...

  6. 读匿名object对象的属性值

    读匿名object对象的属性值 1.定义读object对象值的功能方法 public static class StaticClass { public static string ValueByKe ...

  7. lua中常量的实现及表的深拷贝实现

    废话:好久没在这里写博客了...主要原因是我买了个域名hanxi.info并在github上搭建了个人博客... lua中默认是没有c中的const常量的,在csdn上找到了一个使用setmetata ...

  8. Asp.net Core 1.0.1升级到Asp.net Core 1.1.0 Preview版本发布到Windows Server2008 R2 IIS中的各种坑

    Asp.net Core 1.0.1升级到Asp.net Core 1.1.0后,程序无法运行了 解决方案:在project.json中加入runtime节点 "runtimes" ...

  9. android 中handler的用法分析 (二)

    .Looper 的构造方法是私有的,不能在package外面直接初始化.一般通过Looper.prepare()初始化.Looper.myLooper()获取.2.Looper 中的静态变量 Thre ...

  10. State状态设计模式

    1.状态模式:改变对象的行为 一个用来改变类的(状态的)对象. 2:问题:当你自己实现 State 模式的时候就会碰到很多细节的问题,你必须根据自己的需要选择合适的实现方法, 比如用到的状态(Stat ...