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

发布时间 - 2026-01-11 02:27:46    点击率:

本文实例讲述了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程序设计有所帮助。


# Android  # BaseExpandableListAdapter  # 二级列表  # 仿通话记录  # ExpandableListView实现简单二级列表  # Android实现二级列表购物车功能  # ExpandableListView实现二级列表购物车  # 通话记录  # 进阶  # 操作技巧  # 运行环境  # 相关内容  # 感兴趣  # 给大家  # 转来转去  # 找了  # 多说  # 更多关于  # 来实现  # 所述  # 先看  # 一篇文章  # 程序设计  # 开发工具  # 实际应用  # 那篇  # 讲述了 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Laravel如何正确地在控制器和模型之间分配逻辑_Laravel代码职责分离与架构建议  html5的keygen标签为什么废弃_替代方案说明【解答】  如何在阿里云域名上完成建站全流程?  Laravel如何与Vue.js集成_Laravel + Vue前后端分离项目搭建指南  Laravel的.env文件有什么用_Laravel环境变量配置与管理详解  如何在云主机上快速搭建多站点网站?  如何实现建站之星域名转发设置?  Laravel Facade的原理是什么_深入理解Laravel门面及其工作机制  Laravel如何自定义分页视图?(Pagination示例)  Laravel如何创建和注册中间件_Laravel中间件编写与应用流程  重庆市网站制作公司,重庆招聘网站哪个好?  IOS倒计时设置UIButton标题title的抖动问题  PHP怎么接收前端传的文件路径_处理文件路径参数接收方法【汇总】  武汉网站设计制作公司,武汉有哪些比较大的同城网站或论坛,就是里面都是武汉人的?  Laravel如何集成Inertia.js与Vue/React?(安装配置)  教学论文网站制作软件有哪些,写论文用什么软件 ?  Laravel如何使用查询构建器?(Query Builder高级用法)  Laravel Eloquent:优雅地将关联模型字段扁平化到主模型中  教你用AI将一段旋律扩展成一首完整的曲子  利用python获取某年中每个月的第一天和最后一天  Laravel如何使用Seeder填充数据_Laravel模型工厂Factory批量生成测试数据【方法】  php打包exe后无法访问网络共享_共享权限设置方法【教程】  如何在云主机上快速搭建网站?  如何构建满足综合性能需求的优质建站方案?  高端企业智能建站程序:SEO优化与响应式模板定制开发  如何在阿里云香港服务器快速搭建网站?  免费视频制作网站,更新又快又好的免费电影网站?  Python制作简易注册登录系统  香港服务器部署网站为何提示未备案?  英语简历制作免费网站推荐,如何将简历翻译成英文?  如何用PHP工具快速搭建高效网站?  如何快速搭建FTP站点实现文件共享?  Laravel中Service Container是做什么的_Laravel服务容器与依赖注入核心概念解析  如何用AI一键生成爆款短视频文案?小红书AI文案写作指令【教程】  Microsoft Edge如何解决网页加载问题 Edge浏览器加载问题修复  php在windows下怎么调试_phpwindows环境调试操作说明【操作】  laravel怎么配置和使用PHP-FPM来优化性能_laravel PHP-FPM配置与性能优化方法  LinuxShell函数封装方法_脚本复用设计思路【教程】  Laravel怎么实现观察者模式Observer_Laravel模型事件监听与解耦开发【指南】  Laravel如何处理跨站请求伪造(CSRF)保护_Laravel表单安全机制与令牌校验  javascript中的数组方法有哪些_如何利用数组方法简化数据处理  微信小程序 require机制详解及实例代码  EditPlus中的正则表达式 实战(1)  iOS UIView常见属性方法小结  JS中对数组元素进行增删改移的方法总结  EditPlus中的正则表达式 实战(2)  Laravel如何使用Collections进行数据处理?(实用方法示例)  如何用AI帮你把自己的生活经历写成一个有趣的故事?  三星、SK海力士获美批准:可向中国出口芯片制造设备  米侠浏览器网页图片不显示怎么办 米侠图片加载修复