asp.net-mvc – ASP.NET MVC 5模型绑定编辑视图

前端之家收集整理的这篇文章主要介绍了asp.net-mvc – ASP.NET MVC 5模型绑定编辑视图前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
我无法想出一个最好用口头和一些代码描述的问题的解决方案.我正在使用VS 2013,MVC 5和EF6代码优先;我还使用MvcControllerWithContext脚手架,它生成一个支持CRUD操作的控制器和视图.

简单地说,我有一个包含CreatedDate值的简单模型:

public class WarrantyModel
{
    [Key]
    public int Id { get; set; }
    public string Description { get; set; }
    DateTime CreatedDate { get; set; }
    DateTime LastModifiedDate { get; set; }
}

包含的MVC脚手架为其索引,创建,删除,详细信息和编辑视图使用相同的模型.我想在’创建’视图中创建CreatedDate;我不希望它在“编辑”视图中,因为我不希望在编辑视图发回服务器时更改其值,并且我不希望任何人在表单发布期间能够篡改该值.

理想情况下,我不希望CreatedDate进入“编辑”视图.我已经在模型的CreatedDate属性上找到了一些属性(例如,[ScaffoldColumn(false)]),这些属性阻止它出现在Edit视图中,但随后我在回发时遇到绑定错误,因为CreatedDate结束了值为1/1/0001 12:00:00 AM.这是因为编辑视图没有将值传回给CreatedDate字段的控制器.

我不想实现需要任何sql Server更改的解决方案,例如在包含CreatedDate值的表上添加触发器.如果我想进行快速修复,我会在呈现编辑视图之前存储CreatedDate(服务器端),然后在回发时恢复CreatedDate – 这样我就可以更改1/1/0001日期在渲染视图之前从数据库提取到CreatedDate EF6.这样,我可以将CreatedDate作为隐藏的表单字段发送,然后在回发后覆盖控制器中的值,但我没有一个很好的策略来存储服务器端值(我不想使用Session变量或者ViewBag).

我看着使用[Bind(Exclude =“CreatedDate”)],但这没有帮助.

我的控制器的编辑回发函数中的代码如下所示:

public ActionResult Edit([Bind(Include="Id,Description,CreatedDate,LastModifiedDate")] WarrantyModel warrantymodel)
{
    if (ModelState.IsValid)
    {
        db.Entry(warrantymodel).State = EntityState.Modified;
        db.SaveChanges();
        return RedirectToAction("Index");
    }
    return View(warrantymodel);
}

我以为我可能能够检查上面if块中的db.Entry(warrantymodel)对象,并在OriginalValue中检查CreatedDate,但是当我尝试访问该值时(如下所示),我得到了类型’的异常’ System.InvalidOperationException“:

var originalCreatedDate = db.Entry(warrantymodel).Property("CreatedDate").OriginalValue;

如果我可以成功检查原始的CreatedDate值(即已经在数据库中的值),我可以覆盖CurrentValue的任何内容.但由于上面的代码生成异常,我不知道还能做什么. (我考虑过查询数据库中的值,但这很愚蠢,因为在呈现编辑视图之前已经查询数据库的值).

我的另一个想法是将CreatedDate值的IsModified值更改为false,但是当我调试时,我发现它在前面显示的’if’块中已经设置为false:

bool createdDateIsModified = db.Entry(warrantymodel).Property("CreatedDate").IsModified;

我对如何处理这个看似简单的问题缺乏想法.总之,我不想将模型字段传递给Edit视图,并且我希望该视图(在此示例中为CreatedDate)在视图中的其他Edit字段被回发并使用持久化到数据库时保持其原始值db.SaveChanges().

任何帮助/想法将非常感激.

谢谢.

解决方法

您应该利用viewmodels:
public class WarrantyModelCreateviewmodel
{
    public int Id { get; set; }
    public string Description { get; set; }
    DateTime CreatedDate { get; set; }
    DateTime LastModifiedDate { get; set; }
}

public class WarrantyModelEditviewmodel
{
    public int Id { get; set; }
    public string Description { get; set; }
    DateTime LastModifiedDate { get; set; }
}

viewmodel的意图与域模型的意图略有不同.它为视图提供了正确呈现所需的足够信息.

viewmodels还可以保留与您的域无关的信息.它可以保存对表或搜索过滤器上的排序属性的引用.穿上你的领域模型肯定没有意义!

现在,在控制器中,您将viewmodel中的属性映射到域模型并保留更改:

public ActionResult Edit(WarrantyModelEditviewmodel vm)
{
    if (ModelState.IsValid)
    {
        var warrant = db.Warranties.Find(vm.Id);
        warrant.Description = vm.Description;
        warrant.LastModifiedDate = vm.LastModifiedDate;
        db.SaveChanges();
        return RedirectToAction("Index");
    }
    return View(warrantymodel);
}

此外,viewmodel非常适合合并来自多个模型的数据.如果您有关于保修的详细信息视图,但您还希望在保修期内完成所有服务,该怎么办?你可以简单地使用像这样的viewmodel:

public class WarrantyModelDetailsviewmodel
{
    public int Id { get; set; }
    public string Description { get; set; }
    DateTime CreatedDate { get; set; }
    DateTime LastModifiedDate { get; set; }
    List<Services> Services { get; set; }
}

viewmodels简单,灵活,非常受欢迎.以下是对它们的一个很好的解释:http://lostechies.com/jimmybogard/2009/06/30/how-we-do-mvc-view-models/

你最终会编写很多映射代码. Automapper非常棒,可以完成大部分繁重工作:http://automapper.codeplex.com/

猜你在找的asp.Net相关文章