這篇文章給大家介紹通過在Android中自定義StickinessView實(shí)現(xiàn)一個(gè)粘性滑動(dòng)效果,內(nèi)容非常詳細(xì),感興趣的小伙伴們可以參考借鑒,希望對(duì)大家能有所幫助。
成都創(chuàng)新互聯(lián)服務(wù)項(xiàng)目包括梁山網(wǎng)站建設(shè)、梁山網(wǎng)站制作、梁山網(wǎng)頁制作以及梁山網(wǎng)絡(luò)營(yíng)銷策劃等。多年來,我們專注于互聯(lián)網(wǎng)行業(yè),利用自身積累的技術(shù)優(yōu)勢(shì)、行業(yè)經(jīng)驗(yàn)、深度合作伙伴關(guān)系等,向廣大中小型企業(yè)、政府機(jī)構(gòu)等提供互聯(lián)網(wǎng)行業(yè)的解決方案,梁山網(wǎng)站推廣取得了明顯的社會(huì)效益與經(jīng)濟(jì)效益。目前,我們服務(wù)的客戶以成都為中心已經(jīng)輻射到梁山省份的部分城市,未來相信會(huì)繼續(xù)擴(kuò)大服務(wù)區(qū)域并繼續(xù)獲得客戶的支持與信任!
一、首先,要確定HeadLayout什么時(shí)候可以攔截事件,那么就要確定ListView到達(dá)頂部和底部的時(shí)機(jī)。
@Override public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) { View v = mListView.getChildAt(0); //當(dāng)firstItem的top為0的時(shí)候就認(rèn)為已經(jīng)到達(dá)ListView的頂部了 if (mListView.getChildCount() > 0 && firstVisibleItem == 0) { //滑動(dòng)到頂部 if (v.getTop() == 0) { //滑動(dòng)到頂部了 isListViewTop = true; } else { isListViewBottom = false; } }else if (mListView.getChildCount()>0&&firstVisibleItem+visibleItemCount==totalItemCount){ final View bottomChildView = mListView.getChildAt(mListView.getChildCount()-1); //當(dāng)最后一個(gè)itemView的bottom>=ListView的高度的時(shí)候,那么就認(rèn)為到達(dá)底部了 if (mListView.getHeight()>=bottomChildView.getBottom()){ isListViewBottom = true; }else { isListViewBottom = false; } }else { isListViewBottom = false; isListViewTop = false; }
原因很簡(jiǎn)單,因?yàn)閂iew的getTop和getBottom方法是相對(duì)父容器的位置,熟悉Layout方法的,想必就會(huì)很明白了。
二、知道了HeadView攔截事件的時(shí)機(jī),我們就要搞清楚在此基礎(chǔ)之上,我們到底啥時(shí)候攔擊點(diǎn)擊事件,進(jìn)行滑動(dòng)。
@Override public boolean onInterceptTouchEvent(MotionEvent ev) { switch (ev.getAction()) { case MotionEvent.ACTION_DOWN: touchY = ev.getRawY(); isIntercept = false; break; case MotionEvent.ACTION_MOVE: float distant = ev.getRawY() - touchY; if (isListViewTop) { switch (mHeadPosition) { case TOP: if (distant > 0) isIntercept = true; break; case CENTER: isIntercept = true; break; } } if (isListViewBottom){ switch (mHeadPosition) { case CENTER: isIntercept = true; break; case BOTTOM: if (distant < 0) isIntercept = true; break; } } break; case MotionEvent.ACTION_UP: isIntercept = true; break; } return isIntercept; }
跟大家講解一下onInterceptTouchEvent(MotionEvent ev),這個(gè)方法會(huì)最先調(diào)用,當(dāng)一個(gè)事件序列攔截一次后,那么這個(gè)事件的后續(xù)事件動(dòng)作就不會(huì)再調(diào)用該方法,也就是說,當(dāng)該ViewGroup決定攔截某個(gè)事件后,那么它注定要消費(fèi)后續(xù)的事件動(dòng)作。這里貼出HeadView的位置狀態(tài)
public static final int TOP = 0;//收縮狀態(tài) public static final int CENTER = 1;//中間狀態(tài) public static final int BOTTOM = 2;//展開狀態(tài)
關(guān)于細(xì)節(jié),想必大家畫個(gè)圖就可以知道了,注意一點(diǎn):在攔截事件序列的時(shí)候,一般ACTION_DOWN事件不可以被攔截,因?yàn)閿r截的話,沒得意義了,后續(xù)事件就無法控制了,不可能繼續(xù)往ChildView傳遞事件序列。
三、移動(dòng)HeadView。
@Override public boolean onTouchEvent(MotionEvent event) { switch (event.getAction()) { case MotionEvent.ACTION_DOWN: //獲取不到的 break; case MotionEvent.ACTION_MOVE: int distant = (int) (touchY - event.getRawY()); if (getScrollY() + distant-1 < MAXY && getScrollY() + distant > 0) { scrollTo(0, getScrollY() + distant); } break; case MotionEvent.ACTION_UP: if (getScrollY() == 0) mHeadPosition = BOTTOM; if (getScrollY() == MAXY) mHeadPosition = TOP; if (getScrollY() > 0 && getScrollY() < MAXY) mHeadPosition = CENTER; if (getScrollY() > MAXY / 2) { mScroll.startScroll(0, getScrollY(), 0, MAXY-getScrollY(),100); invalidate(); mHeadPosition = TOP; } if (getScrollY() < MAXY / 2) { mScroll.startScroll(0, getScrollY(),0,-getScrollY(),100); invalidate(); mHeadPosition = BOTTOM; } break; } return super.onTouchEvent(event); }
這里為了使得滑動(dòng)跟家順暢我使用了Scroller這個(gè)類,該類是專門處理彈性滑動(dòng)的工具類,先初始化構(gòu)造器,在調(diào)用startScroll()方法(其中四個(gè)參數(shù):滑動(dòng)的x,滑動(dòng)的y,滑動(dòng)x的偏移量,滑動(dòng)y的偏移量),然后刷新視圖,最后重寫computeScroll()方法,
@Override public void computeScroll() { super.computeScroll(); if (mScroll.computeScrollOffset()){ scrollTo(mScroll.getCurrX(),mScroll.getCurrY()); postInvalidate(); } }
好了,基本完成,我們還要第一時(shí)間獲取HeadView的高度,那么在onMeasure()中獲取比較好,并且只獲取一次如下
if (MAXY == -1) MAXY = mHeadSecond.getMeasuredHeight();
在onFinishInflate()方法中,該方法的執(zhí)行標(biāo)志著所有的View都已經(jīng)add完畢,這里我們進(jìn)行初始化是比較妥當(dāng)?shù)摹?/p>
@Override protected void onFinishInflate() { super.onFinishInflate(); int count = getChildCount(); //本粘性布局只支持ListView if (count == 3 && getChildAt(2) instanceof ListView) init(); }
/** * 初始化 */ private void init() { //獲得子元素 mHeadFiest = getChildAt(0); mHeadSecond = getChildAt(1); mListView = (ListView) getChildAt(2); mListView.setOnScrollListener(this); mScroll = new Scroller(getContext()); }
關(guān)于通過在Android中自定義StickinessView實(shí)現(xiàn)一個(gè)粘性滑動(dòng)效果就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,可以學(xué)到更多知識(shí)。如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到。