摘要:以前最早實現了一個類似的時間選擇插件,但是適用范圍太窄,索性最近要把這個實現方式發布出來,就重寫了一個高復用的組件。
以前最早實現了一個類似的時間選擇插件,但是適用范圍太窄,索性最近要把這個實現方式發布出來,就重寫了一個高復用的vue組件。
支持安卓4.0以上,safari 7以上
props
- 1
- 1
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節點 會自動隱藏)
盒子主要這句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結構也增加了對應的響應
- {{el.value}}
- {{el.value}}
如有不明白的地方,請在下方留言,或者郵箱聯系.k1868548@163.com
代碼還有優化空間,歡迎提出 謝謝
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/82831.html
摘要:實現原生效果若是想看原理,可以看這篇文章若是想直接上手用,這里的組件可以直接拿來用效果組件的運用騰訊地圖獲取當前位置想要使用騰訊地圖,首先當然得看騰訊地圖的開發指南我這邊使用的是方法一通過內嵌調用參照文檔寫的例子來用就行,關鍵是看你拿到數 vue 實現 ios 原生picker 效果 若是想看原理,可以看這篇文章https://www.oudahe.com/p/43627/若是想直接上...
摘要:我的書簽我的書簽謹慎導入,小心覆蓋工具類版本管理快速切換源配置教程指南可視化工具前端工具集前端助手網絡封包截取工具格式化工具標注工具模擬請求類深入淺出布局你所不知道的動畫技巧與細節常用代碼黑魔法小技巧,讓你少寫不必要的,代碼更優雅一勞永 我的書簽 我的書簽(謹慎導入,小心覆蓋) 工具類 nvm: node版本管理 nrm: 快速切換npm源 shell: zsh+on-my-zsh配...
摘要:我的書簽我的書簽謹慎導入,小心覆蓋工具類版本管理快速切換源配置教程指南可視化工具前端工具集前端助手網絡封包截取工具格式化工具標注工具模擬請求類深入淺出布局你所不知道的動畫技巧與細節常用代碼黑魔法小技巧,讓你少寫不必要的,代碼更優雅一勞永 我的書簽 我的書簽(謹慎導入,小心覆蓋) 工具類 nvm: node版本管理 nrm: 快速切換npm源 shell: zsh+on-my-zsh配...
閱讀 1332·2021-11-11 11:00
閱讀 3059·2021-09-24 09:47
閱讀 4977·2021-09-22 15:53
閱讀 966·2021-09-10 10:50
閱讀 3211·2021-09-01 11:40
閱讀 1170·2019-08-30 15:55
閱讀 479·2019-08-30 12:49
閱讀 1055·2019-08-29 17:12