搜尋

首頁  >  問答  >  主體

android - 7.0系统拍照后,使用系统截图功能,截图保存时崩溃如何解决

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_leeringa_lee2823 天前1048

全部回覆(5)我來回復

  • 怪我咯

    怪我咯2017-04-17 17:59:50

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

    回覆
    0
  • 怪我咯

    怪我咯2017-04-17 17:59:50

    對啊 權限錯誤。 6.0後要動態取得權限。

    回覆
    0
  • 黄舟

    黄舟2017-04-17 17:59:50

    現在不會提示,需要主動發起權限申請,一般關於拍照就是SD卡和攝影機兩個權限。 6.0的權限模型相關的知識可以網路查詢下。 http://www.tuicool.com/articl... 網上查到,你看是否能解決,

    回覆
    0
  • 巴扎黑

    巴扎黑2017-04-17 17:59:50

    我目前的思路和做法:

    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

    回覆
    0
  • 天蓬老师

    天蓬老师2017-04-17 17:59:50

    安全異常,6.0以上要使用者授權的。

    回覆
    0
  • 取消回覆