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

Unity Avatar Foot IK - Avatar Foot Placement Resolution

文章目录

  • 简介
  • 实现
    • Avatar FBX Import Settings
    • Animator Settings
    • On Animator IK
    • Calculate IK Position & Rotation
    • Body Position
    • Apply IK Position & Rotation


简介

通过Unity内部的Mecanim动画系统实现的FootIK功能,效果如图所示,左右分别为开启和关闭FootIK的效果:

效果图(一)

效果图(二)

效果图(三)

效果图(四)

初版1.0.0代码已上传至SKFramework框架PackageManager中:

SKFramework PackageManager

相关变量说明:

Avatar Foot IK

  • Enable Foot Ik:是否启用FootIK
  • Foot Ik Pass Layer Index:Animator启用IKPass对应的层级
  • Layer Mask:射线检测时所有的层级
  • Body Y Offset:身体Y坐标的偏移量
  • Body Position Lerp Speed:身体坐标插值的速度
  • Foot Position Lerp Speed:脚部坐标插值的速度
  • Raycast Distance:射线检测的最大距离
  • Raycast Origin Height:射线检测的高度

实现

Avatar FBX Import Settings

  • Animation Type: 需要Humanoid人形动画

Animation Type

  • Avatar Configuration:确保配置正确

Avatar Configuration

Animator Settings

  • Foot IK:相应的Animator State中需要开启Foot IK

Foot IK

  • IK Pass:相应的Animator Layer中需要开启IK Pass通道

IK Pass

On Animator IK

动画IK回调函数,Unity Documentation中这样介绍:OnAnimatorIK() 在即将更新其内部反向动力学系统前由动画器组件调用。该回调可用于设置反向动力学目标的位置及其各自的权重。

参数LayerIndex指的是Animator中的Layer层级的索引值。

OnAnimatorIK
如何设置IK目标的位置及其权重?需要用到Animator中的函数:

  • SetIKPosition:设置一个IK Goal的位置
  • SetIKPositionWeight:设置IK Goal的过渡权重(0表示IK之前的原始动画,1表示在goal)
  • SetIKRotation:设置一个IK Goal的旋转
  • SetIKRotationWeight:设置IK Goal的旋转权重

Calculate IK Position & Rotation

如何获取IK目标位置及旋转?可以通过在脚部加上一定单位的高度上向下进行Raycast射线检测,RaycastHit中的point碰撞点即是IK的目标位置,并且通过normal法线方向获得IK的目标旋转,代码如下所示:

#region 计算左脚IK
//左脚坐标
leftFootPosition = animator.GetBoneTransform(HumanBodyBones.LeftFoot).position;
leftFootPosition.y = transform.position.y + raycastOriginHeight;//左脚 射线检测
leftFootRaycast = Physics.Raycast(leftFootPosition, Vector3.down, out RaycastHit hit, raycastDistance + raycastOriginHeight, layerMask);
if (leftFootRaycast)
{leftFootIkPosition = leftFootPosition;leftFootIkPosition.y = hit.point.y + bodyYOffset;leftFootIkRotation = Quaternion.FromToRotation(transform.up, hit.normal);
#if UNITY_EDITOR//射线Debug.DrawLine(leftFootPosition, leftFootPosition + Vector3.down * (raycastDistance + raycastOriginHeight), Color.yellow);//法线Debug.DrawLine(hit.point, hit.point + hit.normal * .5f, Color.cyan);
#endif
}
else
{leftFootIkPosition = Vector3.zero;
}
#endregion

Body Position

在设置IK目标位置之前,需要先计算和调整身体的高度,原因如下图所示,当射线检测到的IK Position,腿的长度达不到时,需要将身体的Y坐标减去相应距离。

身体高度通过Animator中的bodyPosition去调整:

Animator bodyPosition

代码如下所示:

#region 身体
if (leftFootRaycast && rightFootRaycast)
{//左脚坐标Y差值float leftPosYDelta = leftFootIkPosition.y - transform.position.y;//右脚坐标Y差值float rightPosYDelta = rightFootIkPosition.y - transform.position.y;//身体坐标Y差值取二者最小值float bodyPosYDelta = Mathf.Min(leftPosYDelta, rightPosYDelta);//目标身体坐标Vector3 targetBodyPosition = animator.bodyPosition + Vector3.up * bodyPosYDelta;//插值运算targetBodyPosition.y = Mathf.Lerp(lastBodyPositionY, targetBodyPosition.y, bodyPositionLerpSpeed);//设置身体坐标animator.bodyPosition = targetBodyPosition;
}
//缓存身体Y坐标
lastBodyPositionY = animator.bodyPosition.y;
#endregion

Apply IK Position & Rotation

求得目标位置和旋转并调整完身体高度后,应用目标位置和旋转即可:

#region 应用左脚IK
//权重
animator.SetIKPositionWeight(AvatarIKGoal.LeftFoot, 1f);
animator.SetIKRotationWeight(AvatarIKGoal.LeftFoot, 1f);Vector3 targetIkPosition = animator.GetIKPosition(AvatarIKGoal.LeftFoot);
if (leftFootRaycast)
{//转局部坐标targetIkPosition = transform.InverseTransformPoint(targetIkPosition);Vector3 world2Local = transform.InverseTransformPoint(leftFootIkPosition);//插值计算float y = Mathf.Lerp(lastLeftFootPositionY, world2Local.y, footPositionLerpSpeed);targetIkPosition.y += y;lastLeftFootPositionY = y;//转全局坐标targetIkPosition = transform.TransformPoint(targetIkPosition);//当前旋转Quaternion currRotation = animator.GetIKRotation(AvatarIKGoal.LeftFoot);//目标旋转Quaternion nextRotation = leftFootIkRotation * currRotation;animator.SetIKRotation(AvatarIKGoal.LeftFoot, nextRotation);
}
animator.SetIKPosition(AvatarIKGoal.LeftFoot, targetIkPosition);
#endregion

http://www.lryc.cn/news/15918.html

相关文章:

  • 是时候告别这些 Python 库了
  • nodejs基于vue论坛交流管理系统
  • 企业电子招投标采购系统源码之系统的首页设计
  • 华为OD机试真题Python实现【竖直四子棋】真题+解题思路+代码(20222023)
  • LeetCode 73. 矩阵置零
  • 「TCG 规范解读」第10章 TPM工作组 保护你的数字环境
  • 华为OD机试真题Python实现【 找字符】真题+解题思路+代码(20222023)
  • 如何解决多继承下的 菱形继承 问题
  • rk3288-android8.1-以太网ethernet和蓝牙Bluetooth
  • 算法比赛——必备的数论知识
  • Docker概述
  • 实验室设计建设方案主要内容
  • 华为OD机试真题Python实现【日志采集系统】真题+解题思路+代码(20222023)
  • Python的模块与工具包
  • 联合熵和条件熵
  • 华为OD机试真题Python实现【求最大数字】真题+解题思路+代码(20222023)
  • Python爬虫(10)selenium爬虫后数据,存入csv、txt并将存入数据并对数据进行查询
  • Python 之 Pandas 时间函数 time 、datetime 模块和时间处理基础
  • C语言学习及复习笔记-【5】C 运算符
  • 数仓、数据湖、湖仓一体、数据网格
  • C语言【atoi函数】
  • 一起学习用Verilog在FPGA上实现CNN----(八)integrationFC设计
  • 面试题总结
  • go进阶(1) -深入理解goroutine并发运行机制
  • mongodb 操作记录
  • JDBC简单的示例
  • Spring架构篇--2.3 远程通信基础--IO多路复用select,poll,epoll模型
  • python--matplotlib(4)
  • 【项目精选】城市公交查询系统(论文+视频+源码)
  • less、sass、webpack(前端工程化)