// Copyright (c) AlphaSierraPapa for the SharpDevelop Team (for details please see \doc\copyright.txt) // This code is distributed under MIT X11 license (for details please see \doc\license.txt) using System; using System.Collections.Generic; using System.Diagnostics; using System.Drawing; using System.IO; using System.Reflection; using System.Text; using System.Threading; using System.Threading.Tasks; using System.Windows.Forms; using ICSharpCode.NRefactory.CSharp; using ICSharpCode.NRefactory.CSharp.Resolver; using ICSharpCode.NRefactory.TypeSystem; using ICSharpCode.NRefactory.TypeSystem.Implementation; namespace ICSharpCode.NRefactory.Demo { /// /// Description of MainForm. /// public partial class MainForm : Form { public MainForm() { // // The InitializeComponent() call is required for Windows Forms designer support. // InitializeComponent(); csharpCodeTextBox.SelectAll(); CSharpParseButtonClick(null, null); resolveButton.UseWaitCursor = true; ThreadPool.QueueUserWorkItem( delegate { builtInLibs.Value.ToString(); BeginInvoke(new Action(delegate { resolveButton.UseWaitCursor = false; })); }); } CompilationUnit compilationUnit; void CSharpParseButtonClick(object sender, EventArgs e) { CSharpParser parser = new CSharpParser(); compilationUnit = parser.Parse(new StringReader(csharpCodeTextBox.Text)); csharpTreeView.Nodes.Clear(); foreach (var element in compilationUnit.Children) { csharpTreeView.Nodes.Add(MakeTreeNode(element)); } SelectCurrentNode(csharpTreeView.Nodes); resolveButton.Enabled = true; } TreeNode MakeTreeNode(INode node) { TreeNode t = new TreeNode(GetNodeTitle(node)); t.Tag = node; foreach (INode child in node.Children) { t.Nodes.Add(MakeTreeNode(child)); } return t; } string GetNodeTitle(INode node) { StringBuilder b = new StringBuilder(); b.Append(DecodeRole(node.Role, node.Parent != null ? node.Parent.GetType() : null)); b.Append(": "); b.Append(node.GetType().Name); bool hasProperties = false; foreach (PropertyInfo p in node.GetType().GetProperties(BindingFlags.Public | BindingFlags.Instance)) { if (p.PropertyType == typeof(string) || p.PropertyType.IsEnum) { if (!hasProperties) { hasProperties = true; b.Append(" ("); } else { b.Append(", "); } b.Append(p.Name); b.Append(" = "); try { object val = p.GetValue(node, null); b.Append(val != null ? val.ToString() : "**null**"); } catch (TargetInvocationException ex) { b.Append("**" + ex.InnerException.GetType().Name + "**"); } } } if (hasProperties) b.Append(")"); return b.ToString(); } string DecodeRole(int role, Type type) { if (type != null) { foreach (FieldInfo field in type.GetFields(BindingFlags.Public | BindingFlags.Static | BindingFlags.FlattenHierarchy)) { if (field.FieldType == typeof(int) && (int)field.GetValue(null) == role) return field.Name; } } foreach (FieldInfo field in typeof(AbstractNode.Roles).GetFields(BindingFlags.Public | BindingFlags.Static)) { if (field.FieldType == typeof(int) && (int)field.GetValue(null) == role) return field.Name; } return role.ToString(); } bool SelectCurrentNode(TreeNodeCollection c) { int selectionStart = csharpCodeTextBox.SelectionStart; int selectionEnd = selectionStart + csharpCodeTextBox.SelectionLength; foreach (TreeNode t in c) { INode node = t.Tag as INode; if (node != null && selectionStart >= GetOffset(csharpCodeTextBox, node.StartLocation) && selectionEnd <= GetOffset(csharpCodeTextBox, node.EndLocation)) { if (selectionStart == selectionEnd && (selectionStart == GetOffset(csharpCodeTextBox, node.StartLocation) || selectionStart == GetOffset(csharpCodeTextBox, node.EndLocation))) { // caret is on border of this node; don't expand csharpTreeView.SelectedNode = t; } else { t.Expand(); if (!SelectCurrentNode(t.Nodes)) csharpTreeView.SelectedNode = t; } return true; } } return false; } void CSharpGenerateCodeButtonClick(object sender, EventArgs e) { throw new NotImplementedException(); } int GetOffset(TextBox textBox, DomLocation location) { return textBox.GetFirstCharIndexFromLine(location.Line - 1) + location.Column - 1; } void CSharpTreeViewAfterSelect(object sender, TreeViewEventArgs e) { INode node = e.Node.Tag as INode; if (node != null) { int startOffset = GetOffset(csharpCodeTextBox, node.StartLocation); int endOffset = GetOffset(csharpCodeTextBox, node.EndLocation); csharpCodeTextBox.Select(startOffset, endOffset - startOffset); } } Lazy> builtInLibs = new Lazy>( delegate { Assembly[] assemblies = { typeof(object).Assembly, // mscorlib typeof(Uri).Assembly, // System.dll typeof(System.Linq.Enumerable).Assembly, // System.Core.dll // typeof(System.Xml.XmlDocument).Assembly, // System.Xml.dll // typeof(System.Drawing.Bitmap).Assembly, // System.Drawing.dll // typeof(Form).Assembly, // System.Windows.Forms.dll typeof(ICSharpCode.NRefactory.TypeSystem.IProjectContent).Assembly, }; IProjectContent[] projectContents = new IProjectContent[assemblies.Length]; Stopwatch total = Stopwatch.StartNew(); Parallel.For( 0, assemblies.Length, delegate (int i) { Stopwatch w = Stopwatch.StartNew(); CecilLoader loader = new CecilLoader(); projectContents[i] = loader.LoadAssemblyFile(assemblies[i].Location); Debug.WriteLine(Path.GetFileName(assemblies[i].Location) + ": " + w.Elapsed); }); Debug.WriteLine("Total: " + total.Elapsed); return projectContents; }); void ResolveButtonClick(object sender, EventArgs e) { SimpleProjectContent project = new SimpleProjectContent(); TypeSystemConvertVisitor convertVisitor = new TypeSystemConvertVisitor(project, "dummy.cs"); convertVisitor.VisitCompilationUnit(compilationUnit, null); project.UpdateProjectContent(null, convertVisitor.ParsedFile.TopLevelTypeDefinitions, null, null); List projects = new List(); projects.Add(project); projects.AddRange(builtInLibs.Value); ITypeResolveContext context = new CompositeTypeResolveContext(projects); CSharpResolver resolver = new CSharpResolver(context); IResolveVisitorNavigator navigator = null; if (csharpTreeView.SelectedNode != null) { navigator = new NodeListResolveVisitorNavigator(new[] { (INode)csharpTreeView.SelectedNode.Tag }); } ResolveVisitor visitor = new ResolveVisitor(resolver, convertVisitor.ParsedFile, navigator); visitor.Scan(compilationUnit); csharpTreeView.BeginUpdate(); ShowResolveResultsInTree(csharpTreeView.Nodes, visitor); csharpTreeView.EndUpdate(); } void ShowResolveResultsInTree(TreeNodeCollection c, ResolveVisitor v) { foreach (TreeNode t in c) { INode node = t.Tag as INode; if (node != null) { ResolveResult rr = v.GetResolveResult(node); if (rr != null) t.Text = GetNodeTitle(node) + " " + rr.ToString(); else t.Text = GetNodeTitle(node); } ShowResolveResultsInTree(t.Nodes, v); } } void CSharpCodeTextBoxKeyDown(object sender, KeyEventArgs e) { if (e.Control && e.KeyCode == Keys.A) { e.Handled = true; csharpCodeTextBox.SelectAll(); } } void CsharpCodeTextBoxTextChanged(object sender, EventArgs e) { resolveButton.Enabled = false; } } }