做者:Matt
譯者:前端小智
來源:medium
點贊再看,養成習慣本文
GitHub
https://github.com/qq44924588... 上已經收錄,更多往期高贊文章的分類,也整理了不少個人文檔,和教程資料。歡迎Star和完善,你們面試能夠參照考點複習,但願咱們一塊兒有點東西。css
Vue.js中的transition
確實很棒。 毫無疑問,它們能夠很是輕鬆地讓應用程序栩栩如生,可是一般必須在每一個項目中從頭開始編寫它們,甚至還須要引入animate.css
之類的CSS庫來使它們功能更強大。html
若是咱們能夠將它們封裝到組件中,並在多個項目中簡單地重用它們,結果會怎樣呢?咱們將介紹幾種定義transition
的方法,並深刻研究如何使它們真正可重用。前端
transition
組件和CSS定義transition
的最簡單方法是使用transition·或
transition-group 組件。這須要爲
transition定義一個
name`和一些CSS。vue
<template> <div id="app"> <button v-on:click="show = !show"> Toggle </button> <transition name="fade"> <p v-if="show">hello</p> </transition> </div> </template> <script> export default { name: "App", data() { return { show: true }; } }; </script> <style> .fade-enter-active, .fade-leave-active { transition: opacity 0.3s; } .fade-enter, .fade-leave-to { opacity: 0; } </style>
看起來容易,對吧?然而,這種方法有一個問題。咱們不能在另外一個項目中真正重用這個transition
。git
transition
組件若是咱們將前面的邏輯封裝到一個組件中,並將其用做一個組件,結果會怎樣呢?github
// FadeTransition.vue <template> <transition name="fade"> <slot></slot> </transition> </template> <script> export default { }; </script> <style> .fade-enter-active, .fade-leave-active { transition: opacity 0.3s; } .fade-enter, .fade-leave-to { opacity: 0; } </style> // App.vue <template> <div id="app"> <button v-on:click="show = !show"> Toggle transition </button> <fade-transition> <div v-if="show" class="box"></div> </fade-transition> </div> </template> <script>...</script> <style>...</style>
經過在transition
組件中提供一個slot
,咱們幾乎能夠像使用基本transition
組件同樣使用它。這比前面的例子稍微好一點,可是若是咱們想要傳遞其餘特定於transition
的prop
,好比mode
或者一些hook,該怎麼辦呢面試
transition
組件幸運的是,Vue 中有一個功能,使咱們能夠將用戶指定的全部額外props
和監聽器傳遞給咱們的內部標籤/組件。 若是你還不知道,則能夠經過$attrs
訪問額外傳遞的 props,並將它們與v-bind
結合使用以將它們綁定爲props。 這一樣適用於經過$listeners
進行的事件,並經過v-on
對其進行應用。微信
// FadeTransition.vue <template> <transition name="fade" v-bind="$attrs" v-on="$listeners"> <slot></slot> </transition> </template> <script> export default {}; </script> <style> .fade-enter-active, .fade-leave-active { transition: opacity 0.3s; } .fade-enter, .fade-leave-to { opacity: 0; } </style> // App.vue ... <fade-transition mode="out-in"> <div key="blue" v-if="show" class="box"></div> <div key="red" v-else class="red-box"></div> </fade-transition> ...
完整事例地址:https://codesandbox.io/s/yjl1...app
如今,咱們能夠傳遞普通transition
組件能夠接受的任何事件和支持,這使得咱們的組件更加可重用。但爲何不更進一步,增長經過 prop 輕鬆定製持續時間的可能性。jvm
Vue 爲transition
組件提供了一個duration
prop,然而,它是爲更復雜的動畫連接而設計的,它幫助 Vue 正確地將它們連接在一塊兒。
在咱們的案例中,咱們真正須要的是經過組件prop
控制CSS animation/transition。 咱們能夠經過不在CSS中指定顯式的CSS動畫持續時間,而是將其做爲樣式來實現。 咱們能夠藉助transition hook
來作到這一點,該transition hook
與組件生命週期 hook 很是類似,可是它們在過渡所需元素以前和以後被調用。 讓咱們看看效果如何。
// FadeTransition.vue <template> <transition name="fade" enter-active-class="fadeIn" leave-active-class="fadeOut" v-bind="$attrs" v-on="hooks"> <slot></slot> </transition> </template> <script> export default { props: { duration: { type: Number, default: 300 } }, computed: { hooks() { return { beforeEnter: this.setDuration, afterEnter: this.cleanUpDuration, beforeLeave: this.setDuration, afterLeave: this.cleanUpDuration, ...this.$listeners }; } }, methods: { setDuration(el) { el.style.animationDuration = `${this.duration}ms`; }, cleanUpDuration(el) { el.style.animationDuration = ""; } } }; </script> <style> @keyframes fadeIn { from { opacity: 0; } to { opacity: 1; } } .fadeIn { animation-name: fadeIn; } @keyframes fadeOut { from { opacity: 1; } to { opacity: 0; } } .fadeOut { animation-name: fadeOut; } </style>
完整事例地址:https://codesandbox.io/s/j4qn...
如今,咱們能夠控制實際的可見過渡時間,這使咱們可重用的過渡變得靈活且易於使用。 可是,如何過渡多個元素(如列表項)呢?
你想到的最直接的方法多是建立一個新組件,好比fade-transition-group
,而後將當前transition
標籤替換爲transition-group
標籤,以實現 group transition。若是咱們能夠在相同的組件中這樣作,並公開一個將切換到transition-group
實現的group
prop,那會怎麼樣呢?幸運的是,咱們能夠經過render
函數或component
和is
屬性來實現這一點。
// FadeTransition.vue <template> <component :is="type" :tag="tag" enter-active-class="fadeIn" leave-active-class="fadeOut" move-class="fade-move" v-bind="$attrs" v-on="hooks"> <slot></slot> </component> </template> <script> export default { props: { duration: { type: Number, default: 300 }, group: { type: Boolean, default: false }, tag: { type: String, default: "div" } }, computed: { type() { return this.group ? "transition-group" : "transition"; }, hooks() { return { beforeEnter: this.setDuration, afterEnter: this.cleanUpDuration, beforeLeave: this.setDuration, afterLeave: this.cleanUpDuration, leave: this.setAbsolutePosition, ...this.$listeners }; } }, methods: { setDuration(el) { el.style.animationDuration = `${this.duration}ms`; }, cleanUpDuration(el) { el.style.animationDuration = ""; }, setAbsolutePosition(el) { if (this.group) { el.style.position = "absolute"; } } } }; </script> <style> @keyframes fadeIn { from { opacity: 0; } to { opacity: 1; } } .fadeIn { animation-name: fadeIn; } @keyframes fadeOut { from { opacity: 1; } to { opacity: 0; } } .fadeOut { animation-name: fadeOut; } .fade-move { transition: transform 0.3s ease-out; } </style> // App.vue ... <div class="box-wrapper"> <fade-transition group :duration="300"> <div class="box" v-for="(item, index) in list" @click="remove(index)" :key="item" > </div> </fade-transition> </div> ...
完整事例地址:https://codesandbox.io/s/pk9r...
文檔中介紹了一個帶有transition-group
元素的警告。 咱們基本上必須在元素離開時將每一個項目的定位設置爲absolute
,以實現其餘項目的平滑移動動畫。 咱們也必須添加一個move-class
並手動指定過渡持續時間,由於沒有用於移動的 JS hook。咱們將這些調整添加到咱們的上一個示例中。
再作一些調整,經過在mixin
中提取 JS 邏輯,咱們能夠將其應用於輕鬆建立新的transition
組件,只需將其放入下一個項目中便可。
在此以前描述的全部內容基本上都是這個小型 transition 集合所包含的內容。它有 10 個封裝的transition
組件,每一個約1kb(縮小)。我認爲它很是方便,能夠輕鬆地在不一樣的項目中使用。你能夠試一試:)
咱們從一個基本的過渡示例開始,並最終經過可調整的持續時間和transition-group
支持來建立可重用的過渡組件。 咱們可使用這些技巧根據並根據自身的需求建立本身的過渡組件。 但願讀者從本文中學到了一些知識,而且能夠幫助大家創建功能更好的過渡組件。
代碼部署後可能存在的BUG無法實時知道,過後爲了解決這些BUG,花了大量的時間進行log 調試,這邊順便給你們推薦一個好用的BUG監控工具 Fundebug。
原文:https://levelup.gitconnected....
文章每週持續更新,能夠微信搜索「 大遷世界 」第一時間閱讀和催更(比博客早一到兩篇喲),本文 GitHub https://github.com/qq449245884/xiaozhi 已經收錄,整理了不少個人文檔,歡迎Star和完善,你們面試能夠參照考點複習,另外關注公衆號,後臺回覆福利,便可看到福利,你懂的。