网上Android联系人列表的样例也非常多,都和微信的联系人差点儿相同,因为项目用到了联系人列表索引功能(产品把字母item给去掉了),只是也还是好实现。这里我也来分享分享我的实现,免得以后忘了。那先看看效果(Demo在结尾有下载地址):

要达到的效果就是这么简单。

先说说思路吧:首先为联系人对象加入一个pinyin字段,当获取到了联系人原始数据后,把每一个联系人的名字转换为拼音。并为pinyin字段设置值。

然后获取联系人中出现过哪些字母的拼音保存为数组(这就是字母的item),然后和联系人拼音再组合成新的数组,利用Arrays.sort排序功能,就依次按字母顺序拍好了联系人列表,接下来就是把字母在联系人列表中出现的位置给保存起来,绘制右边字母索引时,给对应的字母设置当前字母在联系人列表的位置,滑动到某一个字母是,就把listview移动到那个位置即可了。

一:先看看提取联系人中出现的字母,并排序

/**
* 把数据排序。并把A-Z顺序加进去
*
* @param carTypes
* @return
*/
public String[] sortIndex(List<ConstastBean> constastBeans) {
TreeSet<String> set = new TreeSet<String>();
for (ConstastBean constastBean : constastBeans) {
char ch = constastBean.getPinyin().charAt(0);
set.add(String.valueOf(ch).toUpperCase(Locale.getDefault()));// 获取出现的首字母
}
String[] names = new String[constastBeans.size() + set.size()];// 新数组,用于保存首字母 + 联系人拼音
int i = 0;
for (String string : set) { // 把set中的字母加入到新数组中(前面)
names[i] = string;
i++;
}
<span style="white-space:pre"> </span>//把联系人拼音提取到一个数组中
String[] pyheader = new String[constastBeans.size()];
for (int j = 0; j < constastBeans.size(); j++) {
pyheader[j] = constastBeans.get(j).getPinyin();
} System.arraycopy(pyheader, 0, names, set.size(), pyheader.length);// <span style="font-family: Arial, Helvetica, sans-serif;">把联系人拼音加入到后面,结果就是联系人拼音和出现过的首字母在一个数组里面(是无序的)</span> // 自己主动依照首字母排序
Arrays.sort(names, String.CASE_INSENSITIVE_ORDER);// 严格依照字母顺序排序,忽略字母大写和小写,结果为按拼音排序的数组返回
return names; }

这个返回的就是排好序的字母顺序了(如[A, 安妮,G,盖伦...Z])。

二:接下来就是依据返回的顺序,把联系人对象排序

/**
* 依据名字排序对数据进行排序 由于默认是数字在首位,为了把数字排到末尾,须要进行转换
*
* @param arry
* @return
*/
public ArrayList<ConstastBean> getAllLists(String[] arry)
{
ArrayList<ConstastBean> lists = new ArrayList<ConstastBean>();// 保存排好序的数据
ArrayList<ConstastBean> lists2 = new ArrayList<ConstastBean>();// 保存数字开头的数据
ArrayList<ConstastBean> lists3 = new ArrayList<ConstastBean>();// 保存字母数据
// 对数据进行排序
for(int i = 0; i < arry.length; i++)
{
for(int j = 0; j < sourceData.size(); j++)
{
if(arry[i].equals(sourceData.get(j).getPinyin()))
{
lists.add(sourceData.get(j));
break;
}
// else //须要显示单个字母的item,这里就不用凝视。在adapter中应给为这个item单独设置一个布局
// {
// ConstastBean contactBean = new ConstastBean();
// contactBean.setPinyin(arry[i]);
// contactBean.setNickName(arry[i]);
// lists.add(contactBean);
// break;
// }
}
}
// 分离出数字数据和字母数据
int index = getLetter(lists);// 获取字母开头的位置
for(int i = 0; i < lists.size(); i++)
{
if(i < index)
{
lists2.add(lists.get(i));
}
else
{
lists3.add(lists.get(i));
}
}
lists.clear();
lists.addAll(lists3);
lists.addAll(lists2); return lists;
}

因为排好序的是数字在字母前面,可是这里须要数字在后面,所以先查询第一个字母出现的位置。然后从那个位置截取为2个集合。再把数字集合加入到字母集合后面即可了。

三:初始化工作都做好了。就该遍历获取每一个字母所相应的位置了

<span style="white-space:pre">	</span>selector = new HashMap<String, Integer>();
// 遍历排好序的数据,获取每一个字母的位置
for (int i = 0; i < indexStr.length; i++) {
for (int j = 0; j < datas.size(); j++) {
if (datas.get(j).getPinyin().toLowerCase(Locale.getDefault()).startsWith(indexStr[i<span style="white-space:pre"> </span>].toLowerCase(Locale.getDefault()))) {
selector.put(indexStr[i], j);
break;
}
String pinyin = datas.get(j).getPinyin();
if(indexStr[i].equals("#") && isNumeric(pinyin.substring(0,1)))
{
selector.put(indexStr[i], j);
return;
}
}
}

返回的结果就类似于这样的:{D=2, #=23, E=4, G=5, A=0, L=11, M=12, N=14, H=6, J=8, K=10, T=19, V=20, S=16, R=15, Z=21}

/**
* 绘制索引条
*/
public void drawIndexView() {
LinearLayout.LayoutParams params = new LayoutParams(LayoutParams.MATCH_PARENT, height);
for (int i = 0; i < indexStr.length; i++) {
TextView tv = new TextView(this);
tv.setLayoutParams(params);
tv.setText(indexStr[i]);
tv.setGravity(Gravity.CENTER);
tv.setTextColor(this.getResources().getColor(R.color.indexs_color));
tv.setTextSize(13); layoutIndex.addView(tv); layoutIndex.setOnTouchListener(new OnTouchListener() { @Override
public boolean onTouch(View v, MotionEvent event) {
// TODO Auto-generated method stub
float y = event.getY();
int index = (int) y / height;// 得到点击字母位置的索引
String key = "";
if (index < indexStr.length && index > -1) {
key = indexStr[index];
if (selector.containsKey(key)) {
int position = selector.get(key);
if (listView.getHeaderViewsCount() > 0) {// 加入的header给去掉
listView.setSelectionFromTop(
position + listView.getHeaderViewsCount(), 0);
} else {
listView.setSelectionFromTop(position, 0);// 滑动到第一项
}
}
if (key.equals("↑"))
{
listView.setSelectionFromTop(0, 0);// 滑动到第一项
}
}
if(!key.equals(""))
{
showTv.setText(key);
showTv.setVisibility(View.VISIBLE);
} switch (event.getAction()) {
case MotionEvent.ACTION_UP:
case MotionEvent.ACTION_CANCEL:
case MotionEvent.ACTION_OUTSIDE:
showTv.setVisibility(View.GONE);
break;
case MotionEvent.ACTION_DOWN:
// layoutIndex.setBackground();
break;
} return true;
}
});
}
}

这样就完毕了带索引的联系人列表的实现:Demo源代码下载

Android带索引联系人列表的更多相关文章

  1. Android UI(五)云通讯录项目之联系人列表,带侧滑选择,带搜索框

    作者:泥沙砖瓦浆木匠网站:http://blog.csdn.net/jeffli1993个人签名:打算起手不凡写出鸿篇巨作的人,往往坚持不了完成第一章节.交流QQ群:[编程之美 365234583]h ...

  2. 【Android Developers Training】 98. 获取联系人列表

    注:本文翻译自Google官方的Android Developers Training文档,译者技术一般,由于喜爱安卓而产生了翻译的念头,纯属个人兴趣爱好. 原文链接:http://developer ...

  3. Android实现SQLite数据库联系人列表

    Android实现SQLite数据库联系人列表 开发工具:Andorid Studio 1.3 运行环境:Android 4.4 KitKat 工程内容 实现一个通讯录查看程序: 要求使用SQLite ...

  4. Android实现“是否退出”对话框和“带图标的列表”对话框

    今天我们学习的内容是实现两种对话框(Dialog),第一种是询问是否退出对话框,另外一种是带图标的列表对话框,程序的执行效果是,我们点击button1的时候,弹出第一种对话框,我们点击button2的 ...

  5. Android Phonebook编写联系人UI加载及联系人保存流程(三)

    2014-01-07 09:54:13  将百度空间里的东西移过来. 本文从点击“添加联系人”Button开始,分析新建联系人页面UI是如何加载,以及新的联系人信息是如何保存的,借此,我们一探Phon ...

  6. Android 读取手机联系人、拨号、发送短信及长按菜单的操作

    本示例实现了读取手机联系人,拨号.发送短信及长按出现菜单选项的操作↓ 1.Andrid项目结构图↓主要操作图中红色方框内的文件. 2.首先布局代码如下↓ a, main.xml 程序运行的主界面,主要 ...

  7. 动手分析安卓仿QQ联系人列表TreeView控件

    因项目需要需要用到仿QQ联系人列表的控件样式,于是网上找到一个轮子(https://github.com/TealerProg/TreeView),工作完成现在简单分析一下这个源码.   一. 需要用 ...

  8. 列表的初识,列表的索引切片,列表的增删改查,列表的嵌套,元组的初识,range

    1 内容总览 列表的初识 列表的索引切片 列表的增删改查 列表的嵌套 元组的初识(了解) 元组的简单应用(了解) range 2 具体内容 列表的初识 why: str: 存储少量的数据.切片出来全都 ...

  9. C#:基于WMI查询USB设备信息 及 Android设备厂商VID列表

    /* ---------------------------------------------------------- 文件名称:WMIUsbQuery.cs 作者:秦建辉 MSN:splashc ...

随机推荐

  1. 3ds Max 2018 在安装后无法启动或出现不稳定

    问题: 安装 3ds Max 2018 后,软件无法正常启动,或在打开后不久出现不稳定和崩溃. 原因: 有多种原因可能会导致这些错误: ▪ 3ds Max.Windows 更新和 ProSound.d ...

  2. js悬浮吸顶

    <!DOCTYPE html> <head> <meta charset="UTF-8"> <title>吸顶和锚点链接</t ...

  3. 关于wdsl

    WSDL元素 WSDL元素基于XML语法描述了与服务进行交互的基本元素: Type(消息类型):数据类型定义的容器,它使用某种类型系统(如XSD). Message(消息):通信数据的抽象类型化定义, ...

  4. [agc015c]nuske vs phantom thnook

    题意: 有一个n*m的网格图,每个格子是蓝色或白色.四相邻的两个格子连一条边,保证蓝格子构成一个森林. 有q组询问,每次询问给出一个矩形,问矩形内蓝格子组成的联通块个数. $1\leq n,m\leq ...

  5. Mint-UI 没有样式?

    如果用mint-ui组件,如toast没有样式,是因为没有映入全局样式和导入MintUI 方法如下: 1.安装 npm install mint-ui -S -S表示 --save 2.在main.j ...

  6. 拓展Lucas小结

    拓展Lucas是解决大组合数取模非质数(尤其是含平方因子的合数)问题的有力工具... 首先对模数质因数分解,把每个质因子单独拎出来处理答案,然后用中国剩余定理(excrt)合并 问题转化为,对于每个质 ...

  7. Python格式化字符串、占位符、合并数组

    合并数组 参考链接:https://www.cnblogs.com/chaihy/p/7243143.html >>> a=[2] >>> b=[3] >&g ...

  8. HDU 4945 2048 DP 组合

    思路: 这个题写了一个背包的解法,超时了.搜了下题解才发现我根本不会做. 思路参见这个: 其实我们可以这样来考虑,求补集,用全集减掉不能组成2048的集合就是答案了. 因为只要达到2048就可以了,所 ...

  9. 【codeforces 589G】Hiring

    [题目链接]:http://codeforces.com/problemset/problem/589/G [题意] 有n个人; 每个人每天在开始工作之前,都需要di单位的准备时间,然后才能开始工作; ...

  10. RPC架构

    RPC架构 学习了: https://www.cnblogs.com/ChrisMurphy/p/6550184.html RPC架构简介,有一个Java例子: http://blog.csdn.ne ...