mirror of
https://github.com/bdunderscore/modular-avatar.git
synced 2025-01-19 21:00:08 +08:00
8418f8e047
* feat: add target renderer to ChangedShape * chore: add test for ShapeChanger target renderer * feat: add override target to MaterialSetter * fix: resolve added AvatarObjectReference * fix: record prefab instance property modifications * refactor: remove unused setter for AvatarObjectReference * refactor: change ChangedShape and MaterialSwitchObject from struct to class * feat: remove override target from ShapeChanger and MaterialSetter * refactor: align flow and code style of ShapeChanger and MaterialSetter * feat: ShapeChanger target migration * fix: add null check * chore: added some comments and nullchecks --------- Co-authored-by: bd_ <bd_@nadena.dev>
157 lines
6.0 KiB
C#
157 lines
6.0 KiB
C#
using System.Collections.Generic;
|
|
using nadena.dev.modular_avatar.animation;
|
|
using UnityEngine;
|
|
|
|
namespace nadena.dev.modular_avatar.core.editor
|
|
{
|
|
partial class ReactiveObjectAnalyzer
|
|
{
|
|
|
|
private Dictionary<TargetProp, AnimatedProperty> FindShapes(GameObject root)
|
|
{
|
|
var changers = root.GetComponentsInChildren<ModularAvatarShapeChanger>(true);
|
|
|
|
Dictionary<TargetProp, AnimatedProperty> shapeKeys = new();
|
|
|
|
foreach (var changer in changers)
|
|
{
|
|
if (changer.Shapes == null) continue;
|
|
|
|
foreach (var shape in changer.Shapes)
|
|
{
|
|
var renderer = shape.Object.Get(changer)?.GetComponent<SkinnedMeshRenderer>();
|
|
if (renderer == null) continue;
|
|
|
|
var mesh = renderer.sharedMesh;
|
|
if (mesh == null) continue;
|
|
|
|
var shapeId = mesh.GetBlendShapeIndex(shape.ShapeName);
|
|
if (shapeId < 0) continue;
|
|
|
|
var key = new TargetProp
|
|
{
|
|
TargetObject = renderer,
|
|
PropertyName = "blendShape." + shape.ShapeName,
|
|
};
|
|
|
|
var value = shape.ChangeType == ShapeChangeType.Delete ? 100 : shape.Value;
|
|
if (!shapeKeys.TryGetValue(key, out var info))
|
|
{
|
|
info = new AnimatedProperty(key, renderer.GetBlendShapeWeight(shapeId));
|
|
shapeKeys[key] = info;
|
|
|
|
// Add initial state
|
|
var agk = new ReactionRule(context, key, null, value);
|
|
agk.Value = renderer.GetBlendShapeWeight(shapeId);
|
|
info.actionGroups.Add(agk);
|
|
}
|
|
|
|
var action = new ReactionRule(context, key, changer.gameObject, value);
|
|
action.Inverted = changer.Inverted;
|
|
var isCurrentlyActive = changer.gameObject.activeInHierarchy;
|
|
|
|
if (shape.ChangeType == ShapeChangeType.Delete)
|
|
{
|
|
action.IsDelete = true;
|
|
|
|
if (isCurrentlyActive) info.currentState = 100;
|
|
|
|
info.actionGroups.Add(action); // Never merge
|
|
|
|
continue;
|
|
}
|
|
|
|
if (changer.gameObject.activeInHierarchy) info.currentState = action.Value;
|
|
|
|
Debug.Log("Trying merge: " + action);
|
|
if (info.actionGroups.Count == 0)
|
|
{
|
|
info.actionGroups.Add(action);
|
|
}
|
|
else if (!info.actionGroups[^1].TryMerge(action))
|
|
{
|
|
Debug.Log("Failed merge");
|
|
info.actionGroups.Add(action);
|
|
}
|
|
else
|
|
{
|
|
Debug.Log("Post merge: " + info.actionGroups[^1]);
|
|
}
|
|
}
|
|
}
|
|
|
|
return shapeKeys;
|
|
}
|
|
|
|
private void FindMaterialSetters(Dictionary<TargetProp, AnimatedProperty> objectGroups, GameObject root)
|
|
{
|
|
var materialSetters = root.GetComponentsInChildren<ModularAvatarMaterialSetter>(true);
|
|
|
|
foreach (var setter in materialSetters)
|
|
{
|
|
if (setter.Objects == null) continue;
|
|
|
|
foreach (var obj in setter.Objects)
|
|
{
|
|
var renderer = obj.Object.Get(setter)?.GetComponent<Renderer>();
|
|
if (renderer == null || renderer.sharedMaterials.Length < obj.MaterialIndex) continue;
|
|
|
|
var key = new TargetProp
|
|
{
|
|
TargetObject = renderer,
|
|
PropertyName = "m_Materials.Array.data[" + obj.MaterialIndex + "]",
|
|
};
|
|
|
|
if (!objectGroups.TryGetValue(key, out var group))
|
|
{
|
|
group = new AnimatedProperty(key, renderer.sharedMaterials[obj.MaterialIndex]);
|
|
objectGroups[key] = group;
|
|
}
|
|
|
|
var action = new ReactionRule(context, key, setter.gameObject, obj.Material);
|
|
action.Inverted = setter.Inverted;
|
|
|
|
if (group.actionGroups.Count == 0)
|
|
group.actionGroups.Add(action);
|
|
else if (!group.actionGroups[^1].TryMerge(action)) group.actionGroups.Add(action);
|
|
}
|
|
}
|
|
}
|
|
|
|
private void FindObjectToggles(Dictionary<TargetProp, AnimatedProperty> objectGroups, GameObject root)
|
|
{
|
|
var toggles = root.GetComponentsInChildren<ModularAvatarObjectToggle>(true);
|
|
|
|
foreach (var toggle in toggles)
|
|
{
|
|
if (toggle.Objects == null) continue;
|
|
|
|
foreach (var obj in toggle.Objects)
|
|
{
|
|
var target = obj.Object.Get(toggle);
|
|
if (target == null) continue;
|
|
|
|
var key = new TargetProp
|
|
{
|
|
TargetObject = target,
|
|
PropertyName = "m_IsActive"
|
|
};
|
|
|
|
if (!objectGroups.TryGetValue(key, out var group))
|
|
{
|
|
group = new AnimatedProperty(key, target.activeSelf ? 1 : 0);
|
|
objectGroups[key] = group;
|
|
}
|
|
|
|
var value = obj.Active ? 1 : 0;
|
|
var action = new ReactionRule(context, key, toggle.gameObject, value);
|
|
action.Inverted = toggle.Inverted;
|
|
|
|
if (group.actionGroups.Count == 0)
|
|
group.actionGroups.Add(action);
|
|
else if (!group.actionGroups[^1].TryMerge(action)) group.actionGroups.Add(action);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} |