在我的视图中,我正在使整个 html 表可编辑。表中的每个字段都有验证。这是一些示例代码...
<tbody>
foreach (Item item in Model.Items)
{
<tr>
<td>
@Html.TextBoxFor(x => em.Value)
@Html.ValidationMessageFor(x => em.Value)
</td>
</tr>
}
</tbody>
这在浏览器上产生了两个问题。
1)由于所有控件的name属性相同,当一个字段无效时,所有同名字段都会显示错误信息。
2)可能也和名字有关,在验证表单时,如果第一行的控件无效,则表单无效。但是所有其他行中的控件不会使表单无效(即使显示了错误消息)。
if (!$(form).valid()) {
return false;
}
任何想法都会有所帮助。
更新:解决方案
根据 bobek 的建议,我为 TextBoxInLoopFor 和 ValidationMessageInLoop 创建了自定义扩展。以下是解决方案:
public static MvcHtmlString TextBoxInLoopFor<TModel, IItem, TValue>(this HtmlHelper<TModel> html, Expression<Func<TModel, TValue>> expression, IItem item, object htmlAttributes)
{
var model = item as IInLoopForModel;
if (model == null)
return MvcHtmlString.Empty;
var metadata = ModelMetadata.FromLambdaExpression(expression, html.ViewData);
var fieldName = ExpressionHelper.GetExpressionText(expression);
var name = model.Name + fieldName;
var tag = new TagBuilder("input");
tag.MergeAttributes(new RouteValueDictionary(htmlAttributes));
tag.Attributes.Add("id", name);
tag.Attributes.Add("for", name);
tag.Attributes.Add("name", name);
tag.Attributes.Add("value", metadata.Model.ToString());
// Add the validation attributes
ModelState modelState;
if (html.ViewData.ModelState.TryGetValue(name, out modelState))
{
if (modelState.Errors.Count > 0)
tag.AddCssClass(HtmlHelper.ValidationInputCssClassName);
}
tag.MergeAttributes(html.GetUnobtrusiveValidationAttributes(name, metadata));
return MvcHtmlString.Create(tag.ToString(TagRenderMode.Normal));
}
public static MvcHtmlString ValidationMessageInLoopFor<TModel, IItem, TValue>(this HtmlHelper<TModel> html, Expression<Func<TModel, TValue>> expression, IItem item)
{
var model = item as IInLoopForModel;
if (model == null)
return MvcHtmlString.Empty;
var fieldName = ExpressionHelper.GetExpressionText(expression);
var name = model.Name + fieldName;
var tag = new TagBuilder("span");
tag.Attributes.Add("class", "field-validation-valid");
tag.Attributes.Add("data-valmsg-replace", "true");
tag.Attributes.Add("data-valmsg-for", name);
return MvcHtmlString.Create(tag.ToString(TagRenderMode.Normal));
}