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; #terminals: Array<{ terminal: ITerminal; process: WebContainerProcess }> = []; #boltTerminal = newBoltShellProcess(); showTerminal: WritableAtom = import.meta.hot?.data.showTerminal ?? atom(true); constructor(webcontainerPromise: Promise) { 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 { const 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 }); } } }