android - 7.0系统拍照后,使用系统截图功能,截图保存时崩溃如何解决
ringa_lee
ringa_lee 2017-04-17 17:57:50
0
5
1044
java.lang.SecurityException: Permission Denial: writing android.support.v4.content.FileProvider uri content://com.tianshaokai.demo.fileprovider/camera_photos/temp/1480414713257.jpg from pid=23075, uid=10041 requires the provider be exported, or grantUriPermission()
 public void cropPhoto(File file) {
        cropfile = new File(Environment.getExternalStorageDirectory(), "/temp/" + System.currentTimeMillis() + ".jpg");
        if (!cropfile.getParentFile().exists()) cropfile.getParentFile().mkdirs();
        Uri outputUri = FileProvider.getUriForFile(this, getProvider(), cropfile);
        Uri imageUri = FileProvider.getUriForFile(this, getProvider(), file);//通过FileProvider创建一个content类型的Uri
        Intent intent = new Intent("com.android.camera.action.CROP");
        intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
        intent.setDataAndType(imageUri, "image/*");
        intent.putExtra("crop", "true");
        intent.putExtra("aspectX", 1);
        intent.putExtra("aspectY", 1);
        intent.putExtra("outputX", 300);
        intent.putExtra("outputY", 400);
//        intent.putExtra("scale", true);
        intent.putExtra(MediaStore.EXTRA_OUTPUT, outputUri);
        intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
//        intent.putExtra("noFaceDetection", true); // no face detection
        startActivityForResult(intent, 1002);
 }
file = new File(Environment.getExternalStorageDirectory(), "/temp/" + System.currentTimeMillis() + ".jpg");
        if (!file.getParentFile().exists()) file.getParentFile().mkdirs();
        Uri imageUri = FileProvider.getUriForFile(this, getProvider(), file);//通过FileProvider创建一个content类型的Uri
        Log.d(TAG, "imageUri: " + imageUri);
        Intent intent = new Intent();
        intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); //添加这一句表示对目标应用临时授权该Uri所代表的文件
        intent.setAction(MediaStore.ACTION_IMAGE_CAPTURE);//设置Action为拍照
        intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);//将拍取的照片保存到指定URI
        startActivityForResult(intent, 1001);

现在就是拍照完成后,裁剪 就报最上边的 权限错误,测试机是 7.0 的模拟器

看到大家的回答我补充一下啊,我已经添加了6.0 的 SD卡和摄像头 动态权限,我现在的问题是裁剪完 保存时 报上边的错误。

ringa_lee
ringa_lee

ringa_lee

全部回复(5)
刘奇

照片 截取输出的outputUri, 只能使用 Uri.fromFile,不能用FileProvider.getUriForFile

刘奇

对啊 权限错误。 6.0后要动态获取权限。

黄舟

现在不会提示,需要主动发起权限申请,一般关于拍照就是SD卡和摄像头两个权限。6.0的权限模型相关的知识可以网上查询下。http://www.tuicool.com/articl... 网上查到,你看是否能解决,

巴扎黑

我目前的思路和做法:

1、使用FileProvider兼容Android N及其以上版本

2、用一个文件来保存拍照以及剪裁的图片

步骤:

1、在AndroidManifest.xml中申明FileProvider

<provider
    android:name="android.support.v4.content.FileProvider"
    android:authorities="包名.fileprovider"
    android:exported="false"
    android:grantUriPermissions="true">
    <meta-data
        android:name="android.support.FILE_PROVIDER_PATHS"
        android:resource="@xml/filepaths" />
    </provider>

2、在res下创建xml目录,并创建filepaths.xml,其内容如下:

<paths>
    <external-path
        name="my_images"
        path="Android/data/包名/files/header/" />
</paths>

注:用该路径保存的好处是,文件会随着应用的卸载而删除,file/header/为自定义路径

3、调用处需要处理外部存储权限(由于使用系统的拍照,所以不需要提供相机权限)

int permission = ContextCompat.checkSelfPermission(context, Manifest.permission.WRITE_EXTERNAL_STORAGE);
if (permission != PackageManager.PERMISSION_GRANTED) {
    ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, PERMISSION_HEAD);
} else {
    //调用拍照或者从相册选取
}
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
    switch (requestCode) {
        case PERMISSION_HEAD:
        if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
            //调用拍照或者从相册选取
        } else {
            //提示:"要使用该功能,必须允许或者在应用访问授权中打开存储空间权限"
        }
        break;
        default:
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        }
    }

4、URI处理,路径已经在filepath.xml中申明好了,需要定义个文件名来保存图片(save.jpg)

private Uri getUri() {
    File path = new File(Environment.getExternalStorageDirectory(), "Android/data/包名/files/header");
    if (!path.exists()) {
        path.mkdirs();
    }
    File file = new File(path, "save.jpg");
    //由于一些Android 7.0以下版本的手机在剪裁保存到URI会有问题,所以根据版本处理下兼容性
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
        return FileProvider.getUriForFile(context, "com.maidouvr.fileprovider", file);
    } else {
        return Uri.fromFile(file);
    }
}

5、拍照处理

Uri uri = getUri();
Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);

//将存储图片的uri读写权限授权给相机应用
List<ResolveInfo> resInfoList = getActivity().getPackageManager().queryIntentActivities(intent, PackageManager.MATCH_DEFAULT_ONLY);
for (ResolveInfo resolveInfo : resInfoList) {
    String packageName = resolveInfo.activityInfo.packageName;
    getActivity().grantUriPermission(packageName, uri , Intent.FLAG_GRANT_WRITE_URI_PERMISSION | Intent.FLAG_GRANT_READ_URI_PERMISSION);
}
startActivityForResult(Intent.createChooser(intent, "选择拍照工具"), 1001);

6、剪裁处理

Uri uri = getUri();
Intent intent = new Intent("com.android.camera.action.CROP");
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);

intent.setDataAndType(uri , "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(MediaStore.EXTRA_OUTPUT, getUri());
intent.putExtra("return-data", false);
intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
intent.putExtra("noFaceDetection", true);

//将存储图片的uri读写权限授权给剪裁工具应用
List<ResolveInfo> resInfoList = getActivity().getPackageManager().queryIntentActivities(intent, PackageManager.MATCH_DEFAULT_ONLY);
for (ResolveInfo resolveInfo : resInfoList) {
    String packageName = resolveInfo.activityInfo.packageName;
    getActivity().grantUriPermission(packageName, uri, Intent.FLAG_GRANT_WRITE_URI_PERMISSION | Intent.FLAG_GRANT_READ_URI_PERMISSION);
}
startActivityForResult(Intent.createChooser(intent, "选择剪裁工具"), 1002);

最后再补充一点:有些手机可能剪裁后的resultCode总是为RESULT_CANCEL,所以建议将使用该功能Activity的launchMode设置为singleTask

Peter_Zhu

安全异常,6.0以上要用户授权的。

热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板