Android調用系統圖片分享以及解決Android11上分享圖片失敗的問題

至於調用系統分享這個功能就很簡單了,直接上代碼,主要如果項目已經使用了分區的概念,那麼在Android 11手機上面分享就會提示"獲取資源失敗"的情況

    //分享文字
    public static void shareText(Context context, String text, String title) {
        new Handler(Looper.getMainLooper()).post(new Runnable() {
            @Override
            public void run() {
                Intent intent = new Intent(Intent.ACTION_SEND);
                intent.setType("text/plain");
                intent.putExtra(Intent.EXTRA_TEXT, text);
                context.startActivity(Intent.createChooser(intent, title));
            }
        });}
    //分享圖片
    public static void shareImage(Context context, Uri uri, String title) {
        Intent intent = new Intent(Intent.ACTION_SEND);
        intent.setType("image/png");
        intent.putExtra(Intent.EXTRA_STREAM, uri);
        context.startActivity(Intent.createChooser(intent, title));
    }

1.先分析問題原因

 * Android 10 中,Google首次引入了分區存儲,將公共區域劃分成了不同的集合,並且在媒體文件和其他文檔之間建立了清楚的分割。經過劃分之後應用不可以隨意訪問外部存儲區中的文件,而只能訪問媒體文件。
 * 1.應用私有目錄:存儲應用私有數據,外部存儲應用私有目錄對應Android/data/packagename,內部存儲應用私有目錄對應data/data/packagename;
 * 2.共享目錄:存儲其他應用可訪問文件, 包含媒體文件、文檔文件以及其他文件,對應設備DCIM、Pictures、Alarms, Music, Notifications,Podcasts, Ringtones、Movies、Download等目錄。
 * Android 11 (API 級別 30) 進一步增強了平臺功能,爲外部存儲中的應用和用戶數據提供了更好的保護。
 * 從 Android 11 開始,使用 分區存儲模式 的應用即使擁有 READ_EXTERNAL_STORAGE 權限,也無法再訪問外部存儲中的任何其他應用的 專屬目錄 中的文件,受此影響應用間分享就需要使用通過應用間共享文件適配的方式( FileProvider進行分享),通過FileProvider,就允許第三方應用讀取你的應用所分享的文件,而不會受到分區存儲的限制。

首先分享應用數據給第三方應用需要在用到文件共享,就需要在AndroidManifest.xml中配置FileProvider,但是配置了只代表你可以使用FileProvider存儲了應用間共享文件,但是在並不代表你使用了FileProvider(後邊會講怎麼使用)。

2.如何使用FileProvider

2.1 首先,在項目的AndroidManifest.xml添加相關配置,示例如下:
<!--
    1.applicationId: 應用包名;
    2.android:exported:要求必須爲false,不需要暴露組件;
    3.android:grantUriPermissions:true,表示授予URI臨時訪問權限;
    4.android:authorities這個屬性的值,建議寫包名+fileprovider,當然也可以起別的字符串, 但是在設備中不能出現2個及以上的APP使用到同一個authorities屬性值,因爲無法共存;
  -->
<provider
    android:name="androidx.core.content.FileProvider"
    android:authorities="${applicationId}.fileprovider"
    android:exported="false"
    android:grantUriPermissions="true">
  
    <meta-data
        android:name="android.support.FILE_PROVIDER_PATHS"
        android:resource="@xml/file_provider_paths" />
</provider>

在res/xml目錄(如果沒有xml目錄,則新建一個)下,添加文件file_provider_paths.xml,內容如下:

<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <external-files-path name="sharedata" path="shareData/"/>
</paths>

external-files-path表示通過 Context.getExternalFilesDir(null) 接口獲取到的目錄下的文件纔可被共享,其他未配置的路徑均不可被分享。同樣的節點可以配置多個,以支持多個不同的子目錄,如下所示:

第二步使用FileProvider接口,需要判斷sdk

將路徑通過FileProvider的接口轉換成content://URI形式,示例如下:

    public static void shareBitmap(Context context, Bitmap bitmap, String title) {
        Uri shareUri = null;
        try {
        //只能在android 11上面才能使用FileProvider存儲圖片,在android10上面同樣取不到
            if (Build.VERSION.SDK_INT > 29) {
                String sharedata = context.getExternalFilesDir(null) + "/sharedata/" + System.currentTimeMillis() + ".jpg";
                ShareHelper.save(bitmap, sharedata, Bitmap.CompressFormat.JPEG);
                File file = new File(sharedata);
                shareUri=getFileProvider(context,file);
            } else {
                String filePath = ImageUtil.getNewPhotoPath();
                ShareHelper.save(bitmap, filePath, Bitmap.CompressFormat.JPEG);
                shareUri=Uri.fromFile(new File(filePath));
            }
            Intent intent = new Intent(Intent.ACTION_SEND);
            intent.setType("image/*");
            intent.putExtra(Intent.EXTRA_STREAM,shareUri);
            context.startActivity(Intent.createChooser(intent, title));

        } catch (Exception e) {
            e.printStackTrace();
        }

    }

    // 將File 轉化爲 content://URI
    public static Uri getFileProvider(Context context, File file) {
        // ‘authority’要與`AndroidManifest.xml`中`provider`配置的`authorities`一致,假設你的應用包名爲com.example.app
        String authority = context.getPackageName() + ".provider";
        Uri contentUri = FileProvider.getUriForFile(context, authority, file);

        // 授權給微信訪問路徑
        context.grantUriPermission("com.tencent.mm",  // 這裏填微信包名
                contentUri, Intent.FLAG_GRANT_READ_URI_PERMISSION);

        return contentUri;
    }

    public static final String getDirPath() {
        File cameraPhotoPath = AppMain.getApp().getExternalFilesDir("UploadImage");
        if (!cameraPhotoPath.exists()){
            cameraPhotoPath.mkdirs();
        }
        return cameraPhotoPath.getAbsolutePath();
    }

    public static final String getNewPhotoPath() {
        return getDirPath() + "/" + System.currentTimeMillis() + ".jpg";
    }


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章