Android 7.0 文件读取适配,适配相机及裁剪图片

作者: cnbzlj 发布时间: 2019-09-24 浏览: 2161 次 编辑

Android 7.0 打开文件的错误,调用相机及裁剪图片,还有动态权限

Github代码
码云链接

  • 华为荣耀V8,小米5,小米6,R9s,测试通过,其他机子没机会测试。。。。

核心代码

  • 7.0以上除了相机和裁剪图片,只要是打开文件的,uri都要更改,包括更新安装包,记得要改uri,不然解析异常,但裁剪图片outputUri有点不一样
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N)
        {
            //赋予权限
            intent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
            Uri uri = FileProvider.getUriForFile(activity, Constants.FILE_CONTENT_FILEPROVIDER, file);
            //举个栗子
            intent.setDataAndType(uri,"application/" + fileType);
        } else
        {
            intent.setDataAndType(Uri.fromFile(file),"application/" + fileType);
        }
  • 因为公司项目需要能修改头像功能,之前的测试机一直都是Android 5.1的机子,一直没问题,突然有一天,小米5升级到Android 7.0,华为机子也升级上去,然后就一直奔溃,然后就开始上网查资料啦。。。
    其中相机调用正常通过,但是裁剪就不是了,这次重点就是裁剪EXTRA_OUTPUT,
    别用 FileProvider.getUriForFile,否者在onActivityResult时,返回的resultCode为0,即取消。
  • 首先Android6.0以上开始要加入动态权限管理,所以开启相机之前,动态权限要去申请,这里我就不多说了,网上很多,推荐一个大牛的:
    转自严振杰的博客:http://blog.csdn.net/yanzhenjie1003/article/details/52503533
    以及他的Github:https://github.com/yanzhenjie/AndPermission#userconsent#
  • 网上关于7.0的适配,我也是看到简书上有看到:http://www.jianshu.com/p/56b9fb319310
    这个大家也能看看,我的就是参考他的。

上面2个建议大家都看看,很有用的,我的就是基于他们做的。

1.首先res文件夹下,新建一个xml文件夹,名字就是 android:resource="@xml/file_paths"对应的内容

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <paths>
        <external-path path="" name="camera_photos" />
    </paths>
</resources>

2.在Manifest <application>里面添加下面一段

        <provider
            android:name="android.support.v4.content.FileProvider"
            android:authorities="com.goodbao.furniture.fileprovider"
            android:grantUriPermissions="true"
            android:exported="false">
            <meta-data
                android:name="android.support.FILE_PROVIDER_PATHS"
                android:resource="@xml/file_paths" />
        </provider>

android:authorities这个属性的值,我写的是包名+fileprovider,其实不是很重要,后面会用的到

其他属性的介绍,http://www.jianshu.com/p/56b9fb319310 这篇里面讲的听清楚的了。

3.调用系统相机

photo_image = createImagePath(APP_NAME + DATE);
        File file = new File(photo_image);
        if (!file.getParentFile().exists())
        {
            file.getParentFile().mkdirs();
        }
        Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
        //Android7.0以上URI
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N)
        {
            //添加这一句表示对目标应用临时授权该Uri所代表的文件
            intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
            //通过FileProvider创建一个content类型的Uri
            Uri uri = FileProvider.getUriForFile(activity, Constants.FILE_CONTENT_FILEPROVIDER, file);
            intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);
        } else
        {
            intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(file));
        }
        try
        {
            activity.startActivityForResult(intent, REQUEST_CODE_TAKE_PHOTO);
        } catch (ActivityNotFoundException anf)
        {
            ToastUtils.showShortToast("摄像头尚未准备好"));
        }

其中

//添加这一句表示对目标应用临时授权该Uri所代表的文件
    intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
    //通过FileProvider创建一个content类型的Uri
    Uri uri = FileProvider.getUriForFile(activity, Constants.FILE_CONTENT_FILEPROVIDER, file);
    intent.putExtra(MediaStore.EXTRA_OUTPUT,uri);
  • Constants.FILE_CONTENT_FILEPROVIDER这个值
    就是之前 Manifest 里添加的 android:authorities="com.goodbao.furniture.fileprovider",
  • 所以你也能写成
    Uri uri = FileProvider.getUriForFile(activity, "com.goodbao.furniture.fileprovider", file);
    只是多次用到,我就弄成共用的。
  • Android7.0以上,相机调用时,intent.putExtra(MediaStore.EXTRA_OUTPUT, uri),
    Uri就不能用Uri.fromFile(file)
    而是要FileProvider.getUriForFile(activity, Constants.FILE_CONTENT_FILEPROVIDER, file);
    但是裁剪的时候就不一样,intent.setData,即打开图片的Uri需要使用ContentUri,
    而intent.putExtra(MediaStore.EXTRA_OUTPUT, outputUri);里的outputUri继续使用 Uri.fromFile(file),
    估计是裁剪操作一整个流程,应该是授权一次就好,在访问时先授权,而写入就不需要了。

4.裁剪图片

/**
     * 调用系统剪裁功能
     */
    public void cropPicture(Activity activity, String path)
    {
        File file = new File(path);
        if (!file.getParentFile().exists())
        {
            file.getParentFile().mkdirs();
        }
        Uri imageUri;
        Uri outputUri;
        crop_image = createImagePath(APP_NAME + "_crop_" + DATE);
 
        Intent intent = new Intent("com.android.camera.action.CROP");
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N)
        {
            //添加这一句表示对目标应用临时授权该Uri所代表的文件
            intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
           //TODO:访问相册需要被限制,需要通过FileProvider创建一个content类型的Uri
            imageUri = FileProvider.getUriForFile(activity, FILE_CONTENT_FILEPROVIDER, file);
            outputUri = Uri.fromFile(new File(crop_image));
            //TODO:裁剪整个流程,估计授权一次就好outputUri不需要ContentUri,否则失败
            //outputUri = FileProvider.getUriForFile(activity, "com.solux.furniture.fileprovider", new File(crop_image));
        } else
        {
            imageUri = Uri.fromFile(file);
            outputUri = Uri.fromFile(new File(crop_image));
        }
        intent.setDataAndType(imageUri, "image/*");
        intent.putExtra("crop", "true");
        //设置宽高比例
        intent.putExtra("aspectX", 1);
        intent.putExtra("aspectY", 1);
        //设置裁剪图片宽高
        intent.putExtra("outputX", 300);
        intent.putExtra("outputY", 300);
        intent.putExtra("scale", true);
        intent.putExtra(MediaStore.EXTRA_OUTPUT, outputUri);
        intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
        intent.putExtra("noFaceDetection", true);
        activity.startActivityForResult(intent, REQUEST_CODE_CROP_PICTURE);
    }
  • 要处理的图片imageUri , intent.setDataAndType(imageUri, "image/*");
    imageUri = FileProvider.getUriForFile(activity, Constants.FILE_CONTENT_FILEPROVIDER, file);
  • 处理完的图片outputUri ,intent.putExtra(MediaStore.EXTRA_OUTPUT, outputUri);
    outputUri = Uri.fromFile(new File(crop_image));就不需要,切记。