Merge branch 'refs/heads/feature/ai-chat'
# Conflicts: # frontend/components/editor/index.tsx
This commit is contained in:
commit
428d2366ff
@ -1,4 +1,5 @@
|
|||||||
import { Anthropic } from "@anthropic-ai/sdk";
|
import { Anthropic } from "@anthropic-ai/sdk";
|
||||||
|
import { MessageParam } from "@anthropic-ai/sdk/src/resources/messages.js";
|
||||||
|
|
||||||
export interface Env {
|
export interface Env {
|
||||||
ANTHROPIC_API_KEY: string;
|
ANTHROPIC_API_KEY: string;
|
||||||
@ -6,69 +7,112 @@ export interface Env {
|
|||||||
|
|
||||||
export default {
|
export default {
|
||||||
async fetch(request: Request, env: Env): Promise<Response> {
|
async fetch(request: Request, env: Env): Promise<Response> {
|
||||||
if (request.method !== "GET") {
|
// Handle CORS preflight requests
|
||||||
|
if (request.method === "OPTIONS") {
|
||||||
|
return new Response(null, {
|
||||||
|
headers: {
|
||||||
|
"Access-Control-Allow-Origin": "*",
|
||||||
|
"Access-Control-Allow-Methods": "GET, POST, OPTIONS",
|
||||||
|
"Access-Control-Allow-Headers": "Content-Type",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (request.method !== "GET" && request.method !== "POST") {
|
||||||
return new Response("Method Not Allowed", { status: 405 });
|
return new Response("Method Not Allowed", { status: 405 });
|
||||||
}
|
}
|
||||||
|
|
||||||
const url = new URL(request.url);
|
let body;
|
||||||
// const fileName = url.searchParams.get("fileName");
|
let isEditCodeWidget = false;
|
||||||
// const line = url.searchParams.get("line");
|
if (request.method === "POST") {
|
||||||
const instructions = url.searchParams.get("instructions");
|
body = await request.json() as { messages: unknown; context: unknown; activeFileContent: string };
|
||||||
const code = url.searchParams.get("code");
|
} else {
|
||||||
|
const url = new URL(request.url);
|
||||||
|
const fileName = url.searchParams.get("fileName") || "";
|
||||||
|
const code = url.searchParams.get("code") || "";
|
||||||
|
const line = url.searchParams.get("line") || "";
|
||||||
|
const instructions = url.searchParams.get("instructions") || "";
|
||||||
|
|
||||||
const prompt = `
|
body = {
|
||||||
Make the following changes to the code below:
|
messages: [{ role: "human", content: instructions }],
|
||||||
- ${instructions}
|
context: `File: ${fileName}\nLine: ${line}\nCode:\n${code}`,
|
||||||
|
activeFileContent: code,
|
||||||
|
};
|
||||||
|
isEditCodeWidget = true;
|
||||||
|
}
|
||||||
|
|
||||||
Return the complete code chunk. Do not refer to other code files. Do not add code before or after the chunk. Start your reponse with \`\`\`, and end with \`\`\`. Do not include any other text.
|
const messages = body.messages;
|
||||||
|
const context = body.context;
|
||||||
|
const activeFileContent = body.activeFileContent;
|
||||||
|
|
||||||
|
if (!Array.isArray(messages) || messages.length === 0) {
|
||||||
|
return new Response("Invalid or empty messages", { status: 400 });
|
||||||
|
}
|
||||||
|
|
||||||
|
let systemMessage;
|
||||||
|
if (isEditCodeWidget) {
|
||||||
|
systemMessage = `You are an AI code editor. Your task is to modify the given code based on the user's instructions. Only output the modified code, without any explanations or markdown formatting. The code should be a direct replacement for the existing code.
|
||||||
|
|
||||||
|
Context:
|
||||||
|
${context}
|
||||||
|
|
||||||
|
Active File Content:
|
||||||
|
${activeFileContent}
|
||||||
|
|
||||||
|
Instructions: ${messages[0].content}
|
||||||
|
|
||||||
|
Respond only with the modified code that can directly replace the existing code.`;
|
||||||
|
} else {
|
||||||
|
systemMessage = `You are an intelligent programming assistant. Please respond to the following request concisely. If your response includes code, please format it using triple backticks (\`\`\`) with the appropriate language identifier. For example:
|
||||||
|
|
||||||
|
\`\`\`python
|
||||||
|
print("Hello, World!")
|
||||||
\`\`\`
|
\`\`\`
|
||||||
${code}
|
|
||||||
\`\`\`
|
|
||||||
`;
|
|
||||||
console.log(prompt);
|
|
||||||
|
|
||||||
try {
|
Provide a clear and concise explanation along with any code snippets. Keep your response brief and to the point.
|
||||||
|
|
||||||
|
${context ? `Context:\n${context}\n` : ''}
|
||||||
|
${activeFileContent ? `Active File Content:\n${activeFileContent}\n` : ''}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
const anthropicMessages = messages.map(msg => ({
|
||||||
|
role: msg.role === 'human' ? 'user' : 'assistant',
|
||||||
|
content: msg.content
|
||||||
|
})) as MessageParam[];
|
||||||
|
|
||||||
|
try {
|
||||||
const anthropic = new Anthropic({ apiKey: env.ANTHROPIC_API_KEY });
|
const anthropic = new Anthropic({ apiKey: env.ANTHROPIC_API_KEY });
|
||||||
|
|
||||||
interface TextBlock {
|
const stream = await anthropic.messages.create({
|
||||||
type: "text";
|
|
||||||
text: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface ToolUseBlock {
|
|
||||||
type: "tool_use";
|
|
||||||
tool_use: {
|
|
||||||
// Add properties if needed
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
type ContentBlock = TextBlock | ToolUseBlock;
|
|
||||||
|
|
||||||
function getTextContent(content: ContentBlock[]): string {
|
|
||||||
for (const block of content) {
|
|
||||||
if (block.type === "text") {
|
|
||||||
return block.text;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return "No text content found";
|
|
||||||
}
|
|
||||||
|
|
||||||
const response = await anthropic.messages.create({
|
|
||||||
model: "claude-3-5-sonnet-20240620",
|
model: "claude-3-5-sonnet-20240620",
|
||||||
max_tokens: 1024,
|
max_tokens: 1024,
|
||||||
messages: [{ role: "user", content: prompt }],
|
system: systemMessage,
|
||||||
|
messages: anthropicMessages,
|
||||||
|
stream: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
const message = response.content as ContentBlock[];
|
const encoder = new TextEncoder();
|
||||||
const textBlockContent = getTextContent(message);
|
|
||||||
|
|
||||||
const pattern = /```[a-zA-Z]*\n([\s\S]*?)\n```/;
|
const streamResponse = new ReadableStream({
|
||||||
const match = textBlockContent.match(pattern);
|
async start(controller) {
|
||||||
|
for await (const chunk of stream) {
|
||||||
|
if (chunk.type === 'content_block_delta' && chunk.delta.type === 'text_delta') {
|
||||||
|
const bytes = encoder.encode(chunk.delta.text);
|
||||||
|
controller.enqueue(bytes);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
controller.close();
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
const codeContent = match ? match[1] : "Error: Could not extract code.";
|
return new Response(streamResponse, {
|
||||||
|
headers: {
|
||||||
return new Response(JSON.stringify({ "response": codeContent }))
|
"Content-Type": "text/plain; charset=utf-8",
|
||||||
|
"Access-Control-Allow-Origin": "*",
|
||||||
|
"Cache-Control": "no-cache",
|
||||||
|
"Connection": "keep-alive",
|
||||||
|
},
|
||||||
|
});
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Error:", error);
|
console.error("Error:", error);
|
||||||
return new Response("Internal Server Error", { status: 500 });
|
return new Response("Internal Server Error", { status: 500 });
|
||||||
|
@ -815,12 +815,28 @@ io.on("connection", async (socket) => {
|
|||||||
generateCodePromise,
|
generateCodePromise,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
const json = await generateCodeResponse.json();
|
if (!generateCodeResponse.ok) {
|
||||||
|
throw new Error(`HTTP error! status: ${generateCodeResponse.status}`);
|
||||||
|
}
|
||||||
|
|
||||||
callback({ response: json.response, success: true });
|
const reader = generateCodeResponse.body?.getReader();
|
||||||
|
const decoder = new TextDecoder();
|
||||||
|
let result = '';
|
||||||
|
|
||||||
|
if (reader) {
|
||||||
|
while (true) {
|
||||||
|
const { done, value } = await reader.read();
|
||||||
|
if (done) break;
|
||||||
|
result += decoder.decode(value, { stream: true });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// The result should now contain only the modified code
|
||||||
|
callback({ response: result.trim(), success: true });
|
||||||
} catch (e: any) {
|
} catch (e: any) {
|
||||||
console.error("Error generating code:", e);
|
console.error("Error generating code:", e);
|
||||||
io.emit("error", `Error: code generation. ${e.message ?? e}`);
|
io.emit("error", `Error: code generation. ${e.message ?? e}`);
|
||||||
|
callback({ response: "Error generating code. Please try again.", success: false });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
36
frontend/components/editor/AIChat/ChatInput.tsx
Normal file
36
frontend/components/editor/AIChat/ChatInput.tsx
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import { Button } from '../../ui/button';
|
||||||
|
import { Send, StopCircle } from 'lucide-react';
|
||||||
|
|
||||||
|
interface ChatInputProps {
|
||||||
|
input: string;
|
||||||
|
setInput: (input: string) => void;
|
||||||
|
isGenerating: boolean;
|
||||||
|
handleSend: () => void;
|
||||||
|
handleStopGeneration: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function ChatInput({ input, setInput, isGenerating, handleSend, handleStopGeneration }: ChatInputProps) {
|
||||||
|
return (
|
||||||
|
<div className="flex space-x-2 min-w-0">
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
value={input}
|
||||||
|
onChange={(e) => setInput(e.target.value)}
|
||||||
|
onKeyPress={(e) => e.key === 'Enter' && !isGenerating && handleSend()}
|
||||||
|
className="flex-grow p-2 border rounded-lg min-w-0 bg-input"
|
||||||
|
placeholder="Type your message..."
|
||||||
|
disabled={isGenerating}
|
||||||
|
/>
|
||||||
|
{isGenerating ? (
|
||||||
|
<Button onClick={handleStopGeneration} variant="destructive" size="icon" className="h-10 w-10">
|
||||||
|
<StopCircle className="w-4 h-4" />
|
||||||
|
</Button>
|
||||||
|
) : (
|
||||||
|
<Button onClick={handleSend} disabled={isGenerating} size="icon" className="h-10 w-10">
|
||||||
|
<Send className="w-4 h-4" />
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
201
frontend/components/editor/AIChat/ChatMessage.tsx
Normal file
201
frontend/components/editor/AIChat/ChatMessage.tsx
Normal file
@ -0,0 +1,201 @@
|
|||||||
|
import React, { useState } from 'react';
|
||||||
|
import { Button } from '../../ui/button';
|
||||||
|
import { ChevronUp, ChevronDown, Copy, Check, CornerUpLeft } from 'lucide-react';
|
||||||
|
import ReactMarkdown from 'react-markdown';
|
||||||
|
import { Prism as SyntaxHighlighter } from 'react-syntax-highlighter';
|
||||||
|
import { vscDarkPlus } from 'react-syntax-highlighter/dist/esm/styles/prism';
|
||||||
|
import remarkGfm from 'remark-gfm';
|
||||||
|
import { copyToClipboard, stringifyContent } from './lib/chatUtils';
|
||||||
|
|
||||||
|
interface MessageProps {
|
||||||
|
message: {
|
||||||
|
role: 'user' | 'assistant';
|
||||||
|
content: string;
|
||||||
|
context?: string;
|
||||||
|
};
|
||||||
|
setContext: (context: string | null) => void;
|
||||||
|
setIsContextExpanded: (isExpanded: boolean) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function ChatMessage({ message, setContext, setIsContextExpanded }: MessageProps) {
|
||||||
|
const [expandedMessageIndex, setExpandedMessageIndex] = useState<number | null>(null);
|
||||||
|
const [copiedText, setCopiedText] = useState<string | null>(null);
|
||||||
|
|
||||||
|
const renderCopyButton = (text: any) => (
|
||||||
|
<Button
|
||||||
|
onClick={() => copyToClipboard(stringifyContent(text), setCopiedText)}
|
||||||
|
size="sm"
|
||||||
|
variant="ghost"
|
||||||
|
className="p-1 h-6"
|
||||||
|
>
|
||||||
|
{copiedText === stringifyContent(text) ? (
|
||||||
|
<Check className="w-4 h-4 text-green-500" />
|
||||||
|
) : (
|
||||||
|
<Copy className="w-4 h-4" />
|
||||||
|
)}
|
||||||
|
</Button>
|
||||||
|
);
|
||||||
|
|
||||||
|
const askAboutCode = (code: any) => {
|
||||||
|
const contextString = stringifyContent(code);
|
||||||
|
setContext(`Regarding this code:\n${contextString}`);
|
||||||
|
setIsContextExpanded(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
const renderMarkdownElement = (props: any) => {
|
||||||
|
const { node, children } = props;
|
||||||
|
const content = stringifyContent(children);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="relative group">
|
||||||
|
<div className="absolute top-0 right-0 flex opacity-0 group-hover:opacity-30 transition-opacity">
|
||||||
|
{renderCopyButton(content)}
|
||||||
|
<Button
|
||||||
|
onClick={() => askAboutCode(content)}
|
||||||
|
size="sm"
|
||||||
|
variant="ghost"
|
||||||
|
className="p-1 h-6"
|
||||||
|
>
|
||||||
|
<CornerUpLeft className="w-4 h-4" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
{React.createElement(node.tagName, {
|
||||||
|
...props,
|
||||||
|
className: `${props.className || ''} hover:bg-transparent rounded p-1 transition-colors`
|
||||||
|
}, children)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="text-left relative">
|
||||||
|
<div className={`relative p-2 rounded-lg ${
|
||||||
|
message.role === 'user'
|
||||||
|
? 'bg-[#262626] text-white'
|
||||||
|
: 'bg-transparent text-white'
|
||||||
|
} max-w-full`}>
|
||||||
|
{message.role === 'user' && (
|
||||||
|
<div className="absolute top-0 right-0 flex opacity-0 group-hover:opacity-30 transition-opacity">
|
||||||
|
{renderCopyButton(message.content)}
|
||||||
|
<Button
|
||||||
|
onClick={() => askAboutCode(message.content)}
|
||||||
|
size="sm"
|
||||||
|
variant="ghost"
|
||||||
|
className="p-1 h-6"
|
||||||
|
>
|
||||||
|
<CornerUpLeft className="w-4 h-4" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{message.context && (
|
||||||
|
<div className="mb-2 bg-input rounded-lg">
|
||||||
|
<div
|
||||||
|
className="flex justify-between items-center cursor-pointer"
|
||||||
|
onClick={() => setExpandedMessageIndex(expandedMessageIndex === 0 ? null : 0)}
|
||||||
|
>
|
||||||
|
<span className="text-sm text-gray-300">
|
||||||
|
Context
|
||||||
|
</span>
|
||||||
|
{expandedMessageIndex === 0 ? (
|
||||||
|
<ChevronUp size={16} />
|
||||||
|
) : (
|
||||||
|
<ChevronDown size={16} />
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
{expandedMessageIndex === 0 && (
|
||||||
|
<div className="relative">
|
||||||
|
<div className="absolute top-0 right-0 flex p-1">
|
||||||
|
{renderCopyButton(message.context.replace(/^Regarding this code:\n/, ''))}
|
||||||
|
</div>
|
||||||
|
{(() => {
|
||||||
|
const code = message.context.replace(/^Regarding this code:\n/, '');
|
||||||
|
const match = /language-(\w+)/.exec(code);
|
||||||
|
const language = match ? match[1] : 'typescript';
|
||||||
|
return (
|
||||||
|
<div className="pt-6">
|
||||||
|
<textarea
|
||||||
|
value={code}
|
||||||
|
onChange={(e) => {
|
||||||
|
const updatedContext = `Regarding this code:\n${e.target.value}`;
|
||||||
|
setContext(updatedContext);
|
||||||
|
}}
|
||||||
|
className="w-full p-2 bg-[#1e1e1e] text-white font-mono text-sm rounded"
|
||||||
|
rows={code.split('\n').length}
|
||||||
|
style={{
|
||||||
|
resize: 'vertical',
|
||||||
|
minHeight: '100px',
|
||||||
|
maxHeight: '400px',
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
})()}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{message.role === 'assistant' ? (
|
||||||
|
<ReactMarkdown
|
||||||
|
remarkPlugins={[remarkGfm]}
|
||||||
|
components={{
|
||||||
|
code({node, className, children, ...props}) {
|
||||||
|
const match = /language-(\w+)/.exec(className || '');
|
||||||
|
return match ? (
|
||||||
|
<div className="relative border border-input rounded-md my-4">
|
||||||
|
<div className="absolute top-0 left-0 px-2 py-1 text-xs font-semibold text-gray-200 bg-#1e1e1e rounded-tl">
|
||||||
|
{match[1]}
|
||||||
|
</div>
|
||||||
|
<div className="absolute top-0 right-0 flex">
|
||||||
|
{renderCopyButton(children)}
|
||||||
|
<Button
|
||||||
|
onClick={() => askAboutCode(children)}
|
||||||
|
size="sm"
|
||||||
|
variant="ghost"
|
||||||
|
className="p-1 h-6"
|
||||||
|
>
|
||||||
|
<CornerUpLeft className="w-4 h-4" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
<div className="pt-6">
|
||||||
|
<SyntaxHighlighter
|
||||||
|
style={vscDarkPlus as any}
|
||||||
|
language={match[1]}
|
||||||
|
PreTag="div"
|
||||||
|
customStyle={{
|
||||||
|
margin: 0,
|
||||||
|
padding: '0.5rem',
|
||||||
|
fontSize: '0.875rem',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{stringifyContent(children)}
|
||||||
|
</SyntaxHighlighter>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<code className={className} {...props}>
|
||||||
|
{children}
|
||||||
|
</code>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
p: renderMarkdownElement,
|
||||||
|
h1: renderMarkdownElement,
|
||||||
|
h2: renderMarkdownElement,
|
||||||
|
h3: renderMarkdownElement,
|
||||||
|
h4: renderMarkdownElement,
|
||||||
|
h5: renderMarkdownElement,
|
||||||
|
h6: renderMarkdownElement,
|
||||||
|
ul: (props) => <ul className="list-disc pl-6 mb-4 space-y-2">{props.children}</ul>,
|
||||||
|
ol: (props) => <ol className="list-decimal pl-6 mb-4 space-y-2">{props.children}</ol>,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{message.content}
|
||||||
|
</ReactMarkdown>
|
||||||
|
) : (
|
||||||
|
<div className="whitespace-pre-wrap group">
|
||||||
|
{message.content}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
48
frontend/components/editor/AIChat/ContextDisplay.tsx
Normal file
48
frontend/components/editor/AIChat/ContextDisplay.tsx
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import { ChevronUp, ChevronDown, X } from 'lucide-react';
|
||||||
|
|
||||||
|
interface ContextDisplayProps {
|
||||||
|
context: string | null;
|
||||||
|
isContextExpanded: boolean;
|
||||||
|
setIsContextExpanded: (isExpanded: boolean) => void;
|
||||||
|
setContext: (context: string | null) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function ContextDisplay({ context, isContextExpanded, setIsContextExpanded, setContext }: ContextDisplayProps) {
|
||||||
|
if (!context) return null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="mb-2 bg-input p-2 rounded-lg">
|
||||||
|
<div className="flex justify-between items-center">
|
||||||
|
<div
|
||||||
|
className="flex-grow cursor-pointer"
|
||||||
|
onClick={() => setIsContextExpanded(!isContextExpanded)}
|
||||||
|
>
|
||||||
|
<span className="text-sm text-gray-300">
|
||||||
|
Context
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div className="flex items-center">
|
||||||
|
{isContextExpanded ? (
|
||||||
|
<ChevronUp size={16} className="cursor-pointer" onClick={() => setIsContextExpanded(false)} />
|
||||||
|
) : (
|
||||||
|
<ChevronDown size={16} className="cursor-pointer" onClick={() => setIsContextExpanded(true)} />
|
||||||
|
)}
|
||||||
|
<X
|
||||||
|
size={16}
|
||||||
|
className="ml-2 cursor-pointer text-gray-400 hover:text-gray-200"
|
||||||
|
onClick={() => setContext(null)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{isContextExpanded && (
|
||||||
|
<textarea
|
||||||
|
value={context.replace(/^Regarding this code:\n/, '')}
|
||||||
|
onChange={(e) => setContext(`Regarding this code:\n${e.target.value}`)}
|
||||||
|
className="w-full mt-2 p-2 bg-#1e1e1e text-white rounded"
|
||||||
|
rows={5}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
73
frontend/components/editor/AIChat/index.tsx
Normal file
73
frontend/components/editor/AIChat/index.tsx
Normal file
@ -0,0 +1,73 @@
|
|||||||
|
import React, { useState, useEffect, useRef } from 'react';
|
||||||
|
import LoadingDots from '../../ui/LoadingDots';
|
||||||
|
import ChatMessage from './ChatMessage';
|
||||||
|
import ChatInput from './ChatInput';
|
||||||
|
import ContextDisplay from './ContextDisplay';
|
||||||
|
import { handleSend, handleStopGeneration } from './lib/chatUtils';
|
||||||
|
|
||||||
|
interface Message {
|
||||||
|
role: 'user' | 'assistant';
|
||||||
|
content: string;
|
||||||
|
context?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function AIChat({ activeFileContent, activeFileName }: { activeFileContent: string, activeFileName: string }) {
|
||||||
|
const [messages, setMessages] = useState<Message[]>([]);
|
||||||
|
const [input, setInput] = useState('');
|
||||||
|
const [isGenerating, setIsGenerating] = useState(false);
|
||||||
|
const chatContainerRef = useRef<HTMLDivElement>(null);
|
||||||
|
const abortControllerRef = useRef<AbortController | null>(null);
|
||||||
|
const [context, setContext] = useState<string | null>(null);
|
||||||
|
const [isContextExpanded, setIsContextExpanded] = useState(false);
|
||||||
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
scrollToBottom();
|
||||||
|
}, [messages]);
|
||||||
|
|
||||||
|
const scrollToBottom = () => {
|
||||||
|
if (chatContainerRef.current) {
|
||||||
|
setTimeout(() => {
|
||||||
|
chatContainerRef.current?.scrollTo({
|
||||||
|
top: chatContainerRef.current.scrollHeight,
|
||||||
|
behavior: 'smooth'
|
||||||
|
});
|
||||||
|
}, 100);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex flex-col h-screen w-full">
|
||||||
|
<div className="flex justify-between items-center p-2 border-b">
|
||||||
|
<span className="text-muted-foreground/50 font-medium">CHAT</span>
|
||||||
|
<span className="text-muted-foreground/50 font-medium truncate max-w-[50%]" title={activeFileName}>{activeFileName}</span>
|
||||||
|
</div>
|
||||||
|
<div ref={chatContainerRef} className="flex-grow overflow-y-auto p-4 space-y-4">
|
||||||
|
{messages.map((message, messageIndex) => (
|
||||||
|
<ChatMessage
|
||||||
|
key={messageIndex}
|
||||||
|
message={message}
|
||||||
|
setContext={setContext}
|
||||||
|
setIsContextExpanded={setIsContextExpanded}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
{isLoading && <LoadingDots />}
|
||||||
|
</div>
|
||||||
|
<div className="p-4 border-t mb-14">
|
||||||
|
<ContextDisplay
|
||||||
|
context={context}
|
||||||
|
isContextExpanded={isContextExpanded}
|
||||||
|
setIsContextExpanded={setIsContextExpanded}
|
||||||
|
setContext={setContext}
|
||||||
|
/>
|
||||||
|
<ChatInput
|
||||||
|
input={input}
|
||||||
|
setInput={setInput}
|
||||||
|
isGenerating={isGenerating}
|
||||||
|
handleSend={() => handleSend(input, context, messages, setMessages, setInput, setIsContextExpanded, setIsGenerating, setIsLoading, abortControllerRef, activeFileContent)}
|
||||||
|
handleStopGeneration={() => handleStopGeneration(abortControllerRef)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
162
frontend/components/editor/AIChat/lib/chatUtils.ts
Normal file
162
frontend/components/editor/AIChat/lib/chatUtils.ts
Normal file
@ -0,0 +1,162 @@
|
|||||||
|
import React from 'react';
|
||||||
|
|
||||||
|
export const stringifyContent = (content: any, seen = new WeakSet()): string => {
|
||||||
|
if (typeof content === 'string') {
|
||||||
|
return content;
|
||||||
|
}
|
||||||
|
if (content === null) {
|
||||||
|
return 'null';
|
||||||
|
}
|
||||||
|
if (content === undefined) {
|
||||||
|
return 'undefined';
|
||||||
|
}
|
||||||
|
if (typeof content === 'number' || typeof content === 'boolean') {
|
||||||
|
return content.toString();
|
||||||
|
}
|
||||||
|
if (typeof content === 'function') {
|
||||||
|
return content.toString();
|
||||||
|
}
|
||||||
|
if (typeof content === 'symbol') {
|
||||||
|
return content.toString();
|
||||||
|
}
|
||||||
|
if (typeof content === 'bigint') {
|
||||||
|
return content.toString() + 'n';
|
||||||
|
}
|
||||||
|
if (React.isValidElement(content)) {
|
||||||
|
return React.Children.toArray((content as React.ReactElement).props.children)
|
||||||
|
.map(child => stringifyContent(child, seen))
|
||||||
|
.join('');
|
||||||
|
}
|
||||||
|
if (Array.isArray(content)) {
|
||||||
|
return '[' + content.map(item => stringifyContent(item, seen)).join(', ') + ']';
|
||||||
|
}
|
||||||
|
if (typeof content === 'object') {
|
||||||
|
if (seen.has(content)) {
|
||||||
|
return '[Circular]';
|
||||||
|
}
|
||||||
|
seen.add(content);
|
||||||
|
try {
|
||||||
|
const pairs = Object.entries(content).map(
|
||||||
|
([key, value]) => `${key}: ${stringifyContent(value, seen)}`
|
||||||
|
);
|
||||||
|
return '{' + pairs.join(', ') + '}';
|
||||||
|
} catch (error) {
|
||||||
|
return Object.prototype.toString.call(content);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return String(content);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const copyToClipboard = (text: string, setCopiedText: (text: string | null) => void) => {
|
||||||
|
navigator.clipboard.writeText(text).then(() => {
|
||||||
|
setCopiedText(text);
|
||||||
|
setTimeout(() => setCopiedText(null), 2000);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
export const handleSend = async (
|
||||||
|
input: string,
|
||||||
|
context: string | null,
|
||||||
|
messages: any[],
|
||||||
|
setMessages: React.Dispatch<React.SetStateAction<any[]>>,
|
||||||
|
setInput: React.Dispatch<React.SetStateAction<string>>,
|
||||||
|
setIsContextExpanded: React.Dispatch<React.SetStateAction<boolean>>,
|
||||||
|
setIsGenerating: React.Dispatch<React.SetStateAction<boolean>>,
|
||||||
|
setIsLoading: React.Dispatch<React.SetStateAction<boolean>>,
|
||||||
|
abortControllerRef: React.MutableRefObject<AbortController | null>,
|
||||||
|
activeFileContent: string
|
||||||
|
) => {
|
||||||
|
if (input.trim() === '' && !context) return;
|
||||||
|
|
||||||
|
const newMessage = {
|
||||||
|
role: 'user' as const,
|
||||||
|
content: input,
|
||||||
|
context: context || undefined
|
||||||
|
};
|
||||||
|
const updatedMessages = [...messages, newMessage];
|
||||||
|
setMessages(updatedMessages);
|
||||||
|
setInput('');
|
||||||
|
setIsContextExpanded(false);
|
||||||
|
setIsGenerating(true);
|
||||||
|
setIsLoading(true);
|
||||||
|
|
||||||
|
abortControllerRef.current = new AbortController();
|
||||||
|
|
||||||
|
try {
|
||||||
|
const anthropicMessages = updatedMessages.map(msg => ({
|
||||||
|
role: msg.role === 'user' ? 'human' : 'assistant',
|
||||||
|
content: msg.content
|
||||||
|
}));
|
||||||
|
|
||||||
|
const response = await fetch(`${process.env.NEXT_PUBLIC_AI_WORKER_URL}/api`, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
body: JSON.stringify({
|
||||||
|
messages: anthropicMessages,
|
||||||
|
context: context || undefined,
|
||||||
|
activeFileContent: activeFileContent,
|
||||||
|
}),
|
||||||
|
signal: abortControllerRef.current.signal,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
throw new Error('Failed to get AI response');
|
||||||
|
}
|
||||||
|
|
||||||
|
const reader = response.body?.getReader();
|
||||||
|
const decoder = new TextDecoder();
|
||||||
|
const assistantMessage = { role: 'assistant' as const, content: '' };
|
||||||
|
setMessages([...updatedMessages, assistantMessage]);
|
||||||
|
setIsLoading(false);
|
||||||
|
|
||||||
|
let buffer = '';
|
||||||
|
const updateInterval = 100;
|
||||||
|
let lastUpdateTime = Date.now();
|
||||||
|
|
||||||
|
if (reader) {
|
||||||
|
while (true) {
|
||||||
|
const { done, value } = await reader.read();
|
||||||
|
if (done) break;
|
||||||
|
buffer += decoder.decode(value, { stream: true });
|
||||||
|
|
||||||
|
const currentTime = Date.now();
|
||||||
|
if (currentTime - lastUpdateTime > updateInterval) {
|
||||||
|
setMessages(prev => {
|
||||||
|
const updatedMessages = [...prev];
|
||||||
|
const lastMessage = updatedMessages[updatedMessages.length - 1];
|
||||||
|
lastMessage.content = buffer;
|
||||||
|
return updatedMessages;
|
||||||
|
});
|
||||||
|
lastUpdateTime = currentTime;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setMessages(prev => {
|
||||||
|
const updatedMessages = [...prev];
|
||||||
|
const lastMessage = updatedMessages[updatedMessages.length - 1];
|
||||||
|
lastMessage.content = buffer;
|
||||||
|
return updatedMessages;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} catch (error: any) {
|
||||||
|
if (error.name === 'AbortError') {
|
||||||
|
console.log('Generation aborted');
|
||||||
|
} else {
|
||||||
|
console.error('Error fetching AI response:', error);
|
||||||
|
const errorMessage = { role: 'assistant' as const, content: 'Sorry, I encountered an error. Please try again.' };
|
||||||
|
setMessages(prev => [...prev, errorMessage]);
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
setIsGenerating(false);
|
||||||
|
setIsLoading(false);
|
||||||
|
abortControllerRef.current = null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const handleStopGeneration = (abortControllerRef: React.MutableRefObject<AbortController | null>) => {
|
||||||
|
if (abortControllerRef.current) {
|
||||||
|
abortControllerRef.current.abort();
|
||||||
|
}
|
||||||
|
};
|
@ -18,7 +18,7 @@ import {
|
|||||||
ResizablePanel,
|
ResizablePanel,
|
||||||
ResizablePanelGroup,
|
ResizablePanelGroup,
|
||||||
} from "@/components/ui/resizable"
|
} from "@/components/ui/resizable"
|
||||||
import { FileJson, Loader2, Sparkles, TerminalSquare } from "lucide-react"
|
import { FileJson, Loader2, Sparkles, TerminalSquare, ArrowDownToLine, ArrowRightToLine } from "lucide-react"
|
||||||
import Tab from "../ui/tab"
|
import Tab from "../ui/tab"
|
||||||
import Sidebar from "./sidebar"
|
import Sidebar from "./sidebar"
|
||||||
import GenerateInput from "./generate"
|
import GenerateInput from "./generate"
|
||||||
@ -37,6 +37,7 @@ import { Button } from "../ui/button"
|
|||||||
import React from "react"
|
import React from "react"
|
||||||
import { parseTSConfigToMonacoOptions } from "@/lib/tsconfig"
|
import { parseTSConfigToMonacoOptions } from "@/lib/tsconfig"
|
||||||
import { cn, deepMerge } from "@/lib/utils"
|
import { cn, deepMerge } from "@/lib/utils"
|
||||||
|
import AIChat from "./AIChat"
|
||||||
|
|
||||||
export default function CodeEditor({
|
export default function CodeEditor({
|
||||||
userData,
|
userData,
|
||||||
@ -73,6 +74,12 @@ export default function CodeEditor({
|
|||||||
message: "",
|
message: "",
|
||||||
})
|
})
|
||||||
|
|
||||||
|
// Layout state
|
||||||
|
const [isHorizontalLayout, setIsHorizontalLayout] = useState(false);
|
||||||
|
|
||||||
|
// AI Chat state
|
||||||
|
const [isAIChatOpen, setIsAIChatOpen] = useState(false);
|
||||||
|
|
||||||
// File state
|
// File state
|
||||||
const [files, setFiles] = useState<(TFolder | TFile)[]>([])
|
const [files, setFiles] = useState<(TFolder | TFile)[]>([])
|
||||||
const [tabs, setTabs] = useState<TTab[]>([])
|
const [tabs, setTabs] = useState<TTab[]>([])
|
||||||
@ -145,7 +152,7 @@ export default function CodeEditor({
|
|||||||
const generateRef = useRef<HTMLDivElement>(null)
|
const generateRef = useRef<HTMLDivElement>(null)
|
||||||
const suggestionRef = useRef<HTMLDivElement>(null)
|
const suggestionRef = useRef<HTMLDivElement>(null)
|
||||||
const generateWidgetRef = useRef<HTMLDivElement>(null)
|
const generateWidgetRef = useRef<HTMLDivElement>(null)
|
||||||
const previewPanelRef = useRef<ImperativePanelHandle>(null)
|
const { previewPanelRef } = usePreview();
|
||||||
const editorPanelRef = useRef<ImperativePanelHandle>(null)
|
const editorPanelRef = useRef<ImperativePanelHandle>(null)
|
||||||
const previewWindowRef = useRef<{ refreshIframe: () => void }>(null)
|
const previewWindowRef = useRef<{ refreshIframe: () => void }>(null)
|
||||||
|
|
||||||
@ -514,20 +521,23 @@ export default function CodeEditor({
|
|||||||
[socket, fileContents]
|
[socket, fileContents]
|
||||||
)
|
)
|
||||||
|
|
||||||
// Keydown event listener to trigger file save on Ctrl+S or Cmd+S
|
// Keydown event listener to trigger file save on Ctrl+S or Cmd+S, and toggle AI chat on Ctrl+L or Cmd+L
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const down = (e: KeyboardEvent) => {
|
const down = (e: KeyboardEvent) => {
|
||||||
if (e.key === "s" && (e.metaKey || e.ctrlKey)) {
|
if (e.key === "s" && (e.metaKey || e.ctrlKey)) {
|
||||||
e.preventDefault()
|
e.preventDefault()
|
||||||
debouncedSaveData(activeFileId)
|
debouncedSaveData(activeFileId);
|
||||||
|
} else if (e.key === "l" && (e.metaKey || e.ctrlKey)) {
|
||||||
|
e.preventDefault()
|
||||||
|
setIsAIChatOpen(prev => !prev);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
document.addEventListener("keydown", down)
|
document.addEventListener("keydown", down)
|
||||||
|
|
||||||
return () => {
|
return () => {
|
||||||
document.removeEventListener("keydown", down)
|
document.removeEventListener("keydown", down)
|
||||||
}
|
}
|
||||||
}, [activeFileId, tabs, debouncedSaveData])
|
}, [activeFileId, tabs, debouncedSaveData, setIsAIChatOpen])
|
||||||
|
|
||||||
// Liveblocks live collaboration setup effect
|
// Liveblocks live collaboration setup effect
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
@ -831,6 +841,20 @@ export default function CodeEditor({
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const togglePreviewPanel = () => {
|
||||||
|
if (isPreviewCollapsed) {
|
||||||
|
previewPanelRef.current?.expand();
|
||||||
|
setIsPreviewCollapsed(false);
|
||||||
|
} else {
|
||||||
|
previewPanelRef.current?.collapse();
|
||||||
|
setIsPreviewCollapsed(true);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const toggleLayout = () => {
|
||||||
|
setIsHorizontalLayout(prev => !prev);
|
||||||
|
};
|
||||||
|
|
||||||
// On disabled access for shared users, show un-interactable loading placeholder + info modal
|
// On disabled access for shared users, show un-interactable loading placeholder + info modal
|
||||||
if (disableAccess.isDisabled)
|
if (disableAccess.isDisabled)
|
||||||
return (
|
return (
|
||||||
@ -953,7 +977,6 @@ export default function CodeEditor({
|
|||||||
/>
|
/>
|
||||||
) : null}
|
) : null}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Main editor components */}
|
{/* Main editor components */}
|
||||||
<Sidebar
|
<Sidebar
|
||||||
sandboxData={sandboxData}
|
sandboxData={sandboxData}
|
||||||
@ -967,143 +990,175 @@ export default function CodeEditor({
|
|||||||
addNew={(name, type) => addNew(name, type, setFiles, sandboxData)}
|
addNew={(name, type) => addNew(name, type, setFiles, sandboxData)}
|
||||||
deletingFolderId={deletingFolderId}
|
deletingFolderId={deletingFolderId}
|
||||||
/>
|
/>
|
||||||
|
{/* Outer ResizablePanelGroup for main layout */}
|
||||||
{/* Shadcn resizeable panels: https://ui.shadcn.com/docs/components/resizable */}
|
|
||||||
<ResizablePanelGroup direction="horizontal">
|
<ResizablePanelGroup direction="horizontal">
|
||||||
<ResizablePanel
|
{/* Left side: Editor and Preview/Terminal */}
|
||||||
className="p-2 flex flex-col"
|
<ResizablePanel defaultSize={isAIChatOpen ? 80 : 100} minSize={50}>
|
||||||
maxSize={80}
|
<ResizablePanelGroup direction={isHorizontalLayout ? "vertical" : "horizontal"}>
|
||||||
minSize={30}
|
|
||||||
defaultSize={60}
|
|
||||||
ref={editorPanelRef}
|
|
||||||
>
|
|
||||||
<div className="h-10 w-full flex gap-2 overflow-auto tab-scroll">
|
|
||||||
{/* File tabs */}
|
|
||||||
{tabs.map((tab) => (
|
|
||||||
<Tab
|
|
||||||
key={tab.id}
|
|
||||||
saved={tab.saved}
|
|
||||||
selected={activeFileId === tab.id}
|
|
||||||
onClick={(e) => {
|
|
||||||
selectFile(tab)
|
|
||||||
}}
|
|
||||||
onClose={() => closeTab(tab.id)}
|
|
||||||
>
|
|
||||||
{tab.name}
|
|
||||||
</Tab>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
{/* Monaco editor */}
|
|
||||||
<div
|
|
||||||
ref={editorContainerRef}
|
|
||||||
className="grow w-full overflow-hidden rounded-md relative"
|
|
||||||
>
|
|
||||||
{!activeFileId ? (
|
|
||||||
<>
|
|
||||||
<div className="w-full h-full flex items-center justify-center text-xl font-medium text-muted-foreground/50 select-none">
|
|
||||||
<FileJson className="w-6 h-6 mr-3" />
|
|
||||||
No file selected.
|
|
||||||
</div>
|
|
||||||
</>
|
|
||||||
) : // Note clerk.loaded is required here due to a bug: https://github.com/clerk/javascript/issues/1643
|
|
||||||
clerk.loaded ? (
|
|
||||||
<>
|
|
||||||
{provider && userInfo ? (
|
|
||||||
<Cursors yProvider={provider} userInfo={userInfo} />
|
|
||||||
) : null}
|
|
||||||
<Editor
|
|
||||||
height="100%"
|
|
||||||
language={editorLanguage}
|
|
||||||
beforeMount={handleEditorWillMount}
|
|
||||||
onMount={handleEditorMount}
|
|
||||||
onChange={(value) => {
|
|
||||||
// If the new content is different from the cached content, update it
|
|
||||||
if (value !== fileContents[activeFileId]) {
|
|
||||||
setActiveFileContent(value ?? "") // Update the active file content
|
|
||||||
// Mark the file as unsaved by setting 'saved' to false
|
|
||||||
setTabs((prev) =>
|
|
||||||
prev.map((tab) =>
|
|
||||||
tab.id === activeFileId
|
|
||||||
? { ...tab, saved: false }
|
|
||||||
: tab
|
|
||||||
)
|
|
||||||
)
|
|
||||||
} else {
|
|
||||||
// If the content matches the cached content, mark the file as saved
|
|
||||||
setTabs((prev) =>
|
|
||||||
prev.map((tab) =>
|
|
||||||
tab.id === activeFileId
|
|
||||||
? { ...tab, saved: true }
|
|
||||||
: tab
|
|
||||||
)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
options={{
|
|
||||||
tabSize: 2,
|
|
||||||
minimap: {
|
|
||||||
enabled: false,
|
|
||||||
},
|
|
||||||
padding: {
|
|
||||||
bottom: 4,
|
|
||||||
top: 4,
|
|
||||||
},
|
|
||||||
scrollBeyondLastLine: false,
|
|
||||||
fixedOverflowWidgets: true,
|
|
||||||
fontFamily: "var(--font-geist-mono)",
|
|
||||||
}}
|
|
||||||
theme="vs-dark"
|
|
||||||
value={activeFileContent}
|
|
||||||
/>
|
|
||||||
</>
|
|
||||||
) : (
|
|
||||||
<div className="w-full h-full flex items-center justify-center text-xl font-medium text-muted-foreground/50 select-none">
|
|
||||||
<Loader2 className="animate-spin w-6 h-6 mr-3" />
|
|
||||||
Waiting for Clerk to load...
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</ResizablePanel>
|
|
||||||
<ResizableHandle />
|
|
||||||
<ResizablePanel defaultSize={40}>
|
|
||||||
<ResizablePanelGroup direction="vertical">
|
|
||||||
<ResizablePanel
|
<ResizablePanel
|
||||||
ref={usePreview().previewPanelRef}
|
|
||||||
defaultSize={4}
|
|
||||||
collapsedSize={4}
|
|
||||||
minSize={25}
|
|
||||||
collapsible
|
|
||||||
className="p-2 flex flex-col"
|
className="p-2 flex flex-col"
|
||||||
onCollapse={() => setIsPreviewCollapsed(true)}
|
maxSize={80}
|
||||||
onExpand={() => setIsPreviewCollapsed(false)}
|
minSize={30}
|
||||||
|
defaultSize={70}
|
||||||
|
ref={editorPanelRef}
|
||||||
>
|
>
|
||||||
<PreviewWindow
|
<div className="h-10 w-full flex gap-2 overflow-auto tab-scroll">
|
||||||
open={() => {
|
{/* File tabs */}
|
||||||
usePreview().previewPanelRef.current?.expand()
|
{tabs.map((tab) => (
|
||||||
setIsPreviewCollapsed(false)
|
<Tab
|
||||||
}}
|
key={tab.id}
|
||||||
collapsed={isPreviewCollapsed}
|
saved={tab.saved}
|
||||||
src={previewURL}
|
selected={activeFileId === tab.id}
|
||||||
ref={previewWindowRef}
|
onClick={(e) => {
|
||||||
/>
|
selectFile(tab)
|
||||||
|
}}
|
||||||
|
onClose={() => closeTab(tab.id)}
|
||||||
|
>
|
||||||
|
{tab.name}
|
||||||
|
</Tab>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
{/* Monaco editor */}
|
||||||
|
<div
|
||||||
|
ref={editorContainerRef}
|
||||||
|
className="grow w-full overflow-hidden rounded-md relative"
|
||||||
|
>
|
||||||
|
{!activeFileId ? (
|
||||||
|
<>
|
||||||
|
<div className="w-full h-full flex items-center justify-center text-xl font-medium text-muted-foreground/50 select-none">
|
||||||
|
<FileJson className="w-6 h-6 mr-3" />
|
||||||
|
No file selected.
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
) : // Note clerk.loaded is required here due to a bug: https://github.com/clerk/javascript/issues/1643
|
||||||
|
clerk.loaded ? (
|
||||||
|
<>
|
||||||
|
{provider && userInfo ? (
|
||||||
|
<Cursors yProvider={provider} userInfo={userInfo} />
|
||||||
|
) : null}
|
||||||
|
<Editor
|
||||||
|
height="100%"
|
||||||
|
language={editorLanguage}
|
||||||
|
beforeMount={handleEditorWillMount}
|
||||||
|
onMount={handleEditorMount}
|
||||||
|
onChange={(value) => {
|
||||||
|
// If the new content is different from the cached content, update it
|
||||||
|
if (value !== fileContents[activeFileId]) {
|
||||||
|
setActiveFileContent(value ?? ""); // Update the active file content
|
||||||
|
// Mark the file as unsaved by setting 'saved' to false
|
||||||
|
setTabs((prev) =>
|
||||||
|
prev.map((tab) =>
|
||||||
|
tab.id === activeFileId
|
||||||
|
? { ...tab, saved: false }
|
||||||
|
: tab
|
||||||
|
)
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
// If the content matches the cached content, mark the file as saved
|
||||||
|
setTabs((prev) =>
|
||||||
|
prev.map((tab) =>
|
||||||
|
tab.id === activeFileId
|
||||||
|
? { ...tab, saved: true }
|
||||||
|
: tab
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
options={{
|
||||||
|
tabSize: 2,
|
||||||
|
minimap: {
|
||||||
|
enabled: false,
|
||||||
|
},
|
||||||
|
padding: {
|
||||||
|
bottom: 4,
|
||||||
|
top: 4,
|
||||||
|
},
|
||||||
|
scrollBeyondLastLine: false,
|
||||||
|
fixedOverflowWidgets: true,
|
||||||
|
fontFamily: "var(--font-geist-mono)",
|
||||||
|
}}
|
||||||
|
theme="vs-dark"
|
||||||
|
value={activeFileContent}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<div className="w-full h-full flex items-center justify-center text-xl font-medium text-muted-foreground/50 select-none">
|
||||||
|
<Loader2 className="animate-spin w-6 h-6 mr-3" />
|
||||||
|
Waiting for Clerk to load...
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
</ResizablePanel>
|
</ResizablePanel>
|
||||||
<ResizableHandle />
|
<ResizableHandle />
|
||||||
<ResizablePanel
|
<ResizablePanel defaultSize={30}>
|
||||||
defaultSize={50}
|
<ResizablePanelGroup direction={
|
||||||
minSize={20}
|
isAIChatOpen && isHorizontalLayout ? "horizontal" :
|
||||||
className="p-2 flex flex-col"
|
isAIChatOpen ? "vertical" :
|
||||||
>
|
isHorizontalLayout ? "horizontal" :
|
||||||
{isOwner ? (
|
"vertical"
|
||||||
<Terminals />
|
}>
|
||||||
) : (
|
<ResizablePanel
|
||||||
<div className="w-full h-full flex items-center justify-center text-lg font-medium text-muted-foreground/50 select-none">
|
ref={previewPanelRef}
|
||||||
<TerminalSquare className="w-4 h-4 mr-2" />
|
defaultSize={isPreviewCollapsed ? 4 : 20}
|
||||||
No terminal access.
|
minSize={25}
|
||||||
</div>
|
collapsedSize={isHorizontalLayout ? 20 : 4}
|
||||||
)}
|
className="p-2 flex flex-col"
|
||||||
|
collapsible
|
||||||
|
onCollapse={() => setIsPreviewCollapsed(true)}
|
||||||
|
onExpand={() => setIsPreviewCollapsed(false)}
|
||||||
|
>
|
||||||
|
<div className="flex items-center justify-between">
|
||||||
|
<Button onClick={toggleLayout} size="sm" variant="ghost" className="mr-2 border">
|
||||||
|
{isHorizontalLayout ? <ArrowRightToLine className="w-4 h-4" /> : <ArrowDownToLine className="w-4 h-4" />}
|
||||||
|
</Button>
|
||||||
|
<PreviewWindow
|
||||||
|
open={togglePreviewPanel}
|
||||||
|
collapsed={isPreviewCollapsed}
|
||||||
|
src={previewURL}
|
||||||
|
ref={previewWindowRef}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
{!isPreviewCollapsed && (
|
||||||
|
<div className="w-full grow rounded-md overflow-hidden bg-foreground mt-2">
|
||||||
|
<iframe
|
||||||
|
width={"100%"}
|
||||||
|
height={"100%"}
|
||||||
|
src={previewURL}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</ResizablePanel>
|
||||||
|
<ResizableHandle />
|
||||||
|
<ResizablePanel
|
||||||
|
defaultSize={50}
|
||||||
|
minSize={20}
|
||||||
|
className="p-2 flex flex-col"
|
||||||
|
>
|
||||||
|
{isOwner ? (
|
||||||
|
<Terminals />
|
||||||
|
) : (
|
||||||
|
<div className="w-full h-full flex items-center justify-center text-lg font-medium text-muted-foreground/50 select-none">
|
||||||
|
<TerminalSquare className="w-4 h-4 mr-2" />
|
||||||
|
No terminal access.
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</ResizablePanel>
|
||||||
|
</ResizablePanelGroup>
|
||||||
</ResizablePanel>
|
</ResizablePanel>
|
||||||
</ResizablePanelGroup>
|
</ResizablePanelGroup>
|
||||||
</ResizablePanel>
|
</ResizablePanel>
|
||||||
|
{/* Right side: AIChat (if open) */}
|
||||||
|
{isAIChatOpen && (
|
||||||
|
<>
|
||||||
|
<ResizableHandle />
|
||||||
|
<ResizablePanel defaultSize={30} minSize={15}>
|
||||||
|
<AIChat
|
||||||
|
activeFileContent={activeFileContent}
|
||||||
|
activeFileName={tabs.find(tab => tab.id === activeFileId)?.name || 'No file selected'}
|
||||||
|
/>
|
||||||
|
</ResizablePanel>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
</ResizablePanelGroup>
|
</ResizablePanelGroup>
|
||||||
</PreviewProvider>
|
</PreviewProvider>
|
||||||
</>
|
</>
|
||||||
@ -1123,4 +1178,4 @@ const defaultCompilerOptions: monaco.languages.typescript.CompilerOptions = {
|
|||||||
module: monaco.languages.typescript.ModuleKind.ESNext,
|
module: monaco.languages.typescript.ModuleKind.ESNext,
|
||||||
moduleResolution: monaco.languages.typescript.ModuleResolutionKind.NodeJs,
|
moduleResolution: monaco.languages.typescript.ModuleResolutionKind.NodeJs,
|
||||||
target: monaco.languages.typescript.ScriptTarget.ESNext,
|
target: monaco.languages.typescript.ScriptTarget.ESNext,
|
||||||
}
|
}
|
@ -4,6 +4,7 @@ import {
|
|||||||
Link,
|
Link,
|
||||||
RotateCw,
|
RotateCw,
|
||||||
TerminalSquare,
|
TerminalSquare,
|
||||||
|
UnfoldVertical,
|
||||||
} from "lucide-react"
|
} from "lucide-react"
|
||||||
import { useEffect, useRef, useState, useImperativeHandle, forwardRef } from "react"
|
import { useEffect, useRef, useState, useImperativeHandle, forwardRef } from "react"
|
||||||
import { toast } from "sonner"
|
import { toast } from "sonner"
|
||||||
@ -32,24 +33,18 @@ ref: React.Ref<{
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div
|
|
||||||
className={`${collapsed ? "h-full" : "h-10"
|
|
||||||
} select-none w-full flex gap-2`}
|
|
||||||
>
|
|
||||||
<div className="h-8 rounded-md px-3 bg-secondary flex items-center w-full justify-between">
|
<div className="h-8 rounded-md px-3 bg-secondary flex items-center w-full justify-between">
|
||||||
<div className="text-xs">Preview</div>
|
<div className="text-xs">Preview</div>
|
||||||
<div className="flex space-x-1 translate-x-1">
|
<div className="flex space-x-1 translate-x-1">
|
||||||
{collapsed ? (
|
{collapsed ? (
|
||||||
<PreviewButton disabled onClick={() => { }}>
|
<PreviewButton onClick={open}>
|
||||||
<TerminalSquare className="w-4 h-4" />
|
<UnfoldVertical className="w-4 h-4" />
|
||||||
</PreviewButton>
|
</PreviewButton>
|
||||||
) : (
|
) : (
|
||||||
<>
|
<>
|
||||||
{/* Removed the unfoldvertical button since we have the same thing via the run button.
|
|
||||||
|
|
||||||
<PreviewButton onClick={open}>
|
<PreviewButton onClick={open}>
|
||||||
<UnfoldVertical className="w-4 h-4" />
|
<UnfoldVertical className="w-4 h-4" />
|
||||||
</PreviewButton> */}
|
</PreviewButton>
|
||||||
|
|
||||||
<PreviewButton
|
<PreviewButton
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
@ -66,18 +61,6 @@ ref: React.Ref<{
|
|||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
|
||||||
{collapsed ? null : (
|
|
||||||
<div className="w-full grow rounded-md overflow-hidden bg-foreground">
|
|
||||||
<iframe
|
|
||||||
key={iframeKey}
|
|
||||||
ref={frameRef}
|
|
||||||
width={"100%"}
|
|
||||||
height={"100%"}
|
|
||||||
src={src}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</>
|
</>
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
|
@ -90,9 +90,9 @@ export default function SidebarFile({
|
|||||||
if (!editing && !pendingDelete && !isMoving)
|
if (!editing && !pendingDelete && !isMoving)
|
||||||
selectFile({ ...data, saved: true });
|
selectFile({ ...data, saved: true });
|
||||||
}}
|
}}
|
||||||
// onDoubleClick={() => {
|
onDoubleClick={() => {
|
||||||
// setEditing(true)
|
setEditing(true)
|
||||||
// }}
|
}}
|
||||||
className={`${
|
className={`${
|
||||||
dragging ? "opacity-50 hover:!bg-background" : ""
|
dragging ? "opacity-50 hover:!bg-background" : ""
|
||||||
} data-[state=open]:bg-secondary/50 w-full flex items-center h-7 px-1 hover:bg-secondary rounded-sm cursor-pointer transition-colors focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring`}
|
} data-[state=open]:bg-secondary/50 w-full flex items-center h-7 px-1 hover:bg-secondary rounded-sm cursor-pointer transition-colors focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring`}
|
||||||
|
32
frontend/components/ui/LoadingDots.tsx
Normal file
32
frontend/components/ui/LoadingDots.tsx
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
import React from 'react';
|
||||||
|
|
||||||
|
const LoadingDots: React.FC = () => {
|
||||||
|
return (
|
||||||
|
<span className="loading-dots">
|
||||||
|
<span className="dot">.</span>
|
||||||
|
<span className="dot">.</span>
|
||||||
|
<span className="dot">.</span>
|
||||||
|
<style jsx>{`
|
||||||
|
.loading-dots {
|
||||||
|
display: inline-block;
|
||||||
|
font-size: 24px;
|
||||||
|
}
|
||||||
|
.dot {
|
||||||
|
opacity: 0;
|
||||||
|
animation: showHideDot 1.5s ease-in-out infinite;
|
||||||
|
}
|
||||||
|
.dot:nth-child(1) { animation-delay: 0s; }
|
||||||
|
.dot:nth-child(2) { animation-delay: 0.5s; }
|
||||||
|
.dot:nth-child(3) { animation-delay: 1s; }
|
||||||
|
@keyframes showHideDot {
|
||||||
|
0% { opacity: 0; }
|
||||||
|
50% { opacity: 1; }
|
||||||
|
100% { opacity: 0; }
|
||||||
|
}
|
||||||
|
`}</style>
|
||||||
|
</span>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default LoadingDots;
|
||||||
|
|
2012
frontend/package-lock.json
generated
2012
frontend/package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@ -12,6 +12,7 @@
|
|||||||
"@atlaskit/pragmatic-drag-and-drop": "^1.1.7",
|
"@atlaskit/pragmatic-drag-and-drop": "^1.1.7",
|
||||||
"@clerk/nextjs": "^4.29.12",
|
"@clerk/nextjs": "^4.29.12",
|
||||||
"@clerk/themes": "^1.7.12",
|
"@clerk/themes": "^1.7.12",
|
||||||
|
"@codemirror/lang-javascript": "^6.2.2",
|
||||||
"@hookform/resolvers": "^3.3.4",
|
"@hookform/resolvers": "^3.3.4",
|
||||||
"@liveblocks/client": "^1.12.0",
|
"@liveblocks/client": "^1.12.0",
|
||||||
"@liveblocks/node": "^1.12.0",
|
"@liveblocks/node": "^1.12.0",
|
||||||
@ -30,6 +31,8 @@
|
|||||||
"@radix-ui/react-slot": "^1.0.2",
|
"@radix-ui/react-slot": "^1.0.2",
|
||||||
"@radix-ui/react-switch": "^1.0.3",
|
"@radix-ui/react-switch": "^1.0.3",
|
||||||
"@react-three/fiber": "^8.16.6",
|
"@react-three/fiber": "^8.16.6",
|
||||||
|
"@uiw/codemirror-theme-vscode": "^4.23.5",
|
||||||
|
"@uiw/react-codemirror": "^4.23.5",
|
||||||
"@vercel/analytics": "^1.2.2",
|
"@vercel/analytics": "^1.2.2",
|
||||||
"@xterm/addon-fit": "^0.10.0",
|
"@xterm/addon-fit": "^0.10.0",
|
||||||
"@xterm/xterm": "^5.5.0",
|
"@xterm/xterm": "^5.5.0",
|
||||||
@ -48,7 +51,10 @@
|
|||||||
"react": "^18.3.1",
|
"react": "^18.3.1",
|
||||||
"react-dom": "^18",
|
"react-dom": "^18",
|
||||||
"react-hook-form": "^7.51.3",
|
"react-hook-form": "^7.51.3",
|
||||||
|
"react-markdown": "^9.0.1",
|
||||||
"react-resizable-panels": "^2.0.16",
|
"react-resizable-panels": "^2.0.16",
|
||||||
|
"react-syntax-highlighter": "^15.5.0",
|
||||||
|
"remark-gfm": "^4.0.0",
|
||||||
"socket.io-client": "^4.7.5",
|
"socket.io-client": "^4.7.5",
|
||||||
"sonner": "^1.4.41",
|
"sonner": "^1.4.41",
|
||||||
"tailwind-merge": "^2.3.0",
|
"tailwind-merge": "^2.3.0",
|
||||||
@ -61,9 +67,11 @@
|
|||||||
"zod": "^3.23.8"
|
"zod": "^3.23.8"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
"@types/estree": "^1.0.6",
|
||||||
"@types/node": "^20",
|
"@types/node": "^20",
|
||||||
"@types/react": "^18.3.3",
|
"@types/react": "^18.3.3",
|
||||||
"@types/react-dom": "^18",
|
"@types/react-dom": "^18",
|
||||||
|
"@types/react-syntax-highlighter": "^15.5.13",
|
||||||
"@types/three": "^0.164.0",
|
"@types/three": "^0.164.0",
|
||||||
"autoprefixer": "^10.0.1",
|
"autoprefixer": "^10.0.1",
|
||||||
"postcss": "^8",
|
"postcss": "^8",
|
||||||
|
3
frontend/react-syntax-highlighter.d.ts
vendored
Normal file
3
frontend/react-syntax-highlighter.d.ts
vendored
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
declare module 'react-syntax-highlighter';
|
||||||
|
declare module 'react-syntax-highlighter/dist/esm/styles/prism';
|
||||||
|
|
@ -1,6 +1,7 @@
|
|||||||
{
|
{
|
||||||
"compilerOptions": {
|
"compilerOptions": {
|
||||||
"lib": ["dom", "dom.iterable", "esnext"],
|
"lib": ["dom", "dom.iterable", "esnext"],
|
||||||
|
"types": ["node"],
|
||||||
"allowJs": true,
|
"allowJs": true,
|
||||||
"skipLibCheck": true,
|
"skipLibCheck": true,
|
||||||
"strict": true,
|
"strict": true,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user