在游戏项目中换装属于一个比较常见的场景,对于有经验的开发者来说肯定会容易,当然还有一些人还不会,为了帮助这部分开发者,下面就给大家介绍下在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);
- }
- }
-
-
- 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);
- }
-
-
- GameObject newPart = new GameObject(partName);
- newPart.transform.parent = _body.transform;
- SkinnedMeshRenderer newPartRender = newPart.AddComponent();
- SkinnedMeshRenderer avatarRender = avatarPart.GetComponent();
-
-
- 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;
- }
-
-
- public static void SetBones(GameObject goBodyPart, GameObject goAvatarPart, GameObject root)
- {
- var bodyRender = goBodyPart.GetComponent();
- var avatarRender = goAvatarPart.GetComponent();
- 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的插件很酷。如果要进行千人战,就必须考虑这方面的优化。

