可排序、可拖动、可排序的表 + mvc 未正确更新顺序

Posted

技术标签:

【中文标题】可排序、可拖动、可排序的表 + mvc 未正确更新顺序【英文标题】:Sortable, draggable, orderable table + mvc not updating order correctly 【发布时间】:2019-03-06 22:47:03 【问题描述】:

我正在尝试向我的视图添加一个可订购的表。在 View New.cshtml 中进行排序时,排序有效。它使用jQuerys sortable。如果我添加一些指令(在这种情况下为 1-5),我可以通过可排序的脚本来移动它们。但是当我尝试在重新排序后添加新指令时,顺序从:

我猜这和隐藏表格有关

@Html.HiddenFor(m => m.Recipe.Instructions[i].Number, new  @id = "Number")

当我对可排序表中的字段重新排序时,它需要更新其值。

我在这里错过了什么?

RecipeController.cs

public ActionResult Create(NewRecipeViewModel viewModel, string command)


//...more code...

    var instructionList = new List<Instruction>();

    if (viewModel.Recipe.Instructions != null)
    
        for (int i = 0; i < viewModel.Recipe.Instructions.Count; i++)
        
            var instruction = new Instruction
            
                Id = viewModel.Recipe.Instructions[i].Id,
                Name = viewModel.Recipe.Instructions[i].Name,
                Number = viewModel.Recipe.Instructions[i].Number

            ;
            instructionList.Add(instruction);
        
    

    if (command.Equals("AddInstruction"))
    
        var instruction = new Instruction
        
            Name = viewModel.NewInstruction.Name,
            Number = viewModel.Recipe.Instructions.Count + 1
        ;

        recipe.Instructions.Add(instruction);
        viewModel.Recipe = recipe;

        return View("New", viewModel);
    

//...more code...

新建.cshtml

<div class="form-group" style="margin-top: 170px;">
    <label class="col-md-2 control-label">
        Lägg till instruktion:
    </label>
    <div class="col-md-10">
        @Html.TextBoxFor(m => m.NewInstruction.Name, new  @class = "form-control" )
    </div>
</div>
<div class="form-group" style="margin-top: 300px;">
    <label class="col-md-2 control-label">
    </label>
    <div class="col-md-10">
        <button type="submit" name="command" value="AddInstruction" class="btn btn-primary">Add instruction</button>
    </div>
</div>
<div class="form-group" style="margin-top: 100px;">

    <label class="col-md-2 control-label">
        Instructions:
    </label>
    <div class="col-md-10">
        <table class="table table-bordered table-hover pagin-table" style="margin-top: 10px">
            <thead>
                <tr bgcolor="#f5f5f5">
                    <th>Order:</th>
                    <th>Instruction:</th>
                </tr>
            </thead>
            <tbody id="sortable">
                @if (Model.Recipe.Instructions != null)
                
                    for (int i = 0; i < Model.Recipe.Instructions.Count; i++)
                    
                        <tr>
                            @Html.HiddenFor(m => Model.Recipe.Instructions[i].Id)
                            <td class="order">
                                @Model.Recipe.Instructions[i].Number
                                @Html.HiddenFor(m => m.Recipe.Instructions[i].Number, new  @id = "Number")
                            </td>
                            <td>
                                @Model.Recipe.Instructions[i].Name
                                @Html.HiddenFor(m => m.Recipe.Instructions[i].Name)
                            </td>
                        </tr>
                    
                
            </tbody>
        </table>
    </div>
</div>
 <script type="text/javascript">
     $(document).ready(function() 
             $('#sortable').sortable(
                 update : function(event, ui)  
                     $('td.order').each(function(index) 
                        var order = index + 1;
                        $(this).find('span').text(order);
                        $(this).find('.Number').val(order);
                     );
                 
             );
     );
 </script

编辑: 已添加

instructionList.Sort((s1, s2) => s1.Number.CompareTo(s2.Number));

到RecipeController.cs。其余的归功于 Stephen Muecke。

RecipeController.cs

public ActionResult Create(NewRecipeViewModel viewModel, string command)


//...more code...

    var instructionList = new List<Instruction>();

    if (viewModel.Recipe.Instructions != null)
    
        for (int i = 0; i < viewModel.Recipe.Instructions.Count; i++)
        
            var instruction = new Instruction
            
                Id = viewModel.Recipe.Instructions[i].Id,
                Name = viewModel.Recipe.Instructions[i].Name,
                Number = viewModel.Recipe.Instructions[i].Number

            ;
            instructionList.Add(instruction);
        
        instructionList.Sort((s1, s2) => s1.Number.CompareTo(s2.Number));
    

//...more code...

新建.cshtml

 @* More code *@
 <td class="order">
      <span>@Model.Recipe.Instructions[i].Number</span>
      @Html.HiddenFor(m => m.Recipe.Instructions[i].Number, new @class = "Number")
 </td>

 @* More code *@    

 <script type="text/javascript">
     $(document).ready(function() 
             $('#sortable').sortable(
                 update : function(event, ui)  
                     $('td.order').each(function(index) 
                        var order = index + 1;
                        $(this).find('span').text(order);
                        $(this).find('.Number').val(order);
                     );
                 
             );
     );
 </script

【问题讨论】:

删除 new @id = "Number" - 您创建的重复 id 属性是无效的 html。是否要更新脚本 update 函数中 Number 的值? 当我对视图表中的字段重新排序时,我想更新控制器中的 Number 属性。 【参考方案1】:

update 函数中的选择器不正确,将返回 undefined(您缺少 #(id 选择器),无论如何$ 创建了一个 jQuery 对象,因此它将是 .val(...) ,而不是value=...

但是使用$('#Number').val(index + 1) 将无法正常工作,因为它只会更新具有id="Number". Duplicateid 属性的第一个元素是无效的html。

改用类名和相对选择器。将html更改为

<td class="order">
    <span>@Model.Recipe.Instructions[i].Number</span>
    @Html.HiddenFor(m => m.Recipe.Instructions[i].Number, new  @class = "Number")
</td>

然后是脚本

$('#sortable').sortable(
    update : function(event, ui)  
        $('td.order').each(function(index) 
            var order = index + 1;
            $(this).find('span').text(order );
            $(this).find('.Number').val(order );
        );
    
);

【讨论】:

做到了!非常感谢!更新了 OP。【参考方案2】:

在您的标记中,将 Model 更改为您的“Html.HiddenFor()”HTML 帮助程序中的投影变量 m

<tr>
    @Html.HiddenFor(m => m.Recipe.Instructions[i].Id)
    <td class="order">
        @Model.Recipe.Instructions[i].Number
        @Html.HiddenFor(m => m.Recipe.Instructions[i].Number, new  @id = "Number")
    </td>
    <td>
        @Model.Recipe.Instructions[i].Name
        @Html.HiddenFor(m => m.Recipe.Instructions[i].Name)
    </td>
</tr>

【讨论】:

恐怕这并不会改变任何事情。

以上是关于可排序、可拖动、可排序的表 + mvc 未正确更新顺序的主要内容,如果未能解决你的问题,请参考以下文章

重新排序 UITableView 行时,详细视图未更新

JQueryUI可排序的thead和tbody在拖动隐藏两个字段的行时缩小

jQuery:可拖动连接到可排序。可拖动项目与可排序列表具有不同的 DOM

可拖动+可排序和可滚动的 div

Jquery ui - 可排序:在可排序元素中按图标“句柄”拖动

Jquery UI 结合了可排序和可拖动