#region License Information
/* HeuristicLab
* Copyright (C) 2002-2018 Heuristic and Evolutionary Algorithms Laboratory (HEAL)
*
* This file is part of HeuristicLab.
*
* HeuristicLab is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* HeuristicLab is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with HeuristicLab. If not, see .
*/
#endregion
using System;
using System.Collections.Generic;
using System.Drawing;
using System.Linq;
using System.Reflection;
using System.Text;
using System.Windows.Forms;
using HeuristicLab.CodeEditor;
using HeuristicLab.Common.Resources;
using HeuristicLab.Core.Views;
using HeuristicLab.MainForm;
using HeuristicLab.PluginInfrastructure;
namespace HeuristicLab.Operators.Programmable {
[View("ProgrammableOperator View")]
[Content(typeof(ProgrammableOperator), true)]
public partial class ProgrammableOperatorView : NamedItemView {
public ProgrammableOperator ProgrammableOperator {
get { return (ProgrammableOperator)base.Content; }
set { base.Content = (ProgrammableOperator)value; }
}
public ProgrammableOperatorView() {
InitializeComponent();
namespacesTreeView.ImageList = new ImageList();
namespacesTreeView.ImageList.Images.Add(VSImageLibrary.Namespace);
assembliesTreeView.ImageList = new ImageList();
assembliesTreeView.ImageList.Images.Add(VSImageLibrary.Assembly);
assembliesTreeView.ImageList.Images.Add(VSImageLibrary.Module);
}
protected override void RegisterContentEvents() {
base.RegisterContentEvents();
ProgrammableOperator.CodeChanged += ProgrammableOperator_CodeChanged;
ProgrammableOperator.SignatureChanged += ProgrammableOperator_SignatureChanged;
ProgrammableOperator.BreakpointChanged += ProgrammableOperator_BreakpointChanged;
}
protected override void DeregisterContentEvents() {
ProgrammableOperator.CodeChanged -= ProgrammableOperator_CodeChanged;
ProgrammableOperator.SignatureChanged -= ProgrammableOperator_SignatureChanged;
ProgrammableOperator.BreakpointChanged -= ProgrammableOperator_BreakpointChanged;
base.DeregisterContentEvents();
}
#region Content event handlers
void ProgrammableOperator_BreakpointChanged(object sender, EventArgs e) {
breakpointCheckBox.Checked = ProgrammableOperator.Breakpoint;
}
private void ProgrammableOperator_CodeChanged(object sender, EventArgs e) {
codeEditor.UserCode = ProgrammableOperator.Code;
}
private void ProgrammableOperator_SignatureChanged(object sender, EventArgs args) {
codeEditor.Prefix = GetGeneratedPrefix();
}
#endregion
protected override void OnContentChanged() {
base.OnContentChanged();
if (ProgrammableOperator == null) {
codeEditor.UserCode = string.Empty;
codeEditor.ClearEditHistory();
assembliesTreeView.Nodes.Clear();
parameterCollectionView.Content = null;
} else {
codeEditor.Prefix = GetGeneratedPrefix();
codeEditor.Suffix = String.Format(" {0}{1} }}{1}}}", ProgrammableOperator.MethodSuffix, Environment.NewLine);
codeEditor.UserCode = ProgrammableOperator.Code;
if (codeEditor.UserCode == string.Empty)
codeEditor.UserCode = string.Format(" {0}", Environment.NewLine);
codeEditor.ClearEditHistory();
InitializeAssemblyList();
InitializeNamespacesList();
codeEditor.AddAssemblies(ProgrammableOperator.SelectedAssemblies);
codeEditor.ScrollAfterPrefix();
codeEditor.ShowCompileErrors(ProgrammableOperator.CompileErrors);
showCodeButton.Enabled = !string.IsNullOrEmpty(ProgrammableOperator.CompilationUnitCode);
parameterCollectionView.Content = ProgrammableOperator.Parameters;
}
UpdateCompilationLabel();
}
private void UpdateCompilationLabel() {
if (ProgrammableOperator == null || ProgrammableOperator.CompileErrors == null) {
compilationLabel.ForeColor = SystemColors.ControlDarkDark;
compilationLabel.Text = "Not compiled";
} else if (ProgrammableOperator.CompileErrors.HasErrors) {
compilationLabel.ForeColor = Color.DarkRed;
compilationLabel.Text = "Compilation failed";
} else {
compilationLabel.ForeColor = Color.DarkGreen;
compilationLabel.Text = "Compilation successful";
}
}
protected override void SetEnabledStateOfControls() {
base.SetEnabledStateOfControls();
breakpointCheckBox.Enabled = Content != null && !Locked;
parameterCollectionView.Enabled = Content != null;
assembliesTreeView.Enabled = Content != null && !ReadOnly;
namespacesTreeView.Enabled = Content != null && !ReadOnly;
compileButton.Enabled = Content != null && !ReadOnly;
codeEditor.Enabled = Content != null && !ReadOnly;
showCodeButton.Enabled = Content != null && !string.IsNullOrEmpty(ProgrammableOperator.CompilationUnitCode) && !ReadOnly;
}
#region Child Control event handlers
private void compileButton_Click(object sender, EventArgs e) {
Recompile();
}
private void breakpointCheckBox_CheckedChanged(object sender, EventArgs e) {
if (ProgrammableOperator != null) ProgrammableOperator.Breakpoint = breakpointCheckBox.Checked;
}
private void showCodeButton_Click(object sender, EventArgs e) {
#if __MonoCS__
new TextDialog("CodeViewer", ProgrammableOperator.CompilationUnitCode, true).ShowDialog(this);
#else
new CodeViewer(ProgrammableOperator.CompilationUnitCode).ShowDialog(this);
#endif
}
private void codeEditor_Validated(object sender, EventArgs e) {
ProgrammableOperator.Code = codeEditor.UserCode;
}
private void assembliesTreeView_AfterCheck(object sender, TreeViewEventArgs e) {
if (initializing)
return;
Assembly a = e.Node.Tag as Assembly;
if (a == null && e.Node.Nodes.Count > 0) {
foreach (TreeNode n in e.Node.Nodes)
n.Checked = e.Node.Checked;
return;
} else {
if (e.Node.Checked) {
ProgrammableOperator.SelectAssembly(a);
codeEditor.AddAssembly(a);
} else {
ProgrammableOperator.UnselectAssembly(a);
codeEditor.RemoveAssembly(a);
}
}
InitializeNamespacesList();
}
private void namespacesTreeView_AfterCheck(object sender, TreeViewEventArgs e) {
if (initializing)
return;
if (e.Node.Checked) {
ProgrammableOperator.SelectNamespace(e.Node.FullPath);
} else {
ProgrammableOperator.UnselectNamespace(e.Node.FullPath);
}
}
#endregion
#region global HotKeys
protected override bool ProcessCmdKey(ref Message msg, Keys keyData) {
if (keyData == Keys.F6) {
Control activeControl = ActiveControl;
compileButton.Focus();
Recompile();
if (activeControl != null)
activeControl.Focus();
return true;
}
return base.ProcessCmdKey(ref msg, keyData);
}
#endregion
#region Auxiliary functions
private string GetGeneratedPrefix() {
StringBuilder prefix = new StringBuilder();
prefix.Append("public class ").Append(ProgrammableOperator.CompiledTypeName).AppendLine(" {");
prefix.Append(" ").Append(ProgrammableOperator.Signature).AppendLine(" {");
return prefix.ToString();
}
private void Recompile() {
this.Enabled = false;
try {
ProgrammableOperator.Compile();
} catch (Exception ex) {
ErrorHandling.ShowErrorDialog(this, ex);
}
this.Enabled = true;
UpdateCompilationLabel();
codeEditor.ShowCompileErrors(ProgrammableOperator.CompileErrors);
}
private bool initializing = false;
private void InitializeAssemblyList() {
initializing = true;
assembliesTreeView.Enabled = false;
namespacesTreeView.Enabled = false;
assembliesTreeView.BeginUpdate();
assembliesTreeView.Nodes.Clear();
var selectedAssemblies = new HashSet(ProgrammableOperator.SelectedAssemblies);
foreach (var p in ProgrammableOperator.Plugins) {
var node = assembliesTreeView.Nodes.Add(p.Key);
node.Tag = p;
node.ImageIndex = 1;
foreach (var a in p.Value) {
var aNode = node.Nodes.Add(a.GetName().Name);
aNode.Tag = a;
aNode.ImageIndex = 0;
if (selectedAssemblies.Contains(a))
aNode.Checked = true;
}
if (node.Nodes.Count == 1 && node.Nodes[0].Name == node.Nodes[0].Name) {
node.Tag = node.Nodes[0].Tag;
node.ImageIndex = node.Nodes[0].ImageIndex;
node.Checked = node.Nodes[0].Checked;
node.Nodes.Clear();
} else if (node.Nodes.Count > 0 && node.Nodes.Cast().All(n => n.Checked)) {
node.Checked = true;
}
}
assembliesTreeView.Sort();
assembliesTreeView.EndUpdate();
assembliesTreeView.Enabled = true;
namespacesTreeView.Enabled = true;
initializing = false;
}
private void InitializeNamespacesList() {
initializing = true;
namespacesTreeView.Enabled = false;
namespacesTreeView.BeginUpdate();
TreeNode oldTree = new TreeNode("root");
CloneTreeNodeCollection(oldTree, namespacesTreeView.Nodes);
namespacesTreeView.Nodes.Clear();
var selectedNamespaces = new HashSet(ProgrammableOperator.Namespaces);
foreach (var ns in ProgrammableOperator.GetAllNamespaces(true))
AddNamespace(namespacesTreeView.Nodes, ns, selectedNamespaces.Contains(ns), oldTree);
codeEditor.Prefix = GetGeneratedPrefix();
namespacesTreeView.Sort();
namespacesTreeView.EndUpdate();
namespacesTreeView.Enabled = true;
initializing = false;
}
private void CloneTreeNodeCollection(TreeNode root, TreeNodeCollection nodes) {
foreach (TreeNode n in nodes) {
TreeNode newNode = root.Nodes.Add(n.Text, n.Text);
newNode.Checked = n.Checked;
newNode.ImageIndex = n.ImageIndex;
CloneTreeNodeCollection(newNode, n.Nodes);
if (n.IsExpanded)
newNode.Expand();
}
}
private bool AddNamespace(TreeNodeCollection parentNodes, string ns, bool isSelected, TreeNode oldTree) {
int dotIndex = ns.IndexOf('.');
string prefix = ns;
if (dotIndex != -1)
prefix = ns.Substring(0, dotIndex);
TreeNode node = GetOrCreateNode(parentNodes, prefix);
TreeNode oldNode = MaybeGetNode(oldTree, prefix);
bool isNew = oldNode == null;
if (dotIndex != -1 && dotIndex + 1 < ns.Length) {
isNew = AddNamespace(node.Nodes, ns.Substring(dotIndex + 1, ns.Length - (dotIndex + 1)), isSelected, oldNode);
} else {
node.Checked = isSelected;
}
if (isNew || oldNode != null && oldNode.IsExpanded)
node.Expand();
if (isNew) {
namespacesTreeView.SelectedNode = node;
node.ImageIndex = 0;
} else {
node.ImageIndex = oldNode.ImageIndex;
}
return isNew;
}
private static TreeNode MaybeGetNode(TreeNode parentNode, string key) {
if (parentNode == null)
return null;
if (parentNode.Nodes.ContainsKey(key))
return parentNode.Nodes[key];
return null;
}
private static TreeNode GetOrCreateNode(TreeNodeCollection parentNodes, string key) {
TreeNode node = null;
if (parentNodes.ContainsKey(key)) {
node = parentNodes[key];
} else {
node = parentNodes.Add(key, key);
}
return node;
}
#endregion
}
}