您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
809 行
32 KiB
809 行
32 KiB
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using UnityEngine;
|
|
using UnityEditor.Graphing;
|
|
using UnityEditor.Graphing.Util;
|
|
using UnityEditor.ShaderGraph.Drawing.Inspector;
|
|
using Object = UnityEngine.Object;
|
|
|
|
using UnityEditor.Experimental.GraphView;
|
|
using UnityEngine.UIElements;
|
|
using Edge = UnityEditor.Experimental.GraphView.Edge;
|
|
|
|
|
|
namespace UnityEditor.ShaderGraph.Drawing
|
|
{
|
|
[Serializable]
|
|
class FloatingWindowsLayout
|
|
{
|
|
public WindowDockingLayout previewLayout = new WindowDockingLayout();
|
|
public WindowDockingLayout blackboardLayout = new WindowDockingLayout();
|
|
public Vector2 masterPreviewSize = new Vector2(400, 400);
|
|
}
|
|
|
|
[Serializable]
|
|
class ToggleSettings
|
|
{
|
|
public bool isBlackboardVisible = true;
|
|
public bool isPreviewVisible = true;
|
|
}
|
|
|
|
class GraphEditorView : VisualElement, IDisposable
|
|
{
|
|
MaterialGraphView m_GraphView;
|
|
MasterPreviewView m_MasterPreviewView;
|
|
|
|
GraphData m_Graph;
|
|
PreviewManager m_PreviewManager;
|
|
MessageManager m_MessageManager;
|
|
SearchWindowProvider m_SearchWindowProvider;
|
|
EdgeConnectorListener m_EdgeConnectorListener;
|
|
BlackboardProvider m_BlackboardProvider;
|
|
|
|
public BlackboardProvider blackboardProvider
|
|
{
|
|
get { return m_BlackboardProvider; }
|
|
}
|
|
|
|
const string k_ToggleSettings = "UnityEditor.ShaderGraph.ToggleSettings";
|
|
ToggleSettings m_ToggleSettings;
|
|
|
|
const string k_FloatingWindowsLayoutKey = "UnityEditor.ShaderGraph.FloatingWindowsLayout2";
|
|
FloatingWindowsLayout m_FloatingWindowsLayout;
|
|
|
|
public Action saveRequested { get; set; }
|
|
|
|
public Action convertToSubgraphRequested
|
|
{
|
|
get { return m_GraphView.onConvertToSubgraphClick; }
|
|
set { m_GraphView.onConvertToSubgraphClick = value; }
|
|
}
|
|
|
|
public Action showInProjectRequested { get; set; }
|
|
|
|
public MaterialGraphView graphView
|
|
{
|
|
get { return m_GraphView; }
|
|
}
|
|
|
|
PreviewManager previewManager
|
|
{
|
|
get { return m_PreviewManager; }
|
|
set { m_PreviewManager = value; }
|
|
}
|
|
|
|
public string assetName
|
|
{
|
|
get { return m_BlackboardProvider.assetName; }
|
|
set
|
|
{
|
|
m_BlackboardProvider.assetName = value;
|
|
}
|
|
}
|
|
|
|
public GraphEditorView(EditorWindow editorWindow, GraphData graph, MessageManager messageManager)
|
|
{
|
|
m_Graph = graph;
|
|
m_MessageManager = messageManager;
|
|
styleSheets.Add(Resources.Load<StyleSheet>("Styles/GraphEditorView"));
|
|
previewManager = new PreviewManager(graph, messageManager);
|
|
previewManager.onPrimaryMasterChanged = OnPrimaryMasterChanged;
|
|
|
|
string serializedToggle = EditorUserSettings.GetConfigValue(k_ToggleSettings);
|
|
if (!string.IsNullOrEmpty(serializedToggle))
|
|
{
|
|
m_ToggleSettings = JsonUtility.FromJson<ToggleSettings>(serializedToggle);
|
|
}
|
|
|
|
string serializedWindowLayout = EditorUserSettings.GetConfigValue(k_FloatingWindowsLayoutKey);
|
|
if (!string.IsNullOrEmpty(serializedWindowLayout))
|
|
{
|
|
m_FloatingWindowsLayout = JsonUtility.FromJson<FloatingWindowsLayout>(serializedWindowLayout);
|
|
}
|
|
else
|
|
{
|
|
m_FloatingWindowsLayout = new FloatingWindowsLayout
|
|
{
|
|
blackboardLayout =
|
|
{
|
|
dockingTop = true,
|
|
dockingLeft = true,
|
|
verticalOffset = 16,
|
|
horizontalOffset = 16,
|
|
size = new Vector2(200, 400)
|
|
}
|
|
};
|
|
}
|
|
|
|
if (m_FloatingWindowsLayout.masterPreviewSize.x > 0f && m_FloatingWindowsLayout.masterPreviewSize.y > 0f)
|
|
{
|
|
previewManager.ResizeMasterPreview(m_FloatingWindowsLayout.masterPreviewSize);
|
|
}
|
|
|
|
previewManager.RenderPreviews();
|
|
var toolbar = new IMGUIContainer(() =>
|
|
{
|
|
GUILayout.BeginHorizontal(EditorStyles.toolbar);
|
|
if (GUILayout.Button("Save Asset", EditorStyles.toolbarButton))
|
|
{
|
|
if (saveRequested != null)
|
|
saveRequested();
|
|
}
|
|
GUILayout.Space(6);
|
|
if (GUILayout.Button("Show In Project", EditorStyles.toolbarButton))
|
|
{
|
|
if (showInProjectRequested != null)
|
|
showInProjectRequested();
|
|
}
|
|
|
|
GUILayout.FlexibleSpace();
|
|
|
|
EditorGUI.BeginChangeCheck();
|
|
m_ToggleSettings.isBlackboardVisible = GUILayout.Toggle(m_ToggleSettings.isBlackboardVisible, "Blackboard", EditorStyles.toolbarButton);
|
|
|
|
GUILayout.Space(6);
|
|
|
|
m_ToggleSettings.isPreviewVisible = GUILayout.Toggle(m_ToggleSettings.isPreviewVisible, "Main Preview", EditorStyles.toolbarButton);
|
|
if (EditorGUI.EndChangeCheck())
|
|
{
|
|
m_MasterPreviewView.visible = m_ToggleSettings.isPreviewVisible;
|
|
m_BlackboardProvider.blackboard.visible = m_ToggleSettings.isBlackboardVisible;
|
|
string serializedToggleables = JsonUtility.ToJson(m_ToggleSettings);
|
|
EditorUserSettings.SetConfigValue(k_ToggleSettings, serializedToggleables);
|
|
}
|
|
GUILayout.EndHorizontal();
|
|
});
|
|
Add(toolbar);
|
|
|
|
var content = new VisualElement { name = "content" };
|
|
{
|
|
m_GraphView = new MaterialGraphView(graph) { name = "GraphView", viewDataKey = "MaterialGraphView" };
|
|
m_GraphView.SetupZoom(0.05f, ContentZoomer.DefaultMaxScale);
|
|
m_GraphView.AddManipulator(new ContentDragger());
|
|
m_GraphView.AddManipulator(new SelectionDragger());
|
|
m_GraphView.AddManipulator(new RectangleSelector());
|
|
m_GraphView.AddManipulator(new ClickSelector());
|
|
m_GraphView.RegisterCallback<KeyDownEvent>(OnSpaceDown);
|
|
m_GraphView.groupTitleChanged = OnGroupTitleChanged;
|
|
m_GraphView.elementsAddedToGroup = OnElementsAddedToGroup;
|
|
m_GraphView.elementsRemovedFromGroup = OnElementsRemovedFromGroup;
|
|
content.Add(m_GraphView);
|
|
|
|
m_BlackboardProvider = new BlackboardProvider(graph);
|
|
m_GraphView.Add(m_BlackboardProvider.blackboard);
|
|
|
|
// Initialize toggle settings if it doesnt exist.
|
|
if (m_ToggleSettings == null)
|
|
{
|
|
m_ToggleSettings = new ToggleSettings();
|
|
}
|
|
m_BlackboardProvider.blackboard.visible = m_ToggleSettings.isBlackboardVisible;
|
|
|
|
CreateMasterPreview();
|
|
|
|
m_GraphView.graphViewChanged = GraphViewChanged;
|
|
|
|
RegisterCallback<GeometryChangedEvent>(ApplySerializewindowLayouts);
|
|
}
|
|
|
|
m_SearchWindowProvider = ScriptableObject.CreateInstance<SearchWindowProvider>();
|
|
m_SearchWindowProvider.Initialize(editorWindow, m_Graph, m_GraphView);
|
|
m_GraphView.nodeCreationRequest = (c) =>
|
|
{
|
|
m_SearchWindowProvider.connectedPort = null;
|
|
SearchWindow.Open(new SearchWindowContext(c.screenMousePosition), m_SearchWindowProvider);
|
|
};
|
|
|
|
m_EdgeConnectorListener = new EdgeConnectorListener(m_Graph, m_SearchWindowProvider);
|
|
|
|
foreach (var graphGroup in graph.groups)
|
|
{
|
|
AddGroup(graphGroup);
|
|
}
|
|
|
|
foreach (var node in graph.GetNodes<AbstractMaterialNode>())
|
|
AddNode(node);
|
|
|
|
foreach (var edge in graph.edges)
|
|
AddEdge(edge);
|
|
|
|
Add(content);
|
|
}
|
|
|
|
void CreateMasterPreview()
|
|
{
|
|
m_MasterPreviewView = new MasterPreviewView(previewManager, m_Graph) {name = "masterPreview"};
|
|
|
|
var masterPreviewViewDraggable = new WindowDraggable(null, this);
|
|
m_MasterPreviewView.AddManipulator(masterPreviewViewDraggable);
|
|
m_GraphView.Add(m_MasterPreviewView);
|
|
|
|
masterPreviewViewDraggable.OnDragFinished += UpdateSerializedWindowLayout;
|
|
m_MasterPreviewView.previewResizeBorderFrame.OnResizeFinished += UpdateSerializedWindowLayout;
|
|
m_MasterPreviewView.visible = m_ToggleSettings.isPreviewVisible;
|
|
}
|
|
|
|
void OnSpaceDown(KeyDownEvent evt)
|
|
{
|
|
if (evt.keyCode == KeyCode.F1)
|
|
{
|
|
if (m_GraphView.selection.OfType<MaterialNodeView>().Count() == 1)
|
|
{
|
|
var nodeView = (MaterialNodeView)graphView.selection.First();
|
|
if (nodeView.node.documentationURL != null)
|
|
System.Diagnostics.Process.Start(nodeView.node.documentationURL);
|
|
}
|
|
}
|
|
}
|
|
|
|
GraphViewChange GraphViewChanged(GraphViewChange graphViewChange)
|
|
{
|
|
if (graphViewChange.edgesToCreate != null)
|
|
{
|
|
foreach (var edge in graphViewChange.edgesToCreate)
|
|
{
|
|
var leftSlot = edge.output.GetSlot();
|
|
var rightSlot = edge.input.GetSlot();
|
|
if (leftSlot != null && rightSlot != null)
|
|
{
|
|
m_Graph.owner.RegisterCompleteObjectUndo("Connect Edge");
|
|
m_Graph.Connect(leftSlot.slotReference, rightSlot.slotReference);
|
|
}
|
|
}
|
|
graphViewChange.edgesToCreate.Clear();
|
|
}
|
|
|
|
if (graphViewChange.movedElements != null)
|
|
{
|
|
m_Graph.owner.RegisterCompleteObjectUndo("Move Elements");
|
|
|
|
List<GraphElement> nodesInsideGroup = new List<GraphElement>();
|
|
foreach (var element in graphViewChange.movedElements)
|
|
{
|
|
var groupNode = element as ShaderGroup;
|
|
if (groupNode == null)
|
|
continue;
|
|
|
|
foreach (GraphElement graphElement in groupNode.containedElements)
|
|
{
|
|
nodesInsideGroup.Add(graphElement);
|
|
}
|
|
}
|
|
|
|
if(nodesInsideGroup.Any())
|
|
graphViewChange.movedElements.AddRange(nodesInsideGroup);
|
|
|
|
foreach (var element in graphViewChange.movedElements)
|
|
{
|
|
var node = element.userData as AbstractMaterialNode;
|
|
if (node == null)
|
|
continue;
|
|
|
|
var drawState = node.drawState;
|
|
drawState.position = element.parent.ChangeCoordinatesTo(m_GraphView.contentViewContainer, element.GetPosition());
|
|
node.drawState = drawState;
|
|
}
|
|
}
|
|
|
|
var nodesToUpdate = m_NodeViewHashSet;
|
|
nodesToUpdate.Clear();
|
|
|
|
if (graphViewChange.elementsToRemove != null)
|
|
{
|
|
m_Graph.owner.RegisterCompleteObjectUndo("Remove Elements");
|
|
m_Graph.RemoveElements(graphViewChange.elementsToRemove.OfType<MaterialNodeView>().Select(v => (AbstractMaterialNode)v.node),
|
|
graphViewChange.elementsToRemove.OfType<Edge>().Select(e => (IEdge)e.userData),
|
|
graphViewChange.elementsToRemove.OfType<ShaderGroup>().Select(g => (GroupData)g.userData));
|
|
foreach (var edge in graphViewChange.elementsToRemove.OfType<Edge>())
|
|
{
|
|
if (edge.input != null)
|
|
{
|
|
var materialNodeView = edge.input.node as MaterialNodeView;
|
|
if (materialNodeView != null)
|
|
nodesToUpdate.Add(materialNodeView);
|
|
}
|
|
if (edge.output != null)
|
|
{
|
|
var materialNodeView = edge.output.node as MaterialNodeView;
|
|
if (materialNodeView != null)
|
|
nodesToUpdate.Add(materialNodeView);
|
|
}
|
|
}
|
|
}
|
|
|
|
foreach (var node in nodesToUpdate)
|
|
node.UpdatePortInputVisibilities();
|
|
|
|
UpdateEdgeColors(nodesToUpdate);
|
|
|
|
return graphViewChange;
|
|
}
|
|
|
|
void OnGroupTitleChanged(Group graphGroup, string title)
|
|
{
|
|
var groupData = graphGroup.userData as GroupData;
|
|
if (groupData != null)
|
|
{
|
|
groupData.title = graphGroup.title;
|
|
}
|
|
}
|
|
|
|
void OnElementsAddedToGroup(Group graphGroup, IEnumerable<GraphElement> element)
|
|
{
|
|
var groupData = graphGroup.userData as GroupData;
|
|
if (groupData != null)
|
|
{
|
|
var anyChanged = false;
|
|
foreach (var materialNodeView in element.Select(e => e).OfType<MaterialNodeView>())
|
|
{
|
|
if (materialNodeView.node.groupGuid != groupData.guid)
|
|
{
|
|
anyChanged = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!anyChanged)
|
|
return;
|
|
|
|
m_Graph.owner.RegisterCompleteObjectUndo(groupData.title);
|
|
|
|
foreach (var materialNodeView in element.Select(e => e).OfType<MaterialNodeView>())
|
|
{
|
|
m_Graph.SetNodeGroup(materialNodeView.node, groupData);
|
|
}
|
|
}
|
|
}
|
|
|
|
void OnElementsRemovedFromGroup(Group graphGroup, IEnumerable<GraphElement> element)
|
|
{
|
|
var groupData = graphGroup.userData as GroupData;
|
|
if (groupData != null)
|
|
{
|
|
var anyChanged = false;
|
|
foreach (var materialNodeView in element.Select(e => e).OfType<MaterialNodeView>())
|
|
{
|
|
if (materialNodeView.userData != null && materialNodeView.node.groupGuid == groupData.guid)
|
|
{
|
|
anyChanged = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!anyChanged)
|
|
return;
|
|
|
|
m_Graph.owner.RegisterCompleteObjectUndo("Ungroup Node(s)");
|
|
|
|
foreach (var materialNodeView in element.Select(e => e).OfType<MaterialNodeView>())
|
|
{
|
|
if (materialNodeView.node != null)
|
|
m_Graph.SetNodeGroup(materialNodeView.node, null);
|
|
}
|
|
}
|
|
}
|
|
|
|
void OnNodeChanged(AbstractMaterialNode inNode, ModificationScope scope)
|
|
{
|
|
if (m_GraphView == null)
|
|
return;
|
|
|
|
var dependentNodes = new List<AbstractMaterialNode>();
|
|
NodeUtils.CollectNodesNodeFeedsInto(dependentNodes, inNode);
|
|
foreach (var node in dependentNodes)
|
|
{
|
|
var theViews = m_GraphView.nodes.ToList().OfType<MaterialNodeView>();
|
|
var viewsFound = theViews.Where(x => x.node.guid == node.guid).ToList();
|
|
foreach (var drawableNodeData in viewsFound)
|
|
drawableNodeData.OnModified(scope);
|
|
}
|
|
}
|
|
|
|
HashSet<MaterialNodeView> m_NodeViewHashSet = new HashSet<MaterialNodeView>();
|
|
|
|
public void UpdatePreviewShaders()
|
|
{
|
|
previewManager.ForceShaderUpdate();
|
|
}
|
|
|
|
public void HandleGraphChanges()
|
|
{
|
|
previewManager.HandleGraphChanges();
|
|
previewManager.RenderPreviews();
|
|
m_BlackboardProvider.HandleGraphChanges();
|
|
|
|
foreach (GroupData groupData in m_Graph.removedGroups)
|
|
{
|
|
var group = m_GraphView.graphElements.ToList().OfType<ShaderGroup>().ToList().First(g => g.userData == groupData);
|
|
m_GraphView.RemoveElement(group);
|
|
}
|
|
|
|
foreach (var node in m_Graph.removedNodes)
|
|
{
|
|
node.UnregisterCallback(OnNodeChanged);
|
|
var nodeView = m_GraphView.nodes.ToList().OfType<MaterialNodeView>()
|
|
.FirstOrDefault(p => p.node != null && p.node.guid == node.guid);
|
|
if (nodeView != null)
|
|
{
|
|
nodeView.Dispose();
|
|
m_GraphView.RemoveElement(nodeView);
|
|
}
|
|
}
|
|
|
|
foreach (var groupData in m_Graph.addedGroups)
|
|
{
|
|
AddGroup(groupData);
|
|
}
|
|
|
|
foreach (var node in m_Graph.addedNodes)
|
|
{
|
|
AddNode(node);
|
|
}
|
|
|
|
foreach (var groupChange in m_Graph.nodeGroupChanges)
|
|
{
|
|
var nodeView = (MaterialNodeView)m_GraphView.GetNodeByGuid(groupChange.nodeGuid.ToString());
|
|
if (nodeView != null)
|
|
{
|
|
var groupView = nodeView.GetContainingScope() as ShaderGroup;
|
|
if (groupView?.userData.guid != groupChange.newGroupGuid)
|
|
{
|
|
groupView?.RemoveElement(nodeView);
|
|
if (groupChange.newGroupGuid != Guid.Empty)
|
|
{
|
|
var newGroupView = m_GraphView.graphElements.ToList()
|
|
.OfType<ShaderGroup>()
|
|
.First(x => x.userData.guid == groupChange.newGroupGuid);
|
|
newGroupView.AddElement(nodeView);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
foreach (var groupData in m_Graph.pastedGroups)
|
|
{
|
|
var group = m_GraphView.graphElements.ToList().OfType<ShaderGroup>().ToList().First(g => g.userData == groupData);
|
|
m_GraphView.AddToSelection(group);
|
|
}
|
|
|
|
foreach (var node in m_Graph.pastedNodes)
|
|
{
|
|
var nodeView = m_GraphView.nodes.ToList().OfType<MaterialNodeView>()
|
|
.FirstOrDefault(p => p.node != null && p.node.guid == node.guid);
|
|
m_GraphView.AddToSelection(nodeView);
|
|
}
|
|
|
|
var nodesToUpdate = m_NodeViewHashSet;
|
|
nodesToUpdate.Clear();
|
|
|
|
foreach (var edge in m_Graph.removedEdges)
|
|
{
|
|
var edgeView = m_GraphView.graphElements.ToList().OfType<Edge>()
|
|
.FirstOrDefault(p => p.userData is IEdge && Equals((IEdge) p.userData, edge));
|
|
if (edgeView != null)
|
|
{
|
|
var nodeView = (MaterialNodeView)edgeView.input.node;
|
|
if (nodeView?.node != null)
|
|
{
|
|
nodesToUpdate.Add(nodeView);
|
|
}
|
|
|
|
edgeView.output.Disconnect(edgeView);
|
|
edgeView.input.Disconnect(edgeView);
|
|
|
|
edgeView.output = null;
|
|
edgeView.input = null;
|
|
|
|
m_GraphView.RemoveElement(edgeView);
|
|
}
|
|
}
|
|
|
|
foreach (var edge in m_Graph.addedEdges)
|
|
{
|
|
var edgeView = AddEdge(edge);
|
|
if (edgeView != null)
|
|
nodesToUpdate.Add((MaterialNodeView)edgeView.input.node);
|
|
}
|
|
|
|
foreach (var node in nodesToUpdate)
|
|
node.UpdatePortInputVisibilities();
|
|
|
|
UpdateEdgeColors(nodesToUpdate);
|
|
|
|
UpdateBadges();
|
|
}
|
|
|
|
void UpdateBadges()
|
|
{
|
|
if (!m_MessageManager.nodeMessagesChanged)
|
|
return;
|
|
|
|
foreach (var messageData in m_MessageManager.GetNodeMessages())
|
|
{
|
|
var node = m_Graph.GetNodeFromTempId(messageData.Key);
|
|
|
|
if (!(m_GraphView.GetNodeByGuid(node.guid.ToString()) is MaterialNodeView nodeView))
|
|
continue;
|
|
|
|
if (messageData.Value.Count == 0)
|
|
{
|
|
var badge = nodeView.Q<IconBadge>();
|
|
badge?.Detach();
|
|
badge?.RemoveFromHierarchy();
|
|
}
|
|
else
|
|
{
|
|
var foundMessage = messageData.Value.First();
|
|
nodeView.AttachMessage(foundMessage.message, foundMessage.severity);
|
|
}
|
|
}
|
|
}
|
|
|
|
List<GraphElement> m_AddNodeGraphElements = new List<GraphElement>();
|
|
|
|
void AddNode(AbstractMaterialNode node)
|
|
{
|
|
var nodeView = new MaterialNodeView { userData = node };
|
|
|
|
m_GraphView.AddElement(nodeView);
|
|
var materialNode = (AbstractMaterialNode)node;
|
|
nodeView.Initialize(materialNode, m_PreviewManager, m_EdgeConnectorListener);
|
|
node.RegisterCallback(OnNodeChanged);
|
|
|
|
nodeView.MarkDirtyRepaint();
|
|
|
|
if (m_SearchWindowProvider.nodeNeedsRepositioning && m_SearchWindowProvider.targetSlotReference.nodeGuid.Equals(node.guid))
|
|
{
|
|
m_SearchWindowProvider.nodeNeedsRepositioning = false;
|
|
foreach (var element in nodeView.inputContainer.Children().Union(nodeView.outputContainer.Children()))
|
|
{
|
|
var port = element as ShaderPort;
|
|
if (port == null)
|
|
continue;
|
|
if (port.slot.slotReference.Equals(m_SearchWindowProvider.targetSlotReference))
|
|
{
|
|
port.RegisterCallback<GeometryChangedEvent>(RepositionNode);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
m_AddNodeGraphElements.Clear();
|
|
m_GraphView.graphElements.ToList(m_AddNodeGraphElements);
|
|
|
|
if (materialNode.groupGuid != Guid.Empty)
|
|
{
|
|
foreach (var element in m_AddNodeGraphElements)
|
|
{
|
|
if (element is ShaderGroup groupView && groupView.userData.guid == materialNode.groupGuid)
|
|
{
|
|
groupView.AddElement(nodeView);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void AddGroup(GroupData groupData)
|
|
{
|
|
ShaderGroup graphGroup = new ShaderGroup();
|
|
|
|
graphGroup.userData = groupData;
|
|
graphGroup.title = groupData.title;
|
|
|
|
m_GraphView.AddElement(graphGroup);
|
|
}
|
|
|
|
static void RepositionNode(GeometryChangedEvent evt)
|
|
{
|
|
var port = evt.target as ShaderPort;
|
|
if (port == null)
|
|
return;
|
|
port.UnregisterCallback<GeometryChangedEvent>(RepositionNode);
|
|
var nodeView = port.node as MaterialNodeView;
|
|
if (nodeView == null)
|
|
return;
|
|
var offset = nodeView.mainContainer.WorldToLocal(port.GetGlobalCenter() + new Vector3(3f, 3f, 0f));
|
|
var position = nodeView.GetPosition();
|
|
position.position -= offset;
|
|
nodeView.SetPosition(position);
|
|
var drawState = nodeView.node.drawState;
|
|
drawState.position = position;
|
|
nodeView.node.drawState = drawState;
|
|
nodeView.MarkDirtyRepaint();
|
|
port.MarkDirtyRepaint();
|
|
}
|
|
|
|
Edge AddEdge(IEdge edge)
|
|
{
|
|
var sourceNode = m_Graph.GetNodeFromGuid(edge.outputSlot.nodeGuid);
|
|
if (sourceNode == null)
|
|
{
|
|
Debug.LogWarning("Source node is null");
|
|
return null;
|
|
}
|
|
var sourceSlot = sourceNode.FindOutputSlot<MaterialSlot>(edge.outputSlot.slotId);
|
|
|
|
var targetNode = m_Graph.GetNodeFromGuid(edge.inputSlot.nodeGuid);
|
|
if (targetNode == null)
|
|
{
|
|
Debug.LogWarning("Target node is null");
|
|
return null;
|
|
}
|
|
var targetSlot = targetNode.FindInputSlot<MaterialSlot>(edge.inputSlot.slotId);
|
|
|
|
var sourceNodeView = m_GraphView.nodes.ToList().OfType<MaterialNodeView>().FirstOrDefault(x => x.node == sourceNode);
|
|
if (sourceNodeView != null)
|
|
{
|
|
var sourceAnchor = sourceNodeView.outputContainer.Children().OfType<ShaderPort>().FirstOrDefault(x => x.slot.Equals(sourceSlot));
|
|
|
|
var targetNodeView = m_GraphView.nodes.ToList().OfType<MaterialNodeView>().FirstOrDefault(x => x.node == targetNode);
|
|
var targetAnchor = targetNodeView.inputContainer.Children().OfType<ShaderPort>().FirstOrDefault(x => x.slot.Equals(targetSlot));
|
|
|
|
var edgeView = new Edge
|
|
{
|
|
userData = edge,
|
|
output = sourceAnchor,
|
|
input = targetAnchor
|
|
};
|
|
edgeView.output.Connect(edgeView);
|
|
edgeView.input.Connect(edgeView);
|
|
m_GraphView.AddElement(edgeView);
|
|
sourceNodeView.RefreshPorts();
|
|
targetNodeView.RefreshPorts();
|
|
sourceNodeView.UpdatePortInputTypes();
|
|
targetNodeView.UpdatePortInputTypes();
|
|
|
|
return edgeView;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
Stack<MaterialNodeView> m_NodeStack = new Stack<MaterialNodeView>();
|
|
|
|
void UpdateEdgeColors(HashSet<MaterialNodeView> nodeViews)
|
|
{
|
|
var nodeStack = m_NodeStack;
|
|
nodeStack.Clear();
|
|
foreach (var nodeView in nodeViews)
|
|
nodeStack.Push(nodeView);
|
|
while (nodeStack.Any())
|
|
{
|
|
var nodeView = nodeStack.Pop();
|
|
nodeView.UpdatePortInputTypes();
|
|
foreach (var anchorView in nodeView.outputContainer.Children().OfType<Port>())
|
|
{
|
|
foreach (var edgeView in anchorView.connections.OfType<Edge>())
|
|
{
|
|
var targetSlot = edgeView.input.GetSlot();
|
|
if (targetSlot.valueType == SlotValueType.DynamicVector || targetSlot.valueType == SlotValueType.DynamicMatrix || targetSlot.valueType == SlotValueType.Dynamic)
|
|
{
|
|
var connectedNodeView = edgeView.input.node as MaterialNodeView;
|
|
if (connectedNodeView != null && !nodeViews.Contains(connectedNodeView))
|
|
{
|
|
nodeStack.Push(connectedNodeView);
|
|
nodeViews.Add(connectedNodeView);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
foreach (var anchorView in nodeView.inputContainer.Children().OfType<Port>())
|
|
{
|
|
var targetSlot = anchorView.GetSlot();
|
|
if (targetSlot.valueType != SlotValueType.DynamicVector)
|
|
continue;
|
|
foreach (var edgeView in anchorView.connections.OfType<Edge>())
|
|
{
|
|
var connectedNodeView = edgeView.output.node as MaterialNodeView;
|
|
if (connectedNodeView != null && !nodeViews.Contains(connectedNodeView))
|
|
{
|
|
nodeStack.Push(connectedNodeView);
|
|
nodeViews.Add(connectedNodeView);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void OnPrimaryMasterChanged()
|
|
{
|
|
m_MasterPreviewView?.RemoveFromHierarchy();
|
|
CreateMasterPreview();
|
|
ApplyMasterPreviewLayout();
|
|
}
|
|
|
|
void HandleEditorViewChanged(GeometryChangedEvent evt)
|
|
{
|
|
m_BlackboardProvider.blackboard.SetPosition(m_FloatingWindowsLayout.blackboardLayout.GetLayout(m_GraphView.layout));
|
|
}
|
|
|
|
void StoreBlackboardLayoutOnGeometryChanged(GeometryChangedEvent evt)
|
|
{
|
|
UpdateSerializedWindowLayout();
|
|
}
|
|
|
|
void ApplySerializewindowLayouts(GeometryChangedEvent evt)
|
|
{
|
|
UnregisterCallback<GeometryChangedEvent>(ApplySerializewindowLayouts);
|
|
|
|
if (m_FloatingWindowsLayout != null)
|
|
{
|
|
ApplyMasterPreviewLayout();
|
|
|
|
// Restore blackboard layout, and make sure that it remains in the view.
|
|
Rect blackboardRect = m_FloatingWindowsLayout.blackboardLayout.GetLayout(this.layout);
|
|
|
|
// Make sure the dimensions are sufficiently large.
|
|
blackboardRect.width = Mathf.Clamp(blackboardRect.width, 160f, m_GraphView.contentContainer.layout.width);
|
|
blackboardRect.height = Mathf.Clamp(blackboardRect.height, 160f, m_GraphView.contentContainer.layout.height);
|
|
|
|
// Make sure that the positionining is on screen.
|
|
blackboardRect.x = Mathf.Clamp(blackboardRect.x, 0f, Mathf.Max(1f, m_GraphView.contentContainer.layout.width - blackboardRect.width - blackboardRect.width));
|
|
blackboardRect.y = Mathf.Clamp(blackboardRect.y, 0f, Mathf.Max(1f, m_GraphView.contentContainer.layout.height - blackboardRect.height - blackboardRect.height));
|
|
|
|
// Set the processed blackboard layout.
|
|
m_BlackboardProvider.blackboard.SetPosition(blackboardRect);
|
|
|
|
previewManager.ResizeMasterPreview(m_FloatingWindowsLayout.masterPreviewSize);
|
|
}
|
|
else
|
|
{
|
|
m_FloatingWindowsLayout = new FloatingWindowsLayout();
|
|
}
|
|
|
|
// After the layout is restored from the previous session, start tracking layout changes in the blackboard.
|
|
m_BlackboardProvider.blackboard.RegisterCallback<GeometryChangedEvent>(StoreBlackboardLayoutOnGeometryChanged);
|
|
|
|
// After the layout is restored, track changes in layout and make the blackboard have the same behavior as the preview w.r.t. docking.
|
|
RegisterCallback<GeometryChangedEvent>(HandleEditorViewChanged);
|
|
}
|
|
|
|
void ApplyMasterPreviewLayout()
|
|
{
|
|
m_FloatingWindowsLayout.previewLayout.ApplyPosition(m_MasterPreviewView);
|
|
m_MasterPreviewView.previewTextureView.style.width = m_FloatingWindowsLayout.masterPreviewSize.x;
|
|
m_MasterPreviewView.previewTextureView.style.height = m_FloatingWindowsLayout.masterPreviewSize.y;
|
|
}
|
|
|
|
void UpdateSerializedWindowLayout()
|
|
{
|
|
m_FloatingWindowsLayout.previewLayout.CalculateDockingCornerAndOffset(m_MasterPreviewView.layout, m_GraphView.layout);
|
|
m_FloatingWindowsLayout.previewLayout.ClampToParentWindow();
|
|
|
|
m_FloatingWindowsLayout.blackboardLayout.CalculateDockingCornerAndOffset(m_BlackboardProvider.blackboard.layout, m_GraphView.layout);
|
|
m_FloatingWindowsLayout.blackboardLayout.ClampToParentWindow();
|
|
|
|
if (m_MasterPreviewView.expanded)
|
|
{
|
|
m_FloatingWindowsLayout.masterPreviewSize = m_MasterPreviewView.previewTextureView.layout.size;
|
|
}
|
|
|
|
string serializedWindowLayout = JsonUtility.ToJson(m_FloatingWindowsLayout);
|
|
EditorUserSettings.SetConfigValue(k_FloatingWindowsLayoutKey, serializedWindowLayout);
|
|
}
|
|
|
|
public void Dispose()
|
|
{
|
|
if (m_GraphView != null)
|
|
{
|
|
saveRequested = null;
|
|
convertToSubgraphRequested = null;
|
|
showInProjectRequested = null;
|
|
foreach (var node in m_GraphView.Children().OfType<MaterialNodeView>())
|
|
node.Dispose();
|
|
m_GraphView = null;
|
|
}
|
|
if (previewManager != null)
|
|
{
|
|
previewManager.Dispose();
|
|
previewManager = null;
|
|
}
|
|
if (m_SearchWindowProvider != null)
|
|
{
|
|
Object.DestroyImmediate(m_SearchWindowProvider);
|
|
m_SearchWindowProvider = null;
|
|
}
|
|
}
|
|
}
|
|
}
|