【问题标题】:How to change attributes with Roslyn如何使用 Roslyn 更改属性
【发布时间】:2017-07-10 09:11:49
【问题描述】:

我很难在 roslyn 库上找到正确的方法来更改属性并重写属性:

class Program
{ 
    static void Main(string[] args)
    {

        //this is the original class 
        var str = @"public class OfferModel
        {
            public int Id { get; set; }

            [Required(ErrorMessage = \""Required\"")]
            [Display(Name = \""Date of Offer *\"")]
            public DateTime? OfferDate { get; set; }

            [Required(ErrorMessage = \""Required\"")]
            [Display(Name = \""Amount *\"")]
            [Range(1, double.MaxValue, ErrorMessage = \""Invalid Amount\"")]
            public decimal? Amount { get; set; }
        }";

        var tree = CSharpSyntaxTree.ParseText(str);

        var myWriter = new MyReWriter();
        var newRoot = myWriter.Visit(tree.GetRoot());

        Console.WriteLine(newRoot.ToFullString());

    }
}

public class MyReWriter : CSharpSyntaxRewriter
{
    public MyReWriter() : base() { }

    public override SyntaxNode VisitAttribute(AttributeSyntax node)
    {
        var newNode = node;

        if (node.Name.ToString() == "Required")
        {
            newNode = PrepareNewRequiredArgument(node);
        }

        return base.VisitAttribute(newNode);
    }

    private AttributeSyntax PrepareNewRequiredArgument(AttributeSyntax node)
    {
        var newNode = node;
        //ErrorMessageResourceType
        //ErrorMessageResourceName
        var argResType = node.ArgumentList.Arguments.FirstOrDefault(aa => aa.NameEquals.Name.Identifier.Text == "ErrorMessageResourceType");
        var argResName = node.ArgumentList.Arguments.FirstOrDefault(aa => aa.NameEquals.Name.Identifier.Text == "ErrorMessageResourceName");
        if (argResType != null && argResName != null)
        {
            //already exists, don't do anything
            return newNode;
        }

        var argErrorMessage = node.ArgumentList.Arguments.FirstOrDefault(aa => aa.NameEquals.Name.Identifier.Text == "ErrorMessage");

        if (argErrorMessage != null)
        {
            var name = ParseName("Required");
            var tokenValue = ((LiteralExpressionSyntax)argErrorMessage.Expression).Token.Value;

            var argErrorMessageResourceName = AttributeArgument(
                LiteralExpression(SyntaxKind.StringLiteralExpression,
                    Token(default(SyntaxTriviaList),
                            SyntaxKind.StringLiteralToken, "ErrorMessageResourceName", tokenValue.ToString(), default(SyntaxTriviaList))));

            //var argErrorMessageResourceType = AttributeArgument(
            //   LiteralExpression(SyntaxKind.NumericLiteralExpression, 
            //        Token(default(SyntaxTriviaList),
            //                SyntaxKind.NumericLiteralToken, "ErrorMessageResourceType", "typeof(WebModels)", default(SyntaxTriviaList))));

            var otherList = new SeparatedSyntaxList<AttributeArgumentSyntax>();
            otherList = otherList.AddRange(new[] { argErrorMessageResourceName });
            //otherList = otherList.AddRange(new[] { argErrorMessageResourceName, argErrorMessageResourceType });
            var argumentList = AttributeArgumentList(otherList);
            var newAttribute = Attribute(name, argumentList);

            newNode = node.ReplaceNode(node, newAttribute);
        }
        return newNode;
    }
}

我需要的输出必须如下所示:

public class OfferModel
{
    public int Id { get; set; }

    [Required(ErrorMessageResourceType = typeof(WebModels), ErrorMessageResourceName = \""Required\"")]
    [Display(Name = \""Date of Offer *\"")]
    public DateTime? OfferDate { get; set; }

    [Required(ErrorMessageResourceType = typeof(WebModels), ErrorMessageResourceName = "Required")]
    [Display(ResourceType = typeof(WebModels), Name = "Amount *")]
    [Range(1, double.MaxValue, ErrorMessageResourceName = "InvalidAmount")]
    public decimal? Amount { get; set; }
}

这同样适用于 [Display] / [Range] 属性。如果我能让 [Required] 工作,我可以将其应用于 [Display] / [Range]

【问题讨论】:

  • 您当前的代码究竟是如何失败的?如果你调试你的代码,它的行为在什么时候会变成你所期望的?
  • 以上代码生成属性如下: [Required(ErrorMessageResourceName)] 应该是:[Required(ErrorMessageResourceType = typeof(WebModels), ErrorMessageResourceName = "Required")]

标签: c# roslyn custom-attributes


【解决方案1】:

你的最后一行是:

newNode = node.ReplaceNode(node, newAttribute);

您似乎只是想用不同的实例(newAttribute)替换子树(节点)的根。 ReplaceNode 在根节点上不起作用。相反,您应该只使用新节点。

应该是:

newNode = newAttribute;

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2013-02-20
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多