NodeUtils.cs 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Globalization;
  4. using System.Linq;
  5. using System.Text;
  6. using UnityEditor.ShaderGraph;
  7. using UnityEditor.ShaderGraph.Drawing;
  8. using UnityEngine;
  9. using UnityEngine.Rendering.ShaderGraph;
  10. namespace UnityEditor.Graphing
  11. {
  12. class SlotConfigurationException : Exception
  13. {
  14. public SlotConfigurationException(string message)
  15. : base(message)
  16. {}
  17. }
  18. static class NodeUtils
  19. {
  20. static string NodeDocSuffix = "-Node";
  21. public static void SlotConfigurationExceptionIfBadConfiguration(AbstractMaterialNode node, IEnumerable<int> expectedInputSlots, IEnumerable<int> expectedOutputSlots)
  22. {
  23. var missingSlots = new List<int>();
  24. var inputSlots = expectedInputSlots as IList<int> ?? expectedInputSlots.ToList();
  25. missingSlots.AddRange(inputSlots.Except(node.GetInputSlots<ISlot>().Select(x => x.id)));
  26. var outputSlots = expectedOutputSlots as IList<int> ?? expectedOutputSlots.ToList();
  27. missingSlots.AddRange(outputSlots.Except(node.GetOutputSlots<ISlot>().Select(x => x.id)));
  28. if (missingSlots.Count == 0)
  29. return;
  30. var toPrint = missingSlots.Select(x => x.ToString());
  31. throw new SlotConfigurationException(string.Format("Missing slots {0} on node {1}", string.Join(", ", toPrint.ToArray()), node));
  32. }
  33. public static IEnumerable<IEdge> GetAllEdges(AbstractMaterialNode node)
  34. {
  35. var result = new List<IEdge>();
  36. var validSlots = ListPool<ISlot>.Get();
  37. validSlots.AddRange(node.GetInputSlots<ISlot>());
  38. for (int index = 0; index < validSlots.Count; index++)
  39. {
  40. var inputSlot = validSlots[index];
  41. result.AddRange(node.owner.GetEdges(inputSlot.slotReference));
  42. }
  43. validSlots.Clear();
  44. validSlots.AddRange(node.GetOutputSlots<ISlot>());
  45. for (int index = 0; index < validSlots.Count; index++)
  46. {
  47. var outputSlot = validSlots[index];
  48. result.AddRange(node.owner.GetEdges(outputSlot.slotReference));
  49. }
  50. ListPool<ISlot>.Release(validSlots);
  51. return result;
  52. }
  53. public static string GetDuplicateSafeNameForSlot(AbstractMaterialNode node, int slotId, string name)
  54. {
  55. List<MaterialSlot> slots = new List<MaterialSlot>();
  56. node.GetSlots(slots);
  57. name = name.Trim();
  58. return GraphUtil.SanitizeName(slots.Where(p => p.id != slotId).Select(p => p.RawDisplayName()), "{0} ({1})", name);
  59. }
  60. // CollectNodesNodeFeedsInto looks at the current node and calculates
  61. // which child nodes it depends on for it's calculation.
  62. // Results are returned depth first so by processing each node in
  63. // order you can generate a valid code block.
  64. public enum IncludeSelf
  65. {
  66. Include,
  67. Exclude
  68. }
  69. public static void DepthFirstCollectNodesFromNode(List<AbstractMaterialNode> nodeList, AbstractMaterialNode node,
  70. IncludeSelf includeSelf = IncludeSelf.Include, IEnumerable<int> slotIds = null, List<KeyValuePair<ShaderKeyword, int>> keywordPermutation = null)
  71. {
  72. // no where to start
  73. if (node == null)
  74. return;
  75. // already added this node
  76. if (nodeList.Contains(node))
  77. return;
  78. IEnumerable<int> ids;
  79. // If this node is a keyword node and we have an active keyword permutation
  80. // The only valid port id is the port that corresponds to that keywords value in the active permutation
  81. if(node is KeywordNode keywordNode && keywordPermutation != null)
  82. {
  83. var valueInPermutation = keywordPermutation.Where(x => x.Key.guid == keywordNode.keywordGuid).FirstOrDefault();
  84. ids = new int[] { keywordNode.GetSlotIdForPermutation(valueInPermutation) };
  85. }
  86. else if (slotIds == null)
  87. {
  88. ids = node.GetInputSlots<ISlot>().Select(x => x.id);
  89. }
  90. else
  91. {
  92. ids = node.GetInputSlots<ISlot>().Where(x => slotIds.Contains(x.id)).Select(x => x.id);
  93. }
  94. foreach (var slot in ids)
  95. {
  96. foreach (var edge in node.owner.GetEdges(node.GetSlotReference(slot)))
  97. {
  98. var outputNode = node.owner.GetNodeFromGuid(edge.outputSlot.nodeGuid);
  99. if (outputNode != null)
  100. DepthFirstCollectNodesFromNode(nodeList, outputNode, keywordPermutation: keywordPermutation);
  101. }
  102. }
  103. if (includeSelf == IncludeSelf.Include)
  104. nodeList.Add(node);
  105. }
  106. public static void CollectNodeSet(HashSet<AbstractMaterialNode> nodeSet, MaterialSlot slot)
  107. {
  108. var node = slot.owner;
  109. var graph = node.owner;
  110. foreach (var edge in graph.GetEdges(node.GetSlotReference(slot.id)))
  111. {
  112. var outputNode = graph.GetNodeFromGuid(edge.outputSlot.nodeGuid);
  113. if (outputNode != null)
  114. {
  115. CollectNodeSet(nodeSet, outputNode);
  116. }
  117. }
  118. }
  119. public static void CollectNodeSet(HashSet<AbstractMaterialNode> nodeSet, AbstractMaterialNode node)
  120. {
  121. if (!nodeSet.Add(node))
  122. {
  123. return;
  124. }
  125. using (var slotsHandle = ListPool<MaterialSlot>.GetDisposable())
  126. {
  127. var slots = slotsHandle.value;
  128. node.GetInputSlots(slots);
  129. foreach (var slot in slots)
  130. {
  131. CollectNodeSet(nodeSet, slot);
  132. }
  133. }
  134. }
  135. public static void CollectNodesNodeFeedsInto(List<AbstractMaterialNode> nodeList, AbstractMaterialNode node, IncludeSelf includeSelf = IncludeSelf.Include)
  136. {
  137. if (node == null)
  138. return;
  139. if (nodeList.Contains(node))
  140. return;
  141. foreach (var slot in node.GetOutputSlots<ISlot>())
  142. {
  143. foreach (var edge in node.owner.GetEdges(slot.slotReference))
  144. {
  145. var inputNode = node.owner.GetNodeFromGuid(edge.inputSlot.nodeGuid);
  146. CollectNodesNodeFeedsInto(nodeList, inputNode);
  147. }
  148. }
  149. if (includeSelf == IncludeSelf.Include)
  150. nodeList.Add(node);
  151. }
  152. public static string GetDocumentationString(string pageName)
  153. {
  154. return Documentation.GetPageLink(pageName.Replace(" ", "-") + NodeDocSuffix);
  155. }
  156. static Stack<MaterialSlot> s_SlotStack = new Stack<MaterialSlot>();
  157. public static ShaderStage GetEffectiveShaderStage(MaterialSlot initialSlot, bool goingBackwards)
  158. {
  159. var graph = initialSlot.owner.owner;
  160. s_SlotStack.Clear();
  161. s_SlotStack.Push(initialSlot);
  162. while (s_SlotStack.Any())
  163. {
  164. var slot = s_SlotStack.Pop();
  165. ShaderStage stage;
  166. if (slot.stageCapability.TryGetShaderStage(out stage))
  167. return stage;
  168. if (goingBackwards && slot.isInputSlot)
  169. {
  170. foreach (var edge in graph.GetEdges(slot.slotReference))
  171. {
  172. var node = graph.GetNodeFromGuid(edge.outputSlot.nodeGuid);
  173. s_SlotStack.Push(node.FindOutputSlot<MaterialSlot>(edge.outputSlot.slotId));
  174. }
  175. }
  176. else if (!goingBackwards && slot.isOutputSlot)
  177. {
  178. foreach (var edge in graph.GetEdges(slot.slotReference))
  179. {
  180. var node = graph.GetNodeFromGuid(edge.inputSlot.nodeGuid);
  181. s_SlotStack.Push(node.FindInputSlot<MaterialSlot>(edge.inputSlot.slotId));
  182. }
  183. }
  184. else
  185. {
  186. var ownerSlots = Enumerable.Empty<MaterialSlot>();
  187. if (goingBackwards && slot.isOutputSlot)
  188. ownerSlots = slot.owner.GetInputSlots<MaterialSlot>();
  189. else if (!goingBackwards && slot.isInputSlot)
  190. ownerSlots = slot.owner.GetOutputSlots<MaterialSlot>();
  191. foreach (var ownerSlot in ownerSlots)
  192. s_SlotStack.Push(ownerSlot);
  193. }
  194. }
  195. // We default to fragment shader stage if all connected nodes were compatible with both.
  196. return ShaderStage.Fragment;
  197. }
  198. public static ShaderStageCapability GetEffectiveShaderStageCapability(MaterialSlot initialSlot, bool goingBackwards)
  199. {
  200. var graph = initialSlot.owner.owner;
  201. s_SlotStack.Clear();
  202. s_SlotStack.Push(initialSlot);
  203. while (s_SlotStack.Any())
  204. {
  205. var slot = s_SlotStack.Pop();
  206. ShaderStage stage;
  207. if (slot.stageCapability.TryGetShaderStage(out stage))
  208. return slot.stageCapability;
  209. if (goingBackwards && slot.isInputSlot)
  210. {
  211. foreach (var edge in graph.GetEdges(slot.slotReference))
  212. {
  213. var node = graph.GetNodeFromGuid(edge.outputSlot.nodeGuid);
  214. s_SlotStack.Push(node.FindOutputSlot<MaterialSlot>(edge.outputSlot.slotId));
  215. }
  216. }
  217. else if (!goingBackwards && slot.isOutputSlot)
  218. {
  219. foreach (var edge in graph.GetEdges(slot.slotReference))
  220. {
  221. var node = graph.GetNodeFromGuid(edge.inputSlot.nodeGuid);
  222. s_SlotStack.Push(node.FindInputSlot<MaterialSlot>(edge.inputSlot.slotId));
  223. }
  224. }
  225. else
  226. {
  227. var ownerSlots = Enumerable.Empty<MaterialSlot>();
  228. if (goingBackwards && slot.isOutputSlot)
  229. ownerSlots = slot.owner.GetInputSlots<MaterialSlot>();
  230. else if (!goingBackwards && slot.isInputSlot)
  231. ownerSlots = slot.owner.GetOutputSlots<MaterialSlot>();
  232. foreach (var ownerSlot in ownerSlots)
  233. s_SlotStack.Push(ownerSlot);
  234. }
  235. }
  236. return ShaderStageCapability.All;
  237. }
  238. public static string GetSlotDimension(ConcreteSlotValueType slotValue)
  239. {
  240. switch (slotValue)
  241. {
  242. case ConcreteSlotValueType.Vector1:
  243. return String.Empty;
  244. case ConcreteSlotValueType.Vector2:
  245. return "2";
  246. case ConcreteSlotValueType.Vector3:
  247. return "3";
  248. case ConcreteSlotValueType.Vector4:
  249. return "4";
  250. case ConcreteSlotValueType.Matrix2:
  251. return "2x2";
  252. case ConcreteSlotValueType.Matrix3:
  253. return "3x3";
  254. case ConcreteSlotValueType.Matrix4:
  255. return "4x4";
  256. default:
  257. return "Error";
  258. }
  259. }
  260. public static string GetHLSLSafeName(string input)
  261. {
  262. char[] arr = input.ToCharArray();
  263. arr = Array.FindAll<char>(arr, (c => (Char.IsLetterOrDigit(c))));
  264. var safeName = new string(arr);
  265. if (safeName.Length > 1 && char.IsDigit(safeName[0]))
  266. {
  267. safeName = $"var{safeName}";
  268. }
  269. return safeName;
  270. }
  271. private static string GetDisplaySafeName(string input)
  272. {
  273. //strip valid display characters from slot name
  274. //current valid characters are whitespace and ( ) _ separators
  275. StringBuilder cleanName = new StringBuilder();
  276. foreach (var c in input)
  277. {
  278. if (c != ' ' && c != '(' && c != ')' && c != '_')
  279. cleanName.Append(c);
  280. }
  281. return cleanName.ToString();
  282. }
  283. public static bool ValidateSlotName(string inName, out string errorMessage)
  284. {
  285. //check for invalid characters between display safe and hlsl safe name
  286. if (GetDisplaySafeName(inName) != GetHLSLSafeName(inName))
  287. {
  288. errorMessage = "Slot name(s) found invalid character(s). Valid characters: A-Z, a-z, 0-9, _ ( ) ";
  289. return true;
  290. }
  291. //if clean, return null and false
  292. errorMessage = null;
  293. return false;
  294. }
  295. public static string FloatToShaderValue(float value)
  296. {
  297. if (Single.IsPositiveInfinity(value))
  298. return "1.#INF";
  299. else if (Single.IsNegativeInfinity(value))
  300. return "-1.#INF";
  301. else if (Single.IsNaN(value))
  302. return "NAN";
  303. else
  304. {
  305. return value.ToString(CultureInfo.InvariantCulture);
  306. }
  307. }
  308. }
  309. }