在运行时将属性添加到对象的实例

时间:2015-02-12 14:11:18

标签: c# razor model-view-controller data-annotations custom-attributes

我正在尝试使用来自数据库的问题生成动态表单。 每个问题都有其类型和其他属性。

我的课程示例:

    public class QuestionnaireBase
    {
        public string Text { get; set; }
        public int Sequence { get; set; }
    }       

    public abstract class Question : QuestionnaireBase
    {
        public bool IsRequired { get; set; }
        public bool WithComment { get; set; }
        public abstract object Answer { get; set; }
    }

    public class TextQuestion : Question
    {
        public string DefaultText { get; set; }
        public int MaxLength { get; set; }
        public override object Answer { get; set; }
    }

我想在Answer字段中添加一些属性(DisplayAttribute,MaxLenght),所以当我使用EditorFor和LabelFor时,会考虑这些属性。

在检索我的问题时,我尝试在我的字段“回答”(这是一个存根)上添加一个属性:

Enumerable.Range(1, 5).Select(questionSeq                                               => new TextQuestion {
Text = string.Format("Question {0}" questionSeq), 
Answer = "TEXTVALUE" + questionSeq
}).Select(w => {
   var skd = new DisplayAttribute();
   skd.Name =  w.Text; 
   TypeDescriptor.AddAttributes(w.Answer,skd );                                                 
   return w;
   })

现在,在我的视图中,我想使用LabelFor来显示此DisplayAttribute:

 @Html.LabelFor(model => model.Questions[questionIndex].Answer)

这将输出'Answer'作为文本。 我可以通过这样做来绕过这个问题:

   @{
     var attribute = 
TypeDescriptor.GetAttributes(Model.Questions[questionIndex].Answer)[typeof(DisplayAttribute)];

    var displayAttribute = ((DisplayAttribute) attribute);                          
    }
    @Html.LabelFor(model => 
model.Questions[questionIndex].Answer, displayAttribute.Name)

我的第一个猜测是,LabelFor会在我的类型上使用DisplayAttribute,而不是在我的实例上。

显然,我不想为每个属性做这项工作,或者在运行时创建属性是完全没用的。

我该怎么做才能解决这个问题? 我想对MaxLenghtAttribute / Range做同样的事情。 谢谢你的时间

1 个答案:

答案 0 :(得分:1)

我认为你无法完全实现这一点,因为属性是静态元数据。

Can attributes be added dynamically in C#?

可能的解决方法是创建自己的自定义属性,该属性实现所有可能的验证(MaxLength,Range等),并在对象上附加一些触发器以启用/禁用它们。这将处理验证,但不处理您的问题中提到的特定实例(Html.LabelFor + DisplayName)。为此,您可以创建自己的扩展方法,以与将其添加到实例相同的方式访问DisplayName属性。