【问题标题】:How can I wrap a SyntaxTree or UnitCompilationRoot in a region or comment using Roslyn?如何使用 Roslyn 将 SyntaxTree 或 UnitCompilationRoot 包装在区域或注释中?
【发布时间】:2020-02-28 22:21:06
【问题描述】:
我正在使用CSharpSyntaxTree.ParseText 从任意代码块创建语法树;我希望能够将SyntaxTree 或其UnitCompilationRoot 包装或封装在一个区域中。我知道 Quoter 网站允许您查看如何使用 Roslyn API 编写任意代码,但是它将代码分解为其原始组件的方式对添加区域的用例没有帮助到SyntaxTree 或UnitCompilationRoot。
使用任意代码的示例
// Create syntax tree from template
var str = @"public class Foo { }"
var syntaxTree = CSharpSyntaxTree.ParseText(str);
// Add Region
// ????
【问题讨论】:
标签:
c#
code-generation
roslyn
region
【解决方案1】:
区域由 Roslyn 表示为特殊类型的 SyntaxTrivia。
因此,为了将代码块包装到区域中,您可能会获得目标 SyntaxNode 并将其前导和尾随琐事替换为区域琐事。
以下代码演示了如何做到这一点:
public static T AddRegion<T>(this T node, string regionName) where T : SyntaxNode
{
return node
// inserts #region RegionName before node
.WithLeadingTrivia(node.GetLeadingTrivia().Insert(0, GetRegionLeadingTrivia(regionName)))
// inserts #endregion after node
.WithTrailingTrivia(node.GetTrailingTrivia().Add(GetRegionTrailingTrivia()));
}
public static SyntaxTrivia GetRegionLeadingTrivia(string regionName)
{
return SyntaxFactory.Trivia(
SyntaxFactory
.RegionDirectiveTrivia(true)
.WithEndOfDirectiveToken(
SyntaxFactory.Token(
SyntaxFactory.TriviaList(SyntaxFactory.PreprocessingMessage(regionName)),
SyntaxKind.EndOfDirectiveToken,
SyntaxFactory.TriviaList())));
}
public static SyntaxTrivia GetRegionTrailingTrivia()
{
return SyntaxFactory.Trivia(SyntaxFactory.EndRegionDirectiveTrivia(true));
}
要执行更复杂的操作,您可以使用CSharpSyntaxRewriter class。可以在here 找到使用该类的一些示例。