我也不得不处理完全相同的情况。我想出的最简单的解决方案是使用包装模型,类似于:
public class QuestionListModel
{
public IList<QuestionModel> Questions { get; set; }
public IList<QuestionModel> Template
{
get
{
return new List<QuestionModel>
{
new QuestionModel {/* defaults for new question */}
};
}
}
public QuestionListModel()
{
Questions = new List<QuestionModel>();
}
}
重要的部分是拥有Template属性,它也是IEnumerable<T>
与您想要的实际模型相同的类型。这样,MVC 将为您完成自动编号。所以使用这个模型,你得到的是你的集合,带有“Questions_0__Title”编号,你还得到一个带有“Template_0__Title”命名的模板行。
我将我的模板行从 UI 中隐藏起来,并在添加新行时使用它。
在 razor 中,您可以像绑定常规问题一样绑定模板,唯一的区别是它会在隐藏的<div>
. 您还需要将Count
您的问题列表绑定到隐藏字段。就我而言,我有一个问题的编辑器模板,它<div>
使用特定的选择器将其呈现在内部,以便以后轻松访问。所以你最终得到的是类似于:
<div class="templateContainer">
<div class="question">
[Template]
</div>
</div>
<div class="items">
[for each of your items]
<div class="question">
[Question]
</div>
</div>
添加行时,诀窍是使用 javascript:
从隐藏字段中获取计数,将其递增。
var counter = $("#QuestionsListCount");
var count = parseInt(counter.val());
count++;
获取整个模板块,克隆它(.clone(true)
例如使用 jquery),将其命名为唯一的东西(例如使用步骤 1 中的计数器值),并将其附加到您的问题所在的部分。
var template = $("#templateContainer");
var newItem = template.clone(true);
var newId = "item_" + count;
var newQuestion = newItem.children().first();
newQuestion.attr("id", newId);
newQuestion.appendTo('#items');
对于每个项目,例如新附加块中的输入(您可以使用分配给它的新 id 找到它),您将 ids => "Template_0" 替换为 "Questions__count from step 2",并且 names => "Template[ 0]”和“问题[从步骤 2 开始计数]”。
$("#" + newId + " :input").each(function (index, input) {
input.id = input.id.replace("Template_0", "Questions_" + (count - 1));
input.name = input.name.replace("Template[0]", "Questions[" + (count - 1) + "]");
});
更新计数器的隐藏字段=>counter.val(count);
- ...
- 利润!
现在,关于删除,我这样做的方式是,我的 ViewModel 实际上有一个IsDeleted
标志,我也将它绑定到问题编辑器模板中的隐藏字段。这样,删除就像隐藏特定的问题一样简单(问题选择器很方便),然后将该IsDeleted
字段设置为 true。当您通过默认模型绑定器取回整个列表时,您需要丢弃所有已删除的列表(或发出实际删除,具体取决于您的后端数据模型)。
通过这种方式,我避免了处理识别已删除项目的单独方法,也避免了重新编号 UI 中的所有项目。此外,您还可以让服务器端代码确定删除时实际应该发生的情况(例如验证或撤消操作)。
这是一篇很长的文章,但实现并不是那么困难(或长),并且可以很容易地以通用方式重用。
干杯!