今天就跟大家聊聊有關(guān)如何在Android中嵌套滾動NestedScroll,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結(jié)了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。
創(chuàng)新互聯(lián)是專業(yè)的榮成網(wǎng)站建設(shè)公司,榮成接單;提供成都網(wǎng)站設(shè)計、成都網(wǎng)站建設(shè)、外貿(mào)網(wǎng)站建設(shè),網(wǎng)頁設(shè)計,網(wǎng)站設(shè)計,建網(wǎng)站,PHP網(wǎng)站建設(shè)等專業(yè)做網(wǎng)站服務(wù);采用PHP框架,可快速的進(jìn)行榮成網(wǎng)站開發(fā)網(wǎng)頁制作和功能擴(kuò)展;專業(yè)做搜索引擎喜愛的網(wǎng)站,專業(yè)的做網(wǎng)站團(tuán)隊,希望更多企業(yè)前來合作!
原理
NestedScroll的其實很簡單.
一般的觸摸消息的分發(fā)都是從外向內(nèi)的,由外層的ViewGroup的dispatchTouchEvent方法調(diào)用到內(nèi)層的View的dispatchTouchEvent方法.
而NestedScroll提供了一個反向的機(jī)制,內(nèi)層的view在接收到ACTION_MOVE的時候,將滾動消息先傳回給外層的ViewGroup,看外層的ViewGroup是不是需要消耗一部分的移動,然后內(nèi)層的View再去消耗剩下的移動.內(nèi)層view可以消耗剩下的滾動的一部分,如果還沒有消耗完,外層的view可以再選擇把最后剩下的滾動消耗掉.
上面的描述可能有點(diǎn)繞,可以看下面的圖來幫助理解:
具體實現(xiàn)
NestedScroll機(jī)制會涉及到四個類:
NestedScrollingChild, NestedScrollingChildHelper 和 NestedScrollingParent , NestedScrollingParentHelper
NestedScrollingChild和NestedScrollingParent是兩個接口,我們先看看他們的聲明:
public interface NestedScrollingChild { public void setNestedScrollingEnabled(boolean enabled); public boolean isNestedScrollingEnabled(); public boolean startNestedScroll(int axes); public void stopNestedScroll(); public boolean hasNestedScrollingParent(); public boolean dispatchNestedScroll(int dxConsumed, int dyConsumed, int dxUnconsumed, int dyUnconsumed, int[] offsetInWindow); public boolean dispatchNestedPreScroll(int dx, int dy, int[] consumed, int[] offsetInWindow); public boolean dispatchNestedFling(float velocityX, float velocityY, boolean consumed); public boolean dispatchNestedPreFling(float velocityX, float velocityY); } public interface NestedScrollingParent { public boolean onStartNestedScroll(View child, View target, int nestedScrollAxes); public void onNestedScrollAccepted(View child, View target, int nestedScrollAxes); public void onStopNestedScroll(View target); public void onNestedScroll(View target, int dxConsumed, int dyConsumed, int dxUnconsumed, int dyUnconsumed); public void onNestedPreScroll(View target, int dx, int dy, int[] consumed); public boolean onNestedFling(View target, float velocityX, float velocityY, boolean consumed); public boolean onNestedPreFling(View target, float velocityX, float velocityY); public int getNestedScrollAxes(); }
這里真正重要的其實是NestedScrollingParent的幾個方法,因為其他方法都能直接讓NestedScrollingChildHelper或者NestedScrollingParentHelper去代理:
onStartNestedScroll 是否接受嵌套滾動,只有它返回true,后面的其他方法才會被調(diào)用
onNestedPreScroll 在內(nèi)層view處理滾動事件前先被調(diào)用,可以讓外層view先消耗部分滾動
onNestedScroll 在內(nèi)層view將剩下的滾動消耗完之后調(diào)用,可以在這里處理最后剩下的滾動
onNestedPreFling 在內(nèi)層view的Fling事件處理之前被調(diào)用
onNestedFling 在內(nèi)層view的Fling事件處理完之后調(diào)用
我們只要讓子view和父view分別實現(xiàn)NestedScrollingChild和NestedScrollingParent接口,然后分別調(diào)用NestedScrollingChildHelper和NestedScrollingParentHelper的對應(yīng)方法去代理一些具體功能,然后在NestedScrollingChild的onTouchEvent那里根據(jù)需求調(diào)用startNestedScroll/dispatchNestedPreScroll/stopNestedScroll就能實現(xiàn)嵌套滾動了:
//NestedScrollingChild private NestedScrollingChildHelper mHelper = new NestedScrollingChildHelper(this); public boolean startNestedScroll(int axes) { return mHelper.startNestedScroll(axes); } public boolean dispatchNestedScroll(int dxConsumed, int dyConsumed, int dxUnconsumed, int dyUnconsumed, int[] offsetInWindow) { return mHelper.dispatchNestedScroll(dxConsumed, dyConsumed, dxUnconsumed, dyUnconsumed, offsetInWindow); } ...
//NestedScrollingParent private NestedScrollingParentHelper mHelper = new NestedScrollingParentHelper(this); public void onNestedScrollAccepted(View child, View target, int axes) { mHelper.onNestedScrollAccepted(child, target, axes); } public int getNestedScrollAxes() { return mHelper.getNestedScrollAxes(); } ...
但是如果你使用sdk21及以上的版本,NestedScroll機(jī)制已經(jīng)直接集成到了View中了,你只需要直接重寫View的對應(yīng)方法就好
布局
我們先看布局文件
最外層是我們自定義的NestedScrollParentView,其實它是一個LinearLayout,內(nèi)部豎直排列了三個子view:
一個由FrameLayout包裹的ImageView
一個TextView
一個RecyclerView
代碼
為了簡便起見,我們先直接用sdk22的版本用重寫View方法的方式去實現(xiàn)它.
NestedScrollParentView中有兩個方法比較重要,嵌套滾動基本上就是由這兩個方法實現(xiàn)的:
@Override public boolean onStartNestedScroll(View child, View target, int nestedScrollAxes) { return true; } @Override public void onNestedPreScroll(View target, int dx, int dy, int[] consumed) { super.onNestedPreScroll(target, dx, dy, consumed); boolean headerScrollUp = dy > 0 && getScrollY() < mHeaderHeight; boolean headerScrollDown = dy < 0 && getScrollY() > 0 && !target.canScrollVertically(-1); if (headerScrollUp || headerScrollDown) { scrollBy(0, dy); consumed[1] = dy; } }
onStartNestedScroll 這個方法如果返回true的話代表接受由內(nèi)層傳來的滾動消息,我們直接返回true就好,否則后面的消息都接受不到
onNestedPreScroll 這個方法用于消耗內(nèi)層view的一部分滾動.我們需要將消耗掉的滾動存到counsumed中讓consumed知道.例如我們這里在頂部的FrameLayout需要移動的情況下會消耗掉所有的dy,這樣內(nèi)層的view(即RecyclerView)就不會滾動了.
這里的mHeaderHeight保存的是頂部的FrameLayout的高度:
@Override protected void onSizeChanged(int w, int h, int oldw, int oldh) { super.onSizeChanged(w, h, oldw, oldh); mHeaderHeight = mHeader.getMeasuredHeight(); }
Android是一種基于Linux內(nèi)核的自由及開放源代碼的操作系統(tǒng),主要使用于移動設(shè)備,如智能手機(jī)和平板電腦,由美國Google公司和開放手機(jī)聯(lián)盟領(lǐng)導(dǎo)及開發(fā)。
看完上述內(nèi)容,你們對如何在Android中嵌套滾動NestedScroll有進(jìn)一步的了解嗎?如果還想了解更多知識或者相關(guān)內(nèi)容,請關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝大家的支持。