国产无遮挡裸体免费直播视频,久久精品国产蜜臀av,动漫在线视频一区二区,欧亚日韩一区二区三区,久艹在线 免费视频,国产精品美女网站免费,正在播放 97超级视频在线观看,斗破苍穹年番在线观看免费,51最新乱码中文字幕

使用 Vue 實現(xiàn)一個虛擬列表的方法

 更新時間:2019年08月20日 08:57:51   作者:dawencilv-1  
這篇文章主要介紹了使用 Vue 實現(xiàn)一個虛擬列表的方法,本文通過實例代碼給大家介紹的非常詳細,具有一定的參考借鑒價值,需要的朋友可以參考下

因為 DOM 性能瓶頸,大型列表存在難以克服的性能問題。 因此,就有了 “局部渲染” 的優(yōu)化方案,這就是虛擬列表的核心思想。

虛擬列表的實現(xiàn),需要重點關(guān)注的問題一有以下幾點:

  • 可視區(qū)域的計算方法
  • 可視區(qū)域的 DOM 更新方案
  • 事件的處理方案

下面逐一分解說明。

可視區(qū)域計算

可視區(qū)域的計算,就是使用當前視口的高度、當前滾動條滾過的距離,得到一個可視區(qū)域的坐標區(qū)間。 算出可視區(qū)域的坐標區(qū)間之后,在去過濾出落在該區(qū)間內(nèi)的列表項,這個過程,列表項的坐標也是必須能算出的。

思考以下情況,

  • 我們的視口高度為 100px
  • 我們當前已經(jīng)滾動了 100px
  • 我們的列表項,每一項高度為 20px

根據(jù)這些條件,我們可以計算出,當前可視區(qū)域為第 11 項至第 20 項。

01 - 05,可視區(qū)域上方
+----+-----------+--------
| 06 | 100 ~ 120 |
+----+-----------+
| 07 | 120 ~ 140 |
+----+-----------+
| 08 | 140 ~ 160 | 可視區(qū)域
+----+-----------+
| 09 | 160 ~ 180 |
+----+-----------+
| 10 | 180 ~ 200 |
+----+-----------+--------
  11 - N,可視區(qū)域下方

這是因為列表項高度是固定的,我們可以通過簡單的四則運算算出已經(jīng)滾動過去的 100px 中,已經(jīng)滾動走了 5 個列表項,因此可視區(qū)域是從第 6 項開始,而視口高度為 100px,能容納 100 / 20 即 5 個條目。

上面例子的情況非常簡單,也不存在性能問題,因此實際上并沒有展開討論的價值。 而還有另一種復(fù)雜很多的情況,那就是,列表項高度不固定,根據(jù)內(nèi)容決定高度。

此時,我們就沒辦法直接使用四則運算一步到位算出可視區(qū)域?qū)?yīng)的條目了。

而必須實現(xiàn)一種機制,記錄所有列表項的坐標,再通過檢查列表項是否落在視口內(nèi)。

下面重點討論該問題。

列表項的坐標

列表項的坐標,可以通過以下公式定義:

<列表項 top 坐標值> = <上一項 top 坐標值> + <上一項的高度值>

第一個列表項的 top 坐標值為 0,因此,只要記錄所有列表項目的高度,即可算出任意一個列表項的 top 坐標值。 于是,問題就變成了,必須使用某種方式來存儲每個條目的高度。

我想,最容易想到的方案就是,使用一個數(shù)組,一一對應(yīng)地存儲列表每項的高度值。 然后獲取特定項的坐標值時,提取第一項到目標項的值,進行累加運算。參考下面代碼進行理解:

// 假設(shè)使用該數(shù)組存儲列表每一項的高度
const itemHeightStore = [20, 20, 20, 20, 20]

// 使用該方法,可以算出列表中指定項的 top 坐標值
const getTop = (index) => {
 let sum = 0
 while (index--) sum += itemHeightStore[index] || 0
 return sum
}
// 第一項
getTop(0) // 0

// 第二項
getTop(1) // 20
// ...

該實現(xiàn)可以很好地工作。

但是,該算法存在嚴重的性能問題,每獲取一個列表項的坐標都要遍歷列表,復(fù)雜度 O(n),非常不劃算。

如果換一種方式,直接存儲每一項的坐標呢?

其實本質(zhì)是一樣的。因為我們的列表項高度是不固定的,我們快速拖動滾動條到不同的區(qū)域時,需要根據(jù)局部渲染結(jié)果算出高度用于更新數(shù)組記錄,而在更新某一項時,該項后續(xù)的所有條目也需要全部更新,復(fù)雜度一樣是 O(n)。

所以,使用數(shù)組來維護每一項的高度或者坐標,在列表規(guī)模比較大的時候,就會消耗大量的 CPU 時間。

也許使用 TypedArray 會有好的表現(xiàn)?

仔細觀察上面例子中的數(shù)組,結(jié)合現(xiàn)實中列表的情況,我們可以觀察到一個現(xiàn)象:

列表項往往是相似的,在許多情況下,高度也很可能是一致的。
基于這種經(jīng)驗,我們可以采用區(qū)間來存儲列表項的高度。

通過折疊記錄相鄰的,相同高度的列表項,來減少列表遍歷操作。
比如以下表示方式:

const range = {
 start: 0,
 end: 4,
 value: 20
}

可以很好地表達列表第 1 項至第 5 項的高度都為 20px。

如果我們需要求第 6 項的高度的話,就只需進行一次簡單的四則運算即可,無需遍歷累加這 5 項。

很容易得出結(jié)論,如果列表大部分情況是相同高度,只有個別條目高度不一致時(例如文本換行),將會有非常優(yōu)異的性能表現(xiàn)。

當然使用區(qū)間,也不是沒有代價的。這又會帶來數(shù)據(jù)結(jié)構(gòu)的復(fù)雜性。

由于折疊了相鄰相同高度的結(jié)點,會導(dǎo)致存儲的列表無法跟原始條目一一對應(yīng)。所以,我們就不能簡單得知我們想查詢的列表項的高度存儲在哪里了, 為此需要設(shè)計一種專門的存儲機制。

這種存儲機制,需要擁有這些特性:

  • 高效的查詢??梢酝ㄟ^列表項序號,快速獲得對應(yīng)的 range,以及該 range 之前的所有 range。
  • 高效地修改??梢愿咝У夭迦?、移除 range,合并 range、拆分 range。

結(jié)合我們學過的數(shù)據(jù)結(jié)構(gòu)知識,可以考慮使用某種 BST 來存儲,從而獲得良好的查詢、插入性能。 而 range 的合并、拆分等,則可以實現(xiàn)一個專門的類來管理。

下面直接給出一個簡單的代碼實現(xiàn)供參考,代碼中已經(jīng)加上了大量的注釋,直接閱讀應(yīng)該比解說要更清晰。

// Avl.ts
const SLIGHTLY_UNBALANCED_RIGHT = -1
const SLIGHTLY_UNBALANCED_LEFT = 1
const UNBALANCED_RIGHT = -2
const UNBALANCED_LEFT = 2
// 樹結(jié)點
class AvlNode<K extends any = any> {
 key: any
 left: AvlNode<K> | null
 right: AvlNode<K> | null
 parent: AvlNode<K> | null
 _height: number
 _prevHeight: number
 constructor(key: K) {
  this.key = key
  this.left = null
  this.right = null
  this.parent = null
  this._height = 0
  this._prevHeight = 0
 }
 // 刷新前的高度,方便平衡操作
 get prevHeight() {
  return this._prevHeight | 0
 }
 get height() {
  return this._height | 0
 }
 set height(value) {
  this._prevHeight = this._height | 0
  this._height = value | 0
 }
 // 左子樹高度
 get leftHeight() {
  if (this.left === null) return -1
  return this.left.height | 0
 }
 // 右子樹高度
 get rightHeight() {
  if (this.right === null) return -1
  return this.right.height | 0
 }
 // 平衡因子
 get balanceFactor() {
  return this.leftHeight - this.rightHeight
 }
 updateHeight() {
  const { leftHeight, rightHeight } = this
  const height = ((leftHeight > rightHeight ? leftHeight : rightHeight) + 1) | 0
  this.height = height
 }
}
// AVL 樹
export class Avl<K extends any = any> {
 _root: AvlNode<K> | null
 _size: number
 constructor() {
  this._root = null
  this._size = 0
 }
 get size() {
  return this._size
 }
 // 插入節(jié)點
 insert(key: K) {
  const node = new AvlNode<K>(key)
  const insertPoint = this._nodeInsert(node)
  // 本次插入是重復(fù)結(jié)點,直接更新 key / value
  // 無新結(jié)點插入,所以無需進行插入后的調(diào)整
  if (insertPoint == null) return
  // 新增結(jié)點成功時,回溯調(diào)整搜索路徑上的結(jié)點
  this._adjustAfterInsertion(insertPoint)
 }
 // 刪除節(jié)點,返回是否成功刪除結(jié)點
 delete(key: K): boolean {
  // 搜索待刪除結(jié)點
  const targetNode = this._nodeSearch(key)
  // 未找到 value 對應(yīng)結(jié)點
  if (targetNode == null) return false
  // 執(zhí)行刪除結(jié)點操作
  const backtracking = this._nodeErase(targetNode)
  const parent = backtracking[0]
  // 回溯調(diào)整搜索路徑上的結(jié)點
  if (parent !== null) {
   this._adjustAfterRemoval(parent)
  }
  return true
 }
 // 通過 key 查找包含該 key 范圍的節(jié)點 key
 search(key: K) {
  const node = this._nodeSearch(key)
  if (node !== null) return node.key
  return null
 }
 // 搜索 start 、end 兩個 key 之間的所有 key 列表
 searchRange(start: K, end: K) {
  const results: K[] = []
  // 找到符合條件的 root 節(jié)點
  let root = this._root
  while (root !== null) {
   const result1 = start.compareTo(root.key)
   const result2 = end.compareTo(root.key)
   // 當前節(jié)點比 start 小,不再搜索左子樹
   if (result1 > 0) {
    root = root.right
    continue
   }
   // 當前節(jié)點大于 end,不再搜索右子樹
   if (result2 < 0) {
    root = root.left
    continue
   }
   break
  }
  if (!root) return results
  const stack = []
  let current: AvlNode<K> | null = root
  while (stack.length || current) {
   while (current) {
    stack.push(current)
    // 當前節(jié)點比 start 小,不再搜索 current 的左子樹
    if (start.compareTo(current.key) > 0) break
    current = current.left
   }
   if (stack.length) {
    // 指向棧頂
    current = stack[stack.length - 1]
    const gteStart = start.compareTo(current.key) <= 0
    const lteEnd = end.compareTo(current.key) >= 0
    if (gteStart && lteEnd) {
     results.push(current.key)
    }
    stack.pop()
    // 只有 current 比 end 小,才繼續(xù)搜索 current 的右子樹
    if (lteEnd) {
     current = current.right
    }
    else {
     current = null
    }
   }
  }
  return results
 }
 // 增加結(jié)點數(shù)量
 _increaseSize() {
  this._size += 1
 }
 // 減少結(jié)點數(shù)量
 _decreaseSize() {
  this._size -= 1
 }
 // 設(shè)置左子結(jié)點,同時維護 parent 關(guān)系
 _setLeft(node: AvlNode<K>, child: AvlNode<K> | null) {
  // 斷開舊 left 結(jié)點
  if (node.left !== null) {
   node.left.parent = null
  }
  // 連接新結(jié)點
  if (child !== null) {
   // 從舊 parent 中斷開
   if (child.parent !== null) {
    child.parent.left === child ? (child.parent.left = null) : (child.parent.right = null)
   }
   child.parent = node
  }
  node.left = child
 }
 // 設(shè)置右子結(jié)點,同時維護 parent 關(guān)系
 _setRight(node: AvlNode<K>, child: AvlNode<K> | null) {
  // 斷開舊 right 結(jié)點
  if (node.right !== null) {
   node.right.parent = null
  }
  // 連接新結(jié)點
  if (child !== null) {
   // 從舊 parent 中斷開
   if (child.parent !== null) {
    child.parent.left === child ? (child.parent.left = null) : (child.parent.right = null)
   }
   child.parent = node
  }
  node.right = child
 }
 // 獲取中序遍歷順序的前驅(qū)結(jié)點
 _inorderPredecessor(node: AvlNode<K> | null) {
  if (node == null) return null
  // 1. 有左子樹,找到左子樹最大元素
  if (node.left !== null) {
   return this._maximumNode(node.left)
  }
  // 2. 沒有左子樹,往上搜索
  let parent = node.parent
  while (parent != null) {
   if (node == parent.right) {
    return parent
   }
   node = parent
   parent = node.parent
  }
  // 4. 搜索到根
  return null
 }
 // 獲取最大的結(jié)點
 _maximumNode(subRoot: AvlNode<K>) {
  let current = subRoot
  while (current.right !== null) current = current.right
  return current
 }
 // 設(shè)置根結(jié)點
 _setRoot(node: AvlNode<K> | null) {
  if (node === null) {
   this._root = null
   return
  }
  this._root = node
  // 如果本身在樹中,則從樹中脫落,成為獨立的樹根
  if (node.parent !== null) {
   node.parent.left === node ? (node.parent.left = null) : (node.parent.right = null)
   node.parent = null
  }
 }
 // 將樹上某個結(jié)點替換成另一個結(jié)點
 private _replaceNode(node: AvlNode<K>, replacer: AvlNode<K> | null) {
  if (node === replacer) return node
  // node 為 root 的情況
  if (node === this._root) {
   this._setRoot(replacer)
  }
  else {
   // 非 root,有父結(jié)點的情況
   const parent = node.parent as AvlNode<K>
   if (parent.left === node) this._setLeft(parent, replacer)
   else this._setRight(parent, replacer)
  }
  return node
 }
 // 左旋,返回新頂點,注意旋轉(zhuǎn)完畢會從原本的樹上脫落
 private _rotateLeft(node: AvlNode<K>) {
  const parent = node.parent
  // 記錄原本在樹上的位置
  const isLeft = parent !== null && parent.left == node
  // 旋轉(zhuǎn)
  const pivot = node.right as AvlNode<K>
  const pivotLeft = pivot.left
  this._setRight(node, pivotLeft)
  this._setLeft(pivot, node)
  // 旋轉(zhuǎn)完畢
  // 新頂點接上樹上原本的位置
  if (parent !== null) {
   if (isLeft) this._setLeft(parent, pivot)
   else this._setRight(parent, pivot)
  }
  // ---
  if (node === this._root) {
   this._setRoot(pivot)
  }
  node.updateHeight()
  pivot.updateHeight()
  return pivot
 }
 // 右旋,返回新頂點,注意旋轉(zhuǎn)完畢會從原本的樹上脫落
 private _rotateRight(node: AvlNode<K>) {
  const parent = node.parent
  // 記錄原本在樹上的位置
  const isLeft = parent !== null && parent.left === node
  // 旋轉(zhuǎn)
  const pivot = node.left as AvlNode<K>
  const pivotRight = pivot.right
  this._setLeft(node, pivotRight)
  this._setRight(pivot, node)
  // 旋轉(zhuǎn)完畢
  // 新頂點接上樹上原本的位置
  if (parent !== null) {
   if (isLeft) this._setLeft(parent, pivot)
   else this._setRight(parent, pivot)
  }
  // ---
  if (node === this._root) {
   this._setRoot(pivot)
  }
  node.updateHeight()
  pivot.updateHeight()
  return pivot
 }
 // 搜索 Node
 private _nodeSearch(key: K) {
  let current = this._root
  while (current !== null) {
   let result = key.compareTo(current.key)
   if (result === 0) return current
   if (result < 0) current = current.left
   else current = current.right
  }
  return null
 }
 // 在樹里插入結(jié)點或者刷新重復(fù)結(jié)點
 // 返回新插入(或刷新)的結(jié)點
 private _nodeInsert(node: AvlNode<K>) {
  // 空樹
  if (this._root === null) {
   this._setRoot(node)
   this._increaseSize()
   return null
  }
  const key = node.key
  let current = this._root
  // 查找待插入的位置
  while (true) {
   const result = key.compareTo(current.key)
   if (result > 0) {
    if (current.right === null) {
     this._setRight(current, node)
     this._increaseSize()
     return current
    }
    current = current.right
   }
   else if (result < 0) {
    if (current.left === null) {
     this._setLeft(current, node)
     this._increaseSize()
     return current
    }
    current = current.left
   }
   else {
    // No duplicates, just update key
    current.key = key
    return null
   }
  }
 }
 // 從樹上移除一個結(jié)點
 private _nodeErase(node: AvlNode<K>) {
  // 同時擁有左右子樹
  // 先轉(zhuǎn)換成只有一顆子樹的情況再統(tǒng)一處理
  if (node.left !== null && node.right !== null) {
   const replacer = this._inorderPredecessor(node)!
   // 使用前驅(qū)結(jié)點替換身份
   // 此時問題轉(zhuǎn)換成刪掉替代結(jié)點(前驅(qū)),
   // 從而簡化成只有一個子結(jié)點的刪除情況
   node.key = replacer.key
   // 修改 node 指針
   node = replacer
  }
  // 刪除點的父結(jié)點
  const parent = node.parent
  // 待刪結(jié)點少于兩顆子樹時,使用子樹 (或 null,沒子樹時) 頂替移除的結(jié)點即可
  const child = node.left || node.right
  this._replaceNode(node, child)
  this._decreaseSize()
  return [ parent, child, node ]
 }
 // AVL 樹插入結(jié)點后調(diào)整動作
 // 自底向上調(diào)整結(jié)點的高度
 // 遇到離 current 最近的不平衡點需要做旋轉(zhuǎn)調(diào)整 
 // 注意: 對最近的不平衡點調(diào)整后 或者 結(jié)點的高度值沒有變化時
 // 上層結(jié)點便不需要更新 
 // 調(diào)整次數(shù)不大于1
 _adjustAfterInsertion(backtracking: AvlNode<K> | null) {
  let current = backtracking
  // 往上回溯,查找最近的不平衡結(jié)點
  while (current !== null) {
   // 更新高度
   current.updateHeight()
   // 插入前后,回溯途徑結(jié)點的高度沒有變化,則無需繼續(xù)回溯調(diào)整
   if (current.height === current.prevHeight) break
   // 若找到不平衡結(jié)點,執(zhí)行一次調(diào)整即可
   if (this._isUnbalanced(current)) {
    this._rebalance(current)
    // 調(diào)整過,則上層無需再調(diào)整了
    break
   }
   current = current.parent
  }
 }
 // AVL樹刪除結(jié)點后調(diào)整動作
 // 自底向上調(diào)整結(jié)點的高度
 // 遇到離 current 最近的不平衡點需要做旋轉(zhuǎn)調(diào)整
 // 注意: 對最近的不平衡點調(diào)整后,其上層結(jié)點仍然可能需要調(diào)整
 // 調(diào)整次數(shù)可能不止一次
 _adjustAfterRemoval(backtracking: AvlNode<K> | null) {
  let current = backtracking
  while (current !== null) {
   // 更新高度
   current.updateHeight()
   // 刪除前后,回溯途徑結(jié)點的高度沒有變化,則無需繼續(xù)回溯調(diào)整
   if (current.height === current.prevHeight) break
   if (this._isUnbalanced(current)) {
    this._rebalance(current)
   }
   // 與插入不同,調(diào)整過后,仍然需要繼續(xù)往上回溯
   // 上層結(jié)點(若有)仍需判斷是否需要調(diào)整
   current = current.parent
  }
 }
 // LL
 _adjustLeftLeft(node: AvlNode<K>) {
  return this._rotateRight(node)
 }
 // RR
 _adjustRightRight(node: AvlNode<K>) {
  return this._rotateLeft(node)
 }
 // LR
 _adjustLeftRight(node: AvlNode<K>) {
  this._rotateLeft(node.left!)
  return this._rotateRight(node)
 }
 // RL
 _adjustRightLeft(node: AvlNode<K>) {
  this._rotateRight(node.right!)
  return this._rotateLeft(node)
 }
 // 檢查結(jié)點是否平衡
 _isUnbalanced(node: AvlNode<K>) {
  const factor = node.balanceFactor
  return factor === UNBALANCED_RIGHT || factor === UNBALANCED_LEFT
 }
 // 重新平衡
 _rebalance(node: AvlNode<K>) {
  const factor = node.balanceFactor
  // Right subtree longer (node.factor: -2)
  if (factor === UNBALANCED_RIGHT) {
   let right = node.right!
   // RL, node.right.factor: 1
   if (right.balanceFactor === SLIGHTLY_UNBALANCED_LEFT) {
    return this._adjustRightLeft(node)
   }
   else {
    // RR, node.right.factor: 0|-1
    // 即 right.rightHeight >= right.leftHeight
    return this._adjustRightRight(node)
   }
  }
  else if (factor === UNBALANCED_LEFT) {
   // Left subtree longer (node.factor: 2)
   let left = node.left!
   // LR, node.left.factor: -1
   if (left.balanceFactor === SLIGHTLY_UNBALANCED_RIGHT) {
    return this._adjustLeftRight(node)
   }
   else {
    // LL, node.left.factor: 1 | 0
    // 即 left.leftHeight >= left.rightHeight
    return this._adjustLeftLeft(node)
   }
  }
  return node
 }
}
export function createAvl() {
 return new Avl()
}
// SparseRangeList.ts
import { createAvl, Avl } from './Avl'
// 區(qū)間類
class Range {
 start: number
 end: number
 constructor(start: number, end?: number) {
  this.start = start
  this.end = end || start
 }
 // 用于 Avl 中節(jié)點的比較
 //
 // 列表中項目范圍是連續(xù)的,必定不會重疊的
 // 如果傳入的 key 為重疊的,則意味著希望通過構(gòu)造一個子 Range 搜索所在的 RangeValue
 // 例如構(gòu)造一個 { start: 10, end: 10, value: 'any' },搜索樹中
 // 范圍包含 10~10 的 RangeValue,如 { start: 0, end: 20, value: 'any' }
 compareTo(other: Range) {
  if (other.start > this.end!) return -1
  if (other.end! < this.start) return 1
  return 0
 }
}
// 區(qū)間-值 類
class RangeValue<T> extends Range {
 value: T
 constructor(start: number, end: number, value: T) {
  super(start, end)
  this.value = value
 }
 clone(): RangeValue<T> {
  return new RangeValue(this.start, this.end!, this.value)
 }
}
// 最終存儲區(qū)間-值的類,內(nèi)部使用 Avl 存儲所有 RangeValue
export default class SparseRangeList<T> {
 _size: number
 defaultValue: T
 valueTree: Avl
 constructor(size: number, defaultValue: T) {
  this._size = size
  this.defaultValue = defaultValue
  this.valueTree = createAvl()
 }
 get size() {
  return this._size
 }
 resize(newSize: number) {
  newSize = newSize | 0
  // 無調(diào)整
  if (this._size === newSize) return
  // 擴容
  if (this._size < newSize) {
   this._size = newSize
   return
  }
  // 縮小,清空超出的部分,再縮小
  this.setRangeValue(newSize - 1, this._size - 1, this.defaultValue)
  this._size = newSize
 }
 // 返回區(qū)間包含 index 的 RangeValue 的值
 getValueAt(index: number): T {
  const result = this.valueTree.search(new Range(index))
  if (result) return result.value
  return this.defaultValue
 } 
 /**
  * 設(shè)值方法,
  * 自動與相鄰的相同值的合并成更大的 RangeValue,
  * 導(dǎo)致原本的 RangeValue 不連續(xù),則會
  * 自動切分成兩個或者三個 RangeValue。
  *
  *     a-------------a
  * |a------------|b------------|c-----------|...
  * 
  * 結(jié)果:
  * |a-------------------|b-----|c-----------|...
  * 
  * 
  *     d-------------d
  * |a------------|b------------|c-----------|...
  * 
  * 結(jié)果:
  * |a-----|d------------|b-----|c-----------|...
  * 
  */
 setRangeValue(start: number, end: number, value: T) {
  if (!this.size) return
  if (end >= this.size) end = this.size - 1
  // 所有與當前傳入?yún)^(qū)間范圍有重疊部分,
  // -1,+1 將接壤的毗鄰 RangeValue 也納入(如果存在的話),
  // 毗鄰的 RangeValue 要檢查否要合并。
  let prevSiblingEnd = start - 1
  let nextSiblingStart = end + 1
  let rangeValues = this.treeIntersecting(prevSiblingEnd, nextSiblingStart)
  // 如果沒有重疊的部分,則作為新的 RangeValue 插入,直接結(jié)束
  // 如果有重疊的部分,就要處理合并、拆分
  if (rangeValues.length) {
   let firstRange = rangeValues[0]
   let lastRange = rangeValues[rangeValues.length - 1]
   // end 邊界比傳入的 start 小,說明是接壤毗鄰的更小的 RangeValue
   //
   // 1. 如果毗鄰的 RangeValue 的值跟當前帶插入的值不一致,
   // 則直接將毗鄰的 RangeValue 從列表中移除,
   // 不需要做任何特殊操作,正常的插入操作即可
   //
   // 2. 否則如果毗鄰的 RangeValue 的值跟當前待插入的值一致,
   // 則將兩個 RangeValue 的 Range 合并(修改 start即可),
   // 然后這個毗鄰的 RangeValue 也自然變成重疊的,正常執(zhí)行后續(xù)
   // 的重疊處理邏輯即可(拆分)
   if (firstRange.end < start) {
    if (firstRange.value !== value) {
     rangeValues.shift()
    }
    else {
     start = firstRange.start
    }
   }
   // 接壤毗鄰的更大的 RangeValue,處理思路
   // 跟上面處理毗鄰的更小的 RangeValue 一樣的
   if (lastRange.start > end) {
    if (lastRange.value !== value) {
     rangeValues.pop()
    }
    else {
     end = lastRange.end
    }
   }
   // 結(jié)束毗鄰 RangeValue 合并邏輯
   // 開始處理相交的 RangeValue 流程
   const length = rangeValues.length
   let index = 0
   while (index < length) {
    const currentRangeValue = rangeValues[index]
    const { value: currentValue, start: currentStart, end: currentEnd } = currentRangeValue
    // 先移除掉該重疊的 RangeValue,然后:
    this.valueTree.delete(currentRangeValue)
    // Case 1. 如果是當前 RangeValue 完整包含在傳入的范圍內(nèi),
    // 則不需要處理,因為整個范圍都將被傳入的值覆蓋。
    if (currentStart >= start && currentEnd <= end) {
     index += 1
     continue
    }
    // Case2. 部分相交,該 RangeValue 的大的一側(cè)在傳入的范圍內(nèi),而小的一側(cè)不在。
    // 需要做切分操作,以重疊的位置作為切分點,比較小的一側(cè)(不重疊的部分)重新插入,
    // 比較大的的那一部分,會被傳入的值覆蓋掉
    if (currentStart < start) {
     // 如果值不一樣,則以相交的位置作為切分點,非重疊部分重新插入,重疊部分用待插入的值覆蓋。
     if (currentValue !== value) {
      this._insert(currentStart, start - 1, currentValue)
     }
     else {
      start = currentStart
     }
    }
    // Case3. 部分相交,該 RangeValue 的小的一側(cè)在傳入的范圍內(nèi),而大的一側(cè)不在。
    // 同 Case 2 做切分操作,只是反向。
    if (currentEnd > end) {
     if (currentValue !== value) {
      this._insert(end + 1, currentEnd, currentValue)
     }
     else {
      end = currentEnd
     }
    }
    index += 1
   }
  }
  this._insert(start, end, value)
 }
 setValue(index: number, value: T) {
  this.setRangeValue(index, index, value)
 }
 /**
  * 篩選出與指定區(qū)間有重疊的 RangeValue,即:
  * 
  *       1. 相互部分重疊
  * 
  * o----------o       o---------o
  *   >start------------------end<
  * 
  * 
  *      2. 相互完全重疊關(guān)系
  * 
  *      o----------------o
  *      >start--------end<
  * 
  * 
  *      3. 包含或被包含關(guān)系
  * 
  * o--------------------------------------o
  *  o-------------------------------o
  *   o-------------------------------o
  *   o-----o   o-----o   o----o
  *   >start--------------------end<
  * 
  */
 treeIntersecting(start: number, end: number): RangeValue[] {
  const startRange = new Range(start)
  const endRange = new Range(end)
  return this.valueTree.searchRange(startRange, endRange)
 }
 /**
  * 返回指定范圍內(nèi)所有 RangeValue
  * 范圍內(nèi)有無值的 Range 的話,則使用
  * 攜帶默認值的 RangeValue 代替
  * 從而確保返回的結(jié)果是線性的、每個區(qū)間都有值的,如:
  * 
  * start>...<end 范圍內(nèi)有 A、B 兩個 RangeValue,所有空洞都用 Default 補足
  * +-----------|-----|-----------|-----|-----------+
  * | Default | A | Default | B | Default |
  * >start------|-----|-----------|-----|--------end<
  *
  */
 intersecting(start: number, end: number): RangeValue[] {
  const ranges = this.treeIntersecting(start, end)
  if (!ranges.length) {
   if (!this.size) return []
   return [ new RangeValue(start, end, this.defaultValue) ]
  }
  let result = []
  let range
  let index = 0
  let length = ranges.length
  while (index < length) {
   range = ranges[index].clone()
   // 傳入的 (start, end) 右側(cè)與某個 RangeValue 重疊,
   // 左側(cè)沒有命中,則左側(cè)區(qū)域手動塞入一個攜帶默認
   // 值的 RangeValue
   if (range.start > start) {
    result.push(new RangeValue(start, range.start - 1, this.defaultValue))
   }
   result.push(range)
   // 將 start 的位置右移,
   // 以便下個 range 的比較
   start = range.end + 1
   index += 1
  }
  // 如果最后一個 range,與傳入的范圍只有左側(cè)重疊,
  // 而右側(cè)沒有重疊的地方,則手動塞入一個攜帶默認值
  // 的 RangeValue
  if (range.end < end) {
   result.push(new RangeValue(range.end + 1, end, this.defaultValue))
  }
  else if (range.end > end) {
   // 否則如果最后一個 range 的范圍已經(jīng)超出需要的范圍,則裁剪
   range.end = end
  }
  return result
 }
 values() {
  if (!this.size) return []
  return this.intersecting(0, this.size - 1)
 }
 _insert(start: number, end: number, value: T) {
  if (value !== this.defaultValue) {
   const rangeValue = new RangeValue(start, end, value)
   this.valueTree.insert(rangeValue)
  }
 }
}
export function create<T>(size: number, value: T) {
 return new SparseRangeList(size, value)
}

有了這套存儲機制之后,我們就可以更高效地管理列表項的高度,和統(tǒng)計列表高度了。

看代碼理解:

import { create as createSparseRangeList } from './SparseRangeList'
// 創(chuàng)建一個默認預(yù)估高度為 20 的列表項存儲對象
const itemHeightStore = createSparseRangeList(wrappedItems.length, 20)
// 設(shè)置第二項為 40px
itemHeightStore.setValue(1, 40)
// 獲取第二項的高度
itemHeightStore.getValueAt(1) // 40
// 獲取列表項的 top 坐標
const top = (index: number): number => {
 if (index === 0) return 0
 // 0 ~ 上一項的高度累加
 const rangeValues = itemHeightStore.intersecting(0, index - 1)
 const sumHeight = rangeValues.reduce((sum: number, rangeValue: any) => {
  const span = rangeValue.end - rangeValue.start + 1
  return sum + rangeValue.value * span
 }, 0)
 return sumHeight
}
top(1) // 20
// 計算列表總高度:
const listHeight = itemHeightStore
 .values()
 .reduce((acc: number, rangeValue: any) => {
  const span = rangeValue.end - rangeValue.start + 1
  const height = rangeValue.value * span
  return acc + height
 }, 0)

計算可視條目

完成了列表項高度的管理,接下來需要解決的重點,就是計算出哪些條目是可視的。

最簡單的實現(xiàn)方式,就是直接遍歷我們的結(jié)點高度存儲列表,逐個去跟視口的坐標區(qū)間比較,過濾出落在(或部分落在)視口內(nèi)部的條目。 基于性能考慮,我們當然不能這么簡單粗暴。我們可以做以下嘗試來提高性能:

一、預(yù)估起點條目 + 二分法修正。

通過條目的預(yù)估高度或默認高度,算出可能出現(xiàn)在視口的第一條條目。 比如,我們視口上沿坐標(即滾動條滾過的距離)為 100px,我們條目預(yù)估高度為 20px,那么,我們可以猜測第一個出現(xiàn)在視口中的條目為 100 / 20 + 1,即第 6 條。 我們直接計算第 6 條的坐標,檢查是否落在視口中,根據(jù)結(jié)果差距,再進行二分法猜測,直到找到真正的起點條目。

二、預(yù)估終點條目 + 二分法修正

在算出起點條目后,在使用視口高度除以預(yù)估條目高度,算出視口內(nèi)部可能顯示多少項,將起點序號加上這個數(shù)量,就是預(yù)估的終點條目序號。使用上述一樣的修正邏輯,直到找到正確的視口終點條目。

描述可能比較難以理解,下面給出關(guān)鍵片段:

// 內(nèi)部方法,計算局部渲染數(shù)據(jù)切片的起止點
private _calcSliceRange() {
 if (!this.dataView.length) {
  return { sliceFrom: 0, sliceTo: 0 }
 }
 // 數(shù)據(jù)總量
 const MAX = this.dataView.length
 // 視口上邊界
 const viewportTop = (this.$refs.viewport as any).scrollTop || 0
 // 視口下邊界
 const viewportBottom = viewportTop + this.viewportHeight
 // 預(yù)估條目高度
 const estimatedItemHeight = this.defaultItemHeight
 // 從估算值開始計算起始序號
 let sliceFrom = Math.floor(viewportTop / estimatedItemHeight!)
 if (sliceFrom > MAX - 1) sliceFrom = MAX - 1
 while (sliceFrom >= 0 && sliceFrom <= MAX - 1) {
  const itemTop = this._top(sliceFrom)
  // 條目頂部相對于 viewport 頂部的偏移
  const itemOffset = itemTop - viewportTop
  // 1. 該條目距離視口頂部有距離,說明上方還有條目元素需要顯示,繼續(xù)測試上一條
  if (itemOffset > 0) {
   // 二分法快速估算下一個嘗試位置
   const diff = itemOffset / estimatedItemHeight!
   sliceFrom -= Math.ceil(diff / 2)
   continue
  }
  // 2. 恰好顯示該條目的頂部,則該條目為本次視口的首條元素
  if (itemOffset === 0) break
  // 以下都是 itemOffset < 0
  const itemHeight = this._itemHeight(sliceFrom)
  // 3. 該條目在頂部露出了一部分,則該條目為本次視口的首條元素
  if (itemOffset < itemHeight) break
  // 4. 該條目已被滾出去視口,繼續(xù)測試下一條
  // 二分法快速估算下一個嘗試位置
  const diff = -itemOffset / estimatedItemHeight!
  sliceFrom += Math.ceil(diff / 2)
 }
 // 從估算值開始計算結(jié)束序號
 let sliceTo = sliceFrom + 1 + Math.floor(this.viewportHeight / estimatedItemHeight!)
 if (sliceTo > MAX) sliceTo = MAX
 while (sliceTo > sliceFrom && sliceTo <= MAX) {
  const itemTop = this._top(sliceTo)
  const itemHeight = this._itemHeight(sliceTo)
  const itemBottom = itemTop + itemHeight
  // 條目底部相對于 viewport 底部的偏移
  const itemOffset = itemBottom - viewportBottom
  // 1. 該條目的底部距離視口底部有距離,說明下方還有條目元素需要顯示,繼續(xù)測試下一條
  if (itemOffset < 0) {
   // 二分法快速估算下一個嘗試位置
   const diff = -itemOffset / estimatedItemHeight!
   sliceTo += Math.ceil(diff / 2)
   continue
  }
  // 2. 恰好顯示該條目的底部,則該條目為視口中最后一項
  if (itemOffset === 0) break
  // 3. 該條目在底部被裁剪了一部分,則該條目為本次視口的末項
  if (itemOffset < itemHeight) break
  // 該條目還未出場,繼續(xù)測試上一條
  // 二分法快速估算下一個嘗試位置
  const diff = itemOffset / estimatedItemHeight!
  sliceTo -= Math.ceil(diff / 2)
 }
 // slice 的時候,不含 end,所以 + 1
 sliceTo += 1
 return { sliceFrom, sliceTo }
}

以上就是計算可視區(qū)域的核心部分。完整的代碼,會在后續(xù)給出。

DOM 更新
由于我們是使用 Vue 來實現(xiàn)虛擬列表的,所以 DOM 的更新方面,可以省去大量繁瑣的細節(jié)管理。 我們只需要關(guān)心列表滾動到某處之后,如何計算出當前視口應(yīng)該出現(xiàn)哪些條目即可。

盡管如此,考慮到滾動的流暢性,以及 IE11 等瀏覽器的 DOM 操作性能,我們不得不多做很多事情。

批量 DOM 操作

我們可以在 IE11 的開發(fā)者工具面板中看到,滾動過程,頻繁地往虛擬列表首尾插入、移除結(jié)點,會帶來非常嚴重的性能問題。 所以,我們必須控制 DOM 操作的頻率。

批量可以部分解決這個問題。

具體的思路是,在滾動回調(diào)中,我們計算出可視區(qū)域的結(jié)點起止序號,不直接應(yīng)用,而是加上一個額外渲染的數(shù)量。 比如我們計算出當前應(yīng)該渲染 20 ~ 30 這些條目,我們可以在前后各加上 10 個額外渲染的條目,即 10 ~ 40,這樣就一次性渲染了 30 個結(jié)點。在繼續(xù)滾動時,我們檢查新的起止范圍,是否還在 10 ~ 40 范圍內(nèi),如果是,我們就不做新的結(jié)點增刪操作。

核心實現(xiàn):

// 刷新局部渲染數(shù)據(jù)切片范圍
private _updateSliceRange(forceUpdate?: boolean) {
 // 上下方額外多渲染的條目波動量
 const COUNT = this._preRenderingCount()

 // 預(yù)渲染觸發(fā)閾值
 const THRESHOLD = this._preRenderingThreshold()  

 // 數(shù)據(jù)總量
 const MAX = this.dataView.length

 // 計算出準確的切片區(qū)間
 const range = this._calcSliceRange()  

 // 檢查計算出來的切片范圍,是否被當前已經(jīng)渲染的切片返回包含了
 // 如果是,無需更新切片,(如果 forceUpdate,則無論如何都需要重新切片)
 let fromThreshold = range.sliceFrom - THRESHOLD
 if (fromThreshold < 0) fromThreshold = 0
 let toThreshold = range.sliceTo + THRESHOLD
 if (toThreshold > MAX) toThreshold = MAX

 // 無需強制刷新,且上下兩端都沒有觸達閾值時,無需重新切片
 if (!forceUpdate && ((this.sliceFrom <= fromThreshold) && (this.sliceTo >= toThreshold))) {
  return
 }

 // 下面是更新切片的情況

 // 在切片區(qū)間頭部、尾部,追加預(yù)渲染的條目
 let { sliceFrom, sliceTo } = range
 sliceFrom = sliceFrom > COUNT ? sliceFrom - COUNT : 0
 sliceTo = sliceTo + COUNT > MAX ? MAX : sliceTo + COUNT

 this.sliceFrom = sliceFrom
 this.sliceTo = sliceTo
 if (forceUpdate) this._doSlice()
}

使用了這種批量操作之后,可以看到,正常的鼠標滾動下,IE 也能比較順暢地滾動了。

事件

由于虛擬列表的 DOM 需要不停地生成和銷毀,因此,直接在列表項目上綁定事件是非常低效的。 所以,使用事件代理就成了很不錯的方案,將事件注冊在組件根結(jié)點上,再根據(jù) event.target 來區(qū)分是由哪個列表項冒泡出來的事件,即可高效處理。

組件實現(xiàn)

import { Component, Vue, Prop, Watch } from 'vue-property-decorator'
import { createSparseRangeList } from './SparseRangeList'
// 列表項數(shù)據(jù)包裹,data 字段存放原始數(shù)據(jù)
// 組件所有操作不應(yīng)該改變 data 的內(nèi)容,而是修改該包裹對象的屬性
class ItemWrapper {
 // 原始數(shù)據(jù)
 data: any
 // 數(shù)據(jù)唯一 key
 key: any
 // 條目高度
 // 1. 正數(shù)代表已經(jīng)計算出來的高度
 // 2. 0 代表未計算的高度,不顯示
 // 3. 負數(shù)代表需要隱藏的高度,絕對值為已經(jīng)計算出來的高度,方便取消隱藏
 height: number
 // 記錄是否已經(jīng)根據(jù)實際 DOM 計算過高度
 realHeight: boolean
 // 條目在當前過濾視圖中的序號
 viewIndex: number
 constructor(data: any, key: any, height: number) {
  this.data = data
  // 數(shù)據(jù)的唯一id,是初始化數(shù)據(jù)時候的序號
  // 每次傳入的 data 改變,都會重新生成
  this.key = key
  // 條目的高度緩存
  // 1. 用于重建高度存儲時快速恢復(fù)
  // 2. 用于快速通過數(shù)據(jù)取高度
  this.height = height >> 0
  this.realHeight = false
  // 每次生成 dataView 都刷新
  this.viewIndex = -1
 }
}
@Component({ name: 'VList' })
export default class VList extends Vue {
 [key: string]: any
 // 高度存儲 不響應(yīng)式
 private itemHeightStore: any
 // 組件寬度,不設(shè)置則為容器的 100%
 @Prop({ type: Number })
 private width?: number
 // 組件高度,不設(shè)置則為容器的 100%
 @Prop({ type: Number })
 private height?: number
 // 傳入高度值,固定條目高度
 @Prop({ type: Number })
 private fixedItemHeight?: number
 // 預(yù)估元素高度,
 // 在高度不確定的列表中,未計算出高度時使用,
 // 該值與元素平均高度越相近,則越高效(修正時估算次數(shù)越少)
 @Prop({ type: Number, default: 30 })
 private estimatedItemHeight!: number
 // 數(shù)據(jù)列表
 @Prop({ type: Array, default: () => ([]) })
 private data!: any[]
 // 計算條目高度的方法
 @Prop({
  type: Function,
  default(node: Node, wrappedData: ItemWrapper) {
   return (node as HTMLElement).clientHeight
  }
 })
 private itemHeightMethod!: (node: Node, wrappedItem: ItemWrapper) => number
 // 數(shù)據(jù)過濾方法(可以用于外部實現(xiàn)搜索框過濾)
 @Prop({ type: Function })
 private filterMethod?: (data: any) => boolean
 // 數(shù)據(jù)排序方法(可以用于外部實現(xiàn)數(shù)據(jù)自定義過濾)
 @Prop({ type: Function })
 private sortMethod?: (a: any, b: any) => number
 // 包裹后的數(shù)據(jù)列表(必須 freeze,否則大列表性能撐不?。?
 private wrappedData: ReadonlyArray<ItemWrapper> = Object.freeze(this._wrapData(this.data))
 // 真實渲染上屏的數(shù)據(jù)列表切片
 private dataSlice: ReadonlyArray<ItemWrapper> = []
 // viewport 寬度
 private viewportWidth = this.width || 0
 // viewport 高度
 private viewportHeight = this.height || 0
 // 當前 viewport 中第一條數(shù)據(jù)的序號
 private sliceFrom = 0
 // 當前 viewport 中最后一條數(shù)據(jù)的序號
 private sliceTo = 0
 // 列表高度
 private listHeight = 0
 // 檢查是否固定高度模式
 private get isFixedHeight() {
  return this.fixedItemHeight! >= 0
 }
 // 獲取默認條目高度
 private get defaultItemHeight() {
  return this.isFixedHeight ? this.fixedItemHeight! : this.estimatedItemHeight
 }
 // 當前篩選條件下的數(shù)據(jù)列表
 // 依賴:wrappedData, filterMethod, sortMethod
 private get dataView() {
  const { wrappedData, filterMethod, sortMethod } = this
  let data = []
  if (typeof filterMethod === 'function') {
   const len = wrappedData.length
   for (let index = 0; index < len; index += 1) {
    const item = wrappedData[index]
    if (filterMethod(item.data)) {
     data.push(item)
    }
   }
  } else {
   data = wrappedData.map(i => i)
  }
  if (typeof sortMethod === 'function') {
   data.sort((a, b) => {
    return sortMethod(a, b)
   })
  }
  // 重新記錄數(shù)據(jù)在視圖中的位置,用于隱藏部分條目時,可以精確計算高度、坐標
  const size = data.length
  for (let index = 0; index < size; index += 1) {
   const wrappedItem = data[index]
   wrappedItem.viewIndex = index
  }
  return Object.freeze(data)
 }
 // 原始列表數(shù)據(jù)變化,重新包裹數(shù)據(jù)
 @Watch('data')
 private onDataChange(data: any[]) {
  this.wrappedData = Object.freeze(this._wrapData(data))
 }
 // 當前過濾、排序視圖變化,重新布局
 @Watch('dataView')
 private onDataViewChange(wrappedItems: ItemWrapper[]) {
  // 重建高度存儲
  const estimatedItemHeight = this.defaultItemHeight
  this.itemHeightStore = createSparseRangeList(wrappedItems.length, estimatedItemHeight)
  // 從緩存中快速恢復(fù)已計算出高度的條目的高度
  wrappedItems.forEach((wrappedItem, index) => {
   // 小于零的需要隱藏,所以高度為 0
   this.itemHeightStore.setValue(index,
    wrappedItem.height > 0 ? wrappedItem.height : 0)
  })
  // 刷新列表高度
  this.updateListHeight()
  // 重置滾動位置
  // TODO, 錨定元素
  const { viewport } = this.$refs as any
  if (viewport) viewport.scrollTop = 0
  // 重新切片當前 viewport 需要的數(shù)據(jù)
  this._updateSliceRange(true)
  this.$emit('data-view-change', this.dataSlice.map((wrappedItem) => wrappedItem.data))
 }
 private created() {
  const estimatedItemHeight = this.defaultItemHeight
  this.itemHeightStore = createSparseRangeList(this.dataView.length, estimatedItemHeight)
  this.layoutObserver = new MutationObserver(this.redraw.bind(this))
  this.childObserver = new MutationObserver((mutations: MutationRecord[]) => {
   this._updateHeightWhenItemInserted(mutations)
  })
  this.$watch(((vm: any) => `${vm.sliceFrom},${vm.sliceTo}`) as any, this._doSlice)
 }
 private mounted() {
  this.redraw()
  this.layoutObserver.observe(this.$el, { attributes: true })
  // 非固定高度場景,監(jiān)聽子元素插入,提取高度
  if (!this.isFixedHeight) {
   this.childObserver.observe(this.$refs.content, { childList: true })
  }
 }
 private beforeDestory() {
  this.layoutObserver.disconnect()
  if (!this.isFixedHeight) {
   this.childObserver.disconnect()
  }
  this.itemHeightStore = null
 }
 // DOM 結(jié)構(gòu)比較簡單,無需 template,直接使用渲染函數(shù)輸出 VDOM
 private render(createElement: any) {
  return createElement(
   'div', // 組件容器,與外部布局
   {
    class: 'VList',
    style: {
     'box-sizing': 'border-box',
     display: 'inline-block',
     margin: '0',
     padding: '0',
     width: this.width ? this.width + 'px' : '100%',
     height: this.height ? this.height + 'px' : '100%',
    }
   },
   [
    createElement(
     'div', // 滾動區(qū)域的可見范圍
     {
      ref: 'viewport',
      class: 'VList_viewport',
      style:
       'box-sizing:border-box;position:relative;overflow:hidden;width:100%;height:100%;margin:0;padding:0;overflow:auto;overflow-scrolling:touch;',
      on: { scroll: this._onScroll }
     },
     [
      createElement(
       'div', // 內(nèi)容容器,內(nèi)容真實高度由此容器體現(xiàn)
       {
        class: 'VList_scollable',
        ref: 'content',
        style: {
         'box-sizing': 'border-box',
         position: 'relative',
         margin: '0',
         padding: '0',
         height: this.listHeight + 'px'
        }
       },
       // 列表項
       this.dataSlice.map((wrappedItem) => {
        return createElement(
         'div',
         {
          key: wrappedItem.key,
          class: `VList_item VList_item-${wrappedItem.key % 2 === 0 ? 'even' : 'odd'}`,
          attrs: {
           'data-key': wrappedItem.key
          },
          style: {
           'box-sizing': 'border-box',
           'z-index': '1',
           position: 'absolute',
           right: '0',
           bottom: 'auto',
           left: '0',
           margin: '0',
           padding: '0',
           cursor: 'default',
           // 注:使用 transfrom 有黑屏 bug
           // transform: `translate(0, ${top})`
           // transform: `translate3d(0, ${top}, 0)`
           top: this._top(wrappedItem.viewIndex) + 'px'
          }
         },
         // 將原始數(shù)據(jù),key 注入到 slot 里,
         // 以便自定義條目內(nèi)容使用
         this.$scopedSlots.default!({
          item: wrappedItem.data,
          listKey: wrappedItem.key
         })
        )
       })
      )
     ]
    )
   ]
  )
 }
 // 重繪界面,確保列表渲染正確
 public redraw() {
  const viewport = this.$refs.viewport as HTMLElement
  const { clientWidth, clientHeight } = viewport
  this.viewportWidth = clientWidth
  this.viewportHeight = clientHeight
  this.updateListHeight()
  this._updateSliceRange(true)
 }
 // 刷新列表總高度
 public updateListHeight() {
  const { itemHeightStore } = this
  const rangeValues = itemHeightStore.values()
  if (!rangeValues.length) {
   this.listHeight = 0
   return
  }
  const listHeight = rangeValues.reduce((sum: number, rangeValue: any) => {
   const span = rangeValue.end - rangeValue.start + 1
   const height = rangeValue.value * span
   return sum + height
  }, 0)
  this.listHeight = listHeight
 }
 // Dom 插入時候,計算高度,然后
 // 批量刷新高度,避免頻繁調(diào)整列表高度帶來性能問題
 public batchUpdateHeight(records: Array<{ wrappedItem: ItemWrapper, height: number }>) {
  records.forEach(({ wrappedItem, height }) => {
   this._updateHeight(wrappedItem, height, true)
  })
  this.updateListHeight()
  this._updateSliceRange()
 }
 // 通過數(shù)據(jù) key,設(shè)置對應(yīng)條目的高度
 public updateHeightByKey(key: any, height: number) {
  const wrappedItem = this.wrappedData[key]
  if (!wrappedItem) return
  this._updateHeight(wrappedItem, height)
  this.updateListHeight()
  this._updateSliceRange()
 }
 // 通過數(shù)據(jù) key,設(shè)置對應(yīng)條目的顯示狀態(tài)
 public showByKey(key: any) {
  const wrappedItem = this.wrappedData[key]
  if (!wrappedItem) return
  if (wrappedItem.height <= 0) {
   const height = -wrappedItem.height || this.defaultItemHeight
   this._updateHeight(wrappedItem, height!)
   this.updateListHeight()
   this._updateSliceRange()
   // 強制重繪
   this._doSlice()
  }
 }
 // 通過數(shù)據(jù) key,設(shè)置對應(yīng)條目的顯示狀態(tài)
 public hideByKey(key: any) {
  const wrappedItem = this.wrappedData[key]
  if (!wrappedItem) return
  if (wrappedItem.height > 0) {
   const height = -wrappedItem.height
   wrappedItem.height = height
   this._updateHeight(wrappedItem, height)
   this.updateListHeight()
   // 強制重繪
   this._updateSliceRange(true)
  }
 }
 // 通過數(shù)據(jù) key 列表,設(shè)置對應(yīng)條目的顯示狀態(tài)
 public showByKeys(keys: any[]) {
  const wrappedItems = keys.map((key) => this.wrappedData[key])
   .filter((wrappedItem) => wrappedItem && wrappedItem.height <= 0)
  wrappedItems.forEach((wrappedItem) => {
   const height = (-wrappedItem.height || this.defaultItemHeight)!
   this._updateHeight(wrappedItem, height)
  })
  this.updateListHeight()
  // 強制重繪
  this._updateSliceRange(true)
 }
 // 通過數(shù)據(jù) key 列表,設(shè)置對應(yīng)條目的顯示狀態(tài)
 public hideByKeys(keys: any[]) {
  const wrappedItems = keys.map((key) => this.wrappedData[key])
   .filter(wrappedItem => wrappedItem && wrappedItem.height > 0)
  wrappedItems.forEach((wrappedItem) => {
   // 設(shè)置為負數(shù),表示隱藏
   const height = -wrappedItem.height
   wrappedItem.height = height
   this._updateHeight(wrappedItem, height)
  })
  this.updateListHeight()
  // 強制重繪
  this._updateSliceRange(true)
 }
 // 內(nèi)部方法,計算局部渲染數(shù)據(jù)切片的起止點
 private _calcSliceRange() {
  if (!this.dataView.length) {
   return { sliceFrom: 0, sliceTo: 0 }
  }
  // 數(shù)據(jù)總量
  const MAX = this.dataView.length
  // 視口上邊界
  const viewportTop = (this.$refs.viewport as any).scrollTop || 0
  // 視口下邊界
  const viewportBottom = viewportTop + this.viewportHeight
  // 預(yù)估條目高度
  const estimatedItemHeight = this.defaultItemHeight
  // 從估算值開始計算起始序號
  let sliceFrom = Math.floor(viewportTop / estimatedItemHeight!)
  if (sliceFrom > MAX - 1) sliceFrom = MAX - 1
  while (sliceFrom >= 0 && sliceFrom <= MAX - 1) {
   const itemTop = this._top(sliceFrom)
   // 條目頂部相對于 viewport 頂部的偏移
   const itemOffset = itemTop - viewportTop
   // 1. 該條目距離視口頂部有距離,說明上方還有條目元素需要顯示,繼續(xù)測試上一條
   if (itemOffset > 0) {
    // 二分法快速估算下一個嘗試位置
    const diff = itemOffset / estimatedItemHeight!
    sliceFrom -= Math.ceil(diff / 2)
    continue
   }
   // 2. 恰好顯示該條目的頂部,則該條目為本次視口的首條元素
   if (itemOffset === 0) break
   // 以下都是 itemOffset < 0
   const itemHeight = this._itemHeight(sliceFrom)
   // 3. 該條目在頂部露出了一部分,則該條目為本次視口的首條元素
   if (itemOffset < itemHeight) break
   // 4. 該條目已被滾出去視口,繼續(xù)測試下一條
   // 二分法快速估算下一個嘗試位置
   const diff = -itemOffset / estimatedItemHeight!
   sliceFrom += Math.ceil(diff / 2)
  }
  // 從估算值開始計算結(jié)束序號
  let sliceTo = sliceFrom + 1 + Math.floor(this.viewportHeight / estimatedItemHeight!)
  if (sliceTo > MAX) sliceTo = MAX
  while (sliceTo > sliceFrom && sliceTo <= MAX) {
   const itemTop = this._top(sliceTo)
   const itemHeight = this._itemHeight(sliceTo)
   const itemBottom = itemTop + itemHeight
   // 條目底部相對于 viewport 底部的偏移
   const itemOffset = itemBottom - viewportBottom
   // 1. 該條目的底部距離視口底部有距離,說明下方還有條目元素需要顯示,繼續(xù)測試下一條
   if (itemOffset < 0) {
    // 二分法快速估算下一個嘗試位置
    const diff = -itemOffset / estimatedItemHeight!
    sliceTo += Math.ceil(diff / 2)
    continue
   }
   // 2. 恰好顯示該條目的底部,則該條目為視口中最后一項
   if (itemOffset === 0) break
   // 3. 該條目在底部被裁剪了一部分,則該條目為本次視口的末項
   if (itemOffset < itemHeight) break
   // 該條目還未出場,繼續(xù)測試上一條
   // 二分法快速估算下一個嘗試位置
   const diff = itemOffset / estimatedItemHeight!
   sliceTo -= Math.ceil(diff / 2)
  }
  // slice 的時候,不含 end,所以 + 1
  sliceTo += 1
  return { sliceFrom, sliceTo }
 }
 // 上下兩端預(yù)先批量渲染的項目波動量
 // 原理是,每次插入刪除都是一個小批量動作,
 // 而不是每次只插入一條、銷毀一條
 // 計算出的局部渲染數(shù)據(jù)范圍,跟上一次計算出來的結(jié)果,差距
 // 在這個波動量范圍內(nèi),則不重新切片渲染,用于
 // 防止 IE 11 頻繁插入內(nèi)容導(dǎo)致性能壓力
 private _preRenderingCount() {
  // 默認預(yù)渲染 2 屏
  return Math.ceil(this.viewportHeight / this.defaultItemHeight!) * 2
 }
 // 滾動到上下方剩下多少個條目時,加載下一批
 // 緩解 Macbook & iOS 觸摸滾動時的白屏
 private _preRenderingThreshold() {
  // 默認觸達預(yù)渲染的一半數(shù)量時,加載下一批切片
  return Math.floor(this._preRenderingCount() / 2)
 }
 // 刷新局部渲染數(shù)據(jù)切片范圍
 private _updateSliceRange(forceUpdate?: boolean) {
  // 上下方額外多渲染的條目波動量
  const COUNT = this._preRenderingCount()
  // 預(yù)渲染觸發(fā)閾值
  const THRESHOLD = this._preRenderingThreshold()  
  // 數(shù)據(jù)總量
  const MAX = this.dataView.length
  // 計算出準確的切片區(qū)間
  const range = this._calcSliceRange()  
  // 檢查計算出來的切片范圍,是否被當前已經(jīng)渲染的切片返回包含了
  // 如果是,無需更新切片,(如果 forceUpdate,則無論如何都需要重新切片)
  let fromThreshold = range.sliceFrom - THRESHOLD
  if (fromThreshold < 0) fromThreshold = 0
  let toThreshold = range.sliceTo + THRESHOLD
  if (toThreshold > MAX) toThreshold = MAX
  // 無需強制刷新,且上下兩端都沒有觸達閾值時,無需重新切片
  if (!forceUpdate && ((this.sliceFrom <= fromThreshold) && (this.sliceTo >= toThreshold))) {
   return
  }
  // 更新切片的情況
  // 在切片區(qū)間頭部、尾部,追加預(yù)渲染的條目
  let { sliceFrom, sliceTo } = range
  sliceFrom = sliceFrom > COUNT ? sliceFrom - COUNT : 0
  sliceTo = sliceTo + COUNT > MAX ? MAX : sliceTo + COUNT
  this.sliceFrom = sliceFrom
  this.sliceTo = sliceTo
  if (forceUpdate) this._doSlice()
 }
 // 當前需要渲染的數(shù)據(jù)切片
 private _doSlice() {
  const { dataView, sliceFrom, sliceTo } = this
  const slice = dataView.slice(sliceFrom, sliceTo)
   .filter((wrappedItem) => wrappedItem.height > 0)
  this.dataSlice = Object.freeze(slice)
  this.$emit('slice', slice.map((wrappedItem) => wrappedItem.data))
 }
 // `index` 數(shù)據(jù)在 dataView 中的 index
 private _itemHeight(index: number): number {
  return this.itemHeightStore.getValueAt(index)
 }
 // `index` 數(shù)據(jù)在 dataView 中的 index
 private _top(index: number): number {
  if (index === 0) return 0
  // 0 ~ 上一項的高度累加
  const rangeValues = this.itemHeightStore.intersecting(0, index - 1)
  const sumHeight = rangeValues.reduce((sum: number, rangeValue: any) => {
   const span = rangeValue.end - rangeValue.start + 1
   return sum + rangeValue.value * span
  }, 0)
  return sumHeight
 }
 // 包裹原始數(shù)據(jù)列表
 private _wrapData(list: any[]): ItemWrapper[] {
  return list.map((item, index) => new ItemWrapper(item, index, this.defaultItemHeight!))
 }
 // 通過 DOM Node 獲取對應(yīng)的數(shù)據(jù)
 private _getDataByNode(node: Node): ItemWrapper {
  return this.wrappedData[(node as any).dataset.key]
 }
 // 刷新列表項高度
 private _updateHeight(wrappedItem: ItemWrapper, height: number, isRealHeight?: boolean) {
  height = height >> 0
  // 更新結(jié)點高度緩存
  wrappedItem.height = height
  if (isRealHeight) {
   wrappedItem.realHeight = true
  }
  // 如果 wrappedItem 為當前過濾下的項目,
  // 則同時刷新高度存儲 store
  const index = this.dataView.indexOf(wrappedItem)
  if (index !== -1) {
   // 小于等于零表示折疊不顯示,計算高度為零
   // 負值存在 wrappedItem 中,用于反折疊時恢復(fù)
   this.itemHeightStore.setValue(index, height > 0 ? height : 0)
  }
 }
 // 節(jié)點插入時,檢查是否首次插入,如果是,計算高度并更新對應(yīng)的 ItemWrapper
 private _updateHeightWhenItemInserted(mutations: MutationRecord[]) {
  const addedNodes: Node[] = mutations
   .map((mutation: MutationRecord) => mutation.addedNodes)
   .reduce((result: any, items: NodeList) => {
    result.push(...items)
    return result
   }, [])
  const batch: Array<{ wrappedItem: ItemWrapper, height: number }> = []
  addedNodes.forEach((node: Node) => {
   const wrappedItem = this._getDataByNode(node)
   // 如果 wrappedItem 中已經(jīng)存儲了計算過的高度,
   // 則直接返回,不訪問 clientHeight
   // 以避免性能開銷(IE 11 中訪問 clientHeight 性能非常差)
   if (wrappedItem.realHeight) {
    return
   }
   const height = this.itemHeightMethod(node, wrappedItem) >> 0
   if (wrappedItem.height !== height) {
    batch.push({ wrappedItem, height })
   } else {
    // 計算出來的高度跟默認值一致,
    // 則無需更新,但是設(shè)置已經(jīng)計算狀態(tài)
    // 以便下次可以直接使用緩存
    wrappedItem.realHeight = true
   }
  })
  if (batch.length) {
   this.batchUpdateHeight(batch)
  }
 }
 // 滾動事件處理器
 private _onScroll() {
  this._updateSliceRange()
 }
}

總結(jié)

以上所說是小白給大家介紹的使用 Vue 實現(xiàn)一個虛擬列表的方法,希望對大家有所幫助,如果大家有任何疑問歡迎給我留言,小編會及時回復(fù)大家的!

相關(guān)文章

最新評論

偷拍自拍 中文字幕| 哥哥姐姐综合激情小说| 日本黄在免费看视频| 骚逼被大屌狂草视频免费看| 99国内小视频在现欢看| 蜜臀av久久久久蜜臀av麻豆| 欧美少妇性一区二区三区| 亚洲欧美久久久久久久久| 黄色黄色黄片78在线| 可以在线观看的av中文字幕| 自拍偷拍亚洲另类色图| 青草青永久在线视频18| 亚洲最大黄了色网站| 99亚洲美女一区二区三区| 免费啪啪啪在线观看视频| 久久久制服丝袜中文字幕| 日韩中文字幕在线播放第二页| 一级a看免费观看网站| 2020国产在线不卡视频| 中文字幕网站你懂的| 玖玖一区二区在线观看| av在线免费观看亚洲天堂| 2019av在线视频| 日本性感美女写真视频| 特级无码毛片免费视频播放| 不卡精品视频在线观看| 午夜国产免费福利av| 91麻豆精品久久久久| 五月激情婷婷久久综合网| 桃色视频在线观看一区二区 | 美女福利视频网址导航| 国产使劲操在线播放| 99精品久久久久久久91蜜桃| 日韩欧美一级aa大片| 一区二区三区毛片国产一区| 青娱乐最新视频在线| 精品国产成人亚洲午夜| 午夜免费体验区在线观看| 5528327男人天堂| 精品成人午夜免费看| 中文字幕国产专区欧美激情| 亚洲码av无色中文| 亚洲精品一区二区三区老狼| 亚洲一区av中文字幕在线观看| 中文字幕+中文字幕| 免费观看理论片完整版| 1769国产精品视频免费观看| 成人资源在线观看免费官网| 日本少妇精品免费视频| 免费国产性生活视频| 黄色成人在线中文字幕| 久久久久91精品推荐99| 成人免费公开视频无毒| 亚洲精品乱码久久久久久密桃明 | 最新中文字幕乱码在线| 鸡巴操逼一级黄色气| 在线观看操大逼视频| 中文字幕一区二 区二三区四区 | 91自产国产精品视频| 啪啪啪18禁一区二区三区 | 国产1区,2区,3区| 日本五十路熟新垣里子| 在线免费91激情四射 | 99精品国产免费久久| 国产精品一区二区久久久av| 人妻丰满熟妇综合网| 国产普通话插插视频| 国产成人精品福利短视频| 天天日夜夜操天天摸| 国产视频精品资源网站| 社区自拍揄拍尻屁你懂的| 亚洲av黄色在线网站| 国产又色又刺激在线视频 | 亚洲午夜福利中文乱码字幕 | 青青尤物在线观看视频网站| 人妻激情图片视频小说| 日韩精品中文字幕福利| 亚洲欧美一区二区三区电影| 人妻素人精油按摩中出| 欧美精品中文字幕久久二区| 中文字母永久播放1区2区3区| 99热这里只有国产精品6| av一区二区三区人妻| 青青草成人福利电影| 天天想要天天操天天干| 熟女在线视频一区二区三区| 日日夜夜精品一二三| 美女日逼视频免费观看| 大屁股肉感人妻中文字幕在线| 国产第一美女一区二区三区四区| 蜜臀av久久久久久久| 在线国产精品一区二区三区| yy6080国产在线视频| 日韩精品中文字幕福利| 亚洲成人情色电影在线观看| 国内自拍第一页在线观看| 91成人在线观看免费视频| 国产黄色片在线收看| 精品一区二区亚洲欧美| 亚洲精品 欧美日韩| 色吉吉影音天天干天天操| 伊人网中文字幕在线视频| 久久三久久三久久三久久| 青青草精品在线视频观看| 日韩欧美国产一区不卡| 97年大学生大白天操逼| 国产极品美女久久久久久| 国产V亚洲V天堂无码欠欠| 初美沙希中文字幕在线 | 亚洲一区自拍高清免费视频| 男人的天堂一区二区在线观看| 一级黄片大鸡巴插入美女| 日本少妇在线视频大香蕉在线观看 | 性感美女诱惑福利视频| 91高清成人在线视频| 亚洲欧美精品综合图片小说| 狠狠躁夜夜躁人人爽天天天天97| 最新国产亚洲精品中文在线| 91精品免费久久久久久| 色婷婷六月亚洲综合香蕉| 全国亚洲男人的天堂| heyzo蜜桃熟女人妻| 一区二区麻豆传媒黄片| 国产精品3p和黑人大战| 日本午夜久久女同精女女| 97小视频人妻一区二区| 99精品国产aⅴ在线观看| 黄色视频在线观看高清无码| 久久久久久性虐视频| 国产精品亚洲а∨天堂免| 国产精品久久9999| 一级A一级a爰片免费免会员| chinese国产盗摄一区二区| 日噜噜噜夜夜噜噜噜天天噜噜噜| 国产夫妻视频在线观看免费| 国产美女精品福利在线| 青青热久免费精品视频在线观看 | 精品黑人一区二区三区久久国产| 三级黄色亚洲成人av| 国产卡一卡二卡三乱码手机| 免费男阳茎伸入女阳道视频| 亚洲国产成人最新资源| 无码国产精品一区二区高潮久久4| 天天做天天干天天舔| 日韩精品中文字幕播放| 黄色视频成年人免费观看| 日韩人妻在线视频免费| 大香蕉大香蕉大香蕉大香蕉大香蕉 | 国产日韩精品一二三区久久久 | 天天日夜夜操天天摸| 日本男女操逼视频免费看| 精品一区二区亚洲欧美| 亚洲伊人av天堂有码在线| 成年午夜免费无码区| 岳太深了紧紧的中文字幕| 啊啊啊想要被插进去视频| 人妻少妇精品久久久久久| 国产超码片内射在线| 亚洲av午夜免费观看| 在线免费观看黄页视频| 激情小视频国产在线| 岛国一区二区三区视频在线| 国产一区二区视频观看| 岛国青草视频在线观看| 人人爱人人妻人人澡39| 久久精品国产999| 在线免费观看日本片| 又色又爽又黄的美女裸体| gay gay男男瑟瑟在线网站| 喷水视频在线观看这里只有精品| 日本人妻欲求不满中文字幕| 欧美精品国产综合久久| 国产av自拍偷拍盛宴| 国产美女一区在线观看| 婷婷久久一区二区字幕网址你懂得| 又粗又硬又猛又爽又黄的| 精品老妇女久久9g国产| 亚洲少妇高潮免费观看| 88成人免费av网站| 亚洲国产欧美一区二区丝袜黑人| 国产免费av一区二区凹凸四季| 开心 色 六月 婷婷| 日本少妇精品免费视频| av网址国产在线观看| 影音先锋女人av噜噜色| 大鸡巴插入美女黑黑的阴毛| caoporn蜜桃视频| 人妻激情图片视频小说| 国产又色又刺激在线视频| 天天艹天天干天天操| 91免费福利网91麻豆国产精品| 亚洲天堂精品久久久| 老司机你懂得福利视频| 任我爽精品视频在线播放| 国产亚洲天堂天天一区| 女同久久精品秋霞网| 中文字幕AV在线免费看 | 亚洲女人的天堂av| 日本av熟女在线视频| 日本免费视频午夜福利视频| 91福利视频免费在线观看| 天天摸天天干天天操科普| 婷婷久久一区二区字幕网址你懂得| 中文字幕第1页av一天堂网| 国产美女午夜福利久久| 国产视频一区在线观看| 欧美少妇性一区二区三区| 婷婷综合蜜桃av在线| 久久久精品欧洲亚洲av| 天天干夜夜操啊啊啊| 免费在线黄色观看网站| 亚洲综合图片20p| 都市激情校园春色狠狠| 精品人妻一二三区久久| 中文字幕av男人天堂| 免费一级特黄特色大片在线观看 | 成人区人妻精品一区二视频| 精品国产污污免费网站入口自| 日韩成人综艺在线播放| 99精品国自产在线人| 在线观看日韩激情视频| 青青青青青操视频在线观看| 伊人精品福利综合导航| 亚洲熟妇x久久av久久| 天天色天天舔天天射天天爽| 免费岛国喷水视频在线观看| 亚洲麻豆一区二区三区| 亚洲男人在线天堂网| 黑人性生活视频免费看| 青青草人人妻人人妻| 亚洲一区二区三区uij| av在线shipin| 亚洲成av人无码不卡影片一| 亚洲精品国产综合久久久久久久久| 国产激情av网站在线观看| 婷婷综合亚洲爱久久| 亚洲成人精品女人久久久| 熟女人妻在线观看视频| 女同性ⅹxx女同hd| 在线免费观看欧美小视频| 亚洲第一黄色在线观看| 天天干夜夜操天天舔| 国内自拍第一页在线观看| 动色av一区二区三区| 欧美xxx成人在线| 日韩人妻xxxxx| 亚洲2021av天堂| 高潮视频在线快速观看国家快速| 亚洲成高清a人片在线观看| 精品国产成人亚洲午夜| 中文字幕综合一区二区| 97少妇精品在线观看| 亚洲精品 欧美日韩| 久久精品久久精品亚洲人| 午夜频道成人在线91| 亚洲精品成人网久久久久久小说 | 中文字幕高清在线免费播放| 久草视频在线一区二区三区资源站 | 少妇人妻100系列| 欧美色婷婷综合在线| 中文字母永久播放1区2区3区| 免费十精品十国产网站| 欧美精品伦理三区四区| 亚洲天堂精品久久久| 午夜毛片不卡免费观看视频| 天天日天天干天天干天天日| 青娱乐最新视频在线| 激情图片日韩欧美人妻| 国产高潮无码喷水AV片在线观看| 老司机午夜精品视频资源| 亚洲人妻av毛片在线| 久久综合老鸭窝色综合久久| av天堂中文字幕最新| 国产日韩欧美视频在线导航| 在线视频精品你懂的| 欧洲黄页网免费观看| 亚洲无线观看国产高清在线| 搡老妇人老女人老熟女| 青青擦在线视频国产在线| 男人操女人逼逼视频网站| 青青在线视频性感少妇和隔壁黑丝| 日本裸体熟妇区二区欧美| 91精品国产高清自在线看香蕉网| 人妻激情图片视频小说| 福利午夜视频在线合集| yy96视频在线观看| 亚洲人妻30pwc| 精品国产成人亚洲午夜| 亚洲欧美国产综合777| 中文字幕av熟女人妻| 亚洲一区av中文字幕在线观看| 精品视频一区二区三区四区五区| 亚洲欧美精品综合图片小说| 国产精品精品精品999| 日日摸夜夜添夜夜添毛片性色av| 久久午夜夜伦痒痒想咳嗽P| 女生被男生插的视频网站| 沈阳熟妇28厘米大战黑人| 97精品成人一区二区三区 | 亚洲欧美一区二区三区电影| 久久精品美女免费视频| 1区2区3区4区视频在线观看| 国产伊人免费在线播放| 韩国亚洲欧美超一级在线播放视频 | 欧美80老妇人性视频| 欧美精品一区二区三区xxxx| 欧美80老妇人性视频| 欧美亚洲自偷自拍 在线| 亚洲人人妻一区二区三区| 首之国产AV医生和护士小芳| 欧美久久久久久三级网| 免费av岛国天堂网站| 91在线免费观看成人| 99国产精品窥熟女精品| av一区二区三区人妻| 精品视频国产在线观看| 888欧美视频在线| 日韩人妻xxxxx| 久久精品国产亚洲精品166m| 夏目彩春在线中文字幕| 蜜臀av久久久久久久| 免费费一级特黄真人片| 国产九色91在线观看精品| 男人操女人的逼免费视频| 日本精品一区二区三区在线视频。| 动漫黑丝美女的鸡巴| 538精品在线观看视频| 成人网18免费视频版国产| aⅴ精产国品一二三产品| 不卡精品视频在线观看| 亚洲天天干 夜夜操| 最新中文字幕免费视频| 日韩精品中文字幕播放| 欧美80老妇人性视频| 在线新三级黄伊人网| 5528327男人天堂| 93精品视频在线观看| 欧洲日韩亚洲一区二区三区| 欧美精品资源在线观看| 亚洲国产欧美一区二区三区久久| 成人动漫大肉棒插进去视频| 91成人在线观看免费视频| 偷拍自拍 中文字幕| 欧美精品中文字幕久久二区| 操日韩美女视频在线免费看| 人妻自拍视频中国大陆| 啪啪啪18禁一区二区三区| 91国内精品久久久久精品一| 在线观看操大逼视频| 欧美另类重口味极品在线观看| 岛国青草视频在线观看| 亚洲免费国产在线日韩| 欧美精品中文字幕久久二区| 欧美日韩一区二区电影在线观看| 一区二区视频视频视频| 国产午夜男女爽爽爽爽爽视频| 女同久久精品秋霞网| 亚洲va国产va欧美va在线| 欧美日韩中文字幕欧美| 2012中文字幕在线高清| 天天艹天天干天天操| 国产成人精品久久二区91| 亚洲天堂av最新网址| 国产黄色大片在线免费播放| 日本丰满熟妇大屁股久久| 中文字幕午夜免费福利视频| 综合页自拍视频在线播放| av手机在线观播放网站| 亚洲专区激情在线观看视频| 黄色录像鸡巴插进去| 亚洲美女自偷自拍11页| 日本性感美女三级视频| 色秀欧美视频第一页| 亚洲图片偷拍自拍区| 人妻av无码专区久久绿巨人| 欧美一区二区中文字幕电影 | 日本高清在线不卡一区二区| 人妻爱爱 中文字幕| 插小穴高清无码中文字幕| 中文 成人 在线 视频| 特级欧美插插插插插bbbbb| 老师让我插进去69AV| 97精品人妻一区二区三区精品| 亚洲欧美色一区二区| 岛国毛片视频免费在线观看| 黄色片年轻人在线观看| 黄色大片男人操女人逼| 日本一二三区不卡无| 天堂av在线最新版在线| 888亚洲欧美国产va在线播放| 在线免费观看日本伦理| 偷拍自拍亚洲美腿丝袜| 亚洲 欧美 自拍 偷拍 在线| 又大又湿又爽又紧A视频| 在线不卡成人黄色精品| 精品av久久久久久久| 天天做天天干天天舔| 91九色国产porny蝌蚪| 91人妻精品久久久久久久网站 | 国产极品美女久久久久久| 精品91自产拍在线观看一区| 97年大学生大白天操逼| 爆乳骚货内射骚货内射在线 | 少妇人妻二三区视频| 欧美中国日韩久久精品| 日本高清撒尿pissing| 狠狠地躁夜夜躁日日躁| 91麻豆精品91久久久久同性 | 涩涩的视频在线观看视频| 啪啪啪啪啪啪啪免费视频| 91高清成人在线视频| 亚洲男人在线天堂网| 搡老熟女一区二区在线观看| 国产精品一二三不卡带免费视频| 鸡巴操逼一级黄色气| 免费在线播放a级片| 成人H精品动漫在线无码播放| 国产精品入口麻豆啊啊啊| 天天干天天日天天干天天操| 激情色图一区二区三区| 日本啪啪啪啪啪啪啪| 鸡巴操逼一级黄色气| 青青青青视频在线播放| jiuse91九色视频| 在线免费观看欧美小视频| 又色又爽又黄的美女裸体| 日韩二区视频一线天婷婷五| 3337p日本欧洲大胆色噜噜| 亚洲熟女久久久36d| 91中文字幕免费在线观看| 中文字幕欧美日韩射射一| 40道精品招牌菜特色| 青青社区2国产视频| 男人的天堂在线黄色| 93精品视频在线观看| 日本少妇人妻xxxxx18| 开心 色 六月 婷婷| 亚洲熟女女同志女同| 亚洲一区二区激情在线| 国产午夜亚洲精品麻豆| 自拍偷拍,中文字幕| 色花堂在线av中文字幕九九| 大香蕉大香蕉大香蕉大香蕉大香蕉| 人人妻人人爽人人添夜| av线天堂在线观看| 亚洲成人黄色一区二区三区 | 青青青青在线视频免费观看| 日本特级片中文字幕| 黑人进入丰满少妇视频| 韩国一级特黄大片做受| 亚洲高清视频在线不卡| 五月天色婷婷在线观看视频免费| 国产一区自拍黄视频免费观看| 欧美成人猛片aaaaaaa| 无码国产精品一区二区高潮久久4| 中文字幕综合一区二区| 欧美一区二区三区高清不卡tv| av久久精品北条麻妃av观看 | 亚洲精品亚洲人成在线导航| 一区二区熟女人妻视频| 亚洲 欧美 精品 激情 偷拍| 国产精品自偷自拍啪啪啪| 唐人色亚洲av嫩草| 久久久久久9999久久久久| 国产av国片精品一区二区| 天堂资源网av中文字幕| 中国产一级黄片免费视频播放| 欧美一区二区三区啪啪同性| 97资源人妻免费在线视频| 青青色国产视频在线| wwwxxx一级黄色片| 日本少妇在线视频大香蕉在线观看| 美女骚逼日出水来了| 激情人妻校园春色亚洲欧美 | 五十路丰满人妻熟妇| 精品乱子伦一区二区三区免费播| 亚洲图片欧美校园春色| 在线观看av2025| 全国亚洲男人的天堂| 在线免费观看欧美小视频| 93精品视频在线观看| 91超碰青青中文字幕| 极品丝袜一区二区三区| 40道精品招牌菜特色| 日本福利午夜电影在线观看| 一区二区三区在线视频福利| 女生自摸在线观看一区二区三区 | 欧美区一区二区三视频| 成年人啪啪视频在线观看| 亚洲国产欧美一区二区三区久久| 黑人巨大精品欧美视频| 精品一区二区三区三区88| 精品成人啪啪18免费蜜臀| 天天做天天爽夜夜做少妇| 国产老熟女伦老熟妇ⅹ| 国产精品精品精品999| 97黄网站在线观看| 天天色天天爱天天爽| 天天操天天干天天插| 亚洲综合另类精品小说| 日本韩国免费福利精品| 中文字幕一区二 区二三区四区| 丝袜美腿欧美另类 中文字幕| 极品性荡少妇一区二区色欲| 午夜成午夜成年片在线观看| 青青草原网站在线观看| 国产 在线 免费 精品| 大胸性感美女羞爽操逼毛片| 一区二区三区欧美日韩高清播放| 欧美香蕉人妻精品一区二区| 91色九色porny| 亚洲综合一区成人在线| 国产自拍在线观看成人| 国产老熟女伦老熟妇ⅹ| 绝顶痉挛大潮喷高潮无码| 护士特殊服务久久久久久久| 中文乱理伦片在线观看| 在线免费91激情四射 | 久久久久久九九99精品| 岛国免费大片在线观看 | 加勒比视频在线免费观看| 精品一区二区三区欧美| 日本人妻精品久久久久久| 亚洲伊人av天堂有码在线| 亚洲另类在线免费观看| 老鸭窝日韩精品视频观看| 顶级尤物粉嫩小尤物网站| 中国把吊插入阴蒂的视频| 亚洲图片欧美校园春色| 色av色婷婷人妻久久久精品高清| 亚洲国产精品久久久久久6| 男人的天堂在线黄色| 人人爽亚洲av人人爽av| 亚洲高清自偷揄拍自拍| 欧美一区二区三区啪啪同性| 亚洲日本一区二区久久久精品| 91麻豆精品秘密入口在线观看 | 国产精品久久久久久久精品视频| 日本韩国免费福利精品| 天天射夜夜操综合网| 中文字幕一区二区人妻电影冢本| 丝袜肉丝一区二区三区四区在线 | 亚洲另类在线免费观看| 欧美一区二区三区乱码在线播放| 日日爽天天干夜夜操| 国产使劲操在线播放| 国产V亚洲V天堂无码欠欠 | 一区二区三区 自拍偷拍| 香港三日本三韩国三欧美三级| 国产高清在线在线视频| 欧美亚洲中文字幕一区二区三区| 天天艹天天干天天操| 美女福利写真在线观看视频| 丝袜长腿第一页在线| av一区二区三区人妻| 水蜜桃一区二区三区在线观看视频| 人妻av无码专区久久绿巨人| 伊人精品福利综合导航| 毛片一级完整版免费| 涩涩的视频在线观看视频| 成年女人免费播放视频| 热99re69精品8在线播放| av在线播放国产不卡| 沈阳熟妇28厘米大战黑人| 亚洲最大黄 嗯色 操 啊| 国产使劲操在线播放| 国产精品黄片免费在线观看| 久久99久久99精品影院| 亚洲少妇高潮免费观看| 看一级特黄a大片日本片黑人| 大香蕉大香蕉大香蕉大香蕉大香蕉 | av天堂资源最新版在线看| 亚洲成a人片777777| 亚洲午夜电影在线观看| 大屁股熟女一区二区三区| 在线免费观看日本片| 免费黄页网站4188| 精品美女久久久久久| 日韩二区视频一线天婷婷五| 青青青青爽手机在线| 欧美成人黄片一区二区三区 | 国产女人被做到高潮免费视频| 亚洲偷自拍高清视频| 漂亮 人妻被中出中文| 午夜精品一区二区三区福利视频| 欧美80老妇人性视频| 国产丰满熟女成人视频| 啊慢点鸡巴太大了啊舒服视频| 色婷婷综合激情五月免费观看| 欧美成人综合色在线噜噜| 欧美第一页在线免费观看视频| 丝袜国产专区在线观看| 亚洲中文精品人人免费| 亚洲国产欧美一区二区三区…| 天天做天天干天天舔| 人妻熟女在线一区二区| 在线观看国产网站资源| yellow在线播放av啊啊啊| 日本人妻精品久久久久久| 97国产在线观看高清| 大鸡八强奸视频在线观看| 女同互舔一区二区三区| 欧美特色aaa大片| 成人蜜臀午夜久久一区| 午夜精彩视频免费一区| 91破解版永久免费| 中文字幕奴隷色的舞台50| 91久久综合男人天堂| 青青草原网站在线观看| 夜夜骑夜夜操夜夜奸| 亚洲欧美清纯唯美另类| 99热碰碰热精品a中文| 精品一区二区三区三区88| 黄色的网站在线免费看 | 国产一区自拍黄视频免费观看| 蜜臀成人av在线播放| 丝袜美腿欧美另类 中文字幕| 97超碰免费在线视频| 春色激情网欧美成人| 视频一区 二区 三区 综合| 成年人免费看在线视频| 偷拍自拍亚洲视频在线观看| 日韩精品啪啪视频一道免费| 欧美日韩一级黄片免费观看| 成人高潮aa毛片免费| 不卡一区一区三区在线| 亚洲一级av无码一级久久精品| 精产国品久久一二三产区区别 | 天天日天天干天天插舔舔| 日本真人性生活视频免费看| 亚洲国产中文字幕啊啊啊不行了 | 久久久久久久久久性潮| 国产精彩福利精品视频| 天天色天天爱天天爽| 久草福利电影在线观看| 晚上一个人看操B片| 中文字幕人妻被公上司喝醉在线| 亚洲av第国产精品| 国产高清女主播在线| av在线shipin| 激情内射在线免费观看| 免费岛国喷水视频在线观看| 91免费观看在线网站| 天天操夜夜操天天操天天操| 日韩欧美在线观看不卡一区二区 | 真实国产乱子伦一区二区| 欧美成人黄片一区二区三区| 清纯美女在线观看国产| 老司机免费福利视频网| 国产变态另类在线观看| 激情五月婷婷免费视频| 国产第一美女一区二区三区四区 | 2o22av在线视频| 阿v天堂2014 一区亚洲| 社区自拍揄拍尻屁你懂的| 欧美日韩熟女一区二区三区| 韩国亚洲欧美超一级在线播放视频| 中文字幕成人日韩欧美| 欧美偷拍自拍色图片| 欧美美女人体视频一区| 日本人妻少妇18—xx| 亚洲精品一区二区三区老狼| 任你操视频免费在线观看| 又黄又刺激的午夜小视频| 免费在线观看视频啪啪| 粉嫩av蜜乳av蜜臀| 成人国产激情自拍三区| 日本一区二区三区免费小视频| 欧美一级色视频美日韩| 福利视频广场一区二区| 欧美精品中文字幕久久二区| 日韩特级黄片高清在线看| 久久三久久三久久三久久| 夏目彩春在线中文字幕| 亚洲一区二区久久久人妻| 国产极品精品免费视频| 直接能看的国产av| 91九色国产porny蝌蚪| 日韩欧美在线观看不卡一区二区 | 日本人妻精品久久久久久| 亚洲嫩模一区二区三区| 欧美在线精品一区二区三区视频 | 岛国青草视频在线观看| 成人av久久精品一区二区| 一区二区麻豆传媒黄片| okirakuhuhu在线观看| 人妻久久无码中文成人| 亚洲精品久久视频婷婷| 亚洲欧美国产麻豆综合| 一本久久精品一区二区| 亚洲男人让女人爽的视频| 青青操免费日综合视频观看| 中文字幕成人日韩欧美| 亚洲在线免费h观看网站| 大鸡八强奸视频在线观看| 2020韩国午夜女主播在线| 国产性生活中老年人视频网站| 亚洲最大免费在线观看| 1769国产精品视频免费观看| 午夜精品在线视频一区| 男人天堂av天天操| 三级av中文字幕在线观看| 在线视频国产欧美日韩| 色狠狠av线不卡香蕉一区二区 | 夜夜嗨av一区二区三区中文字幕| 一本久久精品一区二区| 欧美特色aaa大片| 精内国产乱码久久久久久| 中文字幕在线观看国产片| 这里只有精品双飞在线播放| 久草视频福利在线首页| 久草视频在线免播放| 亚洲精品欧美日韩在线播放| 91p0rny九色露脸熟女| 91亚洲手机在线视频播放| 99精品免费久久久久久久久a| 日本成人一区二区不卡免费在线| 五十路丰满人妻熟妇| 久久久久五月天丁香社区| av完全免费在线观看av| 久久久久五月天丁香社区 | 日本人妻精品久久久久久| av成人在线观看一区| 亚洲精品国偷自产在线观看蜜桃 | 中文字母永久播放1区2区3区| 欧美另类一区二区视频| 男生舔女生逼逼视频| 亚洲人妻视频在线网| 欧美一级片免费在线成人观看| 在线免费观看视频一二区| 欧美日韩在线精品一区二区三| 欧美亚洲国产成人免费在线| 少妇人妻100系列| 涩爱综合久久五月蜜臀| 中文字幕人妻一区二区视频| 91精品国产91青青碰| 亚洲高清国产自产av| 国产在线一区二区三区麻酥酥| 午夜免费体验区在线观看| 青青操免费日综合视频观看| 国产精品自拍偷拍a| 日本少妇在线视频大香蕉在线观看| 免费无码人妻日韩精品一区二区| 亚洲高清视频在线不卡| 扒开腿挺进肉嫩小18禁视频| 在线观看黄色成年人网站| 一色桃子人妻一区二区三区| 青青青青青青青青青青草青青 | 1024久久国产精品| 久久一区二区三区人妻欧美| 欧洲亚洲欧美日韩综合| 中文字幕—97超碰网| 欧美精品黑人性xxxx| 亚洲国产欧美国产综合在线| 孕妇奶水仑乱A级毛片免费看 | 熟女妇女老妇一二三区| 大学生A级毛片免费视频| 亚洲欧美另类手机在线| 亚洲av成人网在线观看| 91she九色精品国产| 欧美另类一区二区视频| 啪啪啪18禁一区二区三区| 国产精品久久久久久久久福交| 韩国三级aaaaa高清视频| 日韩av有码中文字幕| 久久久久久九九99精品| 久久精品亚洲成在人线a| 大鸡吧插逼逼视频免费看| 日本黄色特一级视频| 人妻3p真实偷拍一二区| 78色精品一区二区三区| 中文字幕—97超碰网| 亚洲公开视频在线观看| 国产三级精品三级在线不卡| 人妻少妇性色欲欧美日韩| 九色精品视频在线播放| 韩国三级aaaaa高清视频| 91亚洲手机在线视频播放| 超pen在线观看视频公开97| 99热99re在线播放| 91chinese在线视频| 护士特殊服务久久久久久久| 久草极品美女视频在线观看| 黄色录像鸡巴插进去| 亚洲丝袜老师诱惑在线观看| 水蜜桃一区二区三区在线观看视频| 午夜美女福利小视频| 国产自拍在线观看成人| 日日夜夜狠狠干视频| 伊人综合aⅴ在线网| 激情国产小视频在线| 人妻少妇亚洲一区二区| 午夜精品福利91av| 国产在线免费观看成人| 久久久久久久一区二区三| av中文字幕在线导航| 在线观看视频污一区| 国产一区av澳门在线观看| 午夜频道成人在线91| 80电影天堂网官网| 97人人模人人爽人人喊| 国产黄色片蝌蚪九色91| 成人亚洲精品国产精品| 中文字幕 亚洲av| 2022天天干天天操| 午夜成午夜成年片在线观看| 亚洲欧美福利在线观看| 2022国产精品视频| 国产综合视频在线看片| 天堂v男人视频在线观看| 老司机午夜精品视频资源| 大鸡吧插入女阴道黄色片| 国产麻豆乱子伦午夜视频观看| 一本久久精品一区二区| 亚洲激情av一区二区| 黑人性生活视频免费看| 国产一区二区神马久久| 亚洲午夜精品小视频| 青草青永久在线视频18| 91香蕉成人app下载| 男人在床上插女人视频| 亚洲人成精品久久久久久久| 亚洲中文精品字幕在线观看| 热久久只有这里有精品| 中文字幕1卡1区2区3区| 亚洲成人激情av在线| 免费无码人妻日韩精品一区二区| 三级av中文字幕在线观看| 九色精品视频在线播放| 国产卡一卡二卡三乱码手机| 在线观看欧美黄片一区二区三区| 免费看国产av网站| 午夜美女福利小视频| 午夜久久久久久久99| 青青青视频自偷自拍38碰| 99久久99久国产黄毛片| 色综合久久久久久久久中文| 亚洲最大黄了色网站| 人妻无码色噜噜狠狠狠狠色| 最新的中文字幕 亚洲| 日本av熟女在线视频| 欧美综合婷婷欧美综合| 国产黄网站在线观看播放| 天天日天天干天天插舔舔| 91麻豆精品91久久久久同性 | 欧美特级特黄a大片免费| 精品成人啪啪18免费蜜臀| 国产日韩欧美美利坚蜜臀懂色| 一区二区三区国产精选在线播放 | 2025年人妻中文字幕乱码在线| 3D动漫精品啪啪一区二区下载| 青青伊人一精品视频| 爆乳骚货内射骚货内射在线| 国产美女一区在线观看| 红杏久久av人妻一区| 国产视频网站国产视频| 91九色porny国产在线| 综合一区二区三区蜜臀| 亚洲av天堂在线播放| 在线可以看的视频你懂的| 熟妇一区二区三区高清版| 97黄网站在线观看| 亚洲变态另类色图天堂网| 欧美久久一区二区伊人| 久久免看30视频口爆视频| 亚洲精品三级av在线免费观看| 久久久久久久亚洲午夜综合福利| 一区二区三区四区中文| 青娱乐蜜桃臀av色| 人人在线视频一区二区| 天干天天天色天天日天天射| 3337p日本欧洲大胆色噜噜| 国产实拍勾搭女技师av在线| 99亚洲美女一区二区三区| 91www一区二区三区| 日本在线一区二区不卡视频| 亚洲成高清a人片在线观看| 91色老99久久九九爱精品| 日日夜夜精品一二三| 天天日天天日天天射天天干| 神马午夜在线观看视频| 沈阳熟妇28厘米大战黑人| 一本久久精品一区二区| 在线观看国产网站资源| 视频 一区二区在线观看| 极品粉嫩小泬白浆20p主播| 中文字幕乱码人妻电影| 亚洲中文字字幕乱码| 初美沙希中文字幕在线 | 桃色视频在线观看一区二区| 成人午夜电影在线观看 久久| 国产丰满熟女成人视频| 中文字幕在线第一页成人| 国产午夜亚洲精品不卡在线观看| 非洲黑人一级特黄片| 亚洲 人妻 激情 中文| 欧美激情精品在线观看| 又粗又硬又猛又爽又黄的| 婷婷色国产黑丝少妇勾搭AV| av男人天堂狠狠干| 欧美成人黄片一区二区三区 | 亚洲成人国产综合一区| 亚洲av自拍偷拍综合| 日本最新一二三区不卡在线| 国产揄拍高清国内精品对白| 青草久久视频在线观看| 精品区一区二区三区四区人妻| 黄色片黄色片wyaa| 超级av免费观看一区二区三区| 韩国爱爱视频中文字幕| 国产aⅴ一线在线观看| 日本一区二区三区免费小视频| 2022国产综合在线干| 国产欧美精品免费观看视频| 亚洲美女高潮喷浆视频| 1区2区3区不卡视频| 美女福利视频导航网站| 久久精品国产23696| 亚洲精品 日韩电影| 欧美日本在线视频一区| eeuss鲁片一区二区三区| 国产高清在线在线视频| 巨乳人妻日下部加奈被邻居中出 | 欧美熟妇一区二区三区仙踪林| 亚洲特黄aaaa片| 宅男噜噜噜666免费观看| 少妇人妻100系列| 五月天久久激情视频| 国产1区,2区,3区| 天天躁日日躁狠狠躁躁欧美av| 国产 在线 免费 精品| 爆乳骚货内射骚货内射在线| 天天色天天爱天天爽| 人人妻人人人操人人人爽| 2021年国产精品自拍| 中文字幕中文字幕人妻| 午夜在线观看一区视频| 人妻久久无码中文成人| 在线免费观看日本片| 亚洲中文字幕乱码区| 93精品视频在线观看| 黄色大片免费观看网站| 97人妻无码AV碰碰视频| 香港一级特黄大片在线播放| 福利视频一区二区三区筱慧| 国产麻豆剧果冻传媒app| 日本黄在免费看视频| 97年大学生大白天操逼| 久久丁香婷婷六月天| av网址国产在线观看| 1000部国产精品成人观看视频| 亚洲高清自偷揄拍自拍| 好男人视频在线免费观看网站| 成人区人妻精品一区二视频 | 午夜精品一区二区三区更新| 亚洲精品无码色午夜福利理论片| 久久香蕉国产免费天天| 国产女人露脸高潮对白视频| 男人天堂色男人av| 欧美综合婷婷欧美综合| 夜色17s精品人妻熟女| 日本丰满熟妇BBXBBXHD| 亚洲久久午夜av一区二区| 天天日天天玩天天摸| 成人av免费不卡在线观看| 欧美天堂av无线av欧美| 中文亚洲欧美日韩无线码| 国产麻豆国语对白露脸剧情| 欧美va不卡视频在线观看| 亚洲精品午夜久久久久| 中文字幕—97超碰网| 欧美交性又色又爽又黄麻豆| 天天综合天天综合天天网| 在线免费观看av日韩| av在线免费资源站| 视频一区 二区 三区 综合| 国产精品3p和黑人大战| 黄页网视频在线免费观看| 自拍偷拍 国产资源| 老有所依在线观看完整版| 日本乱人一区二区三区| 成人亚洲精品国产精品| mm131美女午夜爽爽爽| 亚洲午夜伦理视频在线| 国产一区二区火爆视频| 99久久超碰人妻国产| 中文字幕亚洲久久久| 大鸡巴后入爆操大屁股美女| 免费在线福利小视频| 中文字幕—97超碰网| 在线视频这里只有精品自拍| 大屁股肉感人妻中文字幕在线| 日本高清成人一区二区三区| 少妇人妻久久久久视频黄片| 久久精品美女免费视频| 青青草视频手机免费在线观看| 国产高清在线观看1区2区| 极品丝袜一区二区三区| 中文字幕免费福利视频6| 午夜精品一区二区三区城中村| 国产精品女邻居小骚货| 精品亚洲国产中文自在线| 国产午夜福利av导航| 99人妻视频免费在线| 久久艹在线观看视频| 伊人成人综合开心网| 免费观看国产综合视频| 丰满少妇人妻xxxxx| 国产一线二线三线的区别在哪| 中文字幕无码一区二区免费| 欧美特级特黄a大片免费| 久久麻豆亚洲精品av| 一区二区三区四区视频在线播放 | 男生舔女生逼逼视频| 91 亚洲视频在线观看| 大香蕉福利在线观看| 亚洲图片偷拍自拍区| 日韩av熟妇在线观看| 中国把吊插入阴蒂的视频| 亚洲av自拍偷拍综合| 中文字幕高清在线免费播放 | 日本韩国免费一区二区三区视频 | 日本丰满熟妇大屁股久久| 国产精品大陆在线2019不卡| sejizz在线视频| 中文字幕日韩精品就在这里| 日本一区美女福利视频| 大陆av手机在线观看| 青青色国产视频在线| 国产欧美日韩在线观看不卡| 欧美日韩国产一区二区三区三州| 国产真实乱子伦a视频| 毛茸茸的大外阴中国视频| 午夜激情精品福利视频| 天天日天天敢天天干| 东京热男人的av天堂| 免费岛国喷水视频在线观看| 国产免费av一区二区凹凸四季| 久久这里只有精品热视频| 亚洲精品国产久久久久久| 性生活第二下硬不起来| 操操网操操伊剧情片中文字幕网| 2020中文字幕在线播放| 38av一区二区三区| 黄色录像鸡巴插进去| 国产精品国产三级国产午| 久久农村老妇乱69系列| 超碰在线中文字幕一区二区| 亚洲最大黄了色网站| 午夜久久久久久久99| 成人av久久精品一区二区| 中出中文字幕在线观看| av破解版在线观看| 人人妻人人爽人人添夜| 黑人3p华裔熟女普通话| 人妻丝袜精品中文字幕| 视频一区二区三区高清在线| 一区二区三区久久久91| 丰满少妇翘臀后进式| 欧美亚洲一二三区蜜臀| 国产精品一区二区av国| 2021年国产精品自拍| 国产午夜激情福利小视频在线| 2021年国产精品自拍| 久草极品美女视频在线观看| 日本黄色三级高清视频| 国产91久久精品一区二区字幕| 精品一区二区三四区| 91中文字幕免费在线观看| 国产视频一区在线观看| 国产真实灌醉下药美女av福利| 日韩加勒比东京热二区| 插小穴高清无码中文字幕| 日本免费午夜视频网站| 99热这里只有国产精品6| 日本女大学生的黄色小视频| 成人蜜桃美臀九一一区二区三区| 粗大的内捧猛烈进出爽大牛汉子| 成人免费毛片aaaa| 天天操天天干天天日狠狠插| 午夜影院在线观看视频羞羞羞| 亚洲欧美清纯唯美另类 | 国产97在线视频观看| 好吊视频—区二区三区| 国产在线免费观看成人| 一区二区三区日韩久久| 2020久久躁狠狠躁夜夜躁| 久久久久久久久久久免费女人| 黄色在线观看免费观看在线| 日韩人妻丝袜中文字幕| 国产精品人妻66p| av高潮迭起在线观看| 日本丰满熟妇BBXBBXHD| 国产欧美日韩在线观看不卡| 在线播放 日韩 av| 成年午夜影片国产片| 国产品国产三级国产普通话三级| 欧亚乱色一区二区三区| 美女视频福利免费看| 亚洲的电影一区二区三区| 一区二区视频视频视频| 亚洲福利精品福利精品福利| 久久久久久久久久性潮| 女同久久精品秋霞网| 亚洲av黄色在线网站| 国产又粗又硬又猛的毛片视频| 亚洲福利精品福利精品福利| 国产女人露脸高潮对白视频| 九九热99视频在线观看97| 亚洲精品一线二线在线观看| 美女操逼免费短视频下载链接| 天天夜天天日天天日| 888欧美视频在线| 国产精品免费不卡av| 在线观看av观看av| 最新91精品视频在线| av日韩在线免费播放| 99热这里只有精品中文| 欧美激情精品在线观看| 国产精品自拍在线视频| 亚洲免费在线视频网站| 红桃av成人在线观看| 国产九色91在线视频| 免费观看成年人视频在线观看| 成人国产激情自拍三区| 国产污污污污网站在线| 中文字幕1卡1区2区3区| 久久久极品久久蜜桃| 午夜美女福利小视频| 国产三级片久久久久久久| 99的爱精品免费视频| 姐姐的朋友2在线观看中文字幕| 噜噜色噜噜噜久色超碰| 少妇ww搡性bbb91| 日本xx片在线观看| 精品一区二区亚洲欧美| 日本高清成人一区二区三区| 综合页自拍视频在线播放| 青青草精品在线视频观看| 偷拍3456eee| 51国产成人精品视频| 成人av久久精品一区二区| 天天做天天干天天舔| 天天想要天天操天天干| 男人在床上插女人视频| 欧美天堂av无线av欧美| 激情伦理欧美日韩中文字幕| 亚洲美女美妇久久字幕组| av天堂中文免费在线| 亚洲青青操骚货在线视频| 做爰视频毛片下载蜜桃视频1| 欧美偷拍自拍色图片| 国产美女精品福利在线| 免费观看污视频网站| 护士特殊服务久久久久久久| 久草视频在线一区二区三区资源站 | 大香蕉福利在线观看| 青青草在观免费国产精品| 国产av自拍偷拍盛宴| 亚洲狠狠婷婷综合久久app| 少妇ww搡性bbb91| 香蕉av影视在线观看| 亚洲va天堂va国产va久| 在线观看av观看av| 强行扒开双腿猛烈进入免费版| 91精品资源免费观看| 偷拍自拍视频图片免费| 亚洲国产40页第21页| 久久综合老鸭窝色综合久久| 超碰中文字幕免费观看| 亚洲 欧美 精品 激情 偷拍| 5528327男人天堂| 国产乱子伦一二三区| 亚洲精品亚洲人成在线导航 | av天堂加勒比在线| 同居了嫂子在线播高清中文| 精品一区二区三区三区色爱| avjpm亚洲伊人久久| 特级欧美插插插插插bbbbb| 超黄超污网站在线观看| 亚洲综合在线观看免费| 11久久久久久久久久久| 男人在床上插女人视频| 亚洲偷自拍高清视频| 后入美女人妻高清在线| jiujiure精品视频在线| 五月激情婷婷久久综合网| 女警官打开双腿沦为性奴| 国产V亚洲V天堂无码欠欠| 国产精品自拍偷拍a| 日韩中文字幕在线播放第二页| 91一区精品在线观看| 国产品国产三级国产普通话三级| 大香蕉伊人国产在线| 91极品大一女神正在播放| 亚洲粉嫩av一区二区三区| av日韩在线观看大全| 免费黄色成人午夜在线网站| 亚洲一区二区三区精品视频在线| 欧美亚洲国产成人免费在线| 99久久久无码国产精品性出奶水 | 99re国产在线精品| 亚洲日产av一区二区在线| av视网站在线观看| 在线观看免费视频网| 婷婷色国产黑丝少妇勾搭AV| 青青擦在线视频国产在线| 夜女神免费福利视频| 亚洲成人三级在线播放| 在线免费视频 自拍| 九色精品视频在线播放| 亚国产成人精品久久久| 中文字幕AV在线免费看 | 亚洲欧美清纯唯美另类| 久久这里只有精彩视频免费| 国产精品亚洲在线观看| 午夜久久久久久久精品熟女| 亚洲码av无色中文| 久草视频在线免播放| 综合国产成人在线观看| 特级无码毛片免费视频播放| 91天堂精品一区二区| 操日韩美女视频在线免费看 | 国产一区二区欧美三区| 蜜臀av久久久久久久| 中文字幕欧美日韩射射一| 懂色av蜜桃a v| 亚洲蜜臀av一区二区三区九色| 一区二区三区视频,福利一区二区| 国产在线一区二区三区麻酥酥| 偷拍自拍福利视频在线观看| avjpm亚洲伊人久久| 北条麻妃高跟丝袜啪啪| 一区二区三区四区视频| 无套猛戳丰满少妇人妻| 亚洲成人激情视频免费观看了| www日韩a级s片av| 亚洲成人三级在线播放| 五十路熟女av天堂| 日本又色又爽又黄又粗| 国产av自拍偷拍盛宴| 丝袜亚洲另类欧美变态| 亚洲免费视频欧洲免费视频| 午夜福利人人妻人人澡人人爽| 人妻久久无码中文成人| 国产精品人妻熟女毛片av久| 日本性感美女写真视频| 女生自摸在线观看一区二区三区 | 中文字幕日韩无敌亚洲精品| 美女骚逼日出水来了| 老司机免费福利视频网| 3344免费偷拍视频| 人妻少妇精品久久久久久| 国产熟妇一区二区三区av| 都市激情校园春色狠狠| 欧美成人小视频在线免费看| 18禁无翼鸟成人在线| 大白屁股精品视频国产| 青娱乐极品视频青青草| 欧美男同性恋69视频| 91人妻人人做人人爽在线| 懂色av蜜桃a v| 天天日天天舔天天射进去| 青青社区2国产视频| 亚洲公开视频在线观看| 亚洲欧美自拍另类图片| 老司机福利精品免费视频一区二区| 国产精品人妻66p| 五月天色婷婷在线观看视频免费| 欧美色婷婷综合在线| 和邻居少妇愉情中文字幕| 无码中文字幕波多野不卡| 福利视频广场一区二区| 精品黑人巨大在线一区| 精品人妻一二三区久久| 激情综合治理六月婷婷| 欧美成人综合视频一区二区| 欧美天堂av无线av欧美| 免费看国产又粗又猛又爽又黄视频| 天天操夜夜操天天操天天操| 中文字幕av第1页中文字幕| 国产九色91在线视频| 天天日天天天天天天天天天天| 久久香蕉国产免费天天| 欧美aa一级一区三区四区 | 97黄网站在线观看| 中文字幕高清免费在线人妻| 男人的网址你懂的亚洲欧洲av| 视频 一区二区在线观看| 性色av一区二区三区久久久| 91中文字幕免费在线观看| 2020国产在线不卡视频| 热99re69精品8在线播放| 蜜桃视频17c在线一区二区| 欧美乱妇无乱码一区二区| 2020久久躁狠狠躁夜夜躁| 这里有精品成人国产99| 五十路老熟女码av| 国产精品伦理片一区二区| 男大肉棒猛烈插女免费视频| 鸡巴操逼一级黄色气| 欧美色婷婷综合在线| 久久久久五月天丁香社区| 欧美在线一二三视频| 一区二区三区四区中文| 午夜精品福利91av| 人妻无码中文字幕专区| 天堂资源网av中文字幕| 一区二区久久成人网| 亚洲精品av在线观看| 亚洲精品福利网站图片| 天天射夜夜操综合网| 亚洲精品亚洲人成在线导航| 日本熟妇一区二区x x| 欧美国品一二三产区区别| 日本熟妇一区二区x x| 免费看美女脱光衣服的视频| 北条麻妃肉色丝袜视频| 国产精品久久久久久久女人18| 久久久噜噜噜久久熟女av| 国产揄拍高清国内精品对白| 日本成人一区二区不卡免费在线| 国产精品亚洲在线观看| 大香蕉大香蕉大香蕉大香蕉大香蕉| 国产亚洲精品欧洲在线观看| 天天插天天色天天日| 日韩成人免费电影二区| 又大又湿又爽又紧A视频| 38av一区二区三区| 午夜福利人人妻人人澡人人爽| 韩国爱爱视频中文字幕| 制丝袜业一区二区三区| 91国内精品自线在拍白富美| 97黄网站在线观看| 蜜桃色婷婷久久久福利在线| gay gay男男瑟瑟在线网站| 91人妻精品一区二区久久| 在线观看免费视频网| 女生被男生插的视频网站| 91免费福利网91麻豆国产精品| 亚洲午夜伦理视频在线| 精彩视频99免费在线| 天码人妻一区二区三区在线看| 亚洲美女自偷自拍11页| 色av色婷婷人妻久久久精品高清| 午夜精品亚洲精品五月色| 在线观看免费岛国av| 天天日天天透天天操| 青青青青操在线观看免费| 国产久久久精品毛片| 毛茸茸的大外阴中国视频| 久久精品亚洲国产av香蕉| 91精品国产黑色丝袜| 国产高清97在线观看视频| 免费看国产又粗又猛又爽又黄视频| 亚洲Av无码国产综合色区| 免费人成黄页网站在线观看国产 | av乱码一区二区三区| 人妻少妇精品久久久久久| 亚洲中文字幕人妻一区| 制服丝袜在线人妻中文字幕| 99视频精品全部15| 日本高清撒尿pissing| 国产精品精品精品999| 啪啪啪操人视频在线播放| 欧美一区二区三区久久久aaa| 最新中文字幕免费视频| 欧美另类z0z变态| 国产精品熟女久久久久浪潮| 精品国产高潮中文字幕| 精品国产在线手机在线| 91人妻精品一区二区久久| 黄色资源视频网站日韩| 偷拍美女一区二区三区| 人妻另类专区欧美制服| 欧美aa一级一区三区四区| 懂色av之国产精品| 国产亚洲四十路五十路| 91精品一区二区三区站长推荐| 在线观看欧美黄片一区二区三区 | 久久久制服丝袜中文字幕| 国产成人小视频在线观看无遮挡| 亚洲欧美自拍另类图片| 国产免费av一区二区凹凸四季| 天天操天天污天天射| 青娱乐极品视频青青草| 日韩美女综合中文字幕pp| 亚洲中文字字幕乱码| 51国产成人精品视频| mm131美女午夜爽爽爽| 亚洲女人的天堂av| 国产卡一卡二卡三乱码手机| 亚洲av香蕉一区区二区三区犇| 亚洲天堂精品福利成人av| 狠狠操狠狠操免费视频| 午夜福利人人妻人人澡人人爽| 亚洲精品ww久久久久久| 欧洲亚洲欧美日韩综合| 日本性感美女三级视频| 污污小视频91在线观看| 精品久久久久久久久久久a√国产 日本女大学生的黄色小视频 | 美女骚逼日出水来了| 亚洲午夜电影之麻豆| 亚洲成人黄色一区二区三区| 精品高跟鞋丝袜一区二区| 青青青青青青青青青青草青青| 亚洲中文字幕人妻一区| 亚洲av无码成人精品区辽| 久久香蕉国产免费天天| 日本av在线一区二区三区| 激情小视频国产在线| 日韩美女搞黄视频免费| 亚洲视频在线视频看视频在线| 中文字幕之无码色多多| 在线免费观看日本片| 热思思国产99re| 国产超码片内射在线| 亚洲激情唯美亚洲激情图片| 馒头大胆亚洲一区二区| 91欧美在线免费观看| 北条麻妃肉色丝袜视频| 久久久久久99国产精品| 日韩写真福利视频在线观看| 天天日天天玩天天摸| 亚洲av自拍天堂网| 中文字母永久播放1区2区3区| 亚洲欧美精品综合图片小说| 91九色porny蝌蚪国产成人| 午夜国产福利在线观看| 国产日韩精品一二三区久久久 | 欧美精品激情在线最新观看视频| 成人亚洲精品国产精品| 免费岛国喷水视频在线观看| 99精品国自产在线人| 2017亚洲男人天堂| yy6080国产在线视频| 亚洲黄色av网站免费播放| 日韩国产乱码中文字幕| 天天日天天做天天日天天做| 人妻激情图片视频小说| 中文字幕成人日韩欧美| 操人妻嗷嗷叫视频一区二区| 日日爽天天干夜夜操| 中文字幕亚洲久久久| 三级黄色亚洲成人av| 午夜免费体验区在线观看| 91麻豆精品久久久久| 亚洲人妻视频在线网| 中文字幕一区二区三区蜜月| 亚洲2021av天堂| 亚洲人妻av毛片在线| 黄色男人的天堂视频| 日本美女性生活一级片| 天天想要天天操天天干| 精品乱子伦一区二区三区免费播| 欧美激情精品在线观看| 超pen在线观看视频公开97 | 大学生A级毛片免费视频| 精品国产在线手机在线| 又粗又硬又猛又爽又黄的| 搡老熟女一区二区在线观看| 久久精品36亚洲精品束缚| mm131美女午夜爽爽爽| 视频 一区二区在线观看| 一区二区在线视频中文字幕| 孕妇奶水仑乱A级毛片免费看| 青草久久视频在线观看| 日韩加勒比东京热二区| 国产日韩精品一二三区久久久| 天堂v男人视频在线观看| aⅴ精产国品一二三产品| 日韩黄色片在线观看网站| 久久美欧人妻少妇一区二区三区| 国产亚洲成人免费在线观看| 9色精品视频在线观看| 黑人进入丰满少妇视频| av无限看熟女人妻另类av| 天天日天天爽天天爽| 91小伙伴中女熟女高潮| 亚洲av色香蕉一区二区三区 | 1区2区3区不卡视频| 国产麻豆剧果冻传媒app| 免费成人av中文字幕| 国产黄网站在线观看播放| 亚洲免费av在线视频| 黄色男人的天堂视频| 中文字幕—97超碰网| 久久精品在线观看一区二区| 欧美激情精品在线观看| 国产 在线 免费 精品| 中文字幕 亚洲av| 亚洲国产欧美一区二区三区久久| 亚洲第一伊人天堂网| 欧美少妇性一区二区三区| 91老师蜜桃臀大屁股| 青青青青爽手机在线| 国产黄色a级三级三级三级| 亚洲卡1卡2卡三卡四老狼| 中文字幕一区二区人妻电影冢本 | 免费在线黄色观看网站| 成人高潮aa毛片免费| 精品久久久久久久久久久99| 国产真实灌醉下药美女av福利| 传媒在线播放国产精品一区| 亚洲一区二区三区久久午夜 | 大陆av手机在线观看| 99热色原网这里只有精品| 自拍偷拍亚洲欧美在线视频| 国产又粗又硬又大视频| 亚洲av日韩精品久久久久久hd| 人妻激情图片视频小说| 天堂中文字幕翔田av| 白白操白白色在线免费视频| 中文字幕午夜免费福利视频| 40道精品招牌菜特色| 香蕉91一区二区三区| 天天日天天玩天天摸| 最新中文字幕免费视频| 9国产精品久久久久老师| 国产中文精品在线观看| 懂色av蜜桃a v| 亚洲老熟妇日本老妇| 亚洲伊人av天堂有码在线| 午夜精品福利一区二区三区p| 亚洲变态另类色图天堂网| 5528327男人天堂| 国产av一区2区3区| 国产1区,2区,3区| 欧美精品久久久久久影院| 九九热99视频在线观看97| 999久久久久999| 亚洲av无码成人精品区辽| 亚洲蜜臀av一区二区三区九色| 久久尻中国美女视频| 男女第一次视频在线观看| 精品成人啪啪18免费蜜臀| 色综合久久久久久久久中文| 伊人综合免费在线视频| 亚洲久久午夜av一区二区| 午夜毛片不卡在线看| 久久美欧人妻少妇一区二区三区| 黑人性生活视频免费看| 国产成人精品福利短视频| 91久久综合男人天堂| 在线国产中文字幕视频| 特大黑人巨大xxxx| 欧美日韩一级黄片免费观看| 自拍 日韩 欧美激情| 18禁精品网站久久| 桃色视频在线观看一区二区| 天天操天天爽天天干| 早川濑里奈av黑人番号| 亚洲精品无码久久久久不卡| 天堂v男人视频在线观看| 欧美va亚洲va天堂va| 播放日本一区二区三区电影| 桃色视频在线观看一区二区| jiujiure精品视频在线| 大香蕉日本伊人中文在线| 日本一区二区三区免费小视频| av一区二区三区人妻| av中文字幕在线导航| 黄工厂精品视频在线观看| 在线国产中文字幕视频| 啊啊好慢点插舔我逼啊啊啊视频| 99精品视频在线观看免费播放 | 久久精品视频一区二区三区四区| 欧美日韩亚洲国产无线码| 国产男女视频在线播放| 亚洲另类伦春色综合小| 国产欧美精品免费观看视频| 亚洲国产精品美女在线观看| 97成人免费在线观看网站| 熟女人妻在线观看视频| 亚洲福利精品福利精品福利| 欧美精品资源在线观看| 日本欧美视频在线观看三区| 亚洲在线观看中文字幕av| 欧美va亚洲va天堂va| 中文字幕,亚洲人妻| 亚洲日本一区二区久久久精品| 偷拍自拍视频图片免费| 天天干天天日天天干天天操| 无码中文字幕波多野不卡| caoporn蜜桃视频| 国产白袜脚足J棉袜在线观看| 亚洲精品午夜久久久久| 动漫黑丝美女的鸡巴| 77久久久久国产精产品| 9l人妻人人爽人人爽| 亚洲免费在线视频网站| 天天日天天干天天舔天天射| 国产精品亚洲在线观看| 亚洲福利精品福利精品福利| 亚洲国产欧美一区二区三区…| 日韩精品中文字幕播放| 日本韩国亚洲综合日韩欧美国产| 大学生A级毛片免费视频| 欧美特级特黄a大片免费| 成年女人免费播放视频| av在线免费中文字幕| 在线观看av2025| 国产午夜亚洲精品不卡在线观看| 少妇高潮一区二区三区| 自拍偷拍 国产资源| 一区二区三区毛片国产一区| 亚洲一级特黄特黄黄色录像片| 亚洲最大黄了色网站| 啊啊好慢点插舔我逼啊啊啊视频| 骚货自慰被发现爆操| 成人亚洲精品国产精品| 日韩精品一区二区三区在线播放| 中文字幕奴隷色的舞台50| 九色精品视频在线播放| 久草视频 久草视频2| 中文字幕第1页av一天堂网| 97人妻总资源视频| 亚洲另类综合一区小说| 福利片区一区二体验区| av天堂中文字幕最新| 免费看美女脱光衣服的视频| 亚洲欧洲av天堂综合| 中文字幕乱码av资源| 水蜜桃国产一区二区三区| 性欧美日本大妈母与子| 美女少妇亚洲精选av| 日韩美女福利视频网| 日本熟妇色熟妇在线观看| 55夜色66夜色国产精品站| 综合国产成人在线观看| 国产乱子伦一二三区| 午夜大尺度无码福利视频| 55夜色66夜色国产精品站| 天堂av在线最新版在线| 欧美日韩国产一区二区三区三州| 深夜男人福利在线观看| 男女之间激情网午夜在线| 国际av大片在线免费观看| 亚洲日本一区二区三区| 硬鸡巴动态操女人逼视频| 黑人性生活视频免费看| av俺也去在线播放| 最近中文2019年在线看| 人妻凌辱欧美丰满熟妇| 视频二区在线视频观看| 激情国产小视频在线| 三级等保密码要求条款| 夜夜骑夜夜操夜夜奸| 亚洲综合另类精品小说| 插小穴高清无码中文字幕| 欧美综合婷婷欧美综合| 高清一区二区欧美系列| 中文字幕 亚洲av| 久久永久免费精品人妻专区 | 亚洲av自拍天堂网| 超碰97人人做人人爱| 婷婷五月亚洲综合在线| 久久久久久97三级| 熟女人妻在线观看视频| 国产精品一区二区三区蜜臀av| 色综合久久无码中文字幕波多| 久久农村老妇乱69系列| 国产av欧美精品高潮网站| 亚洲蜜臀av一区二区三区九色 | 超碰中文字幕免费观看| 超污视频在线观看污污污| 国产精品自拍在线视频| 99人妻视频免费在线| 免费在线观看视频啪啪| 国产精品人妻熟女毛片av久| 日韩美女福利视频网| 色花堂在线av中文字幕九九| 午夜蜜桃一区二区三区| 久久久久久久精品老熟妇| 亚洲一区二区人妻av| 天天草天天色天天干| 91免费福利网91麻豆国产精品| 亚洲欧美综合另类13p| 欧美熟妇一区二区三区仙踪林| 91chinese在线视频| 免费岛国喷水视频在线观看| 在线 中文字幕 一区| 国产精品中文av在线播放 | 日本免费一级黄色录像| 三级av中文字幕在线观看| 中国视频一区二区三区| 中文字幕一区二区自拍| 熟女91pooyn熟女| 欧美美女人体视频一区| 93精品视频在线观看| 国产妇女自拍区在线观看| 国产在线免费观看成人| 日本一本午夜在线播放| 欧美一区二区三区在线资源 | 中文字幕视频一区二区在线观看| 五十路在线观看完整版| 国产精品亚洲在线观看| 丰满少妇翘臀后进式| 九一传媒制片厂视频在线免费观看| 91大屁股国产一区二区| yellow在线播放av啊啊啊| 55夜色66夜色国产精品站| www骚国产精品视频| 国产之丝袜脚在线一区二区三区| 蜜臀av久久久久久久| 精产国品久久一二三产区区别| 午夜美女福利小视频| 日韩欧美国产一区ab| 天天干天天操天天玩天天射| 欧美视频中文一区二区三区| 久久久久久久久久一区二区三区| 制丝袜业一区二区三区| 91精品国产黑色丝袜| 久久久久久久亚洲午夜综合福利| 任你操视频免费在线观看| 成人sm视频在线观看| 国产午夜激情福利小视频在线| 91极品新人『兔兔』精品新作| 精品成人啪啪18免费蜜臀| 国产第一美女一区二区三区四区| 91片黄在线观看喷潮| 91成人精品亚洲国产| 可以在线观看的av中文字幕| 直接观看免费黄网站| 欧美精品黑人性xxxx| 91啪国自产中文字幕在线| 国产激情av网站在线观看| 久久精品亚洲国产av香蕉| 中文字幕第1页av一天堂网| 2o22av在线视频| 大鸡吧插逼逼视频免费看 | av手机在线免费观看日韩av| 在线观看国产免费麻豆| chinese国产盗摄一区二区| 男人操女人逼逼视频网站| 亚洲激情av一区二区| 性感美女高潮视频久久久| 色哟哟在线网站入口| 婷婷色国产黑丝少妇勾搭AV| 免费在线看的黄片视频| 精品视频中文字幕在线播放| 99婷婷在线观看视频| 91chinese在线视频| 国产精品女邻居小骚货| 黑人变态深video特大巨大| 精品美女在线观看视频在线观看 | 天天做天天爽夜夜做少妇| 日本丰满熟妇BBXBBXHD| 91av中文视频在线| 国产精品一二三不卡带免费视频| 人人妻人人人操人人人爽| 亚洲国产欧美一区二区三区久久| 18禁美女无遮挡免费| 97a片免费在线观看| 美女av色播在线播放| 精内国产乱码久久久久久| 日本真人性生活视频免费看| 顶级尤物粉嫩小尤物网站| 青青草国内在线视频精选| 2020av天堂网在线观看| 直接能看的国产av| 2018在线福利视频| 亚洲中文精品人人免费| 大香蕉大香蕉在线看| 97青青青手机在线视频 | 2012中文字幕在线高清| 91免费福利网91麻豆国产精品| 精品美女在线观看视频在线观看| 黄色的网站在线免费看| 日韩a级黄色小视频| 91久久精品色伊人6882| 硬鸡巴动态操女人逼视频| 亚洲天堂精品久久久| 91精品国产观看免费| 日本www中文字幕| 91自产国产精品视频| 老熟妇凹凸淫老妇女av在线观看| 久久久久久久久久久久久97| 欧美一区二区三区高清不卡tv| 含骚鸡巴玩逼逼视频| 3344免费偷拍视频| 女蜜桃臀紧身瑜伽裤| 亚洲欧美激情国产综合久久久| 亚洲天堂成人在线观看视频网站| 中文字幕第一页国产在线| 人妻最新视频在线免费观看| 国产女人露脸高潮对白视频| 色花堂在线av中文字幕九九 | 国产又粗又硬又猛的毛片视频 | 99av国产精品欲麻豆| 人妻激情图片视频小说| 欧美一区二区三区久久久aaa| 成年午夜免费无码区| 美女被肏内射视频网站| 国产精品久久久黄网站| 91亚洲手机在线视频播放| 99久久久无码国产精品性出奶水 | 在线观看成人国产电影| 99国内精品永久免费视频| 日本真人性生活视频免费看| 人人妻人人澡人人爽人人dvl| 中文字幕成人日韩欧美| 午夜大尺度无码福利视频 | 黄色成年网站午夜在线观看| 日本男女操逼视频免费看| 在线播放 日韩 av| av网址国产在线观看| 亚洲少妇高潮免费观看| 国产成人综合一区2区| 日本又色又爽又黄又粗| 亚洲av男人天堂久久| 人妻无码色噜噜狠狠狠狠色| 神马午夜在线观看视频| 国产精品一区二区久久久av| 一区二区三区久久中文字幕| 午夜美女少妇福利视频| 国产又粗又硬又猛的毛片视频 | 精品国产高潮中文字幕| 日韩伦理短片在线观看| 蜜桃色婷婷久久久福利在线| 亚洲一区二区三区av网站| 沙月文乃人妻侵犯中文字幕在线| 日本18禁久久久久久| 欧美中文字幕一区最新网址 | 日韩美在线观看视频黄| 一区国内二区日韩三区欧美| 大胆亚洲av日韩av| 日韩亚国产欧美三级涩爱| 亚洲人妻av毛片在线| 亚洲 图片 欧美 图片| 人妻熟女在线一区二区| 国产免费高清视频视频| 夜色撩人久久7777| 免费大片在线观看视频网站| 日韩精品一区二区三区在线播放| 中国产一级黄片免费视频播放| 国产女人露脸高潮对白视频| 99热国产精品666| 性欧美日本大妈母与子| 国产精品熟女久久久久浪潮| h国产小视频福利在线观看| 天天躁日日躁狠狠躁躁欧美av | av男人天堂狠狠干| 日韩一个色综合导航| 91福利视频免费在线观看| 午夜av一区二区三区| 亚洲精品无码久久久久不卡| av手机在线观播放网站| 国产亚洲欧美45p| 欧美日韩情色在线观看| 精品一区二区三四区| 国产黄色片在线收看| 亚洲va天堂va国产va久| 天天色天天舔天天射天天爽| 日本韩国免费福利精品| 日韩激情文学在线视频| 人妻丝袜榨强中文字幕| 亚洲天堂精品久久久| 加勒比视频在线免费观看| 91传媒一区二区三区| 适合午夜一个人看的视频| 美女福利视频网址导航| 香蕉片在线观看av| 欧美女同性恋免费a| 99av国产精品欲麻豆| 精品国产成人亚洲午夜| 91福利视频免费在线观看| 在线播放一区二区三区Av无码| 免费看国产av网站| 在线观看日韩激情视频| 欧美日韩精品永久免费网址| 韩国三级aaaaa高清视频| 欧美黑人性暴力猛交喷水| 亚洲欧美另类自拍偷拍色图| 亚洲日本一区二区久久久精品| 午夜在线观看一区视频| eeuss鲁片一区二区三区| 97瑟瑟超碰在线香蕉| 亚洲无码一区在线影院| 日韩亚洲高清在线观看| av在线免费中文字幕| 无码中文字幕波多野不卡| 黄网十四区丁香社区激情五月天| 婷婷色国产黑丝少妇勾搭AV| 91‖亚洲‖国产熟女| 蜜臀av久久久久蜜臀av麻豆| 五十路老熟女码av| 天天射夜夜操狠狠干| 天天通天天透天天插| 欧美视频不卡一区四区| 欲乱人妻少妇在线视频裸| 天天插天天狠天天操| 免费啪啪啪在线观看视频| 一区二区三区综合视频| 亚洲图片欧美校园春色| 99热这里只有精品中文| 国产精彩福利精品视频| 中文字幕日本人妻中出| 亚洲精品麻豆免费在线观看| 91传媒一区二区三区| 精品久久婷婷免费视频| 日本熟妇色熟妇在线观看| 国产午夜激情福利小视频在线| 91超碰青青中文字幕| 99精品国自产在线人| 动漫美女的小穴视频| 人人妻人人澡人人爽人人dvl| 蜜臀成人av在线播放| 国产视频网站一区二区三区 | 亚洲av自拍天堂网| jul—619中文字幕在线| 国产使劲操在线播放| 888亚洲欧美国产va在线播放| 香蕉aⅴ一区二区三区| 久久久久91精品推荐99| 欧美日韩中文字幕欧美| 精彩视频99免费在线| 国产精品久久久久国产三级试频| 六月婷婷激情一区二区三区| 日韩北条麻妃一区在线| 99国产精品窥熟女精品| 在线观看av2025| 天天躁日日躁狠狠躁躁欧美av| 老司机99精品视频在线观看 | 久草视频在线看免费| av手机在线观播放网站| 欧美亚洲国产成人免费在线| 久久久极品久久蜜桃| 又大又湿又爽又紧A视频| 中文字幕之无码色多多| 91精品一区二区三区站长推荐| 亚洲美女高潮喷浆视频| 99精品一区二区三区的区| 亚洲高清视频在线不卡| 在线观看免费视频色97| 亚洲av香蕉一区区二区三区犇| 免费在线看的黄网站| 日本xx片在线观看| 极品丝袜一区二区三区| 日日夜夜精品一二三| 国产九色91在线视频| 又色又爽又黄的美女裸体| 97人妻无码AV碰碰视频| 成人资源在线观看免费官网| 欧美80老妇人性视频| 亚洲一级 片内射视正片| 青青青青青青青在线播放视频| 动漫精品视频在线观看| 亚洲一区二区久久久人妻| 麻豆性色视频在线观看| 午夜免费观看精品视频| 欧美成人精品欧美一级黄色| 亚洲图库另类图片区| 91在线视频在线精品3| 国产伊人免费在线播放| 国产三级片久久久久久久| 午夜久久香蕉电影网| 美味人妻2在线播放| 青青擦在线视频国产在线| 小穴多水久久精品免费看| 3D动漫精品啪啪一区二区下载| 在线免费观看日本片| 久碰精品少妇中文字幕av| 中文字幕在线乱码一区二区| 日韩人妻在线视频免费| 熟女人妻在线中出观看完整版| 欧美日韩人妻久久精品高清国产 | 成人免费毛片aaaa| 在线观看黄色成年人网站| 首之国产AV医生和护士小芳| 一区二区三区美女毛片| www骚国产精品视频| 不卡精品视频在线观看| 日日夜夜大香蕉伊人| 亚洲国产中文字幕啊啊啊不行了| 999九九久久久精品| 亚洲天天干 夜夜操| 欧美香蕉人妻精品一区二区| 国产在线自在拍91国语自产精品| 性生活第二下硬不起来| 好男人视频在线免费观看网站| 最新91精品视频在线| 亚洲av一妻不如妾| 中文字幕国产专区欧美激情| 久久这里有免费精品| 国产91久久精品一区二区字幕| 天天爽夜夜爽人人爽QC| 人人超碰国字幕观看97| 啊啊啊视频试看人妻| 天天日天天透天天操| 少妇人妻久久久久视频黄片| 国产极品精品免费视频 | 国产在线观看免费人成短视频| 色婷婷六月亚洲综合香蕉| 丰满少妇翘臀后进式| 亚洲另类伦春色综合小| 啪啪啪18禁一区二区三区| 在线免费观看99视频| 天天爽夜夜爽人人爽QC| av久久精品北条麻妃av观看| 国产使劲操在线播放| 一区二区三区另类在线| 一区二区三区的久久的蜜桃的视频| 日韩不卡中文在线视频网站| 成年人午夜黄片视频资源| 又黄又刺激的午夜小视频| 黄色男人的天堂视频| 91精品啪在线免费| 好吊操视频这里只有精品| 在线观看免费岛国av| 国产三级片久久久久久久| 天天色天天操天天透| 亚洲 中文字幕在线 日韩| 国产在线免费观看成人| 午夜毛片不卡免费观看视频| 日日爽天天干夜夜操| 免费观看丰满少妇做受| 在线观看免费av网址大全| 日韩在线中文字幕色| 中文字幕在线免费第一页| 天天艹天天干天天操| 午夜频道成人在线91| 亚洲伊人久久精品影院一美女洗澡| 婷婷综合亚洲爱久久| 精品欧美一区二区vr在线观看 | 激情伦理欧美日韩中文字幕| 成人av在线资源网站| 亚洲国产欧美一区二区三区…| 91精品资源免费观看| 狠狠躁狠狠爱网站视频| 亚洲国产欧美一区二区三区久久| 欧美精产国品一二三区| 天天日天天日天天擦| 麻豆性色视频在线观看| 激情小视频国产在线 | 午夜毛片不卡在线看| 日韩美av高清在线| 少妇人妻100系列| 成年午夜免费无码区| 国产欧美精品一区二区高清| 看一级特黄a大片日本片黑人| 国产白嫩美女一区二区| 摧残蹂躏av一二三区| 日韩美av高清在线| 国产精品亚洲在线观看| 欧美伊人久久大香线蕉综合| 999久久久久999| 揄拍成人国产精品免费看视频| 久久午夜夜伦痒痒想咳嗽P| 日韩少妇人妻精品无码专区| 国产亚州色婷婷久久99精品| av天堂加勒比在线| 久久久久久国产精品| 99re久久这里都是精品视频| a v欧美一区=区三区| 亚洲一区二区久久久人妻| 国产91精品拍在线观看| 国产密臀av一区二区三| 人妻激情图片视频小说| 超碰97人人做人人爱| 白嫩白嫩美女极品国产在线观看| 大香蕉日本伊人中文在线| 六月婷婷激情一区二区三区| 精品av久久久久久久| 日本熟妇一区二区x x| av在线免费观看亚洲天堂| 亚洲激情唯美亚洲激情图片| 日本www中文字幕| 中文字幕在线一区精品| 强行扒开双腿猛烈进入免费版| 自拍偷拍 国产资源| 可以在线观看的av中文字幕| 日韩成人性色生活片| 91p0rny九色露脸熟女| 久久热这里这里只有精品| 天天做天天干天天操天天射| 一二三中文乱码亚洲乱码one| 亚洲精品午夜久久久久| 中文字幕高清免费在线人妻| 国产一区二区视频观看| 特级无码毛片免费视频播放| 精品亚洲中文字幕av| 真实国产乱子伦一区二区| 午夜在线精品偷拍一区二| 中文字幕人妻一区二区视频| 91she九色精品国产| 97a片免费在线观看| 亚洲高清国产拍青青草原| 91天堂精品一区二区| 在线免费观看欧美小视频| 中文字幕,亚洲人妻| 青青尤物在线观看视频网站| 亚洲成人黄色一区二区三区| av破解版在线观看| 国产高清在线观看1区2区| 欧美黑人性猛交xxxxⅹooo| 中文字母永久播放1区2区3区 | 2020中文字幕在线播放| 超级av免费观看一区二区三区| 久久久久久9999久久久久| 爱有来生高清在线中文字幕| 精品乱子伦一区二区三区免费播| 亚洲精品成人网久久久久久小说 | 一区二区三区四区五区性感视频| 在线免费观看av日韩| 国产免费高清视频视频| 国产在线免费观看成人| 中文字幕av第1页中文字幕| 天天日天天摸天天爱| 久草视频在线看免费| 亚洲日本一区二区久久久精品| av破解版在线观看| av中文在线天堂精品| 亚洲区美熟妇久久久久| 99热国产精品666| 日本免费视频午夜福利视频| 亚洲国产欧美一区二区三区久久| 国产综合精品久久久久蜜臀| 极品性荡少妇一区二区色欲| 黄色片黄色片wyaa| 亚洲专区激情在线观看视频| 天天日天天爽天天爽| 成人综合亚洲欧美一区| 91高清成人在线视频| 东京干手机福利视频| av中文字幕在线导航| 天天日天天摸天天爱| 亚洲国产香蕉视频在线播放| 亚洲欧美一区二区三区电影| 在线不卡日韩视频播放| 欧美综合婷婷欧美综合| 91人妻精品久久久久久久网站| 日日夜夜狠狠干视频| 热思思国产99re| 在线免费观看日本片| 国产性色生活片毛片春晓精品| 91免费放福利在线观看| 久久精品亚洲国产av香蕉| 国产视频精品资源网站| 国产精品入口麻豆啊啊啊 | 亚洲av自拍天堂网| 经典av尤物一区二区| wwwxxx一级黄色片| 午夜蜜桃一区二区三区| 中文字幕亚洲中文字幕| 精品视频中文字幕在线播放 | 97国产福利小视频合集| 亚洲无线观看国产高清在线| 97a片免费在线观看| 亚洲av午夜免费观看| 快点插进来操我逼啊视频| 亚洲图片偷拍自拍区| 一区二区三区另类在线 | 在线播放 日韩 av| 97超碰最新免费在线观看| 四川五十路熟女av| 性感美女诱惑福利视频| 大胸性感美女羞爽操逼毛片| 天天操天天弄天天射| 亚洲一区二区三区久久午夜| 午夜精品九一唐人麻豆嫩草成人| 欧美成人黄片一区二区三区| 在线 中文字幕 一区| 五色婷婷综合狠狠爱| 精品少妇一二三视频在线| 91中文字幕免费在线观看| 91大神福利视频网| 午夜久久香蕉电影网| 欧洲国产成人精品91铁牛tv| 在线观看的黄色免费网站| 欧美精品一二三视频| 中英文字幕av一区| 一个色综合男人天堂| 熟女人妻一区二区精品视频| 老司机福利精品免费视频一区二区| 中文字幕1卡1区2区3区| 日本a级视频老女人| 在线观看av2025| 国产一线二线三线的区别在哪| 亚洲成高清a人片在线观看| 亚洲成人激情视频免费观看了| 天天日天天操天天摸天天舔| 国产高清精品一区二区三区| 国产露脸对白在线观看| 天天干天天操天天爽天天摸| 亚洲成人免费看电影| 肏插流水妹子在线乐播下载| 免费男阳茎伸入女阳道视频| 91国内精品自线在拍白富美| 中文字幕av第1页中文字幕| 天天干夜夜操天天舔| 久久精品久久精品亚洲人| 亚洲超碰97人人做人人爱| 2021年国产精品自拍| 日本免费午夜视频网站| 93人妻人人揉人人澡人人| 亚洲av自拍偷拍综合| 天天日天天干天天干天天日| 人人人妻人人澡人人| 国产视频一区二区午夜| 人妻少妇亚洲精品中文字幕| 中文字幕av熟女人妻| 欧美亚洲自偷自拍 在线| 国产黑丝高跟鞋视频在线播放| 精品亚洲在线免费观看| 亚洲国产成人在线一区| 国产又色又刺激在线视频| 又粗又硬又猛又爽又黄的| 人妻另类专区欧美制服| av一区二区三区人妻| 欧美交性又色又爽又黄麻豆| 青青青国产免费视频| 一区二区三区日韩久久| 亚洲一级av无码一级久久精品| 大屁股熟女一区二区三区| 一区二区三区视频,福利一区二区 丰满的子国产在线观看 | 青娱乐蜜桃臀av色| 国产精品手机在线看片| 插逼视频双插洞国产操逼插洞| 80电影天堂网官网| 国产视频网站国产视频| 国产精品久久久黄网站| 亚洲av琪琪男人的天堂| 美女日逼视频免费观看| 中文字幕人妻一区二区视频| 亚洲国产欧美一区二区丝袜黑人| 亚洲青青操骚货在线视频| 青青青青青青青在线播放视频| av无限看熟女人妻另类av| 香蕉片在线观看av| 亚洲国产欧美国产综合在线| 美女张开腿让男生操在线看| 97精品成人一区二区三区| 懂色av蜜桃a v| 国产福利小视频二区| 日韩精品中文字幕在线| 91国产资源在线视频| 免费男阳茎伸入女阳道视频 | 班长撕开乳罩揉我胸好爽| 天天色天天爱天天爽| 美日韩在线视频免费看| 午夜免费观看精品视频| 久久久久久国产精品| 一区二区三区四区中文| 免费黄色成人午夜在线网站| 九九视频在线精品播放| 91精品国产黑色丝袜| 九九热99视频在线观看97| 亚洲免费成人a v| 国语对白xxxx乱大交| av在线播放国产不卡| 亚洲av天堂在线播放| av资源中文字幕在线观看| av资源中文字幕在线观看| 大屁股肉感人妻中文字幕在线| 做爰视频毛片下载蜜桃视频1| 亚洲av日韩精品久久久久久hd| 在线不卡成人黄色精品| 少妇露脸深喉口爆吞精| 大骚逼91抽插出水视频| 亚洲av天堂在线播放| 揄拍成人国产精品免费看视频| 日韩av大胆在线观看| 亚洲专区激情在线观看视频| 亚洲欧美日韩视频免费观看| 国产va在线观看精品| 青青伊人一精品视频| 一区二区三区国产精选在线播放| 亚洲免费国产在线日韩| 国产一区二区视频观看| 亚洲av自拍天堂网| 91啪国自产中文字幕在线| 久久香蕉国产免费天天| 欧美天堂av无线av欧美| 孕妇奶水仑乱A级毛片免费看| 天天摸天天日天天操| 亚洲欧美另类自拍偷拍色图| 日本av在线一区二区三区| 欧美精产国品一二三产品价格| 欧美亚洲中文字幕一区二区三区| 人妻少妇亚洲一区二区| 精品久久久久久高潮| 一本一本久久a久久精品综合不卡| 熟女妇女老妇一二三区| 久久久久久久久久久久久97| 中文字幕最新久久久| 韩国女主播精品视频网站| 午夜精品一区二区三区4| 欧美综合婷婷欧美综合| 国产女人叫床高潮大片视频| 人妻爱爱 中文字幕| 亚洲成人激情av在线| 一级黄片久久久久久久久| 1024久久国产精品| 大香蕉大香蕉大香蕉大香蕉大香蕉| 精内国产乱码久久久久久| 粗大的内捧猛烈进出爽大牛汉子| 久久香蕉国产免费天天| 国产精品人妻一区二区三区网站| 夜鲁夜鲁狠鲁天天在线| mm131美女午夜爽爽爽| 97人妻人人澡爽人人精品| 在线观看的黄色免费网站| 最后99天全集在线观看| 性欧美日本大妈母与子| www,久久久,com| 亚洲福利天堂久久久久久| 中英文字幕av一区| 中文字幕午夜免费福利视频| 天天干天天日天天谢综合156| 一区二区熟女人妻视频| 中文字幕欧美日韩射射一| 啊啊啊想要被插进去视频| 精品亚洲国产中文自在线| 大香蕉大香蕉大香蕉大香蕉大香蕉| 激情五月婷婷综合色啪| 青青青爽视频在线播放| 91欧美在线免费观看| 国产精品中文av在线播放 | 精品国产亚洲av一淫| 久久亚洲天堂中文对白| 成人sm视频在线观看| 91精品国产高清自在线看香蕉网| 大胸性感美女羞爽操逼毛片| 91天堂天天日天天操| 91小伙伴中女熟女高潮| 超级福利视频在线观看| 在线视频自拍第三页| 亚洲日本一区二区三区 | 久久久久久9999久久久久| 欧洲黄页网免费观看| 亚洲天天干 夜夜操| 久久久噜噜噜久久熟女av| 中文字幕成人日韩欧美| 可以免费看的www视频你懂的| 亚洲免费在线视频网站| 国产不卡av在线免费| 人妻丝袜精品中文字幕| 亚洲精品三级av在线免费观看| 亚洲成av人无码不卡影片一| 精彩视频99免费在线| 1769国产精品视频免费观看| 黄页网视频在线免费观看| 中文字幕一区二区自拍| 91大屁股国产一区二区| 无码日韩人妻精品久久| 欧美乱妇无乱码一区二区| 青青伊人一精品视频| 国产精品国产三级国产精东| 国产刺激激情美女网站| 中文字幕 码 在线视频| 国产福利在线视频一区| 国产视频在线视频播放| 欧美男人大鸡吧插女人视频| 亚洲卡1卡2卡三卡四老狼| 午夜免费观看精品视频| 色呦呦视频在线观看视频| 日本www中文字幕| 精品av国产一区二区三区四区 | 18禁美女无遮挡免费| 天天日天天爽天天爽| avjpm亚洲伊人久久| 自拍偷区二区三区麻豆| 国产刺激激情美女网站| 老师啊太大了啊啊啊尻视频| 99视频精品全部15| 国产麻豆91在线视频| 午夜成午夜成年片在线观看| 国产午夜福利av导航| 日韩一个色综合导航| 亚洲福利天堂久久久久久| a v欧美一区=区三区| 一区二区三区精品日本| 日韩在线中文字幕色| 欧美亚洲少妇福利视频| 五十路熟女人妻一区二| 欧美中国日韩久久精品| 欧美成人综合视频一区二区| 亚洲 清纯 国产com| 免费十精品十国产网站| 中文字幕在线视频一区二区三区| 天天干天天爱天天色| 东京热男人的av天堂| 老司机免费福利视频网| 91亚洲手机在线视频播放| 亚洲精品亚洲人成在线导航| 国产精品自拍视频大全| av破解版在线观看| 亚洲av无码成人精品区辽| 人妻自拍视频中国大陆| 午夜影院在线观看视频羞羞羞| 久久久久久久精品成人热| 亚洲av男人天堂久久| 国产熟妇乱妇熟色T区| 日本人妻少妇18—xx| 99热99这里精品6国产| 天天插天天色天天日| 久久精品国产999| 免费观看污视频网站| 亚洲国产40页第21页| 无码中文字幕波多野不卡| 老司机免费福利视频网| av欧美网站在线观看| 亚洲欧美人精品高清| 国产成人精品av网站| 最新中文字幕免费视频| 日本特级片中文字幕| 天天干天天日天天谢综合156| 亚洲第一黄色在线观看| 99久久激情婷婷综合五月天| 夜色撩人久久7777| 日韩二区视频一线天婷婷五| 农村胖女人操逼视频| 538精品在线观看视频| 大鸡吧插逼逼视频免费看| 新婚人妻聚会被中出| 天天操天天射天天操天天天| 成人av在线资源网站| 一区二区在线视频中文字幕 | 中文字幕高清在线免费播放 | 国产av国片精品一区二区| 高清成人av一区三区| 狠狠鲁狠狠操天天晚上干干| 亚洲av第国产精品| 99久久激情婷婷综合五月天| 18禁无翼鸟成人在线| 成人亚洲精品国产精品| 色吉吉影音天天干天天操| 男生用鸡操女生视频动漫|