项目用到了app拍照相册裁剪,因为测试机是android10的,一顿操作猛如虎,功能完美呀。然后测试小姐姐用android11测试,直接GG了。
看了之后才发现android11强制分区处理。
android 适配拍照主要经历了 6.0 7.0 10和11
6.0主要是处理运行时权限 7.0为FileProvider 文件共享 10和11则是分区处理。
主要说下7.0和10,11的适配方法。
参考:鸿洋_: Android 7.0 行为变更 通过FileProvider在应用间共享文件吧
官方文档:Android 7.0 行为变更
android7.0 主要是适配其中文件访问权限
在官方7.0的以上的系统中,尝试传递 file://URI可能会触发FileUriExposedException。
清单文件配置provider ,因为它是ContentProvider的子类。
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="com.edu.lzdx.liangjianpro.fileProvider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/provider_paths" />
</provider>
<?xml version="1.0" encoding="utf-8"?>
android:authorities 表示授权者,这里的格式一般是[appId].fileprovider
android:exported 只能为false
android:grantUriPermissions="true" 表示授权Uri权限 ,且必须为true
meta-data里设置指定的文件目录,为引用provider_paths文件
provider_paths文件
<paths xmlns:android="http://schemas.android.com/apk/res/android">
<root-path name="root" path="" />
<files-path name="files" path="" />
<cache-path name="cache" path="" />
<external-path name="external" path="" />
<external-files-path name="name" path="path" />
<external-cache-path name="name" path="path" />
</paths>
<root-path/> 代表设备的根目录new File("/");
<files-path/> 代表context.getFilesDir()
<cache-path/> 代表context.getCacheDir()
<external-path/> 代表Environment.getExternalStorageDirectory()
<external-files-path>代表context.getExternalFilesDirs()
<external-cache-path>代表getExternalCacheDirs()
常用到的 为 external-path 和external-files-path name和path按照自己需求编写。
private void openCamera() {
Intent captureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
// 判断是否有相机
if (captureIntent.resolveActivity(getPackageManager()) != null) {
File photoFile = null;
try {
photoFile = new File(getExternalCacheDir(), "head_image.jpg");
} catch (IOException e) {
e.printStackTrace();
}
if (photoFile != null) {
//根据判断版本 4.4也能安全运行 或者 授权
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
//适配Android 7.0文件权限,通过FileProvider创建一个content类型的Uri 主要是这段代码
photoUri = FileProvider.getUriForFile(this, getPackageName() + ".fileProvider", photoFile);
} else {
photoUri = Uri.fromFile(photoFile);
}
}
if (photoUri != null) {
Log.e("TAG", "openCamera: 拍照初始路径===" + photoUri);
captureIntent.putExtra(MediaStore.EXTRA_OUTPUT, photoUri);
captureIntent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
startActivityForResult(captureIntent, CODE_CAMERA_REQUEST);
}
}
}
7.0大致就是这样。
android10 如果 targetSdkVersion<=29 在清单文件中添加
android:requestLegacyExternalStorage="true"
可以让android10忽略分区,按照之前的方式。但是运行在android11的机型就会强制执行分区处理,所以这个只是过度方式,还是得适配。
我遇到得问题就是一个拍照裁剪------加载图片失败,追其原因,为系统裁剪不能访问到app得私有目录,所以我们
主要进行一个裁剪公域得处理。
public static Intent getCropIntent(Uri uri,Context context,String path){
Log.d("裁剪的Url", "cropRawPhoto: " + uri.toString());
Intent intent = new Intent("com.android.camera.action.CROP");
intent.setDataAndType(uri, "image/*");
intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION | Intent.FLAG_GRANT_READ_URI_PERMISSION);
// 设置裁剪
intent.putExtra("crop", "true");
intent.putExtra("aspectX", 1);
intent.putExtra("aspectY", 1);
intent.putExtra("outputX", output_X);
intent.putExtra("outputY", output_Y);
intent.putExtra("return-data", false);
context.getExternalFilesDir(Environment.DIRECTORY_PICTURES);
//android11 分区存储
if (Build.VERSION.SDK_INT >= 29) {
Log.e("TAG", "裁剪公域::"+path);
File mOnputFile11 = new File(path, SpUtils.getInstance(context).getString("token","") + ".png");
intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.parse("file://" + mOnputFile11.getAbsolutePath()));
} else {
intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(
new File(context.getExternalCacheDir(), "face-cropped.jpg")));
}
return intent;
}
//等会儿。