ListView实现聊天列表之处理不同数据项

通常我们用惯的ListView每一项的布局都是相同的,只是控件所绑定的数据不同。但单单只是如此并不能满足我们某些特殊需求,比如我们常见的QQ、微信的聊天列表,除了有左右之分外,内容更是有很大区别,有文字、语音、图片、视频等等,他们真的是ListView可以实现的吗?答案是肯定的,只要我们做一下类型区别即可。

实现效果如下所示:

大家不要在意布局,这里为了方便就随意了。大家可以看到,这里有两种布局,一种头像在左,一种头像在右,虽然这是一种简单的情况,但我们只需要了解其中的原理,再复杂的情况都可以迎刃而解。
我们只要将每一种布局划为一种类型进行区分,根据我们所区分的类型在我们自定义的Adapter中加载不同布局即可,代码如下所示:

if (bean.getType() == 1) {
      convertView = mInflater.inflate(R.layout.item_chat_left, null);
      holder.mIvAvatar = (ImageView) convertView.findViewById(R.id.iv_chatLeftItem_avatar);
      holder.mTvContent = (TextView) convertView.findViewById(R.id.tv_chatLeftItem_content);
  } else if (bean.getType() == 2) {
      convertView = mInflater.inflate(R.layout.item_chat_right, null);
      holder.mIvAvatar = (ImageView) convertView.findViewById(R.id.iv_chatRightItem_avatar);
      holder.mTvContent = (TextView) convertView.findViewById(R.id.tv_chatRightItem_content);
  }

是不是发现很简单,最后,我再将所有代码一同贴出来。

item_chat_left.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  android:layout_width="match_parent"
  android:layout_height="wrap_content"
  android:padding="6dp" > 

  <ImageView
    android:id="@+id/iv_chatLeftItem_avatar"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:background="#dd00f0"
    android:src="@drawable/ic_launcher" /> 

  <TextView
    android:id="@+id/tv_chatLeftItem_content"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:textSize="16sp"
    android:padding="16dp"
    android:layout_toRightOf="@id/iv_chatLeftItem_avatar"
    android:background="@drawable/qfav_list_bubble_nor"
    android:layout_marginRight="64dp"
    android:gravity="center_vertical" /> 

</RelativeLayout>

item_chat_right.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  android:layout_width="match_parent"
  android:layout_height="wrap_content"
  android:padding="6dp" > 

  <ImageView
    android:id="@+id/iv_chatRightItem_avatar"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:layout_alignParentRight="true"
    android:background="#ddf000"
    android:src="@drawable/ic_launcher" /> 

  <TextView
    android:id="@+id/tv_chatRightItem_content"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:textSize="16sp"
    android:padding="16dp"
    android:layout_toLeftOf="@id/iv_chatRightItem_avatar"
    android:background="@drawable/skin_aio_user_bubble_pressed"
    android:layout_marginLeft="64dp"
    android:gravity="center_vertical" /> 

</RelativeLayout>

activity_chat.xml

<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"
  tools:context="${relativePackage}.${activityClass}" > 

  <ListView
    android:id="@+id/lv_content"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:scrollbars="none"
    android:divider="@null"
    android:listSelector="@android:color/transparent" >
  </ListView> 

</RelativeLayout> 

ChatItemBean.java

import android.graphics.Bitmap; 

public class ChatItemBean {
  private int type;
  private String content;
  private Bitmap avatar;
  public int getType() {
    return type;
  }
  public void setType(int type) {
    this.type = type;
  }
  public String getContent() {
    return content;
  }
  public void setContent(String content) {
    this.content = content;
  }
  public Bitmap getAvatar() {
    return avatar;
  }
  public void setAvatar(Bitmap avatar) {
    this.avatar = avatar;
  }
}

ChatLVAdapter.java

import java.util.List; 

import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.ImageView;
import android.widget.TextView; 

public class ChatLVAdapter extends BaseAdapter { 

  private List<ChatItemBean> mDatas;
  private LayoutInflater mInflater; 

  public ChatLVAdapter(Context context, List<ChatItemBean> datas) {
    this.mInflater = LayoutInflater.from(context);
    this.mDatas = datas;
  } 

  @Override
  public int getCount() {
    return mDatas.size();
  } 

  @Override
  public Object getItem(int position) {
    return mDatas.get(position);
  } 

  @Override
  public long getItemId(int position) {
    return position;
  } 

  @Override
  public View getView(int position, View convertView, ViewGroup parent) {
    ViewHolder holder = null;
    ChatItemBean bean = mDatas.get(position);
    if (convertView == null) {
      holder = new ViewHolder();
      if (bean.getType() == 1) {
        convertView = mInflater.inflate(R.layout.item_chat_left, null);
        holder.mIvAvatar = (ImageView) convertView.findViewById(R.id.iv_chatLeftItem_avatar);
        holder.mTvContent = (TextView) convertView.findViewById(R.id.tv_chatLeftItem_content);
      } else if (bean.getType() == 2) {
        convertView = mInflater.inflate(R.layout.item_chat_right, null);
        holder.mIvAvatar = (ImageView) convertView.findViewById(R.id.iv_chatRightItem_avatar);
        holder.mTvContent = (TextView) convertView.findViewById(R.id.tv_chatRightItem_content);
      }
      convertView.setTag(holder);
    } else { // 通过tag找到缓存的布局
      holder = (ViewHolder) convertView.getTag();
    }
    holder.mIvAvatar.setImageBitmap(bean.getAvatar());
    holder.mTvContent.setText(bean.getContent()); 

    return convertView;
  } 

  public final class ViewHolder {
    public ImageView mIvAvatar;
    public TextView mTvContent;
  }

ChatActivity.java

import java.util.ArrayList;
import java.util.List; 

import android.app.Activity;
import android.graphics.BitmapFactory;
import android.os.Bundle;
import android.widget.ListView; 

public class ChatActivity extends Activity {
  private ListView mLv;
  private List<ChatItemBean> mDatas ;
  private ChatLVAdapter mAdapter; 

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_chat); 

    mLv = (ListView) findViewById(R.id.lv_content); 

    initData();
    mAdapter = new ChatLVAdapter(this, mDatas);
    mLv.setAdapter(mAdapter);
  } 

  private void initData() {
    mDatas = new ArrayList<ChatItemBean>();
    ChatItemBean chat1 = new ChatItemBean();
    chat1.setType(1);
    chat1.setContent("早啊!");
    chat1.setAvatar(BitmapFactory.decodeResource(getResources(), R.drawable.ic_item1));
    mDatas.add(chat1);
    ChatItemBean chat2 = new ChatItemBean();
    chat2.setType(2);
    chat2.setContent("早!一大早找我有啥事?");
    chat2.setAvatar(BitmapFactory.decodeResource(getResources(), R.drawable.ic_item2));
    mDatas.add(chat2);
    ChatItemBean chat3 = new ChatItemBean();
    chat3.setType(1);
    chat3.setContent("没事就不能找你谈情说爱吗?也没什么事,看你有没有在撸代码。");
    chat3.setAvatar(BitmapFactory.decodeResource(getResources(), R.drawable.ic_item1));
    mDatas.add(chat3);
    ChatItemBean chat4 = new ChatItemBean();
    chat4.setType(2);
    chat4.setContent("算了吧,别找我!我害怕。");
    chat4.setAvatar(BitmapFactory.decodeResource(getResources(), R.drawable.ic_item2));
    mDatas.add(chat4);
    ChatItemBean chat5 = new ChatItemBean();
    chat5.setType(2);
    chat5.setContent("都被代码撸惨了,我哪敢一早起来找虐。");
    chat5.setAvatar(BitmapFactory.decodeResource(getResources(), R.drawable.ic_item2));
    mDatas.add(chat5);
  }
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • android 支持的语言列表(汇总)

    Arabic, Egypt (ar_EG) -----------------------------阿拉伯语,埃及Arabic, Israel (ar_IL) -------------------------------阿拉伯语,以色列Bulgarian, Bulgaria (bg_BG) ---------------------保加利亚语,保加利亚Catalan, Spain (ca_ES) ---------------------------加泰隆语,西班牙Czech, Czech

  • Android实现三级联动下拉框 下拉列表spinner的实例代码

    主要实现办法:动态加载各级下拉值的适配器 在监听本级下拉框,当本级下拉框的选中值改变时,随之修改下级的适配器的绑定值              XML布局: 复制代码 代码如下: <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"    xmlns:tools="http://schemas.android.com/tools"    android:layout_w

  • Android uses-permission权限列表中文注释版

    android同时也限定了系统资源的使用,像网络设备,SD卡,录音设备等.如果你的应用希望去使用任何系统资源,我们必须去申请Android的权限.这就是<uses-permission>元素的作用. 一个权限通常有以下格式,用一个名字为name 的字符串去指导我们希望使用的权限. 复制代码 代码如下: <uses-permission android:name="string"/> 例如:想要获得networking APIs的使用权限,我们指定如下的元素作为

  • android二级listview列表实现代码

    今天来实现以下大众点评客户端的横向listview二级列表,先看一下样式.  这种横向的listview二级列表在手机软件上还不太常见,但是使用过平板的都应该知道,在平板上市比较常见的.可能是因为平板屏幕比较大,而且也能展现更多的内容. 下面来看一下我的实现步骤. 首先自定义一个listview,代码如下: 复制代码 代码如下: public class MyListView extends ListView implements Runnable { private float mLastDo

  • Android实现获取应用程序相关信息列表的方法

    本文所述为Androdi获取手机应用列表的方法,比如获取到Android应用的软件属性.大小和应用程序路径.应用名称等,获取所有已安装的Android应用列表,包括那些卸载了的,但没有清除数据的应用程序,同时在获取到应用信息的时候,判断是不是系统的应用程序,这是一个应用管理器所必需具备的功能. 具体实现代码如下: //AppInfoProvider.java package com.xh.ui; import java.util.ArrayList; import java.util.List;

  • Android实现带列表的地图POI周边搜索功能

    先看效果图:(以公司附近的国贸为中心点) 上面是地图,下面是地理位置列表,有的只有地理位置列表(QQ动态的位置),这是个很常见的功能.它有个专门的叫法:POI周边搜索. 实现: 这个效果实现起来其实很简单,不过需要你先阅读下地图的API,这里使用的是高德地图的Android SDK,SDK的配置这里不作讲解,文末会放一些链接供学习. 思路: 1.利用地图的定位功能,获取用户当前的位置 2.根据获得的位置信息调用POI搜索,获取位置列表 3.ListView展示位置列表 4.用户拖动地图,获取地图

  • Android通过LIstView显示文件列表的两种方法介绍

    在Android中通过ListView显示SD卡中的文件列表一共有两种方法,一是:通过继承ListActivity显示;二是:利用BaseAdapter显示.BaseAdapter是一个公共基类适配器,用于对ListView和Spinner等 一些控件提供显示数据.下面是利用BaseAdapter类来实现通过LIstView显示SD卡的步骤: 1.main.xml界面设计,如下图 复制代码 代码如下: <?xml version="1.0" encoding="utf-

  • android开发教程之使用listview显示qq联系人列表

    首先还是xml布局文件,在其中添加ListView控件: 主布局layout_main.xml 复制代码 代码如下: <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"    xmlns:tools="http://schemas.android.com/tools"    android:layout_width="match_parent"

  • Android ExpandableListView展开列表控件使用实例

    你是否觉得手机QQ上的好友列表那个控件非常棒? 不是..... 那也没关系,学多一点知识对自己也有益无害. 那么我们就开始吧. 展开型列表控件, 原名ExpandableListView 是普通的列表控件进阶版, 可以自由的把列表进行收缩, 非常的方便兼好看. 首先看看我完成的截图, 虽然界面不漂亮, 但大家可以自己去修改界面. 该控件需要一个主界面XML 一个标题界面XML及一个列表内容界面XML 首先我们来看看 mian.xml 主界面 复制代码 代码如下: //该界面非常简单, 只要一个E

  • Android用ListView显示SDCard文件列表的小例子

    复制代码 代码如下: filePath = Environment.getExternalStorageDirectory().getAbsolutePath() + "/ADASiteMaps/SigRecord";        File file=new File(filePath);        File[] files = file.listFiles(); 构造Adapter, 复制代码 代码如下: for(File mCurrentFile:files){       

随机推荐