mirror of https://github.com/icsharpcode/ILSpy.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
254 lines
6.9 KiB
254 lines
6.9 KiB
// Copyright (c) 2011 AlphaSierraPapa for the SharpDevelop Team |
|
// |
|
// Permission is hereby granted, free of charge, to any person obtaining a copy of this |
|
// software and associated documentation files (the "Software"), to deal in the Software |
|
// without restriction, including without limitation the rights to use, copy, modify, merge, |
|
// publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons |
|
// to whom the Software is furnished to do so, subject to the following conditions: |
|
// |
|
// The above copyright notice and this permission notice shall be included in all copies or |
|
// substantial portions of the Software. |
|
// |
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, |
|
// INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR |
|
// PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE |
|
// FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR |
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER |
|
// DEALINGS IN THE SOFTWARE. |
|
|
|
using System; |
|
using System.Collections.Generic; |
|
using ICSharpCode.Decompiler; |
|
using ICSharpCode.Decompiler.ILAst; |
|
using ICSharpCode.NRefactory.VB; |
|
using ICSharpCode.NRefactory.VB.Ast; |
|
using Mono.Cecil; |
|
|
|
namespace ICSharpCode.ILSpy.VB |
|
{ |
|
/// <summary> |
|
/// Description of VBTextOutputFormatter. |
|
/// </summary> |
|
public class VBTextOutputFormatter : IOutputFormatter |
|
{ |
|
readonly ITextOutput output; |
|
readonly Stack<AstNode> nodeStack = new Stack<AstNode>(); |
|
|
|
bool firstImport, lastImport; |
|
|
|
public VBTextOutputFormatter(ITextOutput output) |
|
{ |
|
if (output == null) |
|
throw new ArgumentNullException("output"); |
|
this.output = output; |
|
} |
|
|
|
public void StartNode(AstNode node) |
|
{ |
|
// var ranges = node.Annotation<List<ILRange>>(); |
|
// if (ranges != null && ranges.Count > 0) |
|
// { |
|
// // find the ancestor that has method mapping as annotation |
|
// if (node.Ancestors != null && node.Ancestors.Count() > 0) |
|
// { |
|
// var n = node.Ancestors.FirstOrDefault(a => a.Annotation<MemberMapping>() != null); |
|
// if (n != null) { |
|
// MemberMapping mapping = n.Annotation<MemberMapping>(); |
|
// |
|
// // add all ranges |
|
// foreach (var range in ranges) { |
|
// mapping.MemberCodeMappings.Add(new SourceCodeMapping { |
|
// ILInstructionOffset = range, |
|
// SourceCodeLine = output.CurrentLine, |
|
// MemberMapping = mapping |
|
// }); |
|
// } |
|
// } |
|
// } |
|
// } |
|
if (nodeStack.Count == 0) { |
|
if (node is ImportsStatement) { |
|
firstImport = !(node.PrevSibling is ImportsStatement); |
|
lastImport = !(node.NextSibling is ImportsStatement); |
|
} else { |
|
firstImport = false; |
|
lastImport = false; |
|
} |
|
} |
|
nodeStack.Push(node); |
|
} |
|
|
|
public void EndNode(AstNode node) |
|
{ |
|
if (nodeStack.Pop() != node) |
|
throw new InvalidOperationException(); |
|
} |
|
|
|
public void WriteIdentifier(string identifier) |
|
{ |
|
var definition = GetCurrentDefinition(); |
|
if (definition != null) { |
|
output.WriteDefinition(identifier, definition); |
|
return; |
|
} |
|
|
|
object memberRef = GetCurrentMemberReference(); |
|
if (memberRef != null) { |
|
output.WriteReference(identifier, memberRef); |
|
return; |
|
} |
|
|
|
definition = GetCurrentLocalDefinition(); |
|
if (definition != null) { |
|
output.WriteDefinition(identifier, definition); |
|
return; |
|
} |
|
|
|
memberRef = GetCurrentLocalReference(); |
|
if (memberRef != null) { |
|
output.WriteReference(identifier, memberRef, true); |
|
return; |
|
} |
|
|
|
if (firstImport) { |
|
output.MarkFoldStart(defaultCollapsed: true); |
|
firstImport = false; |
|
} |
|
|
|
output.Write(identifier); |
|
} |
|
|
|
MemberReference GetCurrentMemberReference() |
|
{ |
|
AstNode node = nodeStack.Peek(); |
|
MemberReference memberRef = node.Annotation<MemberReference>(); |
|
if (memberRef == null && node.Role == AstNode.Roles.TargetExpression && (node.Parent is InvocationExpression || node.Parent is ObjectCreationExpression)) { |
|
memberRef = node.Parent.Annotation<MemberReference>(); |
|
} |
|
return memberRef; |
|
} |
|
|
|
object GetCurrentLocalReference() |
|
{ |
|
AstNode node = nodeStack.Peek(); |
|
ILVariable variable = node.Annotation<ILVariable>(); |
|
if (variable != null) { |
|
if (variable.OriginalParameter != null) |
|
return variable.OriginalParameter; |
|
//if (variable.OriginalVariable != null) |
|
// return variable.OriginalVariable; |
|
return variable; |
|
} |
|
return null; |
|
} |
|
|
|
object GetCurrentLocalDefinition() |
|
{ |
|
AstNode node = nodeStack.Peek(); |
|
var parameterDef = node.Annotation<ParameterDefinition>(); |
|
if (parameterDef != null) |
|
return parameterDef; |
|
|
|
if (node is VariableInitializer || node is CatchBlock || node is ForEachStatement) { |
|
var variable = node.Annotation<ILVariable>(); |
|
if (variable != null) { |
|
if (variable.OriginalParameter != null) |
|
return variable.OriginalParameter; |
|
//if (variable.OriginalVariable != null) |
|
// return variable.OriginalVariable; |
|
return variable; |
|
} else { |
|
|
|
} |
|
} |
|
|
|
return null; |
|
} |
|
|
|
object GetCurrentDefinition() |
|
{ |
|
if (nodeStack == null || nodeStack.Count == 0) |
|
return null; |
|
|
|
var node = nodeStack.Peek(); |
|
if (IsDefinition(node)) |
|
return node.Annotation<MemberReference>(); |
|
|
|
node = node.Parent; |
|
if (IsDefinition(node)) |
|
return node.Annotation<MemberReference>(); |
|
|
|
return null; |
|
} |
|
|
|
public void WriteKeyword(string keyword) |
|
{ |
|
output.Write(keyword); |
|
} |
|
|
|
public void WriteToken(string token) |
|
{ |
|
// Attach member reference to token only if there's no identifier in the current node. |
|
MemberReference memberRef = GetCurrentMemberReference(); |
|
if (memberRef != null && nodeStack.Peek().GetChildByRole(AstNode.Roles.Identifier).IsNull) |
|
output.WriteReference(token, memberRef); |
|
else |
|
output.Write(token); |
|
} |
|
|
|
public void Space() |
|
{ |
|
output.Write(' '); |
|
} |
|
|
|
public void Indent() |
|
{ |
|
output.Indent(); |
|
} |
|
|
|
public void Unindent() |
|
{ |
|
output.Unindent(); |
|
} |
|
|
|
public void NewLine() |
|
{ |
|
if (lastImport) { |
|
output.MarkFoldEnd(); |
|
lastImport = false; |
|
} |
|
output.WriteLine(); |
|
} |
|
|
|
public void WriteComment(bool isDocumentation, string content) |
|
{ |
|
if (isDocumentation) |
|
output.Write("'''"); |
|
else |
|
output.Write("'"); |
|
output.WriteLine(content); |
|
} |
|
|
|
public void MarkFoldStart() |
|
{ |
|
output.MarkFoldStart(); |
|
} |
|
|
|
public void MarkFoldEnd() |
|
{ |
|
output.MarkFoldEnd(); |
|
} |
|
|
|
private static bool IsDefinition(AstNode node) |
|
{ |
|
return |
|
node is FieldDeclaration || |
|
node is ConstructorDeclaration || |
|
node is EventDeclaration || |
|
node is DelegateDeclaration || |
|
node is OperatorDeclaration|| |
|
node is MemberDeclaration || |
|
node is TypeDeclaration; |
|
} |
|
} |
|
}
|
|
|