加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 编程开发 > asp.Net > 正文

asp.net-mvc-3 – 使用ViewModel时不显示MVC3自定义验证错误消息

发布时间:2020-12-16 09:51:37 所属栏目:asp.Net 来源:网络整理
导读:摘要 问题:使用ViewModel时,为什么不显示自定义验证错误消息. 答:自定义验证应该应用于ViewModel而不是Class.有关示例代码,请参阅@ JaySilk84的答案结尾. MVC3,项目使用 jquery-1.7.2.min.js modernizr-2.5.3.js jquery-ui-1.8.22.custom.min.js(由jQuery.
摘要

问题:使用ViewModel时,为什么不显示自定义验证错误消息.

答:自定义验证应该应用于ViewModel而不是Class.有关示例代码,请参阅@ JaySilk84的答案结尾.

MVC3,项目使用

> jquery-1.7.2.min.js
> modernizr-2.5.3.js
> jquery-ui-1.8.22.custom.min.js(由jQuery.com为Accordion插件生成)
> jquery.validate.min.js和
> jquery.validate.unobtrusive.min.js

我在我的项目中验证了View中的数据注释和Controller中的ModelState.AddModelError,因此我知道我已经正确配置了所有验证代码.

但是使用自定义验证时,代码中会生成错误,但不会显示错误消息.

public IEnumerable<ValidationResult> Validate(ValidationContext validationContext)
{ if (DOB > DateTime.Now.AddYears(-18))
  { yield return new ValidationResult("Must be 18 or over."); }      }

在POST操作中向下钻取,自定义验证会导致模型状态失败,并且错误消息被放置在正确的值字段中,但是当模型被发送回视图时,错误消息不会显示.在控制器中,我还有ModelState.AddModelError代码,并显示其消息.如何处理一个人会工作而不是另一个人?如果没有,还有什么可以阻止显示错误消息?

更新1:

我正在使用ViewModel在视图中创建模型.我删除了ViewModel并显示错误消息,一旦我将ViewModel添加回消息中再次停止显示.有没有人成功使用ViewModel进行自定义验证?你有什么必须做的额外工作吗?

更新2:

我用这两个简单的类(代理和人)创建了一个新的MVC3项目.

public class Agency : IValidatableObject
  {

    public int Id { get; set; }

    public string Name { get; set; }

    public DateTime DOB { get; set; }

    public IEnumerable<ValidationResult> Validate(ValidationContext validationContext)
    {
      if (DOB > DateTime.Now.AddYears(-18)) { yield return new ValidationResult("Must be over 18."); }
    }
  }

  public class Person
  {
    public int Id { get; set; }

    public string Name { get; set; }
  }

这是控制器代码

public ActionResult Create()
    {
        return View();
    } 

    //
    // POST: /Agency/Create

    [HttpPost]
    public ActionResult Create(Agency agency)
    {
      if (ModelState.IsValid)
      {

        db.Agencies.Add(agency);
        db.SaveChanges();
        return RedirectToAction("Index");
      }

      return View(agency);
    }

    //[HttpPost]
    //public ActionResult Create(AgencyVM agencyVM)
    //{
    //  if (ModelState.IsValid)
    //  {
    //    var agency = agencyVM.Agency;
    //    db.Agencies.Add(agency);
    //    db.SaveChanges();
    //    return RedirectToAction("Index");
    //  }

    //  return View(agencyVM);
    //}

风景

@model CustValTest.Models.Agency
@*@model CustValTest.Models.AgencyVM*@
@* When using VM (model => model.Name)  becomes (model => model.Agency.Name) etc. *@

@{
    ViewBag.Title = "Create";
}

<h2>Create</h2>

<script src="@Url.Content("~/Scripts/jquery.validate.min.js")" type="text/javascript"></script>
<script src="@Url.Content("~/Scripts/jquery.validate.unobtrusive.min.js")" type="text/javascript"></script>

@using (Html.BeginForm()) {
    @Html.ValidationSummary(true)
    <fieldset>
        <legend>Agency</legend>

        <div class="editor-label">
            @Html.LabelFor(model => model.Name)
        </div>
        <div class="editor-field">
            @Html.EditorFor(model => model.Name)
            @Html.ValidationMessageFor(model => model.Name)
        </div>

        <div class="editor-label">
            @Html.LabelFor(model => model.DOB)
        </div>
        <div class="editor-field">
            @Html.EditorFor(model => model.DOB)
            @Html.ValidationMessageFor(model => model.DOB)
        </div>

        <p>
            <input type="submit" value="Create" />
        </p>
    </fieldset>
}

<div>
    @Html.ActionLink("Back to List","Index")
</div>

ViewModel

public class AgencyVM
  {
    public Agency Agency { get; set; }

    public Person Person { get; set; }

  }

当只在代理中显示代理时,显示验证错误(DOB低于18).显示ViewModel时,不显示错误.自定义验证总是捕获错误,导致ModelState.IsValid失败并重新显示视图.任何人都可以复制这个吗?关于为什么以及如何修复的任何想法?

更新3:

作为临时工作,我通过向ValidationResult添加一个参数,将Validation更改为字段级别(与模型级别1):

if (DOB > DateTime.Now.AddYears(-18)) { yield return new ValidationResult("Must be over 18.",new [] { "DOB" }); }

现在的问题是错误消息显示在字段旁边而不是在表单的顶部(由于用户将返回到没有可见错误消息的表单,因此表示手风琴视图不好).为了解决这个次要问题,我将此代码添加到Controller POST操作中.

ModelState.AddModelError(string.Empty,errMsgInvld);
      return View(agencyVM);
    }
    string errMsgInvld = "There was an entry error,please review the entire form. Invalid entries will be noted in red.";

问题仍然没有答案,为什么模型级错误消息不会显示在ViewModel上(有关此问题,请参阅我对JaySilk84的回复)?

解决方法

现在问题是您的模型是嵌套的,错误消息被放置在没有.DOB的代理下的ModelState中,因为您没有在ValidationResult中指定它. ValidationMessageFor()帮助程序正在查找名为Agency.DOB的密钥(请参阅以下ValidationMessageFor()帮助程序中的相关代码):

string fullHtmlFieldName = htmlHelper.ViewContext.ViewData.TemplateInfo.GetFullHtmlFieldName(expression);
FormContext clientValidation = htmlHelper.ViewContext.GetFormContextForClientValidation();
if (!htmlHelper.ViewData.ModelState.ContainsKey(fullHtmlFieldName) && clientValidation == null)
    return (MvcHtmlString) null;

GetFullHtmlFieldName()返回Agency.DOB,而不是代理

我想如果你把DOB添加到ValidationResult它会工作:

public IEnumerable<ValidationResult> Validate(ValidationContext validationContext)
{
    if (DOB > DateTime.Now.AddYears(-18)) { yield return new ValidationResult("Must be over 18.",new List<string>() { "DOB" }); }
}

ValidationResult的第二个参数将告诉它在ModelState中使用哪个键(默认情况下它将附加作为Agency的父对象),因此ModelState将有一个名为Agency.DOB的键,这是您的ValidationMessageFor()正在寻找的.

编辑:

如果您不想要字段级验证,那么您不需要Html.ValidationMessageFor().你只需要ValidationSummary().

该视图将AgencyVM视为模型.如果您希望它正确验证,那么将验证放在AgencyVM级别并让它验证子对象.或者,您可以对子对象进行验证,但父对象(AgencyVM)必须将其聚合到视图中.您可以做的另一件事是保持原样并将ValidationSummary(true)更改为ValidationSummary(false).这会将ModelState中的所有内容打印到摘要中.我认为从代理商中删除验证并将其放在AgencyVM上可能是最好的方法:

public class AgencyVM : IValidatableObject

    {
        public Agency Agency { get; set; }

        public Person Person { get; set; }

        public IEnumerable<ValidationResult> Validate(ValidationContext validationContext)
        {
            if (Agency.DOB > DateTime.Now.AddYears(-18)) { yield return new ValidationResult("Must be over 18."); }
            if (string.IsNullOrEmpty(Agency.Name)) { yield return new ValidationResult("Need a name"); }
        }
    }

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读