美文网首页安卓技巧Android知识开源项目
Android 7.0 文件读取适配,适配相机及裁剪图片

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

作者: 进击的包籽 | 来源:发表于2017-03-09 17:05 被阅读5092次

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

  • Demo挂载在码云上 码云链接
    华为荣耀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>
xml.jpg

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));就不需要,切记。

相关文章

网友评论

  • 3d13e2991f92:真的感谢,调试了好几个小时,就是outputUri 的问题
    进击的包籽:@zstone1010 哈,不客气,我也是弄了好久,觉得这是个坑
  • StillForward:裁剪后的输出uri真是个坑
  • Smashing丶:你好~关于在裁剪的时候,设置输出的uri不能使用ContentUri,这个地方我有点不太明白,为什么是这样,7.0以上不是要用ContentUri么?另外,我想看一下源码到底是怎么回事,但是不知道系统裁剪图片的源码在哪里.....
    Smashing丶:@Good包籽 那你知道源码在哪里么....我找了好久没找到,不知道是哪个Activity....
    进击的包籽:@Smashing丶 其实这个是我自己实验出来,之前也是用contenturi,但是不行,具体源码,还要研究下。😂
  • d5ba366035a2:返回的裁剪图片有点模糊怎么解决,产品就盯着这个不放:joy:
    进击的包籽: @cold初养成 哈,那就原图出来,不走截图了,然后再质量压缩
    d5ba366035a2:设置大了会oom吧,不清楚该设多少合适
    进击的包籽:@cold初养成 你可以把裁剪的尺寸调大一些,我里面好久就弄了几百个像素点
  • 简安理思:你好,我想问一下,我按照你的写了一个demo,但是每当我去拍完照,点击确认不能返回到上一个Activity,然后如果拍完照片后点击取消,才会回调onActivityResult方法,请问您知道大概是哪个方面的错误导致的吗?
    进击的包籽:@waitingwx 。。。不客气
    简安理思: @Good包籽 恩恩,我在手机上是完全可以的,但是放在了联想yoga平板上就不行了,拍完照店里确认无法返回上一级。应该是这个机器的问题~非常感谢
    进击的包籽: @waitingwx 我的demo是拍完照,再截图,你可以不调用截图,直接用照片
  • gsn_hear:相当不错,在api-18以上都能运行,api-17会提示没有sd卡这个怎么解决
    进击的包籽:@樹头 这样吧,我这写的有点久了,你在拍照和裁剪图片回调的地方,加上更新相册,试试可以不
    // 最后通知图库更新
    activity.sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE,Uri.fromFile(file)));
    gsn_hear:@Good包籽 api-17的虚拟机,
    public boolean isSdCardExist()
    {
    return Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED);
    }
    这里的判断貌似决定了17之下要插上sd卡才能使用功能。
    进击的包籽:我的项目是15以上,没出现问题,你的是什么机子?
  • d6116eca5146:verygood
  • 南歌ccc:不错不错!

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

本文链接:https://www.haomeiwen.com/subject/dqswgttx.html