我如何使用 setInterval() 自动更改此 React Spring 项目的幻灯片

How can I use setInterval() to automatically change slides on this React Spring project

我实现了这个 React Spring 组件,点击它后会出现不同的幻灯片。我想使用 setInterval 方法让它每隔 X 秒自动更改一次。

我尝试使用 setInterval 但我没有得到想要的结果,因为 setInterval 函数被多次调用并且随着时间的增加幻灯片开始疯狂变化直到浏览器冻结.

这里是codesandbox原项目https://codesandbox.io/embed/1y3yyqpq7q

这是我目前运行的代码

import { useTransition, animated } from 'react-spring'
import clientStyles from '../../styles/modules_scss/clients.module.scss'

const pages = [
    ({ style }) => <animated.div style={{ ...style, background: 'lightpink' }}>Andres Mejias</animated.div>,
    ({ style }) => <animated.div style={{ ...style, background: 'lightblue' }}>Andres Urdaneta</animated.div>,
    ({ style }) => <animated.div style={{ ...style, background: 'lightgreen' }}>Gustavo Abdelnour</animated.div>,
]

export default function SlideTransition() {
    const [index, set] = useState(0)
    const onClick = useCallback(() => set(state => (state + 1) % 3), [])
    const transitions = useTransition(index, p => p, {
        from: { opacity: 0, transform: 'translate3d(100%,0,0)' },
        enter: { opacity: 1, transform: 'translate3d(0%,0,0)' },
        leave: { opacity: 0, transform: 'translate3d(-50%,0,0)' },
    })
    return (
        <div className={clientStyles.simpleTransMain} onClick={setInterval(onClick, 3000)}> //Here's my approach
            {transitions.map(({ item, props, key }) => {
                const Page = pages[item]
                return <Page key={key} style={props} />
            })}
        </div>
    )
}

我得到的最好结果是将 setInterval 包装在一个匿名函数中,并且在单击第一张幻灯片后我每隔 3000ms 得到一张新幻灯片。我希望在页面加载后立即发生这种情况,而无需单击幻灯片。

onClick={() => { setInterval(onClick, 3000) }}

您在每次渲染时调用 setInterval,因此每次都会创建一个新的间隔,而您的第一个间隔仍然是 运行ning。所以你只会得到越来越多的添加,直到它使浏览器超载。

由于您希望间隔在单击任何内容之前开始,请将其添加到效果中,以便您可以 运行 在应用程序安装时将其清除,并在卸载时将其清除。

useEffect(() => {
  const interval_id = setInterval(onClick, 3000);
   return () => {
     // Stop the interval when the component unmounts. 
     // Otherwise it will keeep going and you will get an error.
     clearInterval(interval_id)
   }
}, []); // empty array of dependencies will cause it to run on mount and unmount only

然后只需删除 onClick 中的 setInterval 即可。

你可以使用useEffect来做出和生命周期一样的效果"componentdidmount"。 首先,你不能直接调用 setinterval 因为它会一直导致渲染,这就是为什么你需要将它包装在一个匿名函数中,其次,即使你这样使用它,你的 onclick 也会改变状态,这也会导致渲染每次都没有完成第一个,继续这样做会导致浏览器因为它创建的大量间隔而冻结,所以最好的方法是在卸载时清除你的 setinterval,这样每个渲染只会有一个工作间隔。