美文网首页菜菜AndroidAndroid记录安卓Android基础开发
完全掌握 Android 选择图片、拍照以及图片裁剪

完全掌握 Android 选择图片、拍照以及图片裁剪

作者: VitaminChen | 来源:发表于2016-05-19 17:18 被阅读3890次

    最近深入了解了关于图片的选择和裁剪部分的内容,发现还是有很多需要注意的点,需要及时的记录一下。
    根据使用场景的不同,需要分情况讨论。

    先贴代码再分析:

    
         Uri imageUri;
        /* 场景1:选择一张图片 */
        private void gotoPickImage() {
            Intent intent = new Intent(Intent.ACTION_PICK, MediaStore.Images.Media.EXTERNAL_CONTENT_URI);
            startActivityForResult(intent, REQUEST_PICK_IMAGE);
        }
         
        /* 场景2:选择一张图片并裁剪获得一个小图 */
        private void gotoPickAndCropSmallBitmap() {
            Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
            intent.setType("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("return-data", true);
            intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
            intent.putExtra("noFaceDetection", true); // no face detection
            startActivityForResult(intent, REQUEST_CROP_IMAGE_SMALL);
        }
    
        /* 场景3:选择一张图片并裁剪获得一个大图 */
        private void gotoPickAndCropBigBitmap() {
            imageUri = getTmpUri();
            Intent intent = new Intent(Intent.ACTION_GET_CONTENT, null);
            intent.setType("image/*");
            intent.putExtra("crop", "true");
            intent.putExtra("aspectX", 1);
            intent.putExtra("aspectY", 1);
            intent.putExtra("outputX", 2000);
            intent.putExtra("outputY", 2000);
            intent.putExtra("scale", true);
            intent.putExtra("return-data", false);
            intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
            intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
            intent.putExtra("noFaceDetection", true); // no face detection
            startActivityForResult(intent, REQUEST_CROP_IMAGE_BIG);
        }
    
        /* 场景4:拍照并裁剪 */
        private void startImageCapture() {
            String IMAGE_FILE_LOCATION = Environment.getExternalStorageDirectory() + "/" + "posprint" + "/tmp.jpg";//temp file
    
            imageUri = Uri.fromFile(new File(IMAGE_FILE_LOCATION));//The Uri to store the big bitmap
            Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
            intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
            startActivityForResult(intent, REQUEST_CAPTURE_AND_CROP);
        }
    
        @Override
        protected void onActivityResult(int requestCode, int resultCode, Intent data) {
            if (resultCode == Activity.RESULT_OK) {
                Bitmap bitmap = null;
                try {
                    switch (requestCode) {
                        case REQUEST_PICK_IMAGE:
                            //选择的图片的Uri
                            imageUri = data.getData(); 
                            bitmap = MediaStore.Images.Media.getBitmap(
                                    getContentResolver(), imageUri);
                        case REQUEST_CROP_IMAGE_SMALL:
                            //裁剪后的小图
                            bitmap = data.getParcelableExtra("data");  
                            break;
                        case REQUEST_CROP_IMAGE_BIG:
                            //裁剪后的大图
                            bitmap = MediaStore.Images.Media.getBitmap(getContentResolver(), imageUri);
                            break;
                        case REQUEST_CAPTURE_AND_CROP:
                            //得到拍照后的图片并裁剪
                            cropImageUri(imageUri, REQUEST_CROP_IMAGE_BIG);
                            break;
    
      
                    }
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            doSomething(bitmap);
        }
        
        //获得临时保存图片的Uri,用当前的毫秒值作为文件名
        private Uri getTmpUri() {
            String IMAGE_FILE_DIR = Environment.getExternalStorageDirectory() + "/" + "app_name";
            File dir = new File(IMAGE_FILE_DIR);
            File file = new File(IMAGE_FILE_DIR, Long.toString(System.currentTimeMillis()));
            //非常重要!!!如果文件夹不存在必须先手动创建
            if (!dir.exists()) {
                dir.mkdirs();
            }
            return Uri.fromFile(file);
        }
        
        //裁剪拍照后得到的图片
        private void cropImageUri(Uri uri, int requestCode) {
            Intent intent = new Intent("com.android.camera.action.CROP");
            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, uri);
            intent.putExtra("return-data", false);
            intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
            intent.putExtra("noFaceDetection", true); // no face detection
            intent = Intent.createChooser(intent, "裁剪图片");
            startActivityForResult(intent, requestCode);
        }
    

    场景1:

    直接从图库中选择一张照片,在 onActivityResult() 中就可以通过 data.getData() 得到该图片的Uri,没什么好说的

    场景2:

    从图库中选择一张照片并 裁剪得到一个小图。这里为什么强调是小图,这是因为当图片过大时(我在魅族 Pro5 上测试分辨率 300 * 300 时正常,当设为 400 * 400 就出错了,具体临界点不明)。

    这里要注意几个 extra 字段,先上一个图:


    裁剪图片的 extra 字段
    • return-data: 设为 true 的时候,在 onActivityResult() 中可以直接通过 data.getParcelableExtra("data") 得到裁剪后的 Bitmap 对象。但是当 Bitmap 过大时,就不能使用这种方法了,必须使用 场景3 中采用的方法。
    • MediaStore.EXTRA_OUTPUT: 设置裁剪图片的输入 Uri。可以通过 Uri.fromFile(tmpFile) 获得

    场景3:

    和场景2十分类似,唯一的不同只是此时可以裁剪得到一个分辨率较高的图片。区别只在于 return-dataMediaStore.EXTRA_OUTPUT这两个 extra 的设置。

    这里把return-data 设为 false ,同时向 MediaStore.EXTRA_OUTPUT 设置一个临时构造的 Uri,这个 Uri 就用来保存裁剪后的大图。裁剪之后,在 onActivityResult() 中就可以通过 MediaStore.Images.Media.getBitmap( getContentResolver(), imageUri) 得到裁剪后的 Bitmap 大图。

    场景4:

    拍照后并裁剪,其实是分了两步。第一步拍照得到图片的 Uri,第二部把该图片的 Uri 传给裁剪图片的程序处理。
    几点需要注意的地方:

    • 这里的的 action 变成了 com.android.camera.action.CROP, 并且不再需要设置 extra 字段 crop
    • intent.setDataAndType(uri, "image/*"),这里的 uri 指向拍照后得到的原图,intent.putExtra(MediaStore.EXTRA_OUTPUT, uri) 裁剪后的图片也保存在这个 uri,所以原图就被覆盖了。如果想同时保存原图和裁剪后的图片,这里需要再提供另外一个 Uri 对象
    • 最终得到裁剪后的图片的方法和场景3是一样的。当然如果只是想得到一个裁剪后的小图的话,那么也可以通过场景2的方法得到。

    注意事项:

    1. 注意添加读写权限及拍照权限:
        <uses-permission android:name="android.permission.CAMERA"/>
        <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    
    1. 注意 getTmpUri() 函数里,如果保存图片的目录不存在,需要手动创建,否则无法保存裁剪后的图片。

    相关文章

      网友评论

      • 吧主:这篇文章,我给你在公众号原创发布可以吗?公众号:杨守乐
        吧主: @VitaminChen 好的,谢谢你的分享精神
        VitaminChen:@吧主 没问题
      • 倔强的炉包:楼主试下兼容小米手机。
      • xiasuhuei321:我自己的小米4,6.0调不出来裁剪……魅族 5.0,调裁剪图片加载不出来,想放弃原生裁剪了……
      • cf908b4323ca:Good.赞一个
      • JC_Hou:为什么我裁剪小的图片不会出现裁剪区域呢
      • 捡淑:这种跑原生系统完全没问题,关键很多国内厂商rom改过的,适配机型的道路上各种坑
        恨自己不能小清新:@你家鹏大大 666刺激用得好
        你家鹏大大:@AlaricNorris 同感,小米坑死人。魅族又可以用。:joy:。开发android真是刺激。
      • 你家鹏大大:我打印了Log,当机子拍照成功后。点击确定按钮,返回的时候。onActivityResult方法接收到的resultCode = 0 。而Activity.RESULT_OK 是为 -1 的。这样根本就没有办法进入到 case REQUEST_CAPTURE_AND_CROP 这个循环中。也就是说拍完照,从相机返回的时候哪里出问题了。
        如果可以的话。能否在GitHub上给个Demo地址~
        我的确是Copy了的。前面三个从相册中获取,裁剪做的很好。还要谢谢你呢~
        looooker:你好,请问你最后是怎么处理的??
      • 你家鹏大大:写的很好。。。但是我测试了拍照,裁剪功能,小米note3 和小米4 华为的机型都不成功啊。有没有做过真机测试呢?或者直接给个Demo代码也好 0.0 哈哈。如果你有时间的话
        你家鹏大大:@VitaminChen :joy:
        VitaminChen:@你家鹏大大 试过魅族和华为没什么问题。不成功是什么表现能具体描述下么?代码基本上就是上面那些了,只用copy到一个activity里面加个按钮跳转就行了。
      • 1e592c21f6de:写的很好,👍

      本文标题:完全掌握 Android 选择图片、拍照以及图片裁剪

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