GAMESH

游戏美术动画unity资源搬运工!

剧情党,完美控!
桐崎千棘
当前位置:首页 > 学习 > 正文内容

unity使用json三维模型的变形动画

admin1年前 (2020-04-20)学习646

有时我们只需要网格中顶点的子集即可动画,而无需完整的骨架,例如用于面部动画的一组嘴形或脸部顶点。一种简单的方法是使用“变形目标动画”。在此动画中,我们混合顶点而不是骨骼,这就是为什么它也被称为逐顶点动画的原因。动画存储为原始网格顶点的一系列变形版本。变形版本称为“变形目标”,而原始版本称为“基础”。我们可以使用权重数组在基本目标和变形目标之间进行混合:

v e r t e x = b a s e + i = 1 n ( w i ( t a r g e t i b a s e ) )

通过使用jd文件格式,我们可以在pos0和pos1,normal0和normal1之间进行插值

来自Unity Wiki的以下示例仅使用Lerp在2个网格之间混合顶点:

using UnityEngine;
 /// REALLY IMPORTANT NOTE./// When using the mesh morpher you should absolutely make sure that you turn/// off generate normals automatically in the importer, or set the normal angle to 180 degrees./// When importing a mesh Unity automatically splits vertices based on normal creases./// However the mesh morpher requires that you use the same amount of vertices for each mesh and that/// those vertices are laid out in the exact same way. Thus it wont work if unity autosplits vertices based on normals.[RequireComponent(typeof(MeshFilter))]public class MeshMorpher : MonoBehaviour{   public Mesh[] m_Meshes;
   public bool m_AnimateAutomatically = true;
   public float m_OneLoopLength = 1.0F; /// The time it takes for one loop to complete   public WrapMode m_WrapMode = WrapMode.Loop;
   private float m_AutomaticTime = 0;
 
   private int m_SrcMesh = -1;
   private int m_DstMesh = -1;
   private float m_Weight = -1;
   private Mesh m_Mesh;
 
   /// Set the current morph in       public void SetComplexMorph(int srcIndex, int dstIndex, float t)   {      if (m_SrcMesh == srcIndex && m_DstMesh == dstIndex && Mathf.Approximately(m_Weight, t))         return;
 
      Vector3[] v0 = m_Meshes[srcIndex].vertices;
      Vector3[] v1 = m_Meshes[dstIndex].vertices;
      Vector3[] vdst = new Vector3[m_Mesh.vertexCount];
      for (int i = 0; i < vdst.Length; i++)         vdst[i] = Vector3.Lerp(v0[i], v1[i], t);
 
      m_Mesh.vertices = vdst;
      m_Mesh.RecalculateBounds();
   } 
   /// t is between 0 and m_Meshes.Length - 1.   /// 0 means the first mesh, m_Meshes.Length - 1 means the last mesh.   /// 0.5 means half of the first mesh and half of the second mesh.   public void SetMorph(float t)   {      int floor = (int)t;
      floor = Mathf.Clamp(floor, 0, m_Meshes.Length - 2);
      float fraction = t - floor;
      fraction = Mathf.Clamp(t - floor, 0.0F, 1.0F);
      SetComplexMorph(floor, floor + 1, fraction);
   } 
   void Awake()   {      enabled = m_AnimateAutomatically;
      MeshFilter filter = GetComponent(typeof(MeshFilter)) as MeshFilter;
 
      // Make sure all meshes are assigned!      for (int i = 0; i < m_Meshes.Length; i++)      {         if (m_Meshes[i] == null)         {            Debug.Log("MeshMorpher mesh  " + i + " has not been setup correctly");
            m_AnimateAutomatically = false;
            return;
         }      } 
      //  At least two meshes      if (m_Meshes.Length < 2)      {         Debug.Log("The mesh morpher needs at least 2 source meshes");
         m_AnimateAutomatically = false;
         return;
      } 
      filter.sharedMesh = m_Meshes[0];
      m_Mesh = filter.mesh;
      int vertexCount = m_Mesh.vertexCount;
      for (int i = 0; i < m_Meshes.Length; i++)      {         if (m_Meshes[i].vertexCount != vertexCount)         {            Debug.Log("Mesh " + i + " doesn't have the same number of vertices as the first mesh");
            m_AnimateAutomatically = false;
            return;
         }      }   } 
   void Update()   {      if (m_AnimateAutomatically)      {         float deltaTime = Time.deltaTime * (m_Meshes.Length - 1) / m_OneLoopLength;
         m_AutomaticTime += deltaTime;
         float time;
         if (m_WrapMode == WrapMode.Loop)            time = Mathf.Repeat(m_AutomaticTime, m_Meshes.Length - 1);
         else if (m_WrapMode == WrapMode.PingPong)            time = Mathf.PingPong(m_AutomaticTime, m_Meshes.Length - 1);
         else            time = Mathf.Clamp(m_AutomaticTime, 0, m_Meshes.Length - 1);
 
         SetMorph(time);
      }   }}

转载http://www.cgdev.net/blog/476.html

扫描二维码推送至手机访问。

本サイト上に掲載の文章、画像、写真などを無断で複製することは法律で禁じられています。全ての著作権はGAMESHに帰属します。

本文链接:https://www.pylblog.com/post/178.html

标签: 3dmaxunity3d
分享给朋友:

相关文章

我的AE作品合集

我的AE作品合集

接触AE快五年了,当初因为一个特殊原因开始使用AE!发现自己很容易就上手了,无论是视频修剪处理,还是特效合成都能玩转!近几年由于改做游戏动画行业对AE接触渐渐减少,也慢慢地感到陌生了,电脑里的AE也很...

运动规律的一些基本概念

运动规律的一些基本概念

动画片中的活动形象,不象其它影片那样,用胶片直接拍摄客观物体的运动,而是通过对客观物体运动的观察、分析、研究,用动画片的表现手法(主要是夸张、强调动作过中的某些方面),一张张地画出来,一格格地拍出来,...

unity制作2d横板过关demo

unity制作2d横板过关demo

unity制作的一款2d横板游戏demo,目前只是做着玩只有一关素材来源于网络!   有兴趣的可以下载来体验一下地址:https://www.pylblog.com/up...

100个HTML5纯css写的图标

100个HTML5纯css写的图标

width:0 12px 0 6px;border-style:solid;bottom:0;left:0}.icono-comment{width:30px;height:20px;box-shad...

RealFlow流体模拟牛奶

RealFlow流体模拟牛奶

RealFlow流体模拟牛奶(一)学习重点:学习目的:Maya中场景文件的准备Maya和RealFlow的交互RealFlow系统设置本案例通过牛奶作掌握Maya与RealFlow的交互,RealFl...

安卓使用zanti进行中间人攻击

安卓使用zanti进行中间人攻击

ZANTI是一款安卓平台上的网路渗透工具,他可以截获局域网内网络数据包(替换目标网页图片、视频、下载文件、插入恶意代码、查看对方浏览的网页、登陆的QQ、微信、微博、获取对方密码、踢对方下线、截获coo...

发表评论

访客

看不清,换一张

◎欢迎参与讨论,请在这里发表您的看法和观点。