亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

怎么在vue中利用瀑布流組件實現一個上拉加載更多功能

發布時間:2021-04-16 17:19:08 來源:億速云 閱讀:149 作者:Leah 欄目:web開發

怎么在vue中利用瀑布流組件實現一個上拉加載更多功能?很多新手對此不是很清楚,為了幫助大家解決這個難題,下面小編將為大家詳細講解,有這方面需求的人可以來學習下,希望你能有所收獲。

<template>
 <div class="loadmore">
 <div class="loadmore__body">
  <slot></slot>
 </div>
 <div class="loadmore__footer">
  <span v-if="loading">
  <i class="tc-loading"></i>
  <span>正在加載</span>
  </span>
  <span v-else-if="loadable">上拉加載更多</span>
  <span v-else>沒有更多了</span>
 </div>
 </div>
</template>

然后就是業務部分了

在動手寫組件之前,先理清需求:

加載頁面 -> 滑到底部 -> 上拉一定距離 -> 加載第二頁 -> 繼續前面步驟 -> 沒有更多

這是一個用戶交互邏輯,而我們需要將其映射為代碼邏輯:

首屏自動加載第一頁 -> 滑動到底部&&按下時候滑動距離Y軸有一定偏移量 -> 請求后端加載第二頁 -> 根據返回字段判斷是否還有下一頁

有了代碼邏輯,主干就出來了,加載和判斷由事件來控制,而又作為一個vue組件,我們需要配合vue生命周期來掛載事件和銷毀事件

export default {
 mounted() {
  // 確定容器
  // 容器綁定事件
 },
 beforeDestory() {
  // 解綁事件
 },
}

如果沒有解綁的話,每次你加載組件,就會綁定一次事件…

然后我們需要一些核心事件回調方法來在合適的時間加載數據渲染頁面, 回想一下,第一我們需要http獲取數據的load函數,然后我們需要三個綁定事件的回調函數pointDown(), pointMove(), pointUp(),分別對應用戶按下、移動、彈起手指操作:

export default {
 ···
 methods:{
  /**
  * 加載一組數據的方法
  */
  load() {
   // 設置options
  this.$axios.request(options).then((res) => {
   // 獲取數據后的處理
  }).catch((e) => {
   // 異常處理
  })
  },
  /**
  * 鼠標按下事件處理函數
  * @param {Object} e - 事件對象
  */
  pointerdown(e) {
  // 獲取按下的位置
  this.pageY = e.changedTouches ? e.changedTouches[0].pageY : e.pageY
  },
  /**
  * 鼠標移動事件處理函數
  * @param {Object} e - 事件對象
  */
  pointermove(e) {
  const container = this.$container
  const pageY = e.changedTouches ? e.changedTouches[0].pageY : e.pageY
  const moveY = pageY - this.pageY

  // 如果已經向下滾動到頁面最底部
  if (moveY < 0 && (container.scrollTop + Math.min(
   global.innerHeight,
   container.clientHeight,
  )) >= container.scrollHeight) {
   // 阻止原生的上拉拖動會露出頁面底部空白區域的行為(主要針對iOS版微信)
   e.preventDefault()

   // 如果上拉距離超過50像素,則加載下一頁
   if (moveY < -50) {
   this.pageY = pageY
   this.load()
   }
  }
  },
  /**
  * 鼠標松開事件處理函數
  */
  pointerup() {
  // 這邊就是取消拖動狀態,需要注意在拖動過程中不要再次觸發一些事件回調,否側亂套
  this.dragging = false
  },
 },
 ···
}

基本上主干已經算完工了,一些props傳入或者一些邏輯控制細節需要再額外添加,貼出整個組件的源碼:

<template>
 <div class="loadmore">
 <!-- <div class="loadmore__header"></div> -->
 <div class="loadmore__body">
  <slot></slot>
 </div>
 <div class="loadmore__footer">
  <span v-if="loading">
  <i class="tc-loading"></i>
  <span>正在加載</span>
  </span>
  <span v-else-if="loadable">上拉加載更多</span>
  <span v-else>沒有更多了</span>
 </div>
 </div>
</template>

<script type="text/babel">
 import axios from 'axios'

 const CancelToken = axios.CancelToken

 export default {
 data() {
  return {
  /**
   * 總頁數(由服務端返回)
   * @type {number}
   */
  count: 0,

  /**
   * 是否正在拖拽中
   * @type {boolean}
   */
  dragging: false,

  /**
   * 已加載次數
   * @type {number}
   */
  times: 0,

  /**
   * 已開始記載
   * @type {boolean}
   */
  started: false,

  /**
   * 正在加載中
   * @type {boolean}
   */
  loading: false,
  }
 },

 props: {
  /**
  * 初始化后自動開始加載數據
  */
  autoload: {
  type: Boolean,
  default: true,
  },

  /**
  * 離組件最近的可滾動父級元素(用于監聽事件及獲取滾動條位置)
  */
  container: {
  // Selector or Element
  default: 'body',
  },

  /**
  * 禁用組件
  */
  disabled: {
  type: Boolean,
  default: false,
  },

  /**
  * Axios請求參數配置對象
  * {@link https://github.com/mzabriskie/axios#request-config}
  */
  options: {
  type: Object,
  default: null,
  },

  /**
  * 起始頁碼
  */
  page: {
  type: Number,
  default: 1,
  },

  /**
  * 每頁加載數據條數
  */
  rows: {
  type: Number,
  default: 10,
  },

  /**
  * 數據加載請求地址
  */
  url: {
  type: String,
  default: '',
  },
 },

 computed: {
  /**
  * 是否可以加載
  * @returns {boolean} 是與否
  */
  loadable() {
  return !this.disabled && (!this.started || (this.page + this.times) <= this.count)
  },
 },

 mounted() {
  let container = this.container

  if (container) {
  if (typeof container === 'string') {
   container = document.querySelector(container)
  } else if (!container.querySelector) {
   container = document.body
  }
  }

  if (!container) {
  container = document.body
  }

  this.$container = container
  this.onPointerDown = this.pointerdown.bind(this)
  this.onPointerMove = this.pointermove.bind(this)
  this.onPointerUp = this.pointerup.bind(this)

  if (global.PointerEvent) {
  container.addEventListener('pointerdown', this.onPointerDown, false)
  container.addEventListener('pointermove', this.onPointerMove, false)
  container.addEventListener('pointerup', this.onPointerUp, false)
  container.addEventListener('pointercancel', this.onPointerUp, false)
  } else {
  container.addEventListener('touchstart', this.onPointerDown, false)
  container.addEventListener('touchmove', this.onPointerMove, false)
  container.addEventListener('touchend', this.onPointerUp, false)
  container.addEventListener('touchcancel', this.onPointerUp, false)
  container.addEventListener('mousedown', this.onPointerDown, false)
  container.addEventListener('mousemove', this.onPointerMove, false)
  container.addEventListener('mouseup', this.onPointerUp, false)
  }

  if (this.autoload) {
  this.load()
  }
 },

 // eslint-disable-next-line
 beforeDestroy() {
  const container = this.$container

  if (global.PointerEvent) {
  container.removeEventListener('pointerdown', this.onPointerDown, false)
  container.removeEventListener('pointermove', this.onPointerMove, false)
  container.removeEventListener('pointerup', this.onPointerUp, false)
  container.removeEventListener('pointercancel', this.onPointerUp, false)
  } else {
  container.removeEventListener('touchstart', this.onPointerDown, false)
  container.removeEventListener('touchmove', this.onPointerMove, false)
  container.removeEventListener('touchend', this.onPointerUp, false)
  container.removeEventListener('touchcancel', this.onPointerUp, false)
  container.removeEventListener('mousedown', this.onPointerDown, false)
  container.removeEventListener('mousemove', this.onPointerMove, false)
  container.removeEventListener('mouseup', this.onPointerUp, false)
  }

  if (this.loading && this.cancel) {
  this.cancel()
  }
 },

 methods: {
  /**
  * 加載一組數據的方法
  */
  load() {
  if (this.disabled || this.loading) {
   return
  }

  this.started = true
  this.loading = true

  const params = {
   currentPage: this.page + this.times,
   pageSize: this.rows,
  }
  const options = Object.assign({}, this.options, {
   url: this.url,
   cancelToken: new CancelToken((cancel) => {
   this.cancel = cancel
   }),
  })

  if (String(options.method).toUpperCase() === 'POST') {
   options.data = Object.assign({}, options.data, params)
  } else {
   options.params = Object.assign({}, options.params, params)
  }

  this.$axios.request(options).then((res) => {
   const data = res.result

   this.times += 1
   this.loading = false
   this.count = data.pageCount
   this.$emit('success', data.list)
   this.$emit('complete')
  }).catch((e) => {
   this.loading = false
   this.$emit('error', e)
   this.$emit('complete')
  })
  },

  /**
  * 重置加載相關變量
  */
  reset() {
  this.count = 0
  this.times = 0
  this.started = false
  this.loading = false
  },

  /**
  *重新開始加載
  */
  restart() {
  this.reset()
  this.load()
  },

  /**
  * 鼠標按下事件處理函數
  * @param {Object} e - 事件對象
  */
  pointerdown(e) {
  if (this.disabled || !this.loadable || this.loading) {
   return
  }

  this.dragging = true
  this.pageY = e.changedTouches ? e.changedTouches[0].pageY : e.pageY
  },

  /**
  * 鼠標移動事件處理函數
  * @param {Object} e - 事件對象
  */
  pointermove(e) {
  if (!this.dragging) {
   return
  }

  const container = this.$container
  const pageY = e.changedTouches ? e.changedTouches[0].pageY : e.pageY
  const moveY = pageY - this.pageY

  // 如果已經向下滾動到頁面最底部
  if (moveY < 0 && (container.scrollTop + Math.min(
   global.innerHeight,
   container.clientHeight,
  )) >= container.scrollHeight) {
   // 阻止原生的上拉拖動會露出頁面底部空白區域的行為(主要針對iOS版微信)
   e.preventDefault()

   // 如果上拉距離超過50像素,則加載下一頁
   if (moveY < -50) {
   this.pageY = pageY
   this.load()
   }
  }
  },

  /**
  * 鼠標松開事件處理函數
  */
  pointerup() {
  this.dragging = false
  },
 },
 }
</script>

看完上述內容是否對您有幫助呢?如果還想對相關知識有進一步的了解或閱讀更多相關文章,請關注億速云行業資訊頻道,感謝您對億速云的支持。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

vue
AI

南乐县| 蓝山县| 民权县| 澳门| 上林县| 英吉沙县| 天柱县| 进贤县| 威海市| 广灵县| 永德县| 兴隆县| 章丘市| 浏阳市| 广河县| 镶黄旗| 莱西市| 平武县| 金沙县| 克拉玛依市| 荔浦县| 巴南区| 稻城县| 玛纳斯县| 同心县| 巴中市| 措美县| 合作市| 康马县| 漯河市| 密山市| 台南市| 雅江县| 仪征市| 尖扎县| 安丘市| 汝阳县| 玉屏| 犍为县| 青阳县| 惠州市|