Загрузка фотографий на Google Диск через приложение для Android + не могу выйти из приложения, пока не нажму кнопку «Домой»

Привет, я хочу разработать приложение, которое делает фотографии и загружает их на Google Диск. Сегодня я нашел мастер (исходный код) из Github, который Google https://github.com/googledrive/android-quickstart

Это очень полезно. Но я обнаружил некоторые проблемы: если я нажму кнопку «Назад», приложение все равно не завершит свою работу. По умолчанию он всегда открывает камеру, делает фотографии и сохраняет их на Google Диске. Он делает одно и то же снова и снова. Если я хочу выйти из приложения, я не могу это сделать, пока не нажму кнопку "Домой". Есть какое-нибудь решение? Также есть еще одна проблема: после фотографирования появляется диалоговое окно с вопросом, где сохранить изображение и какое у него будет имя. Проблема в том, что если я нажму кнопку отмены, появится то же самое. диалог снова и снова. Если я нажму «ОК», диалоговое окно не появится, но если я нажму «Отмена», снова отобразится тот же диалог. Я хочу избавиться от него, когда я нажимаю «Отмена». Любое решение? Это код:

package com.randb.uploadtogdrive;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.OutputStream;

import android.app.Activity;
import android.content.Intent;
import android.content.IntentSender;
import android.content.IntentSender.SendIntentException;
import android.graphics.Bitmap;
import android.os.Bundle;
import android.provider.MediaStore;
import android.util.Log;
import android.widget.Toast;

import com.google.android.gms.common.ConnectionResult;
import com.google.android.gms.common.GooglePlayServicesUtil;
import com.google.android.gms.common.api.GoogleApiClient;
import com.google.android.gms.common.api.GoogleApiClient.ConnectionCallbacks;
import com.google.android.gms.common.api.GoogleApiClient.OnConnectionFailedListener;
import com.google.android.gms.common.api.ResultCallback;
import com.google.android.gms.drive.Drive;
import com.google.android.gms.drive.DriveApi.ContentsResult;
import com.google.android.gms.drive.MetadataChangeSet;
public class MainActivity extends Activity implements ConnectionCallbacks,
OnConnectionFailedListener {

private static final String TAG = "android-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.newContents(mGoogleApiClient).setResultCallback(new ResultCallback<ContentsResult>() {

        @Override
        public void onResult(ContentsResult 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.getContents().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("myPhoto.png").build();
            // Create an intent for the file chooser, and start it.
            IntentSender intentSender = Drive.DriveApi
                    .newCreateFileActivityBuilder()
                    .setInitialMetadata(metadataChangeSet)
                    .setInitialContents(result.getContents())
                    .build(mGoogleApiClient);
            try {
                startIntentSenderForResult(
                        intentSender, REQUEST_CODE_CREATOR, null, 0, 0, 0);
            } catch (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)
        .addScope(Drive.SCOPE_FILE)
        .addConnectionCallbacks(this)
        .addOnConnectionFailedListener(this)
        .build();
    }
    // Connect the client. Once connected, the camera is launched.
    mGoogleApiClient.connect();
}

@Override
protected void onPause() {
    if (mGoogleApiClient != null) {
        mGoogleApiClient.disconnect();
    }
    super.onPause();
}
@Override
public void onBackPressed(){
    Toast.makeText(MainActivity.this,"Going Somehwere?", Toast.LENGTH_LONG).show();
    finish();
}
@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.
        GooglePlayServicesUtil.getErrorDialog(result.getErrorCode(), this, 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 (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");
}

person Anish Ubais    schedule 09.07.2014    source источник


Ответы (2)


Это должно решить вашу проблему с миниатюрами. По сути, замените растровое изображение mBitmapToSave файлом '_picFl'. Приведенный ниже код изменен, имена переменных другие, но по сути он делает то, что вы просите.

private File            _picFl;
private GoogleApiClient _gac;

@Override public void onConnected(Bundle connectionHint) {
    if (_picFl == null)
      takePic();
    else 
      save2GooDrv();
  }
  //-----------------------------------------------------------------------------------------------
  private void takePic() {
    Intent icIt = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
    if (icIt.resolveActivity(getPackageManager()) != null) try {
      _picFl =  new File(getCcheDir(), tm2FlNm(null));
      if (_picFl != null) {
        icIt.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(_picFl));
        startActivityForResult(icIt, RC_GETIMAGE);
      }
    } catch (Exception e) {le(e);}
  }
  //-----------------------------------------------------------------------------------------------
  private synchronized void save2GooDrv() {
    Drive.DriveApi.newContents(_gac).setResultCallback(new ResultCallback<ContentsResult>() {
      @Override public void onResult(ContentsResult rslt) {
        if (rslt.getStatus().isSuccess()) try {
          OutputStream os = rslt.getContents().getOutputStream();
          os.write(file2Bytes(_picFl));
          MetadataChangeSet metadataChangeSet = new MetadataChangeSet.Builder()
                                   .setMimeType("image/jpeg").setTitle(_picFl.getName()).build();
          _picFl.delete();
          _picFl = null;

          IntentSender intentSender = Drive.DriveApi
              .newCreateFileActivityBuilder()
              .setInitialMetadata(metadataChangeSet)
              .setInitialContents(rslt.getContents())
              .build(_gac);
          try {
            startIntentSenderForResult( intentSender, RC_CREATOR, null, 0, 0, 0);
          } catch (SendIntentException e) {le(e);}
        } catch (Exception e) {le(e);}
      }
    });
  }

  //***********************************************************************************************
  public String getCcheDir() {
    Context actx = getApplicationContext();

    return Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState()) ||
                                                      !Environment.isExternalStorageRemovable() ?
     actx.getExternalCacheDir().getPath() : actx.getCacheDir().getPath();
  }

  //***********************************************************************************************
  public byte[] file2Bytes(File file) {
    byte[] buf = null;
    RandomAccessFile raFl = null;
    if (file != null) try {
      raFl = new RandomAccessFile(file, "r");
      buf = new byte[(int)raFl.length()];
      raFl.readFully(buf);
    } catch (Exception e) {le(e);}
    finally { 
       if (raFl != null) try { 
         raFl.close();
       } catch (Exception e) {le(e);}
     }
    return buf;
  }

  //***********************************************************************************************
  public String tm2FlNm(Long milis) {       // time -> yymmdd-hhmmss
    try {
      return  new SimpleDateFormat("yyMMdd-HHmmss",Locale.US)
                                          .format((milis == null) ? new Date() : new Date(milis));
    } catch (Exception e) {le(e);}
    return null;
  }  

  //***********************************************************************************************
  public void le(Exception e){
    try { 
      Log.e("_", (e==null) ? "NULL" : Log.getStackTraceString(e));
    }catch (Exception f) { try { Log.e("_", "ERR on err");}  catch (Exception g) {} }
  }
person seanpj    schedule 11.07.2014
comment
Спасибо, seanpj. Этот код сработал. Я хочу каждый раз загружать изображения в папку, созданную из моего приложения. Поэтому каждый раз, когда приложение должно проверять, существует ли папка с определенным именем. Если папка найдена, загрузите ее, иначе создайте папку и загрузить. Я попробовал свой собственный код, но из-за проблемы с синхронизацией я не могу добиться требуемого результата. Любое решение, пожалуйста? - person Anish Ubais; 19.07.2014
comment
Я уже несколько месяцев катаюсь по этому тонкому льду. Моя первоначальная идея заключалась в следующем: 1/обновить файл конфигурации в папке приложения, использовать requestSync() 2/получить уведомление на других устройствах (с тем же приложением) через «DriveEvent.Listener». Не работает. Вы получаете уведомление «когда-нибудь, когда GOOPlaySvcs решит». Вам лучше опросить Диск. И вы должны сделать это, используя «старый» RESTful API. - person seanpj; 19.07.2014
comment
Использование REST и GDAA приводит к целому ряду проблем со временем, поскольку GDAA кэширует и управляет связью с Drive асинхронно. Кроме того, GDAA поддерживает только область FILE, не имеет DELETE, не дает вам ссылку на миниатюру, не поддерживает полнотекстовый поиск. Большинство из этих вещей были обещаны полгода назад и так и не были доставлены. По SA куча вопросов (22515028,22874657,23073474,24562387,23711686,23626827) - person seanpj; 19.07.2014

Вот быстрое решение вашей проблемы. Кнопка «Назад» в обоих действиях, о которых вы говорите (камера, создатель), возвращает «Activity.RESULT_CANCELED», поэтому просто убейте свою активность (используя finish()), когда вы не получите «Activity.RESULT_OK».

  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");
      } else 
        finish();
      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);
      } else 
        finish();
      break;
  }

Но в целом «быстрый старт» — это просто проверка концепции, а не то, на чем вы должны строить свое приложение.

person seanpj    schedule 11.07.2014
comment
спасибо, это работает!!. У меня есть еще одно сомнение. В настоящее время загруженное изображение имеет небольшой размер. Вероятно, потому, что код содержит сжатие изображения (image.compress (Bitmap.CompressFormat.PNG, 100, bitmapStream);). Так что вы можете подскажите как сохранить картинку в исходном размере,без сжатия? - person Anish Ubais; 11.07.2014
comment
Get(data) дает вам только миниатюру. Если вам нужен полный размер, вам нужно создать файл, передать его в действие камеры, а затем захватить его, в нем есть изображение. Я прикреплю код к другому ответу ниже - person seanpj; 11.07.2014