转载请注明出处:
http://www.goteny.com/articles/2013/11/8.html
http://www.cnblogs.com/zjjne/p/3428480.html
当同一个页面布局中的ScrollView中包含有ListView时,两个布局由于都有滑动而导致冲突,最明显的特征就是当ListView中有多个子项时,会出现显示不全的情况,只会显示一两个子项。
以前查到一个简单的解决办法setListViewHeightBasedOnChildren(ListView listView)的那个办法,是测出ListView中每一个子项视图的高度,然后再相加起来,以这个值来设定整个ListView的高度。这种方法的优点是比较简单,能解决子项视图布局比较简单且文字较少的情况;但不足之处是每次刷新ListView时都要调用这个函数来重新设定ListView的高度,重点是:若子项视图中文字过多,出现文字自动换行的时候,此时测出来的高度就不准确了,难以做到准确设置ListView的高度。
这次要介绍的方法虽说有点麻烦,但是相对来说比较治本的方法,其思想是继承并扩展线性布局LinearLayout,用LinearLayout替代ListView来实现ListView的功能和效果。
下面是效果示例:
这是修改前的冲突情况(ListView显示只能显示第一行)
这是修改后的效果
本文的实现思想和代码是在
【Android 解决ListView 和 ScrollView 共存冲突的问题http://terryblog.blog.51cto.com/1764499/373509 】
这篇文章的基础上作了少许修改和“添加子项间分隔线、子项点击事件监听”而成的。
主要思想是:
给LinearLayout添加BaseAdapter、OnItemClickListener成员,
然后在绑定布局时调用BaseAdapter的getView方法取得子项的视图View,
将此View用addView方法添加进LinearLayout中;
并设置每一个子项视图View的监听点击事件OnClickListener,
在OnClickListener的onClick方法中调用OnItemClickListener的
onItemClick方法,即在将子项的点击事件转发到LinearLayout的onItemClick事件中。
下面是用来替代ListView的LinearLayout代码,此LinearLayoutForListView作为自定义控件来替代ListView,
用法和ListView的用法一样,使用BaseAdapter来做适配器
1 package com.and.mine;
2
3 import android.content.Context;
4 import android.util.AttributeSet;
5 import android.util.Log;
6 import android.util.TypedValue;
7 import android.view.View;
8 import android.widget.AdapterView.OnItemClickListener;
9 import android.widget.BaseAdapter;
10 import android.widget.LinearLayout;
11
12 public class LinearLayoutForListView extends LinearLayout
13 {
14 private BaseAdapter adapter;
15 private OnItemClickListener onItemClickListener;
16
17
18 /**
19 * 通过 Java代码 实例化
20 * @param context
21 */
22 public LinearLayoutForListView(Context context)
23 {
24 super(context);
25 //设置LinearLayoutForListView为垂直布局,否者默认为水平布局,容易疏忽导致子项显示不全
26 LinearLayoutForListView.this.setOrientation(LinearLayout.VERTICAL);
27 }
28
29
30 /**
31 * 此构造函数可以允许我们通过 XML的方式注册 控件
32 * @param context
33 * @param attrs
34 */
35 public LinearLayoutForListView(Context context, AttributeSet attrs)
36 {
37 super(context, attrs);
38 LinearLayoutForListView.this.setOrientation(LinearLayout.VERTICAL);
39 }
40
41
42
43 /**
44 * 设置适配器
45 *
46 * @param adpater
47 */
48 public void setAdapter(BaseAdapter adpater)
49 {
50 this.adapter = adpater;
51 bindLinearLayout();
52 }
53
54 /**
55 * 获取适配器Adapter
56 *
57 * @return adapter
58 */
59 public BaseAdapter getAdpater()
60 {
61 return adapter;
62 }
63
64
65
66 /**
67 * 绑定布局:将每个子项的视图view添加进此线性布局LinearLayout中
68 */
69 public void bindLinearLayout()
70 {
71 int count = adapter.getCount();
72 for (int i = 0; i < count; i++)
73 {
74 View v = adapter.getView(i, null, null);
75
76 if (i != count - 1)
77 { //添加每项item之间的分割线
78 v = addLine(v);
79 }
80 addView(v, i);
81 }
82 setItemClickListener();
83 Log.v("countTAG", "" + count);
84 }
85
86 /**
87 * 添加每项item之间的分割线
88 *
89 * @param view
90 * @return
91 */
92 public View addLine(View view)
93 {
94 //分割线view
95 View lineView = new View(view.getContext());
96
97 // 将数据从dip(即dp)转换到px,第一参数为数据原单位(此为DIP),第二参数为要转换的数据值
98 float fPx = TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP,
99 (float) 0.5, view.getResources().getDisplayMetrics());
100 int iPx = Math.round(fPx);
101
102 LayoutParams layoutParams = new LayoutParams(
103 LinearLayout.LayoutParams.MATCH_PARENT, iPx);
104 lineView.setLayoutParams(layoutParams);
105 lineView.setBackgroundColor(view.getSolidColor());
106
107 LinearLayout ly = new LinearLayout(view.getContext());
108 ly.setOrientation(LinearLayout.VERTICAL);
109
110 ly.addView(view);
111 ly.addView(lineView);
112
113 return ly;
114 }
115
116
117 /**
118 * 设置点击子项事件监听对象
119 * @param onItemClickListener
120 */
121 public void setOnItemClickListener(OnItemClickListener onItemClickListener)
122 {
123 this.onItemClickListener = onItemClickListener;
124 setItemClickListener();
125 }
126
127 /**
128 * 获取点击子项事件监听对象
129 * @return
130 */
131 public OnItemClickListener getOnItemClickListener()
132 {
133 return onItemClickListener;
134 }
135
136
137 /**
138 * 设置子项点击事件
139 */
140 private void setItemClickListener()
141 {
142 if (adapter != null)
143 {
144 for (int i = 0; i < LinearLayoutForListView.this.getChildCount(); i++)
145 {
146 View view = LinearLayoutForListView.this.getChildAt(i);
147 if (onItemClickListener != null)
148 {
149 //设置子项点击事件
150 view.setOnClickListener(new ItemClickListener(view, i, adapter.getItemId(i)));
151 }
152 }
153 }
154 }
155
156 class ItemClickListener implements OnClickListener
157 {
158 View view;
159 int position;
160 long id;
161
162 public ItemClickListener(View view, int position, long id)
163 {
164 this.view = view;
165 this.position = position;
166 this.id = id;
167 }
168
169 @Override
170 public void onClick(View v)
171 {
172 //将子项视图的点击事件转发到整个listview的OnItemClick事件中
173 //此方法有局限性,第一个参数 AdapterView<?> parent(即当前listView的视图)没传入onItemClick()中
174 onItemClickListener.onItemClick(null, view, position, id);
175 }
176 }
177 }
在XML中使用自定义控件来调用LinearLayoutForListView
<?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="wrap_content"
android:orientation="vertical" >
<com.and.mine.LinearLayoutForListView
android:id="@+id/list"
android:layout_width="match_parent"
android:layout_height="wrap_content" >
</com.and.mine.LinearLayoutForListView>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="使用方法" />
</LinearLayout>
当然这种这种方法也是存在不足之处的,欢迎各位指正
转载请注明出处:
http://www.goteny.com/articles/2013/11/8.html
http://www.cnblogs.com/zjjne/p/3428480.html
THE END
|