using System; using System.Collections; using System.Collections.Generic; using System.Linq; using System.Reflection; using UnityEngine; using UnityEditor.Experimental.UIElements; using UnityEngine.Experimental.UIElements; using UnityEditor.Graphing; using Object = UnityEngine.Object; namespace UnityEditor.ShaderGraph.Drawing.Inspector { public class MasterPreviewView : VisualElement { AbstractMaterialGraph m_Graph; PreviewRenderData m_PreviewRenderHandle; PreviewTextureView m_PreviewTextureView; Vector2 m_PreviewScrollPosition; ObjectField m_PreviewMeshPicker; MasterNode m_MasterNode; Mesh m_PreviousMesh; List m_DoNotShowPrimitives = new List( new string[] {PrimitiveType.Plane.ToString()}); static Type s_ContextualMenuManipulator = AppDomain.CurrentDomain.GetAssemblies().SelectMany(x => x.GetTypes()).FirstOrDefault(t => t.FullName == "UnityEngine.Experimental.UIElements.ContextualMenuManipulator"); static Type s_ObjectSelector = AppDomain.CurrentDomain.GetAssemblies().SelectMany(x => x.GetTypes()).FirstOrDefault(t => t.FullName == "UnityEditor.ObjectSelector"); public MasterPreviewView(string assetName, PreviewManager previewManager, AbstractMaterialGraph graph) { this.clippingOptions = ClippingOptions.ClipAndCacheContents; m_Graph = graph; AddStyleSheetPath("Styles/MaterialGraph"); m_PreviewRenderHandle = previewManager.masterRenderData; m_PreviewRenderHandle.onPreviewChanged += OnPreviewChanged; var topContainer = new VisualElement() { name = "top" }; { var title = new Label(assetName.Split('/').Last()) { name = "title" }; topContainer.Add(title); } Add(topContainer); var middleContainer = new VisualElement {name = "middle"}; { m_PreviewTextureView = new PreviewTextureView { name = "preview", image = Texture2D.blackTexture }; m_PreviewTextureView.AddManipulator(new Draggable(OnMouseDragPreviewMesh, true)); m_PreviewTextureView.AddManipulator((IManipulator)Activator.CreateInstance(s_ContextualMenuManipulator, (Action)BuildContextualMenu)); middleContainer.Add(m_PreviewTextureView); m_PreviewScrollPosition = new Vector2(0f, 0f); middleContainer.Add(m_PreviewTextureView); middleContainer.AddManipulator(new Scrollable(OnScroll)); } Add(middleContainer); } void OnScroll(float scrollValue) { float rescaleAmount = -scrollValue * .03f; m_Graph.previewData.scale = Mathf.Clamp(m_Graph.previewData.scale + rescaleAmount, 0.2f, 5f); } void BuildContextualMenu(ContextualMenuPopulateEvent evt) { foreach (var primitiveTypeName in Enum.GetNames(typeof(PrimitiveType))) { if(m_DoNotShowPrimitives.Contains(primitiveTypeName)) continue; evt.menu.AppendAction(primitiveTypeName, e => ChangePrimitiveMesh( primitiveTypeName ), ContextualMenu.MenuAction.AlwaysEnabled); } evt.menu.AppendAction("Custom Mesh", e => ChangeMeshCustom(), ContextualMenu.MenuAction.AlwaysEnabled); } MasterNode masterNode { get { return m_PreviewRenderHandle.shaderData.node as MasterNode; } } void OnPreviewChanged() { m_PreviewTextureView.image = m_PreviewRenderHandle.texture ?? Texture2D.blackTexture; m_PreviewTextureView.Dirty(ChangeType.Repaint); } void ChangePrimitiveMesh(string primitiveName) { Mesh changedPrimitiveMesh = Resources.GetBuiltinResource(typeof(Mesh), string.Format("{0}.fbx", primitiveName)) as Mesh; ChangeMesh(changedPrimitiveMesh); } void ChangeMesh(Mesh mesh) { Mesh changedMesh = mesh; masterNode.Dirty(ModificationScope.Node); if (m_Graph.previewData.serializedMesh.mesh != changedMesh) { m_Graph.previewData.rotation = Quaternion.identity; } m_Graph.previewData.serializedMesh.mesh = changedMesh; } private static EditorWindow Get() { PropertyInfo P = s_ObjectSelector.GetProperty("get", BindingFlags.Public | BindingFlags.Static); return P.GetValue(null,null) as EditorWindow; } void OnMeshChanged(Object obj) { var mesh = obj as Mesh; if (mesh == null) mesh = m_PreviousMesh; ChangeMesh(mesh); } void ChangeMeshCustom() { MethodInfo ShowMethod = s_ObjectSelector.GetMethod("Show", BindingFlags.NonPublic | BindingFlags.Instance | BindingFlags.DeclaredOnly, Type.DefaultBinder, new[] {typeof(Object), typeof(Type), typeof(SerializedProperty), typeof(bool), typeof(List), typeof(Action), typeof(Action)}, new ParameterModifier[7]); m_PreviousMesh = m_Graph.previewData.serializedMesh.mesh; ShowMethod.Invoke(Get(), new object[] { null, typeof(Mesh), null, false, null, (Action)OnMeshChanged, (Action)OnMeshChanged }); } public void RefreshRenderTextureSize() { RenderTextureDescriptor descriptor = m_PreviewRenderHandle.renderTexture.descriptor; var targetWidth = m_PreviewTextureView.contentRect.width; var targetHeight = m_PreviewTextureView.contentRect.height; if (Mathf.Approximately(descriptor.width, targetHeight) && Mathf.Approximately(descriptor.height, targetWidth)) { return; } descriptor.width = (int)m_PreviewTextureView.contentRect.width; descriptor.height = (int)m_PreviewTextureView.contentRect.height; m_PreviewRenderHandle.renderTexture.Release(); Object.DestroyImmediate(m_PreviewRenderHandle.renderTexture); m_PreviewRenderHandle.renderTexture = new RenderTexture(descriptor); } public void UpdateRenderTextureOnNextLayoutChange() { RegisterCallback(AdaptRenderTextureOnLayoutChange); } void AdaptRenderTextureOnLayoutChange(PostLayoutEvent evt) { UnregisterCallback(AdaptRenderTextureOnLayoutChange); RefreshRenderTextureSize(); } void OnMouseDragPreviewMesh(Vector2 deltaMouse) { Vector2 previewSize = m_PreviewTextureView.contentRect.size; m_PreviewScrollPosition -= deltaMouse * (Event.current.shift ? 3f : 1f) / Mathf.Min(previewSize.x, previewSize.y) * 140f; m_PreviewScrollPosition.y = Mathf.Clamp(m_PreviewScrollPosition.y, -90f, 90f); Quaternion previewRotation = Quaternion.Euler(m_PreviewScrollPosition.y, 0, 0) * Quaternion.Euler(0, m_PreviewScrollPosition.x, 0); m_Graph.previewData.rotation = previewRotation; masterNode.Dirty(ModificationScope.Node); } } }