国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

Android源碼進階之ViewDragHelper原理機制解析

移動開發 Android
tryCaptureView(View child, int pointerId) 這是一個抽象類,必須去實現,也只有在這個方法返回true的時候下面的方法才會生效。

[[422791]]

本文轉載自微信公眾號「Android開發編程」,作者Android開發編程 。轉載本文請聯系Android開發編程公眾號。

前言

ViewDragHelper類,是用來處理View邊界拖動相關的類;

主要功能處理在View上的觸摸事件,記錄觸摸點、計算距離、滾動動畫、狀態回調等,如果我們自己手動實現自然會很麻煩還可能出錯,而這個類會幫助我們大大簡化工作量;

今天我們就來分析一波;

一、ViewDragHelper的中主要API介紹

1、ViewDragHelper create(ViewGroup forParent, Callback cb)

一個靜態的創建方法;

  • 參數1:出入的是相應的ViewGroup;
  • 參數2:是一個回掉,需要自己實現;

2、shouldInterceptTouchEvent(MotionEvent ev)

處理事件分發的(怎么說這個方法呢?主要是將ViewGroup的事件分發,委托給ViewDragHelper進行處理);

  • 參數1:MotionEvent ev 主要是ViewGroup的事件;

3、processTouchEvent(MotionEvent event)

處理相應TouchEvent的方法,這里要注意一個問題,處理相應的TouchEvent的時候要將結果返回為true,消費本次事件,否則將無法使用ViewDragHelper處理相應的拖拽事件;

4、ViewDragHelper.Callback的API

tryCaptureView(View child, int pointerId) 這是一個抽象類,必須去實現,也只有在這個方法返回true的時候下面的方法才會生效;

onViewDragStateChanged(int state) 當狀態改變的時候回調,返回相應的狀態(這里有三種狀態);

  • STATE_IDLE 閑置狀態;
  • STATE_DRAGGING 正在拖動;
  • STATE_SETTLING 放置到某個位置;

onViewPositionChanged(View changedView, int left, int top, int dx, int dy) 當你拖動的View位置發生改變的時候回調;

  • 參數1:你當前拖動的這個View
  • 參數2:距離左邊的距離
  • 參數3:距離右邊的距離
  • 參數4:x軸的變化量
  • 參數5:y軸的變化量

onViewCaptured(View capturedChild, int activePointerId)捕獲View的時候調用的方法

  • 參數1:捕獲的View(也就是你拖動的這個View);
  • 參數2:這個參數我也不知道什么意思API中寫的一個什么指針,這里沒有到也沒有注意;

onViewReleased(View releasedChild, float xvel, float yvel) 當View停止拖拽的時候調用的方法

  • 參數1:你拖拽的這個View
  • 參數2:x軸的速率
  • 參數3:y軸的速率

clampViewPositionVertical(View child, int top, int dy) 豎直拖拽的時候回調的方法

  • 參數1:拖拽的View
  • 參數2:距離頂部的距離
  • 參數3:變化量

clampViewPositionHorizontal(View child, int left, int dx) 水平拖拽的時候回調的方法

  • 參數1:拖拽的View
  • 參數2:距離左邊的距離
  • 參數3:變化量

二、實現原理介紹

1、初始化

  1. private ViewDragHelper(Context context, ViewGroup forParent, Callback cb) { 
  2.         ... 
  3.         mParentView = forParent;//BaseView 
  4.         mCallback = cb;//callback 
  5.         final ViewConfiguration vc = ViewConfiguration.get(context); 
  6.         final float density = context.getResources().getDisplayMetrics().density; 
  7.         mEdgeSize = (int) (EDGE_SIZE * density + 0.5f);//邊界拖動距離范圍 
  8.         mTouchSlop = vc.getScaledTouchSlop();//拖動距離閾值 
  9.         mScroller = new OverScroller(context, sInterpolator);//滾動器 
  10.     } 
  • mParentView是指基于哪個View進行觸摸處理;
  • mCallback是觸摸處理的各個階段的回調;
  • mEdgeSize是指在邊界多少距離內算作拖動,默認為20dp;
  • mTouchSlop指滑動多少距離算作拖動,用的系統默認值;
  • mScroller是View滾動的Scroller對象,用于處理釋觸摸放后,View的滾動行為,比如滾動回原始位置或者滾動出屏幕;

2.攔截事件處理

該類提供了boolean shouldInterceptTouchEvent(MotionEvent)方法:

  1. override fun onInterceptTouchEvent(ev: MotionEvent?) = 
  2.             dragHelper?.shouldInterceptTouchEvent(ev) ?: super.onInterceptTouchEvent(ev) 

該方法用于處理mParentView是否攔截此次事件

  1. public boolean shouldInterceptTouchEvent(MotionEvent ev) { 
  2.         ... 
  3.         switch (action) { 
  4.             ... 
  5.             case MotionEvent.ACTION_MOVE: { 
  6.                 if (mInitialMotionX == null || mInitialMotionY == null) break; 
  7.                 // First to cross a touch slop over a draggable view wins. Also report edge drags. 
  8.                 final int pointerCount = ev.getPointerCount(); 
  9.                 for (int i = 0; i < pointerCount; i++) { 
  10.                     final int pointerId = ev.getPointerId(i); 
  11.                     // If pointer is invalid then skip the ACTION_MOVE. 
  12.                     if (!isValidPointerForActionMove(pointerId)) continue
  13.                     final float x = ev.getX(i); 
  14.                     final float y = ev.getY(i); 
  15.                     final float dx = x - mInitialMotionX[pointerId]; 
  16.                     final float dy = y - mInitialMotionY[pointerId]; 
  17.                     final View toCapture = findTopChildUnder((int) x, (int) y); 
  18.                     final boolean pastSlop = toCapture != null && checkTouchSlop(toCapture, dx, dy); 
  19.                     ... 
  20.                     //判斷pointer的拖動邊界 
  21.                     reportNewEdgeDrags(dx, dy, pointerId); 
  22.                     ... 
  23.                 } 
  24.                 saveLastMotion(ev); 
  25.                 break; 
  26.             } 
  27.             ... 
  28.         } 
  29.         return mDragState == STATE_DRAGGING; 

攔截事件的前提是mDragState為STATE_DRAGGING,也就是正在拖動狀態下才會攔截,那么什么時候會變為拖動狀態呢?當ACTION_MOVE時,調用reportNewEdgeDrags方法:

  1. private void reportNewEdgeDrags(float dx, float dy, int pointerId) { 
  2.         int dragsStarted = 0; 
  3.   //判斷是否在Left邊緣進行滑動 
  4.         if (checkNewEdgeDrag(dx, dy, pointerId, EDGE_LEFT)) { 
  5.             dragsStarted |= EDGE_LEFT; 
  6.         } 
  7.         if (checkNewEdgeDrag(dy, dx, pointerId, EDGE_TOP)) { 
  8.             dragsStarted |= EDGE_TOP; 
  9.         } 
  10.         ... 
  11.         if (dragsStarted != 0) { 
  12.             mEdgeDragsInProgress[pointerId] |= dragsStarted; 
  13.           //回調拖動的邊 
  14.             mCallback.onEdgeDragStarted(dragsStarted, pointerId); 
  15.         } 
  16. private boolean checkNewEdgeDrag(float delta, float odelta, int pointerId, int edge) { 
  17.         final float absDelta = Math.abs(delta); 
  18.         final float absODelta = Math.abs(odelta); 
  19. //是否支持edge的拖動以及是否滿足拖動距離的閾值 
  20.         if ((mInitialEdgesTouched[pointerId] & edge) != edge  || (mTrackingEdges & edge) == 0 
  21.                 || (mEdgeDragsLocked[pointerId] & edge) == edge 
  22.                 || (mEdgeDragsInProgress[pointerId] & edge) == edge 
  23.                 || (absDelta <= mTouchSlop && absODelta <= mTouchSlop)) { 
  24.             return false
  25.         } 
  26.         if (absDelta < absODelta * 0.5f && mCallback.onEdgeLock(edge)) { 
  27.             mEdgeDragsLocked[pointerId] |= edge; 
  28.             return false
  29.         } 
  30.         return (mEdgeDragsInProgress[pointerId] & edge) == 0 && absDelta > mTouchSlop; 

可以看到,當ACTION_MOVE時,會嘗試找到pointer對應的拖動邊界,這個邊界可以由我們來制定,比如側滑關閉頁面是從左側開始的,所以我們可以調用setEdgeTrackingEnabled(ViewDragHelper.EDGE_LEFT)來設置只支持左側滑動。而一旦有滾動發生,就會回調callback的onEdgeDragStarted方法,交由我們做如下操作:

  1. override fun onEdgeDragStarted(edgeFlags: Int, pointerId: Int) { 
  2.                 super.onEdgeDragStarted(edgeFlags, pointerId) 
  3.                 dragHelper?.captureChildView(getChildAt(0), pointerId) 
  4.             } 
  5. 我們調用了ViewDragHelper的captureChildView方法: 
  6. public void captureChildView(View childView, int activePointerId) { 
  7.         mCapturedView = childView;//記錄拖動view 
  8.         mActivePointerId = activePointerId; 
  9.         mCallback.onViewCaptured(childView, activePointerId); 
  10.         setDragState(STATE_DRAGGING);//設置狀態為開始拖動 

此時,就記錄了拖動的View,并將狀態置為拖動,那么在下次ACTION_MOVE的時候,該mParentView就會攔截事件,交由自己的onTouchEvent方法處理拖動了;

3.拖動事件處理

該類提供了void processTouchEvent(MotionEvent)方法,通常我們需要這么寫:

  1. override fun onTouchEvent(event: MotionEvent?): Boolean { 
  2.         dragHelper?.processTouchEvent(event)//交由ViewDragHelper處理 
  3.         return true 

該方法用于處理mParentView攔截事件后的拖動處理:

  1. public void processTouchEvent(MotionEvent ev) { 
  2.         ... 
  3.         switch (action) { 
  4.             ... 
  5.             case MotionEvent.ACTION_MOVE: { 
  6.                 if (mDragState == STATE_DRAGGING) { 
  7.                     // If pointer is invalid then skip the ACTION_MOVE. 
  8.                     if (!isValidPointerForActionMove(mActivePointerId)) break; 
  9.                     final int index = ev.findPointerIndex(mActivePointerId); 
  10.                     final float x = ev.getX(index); 
  11.                     final float y = ev.getY(index); 
  12.                     //計算距離上次的拖動距離 
  13.                     final int idx = (int) (x - mLastMotionX[mActivePointerId]); 
  14.                     final int idy = (int) (y - mLastMotionY[mActivePointerId]); 
  15.                     dragTo(mCapturedView.getLeft() + idx, mCapturedView.getTop() + idy, idx, idy);//處理拖動 
  16.                     saveLastMotion(ev);//記錄當前觸摸點 
  17.                 }... 
  18.                 break; 
  19.             } 
  20.             ... 
  21.             case MotionEvent.ACTION_UP: { 
  22.                 if (mDragState == STATE_DRAGGING) { 
  23.                     releaseViewForPointerUp();//釋放拖動view 
  24.                 } 
  25.                 cancel(); 
  26.                 break; 
  27.             }... 
  28.         } 

(1)拖動

ACTION_MOVE時,會計算出pointer距離上次的位移,然后計算出capturedView的目標位置,進行拖動處理;

  1. private void dragTo(int leftint topint dx, int dy) { 
  2.         int clampedX = left
  3.         int clampedY = top
  4.         final int oldLeft = mCapturedView.getLeft(); 
  5.         final int oldTop = mCapturedView.getTop(); 
  6.         if (dx != 0) { 
  7.             clampedX = mCallback.clampViewPositionHorizontal(mCapturedView, left, dx);//通過callback獲取真正的移動值 
  8.             ViewCompat.offsetLeftAndRight(mCapturedView, clampedX - oldLeft);//進行位移 
  9.         } 
  10.         if (dy != 0) { 
  11.             clampedY = mCallback.clampViewPositionVertical(mCapturedView, top, dy); 
  12.             ViewCompat.offsetTopAndBottom(mCapturedView, clampedY - oldTop); 
  13.         } 
  14.         if (dx != 0 || dy != 0) { 
  15.             final int clampedDx = clampedX - oldLeft; 
  16.             final int clampedDy = clampedY - oldTop; 
  17.             mCallback.onViewPositionChanged(mCapturedView, clampedX, clampedY, 
  18.                     clampedDx, clampedDy);//callback回調移動后的位置 
  19.         } 

通過callback的clampViewPositionHorizontal方法決定實際移動的水平距離,通常都是返回left值,即拖動了多少就移動多少;

通過callback的onViewPositionChanged方法,可以對View拖動后的新位置做一些處理,如;

  1. override fun onViewPositionChanged(changedView: View?, leftInttopInt, dx: Int, dy: Int) { 
  2.   super.onViewPositionChanged(changedView, lefttop, dx, dy) 
  3.     //當新的left位置到達width時,即滑動除了界面,關閉頁面 
  4.     if (left >= width && context is Activity && !context.isFinishing) { 
  5.       context.finish() 
  6.     } 

(2)釋放

而ACTION_UP動作時,要釋放拖動View

  1. private void releaseViewForPointerUp() { 
  2.         ... 
  3.         dispatchViewReleased(xvel, yvel); 
  4. private void dispatchViewReleased(float xvel, float yvel) { 
  5.         mReleaseInProgress = true
  6.         mCallback.onViewReleased(mCapturedView, xvel, yvel);//callback回調釋放 
  7.         mReleaseInProgress = false
  8.         if (mDragState == STATE_DRAGGING) { 
  9.             // onViewReleased didn't call a method that would have changed this. Go idle. 
  10.             setDragState(STATE_IDLE);//重置狀態 
  11.         } 

通常在callback的onViewReleased方法中,我們可以判斷當前釋放點的位置,從而決定是要回彈頁面還是滑出屏幕

  1. override fun onViewReleased(releasedChild: View?, xvel: Float, yvel: Float) { 
  2.   super.onViewReleased(releasedChild, xvel, yvel) 
  3.     //滑動速度到達一定值時直接關閉 
  4.     if (xvel >= 300) {//滑動頁面到屏幕外,關閉頁面 
  5.       dragHelper?.settleCapturedViewAt(width, 0) 
  6.     } else {//回彈頁面 
  7.       dragHelper?.settleCapturedViewAt(0, 0) 
  8.     } 
  9.   //刷新,開始關閉或重置動畫 
  10.   invalidate() 

如滑動速度大于300時,我們調用settleCapturedViewAt方法將頁面滾動出屏幕,否則調用該方法進行回彈

(3)滾動

  1. public boolean settleCapturedViewAt(int finalLeft, int finalTop) { 
  2.   return forceSettleCapturedViewAt(finalLeft, finalTop, 
  3.                                    (int) mVelocityTracker.getXVelocity(mActivePointerId), 
  4.                                    (int) mVelocityTracker.getYVelocity(mActivePointerId)); 
  5. private boolean forceSettleCapturedViewAt(int finalLeft, int finalTop, int xvel, int yvel) { 
  6.   //當前位置 
  7.   final int startLeft = mCapturedView.getLeft(); 
  8.   final int startTop = mCapturedView.getTop(); 
  9.   //偏移量 
  10.   final int dx = finalLeft - startLeft; 
  11.   final int dy = finalTop - startTop; 
  12.   ... 
  13.   final int duration = computeSettleDuration(mCapturedView, dx, dy, xvel, yvel); 
  14.   //使用Scroller對象開始滾動 
  15.   mScroller.startScroll(startLeft, startTop, dx, dy, duration); 
  16. //重置狀態為滾動 
  17.   setDragState(STATE_SETTLING); 
  18.   return true
  • 其內部使用的是Scroller對象:是View的滾動機制,其回調是View的computeScroll()方法,在其內部通過Scroller對象的computeScrollOffset方法判斷是否滾動完畢,如仍需滾動,需要調用invalidate方法進行刷新;
  • ViewDragHelper據此提供了一個類似的方法continueSettling,需要在computeScroll中調用,判斷是否需要invalidate;
  1. public boolean continueSettling(boolean deferCallbacks) { 
  2.   if (mDragState == STATE_SETTLING) { 
  3.     //是否滾動結束 
  4.     boolean keepGoing = mScroller.computeScrollOffset(); 
  5.     //當前滾動值 
  6.     final int x = mScroller.getCurrX(); 
  7.     final int y = mScroller.getCurrY(); 
  8.     //偏移量 
  9.     final int dx = x - mCapturedView.getLeft(); 
  10.     final int dy = y - mCapturedView.getTop(); 
  11. //便宜操作 
  12.     if (dx != 0) { 
  13.       ViewCompat.offsetLeftAndRight(mCapturedView, dx); 
  14.     } 
  15.     if (dy != 0) { 
  16.       ViewCompat.offsetTopAndBottom(mCapturedView, dy); 
  17.     } 
  18. //回調 
  19.     if (dx != 0 || dy != 0) { 
  20.       mCallback.onViewPositionChanged(mCapturedView, x, y, dx, dy); 
  21.     } 
  22.     //滾動結束狀態 
  23.     if (!keepGoing) { 
  24.       if (deferCallbacks) { 
  25.         mParentView.post(mSetIdleRunnable); 
  26.       } else { 
  27.         setDragState(STATE_IDLE); 
  28.       } 
  29.     } 
  30.   } 
  31.   return mDragState == STATE_SETTLING; 

在我們的View中

  1. override fun computeScroll() { 
  2.   super.computeScroll() 
  3.     if (dragHelper?.continueSettling(true) == true) { 
  4.       invalidate() 
  5.     } 
  6. 以上,就是ViewDragHelper的實現原理和使用方式 
  7. override fun computeScroll() { 
  8.   super.computeScroll() 
  9.     if (dragHelper?.continueSettling(true) == true) { 
  10.       invalidate() 
  11.     } 

以上,就是ViewDragHelper的實現原理和使用方式

總結

ViewDragHelper本質上是對MotionEvent的分析及處理,并提供了一系列的監聽回調方法,來幫助我們減輕開發負擔,更為方便地處理控件的滑動拖拽邏輯;

 

是不是覺得很簡單,一起加油,各位老鐵們;

 

責任編輯:武曉燕 來源: Android開發編程
相關推薦

2021-09-01 06:48:16

AndroidGlide緩存

2021-10-15 09:19:17

AndroidSharedPrefe分析源碼

2021-09-07 06:40:25

AndroidLiveData原理

2021-09-04 07:29:57

Android

2021-08-17 13:41:11

AndroidView事件

2021-08-12 16:28:10

AndroidHandleLooper

2021-09-08 06:51:52

AndroidRetrofit原理

2021-09-05 07:35:58

lifecycleAndroid組件原理

2021-09-14 07:06:12

Android磁盤緩存

2021-09-29 09:42:32

AndroidViewDragHel拖動上下滑卡片

2021-09-03 07:27:38

AndroidGlide管理

2021-09-06 13:12:05

前端JavaScript編程

2021-09-30 07:36:51

AndroidViewDraw

2011-06-23 14:05:32

Qt 事件機制

2021-12-30 22:50:32

KafkaConsumer 源碼

2021-08-05 20:39:34

AndroidKotlinStandard.kt

2021-09-02 07:00:01

Glide流程Android

2021-09-11 07:32:15

Java線程線程池

2017-02-21 12:20:20

Android事件分發機制實例解析

2021-08-10 20:41:33

AndroidApp流程
點贊
收藏

51CTO技術棧公眾號

亚洲欧美精品一区二区| 亚洲成avwww人| 美女黄毛**国产精品啪啪| 97久久亚洲| 91精品国产综合久久精品| 动漫av网站免费观看| 久久福利毛片| 国产亚洲自拍偷拍| 天天做天天爱天天综合网| 国内精品400部情侣激情| 欧美一级大黄| 亚洲福利在线视频| 国产高清在线a视频大全| 日韩一级高清毛片| www视频在线免费观看| 欧美午夜片欧美片在线观看| 日韩欧美亚洲一区| 亚洲天堂免费在线观看视频| 亚洲综合婷婷久久| 国产精品欧美一区喷水| 777视频在线| 亚洲欧美成人一区二区三区| 成人羞羞视频免费看看| 一区二区久久久久久| 免费观看一二区视频网站| 一片黄亚洲嫩模| 美女网站在线| 色网综合在线观看| jizzjizz在线观看| 日韩欧美在线影院| 欧美草逼视频| 亚洲欧美日韩一区在线| 国产一区一一区高清不卡| 国产一区二区三区三区在线观看 | 一本色道久久综合| 欧美精品中文字幕一区二区| 日韩国产精品久久久| 亚洲欧洲日韩精品| 国产精品一区免费在线观看| 草草视频在线免费观看| 国产日韩v精品一区二区| 免费黄色av电影| 欧美午夜精品一区二区三区| 春色校园综合激情亚洲| 久久精品视频在线播放| 国产女人18毛片水真多18精品| 日本久久久久久| 亚洲五月婷婷| 糖心vlog在线免费观看| 99国产精品国产精品毛片| 午夜伦伦电影理论片费看| 午夜久久福利影院| 日本片在线观看| 久久亚洲春色中文字幕| 日韩三级在线| 一区二区三区四区五区精品| 久久你懂得1024| 你懂的视频在线免费| 日韩欧美视频一区| 福利欧美精品在线| 国产一区二区在线观看免费播放 | 91精品天堂| 狂野欧美一区| 亚洲爆乳无码专区| 日本精品视频一区二区| 天堂在线中文网官网| 4438全国成人免费| 亚洲久久视频| 欧美综合在线观看视频| 狠狠干狠狠久久| 成人爱爱网址| 国产精品久久久久久影视| 日韩精品电影一区亚洲| 超碰在线播放91| 制服丝袜成人动漫| 77成人影视| 欧美日韩在线观看一区| 中文字幕永久在线不卡| 久草在线新免费首页资源站| 欧美一区二区.| 日韩电影一区二区三区四区| 成人网免费看| 亚洲欧洲黄色网| 色婷婷热久久| www.浪潮av.com| 91精品国产综合久久久久久久| 激情综合五月| 欧洲精品一区色| 亚洲男人天堂一区| 国产亚洲一区二区手机在线观看| 91精品视频一区| 91麻豆精东视频| 18网站在线观看| 国产精品视频久| 成人免费视频caoporn| 户外极限露出调教在线视频| 欧美另类高清videos| 久久成人免费| 色综合久久网女同蕾丝边| 欧美成人精品一区二区| 99热在线精品观看| 国产精品666| 久久中文字幕国产| 精品影视av免费| 日本中文字幕在线观看| 国产欧美一区二区三区四区| 972aa.com艺术欧美| 高潮在线视频| 免费看成人午夜电影| 黑人巨大精品欧美一区二区三区 | av毛片免费看| 日韩在线播放一区| 麻豆成人av在线| 免费av在线| 99一区二区三区| 亚洲丶国产丶欧美一区二区三区| 亚洲一区电影| 日韩免费视频播放| 亚洲欧洲激情在线| 麻豆成人久久精品二区三区小说| 一广人看www在线观看免费视频| 成人在线播放av| 亚洲最新在线观看| 日韩av中文字幕一区| 日韩av手机版| 精品国内自产拍在线观看| 黑人精品欧美一区二区蜜桃| av软件在线观看| 免费在线成人av电影| 欧美日韩国产成人在线91| 欧美日韩国产综合网| 蜜芽tv福利在线视频| 成人黄色在线播放| 欧美日韩激情美女| 欧美精品一级| 二区三区在线播放| 久久本道综合色狠狠五月| 一本久久a久久精品亚洲| 国产精品99久久精品| 日韩欧美在线观看一区二区| 亚洲一区二区三区成人在线视频精品| 一区二区三区日韩欧美精品 | 18禁男女爽爽爽午夜网站免费| 欧美日韩mp4| 999亚洲国产精| 1区2区3区在线视频| 亚洲国产精品久久久久婷婷老年| 精品三级在线看| 国产麻豆一精品一av一免费| 亚洲高清国产拍精品26u| 国产理论在线播放| 清纯唯美亚洲综合| 狠狠色噜噜狠狠狠狠97| 99成人精品| 樱桃视频成人在线观看| 免费黄色特级片| 91精品久久久久| 欧美一区二区三区四区在线观看 | 亚洲精品乱码久久久久| 欧美日韩精品在线一区| 高清福利在线观看| 亚洲图片都市激情| 久久伊人91精品综合网站| 国产精品久久久久久妇女6080| blacked蜜桃精品一区| av播放在线| 国产一二三四五| 97热精品视频官网| 欧美中文一区二区三区| 国产一区二区三区不卡在线观看| 一区二区在线免费播放| 日本又骚又刺激的视频在线观看| 久久久综合香蕉尹人综合网| 中文字幕亚洲一区| 国产日韩一区二区| 欧美一区二区三区四区在线观看 | zzijzzij亚洲日本成熟少妇| 中文字幕在线一区免费| 欧美三级特黄| 最新日韩三级| 日本高清好狼色视频| 日韩国产高清一区| 久久国产精品久久久久久| 亚洲国产综合人成综合网站| 天堂精品中文字幕在线| 亚洲精品不卡在线观看| 嫩草精品影院| 日韩视频在线视频| 成人精品视频久久久久| 亚洲色图50p| 亚洲最新在线观看| 奇米精品一区二区三区在线观看| 亚洲一区二区电影| sm国产在线调教视频| 免费福利片在线观看| 亚洲欧美久久久久一区二区三区| 午夜精品一区二区三区在线视| 欧美一区二区三区视频在线观看| 国产蜜臀97一区二区三区| 西西人体一区二区|