2023-11-03 23:09:45 +03:00
|
|
|
<script setup lang="ts">
|
|
|
|
import { useProjectStore } from '@/stores/project'
|
|
|
|
import { mockFsDirectoryHandle } from '@/util/convert/fsAccess'
|
|
|
|
import { MockWebSocket, type WebSocketHandler } from '@/util/net'
|
2023-11-27 18:48:37 +03:00
|
|
|
import { type Path as LSPath } from 'shared/languageServerTypes'
|
2023-11-03 23:09:45 +03:00
|
|
|
import { watchEffect } from 'vue'
|
2024-04-19 16:39:45 +03:00
|
|
|
import { mockDataWSHandler } from './dataServer'
|
2023-11-03 23:09:45 +03:00
|
|
|
|
|
|
|
const projectStore = useProjectStore()
|
|
|
|
|
|
|
|
interface FileTree {
|
|
|
|
[name: string]: FileTree | string | ArrayBuffer
|
|
|
|
}
|
|
|
|
|
|
|
|
const props = defineProps<{
|
|
|
|
files: FileTree | undefined
|
|
|
|
directory: FileSystemDirectoryHandle | undefined
|
|
|
|
/** The path of the root directory. */
|
|
|
|
prefix?: string[] | undefined
|
|
|
|
}>()
|
|
|
|
|
|
|
|
let resolveDataWsHandler: ((handler: WebSocketHandler) => void) | undefined
|
|
|
|
let dataWsHandler: Promise<WebSocketHandler> = new Promise((resolve) => {
|
|
|
|
resolveDataWsHandler = resolve
|
|
|
|
})
|
|
|
|
function setDataWsHandler(handler: WebSocketHandler) {
|
|
|
|
if (resolveDataWsHandler) {
|
|
|
|
resolveDataWsHandler(handler)
|
|
|
|
resolveDataWsHandler = undefined
|
|
|
|
} else {
|
|
|
|
dataWsHandler = Promise.resolve(handler)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
MockWebSocket.addMock('data', async (data, send) => {
|
|
|
|
;(await dataWsHandler)(data, send)
|
|
|
|
})
|
|
|
|
|
|
|
|
watchEffect(async (onCleanup) => {
|
|
|
|
let maybeDirectory = props.files ? mockFsDirectoryHandle(props.files, '(root)') : props.directory
|
|
|
|
if (!maybeDirectory) return
|
|
|
|
const prefixLength = props.prefix?.length ?? 0
|
|
|
|
const directory = maybeDirectory
|
|
|
|
const ls = await projectStore.lsRpcConnection
|
2024-03-06 18:34:07 +03:00
|
|
|
const maybeProjectRoot = (await projectStore.contentRoots).find(
|
|
|
|
(root) => root.type === 'Project',
|
|
|
|
)?.id
|
2023-11-03 23:09:45 +03:00
|
|
|
if (!maybeProjectRoot) return
|
|
|
|
const projectRoot = maybeProjectRoot
|
|
|
|
async function walkFiles(
|
|
|
|
dir: FileSystemDirectoryHandle,
|
|
|
|
segments: string[],
|
|
|
|
cb: (path: LSPath) => void,
|
|
|
|
) {
|
|
|
|
for await (const [name, dirOrFile] of dir.entries()) {
|
|
|
|
const newSegments = [...segments, name]
|
|
|
|
if (dirOrFile.kind === 'directory') walkFiles(dirOrFile, newSegments, cb)
|
|
|
|
else {
|
|
|
|
cb({
|
|
|
|
rootId: projectRoot,
|
|
|
|
segments: newSegments.slice(prefixLength),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
walkFiles(directory, props.prefix ?? [], (path) =>
|
|
|
|
ls.emit('file/event', [{ kind: 'Added', path }]),
|
|
|
|
)
|
|
|
|
onCleanup(() => {
|
|
|
|
walkFiles(directory, props.prefix ?? [], (path) =>
|
|
|
|
ls.emit('file/event', [{ kind: 'Removed', path }]),
|
|
|
|
)
|
|
|
|
})
|
2023-11-27 18:48:37 +03:00
|
|
|
setDataWsHandler(
|
|
|
|
mockDataWSHandler(async (segments) => {
|
|
|
|
segments = segments.slice(prefixLength)
|
|
|
|
if (!segments.length) return
|
|
|
|
let file
|
|
|
|
try {
|
|
|
|
let dir = directory
|
|
|
|
for (const segment of segments.slice(0, -1)) {
|
|
|
|
dir = await dir.getDirectoryHandle(segment)
|
2023-11-03 23:09:45 +03:00
|
|
|
}
|
2023-11-27 18:48:37 +03:00
|
|
|
const fileHandle = await dir.getFileHandle(segments.at(-1)!)
|
|
|
|
file = await fileHandle.getFile()
|
|
|
|
} catch {
|
|
|
|
return
|
2023-11-03 23:09:45 +03:00
|
|
|
}
|
2023-11-27 18:48:37 +03:00
|
|
|
return await file?.arrayBuffer()
|
|
|
|
}),
|
|
|
|
)
|
2023-11-03 23:09:45 +03:00
|
|
|
})
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<template>
|
|
|
|
<slot></slot>
|
|
|
|
</template>
|