Представление MVC3 не обновляется с помощью EF 4.1

#asp.net-mvc-3 #entity-framework-4 #razor

#asp.net-mvc-3 #entity-framework-4 #razor

Вопрос:

У меня возникла проблема с обновлением записи. По какой-то причине он даже не выполняет действие post в контроллере, а просто возвращает: «Элемент с тем же ключом уже добавлен».

Похоже, что он ведет себя так, как будто он выполняет вставку, а не обновление. Я был бы признателен за новый взгляд на это. Вероятно, это что-то очень простое, что я пропустил.

Контроллер:

 // GET: /Manage/Regions/Edit/5

    public ActionResult Edit(int id)
    {
        Region_CU regionEdit = (from r in db.Venues_Regions
                                where r.RegionsID == id
                                select new Region_CU
                                {
                                        RegionsID = r.RegionsID,
                                        Name = r.Name,
                                        CalendarLink = r.CalendarLink,
                                        MapIcon = r.MapIcon,
                                        QtrStart = r.QtrStart,
                                        QtrEnd = r.QtrEnd,
                                        FacebookLikeBox = r.FacebookLikeBox,
                                        FacebookId = r.FacebookId
                                    //  region = r 
                                }).Single();
        return View(regionEdit);
    }

    //
    // POST: /Manage/Regions/Edit/5

    [HttpPost]
    public ActionResult Edit(Region_CU r)
    {
        if (ModelState.IsValid)
        {
            var v = db.Venues_Regions.First(i => i.RegionsID == r.RegionsID);
                //v.RegionsID = r.RegionsID;
                v.Name = r.Name;
                v.CalendarLink = r.CalendarLink;
                v.MapIcon = r.MapIcon;
                v.QtrStart = r.QtrStart;
                v.QtrEnd = r.QtrEnd;
                v.FacebookLikeBox = r.FacebookLikeBox;
                v.FacebookId = r.FacebookId;
            //Venues_Regions v = new Venues_Regions
            //{
            //    RegionsID = r.RegionsID,
            //    Name = r.Name,
            //    CalendarLink = r.CalendarLink,
            //    MapIcon = r.MapIcon,
            //    QtrStart = r.QtrStart,
            //    QtrEnd = r.QtrEnd,
            //    FacebookLikeBox = r.FacebookLikeBox,
            //    FacebookId = r.FacebookId
            //};
            //db.Venues_Regions.Attach(v);
            //db.ObjectStateManager.ChangeObjectState(v, EntityState.Modified);
            db.SaveChanges();
            return RedirectToAction("Index");
        }
        return View(r);
    }
  

Вид :

 @model THPT_Razor.Areas.Manage.Models.Region_CU

@{
ViewBag.Title = "Edit Region";
}

<h2>Edit</h2>
<link href="@Url.Content("~/Content/themes/base/jquery.ui.all.css")" rel="stylesheet" type="text/css" />
<script src="@Url.Content("~/Scripts/jquery.validate.min.js")" type="text/javascript">   </script>
  
 <script>
   $(document).ready(function () { $('.date').datepicker({ dateFormat: "mm/dd/yy"   });    });
</script>
@using (Html.BeginForm()) {
  @* @Html.ValidationSummary(true)*@
<fieldset>
    <legend>@Html.DisplayFor(model => model.Name)</legend>

    @Html.HiddenFor(model => model.RegionsID)
    @Html.HiddenFor(model => model.Name)
    @Html.HiddenFor(model => model.CalendarLink)
    <div class="editor-label">
        @Html.Label("Map Icon")
    </div>
    <div class="editor-field">
        @Html.DropDownListFor(model => model.MapIcon, new SelectList(Model.mapicons,"id","Description"))
    </div>
    <div class="editor-label">
        @Html.Label("Quarter Start")
    </div>
    <div class="editor-field">
        @Html.TextBoxFor(model => model.QtrStart, new { @class = "date" })
        @Html.ValidationMessageFor(model => model.QtrStart)
    </div>

    <div class="editor-label">
        @Html.Label("Quarter End")
    </div>
    <div class="editor-field">
        @Html.TextBoxFor(model => model.QtrEnd, new { @class = "date" })
        @Html.ValidationMessageFor(model => model.QtrEnd)
    </div>

    <div class="editor-label">
        @Html.Label("Region Facebook ID")
    </div>
    <div class="editor-field">
        @Html.EditorFor(model => model.FacebookId)
        @Html.ValidationMessageFor(model => model.FacebookId)
    </div>
    <div class="editor-label">
       @Html.Label("Region Facebook LikeBox Code")
    </div>
    <div class="editor-field">
        @Html.EditorFor(model => model.FacebookLikeBox)
        @Html.ValidationMessageFor(model => model.FacebookLikeBox)
    </div>
    <p>
        <input type="submit" value="Update" />
    </p>
</fieldset>
  

Редактировать:

Я получил несколько хороших предложений, но, думаю, я не был ясен. Region_CU не является объектом. Venues_Regions — это то, что я пытаюсь обновить. смотрите Комментарии в классе ниже для пояснения. Первоначальной целью было создать простую оболочку, которая имела объект Venues_Regions и объект list для значков карты. Однако аннотация данных для поля likebox не была передана, в результате чего объект Venues_Regions был выделен. Теперь, когда я пытаюсь сохранить обновление, оно даже не попадает в действие http post. Я надеюсь, что это проясняет то, чего я пытаюсь достичь, и прошу о помощи. Еще раз спасибо за всю помощь и быстрые ответы.

 //create and update
public class Region_CU
{

    public Region_CU()
    {

    }
    public List<MapIcon> mapicons { get; set; }
    //public Venues_Regions region { get; set; }

    // The fields below are what makes up Veunes_Region
    // this was broken out from the above Venues_Region 
    // because the UIHint was not being passed through
    public int RegionsID { get; set; }
    public string Name { get; set; }
    public string CalendarLink { get; set; }
    public int MapIcon { get; set; }
    public DateTime? QtrStart { get; set; }
    public DateTime? QtrEnd { get; set; }

    [UIHint("tinymce_jquery_full"), AllowHtml]
    public string FacebookLikeBox { get; set; }
    public string FacebookId { get; set; }
    public string mapIcon { get; set; }

}
  

Редактировать # 2:
После хорошего ночного сна решение представилось само собой. В действии обновления все, что мне нужно было сделать, это перейти от передаваемой оболочки к передаваемому объекту Venues_Region, и теперь все работает.

Спасибо за всю помощь и предложения.

Заранее спасибо за помощь, Крис

Комментарии:

1. рассматривали ли вы возможность перехода на EF 4.1? Его DbContext API значительно упрощает обновление всего, что вам нужно.

Ответ №1:

На самом деле все, что вам нужно сделать, это загрузить места и вызвать TryUpdateModel. Вам даже не нужно передавать объект. Затем сохраните место проведения. Другой подход заключается в использовании automapper для копирования полей между объектами или использовании метода attach, как упоминалось, но в любом случае копирование полей вручную не требуется.

Комментарии:

1. Спасибо за советы, Адам. Однако проблема заключалась в том, что я передал оболочку обратно действию обновления, а не объекту, который вызывал всю мою путаницу. Я использовал метод attach и видел примеры UpdateModel и TryUpdateModel, но что такое automapper?

2. Automapper находится в codeplex и обычно используется в MVC для копирования полей из одного объекта в другой на основе совпадающих имен свойств. Проверьте это, две строки кода для копирования всех реквизитов довольно хороши.