using System.Collections.Generic; using System.Collections.Immutable; using System.Linq; using System.Threading.Tasks; using nadena.dev.ndmf.preview; using UnityEngine; namespace nadena.dev.modular_avatar.core.editor { internal class ObjectSwitcherPreview : IRenderFilter { static TogglablePreviewNode EnableNode = TogglablePreviewNode.Create( () => "Object Switcher", qualifiedName: "nadena.dev.modular-avatar/ObjectSwitcherPreview", true ); public IEnumerable GetPreviewControlNodes() { yield return EnableNode; } public bool IsEnabled(ComputeContext context) { return context.Observe(EnableNode.IsEnabled); } public ImmutableList GetTargetGroups(ComputeContext context) { var menuItemPreview = new MenuItemPreviewCondition(context); var allToggles = context.GetComponentsByType(); var objectGroups = new Dictionary.Builder>( new ObjectIdentityComparer()); foreach (var toggle in allToggles) { if (!context.ActiveAndEnabled(toggle)) continue; var mami = context.GetComponent(toggle.gameObject); if (mami != null) if (!menuItemPreview.IsEnabledForPreview(mami)) continue; context.Observe(toggle, t => t.Objects.Select(o => o.Object.referencePath).ToList(), (x, y) => x.SequenceEqual(y) ); if (toggle.Objects == null) continue; var index = -1; foreach (var switched in toggle.Objects) { index++; if (switched.Object == null) continue; var target = context.Observe(toggle, _ => switched.Object.Get(toggle)); if (target == null) continue; if (!objectGroups.TryGetValue(target, out var group)) { group = ImmutableList.CreateBuilder<(ModularAvatarObjectToggle, int)>(); objectGroups[target] = group; } group.Add((toggle, index)); } } var affectedRenderers = objectGroups.Keys .SelectMany(go => context.GetComponentsInChildren(go, true)) // If we have overlapping objects, we need to sort by child to parent, so parent configuration overrides // the child. We do this by simply looking at how many times we observe each renderer. .GroupBy(r => r) .Select(g => g.Key) .ToHashSet(); var renderGroups = new List(); foreach (var r in affectedRenderers) { var shouldEnable = true; var obj = r.gameObject; context.ActiveInHierarchy(obj); // observe path changes & object state changes while (obj != null) { var enableAtNode = obj.activeSelf; var group = objectGroups.GetValueOrDefault(obj); if (group == null && !obj.activeSelf) { // always inactive shouldEnable = false; break; } if (group != null) { var (toggle, index) = group[^1]; enableAtNode = context.Observe(toggle, t => t.Objects[index].Active); } if (!enableAtNode) { shouldEnable = false; break; } obj = obj.transform.parent?.gameObject; } if (shouldEnable != r.gameObject.activeInHierarchy) renderGroups.Add(RenderGroup.For(r).WithData(shouldEnable)); } return renderGroups.ToImmutableList(); } public Task Instantiate(RenderGroup group, IEnumerable<(Renderer, Renderer)> proxyPairs, ComputeContext context) { return Task.FromResult(new Node(group.GetData())); } private class Node : IRenderFilterNode { public RenderAspects WhatChanged => 0; private readonly bool _shouldEnable; public Node(bool shouldEnable) { _shouldEnable = shouldEnable; } public void OnFrame(Renderer original, Renderer proxy) { proxy.gameObject.SetActive(_shouldEnable); } } } }