效果一瞥
使用 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” (灭霸) ,点击金手指出现动效