3

我正在使用 aSyntaxRewriter将类从旧库转换为新库,这基本上需要查找具有给定属性的类,然后重写遵循特定约定的属性。重写器的一般骨架如下:

class PropertyConverter : SyntaxRewriter
{
    public override SyntaxNode VisitPropertyDeclaration(PropertyDeclarationSyntax node)
    {
        if (!MeetsUpdateCriteria(node)) return base.VisitPropertyDeclaration(node);

        // these implementations simply return a string
        var name = FigureOutName(node.Identifier);
        var propertyType = FigureOutType(node.Type);

        var getter = Syntax.ParseExpression("this.GetValue<" + propertyType + ">(" + name + ")");
        var setter = Syntax.ParseExpression("this.SetValue(" + name + ", value)");

        return node.WithType(propertyType)
                   .WithAccessorList(
            Syntax.AccessorList(Syntax.List(
                Syntax.AccessorDeclaration(
                    SyntaxKind.GetAccessorDeclaration,
                    Syntax.Block(Syntax.ReturnStatement(getter))),
                Syntax.AccessorDeclaration(
                    SyntaxKind.SetAccessorDeclaration,
                    Syntax.Block(Syntax.ExpressionStatement(setter)))))));
    }
}

这个转换器的结果是一个具有更新属性的类,它在以下代码中输出:

// IDocument csfile <- from a project in a Workspace
var tree = csfile.GetSyntaxTree();
var root = new PropertyConverter().Visit((SyntaxNode)tree.GetRoot())
                                  .NormalizeWhitespace(); // problem!

File.WriteAllText(Path.GetFileName(csfile.FilePath), root.ToFullString());    

此时代码在语法上都是正确的,并且输出的语法树是正确的。我唯一的抱怨是 XML 文档注释周围的空格根本不正确:

/// <summary>
        /// Gets or sets the thickness (TH).
        /// </summary>
public float Thickness
{
    get
    {
        return this.GetValue<float>(TH);
    }

    set
    {
        this.SetValue(TH, value);
    }
}

注意所有多余的缩进。此外,间距也以其他方式损坏,尤其是在方法文档中:

/// <summary>
        /// Initializes a new instance of the <see cref = "X"/> class.
        /// </summary>
        /// <param name = "innerRadius">Inner radius of the X.</param>
        /// <param name = "thickness">Thickness of the X.</param>

我已经验证了输入树没有遇到这些缩进问题,并且我还验证了该树在调用之前没有遇到这些缩进问题NormalizeWhitespace。我也试过elasticTrivia: true了,也没有任何运气。

那么如何让 Roslyn 以一致的方式规范化空白呢?

4

1 回答 1

3

我认为这可能是 Roslyn 中的一个错误。但是,一般来说,我建议使用Format包含在 Roslyn.Services.dll 中的 SyntaxNode 上的扩展方法(添加一个using Roslyn.Services;)。

NormalizeWhitespace是一个非常蛮力的系统,其设计主要是为了确保代码可以往返。Roslyn.Services 层中存在的格式化代码更灵活一些,并且包含了 Visual StudioFormat Document命令的许多行为。

于 2013-04-02T16:46:25.870 回答