3

我正在尝试使用 Windows Phone 8 上的 Facebook C# SDK 对 Facebook 上的用户进行身份验证。为此,我遵循以下代码:FacebookLoginPage.xaml.cs

但我面临的问题是,每当我将用户名和密码输入到打开以验证用户身份的对话框中时,我只会得到以下页面:

WP8 FB 登录问题

在此之后,我的程序不会重定向到作为单独视图的登录页面。我看到的其他建议隐藏 WebView 的解决方案不适用,因为身份验证被抽象为单个 LoginAsync 函数调用。关于做什么的任何建议?

4

3 回答 3

6

当 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));
于 2013-10-02T16:33:54.023 回答
1

在我的项目中,我只是听了 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));
    }
}
于 2013-09-13T09:26:52.817 回答
1

在尝试进行所有建议的更改之前,请检查您的 Facebook 应用程序是否处于沙盒模式。这最终将解决您的问题。

在此处输入图像描述

如果您的 Facebook 应用程序处于沙盒模式,则只有开发人员可以使用他的电子邮件地址登录。任何其他 fb 用户都将获得白页。

于 2013-12-13T12:06:06.263 回答