国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 系統 > Android > 正文

Android 中ScrollView嵌套GridView,ListView的實例

2019-12-12 03:15:57
字體:
來源:轉載
供稿:網友

Android 中ScrollView嵌套GridView,ListView的實例

在Android開發中,經常有一些UI需要進行固定style的動態布局,然而由于現在的UI都喜歡把一個界面拉的很長,所以我們很多情況下需要使用ScrollView來嵌套列表控件來實現UI。這樣就導致了很多不順心的問題。

問題一:列表控件顯示不完全

原因是嵌套情況下,ScrollView不能正確的計算列表控件的高度。

有兩種解決方案

方案一

在適配器賦值完成后代碼動態計算列表的高度。這里貼出ListView的計算代碼,GridView的計算方式類似,不過需要考慮列數,下面代碼沒有加上列表控件padding的計算,如果你設置了這個屬性,需要加上計算代碼

public 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);        // 計算子項View 的寬高        listItem.measure(0, 0);        // 統計所有子項的總高度        totalHeight += listItem.getMeasuredHeight();      }      ViewGroup.LayoutParams params = listView.getLayoutParams();      params.height = totalHeight+ (listView.getDividerHeight() * (listAdapter.getCount() - 1));      // listView.getDividerHeight()獲取子項間分隔符占用的高度      // params.height最后得到整個ListView完整顯示需要的高度      listView.setLayoutParams(params);    }  

方案二

重寫列表控件的onMeasure方法,這種方案不會出現列表控件本身的滾動條,并且viewholder復用機制會失效

@Override  public void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {    int expandSpec = MeasureSpec.makeMeasureSpec(1 << 16, MeasureSpec.AT_MOST);    super.onMeasure(widthMeasureSpec, expandSpec);  }

方案一代碼多,需要多次寫,建議寫成工具類方便調用;方案二在數據量大到不能一屏顯示完的情況下會有性能問題,而且快速滑動的時候ScrollView會不停的去計算列表控件的高度。賊影響繪制性能。
兩種方案有利有弊,大家自己取舍

問題二:列表控件自動獲取焦點,導致ScrollView自動滾動到列表控件所在的位置

這個問題其實有很多種解決方案,歸結起來是兩種。

方案一

等待列表控件數據全部加載完成后(包括圖片加載)調用ScrollView.fullScroll(ScrollView.FOCUS_UP);

方法讓ScrollView滾動到頂部。這個加載完成的時間不好控制,搞得不好會有滾動動畫出現,很尷尬的事情。

方案二

重寫列表控件的如下兩個方法,使之固定返回false

@Override  public boolean isFocused() {    return false;  }  @Override  public boolean requestFocus(int direction, Rect previouslyFocusedRect) {    return false;  }

兩種方案的優缺點很明顯,喔,第二種方案的缺點我目前沒發現。如果你這么使用發現了什么坑,請留言告知

問題三 滑動沖突

這問題就更操蛋了,根據UI的不同,操蛋程度也不同,涉及到view的事件傳遞知識,很難給出所有情況的解決代碼
解決起來也離不開幾個要點,不過首先你得熟悉view的事件傳遞

你需要根據情況決定重寫列表控件與ScrollView的如下幾個方法,根據情況給方法返回不同的bool值來告訴控件是否攔截或者傳遞事件,需要哪個控件相應哪個方向的滾動事件就攔截哪個方向的事件傳遞,作為一個有追求的開發者,切記不要一通亂攔截

 @Override  public boolean onTouchEvent(MotionEvent ev) {    return super.onTouchEvent(ev);  }  @Override  public boolean onInterceptTouchEvent(MotionEvent ev) {    return super.onInterceptTouchEvent(ev);  }  @Override  public boolean dispatchTouchEvent(MotionEvent ev) {    return super.dispatchTouchEvent(ev);  }

給個小彩蛋

getParent().requestDisallowInterceptTouchEvent(boolean b);

這一句代碼可以在子控件里決定是否讓父容器獲取事件

感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 长宁区| 汉川市| 民权县| 大洼县| 灯塔市| 珠海市| 时尚| 宁明县| 新密市| 卢氏县| 墨竹工卡县| 阜新| 左云县| 沅江市| 揭西县| 锡林浩特市| 隆尧县| 高陵县| 凤阳县| 涞水县| 汪清县| 榆社县| 安吉县| 贺兰县| 渝中区| 洛川县| 汾西县| 克东县| 平江县| 翼城县| 宜兰市| 辽阳县| 女性| 剑河县| 环江| 苍梧县| 措勤县| 冕宁县| 普格县| 通江县| 延边|