热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android应用中使用ContentProvider扫描本地图片并显示

这篇文章主要介绍了Android应用中使用ContentProvider扫描本地图片并显示的方法,比调用本地图库的方法更加灵活和可定制,需要的朋友可以参考下

之前群里面有朋友问我,有没有关于本地图片选择的Demo,类似微信的效果,他说网上没有这方面的Demo,问我能不能写一篇关于这个效果的Demo,于是我研究了下微信的本地图片选择的Demo,自己仿照的写了下分享给大家,希望对以后有这样子需求的朋友有一点帮助吧,主要使用的是ContentProvider扫描手机中的图片,并用GridView将图片显示出来,关于GridView和ListView显示图片的问题,一直是一个很头疼的问题,因为我们手机的内存有限,手机给每个应用程序分配的内存也有限,所以图片多的情况下很容易伴随着OOM的发生,不过现在也有很多的开源的图片显示框架,对显示很多图片进行了优化,大家有兴趣的可以去了解了解,今天我的这篇文章使用的是LruCache这个类以及对图片进行相对应的裁剪,这样也可以尽量的避免OOM的发生,我们先看下微信的效果吧

201645151626071.png (480×854)

201645151655089.png (480×854)

接下来我们就来实现这些效果吧,首先我们新建一个项目,取名ImageScan
首先我们先看第一个界面吧,使用将手机中的图片扫描出来,然后根据图片的所在的文件夹将其分类出来,并显示所在文件夹里面的一张图片和文件夹中图片个数,我们根据界面元素(文件夹名, 文件夹图片个数,文件夹中的一张图片)使用一个实体对象ImageBean来封装这三个属性

package com.example.imagescan; 
 
/** 
 * GridView的每个item的数据对象 
 * 
 * @author len 
 * 
 */ 
public class ImageBean{ 
  /** 
   * 文件夹的第一张图片路径 
   */ 
  private String topImagePath; 
  /** 
   * 文件夹名 
   */ 
  private String folderName;  
  /** 
   * 文件夹中的图片数 
   */ 
  private int imageCounts; 
   
  public String getTopImagePath() { 
    return topImagePath; 
  } 
  public void setTopImagePath(String topImagePath) { 
    this.topImagePath = topImagePath; 
  } 
  public String getFolderName() { 
    return folderName; 
  } 
  public void setFolderName(String folderName) { 
    this.folderName = folderName; 
  } 
  public int getImageCounts() { 
    return imageCounts; 
  } 
  public void setImageCounts(int imageCounts) { 
    this.imageCounts = imageCounts; 
  } 
   
} 

接下来就是主界面的布局啦,上面的导航栏我没有加进去,只有下面的GridView,所以说主界面布局中只有一个GridView

 
 
   
   
 
 

接下来就是GridView的Item的布局,看上面的图也行你会认为他的效果是2张图片添加的效果,其实不是,后面的叠加效果只是一张背景图片而已,代码先贴上来

<&#63;xml version="1.0" encoding="UTF-8"&#63;> 
 
 
   
 
   
 
 

看到上面的布局代码,也行你已经发现了,上面使用的是自定义的MyImageView,我先不说这个自定义MyImageView的作用,待会再给大家说,我们继续看代码
第一个界面的主要代码

package com.example.imagescan; 
 
import java.io.File; 
import java.util.ArrayList; 
import java.util.HashMap; 
import java.util.Iterator; 
import java.util.List; 
import java.util.Map; 
 
import android.app.Activity; 
import android.app.ProgressDialog; 
import android.content.ContentResolver; 
import android.content.Intent; 
import android.database.Cursor; 
import android.net.Uri; 
import android.os.Bundle; 
import android.os.Handler; 
import android.os.Message; 
import android.provider.MediaStore; 
import android.view.View; 
import android.widget.AdapterView; 
import android.widget.AdapterView.OnItemClickListener; 
import android.widget.GridView; 


public class MainActivity extends Activity { 
  private HashMap> mGruopMap = new HashMap>(); 
  private List list = new ArrayList(); 
  private final static int SCAN_OK = 1; 
  private ProgressDialog mProgressDialog; 
  private GroupAdapter adapter; 
  private GridView mGroupGridView; 
   
  private Handler mHandler = new Handler(){ 
 
    @Override 
    public void handleMessage(Message msg) { 
      super.handleMessage(msg); 
      switch (msg.what) { 
      case SCAN_OK: 
        //关闭进度条 
        mProgressDialog.dismiss(); 
         
        adapter = new GroupAdapter(MainActivity.this, list = subGroupOfImage(mGruopMap), mGroupGridView); 
        mGroupGridView.setAdapter(adapter); 
        break; 
      } 
    } 
     
  }; 
 
  @Override 
  protected void onCreate(Bundle savedInstanceState) { 
    super.onCreate(savedInstanceState); 
    setContentView(R.layout.activity_main); 
     
    mGroupGridView = (GridView) findViewById(R.id.main_grid); 
     
    getImages(); 
     
    mGroupGridView.setOnItemClickListener(new OnItemClickListener() { 
 
      @Override 
      public void onItemClick(AdapterView<&#63;> parent, View view, 
          int position, long id) { 
        List childList = mGruopMap.get(list.get(position).getFolderName()); 
         
        Intent mIntent = new Intent(MainActivity.this, ShowImageActivity.class); 
        mIntent.putStringArrayListExtra("data", (ArrayList)childList); 
        startActivity(mIntent); 
         
      } 
    }); 
     
  } 


  /** 
   * 利用ContentProvider扫描手机中的图片,此方法在运行在子线程中 
   */ 
  private void getImages() { 
    //显示进度条 
    mProgressDialog = ProgressDialog.show(this, null, "正在加载..."); 
     
    new Thread(new Runnable() { 
       
      @Override 
      public void run() { 
        Uri mImageUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI; 
        ContentResolver mCOntentResolver= MainActivity.this.getContentResolver(); 
 
        //只查询jpeg和png的图片 
        Cursor mCursor = mContentResolver.query(mImageUri, null, 
            MediaStore.Images.Media.MIME_TYPE + "=&#63; or " 
                + MediaStore.Images.Media.MIME_TYPE + "=&#63;", 
            new String[] { "image/jpeg", "image/png" }, MediaStore.Images.Media.DATE_MODIFIED); 
         
        if(mCursor == null){ 
          return; 
        } 
         
        while (mCursor.moveToNext()) { 
          //获取图片的路径 
          String path = mCursor.getString(mCursor 
              .getColumnIndex(MediaStore.Images.Media.DATA)); 
           
          //获取该图片的父路径名 
          String parentName = new File(path).getParentFile().getName(); 
 
           
          //根据父路径名将图片放入到mGruopMap中 
          if (!mGruopMap.containsKey(parentName)) { 
            List chileList = new ArrayList(); 
            chileList.add(path); 
            mGruopMap.put(parentName, chileList); 
          } else { 
            mGruopMap.get(parentName).add(path); 
          } 
        } 
         
        //通知Handler扫描图片完成 
        mHandler.sendEmptyMessage(SCAN_OK); 
        mCursor.close(); 
      } 
    }).start(); 
     
  } 
      
  /** 
   * 组装分组界面GridView的数据源,因为我们扫描手机的时候将图片信息放在HashMap中 
   * 所以需要遍历HashMap将数据组装成List 
   * 
   * @param mGruopMap 
   * @return 
   */ 
  private List subGroupOfImage(HashMap> mGruopMap){ 
    if(mGruopMap.size() == 0){ 
      return null; 
    } 
    List list = new ArrayList(); 
     
    Iterator>> it = mGruopMap.entrySet().iterator(); 
    while (it.hasNext()) { 
      Map.Entry> entry = it.next(); 
      ImageBean mImageBean = new ImageBean(); 
      String key = entry.getKey(); 
      List value = entry.getValue(); 
       
      mImageBean.setFolderName(key); 
      mImageBean.setImageCounts(value.size()); 
      mImageBean.setTopImagePath(value.get(0));//获取该组的第一张图片 
       
      list.add(mImageBean); 
    } 
     
    return list; 
     
  } 
 
 
} 
首先看getImages()这个方法,该方法是使用ContentProvider将手机中的图片扫描出来,我这里只扫描了手机的外部存储中的图片,由于手机中可能存在很多的图片,扫描图片又比较耗时,所以我们在这里开启了子线程去获取图片,扫描的图片都存放在Cursor中,我们先要将图片按照文件夹进行分类,我们使用了HashMap来进行分类并将结果存储到mGruopMap(Key是文件夹名,Value是文件夹中的图片路径的List)中,分类完了关闭Cursor并利用Handler来通知主线程
然后是subGroupOfImage()方法,改方法是将mGruopMap的数据组装到List中,在List中存放GridView中的每个item的数据对象ImageBean, 遍历HashMap对象,具体的逻辑看代码,之后就是给GridView设置Adapter。
设置item点击事件,点击文件夹跳转到展示文件夹图片的Activity, 我们需要传递每个文件夹中的图片的路径的集合
看GroupAdapter的代码之前,我们先看一个比较重要的类,本地图片加载器NativeImageLoader
package com.example.imagescan; 
 
import java.util.concurrent.ExecutorService; 
import java.util.concurrent.Executors; 
 
import android.graphics.Bitmap; 
import android.graphics.BitmapFactory; 
import android.graphics.Point; 
import android.os.Handler; 
import android.os.Message; 
import android.support.v4.util.LruCache; 
 
/** 
 * 本地图片加载器,采用的是异步解析本地图片,单例模式利用getInstance()获取NativeImageLoader实例 
 * 调用loadNativeImage()方法加载本地图片,此类可作为一个加载本地图片的工具类 
 */ 
public class NativeImageLoader { 
  private LruCache mMemoryCache; 
  private static NativeImageLoader mInstance = new NativeImageLoader(); 
  private ExecutorService mImageThreadPool = Executors.newFixedThreadPool(1); 
   
   
  private NativeImageLoader(){ 
    //获取应用程序的最大内存 
    final int maxMemory = (int) (Runtime.getRuntime().maxMemory() / 1024); 
 
    //用最大内存的1/4来存储图片 
    final int cacheSize = maxMemory / 4; 
    mMemoryCache = new LruCache(cacheSize) { 
       
      //获取每张图片的大小 
      @Override 
      protected int sizeOf(String key, Bitmap bitmap) { 
        return bitmap.getRowBytes() * bitmap.getHeight() / 1024; 
      } 
    }; 
  } 
   
  /** 
   * 通过此方法来获取NativeImageLoader的实例 
   * @return 
   */ 
  public static NativeImageLoader getInstance(){ 
    return mInstance; 
  } 
      
      
  /** 
   * 加载本地图片,对图片不进行裁剪 
   * @param path 
   * @param mCallBack 
   * @return 
   */ 
  public Bitmap loadNativeImage(final String path, final NativeImageCallBack mCallBack){ 
    return this.loadNativeImage(path, null, mCallBack); 
  } 
   
  /** 
   * 此方法来加载本地图片,这里的mPoint是用来封装ImageView的宽和高,我们会根据ImageView控件的大小来裁剪Bitmap 
   * 如果你不想裁剪图片,调用loadNativeImage(final String path, final NativeImageCallBack mCallBack)来加载 
   * @param path 
   * @param mPoint 
   * @param mCallBack 
   * @return 
   */ 
  public Bitmap loadNativeImage(final String path, final Point mPoint, final NativeImageCallBack mCallBack){ 
    //先获取内存中的Bitmap 
    Bitmap bitmap = getBitmapFromMemCache(path); 
     
    final Handler mHander = new Handler(){ 
 
      @Override 
      public void handleMessage(Message msg) { 
        super.handleMessage(msg); 
        mCallBack.onImageLoader((Bitmap)msg.obj, path); 
      } 
       
    }; 
     
    //若该Bitmap不在内存缓存中,则启用线程去加载本地的图片,并将Bitmap加入到mMemoryCache中 
    if(bitmap == null){ 
      mImageThreadPool.execute(new Runnable() { 
         
        @Override 
        public void run() { 
          //先获取图片的缩略图 
          Bitmap mBitmap = decodeThumbBitmapForFile(path, mPoint == null &#63; 0: mPoint.x, mPoint == null &#63; 0: mPoint.y); 
          Message msg = mHander.obtainMessage(); 
          msg.obj = mBitmap; 
          mHander.sendMessage(msg); 
           
          //将图片加入到内存缓存 
          addBitmapToMemoryCache(path, mBitmap); 
        } 
      }); 
    } 
    return bitmap; 
     
  } 
 
   
   
  /** 
   * 往内存缓存中添加Bitmap 
   * 
   * @param key 
   * @param bitmap 
   */ 
  private void addBitmapToMemoryCache(String key, Bitmap bitmap) { 
    if (getBitmapFromMemCache(key) == null && bitmap != null) { 
      mMemoryCache.put(key, bitmap); 
    } 
  } 
 
  /** 
   * 根据key来获取内存中的图片 
   * @param key 
   * @return 
   */ 
  private Bitmap getBitmapFromMemCache(String key) { 
    return mMemoryCache.get(key); 
  } 
   
   
  /** 
   * 根据View(主要是ImageView)的宽和高来获取图片的缩略图 
   * @param path 
   * @param viewWidth 
   * @param viewHeight 
   * @return 
   */ 
  private Bitmap decodeThumbBitmapForFile(String path, int viewWidth, int viewHeight){ 
    BitmapFactory.Options optiOns= new BitmapFactory.Options(); 
    //设置为true,表示解析Bitmap对象,该对象不占内存 
    options.inJustDecodeBounds = true; 
    BitmapFactory.decodeFile(path, options); 
    //设置缩放比例 
    options.inSampleSize = computeScale(options, viewWidth, viewHeight); 
     
    //设置为false,解析Bitmap对象加入到内存中 
    options.inJustDecodeBounds = false; 
     
    return BitmapFactory.decodeFile(path, options); 
  } 
   
   
  /** 
   * 根据View(主要是ImageView)的宽和高来计算Bitmap缩放比例。默认不缩放 
   * @param options 
   * @param width 
   * @param height 
   */ 
  private int computeScale(BitmapFactory.Options options, int viewWidth, int viewHeight){ 
    int inSampleSize = 1; 
    if(viewWidth == 0 || viewWidth == 0){ 
      return inSampleSize; 
    } 
    int bitmapWidth = options.outWidth; 
    int bitmapHeight = options.outHeight; 
     
    //假如Bitmap的宽度或高度大于我们设定图片的View的宽高,则计算缩放比例 
    if(bitmapWidth > viewWidth || bitmapHeight > viewWidth){ 
      int widthScale = Math.round((float) bitmapWidth / (float) viewWidth); 
      int heightScale = Math.round((float) bitmapHeight / (float) viewWidth); 
       
      //为了保证图片不缩放变形,我们取宽高比例最小的那个 
      inSampleSize = widthScale 
该类是一个单例类,提供了本地图片加载,内存缓存,裁剪等逻辑,该类在加载本地图片的时候采用的是异步加载的方式,对于大图片的加载也是比较耗时的,所以采用子线程的方式去加载,对于图片的缓存机制使用的是LruCache,使用手机分配给应用程序内存的1/4用来缓存图片,除了使用LruCache缓存图片之外,还对图片进行了裁剪,举个很简单的例子,假如我们的控件大小是100 * 100, 而我们的图片是400*400,我们加载这么大的图片需要很多的内存,所以我们采用了图片裁剪,根据控件的大小来确定图片的裁剪比例,从而减小内存的消耗,提高GridView滑动的流畅度,介绍里面几个比较重要的方法
computeScale()计算图片需要裁剪的比例,根据控件的大小和图片的大小确定比例,如果图片比控件大,我们就进行裁剪,否则不需要。
decodeThumbBitmapForFile()方法是根据计算好了图片裁剪的比例之后从文件中加载图片,我们先设置options.inJustDecodeBounds = true表示解析不占用内存,但是我们能获取图片的具体大小,利用computeScale()计算好比例,在将options.inJustDecodeBounds=false,再次解析Bitmap,这样子就对图片进行了裁剪。
loadNativeImage(final String path, final Point mPoint, final NativeImageCallBack mCallBack)我们在客户端只需要调用该方法就能获取到Bitmap对象,里面的具体逻辑是先判断内存缓存LruCache中是否存在该Bitmap,不存在就开启子线程去读取,为了方便管理加载本地图片线程,这里使用了线程池,池中只能容纳一个线程,读取完了本地图片先将Bitmap加入到LruCache中,保存的Key为图片路径,然后再使用Handler通知主线程图片加载好了,之后将Bitmap和路径回调到方法onImageLoader(Bitmap bitmap, String path)中,该方法的mPoint是用来封装控件的宽和高的对象
如果不对图片进行裁剪直接这个方法的重载方法loadNativeImage(final String path, final NativeImageCallBack mCallBack) 就行了,逻辑是一样的,只是这个方法不对图片进行裁剪
接下来就是GridView的Adapter类的代码
package com.example.imagescan; 
 
import java.util.List; 
 
import android.content.Context; 
import android.graphics.Bitmap; 
import android.graphics.Point; 
import android.view.LayoutInflater; 
import android.view.View; 
import android.view.ViewGroup; 
import android.widget.BaseAdapter; 
import android.widget.GridView; 
import android.widget.ImageView; 
import android.widget.TextView; 
 
import com.example.imagescan.MyImageView.OnMeasureListener; 
import com.example.imagescan.NativeImageLoader.NativeImageCallBack; 
 
public class GroupAdapter extends BaseAdapter{ 
  private List list; 
  private Point mPoint = new Point(0, 0);//用来封装ImageView的宽和高的对象 
  private GridView mGridView; 
  protected LayoutInflater mInflater; 
   
  @Override 
  public int getCount() { 
    return list.size(); 
  } 
 
  @Override 
  public Object getItem(int position) { 
    return list.get(position); 
  } 
 
 
  @Override 
  public long getItemId(int position) { 
    return position; 
  } 
   
  public GroupAdapter(Context context, List list, GridView mGridView){ 
    this.list = list; 
    this.mGridView = mGridView; 
    mInflater = LayoutInflater.from(context); 
  } 
   
 
  @Override 
  public View getView(int position, View convertView, ViewGroup parent) { 
    final ViewHolder viewHolder; 
    ImageBean mImageBean = list.get(position); 
    String path = mImageBean.getTopImagePath(); 
    if(cOnvertView== null){ 
      viewHolder = new ViewHolder(); 
      cOnvertView= mInflater.inflate(R.layout.grid_group_item, null); 
      viewHolder.mImageView = (MyImageView) convertView.findViewById(R.id.group_image); 
      viewHolder.mTextViewTitle = (TextView) convertView.findViewById(R.id.group_title); 
      viewHolder.mTextViewCounts = (TextView) convertView.findViewById(R.id.group_count); 
       
      //用来监听ImageView的宽和高 
      viewHolder.mImageView.setOnMeasureListener(new OnMeasureListener() { 
         
        @Override 
        public void onMeasureSize(int width, int height) { 
          mPoint.set(width, height); 
        } 
      }); 
       
      convertView.setTag(viewHolder); 
    }else{ 
      viewHolder = (ViewHolder) convertView.getTag(); 
      viewHolder.mImageView.setImageResource(R.drawable.friends_sends_pictures_no); 
    } 
     
    viewHolder.mTextViewTitle.setText(mImageBean.getFolderName()); 
    viewHolder.mTextViewCounts.setText(Integer.toString(mImageBean.getImageCounts())); 
    //给ImageView设置路径Tag,这是异步加载图片的小技巧 
    viewHolder.mImageView.setTag(path); 
     
     
    //利用NativeImageLoader类加载本地图片 
    Bitmap bitmap = NativeImageLoader.getInstance().loadNativeImage(path, mPoint, new NativeImageCallBack() { 
       
      @Override 
      public void onImageLoader(Bitmap bitmap, String path) { 
        ImageView mImageView = (ImageView) mGridView.findViewWithTag(path); 
        if(bitmap != null && mImageView != null){ 
          mImageView.setImageBitmap(bitmap); 
        } 
      } 
    }); 
     
    if(bitmap != null){ 
      viewHolder.mImageView.setImageBitmap(bitmap); 
    }else{ 
      viewHolder.mImageView.setImageResource(R.drawable.friends_sends_pictures_no); 
    } 
     
     
    return convertView; 
  } 
   
   
   
  public static class ViewHolder{ 
    public MyImageView mImageView; 
    public TextView mTextViewTitle; 
    public TextView mTextViewCounts; 
  } 
 
   
} 
首先我们将每个item的图片路径设置Tag到该ImageView上面,然后利用NativeImageLoader来加载本地图片,但是我们显示的图片的宽和高可能远大于GirdView item中ImageView的大小,于是为了节省内存,我们需要对图片进行裁剪,需要对图片裁剪我们利用loadNativeImage(final String path, final Point mPoint, final NativeImageCallBack mCallBack)方法,我们就必须要获取ImageView的宽和高了
但是我们想在getView()中获取ImageView的宽和高存在问题,在getView()里面刚开始显示item的时候利用ImageView.getWidth() 获取的都是0,为什么刚开始获取不到宽和高呢,因为我们使用LayoutInflater来将XML布局文件Inflater()成View的时候,View并没有显示在界面上面,表明并没有对View进行onMeasure(), onLayout(), onDraw()等操作,必须等到retrue convertView的时候,表示该item对应的View已经绘制在ListView的位置上了, 此时才对item对应的View进行onMeasure(), onLayout(), onDraw()等操作,这时候才能获取到Item的宽和高,于是我想到了自定义ImageView,在onMeasure()中利用回调的模式主动通知我ImageView测量的宽和高,但是这有一个小小的问题,就是显示GridView的第一个item的时候,获取的宽和高还是0,第二个就能正常获取了,第一个宽和高为0,表示我们不对第一张图片进行裁剪而已,在效率上也没啥问题,不知道大家有没有好的方法,可以在getView()中获取Item中某个控件的宽和高。

自定义MyImageView的代码,我们只需要设置OnMeasureListener监听,当MyImageView测量完毕之后,就会将测量的宽和高回调到onMeasureSize()中,然后我们可以根据MyImageView的大小来裁剪图片

package com.example.imagescan; 
 
import android.content.Context; 
import android.util.AttributeSet; 
import android.widget.ImageView; 
 
public class MyImageView extends ImageView { 
  private OnMeasureListener onMeasureListener; 
   
  public void setOnMeasureListener(OnMeasureListener onMeasureListener) { 
    this.OnMeasureListener= onMeasureListener; 
  } 
 
  public MyImageView(Context context, AttributeSet attrs) { 
    super(context, attrs); 
  } 
 
  public MyImageView(Context context, AttributeSet attrs, int defStyle) { 
    super(context, attrs, defStyle); 
  } 
 
  @Override 
  protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { 
    super.onMeasure(widthMeasureSpec, heightMeasureSpec); 
     
    //将图片测量的大小回调到onMeasureSize()方法中 
    if(onMeasureListener != null){ 
      onMeasureListener.onMeasureSize(getMeasuredWidth(), getMeasuredHeight()); 
    } 
  } 
 
  public interface OnMeasureListener{ 
    public void onMeasureSize(int width, int height); 
  } 
   
} 

上面这些代码就完成了第一个界面的功能了,接下来就是点击GridView的item跳转另一个界面来显示该文件夹下面的所有图片,功能跟第一个界面差不多,也是使用GridView来显示图片,第二个界面的布局代码我就不贴了,直接贴上界面的代码

package com.example.imagescan; 
 
import java.util.List; 
 
import android.app.Activity; 
import android.os.Bundle; 
import android.widget.GridView; 
import android.widget.Toast; 
 
public class ShowImageActivity extends Activity { 
  private GridView mGridView; 
  private List list; 
  private ChildAdapter adapter; 
 
  @Override 
  protected void onCreate(Bundle savedInstanceState) { 
    super.onCreate(savedInstanceState); 
    setContentView(R.layout.show_image_activity); 
     
    mGridView = (GridView) findViewById(R.id.child_grid); 
    list = getIntent().getStringArrayListExtra("data"); 
     
    adapter = new ChildAdapter(this, list, mGridView); 
    mGridView.setAdapter(adapter); 
     
  } 
 
  @Override 
  public void onBackPressed() { 
    Toast.makeText(this, "选中 " + adapter.getSelectItems().size() + " item", Toast.LENGTH_LONG).show(); 
    super.onBackPressed(); 
  } 
   
   
} 

GridView的item上面一个我们自定义的MyImageView用来显示图片,另外还有一个CheckBox来记录我们选中情况,Adapter的代码如下

package com.example.imagescan; 
 
import java.util.ArrayList; 
import java.util.HashMap; 
import java.util.Iterator; 
import java.util.List; 
import java.util.Map; 
 
import android.content.Context; 
import android.graphics.Bitmap; 
import android.graphics.Point; 
import android.view.LayoutInflater; 
import android.view.View; 
import android.view.ViewGroup; 
import android.widget.BaseAdapter; 
import android.widget.CheckBox; 
import android.widget.CompoundButton; 
import android.widget.ImageView; 
import android.widget.CompoundButton.OnCheckedChangeListener; 
import android.widget.GridView; 
 
import com.example.imagescan.MyImageView.OnMeasureListener; 
import com.example.imagescan.NativeImageLoader.NativeImageCallBack; 
import com.nineoldandroids.animation.AnimatorSet; 
import com.nineoldandroids.animation.ObjectAnimator; 
 
public class ChildAdapter extends BaseAdapter { 
  private Point mPoint = new Point(0, 0);//用来封装ImageView的宽和高的对象 
  /** 
   * 用来存储图片的选中情况 
   */ 
  private HashMap mSelectMap = new HashMap(); 
  private GridView mGridView; 
  private List list; 
  protected LayoutInflater mInflater; 
 
  public ChildAdapter(Context context, List list, GridView mGridView) { 
    this.list = list; 
    this.mGridView = mGridView; 
    mInflater = LayoutInflater.from(context); 
  } 
   
  @Override 
  public int getCount() { 
    return list.size(); 
  } 
 
  @Override 
  public Object getItem(int position) { 
    return list.get(position); 
  } 
 
 
  @Override 
  public long getItemId(int position) { 
    return position; 
  } 
   
  @Override 
  public View getView(final int position, View convertView, ViewGroup parent) { 
    final ViewHolder viewHolder; 
    String path = list.get(position); 
     
    if(cOnvertView== null){ 
      cOnvertView= mInflater.inflate(R.layout.grid_child_item, null); 
      viewHolder = new ViewHolder(); 
      viewHolder.mImageView = (MyImageView) convertView.findViewById(R.id.child_image); 
      viewHolder.mCheckBox = (CheckBox) convertView.findViewById(R.id.child_checkbox); 
       
      //用来监听ImageView的宽和高 
      viewHolder.mImageView.setOnMeasureListener(new OnMeasureListener() { 
         
        @Override 
        public void onMeasureSize(int width, int height) { 
          mPoint.set(width, height); 
        } 
      }); 
       
      convertView.setTag(viewHolder); 
    }else{ 
      viewHolder = (ViewHolder) convertView.getTag(); 
      viewHolder.mImageView.setImageResource(R.drawable.friends_sends_pictures_no); 
    } 
    viewHolder.mImageView.setTag(path); 
    viewHolder.mCheckBox.setOnCheckedChangeListener(new OnCheckedChangeListener() { 
       
      @Override 
      public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) { 
        //如果是未选中的CheckBox,则添加动画 
        if(!mSelectMap.containsKey(position) || !mSelectMap.get(position)){ 
          addAnimation(viewHolder.mCheckBox); 
        } 
        mSelectMap.put(position, isChecked); 
      } 
    }); 
     
    viewHolder.mCheckBox.setChecked(mSelectMap.containsKey(position) &#63; mSelectMap.get(position) : false); 
     
    //利用NativeImageLoader类加载本地图片 
    Bitmap bitmap = NativeImageLoader.getInstance().loadNativeImage(path, mPoint, new NativeImageCallBack() { 
       
      @Override 
      public void onImageLoader(Bitmap bitmap, String path) { 
        ImageView mImageView = (ImageView) mGridView.findViewWithTag(path); 
        if(bitmap != null && mImageView != null){ 
          mImageView.setImageBitmap(bitmap); 
        } 
      } 
    }); 
     
    if(bitmap != null){ 
      viewHolder.mImageView.setImageBitmap(bitmap); 
    }else{ 
      viewHolder.mImageView.setImageResource(R.drawable.friends_sends_pictures_no); 
    } 
     
    return convertView; 
  } 
   
  /** 
   * 给CheckBox加点击动画,利用开源库nineoldandroids设置动画 
   * @param view 
   */ 
  private void addAnimation(View view){ 
    float [] vaules = new float[]{0.5f, 0.6f, 0.7f, 0.8f, 0.9f, 1.0f, 1.1f, 1.2f, 1.3f, 1.25f, 1.2f, 1.15f, 1.1f, 1.0f}; 
    AnimatorSet set = new AnimatorSet(); 
    set.playTogether(ObjectAnimator.ofFloat(view, "scaleX", vaules),  
        ObjectAnimator.ofFloat(view, "scaleY", vaules)); 
        set.setDuration(150); 
    set.start(); 
  } 
   
   
  /** 
   * 获取选中的Item的position 
   * @return 
   */ 
  public List getSelectItems(){ 
    List list = new ArrayList(); 
    for(Iterator> it = mSelectMap.entrySet().iterator(); it.hasNext();){ 
      Map.Entry entry = it.next(); 
      if(entry.getValue()){ 
        list.add(entry.getKey()); 
      } 
    } 
     
    return list; 
  } 
   
   
  public static class ViewHolder{ 
    public MyImageView mImageView; 
    public CheckBox mCheckBox; 
  } 
 
 
} 

第二个界面的Adapter跟第一个界面差不多,无非多了一个CheckBox用来记录图片选择情况,我们只需要对CheckBox设置setOnCheckedChangeListener监听,微信的选中之后CheckBox有一个动画效果,所以我利用nineoldandroids动画库也给CheckBox加了一个动画效果,直接调用addAnimation()方法就能添加了,getSelectItems()方法就能获取我们选中的item的position了,知道了选中的position,其他的信息就都知道了,微信有对图片进行预览的功能,我这里就不添加了,如果有这个需求可以自行添加,给大家推荐一个https://github.com/chrisbanes/PhotoView

运行项目,效果如下

看起来还不错吧,采用的是异步读取图片,对图片进行了缓存和裁剪,使得在显示本地图片方面比较流畅,GridView滑动也挺流畅的,也有效的避免OOM的产生。


推荐阅读
  • 解决微信电脑版无法刷朋友圈问题:使用安卓远程投屏方案
    在工作期间想要浏览微信和朋友圈却不太方便?虽然微信电脑版目前不支持直接刷朋友圈,但通过远程投屏技术,可以轻松实现在电脑上操作安卓设备的功能。 ... [详细]
  • RecyclerView初步学习(一)
    RecyclerView初步学习(一)ReCyclerView提供了一种插件式的编程模式,除了提供ViewHolder缓存模式,还可以自定义动画,分割符,布局样式,相比于传统的ListVi ... [详细]
  • Android LED 数字字体的应用与实现
    本文介绍了一种适用于 Android 应用的 LED 数字字体(digital font),并详细描述了其在 UI 设计中的应用场景及其实现方法。这种字体常用于视频、广告倒计时等场景,能够增强视觉效果。 ... [详细]
  • 从零开始构建完整手机站:Vue CLI 3 实战指南(第一部分)
    本系列教程将引导您使用 Vue CLI 3 构建一个功能齐全的移动应用。我们将深入探讨项目中涉及的每一个知识点,并确保这些内容与实际工作中的需求紧密结合。 ... [详细]
  • 基于KVM的SRIOV直通配置及性能测试
    SRIOV介绍、VF直通配置,以及包转发率性能测试小慢哥的原创文章,欢迎转载目录?1.SRIOV介绍?2.环境说明?3.开启SRIOV?4.生成VF?5.VF ... [详细]
  • QUIC协议:快速UDP互联网连接
    QUIC(Quick UDP Internet Connections)是谷歌开发的一种旨在提高网络性能和安全性的传输层协议。它基于UDP,并结合了TLS级别的安全性,提供了更高效、更可靠的互联网通信方式。 ... [详细]
  • Android 九宫格布局详解及实现:人人网应用示例
    本文深入探讨了人人网Android应用中独特的九宫格布局设计,解析其背后的GridView实现原理,并提供详细的代码示例。这种布局方式不仅美观大方,而且在现代Android应用中较为少见,值得开发者借鉴。 ... [详细]
  • 国内BI工具迎战国际巨头Tableau,稳步崛起
    尽管商业智能(BI)工具在中国的普及程度尚不及国际市场,但近年来,随着本土企业的持续创新和市场推广,国内主流BI工具正逐渐崭露头角。面对国际品牌如Tableau的强大竞争,国内BI工具通过不断优化产品和技术,赢得了越来越多用户的认可。 ... [详细]
  • 本文详细介绍了如何在Linux系统上安装和配置Smokeping,以实现对网络链路质量的实时监控。通过详细的步骤和必要的依赖包安装,确保用户能够顺利完成部署并优化其网络性能监控。 ... [详细]
  • 深入理解Cookie与Session会话管理
    本文详细介绍了如何通过HTTP响应和请求处理浏览器的Cookie信息,以及如何创建、设置和管理Cookie。同时探讨了会话跟踪技术中的Session机制,解释其原理及应用场景。 ... [详细]
  • 数据库内核开发入门 | 搭建研发环境的初步指南
    本课程将带你从零开始,逐步掌握数据库内核开发的基础知识和实践技能,重点介绍如何搭建OceanBase的开发环境。 ... [详细]
  • XNA 3.0 游戏编程:从 XML 文件加载数据
    本文介绍如何在 XNA 3.0 游戏项目中从 XML 文件加载数据。我们将探讨如何将 XML 数据序列化为二进制文件,并通过内容管道加载到游戏中。此外,还会涉及自定义类型读取器和写入器的实现。 ... [详细]
  • 本文介绍了如何使用PHP代码实现微信平台的媒体素材上传功能,详细解释了API接口的使用方法和注意事项,确保文件路径正确以避免常见的错误。 ... [详细]
  • 本文详细介绍了 Java 中 org.apache.xmlbeans.SchemaType 类的 getBaseEnumType() 方法,提供了多个代码示例,并解释了其在不同场景下的使用方法。 ... [详细]
  • 解决JAX-WS动态客户端工厂弃用问题并迁移到XFire
    在处理Java项目中的JAR包冲突时,我们遇到了JaxWsDynamicClientFactory被弃用的问题,并成功将其迁移到org.codehaus.xfire.client。本文详细介绍了这一过程及解决方案。 ... [详细]
author-avatar
翔溢_142
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有