15

From within my application, I'm trying to create an email that contains an image contained in a bitmap object.

private void sendEmailWithBitmapAttached(){ 

    final Intent emailIntent = new Intent(android.content.Intent.ACTION_SEND);
    emailIntent.setType("plain/text");
    emailIntent.putExtra(android.content.Intent.EXTRA_SUBJECT, "Email Subject");
    emailIntent.putExtra(android.content.Intent.EXTRA_TEXT, "Email Body");
    emailIntent.setType("image/png");
    ContentResolver cr = getContentResolver();

    // insert the image and create a path
    String imageBitmapPath = MediaStore.Images.Media.insertImage(cr, bitmapForEmail,"title", "description");

    // create a uri
    Uri imageUri = Uri.parse(imageBitmapPath);
    emailIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    emailIntent.putExtra(Intent.EXTRA_STREAM, imageUri);

    // send the email
    startActivity(Intent.createChooser(emailIntent, "Send mail..."));
}

This works fine in Android 2.3.

But using later versions, it produces the following error:

07-13 23:01:01.252: E/MediaStore(5194): Failed to insert image
 07-13 23:01:01.252: E/MediaStore(5194): java.lang.SecurityException: 
     Permission Denial: 
         writing com.android.providers.media.MediaProvider 
         uri content://media/external/images/media from 
             pid=5194, uid=10151 requires 
             android.permission.WRITE_EXTERNAL_STORAGE, 
             or grantUriPermission()

So, taking the suggestion of the error message, I tried to grantUriPermission.

grantUriPermission(String toPackage, Uri uri, int modeFlags)

But I am not sure what to put for toPackage or uri

But again, using the error message, I modified the code as follows:

private void sendEmailWithBitmapAttached(){ 

    final Intent emailIntent = new Intent(android.content.Intent.ACTION_SEND);
    emailIntent.setType("plain/text");
    emailIntent.putExtra(android.content.Intent.EXTRA_SUBJECT, "Email Subject");
    emailIntent.putExtra(android.content.Intent.EXTRA_TEXT, "Email Body");
    emailIntent.setType("image/png");
    ContentResolver cr = getContentResolver();

    // create a Uri for the content provider suggested by the error message
    Uri uri = Uri.parse("content://media/external/images/media");

    // create a package provider string suggested by the error messge.
    String provider = "com.android.providers.media.MediaProvider";


    // grant all three uri permissions!
    grantUriPermission(provider, uri, Intent.FLAG_GRANT_READ_URI_PERMISSION);
    grantUriPermission(provider, uri, Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
    grantUriPermission(provider, uri, Intent.FLAG_GRANT_PERSISTABLE_URI_PERMISSION);

    // insert the image and create a path
    String imageBitmapPath = MediaStore.Images.Media.insertImage(cr, bitmapForEmail,"title", "description");

    // create a uri
    Uri imageUri = Uri.parse(imageBitmapPath);
    emailIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    emailIntent.putExtra(Intent.EXTRA_STREAM, imageUri);

    // send the email
    startActivity(Intent.createChooser(emailIntent, "Send mail..."));
}

And I get the exact same error.

Can a kind soul please give me a hint as how to take care of grantUriPermission's uri and provider items? Is this the correct approach?

Thank you very much for ANY help, hint, guides, or suggestion, you can provide!

user3835673
  • 171
  • 1
  • 1
  • 6

3 Answers3

13

Try including this in your AndroidManifest.xml.

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

On Android 6.0+ the permission must be granted by the user to the application otherwise this will throw the SecurityException.

To do this go into Settings/Apps/[AppName]/Permissions and allow the Storage Permission.

Guy Davis
  • 63
  • 1
  • 5
Badrul
  • 1,582
  • 4
  • 16
  • 27
7

I solved my issue by also adding permission flags to the intent, like this:

intent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);

More details can be found in documentation: https://developer.android.com/reference/android/support/v4/content/FileProvider.html#Permissions

brunaru
  • 86
  • 1
  • 2
  • is it possible to use it without setting flag? – Sagar Aug 12 '19 at 04:06
  • Is it possible to use without using an intent? My app just needs to access cache files of my other app to save on network requests... I want this to happen silently in the background. – JCutting8 Aug 04 '22 at 11:12
3

I had a similar issue. Below is how I solved the problem for my project. You should be able to adapt this for your solution. This solution also has some Firebase code, which you can ignore. The key points are ActivityCompat.requestPermissions and ActivityCompat.checkSelfPermission:

private void shareViaEmail() {
    int permissionCheck = ContextCompat.checkSelfPermission(this, android.Manifest.permission.WRITE_EXTERNAL_STORAGE);
    if (permissionCheck != PackageManager.PERMISSION_GRANTED) {
        if (ActivityCompat.shouldShowRequestPermissionRationale(this, android.Manifest.permission.WRITE_EXTERNAL_STORAGE)) {
            AlertDialog.Builder alertBuilder = new AlertDialog.Builder(this);
            LayoutInflater inflater = this.getLayoutInflater();
            final ViewGroup nullParent = null;
            final View dialogView = inflater.inflate(R.layout.alert_dialog, nullParent);
            alertBuilder.setView(dialogView);

            alertBuilder.setCancelable(true);
            alertBuilder.setTitle("Permission request");

            String message = "\n\n" + getString(R.string.email_images);
            AppCompatTextView notice = (AppCompatTextView) dialogView.findViewById(R.id.notice);
            if (notice != null) {
                notice.setText(message);
                notice.setTextSize(getResources().getInteger(R.integer.dialog_text_size));
            }
            else {
                alertBuilder.setMessage(message);
            }

            alertBuilder.setPositiveButton(android.R.string.yes, new DialogInterface.OnClickListener() {

                @TargetApi(Build.VERSION_CODES.JELLY_BEAN)
                public void onClick(DialogInterface dialog, int which) {
                    mShowingAlert = false;
                    ActivityCompat.requestPermissions(baseActivity(), new String[]{android.Manifest.permission.WRITE_EXTERNAL_STORAGE}, MY_TARGET_WRITE_PERMISSION_REQUEST);
                }
            });

            mAlertDialog = alertBuilder.create();
            mAlertDialog.show();
            return;
        }
    }
    ActivityCompat.requestPermissions(this,
            new String[]{android.Manifest.permission.WRITE_EXTERNAL_STORAGE},
            MY_TARGET_WRITE_PERMISSION_REQUEST);
}

@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
    super.onRequestPermissionsResult(requestCode, permissions, grantResults);
    if (ActivityCompat.checkSelfPermission(this, android.Manifest.permission.WRITE_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED) {
        if (requestCode == MY_TARGET_WRITE_PERMISSION_REQUEST) {
            doShareViaEmail();
        } else {
            failWriteImage();
        }
    }
}

protected void failWriteImage() {
    getHandler().post(new Runnable() {
        @Override

        public void run() {
            String email_failed = getResources().getString(R.string.fail_email_attach);
            @SuppressLint("ShowToast") Toast toast = Toast.makeText(getApplicationContext(), email_failed, Toast.LENGTH_SHORT);
            new CustomToast(toast).invoke();
        }
    });
}

protected void doShareViaEmail() {
    FireUtilities fireUtilities = FireUtilities.getInstance();
    fireUtilities.logEvent(mFirebaseAnalytics, "option_item", "share", "share_string");

    Intent intent = new Intent(Intent.ACTION_SEND_MULTIPLE);
    intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    intent.putExtra(Intent.EXTRA_SUBJECT, getString(R.string.string_share));

    String appStoreLink = getString(R.string.app_store_link);

    String emailText = getString(R.string.share_string_body1)
            + " " + appStoreLink
            + " " + getString(R.string.share_string_body2);

    emailText = emailText + "<br><br>" + getTitleOfSegment(true, mCurrentSegment, mCurrentSegmentIndex);

    Bitmap targetImage = screenShot(mTargetImageView);
    if (targetImage != null) {
        ArrayList<Uri> files = new ArrayList<Uri>();
        String path = MediaStore.Images.Media.insertImage(getContentResolver(), targetImage, "string", null);
        Uri targetUri = Uri.parse(path);
        files.add(targetUri);
        intent.putParcelableArrayListExtra(Intent.EXTRA_STREAM, files);
    }

    intent.setType("message/rfc822");

    Spanned htmlText = fromHtml(getString(R.string.share_score_head1)
            + emailText + getString(R.string.share_score_head2));
    intent.putExtra(Intent.EXTRA_TEXT, htmlText);

    try {
        startActivityForResult(Intent.createChooser(intent, "Email:"), 1234);
    } catch (final android.content.ActivityNotFoundException e) {
        String no_email_client = getResources().getString(R.string.no_email_client);
        @SuppressLint("ShowToast") Toast toast = Toast.makeText(getApplicationContext(), no_email_client, Toast.LENGTH_LONG);
        new CustomToast(toast).invoke();
    }
}

public void onActivityResult(int requestCode, int resultCode, Intent data) {
    if (requestCode == 1234) {
        LogHelper.v(TAG, "e-mail successfully sent");
    }
}

public Bitmap screenShot(View view) {
    if (view != null) {
        Bitmap bitmap = Bitmap.createBitmap(view.getWidth(),
                view.getHeight(), Bitmap.Config.ARGB_8888);
        Canvas canvas = new Canvas(bitmap);
        view.draw(canvas);
        return bitmap;
    }
    return null;
}
Lee Hounshell
  • 842
  • 9
  • 10