ScrollView中巢狀GridView,Listview的辦法

yangxi_001發表於2013-10-16
按照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()時丟擲異常。
  最後,建議大家還是少用這樣的設計,畢竟這種方式是不標準與不規範的。

  原文連結:http://www.eyeandroid.com/thread-13230-1-55.html

相關文章