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