Android调用系统相机、相册、剪裁图片并上传(常用于上传头像,兼容Android7.0)

前言

在我们的日常开发当中,调用相机和从相册中选择照片裁剪并上传是很常见的功能,网上有很多框架,但是导入别人的库, 无疑增加了App的体积,因此这里讲一下如何使用系统自带的相机,相册,并裁剪,个人感觉还行

第一步 : FileProvider准备相关

  1. 在AndroidManifest.xml中增加provider节点,如下:
<provider
    android:name="android.support.v4.content.FileProvider"
    android:authorities="com.dream.takephotodemo"
    android:grantUriPermissions="true"
    android:exported="false">
    <meta-data
        android:name="android.support.FILE_PROVIDER_PATHS"
        android:resource="@xml/filepaths" />
</provider>

其中: android:authorities 表示授权列表,填写你的应用包名,当有多个授权时,用分号隔开 android:exported 表示该内容提供器(ContentProvider)是否能被第三方程序组件使用,必须为false,否则会报异常:ava.lang.RuntimeException: Unable to get provider android.support.v4.content.FileProvider: java.lang.SecurityException: Provider must not be exported android:grantUriPermissions="true" 表示授予 URI 临时访问权限 android:resource 属性指向我们自及创建的xml文件的路径,文件名随便起

  1. 接下来,我们需要在资源(res)目录下创建一个xml目录,并建立一个以上面名字为文件名的xml文件,内容如下:

     <?xml version="1.0" encoding="utf-8"?>
    <paths>
       <external-path path="." name="external_path" />
    </paths>
    

其中: external-path 代表根目录为: Environment.getExternalStorageDirectory() ,也可以写其他的,如: files-path 代表根目录为:Context.getFilesDir() cache-path 代表根目录为:getCacheDir() 其path属性的值代表路径后层级名称,为空则代表就是根目录,假如为“pictures”,就代表对应根目录下的pictures目录

第二步:使用FileProvider

  • 在这之前,我们需要在AndroidManifest.xml中增加必要的读写权限:

    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
    

1. 通过相机获取图片

/**
 * 拍照
 */ 
private void takePhoto() {
    //用于保存调用相机拍照后所生成的文件
    if (!Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {
        return;
    }
    captureFile = new File(rootFile, "temp.jpg");
    //跳转到调用系统相机
    Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
    //判断版本 如果在Android7.0以上,使用FileProvider获取Uri
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
        intent.setFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
        Uri contentUri = FileProvider.getUriForFile(mContext, getPackageName(), captureFile);
        intent.putExtra(MediaStore.EXTRA_OUTPUT, contentUri);
    } else {
        //否则使用Uri.fromFile(file)方法获取Uri
        intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(captureFile));
    }
    startActivityForResult(intent, REQUEST_PERMISSION_CAMERA);
}

2. 通过相册获取图片

 /**
 * 从相册选择
 */
private void choosePhoto() {
    Intent photoPickerIntent = new Intent(Intent.ACTION_PICK);
    photoPickerIntent.setType("image/*");
    startActivityForResult(photoPickerIntent, REQUEST_PERMISSION_WRITE);
}

3. 图片裁剪

  /**
 * 裁剪图片
 */
private void cropPhoto(Uri uri) {
    cropFile = new File(rootFile, "avatar.jpg");
    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", 300);
    intent.putExtra("outputY", 300);
    intent.putExtra("return-data", false);//注意这里返回false,因为在部分手机上获取不到返回的数据
    intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(cropFile));
    intent.putExtra("outputFormat", Bitmap.CompressFormat.PNG.toString());
    intent.putExtra("noFaceDetection", true);
    intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
    intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
    startActivityForResult(intent, CROP_REQUEST_CODE);
}

第三步:接收图片信息

  • 我们在onActivityResult方法中获得返回的图片信息,在这里我们会先调用剪裁去剪裁图片,然后对剪裁返回的图片进行设置、保存、上传等操作

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
      if (resultCode == RESULT_OK) {
          switch (requestCode) {
              case REQUEST_PERMISSION_CAMERA:
                  if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                      Uri contentUri = FileProvider.getUriForFile(mContext, getPackageName(), captureFile);
                      cropPhoto(contentUri);
                  } else {
                      cropPhoto(Uri.fromFile(captureFile));
                  }
                  break;
              case REQUEST_PERMISSION_WRITE:
                  cropPhoto(data.getData());
                  break;
              case CROP_REQUEST_CODE:
                  saveImage(cropFile.getAbsolutePath());
                  ivAvatar.setImageBitmap(BitmapFactory.decodeFile(cropFile.getAbsolutePath()));
                  break;
              default:
                  break;
          }
      }
      super.onActivityResult(requestCode, resultCode, data);
    }
    
  • 保存图片到本地

    public String saveImage(String path) {
      if (!Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {
          return null;
      }
      Bitmap bitmap = BitmapFactory.decodeFile(path);
      try {
          FileOutputStream fos = new FileOutputStream(cropFile);
          bitmap.compress(Bitmap.CompressFormat.JPEG, 100, fos);
          fos.flush();
          fos.close();
          return cropFile.getAbsolutePath();
      } catch (IOException e) {
          e.printStackTrace();
      }
      return null;
    }
    

至此,对Android7.0的兼容就结束了,注意在调用相机和裁剪时,传入的Uri需要使用FileProvider来获取

总结

拍照或从相册中选取时,我们首先要判断是否有拍照或读写SD卡的权限,如果没有则需要动态申请权限,这里我用的一个第三方库传送门,有了权限之后在进行下一步操作,还需要注意判断当前SD卡是否可用,做了这些判断之后,相信调用系统的拍照或者从相册中选择将会变得很简单

最后附上Demo地址传送门

如果大家对我的文章感兴趣的话,请为我点赞,谢谢!!!

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

推荐阅读更多精彩内容