* update eslintrc, and fix most of the errors * more Signed-off-by: Jess Frazelle <github@jessfraz.com> * fix last one Signed-off-by: Jess Frazelle <github@jessfraz.com> * fixes Signed-off-by: Jess Frazelle <github@jessfraz.com> * updates Signed-off-by: Jess Frazelle <github@jessfraz.com> * A snapshot a day keeps the bugs away! 📷🐛 --------- Signed-off-by: Jess Frazelle <github@jessfraz.com> Co-authored-by: Jess Frazelle <github@jessfraz.com> Co-authored-by: Jess Frazelle <jessfraz@users.noreply.github.com> Co-authored-by: github-actions[bot] <github-actions[bot]@users.noreply.github.com>
150 lines
4.9 KiB
TypeScript
150 lines
4.9 KiB
TypeScript
import { useEffect, useMemo, useRef } from 'react'
|
|
import { useHotKeyListener } from './hooks/useHotKeyListener'
|
|
import { Stream } from './components/Stream'
|
|
import { AppHeader } from './components/AppHeader'
|
|
import { useHotkeys } from 'react-hotkeys-hook'
|
|
import { useLoaderData, useNavigate } from 'react-router-dom'
|
|
import { type IndexLoaderData } from 'lib/types'
|
|
import { PATHS } from 'lib/paths'
|
|
import { onboardingPaths } from 'routes/Onboarding/paths'
|
|
import { useEngineConnectionSubscriptions } from 'hooks/useEngineConnectionSubscriptions'
|
|
import { codeManager, engineCommandManager } from 'lib/singletons'
|
|
import { useAbsoluteFilePath } from 'hooks/useAbsoluteFilePath'
|
|
import { isDesktop } from 'lib/isDesktop'
|
|
import { useLspContext } from 'components/LspProvider'
|
|
import { ModelingSidebar } from 'components/ModelingSidebar/ModelingSidebar'
|
|
import { LowerRightControls } from 'components/LowerRightControls'
|
|
import ModalContainer from 'react-modal-promise'
|
|
import useHotkeyWrapper from 'lib/hotkeyWrapper'
|
|
import Gizmo from 'components/Gizmo'
|
|
import { CoreDumpManager } from 'lib/coredump'
|
|
import { UnitsMenu } from 'components/UnitsMenu'
|
|
import { CameraProjectionToggle } from 'components/CameraProjectionToggle'
|
|
import { useCreateFileLinkQuery } from 'hooks/useCreateFileLinkQueryWatcher'
|
|
import { maybeWriteToDisk } from 'lib/telemetry'
|
|
import { takeScreenshotOfVideoStreamCanvas } from 'lib/screenshot'
|
|
import { writeProjectThumbnailFile } from 'lib/desktop'
|
|
import { useRouteLoaderData } from 'react-router-dom'
|
|
import { useEngineCommands } from 'components/EngineCommands'
|
|
import { commandBarActor } from 'machines/commandBarMachine'
|
|
import { useToken } from 'machines/appMachine'
|
|
import { useSettings } from 'machines/appMachine'
|
|
import { rustContext } from 'lib/singletons'
|
|
maybeWriteToDisk()
|
|
.then(() => {})
|
|
.catch(() => {})
|
|
|
|
export function App() {
|
|
const { project, file } = useLoaderData() as IndexLoaderData
|
|
|
|
// Keep a lookout for a URL query string that invokes the 'import file from URL' command
|
|
useCreateFileLinkQuery((argDefaultValues) => {
|
|
commandBarActor.send({
|
|
type: 'Find and select command',
|
|
data: {
|
|
groupId: 'projects',
|
|
name: 'Import file from URL',
|
|
argDefaultValues,
|
|
},
|
|
})
|
|
})
|
|
|
|
const navigate = useNavigate()
|
|
const filePath = useAbsoluteFilePath()
|
|
const { onProjectOpen } = useLspContext()
|
|
// We need the ref for the outermost div so we can screenshot the app for
|
|
// the coredump.
|
|
const ref = useRef<HTMLDivElement>(null)
|
|
|
|
const projectName = project?.name || null
|
|
const projectPath = project?.path || null
|
|
|
|
const [commands] = useEngineCommands()
|
|
const loaderData = useRouteLoaderData(PATHS.FILE) as IndexLoaderData
|
|
const lastCommandType = commands[commands.length - 1]?.type
|
|
|
|
useEffect(() => {
|
|
onProjectOpen({ name: projectName, path: projectPath }, file || null)
|
|
}, [projectName, projectPath])
|
|
|
|
useHotKeyListener()
|
|
|
|
const settings = useSettings()
|
|
const token = useToken()
|
|
|
|
const coreDumpManager = useMemo(
|
|
() =>
|
|
new CoreDumpManager(
|
|
engineCommandManager,
|
|
codeManager,
|
|
rustContext,
|
|
token
|
|
),
|
|
[]
|
|
)
|
|
|
|
const {
|
|
app: { onboardingStatus },
|
|
} = settings
|
|
|
|
useHotkeys('backspace', (e) => {
|
|
e.preventDefault()
|
|
})
|
|
useHotkeyWrapper(
|
|
[isDesktop() ? 'mod + ,' : 'shift + mod + ,'],
|
|
() => navigate(filePath + PATHS.SETTINGS),
|
|
{
|
|
splitKey: '|',
|
|
}
|
|
)
|
|
|
|
const paneOpacity = [onboardingPaths.CAMERA, onboardingPaths.STREAMING].some(
|
|
(p) => p === onboardingStatus.current
|
|
)
|
|
? 'opacity-20'
|
|
: ''
|
|
|
|
useEngineConnectionSubscriptions()
|
|
|
|
// Generate thumbnail.png when loading the app
|
|
useEffect(() => {
|
|
if (isDesktop() && lastCommandType === 'execution-done') {
|
|
setTimeout(() => {
|
|
const projectDirectoryWithoutEndingSlash = loaderData?.project?.path
|
|
if (!projectDirectoryWithoutEndingSlash) {
|
|
return
|
|
}
|
|
const dataUrl: string = takeScreenshotOfVideoStreamCanvas()
|
|
// zoom to fit command does not wait, wait 500ms to see if zoom to fit finishes
|
|
writeProjectThumbnailFile(dataUrl, projectDirectoryWithoutEndingSlash)
|
|
.then(() => {})
|
|
.catch((e) => {
|
|
console.error(
|
|
`Failed to generate thumbnail for ${projectDirectoryWithoutEndingSlash}`
|
|
)
|
|
console.error(e)
|
|
})
|
|
}, 500)
|
|
}
|
|
}, [lastCommandType])
|
|
|
|
return (
|
|
<div className="relative h-full flex flex-col" ref={ref}>
|
|
<AppHeader
|
|
className={'transition-opacity transition-duration-75 ' + paneOpacity}
|
|
project={{ project, file }}
|
|
enableMenu={true}
|
|
/>
|
|
<ModalContainer />
|
|
<ModelingSidebar paneOpacity={paneOpacity} />
|
|
<Stream />
|
|
{/* <CamToggle /> */}
|
|
<LowerRightControls coreDumpManager={coreDumpManager}>
|
|
<UnitsMenu />
|
|
<Gizmo />
|
|
<CameraProjectionToggle />
|
|
</LowerRightControls>
|
|
</div>
|
|
)
|
|
}
|