Android高级控件(五)——如何打造一个企业级应用对话列表,以QQ,微信为例
Android高级控件(五)——如何打造一个企业级应用对话列表,以QQ,微信为例
看标题这么高大上,实际上,还是运用我么拿到listview去扩展,我们讲什么呢,就是研究一下QQ,微信的这种对话列表,我们先看一个传统的ListView是怎么样的,我们做一个通讯录吧,通讯录的组成就是一个头像,一个名字,一个电话号码,一个点击拨打的按钮,既然这样,那我们的item就出来了
call_list_item.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:gravity="center_vertical"
android:orientation="horizontal">
<ImageView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:layout_weight="1"
android:src="@mipmap/ic_launcher" />
<TextView
android:id="@+id/tv_name"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:layout_weight="1" />
<TextView
android:id="@+id/tv_number"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:layout_weight="3" />
<Button
android:id="@+id/mCall"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:layout_weight="1" />
</LinearLayout>
大家都知道,我们一个listview还是需要一个数据集,那我们就写一个java的实体类吧
ContentBean
package com.lgl.wechatlist.bean;
import android.widget.Button;
/**
* 实体类
*
* @author LGL
*/
public class ContentBean {
private String name;
private String number;
private Button mButton;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getNumber() {
return number;
}
public void setNumber(String number) {
this.number = number;
}
public Button getmButton() {
return mButton;
}
public void setmButton(Button mButton) {
this.mButton = mButton;
}
}
传统的ListView,我相信大家都知道写,直接给出Adapter
ContentAdapter
package com.lgl.wechatlist.adapter;
import android.content.Context;
import android.content.Intent;
import android.net.Uri;
import android.view.LayoutInflater;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.Button;
import android.widget.TextView;
import com.lgl.wechatlist.R;
import com.lgl.wechatlist.bean.ContentBean;
import java.util.List;
public class ContentAdapter extends BaseAdapter {
private Context mContext;
private List<ContentBean> mList;
private LayoutInflater mInflater;
private ContentBean bean;
public ContentAdapter(Context mContext, List<ContentBean> mList) {
this.mContext = mContext;
this.mList = mList;
mInflater = (LayoutInflater) mContext
.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
}
@Override
public int getCount() {
// TODO Auto-generated method stub
return mList.size();
}
@Override
public Object getItem(int position) {
// TODO Auto-generated method stub
return mList.get(position);
}
@Override
public long getItemId(int position) {
// TODO Auto-generated method stub
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHodler viewHodler;
if (convertView == null) {
viewHodler = new ViewHodler();
convertView = mInflater.inflate(R.layout.call_list_item, null);
viewHodler.tv_name = (TextView) convertView
.findViewById(R.id.tv_name);
viewHodler.tv_number = (TextView) convertView
.findViewById(R.id.tv_number);
viewHodler.mCall = (Button) convertView.findViewById(R.id.mCall);
convertView.setTag(viewHodler);
} else {
viewHodler = (ViewHodler) convertView.getTag();
}
bean = mList.get(position);
viewHodler.tv_name.setText(bean.getName());
viewHodler.tv_number.setText(bean.getNumber());
viewHodler.mCall.setText("拨打");
viewHodler.mCall.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
Call(bean.getNumber());
}
});
return convertView;
}
class ViewHodler {
private TextView tv_name;
private TextView tv_number;
private Button mCall;
}
/**
* 拨打电话
*
* @param phone
*/
public void Call(String phone) {
Intent intent = new Intent();
intent.setAction("android.intent.action.CALL");
intent.addCategory("android.intent.category.DEFAULT");
intent.setData(Uri.parse("tel:"+phone));
mContext.startActivity(intent);
}
}
当然,还需要在我们的主类中获取数据
package com.lgl.listview;
import java.util.ArrayList;
import java.util.List;
import android.app.Activity;
import android.database.Cursor;
import android.os.Bundle;
import android.provider.ContactsContract;
import android.widget.ListView;
public class MainActivity extends Activity {
private ListView mListView;
// 联系人集合
private List<String> contentName = new ArrayList<String>();
// 号码集合
private List<String> contentPhone = new ArrayList<String>();
private List<ContentBean> list = new ArrayList<ContentBean>();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
getContent();
mListView = (ListView) findViewById(R.id.mListView);
for (int i = 0; i < contentName.size(); i++) {
ContentBean bean = new ContentBean();
bean.setName(contentName.get(i));
bean.setNumber(contentPhone.get(i));
list.add(bean);
}
ContentAdapter adapter = new ContentAdapter(this, list);
mListView.setAdapter(adapter);
}
/**
* 获取联系人
*/
private void getContent() {
Cursor query = getContentResolver().query(
ContactsContract.CommonDataKinds.Phone.CONTENT_URI, null, null,
null, null);
// 创建一个对象进行储存
String name; // 联系人
String phone; // 电话号码
// 我们获取到这些信息之后遍历出来
while (query.moveToNext()) {
// 获取名字就需要Phone.DISPLAY_NAME
name = query
.getString(query
.getColumnIndex(ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME));
// 获取num字段需要Phone.NUMBER
phone = query
.getString(query
.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER));
contentName.add(name);
contentPhone.add(phone);
}
}
}
这样,一个简易的通讯录就做好了
OK,这个没什么问题,但是我们今天要讨论的,是一个对话的ListView,这在我们之前看的群英传一书中也提到过Android群英传笔记——第四章:ListView使用技巧,但是讲的不是很详细,我们这里就多讲一下了,我们要区分左右,也就是type,需要用到getViewTypeCount()和getItemViewType()方法, 根据数据源的position返回需要显示的的layout的type,那我们从易到难,先写一个简单点的,就一个简单的文字对话框,怎么实现呢?我们当然得先定义left和right的item布局了
chatlist_left.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:gravity="bottom"
android:orientation="horizontal"
android:padding="10dp">
<ImageView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginLeft="20dp"
android:src="@mipmap/ic_launcher" />
<TextView
android:id="@+id/chatlist_tv_left"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginLeft="20dp"
android:background="@drawable/pop_left"
android:gravity="center"
android:padding="20dp" />
</LinearLayout>
chatlist_right.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:gravity="right|bottom"
android:orientation="horizontal"
android:padding="10dp">
<TextView
android:id="@+id/chatlist_tv_right"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginRight="20dp"
android:background="@drawable/pop_right"
android:gravity="center"
android:padding="20dp" />
<ImageView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginRight="20dp"
android:src="@mipmap/ic_launcher" />
</LinearLayout>
OK,别忘了这是一个新的listview,所以我们还得写一个实体类
ChatListBean
package com.lgl.wechatlist.bean;
/**
* 对话实体类
*
* @author LGL
*
*/
public class ChatListBean {
// 指定方向
private int type;
// 文本
private String value;
public int getType() {
return type;
}
public void setType(int type) {
this.type = type;
}
public String getValue() {
return value;
}
public void setValue(String value) {
this.value = value;
}
}
这样就对我们的adapter进行分析了
ChatListAdapter
package com.lgl.wechatlist.adapter;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.TextView;
import com.lgl.wechatlist.R;
import com.lgl.wechatlist.bean.ChatListBean;
import java.util.List;
/**
* 对话ListView
*/
public class ChatListAdapter extends BaseAdapter {
public static final String KEY = "key";
public static final String VALUE = "value";
public static final int VALUE_LEFT_TEXT = 1;
public static final int VALUE_RIGHT_TEXT = 2;
private LayoutInflater mInflater;
private List<ChatListBean> myList;
public ChatListAdapter(Context context, List<ChatListBean> myList) {
this.myList = myList;
mInflater = (LayoutInflater) context
.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
}
@Override
public int getCount() {
return myList.size();
}
@Override
public Object getItem(int arg0) {
return myList.get(arg0);
}
@Override
public long getItemId(int arg0) {
return arg0;
}
@Override
public View getView(int position, View convertView, ViewGroup arg2) {
ChatListBean msg = myList.get(position);
int type = getItemViewType(position);
ViewHolderRightText holderRightText = null;
ViewHolderLeftText holderLeftText = null;
if (convertView == null) {
switch (type) {
// 左边
case VALUE_LEFT_TEXT:
holderLeftText = new ViewHolderLeftText();
convertView = mInflater.inflate(R.layout.chatlist_left, null);
holderLeftText.btnLeftText = (TextView) convertView
.findViewById(R.id.chatlist_tv_left);
holderLeftText.btnLeftText.setText(msg.getValue());
convertView.setTag(holderLeftText);
break;
// 右边
case VALUE_RIGHT_TEXT:
holderRightText = new ViewHolderRightText();
convertView = mInflater.inflate(R.layout.chatlist_right, null);
holderRightText.btnRightText = (TextView) convertView
.findViewById(R.id.chatlist_tv_right);
holderRightText.btnRightText.setText(msg.getValue());
convertView.setTag(holderRightText);
break;
}
} else {
switch (type) {
case VALUE_LEFT_TEXT:
holderLeftText = (ViewHolderLeftText) convertView.getTag();
holderLeftText.btnLeftText.setText(msg.getValue());
break;
case VALUE_RIGHT_TEXT:
holderRightText = (ViewHolderRightText) convertView.getTag();
holderRightText.btnRightText.setText(msg.getValue());
break;
}
}
return convertView;
}
/**
* 根据数据源的position返回需要显示的的layout的type
* <p/>
* type的值必须从0开始
*/
@Override
public int getItemViewType(int position) {
ChatListBean msg = myList.get(position);
int type = msg.getType();
return type;
}
/**
* 返回所有的layout的数量
*/
@Override
public int getViewTypeCount() {
return 7;
}
// 左边的文本
class ViewHolderLeftText {
private TextView btnLeftText;
}
// 右边的文本
class ViewHolderRightText {
private TextView btnRightText;
}
}
我们去主类中写点数据
package com.lgl.wechatlist;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.widget.Button;
import android.widget.ListView;
import com.lgl.wechatlist.adapter.ChatListAdapter;
import com.lgl.wechatlist.bean.ChatListBean;
import java.util.ArrayList;
import java.util.List;
/**
* Created by LGL on 2016/5/7.
*/
public class QQActivity extends AppCompatActivity implements View.OnClickListener{
// 会话数据源
private List<ChatListBean> ListData;
// 会话Adapter
private ChatListAdapter chatListAdapter;
// 会话列表
private ListView mListView;
// 消息
private String message;
// 按钮
private Button btn_left, btn_right;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_qq);
mListView = (ListView) findViewById(R.id.mListView);
btn_left = (Button) findViewById(R.id.btn_left);
btn_left.setOnClickListener(this);
btn_right = (Button) findViewById(R.id.btn_right);
btn_right.setOnClickListener(this);
ListData = new ArrayList<ChatListBean>();
ChatListBean chatListBean = new ChatListBean();
chatListBean.setType(ChatListAdapter.VALUE_LEFT_TEXT);
chatListBean.setValue("你好!");
ListData.add(chatListBean);
chatListAdapter = new ChatListAdapter(this, ListData);
mListView.setAdapter(chatListAdapter);
}
@Override
public void onClick(View v) {
switch (v.getId()) {
case R.id.btn_left:
listLeft("left");
break;
case R.id.btn_right:
listRight("right");
break;
}
}
/**
* 显示列表左边
*
* @param text
*/
private void listLeft(String text) {
ChatListBean left = new ChatListBean();
left.setType(ChatListAdapter.VALUE_LEFT_TEXT);
left.setValue(text);
ListData.add(left);
chatListAdapter.notifyDataSetChanged();
// 滚动到最底部
mListView.setSelection(mListView.getBottom());
}
/**
* 显示右边列表
*
* @param text
*/
private void listRight(String text) {
ChatListBean right = new ChatListBean();
right.setType(ChatListAdapter.VALUE_RIGHT_TEXT);
right.setValue(text);
ListData.add(right);
chatListAdapter.notifyDataSetChanged();
// 滚动到最底部
mListView.setSelection(mListView.getBottom());
}
}
好的运行一下
到这里,我们就对这个技巧有点熟悉了吧,既然如此,那我们即系,现在我们有6个item,分别是文字左右,图片左右,语言左右,这里就不贴出来了,要看的直接去Demo里面找哟,也没多少东西,就几个文本布局而已,我们直接来看实体类
MessageBean
package com.lgl.wechatlist.bean;
public class MessageBean {
private int type;//指定是哪种类型
private String value;//值
public int getType() {
return type;
}
public void setType(int type) {
this.type = type;
}
public String getValue() {
return value;
}
public void setValue(String value) {
this.value = value;
}
}
你一看,和上次那个没什么不同,对的,是没什么不同,只不过代码多了一点而已,这样的话,我们可以看Adapter了
package com.lgl.wechatlist.adapter;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.TextView;
import com.lgl.wechatlist.R;
import com.lgl.wechatlist.bean.MessageBean;
import java.util.List;
/**
* 比原来的多了getItemViewType和getViewTypeCount这两个方法,
*
* */
public class WechatAdapter extends BaseAdapter {
public static final String KEY = "key";
public static final String VALUE = "value";
public static final int VALUE_LEFT_TEXT = 1;
public static final int VALUE_LEFT_IMAGE = 2;
public static final int VALUE_LEFT_AUDIO = 3;
public static final int VALUE_RIGHT_TEXT = 4;
public static final int VALUE_RIGHT_IMAGE = 5;
public static final int VALUE_RIGHT_AUDIO = 6;
private LayoutInflater mInflater;
private List<MessageBean> myList;
public WechatAdapter(Context context, List<MessageBean> myList) {
this.myList = myList;
mInflater = (LayoutInflater) context
.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
}
@Override
public int getCount() {
return myList.size();
}
@Override
public Object getItem(int arg0) {
return myList.get(arg0);
}
@Override
public long getItemId(int arg0) {
return arg0;
}
@Override
public View getView(int position, View convertView, ViewGroup arg2) {
MessageBean msg = myList.get(position);
int type = getItemViewType(position);
ViewHolderRightText holderRightText = null;
ViewHolderRightImg holderRightImg = null;
ViewHolderRightAudio holderRightAudio = null;
ViewHolderLeftText holderLeftText = null;
ViewHolderLeftImg holderLeftImg = null;
ViewHolderLeftAudio holderLeftAudio = null;
if (convertView == null) {
switch (type) {
// 左边
case VALUE_LEFT_TEXT:
holderLeftText = new ViewHolderLeftText();
convertView = mInflater.inflate(R.layout.list_item_left_text,
null);
holderLeftText.ivLeftIcon = (ImageView) convertView
.findViewById(R.id.iv_icon);
holderLeftText.btnLeftText = (Button) convertView
.findViewById(R.id.btn_left_text);
holderLeftText.btnLeftText.setText(msg.getValue());
convertView.setTag(holderLeftText);
break;
case VALUE_LEFT_IMAGE:
holderLeftImg = new ViewHolderLeftImg();
convertView = mInflater.inflate(R.layout.list_item_left_iamge,
null);
holderLeftImg.ivLeftIcon = (ImageView) convertView
.findViewById(R.id.iv_icon);
holderLeftImg.ivLeftImage = (ImageView) convertView
.findViewById(R.id.iv_left_image);
holderLeftImg.ivLeftImage.setImageResource(R.mipmap.ic_launcher);
convertView.setTag(holderLeftImg);
break;
case VALUE_LEFT_AUDIO:
holderLeftAudio = new ViewHolderLeftAudio();
convertView = mInflater.inflate(R.layout.list_item_left_audio,
null);
holderLeftAudio.ivLeftIcon = (ImageView) convertView
.findViewById(R.id.iv_icon);
holderLeftAudio.btnLeftAudio = (Button) convertView
.findViewById(R.id.btn_left_audio);
holderLeftAudio.tvLeftAudioTime = (TextView) convertView
.findViewById(R.id.tv_left_audio_time);
holderLeftAudio.tvLeftAudioTime.setText(msg.getValue());
convertView.setTag(holderLeftAudio);
break;
// 右边
case VALUE_RIGHT_TEXT:
holderRightText= new ViewHolderRightText();
convertView = mInflater.inflate(R.layout.list_item_right_text,
null);
holderRightText.ivRightIcon = (ImageView) convertView
.findViewById(R.id.iv_icon);
holderRightText.btnRightText = (Button) convertView
.findViewById(R.id.btn_right_text);
holderRightText.btnRightText.setText(msg.getValue());
convertView.setTag(holderRightText);
break;
case VALUE_RIGHT_IMAGE:
holderRightImg= new ViewHolderRightImg();
convertView = mInflater.inflate(R.layout.list_item_right_iamge,
null);
holderRightImg.ivRightIcon = (ImageView) convertView
.findViewById(R.id.iv_icon);
holderRightImg.ivRightImage = (ImageView) convertView
.findViewById(R.id.iv_right_image);
holderRightImg.ivRightImage.setImageResource(R.mipmap.ic_launcher);
convertView.setTag(holderRightImg);
break;
case VALUE_RIGHT_AUDIO:
holderRightAudio=new ViewHolderRightAudio();
convertView = mInflater.inflate(R.layout.list_item_right_audio,
null);
holderRightAudio.ivRightIcon = (ImageView) convertView
.findViewById(R.id.iv_icon);
holderRightAudio.btnRightAudio = (Button) convertView
.findViewById(R.id.btn_right_audio);
holderRightAudio.tvRightAudioTime = (TextView) convertView
.findViewById(R.id.tv_right_audio_time);
holderRightAudio.tvRightAudioTime.setText(msg.getValue());
convertView.setTag(holderRightAudio);
break;
default:
break;
}
} else {
switch (type) {
case VALUE_LEFT_TEXT:
holderLeftText=(ViewHolderLeftText)convertView.getTag();
holderLeftText.btnLeftText.setText(msg.getValue());
break;
case VALUE_LEFT_IMAGE:
holderLeftImg=(ViewHolderLeftImg)convertView.getTag();
holderLeftImg.ivLeftImage.setImageResource(R.mipmap.ic_launcher);
break;
case VALUE_LEFT_AUDIO:
holderLeftAudio=(ViewHolderLeftAudio)convertView.getTag();
holderLeftAudio.tvLeftAudioTime.setText(msg.getValue());
break;
case VALUE_RIGHT_TEXT:
holderRightText=(ViewHolderRightText)convertView.getTag();
holderRightText.btnRightText.setText(msg.getValue());
break;
case VALUE_RIGHT_IMAGE:
holderRightImg=(ViewHolderRightImg)convertView.getTag();
holderRightImg.ivRightImage.setImageResource(R.mipmap.ic_launcher);
break;
case VALUE_RIGHT_AUDIO:
holderRightAudio=(ViewHolderRightAudio)convertView.getTag();
holderRightAudio.tvRightAudioTime.setText(msg.getValue());
break;
default:
break;
}
//holder = (ViewHolder) convertView.getTag();
}
return convertView;
}
/**
* 根据数据源的position返回需要显示的的layout的type
*
* type的值必须从0开始
*
* */
@Override
public int getItemViewType(int position) {
MessageBean msg = myList.get(position);
int type = msg.getType();
return type;
}
/**
* 返回所有的layout的数量
*
* */
@Override
public int getViewTypeCount() {
return 7;
}
class ViewHolderRightText {
private ImageView ivRightIcon;// 右边的头像
private Button btnRightText;// 右边的文本
}
class ViewHolderRightImg {
private ImageView ivRightIcon;// 右边的头像
private ImageView ivRightImage;// 右边的图像
}
class ViewHolderRightAudio {
private ImageView ivRightIcon;// 右边的头像
private Button btnRightAudio;// 右边的声音
private TextView tvRightAudioTime;// 右边的声音时间
}
class ViewHolderLeftText {
private ImageView ivLeftIcon;// 左边的头像
private Button btnLeftText;// 左边的文本
}
class ViewHolderLeftImg {
private ImageView ivLeftIcon;// 左边的头像
private ImageView ivLeftImage;// 左边的图像
}
class ViewHolderLeftAudio {
private ImageView ivLeftIcon;// 左边的头像
private Button btnLeftAudio;// 左边的声音
private TextView tvLeftAudioTime;// 左边的声音时间
}
}
OK,代码清晰可见
我们的主类
package com.lgl.wechatlist;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.widget.ListView;
import com.lgl.wechatlist.adapter.WechatAdapter;
import com.lgl.wechatlist.bean.MessageBean;
import java.util.ArrayList;
import java.util.List;
public class WechatActivity extends AppCompatActivity {
private ListView lvData;
private List<MessageBean> msgList;
private MessageBean msg;
private WechatAdapter adapter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_wechat);
lvData = (ListView) findViewById(R.id.lv_data);
msgList = new ArrayList<MessageBean>();
adapter = new WechatAdapter(WechatActivity.this, msgList);
lvData.setAdapter(adapter);
}
/**
* 左边文字
*
* @param v
*/
public void tv_left(View v) {
MessageBean msg = new MessageBean();
msg.setType(WechatAdapter.VALUE_LEFT_TEXT);
msg.setValue("我是文字左");
msgList.add(msg);
adapter.notifyDataSetChanged();
//滑动到底部
lvData.setSelection(lvData.getBottom());
}
/**
* 右边文字
*
* @param v
*/
public void tv_right(View v) {
MessageBean msg = new MessageBean();
msg.setType(WechatAdapter.VALUE_RIGHT_TEXT);
msg.setValue("我是文字右");
msgList.add(msg);
adapter.notifyDataSetChanged();
//滑动到底部
lvData.setSelection(lvData.getBottom());
}
/**
* 左边图片
*
* @param v
*/
public void iv_left(View v) {
MessageBean msg = new MessageBean();
msg.setType(WechatAdapter.VALUE_LEFT_IMAGE);
msg.setValue("我是图片左");
msgList.add(msg);
adapter.notifyDataSetChanged();
//滑动到底部
lvData.setSelection(lvData.getBottom());
}
/**
* 右边图片
*
* @param v
*/
public void iv_right(View v) {
msg = new MessageBean();
msg.setType(WechatAdapter.VALUE_RIGHT_IMAGE);
msg.setValue("我是图片右");
msgList.add(msg);
adapter.notifyDataSetChanged();
//滑动到底部
lvData.setSelection(lvData.getBottom());
}
/**
* 左边语音
*
* @param v
*/
public void iv_left_audio(View v) {
msg = new MessageBean();
msg.setType(WechatAdapter.VALUE_LEFT_AUDIO);
msg.setValue("16s");
msgList.add(msg);
adapter.notifyDataSetChanged();
//滑动到底部
lvData.setSelection(lvData.getBottom());
}
/**
* 右边语音
*
* @param v
*/
public void iv_right_audio(View v) {
msg = new MessageBean();
msg.setType(WechatAdapter.VALUE_RIGHT_AUDIO);
msg.setValue("6s");
msgList.add(msg);
adapter.notifyDataSetChanged();
//滑动到底部
lvData.setSelection(lvData.getBottom());
}
}
那这样的话,我们运行一下
行,我们是不是已经有了大致的了解,我Demo也理得比较顺,要是看的不是很懂可以结合Demo来
Demo下载地址:http://download.csdn.net/detail/qq_26787115/9512985
我的群,通往Android的神奇之旅 :555974449,欢迎大家进来交流技术!
Android高级控件(五)——如何打造一个企业级应用对话列表,以QQ,微信为例的更多相关文章
- Android高级控件(六)——自定义ListView高仿一个QQ可拖拽列表的实现
Android高级控件(六)--自定义ListView高仿一个QQ可拖拽列表的实现 我们做一些好友列表或者商品列表的时候,居多的需求可能就是需要列表拖拽了,而我们选择了ListView,也是因为使用L ...
- Android 高级控件(七)——RecyclerView的方方面面
Android 高级控件(七)--RecyclerView的方方面面 RecyclerView出来很长时间了,相信大家都已经比较了解了,这里我把知识梳理一下,其实你把他看成一个升级版的ListView ...
- Android高级控件(四)——VideoView 实现引导页播放视频欢迎效果,超级简单却十分的炫酷
Android高级控件(四)--VideoView 实现引导页播放视频欢迎效果,超级简单却十分的炫酷 是不是感觉QQ空间什么的每次新版本更新那炫炫的引导页就特别的激动,哈哈,其实他实现起来真的很简单很 ...
- Android高级控件(三)—— 使用Google ZXing实现二维码的扫描和生成相关功能体系
Android高级控件(三)-- 使用Google ZXing实现二维码的扫描和生成相关功能体系 摘要 现在的二维码可谓是烂大街了,到处都是二维码,什么都是二维码,扫一扫似乎已经流行到习以为常了,今天 ...
- Android高级控件(二)——SurfaceView实现GIF动画架包,播放GIF动画,自己实现功能的初体现
Android高级控件(二)--SurfaceView实现GIF动画架包,播放GIF动画,自己实现功能的初体现 写这个的原因呢,也是因为项目中用到了gif动画,虽然网上有很多的架包可以实现,不过我们还 ...
- Android高级控件(一)——ListView绑定CheckBox实现全选,增加和删除等功能
Android高级控件(一)--ListView绑定CheckBox实现全选,增加和删除等功能 这个控件还是挺复杂的,也是项目中应该算是比较常用的了,所以写了一个小Demo来讲讲,主要是自定义adap ...
- Android高级控件(三)—— 使用Google ZXing实现二维码的扫描和生成相关功能体系
Android高级控件(三)-- 使用Google ZXing实现二维码的扫描和生成相关功能体系 摘要 如今的二维码可谓是烂大街了.到处都是二维码.什么都是二维码,扫一扫似乎已经流行到习以为常了,今天 ...
- Android高级控件(一)——ListView绑定CheckBox实现全选,添加和删除等功能
Android高级控件(一)--ListView绑定CheckBox实现全选,添加和删除等功能 这个控件还是挺复杂的.也是项目中应该算是比較经常使用的了,所以写了一个小Demo来讲讲,主要是自己定义a ...
- Android高级控件--AdapterView与Adapter
在J2EE中提供过一种非常好的框架--MVC框架,实现原理:数据模型M(Model)存放数据,利用控制器C(Controller)将数据显示在视图V(View)上.在Android中有这样一种高级控件 ...
随机推荐
- JavaScript中的事件模型
JS中的事件 1.鼠标事件 onclick ondbclick onmouseover onmouseout 2.HTML事件 onload onunload onsubmit ...
- Java对象锁和类锁全面解析(多线程synchronized关键字)
最近工作有用到一些多线程的东西,之前吧,有用到synchronized同步块,不过是别人怎么用就跟着用,并没有搞清楚锁的概念.最近也是遇到一些问题,不搞清楚锁的概念,很容易碰壁,甚至有些时候自己连用没 ...
- WebService案例入门(基础篇)
[版权申明:本文系作者原创,转载请注明出处] 文章出处:http://blog.csdn.net/sdksdk0/article/details/52106690 作者:朱培 ID:sdksdk0 邮 ...
- SpriteKit物理引擎碰撞中5个重要信息
我们知道在SpriteKit物理引擎实际是基于Box2D! 在SpriteKit中当你设置好适当的碰撞参数后,通过遵守SKPhysicsContactDelegate,你可以选择实现2各碰撞回调方法: ...
- Hexo写博客
hexo配置github Git Install hexo-deployer-git. $ npm install hexo-deployer-git –save 配置_config.yml文件 de ...
- [Python]print vs sys.stdout.write
之前只是在项目中看到过,没怎么注意,正好跟对象一起看python学习手册,看到了这个部分于是来研究下. python版本 2.7.x os win7 print 一般就是执行脚本的时候,把信息直接 ...
- Spring的DataSource配置、将Hibernate配置全部写到Spring配置
DataSource可以集中管理数据库连接,减少维护工作量,使部署更简单: Spring的DataSource配置:(Spring数据源配置)这里使用dbcp,还有很多其他的如c3p0,jdbc,jn ...
- Android Framework 学习和需要学习的内容
1. 之前的研究太偏向应用层功能实现了,很多原理不了解没有深究,现在研究framework面存一些资料待有空查看. 2.Android系统的层次如下: 3.项目目录简单分析如下: 4.telphony ...
- JBOSS EAP 6 系列三 Oracle、Mysql数据源的配置(驱动)—认识模块的使用
本文介绍JBOSS EAP 6.2中Oracle数据源的配置方式.结合之前JBOSS EAP 6.2新功能,本文初识JBOSS模块申明式容器这一特性. 模块申明式容器:JBOSS EAP不再有lib的 ...
- 即时通讯软件openfire+spark+smack
所以我基本上分为三篇文章来介绍此类软件的开发: 第一篇是关于XMPP 协议是啥,IM 是啥以及一个比较有名的开源实现,该开源实现包括三个部分(Spark.Smack和Openfire): 第二篇讲如何 ...