using System.Collections.Generic;
using System.Linq;
using nadena.dev.ndmf;
using UnityEngine;
using VRC.SDK3.Avatars.ScriptableObjects;
namespace nadena.dev.modular_avatar.core.editor
{
///
/// Creates/allocates parameters to any Menu Items that need them.
///
internal class ParameterAssignerPass : Pass
{
internal static bool ShouldAssignParametersToMami(ModularAvatarMenuItem item)
{
switch (item?.Control?.type)
{
case VRCExpressionsMenu.Control.ControlType.Button:
case VRCExpressionsMenu.Control.ControlType.Toggle:
// ok
break;
default:
return false;
}
foreach (var rc in item.GetComponentsInChildren(true))
{
// Only track components where this is the closest parent
if (rc.transform == item.transform)
{
return true;
}
var parentMami = rc.GetComponentInParent();
if (parentMami == item)
{
return true;
}
}
return false;
}
protected override void Execute(ndmf.BuildContext context)
{
if (!context.AvatarDescriptor) return;
var paramIndex = 0;
var declaredParams = context.AvatarDescriptor.expressionParameters.parameters
.GroupBy(p => p.name).Select(l => l.First())
.ToDictionary(p => p.name);
Dictionary newParameters = new();
Dictionary nextParamValue = new();
Dictionary> _mamiByParam = new();
foreach (var mami in context.AvatarRootTransform.GetComponentsInChildren(true))
{
if (string.IsNullOrWhiteSpace(mami.Control?.parameter?.name))
{
if (!ShouldAssignParametersToMami(mami)) continue;
if (mami.Control == null) mami.Control = new VRCExpressionsMenu.Control();
mami.Control.parameter = new VRCExpressionsMenu.Control.Parameter
{
name = $"__MA/AutoParam/{mami.gameObject.name}${paramIndex++}"
};
}
var paramName = mami.Control.parameter.name;
if (!_mamiByParam.TryGetValue(paramName, out var mamiList))
{
mamiList = new List();
_mamiByParam[paramName] = mamiList;
}
mamiList.Add(mami);
}
foreach (var (paramName, list) in _mamiByParam)
{
// Assign automatic values first
float defaultValue;
if (declaredParams.TryGetValue(paramName, out var p))
{
defaultValue = p.defaultValue;
}
else
{
defaultValue = list.FirstOrDefault(m => m.isDefault && !m.automaticValue)?.Control?.value ?? 0;
if (list.Count == 1)
// If we have only a single entry, it's probably an on-off toggle, so we'll implicitly let 0
// be the 'unselected' default value
defaultValue = 1;
}
HashSet usedValues = new();
usedValues.Add((int)defaultValue);
foreach (var item in list)
if (!item.automaticValue && Mathf.Abs(item.Control.value - Mathf.Round(item.Control.value)) < 0.01f)
usedValues.Add(Mathf.RoundToInt(item.Control.value));
var nextValue = 1;
var canBeBool = true;
var canBeInt = true;
var isSaved = true;
var isSynced = true;
foreach (var mami in list)
{
if (mami.automaticValue)
{
if (mami.isDefault)
{
mami.Control.value = defaultValue;
}
else
{
while (usedValues.Contains(nextValue)) nextValue++;
mami.Control.value = nextValue;
usedValues.Add(nextValue);
}
}
if (Mathf.Abs(mami.Control.value - Mathf.Round(mami.Control.value)) > 0.01f)
canBeInt = false;
else
canBeBool &= mami.Control.value is >= 0 and <= 1;
isSaved &= mami.isSaved;
isSynced &= mami.isSynced;
}
if (!declaredParams.ContainsKey(paramName))
{
VRCExpressionParameters.ValueType newType;
if (canBeBool) newType = VRCExpressionParameters.ValueType.Bool;
else if (canBeInt) newType = VRCExpressionParameters.ValueType.Int;
else newType = VRCExpressionParameters.ValueType.Float;
var newParam = new VRCExpressionParameters.Parameter
{
name = paramName,
valueType = newType,
saved = isSaved,
defaultValue = defaultValue,
networkSynced = isSynced
};
newParameters[paramName] = newParam;
}
}
if (newParameters.Count > 0)
{
var expParams = context.AvatarDescriptor.expressionParameters;
if (!context.IsTemporaryAsset(expParams))
{
expParams = Object.Instantiate(expParams);
context.AvatarDescriptor.expressionParameters = expParams;
}
expParams.parameters = expParams.parameters.Concat(newParameters.Values).ToArray();
}
}
internal static ControlCondition AssignMenuItemParameter(
ModularAvatarMenuItem mami,
Dictionary simulationInitialStates = null,
IDictionary isDefaultOverrides = null)
{
var paramName = mami?.Control?.parameter?.name;
if (mami?.Control != null && simulationInitialStates != null && ShouldAssignParametersToMami(mami))
{
paramName = mami.Control?.parameter?.name;
if (string.IsNullOrEmpty(paramName)) paramName = "___AutoProp/" + mami.GetInstanceID();
var isDefault = mami.isDefault;
if (isDefaultOverrides?.TryGetValue(paramName, out var target) == true)
isDefault = ReferenceEquals(mami, target);
if (isDefault)
{
simulationInitialStates[paramName] = mami.Control.value;
} else if (!simulationInitialStates.ContainsKey(paramName))
{
simulationInitialStates[paramName] = -999;
}
}
if (string.IsNullOrWhiteSpace(paramName)) return null;
return new ControlCondition
{
Parameter = paramName,
DebugName = mami.gameObject.name,
IsConstant = false,
// Note: This slightly odd-looking value is key to making the Auto checkbox work for editor previews;
// we basically force-disable any conditions for nonselected menu items and force-enable any for default
// menu items.
InitialValue = mami.isDefault ? mami.Control.value : -999,
ParameterValueLo = mami.Control.value - 0.5f,
ParameterValueHi = mami.Control.value + 0.5f,
DebugReference = mami,
};
}
}
}