日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > Android >内容正文

Android

android 粘性view_Android自定义StickinessView粘性滑动效果

發布時間:2024/7/5 Android 27 豆豆
生活随笔 收集整理的這篇文章主要介紹了 android 粘性view_Android自定义StickinessView粘性滑动效果 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

design包的出現,Android界面發生了巨大變化,各種滑動配合的效果,下面我就粘性滑動中的一種進行自定義,效果圖如下:

大家看到效果了,這里我是繼承了LinerLayout,方便一點,若果是ViewGroup的話,也就復雜一點點。這里分為三部分:

1.head1,頂部可移動的Layout。

2.head2,固定的頭部,不會滑動除屏幕外。

3.可滑動的Layout(這里只可以是ListView,不過也可以是任何可滑動的View,只要給出Head可滑動的時機即可)

本StickinessView的難點在于,解決滑動沖突和事件的攔截處理,接下來我一一道來。

一、首先,要確定HeadLayout什么時候可以攔截事件,那么就要確定ListView到達頂部和底部的時機。

@Override

public void onScroll(AbsListView view,int firstVisibleItem,int visibleItemCount,int totalItemCount) {

View v = mListView.getChildAt(0);

//當firstItem的top為0的時候就認為已經到達ListView的頂部了

if (mListView.getChildCount() > 0 && firstVisibleItem == 0) {

//滑動到頂部

if (v.getTop() == 0) {

//滑動到頂部了

isListViewTop = true;

} else {

isListViewBottom = false;

}

}else if (mListView.getChildCount()>0&&firstVisibleItem+visibleItemCount==totalItemCount){

final View bottomChildView = mListView.getChildAt(mListView.getChildCount()-1);

//當最后一個itemView的bottom>=ListView的高度的時候,那么就認為到達底部了

if (mListView.getHeight()>=bottomChildView.getBottom()){

isListViewBottom = true;

}else {

isListViewBottom = false;

}

}else {

isListViewBottom = false;

isListViewTop = false;

}

原因很簡單,因為View的getTop和getBottom方法是相對父容器的位置,熟悉Layout方法的,想必就會很明白了。

二、知道了HeadView攔截事件的時機,我們就要搞清楚在此基礎之上,我們到底啥時候攔擊點擊事件,進行滑動。

@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),這個方法會最先調用,當一個事件序列攔截一次后,那么這個事件的后續事件動作就不會再調用該方法,也就是說,當該ViewGroup決定攔截某個事件后,那么它注定要消費后續的事件動作。這里貼出HeadView的位置狀態

public static final int TOP = 0;//收縮狀態

public static final int CENTER = 1;//中間狀態

public static final int BOTTOM = 2;//展開狀態

關于細節,想必大家畫個圖就可以知道了,注意一點:在攔截事件序列的時候,一般ACTION_DOWN事件不可以被攔截,因為攔截的話,沒得意義了,后續事件就無法控制了,不可能繼續往ChildView傳遞事件序列。

三、移動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(),MAXY-getScrollY(),100);

invalidate();

mHeadPosition = TOP;

}

if (getScrollY() < MAXY / 2) {

mScroll.startScroll(0,-getScrollY(),100);

invalidate();

mHeadPosition = BOTTOM;

}

break;

}

return super.onTouchEvent(event);

}

這里為了使得滑動跟家順暢我使用了Scroller這個類,該類是專門處理彈性滑動的工具類,先初始化構造器,在調用startScroll()方法(其中四個參數:滑動的x,滑動的y,滑動x的偏移量,滑動y的偏移量),然后刷新視圖,最后重寫computeScroll()方法,

@Override

public void computeScroll() {

super.computeScroll();

if (mScroll.computeScrollOffset()){

scrollTo(mScroll.getCurrX(),mScroll.getCurrY());

postInvalidate();

}

}

好了,基本完成,我們還要第一時間獲取HeadView的高度,那么在onMeasure()中獲取比較好,并且只獲取一次如下

if (MAXY == -1)

MAXY = mHeadSecond.getMeasuredHeight();

在onFinishInflate()方法中,該方法的執行標志著所有的View都已經add完畢,這里我們進行初始化是比較妥當的。

@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());

}

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持編程小技巧。

總結

以上是生活随笔為你收集整理的android 粘性view_Android自定义StickinessView粘性滑动效果的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

主站蜘蛛池模板: 最近中文字幕在线中文高清版 | 久草a在线 | 长河落日电视连续剧免费观看 | 琪琪免费视频 | 顶级嫩模啪啪呻吟不断好爽 | 黄色日皮视频 | 在线播放精品视频 | av福利社 | 欧美少妇一区二区 | 久久久国产精品一区二区三区 | 韩国电影一区 | 污片免费在线观看 | 男女视频在线免费观看 | 爱插视频| www.久久伊人| 国内毛片视频 | 丝袜人妻一区二区三区 | 美女十八毛片 | 国产日韩欧美中文 | 天堂在线资源网 | 久久午夜精品人妻一区二区三区 | 午夜日韩福利 | 久久精品国产99久久不卡 | 日本三级在线视频 | 999这里只有精品 | 国产精品香蕉在线 | 亚洲一区二区三区不卡视频 | 国产三级三级三级 | 中文字幕在线观看视频一区 | 国产成人精品女人久久久 | 婷婷av在线 | 亚洲天堂伊人网 | 一二三区中文字幕 | 伊人网欧美 | 日韩午夜激情电影 | 97国产精品视频 | 啪啪免费网| 一区二区播放 | 在线中出| 另类欧美亚洲 | 99re在线视频| 热99这里只有精品 | 国产成人高清 | 调教撅屁股啪调教打臀缝av | 99久久婷婷国产一区二区三区 | 亚洲欧美成人一区 | 公侵犯人妻一区二区三区 | 欧美婷婷六月丁香综合色 | 日韩av有码 | 国产chinesehd天美传媒 | 毛片小视频 | 天天草天天爽 | 日韩一区二区三区在线免费观看 | 中文无码精品一区二区三区 | 中国免费黄色片 | 亚洲精品污 | 日韩在线| 国产精品中文字幕在线观看 | 中文天堂资源在线 | 色猫咪av在线 | 日本精品中文字幕 | 91视频国产免费 | 午夜影院福利社 | 日日干天天操 | 国产毛片在线看 | 黑人巨大精品欧美黑白配亚洲 | 色婷亚洲 | 人妻在线一区 | 超碰在线| 半推半就一ⅹ99av | 三上悠亚久久 | 日免费视频 | 亚洲一级免费毛片 | 欧美日韩在线视频观看 | 午夜宅男影院 | 日韩精品无码一本二本三本色 | 欧美成人午夜精品久久久 | 欧美少妇xxxxx | 成人av毛片 | 欧美巨大荫蒂茸毛毛人妖 | 日日夜夜免费 | 国产高清第一页 | 亚洲经典久久 | 国产精品久久久久久一区二区 | 操操操av| 国产丝袜在线播放 | 一区二区三区四区在线观看视频 | www.午夜视频 | 日韩不卡高清 | 91精品国产一区二区三竹菊影视 | 国产精品久久久久久久专区 | 性网站在线观看 | 成人小视频在线播放 | 亚洲激情综合 | 91精品视频免费观看 | 天堂网av中文字幕 | 日本高潮视频 | 日韩一区视频在线 | 久久五月天综合 |