(同样,一个MVC验证问题。我知道,我知道......)
我想使用AutoMapper(http://automapper.codeplex.com/)来验证我的创建视图中的字段,以查找不在我的数据库中的字段(因此不在我的DataModel中)。
示例:我有一个帐户/创建视图供用户创建新帐户,我想要一个密码和确认密码字段,因此用户必须输入两次密码才能进行确认。
数据库中的Account表如下所示:
Account[Id(PK), Name, Password, Email]
我已经生成了一个ADO.NET实体数据模型,我从中使用ADO.NET自跟踪实体生成器生成了模型。
我还为[Required]等验证注释编写了一个自定义AccountViewModel。
总而言之,这是我的项目结构:
Controllers:
AccountController
Models:
Database.edmx (auto-generated from database)
Model.Context.tt (auto-generated from edmx)
Model.tt (auto-generated from edmx)
AccountViewModel.cs
Views:
Account
Create.cshtml
我的AccountViewModel的代码如下所示:
public class AccountViewModel
{
public int Id { get; set; }
[Required]
public string Name { get; set; }
[Required]
public string Password { get; set; }
[Required]
[Compare("Password")]
public string ConfirmPassword { get; set; }
}
现在,我的Create View看起来像这样:
@model AutoMapperTest.Models.Account
<script src="@Url.Content("~/Scripts/jquery-1.4.4.min.js")" type="text/javascript"></script>
<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>Account</legend>
<div class="editor-label">
Name
</div>
<div class="editor-field">
@Html.TextBox("Name")
@Html.ValidationMessageFor(model => model.Name)
</div>
<div class="editor-label">
Email
</div>
<div class="editor-field">
@Html.TextBox("Email")
@Html.ValidationMessageFor(model => model.Email)
</div>
<div class="editor-label">
Password
</div>
<div class="editor-field">
@Html.TextBox("Password")
@Html.ValidationMessageFor(model => model.Password)
</div>
<div class="editor-label">
Confirm your password
</div>
<div class="editor-field">
@Html.TextBox("ConfirmPassword");
@Html.ValidationMessageFor(model => model.ConfirmPassword)
</div>
<p>
<input type="submit" value="Create" />
</p>
</fieldset>
}
<div>
@Html.ActionLink("Back to List", "Index")
</div>
我的代码失败,因为我的模型当然不包含ConfirmPassword字段。 现在,一只小鸟低声对我说,AutoMapper可以解决这个问题。但我无法弄明白......有人可以告诉我,我必须做些什么来使这项工作?我的AccountController现在看起来像这样:
private readonly AccountViewModel _viewModel = new AccountViewModel();
private readonly DatabaseEntities _database = new DatabaseEntities();
//
// GET: /Account/Create
public ActionResult Create()
{
Mapper.CreateMap<AccountViewModel, Account>();
return View("Create", _viewModel);
}
//
// POST: /Account/Create
[HttpPost]
public ActionResult Create(AccountViewModel accountToCreate)
{
try
{
if (ModelState.IsValid)
{
var newAccount = new Account();
Mapper.Map(accountToCreate, newAccount);
_database.Account.AddObject(newAccount);
_database.SaveChanges();
}
return RedirectToAction("Index");
}
catch
{
return View();
}
}
但这不起作用......(从http://weblogs.asp.net/shijuvarghese/archive/2010/02/01/view-model-pattern-and-automapper-in-asp-net-mvc-applications.aspx得到了例子)
任何人都可以请教我这件事吗?非常感谢,我对文本墙和关于同一主题的数百个问题表示道歉......
答案 0 :(得分:18)
关于您的代码的几点评论:
@model
模型的强类型(Account
声明),而应该键入AccountViewModel
视图模型(如果您声明视图模型没有意义不要在视图中使用它。readonly
)声明AccountViewModel
字段。您可以在GET操作中实例化视图模型,并保留默认模型绑定器将其实例化为POST操作的操作参数。Mapper.CreateMap<TSource, TDest>
)一次(理想情况下在Application_Start
中),而不是在控制器操作中所以这就是你的代码的样子:
public ActionResult Create()
{
var model = new AccountViewModel();
return View("Create", model);
}
[HttpPost]
public ActionResult Create(AccountViewModel accountToCreate)
{
try
{
if (ModelState.IsValid)
{
var newAccount = Mapper.Map<AccountViewModel, Account>(accountToCreate);
_database.Account.AddObject(newAccount);
_database.SaveChanges();
}
return RedirectToAction("Index");
}
catch
{
return View();
}
}
答案 1 :(得分:3)
用
替换你查看的第一行@model AutoMapperTest.AccountViewModel
此外,您只需要为应用生命周期调用一次Mapper.CreateMap(例如,在应用开始时)