遍历复选框列表时,Razor代码产生意外结果

时间:2019-07-15 20:17:45

标签: c# razor asp.net-core-mvc asp.net-core-2.2

此剃刀代码可能有什么原因:

// 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>
}

Model.Members.Count = 1时生成此HTML,仅包含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
}

更新

要阐明此操作的流程,请执行以下操作:

  1. 索引视图包含一个带有Selected-复选框的表单,用于列表中的每个成员。

  2. 此表单将成员的整个列表传递到控制器方法ShowSelectedMembers

  3. 控制器方法会过滤掉Selected设置为false的成员,并将选定的成员传递到视图ShowSelectedMembers

  4. 该视图显示已过滤的列表,该列表现在具有损坏的数据。

更新2

我将控制器方法更改为此,创建了一个新实例,而不是重用viewmodel:

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

1 个答案:

答案 0 :(得分:1)

在第一个示例中,您可以添加ModelState.Clear(),它应该会为您提供所需的结果。

public IActionResult ShowSelectedMembers(MembersViewModel vm)
{
    ModelState.Clear();
    vm.Members = vm.Members.Where(s => s.Selected).OrderBy(o => o.LastName).ThenBy(o => o.FirstName).ToList();
    return View(vm);
}

但是,我将遵循https://stackoverflow.com/a/1775185/2030565的建议并采用PRG模式- POST表单,然后重定向到GET操作。如果您无法从数据库中重新补充模型,则需要将模型存储在TempData存储中。