在Vue中建立可重用的 Transition

做者: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>

圖片描述

看起來容易,對吧?然而,這種方法有一個問題。咱們不能在另外一個項目中真正重用這個transitiongit

封裝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組件同樣使用它。這比前面的例子稍微好一點,可是若是咱們想要傳遞其餘特定於transitionprop,好比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

顯式持續時間 prop

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...

如今,咱們能夠控制實際的可見過渡時間,這使咱們可重用的過渡變得靈活且易於使用。 可是,如何過渡多個元素(如列表項)呢?

Transition group 支持

你想到的最直接的方法多是建立一個新組件,好比fade-transition-group,而後將當前transition標籤替換爲transition-group標籤,以實現 group transition。若是咱們能夠在相同的組件中這樣作,並公開一個將切換到transition-group實現的group prop,那會怎麼樣呢?幸運的是,咱們能夠經過render函數或componentis屬性來實現這一點。

// 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組件,只需將其放入下一個項目中便可。

Vue Transition

在此以前描述的全部內容基本上都是這個小型 transition 集合所包含的內容。它有 10 個封裝的transition組件,每一個約1kb(縮小)。我認爲它很是方便,能夠輕鬆地在不一樣的項目中使用。你能夠試一試:)

總結

咱們從一個基本的過渡示例開始,並最終經過可調整的持續時間和transition-group支持來建立可重用的過渡組件。 咱們可使用這些技巧根據並根據自身的需求建立本身的過渡組件。 但願讀者從本文中學到了一些知識,而且能夠幫助大家創建功能更好的過渡組件。


代碼部署後可能存在的BUG無法實時知道,過後爲了解決這些BUG,花了大量的時間進行log 調試,這邊順便給你們推薦一個好用的BUG監控工具 Fundebug

原文:https://levelup.gitconnected....

交流

文章每週持續更新,能夠微信搜索「 大遷世界 」第一時間閱讀和催更(比博客早一到兩篇喲),本文 GitHub https://github.com/qq449245884/xiaozhi 已經收錄,整理了不少個人文檔,歡迎Star和完善,你們面試能夠參照考點複習,另外關注公衆號,後臺回覆福利,便可看到福利,你懂的。

相關文章
相關標籤/搜索