状态过渡

Vue 的过渡系统提供了非常多简单的方法设置进入、离开和列表的动效。那么对于数据元素本身的动效呢,比如:

  • 数字和运算
  • 颜色的显示
  • SVG 节点的位置
  • 元素的大小和其他的 property

这些数据要么本身就以数值形式存储,要么可以转换为数值。有了这些数值后,我们就可以结合 Vue 的响应式和组件系统,使用第三方库来实现切换元素的过渡状态。

状态动画与侦听器

通过侦听器我们能监听到任何数值 property 的数值更新。可能听起来很抽象,所以让我们先来看看使用 GreenSock状态过渡 - 图1 (opens new window) 一个例子:

  1. <script src="https://cdnjs.cloudflare.com/ajax/libs/gsap/3.2.4/gsap.min.js"></script>
  2. <div id="animated-number-demo">
  3. <input v-model.number="number" type="number" step="20" />
  4. <p>{{ animatedNumber }}</p>
  5. </div>
  1. const Demo = {
  2. data() {
  3. return {
  4. number: 0,
  5. tweenedNumber: 0
  6. }
  7. },
  8. computed: {
  9. animatedNumber() {
  10. return this.tweenedNumber.toFixed(0)
  11. }
  12. },
  13. watch: {
  14. number(newValue) {
  15. gsap.to(this.$data, { duration: 0.5, tweenedNumber: newValue })
  16. }
  17. }
  18. }
  19. Vue.createApp(Demo).mount('#animated-number-demo')

更新数字时,更改将在输入下方设置动画。

动态状态过渡

就像 Vue 的过渡组件一样,数据背后状态过渡会实时更新,这对于原型设计十分有用。当你修改一些变量,即使是一个简单的 SVG 多边形也可实现很多难以想象的效果。

把过渡放到组件里

管理太多的状态过渡会很快的增加组件实例复杂性,幸好很多的动画可以提取到专用的子组件。我们来将之前的示例改写一下:

  1. <script src="https://cdnjs.cloudflare.com/ajax/libs/gsap/3.2.4/gsap.min.js"></script>
  2. <div id="app">
  3. <input v-model.number="firstNumber" type="number" step="20" /> +
  4. <input v-model.number="secondNumber" type="number" step="20" /> = {{ result }}
  5. <p>
  6. <animated-integer :value="firstNumber"></animated-integer> +
  7. <animated-integer :value="secondNumber"></animated-integer> =
  8. <animated-integer :value="result"></animated-integer>
  9. </p>
  10. </div>
  1. const app = Vue.createApp({
  2. data() {
  3. return {
  4. firstNumber: 20,
  5. secondNumber: 40
  6. }
  7. },
  8. computed: {
  9. result() {
  10. return this.firstNumber + this.secondNumber
  11. }
  12. }
  13. })
  14. app.component('animated-integer', {
  15. template: '<span>{{ fullValue }}</span>',
  16. props: {
  17. value: {
  18. type: Number,
  19. required: true
  20. }
  21. },
  22. data() {
  23. return {
  24. tweeningValue: 0
  25. }
  26. },
  27. computed: {
  28. fullValue() {
  29. return Math.floor(this.tweeningValue)
  30. }
  31. },
  32. methods: {
  33. tween(newValue, oldValue) {
  34. gsap.to(this.$data, {
  35. duration: 0.5,
  36. tweeningValue: newValue,
  37. ease: 'sine'
  38. })
  39. }
  40. },
  41. watch: {
  42. value(newValue, oldValue) {
  43. this.tween(newValue, oldValue)
  44. }
  45. },
  46. mounted() {
  47. this.tween(this.value, 0)
  48. }
  49. })
  50. app.mount('#app')

我们能在组件中结合使用这一节讲到各种过渡策略和 Vue 内建的过渡系统。总之,对于完成各种过渡动效几乎没有阻碍。

你可以看到我们如何使用它进行数据可视化,物理效果,角色动画和交互,天空是极限。

赋予设计以生命

只要一个动画,就可以带来生命。不幸的是,当设计师创建图标、logo 和吉祥物的时候,他们交付的通常都是图片或静态的 SVG。所以,虽然 GitHub 的章鱼猫、Twitter 的小鸟以及其它许多 logo 类似于生灵,它们看上去实际上并不是活着的。

Vue 可以帮到你。因为 SVG 的本质是数据,我们只需要这些动物兴奋、思考或警戒的样例。然后 Vue 就可以辅助完成这几种状态之间的过渡动画,来制作你的欢迎页面、加载指示、以及更加带有情感的提示。

Sarah Drasner 展示了下面这个 demo,这个 demo 结合了时间和交互相关的状态改变: