這篇文章的效果也是大家常見的,各種通訊應用的對話列表都是這種方式,像微信、whatsapp、易信、米聊等。我們這篇文章也權當為回憶,形成簡單的筆記。這篇文章參考了2009年Google IO中的《TurboChargeYourUI-How to make your AndroidUI fast and efficient》和2010年Google IO中的《The World of List View》。像2009年Google IO的資料還是很前沿的,那會Android開發資料很少,最重要的就是參考google發布的各種資料。
在《TurboChargeYourUI-How to make your AndroidUI fast and efficient》介紹了怎樣提高listview的性能,優化了listview的加載速度。這里的item使用的是單一布局,能夠實現view的重用和回收,那么多種布局文件的怎么辦呢,如果再使用上面的方法,view的重用會出現問題,Android使用的BaseAdapter提供了解決多種布局文件的重用方法。
1)重寫 getViewTypeCount() – 該方法返回多少個不同的布局
2)重寫 getItemViewType(int) – 根據position返回相應的Item
- /**
- * 比原來的多了getItemViewType和getViewTypeCount這兩個方法,
- *
- * */
- public class ChatAdapter extends BaseAdapter {
- public static final String KEY = "key";
- public static final String VALUE = "value";
- public static final int VALUE_TIME_TIP = 0;// 7種不同的布局
- 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<Message> myList;
- public ChatAdapter(Context context, List<Message> 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) {
- Message msg = myList.get(position);
- int type = getItemViewType(position);
- ViewHolderTime holderTime = null;
- 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_TIME_TIP:
- holderTime = new ViewHolderTime();
- convertView = mInflater.inflate(R.layout.list_item_time_tip,
- null);
- holderTime.tvTimeTip = (TextView) convertView
- .findViewById(R.id.tv_time_tip);
- holderTime.tvTimeTip.setText(msg.getValue());
- convertView.setTag(holderTime);
- break;
- // 左邊
- 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.drawable.test);
- 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.drawable.test);
- 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 {
- Log.d("baseAdapter", "Adapter_:"+(convertView == null) );
- switch (type) {
- case VALUE_TIME_TIP:
- holderTime=(ViewHolderTime)convertView.getTag();
- holderTime.tvTimeTip.setText(msg.getValue());
- break;
- 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.drawable.test);
- 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.drawable.test);
- 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) {
- Message msg = myList.get(position);
- int type = msg.getType();
- Log.e("TYPE:", "" + type);
- return type;
- }
- /**
- * 返回所有的layout的數量
- *
- * */
- @Override
- public int getViewTypeCount() {
- return 7;
- }
- class ViewHolderTime {
- private TextView tvTimeTip;// 時間
- }
- 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;// 左邊的聲音時間
- }
- }
分享兩張微信、易信的圖,你也可以做成這樣子。
注意:
使用Listview顯示多樣視圖時,用到了getItemViewType和getViewTypeCount,但是我一運行程序就會報數組越界異常,經過查資料發現,getItemViewType的值一定要從0開始,我開始設置的type類型是從1開始的,結果就悲催了,app一直崩潰,報Java.lang.ArrayIndexOutOfBoundsException。最后把type類型改成從0開始就好了,最后注意一點,getViewTypeCount返回值一定要大於等於getItemViewType的個數。