From e62848cf382679129074e9911af4eb8840a0ae07 Mon Sep 17 00:00:00 2001 From: "Willem-Jan L. van Rootselaar" Date: Thu, 25 Aug 2022 14:30:13 +0200 Subject: [PATCH] Add support for Animator initial support for Unity Animator --- .../uLipSync/Editor/uLipSyncAnimatorEditor.cs | 247 ++++++++++++++++++ .../Editor/uLipSyncAnimatorEditor.cs.meta | 11 + Assets/uLipSync/Runtime/uLipSyncAnimator.cs | 188 +++++++++++++ .../uLipSync/Runtime/uLipSyncAnimator.cs.meta | 11 + 4 files changed, 457 insertions(+) create mode 100644 Assets/uLipSync/Editor/uLipSyncAnimatorEditor.cs create mode 100644 Assets/uLipSync/Editor/uLipSyncAnimatorEditor.cs.meta create mode 100644 Assets/uLipSync/Runtime/uLipSyncAnimator.cs create mode 100644 Assets/uLipSync/Runtime/uLipSyncAnimator.cs.meta diff --git a/Assets/uLipSync/Editor/uLipSyncAnimatorEditor.cs b/Assets/uLipSync/Editor/uLipSyncAnimatorEditor.cs new file mode 100644 index 0000000..77b7a24 --- /dev/null +++ b/Assets/uLipSync/Editor/uLipSyncAnimatorEditor.cs @@ -0,0 +1,247 @@ +using UnityEngine; +using UnityEditor; +using UnityEditorInternal; +using System.Linq; +using System.Collections.Generic; + +namespace uLipSync +{ + + [CustomEditor(typeof(uLipSyncAnimator))] + public class uLipSyncAnimatorEditor : Editor + { + uLipSyncAnimator uAnimator { get { return target as uLipSyncAnimator; } } + ReorderableList _reorderableList = null; + + public override void OnInspectorGUI() + { + serializedObject.Update(); + + if (EditorUtil.Foldout("LipSync Update Method", true)) + { + ++EditorGUI.indentLevel; + EditorUtil.DrawProperty(serializedObject, nameof(uAnimator.updateMethod)); + --EditorGUI.indentLevel; + EditorGUILayout.Separator(); + } + + if (EditorUtil.Foldout("Animator", true)) + { + ++EditorGUI.indentLevel; + DrawAnimator(); + --EditorGUI.indentLevel; + EditorGUILayout.Separator(); + } + + if (EditorUtil.Foldout("Parameters", true)) + { + ++EditorGUI.indentLevel; + DrawParameters(); + --EditorGUI.indentLevel; + EditorGUILayout.Separator(); + } + + if (EditorUtil.Foldout("Animator Controller Parameters", true)) + { + ++EditorGUI.indentLevel; + if (uAnimator.animator != null) + { DrawAnimatorReorderableList(); } + else + { EditorGUILayout.HelpBox("Animator is not available.", MessageType.Warning); } + --EditorGUI.indentLevel; + EditorGUILayout.Separator(); + } + + serializedObject.ApplyModifiedProperties(); + } + + protected void DrawAnimator() + { + var findFromChildren = EditorUtil.EditorOnlyToggle("Find From Children", "uLipSyncAnimator", true); + EditorUtil.DrawProperty(serializedObject, nameof(findFromChildren)); + + if (findFromChildren) + { + DrawAnimatorsInChildren(); + } + else + { + if (uAnimator.animator == null) + { + EditorGUILayout.HelpBox("Animator is not assigned.", MessageType.Warning); + EditorUtil.DrawProperty(serializedObject, nameof(uAnimator.animator)); + } + else + { + EditorUtil.DrawProperty(serializedObject, nameof(uAnimator.animator)); + } + } + } + + protected void DrawAnimatorsInChildren() + { + var animators = uAnimator.GetComponentsInChildren(); + if (animators.Length == 0) + { + EditorGUILayout.HelpBox("Animator is not found in children.", MessageType.Warning); + } + else + { + + int index = 0; + for (int i = 0; i < animators.Length; ++i) + { + var animator = animators[i]; + if (animator == uAnimator.animator) + { + index = i; + break; + } + } + var names = animators.Select(x => x.gameObject.name).ToArray(); + var newIndex = EditorGUILayout.Popup("Animators", index, names); + if (newIndex != index) + { + Undo.RecordObject(target, "Change Animator"); + uAnimator.animator = animators[newIndex]; + } + } + } + + protected void DrawAnimatorReorderableList() + { + if (_reorderableList == null) + { + _reorderableList = new ReorderableList(uAnimator.parameters, typeof(MfccData)); + _reorderableList.drawHeaderCallback = rect => + { + rect.xMin -= EditorGUI.indentLevel * 12f; + EditorGUI.LabelField(rect, "Phoneme - Parameter Table"); + }; + _reorderableList.draggable = true; + _reorderableList.drawElementCallback = (rect, index, isActive, isFocused) => + { + DrawParameterListItem(rect, index); + }; + _reorderableList.elementHeightCallback = index => + { + return GetParameterListItemHeight(index); + }; + } + + EditorGUILayout.Separator(); + EditorGUILayout.BeginHorizontal(); + var indent = EditorGUI.indentLevel * 12f; + EditorGUILayout.Space(indent, false); + _reorderableList.DoLayoutList(); + EditorGUILayout.EndHorizontal(); + } + + protected void DrawParameterListItem(Rect rect, int index) + { + rect.y += 2f; + rect.height = EditorGUIUtility.singleLineHeight; + + var par = uAnimator.animator.parameters; + var uPar = uAnimator.parameters[index]; + float singleLineHeight = + EditorGUIUtility.singleLineHeight + + EditorGUIUtility.standardVerticalSpacing; + + uPar.phoneme = EditorGUI.TextField(rect, "Phoneme", uPar.phoneme); + + rect.y += singleLineHeight; + + var newIndex = EditorGUI.Popup(rect, "Parameter", uPar.index + 1, GetParameterArray()); + if (newIndex != uPar.index + 1 || uPar.name != par[uPar.index + 1].name) + { + Undo.RecordObject(target, "Change Parameter"); + uPar.index = newIndex - 1; + uPar.name = par[uPar.index + 1].name; + uPar.nameHash = Animator.StringToHash(uPar.name); + Debug.Log($"parameter: {uPar.name} - {uPar.nameHash}"); + } + + rect.y += singleLineHeight; + + float weight = EditorGUI.Slider(rect, "Max Weight", uPar.maxWeight, 0f, 1f); + if (weight != uPar.maxWeight) + { + Undo.RecordObject(target, "Change Max Weight"); + uPar.maxWeight = weight; + } + + rect.y += singleLineHeight; + } + + protected virtual float GetParameterListItemHeight(int index) + { + return 64f; + } + + protected virtual string[] GetParameterArray() + { + if (uAnimator.animator == null) + { + return new string[0]; + } + // get parameters from animator + var parAnimator = uAnimator.animator.parameters; + var names = new List(); + for (int i = 0; i < parAnimator.Length; ++i) + { + var name = parAnimator[i].name; + names.Add(name); + } + return names.ToArray(); + } + + protected void DrawParameters() + { + Undo.RecordObject(target, "Change Volume Min/Max"); + EditorGUILayout.MinMaxSlider( + "Volume Min/Max (Log10)", + ref uAnimator.minVolume, + ref uAnimator.maxVolume, + -5f, 0f); + + var rect = EditorGUILayout.GetControlRect(GUILayout.Height(0f)); + rect.x += EditorGUIUtility.labelWidth; + rect.width -= EditorGUIUtility.labelWidth; + rect.height = EditorGUIUtility.singleLineHeight; + EditorGUILayout.BeginHorizontal(); + { + var origColor = GUI.color; + var style = new GUIStyle(GUI.skin.label); + style.fontSize = 9; + GUI.color = Color.gray; + + var minPos = rect; + minPos.x -= 24f; + minPos.y -= 12f; + if (uAnimator.minVolume > -4.5f) + { + minPos.x += (uAnimator.minVolume + 5f) / 5f * rect.width - 30f; + } + EditorGUI.LabelField(minPos, $"{uAnimator.minVolume.ToString("F2")}", style); + + var maxPos = rect; + var maxX = (uAnimator.maxVolume + 5f) / 5f * rect.width; + maxPos.y -= 12f; + if (maxX < maxPos.width - 48f) + { + maxPos.x += maxX; + } + else + { + maxPos.x += maxPos.width - 48f; + } + EditorGUI.LabelField(maxPos, $"{uAnimator.maxVolume.ToString("F2")}", style); + GUI.color = origColor; + } + EditorGUILayout.EndHorizontal(); + + EditorUtil.DrawProperty(serializedObject, nameof(uAnimator.smoothness)); + } + } +} diff --git a/Assets/uLipSync/Editor/uLipSyncAnimatorEditor.cs.meta b/Assets/uLipSync/Editor/uLipSyncAnimatorEditor.cs.meta new file mode 100644 index 0000000..60c66a5 --- /dev/null +++ b/Assets/uLipSync/Editor/uLipSyncAnimatorEditor.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 87c0aca8bdd23114ca349f4e03df3b44 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Assets/uLipSync/Runtime/uLipSyncAnimator.cs b/Assets/uLipSync/Runtime/uLipSyncAnimator.cs new file mode 100644 index 0000000..5b09a27 --- /dev/null +++ b/Assets/uLipSync/Runtime/uLipSyncAnimator.cs @@ -0,0 +1,188 @@ +using UnityEngine; +using System.Collections.Generic; + +namespace uLipSync +{ + [ExecuteAlways] + public class uLipSyncAnimator : MonoBehaviour + { + [System.Serializable] + public class AnimatorInfo + { + public string phoneme; + public int index = -1; + public float maxWeight = 1f; + + public float weight { get; set; } = 0f; + public float weightVelocity { get; set; } = 0f; + public int nameHash; + public string name; + } + + public UpdateMethod updateMethod = UpdateMethod.LateUpdate; + public Animator animator; + + /// + /// Animator and animatorinfo for controlling the Animator parameters. + /// + public List parameters = new List(); + public float minVolume = -2.5f; + public float maxVolume = -1.5f; + [Range(0f, 0.3f)] public float smoothness = 0.05f; + + LipSyncInfo _info = new LipSyncInfo(); + bool _lipSyncUpdated = false; + float _volume = 0f; + float _openCloseVelocity = 0f; + protected float volume => _volume; + +#if UNITY_EDITOR + bool _isAnimationBaking = false; + float _animBakeDeltaTime = 1f / 60; +#endif + + public void OnLipSyncUpdate(LipSyncInfo info) + { + _info = info; + _lipSyncUpdated = true; + if (updateMethod == UpdateMethod.LipSyncUpdateEvent) + { + UpdateVolume(); + UpdateVowels(); + _lipSyncUpdated = false; + OnApplyAnimator(); + } + } + void Awake() + { + foreach (AnimatorInfo par in parameters) + { + par.nameHash = Animator.StringToHash(par.name); + } + } + + void Update() + { +#if UNITY_EDITOR + if (_isAnimationBaking) + return; +#endif + if (updateMethod != UpdateMethod.LipSyncUpdateEvent) + { + UpdateVolume(); + UpdateVowels(); + _lipSyncUpdated = false; + } + + if (updateMethod == UpdateMethod.Update) + { + OnApplyAnimator(); + } + } + + void LateUpdate() + { +#if UNITY_EDITOR + if (_isAnimationBaking) + return; +#endif + if (updateMethod == UpdateMethod.LateUpdate) + { + OnApplyAnimator(); + } + + } + void FixedUpdate() + { +#if UNITY_EDITOR + if (_isAnimationBaking) + return; +#endif + if (updateMethod == UpdateMethod.FixedUpdate) + { + OnApplyAnimator(); + } + } + + float SmoothDamp(float value, float target, ref float velocity) + { +#if UNITY_EDITOR + return Mathf.SmoothDamp(value, target, ref velocity, smoothness, Mathf.Infinity, _animBakeDeltaTime); +#else + return Mathf.SmoothDamp(value, target, ref velocity, smoothness); +#endif + } + + void UpdateVolume() + { + float normVol = 0f; + if (_lipSyncUpdated && _info.rawVolume > 0f) + { + normVol = Mathf.Log10(_info.rawVolume); + normVol = (normVol - minVolume) / Mathf.Max(maxVolume - minVolume, 1e-4f); + normVol = Mathf.Clamp(normVol, 0f, 1f); + } +#if UNITY_EDITOR + _volume = SmoothDamp(_volume, normVol, ref _openCloseVelocity); +#else + _volume = SmoothDamp(_volume, normVol, ref _openCloseVelocity); +#endif + } + + void UpdateVowels() + { + float sum = 0f; + var ratios = _info.phonemeRatios; + + foreach (var par in parameters) + { + float targetWeight = 0f; + // float targetWeight = (par.phoneme == phoneme) ? 1f : 0f; + if (ratios != null && !string.IsNullOrEmpty(par.phoneme)) + { + ratios.TryGetValue(par.phoneme, out targetWeight); + } + float weightVel = par.weightVelocity; + par.weight = SmoothDamp(par.weight, targetWeight, ref weightVel); + par.weightVelocity = weightVel; + sum += par.weight; + } + + foreach (var par in parameters) + { + par.weight = sum > 0f ? par.weight / sum : 0f; + } + } + + public void ApplyAnimator() + { + if (updateMethod == UpdateMethod.External) + { + OnApplyAnimator(); + } + } + + protected virtual void OnApplyAnimator() + { + if (!animator) + return; + + // use hash + foreach (var par in parameters) + { + if (par.index < 0) + continue; + animator.SetFloat(par.nameHash, 0f); + } + + foreach (var par in parameters) + { + if (par.index < 0) + continue; + float weight = animator.GetFloat(par.nameHash); + weight += par.weight * par.maxWeight * volume; + animator.SetFloat(par.nameHash, weight); + } + } + } +} diff --git a/Assets/uLipSync/Runtime/uLipSyncAnimator.cs.meta b/Assets/uLipSync/Runtime/uLipSyncAnimator.cs.meta new file mode 100644 index 0000000..3188fc1 --- /dev/null +++ b/Assets/uLipSync/Runtime/uLipSyncAnimator.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 00cfff4514653254189b8854e8e0fa7b +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: