| import { TerminalWindowIcon, CrossSmallIcon } from './icons'; | |
| import { Loader } from './elements/loader'; | |
| import { Button } from './ui/button'; | |
| import { | |
| type Dispatch, | |
| type SetStateAction, | |
| useCallback, | |
| useEffect, | |
| useRef, | |
| useState, | |
| } from 'react'; | |
| import { cn } from '@/lib/utils'; | |
| import { useArtifactSelector } from '@/hooks/use-artifact'; | |
| export interface ConsoleOutputContent { | |
| type: 'text' | 'image'; | |
| value: string; | |
| } | |
| export interface ConsoleOutput { | |
| id: string; | |
| status: 'in_progress' | 'loading_packages' | 'completed' | 'failed'; | |
| contents: Array<ConsoleOutputContent>; | |
| } | |
| interface ConsoleProps { | |
| consoleOutputs: Array<ConsoleOutput>; | |
| setConsoleOutputs: Dispatch<SetStateAction<Array<ConsoleOutput>>>; | |
| } | |
| export function Console({ consoleOutputs, setConsoleOutputs }: ConsoleProps) { | |
| const [height, setHeight] = useState<number>(300); | |
| const [isResizing, setIsResizing] = useState(false); | |
| const consoleEndRef = useRef<HTMLDivElement>(null); | |
| const isArtifactVisible = useArtifactSelector((state) => state.isVisible); | |
| const minHeight = 100; | |
| const maxHeight = 800; | |
| const startResizing = useCallback(() => { | |
| setIsResizing(true); | |
| }, []); | |
| const stopResizing = useCallback(() => { | |
| setIsResizing(false); | |
| }, []); | |
| const resize = useCallback( | |
| (e: MouseEvent) => { | |
| if (isResizing) { | |
| const newHeight = window.innerHeight - e.clientY; | |
| if (newHeight >= minHeight && newHeight <= maxHeight) { | |
| setHeight(newHeight); | |
| } | |
| } | |
| }, | |
| [isResizing], | |
| ); | |
| useEffect(() => { | |
| window.addEventListener('mousemove', resize); | |
| window.addEventListener('mouseup', stopResizing); | |
| return () => { | |
| window.removeEventListener('mousemove', resize); | |
| window.removeEventListener('mouseup', stopResizing); | |
| }; | |
| }, [resize, stopResizing]); | |
| useEffect(() => { | |
| consoleEndRef.current?.scrollIntoView({ behavior: 'smooth' }); | |
| }, [consoleOutputs]); | |
| useEffect(() => { | |
| if (!isArtifactVisible) { | |
| setConsoleOutputs([]); | |
| } | |
| }, [isArtifactVisible, setConsoleOutputs]); | |
| return consoleOutputs.length > 0 ? ( | |
| <> | |
| <div | |
| className="fixed z-50 w-full h-2 cursor-ns-resize" | |
| onMouseDown={startResizing} | |
| style={{ bottom: height - 4 }} | |
| role="slider" | |
| aria-valuenow={minHeight} | |
| /> | |
| <div | |
| className={cn( | |
| 'flex overflow-x-hidden overflow-y-scroll fixed bottom-0 z-40 flex-col w-full border-t dark:bg-zinc-900 bg-zinc-50 dark:border-zinc-700 border-zinc-200', | |
| { | |
| 'select-none': isResizing, | |
| }, | |
| )} | |
| style={{ height }} | |
| > | |
| <div className="flex sticky top-0 z-50 flex-row justify-between items-center px-2 py-1 w-full border-b h-fit dark:border-zinc-700 border-zinc-200 bg-muted"> | |
| <div className="flex flex-row gap-3 items-center pl-2 text-sm dark:text-zinc-50 text-zinc-800"> | |
| <div className="text-muted-foreground"> | |
| <TerminalWindowIcon /> | |
| </div> | |
| <div>Console</div> | |
| </div> | |
| <Button | |
| variant="ghost" | |
| className="p-1 size-fit hover:dark:bg-zinc-700 hover:bg-zinc-200" | |
| size="icon" | |
| onClick={() => setConsoleOutputs([])} | |
| > | |
| <CrossSmallIcon /> | |
| </Button> | |
| </div> | |
| <div> | |
| {consoleOutputs.map((consoleOutput, index) => ( | |
| <div | |
| key={consoleOutput.id} | |
| className="flex flex-row px-4 py-2 font-mono text-sm border-b dark:border-zinc-700 border-zinc-200 dark:bg-zinc-900 bg-zinc-50" | |
| > | |
| <div | |
| className={cn('w-12 shrink-0', { | |
| 'text-muted-foreground': [ | |
| 'in_progress', | |
| 'loading_packages', | |
| ].includes(consoleOutput.status), | |
| 'text-emerald-500': consoleOutput.status === 'completed', | |
| 'text-red-400': consoleOutput.status === 'failed', | |
| })} | |
| > | |
| [{index + 1}] | |
| </div> | |
| {['in_progress', 'loading_packages'].includes( | |
| consoleOutput.status, | |
| ) ? ( | |
| <div className="flex flex-row gap-2"> | |
| <div className="size-fit self-center mb-auto mt-0.5"> | |
| <Loader size={16} /> | |
| </div> | |
| <div className="text-muted-foreground"> | |
| {consoleOutput.status === 'in_progress' | |
| ? 'Initializing...' | |
| : consoleOutput.status === 'loading_packages' | |
| ? consoleOutput.contents.map((content) => | |
| content.type === 'text' ? content.value : null, | |
| ) | |
| : null} | |
| </div> | |
| </div> | |
| ) : ( | |
| <div className="flex overflow-x-scroll flex-col gap-2 w-full dark:text-zinc-50 text-zinc-900"> | |
| {consoleOutput.contents.map((content, index) => | |
| content.type === 'image' ? ( | |
| <picture key={`${consoleOutput.id}-${index}`}> | |
| <img | |
| src={content.value} | |
| alt="output" | |
| className="w-full rounded-md max-w-screen-toast-mobile" | |
| /> | |
| </picture> | |
| ) : ( | |
| <div | |
| key={`${consoleOutput.id}-${index}`} | |
| className="w-full whitespace-pre-line break-words" | |
| > | |
| {content.value} | |
| </div> | |
| ), | |
| )} | |
| </div> | |
| )} | |
| </div> | |
| ))} | |
| <div ref={consoleEndRef} /> | |
| </div> | |
| </div> | |
| </> | |
| ) : null; | |
| } | |