从MVC视图模型中保存重新排序的列表项

时间:2013-04-24 18:12:23

标签: c# asp.net .net asp.net-mvc asp.net-mvc-4

我有一个视图模型,它绑定到'TreasureHuntDetails'对象,其中包含一个线索列表。这是它的数据模型的一部分。

    public TreasureHuntDetails()
    {
        Clues = new List<Clue>();
    }

    [Key]
    public int TreasureHuntId { get; set; }

    public List<Clue> Clues { get; set; }

在页面上,我有一张桌子。 foreach循环遍历线索列表以将它们添加到表中,例如,

@for (int i = 0; i < Model.Clues.Count; i++)

for循环中的表元素非常大,但这里是一个表元素列的示例:

<td>@Html.DisplayFor(m => Model.Clues[i].Location)</td>

到目前为止一切顺利。然后我使用JQuery UI来允许使用拖放重新排序表的项目,如下所示:

        <script type="text/javascript">
        $(document).ready(function()
        {
            $("#clueTable tbody").sortable().disableSelection();
        });
        </script>

一切都很好,我可以拖放元素。

问题是我不知道如何保存元素的新顺序并将它们保存回数据库。

我尝试的第一件事就是将线索列表传递给控制器​​方法,但我发现一旦线索列表到达控制器方法,它总是为空。

例如:

@Url.Action("ViewCluePage", @Model.Clues)

即使发送整个@Model,其中的线索列表也始终为null。从数据模型的构造函数中删除新的列表实例化并没有解决此问题。

我尝试的另一件事是将整个表格包装成HTML表单,但线索列表仍然为空。

基本上,这个问题实际上是两个问题:

1)为什么在将模型对象发送到控制器后,线索列表始终为空。

2)如何保存物品清单的新订单?

更新:根据@recursive的建议,我在尝试将线索元素提交到HTML表单时看到我在哪里犯了错误。

我在for循环之外使用了这个,它迭代了线索元素:

@Html.HiddenFor(m => m.Clues)

我必须在for循环中添加HiddenFor行(对于每个线索项),并且对于线索项的EACH属性,例如,

@Html.HiddenFor(m => m.Clues[i].Id)

这样才能将列表项目发送到控制器,这是向前迈出的一步,但我认为我仍然需要能够反映线索项目发送到控制器的新顺序的代码。目前,在使用JQuery sortable()方法重新排列屏幕上元素的顺序时,这不会更改元素的顺序,因为它们存储在绑定到视图的数据模型中(@ Model.Clues)。

3 个答案:

答案 0 :(得分:5)

1)正如@resursive在评论中所说,你需要在页面上隐藏元素,这些元素映射到Clue类中的属性。

2)至于保持线索的顺序,你需要在数据库中添加一个列,它保存列表中每个线索的位置,并将position属性添加到你的类中。所以你的班级需要包括

public int Position {get;set;}

在创建页面时应从数据库中提取。然后在渲染页面之前,您应该根据位置变量重新排序线索列表。

编辑:使用jquery的可排序属性。 Check out this thread for reference.在停止拖动事件中(或在提交之前),遍历每个可拖动对象并设置对象的每个隐藏位置属性的值。

var positionIndex = 0;
$('.draggableObjectClass).each(function () {
    $(this).find('input[id$=_Position]').val(positionIndex++);
});

答案 1 :(得分:0)

  

但我认为我仍然需要能够反映线索项目发送到控制器的新顺序的代码。

您现在不会在for循环中迭代它们,它们将按照您将它们发送到视图的顺序编入索引。您的订单必须已经维护。

答案 2 :(得分:0)

从这里发布的答案中获得建议,我提出了以下解决方案。

已经有了这个方法来实现UI元素的拖放重新排序,

    $(document).ready(function()
    {
        $("#clueTable tbody").sortable().disableSelection();
    });

我需要一种能够以新的项目顺序读取并将其发送到MVC控制器的方法。为此,我使用Razor @ Html.AttributeEncode方法将每个项目的Id写入表格每行的一列,如下所示:

<td class="Ids" id="@Html.AttributeEncode(Model.Clues[i].Id)">@{var number = i + 1; @number}</td>

(这是围绕一个遍历项目列表的for循环。)

然后,我创建了以下Javascript函数,该函数是从我放在元素表上方的“SaveNewOrder”按钮调用的(用户在完成对表格中的项目重新排序后按下它):

    function getNewOrder()
    {
        var positions = new Array();
        for (var i = 0; i < $('.Ids').length; i++)
        {
            positions[i] = $('.Ids')[i].id;
        }
        $.ajax(
            {
                type: "POST",
                url: "@Url.Action("ReorderClues", "Clues")",
                data:{ treasureHuntDetails: $("form").serialize(), ids: JSON.stringify(positions) }
                contentType:'application/json'
            }).done(function()
            {
                 window.location.href = '@Url.Action("Clues", Model)';  
            }).   
    }

这样做是从每个表项中读取Id元素,并将它们写入数组 - 因此该数组包含Id的新顺序。包含项目的数据模型在重新排序表元素后不会改变,因此为什么这是必要的。

然后使用JQuery Ajax方法在我的'Clues'MVC控制器上调用'ReOrderClues'方法,传递数据模型的序列化版本(包含原始顺序中的线索项列表)和包含的数组新订单中的线索ID列表。当从控制器(.done)返回结果时,我调用一个刷新页面元素的控制器。

因此,不必维护与每条线索相关联的位置值(这将涉及代码中其他地方的重大重构),我正在做的是交换线索的内容以反映新的顺序,但保持我在同一个位置。

这就是我使用MVC控制器实现的目的:

public ActionResult ReorderClues(TreasureHuntDetails treasureHuntDetails, int[] ids)
{
    using (var db = new TreasureHuntDB())
    {
        var clues = treasureHuntDetails.Clues;
        var newClues = NewOrderList(clues, ids);

        // Save the changes of each clue
        for (var i = 0; i < newClues.Count;i++ )
        {
            db.Entry(clues[i]).CurrentValues.SetValues(newClues[i]);
            db.SaveChanges();
        }

        treasureHuntDetails.Clues = newClues;
        TempData["Success"] = "Clues reordered";
    }

    return RedirectToAction("Clues", treasureHuntDetails);
}

public List<Clue> NewOrderList(List<Clue> clues, int[] ids)
{
    var newClueOrder = new List<Clue>();

    // For each ID in the given order
    for (var i = 0; i < ids.Length; i++)
    {
        // Get the original clue that matches the given ID
        var clue = clues.First(clue1 => clue1.Id == ids[i]);

        var newClue = Clue.Clone(clue);

        // Add the clue to the new list. 
        newClueOrder.Add(newClue);

        // Retain the ID of the clue 
        newClueOrder[i].Id = clues[newClueOrder.Count - 1].Id;
    }

    return newClueOrder;
}

在上面的代码片段中,TreasureHuntDB是我的Entity Framework数据库上下文。