Html.TextBoxFor / Html.TextAreaFor和Html.EditorFor(与[DataType(DataType.MultilineText)])之间的客户端验证差异

时间:2011-06-21 13:56:55

标签: c# asp.net-mvc-3 razor client-side-validation unobtrusive-validation

我在ASP.NET MVC 3 / Razor中使用客户端验证(不引人注意)我通过使用Html.EditorFor并指定DataType.MultilineText让它在<textarea>上工作,但不应该使用Html。 TextAreaFor还有客户端验证吗?

[Required(ErrorMessage = "Foo")]
public string Message { get; set; }

// Does add client-side validation
@Html.TextBoxFor(m => m.Message)

// Does NOT add client-side validation
@Html.TextAreaFor(m => m.Message)

[Required(ErrorMessage = "Foo")]
[DataType(DataType.MultilineText)]
public string Message { get; set; }

// Does add client-side validation (and multiline)
@Html.EditorFor(m => m.Message)

// Does NOT add client-side validation
@Html.TextAreaFor(m => m.Message)

<add key="ClientValidationEnabled" value="true" />
<add key="UnobtrusiveJavaScriptEnabled" value="true" />

2 个答案:

答案 0 :(得分:8)

上面的示例对我来说很好。我想知道 - 这是一个精确的例子,还是从现实世界的问题中简化了?在使用具有嵌套属性的模型时,我发现了这种确切的行为。

因此,例如,如果我将模型更改为如下所示:

public class MyModelObject
{
    [Required(ErrorMessage = "Foo")]
    [DataType(DataType.MultilineText)]
    public string Message { get; set; }
}

public class MyModel
{
    public MyModelObject MyObject { get; set; }
}

然后我重现你提到的确切问题。

@Html.EditorFor(x => x.MyObject.Message)

按预期生成jquery验证属性:

<textarea class="text-box multi-line input-validation-error" data-val="true" data-val-required="Foo" id="MyObject_Message" name="MyObject.Message"></textarea>

但是,这个:

@Html.TextAreaFor(x => x.MyObject.Message)

不:

<textarea cols="20" id="MyObject_Message" name="MyObject.Message" rows="2"></textarea>

如果这实际上描述了您的问题,看起来这已被报告为错误: http://aspnet.codeplex.com/workitem/8576

答案 1 :(得分:0)

正如 obliojoe 在他的回答中所建议的,当表达式比简单属性引用更复杂时会出现此错误。

我已经实现了解决此问题的替换助手。诀窍是获取正确的模型元数据,将其传递给HtmlHelper.GetUnobtrusiveValidationAttributes(),然后将收到的属性传递给原始的TextAreaFor()帮助器:

public static MvcHtmlString TextAreaWithValidationFor<TModel, TProperty>(this HtmlHelper<TModel> helper, Expression<Func<TModel, TProperty>> expression)
{
    var modelMetadata = ModelMetadata.FromLambdaExpression(expression, helper.ViewData);
    var name = ExpressionHelper.GetExpressionText(expression);
    IDictionary<string, object> validationAttributes = helper.GetUnobtrusiveValidationAttributes(name, modelMetadata);
    return TextAreaExtensions.TextAreaFor(helper, expression, validationAttributes);
}

我在bug reported on CodePlex附加了一个更加开发的版本(提供此代码的所有TextAreaFor重载,包括htmlAttributes}重载。