androidMay be useful Android

Android调用系统图片分享以及解决Android11上分享图

2021-07-07  本文已影响0人  唐小鹏

至于调用系统分享这个功能就很简单了,直接上代码,主要如果项目已经使用了分区的概念,那么在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";
    }


上一篇下一篇

猜你喜欢

热点阅读