c# - 为什么发布时模型属性数据会发生变化?
问题描述
OnGet()
填充UnapprovedApplications
来自 的数据AspNetUsers
。此时用户 ID 是正确的。
c4d69cbe-436e-4d74-bc7d-c4a99a8cbf34
e184da25-687b-4e51-9f1d-c3a93a732ec1
OnPostAsync()
,但是,将用户 ID 返回为
20b6af04-ecb5-49c0-b4e4-bdb172bd19c7
5a61fbce-4c69-4a15-be66-725c5ab4b884
为什么数据在UnapprovedApplications
变化?如何防止这些数据发生变化?
一个模型ReviewApplicationModel
是这样构建的
[BindProperty]
public IList<ApplicationUser> UnapprovedApplications { get; set; }
public void OnGet(string returnUrl = null)
{
ReturnUrl = returnUrl;
UnapprovedApplications = new List<ApplicationUser>();
foreach (var user in userManager.Users.Where(x => !x.Approved))
{
System.Diagnostics.Debug.WriteLine(user.Id);
UnapprovedApplications.Add(user);
}
}
public async Task<IActionResult> OnPostAsync()
{
if (!ModelState.IsValid)
{
return Page();
}
foreach (var user in UnapprovedApplications)
System.Diagnostics.Debug.WriteLine(user.Id);
return RedirectToPage();
}
鉴于
@page
@model ReviewApplicationsModel
@{
ViewData["Title"] = "Review Applications";
ViewData["ActivePage"] = ManageNavPages.ReviewApplications;
}
<h4>@ViewData["Title"]</h4>
<div class="row">
<div class="col-md-6">
<form asp-route-returnUrl="@Model.ReturnUrl" method="post">
<button type="submit" class="btn btn-primary">Approve Selected Applications</button>
<table class="table">
<thead>
<tr>
<th>
@Html.DisplayNameFor(x => x.UnapprovedApplications.FirstOrDefault().Id)
</th>
<th>
@Html.DisplayNameFor(x => x.UnapprovedApplications.FirstOrDefault().Organization)
</th>
<th>
@Html.DisplayNameFor(x => x.UnapprovedApplications.FirstOrDefault().FEIEIN)
</th>
<th>
@Html.DisplayNameFor(x => x.UnapprovedApplications.FirstOrDefault().State)
</th>
<th>
@Html.DisplayNameFor(x => x.UnapprovedApplications.FirstOrDefault().PrincipalCity)
</th>
<th>
@Html.DisplayNameFor(x => x.UnapprovedApplications.FirstOrDefault().PhoneNumber)
</th>
<th>
Approved
</th>
<th></th>
</tr>
</thead>
<tbody>
@for (var i = 0; i < Model.UnapprovedApplications.Count(); i++)
{
<tr>
<td>@Model.UnapprovedApplications[i].Id</td>
<td>@Model.UnapprovedApplications[i].Organization</td>
<td>@Model.UnapprovedApplications[i].FEIEIN)</td>
<td>@Model.UnapprovedApplications[i].State</td>
<td>@Model.UnapprovedApplications[i].PrincipalCity</td>
<td>@Model.UnapprovedApplications[i].PhoneNumber</td>
<td>
<input asp-for="@Model.UnapprovedApplications[i].Approved" class="form-control" />
<span asp-validation-for="@Model.UnapprovedApplications[i].Approved" class="text-danger"></span>
</td>
</tr>
}
</tbody>
</table>
</form>
</div>
</div>
@section Scripts {
<partial name="_ValidationScriptsPartial" />
}
解决方案
这将是一个正在寻求解决这个线程的独白。
我发现 IdentityUser 有一个生成 Guid 的默认构造函数。发生的情况是客户端缺少模型数据,并且不同的用户 ID 不同,因为正在生成模型数据,就好像他们是新用户一样。
我发现了@Forty-Two 发布的这个信息 通过 表单提交的整个模型
模型将完整地传递给控制器,但不受输入或隐藏字段绑定的属性值将丢失。
您必须在客户端绑定表单中的属性,或者在服务器端重新获取实体。
您似乎要求@Html.HiddenFor(m => m.Model) 之类的东西,这是不可能的。对不起
需要记住的一件事是,如果您有大量隐藏字段,您可能会向视图发送比您真正需要的更多的数据。考虑使用视图模型
我将实现整个聚合的 User 对象的 ViewModel 以尝试保留通过检索的数据,GET
以便可以POST
按原意保存已编辑的数据。
好的,所以实际上我只是在表单中绑定了必要的数据,一切都按预期工作。
这是一个解决方案
[AllowAnonymous]
public class ReviewApplicationsModel : PageModel
{
readonly UserManager<ApplicationUser> userManager;
public ReviewApplicationsModel(UserManager<ApplicationUser> userManager)
{
this.userManager = userManager;
}
public string ReturnUrl { get; set; }
[BindProperty]
public ObservableCollection<ApplicationUser> UnapprovedApplications { get; set; }
public void OnGet(string returnUrl = null)
{
ReturnUrl = returnUrl;
UnapprovedApplications = new ObservableCollection<ApplicationUser>();
foreach (var user in userManager.Users.Where(x => !x.Approved))
UnapprovedApplications.Add(user);
}
public async Task<IActionResult> OnPostAsync(string returnUrl = null)
{
ReturnUrl = returnUrl;
if (!ModelState.IsValid)
return Page();
foreach (var user in UnapprovedApplications.Where(x => x.Approved))
{
var tmp = await userManager.FindByIdAsync(user.Id);
tmp.Approved = true;
await userManager.UpdateAsync(tmp);
}
return RedirectToPage();
}
}
@page
@model ReviewApplicationsModel
@{
ViewData["Title"] = "Review Applications";
ViewData["ActivePage"] = ManageNavPages.ReviewApplications;
}
<h4>@ViewData["Title"]</h4>
<div class="row">
<form method="post">
<div class="col-md-auto">
<button type="submit" class="btn btn-primary">Approve Selected Applications</button>
<table class="table">
<thead>
<tr>
<th hidden>@Html.DisplayNameFor(x => x.UnapprovedApplications.FirstOrDefault().Id)</th>
<th>@Html.DisplayNameFor(x => x.UnapprovedApplications.FirstOrDefault().Organization)</th>
<th>@Html.DisplayNameFor(x => x.UnapprovedApplications.FirstOrDefault().FEIEIN)</th>
<th>@Html.DisplayNameFor(x => x.UnapprovedApplications.FirstOrDefault().State)</th>
<th>@Html.DisplayNameFor(x => x.UnapprovedApplications.FirstOrDefault().Approved)</th>
</tr>
</thead>
<tbody>
@for (var i = 0; i < Model.UnapprovedApplications.Count(); i++)
{
<tr>
<td hidden><input asp-for="UnapprovedApplications[i].Id" class="form-control" hidden /></td>
<td><input asp-for="UnapprovedApplications[i].Organization" class="form-control" disabled /></td>
<td><input asp-for="UnapprovedApplications[i].FEIEIN" class="form-control" disabled /></td>
<td><input asp-for="UnapprovedApplications[i].State" class="form-control" disabled /></td>
<td><input asp-for="UnapprovedApplications[i].Approved" class="form-control" /></td>
</tr>
}
</tbody>
</table>
</div>
</form>
</div>
@section Scripts {
<partial name="_ValidationScriptsPartial" />
}
必须有一种更灵活的方式来做到这一点,我希望有一天能发现它,但这适用于现在
推荐阅读
- midi - 为什么 MIDI 输入第二次使用 AudioKit 初始化 MIDI() 类?
- python - add_edge() 方法的 python netwokx 权重归一化
- php - 是否可以在 Laravel 中自动添加翻译变量?
- java - 为 Eclipse 中的每个源文件夹定义类路径
- python - 使用“is”在 IDLE 和 python shell 中产生不同的结果
- python - 在python中,如何在程序不等待完成的情况下运行函数?
- java - 来自 Spring Security 的 Auth0 访问令牌
- typescript - 类型错误:“值”是只读的
- svn - 使用日期范围过滤 SVN 修订
- laravel - InvalidArgumentException:json_decode 错误:语法错误