简体   繁体   中英

Crop image from gallery crashes app - Android

I have a code that can crop an image from either the gallery or by taking one with the camera and then cropping it. My code works when I take a photo with the camera and then crop it but if I choose to crop one from the gallery it crashes once I choose what crop app to use. Why does it crash when I do this? It works great if I use just the camera.

My code consists of three Java files.

Main

package com.goboapp;

import java.io.File;
import java.io.InputStream;
import java.net.URL;
import java.util.ArrayList;
import java.util.List;

import android.app.AlertDialog;
import android.content.ActivityNotFoundException;
import android.content.ComponentName;
import android.content.DialogInterface;
import android.content.Intent;
import android.content.pm.ResolveInfo;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.net.Uri;
import android.os.AsyncTask;
import android.os.Bundle;
import android.os.Environment;
import android.provider.MediaStore;
import android.support.v7.app.ActionBarActivity;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.widget.ArrayAdapter;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.Toast;


public class Main extends ActionBarActivity {
    private Uri mImageCaptureUri;

    private static final int PICK_FROM_CAMERA = 1;
    private static final int CROP_FROM_CAMERA = 2;
    private static final int PICK_FROM_FILE = 3;

     ImageView imageView1;
     RoundImage roundedImage;
     Bitmap bitmap;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        imageView1 = (ImageView) findViewById(R.id.imageView1);


        //Camera Stuff

        final String [] items           = new String [] {"Take from camera", "Select from gallery"};                
        ArrayAdapter<String> adapter    = new ArrayAdapter<String> (this, android.R.layout.select_dialog_item,items);
        AlertDialog.Builder builder     = new AlertDialog.Builder(this);

        builder.setTitle("Select Image");
        builder.setAdapter( adapter, new DialogInterface.OnClickListener() {
            public void onClick( DialogInterface dialog, int item ) { //pick from camera
                if (item == 0) {
                    Intent intent    = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

                    mImageCaptureUri = Uri.fromFile(new File(Environment.getExternalStorageDirectory(),
                                       "tmp_avatar_" + String.valueOf(System.currentTimeMillis()) + ".jpg"));

                    intent.putExtra(android.provider.MediaStore.EXTRA_OUTPUT, mImageCaptureUri);

                    try {
                        intent.putExtra("return-data", true);

                        startActivityForResult(intent, PICK_FROM_CAMERA);
                    } catch (ActivityNotFoundException e) {
                        e.printStackTrace();
                    }
                } else { //pick from file
                    Intent intent = new Intent();

                    intent.setType("image/*");
                    intent.setAction(Intent.ACTION_GET_CONTENT);

                    startActivityForResult(Intent.createChooser(intent, "Complete action using"), PICK_FROM_FILE);
                }
            }
        } );

        final AlertDialog dialog = builder.create();

        Button button   = (Button) findViewById(R.id.btn_crop);

        button.setOnClickListener(new View.OnClickListener() {  
            @Override
            public void onClick(View v) {
                dialog.show();
            }
        });

        //END CAMERA STUFF

         }// End OnCreate





    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.main, menu);
        return true;
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        // Handle action bar item clicks here. The action bar will
        // automatically handle clicks on the Home/Up button, so long
        // as you specify a parent activity in AndroidManifest.xml.
        int id = item.getItemId();
        if (id == R.id.action_settings) {
            return true;
        }
        return super.onOptionsItemSelected(item);
    }//end onOptionsItemSelected

    //CAMERA STUFF

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        if (resultCode != RESULT_OK) return;

        switch (requestCode) {
            case PICK_FROM_CAMERA:
                doCrop();

                break;

            case PICK_FROM_FILE: 
                mImageCaptureUri = data.getData();

                doCrop();

                break;          

            case CROP_FROM_CAMERA:          
                Bundle extras = data.getExtras();

                if (extras != null) {               
                    Bitmap photo = extras.getParcelable("data");

                   imageView1.setImageBitmap(photo);
                }

                File f = new File(mImageCaptureUri.getPath());            

                if (f.exists()) f.delete();

                break;

        }
    }

      private void doCrop() {
        final ArrayList<CropOption> cropOptions = new ArrayList<CropOption>();

        Intent intent = new Intent("com.android.camera.action.CROP");
          intent.setType("image/*");

          List<ResolveInfo> list = getPackageManager().queryIntentActivities( intent, 0 );

          int size = list.size();

          if (size == 0) {          
            Toast.makeText(this, "Can not find image crop app", Toast.LENGTH_SHORT).show();

              return;
          } else {
            intent.setData(mImageCaptureUri);

              intent.putExtra("outputX", 200);
              intent.putExtra("outputY", 200);
              intent.putExtra("aspectX", 1);
              intent.putExtra("aspectY", 1);
              intent.putExtra("scale", true);
              intent.putExtra("return-data", true);

            if (size == 1) {
                Intent i        = new Intent(intent);
                ResolveInfo res = list.get(0);

                i.setComponent( new ComponentName(res.activityInfo.packageName, res.activityInfo.name));

                startActivityForResult(i, CROP_FROM_CAMERA);
            } else {
                for (ResolveInfo res : list) {
                    final CropOption co = new CropOption();

                    co.title    = getPackageManager().getApplicationLabel(res.activityInfo.applicationInfo);
                    co.icon     = getPackageManager().getApplicationIcon(res.activityInfo.applicationInfo);
                    co.appIntent= new Intent(intent);

                    co.appIntent.setComponent( new ComponentName(res.activityInfo.packageName, res.activityInfo.name));

                    cropOptions.add(co);
                }

                CropOptionAdapter adapter = new CropOptionAdapter(getApplicationContext(), cropOptions);

                AlertDialog.Builder builder = new AlertDialog.Builder(this);
                builder.setTitle("Choose Crop App");
                builder.setAdapter( adapter, new DialogInterface.OnClickListener() {
                    public void onClick( DialogInterface dialog, int item ) {
                        startActivityForResult( cropOptions.get(item).appIntent, CROP_FROM_CAMERA);
                    }
                });

                builder.setOnCancelListener( new DialogInterface.OnCancelListener() {
                    @Override
                    public void onCancel( DialogInterface dialog ) {

                        if (mImageCaptureUri != null ) {
                            getContentResolver().delete(mImageCaptureUri, null, null );
                            mImageCaptureUri = null;
                        }
                    }
                } );

                AlertDialog alert = builder.create();

                alert.show();
            }
          }
    }

      //END CAMERA STUFF




}//end activity

CropOption

package com.goboapp;

import android.content.Intent;
import android.graphics.drawable.Drawable;

public class CropOption {
    public CharSequence title;
    public Drawable icon;
    public Intent appIntent;
}

CropOptionAdapter

package com.goboapp;

import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;

import android.widget.ArrayAdapter;
import android.widget.ImageView;
import android.widget.TextView;

import android.content.Context;

import java.util.ArrayList;

public class CropOptionAdapter extends ArrayAdapter<CropOption> {
    private ArrayList<CropOption> mOptions;
    private LayoutInflater mInflater;

    public CropOptionAdapter(Context context, ArrayList<CropOption> options) {
        super(context, R.layout.crop_selector, options);

        mOptions    = options;

        mInflater   = LayoutInflater.from(context);
    }

    @Override
    public View getView(int position, View convertView, ViewGroup group) {
        if (convertView == null)
            convertView = mInflater.inflate(R.layout.crop_selector, null);

        CropOption item = mOptions.get(position);

        if (item != null) {
            ((ImageView) convertView.findViewById(R.id.iv_icon)).setImageDrawable(item.icon);
            ((TextView) convertView.findViewById(R.id.tv_name)).setText(item.title);

            return convertView;
        }

        return null;
    }
}

The XML files I use are

activity_main

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context="com.goboapp.Main" >

    <ImageView
        android:id="@+id/imageView1"
        android:layout_width="90dp"
        android:layout_height="90dp"
        android:scaleType="centerCrop"
        android:layout_gravity="center" />

    <Button 
        android:id="@+id/btn_crop"
        android:layout_width="match_parent"          
        android:layout_height="wrap_content"
        android:text="Select image"
        android:layout_marginTop="10dp"
        android:layout_below="@id/imageView1"/>

</RelativeLayout>

crop_selector

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:padding="10dp"
    android:gravity="center_vertical">

    <ImageView
        android:id="@+id/iv_icon"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"/>

    <TextView
        android:id="@+id/tv_name"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text=""/>
</LinearLayout>

Logcat

11-14 00:33:14.164: E/AndroidRuntime(1755): FATAL EXCEPTION: AsyncTask #1
11-14 00:33:14.164: E/AndroidRuntime(1755): Process: com.motorola.MotGallery2, PID: 1755
11-14 00:33:14.164: E/AndroidRuntime(1755): java.lang.RuntimeException: An error occured while executing doInBackground()
11-14 00:33:14.164: E/AndroidRuntime(1755):     at android.os.AsyncTask$3.done(AsyncTask.java:300)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at java.util.concurrent.FutureTask.finishCompletion(FutureTask.java:355)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at java.util.concurrent.FutureTask.setException(FutureTask.java:222)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at java.util.concurrent.FutureTask.run(FutureTask.java:242)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at android.os.AsyncTask$SerialExecutor$1.run(AsyncTask.java:231)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1112)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:587)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at java.lang.Thread.run(Thread.java:841)
11-14 00:33:14.164: E/AndroidRuntime(1755): Caused by: java.lang.SecurityException: Permission Denial: opening provider com.android.providers.media.MediaDocumentsProvider from ProcessRecord{44b96898 1755:com.motorola.MotGallery2/u0a40} (pid=1755, uid=10040) requires android.permission.MANAGE_DOCUMENTS or android.permission.MANAGE_DOCUMENTS
11-14 00:33:14.164: E/AndroidRuntime(1755):     at android.os.Parcel.readException(Parcel.java:1471)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at android.os.Parcel.readException(Parcel.java:1425)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at android.app.ActivityManagerProxy.getContentProvider(ActivityManagerNative.java:2896)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at android.app.ActivityThread.acquireProvider(ActivityThread.java:4491)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at android.app.ContextImpl$ApplicationContentResolver.acquireUnstableProvider(ContextImpl.java:2323)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at android.content.ContentResolver.acquireUnstableProvider(ContentResolver.java:1425)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at android.content.ContentResolver.openTypedAssetFileDescriptor(ContentResolver.java:1047)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at android.content.ContentResolver.openAssetFileDescriptor(ContentResolver.java:904)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at android.content.ContentResolver.openInputStream(ContentResolver.java:629)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at com.android.gallery3d.filtershow.cache.ImageLoader.loadBitmap(ImageLoader.java:331)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at com.android.gallery3d.filtershow.cache.ImageLoader.loadBitmapBounds(ImageLoader.java:306)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at com.android.gallery3d.filtershow.cache.ImageLoader.loadConstrainedBitmap(ImageLoader.java:359)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at com.android.gallery3d.filtershow.crop.CropActivity$LoadBitmapTask.doInBackground(CropActivity.java:277)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at com.android.gallery3d.filtershow.crop.CropActivity$LoadBitmapTask.doInBackground(CropActivity.java:261)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at android.os.AsyncTask$2.call(AsyncTask.java:288)
11-14 00:33:14.164: E/AndroidRuntime(1755):     at java.util.concurrent.FutureTask.run(FutureTask.java:237)
11-14 00:33:14.164: E/AndroidRuntime(1755):     ... 4 more

Thats my entire app.

The app works when I choose to take a picture with my camera, select a crop app and then save the photo.

The app DOES NOT WORK when I choose Get a photo from the Gallery. It opens my gallery and I select an image and then I choose what app I want to use to crop the image and then the app crashes.

There is a problem with Kitkat while using android.permission.MANAGE_DOCUMENTS .You need do this work around.

Try this code:

  public static final String KITKAT_VALUE = 1002;

  Intent intent;

  if (Build.VERSION.SDK_INT < 19){
       intent = new Intent();
       intent.setAction(Intent.ACTION_GET_CONTENT);
       intent.setType("*/*");
       startActivityForResult(intent, KITKAT_VALUE);
  } 
  else 
  {
       intent = new Intent(Intent.ACTION_OPEN_DOCUMENT);
       intent.addCategory(Intent.CATEGORY_OPENABLE);
       intent.setType("*/*");
       startActivityForResult(intent, KITKAT_VALUE);
  }

@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
     if (requestCode == KITKAT_VALUE ) {
         if (resultCode == Activity.RESULT_OK) {
         // do your stuff
         }
     }
}

If this also doesn't work, Other work around
Reference : Android Gallery on KitKat returns different Uri for Intent.ACTION_GET_CONTENT

Hope it will help you :)

Logcat says:

11-14 00:33:14.164: E/AndroidRuntime(1755): Caused by: java.lang.SecurityException: Permission Denial: opening provider com.android.providers.media.MediaDocumentsProvider from ProcessRecord{44b96898 1755:com.motorola.MotGallery2/u0a40} (pid=1755, uid=10040) requires android.permission.MANAGE_DOCUMENTS or android.permission.MANAGE_DOCUMENTS

You need to add this permission in manifest under application tag:

<uses-permission android:name="android.permission.MANAGE_DOCUMENTS" />

This will eliminate crash.

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM