ScrollView 嵌套 listView、gridView 冲突的解决方案

来源:互联网 发布:vivo手机销售数据 编辑:程序博客网 时间:2024/04/20 16:44
按照android的标准,ScrollView中是不能嵌套具有滑动特性的View的,但是有时如果设计真的有这样做的需要,或者为了更方便简单的实现外观(比如在外在的大布局需要有滑动的特性,并且内部有类似于List的UI结构,那么ListView + Adpater的方式来实现里面的效果就很方便,算是违规抄近道的一种方式吧),有时就会不得不采用这种怪异的组合方式。
  先说下这种方式如果不做特殊处理时会出现的冲突和问题:
  1,在SrollView中嵌套ListView,ListView的显示会有问题,只显示一行或显示效果与预期不同,这是因为android禁止这样使用,放入ScrollView中的ListView的高度是无法计 算的。
  2,嵌套中的子ListView和GridvIew是无法滑动的,因为子控件的滑动事件会被外面的ScrollView吃掉,如果想让子控件可以滑动,只能强行的截取滑动的相关事件了。
  言归正传,嵌套的解决方案:
  1,第一种方案,也是我比较推荐的方案,就是重写ListView与GridView,让其失去滑动特性:
  html代码 

  1.     package com.perfect.xiaoao.all.ui;

  2.   import android.content.Context;
  3.   import android.util.AttributeSet;
  4.   import android.widget.GridView;
  5.   /**
  6.   * Created by IntelliJ IDEA.
  7.   * User: zhUser
  8.   * Date: 13-1-24
  9.   * Time: 下午6:53
  10.   */
  11.   public class NoScrollGridView extends GridView{
  12.   public NoScrollGridView(Context context, AttributeSet attrs){
  13.   super(context, attrs);
  14.   }
  15.   public void onMeasure(int widthMeasureSpec, int heightMeasureSpec){
  16.   int mExpandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2, 
  17. MeasureSpec.AT_MOST);
  18.   super.onMeasure(widthMeasureSpec, mExpandSpec);
  19.   }
  20.   }
  21.   package com.perfect.xiaoao.all.ui;
  22.   import android.content.Context;
  23.   import android.util.AttributeSet;
  24.   import android.widget.ListView;
  25.   /**
  26.   * Created by IntelliJ IDEA.
  27.   * User: zhUser
  28.   * Date: 13-1-24
  29.   * Time: 下午6:53
  30.   */
  31.   public class NoScrollListView extends ListView{
  32.   public NoScrollListView(Context context, AttributeSet attrs){
  33.   super(context, attrs);
  34.   }
  35.   public void onMeasure(int widthMeasureSpec, int heightMeasureSpec){
  36.   int mExpandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2, 
  37. MeasureSpec.AT_MOST);
  38.   super.onMeasure(widthMeasureSpec, mExpandSpec);
  39.   }
  40.   }
复制代码

  2,第二种方案,也是网上流行的一种解决办法,人工计算子控件的尺寸,解决办法:
  在listview.setAdapter()之后调用Utility.setListViewHeightBasedOnChilren(listview)就Okay 了。
  html代码 

  1.     public class Utility {

  2.   public static void setListViewHeightBasedOnChildren(ListView listView) 
  3. {
  4.   //获取ListView对应的Adapter
  5.   ListAdapter listAdapter = listView.getAdapter();
  6.   if (listAdapter == null) {
  7.   // pre-condition
  8.   return;
  9.   }
  10.   int totalHeight = 0;
  11.   for (int i = 0, len = listAdapter.getCount(); i < len; i++) { 
  12. //listAdapter.getCount()返回数据项的数目
  13.   View listItem = listAdapter.getView(i, null, listView);
  14.   listItem.measure(0, 0); //计算子项View 的宽高
  15.   totalHeight += listItem.getMeasuredHeight(); //统计所有子项的总高度
  16.   }
  17.   ViewGroup.LayoutParams params = listView.getLayoutParams();
  18.   params.height = totalHeight + (listView.getDividerHeight() * 
  19. (listAdapter.getCount() - 1));
  20.   //listView.getDividerHeight()获取子项间分隔符占用的高度
  21.   //params.height最后得到整个ListView完整显示需要的高度
  22.   listView.setLayoutParams(params);
  23.   }
  24.   }
复制代码

  原理就是:设置完ListView的Adapter后,根据ListView的子项目重新计算ListView的高度,然后把高度再作为LayoutParams设置给ListView,这样它的高度就正确了,通过人工算取控件的应有高度,再设置给ListView
  注意:这个方案中子ListView的每个Item必须是LinearLayout,不能是其他的,因为其他的Layout(如RelativeLayout)没有重写onMeasure(),所以会在onMeasure()时抛出异常。
  最后,建议大家还是少用这样的设计,毕竟这种方式是不标准与不规范的。
0 0