7

我有一个或多或少标准的模型:

public class Project {
  public int ID { get; set; }
  //... some more properties

  public DateTime StartDate { get; set; }
  public int Duration { get; set; }
}

如果用户修改StartDate或项目Duration,我必须调用一个函数来更新模拟。为了实现这一点,我想检测字段StartDateDuration控制器内的状态变化。

像这样的东西:

if(project.StartDate.stateChange() || project.Duration.stateChange())

以下是控制器方法的示例:

[HttpPost]
public ActionResult Edit(Project project)
{
    if (ModelState.IsValid)
    {
        if(project.StartDate.stateChange() || project.Duration.stateChange())
            doSomething();

        db.Entry(project).State = EntityState.Modified;
        db.SaveChanges();
        return RedirectToAction("Index");
    }
    return View(project);
}

任何想法,我怎样才能做到这一点?

4

4 回答 4

10

我相信您可以将编辑后的实体与从数据库中读取的原始实体进行比较。

就像是:

public ActionResult Edit(Project project)
{
    if (ModelState.IsValid)
    {
        var original = db.Find(project.ID);
        bool changed = original.StartDate != project.StartDate || original.Duration != project.Duration;
        if (changed)
        {
            original.StartDate = project.StartDate;
            original.Duration = project.Duration;
            doSomething();
            db.Entry(original).CurrentValues.SetValues(project);
            db.SaveChanges();
        }
    }
    return View(project);
}
于 2013-05-22T09:23:57.317 回答
8

您可以通过 ViewBag 携带旧值来解决它。

在行动:

public ActionResult Edit(int? id)
{
    //...Your Code
    ViewBag.OrigStartDate = project.StartDate;
    ViewBag.OrigDuration = project.Duration;
    return View(project);
}

将隐藏元素添加到视图

...
@Html.Hidden("OrigStartDate", (DateTime)ViewBag.OrigStartDate)
@Html.Hidden("OrigDuration", (int)ViewBag.OrigDuration)
...

将这些参数添加到 post 方法并检查它们是否有更改

[HttpPost]
public ActionResult Edit(DateTime OrigStartDate, int OrigDuration)
{
    if (ModelState.IsValid)
    {
        if (OrigStartDate != project.StartDate || OrigDuration != project.Duration)
            doSomething();

        db.Entry(project).State = EntityState.Modified;
        db.SaveChanges();
        return RedirectToAction("Index");
    }
    ViewBag.FileTypeId = new SelectList(db.FileTypes, "Id", "TypeName", dbinfo.FileTypeId);
    return View(project);
}
于 2014-12-24T00:27:48.120 回答
2

如果您想在不查询持久层的情况下执行此操作,我猜想将旧值作为字段添加到模型中,然后将它们作为隐藏字段保留在页面中是解决此问题的最简单方法。

因此,在您的模型中添加 CurrentStartDate 和 CurrentDuration:

public class Project {
  public int ID { get; set; }
  //... some more properties

  public DateTime StartDate { get; set; }
  public int Duration { get; set; }

  public DateTime CurrentStartDate { get; set; }
  public int CurrentDuration { get; set; }
}

然后在视图中添加具有旧值的隐藏字段:

@Html.HiddenFor(model => model.CurrentStartDate )
@Html.HiddenFor(model => model.CurrentDuration )

这将为您提供一些将所选值与控制器操作中的值进行比较的东西。

于 2013-05-22T09:14:45.467 回答
2

使用 .AsNoTracking().FirstOrDefault 检索原始对象进行比较:

public ActionResult Edit(Project project)
{
    if (ModelState.IsValid)
    {
        Project original = db.AsNoTracking().FirstOrDefault( p => p.ID == project.ID);
        if (original.StartDate != project.StartDate || original.Duration != project.Duration)
            doSomething();

        db.Entry(project).State = EntityState.Modified;
        db.SaveChanges();
   }
   return View(project);
}
于 2017-03-26T20:26:49.660 回答