11

我在新的 Google Drive Android API 中的授权流程如下:

  1. 菜单:选择帐户
  2. 连接();
  3. onConnectionFailed() result.startResolutionForResult() 调用 AccountSelectDialog / DriveAuthorization
  4. onConnected() 做你的事

奇迹般有效。现在重复以切换帐户:

  1. 菜单:选择帐户
  2. 连接();
  3. onConnected()

在这里,我没有机会进入AccountSelectDialog,因为我从来没有得到带有“结果”的onConnectionFailed()来调用startResolutionForResult()。这次我错过了什么?

4

5 回答 5

8

首先,添加 Plus.API:

mGoogleApiClient = new GoogleApiClient.Builder(this).addApi(Drive.API).addApi(Plus.API).addScope(Drive.SCOPE_APPFOLDER).addConnectionCallbacks(this).addOnConnectionFailedListener(this).build();

然后您可以像这样切换帐户:

public void onClick(View view) {
  if (view.getId() == R.id.sign_out_button) {
    if (mGoogleApiClient.isConnected()) {
      Plus.AccountApi.clearDefaultAccount(mGoogleApiClient);
      mGoogleApiClient.disconnect();
      mGoogleApiClient.connect();
    }
  }
}

有关更多信息,请参见此处

于 2014-04-05T13:46:29.983 回答
7

打电话

mGoogleApiClient.clearDefaultAccountAndReconnect();

看看文档

这将调用onConnectionFailed将呈现布局以在可用的 Google 帐户中进行选择的回调:

@Override
public void onConnectionFailed(ConnectionResult connectionResult) 
{
    if (connectionResult.hasResolution()) {
        try {                                              
            connectionResult.startResolutionForResult(this, RESOLVE_CONNECTION_REQUEST_CODE);
        } catch (IntentSender.SendIntentException e) {
            // Unable to resolve, message user appropriately
        }
    } else {                                           
        GooglePlayServicesUtil.getErrorDialog(connectionResult.getErrorCode(), this, 0).show();
    }

}
于 2015-06-11T09:12:10.547 回答
6

我意识到通过打开两个关于基本相同主题的 SO 问题,我搞得一团糟。因此,现在是巩固答案的好时机。我在 GDAA 中搜索直接 getter / setter 方法,但只找到了“setter” - setAccountName()) - SO question 21583828 (实际上没有,但 Burcu 帮助了我)。

另一方面,可以通过从“onActivityResult()”获取帐户名称来替换“getter” -SO question 21501829

还有一个关于同一主题的 SO 问题 - 这个问题 - 也已得到解决。

所以结论是:

  1. 从 'onActivityResult()' 获取帐户
  2. 在 'setAccountName()' 中设置帐户
  3. 保留您当前的帐户电子邮件,以便您可以检测到新的电子邮件(如果用户决定切换)并在必要时重置 Google 帐户客户端。

2014 年 11 月 4 日更新:

这是我用来在我的应用程序中保存和管理 Google 帐户的包装器。

import android.accounts.Account;
import android.accounts.AccountManager;
import android.content.Context;
import android.content.SharedPreferences;
import android.preference.PreferenceManager;
import com.google.android.gms.auth.GoogleAuthUtil;

public class GooAccMgr {
  private static final String ACC_NAME = "account_name";
  public  static final int FAIL = -1;
  public  static final int UNCHANGED =  0;
  public  static final int CHANGED = +1;

  private String mCurrEmail = null;  // cache locally

  public Account[] getAllAccnts(Context ctx) {
    return AccountManager.get(acx(ctx)).getAccountsByType(GoogleAuthUtil.GOOGLE_ACCOUNT_TYPE);
  }

  public Account getPrimaryAccnt(Context ctx) {
    Account[] accts = getAllAccnts(ctx);
    return accts == null || accts.length == 0 ? null : accts[0];
  }

  public Account getActiveAccnt(Context ctx) {
    return email2Accnt(ctx, getActiveEmail(ctx));
  }

  public String getActiveEmail(Context ctx) {
    if (mCurrEmail != null) {
      return mCurrEmail;
    }
    mCurrEmail = ctx == null ? null : pfs(ctx).getString(ACC_NAME, null);
    return mCurrEmail;
  }

  public Account email2Accnt(Context ctx, String emil) {
    if (emil != null) {
      Account[] accounts =
       AccountManager.get(acx(ctx)).getAccountsByType(GoogleAuthUtil.GOOGLE_ACCOUNT_TYPE);
      for (Account account : accounts) {
        if (emil.equalsIgnoreCase(account.name)) {
          return account;
        }
      }
    }
    return null;
  }

  /**
   * Stores a new email in persistent app storage, reporting result
   * @param ctx activity context
   * @param newEmail new email, optionally null
   * @return FAIL, CHANGED or UNCHANGED (based on the following table)
   * OLD    NEW   SAVED   RESULT
   * ERROR                FAIL
   * null   null  null    FAIL
   * null   new   new     CHANGED
   * old    null  old     UNCHANGED
   * old != new   new     CHANGED
   * old == new   new     UNCHANGED
   */
  public int setEmail(Context ctx, String newEmail) {
    int result = FAIL;  // 0  0

    String prevEmail = getActiveEmail(ctx);
    if        ((prevEmail == null) && (newEmail != null)) {
      result = CHANGED;
    } else if ((prevEmail != null) && (newEmail == null)) {
      result = UNCHANGED;
    } else if ((prevEmail != null) && (newEmail != null)) {
      result = prevEmail.equalsIgnoreCase(newEmail) ? UNCHANGED : CHANGED;
    }
    if (result == CHANGED) {
      mCurrEmail = newEmail;
      pfs(ctx).edit().putString(ACC_NAME, newEmail).apply();
    }
    return result;
  }

  private Context acx(Context ctx) {
    return ctx == null ? null : ctx.getApplicationContext();
  }
  private SharedPreferences pfs(Context ctx) {
    return ctx == null ? null : PreferenceManager.getDefaultSharedPreferences(acx(ctx));
  }
}

向亚历克斯洛克伍德致敬以获得最初的灵感。不幸的是,我找不到对他原始代码的引用。

于 2014-02-08T03:17:58.397 回答
0

听起来您依赖于默认帐户选择。在此设置中,系统会提示用户一次选择帐户,并记住此状态。

如果您想在您的应用程序中提供帐户切换功能,则需要从您自己的应用程序中启动帐户选择器并提供实例化 GoogleApiClient 时选择的帐户名称。

您可以将最后选择的帐户名称保留在共享首选项中,以便在用户下次切换帐户之前记住它。

于 2014-02-07T01:05:47.647 回答
0

如果您使用 GoogleApiClient,只需调用mGoogleApiClient.clearDefaultAccountAndReconnect().

如果您将 DriveClient 与 GoogleSignInAccount(驱动库 16.0.0)一起使用,请尝试此操作。

// try connect Drive
fun startSignIn() {
    val requiredScopes = HashSet<Scope>()
    requiredScopes.add(Drive.SCOPE_FILE)
    requiredScopes.add(Drive.SCOPE_APPFOLDER)
    val account = GoogleSignIn.getLastSignedInAccount(this)
    if (account != null && account.grantedScopes.containsAll(requiredScopes)) {
        // TODO: Get DriveClient and DriveResourceClient
    } else {
        val option = GoogleSignInOptions.Builder(GoogleSignInOptions.DEFAULT_SIGN_IN)
            .requestScopes(Drive.SCOPE_FILE, Drive.SCOPE_APPFOLDER)
            .build()
        val client = GoogleSignIn.getClient(this, option)
        startActivityForResult(client.signInIntent, REQUEST_CODE_SIGN_IN)
    }
}

// try change account
fun changeAccount() {
    val option = GoogleSignInOptions.Builder(GoogleSignInOptions.DEFAULT_SIGN_IN)
    .build()
    val client = GoogleSignIn.getClient(activity, option)
    client.signOut()
        .addOnSuccessListener {
            Log.d(TAG, "signOut success")
            // Try again sign-in
            startSignIn()
        }
        .addOnFailureListener {
            Log.e(TAG, "signOut failed $it")
        }
    }
于 2018-12-03T09:00:07.193 回答