Android7.0实现头像更换功能

引言

在实现头像更换功能时,期间发现android7.0和7.0以下的版本有一些区别,这里记录下来希望可以帮助大家。

问题

按照以下的代码运行时调用相机会出现闪退

private void camera(){
        Intent cameraIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
        if (cameraIntent.resolveActivity(getPackageManager()) != null) {
            mTmpFile = new File(FileUtils.createRootPath(getBaseContext()) + "/" + System.currentTimeMillis() + ".jpg");
            cameraIntent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(mTmpFile));
            startActivityForResult(cameraIntent, REQUEST_CAMERA);
        }
    }

Uri.fromFile(mTmpFile);导致闪退错误提示

android.os.FileUriExposedException: file:///storage/emulated/0/Android/data/com.example.zhaoyigang.photodemo3/cache/1496225693867.jpg exposed beyond app through ClipData.Item.getUri()
查找资料发现出于安全原因,Android7.0在这个方面最大变化就在于以前适用的File Uri需要更改为Content Uri。
android7.0行为变更
在应用间共享文件
对于面向 Android 7.0 的应用,Android 框架执行的 StrictMode API 政策禁止在您的应用外部公开 file:// URI。如果一项包含文件 URI 的 intent 离开您的应用,则应用出现故障,并出现 FileUriExposedException 异常。

要在应用间共享文件,您应发送一项 content:// URI,并授予 URI 临时访问权限。进行此授权的最简单方式是使用 FileProvider 类。如需了解有关权限和共享文件的详细信息,请参阅共享文件。

解决

  • 首先在manifest中添加provider
...

    ...

        "android.support.v4.content.FileProvider"
            android:authorities="你的包名.provider"
            android:exported="false"
            android:grantUriPermissions="true">
            "android.support.FILE_PROVIDER_PATHS"
                android:resource="@xml/provider_paths"/>
        
    

  • 其次在资源文件下创建相应的xml文件(如上:则创建provider_paths.xml)
<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <external-path name="external_files" path="."/>
paths>
  • 最后就是在activity中使用
private void camera(){
        Intent cameraIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
        if (cameraIntent.resolveActivity(getPackageManager()) != null) {
            mTmpFile = new File(FileUtils.createRootPath(getBaseContext()) + "/" + System.currentTimeMillis() + ".jpg");
            FileUtils.createFile(mTmpFile);
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N){
                cameraIntent.putExtra(MediaStore.EXTRA_OUTPUT,
                        FileProvider.getUriForFile(getBaseContext(), BuildConfig.APPLICATION_ID + ".provider", mTmpFile));
            }else {
                cameraIntent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(mTmpFile));
            }
            cameraIntent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(mTmpFile));
            startActivityForResult(cameraIntent, REQUEST_CAMERA);
        }
    }

关于截图

我的截图功能是调用的系统截图功能,这里有一个小问题

private void crop(String imagePath){
        //mCropImageFile = FileUtils.createTmpFile(getBaseContext());
        mCropImageFile = getmCropImageFile();
        Intent intent = new Intent("com.android.camera.action.CROP");
        intent.setDataAndType(getImageContentUri(new File(imagePath)), "image/*");
        intent.putExtra("crop", true);
        intent.putExtra("aspectX", 1);
        intent.putExtra("aspectY", 1);
        intent.putExtra("outputX", 500);
        intent.putExtra("outputY", 500);
        intent.putExtra("scale", true);
        intent.putExtra("return-data", false);
        intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(mCropImageFile));
        intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
        intent.putExtra("noFaceDetection", true);
        startActivityForResult(intent, REQUEST_CROP);
    }

intent.putExtra(“return-data”, false);中return-data如果是true,那么intent会返回一个bitmap,而这里截图过程中虽然显示的是原图,但截取后显示的图片却非常模糊。原因是图片没有保存截取的只是缩略图。为了获取到裁切后的原图,我们选择将剪切的图片保存在本地,然后调用本地的图片,并不直接返回Bitmap.

具体的代码

你可能感兴趣的:(android)