NestedScrollView 嵌套 ListView 實現(xiàn)滑動折疊效果

引言

最近,在做公司一個design折疊效果的時候遇到個問題朴译,就是我們本身app的方法數(shù)太多了井佑,dex分包技術(shù)還沒搞定。不得不盡量縮減一些不必要的包眠寿、類躬翁。當(dāng)我們引入RecyclerView的時候,恰好是壓死駱駝的最后一根稻草盯拱,故不得不采用其他方案來代替RecyclerView 和 CollapsingToolbarLayout實現(xiàn)的折疊效果盒发。本文試著采用 NestedScrollView 嵌套 ListView的方法來實現(xiàn)折疊效果。具體結(jié)果如圖所示:

布局文件

如下所示:

<android.support.design.widget.CoordinatorLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <android.support.design.widget.AppBarLayout
        android:layout_width="match_parent"
        android:layout_height="270dp"
        android:fitsSystemWindows="true">

        <android.support.design.widget.CollapsingToolbarLayout
            android:id="@+id/collapsing_toolbar_layout"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:background="#F03843"
            app:contentScrim="#F03843"
            app:layout_scrollFlags="scroll|exitUntilCollapsed">

            <TextView
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                app:layout_collapseMode="parallax"
                app:layout_collapseParallaxMultiplier="0.7"/>
            <android.support.v7.widget.Toolbar
                android:id="@+id/toolbar"
                android:layout_width="match_parent"
                android:layout_height="?attr/actionBarSize"
                android:minHeight="?attr/actionBarSize"
                app:contentInsetEnd="0dp"
                app:contentInsetStart="0dp"
                app:layout_collapseMode="pin">

                <RelativeLayout
                    android:layout_width="match_parent"
                    android:layout_height="match_parent">

                    <ImageButton
                        android:id="@+id/toolbar_import_navigation"
                        android:layout_width="?attr/actionBarSize"
                        android:layout_height="?attr/actionBarSize"
                        android:layout_alignParentLeft="true"
                        android:layout_alignParentStart="true"
                        android:layout_centerVertical="true"
                        android:src="@drawable/icon_toolbar_navigation"/>
                    <TextView
                        android:id="@+id/toolbar_title"
                        android:layout_width="wrap_content"
                        android:layout_height="wrap_content"
                        android:layout_centerInParent="true"
                        android:gravity="center"
                        android:lineSpacingMultiplier="1.5"
                        android:text="@string/select_category"
                        android:textColor="@color/dark_gray"
                        android:textSize="@dimen/text_40px"/>
                    <TextView
                        android:id="@+id/toolbar_import"
                        android:layout_width="wrap_content"
                        android:layout_height="wrap_content"
                        android:layout_alignParentEnd="true"
                        android:layout_alignParentRight="true"
                        android:layout_centerVertical="true"
                        android:layout_marginRight="16dp"
                        android:lineSpacingMultiplier="1.5"
                        android:text="@string/vocabulary_import"
                        android:textColor="@color/dark_gray"
                        android:textSize="@dimen/text_32px"/>
                </RelativeLayout>
            </android.support.v7.widget.Toolbar>
            <android.support.design.widget.TabLayout
                android:id="@+id/tabs"
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                app:tabGravity="fill" />
        </android.support.design.widget.CollapsingToolbarLayout>
    </android.support.design.widget.AppBarLayout>

    <android.support.v4.widget.NestedScrollView
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:fillViewport="true"
        android:layout_marginTop="2dp"
        android:id="@+id/vocabulary_nested_scroll"
        app:layout_behavior="@string/appbar_scrolling_view_behavior">
        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:id="@+id/parent_list_view"
            android:orientation="vertical">
            <NoScrollListView
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                android:id="@+id/list" />
        </LinearLayout>
    </android.support.v4.widget.NestedScrollView>
</android.support.design.widget.CoordinatorLayout>

上述布局需要注意的是:

  • 必須采用CoordinatorLayout作為外層包裹狡逢,至于原因是由于使用Behavior需要宁舰。
  • AppBarLayout做一個兼容。
  • android:fitsSystemWindows奢浑,經(jīng)本人代碼測試蛮艰,表示CollapsingToolbarLayout的上邊界是否擴展到statusbar,這里如果想使用透明的statusbar殷费,這里更新下應(yīng)該該布局的根布局的該屬性為true印荔,不過這里兼容性還是存在問題的低葫,特別是android 4.4版本的折疊布局的透明statusbar,不知道改為有沒更好的方式仍律。
  • app:contentScrim 表示CollapsingToolbarLayout折疊以后嘿悬,toolbar的顏色。
  • app:layout_scrollFlags="scroll|exitUntilCollapsed" 表示CoordinatorLayout的依賴元素滾動的時候水泉,進行折疊善涨。
    scroll - 想滾動就必須設(shè)置這個。 enterAlways - 實現(xiàn)quick return效果, 當(dāng)向下移動時草则,立即顯示View(比如Toolbar). exitUntilCollapsed - 向上滾動時收縮View钢拧,但可以固定Toolbar一直在上面。 enterAlwaysCollapsed - 當(dāng)你的View已經(jīng)設(shè)置minHeight屬性又使用此標志時炕横,你的View只能以最小高度進入源内,只有當(dāng)滾動視圖到達頂部時才擴大到完整高度。(參見: http://android.jobbole.com/82193/
  • layout_collapseMode份殿,pin表示不動膜钓,parallax視差效果
  • 將 app:layout_behavior="@string/appbar_scrolling_view_behavior"指定給NestedScrollView,即當(dāng)該控件滑動的時候卿嘲,其他CollapsingToolbarLayout內(nèi)的子view做相應(yīng)的改變?nèi)纾ㄒ暡睿┗騪in(不變)颂斜。

解決NestedScrollView嵌套listView問題。

   貌似網(wǎng)上一抓一大把拾枣,本文采用其中之一方案沃疮,復(fù)寫listView。
public class NoScrollListView extends ListView {
    public NoScrollListView(Context context) {
        super(context);
    }

    public NoScrollListView(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public NoScrollListView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
    }

    @Override
    public void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2, MeasureSpec.AT_MOST);
        super.onMeasure(widthMeasureSpec, expandSpec);
    }
}

解決NestedScrollView嵌套的listView滾動時無響應(yīng)bug梅肤。

private void adjustParentView() { //兼容NestedscrollView
    int actionBarHeight = 0;
    TypedValue tv = new TypedValue();
    if (getTheme().resolveAttribute(android.R.attr.actionBarSize, tv, true)) {
        actionBarHeight = TypedValue.complexToDimensionPixelSize(tv.data,getResources().getDisplayMetrics());
    }
    Rect outRect = new Rect();  //狀態(tài)欄高度
    getWindow().getDecorView().getWindowVisibleDisplayFrame(outRect);
    mParentView.setMinimumHeight(DeviceUtils.screenHeight() - actionBarHeight - outRect.top);
}

注:直接使用 RecyclerView更省事省力司蔬。

題外話:直接使用RecyclerView折疊動畫不平滑問題

在stackoverflow上找到如下解決方案,大致是由于google官方留的Behavior坑姨蝴。

public final class FlingBehavior extends AppBarLayout.Behavior {
    private static final int TOP_CHILD_FLING_THRESHOLD = 3;
    private boolean isPositive;

    public FlingBehavior() {
    }

    public FlingBehavior(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    @Override
    public boolean onNestedFling(CoordinatorLayout coordinatorLayout, AppBarLayout child, View target, float velocityX, float velocityY, boolean consumed) {
        if (velocityY > 0 && !isPositive || velocityY < 0 && isPositive) {
            velocityY = velocityY * -1;
        }
        if (target instanceof RecyclerView && velocityY < 0) {
            final RecyclerView recyclerView = (RecyclerView) target;
            final View firstChild = recyclerView.getChildAt(0);
            final int childAdapterPosition = recyclerView.getChildAdapterPosition(firstChild);
            consumed = childAdapterPosition > TOP_CHILD_FLING_THRESHOLD;
        }
        return super.onNestedFling(coordinatorLayout, child, target, velocityX, velocityY, consumed);
    }

    @Override
    public void onNestedPreScroll(CoordinatorLayout coordinatorLayout, AppBarLayout child, View target, int dx, int dy, int[] consumed) {
        super.onNestedPreScroll(coordinatorLayout, child, target, dx, dy, consumed);
        isPositive = dy > 0;
    }
}

最后在AppBarLayout如下添加代碼:

 <android.support.design.widget.AppBarLayout
        android:layout_width="match_parent"
        android:layout_height="210dp"
        android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"
        app:layout_behavior="com.youdao.vocabulary.widget.FlingBehavior">

題外話2

java.lang.IllegalStateException: View can not be anchored to the the parent CoordinatorLayout
當(dāng)我們給一個view設(shè)置app:layout_anchor時葱她,在最新的23.2.0包會出現(xiàn)問題,解決辦法是用android:layout_gravity="bottom|end" 替換似扔。

題外話3

23.2.0中在CoordinatorLayout中使用Toolbar 吨些,toolbar無法與頂部對齊,即頂部存在垂直間距炒辉。解決方法在AppBarLayout添加 android:fitsSystemWindows="true"

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末豪墅,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子黔寇,更是在濱河造成了極大的恐慌偶器,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,548評論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異屏轰,居然都是意外死亡颊郎,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,497評論 3 399
  • 文/潘曉璐 我一進店門霎苗,熙熙樓的掌柜王于貴愁眉苦臉地迎上來姆吭,“玉大人,你說我怎么就攤上這事唁盏∧诶辏” “怎么了?”我有些...
    開封第一講書人閱讀 167,990評論 0 360
  • 文/不壞的土叔 我叫張陵厘擂,是天一觀的道長昆淡。 經(jīng)常有香客問我,道長刽严,這世上最難降的妖魔是什么昂灵? 我笑而不...
    開封第一講書人閱讀 59,618評論 1 296
  • 正文 為了忘掉前任,我火速辦了婚禮舞萄,結(jié)果婚禮上倔既,老公的妹妹穿的比我還像新娘。我一直安慰自己鹏氧,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 68,618評論 6 397
  • 文/花漫 我一把揭開白布佩谣。 她就那樣靜靜地躺著把还,像睡著了一般。 火紅的嫁衣襯著肌膚如雪茸俭。 梳的紋絲不亂的頭發(fā)上吊履,一...
    開封第一講書人閱讀 52,246評論 1 308
  • 那天,我揣著相機與錄音调鬓,去河邊找鬼艇炎。 笑死,一個胖子當(dāng)著我的面吹牛腾窝,可吹牛的內(nèi)容都是我干的缀踪。 我是一名探鬼主播,決...
    沈念sama閱讀 40,819評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼虹脯,長吁一口氣:“原來是場噩夢啊……” “哼驴娃!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起循集,我...
    開封第一講書人閱讀 39,725評論 0 276
  • 序言:老撾萬榮一對情侶失蹤唇敞,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體疆柔,經(jīng)...
    沈念sama閱讀 46,268評論 1 320
  • 正文 獨居荒郊野嶺守林人離奇死亡咒精,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,356評論 3 340
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了旷档。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片模叙。...
    茶點故事閱讀 40,488評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖彬犯,靈堂內(nèi)的尸體忽然破棺而出向楼,到底是詐尸還是另有隱情,我是刑警寧澤谐区,帶...
    沈念sama閱讀 36,181評論 5 350
  • 正文 年R本政府宣布湖蜕,位于F島的核電站,受9級特大地震影響宋列,放射性物質(zhì)發(fā)生泄漏昭抒。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,862評論 3 333
  • 文/蒙蒙 一炼杖、第九天 我趴在偏房一處隱蔽的房頂上張望灭返。 院中可真熱鬧,春花似錦坤邪、人聲如沸熙含。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,331評論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽怎静。三九已至,卻和暖如春黔衡,著一層夾襖步出監(jiān)牢的瞬間蚓聘,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,445評論 1 272
  • 我被黑心中介騙來泰國打工盟劫, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留夜牡,地道東北人。 一個月前我還...
    沈念sama閱讀 48,897評論 3 376
  • 正文 我出身青樓侣签,卻偏偏與公主長得像塘装,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子影所,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,500評論 2 359

推薦閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,280評論 25 707
  • 最新項目中用到了些Material效果氢哮,在此對自己的學(xué)習(xí)做個小結(jié)。 首先養(yǎng)成良好的學(xué)習(xí)習(xí)慣-----看源碼: Co...
    風(fēng)少俠閱讀 4,864評論 5 37
  • 昨夜平行世界的我 和此間的你 一樣遙遠 沒有一分鐘擁有的 一千另一個自己 和那一千另一個你 一樣干涸 我丟下了你 ...
    靳非閱讀 284評論 0 0
  • 由暴走人員被追尾致死傷引發(fā)的思考 時間永遠定格在2017年7月8日早上一瞬間型檀,對于臨沂市來說是個難忘的一天...
    我的另一半是你閱讀 239評論 0 1
  • 在文章開始之前,大家可以先思考一個問題:你選擇從事你目前職業(yè)的原因是什么裂七?是迫于無奈皆看?是因為它熱門高薪?是感興趣背零?...
    Neil彭彭閱讀 3,237評論 5 29