Android基于BaseExpandableListAdapter实现的二级列表仿通话记录功能详解

本文实例讲述了Android基于BaseExpandableListAdapter实现的二级列表仿通话记录功能。分享给大家供大家参考,具体如下:

android SDK中带有这样类似的例子,但是那个还是静态数据,没有实际应用价值,参考意义不大。

网上找了很多,还是那样的情况,几乎是同一篇文章,大家转来转去。况且,那篇例子也是静态的数据。

还是自己试试,自己写一个吧。程序读取手机系统的通话记录,按联系人分组,显示到列表。

开发工具:eclipse
运行环境:htc G9 android2.3.3

不多说,先看效果:

展开后的效果:

继续展开的效果:

main.xml:

<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="fill_parent" > <ExpandableListView android:layout_width="fill_parent" android:layout_height="fill_parent" android:id="@+id/list" android:groupIndicator="@drawable/tubiao_button" android:layout_gravity="right" android:indicatorRight="0px" /> </LinearLayout>

lis_item.xml:

<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="fill_parent" android:paddingLeft="2.0dip" > <ImageView android:id="@+id/contact" android:layout_width="wrap_content" android:layout_height="wrap_content" android:src="@drawable/contact" android:layout_alignParentLeft="true" android:layout_centerVertical="true" android:layout_marginRight="10.0dip" /> <ImageView android:id="@+id/open" android:layout_width="wrap_content" android:layout_height="wrap_content" android:src="@drawable/min" android:layout_toRightOf="@id/contact" android:layout_centerVertical="true" android:layout_marginLeft="5.0dip" android:layout_marginRight="5.0dip" /> <TextView android:id="@+id/name" android:textAppearance="?android:textAppearanceLarge" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_marginLeft="10.0dip" android:layout_marginTop="5.0dip" android:layout_marginRight="10.0dip" android:singleLine="true" android:layout_toRightOf="@id/open" android:layout_alignTop="@id/open" /> <ImageView android:id="@+id/type" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_marginTop="10.0dip" android:layout_marginRight="8.0dip" android:layout_marginLeft="5.0dip" android:src="@drawable/call_in" android:layout_alignParentRight="true" /> <TextView android:id="@+id/count" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_marginTop="10.0dip" android:layout_marginRight="8.0dip" android:layout_marginLeft="10.0dip" android:layout_alignParentRight="true" /> <TextView android:id="@+id/number" android:textAppearance="?android:textAppearanceSmall" android:ellipsize="marquee" android:layout_width="wrap_content" android:layout_height="wrap_content" android:singleLine="true" android:layout_below="@id/name" android:layout_alignLeft="@id/name" android:layout_alignWithParentIfMissing="true" /> <TextView android:id="@+id/date" android:textAppearance="?android:textAppearanceSmall" android:ellipsize="marquee" android:layout_width="wrap_content" android:layout_height="wrap_content" android:singleLine="true" android:layout_alignParentBottom="true" android:layout_alignParentRight="true" /> </RelativeLayout>

这里的也是采用继承BaseExpandableListAdapter来实现,具体请看代码。

主角Activity:

public class CollLogActivity extends Activity { private ExpandableListView listView; private MyAsyncQueryHandler queryHandler; private MyExpandableListAdapter listAdapter; private ArrayList<ContentValues> group; private ArrayList<List<ContentValues>> child; private static final Uri uri = Uri.parse("content://call_log/calls"); private static final String[] projection = { "_id", "number", "date", "type", "new", "name" }; private static final int INCOMING_TYPE = 1; private static final int OUTGOING_TYPE = 2; private static final int MISSED_TYPE = 3; private String currentNumber = "-1"; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); listView = (ExpandableListView) findViewById(R.id.list); queryHandler = new MyAsyncQueryHandler(getContentResolver()); group = new ArrayList<ContentValues>(); child = new ArrayList<List<ContentValues>>(); } @Override protected void onResume() { super.onResume(); startQuery(); } private class MyExpandableListAdapter extends BaseExpandableListAdapter { private LayoutInflater inflater; public MyExpandableListAdapter(Context context) { this.inflater = LayoutInflater.from(context); } @Override public Object getChild(int groupPosition, int childPosition) { return child.get(groupPosition).get(childPosition); } @Override public long getChildId(int groupPosition, int childPosition) { return childPosition; } @Override public View getChildView(int groupPosition, int childPosition, boolean isLastChild, View convertView, ViewGroup parent) { ViewHolder holder; if (convertView == null) { convertView = inflater.inflate(R.layout.list_item, null); holder = new ViewHolder(); holder.name = (TextView) convertView.findViewById(R.id.name); holder.number = (TextView) convertView .findViewById(R.id.number); holder.date = (TextView) convertView.findViewById(R.id.date); holder.count = (TextView) convertView.findViewById(R.id.count); holder.type = (ImageView) convertView.findViewById(R.id.type); holder.open = (ImageView) convertView.findViewById(R.id.open); convertView.setTag(holder); } else { holder = (ViewHolder) convertView.getTag(); } convertView.setBackgroundColor(Color.rgb(54, 54, 54)); ContentValues cv = child.get(groupPosition).get(childPosition); String name = cv.getAsString("name"); String number = cv.getAsString("number"); String date = cv.getAsString("date"); int type = cv.getAsInteger("type"); holder.name.setText(name); holder.number.setText(number); holder.date.setText(date); holder.count.setVisibility(View.GONE); setTypeImg(holder.type, type); holder.open.setVisibility(View.GONE); return convertView; } @Override public int getChildrenCount(int groupPosition) { return child.get(groupPosition).size(); } @Override public Object getGroup(int groupPosition) { return group.get(groupPosition); } @Override public int getGroupCount() { return group.size(); } @Override public long getGroupId(int groupPosition) { return groupPosition; } @Override public View getGroupView(int groupPosition, boolean isExpanded, View convertView, ViewGroup parent) { ViewHolder holder; if (convertView == null) { convertView = inflater.inflate(R.layout.list_item, null); holder = new ViewHolder(); holder.name = (TextView) convertView.findViewById(R.id.name); holder.number = (TextView) convertView .findViewById(R.id.number); holder.date = (TextView) convertView.findViewById(R.id.date); holder.count = (TextView) convertView.findViewById(R.id.count); holder.type = (ImageView) convertView.findViewById(R.id.type); holder.open = (ImageView) convertView.findViewById(R.id.open); convertView.setTag(holder); } else { holder = (ViewHolder) convertView.getTag(); } ContentValues cv = (ContentValues) getGroup(groupPosition); String name = cv.getAsString("name"); String number = cv.getAsString("number"); String date = cv.getAsString("date"); holder.name.setText(name); holder.number.setText(number); holder.date.setText(date); holder.type.setVisibility(View.GONE); holder.count.setText("(" + getChildrenCount(groupPosition) + ")"); if (isExpanded) { holder.open.setImageResource(R.drawable.min); } else { holder.open.setImageResource(R.drawable.max); } return convertView; } @Override public boolean hasStableIds() { return false; } @Override public boolean isChildSelectable(int groupPosition, int childPosition) { return true; } } private class ViewHolder { TextView name; TextView number; TextView date; TextView count; ImageView type; ImageView open; } private void setTypeImg(ImageView imageView, int type) { switch (type) { case INCOMING_TYPE: imageView.setImageResource(R.drawable.call_in); break; case OUTGOING_TYPE: imageView.setImageResource(R.drawable.call_out); break; case MISSED_TYPE: imageView.setImageResource(R.drawable.call_miss); break; } } private class MyAsyncQueryHandler extends AsyncQueryHandler { public MyAsyncQueryHandler(ContentResolver cr) { super(cr); } @Override protected void onQueryComplete(int token, Object cookie, Cursor cursor) { if (cursor != null && cursor.getCount() > 0) { cursor.moveToFirst(); for (int i = 0; i < cursor.getCount(); i++) { cursor.moveToPosition(i); String name = cursor.getString(cursor .getColumnIndex("name")); String number = cursor.getString(cursor .getColumnIndex("number")); int type = cursor.getInt(cursor.getColumnIndex("type")); long date = cursor.getLong(cursor.getColumnIndex("date")); ContentValues cv = new ContentValues(); cv.put("name", name); cv.put("number", number); cv.put("type", type); cv.put("date", formatTimeStampString(CollLogActivity.this, date)); addGroupItem(cv); } } if (group.size() > 0) { setAdapter(); } } } private void addGroupItem(ContentValues cv) { String number = cv.getAsString("number"); if (!currentNumber.equals(number)) { group.add(cv); addChildItem(number); currentNumber = number; } } private void addChildItem(String number) { ArrayList<ContentValues> list = new ArrayList<ContentValues>(); Cursor cursor = getContentResolver().query(uri, projection, "number=" + number, null, null); if (cursor != null && cursor.getCount() > 0) { cursor.moveToFirst(); for (int i = 0; i < cursor.getCount(); i++) { cursor.moveToPosition(i); String name = cursor.getString(cursor.getColumnIndex("name")); int type = cursor.getInt(cursor.getColumnIndex("type")); long date = cursor.getLong(cursor.getColumnIndex("date")); ContentValues cv = new ContentValues(); cv.put("name", name); cv.put("number", number); cv.put("type", type); cv.put("date", formatTimeStampString(CollLogActivity.this, date)); list.add(cv); } } child.add(list); } private void setAdapter() { listAdapter = new MyExpandableListAdapter(CollLogActivity.this); listView.setAdapter(listAdapter); } private void startQuery() { queryHandler.startQuery(1, null, uri, projection, null, null, "date desc"); } // 处理日期方法 private static String formatTimeStampString(Context context, long when) { Time then = new Time(); then.set(when); Time now = new Time(); now.setToNow(); int format_flags = DateUtils.FORMAT_NO_NOON_MIDNIGHT | DateUtils.FORMAT_ABBREV_ALL | DateUtils.FORMAT_CAP_AMPM; if (then.year != now.year) { format_flags |= DateUtils.FORMAT_SHOW_YEAR | DateUtils.FORMAT_SHOW_DATE; } else if (then.yearDay != now.yearDay) { format_flags |= DateUtils.FORMAT_SHOW_DATE; } else { format_flags |= DateUtils.FORMAT_SHOW_TIME; } return DateUtils.formatDateTime(context, when, format_flags); } }

更多关于Android相关内容感兴趣的读者可查看本站专题:《Android开发入门与进阶教程》、《Android布局layout技巧总结》、《Android视图View技巧总结》、《Android编程之activity操作技巧总结》、《Android操作json格式数据技巧总结》、《Android资源操作技巧汇总》及《Android控件用法总结》

希望本文所述对大家Android程序设计有所帮助。

时间: 2024-10-23 14:59:51

Android基于BaseExpandableListAdapter实现的二级列表仿通话记录功能详解的相关文章

Android基于widget组件实现物体移动/控件拖动功能示例_Android

本文实例讲述了Android基于widget组件实现物体移动/控件拖动功能.分享给大家供大家参考,具体如下: package com.sky; import android.app.Activity; import android.os.Bundle; import android.view.KeyEvent; import android.view.MotionEvent; import android.view.View; import android.view.View.OnClickLi

Android基于socket实现的简单C/S聊天通信功能_Android

本文实例讲述了Android基于socket实现的简单C/S聊天通信功能.分享给大家供大家参考,具体如下: 主要想法:在客户端上发送一条信息,在后台开辟一个线程充当服务端,收到消息就立即回馈给客户端. 第一步:创建一个继续Activity的SocketClientActity类,包为com.pku.net 编写布局文件socketclient.xml,代码如下: <?xml version="1.0" encoding="utf-8"?> <Lin

Android 模仿iPhone列表数据View刷新动画详解_Android

因为我本人很喜欢在不同的页面之间跳转时加点好玩的动画,今天无意间看到一个动画效果感觉不错,几种效果图如下:既然好玩就写在博客中,直接说就是:该效果类似于iPhone中View的切换动画效果,今天就只介绍上面展示的效果. 废话不多说,先上效果,再看代码!! 效果一: 效果二: 效果三: 效果四:(犯错的效果): 效果五(回旋效果一): 效果六(回旋效果二): 效果看完了,就来看下上面效果实现的具体代码吧, 中间会把我自己试验的.犯的错误都以注释的形式写下来的, 大家使用的时候别出错就行了!先来看下

Android基于socket实现的简单C/S聊天通信功能

本文实例讲述了Android基于socket实现的简单C/S聊天通信功能.分享给大家供大家参考,具体如下: 主要想法:在客户端上发送一条信息,在后台开辟一个线程充当服务端,收到消息就立即回馈给客户端. 第一步:创建一个继续Activity的SocketClientActity类,包为com.pku.net 编写布局文件socketclient.xml,代码如下: <?xml version="1.0" encoding="utf-8"?> <Lin

Android基于widget组件实现物体移动/控件拖动功能示例

本文实例讲述了Android基于widget组件实现物体移动/控件拖动功能.分享给大家供大家参考,具体如下: package com.sky; import android.app.Activity; import android.os.Bundle; import android.view.KeyEvent; import android.view.MotionEvent; import android.view.View; import android.view.View.OnClickLi

Android 实现仿网络直播弹幕功能详解及实例_Android

Android 网络直播弹幕                最近看好多网络电视,播放器及直播都有弹幕功能,自己周末捣鼓下并实现,以下是网上的资料,大家可以看下. 现在网络直播越来越火,网络主播也逐渐成为一种新兴职业,对于网络直播,弹幕功能是必须要有的,如下图: 首先来分析一下,这个弹幕功能是怎么实现的,首先在最下面肯定是一个游戏界面View,然后游戏界面上有弹幕View,弹幕的View必须要做成完全透明的,这样即使覆盖在游戏界面的上方也不会影响到游戏的正常观看,只有当有人发弹幕消息时,再将消息绘

基于java集合中的一些易混淆的知识点(详解)_java

(一) collection和collections 这两者均位于java.util包下,不同的是: collection是一个集合接口,有ListSet等常见的子接口,是集合框架图的第一个节点,,提供了对集合对象进行基本操作的一系列方法. 常见的方法有: boolean add(E e) 往容器中添加元素:int size() 返回collection的元素数:boolean isEmpty() 判断此容器是否为空: boolean contains(Object o) 如果此collecti

Android开发之基本控件和四种布局方式详解_Android

Android中的控件的使用方式和iOS中控件的使用方式基本相同,都是事件驱动.给控件添加事件也有接口回调和委托代理的方式.今天这篇博客就总结一下Android中常用的基本控件以及布局方式.说到布局方式Android和iOS还是区别挺大的,在iOS中有Frame绝对布局和AutoLayout相对布局.而在Android中的布局方式就比较丰富了,今天博客中会介绍四种常用的布局方式.先总结一下控件,然后再搞一搞基本方式,开发环境还是用的Mac下的Android Studio.开始今天的正题, 虽然A

Android xUtils更新到3.0后的基本使用规则详解_Android

 说实话,对于xUtils,是我最近才用到的开发框架(也是刚接触),对于其功能不得不说,简化了很多的开发步骤,可以说是非常好的开发工具,但是其最近更新到3.0也没有解决加载自定义ImageView报错的问题. xUtils简介 xUtils 包含了很多实用的android工具. xUtils 支持大文件上传,更全面的http请求协议支持(10种谓词),拥有更加灵活的ORM,更多的事件注解支持且不受混淆影响... xUitls 最低兼容android 2.2 (api level 8) 我总是喜欢