Android ScrollViewWithStickHeader

前言,一天在點外賣的時候,注意到餓了么列表頁的滑動效果不錯,但是覺得其中的手勢滑動還是挺復雜的垛叨,正好又碰到了在熟悉Touch事件的理解當中,所以就抽空對著餓了么的列表頁面嘗試寫寫這個效果

1.先貼一個實現的效果圖

邏輯是當外部的ScrollView沒有滑到底部的時候柜某,往上滑動的時候嗽元,是滑動外部的ScrollView,當外部的ScrollView到達底部的時候喂击,我們再網上滑还棱,就是滑動內部的列表了,另外在左右滑動的時候惭等,當左右滑動的距離大于minPageSlop的話珍手,那么就執(zhí)行左右滑動。
如下是仿餓了么的列表頁的效果圖:


仿餓了么列表頁.gif

2.引入

在項目根目錄的build.gradle文件下增加jitpack的repo地址
allprojects {
 repositories {
    jcenter()
    maven { url "https://jitpack.io" }
 }
}

在需要引入的module中引入library
dependencies {
    implementation 'com.github.WelliJohn:StickScrollView:0.0.3'
}

3.界面的布局說明

    <wellijohn.org.stickscrollview.ScrollViewWithStickHeader
        android:id="@+id/stick_scroll_view"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_weight="1">

        <LinearLayout
            android:id="@+id/ll"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:descendantFocusability="blocksDescendants"
            android:focusableInTouchMode="true"
            android:orientation="vertical">
            //這里是header部分辞做,可以隨便自定義
            </LinearLayout>

            <LinearLayout
                android:id="@+id/ll_stick_list"
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                android:orientation="vertical">

                <android.support.design.widget.TabLayout
                    android:id="@+id/order_manager_tabs"
                    android:layout_width="match_parent"
                    android:layout_height="50dp"
                    android:background="#FFFFFF"
                    tools:tabGravity="fill"
                    tools:tabMode="fixed" />

                <android.support.v4.view.ViewPager
                    android:id="@+id/vp"
                    android:layout_width="match_parent"
                    android:layout_height="wrap_content" />
            </LinearLayout>
        </LinearLayout>
    </wellijohn.org.stickscrollview.ScrollViewWithStickHeader>

比如我們看到的仿餓了么的列表頁界面琳要,我們就需要在ViewPager設置Fragment,fragment中是左右兩個列表秤茅,看下fragment的xml設置:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/ll"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="horizontal">

    <wellijohn.org.stickscrollview.ChildRecyclerView
        android:id="@+id/child_recyclerview"
        android:layout_width="0dp"
        android:layout_height="wrap_content"
        android:layout_weight="1"
        android:background="#EEEEEE" />

    <wellijohn.org.stickscrollview.ChildRecyclerView
        android:id="@+id/child_recyclerview_right"
        android:layout_width="0dp"
        android:layout_height="wrap_content"
        android:background="#FFFFFF"
        android:layout_weight="3" />
</LinearLayout>

4.注意事項

  • ScrollViewWithStickHeader內部目前支持放置ViewPager稚补,ScrollView,RecyclerView框喳,WebView
  • ScrollView课幕,RecyclerView,WebView需要對應使用ChildScrollView五垮,ChildRecyclerView乍惊,ChildWebView
  • 我們在使用的時候,需要調用mStickScrollView.setContentView(mContentView);mLLStickList就是我們需要StickHeader+列表的部分放仗,如果你沒有StickHeader的話润绎,那么直接設置列表進來也可以,總之,你想滑動到哪個位置接下來滑動就是單純下面的部分滑動莉撇,那你就把下面的View整體設置為mContentView呢蛤。剛剛那個的ContentView是id為ll_stick_list的View。
  • 另外在這里ScrollViewWithStickHeader增加autoscroll屬性棍郎,默認是關閉的其障,如果autoscroll:true的話,在我們手指放開的時候涂佃,contentView會判斷是否自動滑動到頂部還是隱藏不見励翼。


    scrollsnap.gif

5.0.0.3版本修復當有底部有操作欄的時候,界面的滾動出現錯亂的問題巡李。

當我們底部有view需要固定的時候抚笔,我們需要通過mStickScrollView.setBottomView(mViewBottom);就可以了扶认,如下所示:


帶BottomView的操作欄

6.任何控件的使用我們最好都知道它的實現方式侨拦,所以在這里簡單介紹下這款控件的設計思路(ChildScrollView,ChildRecyclerView辐宾,ChildWebView下面的都稱為子ScrollView)狱从?

  • 6.1.我們什么時候應該讓外部的ScrollView執(zhí)行滑動事件,什么時候讓子ScrollView執(zhí)行滑動叠纹。在Android中我們有一個方法getParent().requestDisallowInterceptTouchEvent(true);就是讓view獲取到對應的事件季研。
  • 6.2.既然我們知道了怎么讓view的touch事件,接下來我們就要明白在什么情況下我們應該讓父view執(zhí)行滾動事件誉察,什么時候讓子view執(zhí)行滾動事件与涡。如下,我列了表格:
父ScrollVIew 子ScrollView 手勢滑動方向 滑動事件交由哪個view控制
不在底部 頂部 向上 父ScrollView
不在底部 頂部 向下 父ScrollView
底部 不在頂部 向上 子ScrollView
底部 不在頂部 向下 子ScrollView
底部 頂部 向下 父ScrollView
底部 頂部 向上 子ScrollView

在這里當父ScrollView不在底部的時候持偏,不會出現子ScrollView不在頂部的情況驼卖,所以在這里就不分析了。

  • 6.3.分析了鸿秆,在什么情況我們應該讓子ScrollVIew還是父ScrollView捕獲滑動事件了酌畜,我們就可以在我們的子ScrollView中編寫對應的代碼處理了?
    如下面是一段ChildScrollView的onTouchEvent方法的重寫卿叽,其他的ChildRecyclerView和ChildWebView處理也是一樣的:
@Override
    public boolean onTouchEvent(MotionEvent event) {
        if (mScrollViewWithStickHeader == null) return super.onTouchEvent(event);
        int action = event.getAction();

        if (action == MotionEvent.ACTION_DOWN) {
            mLastX = event.getX();
            mLastY = event.getY();
            //首先判斷外層ScrollView是否滑動到底部
            if (mScrollViewWithStickHeader.isBottom()) {
                getParent().requestDisallowInterceptTouchEvent(true);
                return super.onTouchEvent(event);
            } else {
                //攔截事件 本身不處理
                getParent().requestDisallowInterceptTouchEvent(false);
                return false;
            }
        }
        if (action == MotionEvent.ACTION_MOVE) {
            float nowY = event.getY();
            if (!mScrollViewWithStickHeader.isBottom() && !isScrolledToTop && nowY - mLastY > 0) {
                if (Math.abs(event.getX() - mLastX) < minPageSlop) {
                    getParent().requestDisallowInterceptTouchEvent(true);
                    return super.onTouchEvent(event);
                } else {
                    getParent().requestDisallowInterceptTouchEvent(true);
                    return false;
                }
            } else if (mScrollViewWithStickHeader.isBottom() && !isScrolledToBottom && nowY - mLastY < 0) {
                if (Math.abs(event.getX() - mLastX) < minPageSlop) {
                    getParent().requestDisallowInterceptTouchEvent(true);
                    return super.onTouchEvent(event);
                } else {
                    getParent().requestDisallowInterceptTouchEvent(true);
                    return false;
                }
            } else if (mScrollViewWithStickHeader.isBottom() && !isScrolledToTop && nowY - mLastY > 0) {
                if (Math.abs(event.getX() - mLastX) < minPageSlop) {
                    getParent().requestDisallowInterceptTouchEvent(true);
                    return super.onTouchEvent(event);
                } else {
                    getParent().requestDisallowInterceptTouchEvent(true);
                    return false;
                }
            } else {
                getParent().requestDisallowInterceptTouchEvent(false);
            }
        }

        if (action == MotionEvent.ACTION_UP || action == MotionEvent.ACTION_CANCEL) {
            getParent().requestDisallowInterceptTouchEvent(false);
        }

        return super.onTouchEvent(event);
    }

這樣的話桥胞,我們就能實現固定頭部的ScrollView了。

7.github地址考婴,歡迎star或者fork

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
  • 序言:七十年代末贩虾,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子沥阱,更是在濱河造成了極大的恐慌整胃,老刑警劉巖,帶你破解...
    沈念sama閱讀 207,248評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現場離奇詭異屁使,居然都是意外死亡在岂,警方通過查閱死者的電腦和手機,發(fā)現死者居然都...
    沈念sama閱讀 88,681評論 2 381
  • 文/潘曉璐 我一進店門蛮寂,熙熙樓的掌柜王于貴愁眉苦臉地迎上來蔽午,“玉大人,你說我怎么就攤上這事酬蹋〖袄希” “怎么了?”我有些...
    開封第一講書人閱讀 153,443評論 0 344
  • 文/不壞的土叔 我叫張陵范抓,是天一觀的道長骄恶。 經常有香客問我,道長匕垫,這世上最難降的妖魔是什么僧鲁? 我笑而不...
    開封第一講書人閱讀 55,475評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮象泵,結果婚禮上寞秃,老公的妹妹穿的比我還像新娘。我一直安慰自己偶惠,他們只是感情好春寿,可當我...
    茶點故事閱讀 64,458評論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著忽孽,像睡著了一般绑改。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上兄一,一...
    開封第一講書人閱讀 49,185評論 1 284
  • 那天厘线,我揣著相機與錄音,去河邊找鬼瘾腰。 笑死皆的,一個胖子當著我的面吹牛,可吹牛的內容都是我干的蹋盆。 我是一名探鬼主播费薄,決...
    沈念sama閱讀 38,451評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼栖雾!你這毒婦竟也來了楞抡?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 37,112評論 0 261
  • 序言:老撾萬榮一對情侶失蹤析藕,失蹤者是張志新(化名)和其女友劉穎召廷,沒想到半個月后,有當地人在樹林里發(fā)現了一具尸體,經...
    沈念sama閱讀 43,609評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡竞慢,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 36,083評論 2 325
  • 正文 我和宋清朗相戀三年先紫,在試婚紗的時候發(fā)現自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片筹煮。...
    茶點故事閱讀 38,163評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡遮精,死狀恐怖,靈堂內的尸體忽然破棺而出败潦,到底是詐尸還是另有隱情本冲,我是刑警寧澤,帶...
    沈念sama閱讀 33,803評論 4 323
  • 正文 年R本政府宣布劫扒,位于F島的核電站檬洞,受9級特大地震影響,放射性物質發(fā)生泄漏沟饥。R本人自食惡果不足惜添怔,卻給世界環(huán)境...
    茶點故事閱讀 39,357評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望闷板。 院中可真熱鬧澎灸,春花似錦院塞、人聲如沸遮晚。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,357評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽县遣。三九已至,卻和暖如春汹族,著一層夾襖步出監(jiān)牢的瞬間萧求,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,590評論 1 261
  • 我被黑心中介騙來泰國打工顶瞒, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留夸政,地道東北人。 一個月前我還...
    沈念sama閱讀 45,636評論 2 355
  • 正文 我出身青樓榴徐,卻偏偏與公主長得像守问,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子坑资,可洞房花燭夜當晚...
    茶點故事閱讀 42,925評論 2 344

推薦閱讀更多精彩內容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 171,527評論 25 707
  • 可愛的卡通形象袱贮。
    金頭魚閱讀 1,055評論 0 1
  • 深深地意識到心懷感恩的重要性 當你生病時仿便,例如感冒,你才知道你是多么地念想健康時的你,但當你恢復健康后嗽仪,你往往就會...
    川哥在路上閱讀 126評論 0 0
  • 有位學茶的同學荒勇,她泡的茶總是水甜,無論什么茶闻坚,居然都能泡出甜味來枕屉,百思不得其解。 又有一位同學為我把泡凈茶味的葉底...
    冷漠天性閱讀 854評論 0 2