1. 程式人生 > >vue 實現 ios 原生picker 效果(實現思路分析)

vue 實現 ios 原生picker 效果(實現思路分析)

sta 效果 cursor touchend orm dex tex sub alt

以前最早實現了一個類似的時間選擇插件,但是適用範圍太窄,索性最近要把這個實現方式發布出來,就重寫了一個高復用的vue組件。

支持安卓4.0以上,safari 7以上
技術分享

效果預覽

gitHub

滾輪部分主要dom結構

<template>
  <div class="pd-select-item">
    <div class="pd-select-line"></div>
    <ul class="pd-select-list">
      <li class="pd-select-list-item">1</li>
    </ul>
    <ul class="pd-select-wheel">
      <li class="pd-select-wheel-item">1</li>
    </ul>
  </div>
</template>

props

 props: {
      data: {
        type: Array,
        required: true
      },
      type: {
        type: String,
        default: ‘cycle‘
      },
      value: {}
    }

設置css樣式 使其垂直居中

.pd-select-line, .pd-select-list, .pd-select-wheel {
    position: absolute;
    left: 0;
    right: 0;
    top: 50%;
    transform: translateY(-50%);
}
.pd-select-list {
    overflow: hidden;
}

滾輪3d樣式設置

/* 滾輪盒子 */
.pd-select-wheel {
    transform-style: preserve-3d;
    height: 30px;
}
/* 滾輪單項 */
.pd-select-wheel-item {
    white-space: nowrap;
    text-overflow: ellipsis;
    backface-visibility: hidden;
    position: absolute;
    top: 0px;
    width: 100%;
    overflow: hidden;
}

技術分享

主要註意2個屬性 transform-style: preserve-3d;

backface-visibility: hidden;
第一個是3d布局,讓界面3D化,第二個是讓滾輪背後自動隱藏(上圖紅色部分,背面的dom節點 會自動隱藏)

如何實現3D 滾輪

盒子主要這句css transform: rotate3d(1, 0, 0, x deg);
item主要運用這句css transform: rotate3d(1, 0, 0, xdeg) translate3d(0px, 0px, [x]px);

技術分享

技術分享
技術分享

上面2張圖展示了translate3d(0px, 0px, [x]px);這句話的效果 [x]就是圓的半徑

技術分享

從上面的圖可以看見,我們只需旋轉每個dom自身,然後利用translate3d(0px, 0px, [x]px);把每個dom擴展開
就形成了圓環.α就是每個dom自身旋轉的角度,因為這裏只用了0到180°,所以用了個盒子在裝這些dom

行高 和角度計算

技術分享
已知兩邊和夾角 算第三邊長度 ~=34px
http://tool.520101.com/calcul...

無限滾輪實現

/* 滾輪展示大小限定 */
spin: {start: 0, end: 9, branch: 9}

/* 獲取spin 數據 */
 getSpinData (index) {
   index = index % this.listData.length
   return this.listData[index >= 0 ? index : index + this.listData.length]
 }
 /* 模運算 獲取數組有的索引 這樣就構成 圓環了 */

touchend做特殊處理

在touchend 裏設置setCSS類型 把滾動數據取整,這樣停止的時候就是一格一格的準確轉動到位

 // other code ....
 /* 計算touchEnd移動的整數距離 */
        let endMove = margin
        let endDeg = Math.round(updateDeg / deg) * deg
        if (type === ‘end‘) {
          this.setListTransform(endMove, margin)
          this.setWheelDeg(endDeg)
        } else {
          this.setListTransform(updateMove, margin)
          this.setWheelDeg(updateDeg)
        }
  // other code ....

慣性緩動

// other code ....
setWheelDeg (updateDeg, type, time = 1000) {
        if (type === ‘end‘) {
          this.$refs.wheel.style.webkitTransition = `transform ${time}ms cubic-bezier(0.19, 1, 0.22, 1)`
          this.$refs.wheel.style.webkitTransform = `rotate3d(1, 0, 0, ${updateDeg}deg)`
        } else {
          this.$refs.wheel.style.webkitTransition = ‘‘
          this.$refs.wheel.style.webkitTransform = `rotate3d(1, 0, 0, ${updateDeg}deg)`
        }
      }
setListTransform (translateY = 0, marginTop = 0, type, time = 1000) {
        if (type === ‘end‘) {
          this.$refs.list.style.webkitTransition = `transform ${time}ms cubic-bezier(0.19, 1, 0.22, 1)`
          this.$refs.list.style.webkitTransform = `translateY(${translateY - this.spin.branch * 34}px)`
          this.$refs.list.style.marginTop = `${-marginTop}px`
          this.$refs.list.setAttribute(‘scroll‘, translateY)
          console.log(‘end‘)
        } else {
          this.$refs.list.style.webkitTransition = ‘‘
          this.$refs.list.style.webkitTransform = `translateY(${translateY - this.spin.branch * 34}px)`
          this.$refs.list.style.marginTop = `${-marginTop}px`
          this.$refs.list.setAttribute(‘scroll‘, translateY)
        }
}
// other code ....

獲取當前選中值


 /* 在設置完css後獲取值  */
 
setStyle (move, type, time) {
   // ...other code
   /* 設置$emit 延遲 */
   setTimeout(() => this.getPickValue(endMove), 1000)
  // ...other code
}

/* 獲取選中值 */
      getPickValue (move) {
        let index = Math.abs(move / 34)
        let pickValue = this.getSpinData(index)
        this.$emit(‘input‘, pickValue)
      }

初始化設置

 mounted () {
      /* 事件綁定 */
      this.$el.addEventListener(‘touchstart‘, this.itemTouchStart)
      this.$el.addEventListener(‘touchmove‘, this.itemTouchMove)
      this.$el.addEventListener(‘touchend‘, this.itemTouchEnd)
      /* 初始化狀態 */
      let index = this.listData.indexOf(this.value)
      if (index === -1) {
        console.warn(‘當前初始值不存在,請檢查後listData範圍!!‘)
        this.setListTransform()
        this.getPickValue(0)
      } else {
        let move = index * 34
        /* 因為往上滑動所以是負 */
        this.setStyle(-move)
        this.setListTransform(-move, -move)
      }

當展示為非無限滾輪的時

這裏我們很好判斷,就是滾動的距離不能超過原始數的數組長度*34,且不能小於0(實際代碼中涉及方向)

/* 根據滾輪類型 line or cycle 判斷 updateMove最大距離 */
        if (this.type === ‘line‘) {
          if (updateMove > 0) {
            updateMove = 0
          }
          if (updateMove < -(this.listData.length - 1) * singleHeight) {
            updateMove = -(this.listData.length - 1) * singleHeight
          }
        }
 /* 根據type 控制滾輪顯示效果 */
      setHidden (index) {
        if (this.type === ‘line‘) {
          return index < 0 || index > this.listData.length - 1
        } else {
          return false
        }
      },

dom結構也增加了對應的響應

<div class="pd-select-item">
    <div class="pd-select-line"></div>
    <div class="pd-select-list">
      <ul class="pd-select-ul" ref="list">
        <li class="pd-select-list-item" v-for="el,index in renderData " :class="{‘hidden‘:setHidden(el.index)}" :key="index">{{el.value}}</li>
      </ul>
    </div>
    <ul class="pd-select-wheel" ref="wheel">
      <li class="pd-select-wheel-item" :class="{‘hidden‘:setHidden(el.index)}" :style="setWheelItemDeg(el.index)" :index="el.index" v-for="el,index in renderData " :key="index">{{el.value}}</li>
    </ul>
  </div>

vue 實現 ios 原生picker 效果(實現思路分析)