在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和完善,你们面试能够参照考点复习,另外关注公众号,后台回复福利,便可看到福利,你懂的。

相关文章
相关标签/搜索