using System.Collections.Generic; using nadena.dev.modular_avatar.core; using nadena.dev.modular_avatar.core.menu; using NUnit.Framework; using UnityEngine; using VRC.SDK3.Avatars.Components; namespace nadena.dev.modular_avatar.editor.ErrorReporting { internal static class ComponentValidation { /// /// Validates the provided tag component. /// /// /// Null if valid, otherwise a list of configuration errors internal static List CheckComponent(this AvatarTagComponent tagComponent) { switch (tagComponent) { case ModularAvatarBlendshapeSync bs: return CheckInternal(bs); case ModularAvatarBoneProxy bp: return CheckInternal(bp); case ModularAvatarMenuInstaller mi: return CheckInternal(mi); case ModularAvatarMergeAnimator obj: return CheckInternal(obj); case ModularAvatarMergeArmature obj: return CheckInternal(obj); default: return null; } } internal static List ValidateAll(GameObject root) { List logs = new List(); foreach (var component in root.GetComponentsInChildren(true)) { var componentLogs = component.CheckComponent(); if (componentLogs != null) { logs.AddRange(componentLogs); } } return logs; } private static List CheckInternal(ModularAvatarBlendshapeSync bs) { var localMesh = bs.GetComponent(); if (localMesh == null) { return new List {new ErrorLog(ReportLevel.Validation, "validation.blendshape_sync.no_local_renderer", bs)}; } if (localMesh.sharedMesh == null) { return new List {new ErrorLog(ReportLevel.Validation, "validation.blendshape_sync.no_local_mesh", bs)}; } if (bs.Bindings == null || bs.Bindings.Count == 0) { return new List {new ErrorLog(ReportLevel.Validation, "validation.blendshape_sync.no_bindings", bs)}; } List errorLogs = new List(); foreach (var binding in bs.Bindings) { var localShape = string.IsNullOrWhiteSpace(binding.LocalBlendshape) ? binding.Blendshape : binding.LocalBlendshape; if (localMesh.sharedMesh.GetBlendShapeIndex(localShape) == -1) { errorLogs.Add(new ErrorLog(ReportLevel.Validation, "validation.blendshape_sync.missing_local_shape", new string[] {localShape}, bs)); } var targetObj = binding.ReferenceMesh.Get(bs.transform); if (targetObj == null) { errorLogs.Add(new ErrorLog(ReportLevel.Validation, "validation.blendshape_sync.no_target", bs)); continue; } var targetRenderer = targetObj.GetComponent(); if (targetRenderer == null) { errorLogs.Add(new ErrorLog(ReportLevel.Validation, "validation.blendshape_sync.missing_target_renderer", bs, targetRenderer)); continue; } var targetMesh = targetRenderer.sharedMesh; if (targetMesh == null) { errorLogs.Add(new ErrorLog(ReportLevel.Validation, "validation.blendshape_sync.missing_target_mesh", bs, targetRenderer)); continue; } if (targetMesh.GetBlendShapeIndex(binding.Blendshape) == -1) { errorLogs.Add(new ErrorLog(ReportLevel.Validation, "validation.blendshape_sync.missing_target_shape", new string[] {binding.Blendshape}, bs, targetRenderer)); } } if (errorLogs.Count == 0) { return null; } else { return errorLogs; } } private static List CheckInternal(ModularAvatarBoneProxy bp) { if (bp.target == null) { return new List() { new ErrorLog(ReportLevel.Validation, "validation.bone_proxy.no_target", bp) }; } return null; } private static List CheckInternal(ModularAvatarMenuInstaller mi) { // TODO - check that target menu is in the avatar if (mi.menuToAppend == null && mi.GetComponent() == null) { return new List() { new ErrorLog(ReportLevel.Validation, "validation.menu_installer.no_menu", mi) }; } return null; } private static List CheckInternal(ModularAvatarMergeAnimator ma) { if (ma.animator == null) { return new List() { new ErrorLog(ReportLevel.Validation, "validation.merge_animator.no_animator", ma) }; } return null; } private static List CheckInternal(ModularAvatarMergeArmature ma) { if (ma.mergeTargetObject == null) { return new List() { new ErrorLog(ReportLevel.Validation, "validation.merge_armature.no_target", ma) }; } if (ma.mergeTargetObject == ma.gameObject || ma.mergeTargetObject.transform.IsChildOf(ma.transform)) { return new List() { new ErrorLog(ReportLevel.Validation, "error.merge_armature.merge_into_self", ma, ma.mergeTargetObject) }; } return null; } } }