如何实现 “灭霸” 响指动效

236 阅读8分钟
原文链接: imcuttle.github.io

效果一瞥

使用 snap-fade-away 能够直接灭霸动效

Bye bye

imcuttle 🐟

import snapFadeAway from 'snap-fade-away'

export default class extends Component {
    state = {
        animating: false
    }
    nodeRef;

    fadeAway = async () => {
        this.setState({animating: true})
        await snapFadeAway(this.nodeRef)
        this.setState({animating: false})
    }

    reset = () => {
        this.nodeRef.style.visibility = 'visible'
    }

    render() {
        return <div>
            <button disabled={this.state.animating} onClick={this.fadeAway}>FadeAway</button>
            <button disabled={this.state.animating} onClick={this.reset}>Reset</button>
            <div style={{textAlign:"center"}} ref={ref => this.nodeRef = ref}>
                <h4>Bye bye</h4>
                <p> imcuttle 🐟 </p>
                <img src=""/>
            </div>
        </div>
    }
}

实现原理

下面说明一下 snap-fade-away 的实现要点

首先进行 “DOM粒子化分割”

把输入的 DOM 元素转换为 canvas 画布,得到图像像素点数据,进而进行粒子化分割,如下效果展示: 其中红色表示 原DOM元素 ;绿色表示 转换后的canvas ;蓝色表示 被粒子分割的n个canvas ,同时点击蓝色块可以聚合或分离,点击可以聚合在一起查看效果。

ABCDEFGHIJKLMNOPQRST 💯
import toCanvas from 'html2canvas'
import React from 'react'

const splitFrames = (canvas, frameCount) => {
  const ctx = canvas.getContext('2d')
  const { width, height } = canvas
  // 获取 canvas 的像素数据
  const originalFrame = ctx.getImageData(0, 0, width, height)

  const frames = new Array(frameCount).fill({}).map(() => ctx.createImageData(width, height))
  // 将原始的像素数据,随机分散到多个canvas上面,粒子化
  for (let x = 0; x < width; x++) {
    for (let y = 0; y < height; y++) {
      // 随机获取 像素对象
      const frameIndex = Math.floor(Math.random() * frameCount)
      // 当前的像素位置:通过 x、y 计算出当前遍历到哪个像素点,实际就是从左到右,一行一行的遍历
      const pixelIndex = 4 * (y * width + x)
      // 一个像素 rgba,所以需要设置 4 个值
      for (let z = 0; z < 4; z++) {
        frames[frameIndex].data[pixelIndex + z] = originalFrame.data[pixelIndex + z]
      }
    }
  }
  return frames
}

export default class extends React.Component {
  frameCount = 10
  canvasNodes = []

  componentDidMount() {
    this.initialize()
  }

  async initialize() {
    const container = this.containerRef
    const mirror = this.mirrorRef
    const node = this.nodeRef
    const canvas = await toCanvas(node)
    const frames = splitFrames(canvas, this.frameCount)

    mirror.appendChild(canvas)
    canvas.style.border = '1px solid green'
    canvas.style.marginBottom = '10px'

    this.canvasNodes = frames.map((item, i) => {
      const cloneNode = canvas.cloneNode(true)
      cloneNode.getContext('2d').putImageData(item, 0, 0)
      cloneNode.style.marginBottom = '4px'
      cloneNode.style.border = '1px solid blue'
      container.appendChild(cloneNode)

      this.setUpEachCanvas(cloneNode, i)
      return cloneNode
    })

  }

  setUpEachCanvas(cloneNode) {
    cloneNode.addEventListener('click', () => {
      if ('absolute' === cloneNode.style.position) {
        cloneNode.style.position = ''
        cloneNode.style.left = ''
        cloneNode.style.top = ''
      } else {
        cloneNode.style.position = 'absolute'
        cloneNode.style.left = '0'
        cloneNode.style.top = '0'
      }
    })
  }

  render() {
    return <div>
      <div style={{ border: '1px solid red', marginBottom: 10 }}>
        <div ref={r => this.nodeRef = r}>ABCDEFGHIJKLMNOPQRST 💯</div>
      </div>
      <div ref={r => this.mirrorRef = r}/>
      <div ref={r => this.containerRef = r} style={{ position: 'relative' }}>
      </div>
    </div>
  }
}

进行动画

然后使用你喜欢的方式来随意定义你的动画吧! 下面例子是使用 transition 实现。

ABCDEFGHIJKLMNOPQRST 💯
import SplitFrame from '@/about-snap-fade-away/split-frames'

const duration = '1.5s'

export default class extends SplitFrame {
  setUpEachCanvas(node, i) {
    if (i !== 0) {
      Object.assign(node.style, {
        position: 'absolute',
        left: '0',
        top: '0'
      })
    }

    Object.assign(node.style, {
      border: 'none',
      transition: `transform ${duration} ease-out ${i /
        this.frameCount}s, opacity ${duration} ease-out`,
      userSelect: 'none',
      pointerEvents: 'none',
      opacity: 1,
      transform: 'rotate(0deg) translate(0px)'
    })
  }

  state = {
    ...super.state,
    key: 0
  }

  animate = () => {
    this.canvasNodes.forEach((item, i) => {
      if (i !== 0) {
        let base = i % 2 === 0 ? 1 : -1
        let tx = base * Math.random() * i
        let rotate = -base * i * Math.random()
        item.style.transform = `rotate(${rotate}deg) translate(${tx}px)`
      }
      item.style.opacity = 0
    })
  }

  reset = () => {
    this.setState(
      ({ key }) => ({ key: key + 1 }),
      () => {
        this.initialize()
      }
    )
  }

  render() {
    return (
      <div>
        <button onClick={this.animate}>Animate It!</button>
        <button onClick={this.reset}>Reset!</button>
        <div key={this.state.key}>{super.render()}</div>
      </div>
    )
  }
}

如何进行动画触发

先看一个例子,如下代码执行会有什么效果呢?

dom.style.opacity = 0
dom.style.transition = 'opacity 1s ease'

dom.style.opacity = 1

在书写上述动效的时候遇到一个问题,如何 立即 进行动画的触发呢?

下例子分别对各种情况进行了实践

ABCDEFGHIJKLMNOPQRST 💯
import SplitFrame from '@/about-snap-fade-away/split-frames'

const duration = '2.5s'

const delay = ms => new Promise(resolve => setTimeout(resolve, ms))
const raf = ms => new Promise(resolve => requestAnimationFrame(resolve))

export default class extends SplitFrame {
  async initialize() {
    await super.initialize()
    await this.onAfterInitialized()
  }

  setUpEachCanvas(node, i) {
    if (i !== 0) {
      Object.assign(node.style, {
        position: 'absolute',
        left: '0',
        top: '0'
      })
    }

    Object.assign(node.style, {
      border: 'none',
      transition: `transform ${duration} ease-out ${i /
        this.frameCount}s, opacity ${duration} ease-out`,
      userSelect: 'none',
      pointerEvents: 'none',
      opacity: 1,
      transform: 'rotate(0deg) translate(0px)'
    })
  }

  async onAfterInitialized() {
    // 根据不同 method 触发动画
    switch (this.state.method) {
      case 'setTimeout': {
        await delay()
        break
      }
      case 'RAF': {
        await raf()
        break
      }
      case 'RAF*2': {
        await raf()
        await raf()
        break
      }
      case 'GCS-layout': {
        getComputedStyle(this.nodeRef).margin
        break
      }
      case 'GCS-painting': {
        getComputedStyle(this.nodeRef).transform
        break
      }
      case 'offsetLeft': {
        this.nodeRef.offsetLeft
        break
      }
    }

    this.animate()
  }

  state = {
    ...super.state,
    key: 0,
    method: null
  }

  animate = () => {
    this.canvasNodes.forEach((item, i) => {
      if (i !== 0) {
        let base = i % 2 === 0 ? 1 : -1
        let tx = base * Math.random() * i
        let rotate = -base * i * Math.random()
        item.style.transform = `rotate(${rotate}deg) translate(${tx}px)`
      }
      item.style.opacity = 0
    })
  }

  reset = (method) => () => {
    this.setState(
      ({ key }) => ({ key: key + 1, method }),
      () => {
        this.initialize()
      }
    )
  }

  render() {
    return (
      <div>
        <button onClick={this.reset(null)}>Use *nothing*</button>
        <button onClick={this.reset('setTimeout')}>Use setTimeout</button>
        <button onClick={this.reset('RAF')}>Use requestAnimationFrame</button>
        <button onClick={this.reset('RAF*2')}>Use requestAnimationFrame * 2</button>
        <button onClick={this.reset('GCS-layout')}>Use getComputedStyle layout</button>
        <button onClick={this.reset('GCS-painting')}>Use getComputedStyle painting</button>
        <button onClick={this.reset('offsetLeft')}>Use offsetLeft</button>
        <div key={this.state.key}>{super.render()}</div>
      </div>
    )
  }
}

在第五届 CSS 大会中,就有大佬分享过该 topic

实际应用

趁着复仇者联盟4的上映,Google 也即时加上了彩蛋, 在 Google 搜索 “Thanos” (灭霸) ,点击金手指出现动效