SearchWindowProvider.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Text;
  5. using UnityEditor.Graphing;
  6. using UnityEditor.Graphing.Util;
  7. using UnityEngine;
  8. using UnityEditor.UIElements;
  9. using UnityEditor.Experimental.GraphView;
  10. using UnityEngine.UIElements;
  11. namespace UnityEditor.ShaderGraph.Drawing
  12. {
  13. class SearchWindowProvider : ScriptableObject, ISearchWindowProvider
  14. {
  15. EditorWindow m_EditorWindow;
  16. GraphData m_Graph;
  17. GraphView m_GraphView;
  18. Texture2D m_Icon;
  19. public ShaderPort connectedPort { get; set; }
  20. public bool nodeNeedsRepositioning { get; set; }
  21. public SlotReference targetSlotReference { get; private set; }
  22. public Vector2 targetPosition { get; private set; }
  23. private const string k_HiddenFolderName = "Hidden";
  24. public void Initialize(EditorWindow editorWindow, GraphData graph, GraphView graphView)
  25. {
  26. m_EditorWindow = editorWindow;
  27. m_Graph = graph;
  28. m_GraphView = graphView;
  29. // Transparent icon to trick search window into indenting items
  30. m_Icon = new Texture2D(1, 1);
  31. m_Icon.SetPixel(0, 0, new Color(0, 0, 0, 0));
  32. m_Icon.Apply();
  33. }
  34. void OnDestroy()
  35. {
  36. if (m_Icon != null)
  37. {
  38. DestroyImmediate(m_Icon);
  39. m_Icon = null;
  40. }
  41. }
  42. struct NodeEntry
  43. {
  44. public string[] title;
  45. public AbstractMaterialNode node;
  46. public int compatibleSlotId;
  47. }
  48. List<int> m_Ids;
  49. List<ISlot> m_Slots = new List<ISlot>();
  50. public List<SearchTreeEntry> CreateSearchTree(SearchWindowContext context)
  51. {
  52. // First build up temporary data structure containing group & title as an array of strings (the last one is the actual title) and associated node type.
  53. var nodeEntries = new List<NodeEntry>();
  54. foreach (var assembly in AppDomain.CurrentDomain.GetAssemblies())
  55. {
  56. foreach (var type in assembly.GetTypesOrNothing())
  57. {
  58. if (type.IsClass && !type.IsAbstract && (type.IsSubclassOf(typeof(AbstractMaterialNode)))
  59. && type != typeof(PropertyNode)
  60. && type != typeof(KeywordNode)
  61. && type != typeof(SubGraphNode))
  62. {
  63. var attrs = type.GetCustomAttributes(typeof(TitleAttribute), false) as TitleAttribute[];
  64. if (attrs != null && attrs.Length > 0)
  65. {
  66. var node = (AbstractMaterialNode)Activator.CreateInstance(type);
  67. AddEntries(node, attrs[0].title, nodeEntries);
  68. }
  69. }
  70. }
  71. }
  72. foreach (var guid in AssetDatabase.FindAssets(string.Format("t:{0}", typeof(SubGraphAsset))))
  73. {
  74. var asset = AssetDatabase.LoadAssetAtPath<SubGraphAsset>(AssetDatabase.GUIDToAssetPath(guid));
  75. var node = new SubGraphNode { asset = asset };
  76. var title = asset.path.Split('/').ToList();
  77. if (asset.descendents.Contains(m_Graph.assetGuid) || asset.assetGuid == m_Graph.assetGuid)
  78. {
  79. continue;
  80. }
  81. if (string.IsNullOrEmpty(asset.path))
  82. {
  83. AddEntries(node, new string[1] { asset.name }, nodeEntries);
  84. }
  85. else if (title[0] != k_HiddenFolderName)
  86. {
  87. title.Add(asset.name);
  88. AddEntries(node, title.ToArray(), nodeEntries);
  89. }
  90. }
  91. foreach (var property in m_Graph.properties)
  92. {
  93. var node = new PropertyNode();
  94. node.owner = m_Graph;
  95. node.propertyGuid = property.guid;
  96. node.owner = null;
  97. AddEntries(node, new[] { "Properties", "Property: " + property.displayName }, nodeEntries);
  98. }
  99. foreach (var keyword in m_Graph.keywords)
  100. {
  101. var node = new KeywordNode();
  102. node.owner = m_Graph;
  103. node.keywordGuid = keyword.guid;
  104. node.owner = null;
  105. AddEntries(node, new[] { "Keywords", "Keyword: " + keyword.displayName }, nodeEntries);
  106. }
  107. // Sort the entries lexicographically by group then title with the requirement that items always comes before sub-groups in the same group.
  108. // Example result:
  109. // - Art/BlendMode
  110. // - Art/Adjustments/ColorBalance
  111. // - Art/Adjustments/Contrast
  112. nodeEntries.Sort((entry1, entry2) =>
  113. {
  114. for (var i = 0; i < entry1.title.Length; i++)
  115. {
  116. if (i >= entry2.title.Length)
  117. return 1;
  118. var value = entry1.title[i].CompareTo(entry2.title[i]);
  119. if (value != 0)
  120. {
  121. // Make sure that leaves go before nodes
  122. if (entry1.title.Length != entry2.title.Length && (i == entry1.title.Length - 1 || i == entry2.title.Length - 1))
  123. return entry1.title.Length < entry2.title.Length ? -1 : 1;
  124. return value;
  125. }
  126. }
  127. return 0;
  128. });
  129. //* Build up the data structure needed by SearchWindow.
  130. // `groups` contains the current group path we're in.
  131. var groups = new List<string>();
  132. // First item in the tree is the title of the window.
  133. var tree = new List<SearchTreeEntry>
  134. {
  135. new SearchTreeGroupEntry(new GUIContent("Create Node"), 0),
  136. };
  137. foreach (var nodeEntry in nodeEntries)
  138. {
  139. // `createIndex` represents from where we should add new group entries from the current entry's group path.
  140. var createIndex = int.MaxValue;
  141. // Compare the group path of the current entry to the current group path.
  142. for (var i = 0; i < nodeEntry.title.Length - 1; i++)
  143. {
  144. var group = nodeEntry.title[i];
  145. if (i >= groups.Count)
  146. {
  147. // The current group path matches a prefix of the current entry's group path, so we add the
  148. // rest of the group path from the currrent entry.
  149. createIndex = i;
  150. break;
  151. }
  152. if (groups[i] != group)
  153. {
  154. // A prefix of the current group path matches a prefix of the current entry's group path,
  155. // so we remove everyfrom from the point where it doesn't match anymore, and then add the rest
  156. // of the group path from the current entry.
  157. groups.RemoveRange(i, groups.Count - i);
  158. createIndex = i;
  159. break;
  160. }
  161. }
  162. // Create new group entries as needed.
  163. // If we don't need to modify the group path, `createIndex` will be `int.MaxValue` and thus the loop won't run.
  164. for (var i = createIndex; i < nodeEntry.title.Length - 1; i++)
  165. {
  166. var group = nodeEntry.title[i];
  167. groups.Add(group);
  168. tree.Add(new SearchTreeGroupEntry(new GUIContent(group)) { level = i + 1 });
  169. }
  170. // Finally, add the actual entry.
  171. tree.Add(new SearchTreeEntry(new GUIContent(nodeEntry.title.Last(), m_Icon)) { level = nodeEntry.title.Length, userData = nodeEntry });
  172. }
  173. return tree;
  174. }
  175. void AddEntries(AbstractMaterialNode node, string[] title, List<NodeEntry> nodeEntries)
  176. {
  177. if (m_Graph.isSubGraph && !node.allowedInSubGraph)
  178. return;
  179. if (!m_Graph.isSubGraph && !node.allowedInMainGraph)
  180. return;
  181. if (connectedPort == null)
  182. {
  183. nodeEntries.Add(new NodeEntry
  184. {
  185. node = node,
  186. title = title,
  187. compatibleSlotId = -1
  188. });
  189. return;
  190. }
  191. var connectedSlot = connectedPort.slot;
  192. m_Slots.Clear();
  193. node.GetSlots(m_Slots);
  194. var hasSingleSlot = m_Slots.Count(s => s.isOutputSlot != connectedSlot.isOutputSlot) == 1;
  195. m_Slots.RemoveAll(slot =>
  196. {
  197. var materialSlot = (MaterialSlot)slot;
  198. return !materialSlot.IsCompatibleWith(connectedSlot);
  199. });
  200. m_Slots.RemoveAll(slot =>
  201. {
  202. var materialSlot = (MaterialSlot)slot;
  203. return !materialSlot.IsCompatibleStageWith(connectedSlot);
  204. });
  205. if (hasSingleSlot && m_Slots.Count == 1)
  206. {
  207. nodeEntries.Add(new NodeEntry
  208. {
  209. node = node,
  210. title = title,
  211. compatibleSlotId = m_Slots.First().id
  212. });
  213. return;
  214. }
  215. foreach (var slot in m_Slots)
  216. {
  217. var entryTitle = new string[title.Length];
  218. title.CopyTo(entryTitle, 0);
  219. entryTitle[entryTitle.Length - 1] += ": " + slot.displayName;
  220. nodeEntries.Add(new NodeEntry
  221. {
  222. title = entryTitle,
  223. node = node,
  224. compatibleSlotId = slot.id
  225. });
  226. }
  227. }
  228. public bool OnSelectEntry(SearchTreeEntry entry, SearchWindowContext context)
  229. {
  230. var nodeEntry = (NodeEntry)entry.userData;
  231. var node = nodeEntry.node;
  232. var drawState = node.drawState;
  233. var windowRoot = m_EditorWindow.rootVisualElement;
  234. var windowMousePosition = windowRoot.ChangeCoordinatesTo(windowRoot.parent, context.screenMousePosition - m_EditorWindow.position.position);
  235. var graphMousePosition = m_GraphView.contentViewContainer.WorldToLocal(windowMousePosition);
  236. drawState.position = new Rect(graphMousePosition, Vector2.zero);
  237. node.drawState = drawState;
  238. m_Graph.owner.RegisterCompleteObjectUndo("Add " + node.name);
  239. m_Graph.AddNode(node);
  240. if (connectedPort != null)
  241. {
  242. var connectedSlot = connectedPort.slot;
  243. var connectedSlotReference = connectedSlot.owner.GetSlotReference(connectedSlot.id);
  244. var compatibleSlotReference = node.GetSlotReference(nodeEntry.compatibleSlotId);
  245. var fromReference = connectedSlot.isOutputSlot ? connectedSlotReference : compatibleSlotReference;
  246. var toReference = connectedSlot.isOutputSlot ? compatibleSlotReference : connectedSlotReference;
  247. m_Graph.Connect(fromReference, toReference);
  248. nodeNeedsRepositioning = true;
  249. targetSlotReference = compatibleSlotReference;
  250. targetPosition = graphMousePosition;
  251. }
  252. return true;
  253. }
  254. }
  255. }