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.
439 lines
12 KiB
439 lines
12 KiB
// Copyright (c) 2010-2013 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.Globalization; |
|
using System.IO; |
|
using System.Text; |
|
using ICSharpCode.Decompiler.CSharp.Syntax; |
|
|
|
namespace ICSharpCode.Decompiler.CSharp.OutputVisitor |
|
{ |
|
/// <summary> |
|
/// Writes C# code into a TextWriter. |
|
/// </summary> |
|
public class TextWriterTokenWriter : TokenWriter, ILocatable |
|
{ |
|
readonly TextWriter textWriter; |
|
int indentation; |
|
bool needsIndent = true; |
|
bool isAtStartOfLine = true; |
|
int line, column; |
|
|
|
public int Indentation { |
|
get { return this.indentation; } |
|
set { this.indentation = value; } |
|
} |
|
|
|
public TextLocation Location { |
|
get { return new TextLocation(line, column + (needsIndent ? indentation * IndentationString.Length : 0)); } |
|
} |
|
|
|
public string IndentationString { get; set; } |
|
|
|
public TextWriterTokenWriter(TextWriter textWriter) |
|
{ |
|
if (textWriter == null) |
|
throw new ArgumentNullException("textWriter"); |
|
this.textWriter = textWriter; |
|
this.IndentationString = "\t"; |
|
this.line = 1; |
|
this.column = 1; |
|
} |
|
|
|
public override void WriteIdentifier(Identifier identifier) |
|
{ |
|
WriteIndentation(); |
|
if (identifier.IsVerbatim || CSharpOutputVisitor.IsKeyword(identifier.Name, identifier)) { |
|
textWriter.Write('@'); |
|
column++; |
|
} |
|
textWriter.Write(identifier.Name); |
|
column += identifier.Name.Length; |
|
isAtStartOfLine = false; |
|
} |
|
|
|
public override void WriteKeyword(Role role, string keyword) |
|
{ |
|
WriteIndentation(); |
|
column += keyword.Length; |
|
textWriter.Write(keyword); |
|
isAtStartOfLine = false; |
|
} |
|
|
|
public override void WriteToken(Role role, string token) |
|
{ |
|
WriteIndentation(); |
|
column += token.Length; |
|
textWriter.Write(token); |
|
isAtStartOfLine = false; |
|
} |
|
|
|
public override void Space() |
|
{ |
|
WriteIndentation(); |
|
column++; |
|
textWriter.Write(' '); |
|
} |
|
|
|
protected void WriteIndentation() |
|
{ |
|
if (needsIndent) { |
|
needsIndent = false; |
|
for (int i = 0; i < indentation; i++) { |
|
textWriter.Write(this.IndentationString); |
|
} |
|
column += indentation * IndentationString.Length; |
|
} |
|
} |
|
|
|
public override void NewLine() |
|
{ |
|
textWriter.WriteLine(); |
|
column = 1; |
|
line++; |
|
needsIndent = true; |
|
isAtStartOfLine = true; |
|
} |
|
|
|
public override void Indent() |
|
{ |
|
indentation++; |
|
} |
|
|
|
public override void Unindent() |
|
{ |
|
indentation--; |
|
} |
|
|
|
public override void WriteComment(CommentType commentType, string content) |
|
{ |
|
WriteIndentation(); |
|
switch (commentType) { |
|
case CommentType.SingleLine: |
|
textWriter.Write("//"); |
|
textWriter.WriteLine(content); |
|
column += 2 + content.Length; |
|
needsIndent = true; |
|
isAtStartOfLine = true; |
|
break; |
|
case CommentType.MultiLine: |
|
textWriter.Write("/*"); |
|
textWriter.Write(content); |
|
textWriter.Write("*/"); |
|
column += 2; |
|
UpdateEndLocation(content, ref line, ref column); |
|
column += 2; |
|
isAtStartOfLine = false; |
|
break; |
|
case CommentType.Documentation: |
|
textWriter.Write("///"); |
|
textWriter.WriteLine(content); |
|
column += 3 + content.Length; |
|
needsIndent = true; |
|
isAtStartOfLine = true; |
|
break; |
|
case CommentType.MultiLineDocumentation: |
|
textWriter.Write("/**"); |
|
textWriter.Write(content); |
|
textWriter.Write("*/"); |
|
column += 3; |
|
UpdateEndLocation(content, ref line, ref column); |
|
column += 2; |
|
isAtStartOfLine = false; |
|
break; |
|
default: |
|
textWriter.Write(content); |
|
column += content.Length; |
|
break; |
|
} |
|
} |
|
|
|
static void UpdateEndLocation(string content, ref int line, ref int column) |
|
{ |
|
if (string.IsNullOrEmpty(content)) |
|
return; |
|
for (int i = 0; i < content.Length; i++) { |
|
char ch = content[i]; |
|
switch (ch) { |
|
case '\r': |
|
if (i + 1 < content.Length && content[i + 1] == '\n') |
|
i++; |
|
goto case '\n'; |
|
case '\n': |
|
line++; |
|
column = 0; |
|
break; |
|
} |
|
column++; |
|
} |
|
} |
|
|
|
public override void WritePreProcessorDirective(PreProcessorDirectiveType type, string argument) |
|
{ |
|
// pre-processor directive must start on its own line |
|
if (!isAtStartOfLine) |
|
NewLine(); |
|
WriteIndentation(); |
|
textWriter.Write('#'); |
|
string directive = type.ToString().ToLowerInvariant(); |
|
textWriter.Write(directive); |
|
column += 1 + directive.Length; |
|
if (!string.IsNullOrEmpty(argument)) { |
|
textWriter.Write(' '); |
|
textWriter.Write(argument); |
|
column += 1 + argument.Length; |
|
} |
|
NewLine(); |
|
} |
|
|
|
public static string PrintPrimitiveValue(object value) |
|
{ |
|
TextWriter writer = new StringWriter(); |
|
TextWriterTokenWriter tokenWriter = new TextWriterTokenWriter(writer); |
|
tokenWriter.WritePrimitiveValue(value); |
|
return writer.ToString(); |
|
} |
|
|
|
public override void WritePrimitiveValue(object value, string literalValue = null) |
|
{ |
|
if (literalValue != null) { |
|
textWriter.Write(literalValue); |
|
column += literalValue.Length; |
|
return; |
|
} |
|
|
|
if (value == null) { |
|
// usually NullReferenceExpression should be used for this, but we'll handle it anyways |
|
textWriter.Write("null"); |
|
column += 4; |
|
return; |
|
} |
|
|
|
if (value is bool) { |
|
if ((bool)value) { |
|
textWriter.Write("true"); |
|
column += 4; |
|
} else { |
|
textWriter.Write("false"); |
|
column += 5; |
|
} |
|
return; |
|
} |
|
|
|
if (value is string) { |
|
string tmp = "\"" + ConvertString(value.ToString()) + "\""; |
|
column += tmp.Length; |
|
textWriter.Write(tmp); |
|
} else if (value is char) { |
|
string tmp = "'" + ConvertCharLiteral((char)value) + "'"; |
|
column += tmp.Length; |
|
textWriter.Write(tmp); |
|
} else if (value is decimal) { |
|
string str = ((decimal)value).ToString(NumberFormatInfo.InvariantInfo) + "m"; |
|
column += str.Length; |
|
textWriter.Write(str); |
|
} else if (value is float) { |
|
float f = (float)value; |
|
if (float.IsInfinity(f) || float.IsNaN(f)) { |
|
// Strictly speaking, these aren't PrimitiveExpressions; |
|
// but we still support writing these to make life easier for code generators. |
|
textWriter.Write("float"); |
|
column += 5; |
|
WriteToken(Roles.Dot, "."); |
|
if (float.IsPositiveInfinity(f)) { |
|
textWriter.Write("PositiveInfinity"); |
|
column += "PositiveInfinity".Length; |
|
} else if (float.IsNegativeInfinity(f)) { |
|
textWriter.Write("NegativeInfinity"); |
|
column += "NegativeInfinity".Length; |
|
} else { |
|
textWriter.Write("NaN"); |
|
column += 3; |
|
} |
|
return; |
|
} |
|
if (f == 0 && 1 / f == float.NegativeInfinity) { |
|
// negative zero is a special case |
|
// (again, not a primitive expression, but it's better to handle |
|
// the special case here than to do it in all code generators) |
|
textWriter.Write("-"); |
|
column++; |
|
} |
|
var str = f.ToString("R", NumberFormatInfo.InvariantInfo) + "f"; |
|
column += str.Length; |
|
textWriter.Write(str); |
|
} else if (value is double) { |
|
double f = (double)value; |
|
if (double.IsInfinity(f) || double.IsNaN(f)) { |
|
// Strictly speaking, these aren't PrimitiveExpressions; |
|
// but we still support writing these to make life easier for code generators. |
|
textWriter.Write("double"); |
|
column += 6; |
|
WriteToken(Roles.Dot, "."); |
|
if (double.IsPositiveInfinity(f)) { |
|
textWriter.Write("PositiveInfinity"); |
|
column += "PositiveInfinity".Length; |
|
} else if (double.IsNegativeInfinity(f)) { |
|
textWriter.Write("NegativeInfinity"); |
|
column += "NegativeInfinity".Length; |
|
} else { |
|
textWriter.Write("NaN"); |
|
column += 3; |
|
} |
|
return; |
|
} |
|
if (f == 0 && 1 / f == double.NegativeInfinity) { |
|
// negative zero is a special case |
|
// (again, not a primitive expression, but it's better to handle |
|
// the special case here than to do it in all code generators) |
|
textWriter.Write("-"); |
|
} |
|
string number = f.ToString("R", NumberFormatInfo.InvariantInfo); |
|
if (number.IndexOf('.') < 0 && number.IndexOf('E') < 0) { |
|
number += ".0"; |
|
} |
|
textWriter.Write(number); |
|
} else if (value is IFormattable) { |
|
StringBuilder b = new StringBuilder (); |
|
// if (primitiveExpression.LiteralFormat == LiteralFormat.HexadecimalNumber) { |
|
// b.Append("0x"); |
|
// b.Append(((IFormattable)val).ToString("x", NumberFormatInfo.InvariantInfo)); |
|
// } else { |
|
b.Append(((IFormattable)value).ToString(null, NumberFormatInfo.InvariantInfo)); |
|
// } |
|
if (value is uint || value is ulong) { |
|
b.Append("u"); |
|
} |
|
if (value is long || value is ulong) { |
|
b.Append("L"); |
|
} |
|
textWriter.Write(b.ToString()); |
|
column += b.Length; |
|
} else { |
|
textWriter.Write(value.ToString()); |
|
column += value.ToString().Length; |
|
} |
|
} |
|
|
|
/// <summary> |
|
/// Gets the escape sequence for the specified character within a char literal. |
|
/// Does not include the single quotes surrounding the char literal. |
|
/// </summary> |
|
public static string ConvertCharLiteral(char ch) |
|
{ |
|
if (ch == '\'') { |
|
return "\\'"; |
|
} |
|
return ConvertChar(ch); |
|
} |
|
|
|
/// <summary> |
|
/// Gets the escape sequence for the specified character. |
|
/// </summary> |
|
/// <remarks>This method does not convert ' or ".</remarks> |
|
static string ConvertChar(char ch) |
|
{ |
|
switch (ch) { |
|
case '\\': |
|
return "\\\\"; |
|
case '\0': |
|
return "\\0"; |
|
case '\a': |
|
return "\\a"; |
|
case '\b': |
|
return "\\b"; |
|
case '\f': |
|
return "\\f"; |
|
case '\n': |
|
return "\\n"; |
|
case '\r': |
|
return "\\r"; |
|
case '\t': |
|
return "\\t"; |
|
case '\v': |
|
return "\\v"; |
|
case ' ': |
|
case '_': |
|
case '`': |
|
case '^': |
|
// ASCII characters we allow directly in the output even though we don't use |
|
// other Unicode characters of the same category. |
|
return ch.ToString(); |
|
default: |
|
switch (char.GetUnicodeCategory(ch)) { |
|
case UnicodeCategory.ModifierLetter: |
|
case UnicodeCategory.NonSpacingMark: |
|
case UnicodeCategory.SpacingCombiningMark: |
|
case UnicodeCategory.EnclosingMark: |
|
case UnicodeCategory.LineSeparator: |
|
case UnicodeCategory.ParagraphSeparator: |
|
case UnicodeCategory.Control: |
|
case UnicodeCategory.Format: |
|
case UnicodeCategory.Surrogate: |
|
case UnicodeCategory.PrivateUse: |
|
case UnicodeCategory.ConnectorPunctuation: |
|
case UnicodeCategory.ModifierSymbol: |
|
case UnicodeCategory.OtherNotAssigned: |
|
case UnicodeCategory.SpaceSeparator: |
|
return "\\u" + ((int)ch).ToString("x4"); |
|
default: |
|
return ch.ToString(); |
|
} |
|
} |
|
} |
|
|
|
/// <summary> |
|
/// Converts special characters to escape sequences within the given string. |
|
/// </summary> |
|
public static string ConvertString(string str) |
|
{ |
|
StringBuilder sb = new StringBuilder (); |
|
foreach (char ch in str) { |
|
if (ch == '"') { |
|
sb.Append("\\\""); |
|
} else { |
|
sb.Append(ConvertChar(ch)); |
|
} |
|
} |
|
return sb.ToString(); |
|
} |
|
|
|
public override void WritePrimitiveType(string type) |
|
{ |
|
textWriter.Write(type); |
|
column += type.Length; |
|
if (type == "new") { |
|
textWriter.Write("()"); |
|
column += 2; |
|
} |
|
} |
|
|
|
public override void StartNode(AstNode node) |
|
{ |
|
// Write out the indentation, so that overrides of this method |
|
// can rely use the current output length to identify the position of the node |
|
// in the output. |
|
WriteIndentation(); |
|
} |
|
|
|
public override void EndNode(AstNode node) |
|
{ |
|
} |
|
} |
|
}
|
|
|