最近做界面时发现ScrollView中嵌套ListView 结果界面中最多只能显示一个item,后来多方搜索才知道:android的设计中是不提倡这种设计的。
根据官方标准,ScrollView中是不能嵌套具有滑动效果的View的,比如GridView,Listview,ExpandableListView。但是实际中又需要这种设计:外层具有
滑动效果,内部有List结构。就需要用ScrollView+ListView+Adapter;这样做的会遇到问题,也就是我们要解决的问题。
1,最多只能显示一行,原因是android官方禁止这样做,放入ScrollView中的ListView没办法计算高度。
2,放入ScrollView 的ListView没办法滑动,原因是子控件的滑动监听被ScrollView屏蔽了,如果想子控件可以滑动,只能强行截取滑动事件。
要解决这个问题通常有两种方法:
一,网上流行的,人为的计算listview的高度,
原理:Listview在设置了Adapter以后,更具ListView 的Item重新计算ListView的高度,然后通过LayoutParams把计算得到的高度设置给ListView,这样就正确了。
注意:这ListView 的Item布局必须是LinearLayout,因为只有LinearLayout 重写了onMeasure方法,其他布局会报错,(通常是空指针异常)
import android.view.View;
import android.view.ViewGroup;
import android.widget.ListAdapter;
import android.widget.ListView;
/**
* @version 2012-8-27 上午09:38:54
**/
public class Utility {
public static void setListViewHeightBasedOnChildren(ListView listView) {
// 获取ListView对应的Adapter
ListAdapter listAdapter = listView.getAdapter();
if(listAdapter == null) {
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);
}
}
第二种方法是重写ListView的构造方法(使其失去滑动性)(这种方法是我比较喜欢的,同事GridView,ExpandableListView也可以使用这种方法解决)
package com.soufun.decoration.app.view;
import android.content.Context;
import android.util.AttributeSet;
import android.widget.ListView;
public class FullListView extends ListView {
public FullListView(Context context) {
super(context);
// TODO Auto-generated constructor stub
}
public FullListView(Context context, AttributeSet attrs) {
super(context, attrs);
}
public FullListView(Context context, AttributeSet attrs,
int defStyle) {
super(context, attrs, defStyle);
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2,
MeasureSpec.AT_MOST);
super.onMeasure(widthMeasureSpec, expandSpec);
}
}