Android 拍照/选择照片并剪切成头像

头像裁剪上传功能在现在的App中基本都要用到,今天总结了一下,用比较简单实用的方式(调用系统API)来实现需求。这样做简单,但是不同的手机和系统上裁剪和选择图片的方式有一些区别。

首先来看一下效果图

相册里的图片本身像素比较低,后面是剪切之后强行放大的效果,所以看着比较模糊,实际问题上要计算好剪切的图片的大小。

GIF.gif

分析一下需求

主要有三个步骤:1.获取图片。2.裁剪图片。3.显示和上传

  • 1.获取图片,拍照和从相册获取。没啥好说的,调用系统。

  • 2.裁剪图片。这个可以用系统自带的,也可以用第三方框架。相比来说还是用系统自带的比较简单一些,下面介绍的也是用系统的。第三方框架可以使用这个 android-cropuCrop 或者这个cropper ,怎么使用请看Demo。

  • 3.显示和上传。这个就不用我说了吧。

具体实现

1. 获取图片
  • 拍照,调用相机拍照

    // 创建打开系统相机的意图                
    Intent takeIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
    // 设置参数。将拍照所得的照片存到磁盘中;照片文件的位置,此处是在外置SD卡
    takeIntent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(new File(getExternalCacheDir(), HEAD_ICON_NAME)));
    // 调用系统相机
    startActivityForResult(takeIntent, REQUESTCODE_TAKE);
    
  • 从系统图库中选择一张图片

    
    // 创建打开系统图库的意图
    Intent pickIntent = new Intent(Intent.ACTION_PICK, null);
    // 设置Intent的参数,选择MediaStore中的图片
    pickIntent.setDataAndType(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, "image/*");
    // 开启意图
    startActivityForResult(pickIntent, REQUESTCODE_PICK);
    
2. 裁剪图片
  •  1.得到图片
    
      
      /**
       * 获取图片,根据两种不同的情况下的requestCode分别获取图片
       */
      protected void onActivityResult(int requestCode, int resultCode, Intent data) {
          switch (requestCode) {
              case REQUESTCODE_PICK:// 直接从相册获取
                  try {
                        // 裁剪图片,相机返回的intent就是URI
                      startPhotoZoom(data.getData());
                  } catch (NullPointerException e) {
                      e.printStackTrace();// 用户点击取消操作时的异常
                  }
                  break;
              case REQUESTCODE_TAKE:// 调用相机拍照
                // 我们刚刚调用相机拍照的效果是将图片存到磁盘中,此时把图片拿出来
                  File temp = new File(getExternalCacheDir(), HEAD_ICON_NAME);
                // 转成URI给下面的裁剪图片的方法调用
                  startPhotoZoom(Uri.fromFile(temp));
                  break;
              case REQUESTCODE_CUTTING:// 取得裁剪后的图片
                  if (data != null) {
                      //setPicToView(data);
                      showIcon();
                  }
                  break;
          }
          super.onActivityResult(requestCode, resultCode, data);
      }
  •  2.裁剪图片
    
       mImageUri = Uri.fromFile(new File(getExternalCacheDir(), HEAD_ICON_NAME));

      /**
       * 裁剪图片方法实现
       *
       * @param uri 要裁剪的图片的URI
       */

      public void startPhotoZoom(Uri uri) {
          Intent intent = new Intent("com.android.camera.action.CROP");
          // crop=true是设置在开启的Intent中设置显示的VIEW可裁剪
          intent.setDataAndType(uri, "image/*");
          intent.putExtra("crop", "true");
          // aspectX aspectY 是宽高的比例
          intent.putExtra("aspectX", 1);
          intent.putExtra("aspectY", 1);
          // outputX outputY 是裁剪图片宽高,这里可以将宽高作为参数传递进来
          intent.putExtra("outputX", 600);
          intent.putExtra("outputY", 600);
        
          // 其实加上下面这两句就可以实现基本功能,
          //但是这样做我们会直接得到图片的数据,以bitmap的形式返回,在Intent中。而Intent传递数据大小有限制,1kb=1024字节,这样就对最后的图片的像素有限制。
          //intent.putExtra("return-data", true);
          //intent.putExtra(MediaStore.EXTRA_OUTPUT, tempUri);
             
          // 解决不能传图片,Intent传递数据大小有限制,1kb=1024字节
          // 方法:裁剪后的数据不以bitmap的形式返回,而是放到磁盘中,更方便上传和本地缓存
          // 设置裁剪后的数据不以bitmap的形式返回,剪切后图片的位置,图片是否压缩等
          intent.putExtra("return-data", false);
          intent.putExtra(MediaStore.EXTRA_OUTPUT, mImageUri);
          intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
          intent.putExtra("noFaceDetection", true);
            
          // 调用系统的图片剪切
          startActivityForResult(intent, REQUESTCODE_CUTTING);
      }
3. 获取裁剪后的图片并显示和上传

刚刚我们已经获得了裁剪后的图片的磁盘的位置,直接显示和上传就行了

  protected void onActivityResult(int requestCode, int resultCode, Intent data) {
      switch (requestCode) {
          case REQUESTCODE_PICK:// 直接从相册获取
              ... 
              break;
          case REQUESTCODE_TAKE:// 调用相机拍照
              ... 
              break;
          case REQUESTCODE_CUTTING:// 取得裁剪后的图片
              showIcon();// 显示图片,具体的代码就不写了
              upLoadIcon(); // 上传图片,具体的代码就不写了,随便用什么上传
              break;
      }
      super.onActivityResult(requestCode, resultCode, data);
  }

总的来说过程还是很简单的,一直在调用系统的API,来完成我们的需求

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

推荐阅读更多精彩内容