2023-09-25 19:49:53 -07:00
|
|
|
import { useLayoutEffect, useEffect, useRef } from 'react'
|
2023-11-01 07:39:31 -04:00
|
|
|
import { parse } from '../lang/wasm'
|
2023-09-14 10:47:55 +10:00
|
|
|
import { useStore } from '../useStore'
|
2023-09-25 19:49:53 -07:00
|
|
|
import { engineCommandManager } from '../lang/std/engineConnection'
|
|
|
|
import { deferExecution } from 'lib/utils'
|
2024-02-11 12:59:00 +11:00
|
|
|
import { kclManager } from 'lang/KclSingleton'
|
2023-09-14 10:47:55 +10:00
|
|
|
|
|
|
|
export function useSetupEngineManager(
|
|
|
|
streamRef: React.RefObject<HTMLDivElement>,
|
|
|
|
token?: string
|
|
|
|
) {
|
|
|
|
const {
|
|
|
|
setMediaStream,
|
|
|
|
setIsStreamReady,
|
|
|
|
setStreamDimensions,
|
2023-09-25 19:49:53 -07:00
|
|
|
streamDimensions,
|
2023-09-14 10:47:55 +10:00
|
|
|
} = useStore((s) => ({
|
|
|
|
setMediaStream: s.setMediaStream,
|
|
|
|
setIsStreamReady: s.setIsStreamReady,
|
|
|
|
setStreamDimensions: s.setStreamDimensions,
|
2023-09-25 19:49:53 -07:00
|
|
|
streamDimensions: s.streamDimensions,
|
2023-09-14 10:47:55 +10:00
|
|
|
}))
|
|
|
|
|
|
|
|
const streamWidth = streamRef?.current?.offsetWidth
|
|
|
|
const streamHeight = streamRef?.current?.offsetHeight
|
|
|
|
|
2023-09-25 19:49:53 -07:00
|
|
|
const hasSetNonZeroDimensions = useRef<boolean>(false)
|
2023-09-14 10:47:55 +10:00
|
|
|
|
|
|
|
useLayoutEffect(() => {
|
2023-09-25 19:49:53 -07:00
|
|
|
// Load the engine command manager once with the initial width and height,
|
|
|
|
// then we do not want to reload it.
|
|
|
|
const { width: quadWidth, height: quadHeight } = getDimensions(
|
|
|
|
streamWidth,
|
|
|
|
streamHeight
|
|
|
|
)
|
|
|
|
if (!hasSetNonZeroDimensions.current && quadHeight && quadWidth) {
|
|
|
|
engineCommandManager.start({
|
|
|
|
setMediaStream,
|
|
|
|
setIsStreamReady,
|
|
|
|
width: quadWidth,
|
|
|
|
height: quadHeight,
|
2023-10-11 13:36:54 +11:00
|
|
|
executeCode: (code?: string) => {
|
|
|
|
const _ast = parse(code || kclManager.code)
|
|
|
|
return kclManager.executeAst(_ast, true)
|
|
|
|
},
|
2023-09-25 19:49:53 -07:00
|
|
|
token,
|
|
|
|
})
|
|
|
|
setStreamDimensions({
|
|
|
|
streamWidth: quadWidth,
|
|
|
|
streamHeight: quadHeight,
|
|
|
|
})
|
|
|
|
hasSetNonZeroDimensions.current = true
|
|
|
|
}
|
|
|
|
}, [streamRef?.current?.offsetWidth, streamRef?.current?.offsetHeight])
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
const handleResize = deferExecution(() => {
|
|
|
|
const { width, height } = getDimensions(
|
|
|
|
streamRef?.current?.offsetWidth,
|
|
|
|
streamRef?.current?.offsetHeight
|
|
|
|
)
|
|
|
|
if (
|
|
|
|
streamDimensions.streamWidth !== width ||
|
|
|
|
streamDimensions.streamHeight !== height
|
|
|
|
) {
|
|
|
|
engineCommandManager.handleResize({
|
|
|
|
streamWidth: width,
|
|
|
|
streamHeight: height,
|
|
|
|
})
|
|
|
|
setStreamDimensions({
|
|
|
|
streamWidth: width,
|
|
|
|
streamHeight: height,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}, 500)
|
|
|
|
|
|
|
|
window.addEventListener('resize', handleResize)
|
2023-09-14 10:47:55 +10:00
|
|
|
return () => {
|
2023-09-25 19:49:53 -07:00
|
|
|
window.removeEventListener('resize', handleResize)
|
2023-09-14 10:47:55 +10:00
|
|
|
}
|
2023-09-25 19:49:53 -07:00
|
|
|
}, [])
|
|
|
|
}
|
|
|
|
|
|
|
|
function getDimensions(streamWidth?: number, streamHeight?: number) {
|
2023-11-13 19:32:07 -05:00
|
|
|
const maxResolution = 2000
|
2023-09-25 19:49:53 -07:00
|
|
|
const width = streamWidth ? streamWidth : 0
|
|
|
|
const height = streamHeight ? streamHeight : 0
|
2023-11-13 19:32:07 -05:00
|
|
|
const ratio = Math.min(
|
|
|
|
Math.min(maxResolution / width, maxResolution / height),
|
|
|
|
1.0
|
|
|
|
)
|
|
|
|
const quadWidth = Math.round((width * ratio) / 4) * 4
|
|
|
|
const quadHeight = Math.round((height * ratio) / 4) * 4
|
2023-09-25 19:49:53 -07:00
|
|
|
return { width: quadWidth, height: quadHeight }
|
2023-09-14 10:47:55 +10:00
|
|
|
}
|