Как отобразить несколько ошибок проверки с помощью @Html.ValidationMessageFor?

Я вижу, что мой ModelState.Values[x].Errors правильно заполнен двумя ошибками проверки для одного свойства.

Если я использую @Html.ValidationSummary() в своем представлении, он корректно отображает обе ошибки.... хотя и в верхней части страницы, а не рядом с нарушительным входом.

С помощью @Html.ValidationMessageFor(model => model.MyProperty) отображается только первая ошибка для этого свойства!

Как показать обе ошибки рядом с соответствующим вводом?

Ответ 1

Одним из решений является реализация собственного метода расширения на HtmlHelper, который делает что-то отличное от поведения ValidationMessageFor по умолчанию. Метод sample @Html.ValidationMessagesFor, приведенный ниже, объединит несколько сообщений об ошибках, которые были добавлены в ModelState во время проверки на стороне сервера (только).

public static MvcHtmlString ValidationMessagesFor<TModel, TProperty>(this HtmlHelper<TModel> htmlHelper, Expression<Func<TModel, TProperty>> expression, object htmlAttributes = null)
{
    var propertyName = ModelMetadata.FromLambdaExpression(expression, htmlHelper.ViewData).PropertyName;
    var modelState = htmlHelper.ViewData.ModelState;

    // If we have multiple (server-side) validation errors, collect and present them.
    if (modelState.ContainsKey(propertyName) && modelState[propertyName].Errors.Count > 1)
    {
        var msgs = new StringBuilder();
        foreach (ModelError error in modelState[propertyName].Errors)
        {
            msgs.AppendLine(error.ErrorMessage);
        }

        // Return standard ValidationMessageFor, overriding the message with our concatenated list of messages.
        return htmlHelper.ValidationMessageFor(expression, msgs.ToString(), htmlAttributes as IDictionary<string, object> ?? htmlAttributes);
    }

    // Revert to default behaviour.
    return htmlHelper.ValidationMessageFor(expression, null, htmlAttributes as IDictionary<string, object> ?? htmlAttributes);
}

Это полезно, если у вас есть пользовательская проверка бизнеса, которую вы применяете в коллекции, принадлежащей вашей модели (например, общие контрольные суммы), или проверяя модель в целом.

Пример использования этого, где коллекция Order.LineItems проверяется на стороне сервера:

@using MyHtmlHelperExtensions    // namespace containing ValidationMessagesFor
@using (Html.BeginForm())
{
    @Html.LabelFor(m => m.LineItems)
    <ul>
        @Html.EditorFor(m => m.LineItems)
    </ul>
    @Html.ValidationMessagesFor(m => m.LineItems)
}

Ответ 2

Ответ, предоставленный Dave A-W, по сути, является правильным, но не будет работать во всех случаях. Чтобы получить правильное имя свойства, исходный код MVC 3 использует:

var propertyName = ExpressionHelper.GetExpressionText(expression);

Это касается любых префиксов в данных формы