| import type { WebContainer, WebContainerProcess } from '@webcontainer/api'; | |
| import { atom, type WritableAtom } from 'nanostores'; | |
| import type { ITerminal } from '~/types/terminal'; | |
| import { newBoltShellProcess, newShellProcess } from '~/utils/shell'; | |
| import { coloredText } from '~/utils/terminal'; | |
| export class TerminalStore { | |
| #webcontainer: Promise<WebContainer>; | |
| #terminals: Array<{ terminal: ITerminal; process: WebContainerProcess }> = []; | |
| #boltTerminal = newBoltShellProcess() | |
| showTerminal: WritableAtom<boolean> = import.meta.hot?.data.showTerminal ?? atom(true); | |
| constructor(webcontainerPromise: Promise<WebContainer>) { | |
| this.#webcontainer = webcontainerPromise; | |
| if (import.meta.hot) { | |
| import.meta.hot.data.showTerminal = this.showTerminal; | |
| } | |
| } | |
| get boltTerminal() { | |
| return this.#boltTerminal; | |
| } | |
| toggleTerminal(value?: boolean) { | |
| this.showTerminal.set(value !== undefined ? value : !this.showTerminal.get()); | |
| } | |
| async attachBoltTerminal(terminal: ITerminal) { | |
| try { | |
| let wc = await this.#webcontainer | |
| await this.#boltTerminal.init(wc, terminal) | |
| } catch (error: any) { | |
| terminal.write(coloredText.red('Failed to spawn bolt shell\n\n') + error.message); | |
| return; | |
| } | |
| } | |
| async attachTerminal(terminal: ITerminal) { | |
| try { | |
| const shellProcess = await newShellProcess(await this.#webcontainer, terminal); | |
| this.#terminals.push({ terminal, process: shellProcess }); | |
| } catch (error: any) { | |
| terminal.write(coloredText.red('Failed to spawn shell\n\n') + error.message); | |
| return; | |
| } | |
| } | |
| onTerminalResize(cols: number, rows: number) { | |
| for (const { process } of this.#terminals) { | |
| process.resize({ cols, rows }); | |
| } | |
| } | |
| } | |