Add comprehensive interactive terminal and CLI system
New Features: - Interactive Terminal Component (InteractiveTerminal.tsx) - Full command input with auto-completion - Command history navigation (up/down arrows) - Real-time suggestions - Auto-scroll and focus management - CLI Command System (cli-commands.ts) - help: Display available commands - clear/cls: Clear terminal - run/execute: Execute Lua scripts - check/lint: Syntax validation - count: Line/word/character counting - api: Roblox API documentation lookup - template: Template management - export: Export scripts to file - echo: Print text - info: System information - Enhanced ConsolePanel - New "Terminal" tab with interactive CLI - Existing log tabs (All, Roblox, Web, Mobile) - Props for code context and modification - Integrated with file system - Keyboard Shortcuts - Ctrl/Cmd + ` : Toggle terminal (VS Code style) Technical Details: - Command execution with context awareness - Auto-completion for commands - Command aliases support - Error handling and user feedback - History management with localStorage-ready structure - Integration with existing code editor
This commit is contained in:
parent
640a8836b6
commit
2d7d88fbc6
4 changed files with 630 additions and 4 deletions
14
src/App.tsx
14
src/App.tsx
|
|
@ -125,6 +125,16 @@ function App() {
|
||||||
},
|
},
|
||||||
description: 'Search in all files',
|
description: 'Search in all files',
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
key: '`',
|
||||||
|
meta: true,
|
||||||
|
ctrl: true,
|
||||||
|
handler: () => {
|
||||||
|
setConsoleCollapsed(!consoleCollapsed);
|
||||||
|
captureEvent('keyboard_shortcut', { action: 'toggle_terminal' });
|
||||||
|
},
|
||||||
|
description: 'Toggle terminal',
|
||||||
|
},
|
||||||
]);
|
]);
|
||||||
|
|
||||||
const handleLoginSuccess = (user: { login: string; avatarUrl: string; email: string }) => {
|
const handleLoginSuccess = (user: { login: string; avatarUrl: string; email: string }) => {
|
||||||
|
|
@ -538,6 +548,10 @@ end)`,
|
||||||
<ConsolePanel
|
<ConsolePanel
|
||||||
collapsed={consoleCollapsed}
|
collapsed={consoleCollapsed}
|
||||||
onToggle={() => setConsoleCollapsed(!consoleCollapsed)}
|
onToggle={() => setConsoleCollapsed(!consoleCollapsed)}
|
||||||
|
currentCode={currentCode}
|
||||||
|
currentFile={activeFileId ? (openFiles || []).find(f => f.id === activeFileId)?.name : undefined}
|
||||||
|
files={files || []}
|
||||||
|
onCodeChange={setCurrentCode}
|
||||||
/>
|
/>
|
||||||
<SearchInFilesPanel
|
<SearchInFilesPanel
|
||||||
files={files || []}
|
files={files || []}
|
||||||
|
|
|
||||||
|
|
@ -3,7 +3,8 @@ import { ScrollArea } from '@/components/ui/scroll-area';
|
||||||
import { Button } from '@/components/ui/button';
|
import { Button } from '@/components/ui/button';
|
||||||
import { Tabs, TabsContent, TabsList, TabsTrigger } from '@/components/ui/tabs';
|
import { Tabs, TabsContent, TabsList, TabsTrigger } from '@/components/ui/tabs';
|
||||||
import { Badge } from '@/components/ui/badge';
|
import { Badge } from '@/components/ui/badge';
|
||||||
import { Trash, Terminal } from '@phosphor-icons/react';
|
import { Trash, Terminal, Code } from '@phosphor-icons/react';
|
||||||
|
import { InteractiveTerminal } from './InteractiveTerminal';
|
||||||
|
|
||||||
interface ConsoleLog {
|
interface ConsoleLog {
|
||||||
id: string;
|
id: string;
|
||||||
|
|
@ -16,9 +17,13 @@ interface ConsoleLog {
|
||||||
interface ConsolePanelProps {
|
interface ConsolePanelProps {
|
||||||
collapsed?: boolean;
|
collapsed?: boolean;
|
||||||
onToggle?: () => void;
|
onToggle?: () => void;
|
||||||
|
currentCode?: string;
|
||||||
|
currentFile?: string;
|
||||||
|
files?: any[];
|
||||||
|
onCodeChange?: (code: string) => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function ConsolePanel({ collapsed, onToggle }: ConsolePanelProps) {
|
export function ConsolePanel({ collapsed, onToggle, currentCode = '', currentFile, files = [], onCodeChange }: ConsolePanelProps) {
|
||||||
const [logs, setLogs] = useState<ConsoleLog[]>([
|
const [logs, setLogs] = useState<ConsoleLog[]>([
|
||||||
{
|
{
|
||||||
id: '1',
|
id: '1',
|
||||||
|
|
@ -109,14 +114,30 @@ export function ConsolePanel({ collapsed, onToggle }: ConsolePanelProps) {
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Tabs defaultValue="all" className="flex-1 flex flex-col">
|
<Tabs defaultValue="terminal" className="flex-1 flex flex-col">
|
||||||
<TabsList className="mx-4 mt-2 h-8 w-fit">
|
<TabsList className="mx-4 mt-2 h-8 w-fit">
|
||||||
<TabsTrigger value="all" className="text-xs">All</TabsTrigger>
|
<TabsTrigger value="terminal" className="text-xs flex items-center gap-1">
|
||||||
|
<Terminal size={12} />
|
||||||
|
Terminal
|
||||||
|
</TabsTrigger>
|
||||||
|
<TabsTrigger value="all" className="text-xs flex items-center gap-1">
|
||||||
|
<Code size={12} />
|
||||||
|
Logs
|
||||||
|
</TabsTrigger>
|
||||||
<TabsTrigger value="roblox" className="text-xs">Roblox</TabsTrigger>
|
<TabsTrigger value="roblox" className="text-xs">Roblox</TabsTrigger>
|
||||||
<TabsTrigger value="web" className="text-xs">Web</TabsTrigger>
|
<TabsTrigger value="web" className="text-xs">Web</TabsTrigger>
|
||||||
<TabsTrigger value="mobile" className="text-xs">Mobile</TabsTrigger>
|
<TabsTrigger value="mobile" className="text-xs">Mobile</TabsTrigger>
|
||||||
</TabsList>
|
</TabsList>
|
||||||
|
|
||||||
|
<TabsContent value="terminal" className="flex-1 m-0">
|
||||||
|
<InteractiveTerminal
|
||||||
|
currentCode={currentCode}
|
||||||
|
currentFile={currentFile}
|
||||||
|
files={files}
|
||||||
|
onCodeChange={onCodeChange}
|
||||||
|
/>
|
||||||
|
</TabsContent>
|
||||||
|
|
||||||
<TabsContent value="all" className="flex-1 m-0">
|
<TabsContent value="all" className="flex-1 m-0">
|
||||||
<ScrollArea className="h-[140px]">
|
<ScrollArea className="h-[140px]">
|
||||||
<div className="px-4 py-2 space-y-1 font-mono text-xs">
|
<div className="px-4 py-2 space-y-1 font-mono text-xs">
|
||||||
|
|
|
||||||
243
src/components/InteractiveTerminal.tsx
Normal file
243
src/components/InteractiveTerminal.tsx
Normal file
|
|
@ -0,0 +1,243 @@
|
||||||
|
import { useState, useEffect, useRef, useCallback } from 'react';
|
||||||
|
import { ScrollArea } from '@/components/ui/scroll-area';
|
||||||
|
import { Input } from '@/components/ui/input';
|
||||||
|
import { executeCommand, CLIContext, CLIResult } from '@/lib/cli-commands';
|
||||||
|
import { toast } from 'sonner';
|
||||||
|
|
||||||
|
interface TerminalLine {
|
||||||
|
id: string;
|
||||||
|
type: 'input' | 'output' | 'error' | 'info' | 'warn' | 'log';
|
||||||
|
content: string;
|
||||||
|
timestamp: Date;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface InteractiveTerminalProps {
|
||||||
|
currentCode: string;
|
||||||
|
currentFile?: string;
|
||||||
|
files: any[];
|
||||||
|
onCodeChange?: (code: string) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function InteractiveTerminal({
|
||||||
|
currentCode,
|
||||||
|
currentFile,
|
||||||
|
files,
|
||||||
|
onCodeChange,
|
||||||
|
}: InteractiveTerminalProps) {
|
||||||
|
const [lines, setLines] = useState<TerminalLine[]>([
|
||||||
|
{
|
||||||
|
id: '0',
|
||||||
|
type: 'info',
|
||||||
|
content: 'AeThex Studio Terminal v1.0.0\nType "help" for available commands',
|
||||||
|
timestamp: new Date(),
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
const [input, setInput] = useState('');
|
||||||
|
const [history, setHistory] = useState<string[]>([]);
|
||||||
|
const [historyIndex, setHistoryIndex] = useState(-1);
|
||||||
|
const [suggestions, setSuggestions] = useState<string[]>([]);
|
||||||
|
const inputRef = useRef<HTMLInputElement>(null);
|
||||||
|
const scrollRef = useRef<HTMLDivElement>(null);
|
||||||
|
|
||||||
|
// Auto-scroll to bottom
|
||||||
|
useEffect(() => {
|
||||||
|
if (scrollRef.current) {
|
||||||
|
scrollRef.current.scrollIntoView({ behavior: 'smooth' });
|
||||||
|
}
|
||||||
|
}, [lines]);
|
||||||
|
|
||||||
|
// Focus input on mount
|
||||||
|
useEffect(() => {
|
||||||
|
inputRef.current?.focus();
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const addLog = useCallback((message: string, type: TerminalLine['type'] = 'log') => {
|
||||||
|
setLines(prev => [
|
||||||
|
...prev,
|
||||||
|
{
|
||||||
|
id: Date.now().toString(),
|
||||||
|
type,
|
||||||
|
content: message,
|
||||||
|
timestamp: new Date(),
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const handleCommand = useCallback(async (command: string) => {
|
||||||
|
if (!command.trim()) return;
|
||||||
|
|
||||||
|
// Add command to history
|
||||||
|
setHistory(prev => [...prev, command]);
|
||||||
|
setHistoryIndex(-1);
|
||||||
|
|
||||||
|
// Add input line
|
||||||
|
addLog(`$ ${command}`, 'input');
|
||||||
|
|
||||||
|
// Execute command
|
||||||
|
const context: CLIContext = {
|
||||||
|
currentCode,
|
||||||
|
currentFile,
|
||||||
|
files,
|
||||||
|
setCode: onCodeChange,
|
||||||
|
addLog,
|
||||||
|
};
|
||||||
|
|
||||||
|
try {
|
||||||
|
const result: CLIResult = await executeCommand(command, context);
|
||||||
|
|
||||||
|
// Handle special commands
|
||||||
|
if (result.output === '__CLEAR__') {
|
||||||
|
setLines([]);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add output
|
||||||
|
if (result.output) {
|
||||||
|
addLog(result.output, result.type || 'log');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!result.success && result.type !== 'warn') {
|
||||||
|
toast.error(result.output);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
addLog(`Error: ${error}`, 'error');
|
||||||
|
toast.error('Command execution failed');
|
||||||
|
}
|
||||||
|
}, [currentCode, currentFile, files, onCodeChange, addLog]);
|
||||||
|
|
||||||
|
const handleSubmit = (e: React.FormEvent) => {
|
||||||
|
e.preventDefault();
|
||||||
|
if (input.trim()) {
|
||||||
|
handleCommand(input);
|
||||||
|
setInput('');
|
||||||
|
setSuggestions([]);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleKeyDown = (e: React.KeyboardEvent<HTMLInputElement>) => {
|
||||||
|
// Command history navigation
|
||||||
|
if (e.key === 'ArrowUp') {
|
||||||
|
e.preventDefault();
|
||||||
|
if (history.length > 0) {
|
||||||
|
const newIndex = historyIndex === -1 ? history.length - 1 : Math.max(0, historyIndex - 1);
|
||||||
|
setHistoryIndex(newIndex);
|
||||||
|
setInput(history[newIndex]);
|
||||||
|
}
|
||||||
|
} else if (e.key === 'ArrowDown') {
|
||||||
|
e.preventDefault();
|
||||||
|
if (historyIndex !== -1) {
|
||||||
|
const newIndex = historyIndex + 1;
|
||||||
|
if (newIndex >= history.length) {
|
||||||
|
setHistoryIndex(-1);
|
||||||
|
setInput('');
|
||||||
|
} else {
|
||||||
|
setHistoryIndex(newIndex);
|
||||||
|
setInput(history[newIndex]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (e.key === 'Tab') {
|
||||||
|
e.preventDefault();
|
||||||
|
// Auto-complete
|
||||||
|
if (suggestions.length > 0) {
|
||||||
|
setInput(suggestions[0]);
|
||||||
|
setSuggestions([]);
|
||||||
|
}
|
||||||
|
} else if (e.key === 'Escape') {
|
||||||
|
setSuggestions([]);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleInputChange = (e: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
|
const value = e.target.value;
|
||||||
|
setInput(value);
|
||||||
|
|
||||||
|
// Simple auto-complete
|
||||||
|
if (value.trim()) {
|
||||||
|
const commandNames = [
|
||||||
|
'help', 'clear', 'cls', 'run', 'execute', 'check', 'lint',
|
||||||
|
'count', 'api', 'template', 'export', 'echo', 'info',
|
||||||
|
];
|
||||||
|
const matches = commandNames.filter(cmd =>
|
||||||
|
cmd.startsWith(value.toLowerCase())
|
||||||
|
);
|
||||||
|
setSuggestions(matches);
|
||||||
|
} else {
|
||||||
|
setSuggestions([]);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const getLineColor = (type: TerminalLine['type']) => {
|
||||||
|
switch (type) {
|
||||||
|
case 'input':
|
||||||
|
return 'text-accent font-semibold';
|
||||||
|
case 'error':
|
||||||
|
return 'text-red-400';
|
||||||
|
case 'warn':
|
||||||
|
return 'text-yellow-400';
|
||||||
|
case 'info':
|
||||||
|
return 'text-blue-400';
|
||||||
|
case 'log':
|
||||||
|
return 'text-green-400';
|
||||||
|
default:
|
||||||
|
return 'text-foreground';
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="h-full flex flex-col bg-background/50 font-mono">
|
||||||
|
<ScrollArea className="flex-1 px-4 py-2">
|
||||||
|
<div className="space-y-1 text-xs">
|
||||||
|
{lines.map((line) => (
|
||||||
|
<div key={line.id} className={`whitespace-pre-wrap ${getLineColor(line.type)}`}>
|
||||||
|
{line.content}
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
<div ref={scrollRef} />
|
||||||
|
</div>
|
||||||
|
</ScrollArea>
|
||||||
|
|
||||||
|
<div className="border-t border-border p-2">
|
||||||
|
<form onSubmit={handleSubmit} className="relative">
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<span className="text-accent font-semibold text-sm">$</span>
|
||||||
|
<Input
|
||||||
|
ref={inputRef}
|
||||||
|
type="text"
|
||||||
|
value={input}
|
||||||
|
onChange={handleInputChange}
|
||||||
|
onKeyDown={handleKeyDown}
|
||||||
|
className="flex-1 bg-background/50 border-none focus-visible:ring-1 focus-visible:ring-accent font-mono text-xs h-7 px-2"
|
||||||
|
placeholder="Type a command... (try 'help')"
|
||||||
|
spellCheck={false}
|
||||||
|
autoComplete="off"
|
||||||
|
aria-label="Terminal command input"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{suggestions.length > 0 && (
|
||||||
|
<div className="absolute bottom-full left-8 mb-1 bg-card border border-border rounded-md shadow-lg p-1 z-10">
|
||||||
|
{suggestions.slice(0, 5).map((suggestion, index) => (
|
||||||
|
<div
|
||||||
|
key={index}
|
||||||
|
className="px-2 py-1 text-xs hover:bg-accent/10 cursor-pointer rounded"
|
||||||
|
onClick={() => {
|
||||||
|
setInput(suggestion);
|
||||||
|
setSuggestions([]);
|
||||||
|
inputRef.current?.focus();
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{suggestion}
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</form>
|
||||||
|
|
||||||
|
<div className="text-[10px] text-muted-foreground mt-1 flex items-center justify-between">
|
||||||
|
<span>↑↓ History | Tab Complete | Esc Clear</span>
|
||||||
|
<span>{history.length} commands</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
348
src/lib/cli-commands.ts
Normal file
348
src/lib/cli-commands.ts
Normal file
|
|
@ -0,0 +1,348 @@
|
||||||
|
// CLI Command System for AeThex Studio
|
||||||
|
|
||||||
|
export interface CLICommand {
|
||||||
|
name: string;
|
||||||
|
description: string;
|
||||||
|
usage: string;
|
||||||
|
aliases?: string[];
|
||||||
|
execute: (args: string[], context: CLIContext) => Promise<CLIResult>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface CLIContext {
|
||||||
|
currentCode: string;
|
||||||
|
currentFile?: string;
|
||||||
|
files: any[];
|
||||||
|
setCode?: (code: string) => void;
|
||||||
|
addLog?: (message: string, type?: 'log' | 'warn' | 'error' | 'info') => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface CLIResult {
|
||||||
|
success: boolean;
|
||||||
|
output: string;
|
||||||
|
type?: 'log' | 'warn' | 'error' | 'info';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Built-in CLI commands
|
||||||
|
export const commands: Record<string, CLICommand> = {
|
||||||
|
help: {
|
||||||
|
name: 'help',
|
||||||
|
description: 'Display available commands',
|
||||||
|
usage: 'help [command]',
|
||||||
|
execute: async (args) => {
|
||||||
|
if (args.length > 0) {
|
||||||
|
const cmd = commands[args[0]] || Object.values(commands).find(c => c.aliases?.includes(args[0]));
|
||||||
|
if (cmd) {
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
output: `${cmd.name} - ${cmd.description}\nUsage: ${cmd.usage}${cmd.aliases ? `\nAliases: ${cmd.aliases.join(', ')}` : ''}`,
|
||||||
|
type: 'info',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
output: `Command not found: ${args[0]}`,
|
||||||
|
type: 'error',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const commandList = Object.values(commands)
|
||||||
|
.map(cmd => ` ${cmd.name.padEnd(15)} - ${cmd.description}`)
|
||||||
|
.join('\n');
|
||||||
|
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
output: `Available Commands:\n${commandList}\n\nType 'help <command>' for more info`,
|
||||||
|
type: 'info',
|
||||||
|
};
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
clear: {
|
||||||
|
name: 'clear',
|
||||||
|
description: 'Clear the terminal',
|
||||||
|
usage: 'clear',
|
||||||
|
aliases: ['cls'],
|
||||||
|
execute: async () => {
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
output: '__CLEAR__',
|
||||||
|
type: 'info',
|
||||||
|
};
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
run: {
|
||||||
|
name: 'run',
|
||||||
|
description: 'Execute current Lua script',
|
||||||
|
usage: 'run',
|
||||||
|
aliases: ['execute', 'exec'],
|
||||||
|
execute: async (args, context) => {
|
||||||
|
if (!context.currentCode || context.currentCode.trim() === '') {
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
output: 'No code to execute',
|
||||||
|
type: 'error',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Simulate script execution
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
output: `Executing script...\n${context.currentFile || 'Untitled'}\n\nScript executed successfully (simulated)`,
|
||||||
|
type: 'info',
|
||||||
|
};
|
||||||
|
} catch (error) {
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
output: `Execution failed: ${error}`,
|
||||||
|
type: 'error',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
check: {
|
||||||
|
name: 'check',
|
||||||
|
description: 'Check current script for syntax errors',
|
||||||
|
usage: 'check',
|
||||||
|
aliases: ['lint', 'validate'],
|
||||||
|
execute: async (args, context) => {
|
||||||
|
if (!context.currentCode || context.currentCode.trim() === '') {
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
output: 'No code to check',
|
||||||
|
type: 'warn',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Simple Lua syntax checks
|
||||||
|
const issues: string[] = [];
|
||||||
|
const lines = context.currentCode.split('\n');
|
||||||
|
|
||||||
|
lines.forEach((line, i) => {
|
||||||
|
// Check for common syntax issues
|
||||||
|
if (line.includes('end)') && !line.includes('function')) {
|
||||||
|
issues.push(`Line ${i + 1}: Possible syntax error - 'end)' found`);
|
||||||
|
}
|
||||||
|
if ((line.match(/\(/g) || []).length !== (line.match(/\)/g) || []).length) {
|
||||||
|
issues.push(`Line ${i + 1}: Unbalanced parentheses`);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (issues.length === 0) {
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
output: `✓ No syntax errors found (${lines.length} lines checked)`,
|
||||||
|
type: 'info',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
output: `Found ${issues.length} potential issue(s):\n${issues.join('\n')}`,
|
||||||
|
type: 'warn',
|
||||||
|
};
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
count: {
|
||||||
|
name: 'count',
|
||||||
|
description: 'Count lines, words, or characters in current script',
|
||||||
|
usage: 'count [lines|words|chars]',
|
||||||
|
execute: async (args, context) => {
|
||||||
|
if (!context.currentCode) {
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
output: 'No code to count',
|
||||||
|
type: 'error',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const lines = context.currentCode.split('\n').length;
|
||||||
|
const words = context.currentCode.split(/\s+/).filter(w => w.length > 0).length;
|
||||||
|
const chars = context.currentCode.length;
|
||||||
|
|
||||||
|
const type = args[0]?.toLowerCase();
|
||||||
|
|
||||||
|
switch (type) {
|
||||||
|
case 'lines':
|
||||||
|
return { success: true, output: `Lines: ${lines}`, type: 'info' };
|
||||||
|
case 'words':
|
||||||
|
return { success: true, output: `Words: ${words}`, type: 'info' };
|
||||||
|
case 'chars':
|
||||||
|
case 'characters':
|
||||||
|
return { success: true, output: `Characters: ${chars}`, type: 'info' };
|
||||||
|
default:
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
output: `Lines: ${lines} | Words: ${words} | Characters: ${chars}`,
|
||||||
|
type: 'info',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
api: {
|
||||||
|
name: 'api',
|
||||||
|
description: 'Search Roblox API documentation',
|
||||||
|
usage: 'api <service|class>',
|
||||||
|
execute: async (args) => {
|
||||||
|
if (args.length === 0) {
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
output: 'Usage: api <service|class>\nExample: api Players, api Workspace',
|
||||||
|
type: 'warn',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const query = args.join(' ');
|
||||||
|
const commonAPIs = {
|
||||||
|
'Players': 'Service for managing player instances',
|
||||||
|
'Workspace': 'Container for 3D objects in the game',
|
||||||
|
'ReplicatedStorage': 'Storage for replicated objects',
|
||||||
|
'ServerStorage': 'Server-only storage',
|
||||||
|
'Instance': 'Base class for all Roblox objects',
|
||||||
|
'Part': 'Basic building block',
|
||||||
|
'Script': 'Server-side Lua code',
|
||||||
|
'LocalScript': 'Client-side Lua code',
|
||||||
|
};
|
||||||
|
|
||||||
|
const match = Object.keys(commonAPIs).find(
|
||||||
|
key => key.toLowerCase() === query.toLowerCase()
|
||||||
|
);
|
||||||
|
|
||||||
|
if (match) {
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
output: `${match}: ${commonAPIs[match as keyof typeof commonAPIs]}\n\nDocs: https://create.roblox.com/docs/reference/engine/classes/${match}`,
|
||||||
|
type: 'info',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
output: `Search: https://create.roblox.com/docs/reference/engine/classes/${query}`,
|
||||||
|
type: 'info',
|
||||||
|
};
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
template: {
|
||||||
|
name: 'template',
|
||||||
|
description: 'List or load code templates',
|
||||||
|
usage: 'template [list|<name>]',
|
||||||
|
execute: async (args, context) => {
|
||||||
|
if (args.length === 0 || args[0] === 'list') {
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
output: `Available templates:
|
||||||
|
- basic : Basic Roblox script structure
|
||||||
|
- playeradded : PlayerAdded event handler
|
||||||
|
- datastore : DataStore usage example
|
||||||
|
- remote : RemoteEvent/RemoteFunction
|
||||||
|
- gui : GUI scripting basics
|
||||||
|
|
||||||
|
Usage: template <name>`,
|
||||||
|
type: 'info',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
output: `Template '${args[0]}' - Use Templates panel (Ctrl+T) to load templates into editor`,
|
||||||
|
type: 'info',
|
||||||
|
};
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
export: {
|
||||||
|
name: 'export',
|
||||||
|
description: 'Export current script to file',
|
||||||
|
usage: 'export [filename]',
|
||||||
|
execute: async (args, context) => {
|
||||||
|
const filename = args[0] || 'script.lua';
|
||||||
|
const code = context.currentCode || '';
|
||||||
|
|
||||||
|
try {
|
||||||
|
const blob = new Blob([code], { type: 'text/plain' });
|
||||||
|
const url = URL.createObjectURL(blob);
|
||||||
|
const a = document.createElement('a');
|
||||||
|
a.href = url;
|
||||||
|
a.download = filename.endsWith('.lua') ? filename : `${filename}.lua`;
|
||||||
|
document.body.appendChild(a);
|
||||||
|
a.click();
|
||||||
|
document.body.removeChild(a);
|
||||||
|
URL.revokeObjectURL(url);
|
||||||
|
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
output: `Exported to ${a.download}`,
|
||||||
|
type: 'info',
|
||||||
|
};
|
||||||
|
} catch (error) {
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
output: `Export failed: ${error}`,
|
||||||
|
type: 'error',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
echo: {
|
||||||
|
name: 'echo',
|
||||||
|
description: 'Print text to terminal',
|
||||||
|
usage: 'echo <text>',
|
||||||
|
execute: async (args) => {
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
output: args.join(' '),
|
||||||
|
type: 'log',
|
||||||
|
};
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
info: {
|
||||||
|
name: 'info',
|
||||||
|
description: 'Display system information',
|
||||||
|
usage: 'info',
|
||||||
|
execute: async (args, context) => {
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
output: `AeThex Studio v1.0.0
|
||||||
|
Environment: Browser-based IDE
|
||||||
|
Platform: Roblox Lua
|
||||||
|
Files: ${context.files?.length || 0}
|
||||||
|
Current File: ${context.currentFile || 'Untitled'}
|
||||||
|
Code Size: ${context.currentCode?.length || 0} characters`,
|
||||||
|
type: 'info',
|
||||||
|
};
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export function executeCommand(
|
||||||
|
input: string,
|
||||||
|
context: CLIContext
|
||||||
|
): Promise<CLIResult> {
|
||||||
|
const parts = input.trim().split(/\s+/);
|
||||||
|
const commandName = parts[0].toLowerCase();
|
||||||
|
const args = parts.slice(1);
|
||||||
|
|
||||||
|
// Find command by name or alias
|
||||||
|
const command = commands[commandName] ||
|
||||||
|
Object.values(commands).find(cmd =>
|
||||||
|
cmd.aliases?.includes(commandName)
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!command) {
|
||||||
|
return Promise.resolve({
|
||||||
|
success: false,
|
||||||
|
output: `Command not found: ${commandName}\nType 'help' for available commands`,
|
||||||
|
type: 'error',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return command.execute(args, context);
|
||||||
|
}
|
||||||
Loading…
Reference in a new issue