我正在尝试使用 Windows Phone 8 上的 Facebook C# SDK 对 Facebook 上的用户进行身份验证。为此,我遵循以下代码:FacebookLoginPage.xaml.cs
但我面临的问题是,每当我将用户名和密码输入到打开以验证用户身份的对话框中时,我只会得到以下页面:
在此之后,我的程序不会重定向到作为单独视图的登录页面。我看到的其他建议隐藏 WebView 的解决方案不适用,因为身份验证被抽象为单个 LoginAsync 函数调用。关于做什么的任何建议?
我正在尝试使用 Windows Phone 8 上的 Facebook C# SDK 对 Facebook 上的用户进行身份验证。为此,我遵循以下代码:FacebookLoginPage.xaml.cs
但我面临的问题是,每当我将用户名和密码输入到打开以验证用户身份的对话框中时,我只会得到以下页面:
在此之后,我的程序不会重定向到作为单独视图的登录页面。我看到的其他建议隐藏 WebView 的解决方案不适用,因为身份验证被抽象为单个 LoginAsync 函数调用。关于做什么的任何建议?
当 FB 检测到 Windows Phone 网络浏览器控件时,它似乎对其重定向脚本进行了一些更改。
C# SDK 所做的是将登录页面生成为“ http://www.facebook.com ....”。当您在 webbrowser 控件上打开此 URL 时,它会被重定向到“ http://m.facebook.com ...”,它会显示 FB 登录页面的移动版本。
这以前没有问题,但最近,当 FB 进行重定向时,它还会从 URL 中删除参数“display=page”。然后发生的情况是,当成功登录 FB 时,“login_success.html”页面会在没有此参数的情况下打开。不传入“display=page”参数,默认为“display=touch”。不幸的是,此 URL 不会在 URL 中附加令牌字符串,因此会显示第一个线程中显示的页面。
对此的解决方案是,不要使用以下代码生成登录 URL,而是将其修改为
原来的:
Browser.Navigate(_fb.GetLoginUrl(parameters));
修正:
var URI = _fb.GetLoginUrl(parameters).toString().replace("www.facebook.com","m.facebook.com");
Browser.Navigate(new Uri(URI));
在我的项目中,我只是听了 WebView 的导航事件。如果发生这种情况,则意味着用户在登录页面上做了一些事情(即按下登录按钮)。然后我解析了你提到的应该包含 OAuth 回调 url 的页面的 uri,如果它是正确的并且结果是成功我手动重定向到正确的页面:
//somewhere in the app
private readonly FacebookClient _fb = new FacebookClient();
private void webBrowser1_Navigated(object sender, System.Windows.Navigation.NavigationEventArgs e)
{
FacebookOAuthResult oauthResult;
if (!_fb.TryParseOAuthCallbackUrl(e.Uri, out oauthResult))
{
return;
}
if (oauthResult.IsSuccess)
{
var accessToken = oauthResult.AccessToken;
//you have an access token, you can proceed further
FBLoginSucceded(accessToken);
}
else
{
// errors when logging in
MessageBox.Show(oauthResult.ErrorDescription);
}
}
如果您在异步函数中抽象日志记录,您希望它以异步方式运行,因此事件是可以的。
对不起我的英语不好。
整页代码:
public partial class LoginPageFacebook : PhoneApplicationPage
{
private readonly string AppId = Constants.FacebookAppId;
private const string ExtendedPermissions = "user_birthday,email,user_photos";
private readonly FacebookClient _fb = new FacebookClient();
private Dictionary<string, object> facebookData = new Dictionary<string, object>();
UserIdentity userIdentity = App.Current.Resources["userIdentity"] as UserIdentity;
public LoginPageFacebook()
{
InitializeComponent();
}
private void webBrowser1_Loaded(object sender, RoutedEventArgs e)
{
var loginUrl = GetFacebookLoginUrl(AppId, ExtendedPermissions);
webBrowser1.Navigate(loginUrl);
}
private Uri GetFacebookLoginUrl(string appId, string extendedPermissions)
{
var parameters = new Dictionary<string, object>();
parameters["client_id"] = appId;
parameters["redirect_uri"] = "https://www.facebook.com/connect/login_success.html";
parameters["response_type"] = "token";
parameters["display"] = "touch";
// add the 'scope' only if we have extendedPermissions.
if (!string.IsNullOrEmpty(extendedPermissions))
{
// A comma-delimited list of permissions
parameters["scope"] = extendedPermissions;
}
return _fb.GetLoginUrl(parameters);
}
private void webBrowser1_Navigated(object sender, System.Windows.Navigation.NavigationEventArgs e)
{
if (waitPanel.Visibility == Visibility.Visible)
{
waitPanel.Visibility = Visibility.Collapsed;
webBrowser1.Visibility = Visibility.Visible;
}
FacebookOAuthResult oauthResult;
if (!_fb.TryParseOAuthCallbackUrl(e.Uri, out oauthResult))
{
return;
}
if (oauthResult.IsSuccess)
{
var accessToken = oauthResult.AccessToken;
FBLoginSucceded(accessToken);
}
else
{
// user cancelled
MessageBox.Show(oauthResult.ErrorDescription);
}
}
private void FBLoginSucceded(string accessToken)
{
var fb = new FacebookClient(accessToken);
fb.GetCompleted += (o, e) =>
{
if (e.Error != null)
{
Dispatcher.BeginInvoke(() => MessageBox.Show(e.Error.Message));
return;
}
var result = (IDictionary<string, object>)e.GetResultData();
var id = (string)result["id"];
userIdentity.FBAccessToken = accessToken;
userIdentity.FBID = id;
facebookData["Name"] = result["first_name"];
facebookData["Surname"] = result["last_name"];
facebookData["Email"] = result["email"];
facebookData["Birthday"] = DateTime.Parse((string)result["birthday"]);
facebookData["Country"] = result["locale"];
Dispatcher.BeginInvoke(() =>
{
BitmapImage profilePicture = new BitmapImage(new Uri(string.Format("https://graph.facebook.com/{0}/picture?type={1}&access_token={2}", id, "square", accessToken)));
facebookData["ProfilePicture"] = profilePicture;
userIdentity.FBData = facebookData;
userIdentity.ProfilePicture = profilePicture;
ARLoginOrRegister();
});
};
fb.GetAsync("me");
}
private void ARLoginOrRegister()
{
WebService.ARServiceClient client = new WebService.ARServiceClient();
client.GetUserCompleted += client_GetUserCompleted;
client.GetUserAsync((string)facebookData["Email"]);
client.CloseAsync();
}
void client_GetUserCompleted(object sender, WebService.GetUserCompletedEventArgs e)
{
if (e.Result == null)
NavigationService.Navigate(new Uri("/RegisterPageFacebook.xaml", UriKind.RelativeOrAbsolute));
else if (e.Result.AccountType != (int)AccountType.Facebook)
{
MessageBox.Show("This account is not registered with facebook!");
NavigationService.Navigate(new Uri("/LoginPage.xaml", UriKind.RelativeOrAbsolute));
}
else
{
userIdentity.Authenticated += userIdentity_Authenticated;
userIdentity.FetchARSocialData((string)facebookData["Email"]);
}
}
void userIdentity_Authenticated(bool success)
{
NavigationService.Navigate(new Uri("/MenuPage.xaml", UriKind.RelativeOrAbsolute));
}
}
在尝试进行所有建议的更改之前,请检查您的 Facebook 应用程序是否处于沙盒模式。这最终将解决您的问题。
如果您的 Facebook 应用程序处于沙盒模式,则只有开发人员可以使用他的电子邮件地址登录。任何其他 fb 用户都将获得白页。