为什么会突然想到 选择图片,还有裁剪什么的,
是因为突然看到。联系人模块(6.0),创建用户的时候,当填写完用户信息,他还可以让用户选择自己的头像,当然也是点击选择图片,裁剪了, 但是我看到的是他用到了一个之前没怎么注意过的类 FilePorvider, 查了一下,也试着使用

这样子用 FilePorvider 有什么好处呢, 我看到的模块源码是,实质是将裁剪后的图片保存在了 /data/data/包名/cache 目录下了, 这个样子想想也能想通,毕竟把这个头像数据保存在cache文件夹下了,并且用的是fileprovider,这样其他模块app如果需要获取联系人的信息的话,头像的数据就可以直接从 联系人模块中获取了。

==先把问题放在上面==
测试:

小米的机子是好的 6.0, 移动的一款也是好的 6.0系统(2种跳转方式选择图片 裁剪都是好的)

移动的另外一款机子不行直接权限错误 6.0系统(选完图片后,去裁剪时,跟前面的跳转方式有关,如果用下面跳转方式的,第一种方式,进入com.android.documentsui 模块,然后用FileProvider 进行一系列操作, 在裁剪startActivityResult 就会报错,说是权限不足),但是联系人是好着的,我也不知道这个权限要怎么加 在哪里加,能加的权限都加了

大神看到了一定要帮我解决一下
错误如下:

uri content://media/external/images/media/87 from pid=10707, uid=10108 requires android.permission.READ_EXTERNAL_STORAGE, or grantUriPermission()

java.lang.SecurityException: Uid 10108 does not have permission to uri 0 @ content://com.android.providers.media.documents/document/image%3A71

一、 选择图片首先是跳转选择图片有2种

  1. 使用第一种有点问题
Intent intent = new Intent(Intent.ACTION_GET_CONTENT);  
        intent.setType("image/*");  
        startActivityForResult(intent, 2)

有些机子直接进入的是这个模块 com.android.documentsui

2. Intent.ACTION_PICK

Intent i = new Intent(Intent.ACTION_PICK, android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI);
        startActivityForResult(i,2);

选择图库进入 com.android.gallery3d 模块
选择相册进入 com.google.android.apps.photos 模块

选择图片后, 会返回uri类似
uri: content://media/external/images/media/203474
uri.getpath: /external/images/media/203474

二、 进入裁剪图片模式

2.1 普通正常的裁剪图片方式

intent.setDataAndType(in, "image/*");    
        intent.putExtra("crop", "true");        
        intent.putExtra("scale", true);
        intent.putExtra("scaleUpIfNeeded", true);
        intent.putExtra("aspectX", 1);
        intent.putExtra("aspectY", 1);
        // outputX,outputY 是剪裁图片的宽高
        intent.putExtra("outputX", 150); 
        intent.putExtra("outputY", 150);
        // intent.putExtra("return-data", true);
        startActivityForResult(intent, 3);

        裁剪完成后 会返回uri  也就是 裁剪后的图片的uri 
        Intent { dat=content://media/external/images/media/84 (has extras) }
        content://media/external/images/media/84

2.2 用 FileProvider (这个才是重点)

这是模块创建文件的代码

//创建文件名称
    private static final String PHOTO_DATE_FORMAT = "'IMG'_yyyyMMdd_HHmmss";
    private static String generateTempCroppedPhotoFileName() {
        final Date date = new Date(System.currentTimeMillis());
        SimpleDateFormat dateFormat = new SimpleDateFormat(PHOTO_DATE_FORMAT, Locale.US);
        return "ContactPhoto-" + dateFormat.format(date) + "-cropped.jpg";
    }

     //路径 ontext.getCacheDir()
     private static String pathForTempPhoto(Context context, String fileName) {
        final File dir = context.getCacheDir();
        dir.mkdirs();
        final File f = new File(dir, fileName);
        return f.getAbsolutePath();
     }


    //用 FileProvider 获取uri 
    public Uri getOutUri(Context context){
//      com.example.aaaa.fileprovider
        final String  fileProviderAuthority ="com.example.aaaa.fileprovider" ;
        File ff = new File(pathForTempPhoto(context, generateTempCroppedPhotoFileName()));
        Uri uri = FileProvider.getUriForFile(context, fileProviderAuthority,ff);
        return  uri;
    }

//这里设置 intent 把 fileprovider 生成的参数传过去

out  = getOutUri(MainActivity.this);
        intent.setDataAndType(in, "image/*"); 
        //设置输出文件
        intent.putExtra(MediaStore.EXTRA_OUTPUT, out );  
    intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION |                 Intent.FLAG_GRANT_READ_URI_PERMISSION);
        intent.setClipData(ClipData.newRawUri(MediaStore.EXTRA_OUTPUT, out ));

        intent.putExtra("crop", "true");    
        intent.putExtra("scale", true);
        intent.putExtra("scaleUpIfNeeded", true);
        intent.putExtra("aspectX", 1);
        intent.putExtra("aspectY", 1);
        // outputX,outputY 是剪裁图片的宽高
        intent.putExtra("outputX", 150); 
        intent.putExtra("outputY", 150);
        // intent.putExtra("return-data", true);
        startActivityForResult(intent, 3);

//有可能 onActivityresult 有时候 uri是有路径的, 有时候是null, 可以先对
onActivityresult 中的
intent.getdata() 进行判断 ,或者直接用前面生成好的路径uri。去取数据,看看有没有

final ContentResolver cr = this.getContentResolver();          
        new Thread(){
        public void run() {
             try {  
           // FileProvider 提供的数据
               //bmp = BitmapFactory.decodeStream(cr.openInputStream(out), null ,options);
               bmp = BitmapFactory.decodeStream(cr.openInputStream( out ));
        } catch (Exception e) {
            // TODO Auto-generated catch block
             System.out.println("小米======exception==");  
            e.printStackTrace();
        }  
          aa =  new BitmapDrawable(bmp) ;
          han.sendEmptyMessage(2);
       };
   }.start();

在res文件夹下建立 xml文件夹
随便输个名字,一会要配置到 AndroidManifest.xml 配置文件

<?xml version="1.0" encoding="utf-8"?>

<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <!-- Offer access to files under Context.getCacheDir() -->
    <cache-path name="my_cache" />
</paths>

在AndroidManifest.xml中配置

<!-- 在这里定义共享信息 -->  
        <provider  
            android:name="android.support.v4.content.FileProvider"  
           android:authorities="@string/fileprovider_authority" //这里是作者名称,随便写
            android:exported="false"  
            android:grantUriPermissions="true" >  
            <meta-data  
                android:name="android.support.FILE_PROVIDER_PATHS"  
                android:resource="@xml/filepaths" />  //这个名字就是xml文件夹中新建的文件的名称
        </provider>

PS:看网上的也还有一种方式是这样

//      Uri.fromFile(new File("/mnt/sdcard/temp"))
//      Uri.parse("file://" + Environment.getExternalStorageDirectory().getPath() + "/" +"aaaa.jpg");

//但是这样的方式的 uri其实会有问题, 有的手机会报
java.io.FileNotFoundException: /storage/emulated/0/aa.jpg: open failed: EACCES (Permission denied)

// 路径(关于 android 文件存储文件路径 )

应该改成这样:

//2种方式生成uri
out= Uri.fromFile(new File(this.getExternalCacheDir()+ "/" + "smalaaaal.jpg"));
   //实际路径为  file:///storage/emulated/0/Android/data/com.example.aaaa/cache/smalaaaal.jpg
out= Uri.parse( "file://" +this.getExternalCacheDir()+ "/" + "smalaaaal.jpg");  

这里注意不能写成
      out= Uri.parse(this.getExternalCacheDir()+ "/" + "smalaaaal.jpg");  
       会报:
      FileNotFoundException: No content provider: /storage/emulated/0/Android/data/com.example.aaaa/cache

      // (MediaStore.EXTRA_OUTPUT 的值是  output
      intent.putExtra(MediaStore.EXTRA_OUTPUT, out);
      //intent.putExtra("outputFormat", "JPEG");// 返回格式
      intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());  
      intent.putExtra("noFaceDetection", true);