#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 />
//...
Мой вопрос:
-
Как мне избежать жестко закодированных строк имени свойства «Дата истечения срока действия»?
-
Из-за ViewModel ожидается, что ключ ModelState будет «BusinessObject.ExpiryDate». Как мне добиться соответствия моих свойств? Как мне добавить префикс или удалить автоматически добавляемый префикс?
-
Обратите внимание, что в настоящее время я использую 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.