当前位置: 首页 > news >正文

Unity Metaverse(六)、关于Avatar换装系统的示例工程

文章目录

  • 🎈 简介
  • 🎈 配置表
  • 🎈 关于胡须


🎈 简介

鉴于之前发了一篇关于Avatar换装系统的解决方案的内容后,有朋友反馈对此比较感兴趣,希望能提供源码,因此我专门整理了一个示例项目,已经放在Github上开源,地址:Unity Avatar换装系统示例工程。

衣服

该工程在之前的基础上增加了不少美术资源,当然它们依然都是来源于Ready Player Me,目前已经有衣服15套,然后增加了发型、眼镜、胡须的编辑,在RPM上各扒了几个资源,感兴趣的话大家可以自行拓展。

发型、眼镜、胡须

🎈 配置表

配置表的创建菜单如下:

创建配置表
其中Outlook Config,即衣服的配置表涉及的内容最多,依次手动拖拽赋值较为繁琐,因此使用编辑器DragAndDrop类为其增加了快速拖拽赋值的功能:

快速拖拽赋值
实现并不复杂,因为之前我们使用Avatar Clothes Collector来扒衣服资源的时候已经为资源命好名了,因此只需要根据名称去匹配指定的资源即可,代码如下:

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

#if UNITY_EDITOR
using UnityEditor;
#endif

namespace Metaverse
{
    [Serializable]
    public class AvatarOutlookData
    {
        public Sprite thumb;

        public Mesh headMesh;
        public Material headMaterial;

        public Mesh bodyMesh;
        public Material bodyMaterial;

        public Mesh topMesh;
        public Material topMaterial;

        public Mesh bottomMesh;
        public Material bottomMaterial;

        public Mesh footwearMesh;
        public Material footwearMaterial;
    }

    [CreateAssetMenu]
    public class AvatarOutlookDataConfig : ScriptableObject
    {
        public List<AvatarOutlookData> data = new List<AvatarOutlookData>(0);
    }

#if UNITY_EDITOR
    [CustomEditor(typeof(AvatarOutlookDataConfig))]
    public class AvatarOutlookDataConfigEditor : Editor
    {
        public override void OnInspectorGUI()
        {
            base.OnInspectorGUI();

            OnDragRectGUI();
        }

        private void OnDragRectGUI()
        {
            GUILayout.Space(10f);
            GUILayout.BeginHorizontal();
            GUILayout.Label(GUIContent.none, GUILayout.ExpandWidth(true));
            Rect lastRect = GUILayoutUtility.GetLastRect();
            var dropRect = new Rect(lastRect.x + 2f, lastRect.y - 2f, lastRect.width - 20f, 20f);
            bool containsMouse = dropRect.Contains(Event.current.mousePosition);
            if (containsMouse)
            {
                switch (Event.current.type)
                {
                    case EventType.DragUpdated:
                        bool flag = DragAndDrop.objectReferences.OfType<Mesh>().Any() 
                            || DragAndDrop.objectReferences.OfType<Material>().Any()
                            || DragAndDrop.objectReferences.OfType<Sprite>().Any();
                        DragAndDrop.visualMode = flag ? DragAndDropVisualMode.Copy : DragAndDropVisualMode.Rejected;
                        Event.current.Use();
                        Repaint();
                        break;
                    case EventType.DragPerform:
                        IEnumerable<Mesh> meshes = DragAndDrop.objectReferences.OfType<Mesh>();
                        IEnumerable<Material> materials = DragAndDrop.objectReferences.OfType<Material>();
                        IEnumerable<Texture> textures = DragAndDrop.objectReferences.OfType<Texture>();

                        if (meshes.Any() || materials.Any() || textures.Any())
                        {
                            Undo.RecordObject(target, "Add");
                            AvatarOutlookData data = new AvatarOutlookData();
                            (target as AvatarOutlookDataConfig).data.Add(data);
                            foreach (var mesh in meshes)
                            {
                                if (mesh.name == "mesh_head") data.headMesh = mesh;
                                else if (mesh.name == "mesh_body") data.bodyMesh = mesh;
                                else if (mesh.name == "mesh_top") data.topMesh = mesh;
                                else if (mesh.name == "mesh_bottom") data.bottomMesh = mesh;
                                else if (mesh.name == "mesh_footwear") data.footwearMesh = mesh;
                            }

                            foreach (var material in materials)
                            {
                                if (material.name == "mat_head") data.headMaterial = material;
                                else if (material.name == "mat_body") data.bodyMaterial = material;
                                else if (material.name == "mat_top") data.topMaterial = material;
                                else if (material.name == "mat_bottom") data.bottomMaterial = material;
                                else if (material.name == "mat_footwear") data.footwearMaterial = material;
                            }

                            foreach (var texture in textures)
                            {
                                string path = AssetDatabase.GetAssetPath(texture);
                                TextureImporter importer = AssetImporter.GetAtPath(path) as TextureImporter;
                                if (importer != null && importer.textureType == TextureImporterType.Sprite)
                                {
                                    data.thumb = AssetDatabase.LoadAssetAtPath<Sprite>(path);
                                }
                            }

                            EditorUtility.SetDirty(target);
                        }
                        Event.current.Use();
                        Repaint();
                        break;
                }
            }
            Color color = GUI.color;
            GUI.color = new Color(GUI.color.r, GUI.color.g, GUI.color.b, containsMouse ? .9f : .5f);
            GUI.Box(dropRect, "Drop Here", new GUIStyle(GUI.skin.box) { fontSize = 10 });
            GUI.color = color;
            GUILayout.EndHorizontal();
        }
    }
#endif
}

:当然正式开发工作中不建议使用这些配置表,应该去构建资源的Assets Bundle,实现资源的动态管理。

🎈 关于胡须

编辑胡须的时候不一定是替换Mesh,有的是直接画在了头部的贴图中,因此分为MeshTexture两种类型:

Mesh Beard & Texture Beard

[Serializable]
public class AvatarBeardData
{
    public Sprite thumb;

    public enum Type
    {
        Mesh,
        Texture,
    }

    public Type type = Type.Mesh;

    public Mesh beardMesh;

    public Material beardMaterial;
}

当类型为Mesh时,替换Beard部件的Skin Mesh Renderer中MeshMaterial,当类型为Texture时,替换Head部件的Skin Mesh Renderer中的Material

相关文章:

  • Vue 3 快速上手
  • 30、根据官方教程详解嵌套类、内部类、静态嵌套类、局部类、匿名类 ...
  • 【Java牛客刷题】入门篇(05)
  • docker入门
  • 【NodeJs-5天学习】第四天存储篇① ——安装使用mysql 8.0
  • 【Verilog 流水线设计】以全加器为例阐述流水线设计的影响
  • spring boot 使用Mybatis-plus的查询方法
  • nginx中root和alias的区别
  • pytorch深度学习训练模板
  • 【Qt+FFMPEG】 - 封装 解码音视频 线程
  • Arduino框架下最便宜的开发芯片-CH552初探
  • Java高并发编程实战5,异步注解@Async自定义线程池
  • 前端进阶——ES6
  • (三)centos7案例实战—vmware虚拟机硬盘挂载与卸载
  • 【Python+大数据】第一天:安装VMware及Centos,配置虚拟机网络,学习Linux命令。研究生开学10天的感受。
  • [译]CSS 居中(Center)方法大合集
  • 2017届校招提前批面试回顾
  • angular2开源库收集
  • CODING 缺陷管理功能正式开始公测
  • CSS 提示工具(Tooltip)
  • DataBase in Android
  • hadoop集群管理系统搭建规划说明
  • JDK 6和JDK 7中的substring()方法
  • Laravel深入学习6 - 应用体系结构:解耦事件处理器
  • nginx 负载服务器优化
  • ViewService——一种保证客户端与服务端同步的方法
  • 包装类对象
  • 后端_MYSQL
  • 机器学习学习笔记一
  • 数据仓库的几种建模方法
  • 智能合约开发环境搭建及Hello World合约
  • ionic入门之数据绑定显示-1
  • 资深实践篇 | 基于Kubernetes 1.61的Kubernetes Scheduler 调度详解 ...
  • #图像处理
  • $.extend({},旧的,新的);合并对象,后面的覆盖前面的
  • (2.2w字)前端单元测试之Jest详解篇
  • (26)4.7 字符函数和字符串函数
  • (多级缓存)多级缓存
  • (附源码)springboot金融新闻信息服务系统 毕业设计651450
  • (欧拉)openEuler系统添加网卡文件配置流程、(欧拉)openEuler系统手动配置ipv6地址流程、(欧拉)openEuler系统网络管理说明
  • (十六)一篇文章学会Java的常用API
  • (转) RFS+AutoItLibrary测试web对话框
  • (总结)Linux下的暴力密码在线破解工具Hydra详解
  • .aanva
  • .net CHARTING图表控件下载地址
  • .NET Core 项目指定SDK版本
  • .Net Core 中间件验签
  • .NET/C# 的字符串暂存池
  • .Net高阶异常处理第二篇~~ dump进阶之MiniDumpWriter
  • .Net中的集合
  • .pop ----remove 删除
  • /usr/bin/python: can't decompress data; zlib not available 的异常处理
  • ??在JSP中,java和JavaScript如何交互?
  • @软考考生,这份软考高分攻略你须知道
  • [Android开源]EasySharedPreferences:优雅的进行SharedPreferences数据存储操作