likes
comments
collection
share

React 中更简单的使用 gsap 的方式 useGSAP

作者站长头像
站长
· 阅读数 61

前言

gsap 是一个框架无关的动画库,在vuereact中都可以使用,它提供了一个 @gsap/react 包,里面的 useGSAP 让我们在 react 中使用 gsap 更加简单

useGSP 一次注册多次使用

首先需要像插件一样注册

React 中更简单的使用 gsap 的方式 useGSAP

Next.js 注册好后放到 layout.tsx 中,就可以一次注册多次使用了,如果要用的别的插件也可以在这里注册好,比如 ScrollTrigger

'use client'

import { ScrollTrigger } from 'gsap/dist/ScrollTrigger'
import { gsap } from 'gsap'
import { useGSAP } from '@gsap/react'

gsap.registerPlugin(ScrollTrigger, useGSAP)

export default function GsapPlugin() {
  return null
}

Next.js SSR 中使用动画

framer-motion 中有一个 useIsomorphicLayoutEffect,可以兼容 ssr

import { useIsomorphicLayoutEffect } from 'framer-motion'

但是 GSAP 没有,所以需要自己实现

export const useIsomorphicLayoutEffect =
  typeof window !== 'undefined' ? useLayoutEffect : useEffect

有了 useGSAP 后,它封装了 useIsomorphicLayoutEffect,所以直接用 useGSAP

import gsap from 'gsap'
import { useGSAP } from '@gsap/react'

useGSAP(
  () => {
    gsap.to('.box', { x: 100 })
  },
  { dependencies: [] } // 相当于useEffect的依赖项
)

当然,dependencies 的默认值就是[],所以只需要传入第一个参数就好了

import gsap from 'gsap'
import { useGSAP } from '@gsap/react'

useGSAP(() => {
  gsap.to('.box', { x: 100 })
})

在依赖项变化的时候,它不会重置动画,需要和revertOnUpdate: true一起使用,但平时写动画基本不会用到,具体可以看 Reacting to changes in state

作用域

.item做一个简单的动画,我们可以这样,并且它会给页面所有的 item 加上动画

gsap.to('.item', { x: 100 })

如果我们想限制一下,对.wrapper.item的类,那就可以用 scopescope可以把所有 GSAP 选择器范围都将限定为该容器的后代

import { useRef } from 'react'
import gsap from 'gsap'
import { useGSAP } from '@gsap/react'

const container = useRef()

useGSAP(
  () => {
    gsap.to('.box', { x: 360 })
  },
  { scope: container }
)

<div class='wrapper' ref={container}>>
  <div class='item'></div>
</div>

动画清理

参考 JS Frameworks

GSAP 的动画会一直存在,所以需要 清理动画,防止内存泄漏和其它问题。

对于单个动画

useEffect(() => {
  const t1 = gsap.from('.box', {
    y: 200,
    opacity: 0
  })
  return () => {
    t1.revert()
  }
}, [])

对于多个动画,gsap 提供了 gsap.context()

useEffect(() => {
  const ctx = gsap.context(() => {
    gsap.to(el, { rotation: 360, repeat: -1 })

    let tl = gsap.timeline()

    tl.to(box, { x: 200 })
    tl.to(box, { y: 500, duration: 2 })
    tl.to(box, { rotation: 180, repeat: 2 })
  })
  return () => {
    ctx.revert()
  }
}, [])

可以看到gsap.context()不太优雅,多了一层函数,感觉在套娃,而useGSAP会自动清理动画,所以在 react 中使用 useGSAP 会更简单

非初始化执行的动画清理

useGSAP 会自动清理动画,但是如果是点击事件,则可能内存泄漏

const container = useRef()

useGSAP(
  () => {
    // ✅
    gsap.to('.good', { x: 100 })
  },
  { scope: container }
)

// ❌ 这是不安全的,会存在内存泄漏
const onClickBad = () => {
  gsap.to('.bad', { y: 100 })
}

return (
  <div ref={container}>
    <div className="good"></div>
    <button onClick={onClickBad} className="bad"></button>
  </div>
)

可以通过 contextSafe 来解决

const container = useRef()

const { contextSafe } = useGSAP({ scope: container })

// ✅
const onClickGood = contextSafe(() => {
  gsap.to('.good', { rotation: 180 })
})

return (
  <div ref={container}>
    <button onClick={onClickGood} className="good"></button>
  </div>
)

还有一种方案可能没有那么优雅,可以看官网方案二

转载自:https://juejin.cn/post/7369401900243075110
评论
请登录