最近刚开始在VS2013中使用ASP.Net Identity。我启动了一个MVC5项目,并使用了默认模板,带有个人帐户。我正在尝试用以下字段扩展从IdentityUser继承的ApplicationUser类:
//this is my code first entity
public class ApplicationUser : IdentityUser
{
public string Name { get; set; }
public string EmailAddress { get; set; }
public virtual SellingLocation Location { get; set; }
}
我已经用Name和EmailAddress的文本框扩展了Register.cs.html,更新了registerviewmodel,并添加了一个包含所有SellingLocation的列表,用它填充了一个下拉列表。
public class RegisterViewModel
{
[Required]
[Display(Name = "Användarnamn")]
public string UserName { get; set; }
[Required]
[StringLength(100, ErrorMessage = "The {0} must be at least {2} characters long.", MinimumLength = 6)]
[DataType(DataType.Password)]
[Display(Name = "Lösenord")]
public string Password { get; set; }
[DataType(DataType.Password)]
[Display(Name = "Bekräfta lösenord")]
[Compare("Password", ErrorMessage = "The password and confirmation password do not match.")]
public string ConfirmPassword { get; set; }
[Required]
[Display(Name = "Säljarens namn")]
public string Name { get; set; }
[Required]
[Display(Name = "Epostadress")]
public string EmailAddress { get; set; }
[Display(Name = "Försäljningställe")]
public int SellingLocationId { get; set; }
public List<SellingLocationViewModel> SellingLocations { get; set; }
}
和Register.cs.html代码的下拉:
<div class="form-group">
@Html.LabelFor(m => m.SellingLocationId, new { @class = "col-md-2 control-label" })
<div class="col-md-10">
@Html.DropDownListFor(m => m.SellingLocationId, new SelectList(Model.SellingLocations, "Id", "Name"))
</div>
</div>
问题出在帐户控制器上。当我尝试添加此新用户时,会创建SellingLocation的新副本(数据库中的新行),而不是仅向下拉列表中所选值所指向的行添加引用(外键)。我做错了什么?
// GET: /Account/Register
[AllowAnonymous]
public ActionResult Register()
{
var regvm = new RegisterViewModel();
regvm.SellingLocations = GetSellingLocations();
return View(regvm);
}
private List<SellingLocationViewModel> GetSellingLocations()
{
return Mapper.Map<List<SellingLocation>, List<SellingLocationViewModel>>(db.SellingLocations.ToList());
}
//
// POST: /Account/Register
[HttpPost]
[AllowAnonymous]
[ValidateAntiForgeryToken]
public async Task<ActionResult> Register(RegisterViewModel model)
{
if (ModelState.IsValid)
{
//get the SellingLocation from the id from the selected value in dropdown
var sellingloc = db.SellingLocations.Find(model.SellingLocationId);
//creating user object, adding new data
var user = new ApplicationUser() { UserName = model.UserName, Name = model.Name, EmailAddress = model.EmailAddress, Location = sellingloc };
var result = await UserManager.CreateAsync(user, model.Password);
//this will create a new row in SellingLocations, not just add a reference in AspNetUsers to the old row as expected.
if (result.Succeeded)
{
await SignInAsync(user, isPersistent: false);
return RedirectToAction("Index", "Home");
}
else
{
AddErrors(result);
}
}
// If we got this far, something failed, redisplay form
model.SellingLocations = GetSellingLocations();
return View(model);
}
您需要ApplicationUser中的SellingLocationId并配置您的外键关系。当您创建一个新的ApplicationUser时,您只需填写SellingLocationId,而不是填写虚拟对象位置[/strong>Location由EF在插入后或查询新的ApplicationUser时填写。
此外,要在数据库中复制SellingLocation,必须意味着SelingLocationID不是唯一的ID。它必须是。
我发现了问题。我使用了两个不同的上下文对象,一个用于UserManager,另一个用于在Register-post方法中查找SellingLocation对象。
更改AccountController的构造函数并捕获对传递到UserStore中的上下文的引用,该引用被传递到UserManager中,从而解决了问题。当db对象是对UserManager中上下文的引用时,问题中的代码现在就可以工作了。