summaryrefslogtreecommitdiff
path: root/Erika/Assets/Scripts/Unit/Components/UnitAnimation/UnitAnimation.cs
blob: 653564b7a011d838e7c5ef6c66700e269b4dd3c3 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
#define ANIM_CROSS_FADE
using System;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
#if UNITY_EDITOR
using UnityEditor;
#endif

// http://unitylore.org/wiki/doku.php?id=l3y0gj

// 单独一层动画
public class AnimatorLayerInfo
{
    public int layer;

    public int layerIndex { get { return (int)layer; } }

    public string name { get { return m_Animator.GetLayerName(layerIndex); } }

    public bool isBaseLayer { get { return layerIndex == 0; } }

    private Animator m_Animator;

    // 当前动作的animation data,如果是blendtree选第一个动作
    private AnimationData m_AnimationData;
    public AnimationData animationData
    {
        get
        {
            AnimatorClipInfo[] clipInfos = clipsInfo;
            if (clipInfos == null || clipInfos.Length == 0)
            {
                LogHelper.LogError(LogTag.UnitAnimation, "No current animation.");
                return null;
            }
            var clip = clipInfos[0]; //选第一个
            string folder = m_UnitAnimation.owner.folder;
            string name = clip.clip.name;
            if (m_AnimationData != null && m_AnimationData.animationName == name)
                return m_AnimationData;
            string path = folder + "AnimationData/" + name + ".asset" ; // 要注意这里是用名字区别的,因为最终每个角色的动画都会在一个目录下面
            m_AnimationData  = ResourceManager.Instance.LoadAsset<AnimationData>(path);
            return m_AnimationData;
        }
    }

	// 当前在播放的动作
	// 如果处于transition中,认为当前状态是transition的目标状态(和Animator规则不一样)
	public AnimatorStateInfo stateInfo
    {
        get
        {
			AnimatorStateInfo stateInfo = m_Animator.GetCurrentAnimatorStateInfo(layerIndex);
			if (isInTransition) // 过渡中的动作认为当前动作是下一个动作
			{
				stateInfo = m_Animator.GetNextAnimatorStateInfo(layerIndex);
			}
			return stateInfo;
		}
    }

    private int preStateHash = -1;

    // 当前正在播放和融合的片段信息
    public AnimatorClipInfo[] clipsInfo
    {
        get
        {
            AnimatorClipInfo[] clips = null;
            if (!isInTransition) 
            {
                clips = m_Animator.GetCurrentAnimatorClipInfo(layerIndex);
                if (clips.Length == 0)
                {
                    clips = m_Animator.GetNextAnimatorClipInfo(layerIndex);
                }
            }
            else // 过渡中的动作认为当前动作是下一个动作
            {
                clips = m_Animator.GetNextAnimatorClipInfo(layerIndex);
            }
            return clips;
        }
    }

	public AnimatorClipInfo clipInfo
	{
		get
		{
			return clipsInfo[0];
		}
	}

    public int stateHash
    {
        get
        {
            return stateInfo.shortNameHash;
        }
    }

    // 并非准确的播放时间,只是逻辑时间,因为动画会加速减速
    public float playbackTimeInSeconds
    {
        get
        {
            return playbackNormalizedTime * clipInfo.clip.length; 
        }
    }

    // 这个是真实世界的时间
    private float m_PlaybackRealTime;
    public float playbackRealTimeInSeconds
    {
        get
        {
            return m_PlaybackRealTime;
        }
    }

    // 播放进度百分比,[0-1],是逻辑上的,如果动画不是loop,那么播放完后normalizedTime是1
    public float playbackNormalizedTime
    {
        get
        {
            AnimatorStateInfo state = stateInfo;
            if (!state.loop && state.normalizedTime > 1)
                return 1;
            return state.normalizedTime % 1f;
        }
    }

	public float playbackSpeed
	{
		get
		{
			return m_Animator.GetFloat("PlaybackSpeed" + layerIndex);
		}
		set
		{
			float v = Mathf.Clamp(value, 0, 10);
			m_Animator.SetFloat("PlaybackSpeed" + layerIndex, v);
		}
	}

	public float weight
    {
        get
        {
            return m_Animator.GetLayerWeight(layerIndex);
        }
    }

    public int nextStateHash
    {
        get
        {
            AnimatorStateInfo nextState = m_Animator.GetNextAnimatorStateInfo(layerIndex);
            int hash = nextState.shortNameHash; // 如果不在过渡中,hash是0
            return hash;
        }
    }

    public bool isInTransition
    {
        get
        {
            return m_Animator.IsInTransition(layerIndex);
        }
    }

	public bool applySpeedCurve { get; set; }

	UnitAnimation m_UnitAnimation;

    Coroutine m_CalcPlaybackTimeCoroutine;

	TimelineEventProxy m_TimelineEventProxy;

	public string m_CurrentState;

	public AnimatorLayerInfo(UnitAnimation unitAnimation, Animator animator, int layer)
    {
        this.m_UnitAnimation = unitAnimation;
        this.m_Animator = animator;
        this.layer = layer;
		m_CalcPlaybackTimeCoroutine = unitAnimation.StartCoroutine(CalcPlaybackRealTimeCoroutine());
		m_TimelineEventProxy = new TimelineEventProxy(unitAnimation.owner);
		applySpeedCurve = true;
	}

	IEnumerator CalcPlaybackRealTimeCoroutine()
    {
        while (true)
        {
            if(preStateHash != stateHash)
            {
                m_PlaybackRealTime = 0;
            }
            m_PlaybackRealTime += Time.deltaTime;
            preStateHash = stateHash;
            yield return null; 
        }
    }

    public bool IsToggleOpen(EAnimationToogle toggle)
    {
        if (m_AnimationData == null)
            return false;
        return m_AnimationData.IsToggleOpen(toggle, playbackNormalizedTime);
    }

	public void OnUpdate()
	{
        // 执行事件
        m_TimelineEventProxy.ExecuteAnimationEvents(animationData, playbackTimeInSeconds * TimelineEventProxy.FPS);

        // 播放速度控制
        if (applySpeedCurve && animationData != null && animationData.speedCurve != null)
		{
			playbackSpeed = animationData.speedCurve.Evaluate(playbackNormalizedTime);
		}
	}

	public void OnCrossFade(string animState, float normalizedTransitionDuration, float normalizedTimeOffset, float normalizedTransitionTime )
	{
		m_CurrentState = animState;
		m_Animator.CrossFade(animState.ToString(), normalizedTransitionDuration, layerIndex, normalizedTimeOffset, normalizedTransitionTime);
        m_TimelineEventProxy.ResetPrevAnimationData();

        playbackSpeed = 1;
    }

    public void OnPlay(string animState, float normalizedTime)
	{
		m_CurrentState = animState;
        m_Animator.Play(animState, layerIndex, normalizedTime);
        m_TimelineEventProxy.ResetPrevAnimationData();

        playbackSpeed = 1;
    }
}

[DisallowMultipleComponent]
public class UnitAnimation : UnitComponent
{
    protected AnimatorLayerInfo[] layers { get { return m_LayerInfo; } }
    protected AnimatorLayerInfo[] m_LayerInfo;

    public Animator animator { get { return m_Animator; } }
    protected Animator m_Animator;

	protected virtual AnimatorLayerInfo baseLayer
    {
        get { return layers[0]; }
    }

	protected AnimatorLayerInfo layer
    {
        get
        {
            return baseLayer;
        }
    }

    public bool isInTransition
    {
        get
        {
            return m_Animator.IsInTransition(0);
        }
    }

	public AnimationData animationData
	{
		get
		{
			return layer.animationData;
		}
	}

	public AnimatorClipInfo[] clipsInfo
	{
		get
		{
			return layer.clipsInfo;
		}
	}

	public AnimatorClipInfo clipInfo
	{
		get
		{
			return layer.clipInfo;
		}
	}

	public float playbackNormalizedTime
	{
		get
		{
			return layer.playbackNormalizedTime;
		}
	}

	public AnimatorStateInfo stateInfo
	{
		get
		{
			return layer.stateInfo;
		}
	}

	public int stateHash
	{
		get
		{
			return layer.stateHash;
		}
	}

	public override void OnUpdate()
    {
        base.OnUpdate();
    }

	public bool IsToggleOpen(EAnimationToogle toggle)
	{
		return layer.IsToggleOpen(toggle);
	}

	/// <summary>
	/// 直接播放动作
	/// </summary>
	/// <param name="animState"></param>
	/// <param name="layerIndex"></param>
	/// <param name="normalizedTime"></param>
	public void Play(string animState, int layerIndex = 0, float normalizedTime = float.NegativeInfinity)
    {
        AnimatorLayerInfo layer = this.layers[layerIndex];
        if (layer == null)
            return;
        layer.OnPlay(animState, normalizedTime);

		UnitRootMotion rm = m_Owner.GetComponent<UnitRootMotion>();
		if(rm)
		{
			rm.OnAnimationChange();
		}
	}

    /// <summary>
    /// CrossFade
    /// </summary>
    /// <param name="animState"></param>
    /// <param name="normalizedTransitionDuration"></param>
    /// <param name="layerIndex"></param>
    /// <param name="normalizedTimeOffset"></param>
    /// <param name="normalizedTransitionTime"></param>
    public void CrossFade(string animState, float normalizedTransitionDuration, int layerIndex = 0, float normalizedTimeOffset = float.NegativeInfinity, float normalizedTransitionTime = 0.0f)
    {
        AnimatorLayerInfo layer = this.layers[layerIndex];
        if (layer == null)
            return;
        layer.OnCrossFade(animState, normalizedTransitionDuration, normalizedTimeOffset, normalizedTransitionTime);

		UnitRootMotion rm = m_Owner.GetComponent<UnitRootMotion>();
		if (rm)
		{
			rm.OnAnimationChange();
		}
	}

}