154 lines
5.5 KiB
TypeScript
154 lines
5.5 KiB
TypeScript
import { useFileSystemWatcher } from '@src/hooks/useFileSystemWatcher'
|
|
import { PATHS } from '@src/lib/paths'
|
|
import { systemIOActor, useSettings, useToken } from '@src/lib/singletons'
|
|
import {
|
|
useHasListedProjects,
|
|
useProjectDirectoryPath,
|
|
useRequestedFileName,
|
|
useRequestedProjectName,
|
|
useRequestedTextToCadGeneration,
|
|
useFolders,
|
|
} from '@src/machines/systemIO/hooks'
|
|
import {
|
|
NO_PROJECT_DIRECTORY,
|
|
SystemIOMachineEvents,
|
|
} from '@src/machines/systemIO/utils'
|
|
import { useNavigate } from 'react-router-dom'
|
|
import { useEffect } from 'react'
|
|
import { submitAndAwaitTextToKclSystemIO } from '@src/lib/textToCad'
|
|
import { reportRejection } from '@src/lib/trap'
|
|
import { getUniqueProjectName } from '@src/lib/desktopFS'
|
|
|
|
export function SystemIOMachineLogicListenerDesktop() {
|
|
const requestedProjectName = useRequestedProjectName()
|
|
const requestedFileName = useRequestedFileName()
|
|
const projectDirectoryPath = useProjectDirectoryPath()
|
|
const hasListedProjects = useHasListedProjects()
|
|
const navigate = useNavigate()
|
|
const settings = useSettings()
|
|
const requestedTextToCadGeneration = useRequestedTextToCadGeneration()
|
|
const token = useToken()
|
|
const folders = useFolders()
|
|
|
|
const useGlobalProjectNavigation = () => {
|
|
useEffect(() => {
|
|
if (!requestedProjectName.name) {
|
|
return
|
|
}
|
|
let projectPathWithoutSpecificKCLFile =
|
|
projectDirectoryPath +
|
|
window.electron.path.sep +
|
|
requestedProjectName.name
|
|
|
|
const requestedPath = `${PATHS.FILE}/${encodeURIComponent(
|
|
projectPathWithoutSpecificKCLFile
|
|
)}`
|
|
navigate(requestedPath)
|
|
}, [requestedProjectName])
|
|
}
|
|
|
|
const useGlobalFileNavigation = () => {
|
|
useEffect(() => {
|
|
if (!requestedFileName.file || !requestedFileName.project) {
|
|
return
|
|
}
|
|
const projectPath = window.electron.join(
|
|
projectDirectoryPath,
|
|
requestedFileName.project
|
|
)
|
|
const filePath = window.electron.join(projectPath, requestedFileName.file)
|
|
const requestedPath = `${PATHS.FILE}/${encodeURIComponent(filePath)}`
|
|
navigate(requestedPath)
|
|
}, [requestedFileName])
|
|
}
|
|
|
|
const useApplicationProjectDirectory = () => {
|
|
useEffect(() => {
|
|
systemIOActor.send({
|
|
type: SystemIOMachineEvents.setProjectDirectoryPath,
|
|
data: {
|
|
requestedProjectDirectoryPath:
|
|
settings.app.projectDirectory.current || '',
|
|
},
|
|
})
|
|
}, [settings.app.projectDirectory.current])
|
|
}
|
|
|
|
const useDefaultProjectName = () => {
|
|
useEffect(() => {
|
|
systemIOActor.send({
|
|
type: SystemIOMachineEvents.setDefaultProjectFolderName,
|
|
data: {
|
|
requestedDefaultProjectFolderName:
|
|
settings.projects.defaultProjectName.current || '',
|
|
},
|
|
})
|
|
}, [settings.projects.defaultProjectName.current])
|
|
}
|
|
|
|
const useWatchingApplicationProjectDirectory = () => {
|
|
useFileSystemWatcher(
|
|
async (eventType, path) => {
|
|
// Gotcha: Chokidar is buggy. It will emit addDir or add on files that did not get created.
|
|
// This means while the application initialize and Chokidar initializes you cannot tell if
|
|
// a directory or file is actually created or they are buggy signals. This means you must
|
|
// ignore all signals during initialization because it is ambiguous. Once those signals settle
|
|
// you can actually start listening to real signals.
|
|
// If someone creates folders or files during initialization we ignore those events!
|
|
if (!hasListedProjects) {
|
|
return
|
|
}
|
|
|
|
const folderName =
|
|
systemIOActor.getSnapshot().context.lastProjectDeleteRequest.project
|
|
const folderPath = `${projectDirectoryPath}${window.electron.sep}${folderName}`
|
|
if (
|
|
folderName !== NO_PROJECT_DIRECTORY &&
|
|
(eventType === 'unlinkDir' || eventType === 'unlink') &&
|
|
path.includes(folderPath)
|
|
) {
|
|
// NO OP: The systemIOMachine will be triggering the read in the state transition, don't spam it again
|
|
// once this event is processed after the deletion.
|
|
} else {
|
|
// Prevents spamming reading from disk twice on deletion due to files and folders being deleted async
|
|
systemIOActor.send({
|
|
type: SystemIOMachineEvents.readFoldersFromProjectDirectory,
|
|
})
|
|
}
|
|
},
|
|
settings.app.projectDirectory.current
|
|
? [settings.app.projectDirectory.current]
|
|
: []
|
|
)
|
|
|
|
// TODO: Move this generateTextToCAD to another machine in the future and make a whole machine out of it.
|
|
useEffect(() => {
|
|
const requestedPromptTrimmed =
|
|
requestedTextToCadGeneration.requestedPrompt.trim()
|
|
const requestedProjectName =
|
|
requestedTextToCadGeneration.requestedProjectName
|
|
const isProjectNew = requestedTextToCadGeneration.isProjectNew
|
|
if (!requestedPromptTrimmed || !requestedProjectName) return
|
|
const uniqueNameIfNeeded = isProjectNew
|
|
? getUniqueProjectName(requestedProjectName, folders)
|
|
: requestedProjectName
|
|
submitAndAwaitTextToKclSystemIO({
|
|
trimmedPrompt: requestedPromptTrimmed,
|
|
projectName: uniqueNameIfNeeded,
|
|
navigate,
|
|
token,
|
|
isProjectNew,
|
|
settings: { highlightEdges: settings.modeling.highlightEdges.current },
|
|
}).catch(reportRejection)
|
|
}, [requestedTextToCadGeneration])
|
|
}
|
|
|
|
useGlobalProjectNavigation()
|
|
useGlobalFileNavigation()
|
|
useApplicationProjectDirectory()
|
|
useDefaultProjectName()
|
|
useWatchingApplicationProjectDirectory()
|
|
|
|
return null
|
|
}
|