vue + any-touch實(shí)現(xiàn)一個(gè)iscroll 實(shí)現(xiàn)拖拽和滑動(dòng)動(dòng)畫(huà)效果
https://github.com/383514580/any-touch
先看demo
說(shuō)點(diǎn)濕的
iscroll其實(shí)代碼量挺大的(近2100行, 還有另一個(gè)類(lèi)似的庫(kù) betterScroll
他的代碼量和iscroll差不多, 因?yàn)樵矶际且粯拥?, 閱讀他們的代碼
發(fā)現(xiàn)里面很多邏輯 其實(shí)都是在做手勢(shì)判斷 , 比如拖拽(pan), 和劃(swipe), 還有部分元素(表單元素等)需要單獨(dú)判斷點(diǎn)擊(tap), 這部分代碼接近1/3, 所以我決定用自己開(kāi)發(fā)的手勢(shì)庫(kù)(any-touch)實(shí)現(xiàn)一個(gè)iscroll, 同時(shí)配合文字讓大家 最終都可以以最少的代碼實(shí)現(xiàn)一個(gè)iscroll .
vue
觀察了一段時(shí)間推薦排行, 發(fā)現(xiàn)大家都對(duì) vue 感興趣, 所以本次的"iscroll"將以vue組件的形式實(shí)現(xiàn), 同時(shí)我也希望借助vue強(qiáng)大的抽象能力, 讓最終代碼控制在500行以內(nèi) , 希望大家喜歡.
本文是個(gè)系列文章
本文先實(shí)現(xiàn)拖拽和滑動(dòng)動(dòng)畫(huà), 因?yàn)檫@2部分都依賴 手勢(shì) , 借此用最少的代碼先實(shí)現(xiàn)最核心的功能, 也讓大家對(duì)后續(xù)的內(nèi)容有信心.
簡(jiǎn)單說(shuō)下iscroll原理
添加2個(gè)div, 最內(nèi)的div(子div)通過(guò)設(shè)置css的transform的translate的值來(lái)模擬系統(tǒng)滾動(dòng)效果.
說(shuō)完邏輯再說(shuō)代碼
拖拽的時(shí)候通過(guò)panstart/panmove手勢(shì)返回的 位移增量 (deltaX/Y)進(jìn)行位置變化, 同時(shí)關(guān)閉動(dòng)畫(huà)效果.
發(fā)生快速劃(swipe)的時(shí)候, 開(kāi)啟動(dòng)畫(huà), 同時(shí)通過(guò)計(jì)算 目標(biāo)位置 和 動(dòng)畫(huà)時(shí)間 來(lái)觸發(fā)滑動(dòng)動(dòng)畫(huà).
代碼
<div class="any-scroll-view"> <div ref="body" :style="bodyStyle" class="any-scroll-view__body"><slot></slot></div> </div> .any-scroll-view { position: relative; width: 100%; height: 90vh; overflow: hidden; &__body { transition-timing-function: cubic-bezier(0.1, 0.57, 0.1, 1); background: #eee; position: absolute; width: 100%; height: 100%; } } import AnyTouch from 'any-touch'; export default { name: 'any-scroll-view', props: { // 減速度, 單位px/s² acceleration: { type: Number, default: 3600 } }, data() { return { scrollTop: 0, scrollLeft: 0, transitionDuration: 300 }; }, computed: { bodyStyle() { return { transitionDuration: `${this.transitionDuration}ms`, transform: `translate(${this.scrollLeft}px, ${ this.scrollTop }px)` }; } }, mounted() { const at = new AnyTouch(this.$el); // 第一次觸碰 at.on('inputstart', (ev) => { this.stopRoll(); }); // 拖拽開(kāi)始 at.on('panstart', (ev) => { this.move(ev); }); // 拖拽中 at.on('panmove', (ev) => { this.move(ev); }); // 快速滑動(dòng) at.on('swipe', (ev) => { this.decelerate(ev); }); this.$on('hook:destroy', () => { at.destroy(); }); }, methods: { // https://github.com/nolimits4web/swiper/blob/master/dist/js/swiper.esm.js#L87 // https://github.com/nolimits4web/Swiper/blob/master/src/utils/utils.js#L25 getCurrentTranslate() { const style = getComputedStyle(this.$refs.body, null); const { transform } = style; const array = transform.match(/(\-?)(\d)+(\.\d{0,})?/g); return { x: Math.round(array[4]), y: Math.round(array[5]) }; }, stopRoll() { const { x, y } = this.getCurrentTranslate(); this.moveTo({ scrollTop: y, scrollLeft: x }); }, /** * 移動(dòng)body * @param {Object} 拖拽產(chǎn)生的數(shù)據(jù) * @param {Number} deltaX: x軸位移變化 * @param {Number} deltaY: y軸位移變化 */ move({ deltaX, deltaY }, transitionDuration = 0) { this.transitionDuration = transitionDuration; this.scrollLeft += deltaX; this.scrollTop += deltaY; }, /** * 移動(dòng)到 */ moveTo({ scrollTop, scrollLeft }, transitionDuration = 0) { this.transitionDuration = transitionDuration; this.scrollLeft = scrollLeft; this.scrollTop = scrollTop; }, /** * 拖拽松手后減速移動(dòng)至停止 * velocityX/Y的單位是px/ms */ decelerate(ev) { const directionSign = { up: -1, right: 1, down: 1, left: -1 }[ ev.direction ]; // Top? | Left? let SCROLL_SUFFIX = 'Top'; // x ? | y? let AXIS_SUFFIX = 'Y'; if (ev.velocityX > ev.velocityY) { SCROLL_SUFFIX = 'Left'; AXIS_SUFFIX = 'X'; } // 減速時(shí)間, 單位ms // t = (v₂ - v₁) / a const velocity = ev[`velocity${AXIS_SUFFIX}`]; this.transitionDuration = Math.round( ((velocity * 1000) / this.acceleration) * 1000 ); // 滑動(dòng)距離 // s = (v₂² - v₁²) / (2 * a) const scrollAxis = `scroll${SCROLL_SUFFIX}`; this[scrollAxis] += directionSign * Math.round( Math.pow(velocity * 1000, 2) / (2 * this.acceleration) ); } } };
總結(jié)
以上所述是小編給大家介紹的vue + any-touch實(shí)現(xiàn)一個(gè)iscroll 實(shí)現(xiàn)拖拽和滑動(dòng)動(dòng)畫(huà)效果,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
相關(guān)文章
vue實(shí)現(xiàn)學(xué)生錄入系統(tǒng)之添加刪除功能
本文給大家?guī)?lái)一個(gè)小案例基于vue實(shí)現(xiàn)學(xué)生錄入系統(tǒng)功能,代碼簡(jiǎn)單易懂非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2018-07-07elementui?el-upload一次請(qǐng)求上傳多個(gè)文件的實(shí)現(xiàn)
使用ElementUI?Upload的時(shí)候發(fā)現(xiàn)如果是默認(rèn)方案,上傳多張圖片并不是真正的一次上傳多張,本文就來(lái)介紹一下elementui?el-upload一次請(qǐng)求上傳多個(gè)文件的實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下2023-10-10vue中使用vue-pdf組件實(shí)現(xiàn)文件預(yù)覽及相應(yīng)報(bào)錯(cuò)解決
在需求中,經(jīng)常遇見(jiàn)pdf的在線預(yù)覽效果,很多pdf插件不支持vue3,或者是沒(méi)有集成翻頁(yè)放大縮小功能,比如vue-pdf,下面這篇文章主要給大家介紹了關(guān)于vue中使用vue-pdf組件實(shí)現(xiàn)文件預(yù)覽及相應(yīng)報(bào)錯(cuò)解決的相關(guān)資料,需要的朋友可以參考下2022-09-09Vue實(shí)現(xiàn)一個(gè)動(dòng)態(tài)添加行的表格步驟詳解
在Vue組件中定義表格的數(shù)據(jù)模型,通常使用一個(gè)數(shù)組來(lái)存儲(chǔ)表格的數(shù)據(jù),每一行數(shù)據(jù)可以是一個(gè)對(duì)象,對(duì)象的屬性對(duì)應(yīng)表格的列,這篇文章主要介紹了Vue實(shí)現(xiàn)一個(gè)動(dòng)態(tài)添加行的表格步驟詳解,需要的朋友可以參考下2024-05-05vue3.0+element表格獲取每行數(shù)據(jù)代碼示例
這篇文章主要給大家介紹了關(guān)于vue3.0+element表格獲取每行數(shù)據(jù)的相關(guān)資料,在element-ui中,你可以通過(guò)為表格的行綁定單擊事件來(lái)獲取表格中的一行數(shù)據(jù),需要的朋友可以參考下2023-09-09Vuex模塊化和命名空間namespaced實(shí)例演示
這篇文章主要介紹了Vuex模塊化和命名空間namespaced的相關(guān)知識(shí),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-11-11詳解Vue3.0中ElementPlus<input輸入框自動(dòng)獲取焦點(diǎn)>
這篇文章主要給大家介紹了關(guān)于Vue3.0中ElementPlus<input輸入框自動(dòng)獲取焦點(diǎn)>的相關(guān)資料,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用vue3.0具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2023-04-04富文本編輯器quill.js?開(kāi)發(fā)之自定義插件示例詳解
這篇文章主要為大家介紹了富文本編輯器quill.js?開(kāi)發(fā)之自定義插件示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-08-08vue基于element-ui的三級(jí)CheckBox復(fù)選框功能的實(shí)現(xiàn)代碼
最近vue項(xiàng)目需要用到三級(jí)CheckBox復(fù)選框,需要實(shí)現(xiàn)全選反選不確定三種狀態(tài)。這篇文章主要介紹了vue基于element-ui的三級(jí)CheckBox復(fù)選框功能的實(shí)現(xiàn)方法,需要的朋友可以參考下2018-10-10