ScrollView嵌套 ListView、RecyclerView、GridView 、WebView 出现的高度显示问题

ScrollView-Nested-Problems点击打开链接

解决Android中出现ScrollView嵌套 ListView、RecyclerView、GridView 、WebView出现的高度问题。

开篇语:最近开始想写一些技术总结了,一方面分享给其他同学,另一方面也作为自己的技术积累。 今天我分享的是日常遇到的问题,ScrollView组件里面嵌套GridView、WebView、ListView等本身具有滑动的组件时,所引发的高度显示不全的问题。

对于GridView、WebView、ListView这三类组件来说,大家都知道通常情况下这三类组件本身是具有滑动特性的,其实际占用的高度也只是屏幕内显示的高度,当嵌套在ScrollView组件里时就造成了冲突。 所以解决的思路可以从其onMeasure方法入手,onMeasure方法是重写自定义View中用到的一个非常重要的方法,onMeasure方法的作用就是计算出自定义View的宽度和高度,这个计算的过程参照父布局给出的大小,以及自己特点算出结果。onMeasure方法是在父视图计算子视图大小时被调用的,其中的细节就不在此详细讲述。

@Override 

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {  

  int mExpandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2, MeasureSpec.AT_MOST);       super.onMeasure(widthMeasureSpec, mExpandSpec);

}

上面方法的2个参数,来自于父视图发过来给子视图的限制条件,这涉及到一个知识点MeauseSpec这个类.

一.MeasureSpec的构成

MeasureSpec代表一个32位的int值,前俩位代表SpecMode,后30位代表SpecSize.其中:SpecMode代表测量的模式,SpecSize值在某种测量模式下的规格大小。

共有三种测量模式:

1.EXACTLY: 父容器已经检测出子View所需要的精确大小,这个时候view的大小即为SpecSize的大小,他对应于布局参数中的MATCH_PARENT,或者精确大小值;

2.AT_MOST: 父容器指定了一个大小,即SpecSize,子view的大小不能超过这个SpecSize的大小;

3.UNSPECIFIED: 父容器对子View的大小没有任何要求,子View想多大都可以。

二.如何创建MeasureSpec MeasureSpec内部提供了创建MeasureSpec的方法:

public static int makeMeasureSpec(int size, int mode) {

        if (sUseBrokenMakeMeasureSpec) {

            return size + mode;

        } else {

            return (size & ~MODE_MASK) | (mode & MODE_MASK);

        }

    }

private static final int MODE_SHIFT = 30;

private static final int MODE_MASK = 0x3 << MODE_SHIFT; 二进制 1100....00 32位

public static final int UNSPECIFIED = 0 << MODE_SHIFT;   二进制 0000....00 32位

public static final int EXACTLY     = 1 << MODE_SHIFT;   二进制 0100....00 32位

public static final int AT_MOST = 2 << MODE_SHIFT; 二进制 1000....00 32位

MeasureSpec代表一个32位的int值,前俩位代表SpecMode,后30位代表SpecSize.通过巧妙的位运算,即可通过MeasureSpec来得到SpecSize,SpecMode.

public static int getMode(int measureSpec) {

        return (measureSpec & MODE_MASK); 

      }   public static int getSize(int measureSpec) {

        return (measureSpec & ~MODE_MASK);

    }

对于RecyclerView来说,重写onMeasure()方法就不管用了。

1.一种解决办法是在RecyclerView的外部套上一层RelativeLayout

<RelativeLayout

      android:layout_width="match_parent"

      android:layout_height="wrap_content"

      android:descendantFocusability="blocksDescendants">

android:descendantFocusability属性的值有三种:

beforeDescendants:viewgroup会优先其子类控件而获取到焦点

blocksDescendants:viewgroup会覆盖子类控件而直接获得焦点

afterDescendants:viewgroup只有当其子类控件不需要获取焦点时才获取焦点

但是这个方案recyclerView时有卡顿的问题 原因还是滑动冲突的问题,可以重写LinearLayoutManager,设置让其不可滑动,外部滑动靠ScrollView,这样就解决了滑动时卡顿的问题

代码如下: public class ScrollLinearLayoutManager extends LinearLayoutManager { private boolean isScrollEnabled = true;

    public ScrollLinearLayoutManager(Context context) {

        super(context);

    }

    public ScrollLinearLayoutManager(Context context, int orientation, boolean reverseLayout) {

        super(context, orientation, reverseLayout);

    }

    public ScrollLinearLayoutManager(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {

        super(context, attrs, defStyleAttr, defStyleRes);

    }

    public void setScrollEnabled(boolean flag) {

        this.isScrollEnabled = flag;

    }

    @Override

    public boolean canScrollVertically() {

        return isScrollEnabled && super.canScrollVertically();

    }

}

简单使用: ScrollLinearLayoutManager scrollLinearLayoutManager = new ScrollLinearLayoutManager(this);

scrollLinearLayoutManager.setScrollEnabled(false);

mRecyclerView.setLayoutManager(scrollLinearLayoutManager);

2.完美方案是这样的:首先在xml布局中将你的ScrollView替换成android.support.v4.widget.NestedScrollView,并在java代码中设置recyclerView.setNestedScrollingEnabled(false);

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容