使用GoogleApiClient连接失败循环

时间:2015-12-01 22:10:42

标签: android google-drive-api google-drive-android-api

我在尝试执行此Android Drive API示例googledrive/android-quickstart时遇到错误,应用程序运行良好,然后显示一个Windows对话框以选择Google帐户,我选择一个并在logcat中我得到这个:

I/drive-quickstart﹕ GoogleApiClient connection failed: ConnectionResult{statusCode=SIGN_IN_REQUIRED, resolution=PendingIntent{44c22a28: android.os.BinderProxy@44c1b1c0}} 

然后再次显示对话框似乎是一个无限循环

我已经在google开发者控制台中配置了OAuth和其他参数。

这是我的代码,提前致谢..

public class CloudPaintActivity extends Activity implements GoogleApiClient.ConnectionCallbacks,
    GoogleApiClient.OnConnectionFailedListener {

private static final String TAG = "drive-quickstart";
private static final int REQUEST_CODE_CAPTURE_IMAGE = 1;
private static final int REQUEST_CODE_CREATOR = 2;
private static final int REQUEST_CODE_RESOLUTION = 3;

private GoogleApiClient mGoogleApiClient;
private Bitmap mBitmapToSave;

/**
 * Create a new file and save it to Drive.
 */
private void saveFileToDrive() {
    // Start by creating a new contents, and setting a callback.
    Log.i(TAG, "Creating new contents.");
    final Bitmap image = mBitmapToSave;
    Drive.DriveApi.newDriveContents(mGoogleApiClient)
            .setResultCallback(new ResultCallback<DriveApi.DriveContentsResult>() {

                @Override
                public void onResult(DriveApi.DriveContentsResult result) {
                    // If the operation was not successful, we cannot do anything
                    // and must
                    // fail.
                    if (!result.getStatus().isSuccess()) {
                        Log.i(TAG, "Failed to create new contents.");
                        return;
                    }
                    // Otherwise, we can write our data to the new contents.
                    Log.i(TAG, "New contents created.");
                    // Get an output stream for the contents.
                    OutputStream outputStream = result.getDriveContents().getOutputStream();
                    // Write the bitmap data from it.
                    ByteArrayOutputStream bitmapStream = new ByteArrayOutputStream();
                    image.compress(Bitmap.CompressFormat.PNG, 100, bitmapStream);
                    try {
                        outputStream.write(bitmapStream.toByteArray());
                    } catch (IOException e1) {
                        Log.i(TAG, "Unable to write file contents.");
                    }
                    // Create the initial metadata - MIME type and title.
                    // Note that the user will be able to change the title later.
                    MetadataChangeSet metadataChangeSet = new MetadataChangeSet.Builder()
                            .setMimeType("image/jpeg").setTitle("Android Photo.png").build();
                    // Create an intent for the file chooser, and start it.
                    IntentSender intentSender = Drive.DriveApi
                            .newCreateFileActivityBuilder()
                            .setInitialMetadata(metadataChangeSet)
                            .setInitialDriveContents(result.getDriveContents())
                            .build(mGoogleApiClient);
                    try {
                        startIntentSenderForResult(
                                intentSender, REQUEST_CODE_CREATOR, null, 0, 0, 0);
                    } catch (IntentSender.SendIntentException e) {
                        Log.i(TAG, "Failed to launch file chooser.");
                    }
                }
            });
}

@Override
protected void onResume() {
    super.onResume();

    if (mGoogleApiClient == null) {
        // Create the API client and bind it to an instance variable.
        // We use this instance as the callback for connection and connection
        // failures.
        // Since no account name is passed, the user is prompted to choose.
        mGoogleApiClient = new GoogleApiClient.Builder(this)
                .addApi(Drive.API)
                .addApi(Plus.API)
                .addScope(Drive.SCOPE_FILE)
                .addConnectionCallbacks(this)
                .addOnConnectionFailedListener(this)
                .build();
    }
    // Connect the client. Once connected, the camera is launched.
    mGoogleApiClient.connect();
    // if the api client existed, we terminate it
    if (mGoogleApiClient != null && mGoogleApiClient.isConnected()) {
        Plus.AccountApi.clearDefaultAccount(mGoogleApiClient);
        mGoogleApiClient.disconnect();
    }
}

@Override
protected void onPause() {
    if (mGoogleApiClient != null) {
        mGoogleApiClient.disconnect();
    }
    super.onPause();
}

@Override
protected void onActivityResult(final int requestCode, final int resultCode, final Intent data) {
    switch (requestCode) {
        case REQUEST_CODE_CAPTURE_IMAGE:
            // Called after a photo has been taken.
            if (resultCode == Activity.RESULT_OK) {
                // Store the image data as a bitmap for writing later.
                mBitmapToSave = (Bitmap) data.getExtras().get("data");
            }
            break;
        case REQUEST_CODE_CREATOR:
            // Called after a file is saved to Drive.
            if (resultCode == RESULT_OK) {
                Log.i(TAG, "Image successfully saved.");
                mBitmapToSave = null;
                // Just start the camera again for another photo.
                startActivityForResult(new Intent(MediaStore.ACTION_IMAGE_CAPTURE),
                        REQUEST_CODE_CAPTURE_IMAGE);
            }
            break;
    }
}

@Override
public void onConnectionFailed(ConnectionResult result) {
    // Called whenever the API client fails to connect.
    Log.i(TAG, "GoogleApiClient connection failed: " + result.toString());
    if (!result.hasResolution()) {
        // show the localized error dialog.
        GoogleApiAvailability.getInstance().getErrorDialog(this, result.getErrorCode(), 0).show();
        return;
    }
    // The failure has a resolution. Resolve it.
    // Called typically when the app is not yet authorized, and an
    // authorization
    // dialog is displayed to the user.
    try {
        result.startResolutionForResult(this, REQUEST_CODE_RESOLUTION);
    } catch (IntentSender.SendIntentException e) {
        Log.e(TAG, "Exception while starting resolution activity", e);
    }
}

@Override
public void onConnected(Bundle connectionHint) {
    Log.i(TAG, "API client connected.");
    if (mBitmapToSave == null) {
        // This activity has no UI of its own. Just start the camera.
        startActivityForResult(new Intent(MediaStore.ACTION_IMAGE_CAPTURE),
                REQUEST_CODE_CAPTURE_IMAGE);
        return;
    }
    saveFileToDrive();
}

@Override
public void onConnectionSuspended(int cause) {
    Log.i(TAG, "GoogleApiClient connection suspended");
}

}

3 个答案:

答案 0 :(得分:2)

如果您不需要多个帐户,请尝试此操作:

public class MainActivity extends AppCompatActivity {
  private static final int REQ_CONNECT = 1;

  private Activity mAct;
  private static GoogleApiClient mGAC;

  @Override
  protected void onCreate(Bundle bundle) {
    super.onCreate(bundle);
    mAct = this;
    setContentView(R.layout.activity_main);
    if (bundle == null) try {
      mGAC = new GoogleApiClient.Builder(this)
        .addApi(Drive.API)
        .addScope(Drive.SCOPE_FILE)
        .addConnectionCallbacks(new GoogleApiClient.ConnectionCallbacks() {
          @Override public void onConnectionSuspended(int i) { }
          @Override
          public void onConnected(Bundle bundle) {
            Toast.makeText(mAct, "bingo", Toast.LENGTH_LONG).show();  // connected
            saveFileToDrive();
          }
        })
        .addOnConnectionFailedListener(new GoogleApiClient.OnConnectionFailedListener() {
          @Override
          public void onConnectionFailed(ConnectionResult connResult) {
            if (connResult != null) {
              if (!connResult.hasResolution()) {
                int errCode = connResult.getErrorCode();
                GooglePlayServicesUtil.getErrorDialog(errCode, mAct, 0).show();
                return;
              } else try {
                connResult.startResolutionForResult(mAct, REQ_CONNECT);
                return;
              } catch (Exception e) { e.printStackTrace(); }
            }
            finish();  //--- FAIL - no resolution ---------->>>
          }
        })
        .build();
    } catch (Exception e) { e.printStackTrace(); }
  }

  @Override
  protected void onResume() {  super.onResume();
    mGAC.connect();
  }
  @Override
  protected void onPause() {  super.onPause();
    mGAC.disconnect();
  }

  @Override
  protected void onActivityResult(int request, int result, Intent data) {
    switch (request) {
      case REQ_CONNECT:
        if (result == RESULT_OK)
          mGAC.connect();
        else
          finish();  //--- FAIL, user cancelled  ------------->>>
      break;
    }
    super.onActivityResult(request, result, data);
  }
}

如果您需要切换多个帐户,请添加:

.addApi(Plus.API)

到您的构建者并致电:

Plus.AccountApi.clearDefaultAccount(mGAC);

从哪里(例如菜单)。然后创建一个新的mGAC实例并连接。它将再次弹出帐户选择对话框。但您的应用无法知道用户选择(或创建)的帐户。

如果您需要了解当前用户,可以删除 Plus.API 并使用帐户选择器自行管理GooDrive帐户,但您需要实施帐户管理器并实例化 mGAC

.setAccountName([ACCOUNT EMAIL])

here所示(按照 REQ_ACCPICK 并查看客户经理 UT.AM )。

祝你好运

答案 1 :(得分:0)

根据您的错误,无论出于什么原因,它似乎都无法登录。我相信在这种情况下您应该检查hasResolution是否返回true。如果是这样,您可以调用[startResolutionForResult](https://developers.google.com/android/reference/com/google/android/gms/common/ConnectionResult.html#startResolutionForResult(android.app.Activity,int)),这将提示用户登录。

为什么它没有登录,这有点难以辨别。听起来你正在使用AccountPicker?也许您可以尝试只使用一个登录到设备的帐户。

答案 2 :(得分:0)

我观察到您在Google云端控制台上设置的OAuth 2.0客户端ID与我尝试运行的apk不匹配时所描述的行为,无论是通过签名证书指纹还是通过包名称。