Android-图片的选择,裁剪,压缩,适配高版本

趁着国庆有时间我们来聊一聊最常用的选取用户图片一系列的功能,go!

效果展示
效果展示链接

1.图片的选取

拍照

我们之前设置拍照保存的文件地址的Uri,都是直接Intent.putExtra(MediaStore.EXTRA_OUTPUT,文件保存的Uri路径),但是7.0之后,对用户权限提高了保护,之前那种方式行不通了,所以我们要做7.0的判断,用FileProvider获取设置保存的文件Uri,然后放到Intent.putExtra(MediaStore.EXTRA_OUTPUT,文件保存的Uri路径)中,代码如下:

 //相机拍照的一个标识,后面用
                TAKEPAHTO = 1;
                // 启动系统相机
                Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
                //7.0以下设置保存图片的地址
                Uri norTakePhotoSaveAdr;
                // 判断7.0android系统
                if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                    //临时添加一个拍照权限
                    intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
                    //  //通过FileProvider获取保存图片file的uri(先创建file,然后获取File的Uri)
                    takePhotoSaveAdr = FileProvider.getUriForFile(MainActivity.this,
                            "com.hxzk.bj.photodemo", new File(Environment.getExternalStorageDirectory(), "savephoto.jpg"));
                    //MediaStore.EXTRA_OUTPUT-此设置需要一个保存视频的路径和文件名的Uri
                    intent.putExtra(MediaStore.EXTRA_OUTPUT, takePhotoSaveAdr);
                } else {
                    norTakePhotoSaveAdr = Uri.fromFile(new File(Environment.getExternalStorageDirectory(), "savephoto.jpg"));
                    intent.putExtra(MediaStore.EXTRA_OUTPUT, norTakePhotoSaveAdr);
                }
                //PHOTO_TAKEPHOTO,相机的一个请求码,返回时要用
                startActivityForResult(intent, PHOTO_TAKEPHOTO);

相册

相比较拍照,相册要简单一点,代码中都有注释,直接看:

  //拍照的一个表示
                TAKEPAHTO = 0;
                //调用系统图库,选择图片
                //Intent.ACTION_PICK 意思是选择数据,其具体表达有:
                // Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
                //intent.setType("image/*"); 获取本地图片
                // intent.setType("video/*");  获取本地视频
                //intent.setType("audio/*")  获取本地音乐

                // Intent intent = new Intent(Intent.ACTION_PICK);
                //  intent.setType(ContactsContract.Contacts.CONTENT_TYPE); //获取联系人
                // startActivityForResult(intent, PICK_CONTACT);

                //第二种写法
                Intent intent = new Intent(Intent.ACTION_PICK, null);
                //其中External为sdcard下的多媒体文件,Internal为system下的多媒体文件。
                //使用INTERNAL_CONTENT_URI只能显示存储在内部的照片
                intent.setDataAndType(
                        MediaStore.Images.Media.INTERNAL_CONTENT_URI, "image/*");
                //返回结果和标识
                startActivityForResult(intent, PHOTO_PHOTOALBUM);

.拍照或选取图片的返回结果

不论是拍照还是选取图片,我们都要在Activity的onActivityResult中去获取返回结果,然后进行下一步操作。这里提一下,我们在跳转相册或拍照都有一个requestCode,如图:
startActivityForResult(intent, PHOTO_PHOTOALBUM);
这个,我们全局定义了三个,分别是拍照,相册,裁剪的标识:

 //三个常量全局标识
    //图库
    private static final int PHOTO_PHOTOALBUM = 0;
    //拍照
    private static final int PHOTO_TAKEPHOTO = 1;
    //裁剪
    private static final int PHOTO_PHOTOCLIP = 2;

在onActivityResult中就要用来区分了:


image.png

没啥说的!

2. 图片的裁剪和压缩

图片的裁剪(减小了图片在手机中文件的大小(因为裁剪只是部分嘛),比如1m变为128kb)

图片的裁剪我们主要看一下starPhotoZoom()这个裁剪方法,代码如下:

 public void startPhotoZoom(Uri uri) {
        Log.e("uri=====", "" + uri);
        //com.android.camera.action.CROP,这个action是调用系统自带的图片裁切功能
        Intent intent = new Intent("com.android.camera.action.CROP");
        intent.setDataAndType(uri, "image/*");//裁剪的图片uri和图片类型
        intent.putExtra("crop", "true");//设置允许裁剪,如果不设置,就会跳过裁剪的过程,还可以设置putExtra("crop", "circle")
        intent.putExtra("aspectX", 1);//裁剪框的 X 方向的比例,需要为整数
        intent.putExtra("aspectY", 1);//裁剪框的 Y 方向的比例,需要为整数
        intent.putExtra("outputX", 60);//返回数据的时候的X像素大小。
        intent.putExtra("outputY", 60);//返回数据的时候的Y像素大小。
        //uritempFile为Uri类变量,实例化uritempFile
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
            if (TAKEPAHTO == 1) {//如果是7.0的拍照
                //开启临时访问的读和写权限
                intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION | Intent.FLAG_GRANT_READ_URI_PERMISSION);
                //针对7.0以上的操作
                intent.setClipData(ClipData.newRawUri(MediaStore.EXTRA_OUTPUT, uri));
                uriClipUri = uri;
            } else {//如果是7.0的相册
                //设置裁剪的图片地址Uri
                uriClipUri = Uri.parse("file://" + "/" + Environment.getExternalStorageDirectory().getPath() + "/" + "clip.jpg");
            }

        } else {
            uriClipUri = Uri.parse("file://" + "/" + Environment.getExternalStorageDirectory().getPath() + "/" + "clip.jpg");
        }
        Log.e("uriClipUri=====", "" + uriClipUri);
        //Android 对Intent中所包含数据的大小是有限制的,一般不能超过 1M,否则会使用缩略图 ,所以我们要指定输出裁剪的图片路径
        intent.putExtra(MediaStore.EXTRA_OUTPUT, uriClipUri);
        intent.putExtra("return-data", false);//是否将数据保留在Bitmap中返回
        intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());//输出格式,一般设为Bitmap格式及图片类型
        intent.putExtra("noFaceDetection", true);//人脸识别功能
        startActivityForResult(intent, PHOTO_PHOTOCLIP);//裁剪完成的标识

    }

图片的压缩(减小了图片加载时bitmap的大小,避免oom,但图片本身在手机中大小没有变)

我直接上方法吧:

   /**
     * 图片压缩的方法(只是内存减少,避免oom,图片本身在disk盘体积不变)
     * 显示的Bitmap占用的内存少一点,还是需要去设置加载的像素长度和宽度(变成缩略图)
     */
    public void compressPhto(File mFile){
//        BitmapFactory这个类就提供了多个解析方法(decodeResource、decodeStream、decodeFile等)用于创建Bitmap。
//        比如如果图片来源于网络,就可以使用decodeStream方法;
//        如果是sd卡里面的图片,就可以选择decodeFile方法;
//        如果是资源文件里面的图片,就可以使用decodeResource方法等
        BitmapFactory.Options options = new BitmapFactory.Options();
        options.inJustDecodeBounds = true; // 获取当前图片的边界大小
        //BitmapFactory.decodeResource(getResources(), R.drawable.bg, options);
        BitmapFactory.decodeFile(mFile.getAbsolutePath(),options);
        int outHeight = options.outHeight; //获取图片本身的高像素
        int outWidth = options.outWidth;//获取图片本身的宽的像素
        String outMimeType = options.outMimeType;
        options.inJustDecodeBounds = false;
        //inSampleSize的作用就是可以把图片的长短缩小inSampleSize倍,所占内存缩小inSampleSize的平方
        //对于inSampleSize值的大小有要求,最好是整数且2的倍数
        options.inSampleSize = caculateSampleSize(options, 500, 500);
        //etPath()得到的是构造file的时候的路径。getAbsolutePath()得到的是全路径
        String path =mFile.getPath();
        String absPath=mFile.getAbsolutePath();
        Bitmap bitmap = BitmapFactory.decodeFile(absPath,options);
        ivUserPhoto.setImageBitmap(bitmap);
        //尺寸压缩结果
        ivSize.setImageBitmap(bitmap);



    }

    /**
     * 计算出所需要压缩的大小
     * @param options
     * @param reqWidth  希望的图片宽大小
     * @param reqHeight 希望的图片高大小
     * @return
     */
    private int caculateSampleSize(BitmapFactory.Options options, int reqWidth, int reqHeight) {
        int sampleSize = 1;
        int picWidth = options.outWidth;
        int picHeight = options.outHeight;
        if (picWidth > reqWidth || picHeight > reqHeight) {
            int halfPicWidth = picWidth / 2;
            int halfPicHeight = picHeight / 2;
            while (halfPicWidth / sampleSize > reqWidth || halfPicHeight / sampleSize > reqHeight) {
                sampleSize *= 2;
            }
        }
        return sampleSize;
    }

3.图片的适配

7.0适配问题

不知道,大家注意裁剪方法里的这一段代码没?


image.png

还有拍照的这一段代码:


image.png

为什么要这么写呢?
原来7.0一下版本我们,直接调用相机获取的图片地址是:
file:///storage/emulated/0/temp.jpg的文件
然而7.0之后就变成:
content://........文件,使用 content://代替了 file:///
这是因为:Android 为了提高私有文件的安全性,从 7.0 开始对外传递file://类型的uri会触发FileUriExposedException。因此,在分享私有文件时必须使用FileProvider。
那么如果在使用之前的方法就会报错,我们要给程序在manifest文件中加入FileProvider:

image.png

我们看看provider_paths这个文件中都有啥?
image.png

这注解写的,不会,就转行吧。

裁剪完成

直接加载图片显示

image.png

4.图片的上传

上传图片
但还有中情况是我们要上传加载的图片,我也给大家提供了方法:

Bitmap photoBitmap;
File file;

/**
 * 上传图片
 */
public void upDateFile() {
    try {
        //裁剪后的图像转成BitMap
        photoBitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(uriClipUri));
    } catch (FileNotFoundException e) {
        e.printStackTrace();
    }
    //创建路径
    String path = Environment.getExternalStorageDirectory()
            .getPath() + "/Pic";
    //获取外部储存目录
    file = new File(path);
    //创建新目录, 创建此抽象路径名指定的目录,包括创建必需但不存在的父目录。
    file.mkdirs();
    //以当前时间重新命名文件
    long time = System.currentTimeMillis();
    //生成新的文件
    file = new File(file.toString() + "/" + time+ ".png");
    //创建输出流
    OutputStream out = null;
    try {
        out = new FileOutputStream(file.getPath());
    } catch (FileNotFoundException e) {
        e.printStackTrace();
    }
    //压缩文件,返回结果,true说明成功了
   boolean bCompress = photoBitmap.compress(Bitmap.CompressFormat.JPEG, 100, out);
}

4.其他知识点

  • 点击用户头像弹窗不是我们常用的PupUpWindow,而是自定义的BottomSheetDialog(待优化).
  • 用户头像被我做成了圆角,这个也是自定义ImageView,大家可以下载查看源码。

暂时这么多,以后补充!
Github地址

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352