123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350 |
- using System;
- using System.Collections.Generic;
- using System.Globalization;
- using System.Linq;
- using System.Text;
- using UnityEditor.ShaderGraph;
- using UnityEditor.ShaderGraph.Drawing;
- using UnityEngine;
- using UnityEngine.Rendering.ShaderGraph;
- namespace UnityEditor.Graphing
- {
- class SlotConfigurationException : Exception
- {
- public SlotConfigurationException(string message)
- : base(message)
- {}
- }
- static class NodeUtils
- {
- static string NodeDocSuffix = "-Node";
- public static void SlotConfigurationExceptionIfBadConfiguration(AbstractMaterialNode node, IEnumerable<int> expectedInputSlots, IEnumerable<int> expectedOutputSlots)
- {
- var missingSlots = new List<int>();
- var inputSlots = expectedInputSlots as IList<int> ?? expectedInputSlots.ToList();
- missingSlots.AddRange(inputSlots.Except(node.GetInputSlots<ISlot>().Select(x => x.id)));
- var outputSlots = expectedOutputSlots as IList<int> ?? expectedOutputSlots.ToList();
- missingSlots.AddRange(outputSlots.Except(node.GetOutputSlots<ISlot>().Select(x => x.id)));
- if (missingSlots.Count == 0)
- return;
- var toPrint = missingSlots.Select(x => x.ToString());
- throw new SlotConfigurationException(string.Format("Missing slots {0} on node {1}", string.Join(", ", toPrint.ToArray()), node));
- }
- public static IEnumerable<IEdge> GetAllEdges(AbstractMaterialNode node)
- {
- var result = new List<IEdge>();
- var validSlots = ListPool<ISlot>.Get();
- validSlots.AddRange(node.GetInputSlots<ISlot>());
- for (int index = 0; index < validSlots.Count; index++)
- {
- var inputSlot = validSlots[index];
- result.AddRange(node.owner.GetEdges(inputSlot.slotReference));
- }
- validSlots.Clear();
- validSlots.AddRange(node.GetOutputSlots<ISlot>());
- for (int index = 0; index < validSlots.Count; index++)
- {
- var outputSlot = validSlots[index];
- result.AddRange(node.owner.GetEdges(outputSlot.slotReference));
- }
- ListPool<ISlot>.Release(validSlots);
- return result;
- }
- public static string GetDuplicateSafeNameForSlot(AbstractMaterialNode node, int slotId, string name)
- {
- List<MaterialSlot> slots = new List<MaterialSlot>();
- node.GetSlots(slots);
- name = name.Trim();
- return GraphUtil.SanitizeName(slots.Where(p => p.id != slotId).Select(p => p.RawDisplayName()), "{0} ({1})", name);
- }
- // CollectNodesNodeFeedsInto looks at the current node and calculates
- // which child nodes it depends on for it's calculation.
- // Results are returned depth first so by processing each node in
- // order you can generate a valid code block.
- public enum IncludeSelf
- {
- Include,
- Exclude
- }
- public static void DepthFirstCollectNodesFromNode(List<AbstractMaterialNode> nodeList, AbstractMaterialNode node,
- IncludeSelf includeSelf = IncludeSelf.Include, IEnumerable<int> slotIds = null, List<KeyValuePair<ShaderKeyword, int>> keywordPermutation = null)
- {
- // no where to start
- if (node == null)
- return;
- // already added this node
- if (nodeList.Contains(node))
- return;
- IEnumerable<int> ids;
- // If this node is a keyword node and we have an active keyword permutation
- // The only valid port id is the port that corresponds to that keywords value in the active permutation
- if(node is KeywordNode keywordNode && keywordPermutation != null)
- {
- var valueInPermutation = keywordPermutation.Where(x => x.Key.guid == keywordNode.keywordGuid).FirstOrDefault();
- ids = new int[] { keywordNode.GetSlotIdForPermutation(valueInPermutation) };
- }
- else if (slotIds == null)
- {
- ids = node.GetInputSlots<ISlot>().Select(x => x.id);
- }
- else
- {
- ids = node.GetInputSlots<ISlot>().Where(x => slotIds.Contains(x.id)).Select(x => x.id);
- }
- foreach (var slot in ids)
- {
- foreach (var edge in node.owner.GetEdges(node.GetSlotReference(slot)))
- {
- var outputNode = node.owner.GetNodeFromGuid(edge.outputSlot.nodeGuid);
- if (outputNode != null)
- DepthFirstCollectNodesFromNode(nodeList, outputNode, keywordPermutation: keywordPermutation);
- }
- }
- if (includeSelf == IncludeSelf.Include)
- nodeList.Add(node);
- }
- public static void CollectNodeSet(HashSet<AbstractMaterialNode> nodeSet, MaterialSlot slot)
- {
- var node = slot.owner;
- var graph = node.owner;
- foreach (var edge in graph.GetEdges(node.GetSlotReference(slot.id)))
- {
- var outputNode = graph.GetNodeFromGuid(edge.outputSlot.nodeGuid);
- if (outputNode != null)
- {
- CollectNodeSet(nodeSet, outputNode);
- }
- }
- }
- public static void CollectNodeSet(HashSet<AbstractMaterialNode> nodeSet, AbstractMaterialNode node)
- {
- if (!nodeSet.Add(node))
- {
- return;
- }
- using (var slotsHandle = ListPool<MaterialSlot>.GetDisposable())
- {
- var slots = slotsHandle.value;
- node.GetInputSlots(slots);
- foreach (var slot in slots)
- {
- CollectNodeSet(nodeSet, slot);
- }
- }
- }
- public static void CollectNodesNodeFeedsInto(List<AbstractMaterialNode> nodeList, AbstractMaterialNode node, IncludeSelf includeSelf = IncludeSelf.Include)
- {
- if (node == null)
- return;
- if (nodeList.Contains(node))
- return;
- foreach (var slot in node.GetOutputSlots<ISlot>())
- {
- foreach (var edge in node.owner.GetEdges(slot.slotReference))
- {
- var inputNode = node.owner.GetNodeFromGuid(edge.inputSlot.nodeGuid);
- CollectNodesNodeFeedsInto(nodeList, inputNode);
- }
- }
- if (includeSelf == IncludeSelf.Include)
- nodeList.Add(node);
- }
- public static string GetDocumentationString(string pageName)
- {
- return Documentation.GetPageLink(pageName.Replace(" ", "-") + NodeDocSuffix);
- }
- static Stack<MaterialSlot> s_SlotStack = new Stack<MaterialSlot>();
- public static ShaderStage GetEffectiveShaderStage(MaterialSlot initialSlot, bool goingBackwards)
- {
- var graph = initialSlot.owner.owner;
- s_SlotStack.Clear();
- s_SlotStack.Push(initialSlot);
- while (s_SlotStack.Any())
- {
- var slot = s_SlotStack.Pop();
- ShaderStage stage;
- if (slot.stageCapability.TryGetShaderStage(out stage))
- return stage;
- if (goingBackwards && slot.isInputSlot)
- {
- foreach (var edge in graph.GetEdges(slot.slotReference))
- {
- var node = graph.GetNodeFromGuid(edge.outputSlot.nodeGuid);
- s_SlotStack.Push(node.FindOutputSlot<MaterialSlot>(edge.outputSlot.slotId));
- }
- }
- else if (!goingBackwards && slot.isOutputSlot)
- {
- foreach (var edge in graph.GetEdges(slot.slotReference))
- {
- var node = graph.GetNodeFromGuid(edge.inputSlot.nodeGuid);
- s_SlotStack.Push(node.FindInputSlot<MaterialSlot>(edge.inputSlot.slotId));
- }
- }
- else
- {
- var ownerSlots = Enumerable.Empty<MaterialSlot>();
- if (goingBackwards && slot.isOutputSlot)
- ownerSlots = slot.owner.GetInputSlots<MaterialSlot>();
- else if (!goingBackwards && slot.isInputSlot)
- ownerSlots = slot.owner.GetOutputSlots<MaterialSlot>();
- foreach (var ownerSlot in ownerSlots)
- s_SlotStack.Push(ownerSlot);
- }
- }
- // We default to fragment shader stage if all connected nodes were compatible with both.
- return ShaderStage.Fragment;
- }
- public static ShaderStageCapability GetEffectiveShaderStageCapability(MaterialSlot initialSlot, bool goingBackwards)
- {
- var graph = initialSlot.owner.owner;
- s_SlotStack.Clear();
- s_SlotStack.Push(initialSlot);
- while (s_SlotStack.Any())
- {
- var slot = s_SlotStack.Pop();
- ShaderStage stage;
- if (slot.stageCapability.TryGetShaderStage(out stage))
- return slot.stageCapability;
- if (goingBackwards && slot.isInputSlot)
- {
- foreach (var edge in graph.GetEdges(slot.slotReference))
- {
- var node = graph.GetNodeFromGuid(edge.outputSlot.nodeGuid);
- s_SlotStack.Push(node.FindOutputSlot<MaterialSlot>(edge.outputSlot.slotId));
- }
- }
- else if (!goingBackwards && slot.isOutputSlot)
- {
- foreach (var edge in graph.GetEdges(slot.slotReference))
- {
- var node = graph.GetNodeFromGuid(edge.inputSlot.nodeGuid);
- s_SlotStack.Push(node.FindInputSlot<MaterialSlot>(edge.inputSlot.slotId));
- }
- }
- else
- {
- var ownerSlots = Enumerable.Empty<MaterialSlot>();
- if (goingBackwards && slot.isOutputSlot)
- ownerSlots = slot.owner.GetInputSlots<MaterialSlot>();
- else if (!goingBackwards && slot.isInputSlot)
- ownerSlots = slot.owner.GetOutputSlots<MaterialSlot>();
- foreach (var ownerSlot in ownerSlots)
- s_SlotStack.Push(ownerSlot);
- }
- }
- return ShaderStageCapability.All;
- }
- public static string GetSlotDimension(ConcreteSlotValueType slotValue)
- {
- switch (slotValue)
- {
- case ConcreteSlotValueType.Vector1:
- return String.Empty;
- case ConcreteSlotValueType.Vector2:
- return "2";
- case ConcreteSlotValueType.Vector3:
- return "3";
- case ConcreteSlotValueType.Vector4:
- return "4";
- case ConcreteSlotValueType.Matrix2:
- return "2x2";
- case ConcreteSlotValueType.Matrix3:
- return "3x3";
- case ConcreteSlotValueType.Matrix4:
- return "4x4";
- default:
- return "Error";
- }
- }
- public static string GetHLSLSafeName(string input)
- {
- char[] arr = input.ToCharArray();
- arr = Array.FindAll<char>(arr, (c => (Char.IsLetterOrDigit(c))));
- var safeName = new string(arr);
- if (safeName.Length > 1 && char.IsDigit(safeName[0]))
- {
- safeName = $"var{safeName}";
- }
- return safeName;
- }
- private static string GetDisplaySafeName(string input)
- {
- //strip valid display characters from slot name
- //current valid characters are whitespace and ( ) _ separators
- StringBuilder cleanName = new StringBuilder();
- foreach (var c in input)
- {
- if (c != ' ' && c != '(' && c != ')' && c != '_')
- cleanName.Append(c);
- }
- return cleanName.ToString();
- }
- public static bool ValidateSlotName(string inName, out string errorMessage)
- {
- //check for invalid characters between display safe and hlsl safe name
- if (GetDisplaySafeName(inName) != GetHLSLSafeName(inName))
- {
- errorMessage = "Slot name(s) found invalid character(s). Valid characters: A-Z, a-z, 0-9, _ ( ) ";
- return true;
- }
- //if clean, return null and false
- errorMessage = null;
- return false;
- }
- public static string FloatToShaderValue(float value)
- {
- if (Single.IsPositiveInfinity(value))
- return "1.#INF";
- else if (Single.IsNegativeInfinity(value))
- return "-1.#INF";
- else if (Single.IsNaN(value))
- return "NAN";
- else
- {
- return value.ToString(CultureInfo.InvariantCulture);
- }
- }
- }
- }
|