MaterialGraphEditWindow.cs 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723
  1. using System;
  2. using System.Collections.Generic;
  3. using System.IO;
  4. using System.Linq;
  5. using System.Text;
  6. using UnityEngine;
  7. using UnityEditor.Graphing;
  8. using UnityEditor.Graphing.Util;
  9. using Debug = UnityEngine.Debug;
  10. using Object = UnityEngine.Object;
  11. using UnityEngine.Rendering;
  12. using UnityEditor.UIElements;
  13. using Edge = UnityEditor.Experimental.GraphView.Edge;
  14. using UnityEditor.Experimental.GraphView;
  15. using UnityEditor.ShaderGraph.Internal;
  16. using UnityEngine.UIElements;
  17. using UnityEditor.VersionControl;
  18. namespace UnityEditor.ShaderGraph.Drawing
  19. {
  20. class MaterialGraphEditWindow : EditorWindow
  21. {
  22. [SerializeField]
  23. string m_Selected;
  24. [SerializeField]
  25. GraphObject m_GraphObject;
  26. [NonSerialized]
  27. bool m_HasError;
  28. [NonSerialized]
  29. HashSet<string> m_ChangedFileDependencies = new HashSet<string>();
  30. ColorSpace m_ColorSpace;
  31. RenderPipelineAsset m_RenderPipelineAsset;
  32. bool m_FrameAllAfterLayout;
  33. bool m_ProTheme;
  34. GraphEditorView m_GraphEditorView;
  35. MessageManager m_MessageManager;
  36. MessageManager messageManager
  37. {
  38. get { return m_MessageManager ?? (m_MessageManager = new MessageManager()); }
  39. }
  40. GraphEditorView graphEditorView
  41. {
  42. get { return m_GraphEditorView; }
  43. set
  44. {
  45. if (m_GraphEditorView != null)
  46. {
  47. m_GraphEditorView.RemoveFromHierarchy();
  48. m_GraphEditorView.Dispose();
  49. }
  50. m_GraphEditorView = value;
  51. if (m_GraphEditorView != null)
  52. {
  53. m_GraphEditorView.saveRequested += UpdateAsset;
  54. m_GraphEditorView.convertToSubgraphRequested += ToSubGraph;
  55. m_GraphEditorView.showInProjectRequested += PingAsset;
  56. m_GraphEditorView.isCheckedOut += IsGraphAssetCheckedOut;
  57. m_GraphEditorView.checkOut += CheckoutAsset;
  58. m_GraphEditorView.RegisterCallback<GeometryChangedEvent>(OnGeometryChanged);
  59. m_FrameAllAfterLayout = true;
  60. this.rootVisualElement.Add(graphEditorView);
  61. }
  62. }
  63. }
  64. GraphObject graphObject
  65. {
  66. get { return m_GraphObject; }
  67. set
  68. {
  69. if (m_GraphObject != null)
  70. DestroyImmediate(m_GraphObject);
  71. m_GraphObject = value;
  72. }
  73. }
  74. public string selectedGuid
  75. {
  76. get { return m_Selected; }
  77. private set { m_Selected = value; }
  78. }
  79. public string assetName
  80. {
  81. get { return titleContent.text; }
  82. set
  83. {
  84. titleContent.text = value;
  85. graphEditorView.assetName = value;
  86. }
  87. }
  88. void Update()
  89. {
  90. if (m_HasError)
  91. return;
  92. if (PlayerSettings.colorSpace != m_ColorSpace)
  93. {
  94. graphEditorView = null;
  95. m_ColorSpace = PlayerSettings.colorSpace;
  96. }
  97. if (GraphicsSettings.renderPipelineAsset != m_RenderPipelineAsset)
  98. {
  99. graphEditorView = null;
  100. m_RenderPipelineAsset = GraphicsSettings.renderPipelineAsset;
  101. }
  102. if (EditorGUIUtility.isProSkin != m_ProTheme)
  103. {
  104. if (graphObject != null && graphObject.graph != null)
  105. {
  106. Texture2D icon = GetThemeIcon(graphObject.graph);
  107. // This is adding the icon at the front of the tab
  108. titleContent = EditorGUIUtility.TrTextContentWithIcon(assetName, icon);
  109. m_ProTheme = EditorGUIUtility.isProSkin;
  110. }
  111. }
  112. try
  113. {
  114. if (graphObject == null && selectedGuid != null)
  115. {
  116. var guid = selectedGuid;
  117. selectedGuid = null;
  118. Initialize(guid);
  119. }
  120. if (graphObject == null)
  121. {
  122. Close();
  123. return;
  124. }
  125. var materialGraph = graphObject.graph as GraphData;
  126. if (materialGraph == null)
  127. return;
  128. if (graphEditorView == null)
  129. {
  130. messageManager.ClearAll();
  131. materialGraph.messageManager = messageManager;
  132. var asset = AssetDatabase.LoadAssetAtPath<Object>(AssetDatabase.GUIDToAssetPath(selectedGuid));
  133. graphEditorView = new GraphEditorView(this, materialGraph, messageManager)
  134. {
  135. viewDataKey = selectedGuid,
  136. assetName = asset.name.Split('/').Last()
  137. };
  138. m_ColorSpace = PlayerSettings.colorSpace;
  139. m_RenderPipelineAsset = GraphicsSettings.renderPipelineAsset;
  140. graphObject.Validate();
  141. }
  142. if (m_ChangedFileDependencies.Count > 0 && graphObject != null && graphObject.graph != null)
  143. {
  144. var subGraphNodes = graphObject.graph.GetNodes<SubGraphNode>();
  145. foreach (var subGraphNode in subGraphNodes)
  146. {
  147. subGraphNode.Reload(m_ChangedFileDependencies);
  148. }
  149. if(subGraphNodes.Count() > 0)
  150. {
  151. // Keywords always need to be updated to test against variant limit
  152. // No Keywords may indicate removal and this may have now made the Graph valid again
  153. // Need to validate Graph to clear errors in this case
  154. materialGraph.OnKeywordChanged();
  155. }
  156. foreach (var customFunctionNode in graphObject.graph.GetNodes<CustomFunctionNode>())
  157. {
  158. customFunctionNode.Reload(m_ChangedFileDependencies);
  159. }
  160. m_ChangedFileDependencies.Clear();
  161. }
  162. if (graphObject.wasUndoRedoPerformed)
  163. {
  164. graphEditorView.HandleGraphChanges();
  165. graphObject.graph.ClearChanges();
  166. graphObject.HandleUndoRedo();
  167. }
  168. graphEditorView.HandleGraphChanges();
  169. graphObject.graph.ClearChanges();
  170. }
  171. catch (Exception e)
  172. {
  173. m_HasError = true;
  174. m_GraphEditorView = null;
  175. graphObject = null;
  176. Debug.LogException(e);
  177. throw;
  178. }
  179. }
  180. public void ReloadSubGraphsOnNextUpdate(List<string> changedFiles)
  181. {
  182. foreach (var changedFile in changedFiles)
  183. {
  184. m_ChangedFileDependencies.Add(changedFile);
  185. }
  186. }
  187. void OnEnable()
  188. {
  189. this.SetAntiAliasing(4);
  190. }
  191. void OnDisable()
  192. {
  193. graphEditorView = null;
  194. messageManager.ClearAll();
  195. }
  196. void OnDestroy()
  197. {
  198. if (graphObject != null)
  199. {
  200. string nameOfFile = AssetDatabase.GUIDToAssetPath(selectedGuid);
  201. if (graphObject.isDirty && EditorUtility.DisplayDialog("Shader Graph Has Been Modified", "Do you want to save the changes you made in the Shader Graph?\n" + nameOfFile + "\n\nYour changes will be lost if you don't save them.", "Save", "Don't Save"))
  202. UpdateAsset();
  203. Undo.ClearUndo(graphObject);
  204. graphObject = null;
  205. }
  206. graphEditorView = null;
  207. }
  208. public void PingAsset()
  209. {
  210. if (selectedGuid != null)
  211. {
  212. var path = AssetDatabase.GUIDToAssetPath(selectedGuid);
  213. var asset = AssetDatabase.LoadAssetAtPath<Object>(path);
  214. EditorGUIUtility.PingObject(asset);
  215. }
  216. }
  217. public bool IsGraphAssetCheckedOut()
  218. {
  219. if (selectedGuid != null)
  220. {
  221. var path = AssetDatabase.GUIDToAssetPath(selectedGuid);
  222. var asset = AssetDatabase.LoadAssetAtPath<Object>(path);
  223. if (!AssetDatabase.IsOpenForEdit(asset, StatusQueryOptions.UseCachedIfPossible))
  224. return false;
  225. return true;
  226. }
  227. return false;
  228. }
  229. public void CheckoutAsset()
  230. {
  231. if (selectedGuid != null)
  232. {
  233. var path = AssetDatabase.GUIDToAssetPath(selectedGuid);
  234. var asset = AssetDatabase.LoadAssetAtPath<Object>(path);
  235. Task task = Provider.Checkout(asset, CheckoutMode.Both);
  236. task.Wait();
  237. }
  238. }
  239. public void UpdateAsset()
  240. {
  241. if (selectedGuid != null && graphObject != null)
  242. {
  243. var path = AssetDatabase.GUIDToAssetPath(selectedGuid);
  244. if (string.IsNullOrEmpty(path) || graphObject == null)
  245. return;
  246. var oldShader = AssetDatabase.LoadAssetAtPath<Shader>(path);
  247. if (oldShader != null)
  248. ShaderUtil.ClearShaderMessages(oldShader);
  249. ShaderGraphAnalytics.SendShaderGraphEvent(selectedGuid, graphObject.graph);
  250. UpdateShaderGraphOnDisk(path);
  251. if (GraphData.onSaveGraph != null)
  252. {
  253. var shader = AssetDatabase.LoadAssetAtPath<Shader>(path);
  254. if (shader != null)
  255. {
  256. GraphData.onSaveGraph(shader, (graphObject.graph.outputNode as MasterNode).saveContext);
  257. }
  258. }
  259. graphObject.isDirty = false;
  260. }
  261. }
  262. public void ToSubGraph()
  263. {
  264. var graphView = graphEditorView.graphView;
  265. var path = EditorUtility.SaveFilePanelInProject("Save Sub Graph", "New Shader Sub Graph", ShaderSubGraphImporter.Extension, "");
  266. path = path.Replace(Application.dataPath, "Assets");
  267. if (path.Length == 0)
  268. return;
  269. graphObject.RegisterCompleteObjectUndo("Convert To Subgraph");
  270. var nodes = graphView.selection.OfType<IShaderNodeView>().Where(x => !(x.node is PropertyNode || x.node is SubGraphOutputNode)).Select(x => x.node).Where(x => x.allowedInSubGraph).ToArray();
  271. var bounds = Rect.MinMaxRect(float.PositiveInfinity, float.PositiveInfinity, float.NegativeInfinity, float.NegativeInfinity);
  272. foreach (var node in nodes)
  273. {
  274. var center = node.drawState.position.center;
  275. bounds = Rect.MinMaxRect(
  276. Mathf.Min(bounds.xMin, center.x),
  277. Mathf.Min(bounds.yMin, center.y),
  278. Mathf.Max(bounds.xMax, center.x),
  279. Mathf.Max(bounds.yMax, center.y));
  280. }
  281. var middle = bounds.center;
  282. bounds.center = Vector2.zero;
  283. // Collect graph inputs
  284. var graphInputs = graphView.selection.OfType<BlackboardField>().Select(x => x.userData as ShaderInput);
  285. // Collect the property nodes and get the corresponding properties
  286. var propertyNodeGuids = graphView.selection.OfType<IShaderNodeView>().Where(x => (x.node is PropertyNode)).Select(x => ((PropertyNode)x.node).propertyGuid);
  287. var metaProperties = graphView.graph.properties.Where(x => propertyNodeGuids.Contains(x.guid));
  288. // Collect the keyword nodes and get the corresponding keywords
  289. var keywordNodeGuids = graphView.selection.OfType<IShaderNodeView>().Where(x => (x.node is KeywordNode)).Select(x => ((KeywordNode)x.node).keywordGuid);
  290. var metaKeywords = graphView.graph.keywords.Where(x => keywordNodeGuids.Contains(x.guid));
  291. var copyPasteGraph = new CopyPasteGraph(
  292. graphView.graph.assetGuid,
  293. graphView.selection.OfType<ShaderGroup>().Select(x => x.userData),
  294. graphView.selection.OfType<IShaderNodeView>().Where(x => !(x.node is PropertyNode || x.node is SubGraphOutputNode)).Select(x => x.node).Where(x => x.allowedInSubGraph).ToArray(),
  295. graphView.selection.OfType<Edge>().Select(x => x.userData as IEdge),
  296. graphInputs,
  297. metaProperties,
  298. metaKeywords,
  299. graphView.selection.OfType<StickyNote>().Select(x => x.userData));
  300. var deserialized = CopyPasteGraph.FromJson(JsonUtility.ToJson(copyPasteGraph, false));
  301. if (deserialized == null)
  302. return;
  303. var subGraph = new GraphData { isSubGraph = true };
  304. subGraph.path = "Sub Graphs";
  305. var subGraphOutputNode = new SubGraphOutputNode();
  306. {
  307. var drawState = subGraphOutputNode.drawState;
  308. drawState.position = new Rect(new Vector2(bounds.xMax + 200f, 0f), drawState.position.size);
  309. subGraphOutputNode.drawState = drawState;
  310. }
  311. subGraph.AddNode(subGraphOutputNode);
  312. // Always copy deserialized keyword inputs
  313. foreach (ShaderKeyword keyword in deserialized.metaKeywords)
  314. {
  315. ShaderInput copiedInput = keyword.Copy();
  316. subGraph.SanitizeGraphInputName(copiedInput);
  317. subGraph.SanitizeGraphInputReferenceName(copiedInput, keyword.overrideReferenceName);
  318. subGraph.AddGraphInput(copiedInput);
  319. // Update the keyword nodes that depends on the copied keyword
  320. var dependentKeywordNodes = deserialized.GetNodes<KeywordNode>().Where(x => x.keywordGuid == keyword.guid);
  321. foreach (var node in dependentKeywordNodes)
  322. {
  323. node.owner = graphView.graph;
  324. node.keywordGuid = copiedInput.guid;
  325. }
  326. }
  327. var groupGuidMap = new Dictionary<Guid, Guid>();
  328. foreach (GroupData groupData in deserialized.groups)
  329. {
  330. var oldGuid = groupData.guid;
  331. var newGuid = groupData.RewriteGuid();
  332. groupGuidMap[oldGuid] = newGuid;
  333. subGraph.CreateGroup(groupData);
  334. }
  335. List<Guid> groupGuids = new List<Guid>();
  336. var nodeGuidMap = new Dictionary<Guid, Guid>();
  337. foreach (var node in deserialized.GetNodes<AbstractMaterialNode>())
  338. {
  339. var oldGuid = node.guid;
  340. var newGuid = node.RewriteGuid();
  341. nodeGuidMap[oldGuid] = newGuid;
  342. var drawState = node.drawState;
  343. drawState.position = new Rect(drawState.position.position - middle, drawState.position.size);
  344. node.drawState = drawState;
  345. if (!groupGuids.Contains(node.groupGuid))
  346. {
  347. groupGuids.Add(node.groupGuid);
  348. }
  349. // Checking if the group guid is also being copied.
  350. // If not then nullify that guid
  351. if (node.groupGuid != Guid.Empty)
  352. {
  353. node.groupGuid = !groupGuidMap.ContainsKey(node.groupGuid) ? Guid.Empty : groupGuidMap[node.groupGuid];
  354. }
  355. subGraph.AddNode(node);
  356. }
  357. foreach (var note in deserialized.stickyNotes)
  358. {
  359. if (!groupGuids.Contains(note.groupGuid))
  360. {
  361. groupGuids.Add(note.groupGuid);
  362. }
  363. if (note.groupGuid != Guid.Empty)
  364. {
  365. note.groupGuid = !groupGuidMap.ContainsKey(note.groupGuid) ? Guid.Empty : groupGuidMap[note.groupGuid];
  366. }
  367. note.RewriteGuid();
  368. subGraph.AddStickyNote(note);
  369. }
  370. // figure out what needs remapping
  371. var externalOutputSlots = new List<IEdge>();
  372. var externalInputSlots = new List<IEdge>();
  373. foreach (var edge in deserialized.edges)
  374. {
  375. var outputSlot = edge.outputSlot;
  376. var inputSlot = edge.inputSlot;
  377. Guid remappedOutputNodeGuid;
  378. Guid remappedInputNodeGuid;
  379. var outputSlotExistsInSubgraph = nodeGuidMap.TryGetValue(outputSlot.nodeGuid, out remappedOutputNodeGuid);
  380. var inputSlotExistsInSubgraph = nodeGuidMap.TryGetValue(inputSlot.nodeGuid, out remappedInputNodeGuid);
  381. // pasting nice internal links!
  382. if (outputSlotExistsInSubgraph && inputSlotExistsInSubgraph)
  383. {
  384. var outputSlotRef = new SlotReference(remappedOutputNodeGuid, outputSlot.slotId);
  385. var inputSlotRef = new SlotReference(remappedInputNodeGuid, inputSlot.slotId);
  386. subGraph.Connect(outputSlotRef, inputSlotRef);
  387. }
  388. // one edge needs to go to outside world
  389. else if (outputSlotExistsInSubgraph)
  390. {
  391. externalInputSlots.Add(edge);
  392. }
  393. else if (inputSlotExistsInSubgraph)
  394. {
  395. externalOutputSlots.Add(edge);
  396. }
  397. }
  398. // Find the unique edges coming INTO the graph
  399. var uniqueIncomingEdges = externalOutputSlots.GroupBy(
  400. edge => edge.outputSlot,
  401. edge => edge,
  402. (key, edges) => new { slotRef = key, edges = edges.ToList() });
  403. var externalInputNeedingConnection = new List<KeyValuePair<IEdge, AbstractShaderProperty>>();
  404. var amountOfProps = uniqueIncomingEdges.Count();
  405. const int height = 40;
  406. const int subtractHeight = 20;
  407. var propPos = new Vector2(0, -((amountOfProps / 2) + height) - subtractHeight);
  408. foreach (var group in uniqueIncomingEdges)
  409. {
  410. var sr = group.slotRef;
  411. var fromNode = graphObject.graph.GetNodeFromGuid(sr.nodeGuid);
  412. var fromSlot = fromNode.FindOutputSlot<MaterialSlot>(sr.slotId);
  413. AbstractShaderProperty prop;
  414. switch (fromSlot.concreteValueType)
  415. {
  416. case ConcreteSlotValueType.Texture2D:
  417. prop = new Texture2DShaderProperty();
  418. break;
  419. case ConcreteSlotValueType.Texture2DArray:
  420. prop = new Texture2DArrayShaderProperty();
  421. break;
  422. case ConcreteSlotValueType.Texture3D:
  423. prop = new Texture3DShaderProperty();
  424. break;
  425. case ConcreteSlotValueType.Cubemap:
  426. prop = new CubemapShaderProperty();
  427. break;
  428. case ConcreteSlotValueType.Vector4:
  429. prop = new Vector4ShaderProperty();
  430. break;
  431. case ConcreteSlotValueType.Vector3:
  432. prop = new Vector3ShaderProperty();
  433. break;
  434. case ConcreteSlotValueType.Vector2:
  435. prop = new Vector2ShaderProperty();
  436. break;
  437. case ConcreteSlotValueType.Vector1:
  438. prop = new Vector1ShaderProperty();
  439. break;
  440. case ConcreteSlotValueType.Boolean:
  441. prop = new BooleanShaderProperty();
  442. break;
  443. case ConcreteSlotValueType.Matrix2:
  444. prop = new Matrix2ShaderProperty();
  445. break;
  446. case ConcreteSlotValueType.Matrix3:
  447. prop = new Matrix3ShaderProperty();
  448. break;
  449. case ConcreteSlotValueType.Matrix4:
  450. prop = new Matrix4ShaderProperty();
  451. break;
  452. case ConcreteSlotValueType.SamplerState:
  453. prop = new SamplerStateShaderProperty();
  454. break;
  455. case ConcreteSlotValueType.Gradient:
  456. prop = new GradientShaderProperty();
  457. break;
  458. default:
  459. throw new ArgumentOutOfRangeException();
  460. }
  461. if (prop != null)
  462. {
  463. var materialGraph = (GraphData)graphObject.graph;
  464. var fromPropertyNode = fromNode as PropertyNode;
  465. var fromProperty = fromPropertyNode != null ? materialGraph.properties.FirstOrDefault(p => p.guid == fromPropertyNode.propertyGuid) : null;
  466. prop.displayName = fromProperty != null ? fromProperty.displayName : fromSlot.concreteValueType.ToString();
  467. prop.displayName = GraphUtil.SanitizeName(subGraph.addedInputs.Select(p => p.displayName), "{0} ({1})", prop.displayName);
  468. subGraph.AddGraphInput(prop);
  469. var propNode = new PropertyNode();
  470. {
  471. var drawState = propNode.drawState;
  472. drawState.position = new Rect(new Vector2(bounds.xMin - 300f, 0f) + propPos, drawState.position.size);
  473. propPos += new Vector2(0, height);
  474. propNode.drawState = drawState;
  475. }
  476. subGraph.AddNode(propNode);
  477. propNode.propertyGuid = prop.guid;
  478. foreach (var edge in group.edges)
  479. {
  480. subGraph.Connect(
  481. new SlotReference(propNode.guid, PropertyNode.OutputSlotId),
  482. new SlotReference(nodeGuidMap[edge.inputSlot.nodeGuid], edge.inputSlot.slotId));
  483. externalInputNeedingConnection.Add(new KeyValuePair<IEdge, AbstractShaderProperty>(edge, prop));
  484. }
  485. }
  486. }
  487. var uniqueOutgoingEdges = externalInputSlots.GroupBy(
  488. edge => edge.outputSlot,
  489. edge => edge,
  490. (key, edges) => new { slot = key, edges = edges.ToList() });
  491. var externalOutputsNeedingConnection = new List<KeyValuePair<IEdge, IEdge>>();
  492. foreach (var group in uniqueOutgoingEdges)
  493. {
  494. var outputNode = subGraph.outputNode as SubGraphOutputNode;
  495. AbstractMaterialNode node = graphView.graph.GetNodeFromGuid(group.edges[0].outputSlot.nodeGuid);
  496. MaterialSlot slot = node.FindSlot<MaterialSlot>(group.edges[0].outputSlot.slotId);
  497. var slotId = outputNode.AddSlot(slot.concreteValueType);
  498. var inputSlotRef = new SlotReference(outputNode.guid, slotId);
  499. foreach (var edge in group.edges)
  500. {
  501. var newEdge = subGraph.Connect(new SlotReference(nodeGuidMap[edge.outputSlot.nodeGuid], edge.outputSlot.slotId), inputSlotRef);
  502. externalOutputsNeedingConnection.Add(new KeyValuePair<IEdge, IEdge>(edge, newEdge));
  503. }
  504. }
  505. if(FileUtilities.WriteShaderGraphToDisk(path, subGraph))
  506. AssetDatabase.ImportAsset(path);
  507. var loadedSubGraph = AssetDatabase.LoadAssetAtPath(path, typeof(SubGraphAsset)) as SubGraphAsset;
  508. if (loadedSubGraph == null)
  509. return;
  510. var subGraphNode = new SubGraphNode();
  511. var ds = subGraphNode.drawState;
  512. ds.position = new Rect(middle - new Vector2(100f, 150f), Vector2.zero);
  513. subGraphNode.drawState = ds;
  514. // Add the subgraph into the group if the nodes was all in the same group group
  515. if (groupGuids.Count == 1)
  516. {
  517. subGraphNode.groupGuid = groupGuids[0];
  518. }
  519. graphObject.graph.AddNode(subGraphNode);
  520. subGraphNode.asset = loadedSubGraph;
  521. foreach (var edgeMap in externalInputNeedingConnection)
  522. {
  523. graphObject.graph.Connect(edgeMap.Key.outputSlot, new SlotReference(subGraphNode.guid, edgeMap.Value.guid.GetHashCode()));
  524. }
  525. foreach (var edgeMap in externalOutputsNeedingConnection)
  526. {
  527. graphObject.graph.Connect(new SlotReference(subGraphNode.guid, edgeMap.Value.inputSlot.slotId), edgeMap.Key.inputSlot);
  528. }
  529. graphObject.graph.RemoveElements(
  530. graphView.selection.OfType<IShaderNodeView>().Select(x => x.node).Where(x => x.allowedInSubGraph).ToArray(),
  531. new IEdge[] {},
  532. new GroupData[] {},
  533. graphView.selection.OfType<StickyNote>().Select(x => x.userData).ToArray());
  534. graphObject.graph.ValidateGraph();
  535. }
  536. void UpdateShaderGraphOnDisk(string path)
  537. {
  538. if(FileUtilities.WriteShaderGraphToDisk(path, graphObject.graph))
  539. AssetDatabase.ImportAsset(path);
  540. }
  541. public void Initialize(string assetGuid)
  542. {
  543. try
  544. {
  545. m_ColorSpace = PlayerSettings.colorSpace;
  546. m_RenderPipelineAsset = GraphicsSettings.renderPipelineAsset;
  547. var asset = AssetDatabase.LoadAssetAtPath<Object>(AssetDatabase.GUIDToAssetPath(assetGuid));
  548. if (asset == null)
  549. return;
  550. if (!EditorUtility.IsPersistent(asset))
  551. return;
  552. if (selectedGuid == assetGuid)
  553. return;
  554. var path = AssetDatabase.GetAssetPath(asset);
  555. var extension = Path.GetExtension(path);
  556. if (extension == null)
  557. return;
  558. // Path.GetExtension returns the extension prefixed with ".", so we remove it. We force lower case such that
  559. // the comparison will be case-insensitive.
  560. extension = extension.Substring(1).ToLowerInvariant();
  561. bool isSubGraph;
  562. switch (extension)
  563. {
  564. case ShaderGraphImporter.Extension:
  565. isSubGraph = false;
  566. break;
  567. case ShaderSubGraphImporter.Extension:
  568. isSubGraph = true;
  569. break;
  570. default:
  571. return;
  572. }
  573. selectedGuid = assetGuid;
  574. var textGraph = File.ReadAllText(path, Encoding.UTF8);
  575. graphObject = CreateInstance<GraphObject>();
  576. graphObject.hideFlags = HideFlags.HideAndDontSave;
  577. graphObject.graph = JsonUtility.FromJson<GraphData>(textGraph);
  578. graphObject.graph.assetGuid = assetGuid;
  579. graphObject.graph.isSubGraph = isSubGraph;
  580. graphObject.graph.messageManager = messageManager;
  581. graphObject.graph.OnEnable();
  582. graphObject.graph.ValidateGraph();
  583. graphEditorView = new GraphEditorView(this, m_GraphObject.graph, messageManager)
  584. {
  585. viewDataKey = selectedGuid,
  586. assetName = asset.name.Split('/').Last()
  587. };
  588. Texture2D icon = GetThemeIcon(graphObject.graph);
  589. // This is adding the icon at the front of the tab
  590. titleContent = EditorGUIUtility.TrTextContentWithIcon(asset.name.Split('/').Last(), icon);
  591. Repaint();
  592. }
  593. catch (Exception)
  594. {
  595. m_HasError = true;
  596. m_GraphEditorView = null;
  597. graphObject = null;
  598. throw;
  599. }
  600. }
  601. Texture2D GetThemeIcon(GraphData graphdata)
  602. {
  603. string theme = EditorGUIUtility.isProSkin ? "_dark" : "_light";
  604. Texture2D icon = Resources.Load<Texture2D>("Icons/sg_graph_icon_gray"+theme+"@16");
  605. if (graphdata.isSubGraph)
  606. {
  607. icon = Resources.Load<Texture2D>("Icons/sg_subgraph_icon_gray"+theme+"@16");
  608. }
  609. return icon;
  610. }
  611. void OnGeometryChanged(GeometryChangedEvent evt)
  612. {
  613. graphEditorView.UnregisterCallback<GeometryChangedEvent>(OnGeometryChanged);
  614. if (m_FrameAllAfterLayout)
  615. graphEditorView.graphView.FrameAll();
  616. m_FrameAllAfterLayout = false;
  617. foreach (var node in m_GraphObject.graph.GetNodes<AbstractMaterialNode>())
  618. node.Dirty(ModificationScope.Node);
  619. }
  620. }
  621. }