【发布时间】:2019-11-24 13:30:41
【问题描述】:
这个剃须刀代码可能有什么原因:
// ShowSelectedMembers.cshtml
@model MyApp.Models.MembersViewModel
@for (int m = 0; m < Model.Members.Count; m++)
{
<input type="hidden" asp-for="@Model.Members[m].Id" />
<input type="hidden" asp-for="@Model.Members[m].FirstName" />
<input type="hidden" asp-for="@Model.Members[m].LastName" />
<span>[debug-info: m = @m, id = @Model.Members[m].Id]</span>
<span>@Model.Members[m].LastName</span>,
<span>@Model.Members[m].FirstName</span>
}
要生成这个 HTML,当 Model.Members.Count = 1 时,只包含 id 为 6653 的成员:
<input type="hidden" data-val="true" data-val-required="The Id field is required." id="Members_0__Id" name="Members[0].Id" value="6652" />
<input type="hidden" id="Members_0__FirstName" name="Members[0].FirstName" value="Peter" />
<input type="hidden" id="Members_0__LastName" name="Members[0].LastName" value="Hanson" />
<span>[debug-info: m = 0, id = 6653]</span>
<span>Swanson</span>,
<span>Lisa</span>
Members[0].Id怎么可能在隐藏字段中的值为6652,而<span>中的值为6653?
这是视图的控制器方法:
public IActionResult ShowSelectedMembers(MembersViewModel vm)
{
vm.Members = vm.Members.Where(s => s.Selected).OrderBy(o => o.LastName).ThenBy(o => o.FirstName).ToList();
return View(vm);
}
这是将整个成员列表发送到控制器方法的表单:
// Index.cshtml
@model MyApp.Models.MembersViewModel
<form asp-action="ShowSelectedMembers">
<button type="submit">View selection</button>
@if (Model.Members.Any())
{
for (int i = 0; i < Model.Members.Count; i++)
{
Model.Members[i].Id
Model.Members[i].FirstName
Model.Members[i].LastName
<input type="checkbox" asp-for="@Model.Members[i].Selected" />
<input type="hidden" asp-for="@Model.Members[i].Id" />
<input type="hidden" asp-for="@Model.Members[i].FirstName" />
<input type="hidden" asp-for="@Model.Members[i].LastName" />
}
}
</form>
当表单数据发送到控制器方法时,它包含该视图中的所有成员。然后该方法过滤掉所有将Selected 设置为false 的成员。
这些是 ViewModel:
public class MembersViewModel
{
// ... some more properties
public List<MemberViewModel> Members { get; set; }
}
public class MemberViewModel
{
public int Id { get; set; }
public string FirstName { get; set; }
public string LastName { get; set; }
public bool Selected { get; set; }
// ... some more properties
}
更新
澄清这个操作的流程:
索引视图包含一个表单,列表中的每个成员都有
Selected-复选框。此表单将整个成员列表传递给控制器方法
ShowSelectedMembers。控制器方法过滤掉
Selected设置为false的成员,并将选定的成员传递给视图ShowSelectedMembers。视图显示过滤后的列表,现在有损坏的数据。
更新 2
我把控制器方法改成了这个,创建了一个新实例而不是重用视图模型:
public async Task<IActionResult> ShowSelectedMembers(MembersViewModel vmIn)
{
List<int> memberIds = new List<int>();
foreach (MemberViewModel member in vmIn.Members.Where(s => s.Selected).ToList())
{
memberIds.Add(member.Id);
}
List<Member> selectedMembers = await db.Members
.Where(s => memberIds.Contains(s.Id))
.ToListAsync();
MembersViewModel vmOut = new MembersViewModel {
Members = auto.Map<List<MemberViewModel>>(selectedMembers)
};
return View(vmOut);
}
...但结果完全一样。
更新 3
这太奇怪了!
当我在ShowSelectedMembers.cshtml 中将剃须刀代码更改为此时,它似乎工作正常:
@*<input type="text" asp-for="@Model.Members[i].Id" />*@
@{
string _id = $"Members_{i}__Id";
string _name = $"Members[{i}].Id";
string _value = Model.Members[i].Id.ToString();
}
<input type="hidden"
data-val="true"
data-val-required="The Id field is required."
id=@_id
name=@_name
value=@_value />
即手动生成输入字段,而不是使用asp-for。
【问题讨论】:
-
这是页面的首次加载吗?还是您之前提交过表单?
-
@Jasen 有问题的视图是由另一个视图(索引)上的表单提交调用的 GET。如果我只选择索引视图列表顶部的成员,它似乎可以工作,但如果我选择列表下方的成员,则成员 #1 信息似乎与所选成员一起发送。
-
@Jasen 请参阅问题底部的更新。
-
不要重用视图模型作为返回值,而是创建一个新实例并返回过滤后的集合。
-
@Jasen 我试过了,但无济于事。请参阅更新 2。
标签: c# razor asp.net-core-mvc asp.net-core-2.2