文档章节

【原创】Unity3D 场景导出成 XML 并解析还原场景

MrBlack
 MrBlack
发布于 2015/07/25 12:14
字数 1441
阅读 75
收藏 0

为了尽可能加快从网络加载场景,我们通常可以把场景先导出成 XML,把优先级高的资源优先加载并显示(地形等),把可以进入场景之后再加载的对象放到最后(比如场景里面的怪物等),本篇一部分代码引用自:http://www.xuanyusong.com/archives/1919,导出场景部分在原作者的代码基础进行了优化,并且整理成了更加方便,容易使用的类库。

先来搭建测试场景(测试场景来源网络),并整理场景中的对象,如图:

然后把场景中的对象都设置成预设,方便打包成 assetbundle 文件(如何打包预设请查看),如图:

接着我们编写把场景打包成 XML 的代码,取名 ExportSceneToXml.cs,大家可以先看这篇文章http://www.xuanyusong.com/archives/1919

我在此基础上面进行了优化,全部代码如下:

using UnityEngine;
using UnityEditor;
using System.Collections;
using System.Collections.Generic;
using System.Xml;
using System.IO;
using System.Text;

public class ExportSceneToXml : Editor 
{
	[MenuItem("Assets/Export Scene To XML From Selection")]
	static void ExportXML()
	{
		string path = EditorUtility.SaveFilePanel ("Save Resource", "", "New Resource", "xml");
		if (path.Length != 0) 
		{
			Object[] selectedAssetList = Selection.GetFiltered (typeof(Object), SelectionMode.DeepAssets);
			
			//遍历所有的游戏对象
			foreach (Object selectObject in selectedAssetList) 
			{
				// 场景名称
				string sceneName = selectObject.name;
				// 场景路径
				string scenePath = AssetDatabase.GetAssetPath(selectObject);
				// 场景文件
				//string xmlPath = path; //Application.dataPath + "/AssetBundles/Prefab/Scenes/" + sceneName + ".xml";
				// 如果存在场景文件,删除
				if(File.Exists(path)) File.Delete(path);
				// 打开这个关卡
				EditorApplication.OpenScene(scenePath);

				XmlDocument xmlDocument = new XmlDocument();
				// 创建XML属性
				XmlDeclaration xmlDeclaration = xmlDocument.CreateXmlDeclaration("1.0", "utf-8", null);
				xmlDocument.AppendChild(xmlDeclaration);
				// 创建XML根标志
				XmlElement rootXmlElement = xmlDocument.CreateElement("root");
				// 创建场景标志
				XmlElement sceneXmlElement = xmlDocument.CreateElement("scene");
				sceneXmlElement.SetAttribute("sceneName", sceneName);
				
				foreach (GameObject sceneObject in Object.FindObjectsOfType(typeof(GameObject)))
				{
					// 如果对象是激活状态
					if (sceneObject.transform.parent == null && sceneObject.activeSelf)
					{
						// 判断是否是预设
						if(PrefabUtility.GetPrefabType(sceneObject) == PrefabType.PrefabInstance)
						{
							// 获取引用预设对象
							Object prefabObject = EditorUtility.GetPrefabParent(sceneObject);
							if(prefabObject != null)
							{
								XmlElement gameObjectXmlElement = xmlDocument.CreateElement("gameObject");
								gameObjectXmlElement.SetAttribute("objectName", sceneObject.name);
								gameObjectXmlElement.SetAttribute("objectAsset",  prefabObject.name);
								
								XmlElement transformXmlElement = xmlDocument.CreateElement("transform");
								
								// 位置信息
								XmlElement positionXmlElement = xmlDocument.CreateElement("position");
								positionXmlElement.SetAttribute("x", sceneObject.transform.position.x.ToString());
								positionXmlElement.SetAttribute("y", sceneObject.transform.position.y.ToString());
								positionXmlElement.SetAttribute("z", sceneObject.transform.position.z.ToString());
								
								// 旋转信息
								XmlElement rotationXmlElement = xmlDocument.CreateElement("rotation");
								rotationXmlElement.SetAttribute("x", sceneObject.transform.rotation.eulerAngles.x.ToString());
								rotationXmlElement.SetAttribute("y", sceneObject.transform.rotation.eulerAngles.y.ToString());
								rotationXmlElement.SetAttribute("z", sceneObject.transform.rotation.eulerAngles.z.ToString());
								
								// 缩放信息
								XmlElement scaleXmlElement = xmlDocument.CreateElement("scale");
								scaleXmlElement.SetAttribute("x", sceneObject.transform.localScale.x.ToString());
								scaleXmlElement.SetAttribute("y", sceneObject.transform.localScale.y.ToString());
								scaleXmlElement.SetAttribute("z", sceneObject.transform.localScale.z.ToString());
								
								transformXmlElement.AppendChild(positionXmlElement);
								transformXmlElement.AppendChild(rotationXmlElement);
								transformXmlElement.AppendChild(scaleXmlElement);    
								
								gameObjectXmlElement.AppendChild(transformXmlElement);
								sceneXmlElement.AppendChild(gameObjectXmlElement);
							}
						}
					}
				}
				rootXmlElement.AppendChild(sceneXmlElement);
				xmlDocument.AppendChild(rootXmlElement);
				// 保存场景数据
				xmlDocument.Save(path);
				// 刷新Project视图
				AssetDatabase.Refresh();
			}
		}
	}
}

然后我们选中需要打包的场景,选择把场景打包成 XML 的选项,如图:

生成完成,我们可以查看生成出的 XML 内容,如图:

这儿为什么说是对原作者的代码进行了优化,下面我们可以把场景中的一个对象名称改成与预设名称不同,如图:

然后再次导出成 XML 文件,查看 XML 生成的内容我们可以发现,我们可以正确找到预设的名称,如图:

另外,我们还可以选择场景中的哪些文件不用导出,方法很简单,我们可以先把场景中的对象禁用,再导出,如图:

再次查看新导出的 XML 文件,我们会发现 XML 中已经不包括了被禁用对象的配置信息,如图:

以上两点是对原作者代码的优化,而且我也改成了使用右键导出,个人感觉这样更加方便、实用。

现在回到场景中,我们可以把场景里面的对象全部删除,因为场景中已经不需要这些对象了,我们需要通过代码创建这些对象,如图:

下面我们来看如何还原场景,有了 XML,我们解析 XML 就可以了,资源的加载可以看这篇文章(查看详情),加载场景以及预设资源(assetbundle)的代码如下:

using UnityEngine;
using System.Collections.Generic;

public class LoaderScene : MonoBehaviour 
{
	public UISlider progressBar;
	public UILabel lblStatus;

	private string scenePath;
	
	void Awake()
	{
		string prefabPath = "file:///" + Application.dataPath + "/Assets/{0}.assetbundle";

		this.scenePath = "file:///" + Application.dataPath + "/Assets/MainScene.unity3d";

		IList<WwwLoaderPath> pathList = new List<WwwLoaderPath> ();
		pathList.Add (new WwwLoaderPath (this.scenePath, Random.Range (0, 100), WwwLoaderTypeEnum.UNITY_3D));
		pathList.Add (new WwwLoaderPath (string.Format(prefabPath, "Lights"), Random.Range (0, 100), WwwLoaderTypeEnum.ASSET_BUNDLE));
		pathList.Add (new WwwLoaderPath (string.Format(prefabPath, "Particles"), Random.Range (0, 100), WwwLoaderTypeEnum.ASSET_BUNDLE));
		pathList.Add (new WwwLoaderPath (string.Format(prefabPath, "PhysicsCube"), Random.Range (0, 100), WwwLoaderTypeEnum.ASSET_BUNDLE));
		pathList.Add (new WwwLoaderPath (string.Format(prefabPath, "Player"), Random.Range (0, 100), WwwLoaderTypeEnum.ASSET_BUNDLE));
		pathList.Add (new WwwLoaderPath (string.Format(prefabPath, "Stamps"), Random.Range (0, 100), WwwLoaderTypeEnum.ASSET_BUNDLE));
		pathList.Add (new WwwLoaderPath (string.Format(prefabPath, "Statics"), Random.Range (0, 100), WwwLoaderTypeEnum.ASSET_BUNDLE));
		pathList.Add (new WwwLoaderPath (string.Format(prefabPath, "Terrain"), Random.Range (0, 100), WwwLoaderTypeEnum.ASSET_BUNDLE));
		pathList.Add (new WwwLoaderPath (string.Format(prefabPath, "Trees"), Random.Range (0, 100), WwwLoaderTypeEnum.ASSET_BUNDLE));


		this.lblStatus.text = "场景加载中,请稍候。。。";

		WwwLoaderManager.instance.Loader (pathList, onLoaderProgress, onLoaderComplete, "MainScene");
	}

	private void onLoaderProgress(string path, float currentValue, float totalValue)
	{
		this.progressBar.value = currentValue;
	}

	private void onLoaderComplete()
	{
		this.lblStatus.text = "场景正在初始化,请等待。。。";
		Application.LoadLevelAsync("MainScene");
	}
}

然后新建立一个 C# 文件,取名:InitObject.cs,代码如下:

using UnityEngine;
using System.Collections;
using System.Xml;

public class InitObject : MonoBehaviour 
{
	void Awake()
	{
		string xmlPath = Application.dataPath + "/Assets/MainScene.xml";
		string prefabPath = "file:///" + Application.dataPath + "/Assets/{0}.assetbundle";

		XmlDocument xmlDocument = new XmlDocument();
		xmlDocument.Load (xmlPath);
		
		// 使用 XPATH 获取所有 gameObject 节点
		XmlNodeList xmlNodeList = xmlDocument.SelectNodes("//gameObject");
		foreach(XmlNode xmlNode in xmlNodeList)
		{
			string gameObjectName = xmlNode.Attributes["objectName"].Value;
			string prefabName = xmlNode.Attributes["objectAsset"].Value;

			AssetBundle assetBundle = WwwDataManager.instance.GetDataAssetBundle(string.Format(prefabPath, prefabName));
			if(assetBundle != null)
			{
				GameObject assetObject = (GameObject)assetBundle.Load(prefabName, typeof(GameObject));
				if(assetObject != null)
				{
					GameObject gameObject = (GameObject)Instantiate(assetObject);
					// 使用 XPATH 获取 位置、旋转、缩放数据
					XmlNode positionXmlNode = xmlNode.SelectSingleNode("descendant::position");
					XmlNode rotationXmlNode = xmlNode.SelectSingleNode("descendant::rotation");
					XmlNode scaleXmlNode = xmlNode.SelectSingleNode("descendant::scale");
					
					if(positionXmlNode != null && rotationXmlNode != null && scaleXmlNode != null)
					{
						gameObject.transform.position = new Vector3(float.Parse(positionXmlNode.Attributes["x"].Value), float.Parse(positionXmlNode.Attributes["y"].Value), float.Parse(positionXmlNode.Attributes["z"].Value));
						gameObject.transform.rotation = Quaternion.Euler(new Vector3(float.Parse(rotationXmlNode.Attributes["x"].Value), float.Parse(rotationXmlNode.Attributes["y"].Value), float.Parse(rotationXmlNode.Attributes["z"].Value)));
						gameObject.transform.localScale = new Vector3(float.Parse(scaleXmlNode.Attributes["x"].Value), float.Parse(scaleXmlNode.Attributes["y"].Value), float.Parse(scaleXmlNode.Attributes["z"].Value));
					}
				}
				// 卸载引用的加载资源,释放内存
				assetBundle.Unload(false);
			}
		}
		xmlDocument = null;
	}
}

然后我们在空的场景中新建立一个空对象,并且把代码挂载到这个空对象上面,如图:

再然后我们把这个场景打包成 .unity3d 文件,方便从网络上面加载(详情可以查看这篇文章),这样所有的准备工作都已经做好了,全部的配置文件以及资源文件如下:

我们从加载场景运行项目,我们可以先看到依次在加载主场景资源,加载完成之后进入主场景,根据 XML 的内容,原场景被还原了回来,如图:

下载地址:链接: http://pan.baidu.com/s/1pJMK5r9 密码: gkxu

© 著作权归作者所有

MrBlack
粉丝 0
博文 14
码字总数 11101
作品 0
闸北
高级程序员
私信 提问
Unity3D 游戏引擎之游戏场景的切换与持久化简单数据的储存(十四)

Unity3D 游戏引擎之游戏场景的切换与持久化简单数据的储存 雨松MOMO原创文章如转载,请注明:转载自雨松MOMO的博客原文地址:http://blog.csdn.net/xys289187120/article/details/6983054 持久...

彭博
2012/03/09
322
0
Unity3D 游戏引擎之IOS高级界面发送消息与Unity3D消息的接收(九)

Unity3D 游戏引擎之IOS高级界面发送消息与Unity3D消息的接收 雨松MOMO原创文章如转载,请注明:转载自雨松MOMO的博客原文地址:http://blog.csdn.net/xys289187120/article/details/6926746 ...

彭博
2012/03/09
799
0
Unity3D 游戏引擎之IOS触摸屏手势控制镜头旋转与缩放(八)

Unity3D 游戏引擎之IOS触摸屏手势控制镜头旋转与缩放 雨松MOMO原创文章如转载,请注明:转载自雨松MOMO的博客原文地址:http://blog.csdn.net/xys289187120/article/details/6910267 前几篇文...

彭博
2012/03/09
484
0
ARKit 1.0丨 002:ARKit Remote的使用(手机App与Unity的互联)

版权声明:欢迎转载,转载请注明出处 https://blog.csdn.net/weixin_38239050/article/details/86777118 Remote:远程 ARKit Remote目的:因为Unity不是手机,无法使用ARKit功能,所以做出A...

橙子va
02/08
0
0
Mogoson/CubemapRenderer

CubemapRenderer English Manual 概述 Unity Cubemap渲染器。 需求 有时,发现一些Unity资源包中的场景的环境效果不错,希望将其渲染成Cubemap供天空盒使用。 Unity场景中需要制作一些反射效...

Mogoson
2018/08/22
0
0

没有更多内容

加载失败,请刷新页面

加载更多

UAVStack功能上新:新增JVM监控分析工具

UAVStack推出的JVM监控分析工具提供基于页面的展现方式,以图形化的方式展示采集到的监控数据;同时提供JVM基本参数获取、内存dump、线程分析、内存分配采样和热点方法分析等功能。 引言 作为...

宜信技术学院
18分钟前
4
0
MySQL的5种时间类型的比较

日期时间类型 占用空间 日期格式 最小值 最大值 零值表示 DATETIME 8 bytes YYYY-MM-DD HH:MM:SS 1000-01-01 00:00:00 9999-12-31 23:59:59 0000-00-00 00:00:00 TIMESTAMP 4 bytes YYYY-MM......

物种起源-达尔文
25分钟前
6
0
云服务OpenAPI的7大挑战,架构师如何应对?

阿里妹导读:API 是模块或者子系统之间交互的接口定义。好的系统架构离不开好的 API 设计,而一个设计不够完善的 API 则注定会导致系统的后续发展和维护非常困难。比较好的API设计样板可以参...

阿里云官方博客
28分钟前
5
0
Rancher + VMware PKS实现全球数百站点的边缘K8S集群管理

Sovereign Systems是一家成立于2007年的技术咨询公司,帮助客户将传统数据中心技术和应用程序转换为更高效的、基于云的技术平台,以更好地应对业务挑战。曾连续3年提名CRN,并且在2012年到2...

RancherLabs
33分钟前
5
0
6、根据坐标,判断该坐标是否在地图区域范围内

最近在写配送区域相关的代码,具体需求如下: 根据腾讯地图划分配送区域,总站下边设多个配送分站,然后将订单中的收货地址将其分配给不同的配送分站。 1、地图区域划分(腾讯地图) 1.1、H...

有一个小阿飞
34分钟前
4
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部