基于PinnedSectionListView实现联系人通讯录并且点击打电话
PinnedSectionListView具体下载地址、使用方法和注意事项:http://www.cnblogs.com/zzw1994/p/4997601.html
怎么根据联系人姓名首字符顺序读取联系人:http://www.cnblogs.com/zzw1994/p/4998524.html
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
> <com.zzw.contacts.PinnedSectionListView
android:id="@+id/listView"
android:layout_width="match_parent"
android:layout_height="match_parent" /> </RelativeLayout>
activity_main.xml
权限:
<!-- 写权限 -->
<uses-permission android:name="android.permission.WRITE_CONTACTS" />
<!-- 读权限 -->
<uses-permission android:name="android.permission.READ_CONTACTS" />
<!-- 打电话权限 -->
<uses-permission android:name="android.permission.CALL_PHONE" />
主要实现代码:
package com.zzw.contacts; import java.util.ArrayList; import com.zzw.contacts.PinnedSectionListView.PinnedSectionListAdapter; import android.app.Activity;
import android.content.ContentResolver;
import android.content.Context;
import android.content.Intent;
import android.database.Cursor;
import android.graphics.Color;
import android.net.Uri;
import android.os.Bundle;
import android.provider.ContactsContract;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.ArrayAdapter;
import android.widget.TextView; public class MainActivity extends Activity { @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); ArrayList<Contact> contacts = new ArrayList<Contact>();
readContacts(contacts); final ArrayList<Item> items = new ArrayList<Item>(); // 从字母A开始到Z。26个字母,遍历联系人中的首字符是否相等。
// 相等则归入一组。
int A = (int) 'A';
for (int i = 0; i < 26; i++) {
int letter = A + i;
char c = (char) letter; Item group = new Item();
group.type = Item.GROUP;
group.text = c + "";
items.add(group); for (int j = 0; j < contacts.size(); j++) {
Contact contact = contacts.get(j);
String s = contact.firstLetterOfName();
if (s.equals(group.text)) {
Item child = new Item();
child.type = Item.CHILD;
child.text = contact.name + " " + contact.getPhoneNumbers();
child.contact = contact;
items.add(child);
}
}
} PinnedSectionListView listView = (PinnedSectionListView) findViewById(R.id.listView); ArrayAdapter<Item> adapter = new MyAdapter(this, -1, items);
listView.setAdapter(adapter); listView.setOnItemClickListener(new OnItemClickListener() { @Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
Item item = items.get(position); if (item.type == Item.CHILD) {
Contact contact = item.contact;
// 简单演示期间,我们只选择第一个电话(不管有几个电话)。
String number = contact.phoneNumbers.get(0);
if (!number.equals("")) {
// 拨打电话
Intent intent = new Intent(Intent.ACTION_CALL, Uri.parse("tel:" + number));
startActivity(intent);
}
}
}
}); } private class MyAdapter extends ArrayAdapter<Item>implements PinnedSectionListAdapter { private LayoutInflater inflater;
private ArrayList<Item> items; public MyAdapter(Context context, int resource, ArrayList<Item> items) {
super(context, resource, items);
this.items = items;
inflater = (LayoutInflater) getSystemService(Context.LAYOUT_INFLATER_SERVICE);
} @Override
public int getCount() {
return items.size();
} @Override
public int getItemViewType(int position) { return items.get(position).type;
} @Override
public int getViewTypeCount() { return 2;
} @Override
public Item getItem(int position) {
return items.get(position);
} @Override
public View getView(int position, View convertView, ViewGroup parent) { String text = getItem(position).text; switch (getItemViewType(position)) {
case Item.GROUP: if (convertView == null) {
convertView = inflater.inflate(android.R.layout.simple_list_item_1, null);
} TextView text1 = (TextView) convertView.findViewById(android.R.id.text1); // 首字符,分组的依据
text1.setText(text);
text1.setBackgroundColor(Color.GRAY); break; case Item.CHILD:
if (convertView == null) {
convertView = inflater.inflate(android.R.layout.simple_list_item_2, null);
} TextView text2 = (TextView) convertView.findViewById(android.R.id.text2);
text2.setText(text); break;
}
return convertView;
} // 假设此方法返回皆为false。那么PinnedSectionListView将退化成为一个基础的ListView.
// 只不过退化后的ListView只是一个拥有两个View Type的ListView。
// 从某种角度上讲,此方法对于PinnedSectionListView至关重要,因为返回值true或false,
// 将直接导致PinnedSectionListView是一个PinnedSectionListView,还是一个普通的ListView。
@Override
public boolean isItemViewTypePinned(int viewType) {
boolean type = false;
switch (viewType) {
case Item.GROUP:
type = true;
break; case Item.CHILD:
type = false;
break; default:
type = false;
break;
}
return type;
} } // 读取设备联系人的一般方法。大致流程就是这样,模板化的操作代码。
private void readContacts(ArrayList<Contact> contacts) {
Uri uri = Uri.parse("content://com.android.contacts/contacts");
ContentResolver resolver = this.getContentResolver(); // 在这里我们给query传递进去一个SORT_KEY_PRIMARY
// 告诉ContentResolver获得的结果按照联系人名字的首字母有序排序
Cursor cursor = resolver.query(uri, null, null, null,
android.provider.ContactsContract.Contacts.SORT_KEY_PRIMARY); while (cursor.moveToNext()) {
// 联系人id
String id = cursor.getString(cursor.getColumnIndex(android.provider.ContactsContract.Contacts._ID));
// Sort key,读取的联系人按照姓名从 A->Z 的排序分组
String sork_key_primary = cursor
.getString(cursor.getColumnIndex(android.provider.ContactsContract.Contacts.SORT_KEY_PRIMARY));
// 获取联系人姓名
String name = cursor
.getString(cursor.getColumnIndex(android.provider.ContactsContract.Contacts.DISPLAY_NAME)); Contact mContact = new Contact();
mContact.id = id;
mContact.sort_key_primay = sork_key_primary;
mContact.name = name; // 获取联系人的手机号码
Cursor phone = resolver.query(ContactsContract.CommonDataKinds.Phone.CONTENT_URI, null,
ContactsContract.CommonDataKinds.Phone.CONTACT_ID + "=" + id, null, null);
// 取得电话号码(可能存在多个号码)
// 因为在同一个名字下,用户可能存有一个以上号码
// 遍历
ArrayList<String> phoneNumbers = new ArrayList<String>();
while (phone.moveToNext()) {
int phoneFieldColumnIndex = phone.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER);
String phoneNumber = phone.getString(phoneFieldColumnIndex);
phoneNumbers.add(phoneNumber);
} mContact.phoneNumbers = phoneNumbers;
contacts.add(mContact);
}
} // 用于承载数据块的类。
// 字段分为类型(type)和值(text)。
private class Item {
public static final int GROUP = 0;
public static final int CHILD = 1; public int type;
public String text; public Contact contact = null;
} // 用于装载从联系人数据库中读取到的数据。
// 结构化数据,便于数据操作和访问。
private class Contact {
public String id;
public String name;
public String sort_key_primay;
public ArrayList<String> phoneNumbers; // 获得一个联系人名字的首字母。
// 比如一个人的名字叫做“安卓”,那么这个人的首字母是:A
public String firstLetterOfName() {
String s = sort_key_primay.charAt(0) + ""; return s.toUpperCase();
} public String getPhoneNumbers() {
String phones = "";
for (int i = 0; i < phoneNumbers.size(); i++) {
phones += ": " + phoneNumbers.get(i);
}
return phones;
}
}
}
基于PinnedSectionListView实现联系人通讯录并且点击打电话的更多相关文章
- 【WP8.1开发】基于应用的联系人存储
上一篇文章所吹的牛是访问系统(手机)上的联系人,当然那只是读不能改,这是自然的,要是让你能随便修改用户的联系人信息的话,那后果很严重,有些恶意开发者就有可能把”你的户口改成猪“. 但是,API也允许应 ...
- Android软件开发之获取通讯录联系人信息
Android手机的通讯录联系人全部都存在系统的数据库中,如果须要获得通讯里联系人的信息就须要访问系统的数据库,才能将信息拿出来. 这一篇文章我主要带领同学们熟悉Android的通讯录机制. 图中选中 ...
- Swift - 访问通讯录联系人(使用系统提供的通讯录交互界面)
1,通讯录访问介绍 通讯录(或叫地址簿,电话簿)是一个数据库,里面储存了联系人的相关信息.要实现访问通讯录有如下两种方式: (1)AddressBook.framework框架 : 没有界面,通过代码 ...
- iOS获取通讯录所有联系人信息
以下是2种方式: 第一种方法: GetAddressBook.h #import <Foundation/Foundation.h> @interface GetAddressBook : ...
- Android6.0运行时权限(基于RxPermission开源库)
版权声明:本文为HaiyuKing原创文章,转载请注明出处! 前言 在6.0以前的系统,都是权限一刀切的处理方式,只要用户安装,Manifest申请的权限都会被赋予,并且安装后权限也撤销不了. And ...
- 计算机网络课设之TCP通讯录
这篇文章我主要是想对这学期末计算机网络课程设计所做的一个小项目也就是基于tcp协议的通讯录做一个总结梳理.项目的具体的代码实现是基于C语言,当然在此之前网上也有一些基于c++编写的tcp通讯录,原理都 ...
- Swift - 访问通讯录-使用AddressBook.framework和AddressBookUI.framework框架实现
1,通讯录访问介绍 通讯录(或叫地址簿,电话簿)是一个数据库,里面储存了联系人的相关信息.要实现访问通讯录有如下两种方式: (1)AddressBook.framework框架 : 没有界面,通过代码 ...
- iOS获取通讯录 电话号码与姓名
// 还是导入 #import <AddressBook/AddressBook.h> - (void)fetchAddressBookBeforeIOS9{ ABAddressBookR ...
- Jquery Mobile设计Android通讯录第二章
本文是jQuery Mobile设计Android通讯录系统教程的第二篇,在上一篇教程中(http://publish.itpub.net/a2011/0517/1191/000001191561.s ...
随机推荐
- FileListEntityProcessor
一个简单的实体处理程序,可以用于枚举标准文件系统中的文件列表,它不需要使用DataSource.属性如下: fileName:(必填) 用正则表达式来标记文件名 baseDir:(必填) 基础目录,绝 ...
- C# 使用代码来操作 IIS
由于需要维护网站的时候,可以自动将所有的站点HTTP重定向到指定的静态页面上. 要操作 IIS 主要使用到的是“Microsoft.Web.Administration.dll”. 该类库不可以在引用 ...
- Callable--创建有返回值的线程
转自:JAVA 笔记 Callable 与 FutureTask:有返回值的多线程 常用的Thread类在run方法执行完之后是没有返回值的,要实现子线程完成任务后返回值给主线程需要借助第三方转存.C ...
- webapp新体验Rem实现移动端网页适配详解资源
本来想写一篇,webapp使用Rem的问题,查了一下相关rem的介绍之后,发现很多平台已经解释的很清楚了,图文并茂,于是我便想将其解释资源整理一些,方便以后自己查阅. 腾讯ISUX:web app变革 ...
- 转: ffmpeg 的deom列表
http://blog.csdn.net/leixiaohua1020/article/details/47072861
- JS each 跳出
break 对应的是 return false continue 对应的是 return true 这个问题每次都会记不清,都要去查一遍百度再确定答案,在这里也记一遍好了. function getP ...
- .NET DLL 保护措施应用实例(百度云分享工具)
最近做了个小工具,将保护措施思路全部应用到了此工具中. 点我下载 “百度分享工具”介绍 大家都知道,在百度云盘中分享文件,只能手工一条条地点击“分享”,如果想分享很多文件,操作会非常辛苦.“百度云分享 ...
- 关于使用AIDL出现空指针的解决办法
使用AIDL进行远程调用的时候出现的空指针异常,解决过程稍微有点小曲折.具体安下 1.先贴异常信息 ERROR/AndroidRuntime(9435): FATAL EXCEPTION: main ...
- MySQL之学生名次问题
--对输入的数据进行约束create table t(studentID char(10), [name] varchar(8), startDate char(10) Check (isdate(s ...
- Linux 基本命令学习笔记
1. 文件管理 Ø touch 新建文件.例: touch test.txt 新建一个test.txt 文件. Ø cp 复制文件.例:cp ./user_one/test_one ./user ...