亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 系統 > Android > 正文

Android仿美團拖拽效果實例代碼

2019-10-21 21:25:39
字體:
來源:轉載
供稿:網友

效果圖

Android,仿美團,拖拽,代碼

如上圖,實現了拖拽事件的無縫過渡。效果很流暢很自然,之所以寫輪子因為實在找不到好用的庫,該庫參考了https://github.com/woxingxiao/SlidingUpPanelLayout ,其實在大神的開源庫里就有Issues提到內嵌 scrollView 時滑動沖突的問題。再加上最近項目里面的詳情頁就有這樣的拖拽效果需求,只好自己實現一遍。

在實現的過程中,就遇到幾個比較棘手的問題,也經過了一番掙扎才想出解決的方案。

困難

  • 拖拽釋放的時機,如下拉1/6就自動收縮否則回彈,上拉1/3回彈還是展開
  • 釋放后,在回彈過程中更新背后view的視覺差、漸變效果
  • 處理好上面兩個問題,就可以很流暢的實現拖拽展開和收縮效果,接下來過渡的傳遞問題
  • 點擊漸變區域收縮并把內部scrollView滾回頂部
  • 什么時機該攔截事件還是父view處理
  • 狀態的更新和回調

以上問題也不是一蹴而就就能羅列清楚,這都是每解決一個問題我就萌新另一種想法逐漸完善而得到的結果。就比如在實現這個效果之前,我就想應該和 ViewDragHelper 有關,那么拖拽都有哪些需要重寫的方法以及我自己需要實現哪些?關于重寫 tryCaptureView、getViewVerticalDragRange、clampViewPositionVertical 必須的就不多說了,下面兩方法在本項目中處理的邏輯簡單說一下

onViewPositionChanged:當拖拽view的位置發生改變時觸發

onViewReleased:簡單可以理解為不再拖拽時觸發,但還有其狀態和方法會影響它觸發的時機,我們沒涉及到就不研究

回到開始我們想要的拖拽效果,超過多少就回彈、展開、收縮,在這里我們通過第一個方法可以知道,目前拖拽的view到底是展開還是收縮,我用了一個局部的boolean來記錄狀態,畢竟此方法執行頻繁減少消耗。再在釋放時根據 slideUp 來判斷,至于 onPanelDragged() 方法就用來跟新拖拽狀態和更新視覺差

 @Override public void onViewPositionChanged(View changedView, int left, int top, int dx, int dy) { slideUp = dy > 0;//正為收縮,負為展開 onPanelDragged(top); } @Override public void onViewReleased(View releasedChild, float xvel, float yvel) { int target; if (!slideUp) {  if (mSlideOffset >= mAnchorPoint / 6) {  target = computePanelToPosition(mAnchorPoint);  } else {  target = computePanelToPosition(0f);  } }else {  if (mSlideOffset >= mAnchorPoint / 3) {  target = computePanelToPosition(0f);  } else {  target = computePanelToPosition(mAnchorPoint);  } } if (mDragHelper != null) {  mDragHelper.settleCapturedViewAt(releasedChild.getLeft(), target); } }
 /** * 拖拽狀態更新以及位置的更新 * */ private void onPanelDragged(int newTop) { setPanelStateInternal(PanelState.DRAGGING); //重新計算距離頂部偏移 mSlideOffset = computeSlideOffset(newTop); //更新視覺差效果和分發事件 applyParallaxForCurrentSlideOffset(); //如果偏移是向上,覆蓋則無效,需要增加main的高度 LayoutParams lp = mMainView.getLayoutParams(); int defaultHeight = getHeight() - getPaddingBottom() - getPaddingTop() - mPanelHeight; if (mSlideOffset <= 0 && !mOverlayFlag) {  lp.height = (newTop - getPaddingBottom());  if (lp.height == defaultHeight) {  lp.height = LayoutParams.MATCH_PARENT;  } } else if (lp.height != LayoutParams.MATCH_PARENT && !mOverlayFlag) {  lp.height = LayoutParams.MATCH_PARENT; } mMainView.requestLayout(); }

緊接著,我們點擊展開后漸變層,收縮并將內嵌 scrollView 滾回頂部,點擊肯定就在 onTouchEvent 或者 dispatchTouchEvent 里實現,但有沒有區別呢?首先明確一點的時,不管方法寫在哪個回調里面都可以實現我們需求,但在此我寫在了后者里面,因為在 viewGroup 里面的點擊事件傳遞,dispatchTouchEvent(分發) 會經過詢問 onInterceptTouchEvent(攔截) 是否攔截再到 onTouchEvent(響應),這也算是優化的一點吧。

所有很自然而然地,我在分發里面處理了事件過渡的邏輯,其實說白了就在 MotionEvent.ACTION_MOVE 里決定了到底誰來消化這個事件

case MotionEvent.ACTION_MOVE:{ float dx = x - mPrevMotionX; float dy = y - mPrevMotionY; mPrevMotionX = x; mPrevMotionY = y; //橫向滑動就不分發了 if (Math.abs(dx) > Math.abs(dy)) {  return true; } //滑動向上、向下 if (dy > 0) { //收縮  if (mScrollableViewHelper.getScrollableViewScrollPosition(mScrollView, true) > 0) {  isMyHandleTouch = true;  return super.dispatchTouchEvent(ev);  } //之前子view處理了事件 //我們就需要重新組合一下讓面板得到一個合理的點擊事件 if (isMyHandleTouch) {  MotionEvent up = MotionEvent.obtain(ev);  up.setAction(MotionEvent.ACTION_CANCEL);  super.dispatchTouchEvent(up);  up.recycle();  ev.setAction(MotionEvent.ACTION_DOWN);  }  isMyHandleTouch = false;  return this.onTouchEvent(ev); } else { //展開  //scrollY=0表示沒滑動過,canScroll(1)表示可scroll up  //邏輯或的意義:拖拽到頂后,要不要禁用外部拖拽  if (isOnTopFlag == 1) {  int offset = mDragView.getScrollY();  boolean scroll = mScrollableViewHelper.getScrollableViewScrollPosition(mScrollView, true) > 0;  setEnabled(offset == 0 || scroll);  mDragHelper.abort();  return super.dispatchTouchEvent(ev);  }  //面板是否全部展開  if (mSlideOffset < mAnchorPoint) {  isMyHandleTouch = false;  return this.onTouchEvent(ev);  }  if (!isMyHandleTouch && mDragHelper.isDragging()) {  mDragHelper.cancel();  ev.setAction(MotionEvent.ACTION_DOWN);  }  isMyHandleTouch = true;  return super.dispatchTouchEvent(ev); }}
  • 這里消化了橫向滑動事件,因為內部 scrollView 可以通過橫向滑動優先獲取控制權,不信你注釋那句代碼,在一開始就先右滑不放再上滑,就會出現所謂的 bug
  • getScrollableViewScrollPosition 方法是一個輔助類,用來判斷view在豎直方向還有沒有可滑動的距離
  • 關鍵的 return,是要繼續處理還是給 dragHelper 處理
  • 收縮和展開其核心都圍繞 event 該給誰處理,邏輯條件有點繞

(也因為在這里的處理邏輯,有很多操作的情況沒完全覆蓋,導致不可預知的滑動出現bug,如有發現請給我反饋,我去優化)

處理到這里,需求基本達到了??梢越o設計師秀一波,把手機遞給她然后靜靜地聽她懟iOS了,“為什么 Android 都能做得到,你 iOS 卻做不出來,你看人家多厲害”。

再優化一個小問題,狀態的回調,為了避免裝逼失敗等下要求展開或者收縮時又要做些什么效果,有點危機意識。我縱觀了一些全局,實在沒有合適的方法可做回調,實在沒有方法在任何操作都觸發啊。最后我打起漸變層的主意,這個實現可把我樂了一下,太聰明了哈哈哈哈哈而且狀態都能正確回調。你要知道漸變層繪制可是需要不停的觸發的,回調只能一次

@Overrideprotected boolean drawChild(Canvas canvas, View child, long drawingTime) { ...(省略一些代碼) //沒有合適的回調方法,只能另辟蹊徑了 //在這里判斷dragView有沒有到頂,然后把事件給內嵌view final int targetY = computePanelToPosition(mAnchorPoint); final int originalY = computePanelToPosition(0f); if (mDragView.getTop() == targetY) { //避免多次回調 if (isOnTopFlag != 1 && stateCallback != null) {  stateCallback.onExpandedState(); } isOnTopFlag = 1; }else if (mDragView.getTop() == originalY){ if (isOnTopFlag == -1 && stateCallback != null) {  stateCallback.onCollapsedState(); } isOnTopFlag = 0; }else { isOnTopFlag = -1; }...(省略一些代碼)}

github:https://github.com/BmobSnail/SlideNestedPanelLayout 

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對VEVB武林網的支持。


注:相關教程知識閱讀請移步到Android開發頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
久久亚洲一区二区三区四区五区高| 国产成人在线一区二区| 欧美一级视频免费在线观看| 亚洲日本aⅴ片在线观看香蕉| 91视频免费在线| 亚洲欧洲在线免费| 亚洲欧美日本精品| 色七七影院综合| 欧美综合在线观看| 欧洲永久精品大片ww免费漫画| 欧美做爰性生交视频| 国产中文字幕亚洲| 日韩视频第一页| 97色在线视频| 中文字幕日韩高清| 中文字幕在线精品| 国产视频综合在线| 2021国产精品视频| 亚洲国产成人爱av在线播放| 欧美一区二区大胆人体摄影专业网站| 亚洲欧美综合另类中字| 午夜剧场成人观在线视频免费观看| 美女精品久久久| 国产精品v片在线观看不卡| 久久久久久久国产精品| 国产精品91视频| 亚洲精品日韩丝袜精品| 国产精品久久在线观看| 中文字幕欧美日韩精品| 91精品国产综合久久香蕉922| 色噜噜狠狠色综合网图区| 亚洲香蕉av在线一区二区三区| 成人黄色免费网站在线观看| 国产欧美久久一区二区| 日韩在线视频网站| 日本久久91av| 国内精品久久影院| 欧美成人午夜影院| 国产区亚洲区欧美区| 在线观看成人黄色| 68精品国产免费久久久久久婷婷| 国产91色在线免费| 国产成人av网| 97热精品视频官网| 亚洲精品久久久久中文字幕欢迎你| 日韩欧美在线视频| 日韩精品极品在线观看播放免费视频| 福利视频一区二区| 伊是香蕉大人久久| 91色在线观看| 成人妇女淫片aaaa视频| 色综合男人天堂| 91精品在线看| 久久6免费高清热精品| 国产精品久久久久免费a∨大胸| 亚洲三级黄色在线观看| 欧美一区二区三区四区在线| 欧美国产日韩中文字幕在线| 911国产网站尤物在线观看| 亚洲国产精品嫩草影院久久| 国产精品中文字幕在线| 日韩欧美a级成人黄色| 午夜剧场成人观在线视频免费观看| 不卡在线观看电视剧完整版| 欧美亚洲另类制服自拍| 欧美电影免费观看高清完整| 国内精品久久久久影院 日本资源| 日韩精品中文字幕视频在线| 在线日韩日本国产亚洲| 欧美专区中文字幕| 精品一区二区三区四区| 第一福利永久视频精品| 精品少妇v888av| 狠狠久久五月精品中文字幕| 亚洲免费视频在线观看| 国产日韩精品一区二区| 久久久久久18| 亚洲人午夜色婷婷| 亚洲国产精品系列| 俺去了亚洲欧美日韩| 8x海外华人永久免费日韩内陆视频| 亚洲精品suv精品一区二区| 亚洲嫩模很污视频| 久久视频在线观看免费| 国产精品入口日韩视频大尺度| 精品亚洲男同gayvideo网站| 国产精品无码专区在线观看| 中文字幕亚洲综合久久| 国产精品av免费在线观看| 日韩精品在线免费观看视频| 国产精品一区二区三区久久久| 欧美黄色www| 高清日韩电视剧大全免费播放在线观看| 中文字幕亚洲色图| 国产精品男人爽免费视频1| 久久av资源网站| 欧美日韩国产一区中文午夜| 国产成人精品视频在线观看| 日韩欧美在线视频免费观看| 亚洲一区二区中文| 97国产精品视频人人做人人爱| 91高清视频免费观看| 国产在线视频91| 91久久久在线| 久久97精品久久久久久久不卡| 在线观看成人黄色| 日韩亚洲在线观看| 国产精品美女av| 欧美亚洲伦理www| 日韩欧美一区视频| 91精品国产乱码久久久久久久久| 久久免费视频在线观看| 欧美亚洲视频一区二区| 久久99久久99精品免观看粉嫩| 亚洲国产私拍精品国模在线观看| 色无极亚洲影院| 欧美日韩国产综合视频在线观看中文| 欧美在线一区二区视频| 亚洲电影免费观看高清完整版在线| 亚洲人精品午夜在线观看| 日本高清不卡在线| 久久久久亚洲精品成人网小说| 欧美日韩在线视频观看| 最近2019年手机中文字幕| 成人写真视频福利网| 欧美黑人一级爽快片淫片高清| 欧美性生交大片免网| 国产精品视频免费在线观看| 福利二区91精品bt7086| 国产精品久久电影观看| 亚洲乱码av中文一区二区| 日韩在线观看免费高清完整版| 欧美视频在线免费看| 曰本色欧美视频在线| 欧美性高潮床叫视频| 欧美精品在线网站| 国产精品爽黄69| 91精品国产91久久久久久不卡| 久久久久久久香蕉网| 亚洲免费小视频| 欧美影院在线播放| 茄子视频成人在线| 国模视频一区二区三区| 一区二区三区视频在线| 亚洲mm色国产网站| 欧美视频中文在线看| 夜夜嗨av色一区二区不卡| 欧美床上激情在线观看| 国产精品成人一区二区三区吃奶| 国产日韩综合一区二区性色av| 热99精品里视频精品| 精品日韩美女的视频高清| 国产成+人+综合+亚洲欧美丁香花| 欧美黑人一级爽快片淫片高清| 精品日韩美女的视频高清| 日韩视频中文字幕| www.国产精品一二区| 国产+人+亚洲| 国产亚洲视频在线| 亚洲嫩模很污视频| 日韩在线欧美在线| 中文字幕成人在线| 国内精久久久久久久久久人| 91久久久精品|