Android: How do I attach a temporary, generated image to an email?

11,499

Solution 1

My problem really consisted of two parts:

  1. context.getCacheDir() is private to your app. You can't put something there and expect another app to be able to access it.
  2. I misunderstood what MIME type I should have been using. Even though I was sending email text, I really needed to specify image/png for the sake of my attachment.

Additionally, research indicated that putting (potentially large) images on the primary memory was not a good idea, even if you were going to immediately clean it up.

Once I did these things and wrote my generated images to a public location on the SD Card, it worked just fine.

So, in overview:

Request SD Card Access in your manifest

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

Make sure SD Card is available

if (!Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState())) 
{
    //Bail gracefully
}

Create a directory on the SD Card

File pngDir = new File(
    Environment.getExternalStorageDirectory(),   
    //Loose convention inferred from app examples
    "Android/data/com.somedomain.someapp/flotsam");

if (!pngDir.exists())
    pngDir.mkdirs();

Write your file to that directory and capture the Uri

File pngFile = new File(pngDir, "jetsam.png");
//Save file encoded as PNG
Uri pngUri = Uri.fromFile(pngFile);

Build an ACTION_SEND intent

Intent intent = new Intent(android.content.Intent.ACTION_SEND);
intent.setType("image/png"); //
intent.putExtra(android.content.Intent.EXTRA_EMAIL, new String[] { "[email protected]" });
intent.putExtra(android.content.Intent.EXTRA_SUBJECT, "Portable Network Graphics");
intent.putExtra(android.content.Intent.EXTRA_CC, new String[] { "[email protected]" });
intent.putExtra(Intent.EXTRA_TEXT, "Something textual");
intent.putExtra(Intent.EXTRA_STREAM, pngUri);

And then start the activity

context.startActivity(Intent.createChooser(intent, "Something Pithy"));

And then make sure you clean everything up...

Caveat 1

There appears to be more support coming for app-specific SD Card directories, but alas, not in my required SDK version.

Caveat 2

This is an overview of the solution that eventually worked for me. It is not necessarily a "best practice" approach.

Caveat 3

This does mean that the application has to have an SD Card mounted in order to have the image attachments feature available, but this was totally acceptable for my use case. Your mileage may vary. If the SD Card is not available, I append a friendly note to the email explaining why the images could not be attached and how to rectify the situation.

Solution 2

I've just run into exactly the same issue (wanting to attach a text file in my case). If you look in the Android log, the reason for it is:

02-28 21:01:28.434: E/Gmail(19673): file:// attachment paths must point to file:///mnt/sdcard. Ignoring attachment file:///data/data/com.stephendnicholas.gmailattach/cache/Test.txt

As a workaround (as mentioned by HRJ), you can use a ContentProvider to provide access to files in your application's internal cache so that Gmail can attach them. I've just written up a blog post on how to do it.

Hopefully that's of some help :)

Share:
11,499
el2iot2
Author by

el2iot2

A unicycle gives you an opportunity to fall in every direction, so balance is key.

Updated on June 05, 2022

Comments

  • el2iot2
    el2iot2 almost 2 years

    I have a programmatically generated image that I want to send as an attachment via the ACTION_SEND and EXTRA_STREAM method.

    But how do i do this?

    My first attempt (writing to my context.getCacheDir() based file path) appeared to work in the Gmail preview (no image preview, but attached file name and icon was visible), but the attachment never arrived on the recipient side. I guess this has something to do with permissions on the generated file, but how to avoid this? Do I need to set more permissive settings on these generated files (so that the Gmail activity can access)? Is that even possible for the app's cache folder?

    Is there another file location that would be more suitable to write my files to? I considered the downloads folder, but think it would be an awkward location for something that only needs to exist until it has been emailed.

    I have even tried encoding my image purely in a data:image/png;base64,ABCD... style URI. This, too, showed up in Gmail preview (attachment icon, but no file name), but did not result in a recipient-side attachment.

    Has anyone been able to attach a one-shot generated image to an email intent by any means? What options may I have overlooked?

  • HRJ
    HRJ over 12 years
    Could this be done with a custom ContentProvider. It won't require an extra permission, and also remove dependency on SD card.
  • el2iot2
    el2iot2 over 12 years
    @HRJ can you elaborate on this possibility with an answer? I'd love to see more detail on how this can be done...
  • Paschalis
    Paschalis almost 12 years
    this solution dont works for me. most strange thing is that images file is created on sd card!
  • el2iot2
    el2iot2 almost 12 years
    @Paschalis what version(s) of android are you targeting? It has been a while since I was using this solution...
  • Paschalis
    Paschalis almost 12 years
    target is 4.1 @devfuel . min sdk 2.0, max 4.1, target 4.1
  • Carlos
    Carlos almost 10 years
    ContentProvider was the solution I went with. I didn't want everyone to be able to read my files from the SD card.