Android仿QQ、新浪相册的实现

在移动应用中,很多时候都会用到图片选择、图片裁剪等功能。最近我也在准备一个开源的相册项目,以方便以后开发应用的时候使用,也尽可能的方便需要的人。一个完整的相册,应该包含相册列表、图片列表、图片的单选和多选、图片的裁剪、拍照、多选图片的大图预览等功能。这也是我这个项目将要包含的功能。在本篇博客中,将会讲述下我在这个项目中相册列表和图片列表的大致实现。

实现效果

结合几个常用的APP中的相册效果,当前项目中已经实现了一些基本的功能和UI,在后续完善的过程中还会有所变动。项目在Github上开源,欢迎fork和star。先展示实现的效果(后面会增加拍照功能):
单选效果 单选未选择时的效果 单选已选择的效果

功能分析

在实现相册功能之前,我们先需要明确它的逻辑。参照QQ、新浪、微博这中巨头级的APP,当我们需要用选择图片时,会先打开相册,获取到最新的照片列表。然后点击一个按钮可以展开相册列表,点击列表内容,可以切换相册,刷新当前照片列表中的内容。而且选择这篇的时候,会有单选、多选、单选并裁剪等情况,多选的时候还要出现选择效果和指示器等,单选的时候如果需要裁剪则进入裁剪页,不裁剪则默认确定选择,(拍照功能在后续博客中再说明)。
这样,我们就可以明确我们需要实现的功能有:

1.获取手机中的最新图片
2.获取手机中的相册列表
3.获取制定相册中的所有图片
4.展示图片和相册
5.多图选择时需要有选择效果和指示器
6.单选裁剪时需要用到裁剪功能

另外,扫描手机中的图片也是一个相对耗时的工作,所以这个工作还需要主要避免放到主线程中。

准备数据

为了使用方便,我们可以将相册列表的查询、制定相册的查询、最新图片的查询都放到一个工具类中,主要工具类代码如下:

public class AlbumTool {

 private Handler handler;
 //private Semaphore semaphore;
 private Callback callback;
 private Context context;

 private final int TYPE_FOLDER=1;
 private final int TYPE_ALBUM=2;

 public AlbumTool(Context context){
  this.context=context;
  handler=new Handler(Looper.getMainLooper()){
   @Override
   public void handleMessage(Message msg) {
    if(callback!=null){
     switch (msg.what){
      case TYPE_FOLDER:
       callback.onFolderFinish((ImageFolder) msg.obj);
       break;
      case TYPE_ALBUM:
       callback.onAlbumFinish((ArrayList<ImageFolder>) msg.obj);
       break;
     }
    }
    super.handleMessage(msg);
   }
  };
 }

 public void setCallback(Callback callback){
  this.callback=callback;
 }

 public void findAlbumsAsync(){
  new Thread(new Runnable() {
   @Override
   public void run() {
    getAlbums(context);
   }
  }).start();
 }

 public void findFolderAsync(final ImageFolder folder){
  new Thread(new Runnable() {
   @Override
   public void run() {
    getFolder(context,folder);
   }
  }).start();
 }

 //获取所有图片集
 private ArrayList<ImageFolder> getAlbums(Context context) {
  ArrayList<ImageFolder> albums=new ArrayList<>();
  albums.add(getNewestPhotos(context));
  //利用ContentResolver查询数据库,找出所有包含图片的文件夹,保存到相册列表中
  ContentResolver resolver = context.getContentResolver();
  Cursor cursor = resolver.query(MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
    new String[]{
      MediaStore.Images.Media.DATA,
      MediaStore.Images.ImageColumns.BUCKET_ID,
      MediaStore.Images.Media.DATE_MODIFIED,
      "count(*) as count"
    },
    MediaStore.Images.Media.MIME_TYPE + "=? or " +
      MediaStore.Images.Media.MIME_TYPE + "=? or " +
      MediaStore.Images.Media.MIME_TYPE + "=?) " +
      "group by (" + MediaStore.Images.ImageColumns.BUCKET_ID,
    new String[]{"image/jpeg", "image/png", "image/jpg"},
    MediaStore.Images.Media.DATE_MODIFIED + " desc");
  if (cursor != null) {
   while (cursor.moveToNext()) {
    final File file = new File(cursor.getString(0));
    ImageFolder imageFolder = new ImageFolder();
    imageFolder.setDir(file.getParent());
    imageFolder.setId(cursor.getString(1));
    imageFolder.setFirstImagePath(cursor.getString(0));
    String[] all=file.getParentFile().list(new FilenameFilter() {

     private boolean e(String filename,String ends){
      return filename.toLowerCase().endsWith(ends);
     }

     @Override
     public boolean accept(File dir, String filename) {
      return e(filename,".png") || e(filename,".jpg") || e(filename,"jpeg");
     }
    });
    if(all!=null&&all.length>0){
     imageFolder.setCount(all.length);
     albums.add(imageFolder);
    }
   }
   cursor.close();
  }
  sendMessage(TYPE_ALBUM,albums);
  return albums;
 }

 //获取《最新图片》集
 private ImageFolder getNewestPhotos(Context context) {
  ImageFolder newestFolder=new ImageFolder();
  newestFolder.setName(ChooserSetting.newestAlbumName);
  ArrayList<ImageInfo> imageBeans = new ArrayList<>();
  ContentResolver resolver = context.getContentResolver();
  Cursor cursor = resolver.query(MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
    new String[]{
      MediaStore.Images.Media.DATA,
      MediaStore.Images.Media.DISPLAY_NAME,
      MediaStore.Images.Media.DATE_MODIFIED,
    },
    MediaStore.Images.Media.MIME_TYPE + "=? or "
      + MediaStore.Images.Media.MIME_TYPE + "=? or "
      + MediaStore.Images.Media.MIME_TYPE + "=?",
    new String[]{"image/jpeg", "image/png", "image/jpg"},
    MediaStore.Images.Media.DATE_MODIFIED + " desc"
      + (ChooserSetting.newestAlbumSize < 0 ? ""
      : (" limit " + ChooserSetting.newestAlbumSize)));
  if (cursor != null){
   while (cursor.moveToNext()) {
    ImageInfo info=new ImageInfo();
    info.path=cursor.getString(0);
    info.displayName=cursor.getString(1);
    info.time=cursor.getLong(2);
    imageBeans.add(info);
   }
   cursor.close();
   newestFolder.setFirstImagePath(imageBeans.get(0).path);
   newestFolder.setDatas(imageBeans);
   newestFolder.setCount(imageBeans.size());
  }
  sendMessage(TYPE_FOLDER,newestFolder);
  return newestFolder;
 }

 //获取具体图片集,确保图片数据已被查询
 private ImageFolder getFolder(Context context,ImageFolder folder) {
  ContentResolver resolver = context.getContentResolver();
  Cursor cursor;
  if(folder!=null&&folder.getDatas()!=null&&folder.getDatas().size()>0){
   sendMessage(TYPE_FOLDER,folder);
   return folder;
  }
  if (folder == null) {
   return getNewestPhotos(context);
  } else {
   cursor = resolver.query(MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
     new String[]{
       MediaStore.Images.Media.DATA,
       MediaStore.Images.Media.DISPLAY_NAME,
       MediaStore.Images.Media.DATE_MODIFIED
     },
     MediaStore.Images.ImageColumns.BUCKET_ID + "=? and (" +
       MediaStore.Images.Media.MIME_TYPE + "=? or "
       + MediaStore.Images.Media.MIME_TYPE + "=? or "
       + MediaStore.Images.Media.MIME_TYPE + "=?) ",
     new String[]{folder.getId(), "image/jpeg", "image/png", "image/jpg"},
     MediaStore.Images.Media.DATE_MODIFIED + " desc");
  }
  ArrayList<ImageInfo> datas=new ArrayList<>();
  folder.setDatas(datas);
  if (cursor != null){
   while (cursor.moveToNext()) {
    ImageInfo info=new ImageInfo();
    info.path=cursor.getString(0);
    info.displayName=cursor.getString(1);
    info.time=cursor.getLong(2);
    datas.add(info);
   }
   cursor.close();
  }
  sendMessage(TYPE_FOLDER,folder);
  return folder;
 }

 private void sendMessage(int what,Object obj){
  Message msg=new Message();
  msg.what=what;
  msg.obj=obj;
  handler.sendMessage(msg);
 }

 public interface Callback{

  //文件夹查找完毕
  void onFolderFinish(ImageFolder folder);
  //成功搜索出所有的图片集
  void onAlbumFinish(ArrayList<ImageFolder> albums);

 }

}

这样,我们就可以利用这个工具类方便的获取相册列表、获取制定相册的图片了(最新照片合集当做是一个相册)。里面主要就是使用ContentResolver来做查询,Android入门级问题,四大组件——Activity、Service、ContentProvider和BroadcastReceiver,中的ContentProvider和ContentResolver就是一对CP了,ContentProvider用来提供数据,ContentResolver用来获取数据。

展示相册和相册列表

有了获取相册列表和获取指定相册的方法,展示相册和相册列表就容易了,按照通常的方式,我们直接使用GridView来展示相册,用ListView来展示相册列表。当然,你也可以选择使用RecyclerView来替代掉GridView和ListView,其实也都一样。
显示图片直接使用成熟的第三方框架即可,我使用的是Glide。
值得注意的是,在相册中,我们展示出来的图片都是正方块、并且需要三个(你也可以设置四个或者五个,只要你高兴)铺满宽度。在这里我使用的是比较懒的方式,直接用一个自定义的布局作为Item的跟布局,这个自定义布局继承RelativeLayout,然后将复写它的onMeasure方法:

@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
 super.onMeasure(widthMeasureSpec, widthMeasureSpec);
}

心有多懒,人就能有多懒。这样它的高度就被强制保持为何宽度一致了。

选择指示器

像QQ中,选择图片时,图片会根据选择的顺序,在图片上的那个圈圈里面显示出1234……等数字,然后取消选择时,被选的数字会顺序补位,比如你选了七张图片、然后取消了显示数字3的那张,这时4就变成3了、5变成了4、6变成了5。
像新浪微博中的图片选择,不会出现数字,而是出现一个勾,选中的时候这个勾还有动画效果。
这样的功能怎么实现呢?
我实现的方式是,在每个Item中都有一个固定大小的View,根据图片是否被选中,加载不同的Drawable。当然,写这个项目既然是为了以后在不同的项目中使用,这个自然要方便被使用者自行设置。所以我写一个抽象类:

public abstract class IChooseDrawable{

 private Paint paint;
 protected int width=0;
 protected int height=0;

 private SparseArray<Drawable> drawables;

 public IChooseDrawable(){
  paint=new Paint();
  paint.setAntiAlias(true);
  paint.setColor(0x88000000);
  drawables=new SparseArray<>();
 }

 public Drawable get(int state){
  if(drawables.indexOfKey(state)>=0){
   return drawables.get(state);
  }else{
   InDrawable drawable=new InDrawable(state);
   drawables.put(state,drawable);
   return drawable;
  }
 }

 public void clear(){
  drawables.clear();
 }

 public int getBaseline(Paint paint,int top,int bottom){
  Paint.FontMetrics i=paint.getFontMetrics();
  return (int) ((bottom+top-i.top-i.bottom)/2);
 }

 //state表示第几个被选择,0表示未选中
 public abstract void draw(Canvas canvas,Paint paint,int state);

 private class InDrawable extends Drawable{

  private int state=0;

  InDrawable(int state){
   this.state=state;
  }

  @Override
  public void draw(@NonNull Canvas canvas) {
   IChooseDrawable.this.draw(canvas,paint,state);
  }

  @Override
  public void setAlpha(int alpha) {

  }

  @Override
  public void setColorFilter(ColorFilter colorFilter) {

  }

  @Override
  public int getOpacity() {
   return PixelFormat.TRANSPARENT;
  }
 }
}

在相册的Adapter的构造函数中会传入一个IChooseDrawable实体,在显示每个Item时,会根据当前状态通过drawable.get(int state)取得指定的Drawable,设置为指示器View的背景。
上面效果图中的指示器(也可配置为只显示对号)实现为:

public class CircleChooseDrawable extends IChooseDrawable {

 private boolean isShowNum=true;
 private int chooseBgColor=0xFFFF6600;
 private Path path;

 public CircleChooseDrawable(){
  super();
 }

 public CircleChooseDrawable(boolean isShowNum,int chooseBgColor){
  super();
  this.isShowNum=isShowNum;
  this.chooseBgColor=chooseBgColor;
 }

 @Override
 public void draw(Canvas canvas, Paint paint, int state) {
  width=canvas.getWidth();
  height=canvas.getHeight();
  if(state==0){ //未选择状态
   paint.setColor(0x55000000);
   paint.setStyle(Paint.Style.FILL);
   canvas.drawCircle(width/2,height/2,width/2-2,paint);
   paint.setColor(0xDDFFFFFF);
   paint.setStrokeWidth(2);
   paint.setStyle(Paint.Style.STROKE);
   canvas.drawCircle(width/2,height/2,width/2-2,paint);
  }else{ //选中状态
   paint.setColor(chooseBgColor);
   paint.setStyle(Paint.Style.FILL);
   canvas.drawCircle(width/2,height/2,width/2-2,paint);
   paint.setColor(0xDDFFFFFF);
   paint.setStrokeWidth(2);
   paint.setStyle(Paint.Style.STROKE);
   canvas.drawCircle(width/2,height/2,width/2-2,paint);
   paint.setColor(0xDDFFFFFF);
   if(isShowNum){ //显示数字
    paint.setStyle(Paint.Style.FILL);
    paint.setTextAlign(Paint.Align.CENTER);
    paint.setTextSize(width*0.53f);
    canvas.drawText(state+"",width/2,getBaseline(paint,0,height),paint);
   }else{ //显示一个√号
    paint.setStyle(Paint.Style.STROKE);
    paint.setStrokeWidth(3);
    paint.setStrokeCap(Paint.Cap.ROUND);
    if(path==null){
     path=new Path();
     path.moveTo(width/4f,height/2f);
     path.lineTo(width*2/5f,height*5/7f);
     path.lineTo(width*3/4f,height/3f);
    }
    canvas.drawPath(path,paint);
   }
  }
 }
}

裁剪、单选和多选

单选和多选的区别在于单选的时候,没有选择指示器,选中直接携带数据返回。而多选时,有选择指示器,选择完成后,需要确定后携带数据返回,在确定前可以取消之前所选的内容。
所以实现的时候,只需要判断用户传入的选择意图,做出相应的处理。如果是裁剪,则选择一张图片后,进入到裁剪页面,裁剪结束后携带裁剪结果返回到进入到相册前的页面。如果是单选,则选择一张图片后,直接携带数据返回到进入相册前的页面。如果是多选,则要在点击确认按钮后,携带数据返回到进入相册前的页面。裁剪的实现见上一篇博客——Android 图片裁剪。

其他

其他的一些功能,主要是拍照的功能、和大图切换预览现在还未添加进项目中,目前准备是利用OpenGl做拍照预览和拍照(也许会添加些许常用滤镜),实现的相关细节也会在后续单独写博客来介绍。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android实现调用系统相册和拍照的Demo示例

    本文讲述了Android实现调用系统相册和拍照的Demo示例.分享给大家供大家参考,具体如下: 最近我在群里看到有好几个人在交流说现在网上的一些Android调用系统相册和拍照的demo都有bug,有问题,没有一个完整的.确实是,我记得一个月前,我一同学也遇到了这样的问题,在低版本的系统中没问题,用高于4.4版本的系统就崩溃.所以,我还是想提取出来,给大家整理一下,一个比较完整无bug的demo,让大家收藏,留着以后用. 其实对于调用手机图库,高版本的系统会崩溃,是因为获取方法变了,所以我们应该

  • 解决Android从相册中获取图片出错图片却无法裁剪问题的方法

    在学习获取相册中图片进行裁剪的时候遇到了比较大的问题,在纠结了近半天才真的解决,下面分享一下学习经验. 问题: 选择了相册中的图片之后要进入图片裁剪的时候出错,(华为)手机提示"此图片无法获取",经百度后,明白是版本不同导致的URI的问题的问题,原文如下: 4.3或以下,选了图片之后,根据Uri来做处理,很多帖子都有了,我就不详细说了.主要是4.4,如果使用上面pick的原生方法来选图,返回的uri还是正常的,但如果用ACTION_GET_CONTENT的方法,返回的uri跟4.3是完

  • Android仿微信QQ设置图形头像裁剪功能

    最近在做毕业设计,想有一个功能和QQ一样可以裁剪头像并设置圆形头像,额,这是设计狮的一种潮流. 而纵观现在主流的APP,只要有用户系统这个功能,这个需求一般都是在(bu)劫(de)难(bu)逃(xue)! 图片裁剪实现方式有两种,一种是利用系统自带的裁剪工具,一种是使用开源工具Cropper.本节就为大家带来如何使用系统自带的裁剪工具进行图片裁剪~ 还是先来个简单的运行图. 额,简单说下,我待会会把代码写成小demo分享给大家,在文章末尾会附上github链接,需要的可以自行下载~ 下面来简单分

  • android照相、相册获取图片剪裁报错的解决方法

    这是调用相机 public static File getImageFromCamer(Context context, File cameraFile, int REQUE_CODE_CAMERA, Intent intent) { intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); File fileDir = HelpUtil.getFile(context, "/Tour/user_photos"); cameraFile

  • Android仿QQ圆形头像个性名片

    先看看效果图: 中间的圆形头像和光环波形讲解请看:http://www.jb51.net/article/96508.htm 周围的气泡布局,因为布局RatioLayout是继承自ViewGroup,所以布局layout就可以根据自己的需求来布局其子view,view.layout(int l,int t,int r,int b);用于布局子view在父ViewGroup中的位置(相对于父容器),所以在RatioLayout中计算所有子view的left,top,right,bottom.那么头

  • Android自定义控件仿QQ编辑和选取圆形头像

    android大家都有很多需要用户上传头像的需求,有的是选方形,有的是圆角矩形,有的是圆形. 首先我们要做一个处理图片的自定义控件,把传入的图片,经过用户选择区域,处理成一定的形状. 有的app是通过在图片上画一个矩形区域表示选中的内容,有的则是通过双指放大缩小,拖动图片来选取图片.圆形头像,还是改变图片比较好 圆形区域可调节大小. 这个自定义View的图像部分分为三个,背景图片,半透明蒙层,和亮色区域--还是直接贴代码得了 package com.example.jjj.widget; imp

  • Android 仿QQ头像自定义截取功能

    看了Android版QQ的自定义头像功能,决定自己实现,随便熟悉下android绘制和图片处理这一块的知识. 先看看效果: 思路分析: 这个效果可以用两个View来完成,上层View是一个遮盖物,绘制半透明的颜色,中间挖了一个圆:下层的View用来显示图片,具备移动和缩放的功能,并且能截取某区域内的图片. 涉及到的知识点: 1.Matrix,图片的移动和缩放 2.Paint的setXfermode方法 3.图片放大移动后,截取一部分 编码实现: 自定义三个View: 1.下层View:ClipP

  • android获取相册图片和路径的实现方法

    Android开发获取相册图片的方式网上有很多种,这里说一个Android4.4后的方法,因为版本越高,一些老的api就会被弃用,新的api和老的api不兼容,导致出现很多问题. 比如:managedQuery()现在已经被getContentResolver().query()替代了,不过它们的参数都是一样的 再比如Android4.4后Intent(Intent.ACTION_GET_CONTENT);和Intent(Intent.ACTION_OPEN_DOCUMENT);两个方法所得到的

  • Android实现读取相机(相册)图片并进行剪裁

    我们先说一下思路,在android系统中就自带了图片剪切的应用,所以,我们只需要将我们获取到的相片传给图片剪切应用,再将剪切好的相片返回到我们自己的界面显示就ok了 在开发一些APP的过程中,我们可能涉及到头像的处理,比如从手机或者相册获取头像,剪裁成自己需要的头像,设置或上传头像等.网上一些相关的资料也是多不胜数,但在实际应用中往往会存在各种问题,没有一个完美的解决方案.由于近期项目的需求,就研究了一下,目前看来还没有什么问题. 这里我们只讨论获取.剪裁与设置,上传流程根据自己的业务需求添加.

  • 基于Android实现保存图片到本地并可以在相册中显示出来

    App应用越来越人性化,不仅界面优美而且服务也很多样化,操作也非常方便.比如我们在用app的时候,发现上面有比较的图片想保存到手机,只要点一点app上提供的保存按钮就可以了.那这个图片保存到本地怎么实现的呢? 保存图片很简单,方法如下: /** 首先默认个文件保存路径 */ private static final String SAVE_PIC_PATH=Environment.getExternalStorageState().equalsIgnoreCase(Environment.MED

随机推荐