0

我构建了一个简单的 ASP.NET Core 2.0 应用程序,并且有一个简单的登录视图。我正在使用 ASP.NET Core Identity,这意味着我正在使用“IdentityUser”类

@model LoginViewModel

<h2 class="my-4 text-center text-lg-left">Before contacting us, please log in or
    <a asp-action="Register" asp-controller="Account"> register!</a></h2>

<form asp-action="Login" asp-controller="Account" method="post" role="form">
    <!-- To show form error after submission -->
    <div asp-validation-summary="All" class="text-danger"></div>

    <div class="form-group">
        <label asp-for="UserName">User Name</label>
        <input type="text" class="form-control" placeholder="Username">
        <span asp-validation-for="UserName" class="text-danger"></span>
    </div>
    <div class="form-group">
        <label asp-for="Password">Password</label>
        <input type="password" class="form-control" placeholder="Password">
        <span asp-validation-for="Password" class="text-danger"></span>
    </div>

    <button type="submit" class="btn btn-primary">Log In!</button>
</form>

如您所见,我创建了一个名为“LoginViewModel”的视图模型,它将检测提交表单中传递的参数

登录视图模型

public class LoginViewModel
    {
        [Required]
        [Display(Name = "User name")]
        public string UserName { get; set; }

        [DataType(DataType.Password)]
        public string Password { get; set; }
    }

当用户在表单上点击“登录”时,我的 ASP.NET 标签助手指示发布请求将点击“帐户”控制器中的“登录”操作

这是我的“登录”操作

        private readonly SignInManager<IdentityUser> _signInManager;
        private readonly UserManager<IdentityUser> _userManager;

        public AccountController(SignInManager<IdentityUser> signInManager, UserManager<IdentityUser> userManager)
        {
            _signInManager = signInManager;
            _userManager = userManager;
        }

        [HttpGet]
        public IActionResult Login()
        {
            return View(new LoginViewModel());
        }

        [HttpPost]
        public async Task<IActionResult> Login(LoginViewModel loginViewModel)
        {
            if (!ModelState.IsValid)
            {
                return View(loginViewModel);
            }


            var user = await _userManager.FindByNameAsync(loginViewModel.UserName);

            if (user != null)
            {
                var result = await _signInManager.PasswordSignInAsync(user, loginViewModel.Password, false, false);
                if (result.Succeeded)
                {
                    return RedirectToAction("Index", "Home");
                }
            }

            ModelState.AddModelError("", "User name/password not found");
            return View(loginViewModel);
        }

我传递给“登录”操作的对象“loginViewModel”为空,这意味着它没有在用户提交表单时抓取表单数据。

知道我在这里缺少什么,很确定这是愚蠢的吗?

4

2 回答 2

3

您缺少输入上的标签助手!

<div class="form-group">
    <label asp-for="UserName">User Name</label>
    <input type="text" class="form-control" placeholder="Username"
        asp-for="UserName" />    <!-- You're missing this -->
    <span asp-validation-for="UserName" class="text-danger"></span>
</div>

我的 2 美分:

  • 您无需指明 method="post"。这是默认方法。此外,您可能还想指定区域,以防万一您有多个具有相同名称但位于不同区域的控制器。

    <form asp-area="" asp-controller="account" asp-action="login">
        ....
    </form>
    
  • 当您在标签上使用标签助手时,您不需要将文本放在两者之间。标签助手将根据您的[Display]注释生成它。

    <label asp-for="UserName"></label>
    
  • 如果需要,您也可以使用它HTML helpers来生成占位符文本[Display]。这样您就不需要对视图上的文本进行硬编码。

    <div class="form-group">
        <label asp-for="UserName"></label>
        <input type="text" class="form-control" asp-for="UserName"
            placeholder="@Html.DisplayNameFor(x => x.UserName)" />
    </div>
    
于 2018-05-09T06:09:12.760 回答
2

您正在为您的输入生成手动 html,并且它们没有name属性,因此在提交表单时它们不会发布值。

更改您的输入以使用 TagHelpers,这将生成正确的 html,包括用于客户端验证的name属性和属性data-val-*

// <input type="text" class="form-control" placeholder="Username">
<input asp-for="UserName" class="form-control" placeholder="Username">

// <input type="password" class="form-control" placeholder="Password">
<input asp-for="Password" class="form-control" placeholder="Username">
于 2018-05-09T06:09:53.547 回答