UNPKG

3d-tiles-renderer

Version:

https://github.com/AnalyticalGraphicsInc/3d-tiles/tree/master/specification

123 lines (79 loc) 3.63 kB
import { forwardRef, useMemo, useEffect, useContext } from 'react'; import { useThree, useFrame } from '@react-three/fiber'; import { EnvironmentControls as EnvironmentControlsImpl, GlobeControls as GlobeControlsImpl } from '3d-tiles-renderer/three'; import { useShallowOptions } from '../utilities/useOptions.js'; import { TilesRendererContext } from './TilesRenderer.jsx'; import { useApplyRefs } from '../utilities/useApplyRefs.js'; // Add a base component implementation for both EnvironmentControls and GlobeControls const ControlsBaseComponent = forwardRef( function ControlsBaseComponent( props, ref ) { const { controlsConstructor, domElement, scene, camera, ellipsoid, ellipsoidFrame, tilesRenderer, ...rest } = props; const [ defaultCamera ] = useThree( state => [ state.camera ] ); const [ gl ] = useThree( state => [ state.gl ] ); const [ defaultScene ] = useThree( state => [ state.scene ] ); const [ invalidate ] = useThree( state => [ state.invalidate ] ); const [ get ] = useThree( state => [ state.get ] ); const [ set ] = useThree( state => [ state.set ] ); const contextTilesRenderer = useContext( TilesRendererContext ); const appliedTilesRenderer = tilesRenderer || contextTilesRenderer; const appliedCamera = camera || defaultCamera || null; const appliedScene = scene || defaultScene || null; const appliedDomElement = domElement || gl.domElement || null; const appliedEllipsoid = ellipsoid || appliedTilesRenderer?.ellipsoid || null; const appliedEllipsoidFrame = ellipsoidFrame || appliedTilesRenderer?.group || null; // create a controls instance const controls = useMemo( () => { return new controlsConstructor(); }, [ controlsConstructor ] ); // assign / call the reference useApplyRefs( controls, ref ); // fire invalidate callbacks useEffect( () => { const callback = () => invalidate(); controls.addEventListener( 'change', callback ); controls.addEventListener( 'start', callback ); controls.addEventListener( 'end', callback ); return () => { controls.removeEventListener( 'change', callback ); controls.removeEventListener( 'start', callback ); controls.removeEventListener( 'end', callback ); }; }, [ controls, invalidate ] ); // assign the camera useEffect( () => { controls.setCamera( appliedCamera ); }, [ controls, appliedCamera ] ); // assign the scene useEffect( () => { controls.setScene( appliedScene ); }, [ controls, appliedScene ] ); // assign the tiles renderer useEffect( () => { if ( controls.isGlobeControls ) { controls.setEllipsoid( appliedEllipsoid, appliedEllipsoidFrame ); } }, [ controls, appliedEllipsoid, appliedEllipsoidFrame ] ); // attach to the dom element useEffect( () => { controls.attach( appliedDomElement ); return () => { controls.detach(); }; }, [ controls, appliedDomElement ] ); // set the controls for global use useEffect( () => { const old = get().controls; set( { controls } ); return () => set( { controls: old } ); }, [ controls, get, set ] ); // update the controls with a priority of - 1 so it happens before tiles renderer update useFrame( () => { controls.update(); }, - 1 ); useShallowOptions( controls, rest ); } ); export const EnvironmentControls = forwardRef( function EnvironmentControls( props, ref ) { return <ControlsBaseComponent { ...props } ref={ ref } controlsConstructor={ EnvironmentControlsImpl } />; } ); export const GlobeControls = forwardRef( function GlobeControls( props, ref ) { return <ControlsBaseComponent { ...props } ref={ ref } controlsConstructor={ GlobeControlsImpl } />; } );