IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 游戏开发 -> Unity_建造系统及保存加载 -> 正文阅读

[游戏开发]Unity_建造系统及保存加载

主要是尝试类似RTS的建造系统,?以及通过序列化/反序列化保存和加载游戏数据,?测试版本Unity2020.3


建造物体

移动

生成使用透明材质的的模型到场景,?通过射线检测获取鼠标位置并使模型位置跟随移动

private Camera mCamera;//场景相机
public GameObject PrefabModel;//预制体模型
private GameObject createModel;//用于创建的场景模型

void Start()
{
	//获取场景相机
	GameObject gameObject=GameObject.Find("Main Camera");
	mCamera=gameObject.GetComponent<Camera>();

	createModel = Instantiate(PrefabModel);//生成模型到场景
}

void Update()
{
	/*从摄像机向鼠标方向发射射线,获取鼠标位置*/
	Ray ray = mCamera.ScreenPointToRay(Input.mousePosition);
	RaycastHit hit;
	LayerMask mask = 1 << (LayerMask.NameToLayer("Floor"));//遮罩图层,使射线只检测Floor图层

	if (Physics.Raycast(ray, out hit,Mathf.Infinity,mask.value))
	{
		Debug.DrawLine(ray.origin, hit.point,Color.green);//在编辑器视图中绘制射线
		createModel.transform.position = hit.point;//使模型位置跟随鼠标坐标
	}
}

旋转

用鼠标滚轮控制模型旋转角度

void Update()
{
	if (Input.GetAxis("Mouse ScrollWheel")!=0)//鼠标滚轮
	{
		//旋转模型角度
		newcreateModel.transform.Rotate(0,Input.GetAxis("Mouse ScrollWheel") *Time.deltaTime*20000,0,Space.World);//Space.World为使用全局坐标系
	}
}

设置材质

固定模型到场景中后把模型材质从透明恢复为正常材质

public Material AlbedoMat;//正常材质

...

createModel.GetComponent<MeshRenderer>().sharedMaterial = AlbedoMat;//设置模型为正常材质

碰撞/重叠检测

给预制体模型添加Collider碰撞体和Rigidbody刚体组件并使用触发器来检测是否与其他已经固定的模型有重叠导致穿模,?如果有重叠则不能固定到场景并把模型材质变红提示

[参考的]Unity碰撞和触发

//触发开始时执行一次
public void OnTriggerEnter(Collider collider){
	Debug.Log("与其他模型有重叠,不可建造");
}

//触发过程中一直重复执行
public void OnTriggerStay(Collider collider){
	Debug.Log("与其他模型有重叠,不可建造");
}

//触发结束时执行一次
public void OnTriggerExit(Collider collider){
	Debug.Log("无重叠,可以建造");
}

数据保存/加载

查资料时都管这个叫序列化和反序列化, 听起来好高大上的样子,?Unity自带了JsonUtility用于处理 json数据(JsonUtility文档)

测试用到的主要是获取场景上所有已固定的建筑模型的类型、位置和角度并保存为json文件,?在重新加载场景后可以加载保存的数据并根据数据恢复保存时的状态

保存数据

/*用于保存数据的类*/
[Serializable]//应用可序列化属性
public class BuildStateSave
{
	public List<string> modelType = new List<string>();//模型类型
	public List<double> posX = new List<double>();//X位置
	public List<double> posY = new List<double>();//Y位置
	public List<double> posZ = new List<double>();//Z位置
	public List<double> rotX = new List<double>();//X方向
	public List<double> rotY = new List<double>();//Y方向
	public List<double> rotZ = new List<double>();//Z方向
	
	/*把数据转换为json字符串*/
	public string SaveToString()
	{
		return JsonUtility.ToJson(this,true);//[prettyPrint]为true时返回的字符串保持可读格式,为false时大小最小
	}
}

/*保存按钮*/
public void SaveSence()
{
	GameObject[] buildCubes =GameObject.FindGameObjectsWithTag("Build");//获取所有已建造对象
	
	BuildStateSave buildState = new BuildStateSave();
	
	for (int i = 0; i < buildCubes.Length; i++)
	{
		buildState.modelType.Add(buildCubes[i].name);
		buildState.posX.Add(buildCubes[i].transform.position.x);
		buildState.posY.Add(buildCubes[i].transform.position.y);
		buildState.posZ.Add(buildCubes[i].transform.position.z);
		buildState.rotX.Add(buildCubes[i].transform.rotation.eulerAngles.x);
		buildState.rotY.Add(buildCubes[i].transform.rotation.eulerAngles.y);
		buildState.rotZ.Add(buildCubes[i].transform.rotation.eulerAngles.z);
	}
	
	string filePath = Application.streamingAssetsPath + "/BuildSave.json";

	if (File.Exists(filePath))//判断文件是否存在
	{
		File.Delete(filePath);//删除这个文件
	}

	//找到当前路径
	FileInfo file = new FileInfo(filePath);

	//判断有没有文件,有则打开文件,没有则创建后打开
	StreamWriter sw = file.CreateText();
	
	//数据转换为字符串并写入文件
	sw.WriteLine(buildState.SaveToString());
	
	sw.Close();//关闭文件
	sw.Dispose();
}

执行保存后路径下的json文件中可以看到保存的信息数据

加载数据

/*用于加载数据的类*/
[System.Serializable]//应用可序列化属性
public class BuildStateLoad
{
	public List<string> modelType = new List<string>();//模型类型
	public List<double> posX = new List<double>();//X位置
	public List<double> posY = new List<double>();//Y位置
	public List<double> posZ = new List<double>();//Z位置
	public List<double> rotX = new List<double>();//X方向
	public List<double> rotY = new List<double>();//Y方向
	public List<double> rotZ = new List<double>();//Z方向

	/*把json字符串转换为变量数据*/
	public BuildStateLoad CreateFromJSON(string jsonString)
	{
		return JsonUtility.FromJson<BuildStateLoad>(jsonString);
	}
}

/*加载按钮*/
public void LoadSence()
{
	/*从json文件读取数据*/
	FileStream fs = new FileStream(Application.streamingAssetsPath + "/BuildSave.json", FileMode.Open, FileAccess.Read, FileShare.None);
	StreamReader sr = new StreamReader(fs, System.Text.Encoding.Default);
	if (null == sr) return;
	string str = sr.ReadToEnd();
	sr.Close();

	/*字符串数据反序列化到变量中*/
	BuildStateLoad buildInfo = new BuildStateLoad();
	buildInfo=buildInfo.CreateFromJSON(str);

	for (int i = 0; i < buildInfo.modelType.Count; i++)
	{
		/*生成模型到场景*/
		switch (buildInfo.modelType[i])
		{
			case "Model1":
				createModel = Instantiate(PrefabModel1);
				break;
			case "Model2":
				createModel = Instantiate(PrefabModel2);
				break;
		}
		
		/*重命名*/
		createModel.name = buildInfo.modelType[i];
		
		/*位置*/
		createModel.transform.position = new Vector3((float)buildInfo.posX[i],(float)buildInfo.posY[i],(float)buildInfo.posZ[i]);
		
		/*旋转*/
		createModel.transform.rotation = Quaternion.Euler(new Vector3((float)buildInfo.rotX[i],(float)buildInfo.rotY[i],(float)buildInfo.rotZ[i]));
		
		/*固定到场景*/
		createModel.GetComponent<BoxCollider>().isTrigger=false;//关闭触发器恢复碰撞
		createModel.GetComponent<Rigidbody>().useGravity=true;//恢复刚体组件的重力
	}
}

执行加载后获取保存的数据


完整代码

主要逻辑脚本, 挂载在一个场景空对象上

using System;
using System.Collections;
using System.Collections.Generic;
using System.IO;
using UnityEngine;

public class BuildTest : MonoBehaviour
{
    private Camera mCamera;//场景相机

    public GameObject boxPrefabModel;//预制体模型
    public GameObject flak88PrefabModel;//预制体模型
    
    private GameObject newcreateModel;//用于创建的场景模型
    private GameObject loadcreateModel;//用于加载的场景模型
    
    public Material transparentMat;//透明材质
    public Material unInitMat;//阻挡材质
    
    public Material boxAlbedoMat;//Box正常材质
    public Material flak88AlbedoMat;//88炮正常材质

    private bool iSBuilding = false;//是否为建造模式
    public bool unInitPrefab = false;//是否为建造模式

    private string selectBuildType;//记录当前选择建造的物体类型
    
    void Start()
    {
        /*获取相机*/
        GameObject gameObject=GameObject.Find("Main Camera");
        mCamera=gameObject.GetComponent<Camera>();
    }
    
    void Update()
    {
        if (iSBuilding)
        {
            /*从摄像机向鼠标方向发射射线,获取鼠标位置*/
            Ray ray = mCamera.ScreenPointToRay(Input.mousePosition);
            RaycastHit hit;
            LayerMask mask = 1 << (LayerMask.NameToLayer("Floor"));
            
            /*Raycast:
             [1]射线方向
             [2]碰撞到的对象信息
             [3]射线长度,Mathf.Infinity为无限远
             [4]遮罩图层,不检测其他图层的物体
             */
            if (Physics.Raycast(ray, out hit,Mathf.Infinity,mask.value))
            {
                Debug.DrawLine(ray.origin, hit.point,Color.green);//在编辑器视图中绘制射线
                
                /*模型位置跟随鼠标坐标*/
                newcreateModel.transform.position = hit.point;

                if (unInitPrefab)//若位置与已生成物体重叠,修改显示材质
                {
                    /*设置模型材质为不可生成状态*/
                    MeshRenderer[] Mat = newcreateModel.GetComponentsInChildren<MeshRenderer>();//获取当前对象及所有子对象上的材质组件并更换材质
                    for (int i = 0; i < Mat.Length; i++)
                    {
                        Mat[i].sharedMaterial = unInitMat;
                    }
                }
                else
                {
                    /*设置模型材质为透明*/
                    MeshRenderer[] Mat = newcreateModel.GetComponentsInChildren<MeshRenderer>();//获取当前对象及所有子对象上的材质组件并更换材质
                    for (int i = 0; i < Mat.Length; i++)
                    {
                        Mat[i].sharedMaterial = transparentMat;
                    }
                    
                    if (Input.GetMouseButtonDown(0))//点击鼠标左键
                    {
                        /*固定对象到场景中*/
                        newcreateModel.GetComponent<BoxCollider>().isTrigger=false;//关闭触发器恢复碰撞
                        newcreateModel.GetComponent<Rigidbody>().useGravity=true;//恢复刚体组件的重力
                        //newcreateModel.layer = 20;//设置图层用于接收爆炸力影响

                        switch (selectBuildType)
                        {
                            case "Box":
                                newcreateModel.GetComponent<MeshRenderer>().sharedMaterial = boxAlbedoMat;//设置旧模型为正常材质
                                newcreateModel = Instantiate(boxPrefabModel);//生成新模型到场景
                                break;
                            case "Flak88":
                                /*设置旧模型为正常材质*/
                                for (int i = 0; i < Mat.Length; i++)
                                {
                                    Mat[i].sharedMaterial = flak88AlbedoMat;
                                }
                                
                                newcreateModel = Instantiate(flak88PrefabModel);//生成新模型到场景
                                break;
                        }
                        
                        newcreateModel.name = selectBuildType;//重命名对象
                    }
                }
                
                if (Input.GetAxis("Mouse ScrollWheel")!=0)//滑动鼠标滚轮
                {
                    //旋转模型角度
                    newcreateModel.transform.Rotate(0,Input.GetAxis("Mouse ScrollWheel") *Time.deltaTime*20000,0,Space.World);//加上Space.World为基于全局坐标系变换
                }
            }
        }

        //按下鼠标右键退出建造模式
        if (Input.GetMouseButtonDown(1))
        {
            iSBuilding = false;
            SetCursor(true);
            Destroy(newcreateModel);
            
            /*退出建造模式时如果是重叠状态会导致再次进入建造模式时也是重叠状态,这里在退出时强制取消重叠状态*/
            if (unInitPrefab)
            {
                unInitPrefab = false;
            }
        }
    }
    
    /// <summary>
    /// 设置鼠标状态
    /// </summary>
    /// <param name="">true显示,false隐藏</param>
    public void SetCursor(bool _cursorState)
    {
        //隐藏鼠标
        Cursor.visible = _cursorState;
    }

    /*进入Box建造模式按钮*/
    public void StartBoxBuild()
    {
        selectBuildType = "Box";
        iSBuilding = true;
        SetCursor(false);
        
        /*生成模型到场景*/
        newcreateModel = Instantiate(boxPrefabModel);
        
        /*设置模型材质为透明*/
        newcreateModel.GetComponent<MeshRenderer>().sharedMaterial = transparentMat;
        
        /*重命名对象便于区分*/
        newcreateModel.name = selectBuildType;
    }
    
    /*进入Flak88建造模式按钮*/
    public void StartFlak88Build()
    {
        selectBuildType = "Flak88";
        iSBuilding = true;
        SetCursor(false);
        
        /*生成模型到场景*/
        newcreateModel = Instantiate(flak88PrefabModel);
        
        /*设置模型材质为透明*/
        MeshRenderer[] Mat = newcreateModel.GetComponentsInChildren<MeshRenderer>();

        for (int i = 0; i < Mat.Length; i++)
        {
            Mat[i].sharedMaterial = transparentMat;
        }
        
        /*重命名*/
        newcreateModel.name = selectBuildType;
    }
    
    /*保存按钮*/
    public void ClickSaveGame()
    {
        GameObject[] buildCubes =GameObject.FindGameObjectsWithTag("Build");//寻找对象
        
        BuildStateSave buildState = new BuildStateSave();
        
        for (int i = 0; i < buildCubes.Length; i++)
        {
            buildState.modelType.Add(buildCubes[i].name);
            buildState.posX.Add(buildCubes[i].transform.position.x);
            buildState.posY.Add(buildCubes[i].transform.position.y);
            buildState.posZ.Add(buildCubes[i].transform.position.z);
            buildState.rotX.Add(buildCubes[i].transform.rotation.eulerAngles.x);
            buildState.rotY.Add(buildCubes[i].transform.rotation.eulerAngles.y);
            buildState.rotZ.Add(buildCubes[i].transform.rotation.eulerAngles.z);
        }
        
        string filePath = Application.streamingAssetsPath + "/BuildSave.json";

        if (File.Exists(filePath))//判断文件是否存在
        {
            File.Delete(filePath);//删除这个文件
        }

        //找到当前路径
        FileInfo file = new FileInfo(filePath);

        //判断有没有文件,有则打开文件,没有则创建后打开
        StreamWriter sw = file.CreateText();
        
        //数据转换为字符串并写入文件
        sw.WriteLine(buildState.SaveToString());
        
        sw.Close();//关闭文件
        sw.Dispose();
    }

    /*重新加载场景按钮*/
    public void ReLoadGame()
    {
        Application.LoadLevel(Application.loadedLevel);//重置场景
    }
    
    /*加载按钮*/
    public void ClickLoadGame()
    {
        /*从json文件读取数据*/
        FileStream fs = new FileStream(Application.streamingAssetsPath + "/BuildSave.json", FileMode.Open, FileAccess.Read, FileShare.None);
        StreamReader sr = new StreamReader(fs, System.Text.Encoding.Default);
        if (null == sr) return;
        string str = sr.ReadToEnd();
        sr.Close();

        /*字符串数据反序列化到变量中*/
        BuildStateLoad buildInfo = new BuildStateLoad();
        buildInfo=buildInfo.CreateFromJSON(str);

        for (int i = 0; i < buildInfo.modelType.Count; i++)
        {
            /*生成模型到场景*/
            switch (buildInfo.modelType[i])
            {
                case "Box":
                    loadcreateModel = Instantiate(boxPrefabModel);
                    break;
                case "Flak88":
                    loadcreateModel = Instantiate(flak88PrefabModel);
                    break;
            }
            
            /*重命名*/
            loadcreateModel.name = buildInfo.modelType[i];
            
            /*位置*/
            loadcreateModel.transform.position = new Vector3((float)buildInfo.posX[i],(float)buildInfo.posY[i],(float)buildInfo.posZ[i]);
            
            /*旋转*/
            loadcreateModel.transform.rotation = Quaternion.Euler(new Vector3((float)buildInfo.rotX[i],(float)buildInfo.rotY[i],(float)buildInfo.rotZ[i]));
            
            /*固定到场景*/
            loadcreateModel.GetComponent<BoxCollider>().isTrigger=false;//关闭触发器恢复碰撞
            loadcreateModel.GetComponent<Rigidbody>().useGravity=true;//恢复刚体组件的重力
        }
    }

    /*用于保存数据的类*/
    [Serializable]//应用可序列化属性,不受支持的字段以及私有字段、静态字段和应用了NonSerialized属性的字段会被忽略
    public class BuildStateSave
    {
        /*
         泛型集合List<T>
         大小可按需动态增加,不会强行对值类型进行装箱和拆箱,或对引用类型进行向下强制类型转换,性能提高
         */
        public List<string> modelType = new List<string>();//模型类型
        public List<double> posX = new List<double>();//X位置
        public List<double> posY = new List<double>();//Y位置
        public List<double> posZ = new List<double>();//Z位置
        public List<double> rotX = new List<double>();//X方向
        public List<double> rotY = new List<double>();//Y方向
        public List<double> rotZ = new List<double>();//Z方向
        
        /*把列表数据转换为json字符串*/
        public string SaveToString()
        {
            return JsonUtility.ToJson(this,true);//[prettyPrint]为true时保持可读格式,为false时最小大小
        }
    }
    
    /*用于加载数据的类*/
    [System.Serializable]//应用可序列化属性,不受支持的字段以及私有字段、静态字段和应用了NonSerialized属性的字段会被忽略
    public class BuildStateLoad
    {
        public List<string> modelType = new List<string>();//模型类型
        public List<double> posX = new List<double>();//X位置
        public List<double> posY = new List<double>();//Y位置
        public List<double> posZ = new List<double>();//Z位置
        public List<double> rotX = new List<double>();//X方向
        public List<double> rotY = new List<double>();//Y方向
        public List<double> rotZ = new List<double>();//Z方向

        /*把json字符串转换为变量数据*/
        public BuildStateLoad CreateFromJSON(string jsonString)
        {
            return JsonUtility.FromJson<BuildStateLoad>(jsonString);
        }
    }
}

用于碰撞触发检测的脚本, 挂载在预制体模型上

using System;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class 碰撞检测 : MonoBehaviour
{
    private BuildTest testScript;
    
    private void Start()
    {
        /*获取脚本对象*/
        GameObject gameObject=GameObject.Find("脚本挂载");
        testScript = gameObject.GetComponent<BuildTest>();
    }

    //触发开始时执行一次
    public void OnTriggerEnter(Collider collider){
        testScript.unInitPrefab = true;
    }
 
    //触发过程中一直重复执行
    public void OnTriggerStay(Collider collider){
        testScript.unInitPrefab = true;
    }
 
    //触发结束时执行一次
    public void OnTriggerExit(Collider collider){
        testScript.unInitPrefab = false;
    }
}
  游戏开发 最新文章
6、英飞凌-AURIX-TC3XX: PWM实验之使用 GT
泛型自动装箱
CubeMax添加Rtthread操作系统 组件STM32F10
python多线程编程:如何优雅地关闭线程
数据类型隐式转换导致的阻塞
WebAPi实现多文件上传,并附带参数
from origin ‘null‘ has been blocked by
UE4 蓝图调用C++函数(附带项目工程)
Unity学习笔记(一)结构体的简单理解与应用
【Memory As a Programming Concept in C a
上一篇文章      下一篇文章      查看所有文章
加:2022-10-08 21:12:26  更:2022-10-08 21:12:38 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/17 5:46:04-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码