原帖地址:http://www.cnblogs.com/QLeelulu/archive/2008/10/08/1305962.html
ViewData有一個ModelState的屬性,這是一個類型為ModelStateDictionary的ModelState類型的字典集合。在進行數據驗證的時候這個屬性是比較實用的。在使用Html.ValidationMessage()的時候,就是從ViewData.ModelState中檢測是否有指定的KEY。假設存在。就提示錯誤信息。比如在前一篇文章ASP.NET MVC 入門7、Hellper與數據的提交與綁定中使用到的UpdateModel方法:
我們在View中使用Html.ValidationMessage(string modelName)來對指定的屬性進行驗證:
Html.ValidationMessage()有幾個重載:
當中ValidationSummary()是用於顯示所有的驗證信息的。跟ASP.NET里面的ValidationSummary驗證控件差點兒相同。
我們測試一下/Admin/Setting頁面:
在用UpdateModel方法更新BlogSettings.Instance.PostsPerPage的時候,當我們如圖所看到的填寫"10d"的時候。因為PostsPerPage為整型的,所以UpdateModel方法就會出錯,同一時候會往ViewData.ModelState加入對應的錯誤信息。從而Html.ValidationMessage()方法就能夠從ViewData.ModelState中檢測到錯誤並提示。
同一時候Html.ValidationMessage()方法會為出錯的屬性的輸入框加入一個名為"input-validation-error"的CSS類,同一時候后面的提示信息的CSS類名為"field-validation-error":
CSS類的樣式是能夠由我們自己自由定義的。如上圖的紅色高亮顯示。
好,以下我們來實現發表新隨筆的功能。我們先寫一個提供用戶輸入隨筆內容的表單頁面:
< label for ="Title" > 標題 </ label >
<% = Html.TextBox( " Title " , new { id = " Title " , @class = " required " }) %>
<% = Html.ValidationMessage( " Title " ) %>
</ p >
< p >
< label for ="Content" > 內容 </ label >
<% = Html.TextArea( " Content " ) %>
<% = Html.ValidationMessage( " Content " ) %>
</ p >
< p >
< label for ="Slug" > URL地址別名(假設為空則和標題同名) </ label >
<% = Html.TextBox( " Slug " , new { id = " Slug " , @class = " required " }) %>
<% = Html.ValidationMessage( " Slug " ) %>
</ p >
然后我們對用戶提交過來的數據進行保存:
public ActionResult SaveNewPost(FormCollection form)
{
Post post = new Post();
try
{
UpdateModel(post, new [] { " Title " , " Content " , " Slug " });
}
catch
{
return View(post);
}
post.Save();
return ShowMsg( new List < string > () { " 發表新隨筆成功 " });
}
因為這三個值都是字符串類型,所以假設值為空的話,UpdateModel也是不會出錯的,而我們的Title和Content是不同意為空的。或者我們想我們的Slug的長度不能超過100,也就是須要有我們自己的業務規則。
這時候我們也許會這樣寫:
{
UpdateModel(post, new [] { " Title " , " Content " , " Slug " });
}
catch
{
return View(post);
}
if ( string .IsNullOrEmpty(post.Title))
{
ViewData.ModelState.AddModelError( " Title " , post.Title, " 標題不能為空 " );
}
if ( string .IsNullOrEmpty(post.Content))
{
ViewData.ModelState.AddModelError( " Content " , post.Content, " 內容不能為空 " );
}
if ( ! ViewData.ModelState.IsValid)
{
return View(post);
}
ViewData.ModelState提供了一個AddModelError的方法。方便我們加入驗證失敗的信息。我們能夠如上代碼這樣進行對象的業務規則驗證,可是一旦業務規則多了。這種代碼是非常壯觀的。並且不好控制。
那么我們該怎么更好的進行業務規則的驗證呢?得意於BlogEngine.Net的良好架構,我們能夠非常輕松的完畢這一點。
首先。讓我們改動一下BlogEngine.Core里面BusinessBase的代碼。我們前面說過。BusinessBase實現了IDataErrorInfo接口,該接口有個索引器,導致ViewData.Eval()方法調用時搜索索引器的值時返回String.Empty而使ViewData.Eval()覺得是找到值了,從而失效。
我們能夠將return string.Empty改動為return null。
但我們這里並不須要用到這個接口。所以我們把該接口去掉,並把對應的代碼凝視了。然后我們再暴露一個BrokenRules的屬性,用於返回當前的全部破壞性業務規則(紅框部分代碼為我們加入的):
BusinessBase提供了一個抽象的ValidationRules方法,用於在業務類重寫這種方法往里面加入驗證規則(詳細請看BusinessBase的Validation節)。

我們在Post類中重寫這種方法來加入驗證規則:
然后我們能夠在Controller的Action中非常優雅的書寫我們的代碼來進行業務規則的驗證:
public ActionResult SaveNewPost(FormCollection form)
{
Post post = new Post();
try
{
UpdateModel(post, new [] { " Title " , " Content " , " Slug " });
}
catch
{
return View(post);
}
if ( ! post.IsValid)
{
foreach ( string key in post.BrokenRules.Keys)
{
ViewData.ModelState.AddModelError(key, form[key], post.BrokenRules[key]);
}
return View(post);
}
post.Save();
return ShowMsg( new List < string > () { " 發表新隨筆成功 " });
}
我們注意到上面的Action中用到了一個FormCollection 的參數,這個參數系統會自己主動將Form提交過來的所有表單值(Request.Form)賦給它的。client驗證能夠用jQuery的驗證插件來。這里就不羅嗦了。