Как получить совпадающие имена ключей ModelState / свойств модели?

#asp.net-mvc #validation #data-annotations

#asp.net-mvc #проверка #данные-аннотации

Вопрос:

У меня есть приложение MVC3, я использую классы EF4.1 Code First и POCO.

Мой вопрос заключается в том, чтобы избавиться от жестко закодированных «propertyNames» в моем классе POCO, а также предварительно зафиксировать его соответствующими значениями в 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. Как мне избежать жестко закодированных строк имени свойства «Дата истечения срока действия»?

  2. Из-за ViewModel ожидается, что ключ ModelState будет «BusinessObject.ExpiryDate». Как мне добиться соответствия моих свойств? Как мне добавить префикс или удалить автоматически добавляемый префикс?

  3. Обратите внимание, что в настоящее время я использую ValidationResult.Имена участников.Сначала()). В моем случае мне нужно только одно сообщение в сводке проверки, но я хотел бы, чтобы были выделены ОБА свойства. т. Е. только одно сообщение для «.validation-summary-errors», но оба поля выделены для элементов «.input-validation-error» и «.field-validation-error» в DOM.

Спасибо,

Ответ №1:

Жесткое кодирование имени свойства в атрибутах неплохо, за исключением функций и т.д.Я предлагаю вам реализовать новый класс, унаследованный от CompareAttribute , так что большая часть приведенного выше кода исчезнет, и ваша проблема с жестким кодированием будет решена.

Для выделения проблемы вы можете использовать java script или реализовать некоторую функцию 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>
  

Комментарии:

1. спасибо за информацию… Мне нравится решение с подсветкой. Мне нужно взглянуть на «CompareAttribute», чтобы посмотреть, смогу ли я заставить это работать.

2. Я могу предложить другое решение выделения, поскольку моя проверка выполнялась на стороне сервера с использованием вызова ajax, возвращаемый результат JSON содержит информацию о недопустимом элементе html и сообщение о проверке. Я могу настроить выделение с помощью jquery.