diff --git a/Bonsai.Editor.Tests/MockGraphView.cs b/Bonsai.Editor.Tests/MockGraphView.cs index 296a11c2c..e67e10448 100644 --- a/Bonsai.Editor.Tests/MockGraphView.cs +++ b/Bonsai.Editor.Tests/MockGraphView.cs @@ -18,6 +18,7 @@ public MockGraphView(ExpressionBuilderGraph workflow = null) Workflow = workflow ?? new ExpressionBuilderGraph(); CommandExecutor = new CommandExecutor(); var serviceContainer = new ServiceContainer(); + serviceContainer.AddService(typeof(WorkflowBuilder), new WorkflowBuilder(Workflow)); serviceContainer.AddService(typeof(CommandExecutor), CommandExecutor); ServiceProvider = serviceContainer; } diff --git a/Bonsai.Editor.Tests/WorkflowEditorTests.cs b/Bonsai.Editor.Tests/WorkflowEditorTests.cs index 9c5bd0316..556f7ddf0 100644 --- a/Bonsai.Editor.Tests/WorkflowEditorTests.cs +++ b/Bonsai.Editor.Tests/WorkflowEditorTests.cs @@ -52,7 +52,6 @@ static string ToString(IEnumerable sequence) var editor = new WorkflowEditor(graphView.ServiceProvider, graphView); editor.UpdateLayout.Subscribe(graphView.UpdateGraphLayout); editor.UpdateSelection.Subscribe(graphView.UpdateSelection); - editor.Workflow = graphView.Workflow; var nodeSequence = editor.GetGraphValues().ToArray(); return (editor, assertIsReversible: () => diff --git a/Bonsai.Editor/EditorForm.Designer.cs b/Bonsai.Editor/EditorForm.Designer.cs index fec3e6423..d52171c9e 100644 --- a/Bonsai.Editor/EditorForm.Designer.cs +++ b/Bonsai.Editor/EditorForm.Designer.cs @@ -115,9 +115,9 @@ private void InitializeComponent() this.editExtensionsToolStripButton = new System.Windows.Forms.ToolStripButton(); this.reloadExtensionsToolStripButton = new System.Windows.Forms.ToolStripButton(); this.statusStrip = new System.Windows.Forms.StatusStrip(); + this.statusImageLabel = new System.Windows.Forms.ToolStripStatusLabel(); this.statusContextMenuStrip = new System.Windows.Forms.ContextMenuStrip(this.components); this.statusCopyToolStripMenuItem = new System.Windows.Forms.ToolStripMenuItem(); - this.statusImageLabel = new System.Windows.Forms.ToolStripStatusLabel(); this.toolboxSplitContainer = new Bonsai.Editor.SelectableSplitContainer(); this.toolboxTableLayoutPanel = new Bonsai.Editor.TableLayoutPanel(); this.searchTextBox = new Bonsai.Editor.CueBannerTextBox(); @@ -153,6 +153,10 @@ private void InitializeComponent() this.commandExecutor = new Bonsai.Design.CommandExecutor(); this.workflowFileWatcher = new System.IO.FileSystemWatcher(); this.exportImageDialog = new System.Windows.Forms.SaveFileDialog(); + this.explorerSplitContainer = new System.Windows.Forms.SplitContainer(); + this.explorerLayoutPanel = new Bonsai.Editor.TableLayoutPanel(); + this.explorerLabel = new Bonsai.Editor.Label(); + this.explorerTreeView = new Bonsai.Editor.ExplorerTreeView(); this.menuStrip.SuspendLayout(); this.toolStrip.SuspendLayout(); this.statusStrip.SuspendLayout(); @@ -180,6 +184,11 @@ private void InitializeComponent() this.propertiesLayoutPanel.SuspendLayout(); this.toolboxContextMenuStrip.SuspendLayout(); ((System.ComponentModel.ISupportInitialize)(this.workflowFileWatcher)).BeginInit(); + ((System.ComponentModel.ISupportInitialize)(this.explorerSplitContainer)).BeginInit(); + this.explorerSplitContainer.Panel1.SuspendLayout(); + this.explorerSplitContainer.Panel2.SuspendLayout(); + this.explorerSplitContainer.SuspendLayout(); + this.explorerLayoutPanel.SuspendLayout(); this.SuspendLayout(); // // menuStrip @@ -626,7 +635,7 @@ private void InitializeComponent() // welcomeToolStripMenuItem // this.welcomeToolStripMenuItem.Name = "welcomeToolStripMenuItem"; - this.welcomeToolStripMenuItem.Size = new System.Drawing.Size(152, 22); + this.welcomeToolStripMenuItem.Size = new System.Drawing.Size(180, 22); this.welcomeToolStripMenuItem.Text = "&Welcome..."; this.welcomeToolStripMenuItem.Click += new System.EventHandler(this.welcomeToolStripMenuItem_Click); // @@ -635,7 +644,7 @@ private void InitializeComponent() this.docsToolStripMenuItem.Image = ((System.Drawing.Image)(resources.GetObject("docsToolStripMenuItem.Image"))); this.docsToolStripMenuItem.Name = "docsToolStripMenuItem"; this.docsToolStripMenuItem.ShortcutKeys = System.Windows.Forms.Keys.F1; - this.docsToolStripMenuItem.Size = new System.Drawing.Size(152, 22); + this.docsToolStripMenuItem.Size = new System.Drawing.Size(180, 22); this.docsToolStripMenuItem.Text = "&View Help"; this.docsToolStripMenuItem.Click += new System.EventHandler(this.docsToolStripMenuItem_Click); // @@ -643,7 +652,7 @@ private void InitializeComponent() // this.forumToolStripMenuItem.Image = ((System.Drawing.Image)(resources.GetObject("forumToolStripMenuItem.Image"))); this.forumToolStripMenuItem.Name = "forumToolStripMenuItem"; - this.forumToolStripMenuItem.Size = new System.Drawing.Size(152, 22); + this.forumToolStripMenuItem.Size = new System.Drawing.Size(180, 22); this.forumToolStripMenuItem.Text = "Bonsai &Forums"; this.forumToolStripMenuItem.Click += new System.EventHandler(this.forumToolStripMenuItem_Click); // @@ -651,19 +660,19 @@ private void InitializeComponent() // this.reportBugToolStripMenuItem.Image = ((System.Drawing.Image)(resources.GetObject("reportBugToolStripMenuItem.Image"))); this.reportBugToolStripMenuItem.Name = "reportBugToolStripMenuItem"; - this.reportBugToolStripMenuItem.Size = new System.Drawing.Size(152, 22); + this.reportBugToolStripMenuItem.Size = new System.Drawing.Size(180, 22); this.reportBugToolStripMenuItem.Text = "&Report a Bug"; this.reportBugToolStripMenuItem.Click += new System.EventHandler(this.reportBugToolStripMenuItem_Click); // // toolStripSeparator5 // this.toolStripSeparator5.Name = "toolStripSeparator5"; - this.toolStripSeparator5.Size = new System.Drawing.Size(149, 6); + this.toolStripSeparator5.Size = new System.Drawing.Size(177, 6); // // aboutToolStripMenuItem // this.aboutToolStripMenuItem.Name = "aboutToolStripMenuItem"; - this.aboutToolStripMenuItem.Size = new System.Drawing.Size(152, 22); + this.aboutToolStripMenuItem.Size = new System.Drawing.Size(180, 22); this.aboutToolStripMenuItem.Text = "&About..."; this.aboutToolStripMenuItem.Click += new System.EventHandler(this.aboutToolStripMenuItem_Click); // @@ -900,6 +909,14 @@ private void InitializeComponent() this.statusStrip.TabIndex = 2; this.statusStrip.Text = "statusStrip"; // + // statusImageLabel + // + this.statusImageLabel.DisplayStyle = System.Windows.Forms.ToolStripItemDisplayStyle.Image; + this.statusImageLabel.Image = global::Bonsai.Editor.Properties.Resources.StatusReadyImage; + this.statusImageLabel.Name = "statusImageLabel"; + this.statusImageLabel.Size = new System.Drawing.Size(16, 17); + this.statusImageLabel.Text = "statusImageLabel"; + // // statusContextMenuStrip // this.statusContextMenuStrip.Items.AddRange(new System.Windows.Forms.ToolStripItem[] { @@ -917,14 +934,6 @@ private void InitializeComponent() this.statusCopyToolStripMenuItem.Text = "&Copy"; this.statusCopyToolStripMenuItem.Click += new System.EventHandler(this.statusCopyToolStripMenuItem_Click); // - // statusImageLabel - // - this.statusImageLabel.DisplayStyle = System.Windows.Forms.ToolStripItemDisplayStyle.Image; - this.statusImageLabel.Image = global::Bonsai.Editor.Properties.Resources.StatusReadyImage; - this.statusImageLabel.Name = "statusImageLabel"; - this.statusImageLabel.Size = new System.Drawing.Size(16, 17); - this.statusImageLabel.Text = "statusImageLabel"; - // // toolboxSplitContainer // this.toolboxSplitContainer.Dock = System.Windows.Forms.DockStyle.Fill; @@ -942,8 +951,8 @@ private void InitializeComponent() // this.toolboxSplitContainer.Panel2.Controls.Add(this.toolboxDescriptionPanel); this.toolboxSplitContainer.Selectable = true; - this.toolboxSplitContainer.Size = new System.Drawing.Size(197, 308); - this.toolboxSplitContainer.SplitterDistance = 245; + this.toolboxSplitContainer.Size = new System.Drawing.Size(197, 138); + this.toolboxSplitContainer.SplitterDistance = 75; this.toolboxSplitContainer.TabIndex = 1; this.toolboxSplitContainer.TabStop = false; // @@ -959,7 +968,7 @@ private void InitializeComponent() this.toolboxTableLayoutPanel.RowCount = 2; this.toolboxTableLayoutPanel.RowStyles.Add(new System.Windows.Forms.RowStyle(System.Windows.Forms.SizeType.Absolute, 23F)); this.toolboxTableLayoutPanel.RowStyles.Add(new System.Windows.Forms.RowStyle(System.Windows.Forms.SizeType.Percent, 100F)); - this.toolboxTableLayoutPanel.Size = new System.Drawing.Size(197, 245); + this.toolboxTableLayoutPanel.Size = new System.Drawing.Size(197, 75); this.toolboxTableLayoutPanel.TabIndex = 2; // // searchTextBox @@ -1000,15 +1009,13 @@ private void InitializeComponent() treeNode4, treeNode5, treeNode6}); - this.toolboxTreeView.Size = new System.Drawing.Size(197, 222); + this.toolboxTreeView.Size = new System.Drawing.Size(197, 52); this.toolboxTreeView.TabIndex = 0; - this.toolboxTreeView.ItemDrag += new System.Windows.Forms.ItemDragEventHandler(this.toolboxTreeView_ItemDrag); this.toolboxTreeView.AfterLabelEdit += new System.Windows.Forms.NodeLabelEditEventHandler(this.toolboxTreeView_AfterLabelEdit); + this.toolboxTreeView.ItemDrag += new System.Windows.Forms.ItemDragEventHandler(this.toolboxTreeView_ItemDrag); this.toolboxTreeView.AfterSelect += new System.Windows.Forms.TreeViewEventHandler(this.toolboxTreeView_AfterSelect); this.toolboxTreeView.NodeMouseDoubleClick += new System.Windows.Forms.TreeNodeMouseClickEventHandler(this.toolboxTreeView_NodeMouseDoubleClick); - this.toolboxTreeView.Enter += new System.EventHandler(this.toolboxTreeView_Enter); this.toolboxTreeView.KeyDown += new System.Windows.Forms.KeyEventHandler(this.toolboxTreeView_KeyDown); - this.toolboxTreeView.Leave += new System.EventHandler(this.toolboxTreeView_Leave); this.toolboxTreeView.MouseUp += new System.Windows.Forms.MouseEventHandler(this.toolboxTreeView_MouseUp); // // toolboxDescriptionPanel @@ -1088,9 +1095,9 @@ private void InitializeComponent() this.propertyGrid.Name = "propertyGrid"; this.propertyGrid.Size = new System.Drawing.Size(193, 245); this.propertyGrid.TabIndex = 0; + this.propertyGrid.Refreshed += new System.EventHandler(this.propertyGrid_Refreshed); this.propertyGrid.DragDrop += new System.Windows.Forms.DragEventHandler(this.propertyGrid_DragDrop); this.propertyGrid.DragEnter += new System.Windows.Forms.DragEventHandler(this.propertyGrid_DragEnter); - this.propertyGrid.Refreshed += new System.EventHandler(this.propertyGrid_Refreshed); this.propertyGrid.Validated += new System.EventHandler(this.propertyGrid_Validated); // // propertyGridContextMenuStrip @@ -1128,7 +1135,7 @@ private void InitializeComponent() // // panelSplitContainer.Panel1 // - this.panelSplitContainer.Panel1.Controls.Add(this.toolboxLayoutPanel); + this.panelSplitContainer.Panel1.Controls.Add(this.explorerSplitContainer); // // panelSplitContainer.Panel2 // @@ -1151,7 +1158,7 @@ private void InitializeComponent() this.toolboxLayoutPanel.RowCount = 2; this.toolboxLayoutPanel.RowStyles.Add(new System.Windows.Forms.RowStyle(System.Windows.Forms.SizeType.Absolute, 23F)); this.toolboxLayoutPanel.RowStyles.Add(new System.Windows.Forms.RowStyle(System.Windows.Forms.SizeType.Percent, 100F)); - this.toolboxLayoutPanel.Size = new System.Drawing.Size(200, 340); + this.toolboxLayoutPanel.Size = new System.Drawing.Size(200, 170); this.toolboxLayoutPanel.TabIndex = 1; // // toolboxLabel @@ -1227,7 +1234,7 @@ private void InitializeComponent() this.findPreviousToolStripMenuItem, this.goToDefinitionToolStripMenuItem}); this.toolboxContextMenuStrip.Name = "toolboxContextMenuStrip"; - this.toolboxContextMenuStrip.Size = new System.Drawing.Size(207, 290); + this.toolboxContextMenuStrip.Size = new System.Drawing.Size(207, 268); // // toolboxDocsToolStripMenuItem // @@ -1347,6 +1354,64 @@ private void InitializeComponent() "*.tiff)|*.tif;*.tiff|PNG (*.png)|*.png|SVG (*.svg)|*.svg"; this.exportImageDialog.FilterIndex = 6; // + // explorerSplitContainer + // + this.explorerSplitContainer.Dock = System.Windows.Forms.DockStyle.Fill; + this.explorerSplitContainer.Location = new System.Drawing.Point(0, 0); + this.explorerSplitContainer.Name = "explorerSplitContainer"; + this.explorerSplitContainer.Orientation = System.Windows.Forms.Orientation.Horizontal; + // + // explorerSplitContainer.Panel1 + // + this.explorerSplitContainer.Panel1.Controls.Add(this.toolboxLayoutPanel); + // + // explorerSplitContainer.Panel2 + // + this.explorerSplitContainer.Panel2.Controls.Add(this.explorerLayoutPanel); + this.explorerSplitContainer.Size = new System.Drawing.Size(200, 340); + this.explorerSplitContainer.SplitterDistance = 170; + this.explorerSplitContainer.TabIndex = 2; + // + // explorerLayoutPanel + // + this.explorerLayoutPanel.ColumnCount = 1; + this.explorerLayoutPanel.ColumnStyles.Add(new System.Windows.Forms.ColumnStyle(System.Windows.Forms.SizeType.Percent, 100F)); + this.explorerLayoutPanel.Controls.Add(this.explorerTreeView, 0, 1); + this.explorerLayoutPanel.Controls.Add(this.explorerLabel, 0, 0); + this.explorerLayoutPanel.Dock = System.Windows.Forms.DockStyle.Fill; + this.explorerLayoutPanel.Location = new System.Drawing.Point(0, 0); + this.explorerLayoutPanel.Name = "explorerLayoutPanel"; + this.explorerLayoutPanel.Padding = new System.Windows.Forms.Padding(0, 6, 0, 0); + this.explorerLayoutPanel.RowCount = 2; + this.explorerLayoutPanel.RowStyles.Add(new System.Windows.Forms.RowStyle(System.Windows.Forms.SizeType.Absolute, 23F)); + this.explorerLayoutPanel.RowStyles.Add(new System.Windows.Forms.RowStyle(System.Windows.Forms.SizeType.Percent, 100F)); + this.explorerLayoutPanel.Size = new System.Drawing.Size(200, 166); + this.explorerLayoutPanel.TabIndex = 2; + // + // explorerLabel + // + this.explorerLabel.AutoSize = true; + this.explorerLabel.BackColor = System.Drawing.SystemColors.ScrollBar; + this.explorerLabel.Dock = System.Windows.Forms.DockStyle.Fill; + this.explorerLabel.Location = new System.Drawing.Point(3, 6); + this.explorerLabel.Margin = new System.Windows.Forms.Padding(3, 0, 0, 0); + this.explorerLabel.Name = "explorerLabel"; + this.explorerLabel.Size = new System.Drawing.Size(197, 23); + this.explorerLabel.TabIndex = 2; + this.explorerLabel.Text = "Explorer"; + this.explorerLabel.TextAlign = System.Drawing.ContentAlignment.MiddleLeft; + // + // explorerTreeView + // + this.explorerTreeView.BorderStyle = System.Windows.Forms.BorderStyle.None; + this.explorerTreeView.Dock = System.Windows.Forms.DockStyle.Fill; + this.explorerTreeView.Location = new System.Drawing.Point(0, 29); + this.explorerTreeView.Margin = new System.Windows.Forms.Padding(3, 0, 0, 3); + this.explorerTreeView.Name = "explorerTreeView"; + this.explorerTreeView.Size = new System.Drawing.Size(200, 137); + this.explorerTreeView.TabIndex = 3; + this.explorerTreeView.NodeMouseDoubleClick += new System.Windows.Forms.TreeNodeMouseClickEventHandler(explorerTreeView_NodeMouseDoubleClick); + // // EditorForm // this.AutoScaleDimensions = new System.Drawing.SizeF(6F, 13F); @@ -1398,6 +1463,12 @@ private void InitializeComponent() this.propertiesLayoutPanel.PerformLayout(); this.toolboxContextMenuStrip.ResumeLayout(false); ((System.ComponentModel.ISupportInitialize)(this.workflowFileWatcher)).EndInit(); + this.explorerSplitContainer.Panel1.ResumeLayout(false); + this.explorerSplitContainer.Panel2.ResumeLayout(false); + ((System.ComponentModel.ISupportInitialize)(this.explorerSplitContainer)).EndInit(); + this.explorerSplitContainer.ResumeLayout(false); + this.explorerLayoutPanel.ResumeLayout(false); + this.explorerLayoutPanel.PerformLayout(); this.ResumeLayout(false); this.PerformLayout(); @@ -1522,6 +1593,10 @@ private void InitializeComponent() private System.Windows.Forms.ToolStripMenuItem toolboxDocsToolStripMenuItem; private System.Windows.Forms.ToolStripSeparator toolStripSeparator11; private System.Windows.Forms.ToolStripMenuItem watchToolStripMenuItem; + private System.Windows.Forms.SplitContainer explorerSplitContainer; + private Bonsai.Editor.TableLayoutPanel explorerLayoutPanel; + private Bonsai.Editor.Label explorerLabel; + private Bonsai.Editor.ExplorerTreeView explorerTreeView; } } diff --git a/Bonsai.Editor/EditorForm.cs b/Bonsai.Editor/EditorForm.cs index dbda20623..553c55b3a 100644 --- a/Bonsai.Editor/EditorForm.cs +++ b/Bonsai.Editor/EditorForm.cs @@ -39,6 +39,7 @@ public partial class EditorForm : Form static readonly char[] ToolboxArgumentSeparator = new[] { ' ' }; static readonly object ExtensionsDirectoryChanged = new object(); static readonly object WorkflowValidating = new object(); + static readonly object WorkflowValidated = new object(); int version; int saveVersion; @@ -65,6 +66,7 @@ public partial class EditorForm : Form readonly BehaviorSubject updatesAvailable; readonly FormScheduler formScheduler; readonly TypeVisualizerMap typeVisualizers; + readonly VisualizerLayoutMap visualizerSettings; readonly List workflowElements; readonly List workflowExtensions; readonly WorkflowRuntimeExceptionCache exceptionCache; @@ -73,6 +75,7 @@ public partial class EditorForm : Form AttributeCollection browsableAttributes; DirectoryInfo extensionsPath; WorkflowBuilder workflowBuilder; + VisualizerDialogMap visualizerDialogs; WorkflowException workflowError; IDisposable running; bool debugging; @@ -152,6 +155,7 @@ public EditorForm( regularFont = new Font(toolboxDescriptionTextBox.Font, FontStyle.Regular); selectionFont = new Font(toolboxDescriptionTextBox.Font, FontStyle.Bold); typeVisualizers = new TypeVisualizerMap(); + visualizerSettings = new VisualizerLayoutMap(typeVisualizers); workflowElements = new List(); workflowExtensions = new List(); exceptionCache = new WorkflowRuntimeExceptionCache(); @@ -166,7 +170,6 @@ public EditorForm( definitionsPath = Project.GetDefinitionsTempPath(); editorControl = new WorkflowEditorControl(editorSite); editorControl.Enter += new EventHandler(editorControl_Enter); - editorControl.Workflow = workflowBuilder.Workflow; editorControl.Dock = DockStyle.Fill; workflowSplitContainer.Panel1.Controls.Add(editorControl); propertyGrid.BrowsableAttributes = browsableAttributes = DesignTimeAttributes; @@ -257,6 +260,10 @@ void RestoreEditorSettings() themeRenderer.ActiveTheme = EditorSettings.Instance.EditorTheme; editorControl.AnnotationPanelSize = (int)Math.Round( EditorSettings.Instance.AnnotationPanelSize * scaleFactor.Width); + explorerSplitContainer.SplitterDistance = (int)Math.Round( + EditorSettings.Instance.ExplorerSplitterDistance * scaleFactor.Width); + var toolboxBottomMargin = toolboxSplitContainer.Margin.Bottom; + toolboxSplitContainer.SplitterDistance = toolboxSplitContainer.Height - propertiesSplitContainer.SplitterDistance - toolboxBottomMargin; } void CloseEditorForm() @@ -264,6 +271,8 @@ void CloseEditorForm() Application.RemoveMessageFilter(hotKeys); EditorSettings.Instance.AnnotationPanelSize = (int)Math.Round( editorControl.AnnotationPanelSize * inverseScaleFactor.Width); + EditorSettings.Instance.ExplorerSplitterDistance = (int)Math.Round( + explorerSplitContainer.SplitterDistance * inverseScaleFactor.Width); var desktopBounds = WindowState != FormWindowState.Normal ? RestoreBounds : Bounds; EditorSettings.Instance.DesktopBounds = ScaleBounds(desktopBounds, inverseScaleFactor); if (WindowState == FormWindowState.Minimized) @@ -294,6 +303,7 @@ protected override void OnLoad(EventArgs e) handler => FormClosed -= handler); InitializeSubjectSources().TakeUntil(formClosed).Subscribe(); InitializeWorkflowFileWatcher().TakeUntil(formClosed).Subscribe(); + InitializeWorkflowExplorerWatcher().TakeUntil(formClosed).Subscribe(); updatesAvailable.TakeUntil(formClosed).ObserveOn(formScheduler).Subscribe(HandleUpdatesAvailable); var currentDirectory = Project.GetCurrentBaseDirectory(out bool currentDirectoryRestricted); @@ -311,7 +321,10 @@ protected override void OnLoad(EventArgs e) InitializeEditorToolboxTypes(); var shutdown = ShutdownSequence(); - var initialization = InitializeToolbox().Merge(InitializeTypeVisualizers()).TakeLast(1).Finally(shutdown.Dispose).ObserveOn(Scheduler.Default); + var initialization = InitializeToolbox().Merge(InitializeTypeVisualizers()) + .TakeLast(1) + .Finally(shutdown.Dispose) + .ObserveOn(Scheduler.Default); if (validFileName && OpenWorkflow(initialFileName, false)) { foreach (var assignment in propertyAssignments) @@ -341,14 +354,10 @@ protected override void ScaleControl(SizeF factor, BoundsSpecified specified) inverseScaleFactor = new SizeF(1f / factor.Width, 1f / factor.Height); #if NETFRAMEWORK - const float DefaultToolboxSplitterDistance = 245f; var workflowSplitterScale = EditorSettings.IsRunningOnMono ? 0.5f / factor.Width : 1.0f; - var toolboxSplitterScale = EditorSettings.IsRunningOnMono ? 0.75f / factor.Height : 1.0f; - toolboxSplitterScale *= DefaultToolboxSplitterDistance / toolboxSplitContainer.SplitterDistance; panelSplitContainer.SplitterDistance = (int)(panelSplitContainer.SplitterDistance * factor.Height); workflowSplitContainer.SplitterDistance = (int)(workflowSplitContainer.SplitterDistance * workflowSplitterScale * factor.Height); propertiesSplitContainer.SplitterDistance = (int)(propertiesSplitContainer.SplitterDistance * factor.Height); - toolboxSplitContainer.SplitterDistance = (int)(toolboxSplitContainer.SplitterDistance * toolboxSplitterScale * factor.Height); workflowSplitContainer.Panel1.Padding = new Padding(0, 6, 0, 2); var imageSize = toolStrip.ImageScalingSize; @@ -361,6 +370,9 @@ protected override void ScaleControl(SizeF factor, BoundsSpecified specified) statusStrip.ImageScalingSize = toolStrip.ImageScalingSize; propertyGrid.LargeButtons = scalingFactor >= 2; } +#else + const float PropertiesSplitterScale = 0.4f; // correct for overshoot rescaling in .NET core + propertiesSplitContainer.SplitterDistance = (int)(propertiesSplitContainer.SplitterDistance * PropertiesSplitterScale * factor.Height); #endif base.ScaleControl(factor, specified); } @@ -427,7 +439,7 @@ IEnumerable EditorToolboxTypes() IObservable InitializeSubjectSources() { - var selectionChanged = Observable.FromEventPattern( + var selectedViewChanged = Observable.FromEventPattern( handler => selectionModel.SelectionChanged += handler, handler => selectionModel.SelectionChanged -= handler) .Select(evt => selectionModel.SelectedView) @@ -437,7 +449,7 @@ IObservable InitializeSubjectSources() handler => Events.RemoveHandler(WorkflowValidating, handler)) .Select(evt => selectionModel.SelectedView); return Observable - .Merge(selectionChanged, workflowValidating) + .Merge(selectedViewChanged, workflowValidating) .Do(view => { toolboxTreeView.BeginUpdate(); @@ -468,6 +480,35 @@ IObservable InitializeSubjectSources() .Select(xs => Unit.Default); } + IObservable InitializeWorkflowExplorerWatcher() + { + var selectedViewChanged = Observable.FromEventPattern( + handler => selectionModel.SelectionChanged += handler, + handler => selectionModel.SelectionChanged -= handler) + .Select(evt => selectionModel.SelectedView.WorkflowPath) + .DistinctUntilChanged() + .Do(view => explorerTreeView.SelectNode(editorControl.WorkflowGraphView.WorkflowPath)) + .IgnoreElements() + .Select(xs => Unit.Default); + + var workflowValidated = Observable.FromEventPattern( + handler => Events.AddHandler(WorkflowValidated, handler), + handler => Events.RemoveHandler(WorkflowValidated, handler)) + .Select(evt => selectionModel.SelectedView) + .Merge(Observable.Return(selectionModel.SelectedView)); + return Observable.Merge(selectedViewChanged, workflowValidated.Do(view => + { + if (workflowBuilder.Workflow == null) + return; + + explorerTreeView.UpdateWorkflow( + GetProjectDisplayName(), + workflowBuilder); + }) + .IgnoreElements() + .Select(xs => Unit.Default)); + } + IObservable InitializeWorkflowFileWatcher() { var extensionsDirectoryChanged = Observable.FromEventPattern( @@ -739,8 +780,9 @@ void ClearWorkflow() ClearWorkflowError(); saveWorkflowDialog.FileName = null; workflowBuilder.Workflow.Clear(); - editorControl.VisualizerLayout = null; - editorControl.Workflow = workflowBuilder.Workflow; + editorControl.ResetNavigation(); + editorSite.ValidateWorkflow(); + visualizerSettings.Clear(); ResetProjectStatus(); UpdateTitle(); } @@ -771,7 +813,7 @@ bool OpenWorkflow(string fileName, bool setWorkingDirectory) UpdateWorkflowDirectory(fileName, setWorkingDirectory); if (EditorResult == EditorResult.ReloadEditor) return false; - editorControl.Workflow = workflowBuilder.Workflow; + editorControl.ResetNavigation(); if (workflowBuilder.Workflow.Count > 0 && !editorControl.WorkflowGraphView.GraphView.Nodes.Any()) { try { workflowBuilder.Workflow.Build(); } @@ -785,8 +827,8 @@ bool OpenWorkflow(string fileName, bool setWorkingDirectory) } workflowBuilder = PrepareWorkflow(workflowBuilder, workflowVersion, out bool upgraded); - editorControl.VisualizerLayout = null; - editorControl.Workflow = workflowBuilder.Workflow; + saveWorkflowDialog.FileName = fileName; + editorControl.ResetNavigation(); editorSite.ValidateWorkflow(); #pragma warning disable CS0612 // Support for deprecated layout config files @@ -795,13 +837,16 @@ bool OpenWorkflow(string fileName, bool setWorkingDirectory) { using (var reader = XmlReader.Create(layoutPath)) { - try { editorControl.VisualizerLayout = (VisualizerLayout)VisualizerLayout.Serializer.Deserialize(reader); } + try + { + var visualizerLayout = (VisualizerLayout)VisualizerLayout.Serializer.Deserialize(reader); + visualizerSettings.SetVisualizerLayout(workflowBuilder, visualizerLayout); + } catch (InvalidOperationException) { } } } #pragma warning restore CS0612 // Support for deprecated layout config files - saveWorkflowDialog.FileName = fileName; ResetProjectStatus(); if (upgraded) { @@ -850,13 +895,13 @@ bool SaveWorkflow(string fileName) if (!SaveWorkflowBuilder(fileName, serializerWorkflowBuilder)) return false; saveVersion = version; - editorControl.UpdateVisualizerLayout(); - if (editorControl.VisualizerLayout != null) + var visualizerLayout = visualizerSettings.GetVisualizerLayout(workflowBuilder); + if (visualizerLayout != null) { var layoutPath = new FileInfo(Project.GetLayoutConfigPath(fileName)); layoutPath.Directory?.Create(); - SaveVisualizerLayout(layoutPath.FullName, editorControl.VisualizerLayout); + SaveVisualizerLayout(layoutPath.FullName, visualizerLayout); #pragma warning disable CS0612 // Support for deprecated layout config files var legacyLayoutPath = new FileInfo(Project.GetLegacyLayoutConfigPath(fileName)); if (legacyLayoutPath.Exists) @@ -926,6 +971,11 @@ void OnWorkflowValidating(EventArgs e) (Events[WorkflowValidating] as EventHandler)?.Invoke(this, e); } + void OnWorkflowValidated(EventArgs e) + { + (Events[WorkflowValidated] as EventHandler)?.Invoke(this, e); + } + void OnExtensionsDirectoryChanged(EventArgs e) { (Events[ExtensionsDirectoryChanged] as EventHandler)?.Invoke(this, e); @@ -1145,7 +1195,11 @@ IDisposable ShutdownSequence() running = null; building = false; workflowWatch.Stop(); - editorControl.UpdateVisualizerLayout(); + if (visualizerDialogs != null) + { + visualizerSettings.Update(visualizerDialogs); + visualizerDialogs = null; + } UpdateTitle(); })); } @@ -1157,10 +1211,11 @@ void StartWorkflow(bool debug) building = true; debugging = debug; ClearWorkflowError(); + visualizerDialogs = visualizerSettings.CreateVisualizerDialogs(workflowBuilder); LayoutHelper.SetWorkflowNotifications(workflowBuilder.Workflow, debug); - if (!debug && editorControl.VisualizerLayout != null) + if (!debug) { - LayoutHelper.SetLayoutNotifications(editorControl.VisualizerLayout); + LayoutHelper.SetLayoutNotifications(workflowBuilder.Workflow, visualizerDialogs); } running = Observable.Using( @@ -1176,6 +1231,7 @@ void StartWorkflow(bool debug) workflowWatch.Start(workflowBuilder.Workflow); statusTextLabel.Text = Resources.RunningStatus; statusImageLabel.Image = statusRunningImage; + visualizerDialogs.Show(visualizerSettings, editorSite, this); editorSite.OnWorkflowStarted(EventArgs.Empty); Activate(); })); @@ -1251,7 +1307,7 @@ void ClearWorkflowError() { if (workflowError != null) { - ClearExceptionBuilderNode(editorControl.WorkflowGraphView, workflowError); + ClearExceptionBuilderNode(workflowError); } exceptionCache.Clear(); @@ -1266,91 +1322,34 @@ void HighlightWorkflowError() } } - void ClearExceptionBuilderNode(WorkflowGraphView workflowView, WorkflowException e) + void ClearExceptionBuilderNode(WorkflowException ex) { - GraphNode graphNode = null; - if (workflowView != null) - { - graphNode = workflowView.FindGraphNode(e.Builder); - if (graphNode != null) - { - workflowView.GraphView.Invalidate(graphNode); - graphNode.Highlight = false; - } - } - - if (e.InnerException is WorkflowException nestedException) - { - WorkflowGraphView nestedEditor = null; - if (workflowView != null) - { - var editorLauncher = workflowView.GetWorkflowEditorLauncher(graphNode); - nestedEditor = editorLauncher != null && editorLauncher.Visible ? editorLauncher.WorkflowGraphView : null; - } + var workflowPath = WorkflowEditorPath.GetExceptionPath(workflowBuilder, ex); + var selectedView = selectionModel.SelectedView; + selectedView.ClearGraphNode(workflowPath); - ClearExceptionBuilderNode(nestedEditor, nestedException); - } - else - { - statusStrip.ContextMenuStrip = null; - statusTextLabel.Text = Resources.ReadyStatus; - statusImageLabel.Image = Resources.StatusReadyImage; - } + statusStrip.ContextMenuStrip = null; + statusTextLabel.Text = Resources.ReadyStatus; + statusImageLabel.Image = Resources.StatusReadyImage; + explorerTreeView.SetNodeStatus(ExplorerNodeStatus.Ready); } void HighlightExceptionBuilderNode(WorkflowException ex, bool showMessageBox) { - HighlightExceptionBuilderNode(editorControl.WorkflowGraphView, ex, showMessageBox); - } - - void HighlightExceptionBuilderNode(WorkflowGraphView workflowView, WorkflowException ex, bool showMessageBox) - { - GraphNode graphNode = null; - if (workflowView != null) - { - graphNode = workflowView.FindGraphNode(ex.Builder); - if (graphNode == null) - { - throw new InvalidOperationException(Resources.ExceptionNodeNotFound_Error); - } - - workflowView.GraphView.Invalidate(graphNode); - if (showMessageBox) workflowView.GraphView.SelectedNode = graphNode; - graphNode.Highlight = true; - } - - var nestedException = ex.InnerException as WorkflowException; - if (nestedException != null) - { - WorkflowGraphView nestedEditor = null; - if (workflowView != null) - { - var editorLauncher = workflowView.GetWorkflowEditorLauncher(graphNode); - if (editorLauncher != null) - { - if (building && editorLauncher.Visible) workflowView.LaunchWorkflowView(graphNode); - nestedEditor = editorLauncher.WorkflowGraphView; - } - } + var workflowPath = WorkflowEditorPath.GetExceptionPath(workflowBuilder, ex); + var pathElements = workflowPath.GetPathElements(); + var selectedView = selectionModel.SelectedView; + selectedView.HighlightGraphNode(workflowPath, showMessageBox); - HighlightExceptionBuilderNode(nestedEditor, nestedException, showMessageBox); - } - else + var buildException = ex is WorkflowBuildException; + statusTextLabel.Text = ex.Message; + statusStrip.ContextMenuStrip = statusContextMenuStrip; + statusImageLabel.Image = buildException ? Resources.StatusBlockedImage : Resources.StatusCriticalImage; + explorerTreeView.SetNodeStatus(pathElements, ExplorerNodeStatus.Blocked); + if (showMessageBox) { - if (workflowView != null) - { - workflowView.GraphView.Select(); - } - - var buildException = ex is WorkflowBuildException; var errorCaption = buildException ? Resources.BuildError_Caption : Resources.RuntimeError_Caption; - statusTextLabel.Text = ex.Message; - statusStrip.ContextMenuStrip = statusContextMenuStrip; - statusImageLabel.Image = buildException ? Resources.StatusBlockedImage : Resources.StatusCriticalImage; - if (showMessageBox) - { - editorSite.ShowError(ex.Message, errorCaption); - } + editorSite.ShowError(ex.Message, errorCaption); } } @@ -1385,33 +1384,21 @@ void HandleWorkflowCompleted() else clearErrors(); } - void HighlightExpression(WorkflowGraphView workflowView, ExpressionScope scope) + void SelectBuilderNode(ExpressionBuilder builder) { - if (workflowView == null) + var builderPath = WorkflowEditorPath.GetBuilderPath(workflowBuilder, builder); + if (builderPath != null) { - throw new ArgumentNullException(nameof(workflowView)); - } + var selectedView = selectionModel.SelectedView; + selectedView.WorkflowPath = builderPath.Parent; - var graphNode = workflowView.FindGraphNode(scope.Value); - if (graphNode != null) - { - workflowView.GraphView.SelectedNode = graphNode; - var innerScope = scope.InnerScope; - if (innerScope != null) - { - workflowView.LaunchWorkflowView(graphNode); - var editorLauncher = workflowView.GetWorkflowEditorLauncher(graphNode); - if (editorLauncher != null) - { - HighlightExpression(editorLauncher.WorkflowGraphView, innerScope); - } - } - else + var graphNode = selectedView.FindGraphNode(builderPath.Resolve(workflowBuilder)); + if (graphNode == null) { - var ownerForm = workflowView.EditorControl.ParentForm; - if (ownerForm != null) ownerForm.Activate(); - workflowView.SelectGraphNode(graphNode); + throw new InvalidOperationException(Resources.ExceptionNodeNotFound_Error); } + + selectedView.SelectGraphNode(graphNode); } } @@ -1509,13 +1496,13 @@ selectedNode.Tag is not null && void editorControl_Enter(object sender, EventArgs e) { var selectedView = selectionModel.SelectedView; - if (selectedView != null && selectedView.Launcher != null) + if (selectedView != null) { var container = selectedView.EditorControl; if (container != null && container != editorControl && hotKeys.TabState) { container.ParentForm.Activate(); - var forward = Form.ModifierKeys.HasFlag(Keys.Shift); + var forward = ModifierKeys.HasFlag(Keys.Shift); container.SelectNextControl(container.ActiveControl, forward, true, true, false); } } @@ -1534,6 +1521,13 @@ private void GetSelectionDescription(object[] selectedObjects, out string displa description = objectDescriptions.Length == 1 ? objectDescriptions[0] : string.Empty; } + private string GetProjectDisplayName() + { + return !string.IsNullOrEmpty(saveWorkflowDialog.FileName) + ? Path.GetFileNameWithoutExtension(saveWorkflowDialog.FileName) + : "Workflow"; + } + private void UpdatePropertyGrid() { var selectedObjects = selectionModel.SelectedNodes.Select(node => @@ -1568,18 +1562,16 @@ private void UpdatePropertyGrid() if (!hasSelectedObjects && selectedView != null) { // Select externalized properties - var launcher = selectedView.Launcher; - if (launcher != null) + if (selectedView.WorkflowPath != null) { - displayName = ElementHelper.GetElementName(launcher.Builder); - description = ElementHelper.GetElementDescription(launcher.Builder); + var builder = ExpressionBuilder.Unwrap(selectedView.WorkflowPath.Resolve(workflowBuilder)); + displayName = ElementHelper.GetElementName(builder); + description = ElementHelper.GetElementDescription(builder); } else { description = workflowBuilder.Description ?? Resources.WorkflowPropertiesDescription; - displayName = !string.IsNullOrEmpty(saveWorkflowDialog.FileName) - ? Path.GetFileNameWithoutExtension(saveWorkflowDialog.FileName) - : editorControl.ActiveTab.TabPage.Text; + displayName = GetProjectDisplayName(); } propertyGrid.SelectedObject = selectedView.Workflow; @@ -1768,19 +1760,6 @@ void UpdateTreeViewDescription() else UpdateDescriptionTextBox(string.Empty, string.Empty, toolboxDescriptionTextBox); } - void UpdateTreeViewSelection(bool focused) - { - var selectedNode = toolboxTreeView.SelectedNode; - if (toolboxTreeView.Tag != selectedNode) - { - if (toolboxTreeView.Tag is TreeNode previousNode) previousNode.BackColor = Color.Empty; - toolboxTreeView.Tag = selectedNode; - } - - if (selectedNode == null) return; - selectedNode.BackColor = focused ? Color.Empty : themeRenderer.ToolStripRenderer.ColorTable.InactiveCaption; - } - void SelectTreeViewSubjectNode(string subjectName) { var subjectCategory = toolboxCategories[SubjectCategoryName]; @@ -1841,11 +1820,16 @@ void FindNextMatch(Func predicate, ExpressionBuilder cu var match = workflowBuilder.Find(predicate, current, findPrevious); if (match != null) { - var scope = workflowBuilder.GetExpressionScope(match); - HighlightExpression(editorControl.WorkflowGraphView, scope); + SelectBuilderNode(match); } } + private void explorerTreeView_NodeMouseDoubleClick(object sender, TreeNodeMouseClickEventArgs e) + { + var workflowPath = (WorkflowEditorPath)e.Node?.Tag; + editorControl.WorkflowGraphView.WorkflowPath = workflowPath; + } + private void toolboxTreeView_KeyDown(object sender, KeyEventArgs e) { var selectedNode = toolboxTreeView.SelectedNode; @@ -1888,8 +1872,7 @@ private void toolboxTreeView_KeyDown(object sender, KeyEventArgs e) } else { - var scope = workflowBuilder.GetExpressionScope(definition.Subject); - HighlightExpression(editorControl.WorkflowGraphView, scope); + SelectBuilderNode(definition.Subject); } } } @@ -1929,7 +1912,6 @@ private void toolboxTreeView_NodeMouseDoubleClick(object sender, TreeNodeMouseCl private void toolboxTreeView_AfterSelect(object sender, TreeViewEventArgs e) { UpdateTreeViewDescription(); - UpdateTreeViewSelection(toolboxTreeView.Focused); } private void toolboxTreeView_MouseUp(object sender, MouseEventArgs e) @@ -1982,16 +1964,6 @@ private void toolboxTreeView_MouseUp(object sender, MouseEventArgs e) } } - private void toolboxTreeView_Enter(object sender, EventArgs e) - { - UpdateTreeViewSelection(true); - } - - private void toolboxTreeView_Leave(object sender, EventArgs e) - { - UpdateTreeViewSelection(false); - } - private void insertAfterToolStripMenuItem_Click(object sender, EventArgs e) { toolboxTreeView_KeyDown(sender, new KeyEventArgs(Keys.Return)); @@ -2455,6 +2427,11 @@ public bool DesignMode public string Name { get; set; } + public string GetProjectDisplayName() + { + return siteForm.GetProjectDisplayName(); + } + public object GetService(Type serviceType) { if (serviceType == typeof(ExpressionBuilderGraph)) @@ -2482,6 +2459,16 @@ public object GetService(Type serviceType) return siteForm.typeVisualizers; } + if (serviceType == typeof(VisualizerLayoutMap)) + { + return siteForm.visualizerSettings; + } + + if (serviceType == typeof(VisualizerDialogMap)) + { + return siteForm.visualizerDialogs; + } + if (serviceType == typeof(ThemeRenderer)) { return siteForm.themeRenderer; @@ -2499,11 +2486,10 @@ public object GetService(Type serviceType) if (serviceType == typeof(DialogTypeVisualizer)) { - var selectedView = siteForm.selectionModel.SelectedView; var selectedNode = siteForm.selectionModel.SelectedNodes.FirstOrDefault(); - if (selectedNode != null) + if (selectedNode != null && selectedNode.Value is InspectBuilder builder && + siteForm.visualizerDialogs.TryGetValue(builder, out VisualizerDialogLauncher visualizerDialog)) { - var visualizerDialog = selectedView.GetVisualizerDialogLauncher(selectedNode); var visualizer = visualizerDialog.Visualizer; if (visualizer.IsValueCreated) { @@ -2654,6 +2640,11 @@ public void SelectNextControl(bool forward) siteForm.Activate(); } + public void SelectBuilderNode(ExpressionBuilder builder) + { + siteForm.SelectBuilderNode(builder); + } + public bool ValidateWorkflow() { if (siteForm.running == null) @@ -2663,10 +2654,12 @@ public bool ValidateWorkflow() siteForm.OnWorkflowValidating(EventArgs.Empty); siteForm.ClearWorkflowError(); siteForm.workflowBuilder.Workflow.Build(); + siteForm.OnWorkflowValidated(EventArgs.Empty); } catch (WorkflowBuildException ex) { siteForm.HandleWorkflowError(ex); + siteForm.OnWorkflowValidated(EventArgs.Empty); return false; } } @@ -2728,8 +2721,7 @@ public void ShowDefinition(object component) var definition = siteForm.workflowBuilder.GetSubjectDefinition(model.Workflow, namedElement.Name); if (definition != null) { - var scope = siteForm.workflowBuilder.GetExpressionScope(definition.Subject); - siteForm.HighlightExpression(siteForm.editorControl.WorkflowGraphView, scope); + siteForm.SelectBuilderNode(definition.Subject); return; } } @@ -3003,10 +2995,12 @@ private void InitializeTheme() toolboxSplitContainer.BackColor = panelColor; toolboxLabel.BackColor = colorTable.SeparatorDark; toolboxLabel.ForeColor = ForeColor; - toolboxTreeView.BackColor = panelColor; - toolboxTreeView.ForeColor = windowText; + toolboxTreeView.Renderer = themeRenderer.ToolStripRenderer; toolboxDescriptionTextBox.BackColor = panelColor; toolboxDescriptionTextBox.ForeColor = ForeColor; + explorerTreeView.Renderer = themeRenderer.ToolStripRenderer; + explorerLabel.BackColor = colorTable.SeparatorDark; + explorerLabel.ForeColor = ForeColor; propertiesDescriptionTextBox.BackColor = panelColor; propertiesDescriptionTextBox.ForeColor = ForeColor; menuStrip.ForeColor = SystemColors.ControlText; @@ -3023,7 +3017,7 @@ private void InitializeTheme() } propertiesLayoutPanel.RowStyles[0].Height -= labelOffset; toolboxLayoutPanel.RowStyles[0].Height -= labelOffset; - UpdateTreeViewSelection(toolboxTreeView.Focused); + explorerLayoutPanel.RowStyles[0].Height -= labelOffset; propertyGrid.Refresh(); } diff --git a/Bonsai.Editor/EditorSettings.cs b/Bonsai.Editor/EditorSettings.cs index 1759cf951..9c5edaaa3 100644 --- a/Bonsai.Editor/EditorSettings.cs +++ b/Bonsai.Editor/EditorSettings.cs @@ -21,6 +21,7 @@ sealed class EditorSettings internal EditorSettings(string path) { AnnotationPanelSize = 400; + ExplorerSplitterDistance = 300; settingsPath = path; } @@ -37,6 +38,8 @@ public static EditorSettings Instance public int AnnotationPanelSize { get; set; } + public int ExplorerSplitterDistance { get; set; } + public RecentlyUsedFileCollection RecentlyUsedFiles { get { return recentlyUsedFiles; } @@ -74,6 +77,11 @@ static EditorSettings Load() int.TryParse(reader.ReadElementContentAsString(), out int annotationPanelSize); settings.AnnotationPanelSize = annotationPanelSize; } + else if (reader.Name == nameof(ExplorerSplitterDistance)) + { + int.TryParse(reader.ReadElementContentAsString(), out int explorerSplitterDistance); + settings.ExplorerSplitterDistance = explorerSplitterDistance; + } else if (reader.Name == nameof(DesktopBounds)) { reader.ReadToFollowing(nameof(Rectangle.X)); @@ -120,6 +128,7 @@ public void Save() writer.WriteElementString(nameof(WindowState), WindowState.ToString()); writer.WriteElementString(nameof(EditorTheme), EditorTheme.ToString()); writer.WriteElementString(nameof(AnnotationPanelSize), AnnotationPanelSize.ToString(CultureInfo.InvariantCulture)); + writer.WriteElementString(nameof(ExplorerSplitterDistance), ExplorerSplitterDistance.ToString(CultureInfo.InvariantCulture)); writer.WriteStartElement(nameof(DesktopBounds)); writer.WriteElementString(nameof(Rectangle.X), DesktopBounds.X.ToString(CultureInfo.InvariantCulture)); diff --git a/Bonsai.Editor/ExplorerTreeView.cs b/Bonsai.Editor/ExplorerTreeView.cs new file mode 100644 index 000000000..d49b138d4 --- /dev/null +++ b/Bonsai.Editor/ExplorerTreeView.cs @@ -0,0 +1,212 @@ +using System; +using System.Collections.Generic; +using System.Drawing; +using System.Linq; +using System.Windows.Forms; +using Bonsai.Editor.GraphModel; +using Bonsai.Editor.Properties; +using Bonsai.Expressions; + +namespace Bonsai.Editor +{ + class ExplorerTreeView : ToolboxTreeView + { + bool activeDoubleClick; + readonly ImageList imageList; + readonly ImageList stateImageList; + + public ExplorerTreeView() + { + imageList = new(); + stateImageList = new(); + imageList.Images.Add(Resources.WorkflowEditableImage); + imageList.Images.Add(Resources.WorkflowReadOnlyImage); +#if NETFRAMEWORK + stateImageList.Images.Add(Resources.StatusReadyImage); + stateImageList.Images.Add(Resources.StatusBlockedImage); +#else + // TreeView.StateImageList.ImageSize is internally scaled according to initial system DPI (not font). + // To avoid excessive scaling of images we must prepare correctly sized ImageList beforehand. + const float DefaultDpi = 96f; + using var graphics = CreateGraphics(); + var dpiScale = graphics.DpiY / DefaultDpi; + stateImageList.ImageSize = new Size( + (int)(16 * dpiScale), + (int)(16 * dpiScale)); + stateImageList.Images.Add(ResizeMakeBorder(Resources.StatusReadyImage, stateImageList.ImageSize)); + stateImageList.Images.Add(ResizeMakeBorder(Resources.StatusBlockedImage, stateImageList.ImageSize)); + + static Bitmap ResizeMakeBorder(Bitmap original, Size newSize) + { + //TODO: DrawImageUnscaledAndClipped gives best results but blending is not great + var image = new Bitmap(newSize.Width, newSize.Height, original.PixelFormat); + using var graphics = Graphics.FromImage(image); + var offsetX = (newSize.Width - original.Width) / 2; + var offsetY = (newSize.Height - original.Height) / 2; + graphics.DrawImageUnscaledAndClipped(original, new Rectangle(offsetX, offsetY, original.Width, original.Height)); + return image; + } +#endif + + StateImageList = stateImageList; + ImageList = imageList; + } + + protected override void OnBeforeCollapse(TreeViewCancelEventArgs e) + { + if (activeDoubleClick && e.Action == TreeViewAction.Collapse) + e.Cancel = true; + activeDoubleClick = false; + base.OnBeforeCollapse(e); + } + + protected override void OnBeforeExpand(TreeViewCancelEventArgs e) + { + if (activeDoubleClick && e.Action == TreeViewAction.Expand) + e.Cancel = true; + activeDoubleClick = false; + base.OnBeforeExpand(e); + } + + protected override void OnMouseDown(MouseEventArgs e) + { + activeDoubleClick = e.Clicks > 1; + base.OnMouseDown(e); + } + + public void UpdateWorkflow(string name, WorkflowBuilder workflowBuilder) + { + BeginUpdate(); + Nodes.Clear(); + + var rootNode = Nodes.Add(name); + AddWorkflow(rootNode.Nodes, null, workflowBuilder.Workflow, ExplorerNodeType.Editable); + + static void AddWorkflow( + TreeNodeCollection nodes, + WorkflowEditorPath basePath, + ExpressionBuilderGraph workflow, + ExplorerNodeType parentNodeType) + { + for (int i = 0; i < workflow.Count; i++) + { + var builder = workflow[i].Value; + if (ExpressionBuilder.Unwrap(builder) is IWorkflowExpressionBuilder workflowBuilder && + workflowBuilder.Workflow != null) + { + var nodeType = parentNodeType == ExplorerNodeType.ReadOnly || workflowBuilder is IncludeWorkflowBuilder + ? ExplorerNodeType.ReadOnly + : ExplorerNodeType.Editable; + var displayName = ExpressionBuilder.GetElementDisplayName(builder); + var builderPath = new WorkflowEditorPath(i, basePath); + var node = nodes.Add(displayName); + node.ImageIndex = node.SelectedImageIndex = GetImageIndex(nodeType); + node.Tag = builderPath; + AddWorkflow(node.Nodes, builderPath, workflowBuilder.Workflow, nodeType); + } + } + } + + SetNodeStatus(ExplorerNodeStatus.Ready); + rootNode.Expand(); + EndUpdate(); + } + + public void SelectNode(WorkflowEditorPath path) + { + SelectNode(Nodes, path); + } + + bool SelectNode(TreeNodeCollection nodes, WorkflowEditorPath path) + { + foreach (TreeNode node in nodes) + { + var nodePath = (WorkflowEditorPath)node.Tag; + if (nodePath == path) + { + SelectedNode = node; + return true; + } + + var selected = SelectNode(node.Nodes, path); + if (selected) break; + } + + return false; + } + + private static int GetImageIndex(ExplorerNodeType status) + { + return status switch + { + ExplorerNodeType.Editable => 0, + ExplorerNodeType.ReadOnly => 1, + _ => throw new ArgumentException("Invalid node type.", nameof(status)) + }; + } + + private static int GetStateImageIndex(ExplorerNodeStatus status) + { + return status switch + { + ExplorerNodeStatus.Ready => -1, + ExplorerNodeStatus.Blocked => 1, + _ => throw new ArgumentException("Invalid node status.", nameof(status)) + }; + } + + public void SetNodeStatus(ExplorerNodeStatus status) + { + var imageIndex = GetStateImageIndex(status); + SetNodeImageIndex(Nodes, imageIndex); + + static void SetNodeImageIndex(TreeNodeCollection nodes, int index) + { + foreach (TreeNode node in nodes) + { + if (node.StateImageIndex == index) + continue; + + node.StateImageIndex = index; + SetNodeImageIndex(node.Nodes, index); + } + } + } + + public void SetNodeStatus(IEnumerable pathElements, ExplorerNodeStatus status) + { + var nodes = Nodes; + var imageIndex = GetStateImageIndex(status); + foreach (var path in pathElements.Prepend(null)) + { + var found = false; + for (int n = 0; n < nodes.Count; n++) + { + var groupNode = nodes[n]; + if ((WorkflowEditorPath)groupNode.Tag == path) + { + groupNode.StateImageIndex = imageIndex; + nodes = groupNode.Nodes; + found = true; + break; + } + } + + if (!found) + break; + } + } + } + + enum ExplorerNodeType + { + Editable, + ReadOnly + } + + enum ExplorerNodeStatus + { + Ready, + Blocked + } +} diff --git a/Bonsai.Editor/GraphModel/WorkflowBuilderExtensions.cs b/Bonsai.Editor/GraphModel/WorkflowBuilderExtensions.cs index b6b8b8f0a..3b3750a6b 100644 --- a/Bonsai.Editor/GraphModel/WorkflowBuilderExtensions.cs +++ b/Bonsai.Editor/GraphModel/WorkflowBuilderExtensions.cs @@ -91,47 +91,6 @@ public static IEnumerable GetDependentExpressions(this SubjectD } } } - - public static ExpressionScope GetExpressionScope(this WorkflowBuilder source, ExpressionBuilder target) - { - return GetExpressionScope(source.Workflow, target); - } - - static ExpressionScope GetExpressionScope(ExpressionBuilderGraph source, ExpressionBuilder target) - { - foreach (var node in source) - { - var builder = ExpressionBuilder.Unwrap(node.Value); - if (builder == target) - { - return new ExpressionScope(node.Value, innerScope: null); - } - - if (builder is IWorkflowExpressionBuilder workflowBuilder) - { - var innerScope = GetExpressionScope(workflowBuilder.Workflow, target); - if (innerScope != null) - { - return new ExpressionScope(node.Value, innerScope); - } - } - } - - return null; - } - } - - class ExpressionScope - { - public ExpressionScope(ExpressionBuilder value, ExpressionScope innerScope) - { - Value = value; - InnerScope = innerScope; - } - - public ExpressionBuilder Value { get; } - - public ExpressionScope InnerScope { get; } } class SubjectDefinition diff --git a/Bonsai.Editor/GraphModel/WorkflowEditor.cs b/Bonsai.Editor/GraphModel/WorkflowEditor.cs index 77e5d8660..082f6fcb6 100644 --- a/Bonsai.Editor/GraphModel/WorkflowEditor.cs +++ b/Bonsai.Editor/GraphModel/WorkflowEditor.cs @@ -20,10 +20,10 @@ class WorkflowEditor readonly IGraphView graphView; readonly Subject error; readonly Subject updateLayout; - readonly Subject updateParentLayout; readonly Subject invalidateLayout; + readonly Subject workflowPathChanged; readonly Subject> updateSelection; - readonly Subject closeWorkflowEditor; + WorkflowEditorPath workflowPath; public WorkflowEditor(IServiceProvider provider, IGraphView view) { @@ -32,25 +32,53 @@ public WorkflowEditor(IServiceProvider provider, IGraphView view) commandExecutor = (CommandExecutor)provider.GetService(typeof(CommandExecutor)); error = new Subject(); updateLayout = new Subject(); - updateParentLayout = new Subject(); invalidateLayout = new Subject(); + workflowPathChanged = new Subject(); updateSelection = new Subject>(); - closeWorkflowEditor = new Subject(); + ResetNavigation(); } - public ExpressionBuilderGraph Workflow { get; set; } + public ExpressionBuilderGraph Workflow { get; private set; } + + public bool IsReadOnly { get; private set; } + + public WorkflowEditorPath WorkflowPath + { + get { return workflowPath; } + private set + { + workflowPath = value; + var workflowBuilder = (WorkflowBuilder)serviceProvider.GetService(typeof(WorkflowBuilder)); + if (workflowPath != null) + { + var builder = ExpressionBuilder.Unwrap(workflowPath.Resolve(workflowBuilder, out bool isReadOnly)); + if (builder is not IWorkflowExpressionBuilder workflowExpressionBuilder) + { + throw new ArgumentException(Resources.InvalidWorkflowPath_Error, nameof(value)); + } + + Workflow = workflowExpressionBuilder.Workflow; + IsReadOnly = isReadOnly; + } + else + { + Workflow = workflowBuilder.Workflow; + IsReadOnly = false; + } + updateLayout.OnNext(false); + workflowPathChanged.OnNext(workflowPath); + } + } public IObservable Error => error; public IObservable UpdateLayout => updateLayout; - public IObservable UpdateParentLayout => updateParentLayout; - public IObservable InvalidateLayout => invalidateLayout; - public IObservable> UpdateSelection => updateSelection; + public IObservable WorkflowPathChanged => workflowPathChanged; - public IObservable CloseWorkflowEditor => closeWorkflowEditor; + public IObservable> UpdateSelection => updateSelection; private static Node FindWorkflowValue(ExpressionBuilderGraph workflow, ExpressionBuilder value) { @@ -170,8 +198,6 @@ private void AddWorkflowInput(ExpressionBuilderGraph workflow, Node nodes) @@ -2070,6 +2086,28 @@ public GraphNode FindGraphNode(ExpressionBuilder value) { return graphView.Nodes.SelectMany(layer => layer).FirstOrDefault(n => n.Value == value); } + + public void NavigateTo(WorkflowEditorPath path) + { + if (path == workflowPath) + return; + + var previousPath = workflowPath; + var selectedNodes = graphView.SelectedNodes.ToArray(); + var restoreSelectedNodes = CreateUpdateSelectionDelegate(selectedNodes); + commandExecutor.Execute( + () => WorkflowPath = path, + () => + { + WorkflowPath = previousPath; + restoreSelectedNodes(); + }); + } + + public void ResetNavigation() + { + WorkflowPath = null; + } } enum CreateGraphNodeType diff --git a/Bonsai.Editor/GraphModel/WorkflowEditorPath.cs b/Bonsai.Editor/GraphModel/WorkflowEditorPath.cs new file mode 100644 index 000000000..96c91e3a5 --- /dev/null +++ b/Bonsai.Editor/GraphModel/WorkflowEditorPath.cs @@ -0,0 +1,171 @@ +using System; +using System.Collections.Generic; +using Bonsai.Expressions; + +namespace Bonsai.Editor.GraphModel +{ + class WorkflowEditorPath : IEquatable + { + public WorkflowEditorPath() + { + } + + public WorkflowEditorPath(int index, WorkflowEditorPath parent) + { + Index = index; + Parent = parent; + } + + public int Index { get; } + + public WorkflowEditorPath Parent { get; } + + public IEnumerable GetPathElements() + { + var stack = new Stack(); + var pathElement = this; + while (pathElement != null) + { + stack.Push(pathElement); + pathElement = pathElement.Parent; + } + + foreach (var element in stack) + { + yield return element; + } + } + + public ExpressionBuilder Resolve(WorkflowBuilder workflowBuilder) + { + return Resolve(workflowBuilder, out _); + } + + public ExpressionBuilder Resolve(WorkflowBuilder workflowBuilder, out bool isReadOnly) + { + isReadOnly = false; + var builder = default(ExpressionBuilder); + var workflow = workflowBuilder.Workflow; + foreach (var pathElement in GetPathElements()) + { + if (workflow == null) + { + throw new ArgumentException($"Unable to resolve workflow editor path.", nameof(workflowBuilder)); + } + + builder = workflow[pathElement.Index].Value; + if (ExpressionBuilder.Unwrap(builder) is IWorkflowExpressionBuilder nestedWorkflowBuilder) + { + workflow = nestedWorkflowBuilder.Workflow; + isReadOnly |= nestedWorkflowBuilder is IncludeWorkflowBuilder; + } + else workflow = null; + } + + return builder; + } + + public static WorkflowEditorPath GetExceptionPath(WorkflowBuilder workflowBuilder, WorkflowException ex) + { + return GetExceptionPath(workflowBuilder.Workflow, ex, null); + } + + static WorkflowEditorPath GetExceptionPath(ExpressionBuilderGraph workflow, WorkflowException ex, WorkflowEditorPath parent) + { + for (int i = 0; i < workflow.Count; i++) + { + var builder = workflow[i].Value; + if (builder == ex.Builder) + { + var path = new WorkflowEditorPath(i, parent); + if (ex.InnerException is WorkflowException nestedEx && + ExpressionBuilder.Unwrap(ex.Builder) is IWorkflowExpressionBuilder workflowBuilder) + { + return GetExceptionPath(workflowBuilder.Workflow, nestedEx, path); + } + else return path; + } + } + + return null; + } + + public static WorkflowEditorPath GetBuilderPath(WorkflowBuilder workflowBuilder, ExpressionBuilder builder) + { + return GetBuilderPath(workflowBuilder.Workflow, ExpressionBuilder.Unwrap(builder), new List()); + } + + static WorkflowEditorPath GetBuilderPath(ExpressionBuilderGraph workflow, ExpressionBuilder target, List pathElements) + { + for (int i = 0; i < workflow.Count; i++) + { + var builder = ExpressionBuilder.Unwrap(workflow[i].Value); + if (builder == target) + { + pathElements.Add(i); + return GetBuilderPath(pathElements); + } + + if (builder is IWorkflowExpressionBuilder workflowBuilder) + { + pathElements.Add(i); + var path = GetBuilderPath(workflowBuilder.Workflow, target, pathElements); + if (path is not null) + return path; + pathElements.RemoveAt(pathElements.Count - 1); + } + } + + return null; + } + + static WorkflowEditorPath GetBuilderPath(List pathElements) + { + WorkflowEditorPath path = null; + foreach (var index in pathElements) + { + path = new WorkflowEditorPath(index, path); + } + return path; + } + + public override int GetHashCode() + { + var hash = 107; + hash += Index.GetHashCode() * 13; + hash += (Parent?.GetHashCode()).GetValueOrDefault() * 13; + return hash; + } + + public override bool Equals(object obj) + { + if (obj is WorkflowEditorPath path) + return Equals(path); + + return false; + } + + public bool Equals(WorkflowEditorPath other) + { + if (ReferenceEquals(this, other)) + return true; + + if (other == null) + return false; + + return Index == other.Index && Parent == other.Parent; + } + + public static bool operator ==(WorkflowEditorPath left, WorkflowEditorPath right) + { + if (left is not null) return left.Equals(right); + else return right is null; + } + + public static bool operator !=(WorkflowEditorPath left, WorkflowEditorPath right) + { + if (left is not null) return !left.Equals(right); + else return right is not null; + } + } +} diff --git a/Bonsai.Editor/GraphView/WorkflowEditorControl.Designer.cs b/Bonsai.Editor/GraphView/WorkflowEditorControl.Designer.cs index 92b79d164..d34e5cc7d 100644 --- a/Bonsai.Editor/GraphView/WorkflowEditorControl.Designer.cs +++ b/Bonsai.Editor/GraphView/WorkflowEditorControl.Designer.cs @@ -191,7 +191,6 @@ private void InitializeComponent() this.AutoScaleDimensions = new System.Drawing.SizeF(6F, 13F); this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font; this.Controls.Add(this.splitContainer); - this.MinimumSize = new System.Drawing.Size(250, 125); this.Name = "WorkflowEditorControl"; this.Size = new System.Drawing.Size(300, 200); this.tabContextMenuStrip.ResumeLayout(false); diff --git a/Bonsai.Editor/GraphView/WorkflowEditorControl.cs b/Bonsai.Editor/GraphView/WorkflowEditorControl.cs index 36c5234e8..e11b7d90f 100644 --- a/Bonsai.Editor/GraphView/WorkflowEditorControl.cs +++ b/Bonsai.Editor/GraphView/WorkflowEditorControl.cs @@ -3,7 +3,6 @@ using System.Drawing; using System.Windows.Forms; using Bonsai.Expressions; -using Bonsai.Design; using Bonsai.Editor.Themes; using Bonsai.Editor.GraphModel; @@ -18,17 +17,12 @@ partial class WorkflowEditorControl : UserControl Padding? adjustMargin; public WorkflowEditorControl(IServiceProvider provider) - : this(provider, false) - { - } - - public WorkflowEditorControl(IServiceProvider provider, bool readOnly) { InitializeComponent(); serviceProvider = provider ?? throw new ArgumentNullException(nameof(provider)); editorService = (IWorkflowEditorService)provider.GetService(typeof(IWorkflowEditorService)); themeRenderer = (ThemeRenderer)provider.GetService(typeof(ThemeRenderer)); - workflowTab = InitializeTab(workflowTabPage, readOnly, null); + workflowTab = InitializeTab(workflowTabPage); annotationPanel.ThemeRenderer = themeRenderer; annotationPanel.LinkClicked += (sender, e) => { EditorDialog.OpenUrl(e.LinkText); }; annotationPanel.CloseRequested += delegate { CollapseAnnotationPanel(); }; @@ -60,18 +54,6 @@ public int AnnotationPanelSize } } - public VisualizerLayout VisualizerLayout - { - get { return WorkflowGraphView.VisualizerLayout; } - set { WorkflowGraphView.VisualizerLayout = value; } - } - - public ExpressionBuilderGraph Workflow - { - get { return WorkflowGraphView.Workflow; } - set { WorkflowGraphView.Workflow = value; } - } - public void ExpandAnnotationPanel(ExpressionBuilder builder) { annotationPanel.Tag = builder; @@ -91,11 +73,6 @@ public void CollapseAnnotationPanel() annotationPanel.Tag = null; } - public void UpdateVisualizerLayout() - { - WorkflowGraphView.UpdateVisualizerLayout(); - } - public TabPageController ActiveTab { get; private set; } public int ItemHeight @@ -103,9 +80,9 @@ public int ItemHeight get { return tabControl.DisplayRectangle.Y; } } - TabPageController InitializeTab(TabPage tabPage, bool readOnly, Control container) + TabPageController InitializeTab(TabPage tabPage) { - var workflowGraphView = new WorkflowGraphView(serviceProvider, this, readOnly); + var workflowGraphView = new WorkflowGraphView(serviceProvider, this); workflowGraphView.BackColorChanged += (sender, e) => { tabPage.BackColor = workflowGraphView.BackColor; @@ -118,42 +95,47 @@ TabPageController InitializeTab(TabPage tabPage, bool readOnly, Control containe var tabState = new TabPageController(tabPage, workflowGraphView); tabPage.Tag = tabState; tabPage.SuspendLayout(); - if (container != null) + + var breadcrumbs = new WorkflowPathNavigationControl(serviceProvider); + breadcrumbs.WorkflowPath = null; + breadcrumbs.WorkflowPathMouseClick += (sender, e) => workflowGraphView.WorkflowPath = e.Path; + workflowGraphView.WorkflowPathChanged += (sender, e) => { - container.TextChanged += (sender, e) => tabState.Text = container.Text; - container.Controls.Add(workflowGraphView); - tabPage.Controls.Add(container); - } - else tabPage.Controls.Add(workflowGraphView); + breadcrumbs.WorkflowPath = workflowGraphView.WorkflowPath; + }; + + var navigationPanel = new TableLayoutPanel(); + navigationPanel.Dock = DockStyle.Fill; + navigationPanel.ColumnCount = 1; + navigationPanel.RowCount = 2; + navigationPanel.RowStyles.Add(new RowStyle(SizeType.Absolute, breadcrumbs.Height)); + navigationPanel.RowStyles.Add(new RowStyle(SizeType.AutoSize)); + navigationPanel.Controls.Add(breadcrumbs); + navigationPanel.Controls.Add(workflowGraphView); + + // TODO: This should be handled by docking, but some strange interaction prevents shrinking to min size + navigationPanel.Layout += (sender, e) => breadcrumbs.Width = navigationPanel.Width; + breadcrumbs.Width = navigationPanel.Width; + + tabPage.Controls.Add(navigationPanel); tabPage.BackColor = workflowGraphView.BackColor; tabPage.ResumeLayout(false); tabPage.PerformLayout(); return tabState; } - public TabPageController CreateTab(IWorkflowExpressionBuilder builder, bool readOnly, Control owner) + public TabPageController CreateTab(WorkflowEditorPath workflowPath) { var tabPage = new TabPage(); tabPage.Padding = workflowTabPage.Padding; tabPage.UseVisualStyleBackColor = workflowTabPage.UseVisualStyleBackColor; - var tabState = InitializeTab(tabPage, readOnly || builder is IncludeWorkflowBuilder, owner); - tabState.Text = ExpressionBuilder.GetElementDisplayName(builder); - tabState.WorkflowGraphView.Workflow = builder.Workflow; - tabState.Builder = builder; + var tabState = InitializeTab(tabPage); + tabState.WorkflowGraphView.WorkflowPath = workflowPath; tabControl.TabPages.Add(tabPage); return tabState; } - public void SelectTab(IWorkflowExpressionBuilder builder) - { - var tabPage = FindTab(builder); - if (tabPage != null) - { - tabControl.SelectTab(tabPage); - } - } - public void SelectTab(WorkflowGraphView workflowGraphView) { var tabPage = (TabPage)workflowGraphView.Tag; @@ -164,23 +146,24 @@ public void SelectTab(WorkflowGraphView workflowGraphView) } } - public void CloseTab(IWorkflowExpressionBuilder builder) + public void ResetNavigation() { - var tabPage = FindTab(builder); - if (tabPage != null) + CloseAll(); + WorkflowGraphView.Editor.ResetNavigation(); + } + + void CloseAll() + { + while (tabControl.TabCount > 1) { - var tabState = (TabPageController)tabPage.Tag; - CloseTab(tabState); + CloseTab(tabControl.TabPages[1]); } } void CloseTab(TabPage tabPage) { var tabState = (TabPageController)tabPage.Tag; - if (tabState.Builder != null) - { - CloseTab(tabState); - } + CloseTab(tabState); } void CloseTab(TabPageController tabState) @@ -202,47 +185,12 @@ void CloseTab(TabPageController tabState) } } - public void RefreshTab(IWorkflowExpressionBuilder builder) - { - var tabPage = FindTab(builder); - if (tabPage != null) - { - var tabState = (TabPageController)tabPage.Tag; - RefreshTab(tabState); - } - } - - void RefreshTab(TabPageController tabState) - { - var builder = tabState.Builder; - var workflowGraphView = tabState.WorkflowGraphView; - if (builder != null && builder.Workflow != workflowGraphView.Workflow) - { - CloseTab(tabState); - } - } - - TabPage FindTab(IWorkflowExpressionBuilder builder) - { - foreach (TabPage tabPage in tabControl.TabPages) - { - var tabState = (TabPageController)tabPage.Tag; - if (tabState.Builder == builder) - { - return tabPage; - } - } - - return null; - } - void ActivateTab(TabPage tabPage) { var tabState = tabPage != null ? (TabPageController)tabPage.Tag : null; if (tabState != null && ActiveTab != tabState) { ActiveTab = tabState; - RefreshTab(ActiveTab); ActiveTab.UpdateSelection(); } } @@ -285,8 +233,6 @@ public TabPageController(TabPage tabPage, WorkflowGraphView graphView) public TabPage TabPage { get; private set; } - public IWorkflowExpressionBuilder Builder { get; set; } - public WorkflowGraphView WorkflowGraphView { get; private set; } public string Text @@ -301,7 +247,7 @@ public string Text void UpdateDisplayText() { - TabPage.Text = displayText + (WorkflowGraphView.ReadOnly ? ReadOnlySuffix : string.Empty) + CloseSuffix; + TabPage.Text = displayText + (WorkflowGraphView.IsReadOnly ? ReadOnlySuffix : string.Empty); } public void UpdateSelection() @@ -361,7 +307,7 @@ void tabControl_MouseUp(object sender, MouseEventArgs e) var tabState = (TabPageController)selectedTab.Tag; var tabRect = tabControl.GetTabRect(tabControl.SelectedIndex); - if (tabState.Builder != null && tabRect.Contains(e.Location)) + if (selectedTab != workflowTabPage && tabRect.Contains(e.Location)) { using (var graphics = selectedTab.CreateGraphics()) { @@ -414,10 +360,7 @@ private void closeToolStripMenuItem_Click(object sender, EventArgs e) private void closeAllToolStripMenuItem_Click(object sender, EventArgs e) { - while (tabControl.TabCount > 1) - { - CloseTab(tabControl.TabPages[1]); - } + CloseAll(); } protected override void ScaleControl(SizeF factor, BoundsSpecified specified) diff --git a/Bonsai.Editor/GraphView/WorkflowGraphView.cs b/Bonsai.Editor/GraphView/WorkflowGraphView.cs index b6ac60c26..a40f5ada0 100644 --- a/Bonsai.Editor/GraphView/WorkflowGraphView.cs +++ b/Bonsai.Editor/GraphView/WorkflowGraphView.cs @@ -22,10 +22,10 @@ namespace Bonsai.Editor.GraphView { partial class WorkflowGraphView : UserControl { - static readonly Action EmptyAction = () => { }; static readonly Cursor InvalidSelectionCursor = Cursors.No; static readonly Cursor MoveSelectionCursor = Cursors.SizeAll; static readonly Cursor AlternateSelectionCursor = Cursors.UpArrow; + static readonly object WorkflowPathChangedEvent = new(); const int RightMouseButton = 0x2; const int ShiftModifier = 0x4; @@ -37,7 +37,6 @@ partial class WorkflowGraphView : UserControl public const string BonsaiExtension = ".bonsai"; int dragKeyState; - bool editorLaunching; bool isContextMenuSource; GraphNode dragHighlight; IEnumerable dragSelection; @@ -46,32 +45,18 @@ partial class WorkflowGraphView : UserControl readonly IWorkflowEditorState editorState; readonly IWorkflowEditorService editorService; readonly TypeVisualizerMap typeVisualizerMap; - readonly Dictionary workflowEditorMapping; + readonly VisualizerLayoutMap visualizerSettings; readonly IServiceProvider serviceProvider; readonly IUIService uiService; readonly ThemeRenderer themeRenderer; readonly WorkflowWatch workflowWatch; readonly IDefinitionProvider definitionProvider; - Dictionary visualizerMapping; - VisualizerLayout visualizerLayout; - ExpressionBuilderGraph workflow; - public WorkflowGraphView(IServiceProvider provider, WorkflowEditorControl owner, bool readOnly) + public WorkflowGraphView(IServiceProvider provider, WorkflowEditorControl owner) { - if (provider == null) - { - throw new ArgumentNullException(nameof(provider)); - } - - if (owner == null) - { - throw new ArgumentNullException(nameof(owner)); - } - + EditorControl = owner ?? throw new ArgumentNullException(nameof(owner)); + serviceProvider = provider ?? throw new ArgumentNullException(nameof(provider)); InitializeComponent(); - EditorControl = owner; - serviceProvider = provider; - ReadOnly = readOnly; Editor = new WorkflowEditor(provider, graphView); uiService = (IUIService)provider.GetService(typeof(IUIService)); themeRenderer = (ThemeRenderer)provider.GetService(typeof(ThemeRenderer)); @@ -81,13 +66,12 @@ public WorkflowGraphView(IServiceProvider provider, WorkflowEditorControl owner, selectionModel = (WorkflowSelectionModel)provider.GetService(typeof(WorkflowSelectionModel)); editorService = (IWorkflowEditorService)provider.GetService(typeof(IWorkflowEditorService)); typeVisualizerMap = (TypeVisualizerMap)provider.GetService(typeof(TypeVisualizerMap)); + visualizerSettings = (VisualizerLayoutMap)provider.GetService(typeof(VisualizerLayoutMap)); editorState = (IWorkflowEditorState)provider.GetService(typeof(IWorkflowEditorState)); - workflowEditorMapping = new Dictionary(); workflowWatch = (WorkflowWatch)provider.GetService(typeof(WorkflowWatch)); workflowWatch.Update += WorkflowWatch_Tick; graphView.HandleDestroyed += graphView_HandleDestroyed; - editorState.WorkflowStarted += editorService_WorkflowStarted; themeRenderer.ThemeChanged += themeRenderer_ThemeChanged; InitializeTheme(); InitializeViewBindings(); @@ -131,15 +115,16 @@ private void WorkflowWatch_Tick(object sender, EventArgs e) internal WorkflowEditor Editor { get; } - internal WorkflowEditorLauncher Launcher { get; set; } - internal WorkflowEditorControl EditorControl { get; } - internal bool ReadOnly { get; } + internal bool IsReadOnly + { + get { return Editor.IsReadOnly; } + } internal bool CanEdit { - get { return !ReadOnly && !editorState.WorkflowRunning; } + get { return !Editor.IsReadOnly && !editorState.WorkflowRunning; } } public GraphViewControl GraphView @@ -147,22 +132,26 @@ public GraphViewControl GraphView get { return graphView; } } - public VisualizerLayout VisualizerLayout + public WorkflowEditorPath WorkflowPath + { + get { return Editor.WorkflowPath; } + set { Editor.NavigateTo(value); } + } + + public event EventHandler WorkflowPathChanged { - get { return visualizerLayout; } - set { SetVisualizerLayout(value); } + add { Events.AddHandler(WorkflowPathChangedEvent, value); } + remove { Events.RemoveHandler(WorkflowPathChangedEvent, value); } } public ExpressionBuilderGraph Workflow { - get { return workflow; } - set - { - ClearEditorMapping(); - workflow = value; - Editor.Workflow = value; - UpdateEditorWorkflow(); - } + get { return Editor.Workflow; } + } + + private void OnWorkflowPathChanged(EventArgs e) + { + (Events[WorkflowPathChangedEvent] as EventHandler)?.Invoke(this, e); } public static ElementCategory GetToolboxElementCategory(TreeNode typeNode) @@ -177,109 +166,8 @@ public static ElementCategory GetToolboxElementCategory(TreeNode typeNode) return ElementCategory.Combinator; } - private Func CreateWindowOwnerSelectorDelegate() - { - return Launcher != null ? (Func)(() => Launcher.Owner) : () => graphView; - } - - private Action CreateUpdateEditorMappingDelegate(Action> action) - { - return Launcher != null - ? (Action)(() => action(Launcher.WorkflowGraphView.workflowEditorMapping)) - : () => action(workflowEditorMapping); - } - #region Model - private void HideWorkflowEditorLauncher(WorkflowEditorLauncher editorLauncher) - { - var visible = editorLauncher.Visible; - var serviceProvider = this.serviceProvider; - var windowSelector = CreateWindowOwnerSelectorDelegate(); - var activeTabClosing = editorLauncher.Container != null && - editorLauncher.Container.ActiveTab != null && - editorLauncher.Container.ActiveTab.WorkflowGraphView == editorLauncher.WorkflowGraphView; - commandExecutor.Execute( - editorLauncher.Hide, - () => - { - if (visible && editorLauncher.Builder.Workflow != null) - { - editorLauncher.Show(windowSelector(), serviceProvider); - if (editorLauncher.Container != null && activeTabClosing) - { - editorLauncher.Container.SelectTab(editorLauncher.WorkflowGraphView); - } - } - }); - } - - private void UpdateEditorWorkflow() - { - UpdateGraphLayout(validateWorkflow: false); - if (editorState.WorkflowRunning) - { - InitializeVisualizerMapping(); - } - } - - internal void HideEditorMapping() - { - foreach (var mapping in workflowEditorMapping) - { - mapping.Value.Hide(); - } - } - - private void ClearEditorMapping() - { - HideEditorMapping(); - workflowEditorMapping.Clear(); - } - - private void InitializeVisualizerMapping() - { - if (workflow == null) return; - visualizerMapping = LayoutHelper.CreateVisualizerMapping( - workflow, - visualizerLayout, - typeVisualizerMap, - serviceProvider, - graphView, - this); - } - - private void CloseWorkflowEditorLauncher(IWorkflowExpressionBuilder workflowExpressionBuilder) - { - CloseWorkflowEditorLauncher(workflowExpressionBuilder, true); - } - - private void CloseWorkflowEditorLauncher(IWorkflowExpressionBuilder workflowExpressionBuilder, bool removeEditorMapping) - { - if (workflowEditorMapping.TryGetValue(workflowExpressionBuilder, out WorkflowEditorLauncher editorLauncher)) - { - if (editorLauncher.Visible) - { - var workflowGraphView = editorLauncher.WorkflowGraphView; - foreach (var node in workflowGraphView.workflow) - { - var nestedBuilder = ExpressionBuilder.Unwrap(node.Value) as IWorkflowExpressionBuilder; - if (nestedBuilder != null) - { - workflowGraphView.CloseWorkflowEditorLauncher(nestedBuilder, removeEditorMapping); - } - } - } - - HideWorkflowEditorLauncher(editorLauncher); - var removeMapping = removeEditorMapping - ? CreateUpdateEditorMappingDelegate(editorMapping => editorMapping.Remove(workflowExpressionBuilder)) - : EmptyAction; - var addMapping = CreateUpdateEditorMappingDelegate(editorMapping => editorMapping[workflowExpressionBuilder] = editorLauncher); - commandExecutor.Execute(removeMapping, addMapping); - } - } - private void InsertWorkflow(ExpressionBuilderGraph workflow) { var branch = ModifierKeys.HasFlag(BranchModifier); @@ -425,6 +313,42 @@ internal void SelectGraphNode(GraphNode node) UpdateSelection(); } + internal void ClearGraphNode(WorkflowEditorPath path) + { + SetGraphNodeHighlight(path, false, false); + } + + internal void HighlightGraphNode(WorkflowEditorPath path, bool selectNode) + { + SetGraphNodeHighlight(path, selectNode, true); + } + + private void SetGraphNodeHighlight(WorkflowEditorPath path, bool selectNode, bool highlight) + { + if (selectNode) + WorkflowPath = path?.Parent; + + while (path != null) + { + if (path.Parent == WorkflowPath) + { + var builder = Workflow[path.Index].Value; + var graphNode = FindGraphNode(builder); + if (graphNode == null) + { + throw new InvalidOperationException(Resources.ExceptionNodeNotFound_Error); + } + + GraphView.Invalidate(graphNode); + if (selectNode) GraphView.SelectedNode = graphNode; + graphNode.Highlight = highlight; + break; + } + + path = path.Parent; + } + } + private bool HasDefaultEditor(ExpressionBuilder builder) { if (builder is IWorkflowExpressionBuilder) return true; @@ -536,10 +460,18 @@ private void LaunchVisualizer(GraphNode node) return; } - var visualizerLauncher = GetVisualizerDialogLauncher(node); - if (visualizerLauncher != null) + var builder = (InspectBuilder)Workflow[node.Index].Value; + var visualizerDialogs = (VisualizerDialogMap)serviceProvider.GetService(typeof(VisualizerDialogMap)); + if (visualizerDialogs != null) { - visualizerLauncher.Show(graphView, serviceProvider); + if (!visualizerDialogs.TryGetValue(builder, out VisualizerDialogLauncher visualizerLauncher)) + { + visualizerSettings.TryGetValue(builder, out VisualizerDialogSettings dialogSettings); + visualizerLauncher = visualizerDialogs.Add(builder, Workflow, dialogSettings); + } + + var ownerWindow = uiService.GetDialogOwnerWindow(); + visualizerLauncher.Show(ownerWindow, serviceProvider); } } @@ -581,85 +513,22 @@ private void LaunchDefinition(GraphNode node) } public void LaunchWorkflowView(GraphNode node) - { - CreateWorkflowView(node, null, Rectangle.Empty, launch: true, activate: true); - } - - private void CreateWorkflowView(GraphNode node, VisualizerLayout editorLayout, Rectangle bounds, bool launch, bool activate) { var builder = WorkflowEditor.GetGraphNodeBuilder(node); var disableBuilder = builder as DisableBuilder; var workflowExpressionBuilder = (disableBuilder != null ? disableBuilder.Builder : builder) as IWorkflowExpressionBuilder; - if (workflowExpressionBuilder == null || editorLaunching) return; - - editorLaunching = true; - var parentLaunching = Launcher != null && Launcher.ParentView.editorLaunching; - var compositeExecutor = new Lazy(() => - { - if (!parentLaunching) commandExecutor.BeginCompositeCommand(); - return commandExecutor; - }, false); - - try - { - if (!workflowEditorMapping.TryGetValue(workflowExpressionBuilder, out WorkflowEditorLauncher editorLauncher)) - { - Func parentSelector; - Func containerSelector; - if (workflowExpressionBuilder is IncludeWorkflowBuilder || - workflowExpressionBuilder is GroupWorkflowBuilder) - { - containerSelector = () => Launcher != null ? Launcher.WorkflowGraphView.EditorControl : EditorControl; - } - else containerSelector = () => null; - parentSelector = () => Launcher != null ? Launcher.WorkflowGraphView : this; - - editorLauncher = new WorkflowEditorLauncher(workflowExpressionBuilder, parentSelector, containerSelector); - editorLauncher.VisualizerLayout = editorLayout; - editorLauncher.Bounds = bounds; - var addEditorMapping = CreateUpdateEditorMappingDelegate(editorMapping => editorMapping.Add(workflowExpressionBuilder, editorLauncher)); - var removeEditorMapping = CreateUpdateEditorMappingDelegate(editorMapping => editorMapping.Remove(workflowExpressionBuilder)); - compositeExecutor.Value.Execute(addEditorMapping, removeEditorMapping); - } - - if (launch && (!editorLauncher.Visible || activate)) - { - var highlight = node.Highlight; - var visible = editorLauncher.Visible; - var editorService = this.editorService; - var serviceProvider = this.serviceProvider; - var windowSelector = CreateWindowOwnerSelectorDelegate(); - Action launchEditor = () => - { - if (editorLauncher.Builder.Workflow != null) - { - editorLauncher.Show(windowSelector(), serviceProvider); - if (editorLauncher.Container != null && !parentLaunching && activate) - { - editorLauncher.Container.SelectTab(editorLauncher.WorkflowGraphView); - } - - if (highlight && !visible) - { - editorService.RefreshEditor(); - } - } - }; - - if (visible) launchEditor(); - else compositeExecutor.Value.Execute(launchEditor, editorLauncher.Hide); - } - } - finally + if (workflowExpressionBuilder != null) { - if (compositeExecutor.IsValueCreated && !parentLaunching) - { - compositeExecutor.Value.EndCompositeCommand(); - } - editorLaunching = false; + var newPath = new WorkflowEditorPath(node.Index, WorkflowPath); + LaunchWorkflowPath(newPath); } } + private void LaunchWorkflowPath(WorkflowEditorPath path) + { + Editor.NavigateTo(path); + } + internal void UpdateSelection() { UpdateSelection(forceUpdate: false); @@ -675,166 +544,6 @@ internal void UpdateSelection(bool forceUpdate) } } - internal void CloseWorkflowView(IWorkflowExpressionBuilder workflowExpressionBuilder) - { - commandExecutor.BeginCompositeCommand(); - CloseWorkflowEditorLauncher(workflowExpressionBuilder, false); - commandExecutor.EndCompositeCommand(); - } - - public VisualizerDialogLauncher GetVisualizerDialogLauncher(GraphNode node) - { - VisualizerDialogLauncher visualizerDialog = null; - if (visualizerMapping != null && node?.Value is InspectBuilder inspectBuilder) - { - visualizerMapping.TryGetValue(inspectBuilder, out visualizerDialog); - } - - return visualizerDialog; - } - - public WorkflowEditorLauncher GetWorkflowEditorLauncher(GraphNode node) - { - var builder = WorkflowEditor.GetGraphNodeBuilder(node); - var disableBuilder = builder as DisableBuilder; - if (disableBuilder != null) builder = disableBuilder.Builder; - - var workflowExpressionBuilder = builder as IWorkflowExpressionBuilder; - if (workflowExpressionBuilder != null) - { - workflowEditorMapping.TryGetValue(workflowExpressionBuilder, out WorkflowEditorLauncher editorLauncher); - return editorLauncher; - } - - return null; - } - - private VisualizerDialogSettings CreateLayoutSettings(ExpressionBuilder builder) - { - VisualizerDialogSettings dialogSettings; - if (ExpressionBuilder.GetWorkflowElement(builder) is IWorkflowExpressionBuilder workflowExpressionBuilder && - workflowEditorMapping.TryGetValue(workflowExpressionBuilder, out WorkflowEditorLauncher editorLauncher)) - { - if (editorLauncher.Visible) editorLauncher.UpdateEditorLayout(); - dialogSettings = new WorkflowEditorSettings - { - EditorVisualizerLayout = editorLauncher.Visible ? editorLauncher.VisualizerLayout : null, - EditorDialogSettings = new VisualizerDialogSettings - { - Visible = editorLauncher.Visible, - Bounds = editorLauncher.Bounds, - Tag = editorLauncher - } - }; - } - else dialogSettings = new VisualizerDialogSettings(); - dialogSettings.Tag = builder; - return dialogSettings; - } - - private void SetVisualizerLayout(VisualizerLayout layout) - { - if (workflow == null) - { - throw new InvalidOperationException(Resources.VisualizerLayoutOnNullWorkflow_Error); - } - - visualizerLayout = layout ?? new VisualizerLayout(); - foreach (var node in workflow) - { - var layoutSettings = visualizerLayout.GetLayoutSettings(node.Value); - if (layoutSettings == null) - { - layoutSettings = CreateLayoutSettings(node.Value); - visualizerLayout.DialogSettings.Add(layoutSettings); - } - else layoutSettings.Tag = node.Value; - - var graphNode = graphView.Nodes.SelectMany(layer => layer).First(n => n.Value == node.Value); - if (layoutSettings is WorkflowEditorSettings workflowEditorSettings && - workflowEditorSettings.EditorDialogSettings.Tag == null) - { - var editorLayout = workflowEditorSettings.EditorVisualizerLayout; - var editorVisible = workflowEditorSettings.EditorDialogSettings.Visible; - var editorBounds = workflowEditorSettings.EditorDialogSettings.Bounds; - CreateWorkflowView(graphNode, - editorLayout, - editorBounds, - launch: editorVisible, - activate: false); - } - } - } - - public void UpdateVisualizerLayout() - { - var updatedLayout = new VisualizerLayout(); - var topologicalOrder = workflow.TopologicalSort(); - foreach (var node in topologicalOrder) - { - var builder = node.Value; - VisualizerDialogSettings dialogSettings; - if (visualizerMapping != null && - visualizerMapping.TryGetValue(builder as InspectBuilder, out VisualizerDialogLauncher visualizerDialog)) - { - var visible = visualizerDialog.Visible; - if (!editorState.WorkflowRunning) - { - visualizerDialog.Hide(); - } - - var visualizer = visualizerDialog.Visualizer; - dialogSettings = CreateLayoutSettings(builder); - dialogSettings.Visible = visible; - dialogSettings.Bounds = visualizerDialog.Bounds; - dialogSettings.WindowState = visualizerDialog.WindowState; - - if (visualizer.IsValueCreated) - { - var visualizerType = visualizer.Value.GetType(); - if (visualizerType.IsPublic) - { - dialogSettings.VisualizerTypeName = visualizerType.FullName; - dialogSettings.VisualizerSettings = LayoutHelper.SerializeVisualizerSettings( - visualizer.Value, - topologicalOrder); - } - } - } - else - { - dialogSettings = visualizerLayout.GetLayoutSettings(builder); - if (dialogSettings == null) dialogSettings = CreateLayoutSettings(builder); - else - { - if (ExpressionBuilder.Unwrap(builder) is IWorkflowExpressionBuilder workflowExpressionBuilder) - { - var updatedEditorSettings = CreateLayoutSettings(builder); - updatedEditorSettings.Bounds = dialogSettings.Bounds; - updatedEditorSettings.Visible = dialogSettings.Visible; - updatedEditorSettings.WindowState = dialogSettings.WindowState; - updatedEditorSettings.VisualizerTypeName = dialogSettings.VisualizerTypeName; - updatedEditorSettings.VisualizerSettings = dialogSettings.VisualizerSettings; - foreach (var mashup in dialogSettings.Mashups) - { - updatedEditorSettings.Mashups.Add(mashup); - } - - dialogSettings = updatedEditorSettings; - } - } - } - - updatedLayout.DialogSettings.Add(dialogSettings); - } - - visualizerLayout = updatedLayout; - if (!editorState.WorkflowRunning) - { - visualizerMapping = null; - } - } - public void RefreshSelection() { foreach (var node in graphView.SelectedNodes) @@ -852,12 +561,6 @@ void RefreshEditorNode(GraphNode node) { LaunchVisualizer(node); } - - var editor = GetWorkflowEditorLauncher(node); - if (editor != null && editor.Visible) - { - editor.UpdateEditorText(); - } } private void UpdateGraphLayout() @@ -867,20 +570,20 @@ private void UpdateGraphLayout() private void UpdateGraphLayout(bool validateWorkflow) { - graphView.Nodes = workflow.ConnectedComponentLayering(); + graphView.Nodes = Workflow.ConnectedComponentLayering(); graphView.Invalidate(); if (validateWorkflow) { editorService.ValidateWorkflow(); } - UpdateVisualizerLayout(); if (validateWorkflow) { EditorControl.SelectTab(this); if (EditorControl.AnnotationPanel.Tag is ExpressionBuilder builder) { - if (!EditorControl.Workflow.Descendants().Contains(builder)) + var workflowBuilder = (WorkflowBuilder)serviceProvider.GetService(typeof(WorkflowBuilder)); + if (!workflowBuilder.Workflow.Descendants().Contains(builder)) { EditorControl.AnnotationPanel.NavigateToString(string.Empty); EditorControl.AnnotationPanel.Tag = null; @@ -888,16 +591,13 @@ private void UpdateGraphLayout(bool validateWorkflow) } } UpdateSelection(); + editorService.RefreshEditor(); } private void InvalidateGraphLayout(bool validateWorkflow) { graphView.Refresh(); - if (Launcher != null) - { - Launcher.ParentView.InvalidateGraphLayout(validateWorkflow); - } - else if (validateWorkflow) + if (validateWorkflow) { editorService.ValidateWorkflow(); } @@ -968,8 +668,8 @@ private void graphView_DragEnter(object sender, DragEventArgs e) else if (e.Data.GetDataPresent(typeof(GraphNode))) { var graphViewNode = (GraphNode)e.Data.GetData(typeof(GraphNode)); - var node = WorkflowEditor.GetGraphNodeTag(workflow, graphViewNode, false); - if (node != null && workflow.Contains(node)) + var node = WorkflowEditor.GetGraphNodeTag(Workflow, graphViewNode, false); + if (node != null && Workflow.Contains(node)) { dragSelection = graphView.SelectedNodes; dragHighlight = graphViewNode; @@ -1165,22 +865,7 @@ private void graphView_KeyDown(object sender, KeyEventArgs e) if (e.KeyCode == Keys.Back && e.Modifiers == Keys.Control) { - if (Launcher != null && Launcher.ParentView != null) - { - var parentView = Launcher.ParentView; - var parentEditor = parentView.EditorControl; - var parentEditorForm = parentEditor.ParentForm; - if (EditorControl.ParentForm != parentEditorForm) - { - parentEditorForm.Activate(); - } - - var parentNode = parentView.Workflow.FirstOrDefault(node => ExpressionBuilder.Unwrap(node.Value) == Launcher.Builder); - if (parentNode != null) - { - parentView.SelectBuilderNode(parentNode.Value); - } - } + LaunchWorkflowPath(WorkflowPath?.Parent); } if (CanEdit) @@ -1306,16 +991,10 @@ private void graphView_NodeMouseLeave(object sender, GraphNodeMouseEventArgs e) private void graphView_HandleDestroyed(object sender, EventArgs e) { - editorState.WorkflowStarted -= editorService_WorkflowStarted; themeRenderer.ThemeChanged -= themeRenderer_ThemeChanged; workflowWatch.Update -= WorkflowWatch_Tick; } - private void editorService_WorkflowStarted(object sender, EventArgs e) - { - InitializeVisualizerMapping(); - } - private void themeRenderer_ThemeChanged(object sender, EventArgs e) { InitializeTheme(); @@ -1329,31 +1008,18 @@ private void InitializeTheme() private void InitializeViewBindings() { - Editor.CloseWorkflowEditor.Subscribe(CloseWorkflowEditorLauncher); - Editor.Error.Subscribe(ex => uiService.ShowError(ex)); - Editor.UpdateLayout.Subscribe(validateWorkflow => - { - if (Launcher != null) Launcher.WorkflowGraphView.UpdateGraphLayout(); - else UpdateGraphLayout(); - }); - - Editor.UpdateParentLayout.Subscribe(validateWorkflow => - { - if (Launcher != null) - { - Launcher.ParentView.UpdateGraphLayout(validateWorkflow); - } - }); - - Editor.InvalidateLayout.Subscribe(validateWorkflow => + Editor.Error.Subscribe(uiService.ShowError); + Editor.UpdateLayout.Subscribe(UpdateGraphLayout); + Editor.InvalidateLayout.Subscribe(InvalidateGraphLayout); + Editor.WorkflowPathChanged.Subscribe(path => { - if (Launcher != null) Launcher.WorkflowGraphView.InvalidateGraphLayout(validateWorkflow); - else InvalidateGraphLayout(validateWorkflow); + UpdateSelection(forceUpdate: true); + OnWorkflowPathChanged(EventArgs.Empty); }); Editor.UpdateSelection.Subscribe(selection => { - var activeView = Launcher != null ? Launcher.WorkflowGraphView.GraphView : graphView; + var activeView = graphView; activeView.SelectedNodes = activeView.Nodes.LayeredNodes() .Where(node => { @@ -1361,6 +1027,7 @@ private void InitializeViewBindings() return selection.Any(builder => ExpressionBuilder.Unwrap(builder) == nodeBuilder); }); }); + Editor.ResetNavigation(); } #endregion @@ -1630,7 +1297,7 @@ private ToolStripMenuItem CreateSubjectTypeMenuItem( private HashSet FindMappedProperties(GraphNode node) { var mappedProperties = new HashSet(); - foreach (var predecessor in workflow.Predecessors(WorkflowEditor.GetGraphNodeTag(workflow, node))) + foreach (var predecessor in Workflow.Predecessors(WorkflowEditor.GetGraphNodeTag(Workflow, node))) { var builder = ExpressionBuilder.Unwrap(predecessor.Value); if (builder is ExternalizedProperty externalizedProperty) @@ -1678,7 +1345,7 @@ private ToolStripMenuItem CreateExternalizeMenuItem( var menuItem = new ToolStripMenuItem(text, null, delegate { var mapping = new ExternalizedMapping { Name = memberName, DisplayName = externalizedName }; - var mappingNode = (from predecessor in workflow.Predecessors(WorkflowEditor.GetGraphNodeTag(workflow, selectedNode)) + var mappingNode = (from predecessor in Workflow.Predecessors(WorkflowEditor.GetGraphNodeTag(Workflow, selectedNode)) let builder = ExpressionBuilder.Unwrap(predecessor.Value) as ExternalizedMappingBuilder where builder != null && predecessor.Successors.Count == 1 select new { node = FindGraphNode(predecessor.Value), builder }) @@ -1741,7 +1408,7 @@ private ToolStripMenuItem CreatePropertySourceMenuItem( return menuItem; } - private ToolStripMenuItem CreateVisualizerMenuItem(string typeName, VisualizerDialogSettings layoutSettings, GraphNode selectedNode) + private ToolStripMenuItem CreateVisualizerMenuItem(string typeName, GraphNode selectedNode) { ToolStripMenuItem menuItem = null; var emptyVisualizer = string.IsNullOrEmpty(typeName); @@ -1762,37 +1429,37 @@ private ToolStripMenuItem CreateVisualizerMenuItem(string typeName, VisualizerDi } else if (!menuItem.Checked) { - layoutSettings.VisualizerTypeName = typeName; - layoutSettings.VisualizerSettings = null; - layoutSettings.Visible = !emptyVisualizer; - if (!editorState.WorkflowRunning) + var dialogSettings = emptyVisualizer ? default : new VisualizerDialogSettings { - layoutSettings.Size = Size.Empty; - } - else + Tag = inspectBuilder, + VisualizerTypeName = typeName, + Visible = true, + Bounds = Rectangle.Empty + }; + + if (editorState.WorkflowRunning) { - var visualizerLauncher = visualizerMapping[inspectBuilder]; - var visualizerVisible = visualizerLauncher.Visible; - if (visualizerVisible) + var visualizerDialogs = (VisualizerDialogMap)serviceProvider.GetService(typeof(VisualizerDialogMap)); + if (visualizerDialogs.TryGetValue(inspectBuilder, out VisualizerDialogLauncher visualizerDialog)) { - visualizerLauncher.Hide(); + visualizerDialog.Hide(); + visualizerDialogs.Remove(visualizerDialog); } - var visualizerBounds = visualizerLauncher.Bounds; - visualizerLauncher = LayoutHelper.CreateVisualizerLauncher( - inspectBuilder, - visualizerLayout, - typeVisualizerMap, - workflow, - visualizerLauncher.VisualizerFactory.MashupSources, - workflowGraphView: this); - visualizerLauncher.Bounds = new Rectangle(visualizerBounds.Location, Size.Empty); - visualizerMapping[inspectBuilder] = visualizerLauncher; - if (layoutSettings.Visible) + if (!emptyVisualizer) { - visualizerLauncher.Show(graphView, serviceProvider); + var dialogLauncher = visualizerDialogs.Add(inspectBuilder, Workflow, dialogSettings); + var ownerWindow = uiService.GetDialogOwnerWindow(); + visualizerDialog.Show(ownerWindow, serviceProvider); } } + else + { + if (emptyVisualizer) + visualizerSettings.Remove(inspectBuilder); + else + visualizerSettings[inspectBuilder] = dialogSettings; + } } }); return menuItem; @@ -1901,44 +1568,31 @@ private void contextMenuStrip_Opening(object sender, CancelEventArgs e) createPropertySourceToolStripMenuItem.Enabled = createPropertySourceToolStripMenuItem.DropDownItems.Count > 0; } - var layoutSettings = visualizerLayout.GetLayoutSettings(selectedNode.Value); - if (layoutSettings != null) - { - var activeVisualizer = layoutSettings.VisualizerTypeName; - if (workflowElement is VisualizerMappingBuilder mappingBuilder && - mappingBuilder.VisualizerType != null) - { - activeVisualizer = mappingBuilder.VisualizerType.GetType().GetGenericArguments()[0].FullName; - } + var activeVisualizer = visualizerSettings.TryGetValue(inspectBuilder, out var dialogSettings) + ? dialogSettings.VisualizerTypeName + : null; - if (editorState.WorkflowRunning) - { - if (visualizerMapping.TryGetValue(inspectBuilder, out VisualizerDialogLauncher visualizerLauncher)) - { - var visualizer = visualizerLauncher.Visualizer; - if (visualizer.IsValueCreated) - { - activeVisualizer = visualizer.Value.GetType().FullName; - } - } - } + if (workflowElement is VisualizerMappingBuilder mappingBuilder && + mappingBuilder.VisualizerType != null) + { + activeVisualizer = mappingBuilder.VisualizerType.GetType().GetGenericArguments()[0].FullName; + } - var visualizerElement = ExpressionBuilder.GetVisualizerElement(inspectBuilder); - if (visualizerElement.ObservableType != null && - (!editorState.WorkflowRunning || visualizerElement.PublishNotifications)) + var visualizerElement = ExpressionBuilder.GetVisualizerElement(inspectBuilder); + if (visualizerElement.ObservableType != null && + (!editorState.WorkflowRunning || visualizerElement.PublishNotifications)) + { + var visualizerTypes = Enumerable.Repeat(null, 1); + visualizerTypes = visualizerTypes.Concat(typeVisualizerMap.GetTypeVisualizers(visualizerElement)); + visualizerToolStripMenuItem.Enabled = true; + foreach (var type in visualizerTypes) { - var visualizerTypes = Enumerable.Repeat(null, 1); - visualizerTypes = visualizerTypes.Concat(typeVisualizerMap.GetTypeVisualizers(visualizerElement)); - visualizerToolStripMenuItem.Enabled = true; - foreach (var type in visualizerTypes) - { - var typeName = type != null ? type.FullName : string.Empty; - var menuItem = CreateVisualizerMenuItem(typeName, layoutSettings, selectedNode); - visualizerToolStripMenuItem.DropDownItems.Add(menuItem); - menuItem.Checked = type == null - ? string.IsNullOrEmpty(activeVisualizer) - : typeName == activeVisualizer; - } + var typeName = type?.FullName ?? string.Empty; + var menuItem = CreateVisualizerMenuItem(typeName, selectedNode); + visualizerToolStripMenuItem.DropDownItems.Add(menuItem); + menuItem.Checked = type is null + ? activeVisualizer is null + : typeName == activeVisualizer; } } } diff --git a/Bonsai.Editor/GraphView/WorkflowPathMouseEventArgs.cs b/Bonsai.Editor/GraphView/WorkflowPathMouseEventArgs.cs new file mode 100644 index 000000000..d70832b1c --- /dev/null +++ b/Bonsai.Editor/GraphView/WorkflowPathMouseEventArgs.cs @@ -0,0 +1,16 @@ +using System.Windows.Forms; +using Bonsai.Editor.GraphModel; + +namespace Bonsai.Editor.GraphView +{ + internal class WorkflowPathMouseEventArgs : MouseEventArgs + { + public WorkflowPathMouseEventArgs(WorkflowEditorPath path, MouseButtons button, int clicks, int x, int y, int delta) + : base(button, clicks, x, y, delta) + { + Path = path; + } + + public WorkflowEditorPath Path { get; } + } +} diff --git a/Bonsai.Editor/GraphView/WorkflowPathNavigationControl.Designer.cs b/Bonsai.Editor/GraphView/WorkflowPathNavigationControl.Designer.cs new file mode 100644 index 000000000..41edb79d9 --- /dev/null +++ b/Bonsai.Editor/GraphView/WorkflowPathNavigationControl.Designer.cs @@ -0,0 +1,59 @@ +namespace Bonsai.Editor.GraphView +{ + partial class WorkflowPathNavigationControl + { + /// + /// Required designer variable. + /// + private System.ComponentModel.IContainer components = null; + + /// + /// Clean up any resources being used. + /// + /// true if managed resources should be disposed; otherwise, false. + protected override void Dispose(bool disposing) + { + if (disposing && (components != null)) + { + components.Dispose(); + } + base.Dispose(disposing); + } + + #region Component Designer generated code + + /// + /// Required method for Designer support - do not modify + /// the contents of this method with the code editor. + /// + private void InitializeComponent() + { + this.flowLayoutPanel = new System.Windows.Forms.FlowLayoutPanel(); + this.SuspendLayout(); + // + // flowLayoutPanel + // + this.flowLayoutPanel.Dock = System.Windows.Forms.DockStyle.Fill; + this.flowLayoutPanel.Location = new System.Drawing.Point(0, 0); + this.flowLayoutPanel.Name = "flowLayoutPanel"; + this.flowLayoutPanel.Size = new System.Drawing.Size(452, 29); + this.flowLayoutPanel.TabIndex = 0; + this.flowLayoutPanel.WrapContents = false; + // + // EditorPathNavigationControl + // + this.AutoScaleDimensions = new System.Drawing.SizeF(6F, 13F); + this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font; + this.Controls.Add(this.flowLayoutPanel); + this.Name = "EditorPathNavigationControl"; + this.Size = new System.Drawing.Size(452, 29); + this.ResumeLayout(false); + this.PerformLayout(); + + } + + #endregion + + private System.Windows.Forms.FlowLayoutPanel flowLayoutPanel; + } +} diff --git a/Bonsai.Editor/GraphView/WorkflowPathNavigationControl.cs b/Bonsai.Editor/GraphView/WorkflowPathNavigationControl.cs new file mode 100644 index 000000000..bace621e1 --- /dev/null +++ b/Bonsai.Editor/GraphView/WorkflowPathNavigationControl.cs @@ -0,0 +1,234 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Windows.Forms; +using Bonsai.Editor.GraphModel; +using Bonsai.Editor.Themes; +using Bonsai.Expressions; + +namespace Bonsai.Editor.GraphView +{ + partial class WorkflowPathNavigationControl : UserControl + { + static readonly object WorkflowPathMouseClickEvent = new(); + readonly IServiceProvider serviceProvider; + readonly IWorkflowEditorService editorService; + readonly ThemeRenderer themeRenderer; + WorkflowEditorPath workflowPath; + int totalPathWidth; + + public WorkflowPathNavigationControl(IServiceProvider provider) + { + InitializeComponent(); + serviceProvider = provider; + themeRenderer = (ThemeRenderer)provider.GetService(typeof(ThemeRenderer)); + themeRenderer.ThemeChanged += ThemeRenderer_ThemeChanged; + editorService = (IWorkflowEditorService)provider.GetService(typeof(IWorkflowEditorService)); + } + + public string DisplayName + { + get + { + return flowLayoutPanel.Controls.Count > 0 + ? flowLayoutPanel.Controls[flowLayoutPanel.Controls.Count - 1].Text + : editorService.GetProjectDisplayName(); + } + } + + public WorkflowEditorPath WorkflowPath + { + get { return workflowPath; } + set + { + workflowPath = value; + var workflowBuilder = (WorkflowBuilder)serviceProvider.GetService(typeof(WorkflowBuilder)); + var pathElements = GetPathElements(workflowPath, workflowBuilder); + SetPath(pathElements); + } + } + + public event EventHandler WorkflowPathMouseClick + { + add { Events.AddHandler(WorkflowPathMouseClickEvent, value); } + remove { Events.RemoveHandler(WorkflowPathMouseClickEvent, value); } + } + + private void OnWorkflowPathMouseClick(WorkflowPathMouseEventArgs e) + { + (Events[WorkflowPathMouseClickEvent] as EventHandler)?.Invoke(this, e); + } + + static IEnumerable> GetPathElements(WorkflowEditorPath workflowPath, WorkflowBuilder workflowBuilder) + { + var workflow = workflowBuilder.Workflow; + foreach (var pathElement in workflowPath?.GetPathElements() ?? Enumerable.Empty()) + { + var builder = workflow[pathElement.Index].Value; + if (ExpressionBuilder.Unwrap(builder) is IWorkflowExpressionBuilder nestedWorkflowBuilder) + { + workflow = nestedWorkflowBuilder.Workflow; + } + + yield return new( + key: ExpressionBuilder.GetElementDisplayName(builder), + value: pathElement); + } + } + + private void SetPath(IEnumerable> pathElements) + { + SuspendLayout(); + totalPathWidth = 0; + flowLayoutPanel.Controls.Clear(); + AddPathButton("...", null, createEvent: false, visible: false); + AddPathButton(editorService.GetProjectDisplayName(), null); + foreach (var path in pathElements) + { + AddPathButton(">", null, createEvent: false); + AddPathButton(path.Key, path.Value); + } + CompressPath(); + ResumeLayout(true); + } + + private void CompressPath() + { + if (flowLayoutPanel.Controls.Count <= 4) + return; + + bool compressPath = false; + var totalWidth = totalPathWidth; + if (totalWidth > Width) + { + // adjust for inserting the ellipsis button + totalWidth -= flowLayoutPanel.Controls[1].Width; + totalWidth += flowLayoutPanel.Controls[0].Width; + compressPath = true; + } + + var excessWidth = totalWidth - Width; + for (int i = 2; i < flowLayoutPanel.Controls.Count - 4; i++) + { + // separator and breadcrumb buttons are hidden together + var visible = !compressPath || excessWidth <= 0; + if (i % 2 != 0) visible &= flowLayoutPanel.Controls[i - 1].Visible; + + // hide excess breadcrumb levels + flowLayoutPanel.Controls[i].Visible = visible; + if (excessWidth > 0) + { + excessWidth -= GetControlWidth(flowLayoutPanel.Controls[i]); + } + } + + // either the root or ellipsis button is shown + flowLayoutPanel.Controls[0].Visible = compressPath; + flowLayoutPanel.Controls[1].Visible = !compressPath; + } + + private int GetControlWidth(Control control) + { + return control.Width + control.Margin.Horizontal + flowLayoutPanel.Padding.Right; + } + + private BreadcrumbButtton AddPathButton(string text, WorkflowEditorPath path, bool createEvent = true, bool visible = true) + { + var breadcrumbButton = new BreadcrumbButtton + { + AutoSize = true, + Locked = !createEvent, + AutoSizeMode = AutoSizeMode.GrowAndShrink, + Visible = visible, + Text = text, + Tag = path + }; + if (createEvent) + breadcrumbButton.MouseClick += BreadcrumbButton_MouseClick; + breadcrumbButton.ParentChanged += BreadcrumbButton_ParentChanged; + SetBreadcrumbTheme(breadcrumbButton, themeRenderer); + flowLayoutPanel.Controls.Add(breadcrumbButton); + if (flowLayoutPanel.Controls.Count > 1) + totalPathWidth += GetControlWidth(breadcrumbButton); + return breadcrumbButton; + } + + private void BreadcrumbButton_ParentChanged(object sender, EventArgs e) + { + var button = (Button)sender; + if (button.Parent == null) + button.Dispose(); + } + + private void BreadcrumbButton_MouseClick(object sender, MouseEventArgs e) + { + var button = (Button)sender; + var path = (WorkflowEditorPath)button.Tag; + OnWorkflowPathMouseClick(new WorkflowPathMouseEventArgs(path, e.Button, e.Clicks, e.X, e.Y, e.Delta)); + } + + protected override void OnLayout(LayoutEventArgs e) + { + CompressPath(); + base.OnLayout(e); + } + + protected override void OnHandleDestroyed(EventArgs e) + { + themeRenderer.ThemeChanged -= ThemeRenderer_ThemeChanged; + base.OnHandleDestroyed(e); + } + + private void ThemeRenderer_ThemeChanged(object sender, EventArgs e) + { + InitializeTheme(); + } + + internal void InitializeTheme() + { + foreach (Button button in flowLayoutPanel.Controls) + { + SetBreadcrumbTheme(button, themeRenderer); + } + } + + private static void SetBreadcrumbTheme(Button button, ThemeRenderer themeRenderer) + { + if (themeRenderer == null) + return; + + var colorTable = themeRenderer.ToolStripRenderer.ColorTable; + button.BackColor = colorTable.WindowBackColor; + button.ForeColor = colorTable.WindowText; + } + + class BreadcrumbButtton : Button + { + bool locked; + + public bool Locked + { + get => locked; + set + { + locked = value; + SetStyle(ControlStyles.Selectable, !locked); + } + } + + protected override void OnMouseEnter(EventArgs e) + { + if (Locked) + return; + base.OnMouseEnter(e); + } + + protected override void OnMouseDown(MouseEventArgs mevent) + { + if (Locked) + return; + base.OnMouseDown(mevent); + } + } + } +} diff --git a/Bonsai.Editor/GraphView/WorkflowPathNavigationControl.resx b/Bonsai.Editor/GraphView/WorkflowPathNavigationControl.resx new file mode 100644 index 000000000..1af7de150 --- /dev/null +++ b/Bonsai.Editor/GraphView/WorkflowPathNavigationControl.resx @@ -0,0 +1,120 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + text/microsoft-resx + + + 2.0 + + + System.Resources.ResXResourceReader, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + + System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + \ No newline at end of file diff --git a/Bonsai.Editor/IWorkflowEditorService.cs b/Bonsai.Editor/IWorkflowEditorService.cs index ffb1c4aca..823f42788 100644 --- a/Bonsai.Editor/IWorkflowEditorService.cs +++ b/Bonsai.Editor/IWorkflowEditorService.cs @@ -1,11 +1,14 @@ using System; using System.IO; using System.Windows.Forms; +using Bonsai.Expressions; namespace Bonsai.Editor { interface IWorkflowEditorService { + string GetProjectDisplayName(); + void OnKeyDown(KeyEventArgs e); void OnKeyPress(KeyPressEventArgs e); @@ -22,6 +25,8 @@ interface IWorkflowEditorService void SelectNextControl(bool forward); + void SelectBuilderNode(ExpressionBuilder builder); + bool ValidateWorkflow(); void RefreshEditor(); diff --git a/Bonsai.Editor/Layout/LayoutHelper.cs b/Bonsai.Editor/Layout/LayoutHelper.cs index 5e1f74914..b1e085e72 100644 --- a/Bonsai.Editor/Layout/LayoutHelper.cs +++ b/Bonsai.Editor/Layout/LayoutHelper.cs @@ -4,7 +4,6 @@ using System.Collections.Generic; using System.IO; using System.Linq; -using System.Windows.Forms; using System.Xml.Linq; using System.Xml.Serialization; @@ -33,30 +32,6 @@ public static string GetLayoutPath(string fileName) : Editor.Project.GetLegacyLayoutConfigPath(fileName); } - public static void SetLayoutTags(ExpressionBuilderGraph source, VisualizerLayout layout) - { - foreach (var node in source) - { - var builder = node.Value; - var layoutSettings = layout.GetLayoutSettings(builder); - if (layoutSettings == null) - { - layoutSettings = new VisualizerDialogSettings(); - layout.DialogSettings.Add(layoutSettings); - } - layoutSettings.Tag = builder; - - if (layoutSettings is WorkflowEditorSettings editorSettings && - ExpressionBuilder.Unwrap(builder) is IWorkflowExpressionBuilder workflowBuilder && - editorSettings.EditorVisualizerLayout != null && - editorSettings.EditorDialogSettings.Visible && - workflowBuilder.Workflow != null) - { - SetLayoutTags(workflowBuilder.Workflow, editorSettings.EditorVisualizerLayout); - } - } - } - public static void SetWorkflowNotifications(ExpressionBuilderGraph source, bool publishNotifications) { foreach (var builder in from node in source @@ -73,41 +48,15 @@ public static void SetWorkflowNotifications(ExpressionBuilderGraph source, bool } } - public static void SetLayoutNotifications(VisualizerLayout root) + public static void SetLayoutNotifications(ExpressionBuilderGraph source, VisualizerDialogMap lookup) { - foreach (var settings in root.DialogSettings) + foreach (var builder in source.Descendants()) { - SetLayoutNotifications(settings, root, forcePublish: false); - } - } - - static void SetLayoutNotifications(VisualizerDialogSettings settings, VisualizerLayout root, bool forcePublish = false) - { - var inspectBuilder = settings.Tag as InspectBuilder; - while (inspectBuilder != null && !inspectBuilder.PublishNotifications) - { - if (string.IsNullOrEmpty(settings.VisualizerTypeName) && !forcePublish) - { - break; - } - - SetVisualizerNotifications(inspectBuilder); - foreach (var index in settings.Mashups.Concat(settings.VisualizerSettings? - .Descendants(MashupSourceElement) - .Select(m => int.Parse(m.Value)) - .Distinct() ?? Enumerable.Empty())) + var inspectBuilder = (InspectBuilder)builder; + if (lookup.TryGetValue((InspectBuilder)builder, out VisualizerDialogLauncher _)) { - if (index < 0 || index >= root.DialogSettings.Count) continue; - var mashupSource = root.DialogSettings[index]; - SetLayoutNotifications(mashupSource, root, forcePublish: true); + SetVisualizerNotifications(inspectBuilder); } - - inspectBuilder = ExpressionBuilder.GetVisualizerElement(inspectBuilder); - } - - if (settings is WorkflowEditorSettings editorSettings && editorSettings.EditorVisualizerLayout != null) - { - SetLayoutNotifications(editorSettings.EditorVisualizerLayout); } } @@ -120,18 +69,6 @@ static void SetVisualizerNotifications(InspectBuilder inspectBuilder) } } - static IEnumerable GetMashupSources(this VisualizerFactory visualizerFactory) - { - yield return visualizerFactory; - foreach (var source in visualizerFactory.MashupSources) - { - foreach (var nestedSource in source.GetMashupSources()) - { - yield return nestedSource; - } - } - } - internal static Type GetMashupSourceType(Type mashupVisualizerType, Type visualizerType, TypeVisualizerMap typeVisualizerMap) { Type mashupSource = default; @@ -156,11 +93,9 @@ internal static Type GetMashupSourceType(Type mashupVisualizerType, Type visuali public static VisualizerDialogLauncher CreateVisualizerLauncher( InspectBuilder source, - VisualizerLayout visualizerLayout, + VisualizerDialogSettings layoutSettings, TypeVisualizerMap typeVisualizerMap, - ExpressionBuilderGraph workflow, - IReadOnlyList mashupArguments, - Editor.GraphView.WorkflowGraphView workflowGraphView = null) + ExpressionBuilderGraph workflow) { var inspectBuilder = ExpressionBuilder.GetVisualizerElement(source); if (inspectBuilder.ObservableType == null || !inspectBuilder.PublishNotifications || @@ -169,23 +104,23 @@ public static VisualizerDialogLauncher CreateVisualizerLauncher( return null; } - var layoutSettings = visualizerLayout.GetLayoutSettings(source); - var visualizerType = typeVisualizerMap.GetVisualizerType(layoutSettings?.VisualizerTypeName ?? string.Empty) - ?? typeVisualizerMap.GetTypeVisualizers(inspectBuilder).FirstOrDefault(); - if (visualizerType == null) + var visualizerType = typeVisualizerMap.GetVisualizerType(layoutSettings?.VisualizerTypeName ?? string.Empty); + visualizerType ??= typeVisualizerMap.GetTypeVisualizers(inspectBuilder).FirstOrDefault(); + if (visualizerType is null) { return null; } + var mashupArguments = GetMashupArguments(inspectBuilder, typeVisualizerMap); var visualizerFactory = new VisualizerFactory(inspectBuilder, visualizerType, mashupArguments); var visualizer = new Lazy(() => DeserializeVisualizerSettings( visualizerType, layoutSettings, - visualizerLayout, + workflow, visualizerFactory, typeVisualizerMap)); - var launcher = new VisualizerDialogLauncher(visualizer, visualizerFactory, workflow, source, workflowGraphView); + var launcher = new VisualizerDialogLauncher(visualizer, visualizerFactory, workflow, source); launcher.Text = source != null ? ExpressionBuilder.GetElementDisplayName(source) : null; return launcher; } @@ -206,48 +141,6 @@ static IReadOnlyList GetMashupArguments(InspectBuilder builde }).ToList(); } - public static Dictionary CreateVisualizerMapping( - ExpressionBuilderGraph workflow, - VisualizerLayout visualizerLayout, - TypeVisualizerMap typeVisualizerMap, - IServiceProvider provider = null, - IWin32Window owner = null, - Editor.GraphView.WorkflowGraphView graphView = null) - { - if (workflow == null) return null; - var visualizerMapping = (from node in workflow.TopologicalSort() - let source = (InspectBuilder)node.Value - let mashupArguments = GetMashupArguments(source, typeVisualizerMap) - let visualizerLauncher = CreateVisualizerLauncher( - source, - visualizerLayout, - typeVisualizerMap, - workflow, - mashupArguments, - graphView) - where visualizerLauncher != null - select new { source, visualizerLauncher }) - .ToDictionary(mapping => mapping.source, - mapping => mapping.visualizerLauncher); - foreach (var mapping in visualizerMapping) - { - var key = mapping.Key; - var visualizerLauncher = mapping.Value; - var layoutSettings = visualizerLayout.GetLayoutSettings(key); - if (layoutSettings != null) - { - visualizerLauncher.Bounds = layoutSettings.Bounds; - visualizerLauncher.WindowState = layoutSettings.WindowState; - if (layoutSettings.Visible) - { - visualizerLauncher.Show(owner, provider); - } - } - } - - return visualizerMapping; - } - public static XElement SerializeVisualizerSettings( DialogTypeVisualizer visualizer, IEnumerable> topologicalOrder) @@ -314,11 +207,11 @@ static XElement SerializeMashupSource( public static DialogTypeVisualizer DeserializeVisualizerSettings( Type visualizerType, VisualizerDialogSettings layoutSettings, - VisualizerLayout visualizerLayout, + ExpressionBuilderGraph workflow, VisualizerFactory visualizerFactory, TypeVisualizerMap typeVisualizerMap) { - if (layoutSettings?.VisualizerTypeName != visualizerType.FullName) + if (layoutSettings?.VisualizerTypeName != visualizerType?.FullName) { layoutSettings = default; } @@ -341,7 +234,7 @@ public static DialogTypeVisualizer DeserializeVisualizerSettings( layoutSettings.Mashups.Clear(); } - return visualizerFactory.CreateVisualizer(layoutSettings?.VisualizerSettings, visualizerLayout, typeVisualizerMap); + return visualizerFactory.CreateVisualizer(layoutSettings?.VisualizerSettings, workflow, typeVisualizerMap); } static int? GetMashupSourceIndex( @@ -356,7 +249,7 @@ public static DialogTypeVisualizer DeserializeVisualizerSettings( public static DialogTypeVisualizer CreateVisualizer( this VisualizerFactory visualizerFactory, XElement visualizerSettings, - VisualizerLayout visualizerLayout, + ExpressionBuilderGraph workflow, TypeVisualizerMap typeVisualizerMap) { DialogTypeVisualizer visualizer; @@ -389,19 +282,19 @@ public static DialogTypeVisualizer CreateVisualizer( if (mashupSourceElement == null) continue; var mashupSourceIndex = int.Parse(mashupSourceElement.Value); - var mashupSource = (InspectBuilder)visualizerLayout.DialogSettings[mashupSourceIndex]?.Tag; + var mashupSource = (InspectBuilder)workflow[mashupSourceIndex].Value; var mashupVisualizerTypeName = mashup.Element(nameof(VisualizerDialogSettings.VisualizerTypeName))?.Value; var mashupVisualizerType = typeVisualizerMap.GetVisualizerType(mashupVisualizerTypeName); mashupFactory = new VisualizerFactory(mashupSource, mashupVisualizerType); } - CreateMashupVisualizer(mashupVisualizer, visualizerFactory, mashupFactory, visualizerLayout, typeVisualizerMap, mashup); + CreateMashupVisualizer(mashupVisualizer, visualizerFactory, mashupFactory, workflow, typeVisualizerMap, mashup); } for (int i = index; i < visualizerFactory.MashupSources.Count; i++) { var mashupFactory = visualizerFactory.MashupSources[i]; - CreateMashupVisualizer(mashupVisualizer, visualizerFactory, mashupFactory, visualizerLayout, typeVisualizerMap); + CreateMashupVisualizer(mashupVisualizer, visualizerFactory, mashupFactory, workflow, typeVisualizerMap); } } @@ -412,7 +305,7 @@ static void CreateMashupVisualizer( MashupVisualizer mashupVisualizer, VisualizerFactory visualizerFactory, VisualizerFactory mashupFactory, - VisualizerLayout visualizerLayout, + ExpressionBuilderGraph workflow, TypeVisualizerMap typeVisualizerMap, XElement mashup = null) { @@ -437,7 +330,7 @@ static void CreateMashupVisualizer( } } - var nestedVisualizer = mashupFactory.CreateVisualizer(mashupVisualizerSettings, visualizerLayout, typeVisualizerMap); + var nestedVisualizer = mashupFactory.CreateVisualizer(mashupVisualizerSettings, workflow, typeVisualizerMap); mashupVisualizer.MashupSources.Add(mashupFactory.Source, nestedVisualizer); } } diff --git a/Bonsai.Editor/Layout/VisualizerDialogLauncher.cs b/Bonsai.Editor/Layout/VisualizerDialogLauncher.cs index 6cb186b21..32b6c4d14 100644 --- a/Bonsai.Editor/Layout/VisualizerDialogLauncher.cs +++ b/Bonsai.Editor/Layout/VisualizerDialogLauncher.cs @@ -3,39 +3,38 @@ using Bonsai.Expressions; using System.Reactive.Linq; using System.Windows.Forms; -using Bonsai.Editor.GraphView; using Bonsai.Editor.GraphModel; +using System.Linq; +using Bonsai.Editor; namespace Bonsai.Design { class VisualizerDialogLauncher : DialogLauncher, ITypeVisualizerContext { - readonly ExpressionBuilderGraph workflow; - readonly WorkflowGraphView graphView; ServiceContainer visualizerContext; IDisposable visualizerObserver; public VisualizerDialogLauncher( Lazy visualizer, VisualizerFactory visualizerFactory, - ExpressionBuilderGraph visualizerWorkflow, - InspectBuilder workflowSource, - WorkflowGraphView workflowGraphView) + ExpressionBuilderGraph workflow, + InspectBuilder workflowSource) { Visualizer = visualizer ?? throw new ArgumentNullException(nameof(visualizer)); VisualizerFactory = visualizerFactory ?? throw new ArgumentNullException(nameof(visualizerFactory)); Source = workflowSource ?? throw new ArgumentNullException(nameof(workflowSource)); - workflow = visualizerWorkflow; - graphView = workflowGraphView; + Workflow = workflow ?? throw new ArgumentNullException(nameof(workflow)); } public string Text { get; set; } public InspectBuilder Source { get; } + public ExpressionBuilderGraph Workflow { get; } + public Lazy Visualizer { get; } - public VisualizerFactory VisualizerFactory { get; } + private VisualizerFactory VisualizerFactory { get; } static IDisposable SubscribeDialog(IObservable source, TypeVisualizerDialog visualizerDialog) { @@ -55,7 +54,7 @@ protected override void InitializeComponents(TypeVisualizerDialog visualizerDial visualizerContext.AddService(typeof(ITypeVisualizerContext), this); visualizerContext.AddService(typeof(TypeVisualizerDialog), visualizerDialog); visualizerContext.AddService(typeof(IDialogTypeVisualizerService), visualizerDialog); - visualizerContext.AddService(typeof(ExpressionBuilderGraph), workflow); + visualizerContext.AddService(typeof(ExpressionBuilderGraph), Workflow); Visualizer.Value.Load(visualizerContext); var visualizerOutput = Visualizer.Value.Visualize(VisualizerFactory.Source.Output, visualizerContext); @@ -90,10 +89,10 @@ protected override void InitializeComponents(TypeVisualizerDialog visualizerDial void visualizerDialog_KeyDown(object sender, KeyEventArgs e) { - if (graphView != null && e.KeyCode == Keys.Back && e.Control) + if (e.KeyCode == Keys.Back && e.Control) { - graphView.SelectBuilderNode(Source); - graphView.EditorControl.ParentForm.Activate(); + var editorService = (IWorkflowEditorService)visualizerContext.GetService(typeof(IWorkflowEditorService)); + editorService.SelectBuilderNode(Source); } if (e.KeyCode == Keys.Delete && e.Control) @@ -148,36 +147,40 @@ MashupVisualizer GetMashupContainer(int x, int y, bool allowEmpty = true) return visualizer; } - public void CreateMashup(MashupVisualizer dialogMashup, VisualizerDialogLauncher visualizerDialog, TypeVisualizerMap typeVisualizerMap) + public void CreateMashup(MashupVisualizer dialogMashup, InspectBuilder source, Type visualizerType, TypeVisualizerMap typeVisualizerMap) { - if (visualizerDialog != null) + if (visualizerType == null) + throw new ArgumentNullException(nameof(visualizerType)); + + var dialogMashupType = dialogMashup.GetType(); + var mashupSourceType = LayoutHelper.GetMashupSourceType(dialogMashupType, visualizerType, typeVisualizerMap); + if (mashupSourceType != null) { - var dialogMashupType = dialogMashup.GetType(); - var visualizerFactory = visualizerDialog.VisualizerFactory; - var mashupSourceType = LayoutHelper.GetMashupSourceType(dialogMashupType, visualizerFactory.VisualizerType, typeVisualizerMap); - if (mashupSourceType != null) + UnloadMashups(); + if (mashupSourceType == typeof(DialogTypeVisualizer)) { - UnloadMashups(); - if (mashupSourceType == typeof(DialogTypeVisualizer)) - { - mashupSourceType = visualizerFactory.VisualizerType; - } - var visualizerMashup = (DialogTypeVisualizer)Activator.CreateInstance(mashupSourceType); - dialogMashup.MashupSources.Add(visualizerFactory.Source, visualizerMashup); - ReloadMashups(); + mashupSourceType = visualizerType; } + var visualizerMashup = (DialogTypeVisualizer)Activator.CreateInstance(mashupSourceType); + dialogMashup.MashupSources.Add(source, visualizerMashup); + ReloadMashups(); } } void visualizerDialog_DragDrop(object sender, DragEventArgs e) { - if (graphView != null && visualizerContext != null && e.Data.GetDataPresent(typeof(GraphNode))) + if (visualizerContext != null && e.Data.GetDataPresent(typeof(GraphNode))) { var graphNode = (GraphNode)e.Data.GetData(typeof(GraphNode)); + var inspectBuilder = (InspectBuilder)graphNode.Value; var typeVisualizerMap = (TypeVisualizerMap)visualizerContext.GetService(typeof(TypeVisualizerMap)); - var visualizerDialog = graphView.GetVisualizerDialogLauncher(graphNode); - var visualizer = GetMashupContainer(e.X, e.Y); - CreateMashup(visualizer, visualizerDialog, typeVisualizerMap); + var visualizerDialogMap = (VisualizerDialogMap)visualizerContext.GetService(typeof(VisualizerDialogMap)); + var visualizerType = GetVisualizerType(inspectBuilder, visualizerDialogMap, typeVisualizerMap); + if (visualizerType != null) + { + var visualizer = GetMashupContainer(e.X, e.Y); + CreateMashup(visualizer, inspectBuilder, visualizerType, typeVisualizerMap); + } } } @@ -187,17 +190,17 @@ void visualizerDialog_DragOver(object sender, DragEventArgs e) void visualizerDialog_DragEnter(object sender, DragEventArgs e) { - if (graphView != null && visualizerContext != null && e.Data.GetDataPresent(typeof(GraphNode))) + if (visualizerContext != null && e.Data.GetDataPresent(typeof(GraphNode))) { var graphNode = (GraphNode)e.Data.GetData(typeof(GraphNode)); - var visualizerDialog = graphView.GetVisualizerDialogLauncher(graphNode); - if (visualizerDialog != null && visualizerDialog != this) + var typeVisualizerMap = (TypeVisualizerMap)visualizerContext.GetService(typeof(TypeVisualizerMap)); + var visualizerDialogMap = (VisualizerDialogMap)visualizerContext.GetService(typeof(VisualizerDialogMap)); + var visualizerType = GetVisualizerType((InspectBuilder)graphNode.Value, visualizerDialogMap, typeVisualizerMap); + if (visualizerType != null) { var dialogMashupType = VisualizerFactory.VisualizerType; if (dialogMashupType.IsSubclassOf(typeof(MashupVisualizer))) { - var visualizerType = visualizerDialog.VisualizerFactory.VisualizerType; - var typeVisualizerMap = (TypeVisualizerMap)visualizerContext.GetService(typeof(TypeVisualizerMap)); var mashupVisualizerType = LayoutHelper.GetMashupSourceType(dialogMashupType, visualizerType, typeVisualizerMap); if (mashupVisualizerType != null) { @@ -207,5 +210,16 @@ void visualizerDialog_DragEnter(object sender, DragEventArgs e) } } } + + Type GetVisualizerType(InspectBuilder source, VisualizerDialogMap visualizerDialogMap, TypeVisualizerMap typeVisualizerMap) + { + if (visualizerDialogMap.TryGetValue(source, out VisualizerDialogLauncher dialogLauncher)) + { + if (dialogLauncher == this) + return null; + else return dialogLauncher.VisualizerFactory.VisualizerType; + } + else return typeVisualizerMap.GetTypeVisualizers(source).FirstOrDefault(); + } } } diff --git a/Bonsai.Editor/Layout/VisualizerDialogMap.cs b/Bonsai.Editor/Layout/VisualizerDialogMap.cs new file mode 100644 index 000000000..b4aa0c133 --- /dev/null +++ b/Bonsai.Editor/Layout/VisualizerDialogMap.cs @@ -0,0 +1,75 @@ +using System; +using System.Collections; +using System.Collections.Generic; +using System.Windows.Forms; +using Bonsai.Expressions; + +namespace Bonsai.Design +{ + internal class VisualizerDialogMap : IEnumerable + { + readonly TypeVisualizerMap typeVisualizerMap; + readonly Dictionary lookup; + + public VisualizerDialogMap(TypeVisualizerMap typeVisualizers) + { + typeVisualizerMap = typeVisualizers ?? throw new ArgumentNullException(nameof(typeVisualizers)); + lookup = new(); + } + + public VisualizerDialogLauncher this[InspectBuilder key] + { + get => lookup[key]; + } + + public bool TryGetValue(InspectBuilder key, out VisualizerDialogLauncher value) + { + return lookup.TryGetValue(key, out value); + } + + public void Show(VisualizerLayoutMap visualizerSettings, IServiceProvider provider = null, IWin32Window owner = null) + { + foreach (var dialogLauncher in lookup.Values) + { + var dialogSettings = visualizerSettings[dialogLauncher.Source]; + dialogLauncher.Bounds = dialogSettings.Bounds; + dialogLauncher.WindowState = dialogSettings.WindowState; + if (dialogSettings.Visible) + { + dialogLauncher.Show(owner, provider); + } + } + } + + public VisualizerDialogLauncher Add(InspectBuilder source, ExpressionBuilderGraph workflow, VisualizerDialogSettings dialogSettings) + { + var dialogLauncher = LayoutHelper.CreateVisualizerLauncher( + source, + dialogSettings, + typeVisualizerMap, + workflow); + Add(dialogLauncher); + return dialogLauncher; + } + + public void Add(VisualizerDialogLauncher item) + { + lookup.Add(item.Source, item); + } + + public bool Remove(VisualizerDialogLauncher item) + { + return lookup.Remove(item.Source); + } + + public IEnumerator GetEnumerator() + { + return lookup.Values.GetEnumerator(); + } + + IEnumerator IEnumerable.GetEnumerator() + { + return GetEnumerator(); + } + } +} diff --git a/Bonsai.Editor/Layout/VisualizerDialogSettings.cs b/Bonsai.Editor/Layout/VisualizerDialogSettings.cs index a6aaa0623..9074efff9 100644 --- a/Bonsai.Editor/Layout/VisualizerDialogSettings.cs +++ b/Bonsai.Editor/Layout/VisualizerDialogSettings.cs @@ -6,9 +6,21 @@ namespace Bonsai.Design { +#pragma warning disable CS0612 // Type or member is obsolete [XmlInclude(typeof(WorkflowEditorSettings))] +#pragma warning restore CS0612 // Type or member is obsolete public class VisualizerDialogSettings { + [XmlIgnore] + public int? Index { get; set; } + + [XmlAttribute(nameof(Index))] + public string IndexXml + { + get => Index.HasValue ? Index.GetValueOrDefault().ToString() : null; + set => Index = !string.IsNullOrEmpty(value) ? int.Parse(value) : null; + } + [XmlIgnore] public object Tag { get; set; } @@ -35,12 +47,21 @@ public Rectangle Bounds public XElement VisualizerSettings { get; set; } + public VisualizerLayout NestedLayout { get; set; } + // [Obsolete] public Collection Mashups { get; } = new Collection(); - public bool MashupsSpecified - { - get { return false; } - } + public bool VisibleSpecified => Visible; + + public bool LocationSpecified => !Location.IsEmpty; + + public bool SizeSpecified => !Size.IsEmpty; + + public bool WindowStateSpecified => WindowState != FormWindowState.Normal; + + public bool NestedLayoutSpecified => NestedLayout?.DialogSettings.Count > 0; + + public bool MashupsSpecified => false; } } diff --git a/Bonsai.Editor/Layout/VisualizerLayoutMap.cs b/Bonsai.Editor/Layout/VisualizerLayoutMap.cs new file mode 100644 index 000000000..172fa418c --- /dev/null +++ b/Bonsai.Editor/Layout/VisualizerLayoutMap.cs @@ -0,0 +1,201 @@ +using System.Collections; +using System.Collections.Generic; +using Bonsai.Expressions; + +namespace Bonsai.Design +{ + internal class VisualizerLayoutMap : IEnumerable + { + readonly TypeVisualizerMap typeVisualizerMap; + readonly Dictionary lookup; + + public VisualizerLayoutMap(TypeVisualizerMap typeVisualizers) + { + typeVisualizerMap = typeVisualizers; + lookup = new(); + } + + public VisualizerDialogSettings this[InspectBuilder key] + { + get => lookup[key]; + set => lookup[key] = value; + } + + public bool TryGetValue(InspectBuilder key, out VisualizerDialogSettings value) + { + return lookup.TryGetValue(key, out value); + } + + private void CreateVisualizerDialogs(ExpressionBuilderGraph workflow, VisualizerDialogMap visualizerDialogs) + { + for (int i = 0; i < workflow.Count; i++) + { + var builder = (InspectBuilder)workflow[i].Value; + if (lookup.TryGetValue(builder, out VisualizerDialogSettings dialogSettings)) + { + visualizerDialogs.Add(builder, workflow, dialogSettings); + } + + if (ExpressionBuilder.Unwrap(builder) is IWorkflowExpressionBuilder workflowBuilder) + { + CreateVisualizerDialogs(workflowBuilder.Workflow, visualizerDialogs); + } + } + } + + public VisualizerDialogMap CreateVisualizerDialogs(WorkflowBuilder workflowBuilder) + { + var visualizerDialogs = new VisualizerDialogMap(typeVisualizerMap); + CreateVisualizerDialogs(workflowBuilder.Workflow, visualizerDialogs); + return visualizerDialogs; + } + + public void Update(IEnumerable visualizerDialogs) + { + var unused = new HashSet(lookup.Keys); + foreach (var dialog in visualizerDialogs) + { + unused.Remove(dialog.Source); + if (!lookup.TryGetValue(dialog.Source, out VisualizerDialogSettings dialogSettings)) + { + dialogSettings = new VisualizerDialogSettings(); + dialogSettings.Tag = dialog.Source; + lookup.Add(dialog.Source, dialogSettings); + } + + var visible = dialog.Visible; + dialog.Hide(); + dialogSettings.Visible = visible; + dialogSettings.Bounds = dialog.Bounds; + dialogSettings.WindowState = dialog.WindowState; + + var visualizer = dialog.Visualizer.Value; + var visualizerType = visualizer.GetType(); + if (visualizerType.IsPublic) + { + dialogSettings.VisualizerTypeName = visualizerType.FullName; + dialogSettings.VisualizerSettings = LayoutHelper.SerializeVisualizerSettings( + visualizer, + dialog.Workflow); + } + } + + foreach (var builder in unused) + { + lookup.Remove(builder); + } + } + + public VisualizerLayout GetVisualizerLayout(WorkflowBuilder workflowBuilder) + { + return GetVisualizerLayout(workflowBuilder.Workflow); + } + + private VisualizerLayout GetVisualizerLayout(ExpressionBuilderGraph workflow) + { + var layout = new VisualizerLayout(); + for (int i = 0; i < workflow.Count; i++) + { + var builder = (InspectBuilder)workflow[i].Value; + var layoutSettings = new VisualizerDialogSettings { Index = i }; + + if (lookup.TryGetValue(builder, out VisualizerDialogSettings dialogSettings)) + { + layoutSettings.Visible = dialogSettings.Visible; + layoutSettings.Bounds = dialogSettings.Bounds; + layoutSettings.WindowState = dialogSettings.WindowState; + layoutSettings.VisualizerTypeName = dialogSettings.VisualizerTypeName; + layoutSettings.VisualizerSettings = dialogSettings.VisualizerSettings; + } + + if (ExpressionBuilder.Unwrap(builder) is IWorkflowExpressionBuilder workflowBuilder) + { + layoutSettings.NestedLayout = GetVisualizerLayout(workflowBuilder.Workflow); + } + + if (!layoutSettings.Bounds.IsEmpty || + layoutSettings.VisualizerTypeName != null || + layoutSettings.NestedLayout?.DialogSettings.Count > 0) + { + layout.DialogSettings.Add(layoutSettings); + } + } + + return layout; + } + + public static VisualizerLayoutMap FromVisualizerLayout( + WorkflowBuilder workflowBuilder, + VisualizerLayout layout, + TypeVisualizerMap typeVisualizers) + { + var visualizerSettings = new VisualizerLayoutMap(typeVisualizers); + visualizerSettings.SetVisualizerLayout(workflowBuilder.Workflow, layout); + return visualizerSettings; + } + + public void SetVisualizerLayout(WorkflowBuilder workflowBuilder, VisualizerLayout layout) + { + Clear(); + SetVisualizerLayout(workflowBuilder.Workflow, layout); + } + + private void SetVisualizerLayout(ExpressionBuilderGraph workflow, VisualizerLayout layout) + { + for (int i = 0; i < layout.DialogSettings.Count; i++) + { + var layoutSettings = layout.DialogSettings[i]; + var index = layoutSettings.Index.GetValueOrDefault(i); + if (index < workflow.Count) + { + var builder = (InspectBuilder)workflow[index].Value; + var dialogSettings = new VisualizerDialogSettings(); + dialogSettings.Tag = builder; + dialogSettings.Bounds = layoutSettings.Bounds; + dialogSettings.WindowState = layoutSettings.WindowState; + dialogSettings.Visible = layoutSettings.Visible; + dialogSettings.VisualizerTypeName = layoutSettings.VisualizerTypeName; + dialogSettings.VisualizerSettings = layoutSettings.VisualizerSettings; + Add(dialogSettings); + + if (layoutSettings.NestedLayout != null && + ExpressionBuilder.Unwrap(builder) is IWorkflowExpressionBuilder workflowBuilder) + { + SetVisualizerLayout(workflowBuilder.Workflow, layoutSettings.NestedLayout); + } + } + } + } + + public void Add(VisualizerDialogSettings item) + { + var builder = (InspectBuilder)item.Tag; + lookup.Add(builder, item); + } + + public bool ContainsKey(InspectBuilder builder) + { + return lookup.ContainsKey(builder); + } + + public bool Remove(InspectBuilder builder) + { + return lookup.Remove(builder); + } + + public void Clear() + { + lookup.Clear(); + } + + public IEnumerator GetEnumerator() + { + return lookup.Values.GetEnumerator(); + } + + IEnumerator IEnumerable.GetEnumerator() + { + return GetEnumerator(); + } + } +} diff --git a/Bonsai.Editor/Layout/WorkflowEditorLauncher.cs b/Bonsai.Editor/Layout/WorkflowEditorLauncher.cs deleted file mode 100644 index 87bb1d83a..000000000 --- a/Bonsai.Editor/Layout/WorkflowEditorLauncher.cs +++ /dev/null @@ -1,188 +0,0 @@ -using System; -using System.Windows.Forms; -using Bonsai.Expressions; -using System.ComponentModel; -using Bonsai.Editor.GraphView; -using Bonsai.Editor; - -namespace Bonsai.Design -{ - class WorkflowEditorLauncher : DialogLauncher - { - bool userClosing; - readonly Func parentSelector; - readonly Func containerSelector; - - public WorkflowEditorLauncher(IWorkflowExpressionBuilder builder, Func parentSelector, Func containerSelector) - { - Builder = builder ?? throw new ArgumentNullException(nameof(builder)); - this.parentSelector = parentSelector ?? throw new ArgumentNullException(nameof(parentSelector)); - this.containerSelector = containerSelector ?? throw new ArgumentNullException(nameof(containerSelector)); - } - - internal IWorkflowExpressionBuilder Builder { get; } - - internal WorkflowGraphView ParentView - { - get { return parentSelector(); } - } - - internal WorkflowEditorControl Container - { - get { return containerSelector(); } - } - - internal IWin32Window Owner - { - get { return VisualizerDialog; } - } - - public VisualizerLayout VisualizerLayout { get; set; } - - public WorkflowGraphView WorkflowGraphView { get; private set; } - - public void UpdateEditorLayout() - { - if (WorkflowGraphView != null) - { - WorkflowGraphView.UpdateVisualizerLayout(); - VisualizerLayout = WorkflowGraphView.VisualizerLayout; - if (VisualizerDialog != null) - { - Bounds = VisualizerDialog.LayoutBounds; - } - } - } - - public void UpdateEditorText() - { - if (VisualizerDialog != null) - { - VisualizerDialog.Text = ExpressionBuilder.GetElementDisplayName(Builder); - if (VisualizerDialog.TopLevel == false) - { - Container.RefreshTab(Builder); - } - } - } - - public override void Show(IWin32Window owner, IServiceProvider provider) - { - if (VisualizerDialog != null && VisualizerDialog.TopLevel == false) - { - Container.SelectTab(Builder); - } - else base.Show(owner, provider); - } - - public override void Hide() - { - if (VisualizerDialog != null) - { - userClosing = false; - if (VisualizerDialog.TopLevel == false) - { - Container.CloseTab(Builder); - } - else base.Hide(); - } - } - - void EditorClosing(object sender, CancelEventArgs e) - { - if (userClosing) - { - e.Cancel = true; - ParentView.CloseWorkflowView(Builder); - ParentView.UpdateSelection(); - } - else - { - UpdateEditorLayout(); - WorkflowGraphView.HideEditorMapping(); - } - } - - protected override LauncherDialog CreateVisualizerDialog(IServiceProvider provider) - { - return new NestedEditorDialog(provider); - } - - protected override void InitializeComponents(TypeVisualizerDialog visualizerDialog, IServiceProvider provider) - { - var workflowEditor = Container; - if (workflowEditor == null) - { - workflowEditor = new WorkflowEditorControl(provider, ParentView.ReadOnly); - workflowEditor.SuspendLayout(); - workflowEditor.Dock = DockStyle.Fill; - workflowEditor.Font = ParentView.Font; - workflowEditor.Workflow = Builder.Workflow; - WorkflowGraphView = workflowEditor.WorkflowGraphView; - workflowEditor.ResumeLayout(false); - visualizerDialog.AddControl(workflowEditor); - visualizerDialog.Icon = Editor.Properties.Resources.Icon; - visualizerDialog.ShowIcon = true; - visualizerDialog.Activated += (sender, e) => workflowEditor.ActiveTab.UpdateSelection(); - visualizerDialog.FormClosing += (sender, e) => - { - if (e.CloseReason == CloseReason.UserClosing) - { - EditorClosing(sender, e); - } - }; - } - else - { - visualizerDialog.FormBorderStyle = FormBorderStyle.None; - visualizerDialog.Dock = DockStyle.Fill; - visualizerDialog.TopLevel = false; - visualizerDialog.Visible = true; - var tabState = workflowEditor.CreateTab(Builder, ParentView.ReadOnly, visualizerDialog); - WorkflowGraphView = tabState.WorkflowGraphView; - tabState.TabClosing += EditorClosing; - } - - userClosing = true; - visualizerDialog.BackColor = ParentView.ParentForm.BackColor; - WorkflowGraphView.BackColorChanged += (sender, e) => visualizerDialog.BackColor = ParentView.ParentForm.BackColor; - WorkflowGraphView.Launcher = this; - WorkflowGraphView.VisualizerLayout = VisualizerLayout; - WorkflowGraphView.SelectFirstGraphNode(); - WorkflowGraphView.Select(); - UpdateEditorText(); - } - - class NestedEditorDialog : LauncherDialog - { - IWorkflowEditorService editorService; - - public NestedEditorDialog(IServiceProvider provider) - { - editorService = (IWorkflowEditorService)provider.GetService(typeof(IWorkflowEditorService)); - } - - protected override void OnKeyDown(KeyEventArgs e) - { - if (e.KeyCode == Keys.Escape) - { - e.Handled = true; - } - base.OnKeyDown(e); - } - - protected override bool ProcessTabKey(bool forward) - { - var selected = SelectNextControl(ActiveControl, forward, true, true, false); - if (!selected) - { - var parent = Parent; - if (parent != null) return parent.SelectNextControl(this, forward, true, true, false); - else editorService.SelectNextControl(forward); - } - - return selected; - } - } - } -} diff --git a/Bonsai.Editor/Layout/WorkflowEditorSettings.cs b/Bonsai.Editor/Layout/WorkflowEditorSettings.cs index fa8337182..41dd89b07 100644 --- a/Bonsai.Editor/Layout/WorkflowEditorSettings.cs +++ b/Bonsai.Editor/Layout/WorkflowEditorSettings.cs @@ -1,5 +1,8 @@ -namespace Bonsai.Design +using System; + +namespace Bonsai.Design { + [Obsolete] public class WorkflowEditorSettings : VisualizerDialogSettings { public VisualizerDialogSettings EditorDialogSettings { get; set; } diff --git a/Bonsai.Editor/Properties/Resources.Designer.cs b/Bonsai.Editor/Properties/Resources.Designer.cs index fc2ef1486..ef1ca898c 100644 --- a/Bonsai.Editor/Properties/Resources.Designer.cs +++ b/Bonsai.Editor/Properties/Resources.Designer.cs @@ -287,6 +287,15 @@ internal static string InvalidReplaceGroupNode_Error { } } + /// + /// Looks up a localized string similar to The specified workflow path does not resolve to a workflow expression builder node.. + /// + internal static string InvalidWorkflowPath_Error { + get { + return ResourceManager.GetString("InvalidWorkflowPath_Error", resourceCulture); + } + } + /// /// Looks up a localized string similar to There was an error opening the workflow {0}: ///{1}. @@ -613,6 +622,16 @@ internal static System.Drawing.Bitmap WatchMenuImage { } } + /// + /// Looks up a localized resource of type System.Drawing.Bitmap. + /// + internal static System.Drawing.Bitmap WorkflowEditableImage { + get { + object obj = ResourceManager.GetObject("WorkflowEditableImage", resourceCulture); + return ((System.Drawing.Bitmap)(obj)); + } + } + /// /// Looks up a localized string similar to Externalized properties of this workflow can be configured below.. /// @@ -621,5 +640,15 @@ internal static string WorkflowPropertiesDescription { return ResourceManager.GetString("WorkflowPropertiesDescription", resourceCulture); } } + + /// + /// Looks up a localized resource of type System.Drawing.Bitmap. + /// + internal static System.Drawing.Bitmap WorkflowReadOnlyImage { + get { + object obj = ResourceManager.GetObject("WorkflowReadOnlyImage", resourceCulture); + return ((System.Drawing.Bitmap)(obj)); + } + } } } diff --git a/Bonsai.Editor/Properties/Resources.resx b/Bonsai.Editor/Properties/Resources.resx index 0aa5ca807..c8fb69609 100644 --- a/Bonsai.Editor/Properties/Resources.resx +++ b/Bonsai.Editor/Properties/Resources.resx @@ -134,7 +134,7 @@ iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAABGdBTUEAALGPC/xhBQAAAAlwSFlzAAAO - vwAADr8BOAVTJAAAAK5JREFUOE9jQAffvn1LAOL9QPwfDYPEEqDKMAFQUgGIz3/at/P/q2jf/09FGFAw + vAAADrwBlbxySQAAAK5JREFUOE9jQAffvn1LAOL9QPwfDYPEEqDKMAFQUgGIz3/at/P/q2jf/09FGFAw SAwkB1IDUgvVBgFQze9fZ8ZgaETHb8pzQIa8RzEEyDkPksCmARuGGnIepjkB5DRsCvFhqHcSQAbsx+Zn QhikB6QXZABWBcRgkF4UA4gFtDOAVAwzgOJApCwaoWnhPDGpEIZREhIIADngpExMasSalEEAagh5mQkZ ACVJyM4MDAD69UvNH5WBiAAAAABJRU5ErkJggg== @@ -143,7 +143,7 @@ iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAABGdBTUEAALGPC/xhBQAAAAlwSFlzAAAO - vwAADr8BOAVTJAAAANdJREFUOE+dU7kNwzAM1CjZwQtpBA/gwqVX8CKpAggI4FojuHNrwIVahZeQhh7a + vAAADrwBlbxySQAAANdJREFUOE+dU7kNwzAM1CjZwQtpBA/gwqVX8CKpAggI4FojuHNrwIVahZeQhh7a cXLAFSJ5Rz2UKRFCsERHjAURs1xWg5I3op/mKbb3NjZjkxEx5FCDWpZ9wOK1e3SVsOTwHGCyZia08Eho Ao1s4kVssTWtUNi7PgLLtuwxPo6FgdPOnBJCAEYSgwZaGGTFJbXuQmi/GmjdhapBKjjrDqoGqeisOygG 1SWKEDjqnl5i9YyyC+Co+/6MPAv+yhQKs0ECaPEe5SvTqI4ywCb/faYUlPzhOxvzAkO1WA01cJaNAAAA @@ -153,7 +153,7 @@ iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAABGdBTUEAALGPC/xhBQAAAAlwSFlzAAAO - wAAADsABataJCQAAANRJREFUOE+1U8ENwjAMzBAMwAiMwAJMAI98kfgzGx86QqbgU5CCeOSHgi/YlZNY + vAAADrwBlbxySQAAANRJREFUOE+1U8ENwjAMzBAMwAiMwAJMAI98kfgzGx86QqbgU5CCeOSHgi/YlZNY SFBx0qmNfXdtWse1SCl54kDMDVHzLOtBzSUxPM6nPG43+bJwFVFDDxpo2fYGm+N1v+uMLW/HA0JiFUKL gAYE43pV2Bp1nUOCmD1eTUTPeyzUIVadt+MRMMieRQiI2KoVLXngRcD0JCvEMgvh7QJAHQJYZvA/AdqM q75vQyRg9kec9xt5FoJMIQTaLNT1apAAWpRRlmn8RHOUAQ757TBpUPOL4+zcCzIffKHxkkn8AAAAAElF @@ -162,11 +162,11 @@ - iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8 - YQUAAAAJcEhZcwAADsQAAA7EAZUrDhsAAADBSURBVDhPY0AH3759SwDi/UD8Hw2DxBKgyjABUFIBiM8f - vX/0f8G2gv/GM41RMEgMJAdSA1IL1QYBUM3va/bUYGhExx2HOkCGvEcxBMg5D5LApgEbhhpyHqY5AeQ0 - dEU339z8H7kmEkMchqHeSQAZsB+bn2Fg5pmZGHIgDNID0gsyAKsCZIDLNSC9RBkAA+iuoZ8BhLxAcSBS - Fo3QtHCemFQIwygJCQSAHHBSJiY1Yk3KIAA1hLzMhAyAkiRkZwYGAEcIWvs/bCjHAAAAAElFTkSuQmCC + iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAABGdBTUEAALGPC/xhBQAAAAlwSFlzAAAO + vgAADr4B6kKxwAAAAMFJREFUOE9jQAffvn1LAOL9QPwfDYPEEqDKMAFQUgGIzx+9f/R/wbaC/8YzjVEw + SAwkB1IDUgvVBgFQze9r9tRgaETHHYc6QIa8RzEEyDkPksCmARuGGnIepjkB5DR0RTff3PwfuSYSQxyG + od5JABmwH5ufYWDmmZkYciAM0gPSCzIAqwJkgMs1IL1EGQAD6K6hnwGEvEBxIFIWjdC0cJ6YVAjDKAkJ + BIAccFImJjViTcogADWEvMyEDICSJGRnBgYARwha+z9sKMcAAAAASUVORK5CYII= @@ -220,7 +220,7 @@ Copyright (c) .NET Foundation and Contributors iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAABGdBTUEAALGPC/xhBQAAAAlwSFlzAAAO - wwAADsMBx2+oZAAAABh0RVh0U29mdHdhcmUAcGFpbnQubmV0IDQuMC41ZYUyZQAAAKdJREFUOE+lkMEN + vQAADr0BR/uQrQAAABh0RVh0U29mdHdhcmUAcGFpbnQubmV0IDQuMC41ZYUyZQAAAKdJREFUOE+lkMEN wyAQBCktooV8U4KLyTuklnxSkJ0P300WaZ3jwDaSLY18HLeDTQDQkHPG7f5CuD4LrNnzc6Rp+PCRpFps hUVP8i9+Dzc1PD3eVVD1kEAn2ZAkjYANi8LECvyeKIKUEmKM5V1tOgHxs0ENiw1Y/Byz1S/0ZB6dLNbL UINDy/wpKGTXmlsv0QguowJlSFewx5Dg9BecFuxxKBhBGQDhC/DB5AQ227rCAAAAAElFTkSuQmCC @@ -350,6 +350,26 @@ NOTE: You will have to restart Bonsai for any changes to take effect. Qz0ptsMAih6gaY+AAfMfC34AkgfRaOJgDNIHNgDI+SsnJyeJjpEV45D/BzYAaNILoHMMkCWB/ACg+FWo BRdAfGR5ZWVlI6D4c5gB24A4HlkBkL8aqKkAakAOiI8sD5RLBIptgRmQB3ImMgaKvQKFNkgeqEEQxMei Jg9swFAHDAwAKi9WPOw18+QAAAAASUVORK5CYII= + + + + The specified workflow path does not resolve to a workflow expression builder node. + + + + iVBORw0KGgoAAAANSUhEUgAAAA8AAAAPCAYAAAA71pVKAAAABGdBTUEAALGPC/xhBQAAAAlwSFlzAAAO + wAAADsABataJCQAAAH9JREFUOE/VjdEJwCAMRB2ls7ih/+7gHG7hBrY/+mk9iWKNVAqF0oMjJHmXiFcV + Y9xCCCY7dTaYEzIXBb1SKkkpm9FjfnsAH8bgcMAQypWXDdZaJ2ttqXWGPaFcfdg5V2DUfkYoF+CVCeXC + sn6Z+VF498el/0l49DK8MqGfSogTPS0oz8b0R/8AAAAASUVORK5CYII= + + + + + iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAABGdBTUEAALGPC/xhBQAAAAlwSFlzAAAO + wQAADsEBuJFr7QAAAJVJREFUOE+9kLENAyEQBCnFJbgEi1qoggKok4yAwAmCEHssTkL4/3UE9kob8Ghm + xZu/pJRyp+O4F8Ba65NuSwQOIXS6JZmWP7C1Vi9Zl3PO3Tmnk8wwgEAioaeCI3iVpJSuYe/9F0wvlwmX + 7/YjAfDpsgR4fusWTBAArD9MBRMRyGqMUQ8TBEv1sASgtfYYvY3Pv4wxL5igM/WVQzVaAAAAAElFTkSu + QmCC \ No newline at end of file diff --git a/Bonsai.Editor/ToolboxTreeView.cs b/Bonsai.Editor/ToolboxTreeView.cs index 0c89e4f39..8f3e16136 100644 --- a/Bonsai.Editor/ToolboxTreeView.cs +++ b/Bonsai.Editor/ToolboxTreeView.cs @@ -1,11 +1,24 @@ using System; using System.Drawing; using System.Windows.Forms; +using Bonsai.Editor.Themes; namespace Bonsai.Editor { class ToolboxTreeView : TreeView { + private ToolStripExtendedRenderer renderer; + + public ToolStripExtendedRenderer Renderer + { + get => renderer; + set + { + renderer = value; + UpdateTreeViewSelection(Focused); + } + } + protected override void OnEnabledChanged(EventArgs e) { base.OnEnabledChanged(e); @@ -23,5 +36,43 @@ void SetNodeEnabled(TreeNode node) SetNodeEnabled(child); } } + + protected override void OnEnter(EventArgs e) + { + UpdateTreeViewSelection(true); + base.OnEnter(e); + } + + protected override void OnLeave(EventArgs e) + { + UpdateTreeViewSelection(false); + base.OnLeave(e); + } + + protected override void OnAfterSelect(TreeViewEventArgs e) + { + UpdateTreeViewSelection(Focused); + base.OnAfterSelect(e); + } + + void UpdateTreeViewSelection(bool focused) + { + if (Renderer == null) + return; + + var colorTable = Renderer.ColorTable; + BackColor = colorTable.ContentPanelBackColor; + ForeColor = colorTable.WindowText; + + var selectedNode = SelectedNode; + if (Tag != selectedNode) + { + if (Tag is TreeNode previousNode) previousNode.BackColor = Color.Empty; + Tag = selectedNode; + } + + if (selectedNode == null) return; + selectedNode.BackColor = focused ? Color.Empty : colorTable.InactiveCaption; + } } } diff --git a/Bonsai.Editor/WorkflowRunner.cs b/Bonsai.Editor/WorkflowRunner.cs index adeeae33e..9cee150e2 100644 --- a/Bonsai.Editor/WorkflowRunner.cs +++ b/Bonsai.Editor/WorkflowRunner.cs @@ -32,9 +32,11 @@ static void RunLayout( workflowBuilder = new WorkflowBuilder(workflowBuilder.Workflow.ToInspectableGraph()); BuildAssignProperties(workflowBuilder, propertyAssignments); + + var visualizerSettings = VisualizerLayoutMap.FromVisualizerLayout(workflowBuilder, layout, typeVisualizers); + var visualizerDialogs = visualizerSettings.CreateVisualizerDialogs(workflowBuilder); LayoutHelper.SetWorkflowNotifications(workflowBuilder.Workflow, publishNotifications: false); - LayoutHelper.SetLayoutTags(workflowBuilder.Workflow, layout); - LayoutHelper.SetLayoutNotifications(layout); + LayoutHelper.SetLayoutNotifications(workflowBuilder.Workflow, visualizerDialogs); var services = new System.ComponentModel.Design.ServiceContainer(); services.AddService(typeof(WorkflowBuilder), workflowBuilder); @@ -42,31 +44,14 @@ static void RunLayout( var cts = new CancellationTokenSource(); var contextMenu = new ContextMenuStrip(); - void CreateVisualizerMapping(ExpressionBuilderGraph workflow, VisualizerLayout layout) + foreach (var launcher in visualizerDialogs) { - var mapping = LayoutHelper.CreateVisualizerMapping(workflow, layout, typeVisualizers, services); - foreach (var launcher in mapping.Values.Where(launcher => launcher.Visualizer.IsValueCreated)) + var activeLauncher = launcher; + contextMenu.Items.Add(new ToolStripMenuItem(launcher.Text, null, (sender, e) => { - var activeLauncher = launcher; - contextMenu.Items.Add(new ToolStripMenuItem(launcher.Text, null, (sender, e) => - { - activeLauncher.Show(services); - })); - } - - foreach (var settings in layout.DialogSettings) - { - if (settings is WorkflowEditorSettings editorSettings && - editorSettings.Tag is ExpressionBuilder builder && - ExpressionBuilder.Unwrap(builder) is IWorkflowExpressionBuilder workflowBuilder && - editorSettings.EditorVisualizerLayout != null && - editorSettings.EditorDialogSettings.Visible) - { - CreateVisualizerMapping(workflowBuilder.Workflow, editorSettings.EditorVisualizerLayout); - } - } + activeLauncher.Show(services); + })); } - CreateVisualizerMapping(workflowBuilder.Workflow, layout); contextMenu.Items.Add(new ToolStripSeparator()); contextMenu.Items.Add(new ToolStripMenuItem("Stop", null, (sender, e) => cts.Cancel())); @@ -76,6 +61,7 @@ editorSettings.Tag is ExpressionBuilder builder && notifyIcon.ContextMenuStrip = contextMenu; notifyIcon.Visible = true; + visualizerDialogs.Show(visualizerSettings, services); using var synchronizationContext = new WindowsFormsSynchronizationContext(); runtimeWorkflow.Finally(() => { diff --git a/Bonsai/DependencyInspector.cs b/Bonsai/DependencyInspector.cs index 7076fb1a5..69be47211 100644 --- a/Bonsai/DependencyInspector.cs +++ b/Bonsai/DependencyInspector.cs @@ -34,10 +34,9 @@ static IEnumerable GetVisualizerSettings(VisualizerLay foreach (var settings in layout.DialogSettings) { yield return settings; - var editorSettings = settings as WorkflowEditorSettings; - if (editorSettings != null && editorSettings.EditorVisualizerLayout != null) + if (settings.NestedLayout != null) { - stack.Push(editorSettings.EditorVisualizerLayout); + stack.Push(settings.NestedLayout); } } }