我有以下脚本:
function OpenIdLogon(e) {
$.post("/Account/OpenIdLogOn/", { token: e }, function (data) {
$("#userNavigation").html(data);
$(".auth_box").hide();
$(".kb_fading").hide();
});
}
这是行动:
public ActionResult OpenIdLogOn(string token)
{
WebClient cli = new WebClient();
string json = cli.DownloadString(new Uri("http://ulogin.ru/token.php?token=" + Request.Params["token"] + "&host=" + Request.Url.Host));
var obj = JObject.Parse(json);
if (obj["error"] == null)
{
var userName = obj["nickname"].Value<string>();
var email = obj["email"].Value<string>();
FormsAuthentication.SetAuthCookie(userName, true);
}
return PartialView("UserNavigation");
}
而且,我的UserNavigation:
@if (Request.IsAuthenticated)
{
<a href="#" class="username"><span>@Context.User.Identity.Name</span><i class="icon iUser"></i></a>
<ul class="headLine_link">
<li><a href="#">Profile</a></li>
<li>
@Html.ActionLink("Logg Off", "LogOff", "Account", null, new { @class = "exit" })</li>
</ul>
}
else
{
<ul class="headLine_link">
<li><a id="regLink">Register</a></li>
<li><a id="authLink">Log On</a></li>
</ul>
}
Request.IsAuthenticated
仅在刷新页面后等于true的问题。
答案 0 :(得分:4)
问题如下:
在请求($.post("/Account/OpenIdLogOn/"...)
时,用户未对其进行身份验证。
然后在您的操作方法中对用户进行身份验证,但在表示用户在创建Auth cookie之前所做的请求的Request对象上,用户未经过身份验证。但是,正如您在下一个请求中说的那样,当时用户在生成请求时拥有身份验证Cookie。
此处的一个解决方案是创建一个viewmodel对象,以便从您的控制器操作方法发送到您的视图。此viewModel可以有一个名为authenticated的字段,您可以从action方法中正确设置它。然后在视图中检查此值。我没有在Visual Studio中检查过这个,但它应该是这样的:
创建viewmodel:
public class LoginViewModel{
public bool Authenticated{ get; set; }
}
你的行动方法:
public ActionResult OpenIdLogOn(string token)
{
WebClient cli = new WebClient();
string json = cli.DownloadString(new Uri("http://ulogin.ru/token.php?token=" + Request.Params["token"] + "&host=" + Request.Url.Host));
var obj = JObject.Parse(json);
var viewModel = new LoginViewModel{ Authenticated = Request.IsAuthenticated };
if (obj["error"] == null)
{
var userName = obj["nickname"].Value<string>();
var email = obj["email"].Value<string>();
FormsAuthentication.SetAuthCookie(userName, true);
viewModel.Authenticated = true;
}
return PartialView("UserNavigation");
}
和你的观点
@model LoginViewModel
@if (Model.Authenticated)
{
<a href="#" class="username"><span>@Context.User.Identity.Name</span><i class="icon iUser"></i></a>
<ul class="headLine_link">
<li><a href="#">Profile</a></li>
<li>
@Html.ActionLink("Logg Off", "LogOff", "Account", null, new { @class = "exit" })</li>
</ul>
}
else
{
<ul class="headLine_link">
<li><a id="regLink">Register</a></li>
<li><a id="authLink">Log On</a></li>
</ul>
}
创建一个viewmodel而不是仅仅将bool作为模型发送只是因为我总是把我发送的数据放到viewmodel中的视图中。使得以后扩展更容易,并且更容易在视图内部阅读(例如,您可以编写@if (Model.Authenticated)
而不是@if (Model)
)