天天看点

Android 高仿微信图片选择器1、概述2、图片的列表页3、展现文件夹的PopupWindow4、选择不同的文件夹

转载请标明出处: http://blog.csdn.net/lmj623565791/article/details/39943731,本文出自: 【张鸿洋的博客】

1、概述

关于手机图片加载器,在当今像素随随便便破千万的时代,一张图片占据的内存都相当可观,作为高大尚程序猿的我们,有必要掌握图片的压缩,缓存等处理,以到达纵使你有万张照片,纵使你的像素再高,我们也能正确的显示所有的图片。当然了,单纯显示图片没撒意思,我们决定高仿一下微信的图片选择器,在此,感谢微信!本篇博客将基于以下两篇博客:

Android 快速开发系列 打造万能的ListView GridView 适配器  将使用我们打造的CommonAdapter作为我们例子中GridView以及ListView的适配器

Android Handler 异步消息处理机制的妙用 创建强大的图片加载类 将使用我们自己写的ImageLoader作为我们的图片加载的核心类

如果你没看过也没关系,等看完本篇博客,可以结合以上两篇再进行充分理解一下。

好了,首先贴一下效果图:

Android 高仿微信图片选择器1、概述2、图片的列表页3、展现文件夹的PopupWindow4、选择不同的文件夹
Android 高仿微信图片选择器1、概述2、图片的列表页3、展现文件夹的PopupWindow4、选择不同的文件夹
Android 高仿微信图片选择器1、概述2、图片的列表页3、展现文件夹的PopupWindow4、选择不同的文件夹

动态图实在是录不出来,大家自己打开微信点击发表图片,或者聊天窗口发送图片,大致和微信的效果一样~

简单描述一下:

1、默认显示图片最多的文件夹图片,以及底部显示图片总数量;如上图1;

2、点击底部,弹出popupWindow,popupWindow包含所有含有图片的文件夹,以及显示每个文件夹中图片数量;如上图2;注:此时Activity变暗

3、选择任何文件夹,进入该文件夹图片显示,可以点击选择图片,当然了,点击已选择的图片则会取消选择;如上图3;注:选中图片变暗

当然了,最重要的效果一定流畅,不能动不动OOM~~

本人测试手机小米2s,图片6802张,未出现OOM异常,效果也是非常流畅,堪比图库~

不过存在bug在所难免,大家可以留言说下自己发现的bug;文末会提供源码下载。

好了,下面就可以代码的征程了~

2、图片的列表页

首先对手机中图片进行扫描,拿到图片数量最多的,直接显示在GridView上;并且扫描结束,得到一个所有包含图片的文件夹信息的List;

对于文件夹信息,我们单独创建了一个Bean:

[java]

view plain copy print ?

  1. package com.zhy.bean;  
  2. public class ImageFloder  
  3. {  
  4.     private String dir;  
  5.     private String firstImagePath;  
  6.     private String name;  
  7.     private int count;  
  8.     public String getDir()  
  9.     {  
  10.         return dir;  
  11.     }  
  12.     public void setDir(String dir)  
  13.     {  
  14.         this.dir = dir;  
  15.         int lastIndexOf = this.dir.lastIndexOf(“/”);  
  16.         this.name = this.dir.substring(lastIndexOf);  
  17.     }  
  18.     public String getFirstImagePath()  
  19.     {  
  20.         return firstImagePath;  
  21.     }  
  22.     public void setFirstImagePath(String firstImagePath)  
  23.     {  
  24.         this.firstImagePath = firstImagePath;  
  25.     }  
  26.     public String getName()  
  27.     {  
  28.         return name;  
  29.     }  
  30.     public int getCount()  
  31.     {  
  32.         return count;  
  33.     }  
  34.     public void setCount(int count)  
  35.     {  
  36.         this.count = count;  
  37.     }  
  38. }  
package com.zhy.bean;

public class ImageFloder
{
    /**
     * 图片的文件夹路径
     */
    private String dir;

    /**
     * 第一张图片的路径
     */
    private String firstImagePath;

    /**
     * 文件夹的名称
     */
    private String name;

    /**
     * 图片的数量
     */
    private int count;

    public String getDir()
    {
        return dir;
    }

    public void setDir(String dir)
    {
        this.dir = dir;
        int lastIndexOf = this.dir.lastIndexOf("/");
        this.name = this.dir.substring(lastIndexOf);
    }

    public String getFirstImagePath()
    {
        return firstImagePath;
    }

    public void setFirstImagePath(String firstImagePath)
    {
        this.firstImagePath = firstImagePath;
    }

    public String getName()
    {
        return name;
    }
    public int getCount()
    {
        return count;
    }

    public void setCount(int count)
    {
        this.count = count;
    }



}
           

用来存储当前文件夹的路径,当前文件夹包含多少张图片,以及第一张图片路径用于做文件夹的图标;注:文件夹的名称,我们在set文件夹的路径的时候,自动提取,仔细看下setDir这个方法。

接下来就是扫描手机图片的代码了:

[java]

view plain copy print ?

  1. @Override  
  2.     protected void onCreate(Bundle savedInstanceState)  
  3.     {  
  4.         super.onCreate(savedInstanceState);  
  5.         setContentView(R.layout.activity_main);  
  6.         DisplayMetrics outMetrics = new DisplayMetrics();  
  7.         getWindowManager().getDefaultDisplay().getMetrics(outMetrics);  
  8.         mScreenHeight = outMetrics.heightPixels;  
  9.         initView();  
  10.         getImages();  
  11.         initEvent();  
  12.     }  
  13.     private void getImages()  
  14.     {  
  15.         if (!Environment.getExternalStorageState().equals(  
  16.                 Environment.MEDIA_MOUNTED))  
  17.         {  
  18.             Toast.makeText(this, “暂无外部存储”, Toast.LENGTH_SHORT).show();  
  19.             return;  
  20.         }  
  21.         // 显示进度条  
  22.         mProgressDialog = ProgressDialog.show(this, null, “正在加载…”);  
  23.         new Thread(new Runnable()  
  24.         {  
  25.             @Override  
  26.             public void run()  
  27.             {  
  28.                 String firstImage = null;  
  29.                 Uri mImageUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;  
  30.                 ContentResolver mContentResolver = MainActivity.this  
  31.                         .getContentResolver();  
  32.                 // 只查询jpeg和png的图片  
  33.                 Cursor mCursor = mContentResolver.query(mImageUri, null,  
  34.                         MediaStore.Images.Media.MIME_TYPE + ”=? or ”  
  35.                                 + MediaStore.Images.Media.MIME_TYPE + ”=?”,  
  36.                         new String[] { “image/jpeg”, “image/png” },  
  37.                         MediaStore.Images.Media.DATE_MODIFIED);  
  38.                 Log.e(”TAG”, mCursor.getCount() + “”);  
  39.                 while (mCursor.moveToNext())  
  40.                 {  
  41.                     // 获取图片的路径  
  42.                     String path = mCursor.getString(mCursor  
  43.                             .getColumnIndex(MediaStore.Images.Media.DATA));  
  44.                     Log.e(”TAG”, path);  
  45.                     // 拿到第一张图片的路径  
  46.                     if (firstImage == null)  
  47.                         firstImage = path;  
  48.                     // 获取该图片的父路径名  
  49.                     File parentFile = new File(path).getParentFile();  
  50.                     if (parentFile == null)  
  51.                         continue;  
  52.                     String dirPath = parentFile.getAbsolutePath();  
  53.                     ImageFloder imageFloder = null;  
  54.                     // 利用一个HashSet防止多次扫描同一个文件夹(不加这个判断,图片多起来还是相当恐怖的~~)  
  55.                     if (mDirPaths.contains(dirPath))  
  56.                     {  
  57.                         continue;  
  58.                     } else  
  59.                     {  
  60.                         mDirPaths.add(dirPath);  
  61.                         // 初始化imageFloder  
  62.                         imageFloder = new ImageFloder();  
  63.                         imageFloder.setDir(dirPath);  
  64.                         imageFloder.setFirstImagePath(path);  
  65.                     }  
  66.                     int picSize = parentFile.list(new FilenameFilter()  
  67.                     {  
  68.                         @Override  
  69.                         public boolean accept(File dir, String filename)  
  70.                         {  
  71.                             if (filename.endsWith(“.jpg”)  
  72.                                     || filename.endsWith(”.png”)  
  73.                                     || filename.endsWith(”.jpeg”))  
  74.                                 return true;  
  75.                             return false;  
  76.                         }  
  77.                     }).length;  
  78.                     totalCount += picSize;  
  79.                     imageFloder.setCount(picSize);  
  80.                     mImageFloders.add(imageFloder);  
  81.                     if (picSize > mPicsSize)  
  82.                     {  
  83.                         mPicsSize = picSize;  
  84.                         mImgDir = parentFile;  
  85.                     }  
  86.                 }  
  87.                 mCursor.close();  
  88.                 // 扫描完成,辅助的HashSet也就可以释放内存了  
  89.                 mDirPaths = null;  
  90.                 // 通知Handler扫描图片完成  
  91.                 mHandler.sendEmptyMessage(0x110);  
  92.             }  
  93.         }).start();  
  94.     }  
@Override
    protected void onCreate(Bundle savedInstanceState)
    {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        DisplayMetrics outMetrics = new DisplayMetrics();
        getWindowManager().getDefaultDisplay().getMetrics(outMetrics);
        mScreenHeight = outMetrics.heightPixels;

        initView();
        getImages();
        initEvent();

    }



    /**
     * 利用ContentProvider扫描手机中的图片,此方法在运行在子线程中 完成图片的扫描,最终获得jpg最多的那个文件夹
     */
    private void getImages()
    {
        if (!Environment.getExternalStorageState().equals(
                Environment.MEDIA_MOUNTED))
        {
            Toast.makeText(this, "暂无外部存储", Toast.LENGTH_SHORT).show();
            return;
        }
        // 显示进度条
        mProgressDialog = ProgressDialog.show(this, null, "正在加载...");

        new Thread(new Runnable()
        {
            @Override
            public void run()
            {

                String firstImage = null;

                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 + "=? or "
                                + MediaStore.Images.Media.MIME_TYPE + "=?",
                        new String[] { "image/jpeg", "image/png" },
                        MediaStore.Images.Media.DATE_MODIFIED);

                Log.e("TAG", mCursor.getCount() + "");
                while (mCursor.moveToNext())
                {
                    // 获取图片的路径
                    String path = mCursor.getString(mCursor
                            .getColumnIndex(MediaStore.Images.Media.DATA));

                    Log.e("TAG", path);
                    // 拿到第一张图片的路径
                    if (firstImage == null)
                        firstImage = path;
                    // 获取该图片的父路径名
                    File parentFile = new File(path).getParentFile();
                    if (parentFile == null)
                        continue;
                    String dirPath = parentFile.getAbsolutePath();
                    ImageFloder imageFloder = null;
                    // 利用一个HashSet防止多次扫描同一个文件夹(不加这个判断,图片多起来还是相当恐怖的~~)
                    if (mDirPaths.contains(dirPath))
                    {
                        continue;
                    } else
                    {
                        mDirPaths.add(dirPath);
                        // 初始化imageFloder
                        imageFloder = new ImageFloder();
                        imageFloder.setDir(dirPath);
                        imageFloder.setFirstImagePath(path);
                    }

                    int picSize = parentFile.list(new FilenameFilter()
                    {
                        @Override
                        public boolean accept(File dir, String filename)
                        {
                            if (filename.endsWith(".jpg")
                                    || filename.endsWith(".png")
                                    || filename.endsWith(".jpeg"))
                                return true;
                            return false;
                        }
                    }).length;
                    totalCount += picSize;

                    imageFloder.setCount(picSize);
                    mImageFloders.add(imageFloder);

                    if (picSize > mPicsSize)
                    {
                        mPicsSize = picSize;
                        mImgDir = parentFile;
                    }
                }
                mCursor.close();

                // 扫描完成,辅助的HashSet也就可以释放内存了
                mDirPaths = null;

                // 通知Handler扫描图片完成
                mHandler.sendEmptyMessage(0x110);

            }
        }).start();

    }
           

ps:运行出现空指针的话,在81行的位置添加判断,if(parentFile.list()==null)continue , 切记~~~有些图片比较诡异~~; 

initView就不看了,都是些findViewById;

getImages主要就是扫描图片的代码,我们开启了一个Thread进行扫描,扫描完成以后,我们得到了图片最多文件夹路径(mImgDir),手机中图片数量(totalCount);以及所有包含图片文件夹信息(mImageFloders)

然后我们通过handler发送消息,在handleMessage里面:

1、创建GridView的适配器,为我们的GridView设置适配器,显示图片;

2、有了mImageFloders,就可以创建我们的popupWindow了

看一眼我们的Handler

[java]

view plain copy print ?

  1. private Handler mHandler = new Handler()  
  2.     {  
  3.         public void handleMessage(android.os.Message msg)  
  4.         {  
  5.             mProgressDialog.dismiss();  
  6.             //为View绑定数据  
  7.             data2View();  
  8.             //初始化展示文件夹的popupWindw  
  9.             initListDirPopupWindw();  
  10.         }  
  11.     };  
private Handler mHandler = new Handler()
    {
        public void handleMessage(android.os.Message msg)
        {
            mProgressDialog.dismiss();
            //为View绑定数据
            data2View();
            //初始化展示文件夹的popupWindw
            initListDirPopupWindw();
        }
    };
           

可以看到分别干了上述的两件事:

[java]

view plain copy print ?

  1.     private void data2View()  
  2.     {  
  3.         if (mImgDir == null)  
  4.         {  
  5.             Toast.makeText(getApplicationContext(), ”擦,一张图片没扫描到”,  
  6.                     Toast.LENGTH_SHORT).show();  
  7.             return;  
  8.         }  
  9.         mImgs = Arrays.asList(mImgDir.list());  
  10.         mAdapter = new MyAdapter(getApplicationContext(), mImgs,  
  11.                 R.layout.grid_item, mImgDir.getAbsolutePath());  
  12.         mGirdView.setAdapter(mAdapter);  
  13.         mImageCount.setText(totalCount + ”张”);  
  14.     };  
/**
     * 为View绑定数据
     */
    private void data2View()
    {
        if (mImgDir == null)
        {
            Toast.makeText(getApplicationContext(), "擦,一张图片没扫描到",
                    Toast.LENGTH_SHORT).show();
            return;
        }

        mImgs = Arrays.asList(mImgDir.list());
        /**
         * 可以看到文件夹的路径和图片的路径分开保存,极大的减少了内存的消耗;
         */
        mAdapter = new MyAdapter(getApplicationContext(), mImgs,
                R.layout.grid_item, mImgDir.getAbsolutePath());
        mGirdView.setAdapter(mAdapter);
        mImageCount.setText(totalCount + "张");
    };
           

data2View就是我们当前Activity上所有的View设置数据了。

看到这里还用到了一个Adapter,我们GridView的:

[java]

view plain copy print ?

  1. package com.zhy.imageloader;  
  2. import java.util.LinkedList;  
  3. import java.util.List;  
  4. import android.content.Context;  
  5. import android.graphics.Color;  
  6. import android.view.View;  
  7. import android.view.View.OnClickListener;  
  8. import android.widget.ImageView;  
  9. import com.zhy.utils.CommonAdapter;  
  10. public class MyAdapter extends CommonAdapter<String>  
  11. {  
  12.     public static List<String> mSelectedImage = new LinkedList<String>();  
  13.     private String mDirPath;  
  14.     public MyAdapter(Context context, List<String> mDatas, int itemLayoutId,  
  15.             String dirPath)  
  16.     {  
  17.         super(context, mDatas, itemLayoutId);  
  18.         this.mDirPath = dirPath;  
  19.     }  
  20.     @Override  
  21.     public void convert(final com.zhy.utils.ViewHolder helper, final String item)  
  22.     {  
  23.         // 设置no_pic  
  24.         helper.setImageResource(R.id.id_item_image, R.drawable.pictures_no);  
  25.         // 设置no_selected  
  26.         helper.setImageResource(R.id.id_item_select,  
  27.                 R.drawable.picture_unselected);  
  28.         // 设置图片  
  29.         helper.setImageByUrl(R.id.id_item_image, mDirPath + ”/” + item);  
  30.         final ImageView mImageView = helper.getView(R.id.id_item_image);  
  31.         final ImageView mSelect = helper.getView(R.id.id_item_select);  
  32.         mImageView.setColorFilter(null);  
  33.         // 设置ImageView的点击事件  
  34.         mImageView.setOnClickListener(new OnClickListener()  
  35.         {  
  36.             // 选择,则将图片变暗,反之则反之  
  37.             @Override  
  38.             public void onClick(View v)  
  39.             {  
  40.                 // 已经选择过该图片  
  41.                 if (mSelectedImage.contains(mDirPath + “/” + item))  
  42.                 {  
  43.                     mSelectedImage.remove(mDirPath + ”/” + item);  
  44.                     mSelect.setImageResource(R.drawable.picture_unselected);  
  45.                     mImageView.setColorFilter(null);  
  46.                 } else  
  47.                 // 未选择该图片  
  48.                 {  
  49.                     mSelectedImage.add(mDirPath + ”/” + item);  
  50.                     mSelect.setImageResource(R.drawable.pictures_selected);  
  51.                     mImageView.setColorFilter(Color.parseColor(”#77000000”));  
  52.                 }  
  53.             }  
  54.         });  
  55.         if (mSelectedImage.contains(mDirPath + “/” + item))  
  56.         {  
  57.             mSelect.setImageResource(R.drawable.pictures_selected);  
  58.             mImageView.setColorFilter(Color.parseColor(”#77000000”));  
  59.         }  
  60.     }  
  61. }  
package com.zhy.imageloader;

import java.util.LinkedList;
import java.util.List;

import android.content.Context;
import android.graphics.Color;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.ImageView;

import com.zhy.utils.CommonAdapter;

public class MyAdapter extends CommonAdapter<String>
{

    /**
     * 用户选择的图片,存储为图片的完整路径
     */
    public static List<String> mSelectedImage = new LinkedList<String>();

    /**
     * 文件夹路径
     */
    private String mDirPath;

    public MyAdapter(Context context, List<String> mDatas, int itemLayoutId,
            String dirPath)
    {
        super(context, mDatas, itemLayoutId);
        this.mDirPath = dirPath;
    }

    @Override
    public void convert(final com.zhy.utils.ViewHolder helper, final String item)
    {
        // 设置no_pic
        helper.setImageResource(R.id.id_item_image, R.drawable.pictures_no);
        // 设置no_selected
        helper.setImageResource(R.id.id_item_select,
                R.drawable.picture_unselected);
        // 设置图片
        helper.setImageByUrl(R.id.id_item_image, mDirPath + "/" + item);

        final ImageView mImageView = helper.getView(R.id.id_item_image);
        final ImageView mSelect = helper.getView(R.id.id_item_select);

        mImageView.setColorFilter(null);
        // 设置ImageView的点击事件
        mImageView.setOnClickListener(new OnClickListener()
        {
            // 选择,则将图片变暗,反之则反之
            @Override
            public void onClick(View v)
            {

                // 已经选择过该图片
                if (mSelectedImage.contains(mDirPath + "/" + item))
                {
                    mSelectedImage.remove(mDirPath + "/" + item);
                    mSelect.setImageResource(R.drawable.picture_unselected);
                    mImageView.setColorFilter(null);
                } else
                // 未选择该图片
                {
                    mSelectedImage.add(mDirPath + "/" + item);
                    mSelect.setImageResource(R.drawable.pictures_selected);
                    mImageView.setColorFilter(Color.parseColor("#77000000"));
                }

            }
        });

        /**
         * 已经选择过的图片,显示出选择过的效果
         */
        if (mSelectedImage.contains(mDirPath + "/" + item))
        {
            mSelect.setImageResource(R.drawable.pictures_selected);
            mImageView.setColorFilter(Color.parseColor("#77000000"));
        }

    }
}
           

可以看到我们GridView的Adapter继承了我们的CommonAdapter,如果不知道CommonAdapter为何物,可以去看看万能适配器那篇博文;

我们现在只需要实现convert方法:

在convert中,我们设置图片,设置事件等,对于图片的变暗,我们使用的是ImageView的setColorFilter ;根据Url加载图片的操作封装在helper.setImageByUrl(view,url)中,内部使用的是我们自己定义的ImageLoader,包括错乱处理都已经封装了,图片策略我们使用的是LIFO后进先出;不清楚的可以看文章一开始说明的那两篇博文,对于CommonAdapter以及ImageLoader都有从无到有的详细打造过程;

到此我们的第一个Activity的所有的任务就完成了~~~

3、展现文件夹的PopupWindow

现在我们要实现,点击底部的布局弹出我们的文件夹选择框,并且我们弹出框后面的Activity要变暗;

不急着贴代码,我们先考虑下PopupWindow怎么用最好,我们的PopupWindow需要设置布局文件,需要初始化View,需要初始化事件,还需要和Activity交互~~

那么肯定的,我们使用独立的类,这个类和Activity很相似,在里面initView(),initEvent()之类的。

我们创建了一个popupWindow使用的超类:

[java]

view plain copy print ?

  1. package com.zhy.utils;  
  2. import java.util.List;  
  3. import android.content.Context;  
  4. import android.graphics.drawable.BitmapDrawable;  
  5. import android.view.MotionEvent;  
  6. import android.view.View;  
  7. import android.view.View.OnTouchListener;  
  8. import android.widget.PopupWindow;  
  9. public abstract class BasePopupWindowForListView<T> extends PopupWindow  
  10. {  
  11.     protected View mContentView;  
  12.     protected Context context;  
  13.     protected List<T> mDatas;  
  14.     public BasePopupWindowForListView(View contentView, int width, int height,  
  15.             boolean focusable)  
  16.     {  
  17.         this(contentView, width, height, focusable, null);  
  18.     }  
  19.     public BasePopupWindowForListView(View contentView, int width, int height,  
  20.             boolean focusable, List<T> mDatas)  
  21.     {  
  22.         this(contentView, width, height, focusable, mDatas, new Object[0]);  
  23.     }  
  24.     public BasePopupWindowForListView(View contentView, int width, int height,  
  25.             boolean focusable, List<T> mDatas, Object… params)  
  26.     {  
  27.         super(contentView, width, height, focusable);  
  28.         this.mContentView = contentView;  
  29.         context = contentView.getContext();  
  30.         if (mDatas != null)  
  31.             this.mDatas = mDatas;  
  32.         if (params != null && params.length > 0)  
  33.         {  
  34.             beforeInitWeNeedSomeParams(params);  
  35.         }  
  36.         setBackgroundDrawable(new BitmapDrawable());  
  37.         setTouchable(true);  
  38.         setOutsideTouchable(true);  
  39.         setTouchInterceptor(new OnTouchListener()  
  40.         {  
  41.             @Override  
  42.             public boolean onTouch(View v, MotionEvent event)  
  43.             {  
  44.                 if (event.getAction() == MotionEvent.ACTION_OUTSIDE)  
  45.                 {  
  46.                     dismiss();  
  47.                     return true;  
  48.                 }  
  49.                 return false;  
  50.             }  
  51.         });  
  52.         initViews();  
  53.         initEvents();  
  54.         init();  
  55.     }  
  56.     protected abstract void beforeInitWeNeedSomeParams(Object… params);  
  57.     public abstract void initViews();  
  58.     public abstract void initEvents();  
  59.     public abstract void init();  
  60.     public View findViewById(int id)  
  61.     {  
  62.         return mContentView.findViewById(id);  
  63.     }  
  64.     protected static int dpToPx(Context context, int dp)  
  65.     {  
  66.         return (int) (context.getResources().getDisplayMetrics().density * dp + 0.5f);  
  67.     }  
  68. }  
package com.zhy.utils;

import java.util.List;

import android.content.Context;
import android.graphics.drawable.BitmapDrawable;
import android.view.MotionEvent;
import android.view.View;
import android.view.View.OnTouchListener;
import android.widget.PopupWindow;

public abstract class BasePopupWindowForListView<T> extends PopupWindow
{
    /**
     * 布局文件的最外层View
     */
    protected View mContentView;
    protected Context context;
    /**
     * ListView的数据集
     */
    protected List<T> mDatas;

    public BasePopupWindowForListView(View contentView, int width, int height,
            boolean focusable)
    {
        this(contentView, width, height, focusable, null);
    }

    public BasePopupWindowForListView(View contentView, int width, int height,
            boolean focusable, List<T> mDatas)
    {
        this(contentView, width, height, focusable, mDatas, new Object[0]);

    }

    public BasePopupWindowForListView(View contentView, int width, int height,
            boolean focusable, List<T> mDatas, Object... params)
    {
        super(contentView, width, height, focusable);
        this.mContentView = contentView;
        context = contentView.getContext();
        if (mDatas != null)
            this.mDatas = mDatas;

        if (params != null && params.length > 0)
        {
            beforeInitWeNeedSomeParams(params);
        }

        setBackgroundDrawable(new BitmapDrawable());
        setTouchable(true);
        setOutsideTouchable(true);
        setTouchInterceptor(new OnTouchListener()
        {
            @Override
            public boolean onTouch(View v, MotionEvent event)
            {
                if (event.getAction() == MotionEvent.ACTION_OUTSIDE)
                {
                    dismiss();
                    return true;
                }
                return false;
            }
        });
        initViews();
        initEvents();
        init();
    }

    protected abstract void beforeInitWeNeedSomeParams(Object... params);

    public abstract void initViews();

    public abstract void initEvents();

    public abstract void init();

    public View findViewById(int id)
    {
        return mContentView.findViewById(id);
    }

    protected static int dpToPx(Context context, int dp)
    {
        return (int) (context.getResources().getDisplayMetrics().density * dp + 0.5f);
    }

}
           

也就是封装了一下popupWindow常用的一些设置,然后使用了类似模版方法模式,约束子类,必须实现initView,initEvent,init等方法

[java]

view plain copy print ?

  1. package com.zhy.imageloader;  
  2. import java.util.List;  
  3. import android.view.View;  
  4. import android.widget.AdapterView;  
  5. import android.widget.AdapterView.OnItemClickListener;  
  6. import android.widget.ListView;  
  7. import com.zhy.bean.ImageFloder;  
  8. import com.zhy.utils.BasePopupWindowForListView;  
  9. import com.zhy.utils.CommonAdapter;  
  10. import com.zhy.utils.ViewHolder;  
  11. public class ListImageDirPopupWindow extends BasePopupWindowForListView<ImageFloder>  
  12. {  
  13.     private ListView mListDir;  
  14.     public ListImageDirPopupWindow(int width, int height,  
  15.             List<ImageFloder> datas, View convertView)  
  16.     {  
  17.         super(convertView, width, height, true, datas);  
  18.     }  
  19.     @Override  
  20.     public void initViews()  
  21.     {  
  22.         mListDir = (ListView) findViewById(R.id.id_list_dir);  
  23.         mListDir.setAdapter(new CommonAdapter<ImageFloder>(context, mDatas,  
  24.                 R.layout.list_dir_item)  
  25.         {  
  26.             @Override  
  27.             public void convert(ViewHolder helper, ImageFloder item)  
  28.             {  
  29.                 helper.setText(R.id.id_dir_item_name, item.getName());  
  30.                 helper.setImageByUrl(R.id.id_dir_item_image,  
  31.                         item.getFirstImagePath());  
  32.                 helper.setText(R.id.id_dir_item_count, item.getCount() + ”张”);  
  33.             }  
  34.         });  
  35.     }  
  36.     public interface OnImageDirSelected  
  37.     {  
  38.         void selected(ImageFloder floder);  
  39.     }  
  40.     private OnImageDirSelected mImageDirSelected;  
  41.     public void setOnImageDirSelected(OnImageDirSelected mImageDirSelected)  
  42.     {  
  43.         this.mImageDirSelected = mImageDirSelected;  
  44.     }  
  45.     @Override  
  46.     public void initEvents()  
  47.     {  
  48.         mListDir.setOnItemClickListener(new OnItemClickListener()  
  49.         {  
  50.             @Override  
  51.             public void onItemClick(AdapterView<?> parent, View view,  
  52.                     int position, long id)  
  53.             {  
  54.                 if (mImageDirSelected != null)  
  55.                 {  
  56.                     mImageDirSelected.selected(mDatas.get(position));  
  57.                 }  
  58.             }  
  59.         });  
  60.     }  
  61.     @Override  
  62.     public void init()  
  63.     {  
  64.         // TODO Auto-generated method stub  
  65.     }  
  66.     @Override  
  67.     protected void beforeInitWeNeedSomeParams(Object… params)  
  68.     {  
  69.         // TODO Auto-generated method stub  
  70.     }  
  71. }  
package com.zhy.imageloader;

import java.util.List;

import android.view.View;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.ListView;

import com.zhy.bean.ImageFloder;
import com.zhy.utils.BasePopupWindowForListView;
import com.zhy.utils.CommonAdapter;
import com.zhy.utils.ViewHolder;

public class ListImageDirPopupWindow extends BasePopupWindowForListView<ImageFloder>
{
    private ListView mListDir;

    public ListImageDirPopupWindow(int width, int height,
            List<ImageFloder> datas, View convertView)
    {
        super(convertView, width, height, true, datas);
    }

    @Override
    public void initViews()
    {
        mListDir = (ListView) findViewById(R.id.id_list_dir);
        mListDir.setAdapter(new CommonAdapter<ImageFloder>(context, mDatas,
                R.layout.list_dir_item)
        {
            @Override
            public void convert(ViewHolder helper, ImageFloder item)
            {
                helper.setText(R.id.id_dir_item_name, item.getName());
                helper.setImageByUrl(R.id.id_dir_item_image,
                        item.getFirstImagePath());
                helper.setText(R.id.id_dir_item_count, item.getCount() + "张");
            }
        });
    }

    public interface OnImageDirSelected
    {
        void selected(ImageFloder floder);
    }

    private OnImageDirSelected mImageDirSelected;

    public void setOnImageDirSelected(OnImageDirSelected mImageDirSelected)
    {
        this.mImageDirSelected = mImageDirSelected;
    }

    @Override
    public void initEvents()
    {
        mListDir.setOnItemClickListener(new OnItemClickListener()
        {
            @Override
            public void onItemClick(AdapterView<?> parent, View view,
                    int position, long id)
            {

                if (mImageDirSelected != null)
                {
                    mImageDirSelected.selected(mDatas.get(position));
                }
            }
        });
    }

    @Override
    public void init()
    {
        // TODO Auto-generated method stub

    }

    @Override
    protected void beforeInitWeNeedSomeParams(Object... params)
    {
        // TODO Auto-generated method stub
    }

}
           

好了,现在就是我们正在的popupWindow咯,布局文件夹主要是个ListView,所以在initView里面,我们得设置它的适配器;当然了,这里的适配器依然用我们的CommonAdapter,几行代码搞定~~

然后我们需要和Activity交互,当我们点击某个文件夹的时候,外层的Activity需要改变它GridView的数据源,展示我们点击文件夹的图片;

关于交互,我们从Activity的角度去看弹出框,Activity想知道什么,只想知道选择了别的文件夹来告诉我,所以我们创建一个接口OnImageDirSelected,对Activity设置回调;

这里还可以这么写:就是把popupWindow的ListView公布出去,然后在Activity里面使用popupWindow.getListView(),setOnItemClickListener,这么做,个人觉得不好,耦合度太高,客户简单改下需求“这个文件夹展示,给我们换了,换成GridView”,呵呵,此时,你需要到处去修改Activity里面的代码,因为你Activity里面竟然还有个popupWindow.getListView。

好了,扯多了,初始化事件的代码:

[java]

view plain copy print ?

  1. @Override  
  2.     public void initEvents()  
  3.     {  
  4.         mListDir.setOnItemClickListener(new OnItemClickListener()  
  5.         {  
  6.             @Override  
  7.             public void onItemClick(AdapterView<?> parent, View view,  
  8.                     int position, long id)  
  9.             {  
  10.                 if (mImageDirSelected != null)  
  11.                 {  
  12.                     mImageDirSelected.selected(mDatas.get(position));  
  13.                 }  
  14.             }  
  15.         });  
  16.     }  
@Override
    public void initEvents()
    {
        mListDir.setOnItemClickListener(new OnItemClickListener()
        {
            @Override
            public void onItemClick(AdapterView<?> parent, View view,
                    int position, long id)
            {

                if (mImageDirSelected != null)
                {
                    mImageDirSelected.selected(mDatas.get(position));
                }
            }
        });
    }
           

如果有人设置了回调,我们就调用;

到此,整个popupWindow就出炉了,接下来就看啥时候让它展示了;

4、选择不同的文件夹

上面说道,当扫描图片完成,拿到包含图片的文件夹信息列表;这个列表就是我们popupWindow所需的数据,所以我们的popupWindow的初始化在handleMessage(上面贴了handler的代码)里面:

在handleMessage里面调用initListDirPopupWindw

[java]

view plain copy print ?

  1.     private void initListDirPopupWindw()  
  2.     {  
  3.         mListImageDirPopupWindow = new ListImageDirPopupWindow(  
  4.                 LayoutParams.MATCH_PARENT, (int) (mScreenHeight * 0.7),  
  5.                 mImageFloders, LayoutInflater.from(getApplicationContext())  
  6.                         .inflate(R.layout.list_dir, null));  
  7.         mListImageDirPopupWindow.setOnDismissListener(new OnDismissListener()  
  8.         {  
  9.             @Override  
  10.             public void onDismiss()  
  11.             {  
  12.                 // 设置背景颜色变暗  
  13.                 WindowManager.LayoutParams lp = getWindow().getAttributes();  
  14.                 lp.alpha = 1.0f;  
  15.                 getWindow().setAttributes(lp);  
  16.             }  
  17.         });  
  18.         // 设置选择文件夹的回调  
  19.         mListImageDirPopupWindow.setOnImageDirSelected(this);  
  20.     }  
/**
     * 初始化展示文件夹的popupWindw
     */
    private void initListDirPopupWindw()
    {
        mListImageDirPopupWindow = new ListImageDirPopupWindow(
                LayoutParams.MATCH_PARENT, (int) (mScreenHeight * 0.7),
                mImageFloders, LayoutInflater.from(getApplicationContext())
                        .inflate(R.layout.list_dir, null));

        mListImageDirPopupWindow.setOnDismissListener(new OnDismissListener()
        {

            @Override
            public void onDismiss()
            {
                // 设置背景颜色变暗
                WindowManager.LayoutParams lp = getWindow().getAttributes();
                lp.alpha = 1.0f;
                getWindow().setAttributes(lp);
            }
        });
        // 设置选择文件夹的回调
        mListImageDirPopupWindow.setOnImageDirSelected(this);
    }
           

我们初始化我们的popupWindow,设置了关闭对话框的回调,已经设置了选择不同文件夹的回调;

这里仅仅是初始化,下面看我们合适将其弹出的,其实整个Activity也就一个事件,点击弹出该对话框,所以看Activity的initEvents方法:

[java]

view plain copy print ?

  1. private void initEvent()  
  2.     {  
  3.         mBottomLy.setOnClickListener(new OnClickListener()  
  4.         {  
  5.             @Override  
  6.             public void onClick(View v)  
  7.             {  
  8.                 mListImageDirPopupWindow  
  9.                         .setAnimationStyle(R.style.anim_popup_dir);  
  10.                 mListImageDirPopupWindow.showAsDropDown(mBottomLy, 0, 0);  
  11.                 // 设置背景颜色变暗  
  12.                 WindowManager.LayoutParams lp = getWindow().getAttributes();  
  13.                 lp.alpha = .3f;  
  14.                 getWindow().setAttributes(lp);  
  15.             }  
  16.         });  
  17.     }  
private void initEvent()
    {
        /**
         * 为底部的布局设置点击事件,弹出popupWindow
         */
        mBottomLy.setOnClickListener(new OnClickListener()
        {
            @Override
            public void onClick(View v)
            {
                mListImageDirPopupWindow
                        .setAnimationStyle(R.style.anim_popup_dir);
                mListImageDirPopupWindow.showAsDropDown(mBottomLy, 0, 0);

                // 设置背景颜色变暗
                WindowManager.LayoutParams lp = getWindow().getAttributes();
                lp.alpha = .3f;
                getWindow().setAttributes(lp);
            }
        });
    }
           

可以看到,我们为底部布局设置点击事件;设置popupWindow的弹出与消失的动画;已经让Activity背景变暗变亮,通过改变Window alpha实现的。变亮在弹出框消息的监听里面~~

动画的文件就不贴了,大家自己看源码;

popupWindow弹出了,用户此时可以选择不同的文件夹,那么现在该看选择后的回调的代码了:

我们的Activity实现了该接口,直接看实现的方法:

[java]

view plain copy print ?

  1. @Override  
  2. public void selected(ImageFloder floder)  
  3. {  
  4.     mImgDir = new File(floder.getDir());  
  5.     mImgs = Arrays.asList(mImgDir.list(new FilenameFilter()  
  6.     {  
  7.         @Override  
  8.         public boolean accept(File dir, String filename)  
  9.         {  
  10.             if (filename.endsWith(“.jpg”) || filename.endsWith(“.png”)  
  11.                     || filename.endsWith(”.jpeg”))  
  12.                 return true;  
  13.             return false;  
  14.         }  
  15.     }));  
  16.     mAdapter = new MyAdapter(getApplicationContext(), mImgs,  
  17.             R.layout.grid_item, mImgDir.getAbsolutePath());  
  18.     mGirdView.setAdapter(mAdapter);  
  19.     // mAdapter.notifyDataSetChanged();  
  20.     mImageCount.setText(floder.getCount() + ”张”);  
  21.     mChooseDir.setText(floder.getName());  
  22.     mListImageDirPopupWindow.dismiss();  
  23. }  
@Override
    public void selected(ImageFloder floder)
    {

        mImgDir = new File(floder.getDir());
        mImgs = Arrays.asList(mImgDir.list(new FilenameFilter()
        {
            @Override
            public boolean accept(File dir, String filename)
            {
                if (filename.endsWith(".jpg") || filename.endsWith(".png")
                        || filename.endsWith(".jpeg"))
                    return true;
                return false;
            }
        }));
        /**
         * 可以看到文件夹的路径和图片的路径分开保存,极大的减少了内存的消耗;
         */
        mAdapter = new MyAdapter(getApplicationContext(), mImgs,
                R.layout.grid_item, mImgDir.getAbsolutePath());
        mGirdView.setAdapter(mAdapter);
        // mAdapter.notifyDataSetChanged();
        mImageCount.setText(floder.getCount() + "张");
        mChooseDir.setText(floder.getName());
        mListImageDirPopupWindow.dismiss();

    }
           

我们改变了GridView的适配器,以及底部的控件上的文件夹名称,文件数量等等;

好了,到此结束;整篇由于篇幅原因没有贴任何布局文件,大家自己通过源码查看;

在此希望大家可以通过该案例,能够去其糟粕,取其精华,学习其中值得借鉴的代码风格,不要真的当作一个例子去学习~~

源码点击下载  

ps:请真机测试,反正我的模拟器扫描不到图片~

ps:运行出现空指针的话,在getImages中添加判断,if(parentFile.list()==null)continue , 切记~~~具体位置,上面有说; 

———————————————————————————————————

我建了一个QQ群,方便大家交流。群号:55032675

———————————————————————————————————-

博主部分视频已经上线,如果你不喜欢枯燥的文本,请猛戳(初录,期待您的支持):

1、高仿微信5.2.1主界面及消息提醒

2、高仿QQ5.0侧滑