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程序设计有所帮助。

(0)

相关推荐

  • Android利用Intent读取和更新通讯录

    一.简介 本节演示如何在安卓系统中通过用户配置文件(user profile)读取和更新该手机的所有联系人信息,以及如何导航到用户配置文件中的这些联系人. 二.基本概念  1.什么是 User Profile 用户配置文件(user profile)保存的是机主信息以及该手机中所有联系人的信息. 假定手机所有者的名字为"Mao mao yu",那么,user profile保存的就是"Mao mao yu"的通讯录(即机主所有联系人的姓名.电话.邮箱.--等信息).

  • Android通讯录开发之删除功能的实现方法

    无论是Android开发或者是其他移动平台的开发,ListView肯定是一个大咖,那么对ListView的操作肯定是不会少的,上一篇博客介绍了如何实现全选和反选的功能,本篇博客介绍删除功能,删除列表中的项无谓就是及时刷新列表,这又跟UI线程扯上关系了,还是那句话,数据的更新通知一定要在UI线程上做,不然会出现各种错误,比如出现adapter数据源改变,但没有及时收到通知的情况.在执行遍历删除的时候,最好不要每删一个就直接通知,下面是我的实现方法,将需要删除的contact保存到一个List然后通

  • Android获取手机通话记录的方法

    Android如何获取手机通话记录,本文为大家揭晓. 获取手机通话记录流程: 1. 获取ContentResolver; ContentResolver resolver = getContentResolver(); 2.resolver.query(*); 需要传入通话记录的URI:CallLog.Calls.CONTENT_URI 3.对查询得到的Cursor进行数据获取. 主要代码如下: MainActivity.java package com.noonecode.contentres

  • Android获取手机通讯录、sim卡联系人及调用拨号界面方法

    android获取手机通讯录联系人信息 复制代码 代码如下: private void getPhoneContacts() {        ContentResolver resolver = this.getContentResolver();                // 获取手机联系人       Cursor phoneCursor = resolver.query(Phone.CONTENT_URI,                  new String[] { Phone

  • Android通话记录备份实现代码

    (一) 前言 Android默认提供了联系人备份到sd卡的功能(代码在com.android.vcard包里面),我们可以把联系人导出成.vcf文件存在sd卡中:如果换手机了,我们又可以把联系人从sd卡文件中导入进来.那么,通话记录我们也能不能做出类似的功能呢?答案是肯定的! (二) 导出通话记录 既然是备份通话记录,那就肯定包括导出和导入的功能,这里我们先讲导出通话记录. 1. 根据通话记录导出的规范,导出的文件一般以.vcl后缀结尾,中间的内容是 复制代码 代码如下: BEGIN:VCALL

  • Android实现仿通讯录侧边栏滑动SiderBar效果代码

    本文实例讲述了Android实现仿通讯录侧边栏滑动SiderBar效果代码.分享给大家供大家参考,具体如下: 之前看到某些应用的侧边栏做得不错,想想自己也弄一个出来,现在分享出来,当然里面还有不足的地方,请大家多多包涵. 先上图: 具体实现的代码如下: package com.freesonfish.listview_index; import android.content.Context; import android.graphics.Canvas; import android.grap

  • Android开发实现删除联系人通话记录的方法

    本文实例讲述了Android开发实现删除联系人通话记录的方法.分享给大家供大家参考,具体如下: 1. 负责存放呼叫记录的内容提供者源码在 ContactsProvider 项目下: 源码路径: com/Android/providers/contacts/CallLogProvider.Java 使用到的数据库在: /data/data/com.android.providers.contacts/databases/contacts2.db 表名:calls 呼叫记录有三种类型: 来电:Cal

  • Android实现通讯录效果——获取手机号码和姓名

    首先给大家展示下运行效果图: 由于通讯录在手机里是以数据库贮存的 所以我们可以通过一个方法 context.getContentResolver().query(Phone.CONTENT_URI, null, null, null, null); 来获得通讯录 ,这个方法返回一个游标的数据类型,通过moveToNext()方法来获取所有的手机号码信息 当然读取手机通讯录需要权限 在adnroidManifest文件中声明即可 由于我也实现了打电话的功能 所以也要声明权限 <uses-permi

  • Android破解微信获取聊天记录和通讯录信息(静态方式)

    一.猜想数据存放路径 微信现在是老少皆宜,大街小巷都在使用,已经替代了传统的短信聊天方式了,只要涉及到聊天就肯定有隐私消息,那么本文就来讲解如何获取微信的聊天记录以及通讯录信息. 首先我们在没有网络的时候,打开微信同样可以查看聊天记录,说明微信会把聊天记录保存到本地,那么这么多信息肯定会保存在数据库中,所以我们可以去查看微信的databases目录看看内容: 可惜的是,我们在这个里面并没有发现一些有用的数据,所以这时候就了解到了微信因为把重要信息的数据库存在其他目录下面,我们可以直接把微信的整个

  • Android跳转到通讯录获取用户名称和手机号码的实现思路

    效果图如下所示: 先给大家说下实现android 跳转到通讯录的实现思路: 1.点击跳转到通讯录界面 2.获取通讯录姓名和手机号码 3.回调显示姓名和手机号码 1首先是跳转到通讯录界面 Uri uri = Uri.parse("content://contacts/people"); Intent intent = new Intent(Intent.ACTION_PICK, uri); startActivityForResult(intent, 0); 通过设置通讯录url跳转,可

  • Android手机联系人快速索引(手机通讯录)

    最近需要实现一个手机通讯录的快速索引功能.根据姓名首字母快速索引功能.下面是一个手机联系人快速索引的效果,总体来说代码不算难,拼音转换的地方略有复杂.下面上源码:源码中有注释. 下面是效果图: MainActivity: import java.util.ArrayList; import java.util.Collections; import java.util.List; import android.app.Activity; import android.os.Bundle; imp

随机推荐