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的插件很酷。如果要进行千人战,就必须考虑这方面的优化。
来源:https://blog.csdn.net/langresser_king/article/details/44179901
猜你喜欢
- 废话不多说,直接给大家贴代码了,具体代码如下所示:package com.luo.wctweb.util; import java.awt
- 在进行详解之前,我想先声明一下,本次我们进行讲解说明的是 Kafka 消息存储的信息文件内容,不是所谓的 Kafka 服务器运行产生的日志文
- 最近做了一个功能,里面涉及到了渐变圆形的需求。就是一个颜色可以渐变的圆环,最后实现的效果如下图:左图是带渐变效果,右图是不带渐变效果。原理还
- Spring Cloud feign GET请求无法用实体传参代码如下:@FeignClient(name = "eureka-c
- 昨天遇到了点问题解决浪费了一些时间(导致更新内容较少)回顾下问题项目出现Unable to import maven project: Se
- 本文实例为大家分享了Java打印指定年月日历的具体代码,供大家参考,具体内容如下日历如下:程序如下://打印指定年月的日历public cl
- Timer 详解Timer 和 TimerTask 用于在后台线程中调度任务的 java.
- Java for循环标签跳转到指定位置大家是否见过这种for循环,在for循环前加了个标记的:outerLoop:for (; ; ) {
- 1. 安装JDK解释: JDK是Java编写环境--开发环境注: 安装路径不可出现中文及标点符号。比如:D:\Java\jdk81.1 下载
- c#下压缩解压,主要是用第三方类库进行封装的。ICSharpCode.SharpZipLib.dll类库,链接地址为你官方下载链接。压缩主要
- package com.famous.dark.util;import java.io.File;import java.io.FileFi
- 实现字符串库函数功能有些时候我们可能会被限制无法使用库函数,这个时候我们需要编写自己的库函数。但了解了字符串库函数的功能之后,想要实现并不困
- 引言在多线程并发编程中synchronized和Volatile都扮演着重要的角色,Volatile是轻量级的synchronized,它在
- 在编程过程中一定要注意代码命名的规范性,否则在使用和维护过程中将造成很大的麻烦,这也是一种良好的编码习惯。看下面代码,除了userPass命
- C# WPF ListView控件的实例详解C#的WPF作为现在微软主流的桌面程序开发平台,相比过去的MFC时代,有了非常多的不同。本人刚从
- 一、Mybatis一对多分解式查询分解式查询就是将一条Sql语句拆分成多条在MyBatis多表查询中,使用连接查询时一个Sql语句就可以查询
- C# 获取某个时间的0点0分和23点59分59秒,具体代码如下所示:C#获取当月第一天和最后一天当月第一天0时0分0秒:DateTime.N
- 1.对包名的判断,异常则说明不存在:try {PackageManager pm = getPackageManager();pm.getP
- Android 控制WIFI相关操作WIFI的全称是Wireless Fidelity,又称802.11b标准,是一种高速的无线通信协议,传
- 本文实例讲述了C#快速排序算法。分享给大家供大家参考。具体实现方法如下:public static int[] QuickSort(int[