diff --git a/packages/playground/src/shared/r3f-rocket/App.tsx b/packages/playground/src/shared/r3f-rocket/App.tsx index 19feef1..eb9b650 100644 --- a/packages/playground/src/shared/r3f-rocket/App.tsx +++ b/packages/playground/src/shared/r3f-rocket/App.tsx @@ -53,7 +53,13 @@ function App() { height: '100vh', }} > - + diff --git a/packages/r3f/src/components/ReferenceWindow/ReferenceWindow.tsx b/packages/r3f/src/components/ReferenceWindow/ReferenceWindow.tsx new file mode 100644 index 0000000..fe11ebf --- /dev/null +++ b/packages/r3f/src/components/ReferenceWindow/ReferenceWindow.tsx @@ -0,0 +1,149 @@ +import type {VFC} from 'react' +import React, {useEffect, useLayoutEffect, useRef} from 'react' +import {useEditorStore} from '../../store' +import shallow from 'zustand/shallow' +import type {WebGLRenderer} from 'three' +import useMeasure from 'react-use-measure' +import styled, {keyframes} from 'styled-components' +import {TiWarningOutline} from 'react-icons/ti' +// This is ugly, but pure TS doesn't let you do bundler-stuff +import noiseImageUrl from './noiseImage' + +const Container = styled.div` + position: relative; + width: fit-content; + height: fit-content; + border-radius: 4px; + overflow: hidden; +` + +const Canvas = styled.canvas` + display: block; +` + +const staticAnimation = keyframes` + 0% { transform: translate(0,0) } + 10% { transform: translate(-5%,-5%) } + 20% { transform: translate(-10%,5%) } + 30% { transform: translate(5%,-10%) } + 40% { transform: translate(-5%,15%) } + 50% { transform: translate(-10%,5%) } + 60% { transform: translate(15%,0) } + 70% { transform: translate(0,10%) } + 80% { transform: translate(-15%,0) } + 90% { transform: translate(10%,5%) } + 100% { transform: translate(5%,0) } +` + +const Static = styled.div` + position: relative; + display: flex; + width: 200px; + height: 120px; + padding: 18px; + + ::before { + content: ''; + position: absolute; + z-index: -1; + top: -50%; + left: -50%; + right: -50%; + bottom: -50%; + background: #2f2f2f url(${noiseImageUrl}) repeat 0 0; + animation: ${staticAnimation} 0.2s infinite; + } +` + +const Warning = styled.div` + display: flex; + flex-direction: column; + gap: 12px; + align-items: center; + text-align: center; + opacity: 0.8; +` + +interface ReferenceWindowProps { + height: number +} + +const ReferenceWindow: VFC = ({height}) => { + const canvasRef = useRef(null) + + const [gl] = useEditorStore((state) => [state.gl], shallow) + const [ref, bounds] = useMeasure() + + const preserveDrawingBuffer = + ( + gl?.domElement.getContext('webgl') ?? gl?.domElement.getContext('webgl2') + )?.getContextAttributes()?.preserveDrawingBuffer ?? false + + useLayoutEffect(() => { + if (gl) { + ref(gl?.domElement) + } + }, [gl, ref]) + + useEffect(() => { + let animationHandle: number + const draw = (gl: WebGLRenderer) => () => { + animationHandle = requestAnimationFrame(draw(gl)) + + if (!gl.domElement || !preserveDrawingBuffer) { + cancelAnimationFrame(animationHandle) + return + } + + const width = (gl.domElement.width / gl.domElement.height) * height + + const ctx = canvasRef.current!.getContext('2d')! + + console.log(gl.domElement.getContext('webgl2')!.getContextAttributes()) + + // https://stackoverflow.com/questions/17861447/html5-canvas-drawimage-how-to-apply-antialiasing + ctx.imageSmoothingQuality = 'high' + + ctx.fillStyle = 'white' + ctx.fillRect(0, 0, width, height) + + ctx.drawImage(gl.domElement, 0, 0, width, height) + } + + if (gl) { + draw(gl)() + } + + return () => { + cancelAnimationFrame(animationHandle) + } + }, [gl, height, preserveDrawingBuffer]) + + return ( + + {gl?.domElement && preserveDrawingBuffer ? ( + + ) : ( + + + + + Please set
{`gl={{ preserveDrawingBuffer: true }}`}
on + the r3f Canvas for the reference window to work. +
+
+
+ )} +
+ ) +} + +export default ReferenceWindow diff --git a/packages/r3f/src/components/ReferenceWindow/noise-transparent.png b/packages/r3f/src/components/ReferenceWindow/noise-transparent.png new file mode 100644 index 0000000..45902c9 Binary files /dev/null and b/packages/r3f/src/components/ReferenceWindow/noise-transparent.png differ diff --git a/packages/r3f/src/components/ReferenceWindow/noiseImage.ts b/packages/r3f/src/components/ReferenceWindow/noiseImage.ts new file mode 100644 index 0000000..b4b6c3b --- /dev/null +++ b/packages/r3f/src/components/ReferenceWindow/noiseImage.ts @@ -0,0 +1,4 @@ +const imageDataUrl = + '' + +export default imageDataUrl diff --git a/packages/r3f/src/components/SnapshotEditor.tsx b/packages/r3f/src/components/SnapshotEditor.tsx index 3a5b3ac..a3277d9 100644 --- a/packages/r3f/src/components/SnapshotEditor.tsx +++ b/packages/r3f/src/components/SnapshotEditor.tsx @@ -9,13 +9,15 @@ import ProxyManager from './ProxyManager' import studio, {ToolbarIconButton} from '@theatre/studio' import {useVal} from '@theatre/react' import styled, {createGlobalStyle, StyleSheetManager} from 'styled-components' -import {IoCameraReverseOutline} from 'react-icons/io5' import type {ISheet} from '@theatre/core' import useSnapshotEditorCamera from './useSnapshotEditorCamera' import {getEditorSheet, getEditorSheetObject} from './editorStuff' import type {$IntentionalAny} from '@theatre/shared/utils/types' import {InfiniteGridHelper} from '../InfiniteGridHelper' import {DragDetectorProvider} from './DragDetector' +import TooltipPortalProvider from './TooltipPortalProvider' +import {FiRefreshCw} from 'react-icons/fi' +import ReferenceWindow from './ReferenceWindow/ReferenceWindow' const GlobalStyle = createGlobalStyle` :host { @@ -97,16 +99,26 @@ const Overlay = styled.div` const Tools = styled.div` position: absolute; - left: 8px; - top: 6px; + left: 12px; + top: 12px; pointer-events: auto; ` +const ReferenceWindowContainer = styled.div` + position: absolute; + right: 12px; + top: 12px; + justify-content: center; +` + const SnapshotEditor: React.FC<{paneId: string}> = (props) => { const snapshotEditorSheet = getEditorSheet() const paneId = props.paneId const editorObject = getEditorSheetObject() + const showReferenceWindow = + useVal(editorObject?.props.viewport.showReferenceWindow) ?? true + const [sceneSnapshot, createSnapshot] = useEditorStore( (state) => [state.sceneSnapshot, state.createSnapshot], shallow, @@ -148,43 +160,49 @@ const SnapshotEditor: React.FC<{paneId: string}> = (props) => { <> - - - - - - - - - - {sceneSnapshot ? ( - <> - - { - gl.setClearColor('white') - }} - shadowMap - dpr={[1, 2]} - fog={'red'} - frameloop="demand" - onPointerMissed={onPointerMissed} + + + + + - - - - - ) : null} - - {/* */} + + + + {showReferenceWindow && ( + + + + )} + + + {sceneSnapshot ? ( + <> + + { + gl.setClearColor('white') + }} + shadowMap + dpr={[1, 2]} + fog={'red'} + frameloop="demand" + onPointerMissed={onPointerMissed} + > + + + + + ) : null} + + diff --git a/packages/r3f/src/components/TooltipPortalProvider.tsx b/packages/r3f/src/components/TooltipPortalProvider.tsx new file mode 100644 index 0000000..0955e50 --- /dev/null +++ b/packages/r3f/src/components/TooltipPortalProvider.tsx @@ -0,0 +1,27 @@ +import type {ReactNode} from 'react'; +import React, { useState} from 'react' +import {PortalContext} from 'reakit' +import styled from 'styled-components' + +const PortalHost = styled.div` + position: fixed; + inset: 0; + pointer-events: none; +` + +export interface PortalManagerProps { + children: ReactNode +} + +const TooltipPortalProvider: React.VFC = ({children}) => { + const [wrapper, setWrapper] = useState(null) + + return ( + + {children} + + + ) +} + +export default TooltipPortalProvider diff --git a/packages/r3f/src/components/editorStuff.ts b/packages/r3f/src/components/editorStuff.ts index a039679..9fb4155 100644 --- a/packages/r3f/src/components/editorStuff.ts +++ b/packages/r3f/src/components/editorStuff.ts @@ -22,6 +22,9 @@ const editorSheetObjectConfig = { }, {as: 'menu', label: 'Shading'}, ), + showReferenceWindow: types.boolean(true, { + label: 'Reference', + }), }, {label: 'Viewport Config'}, ),