如何使ModelState键/模型属性名称匹配?

时间:2011-05-19 23:14:53

标签: asp.net-mvc validation data-annotations

我有一个MVC3应用程序,我正在使用EF4.1 Code First和POCO类。

我的问题涉及在我的POCO类中删除硬编码的“PropertyNames”,并使用ModelState中的正确值预先修复它,以便验证消息正确显示。

我的POCO课程中有一个看起来像这样的方法。 您会注意到我也在使用“DataAnnotations.Validator”代码来重用数据注释验证。

public class TaxCode : ModelBusinessObjectBase
{
    //...

    [Required]
    [DataType(DataType.Date)]
    [DisplayName("Effective Date")]        
    public DateTime EffectiveDate
    {
        get { return _effectiveDate; }
        set { _effectiveDate = value; }
    }    

    [Required]
    [DataType(DataType.Date)]
    [DisplayName("Expiry Date")]        
    public DateTime ExpiryDate
    {
        get { return _expiryDate; }
        set { _expiryDate = value; }
    }    

    //...

    public override IEnumerable<ValidationResult> GetValidationResults()
    {
        //---- data annotation validation ----
        ValidationContext validationContext = new ValidationContext(this, null, null);
        IList<ValidationResult> dataAnnotationValidationResults = new List<ValidationResult>();
        bool isValid = Validator.TryValidateObject(this, validationContext, dataAnnotationValidationResults, true);


        foreach (ValidationResult dataAnnotationValidationResult in dataAnnotationValidationResults)
            yield return new ValidationResult(dataAnnotationValidationResult.ErrorMessage, dataAnnotationValidationResult.MemberNames);

        //---- custom business rule validation ----
        // expiry date must be greater than effective date
        if (ExpiryDate <= EffectiveDate)
        {   
            yield return new ValidationResult("Expiry Date must be after Effective Date", new [] {"EffectiveDate", "ExpiryDate"});
        }

        yield break;
    }

在我的服务层,我最终会调用这样的东西:

    public bool TryValidate(TaxCode domainObject)
    {
        if (!domainObject.IsValid) 
        {   
            _validationDictionary.AddValidationResults(domainObject.GetValidationResults());
            isValid = false;
        }
        return isValid;
    }

使用“ModelStateWrapper”,代码执行此操作:

    public virtual void AddValidationResults(IEnumerable<ValidationResult> validationResults)
    {

        foreach (ValidationResult validationResult in validationResults)
        {
            _modelState.AddModelError(validationResult.MemberNames.First(), validationResult.ErrorMessage);
        }
    }

我的观点模型是:

public class TaxCodeViewModel : IPersistantBusinessObjectViewModel<TaxCode>
{
    public TaxCodeViewModel()
    {
    }

    public TaxCodeViewModel(TaxCode domainObj)
    {
        this.BusinessObject = domainObj;
    }
}

我的视图看起来有点像这样:

   <%@ Control Language="C#" AutoEventWireup="true"               Inherits="System.Web.Mvc.ViewUserControl<TaxCodeViewModel>" %>

   //...

    <% using (Html.BeginForm()) {%>

        <%:Html.ValidationSummary(false, "Unable to save. Please correct the errors and try again.")%>


        //...

        <div class="editor-label">
            <%:Html.LabelFor(model => model.BusinessObject.EffectiveDate)%>
        </div>
        <div class="editor-field">
            <%: Html.EditorFor(model => model.BusinessObject.EffectiveDate)%>
            <%: Html.ValidationMessageFor(model => model.BusinessObject.EffectiveDate)%>
        </div>
        <br />  
        <div class="editor-label">
            <%: Html.LabelFor(model => model.BusinessObject.ExpiryDate)%>
        </div>
        <div class="editor-field">
            <%: Html.EditorFor(model => model.BusinessObject.ExpiryDate)%>
            <%: Html.ValidationMessageFor(model => model.BusinessObject.ExpiryDate)%>
        </div>
        <br />  
        //...

我的问题是:

  1. 如何避免硬编码属性名称字符串“Expiry Date”?

  2. 由于ViewModel,期望ModelState键为“BusinessObject.ExpiryDate”。我如何让我的财产匹配?如何添加前缀或删除自动添加的前缀?

  3. 注意我目前使用validationResult.MemberNames.First())。 在我的情况下,我只想在验证摘要中有一条消息,但我希望突出显示BOTH属性。即只有一条消息为“.validation-summary-errors”,但两个字段都突出显示为DOM中的“.input-validation-error”和“.field-validation-error”元素。

  4. 谢谢,

1 个答案:

答案 0 :(得分:0)

属性中的硬编码属性名称也不错,除了函数等。我建议您实现从CompareAttribute继承的新类,因此上述大部分代码都将消失,您的硬编码依赖性问题将得到解决。

对于突出显示问题,你可以使用java脚本,或者实现一些utily函数来检查模型是否在某些字段上有错误并返回一些css类名。像休息一样。

 <div class="editor-field <%: Html.HasValidationMessage(model => model.BusinessObject.ExpiryDate)%>">
            <%: Html.EditorFor(model => model.BusinessObject.ExpiryDate)%>
            <%: Html.ValidationMessageFor(model => model.BusinessObject.ExpiryDate)%>
 </div>