【问题标题】:Combining Facebook and Google auth for Firebase Android为 Firebase Android 结合 Facebook 和 Google 身份验证
【发布时间】:2017-02-08 06:02:23
【问题描述】:

背景

您好,我是 Firebase for Android 的新手,我正在尝试首次实现 Facebook 和 Google 身份验证/登录。我按照这两个教程进行了相应的身份验证:

  • http://firebase.google.com/docs/auth/android/google-signin
  • http://firebase.google.com/docs/auth/android/facebook-login

另外,FacebookSignInActivityGoogleSignInActivity 可以正常工作。

问题

问题是我试图在同一个活动中使用 Google 和 Facebook 身份验证,但它不起作用。像这样:

我做了什么

我试图通过让GoogleSignInActivity 扩展MainActivity 并在其中设置布局来将FacebookSignInActivityGoogleSignInActivity 分开。

但我认为我应该将两者合二为一。所以我尝试了,但我得到了一个奇怪的空指针异常:

java.lang.NullPointerException: Attempt to invoke virtual method 'void com.google.firebase.auth.FirebaseAuth.addAuthStateListener(com.google.firebase.auth.FirebaseAuth$AuthStateListener)' on a null object reference

我不知道为什么 onCreate 中的对象为空,因为我从其他两个正在运行的活动中复制了相同的代码:

 mAuth = FirebaseAuth.getInstance();

 mAuthListener = new FirebaseAuth.AuthStateListener() {
            @Override
            public void onAuthStateChanged(@NonNull FirebaseAuth firebaseAuth) {
                FirebaseUser user = firebaseAuth.getCurrentUser();
                if (user != null) {
                    // User is signed in
                    Log.d(TAG, "onAuthStateChanged:signed_in:" + user.getUid());
                } else {
                    // User is signed out
                    Log.d(TAG, "onAuthStateChanged:signed_out");
                }
                // [START_EXCLUDE]
                updateUI(user);
                // [END_EXCLUDE]
            }
        };

我什至不确定是否应该将两者合并为一项活动。我还检查了这些链接:

但看起来这与我正在尝试做的事情不同。如果有人可以帮助我指出正确的方向,我们将不胜感激。

【问题讨论】:

  • 您发布的代码不是抛出异常的代码。当您调用mAuth.addAuthStateListener(mAuthListener) 并且mAuth 为空时会发生异常。确认您已执行mAuth = FirebaseAuth.getInstance(),然后再尝试添加侦听器。
  • 就像在 firebase 教程中一样,您所指的行:mAuth = FirebaseAuth.getInstance() 在我的代码 sn-p 中。它在 onCreate 中被调用。 mAuth.addAuthStateListener(mAuthListener); 在 onStart 中被调用。

标签: android firebase firebase-authentication facebook-authentication google-authentication


【解决方案1】:

你可以试试这个代码:

public class LoginActivity extends AppCompatActivity implements GoogleApiClient.OnConnectionFailedListener,
        View.OnClickListener {

    private static final String TAG = "SignInActivity";
    private static final int RC_SIGN_IN = 9001;

    private GoogleApiClient mGoogleApiClient;
    private FirebaseAuth mAuth;
    private FirebaseAuth.AuthStateListener mAuthListener;

    private CallbackManager mCallbackManager;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        setContentView(R.layout.activity_login);

        // Facebook Login
        FacebookSdk.sdkInitialize(getApplicationContext());
        mCallbackManager = CallbackManager.Factory.create();

        LoginButton mFacebookSignInButton = (LoginButton) findViewById(R.id.facebook_button);
        mFacebookSignInButton.setReadPermissions("email", "public_profile", "user_birthday", "user_friends");

        mFacebookSignInButton.registerCallback(mCallbackManager, new FacebookCallback<LoginResult>() {
            @Override
            public void onSuccess(LoginResult loginResult) {
                Log.d(TAG, "facebook:onSuccess:" + loginResult);
                firebaseAuthWithFacebook(loginResult.getAccessToken());
            }

            @Override
            public void onCancel() {
                Log.d(TAG, "facebook:onCancel");
            }

            @Override
            public void onError(FacebookException error) {
                Log.d(TAG, "facebook:onError", error);
            }
        });

        // Google Sign-In
        // Assign fields
        Button mGoogleSignInButton = (Button) findViewById(R.id.google_button);

        // Set click listeners
        mGoogleSignInButton.setOnClickListener(this);

        GoogleSignInOptions gso = new GoogleSignInOptions.Builder(GoogleSignInOptions.DEFAULT_SIGN_IN)
                .requestIdToken(getString(R.string.default_web_client_id))
                .requestEmail()
                .build();
        mGoogleApiClient = new GoogleApiClient.Builder(this)
                .enableAutoManage(this /* FragmentActivity */, this /* OnConnectionFailedListener */)
                .addApi(Auth.GOOGLE_SIGN_IN_API, gso)
                .build();

        // Initialize FirebaseAuth
        mAuth = FirebaseAuth.getInstance();

        mAuthListener = new FirebaseAuth.AuthStateListener() {
            @Override
            public void onAuthStateChanged(@NonNull FirebaseAuth firebaseAuth) {
                FirebaseUser user = firebaseAuth.getCurrentUser();
                if (user != null) {
                    // User is signed in
                    Log.d(TAG, "onAuthStateChanged:signed_in:" + user.getUid());
                } else {
                    // User is signed out
                    Log.d(TAG, "onAuthStateChanged:signed_out");
                }
            }
        };
    }

    @Override
    public void onStart() {
        super.onStart();
        mAuth.addAuthStateListener(mAuthListener);
    }

    @Override
    public void onStop() {
        super.onStop();
        if (mAuthListener != null) {
            mAuth.removeAuthStateListener(mAuthListener);
        }
    }

    private void firebaseAuthWithGoogle(GoogleSignInAccount acct) {
        Log.d(TAG, "firebaseAuthWithGooogle:" + acct.getId());
        AuthCredential credential = GoogleAuthProvider.getCredential(acct.getIdToken(), null);
        mAuth.signInWithCredential(credential)
                .addOnCompleteListener(this, new OnCompleteListener<AuthResult>() {
                    @Override
                    public void onComplete(@NonNull Task<AuthResult> task) {
                        Log.d(TAG, "signInWithCredential:onComplete:" + task.isSuccessful());

                        // If sign in fails, display a message to the user. If sign in succeeds
                        // the auth state listener will be notified and logic to handle the
                        // signed in user can be handled in the listener.
                        if (!task.isSuccessful()) {
                            Log.w(TAG, "signInWithCredential", task.getException());
                            Toast.makeText(LoginActivity.this, "Authentication failed.",
                                    Toast.LENGTH_SHORT).show();
                        } else {
                            startActivity(new Intent(LoginActivity.this, MainActivity.class));
                            finish();
                        }
                    }
                });
    }

    private void firebaseAuthWithFacebook(AccessToken token) {
        Log.d(TAG, "handleFacebookAccessToken:" + token);

        final AuthCredential credential = FacebookAuthProvider.getCredential(token.getToken());
        mAuth.signInWithCredential(credential)
                .addOnCompleteListener(this, new OnCompleteListener<AuthResult>() {
                    @Override
                    public void onComplete(@NonNull Task<AuthResult> task) {
                        Log.d(TAG, "signInWithCredential:onComplete:" + task.isSuccessful());

                        // If sign in fails, display a message to the user. If sign in succeeds
                        // the auth state listener will be notified and logic to handle the
                        // signed in user can be handled in the listener.
                        if (!task.isSuccessful()) {
                            Log.w(TAG, "signInWithCredential", task.getException());
                            Toast.makeText(LoginActivity.this, "Authentication failed.",
                                    Toast.LENGTH_SHORT).show();
                        } else {
                            startActivity(new Intent(LoginActivity.this, MainActivity.class));
                            finish();
                        }
                    }
                });
    }

 @Override
    public void onClick(View v) {
        switch (v.getId()) {
            case R.id.google_button:
                signIn();
                break;
            default:
                return;
        }
    }

    private void signIn() {
        Intent signInIntent = Auth.GoogleSignInApi.getSignInIntent(mGoogleApiClient);
        startActivityForResult(signInIntent, RC_SIGN_IN);
    }
  @Override
    public void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);

        mCallbackManager.onActivityResult(requestCode, resultCode, data);

        // Result returned from launching the Intent from GoogleSignInApi.getSignInIntent(...);
        if (requestCode == RC_SIGN_IN) {
            GoogleSignInResult result = Auth.GoogleSignInApi.getSignInResultFromIntent(data);
            if (result.isSuccess()) {
                // Google Sign In was successful, authenticate with Firebase
                GoogleSignInAccount account = result.getSignInAccount();
                firebaseAuthWithGoogle(account);
            } else {
                // Google Sign In failed
                Log.e(TAG, "Google Sign In failed.");
            }
        }
    }

    @Override
    public void onConnectionFailed(@NonNull ConnectionResult connectionResult) {
        // An unresolvable error has occurred and Google APIs (including Sign-In) will not
        // be available.
        Log.d(TAG, "onConnectionFailed:" + connectionResult);
        Toast.makeText(this, "Google Play Services error.", Toast.LENGTH_SHORT).show();
    }

}

如果您有任何问题,请告诉我。

【讨论】:

    【解决方案2】:

    也许有人还在寻找解决方案:

    在活动中(yourLoginActivity.class)

    *创建一个常量int来表示ActivityForResult的requestCode

       // You can change to any Value just be unique
       private static final int RC_SIGN_IN = 1001; 
    

    –> 声明 GoogleApiClient、FirebaseAuth、AuthStateListener 和 CallbackManager

    private static final String TAG = "LoginActivity";
    private static final int RC_SIGN_IN = 1001;
    private GoogleApiClient mGoogleApiClient;
    private FirebaseAuth mAuth;
    private FirebaseAuth.AuthStateListener mAuthListener;
    private CallbackManager mCallbackManager;
    

    –> 初始化上面声明的变量:

     mCallbackManager = CallbackManager.Factory.create();  mAuth = FirebaseAuth.getInstance();  mAuthListener = new FirebaseAuth.AuthStateListener() {
         @Override
         public void onAuthStateChanged(@NonNull FirebaseAuth firebaseAuth) {
             FirebaseUser user = firebaseAuth.getCurrentUser();
             if (user != null) {
                 // User is signed in
                 Log.d(TAG, "onAuthStateChanged:signed_in:" + user.getUid());
             } else {
                 // User is signed out
                 Log.d(TAG, "onAuthStateChanged:signed_out");
             }
         }  };
    

    –> 获取 SingInButton 和 LoginButton 的引用:

      LoginButton facebookLoginButton = findViewById(R.id.login_facebook_button);
      SignInButton mGoogleSignInButton = findViewById(R.id.sign_in_button);
    

    –> 处理 SingInButton 和 LoginButton:

     facebookLoginButton.setReadPermissions("email", "public_profile", "user_birthday");
     facebookLoginButton.registerCallback(mCallbackManager, new FacebookCallback<LoginResult>() {
         @Override
         public void onSuccess(LoginResult loginResult) {
             firebaseAuthWithFacebook(loginResult.getAccessToken());
         }
    
         @Override
         public void onCancel() {
             Toast.makeText(LoginActivity.this, "Succes", Toast.LENGTH_SHORT).show();
         }
    
         @Override
         public void onError(FacebookException error) {
             Toast.makeText(LoginActivity.this, "Succes", Toast.LENGTH_SHORT).show();
         }
     });
    
     mGoogleSignInButton.setOnClickListener(new View.OnClickListener() {
         @Override
         public void onClick(View view) {
             signIn();
         }
     });
    

    –> 管理 GoogleSingInOptions 和 GoogleApiClient

    GoogleSignInOptions gso = new GoogleSignInOptions.Builder(GoogleSignInOptions.DEFAULT_SIGN_IN)
            .requestIdToken(getString(R.string.default_web_client_id))
            .requestEmail()
            .build();
    
    mGoogleApiClient = new GoogleApiClient.Builder(this)
            .enableAutoManage(this /* FragmentActivity */, this /* OnConnectionFailedListener */)
            .addApi(Auth.GOOGLE_SIGN_IN_API, gso)
            .build();
    

    –> 处理 Facebook 登录

       private void authWithFacebook(AccessToken token) {
       Log.d(TAG, "handleFacebookAccessToken:" + token);
       final AuthCredential credential = FacebookAuthProvider.getCredential(token.getToken());
        if (mAuth.getCurrentUser() != null) {
            mAuth.getCurrentUser().linkWithCredential(credential)
                    .addOnSuccessListener(new OnSuccessListener<AuthResult>() {
                @Override
                public void onSuccess(AuthResult authResult) {
                    Toast.makeText(LoginActivity.this, "Logged IN", Toast.LENGTH_LONG).show();
                }
            })
    
                    .addOnFailureListener(new OnFailureListener() {
                        @Override
                        public void onFailure(@NonNull Exception e) {
                            e.printStackTrace();
                            Log.e(TAG, "onFailure: " + e.getMessage());
                            mAuth.signInWithCredential(credential).addOnSuccessListener(new OnSuccessListener<AuthResult>() {
                                @Override
                                public void onSuccess(AuthResult authResult) {
                                    Toast.makeText(LoginActivity.this, "Logged IN", Toast.LENGTH_LONG).show();
                                }
                            });
                        }
                    });
        } else {
            mAuth.signInWithCredential(credential);
        }
    }
    

    –> 处理 Google 登录:

      pprivate void authWithGoogle(final GoogleSignInAccount acct) {
        Log.d(TAG, "firebaseAuthWithGooogle:" + acct.getId());
        final AuthCredential credential = GoogleAuthProvider.getCredential(acct.getIdToken(), null);
        if (mAuth.getCurrentUser() != null) {
            mAuth.getCurrentUser().linkWithCredential(credential).addOnSuccessListener(new OnSuccessListener<AuthResult>() {
                @Override
                public void onSuccess(AuthResult authResult) {
                    Toast.makeText(LoginActivity.this, "Logged IN", Toast.LENGTH_LONG).show();
                }
            })
    
                    .addOnFailureListener(new OnFailureListener() {
                        @Override
                        public void onFailure(@NonNull Exception e) {
                            e.printStackTrace();
                            Log.e(TAG, "onFailure: " + e.getMessage());
                            mAuth.signInWithCredential(credential).addOnSuccessListener(new OnSuccessListener<AuthResult>() {
                                @Override
                                public void onSuccess(AuthResult authResult) {
                                    Toast.makeText(LoginActivity.this, "Logged IN", Toast.LENGTH_LONG).show();
                                }
                            });
                        }
                    });
        }else{
            mAuth.signInWithCredential(credential);
        }
    }
    

    –> 覆盖 onConnectionFailed():

     @Override
    public void onConnectionFailed(@NonNull ConnectionResult connectionResult) {
        // An unresolvable error has occurred and Google APIs (including Sign-In) will not
        // be available.
        Log.d(TAG, "onConnectionFailed:" + connectionResult);
        Toast.makeText(this, "Google Play Services error.", Toast.LENGTH_SHORT).show();
    }
    

    –> 唱歌并调用onActivityForResult()

     private void signIn() {
        Intent signInIntent = Auth.GoogleSignInApi.getSignInIntent(mGoogleApiClient);
        startActivityForResult(signInIntent, RC_SIGN_IN);
    }
    

    –> 实现 GoogleApiClient.OnConnectionFailedListener

    LoginActivity extends AppCompatActivity implements GoogleApiClient.OnConnectionFailedListener{
    ....
     }
    

    确保您已启用了 FACEBOOK SDK 从 FIREBASE 控制台登录并正确设置!

    【讨论】:

    • 你的回答帮了大忙。从过去的两天开始,我一直在挠头来做这件事。非常感谢。
    【解决方案3】:

    有人想在 Kotlin 的片段中做这个

    class LoginFragment(): Fragment() ,  GoogleApiClient.OnConnectionFailedListener {
    
        override fun onConnectionFailed(p0: ConnectionResult) {
            Toast.makeText(activity, "Google Play Services error.", Toast.LENGTH_SHORT).show();
        }
    
        private val TAG = "SignInActivity"
        private val RC_SIGN_IN = 9001
        private var mGoogleApiClient: GoogleApiClient? = null
        private var mAuth: FirebaseAuth? = null
        private var mAuthListener: FirebaseAuth.AuthStateListener? = null
        private var mCallbackManager: CallbackManager? = null
    
        override fun onCreate(savedInstanceState: Bundle?) {
            super.onCreate(savedInstanceState)
        }
    
        override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?,
                                  savedInstanceState: Bundle?): View? {
            return inflater.inflate(R.layout.likes_fragment, container, false)
        }
        override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
            super.onViewCreated(view, savedInstanceState)
            FacebookSdk.sdkInitialize(activity);
            mCallbackManager = CallbackManager.Factory.create()
    
            google_container.setOnClickListener {
                signIn()
            }
    
            // Facebook Button (LoginButton) from xml
            login_button.setReadPermissions("email", "public_profile")
            login_button.fragment = this
            login_button.run {
                registerCallback(mCallbackManager, object:  FacebookCallback<LoginResult> {
                    override fun onSuccess(result: LoginResult?) {
                        Log.e(TAG, "Succesfull Facebook Sign In")
                        firebaseAuthWithFacebook(result?.accessToken!!)
                    }
    
                    override fun onCancel() {
                        Log.e(TAG, "Cancelled Facebook Sign In")
                    }
    
                    override fun onError(error: FacebookException?) {
                        Log.e(TAG, "Facebook sign in error : " + error.toString())
                    }
    
                })
            }
    
            val gso = GoogleSignInOptions.Builder(GoogleSignInOptions.DEFAULT_SIGN_IN)
                .requestIdToken(getString(R.string.default_web_client_id))
                .requestEmail()
                .build()
            mGoogleApiClient = GoogleApiClient.Builder(activity!!.applicationContext)
                .enableAutoManage(this.activity!! /* FragmentActivity */, this /* OnConnectionFailedListener */)
                .addApi(Auth.GOOGLE_SIGN_IN_API, gso)
                .build()
    
            mAuth = FirebaseAuth.getInstance()
            mAuthListener = FirebaseAuth.AuthStateListener { firebaseAuth ->
                val user = firebaseAuth.currentUser
                if (user != null) {
                    // User is signed in
                    Log.e(TAG, "onAuthStateChanged:signed_in:" + user.uid)
                } else {
                    // User is signed out
                    Log.e(TAG, "onAuthStateChanged:signed_out")
                }
            }
        }
    
    
        fun signIn() {
            val signInIntent = Auth.GoogleSignInApi.getSignInIntent(mGoogleApiClient)
            startActivityForResult(signInIntent, RC_SIGN_IN);
        }
    
        override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
            super.onActivityResult(requestCode, resultCode, data)
            mCallbackManager!!.onActivityResult(requestCode, resultCode, data);
    
            // Result returned from launching the Intent from GoogleSignInApi.getSignInIntent(...);
            if (requestCode == RC_SIGN_IN) {
                val result = Auth.GoogleSignInApi.getSignInResultFromIntent(data)
                if (result.isSuccess) {
                    // Google Sign In was successful, authenticate with Firebase
                    val account = result.signInAccount
                    firebaseAuthWithGoogle(account!!)
                } else {
                    // Google Sign In failed
                    Log.e(TAG, "Google Sign In failed.");
                }
            }
        }
    
        private fun firebaseAuthWithGoogle(token : GoogleSignInAccount) {
            Log.e(TAG, "handleGoogleAccessToken:" + token);
            val credential = GoogleAuthProvider.getCredential(token.idToken,null)
            mAuth!!.signInWithCredential(credential)
                .addOnCompleteListener(activity as MainActivity, object : OnCompleteListener<AuthResult> {
                    override fun onComplete(task: Task<AuthResult>) {
                        Log.e(TAG, "signInWithCredential:onComplete:" + task.isSuccessful());
                        if(!task.isSuccessful){
                            Toast.makeText(activity, "Failed", Toast.LENGTH_SHORT).show();
                        }else{
                            Toast.makeText(activity, "Success", Toast.LENGTH_SHORT).show();
                        }
                    }
    
                })
    
        }
    
    
         private fun firebaseAuthWithFacebook(token : AccessToken) {
             Log.e(TAG, "handleFacebookAccessToken:" + token);
             val credential = FacebookAuthProvider.getCredential(token.token)
             mAuth!!.signInWithCredential(credential)
                 .addOnCompleteListener(activity as MainActivity, object : OnCompleteListener<AuthResult> {
                     override fun onComplete(task: Task<AuthResult>) {
                         Log.e(TAG, "signInWithCredential:onComplete:" + task.isSuccessful());
                         if(!task.isSuccessful){
                             Toast.makeText(activity, "Failed", Toast.LENGTH_SHORT).show();
    
                         }else{
                             Toast.makeText(activity, "Success", Toast.LENGTH_SHORT).show();
    
                         }
                     }
    
                 })
    
        }
    }
    

    【讨论】:

      猜你喜欢
      • 2018-02-24
      • 2017-11-27
      • 2013-11-03
      • 1970-01-01
      • 2017-11-08
      • 1970-01-01
      • 2018-02-03
      • 2019-05-20
      • 2017-03-11
      相关资源
      最近更新 更多