新动画系统:
给模型选择动画类型
普通动画:Generic
人形动画:Humanoid

建立动画控制器 - 在Project右击 - 选择Create-AnimatorContorller
将对应动画控制器拖入到Animator的Contorller

双击动画控制器可以打开Animator窗口
将对应动画片段拖入生成新节点,就可以编辑了
橙色节点:相当于旧动画系统的默认片段
灰色节点:
Any State:代表所有动画状态

设置默认片段

建立基础连线

箭头代表两个动画的过渡
Transition Duration:代表融合

任何状态都可以连线到death,甚至包括自己到自己

取消death过渡到自己的选择
动画鬼畜问题,自己过渡到自己

建立过渡条件
添加参数,有4种
通过代码激活参数,实现动画切换

默认过渡条件:Has Exit Time,播放完当前动画才会进行到下一动画的过渡
有条件,取消勾选
无条件,默认勾选,跳跃,浮空

给游戏物体添加脚本,控制参数的激活
代码不能越权:越过动画状态机设置的逻辑(动画连线)的权限

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
public class HellephantAnimator : MonoBehaviour {
public Animator anim;
// Use this for initialization
void Start () { } // Update is called once per frame
void Update () {
if (Input.GetKeyDown(KeyCode.Alpha1))
{
anim.SetBool("isMove", true);
}
if (Input.GetKeyDown(KeyCode.Alpha2))
{
anim.SetBool("isMove", false);
}
if (Input.GetKeyDown(KeyCode.Alpha3))
{
anim.SetTrigger("Death");
}
}
}
用一个怪物管理器,管理两个怪物的父类
两种方法:
1、管理两个怪物的父类
2、把方法写成接口
父类:全部写成虚函数
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.AI;
public enum EnemySta
{
Move,
Idle,
Death
}
public class NewMonsterBase : MonoBehaviour {
public PlayerControl moveTarget;//设置目标点
public MonsterManager manager;//怪物属于那个管理器
public virtual void Action()
{
}
public virtual void Move()
{
}
public virtual void Idle()
{
}
public virtual void Damage()
{
}
public virtual void Death()
{
}
public virtual void DeathEvent()
{
}
}

这两个需要初始化,要放在父类里

public PlayerControl moveTarget;//设置目标点
public MonsterManager manager;//怪物属于那个管理器

子类:继承父类,并重写父类方法

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.AI;
public class HellephantControl : NewMonsterBase
{
public Animator anim;
public EnemySta enemySta;
//因为我们需要经常用到目标的状态,所以把TransForm改成了PlayerControl
public NavMeshAgent navMeshAgent;//引用寻路系统
private int enemyHealth = ;//设置怪物生命值
// Use this for initialization
void Start()//子类添加新动画系统
{
enemySta = EnemySta.Move;
anim.SetBool("isMove", true);
}
// Update is called once per frame
void Update()
{
Action();
}
public override void Action()
{
switch (enemySta)
{
case EnemySta.Move:
Move();
break;
case EnemySta.Idle:
Idle();
break;
case EnemySta.Death:
Death();
break;
}
}
public override void Move()
{
//anim.CrossFade("Move", 0.2f);
navMeshAgent.SetDestination(moveTarget.transform.position);
//如果怪物的目标死了
if (moveTarget.playSta == PlaySta.Death)
{
//寻路停止
navMeshAgent.isStopped = true;
//状态切换到闲置
enemySta = EnemySta.Idle;
anim.SetBool("isMove", false);
}
}
public override void Idle()
{
//anim.CrossFade("Idle", 0.2f);
}
public override void Damage()
{
//受伤减血
enemyHealth -= ;
//Debug.Log("怪物生命"+enemyHealth);
//如果血到了0
if (enemyHealth <= )
{
//状态变成死亡状态
enemySta = EnemySta.Death;
anim.SetTrigger("Death");
navMeshAgent.isStopped = true;
gameObject.GetComponent<SphereCollider>().enabled = false;
//gameObject.tag = "Untagged";
}
}
public override void Death()//子类关闭老动画系统
{
//anim.CrossFade("Death", 0.2f);
}
public override void DeathEvent()
{
//在管理器列表中移除自己
manager.monsterList.Remove(this);
//销毁自己
Destroy(this.gameObject);
}
}
怪物管理器的怪物列表全部改成父类
怪物管理器的怪物生成
怪物预制体改成数组
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
public class MounsterManager : MonoBehaviour {
List<Transform> birthPoint;
public GameObject[] monsterPreGo;
//最大怪物数
public int MaxMonsterCount;
public List<NewMonsterBase> monsterList;
public NewPlayer player;
// Use this for initialization
void Start () {
birthPoint = new List<Transform>();
monsterList = new List<NewMonsterBase>();
//遍历所有的子物体,将其加到出生点列表中
for (int i = ; i < transform.childCount; i++)
{
birthPoint.Add(transform.GetChild(i));
} }
void CreateMonster() {
monsterCont++;
GameObject go;
if (monsterCont%==)
{
go = Instantiate<GameObject>(monsterPreGo[]);
}
else
{
//实例化一个怪物
go = Instantiate<GameObject>(monsterPreGo[]);
} //怪物的位置设置任意出生点
go.transform.position = birthPoint[Random.Range(, birthPoint.Count)].position;
//取得怪物脚本
NewMonsterBase monster = go.GetComponent<NewMonsterBase>();
//把怪物放到管理器列表里
monsterList.Add(monster);
//指定怪物的目标
monster.moveTarget = player;
//指定怪物的管理者
monster.manager = this;
}
int monsterCont = ;
void CheckMonster() {
if (monsterList.Count < MaxMonsterCount)
{
CreateMonster();
}
}
// Update is called once per frame
void Update () { }
private void FixedUpdate()
{
CheckMonster();
}
}

怪物动画添加死亡检测事件
骨骼动画锁定:只允许动画控制器修改
在动画状态机的设置里,取消骨骼动画锁定

怪物添加碰撞体

新动画系统的新功能
融合树功能:一个和多个参数,来决定播放哪个动画
更改动画类型

动画和模型分开

拖拽模型生成预制体

添加动画控制器

任意窗口最大化:shift+空格
中键拖动,滚轮缩放
动画本身带有位移动画,根据实际情况决定
好处:过渡自然
坏处:代码不可控,代码强拉,会显得不自然
勾选动画自带的位移动画

建立三个动画的融合树
添加参数,保存键值,保证参数正确传递

右键新建 融合树

双击融合树,打开融合树窗口,右键可以再添加融合树

添加motion

第一列:默认融合区间0到1 ,三个动画过渡0,0.5,1
第二列:动画速度
默认融合区间均等分开,取消勾选,可自行修改

旧动画系统修改动画片段的速度

添加脚本

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
public class DefaultAvatarAnimator : MonoBehaviour {
public Animator anim;
public int speedHashID;
public int angleHashID;
public int attackHashID;
private float slowSpeed;//速度修正
// Use this for initialization
void Start () {
speedHashID = Animator.StringToHash("Speed");//哈希Code,效能略高,不用每次调用都转换
angleHashID = Animator.StringToHash("Angle");
attackHashID = Animator.StringToHash("Attack");
}
// Update is called once per frame
void Update () {
if (Input.GetKey(KeyCode.LeftShift))
{
slowSpeed = 0.5F;
}
else
{
slowSpeed = 1.0F;
}
anim.SetFloat(speedHashID, Mathf.Clamp01(Input.GetAxis("Vertical")) * slowSpeed);//Mathf.Clamp01,剔除向后走的值
anim.SetFloat(angleHashID, Input.GetAxis("Horizontal") * slowSpeed);
if (Input.GetMouseButtonDown())
{
anim.SetTrigger(attackHashID);
} }
}

给Walk和Run添加融合树

层级关系
两个动画融合,要和动画师沟通,融合动作容易变形,最后让动画师手动新建动画
边跑动,边射箭
添加Attack层,添加射箭节点,添加Attack参数

点击齿轮,修改层级权重为1

Blending融合模式
Override,两个动作折中,一般选则
Aaditive,两个动作叠加

添加Avatar Mask,阿凡达遮罩
如果动画类型是人形动画:Humanoid,可选择Humanoid的Mask
分配骨骼权重:点击区域激活失活
动画模型来自同一骨骼(同一Avatar)

添加代码

Unity3D学习笔记(十六):Animator新动画的更多相关文章

  1. python3.4学习笔记(十六) windows下面安装easy_install和pip教程

    python3.4学习笔记(十六) windows下面安装easy_install和pip教程 easy_install和pip都是用来下载安装Python一个公共资源库PyPI的相关资源包的 首先安 ...

  2. (C/C++学习笔记) 十六. 预处理

    十六. 预处理 ● 关键字typeof 作用: 为一个已有的数据类型起一个或多个别名(alias), 从而增加了代码的可读性. typedef known_type_name new_type_nam ...

  3. python 学习笔记十六 django深入学习一 路由系统,模板,admin,数据库操作

    django 请求流程图 django 路由系统 在django中我们可以通过定义urls,让不同的url路由到不同的处理函数 from . import views urlpatterns = [ ...

  4. SharpGL学习笔记(十六) 多重纹理映射

    多重纹理就把多张贴图隔和在一起.比如下面示例中,一个表现砖墙的纹理,配合一个表现聚光灯效果的灰度图,就形成了砖墙被一个聚光灯照亮的效果,这便是所谓的光照贴图技术. 多重纹理只在OpenGL扩展库中才提 ...

  5. yii2源码学习笔记(十六)

    Module类的最后代码 /** * Registers sub-modules in the current module. * 注册子模块到当前模块 * Each sub-module shoul ...

  6. Swift学习笔记十六:协议

    Protocol(协议)用于统一方法和属性的名称,而不实现不论什么功能. 协议可以被类.枚举.结构体实现.满足协议要求的类,枚举,结构体被称为协议的遵循者. 遵循者须要提供协议指定的成员,如属性,方法 ...

  7. PHP学习笔记十六【方法】

    <?php //给一个函数传递基本数据类型 $a=90; $b=90.8; $c=true; $d="hello world"; function test1($a,$b,$ ...

  8. Java基础学习笔记十六 集合框架(二)

    List List接口的特点: 它是一个元素存取有序的集合.例如,存元素的顺序是11.22.33.那么集合中,元素的存储就是按照11.22.33的顺序完成的. 它是一个带有索引的集合,通过索引就可以精 ...

  9. Java学习笔记十六:Java中的构造方法

    Java中的构造方法 1.使用new+构造方法 创建一个新的对象: 2.构造方法是定义在Java类中的一个用来初始化对象的方法: 3.构造方法与类同名且没有返回值: 4.语法格式: public 构造 ...

  10. JavaScript权威设计--CSS(简要学习笔记十六)

    1.Document的一些特殊属性 document.lastModified document.URL document.title document.referrer document.domai ...

随机推荐

  1. TC命令流量控制测试(针对具体IP地址和IP协议)

    这里测试系统为Linux操作系统,通过简单的TC命令来实现对带宽的控制. 1对具体IP地址的流量控制 这里采用iperf来进行带宽的测试,首先在服务器和客户端都安装上iperf软件,通过该软件下的命令 ...

  2. [py][mx]django get方法返回login页面

    get方法返回login.html users/views.py def login(request): if request.method == "POST": pass eli ...

  3. [py][mx]django xadmin后台配置

    xadmin配置 - 安装 pip install -r https://github.com/sshwsfc/xadmin/blob/django2/requirements.txt 以下被我测试通 ...

  4. [py]监控内存并出图

    监控内存出图 先将内存数据搞到数据库 已使用内存算法 used = int(total) - int(free) - int(butffers) - int(cache) pymysql模块使用 db ...

  5. soapUI-DataSink

    1.1.1  DataSink 1.1.1.1 概述 – DataSink Option Description   Properties DataSink属性表   Toolbar DataSink ...

  6. testng入门教程10 TestNG参数化测试

    在TestNG的另一个有趣的功能是参数测试.在大多数情况下,你会遇到这样一个场景,业务逻辑需要一个巨大的不同数量的测试.参数测试,允许开发人员运行同样的测试,一遍又一遍使用不同的值. TestNG让你 ...

  7. 在MS SQL删除重复行的几种方法

    1.如果有ID字段,就是具有唯一性的字段         delect   table   where   id   not   in   (             select   max(id) ...

  8. vue性能优化1--懒加载

    懒加载也叫延迟加载,即在需要的时候进行加载.随用随载.为什么需要懒加载?像vue这种单页面应用,如果没有应用懒加载,运用webpack打包后的文件将会异常的大,造成进入首页时,需要加载的内容过多,时间 ...

  9. MFC Ribbon界面设计

    Ribbon是类似于office2007样式的界面,它替代了传统的MFC程序里的菜单和工具栏 MFC默认生成的Ribbon功能少,需要我们自己添加一些控件和图片等元素使界面好看 看下面的一个界面,是V ...

  10. Groovy中的闭包

    Closures(闭包) 本节主要讲groovy中的一个核心语法:closurs,也叫闭包.闭包在groovy中是一个处于代码上下文中的开放的,匿名代码块.它可以访问到其外部的变量或方法. 1. 句法 ...