Unity实现简单换装系统

作者:langresser 时间:2022-06-15 16:30:07 

关于Unity的换装,网上有几篇文章,我之前也简单的描述过实现。不过那个时候只是粗略的试验了下。今天好好梳理了下代码。

先上代码(自己的游戏项目,不是公司的,所以放心的贴上项目代码了,部分引用到其他的功能文件,但是核心代码无影响,这里主要看一下细节和思路)


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

public enum AvatarPart
{
   helmet,
   chest,
   shoulders,
   gloves,
   boots,
}

// 人物换装
public class ActorAvatar : MonoBehaviour
{
   // 换装的部件信息
   public class AvatarInfo
   {
       public string partName;
       public GameObject defaultPart;
       public GameObject avatarPart;
   }

protected int _bodyModelId;
   protected GameObject _body;         // 基础模型动画
   protected Dictionary<string, AvatarInfo> _avatarInfo = new Dictionary<string, AvatarInfo>();        // 换装信息

private List<int> _avatarLoadQueue = new List<int>();

void Start()
   {
   }

void Update()
   {
   }

// 创建模型
   public void LoadModel(int modelId)
   {
       _bodyModelId = modelId;
       ResourceMgr.Instance.LoadModel(modelId, (GameObject obj) =>
       {
           _body = obj;

// 换装请求
           if (_avatarLoadQueue.Count > 0) {
               foreach (var avatar in _avatarLoadQueue) {
                   LoadAvatar(avatar);
               }
               _avatarLoadQueue.Clear();
           }
       }, true);
   }

// 给人物换装
   public void LoadAvatar(int avatarId)
   {
       // 如果还没有加载完基础模型,则等待
       if (_body == null) {
           _avatarLoadQueue.Add(avatarId);
           return;
       }

AvatarData adata = DataMgr.Instance.GetAvatarData(avatarId);
       ResourceMgr.Instance.LoadModel(adata.model, (GameObject obj) => {
           ChangeAvatar(obj, adata.addpart);
       });
   }

// 替换部件
   public void ChangeAvatar(GameObject avatarModel, string partName)
   {
       // 先卸载当前部件
       AvatarInfo currentInfo;
       if (_avatarInfo.TryGetValue(partName, out currentInfo)) {
           if (currentInfo.avatarPart != null) {
               Destroy(currentInfo.avatarPart);
               currentInfo.avatarPart = null;
           }

if (currentInfo.defaultPart != null) {
               currentInfo.defaultPart.SetActive(true);
           }
       }

// avatarModel是一个resource,并没有实例化
       if (avatarModel == null) {
           return;
       }

// 需要替换的部件
       Transform avatarPart = GetPart(avatarModel.transform, partName);
       if (avatarPart == null) {
           Debug.LogError(string.Format("Avatar Part Not Found: ", partName));
           return;
       }

// 将原始部件隐藏
       Transform bodyPart = GetPart(_body.transform, partName);
       if (bodyPart != null) {
           bodyPart.gameObject.SetActive(false);
       }

// 设置到body上的新物件
       GameObject newPart = new GameObject(partName);
       newPart.transform.parent = _body.transform;
       SkinnedMeshRenderer newPartRender = newPart.AddComponent<SkinnedMeshRenderer>();
       SkinnedMeshRenderer avatarRender = avatarPart.GetComponent<SkinnedMeshRenderer>();

// 刷新骨骼模型数据
       SetBones(newPart, avatarPart.gameObject, _body);
       newPartRender.sharedMesh = avatarRender.sharedMesh;
       newPartRender.sharedMaterials = avatarRender.sharedMaterials;

// 记录换装信息
       AvatarInfo info = new AvatarInfo();
       info.partName = partName;
       if (bodyPart != null) {
           info.defaultPart = bodyPart.gameObject;
       } else {
           info.defaultPart = null;
       }

info.avatarPart = newPart;
       _avatarInfo[partName] = info;
   }

// 递归遍历子物体
   public static Transform GetPart(Transform t, string searchName)
   {
       foreach (Transform c in t) {
           string partName = c.name.ToLower();

if (partName.IndexOf(searchName) != -1) {
               return c;
           } else {
               Transform r = GetPart(c, searchName);
               if (r != null) {
                   return r;
               }
           }
       }
       return null;
   }

public static Transform FindChild(Transform t, string searchName)
   {
       foreach (Transform c in t) {
           string partName = c.name;
           if (partName == searchName) {
               return c;
           } else {
               Transform r = FindChild(c, searchName);
               if (r != null) {
                   return r;
               }
           }
       }
       return null;
   }

// 刷新骨骼数据   将root物体的bodyPart骨骼更新为avatarPart
   public static void SetBones(GameObject goBodyPart, GameObject goAvatarPart, GameObject root)
   {
       var bodyRender = goBodyPart.GetComponent<SkinnedMeshRenderer>();
       var avatarRender = goAvatarPart.GetComponent<SkinnedMeshRenderer>();
       var myBones = new Transform[avatarRender.bones.Length];
       for (var i = 0; i < avatarRender.bones.Length; i++) {
           myBones[i] = FindChild(root.transform, avatarRender.bones[i].name);
       }
       bodyRender.bones = myBones;
   }

}

1、Unity换装有三种需求:

添加武器的挂载式换装,这个只要创建对应的模型,并且设置好transform.parent就可以了。

替换纹理,这个取到对应的material,然后设置texture就可以了。

模型部件的替换,这个是此处处理的,也是相对最复杂的换装。

2、最核心的部分是ChangeAvatar,它完成了模型换装的功能。模型部件的替换其实就是替换SkinnedMeshRender中的sharedMesh和sharedMaterials。

(这里稍微插一下sharedMaterials   sharedMaterial  Materials  Material这几个变量的区别。sharedMaterials是共享和引用的关系,只要修改这个,所有使用到这个material的模型都会受到影响。如果是在编辑器模式下,它还会修改实际material文件的属性。Materials是sharedMaterials的一份拷贝,只有当前模型使用。materia是materials数组中的第一个对象,这个仅仅是为了方便书写而存在的。)

仅仅替换了sharedMesh还不够,模型会变成一坨麻花。 还应该修改SkinnedMeshRender中的bones属性,它记录了模型的骨骼信息(其实就是一大堆Transform)。  SetBones函数完成了骨骼替换的操作。它查找avatar部件中的所有骨骼名称,然后查找当前模型中的对应骨骼名字,并存储起来。这个数组就是新部件的骨骼信息。

3、一个逻辑上的处理细节。保留了原始模型的对应部件,并没有销毁这个部件,仅仅是隐藏起来。这样卸载装备的时候,只需要删掉装备部件,然后把默认部件设为可见就可以了。

4、可以考虑使用Unity的CombineInstance把模型合并,这样的好处是可以提高运行性能。但是只有材质共用一个的时候才能真正起到优化效果。有个MeshBaker的插件很酷。如果要进行千人战,就必须考虑这方面的优化。

Unity实现简单换装系统

来源:https://blog.csdn.net/langresser_king/article/details/44179901

标签:unity,换装系统
0
投稿

猜你喜欢

  • android TextView加下划线的方法

    2023-09-11 01:12:36
  • Java通过PropertyDescriptor反射调用set和get方法

    2023-10-11 19:34:17
  • Spring Boot @Conditional注解用法示例介绍

    2023-04-18 22:51:51
  • 使用Java将字符串在ISO-8859-1和UTF-8之间相互转换

    2022-09-15 15:05:16
  • java读取properties文件的方法

    2021-12-25 11:30:36
  • C#泛型集合类System.Collections.Generic

    2023-02-24 19:21:18
  • Android Studio使用小技巧:提取方法代码片段

    2023-11-14 19:35:20
  • Windows系统中Java调用cmd命令及执行exe程序的方法

    2021-11-27 23:00:02
  • Android实现锁屏荧光效果

    2023-09-02 03:12:03
  • C#验证身份证的函数

    2022-06-16 04:49:37
  • java学习DongTai被动型IAST工具部署过程

    2023-06-21 09:43:03
  • java String的深入理解

    2022-10-30 23:50:37
  • 详解docker镜像centos7配置Java运行环境

    2022-03-14 04:44:37
  • java实现科研信息管理系统

    2022-05-13 02:49:41
  • Java字符串拼接详解

    2022-10-17 06:22:51
  • java统计字符串中指定元素出现次数方法

    2022-11-02 16:05:40
  • Java中接口和抽象类的区别详解

    2022-09-28 15:21:19
  • Kotlin协程到底是如何切换线程的

    2022-03-03 13:52:39
  • 详解如何使用maven生成可以执行的jar

    2023-08-17 00:18:43
  • Spring component-scan XML配置与@ComponentScan注解配置

    2023-07-04 03:37:12
  • asp之家 软件编程 m.aspxhome.com