From 3205f78e3f293679325f1c80ce11707da2a1eef5 Mon Sep 17 00:00:00 2001 From: Garrett Johnson Date: Fri, 13 Dec 2024 16:39:51 +0900 Subject: [PATCH] CameraTransition: invalidate on change, pass options (#867) * CameraTransition: invalidate on change, pass options * Fix transition --- src/r3f/components/CameraTransition.jsx | 46 +++++++++---------- src/three/controls/CameraTransitionManager.js | 2 + 2 files changed, 24 insertions(+), 24 deletions(-) diff --git a/src/r3f/components/CameraTransition.jsx b/src/r3f/components/CameraTransition.jsx index 876e3d2f1..241935103 100644 --- a/src/r3f/components/CameraTransition.jsx +++ b/src/r3f/components/CameraTransition.jsx @@ -1,10 +1,11 @@ import { forwardRef, useEffect, useMemo } from 'react'; import { useFrame, useThree } from '@react-three/fiber'; import { CameraTransitionManager } from '3d-tiles-renderer'; +import { useDeepOptions } from '../utilities/useOptions'; export const CameraTransition = forwardRef( function CameraTransition( props, ref ) { - const { mode = 'perspective', onTransitionStart, onTransitionEnd, perspectiveCamera, orthographicCamera } = props; + const { mode = 'perspective', perspectiveCamera, orthographicCamera, ...options } = props; const [ set, invalidate, controls, camera, size ] = useThree( state => [ state.set, state.invalidate, state.controls, state.camera, state.size ] ); // create the manager @@ -84,29 +85,6 @@ export const CameraTransition = forwardRef( function CameraTransition( props, re }, [ manager, set ] ); - // register for events - useEffect( () => { - - if ( onTransitionEnd ) { - - manager.addEventListener( 'transition-end', onTransitionEnd ); - return () => manager.removeEventListener( 'transition-end', onTransitionEnd ); - - } - - }, [ onTransitionEnd, manager ] ); - - useEffect( () => { - - if ( onTransitionStart ) { - - manager.addEventListener( 'transition-start', onTransitionStart ); - return () => manager.removeEventListener( 'transition-start', onTransitionStart ); - - } - - }, [ onTransitionStart, manager ] ); - // assign cameras useEffect( () => { @@ -156,6 +134,26 @@ export const CameraTransition = forwardRef( function CameraTransition( props, re }, [ mode, manager, invalidate, controls ] ); + // rerender the frame when the transition animates + useEffect( () => { + + const callback = () => invalidate(); + manager.addEventListener( 'transition-start', callback ); + manager.addEventListener( 'change', callback ); + manager.addEventListener( 'transition-end', callback ); + + return () => { + + manager.removeEventListener( 'transition-start', callback ); + manager.removeEventListener( 'change', callback ); + manager.removeEventListener( 'transition-end', callback ); + + }; + + }, [ manager, invalidate ] ); + + useDeepOptions( manager, options ); + // update animation useFrame( () => { diff --git a/src/three/controls/CameraTransitionManager.js b/src/three/controls/CameraTransitionManager.js index 4c9eacf3f..d323b3aea 100644 --- a/src/three/controls/CameraTransitionManager.js +++ b/src/three/controls/CameraTransitionManager.js @@ -75,7 +75,9 @@ export class CameraTransitionManager extends EventDispatcher { toggle() { + // reset the clock for cases where we're not calling "update" every frame this._target = this._target === 1 ? 0 : 1; + this._clock.getDelta(); }