当前位置:   article > 正文

Android QQ、微信聊天消息界面设计原理与实现_android 聊天页面原理

android 聊天页面原理


Android QQ、微信聊天消息界面设计原理与实现


原理:Android平台上,典型的以腾讯的QQ、微信这些聊天消息界面通常可以采用ListView设计与实现,需要使用ListView 适配器Adapter的getItemViewType()和getViewTypeCount()。
在ListView的适配器中,每一次getView时候,首先要判断view的类型getItemViewType(),然后根据不同的类型加载不同的布局view。
至于底部发送消息的窗口,每次发送完消息,需要将ListView滚动到底部,以免输入键盘遮挡住数据而致使用户看不到刚刚发送的消息。
现在给出一个Android平台上简单的设计与实现方案。效果如图所示:



测试的主Activity MainActivity.java:

  1. package zhangphil.chat;
  2. import java.util.ArrayList;
  3. import java.util.HashMap;
  4. import android.app.Activity;
  5. import android.content.Context;
  6. import android.os.Bundle;
  7. import android.view.ContextMenu;
  8. import android.view.LayoutInflater;
  9. import android.view.View;
  10. import android.view.ViewGroup;
  11. import android.view.ContextMenu.ContextMenuInfo;
  12. import android.widget.ArrayAdapter;
  13. import android.widget.Button;
  14. import android.widget.EditText;
  15. import android.widget.ListView;
  16. import android.widget.TextView;
  17. public class MainActivity extends Activity {
  18. private final int VIEW_TYPE = 0xb01;
  19. private final int VIEW_TYPE_LEFT = -10;
  20. private final int VIEW_TYPE_RIGHT = -11;
  21. private final int MESSAGE = 0xb02;
  22. private ArrayList<HashMap<Integer, Object>> items = null;
  23. @Override
  24. protected void onCreate(Bundle savedInstanceState) {
  25. super.onCreate(savedInstanceState);
  26. setContentView(R.layout.activity_main);
  27. final ListView listView = (ListView) findViewById(android.R.id.list);
  28. items = new ArrayList<HashMap<Integer, Object>>();
  29. for (int i = 0; i < 8; i++) {
  30. if (i % 2 == 0) {
  31. HashMap<Integer, Object> map = new HashMap<Integer, Object>();
  32. map.put(VIEW_TYPE, VIEW_TYPE_LEFT);
  33. map.put(MESSAGE, "对方说的消息" + i);
  34. items.add(map);
  35. } else {
  36. HashMap<Integer, Object> map = new HashMap<Integer, Object>();
  37. map.put(VIEW_TYPE, VIEW_TYPE_RIGHT);
  38. map.put(MESSAGE, "我说的消息" + i);
  39. items.add(map);
  40. }
  41. }
  42. final MyAdapter adapter = new MyAdapter(this, -1);
  43. listView.setAdapter(adapter);
  44. final EditText msgEditText = (EditText) findViewById(R.id.msgEditText);
  45. Button button = (Button) findViewById(R.id.msgSend);
  46. button.setOnClickListener(new View.OnClickListener() {
  47. @Override
  48. public void onClick(View v) {
  49. String msg = msgEditText.getText() + "";
  50. HashMap<Integer, Object> map = new HashMap<Integer, Object>();
  51. map.put(VIEW_TYPE, VIEW_TYPE_RIGHT);
  52. map.put(MESSAGE, msg);
  53. items.add(map);
  54. adapter.notifyDataSetChanged();
  55. // 发送后清空输入框内容
  56. msgEditText.setText(null);
  57. // 输入框发送消息后将ListView滚动到最底部
  58. listView.setSelection(ListView.FOCUS_DOWN);
  59. }
  60. });
  61. }
  62. private class MyAdapter extends ArrayAdapter {
  63. private LayoutInflater layoutInflater;
  64. public MyAdapter(Context context, int resource) {
  65. super(context, resource);
  66. layoutInflater = LayoutInflater.from(context);
  67. }
  68. @Override
  69. public View getView(int pos, View convertView, ViewGroup parent) {
  70. int type = getItemViewType(pos);
  71. String msg = getItem(pos);
  72. switch (type) {
  73. case VIEW_TYPE_LEFT:
  74. convertView = layoutInflater.inflate(R.layout.left, null);
  75. TextView textLeft = (TextView) convertView.findViewById(R.id.textView);
  76. textLeft.setText(msg);
  77. break;
  78. case VIEW_TYPE_RIGHT:
  79. convertView = layoutInflater.inflate(R.layout.right, null);
  80. TextView textRight = (TextView) convertView.findViewById(R.id.textView);
  81. textRight.setText(msg);
  82. break;
  83. }
  84. return convertView;
  85. }
  86. @Override
  87. public String getItem(int pos) {
  88. String s = items.get(pos).get(MESSAGE) + "";
  89. return s;
  90. }
  91. @Override
  92. public int getCount() {
  93. return items.size();
  94. }
  95. @Override
  96. public int getItemViewType(int pos) {
  97. int type = (Integer) items.get(pos).get(VIEW_TYPE);
  98. return type;
  99. }
  100. @Override
  101. public int getViewTypeCount() {
  102. return 2;
  103. }
  104. }
  105. }


MainActivity.java需要的布局文件activity_main.xml:

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="match_parent"
  4. android:layout_height="match_parent" >
  5. <ListView
  6. android:id="@android:id/list"
  7. android:layout_width="match_parent"
  8. android:layout_height="wrap_content"
  9. android:layout_above="@+id/commentLinearLayout"
  10. android:layout_alignParentTop="true"
  11. android:divider="@android:color/transparent"
  12. android:dividerHeight="15dip"
  13. android:scrollbars="none" />
  14. <LinearLayout
  15. android:id="@+id/commentLinearLayout"
  16. android:layout_width="match_parent"
  17. android:layout_height="wrap_content"
  18. android:layout_alignParentBottom="true"
  19. android:background="#e0e0e0"
  20. android:orientation="horizontal" >
  21. <EditText
  22. android:id="@+id/msgEditText"
  23. android:layout_width="0dip"
  24. android:layout_height="wrap_content"
  25. android:layout_weight="8"
  26. android:hint="发送消息" />
  27. <Button
  28. android:id="@+id/msgSend"
  29. style="?android:attr/buttonStyleSmall"
  30. android:layout_width="0dip"
  31. android:layout_height="wrap_content"
  32. android:layout_weight="2"
  33. android:text="发送" />
  34. </LinearLayout>
  35. </RelativeLayout>


ListView适配器Adapter在每一次getView时候,首先判断view的type,然后根据不同的view type加载不同的布局view。


left.xml表示是对方说的消息在ListView界面的左边:

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="match_parent"
  4. android:layout_height="match_parent"
  5. android:orientation="horizontal" >
  6. <ImageView
  7. android:id="@+id/imageView"
  8. android:layout_width="wrap_content"
  9. android:layout_height="wrap_content"
  10. android:layout_alignParentLeft="true"
  11. android:layout_centerVertical="true"
  12. android:singleLine="false"
  13. android:src="@drawable/ic_launcher" />
  14. <TextView
  15. android:id="@+id/textView"
  16. android:layout_width="wrap_content"
  17. android:layout_height="wrap_content"
  18. android:layout_centerVertical="true"
  19. android:layout_toRightOf="@+id/imageView"
  20. android:background="#ff5252"
  21. android:text="left" />
  22. </RelativeLayout>


right.xml表示是自己说的消息,在消息聊天界面的右边:

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="match_parent"
  4. android:layout_height="match_parent"
  5. android:orientation="horizontal" >
  6. <ImageView
  7. android:id="@+id/imageView"
  8. android:layout_width="wrap_content"
  9. android:layout_height="wrap_content"
  10. android:layout_alignParentRight="true"
  11. android:layout_centerVertical="true"
  12. android:src="@drawable/ic_launcher" />
  13. <TextView
  14. android:id="@+id/textView"
  15. android:layout_width="wrap_content"
  16. android:layout_height="wrap_content"
  17. android:layout_centerVertical="true"
  18. android:layout_toLeftOf="@+id/imageView"
  19. android:background="#2196f3"
  20. android:singleLine="false"
  21. android:text="right" />
  22. </RelativeLayout>



另外,QQ、微信这些聊天消息界面的消息背景是一个气泡,这个气泡其实是一个.9.png图片,将这个气泡的.9.png作为TextView的背景衬图衬上去即可,QQ、微信的聊天气泡.9.png不是本文要着重探讨的内容,在次不再展开叙述。


附参考文章:
《Android ListView Adapter的getItemViewType和getViewTypeCount多种布局》文章链接地址:http://blog.csdn.net/zhangphil/article/details/46984367

《Android基于PinnedSectionListView实现联系人通讯录》文章链接地址:http://blog.csdn.net/zhangphil/article/details/47271741

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/你好赵伟/article/detail/565869
推荐阅读
相关标签
  

闽ICP备14008679号