最近在做界面时,想要在ScrollView中嵌入ListView ,发现listview只能显示一个item。网上搜索后发现原来这是android设计中不提倡的一种设计方式,根据android的官方标准,ScrollView中是不能嵌套具有滑动特性的View的,但是实际设计中又想要这样做:最外面的布局需要有滑动的特性,而内部有类似于List的UI结构,那么就会想到用ScrollView+ListView + Adpater的方式来实现了。
既然是违反了标准,说明这样做是有问题的,那有哪些问题呢,通过查找资料总结如下:
首先,在SrollView中嵌套ListView,ListView的显示会有问题,就是我现在遇到的,只显示一行或显示效果与预期不同,这是因为android禁止这样使用,放入ScrollView中的ListView的高度是无法计算的。
其次,嵌套中的子ListView和GridvIew是无法滑动的,因为子控件的滑动事件会被外面的ScrollView屏蔽掉,如果想让子控件可以滑动,只能强行的截取滑动的相关事件了。
如何解决这类问题呢?接下来介绍几种方式
<1>重写ListView与GridView,让其失去滑动特性:
- public class NonScrollGridView extends GridView{
- public <span style="font-family:Arial">Non</span><span style="font-family:Arial">ScrollGridView(Context context, AttributeSet attrs){</span>
- super(context, attrs);
- }
- public void onMeasure(int widthMeasureSpec, int heightMeasureSpec){
- int mExpandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2, MeasureSpec.AT_MOST);
- super.onMeasure(widthMeasureSpec, mExpandSpec);
- }
- }
- public class NonScrollListView extends ListView{
- public NonScrollListView(Context context, AttributeSet attrs){
- super(context, attrs);
- }
- public void onMeasure(int widthMeasureSpec, int heightMeasureSpec){
- int mExpandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2, MeasureSpec.AT_MOST);
- super.onMeasure(widthMeasureSpec, mExpandSpec);
- }
- }
<2>第二种方案也是网上流行的一种解决办法,人工计算子控件的尺寸,解决办法:
原理就是:
设置完ListView的Adapter后,根据ListView的子项目重新计算ListView的高度,然后把高度再作为LayoutParams设置给ListView,这样它的高度就正确了,通过人工算取控件的应有高度,再设置给ListView
注意:这个方案中子ListView的每个Item必须是LinearLayout,不能是其他的,因为其他的Layout(如RelativeLayout)没有重写onMeasure(),所以会在onMeasure()时抛出异常。
- public class Utility {
- public static void setListViewHeightBasedOnChildren(ListView listView) {
- //获取ListView对应的Adapter
- ListAdapter listAdapter = listView.getAdapter();
- if (listAdapter == null) {
- // pre-condition
- return;
- }
- int totalHeight = 0;
- for (int i = 0, len = listAdapter.getCount(); i < len; i++) { //listAdapter.getCount()返回数据项的数目
- View listItem = listAdapter.getView(i, null, listView);
- listItem.measure(0, 0); //计算子项View 的宽高
- totalHeight += listItem.getMeasuredHeight(); //统计所有子项的总高度
- }
- ViewGroup.LayoutParams params = listView.getLayoutParams();
- params.height = totalHeight + (listView.getDividerHeight() * (listAdapter.getCount() - 1));
- //listView.getDividerHeight()获取子项间分隔符占用的高度
- //params.height最后得到整个ListView完整显示需要的高度
- listView.setLayoutParams(params);
- }
- }