参考:http://www.xuanyusong.com/archives/1919

http://www.omuying.com/article/48.aspx
 
主要功能:
1.导出场景的配置文件
2.导出当前场景中资源的AssetBundle
3.客户端从服务器获取配置文件
4.解析配置文件,并根据配置文件下载AssetBundle
5.实例化并还原场景
 
1.场景设置:将需要导出的场景资源设置为预设
 
2.将场景配置导出为XML文件
 

[code]csharpcode:

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("objectAssetURL", 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();
}
}
}
}

导出结果参考:

[code]xmlcode:

<?xml version="1.0" encoding="utf-8"?>
<root>
<scene sceneName="DongTaiJiaZaiTest">
<gameObject objectName="Box" objectAssetURL="Box">
<transform>
<position x="-1.293883" y="-0.07" z="1.41" />
<rotation x="270" y="0" z="0" />
<scale x="1" y="1" z="1" />
</transform>
</gameObject>
<gameObject objectName="meinv" objectAssetURL="meinv">
<transform>
<position x="-1.75" y="1.36" z="11.28" />
<rotation x="0" y="97.43578" z="0" />
<scale x="1.09" y="1.09" z="1.09" />
</transform>
</gameObject>
<gameObject objectName="Envirment" objectAssetURL="Envirment">
<transform>
<position x="0" y="0" z="0" />
<rotation x="0" y="0" z="0" />
<scale x="1" y="1" z="1" />
</transform>
</gameObject>
<gameObject objectName="RigidBodyFPSController" objectAssetURL="RigidBodyFPSController">
<transform>
<position x="4.89" y="1.46" z="0.87" />
<rotation x="0" y="253.2478" z="0" />
<scale x="1" y="1" z="1" />
</transform>
</gameObject>
<gameObject objectName="Sphere" objectAssetURL="Sphere">
<transform>
<position x="-2.63" y="3.28" z="-3.95" />
<rotation x="270" y="0" z="0" />
<scale x="1" y="1" z="1" />
</transform>
</gameObject>
</scene>
</root>
3.将预设打包为AssetBundle,并上传到服务器
打包Assetbundle编辑器脚本:

[code]csharpcode:

using UnityEngine;
using System.Collections;
using UnityEditor; public class Test : Editor
{ [MenuItem("Custom Editor/Create AssetBunldes Main For Android")]
static void CreateAssetBunldesMainForAndroid()
{
//获取在Project视图中选择的所有游戏对象
Object[] SelectedAsset = Selection.GetFiltered(typeof(Object), SelectionMode.DeepAssets); //遍历所有的游戏对象
foreach (Object obj in SelectedAsset)
{
//本地测试:建议最后将Assetbundle放在StreamingAssets文件夹下,如果没有就创建一个,因为移动平台下只能读取这个路径
//StreamingAssets是只读路径,不能写入
//服务器下载:就不需要放在这里,服务器上客户端用www类进行下载。
string targetPath = Application.dataPath + "/StreamingAssets/" + obj.name + "Android" + ".assetbundle";
if (BuildPipeline.BuildAssetBundle(obj, null, targetPath, BuildAssetBundleOptions.CollectDependencies, BuildTarget.Android))
{
Debug.Log(obj.name + "资源打包成功");
}
else
{
Debug.Log(obj.name + "资源打包失败");
}
}
//刷新编辑器
AssetDatabase.Refresh(); }
[MenuItem("Custom Editor/Create AssetBunldes Main For iPhone")]
static void CreateAssetBunldesMainForiPhone()
{
//获取在Project视图中选择的所有游戏对象
Object[] SelectedAsset = Selection.GetFiltered(typeof(Object), SelectionMode.DeepAssets);
//遍历所有的游戏对象
foreach (Object obj in SelectedAsset)
{
string targetPath = Application.dataPath + "/StreamingAssets/" + obj.name + "iPhone" + ".assetbundle";
if (BuildPipeline.BuildAssetBundle(obj, null, targetPath, BuildAssetBundleOptions.CollectDependencies, BuildTarget.iPhone))
{
Debug.Log(obj.name + "资源打包成功");
}
else
{
Debug.Log(obj.name + "资源打包失败");
}
}
//刷新编辑器
AssetDatabase.Refresh(); } [MenuItem("Custom Editor/Create AssetBunldes ALL For Android")]
static void CreateAssetBunldesALLForAndroid()
{ Caching.CleanCache();
string Path = Application.dataPath + "/StreamingAssets/ALLAndroid.assetbundle";
Object[] SelectedAsset = Selection.GetFiltered(typeof(Object), SelectionMode.DeepAssets);
foreach (Object obj in SelectedAsset)
{
Debug.Log("Create AssetBunldes name :" + obj);
} //这里注意第二个参数就行
if (BuildPipeline.BuildAssetBundle(null, SelectedAsset, Path, BuildAssetBundleOptions.CollectDependencies, BuildTarget.Android))
{
AssetDatabase.Refresh();
}
else
{ }
}
[MenuItem("Custom Editor/Create AssetBunldes ALL For iPhone")]
static void CreateAssetBunldesALLForiPhone()
{ Caching.CleanCache(); string Path = Application.dataPath + "/StreamingAssets/ALLiPhone.assetbundle"; Object[] SelectedAsset = Selection.GetFiltered(typeof(Object), SelectionMode.DeepAssets); foreach (Object obj in SelectedAsset)
{
Debug.Log("Create AssetBunldes name :" + obj);
} //这里注意第二个参数就行
if (BuildPipeline.BuildAssetBundle(null, SelectedAsset, Path, BuildAssetBundleOptions.CollectDependencies, BuildTarget.iPhone))
{
AssetDatabase.Refresh();
}
else
{ }
}
[MenuItem("Custom Editor/Create Scene For Android")]
static void CreateSceneALLForAndroid()
{
Caching.CleanCache();
Object[] SelectedAsset = Selection.GetFiltered(typeof(Object), SelectionMode.DeepAssets);
foreach (Object obj in SelectedAsset)
{
string Path = Application.dataPath + "/StreamingAssets/" + obj.name + "Android" + ".unity3d";
string[] levels = { @"Assets/Scenes/ExportedScene/" + obj.name + ".unity" };
BuildPipeline.BuildPlayer(levels, Path, BuildTarget.Android, BuildOptions.BuildAdditionalStreamedScenes);
Debug.Log("Craete Scene" + Path + "Complete!!");
}
AssetDatabase.Refresh();
} [MenuItem("Custom Editor/Create Scene For iPhone")]
static void CreateSceneALLForiPhone()
{
Caching.CleanCache();
Object[] SelectedAsset = Selection.GetFiltered(typeof(Object), SelectionMode.DeepAssets);
foreach (Object obj in SelectedAsset)
{
string Path = Application.dataPath + "/StreamingAssets/" + obj.name + "iPhone" + ".unity3d";
string[] levels = { @"Assets/Scenes/ExportedScene/" + obj.name + ".unity" };
BuildPipeline.BuildPlayer(levels, Path, BuildTarget.iPhone, BuildOptions.BuildAdditionalStreamedScenes);
Debug.Log("Craete Scene" + Path + "Complete!!");
}
AssetDatabase.Refresh();
} }
 
4.将服务器的路径修改到XML配置文件中,并将XML上传到服务器
 
5.在Unity客户端下载XML并解析,下载并实例化对应的资源

[code]csharpcode:

using UnityEngine;
using System.Collections;
using System.IO;
using System.Collections.Generic;
using System.Xml; public class GetXMLDoc : MonoBehaviour
{ string filePath;
public string XMLDocURL = "http://************.xml"; void Awake ()
{
filePath = Application.persistentDataPath + "/XMLDoc1028.xml";
if (File.Exists (filePath)) {
File.Delete (filePath);
}
WWW www = new WWW (XMLDocURL);
StartCoroutine (DownloadXMLDoc (www)); } IEnumerator DownloadXMLDoc (WWW www)
{
yield return www;
if (www.isDone) {
Debug.Log ("WWW is done");
byte[] bts = www.bytes;
int length = bts.Length;
CreateXMLDoc (filePath, bts, length);
} } void CreateXMLDoc (string path, byte[] info, int lenth)
{
Debug.Log ("Start to create XML");
Stream sw;
FileInfo t = new FileInfo (path);
if (!t.Exists) {
sw = t.Create (); } else {
return;
}
sw.Write (info, 0, lenth);
sw.Close ();
sw.Dispose ();
Debug.Log ("XML create sucess");
LoadScene ();
// 下载完毕之后可以读取并进行实例化了
} void LoadScene ()
{
Debug.Log ("开始加载");
if (File.Exists (filePath)) {
XmlDocument xmlDoc = new XmlDocument ();
xmlDoc.Load (filePath);
XmlNodeList nodeList = xmlDoc.SelectSingleNode ("root").ChildNodes;
foreach (XmlElement scene in nodeList) {
//因为我的XML是把所有游戏对象全部导出, 所以这里判断一下只解析需要的场景中的游戏对象
//JSON和它的原理类似
// if (!scene.GetAttribute("name").Equals("Assets/StarTrooper.unity"))
// {
// continue;
// } foreach (XmlElement gameObjects in scene.ChildNodes) {
// 取得资源地址
string assetRUL = gameObjects.GetAttribute ("objectAssetURL");
string assetName = gameObjects.GetAttribute ("objectName");
Vector3 pos = Vector3.zero;
Vector3 rot = Vector3.zero;
Vector3 sca = Vector3.zero;
foreach (XmlElement transform in gameObjects.ChildNodes) {
foreach (XmlElement prs in transform.ChildNodes) {
if (prs.Name == "position") { pos.x = float.Parse(prs.GetAttribute("x"));
pos.y = float.Parse(prs.GetAttribute("y"));
pos.z = float.Parse(prs.GetAttribute("z")); // foreach (XmlElement position in prs.ChildNodes) {
// switch (position.Name) {
// case "x":
// pos.x = float.Parse (position.InnerText);
// break;
// case "y":
// pos.y = float.Parse (position.InnerText);
// break;
// case "z":
// pos.z = float.Parse (position.InnerText);
// break;
// }
// } } else if (prs.Name == "rotation") { rot.x = float.Parse (prs.GetAttribute("x"));
rot.y = float.Parse(prs.GetAttribute("y"));
rot.z = float.Parse(prs.GetAttribute("z")); // foreach (XmlElement rotation in prs.ChildNodes) {
// switch (rotation.Name) {
// case "x":
// rot.x = float.Parse (rotation.InnerText);
// break;
// case "y":
// rot.y = float.Parse (rotation.InnerText);
// break;
// case "z":
// rot.z = float.Parse (rotation.InnerText);
// break;
// }
// }
} else if (prs.Name == "scale") {
sca.x = float.Parse (prs.GetAttribute("x"));
sca.y = float.Parse(prs.GetAttribute("y"));
sca.z = float.Parse(prs.GetAttribute("z")); // foreach (XmlElement scale in prs.ChildNodes) {
// switch (scale.Name) {
// case "x":
// sca.x = float.Parse (scale.InnerText);
// break;
// case "y":
// sca.y = float.Parse (scale.InnerText);
// break;
// case "z":
// sca.z = float.Parse (scale.InnerText);
// break;
// }
// }
}
}
Debug.Log ("准备下载:" + assetRUL);
// 开始下载并实例化对象
Debug.Log (assetName + ":pos=" + pos);
Debug.Log (assetName + ":rot=" + pos);
Debug.Log (assetName + ":sca=" + pos);
StartCoroutine (DownloadAsset (new WWW (assetRUL), pos, rot, sca)); //拿到 旋转 缩放 平移 以后克隆新游戏对象
// GameObject ob = (GameObject)Instantiate(Resources.Load(asset), pos, Quaternion.Euler(rot));
// ob.transform.localScale = sca; }
} } } } IEnumerator DownloadAsset (WWW www, Vector3 pos, Vector3 rot, Vector3 sca)
{ yield return www;
if (www.isDone) {
// yield return Instantiate(www.assetBundle.mainAsset,pos,pos,Quaternion.Euler(rot));
GameObject ob = (GameObject)Instantiate (www.assetBundle.mainAsset, pos, Quaternion.Euler (rot));
ob.transform.localScale = sca;
www.assetBundle.Unload (false);
}
} }

(转)Unity 导出XML配置文件,动态加载场景的更多相关文章

  1. C#遍历XML文件动态加载菜单

    通过遍历XML文件动态加载菜单,顺便利用WebBrowser控件实现一个简单的桌面浏览器 效果如下: 代码如下: XMLFile1.xml <?xml version="1.0&quo ...

  2. Unity3d 动态加载场景物件与缓存池的使用

    聊聊Unity3d动态加载场景物件那些事儿. 众所周知,在策划或美术设计完游戏场景地图后,一个场景中可能会存在成千上万个小的物件,比如石头,木箱子,油桶,栅栏等等等等,这些物件并不是游戏中的道具,仅仅 ...

  3. android sax解析xml 文件 动态加载标题

    要解决一个问题 : 问题描述为 把标题动态的加载到 listView子布局中 我们首先通过 java程序写一个把标题写到xml文件的程序.这个程序会在以后讲解. 现在截图 已经写好的xm文件格式如下 ...

  4. Unity实现精灵资源动态加载

    private Sprite LoadSourceSprite(string relativePath) {         //把资源加载到内存中         UnityEngine.Objec ...

  5. IDEA设置maven修改settings.xml配置文件无法加载仓库

    作为初学者配置maven一般网上搜索.然后你就看到各种配置文件片段,首先配置镜像,然后配置仓库.完事后再IDEA里面配置下maven的路径和配置文件路径. 这些文章属实坑爹,完全没讲一个重要的配置就是 ...

  6. 基于xml 实现动态加载权限功能树列表---EFSFrame企业级开发架构

    在学习EFSFrame框架的过程中,感触最深的就是通过xml来实现前台与后台数据的交互,页面设计灵活,不用管后台如何写的,前台与后台的交互唯一的交互通道都是xml,在我们需要添加页面.添加规定的格式的 ...

  7. java基于xml配置的通用excel单表数据导入组件(二、xml配置文件解析加载)

    1.BN_ImportExcel.java 对应xml主节点属性 package XXXXX.manage.importexcel; import java.io.Serializable; impo ...

  8. Spring IoC容器 XML 配置与加载

    IoC 容器 XML 配置文件 <?xml version="1.0" encoding="UTF-8"?> <beans xmlns=&qu ...

  9. Unity Lightmap动态加载研究

    什么情况下需要Lightmap? 移动平台上目前暂时还不能开实时光影效果,会卡成幻灯片.所以就需要将光影烘焙到贴图上. 什么情况下需要动态加载Lightmap? 1.当项目抛弃了Unity的多场景模式 ...

随机推荐

  1. verynginx +nginx_upstream_check_module模块,负载均衡检查模块。

    yum -y install git yum -y install patch yum -y install pcre-devel yum install -y zlib-devel   mkdir ...

  2. 10个常用的ps命令总结,参数

    Linux系统中10个常用的ps命令总结 PS 命令是什么 查看它的man手册可以看到,ps命令能够给出当前系统中进程的快照.它能捕获系统在某一事件的进程状态.如果你想不断更新查看的这个状态,可以使用 ...

  3. 渗透笔记0x00

    今天朋友在入侵织梦的网站系统,通过最新的Exp获得了织梦后台的账号和密码 账号:admin 密码:abc123456 但是,找来找去找不到后台,就让我提权试试. 我看了下:http://pxc.nci ...

  4. C# SendMail 发送邮件

    最近因为用的发送邮件的地方,就查询了资料,总结以下几个方法 1.利用新浪邮箱发送 2.利用公司邮箱发送 3.利用CDO发送,这种方式要引用Interop.ADODB.dll(http://www.no ...

  5. java源码阅读StringBuilder

    1类签名与注释 public final class StringBuilder extends AbstractStringBuilder implements java.io.Serializab ...

  6. JSON,字符串,MAP转换

    package com.tree.autotest.testcase.IAuditBillDetailService; import com.alibaba.fastjson.JSON;import ...

  7. 神奇的canvas——巧用 canvas 为图片添加水印

    代码地址如下:http://www.demodashi.com/demo/11637.html 很久之前写过一篇关于 canvas 的文章,是通过 canvas 来实现一个绚丽的动画效果,不管看过没看 ...

  8. iOS开发-Object-C学习之结构体使用

    前言:定义结构体并不是定义一个变量,而是定义了种数据类型. 结构体作用: 结构体和其他类型基础数据类型一样,例如int类型,char类型 只不过结构体可以做成你想要的数据类型.以方便日后的使用. 在实 ...

  9. 《windows核心编程》 18章 堆

    堆的优缺点: 优点:让我们专心解决手头问题,不必理会分配粒度和页边界这类事情. 缺点:分配和释放内存块的速度比其他方式慢,而且也无法对物理存储器的调拨和撤销进行直接控制. 什么是堆: 堆就是一块预订的 ...

  10. ibatis 动态查询

    http://www.iteye.com/topic/393042最近做了很多动态的查询,尤其是排序,以及一些状态字段,所以就做了一个总的动态查询,以不变应万变,呵呵 ibatis 里面的sql代码: ...