Android 超高仿微信图片选择器 图片该这么加载

{

this.count = count;

}

}

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

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

@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

private Handler mHandler = new Handler()

{

public void handleMessage(android.os.Message msg)

{

mProgressDialog.dismiss();

//为View绑定数据

data2View();

//初始化展示文件夹的popupWindw

initListDirPopupWindw();

}

};

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

/**

  • 为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的:

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

{

/**

  • 用户选择的图片,存储为图片的完整路径

*/

public static List mSelectedImage = new LinkedList();

/**

  • 文件夹路径

*/

private String mDirPath;

public MyAdapter(Context context, List 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使用的超类:

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 extends PopupWindow

{

/**

  • 布局文件的最外层View

*/

protected View mContentView;

protected Context context;

/**

  • ListView的数据集

*/

protected List 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 mDatas)

{

this(contentView, width, height, focusable, mDatas, new Object[0]);

}

public BasePopupWindowForListView(View contentView, int width, int height,

boolean focusable, List 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等方法

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

{

private ListView mListDir;

public ListImageDirPopupWindow(int width, int height,

List 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(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。

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

@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));

}

}

});

}

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

最后

**一个零基础的新人,我认为坚持是最最重要的。**我的很多朋友都找我来学习过,我也很用心的教他们,可是不到一个月就坚持不下来了。我认为他们坚持不下来有两点主要原因:

他们打算入行不是因为兴趣,而是因为所谓的IT行业工资高,或者说完全对未来没有任何规划。

刚开始学的时候确实很枯燥,这确实对你是个考验,所以说坚持下来也很不容易,但是如果你有兴趣就不会认为这是累,不会认为这很枯燥,总之还是贵在坚持。

技术提升遇到瓶颈了?缺高级Android进阶视频学习提升自己吗?还有大量大厂面试题为你面试做准备!

提升自己去挑战一下BAT面试难关吧

对于很多Android工程师而言,想要提升技能,往往是自己摸索成长,不成体系的学习效果低效漫长且无助。整理的这些知识图谱希望对Android开发的朋友们有所参考以及少走弯路,本文的重点是你有没有收获与成长,其余的都不重要,希望读者们能谨记这一点。

不论遇到什么困难,都不应该成为我们放弃的理由!

如果有什么疑问的可以直接私我,我尽自己最大力量帮助你!

最后祝各位新人都能坚持下来,学有所成。

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

s8fo1j-1713769503868)]

[外链图片转存中…(img-n27XThXg-1713769503869)]

[外链图片转存中…(img-0Ifoizrx-1713769503870)]

[外链图片转存中…(img-roZ9AJUV-1713769503871)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

[外链图片转存中…(img-yMbIGWDx-1713769503872)]

最后

**一个零基础的新人,我认为坚持是最最重要的。**我的很多朋友都找我来学习过,我也很用心的教他们,可是不到一个月就坚持不下来了。我认为他们坚持不下来有两点主要原因:

他们打算入行不是因为兴趣,而是因为所谓的IT行业工资高,或者说完全对未来没有任何规划。

刚开始学的时候确实很枯燥,这确实对你是个考验,所以说坚持下来也很不容易,但是如果你有兴趣就不会认为这是累,不会认为这很枯燥,总之还是贵在坚持。

技术提升遇到瓶颈了?缺高级Android进阶视频学习提升自己吗?还有大量大厂面试题为你面试做准备!

提升自己去挑战一下BAT面试难关吧

[外链图片转存中…(img-nyjtL22z-1713769503872)]

对于很多Android工程师而言,想要提升技能,往往是自己摸索成长,不成体系的学习效果低效漫长且无助。整理的这些知识图谱希望对Android开发的朋友们有所参考以及少走弯路,本文的重点是你有没有收获与成长,其余的都不重要,希望读者们能谨记这一点。

不论遇到什么困难,都不应该成为我们放弃的理由!

如果有什么疑问的可以直接私我,我尽自己最大力量帮助你!

最后祝各位新人都能坚持下来,学有所成。

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

  • 4
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值