使用导航属性扩展ASP.Net标识



最近刚开始在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中上下文的引用时,问题中的代码现在就可以工作了。

相关内容

  • 没有找到相关文章

最新更新