前段时间跟大家分享了ExpandableListView的使用,不知道的童鞋,可以去这里看一下:http://blog.csdn.net/weidi1989/article/details/7995552
但是我最近做那个QQ项目是遇到一个问题,如果给这个ExpandableListView添加动态从网上获取的数据呢?前面跟大家分享的时候,是用了静态的数据,很好处理。大组跟小组就类似于一个一维数组和二维数组,但我们从服务器获取的数据可能并不是这样!比如我做的这个QQ从服务器获取的就是一个List<User>,这么一个用户数组,那些分组信息都包含在每个用户之中,像这样的数据又该如何添加到适配器里面呢?好了,下面跟大家分享一下我的处理办法。先看一张效果图,有图有真相,哈哈:
下面是对应好友在数据库中的分组,自己本人默认为在第一组。QQ好像也是这么处理的
先看一下我们自定义的适配器代码,看看需要什么样的数据,然后根据需求,传递对应的数据。
/**
* 自定义ExpandableListView的适配器
*
* @author way
*
*/
public class MyExAdapter extends BaseExpandableListAdapter {
private int[] imgs = { R.drawable.icon, R.drawable.f1, R.drawable.f2,
R.drawable.f3, R.drawable.f4, R.drawable.f5, R.drawable.f6,
R.drawable.f7, R.drawable.f8, R.drawable.f9 };// 头像资源数组
private Context context;
private List<GroupFriend> group;// 传递过来的经过处理的总数据
public MyExAdapter(Context context, List<GroupFriend> group) {
super();
this.context = context;
this.group = group;
}
// 得到大组成员的view
public View getGroupView(int groupPosition, boolean isExpanded,
View convertView, ViewGroup parent) {
if (convertView == null) {
LayoutInflater inflater = LayoutInflater.from(context);
convertView = inflater.inflate(R.layout.member_listview, null);
}
TextView title = (TextView) convertView.findViewById(R.id.content_001);
title.setText(getGroup(groupPosition).toString());// 设置大组成员名称
ImageView image = (ImageView) convertView.findViewById(R.id.tubiao);// 是否展开大组的箭头图标
if (isExpanded)// 大组展开时的箭头图标
image.setBackgroundResource(R.drawable.group_unfold_arrow);
else
// 大组合并时的箭头图标
image.setBackgroundResource(R.drawable.group_fold_arrow);
return convertView;
}
// 得到大组成员的id
public long getGroupId(int groupPosition) {
return groupPosition;
}
// 得到大组成员名称
public Object getGroup(int groupPosition) {
return group.get(groupPosition).getGroupName();
}
// 得到大组成员总数
public int getGroupCount() {
return group.size();
}
// 得到小组成员的view
public View getChildView(int groupPosition, int childPosition,
boolean isLastChild, View convertView, ViewGroup parent) {
if (convertView == null) {
LayoutInflater inflater = LayoutInflater.from(context);
convertView = inflater.inflate(R.layout.item, null);
}
final TextView title = (TextView) convertView
.findViewById(R.id.name_item);// 显示用户名
final TextView title2 = (TextView) convertView
.findViewById(R.id.id_item);// 显示用户id
ImageView icon = (ImageView) convertView
.findViewById(R.id.imageView_item);// 显示用户头像,其实还可以判断是否在线,选择黑白和彩色头像,我这里未处理,没资源,呵呵
final String name = group.get(groupPosition).getChild(childPosition)
.getName();
final String id = group.get(groupPosition).getChild(childPosition)
.getId()
+ "";
final int img = group.get(groupPosition).getChild(childPosition)
.getImg();
title.setText(name);// 大标题
title2.setText(id);// 小标题
icon.setImageResource(imgs[img]);
convertView.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
// 下面是添加到最近会话列表处理
RecentChatEntity entity = new RecentChatEntity(img, 0, name,
MyDate.getDateEN(), "");
FriendListActivity.mRecentList.add(entity);
FriendListActivity.mRecentAdapter = new RecentChatAdapter(
context, FriendListActivity.mRecentList);
FriendListActivity.mRecentListView
.setAdapter(FriendListActivity.mRecentAdapter);
// 下面是切换到聊天界面处理
User u = new User();
u.setName(name);
u.setId(Integer.parseInt(id));
u.setImg(img);
Intent intent = new Intent(context, ChatActivity.class);
intent.putExtra("user", u);
context.startActivity(intent);
// Toast.makeText(Tab2.this, "开始聊天", 0).show();
}
});
return convertView;
}
// 得到小组成员id
public long getChildId(int groupPosition, int childPosition) {
return childPosition;
}
// 得到小组成员的名称
public Object getChild(int groupPosition, int childPosition) {
return group.get(groupPosition).getChild(childPosition);
}
// 得到小组成员的数量
public int getChildrenCount(int groupPosition) {
return group.get(groupPosition).getChildSize();
}
/**
* Indicates whether the child and group IDs are stable across changes to
* the underlying data. 表明大組和小组id是否稳定的更改底层数据。
*
* @return whether or not the same ID always refers to the same object
* @see Adapter#hasStableIds()
*/
public boolean hasStableIds() {
return true;
}
// 得到小组成员是否被选择
public boolean isChildSelectable(int groupPosition, int childPosition) {
return true;
}
/**
* 这个方法是我自定义的,用于下拉刷新好友的方法
*
* @param group
* 传递进来的新数据
*/
public void updata(List<GroupFriend> group) {
this.group = null;
this.group = group;
}
}
从自定义适配器可以看出,需要一个List<GroupFriend>,这实际上也是我自定义的一个对象GroupFriend数组,为了方便处理,我把大组成员以及每个大组对应的小组成员都封装到GroupFriend这个对象中,下面我们来看一下GroupFriend的这个对象代码:
/**
* 自定义的GroupFriend对象,用来封装大组名称和分配对应的数据
*
* @author way
*
*/
public class GroupFriend {
private String groupName;// 大组名称
private List<User> groupChild;// 对应大组的小组成员对象数组
public GroupFriend() {
super();
}
public GroupFriend(String groupName, List<User> groupChild) {
super();
this.groupName = groupName;
this.groupChild = groupChild;
}
public void add(User u) {// 往小组中添加用户
groupChild.add(u);
}
public void remove(User u) {// 根据用户对象移除用户
groupChild.remove(u);
}
public void remove(int index) {// 根据下标移除用户
groupChild.remove(index);
}
public int getChildSize() {// 小组的大小
return groupChild.size();
}
public User getChild(int index) {// 根据下标得到用户
return groupChild.get(index);
}
// get...set...
public String getGroupName() {
return groupName;
}
public void setGroupName(String groupName) {
this.groupName = groupName;
}
public List<User> getGroupChild() {
return groupChild;
}
public void setGroupChild(List<User> groupChild) {
this.groupChild = groupChild;
}
}
好了,适配都根据我们的需求整完了,下面我们就处理一下服务器传递过来的List<User>对象(代码出自我的QQ项目中,我这里只截取其中重要的一部分):
private List<GroupFriend> group;//需要传递给适配器的数据
private String[] groupName = { "我的好友", "我的同学", "我的家人" };// 大组成员名
/**
* 处理服务器传递过来的用户数组数据,
*
* @param list
* 从服务器获取的用户数组
*/
private void initListViewData(List<User> list) {
group = new ArrayList<GroupFriend>();// 实例化
for (int i = 0; i < groupName.length; ++i) {// 根据大组的数量,循环给各大组分配成员
List<User> child = new ArrayList<User>();// 装小组成员的list
GroupFriend groupInfo = new GroupFriend(groupName[i], child);// 我们自定义的大组成员对象
for (User u : list) {
if (u.getGroup() == i)// 判断一下是属于哪个大组
child.add(u);
}
group.add(groupInfo);// 把自定义大组成员对象放入一个list中,传递给适配器
}
}
经过这个数据加工厂后生成的List<GroupFriend>,我们就可以用来实例化适配器对象了:
// 下面是处理好友列表界面处理
myListView = (MyListView) lay2.findViewById(R.id.tab2_listView);//获取我们自定义的可以下拉刷新的ListView对象
myExAdapter = new MyExAdapter(this, group);//实例化适配器
myListView.setAdapter(myExAdapter);//为我们自定义的ListView设置适配器
myListView.setGroupIndicator(null);// 不设置大组指示器图标,因为我们自定义设置了
myListView.setDivider(null);// 设置图片可拉伸的
myListView.setFocusable(true);// 聚焦才可以下拉刷新
myListView.setonRefreshListener(new MyRefreshListener());//监听下拉刷新状态
OK,大功告成,应该处理数据还有很多其他的办法,这只是我个人的一点思路而已,仅供参考,谢谢大家。
由于上面提到了我们自定义下拉刷新的ListView,下面顺便贴出,下拉刷新后的处理代码:
/**
* 好友列表下拉刷新监听与实现,异步任务
*
* @author way
*
*/
public class MyRefreshListener implements MyListView.OnRefreshListener {
@Override
public void onRefresh() {
new AsyncTask<Void, Void, Void>() {
List<User> list;
protected Void doInBackground(Void... params) {
// 从服务器重新获取好友列表
if (GetMsgService.isStart) {
ClientOutputThread out = GetMsgService.client
.getClientOutputThread();
TranObject o = new TranObject(TranObjectType.REFRESH);
o.setFromUser(Integer.parseInt(util.getId()));
out.setMsg(o);
// 为了及时收到服务器发过来的消息,我这里直接通过监听收消息线程,获取好友列表,就不通过接收广播了
ClientInputThread in = GetMsgService.client
.getClientInputThread();
in.setMessageListener(new MessageListener() {
@Override
public void Message(TranObject msg) {
// TODO Auto-generated method stub
if (msg != null
&& msg.getType() == TranObjectType.REFRESH) {
list = (List<User>) msg.getObject();//获取到的数据
if (list.size() > 0) {
// System.out.println("Friend:" + list);
initListViewData(list);//交给写好的方法加工一下
myExAdapter.updata(group);//调用自定义更新的方法
userDB.updateUser(list);// 保存到数据库
}
}
}
});
}
return null;
}
@Override
protected void onPostExecute(Void result) {
myExAdapter.notifyDataSetChanged();//通知更新了
myListView.onRefreshComplete();//下拉刷新结束
Toast.makeText(FriendListActivity.this, "刷新成功", 0).show();
}
}.execute(null);
}
}
分享到:
相关推荐
Android BaseExpandableListAdapter 教程
NULL 博文链接:https://xieruilin.iteye.com/blog/726494
主要介绍了Android基于BaseExpandableListAdapter实现的二级列表仿通话记录功能,结合具体实例形式分析了Android实现通话记录功能的布局与功能相关操作技巧,需要的朋友可以参考下
可以向QQ好友列表一样,可以有二级列表,使用的是android控件BaseExpandableListAdapter。
在某些android开发群里,看到有些新手问怎么实现QQ好友列表,其实网上一搜挺多的。接触Android,也才一年的时间,大部分时间花在工作上(解bug。。。),界面上开发很少参与。自己维护的系统应用里,有个...
import android.widget.BaseExpandableListAdapter; import android.widget.ExpandableListAdapter; import android.widget.ExpandableListView; import android.widget.TextView; import android.widget.Toast; ...
http://blog.csdn.net/richiezhu/article/details/50906152 BaseExpandableListAdapter
Android开发模拟QQ扩展列表功能的框架,简单易懂,方便使用,比重写 BaseExpandableListAdapter更方面,更灵活可以自定义listView布局!
收集的一些关于Android的学习...Android之Adapter用法总结,Android中图片的处理,BaseExpandableListAdapter的使用,反编译android app,详解 Android 的 Activity 组件,需要的朋友可以下载查看(直接双击html文件查看即可)
android listview SQLite BaseExpandableListAdapter 图片切割 bitmap
在某些android开发群里,看到有些新手问怎么实现QQ好友列表,其实网上一搜挺多的。接触Android,也才一年的时间,大部分时间花在工作上(解bug。。。),界面上开发很少参与。自己维护的系统应用里,有个...
利用ExpandableListView实现多级目录的显示。
好友列表的一个小例子,使用BaseExpandableListAdapter,完成的资源的复用,很简单很漂亮的例子
BaseExpandableListAdapter使用方法
创建ExpandableListView并使用BaseExpandableListAdapter派生类填充它
Android 中使用ExpandableListView 实现分组 一个视图显示垂直滚动两级列表中的条目。这不同于列表视图,允许两个层次,类似于QQ的好友分组。要实现这个效果的整体思路为: 1.要给ExpandableListView 设置适配器,...
9.90、ANDROID-MARKET 使用 365 9.91、传感器 369 9.91.1、获取手机上的传感器 369 9.91.2、 371 9.92、时间类 372 * 获得日期或时间字符串 372 * num天前的日期 373 * num天后的日期 373 * 判断 thingdate 的 ...