欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

Android 调用系统相册选择照片

程序员文章站 2022-03-13 21:39:54
前言在相册里选择图片上传也是很常见的功能了例如微信朋友圈等等。但是他们是自定义的选择器,可以选择多张图片并修改。这里我们讲一个最简单的:调用系统的相册选择一张图片并展示。另外有的读者还想到要通过相机拍...

前言

在相册里选择图片上传也是很常见的功能了例如微信朋友圈等等。但是他们是自定义的选择器,可以选择多张图片并修改。这里我们讲一个最简单的:调用系统的相册选择一张图片并展示。另外有的读者还想到要通过相机拍照来选择图片的功能,也可以参考一下我的另一篇文章android使用系统相机进行拍照

使用步骤

这里我是通过一个简单的demo来讲解怎么去实现这个功能。首先看布局:

 <button
  android:id="@+id/button2"
  android:layout_width="wrap_content"
  android:layout_height="wrap_content"
  android:layout_margintop="5dp"
  android:layout_marginend="52dp"
  android:layout_marginright="52dp"
  android:text="choose"
  app:layout_constraintend_toendof="parent"
  app:layout_constrainttop_totopof="parent" />

 <imageview
  android:id="@+id/imageview"
  android:layout_width="wrap_content"
  android:layout_height="wrap_content"
  android:layout_margintop="29dp"
  app:layout_constraintend_toendof="parent"
  app:layout_constraintstart_tostartof="parent"
  app:layout_constrainttop_tobottomof="@+id/button"
  app:srccompat="@mipmap/ic_launcher_round" />

很简单,就是一个按钮和一个imageview。然后接下来让我们想想这个功能怎么去实现:

首先打开相册,那么肯定要通过隐式启动相册activity;然后相册返回一个路径,我们就拿这个路径把路径上对应的照片展示出来。思路挺简单的,让我们写写看:
首先看代码:

 private uri imageuri;
 private imageview imageview;

 @override
 protected void oncreate(bundle savedinstancestate) {
  super.oncreate(savedinstancestate);
  setcontentview(r.layout.activity_main);
  imageview = findviewbyid(r.id.imageview);
  button button1 = findviewbyid(r.id.button2);
  button1.setonclicklistener(new view.onclicklistener() {
   @override
   public void onclick(view v) {
   //动态申请权限
    if (contextcompat.checkselfpermission(mainactivity.this,manifest.permission
      .write_external_storage)!= packagemanager.permission_granted){
     activitycompat.requestpermissions(mainactivity.this,new string[]{manifest.permission.write_external_storage},1);
    }else{
    //执行启动相册的方法
     openalbum();
    }
   }
  });
  }
//获取权限的结果
@override
 public void onrequestpermissionsresult(int requestcode, @nonnull string[] permissions, @nonnull int[] grantresults) {
  if (requestcode == 1){
   if (grantresults.length>0&&grantresults[0] == packagemanager.permission_granted) openalbum();
   else toast.maketext(mainactivity.this,"你拒绝了",toast.length_short).show();
  }
 }

//启动相册的方法
private void openalbum(){
  intent intent = new intent("android.intent.action.get_content");
  intent.settype("image/*");
  startactivityforresult(intent,2);
 }

这里先初始化控件,然后动态申请权限,因为我们要读取照片肯定是要读取内存的权限,记得在androidmanifest中要写明权限:

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

获取权限后就打开相册选择。相册对应的action是android.intent.action.get_content,settype(“image/*”)这个方法表示把所有照片显示出来,然后开启活动。启动活动选择完照片后就会返回一个intent到onactivityresult方法中,所以接下来的主要工作就是如果获取到返回的路径。

我们知道在安卓4.4以后是不能把文件的真实路径直接给别的应用的,所以返回的uri是经过封装的,所以我们要进行解析取出里面的路径。所以这里我们要进行判断安卓版本来进行不同的逻辑,先看代码:

@override
 protected void onactivityresult(int requestcode, int resultcode, @nullable intent data) {
 if (requestcode == 2){
 //判断安卓版本
    if (resultcode == result_ok&&data!=null){
    if (build.version.sdk_int>=19)
    handimage(data);
    else handimagelow(data);
   }
  }
 }

//安卓版本大于4.4的处理方法
@requiresapi(api = build.version_codes.kitkat)
 private void handimage(intent data){
  string path =null;
  uri uri = data.getdata();
  //根据不同的uri进行不同的解析
  if (documentscontract.isdocumenturi(this,uri)){
   string docid = documentscontract.getdocumentid(uri);
   if ("com.android.providers.media.documents".equals(uri.getauthority())){
    string id = docid.split(":")[1];
    string selection = mediastore.images.media._id+"="+id;
    path = getimagepath(mediastore.images.media.external_content_uri,selection);
   }else if("com.android.providers.downloads.documents".equals(uri.getauthority())){
    uri contenturi = contenturis.withappendedid(uri.parse("content://downloads/public_downloads"),long.valueof(docid));
    path = getimagepath(contenturi,null);
   }
  }else if ("content".equalsignorecase(uri.getscheme())){
   path = getimagepath(uri,null);
  }else if ("file".equalsignorecase(uri.getscheme())){
   path = uri.getpath();
  }
  //展示图片
  displayimage(path);
 }


//安卓小于4.4的处理方法
private void handimagelow(intent data){
  uri uri = data.getdata();
  string path = getimagepath(uri,null);
  displayimage(path);
 }

//content类型的uri获取图片路径的方法
private string getimagepath(uri uri,string selection) {
  string path = null;
  cursor cursor = getcontentresolver().query(uri,null,selection,null,null);
  if (cursor!=null){
   if (cursor.movetofirst()){
    path = cursor.getstring(cursor.getcolumnindex(mediastore.images.media.data));
   }
   cursor.close();
  }
  return path;
 }

//根据路径展示图片的方法
private void displayimage(string imagepath){
  if (imagepath != null){
   bitmap bitmap = bitmapfactory.decodefile(imagepath);
   imageview.setimagebitmap(bitmap);
  }else{
   toast.maketext(this,"fail to set image",toast.length_short).show();
  }
 }

上面的代码很多但是不要慌,咱们一个一个来,不难理解的。首先我们知道不同的版本有两个不同的方法来展示图片,就是:handimage和handimagelow。content类型的uri通过getimagepath这个方法来获取真实路径,真实路径通过displayimage这个方法就可以展示出来了。所以主要的工作就是怎么拿到真实路径。现在思路清晰了,让我们一个个来看:

首先来看一下两个工具方法:getimagepath和displayimage。

  • getimagepath学过内容提供器会知道这个就是通过内容提供器来获取数据。通过这个uri以及selection获取到一个cursor对象。cursor是什么呢?不了解的读者可以查看这篇博客android中的cursor。然后通过这个cursor对象的mediastore.images.media.data这个参数就可以获取到真实路径了。
  • displayimage这个方法收一个真实路径字符串,直接通过bitmapfactory.decodefile这个方法获取到bitmap再显示出来就行了

了解了工具方法后,我们的目的就很明确啦:content类型的uri或者真实路径的string。
首先是版本低于4.4的,因为返回的是真实的uri,也就是content开头的那个,所以直接通过getimagepath获取真实路径再通过displayimage展示即可。

接下来这个可能看起来有点头疼,因为要解析不同类型的uri。我们一个个来看:

  • 第一种是document类型的uri。至于什么是document类型的uri这里就不深入了,只要知道有这种类型的uri,要怎么处理就好了。首先我们要获取一个documentid,然后再分两种情况处理:

第一种的是media格式的,然后我们要取出后半截字符串我们才能获取到真正的id,这里就真正的id指的是对应数据库表中的id,用于selection的。mediastore.images.media.external_content_uri就是这个照片的content类型uri,再把selection放进去即可。
第二种通过contenturis.withappendedid这个方法即可获取到content类型的uri,这个方法负责把id和contenturi连接成一个新的uri。这个方法在这里也不详细讲解。

  • 第二种的是content类型的,那不用说直接用就行了
  • 第三种的是file类型的,这个就是真实路径了,直接getpath就可以获取到了。

好了,到此我们的所有疑问也就解决了。

小结

看完之后是不是发现思路很简单但是实现起来很多的知识盲区呢?确实是这样。但是当我们把这些细节都解决了之后我们就会学到很多的东西,相当于以点带面。文中还有好多没有详解的:
contenturis,bitmapfactory,cursor,documentscontract等等。因为这是另外一块比较大的内容,如果要讲的话将会涉及到很多内容就很容易偏离我们的主题了,所以只要知道大概是什么就可以了。

参考资料

以上就是android 调用系统相册选择照片的详细内容,更多关于android 调用系统相册的资料请关注其它相关文章!