Share image without WRITE_EXTERNAL_STORAGE?

2019-02-23 00:57发布

Is there a way to use Intent.ACTION_SEND to share a screenshot without requiring android.permission.WRITE_EXTERNAL_STORAGE?

Here's the share part:

    Intent shareIntent = new Intent(Intent.ACTION_SEND);
    shareIntent.setType("image/jpeg");
    shareIntent.putExtra(Intent.EXTRA_STREAM, uri);
    Intent chooserIntent = Intent.createChooser(shareIntent, shareTitle);
    startActivity(chooserIntent);

The share works fine when the uri points to a file in getExternalFilesDir(), but I'd prefer a solution that does not require the WRITE_EXTERNAL_STORAGE permission for privacy-concerned users.

I've tried 3 different approaches:

  1. File Provider:

    uri = FileProvider.getUriForFile(context, authority, imageFile);
    

This works for some share styles (Gmail) but not others (Google+).

  1. Upload to a web server:

    uri = Uri.parse("http://my-image-host.com/screenshot.jpg");
    

This fails everywhere, crashing some (Google+).

(I suspect this could work if I implemented sharing logic myself using per-social-network APIs instead of the chooserIntent)

  1. Inject into Media Store:

    uri = MediaStore.Images.Media.insertImage(contentResolver, bitmap, name, description);
    

This throws a SecurityException explaining it requires WRITE_EXTERNAL_STORAGE.

Are there other approaches I'm missing?

1条回答
爷的心禁止访问
2楼-- · 2019-02-23 01:31

Based on work by Stefan Rusek, I created LegacyCompatCursorWrapper, designed to help improve compatibility of FileProvider (and other ContentProvider implementations) with client apps that are looking for _DATA columns and not finding them. The _DATA pattern was used originally by MediaStore, but it was never a good idea for apps to try referring to that column.

To use it in conjunction with FileProvider, add my CWAC-Provider library as a dependency, and then create your own subclass of FileProvider, such as this one:

/***
 Copyright (c) 2015 CommonsWare, LLC
 Licensed under the Apache License, Version 2.0 (the "License"); you may not
 use this file except in compliance with the License. You may obtain a copy
 of the License at http://www.apache.org/licenses/LICENSE-2.0. Unless required
 by applicable law or agreed to in writing, software distributed under the
 License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS
 OF ANY KIND, either express or implied. See the License for the specific
 language governing permissions and limitations under the License.

 From _The Busy Coder's Guide to Android Development_
 http://commonsware.com/Android
 */

package com.commonsware.android.cp.v4file;

import android.database.Cursor;
import android.net.Uri;
import android.support.v4.content.FileProvider;
import com.commonsware.cwac.provider.LegacyCompatCursorWrapper;

public class LegacyCompatFileProvider extends FileProvider {
  @Override
  public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) {
    return(new LegacyCompatCursorWrapper(super.query(uri, projection, selection, selectionArgs, sortOrder)));
  }
}

All this does is wrap the FileProvider query() results in a LegacyCompatCursorWrapper. The rest of your app configuration would be identical to using FileProvider directly (e.g., <meta-data> element), except that your <activity> element's android:name attribute would point to your own class. You can see this in action in this sample app.

查看更多
登录 后发表回答