diff --git a/src/core/webview/__tests__/webviewMessageHandler.selectFiles.test.ts b/src/core/webview/__tests__/webviewMessageHandler.selectFiles.test.ts new file mode 100644 index 0000000000..4378704052 --- /dev/null +++ b/src/core/webview/__tests__/webviewMessageHandler.selectFiles.test.ts @@ -0,0 +1,92 @@ +import { describe, it, expect, vi, beforeEach } from "vitest" +import { webviewMessageHandler } from "../webviewMessageHandler" +import * as processFiles from "../../../integrations/misc/process-files" +import { FileReference, FileAttachment } from "../../../shared/FileTypes" + +// Mock the process-files module +vi.mock("../../../integrations/misc/process-files", () => ({ + selectFiles: vi.fn(), + convertImageReferencesToDataUrls: vi.fn(), +})) + +describe("webviewMessageHandler - selectFiles", () => { + let mockProvider: any + + beforeEach(() => { + mockProvider = { + postMessageToWebview: vi.fn(), + } + vi.clearAllMocks() + }) + + it("should convert FileReference[] to string[] for images when handling selectFiles", async () => { + // Mock data + const mockFileReferences: FileReference[] = [ + { + path: "image1.png", + uri: "file:///path/to/image1.png", + size: 1024, + mimeType: "image/png", + }, + { + path: "image2.jpg", + uri: "file:///path/to/image2.jpg", + size: 2048, + mimeType: "image/jpeg", + }, + ] + + const mockFileAttachments: FileAttachment[] = [ + { + path: "file1.txt", + content: "Hello world", + type: "txt", + }, + ] + + const mockDataUrls = [ + "...", + "...", + ] + + // Setup mocks + vi.mocked(processFiles.selectFiles).mockResolvedValue({ + images: mockFileReferences, + files: mockFileAttachments, + }) + vi.mocked(processFiles.convertImageReferencesToDataUrls).mockResolvedValue(mockDataUrls) + + // Execute + await webviewMessageHandler(mockProvider, { type: "selectFiles" }) + + // Verify + expect(processFiles.selectFiles).toHaveBeenCalledOnce() + expect(processFiles.convertImageReferencesToDataUrls).toHaveBeenCalledWith(mockFileReferences) + expect(mockProvider.postMessageToWebview).toHaveBeenCalledWith({ + type: "selectedFiles", + images: mockDataUrls, + files: mockFileAttachments, + }) + }) + + it("should handle empty file selection", async () => { + // Setup mocks for empty selection + vi.mocked(processFiles.selectFiles).mockResolvedValue({ + images: [], + files: [], + }) + vi.mocked(processFiles.convertImageReferencesToDataUrls).mockResolvedValue([]) + + // Execute + await webviewMessageHandler(mockProvider, { type: "selectFiles" }) + + // Verify + expect(processFiles.selectFiles).toHaveBeenCalledOnce() + expect(processFiles.convertImageReferencesToDataUrls).toHaveBeenCalledWith([]) + expect(mockProvider.postMessageToWebview).toHaveBeenCalledWith({ + type: "selectedFiles", + images: [], + files: [], + }) + }) +}) diff --git a/src/core/webview/webviewMessageHandler.ts b/src/core/webview/webviewMessageHandler.ts index a6577fb2fb..803b9c2371 100644 --- a/src/core/webview/webviewMessageHandler.ts +++ b/src/core/webview/webviewMessageHandler.ts @@ -29,7 +29,7 @@ import { experimentDefault } from "../../shared/experiments" import { Terminal } from "../../integrations/terminal/Terminal" import { openFile } from "../../integrations/misc/open-file" import { openImage, saveImage } from "../../integrations/misc/image-handler" -import { selectImages } from "../../integrations/misc/process-images" +import { selectImages, selectFiles, convertImageReferencesToDataUrls } from "../../integrations/misc/process-files" import { getTheme } from "../../integrations/theme/getTheme" import { discoverChromeHostUrl, tryChromeHostUrl } from "../../services/browser/browserDiscovery" import { searchWorkspaceFiles } from "../../services/search/file-search" @@ -418,6 +418,12 @@ export const webviewMessageHandler = async ( const images = await selectImages() await provider.postMessageToWebview({ type: "selectedImages", images }) break + case "selectFiles": + const { images: fileImages, files } = await selectFiles() + // Convert FileReference[] to string[] (base64 data URLs) + const imageDataUrls = await convertImageReferencesToDataUrls(fileImages) + await provider.postMessageToWebview({ type: "selectedFiles", images: imageDataUrls, files }) + break case "exportCurrentTask": const currentTaskId = provider.getCurrentCline()?.taskId if (currentTaskId) { diff --git a/src/integrations/misc/__tests__/process-files.test.ts b/src/integrations/misc/__tests__/process-files.test.ts new file mode 100644 index 0000000000..ec2041b8e7 --- /dev/null +++ b/src/integrations/misc/__tests__/process-files.test.ts @@ -0,0 +1,241 @@ +import { describe, it, expect, vi, beforeEach } from "vitest" +import * as vscode from "vscode" +import fs from "fs/promises" +import * as path from "path" +import { selectFiles, selectImages, convertImageReferencesToDataUrls } from "../process-files" +import { getMimeType, isImageExtension } from "../../../utils/mimeTypes" + +// Mock vscode +vi.mock("vscode", () => ({ + window: { + showOpenDialog: vi.fn(), + showWarningMessage: vi.fn(), + }, + Uri: { + file: (path: string) => ({ fsPath: path, toString: () => `file://${path}` }), + parse: (uri: string) => ({ fsPath: uri.replace("file://", "") }), + }, +})) + +// Mock fs +vi.mock("fs/promises", () => ({ + default: { + readFile: vi.fn(), + stat: vi.fn(), + }, +})) + +// Mock mimeTypes utils +vi.mock("../../../utils/mimeTypes", () => ({ + getMimeType: vi.fn(), + isImageExtension: vi.fn(), +})) + +describe("process-files", () => { + beforeEach(() => { + vi.clearAllMocks() + }) + + describe("selectFiles", () => { + it("should return empty arrays when no files are selected", async () => { + vi.mocked(vscode.window.showOpenDialog).mockResolvedValue(undefined) + + const result = await selectFiles() + + expect(result).toEqual({ images: [], files: [] }) + }) + + it("should process image files correctly with references", async () => { + const mockUri = { + fsPath: "/test/image.png", + toString: () => "file:///test/image.png", + } + vi.mocked(vscode.window.showOpenDialog).mockResolvedValue([mockUri] as any) + vi.mocked(fs.stat).mockResolvedValue({ size: 1000 } as any) + vi.mocked(isImageExtension).mockReturnValue(true) + vi.mocked(getMimeType).mockReturnValue("image/png") + + const result = await selectFiles() + + expect(result.images).toHaveLength(1) + expect(result.images[0]).toEqual({ + path: "image.png", + uri: "file:///test/image.png", + size: 1000, + mimeType: "image/png", + }) + expect(result.files).toHaveLength(0) + }) + + it("should process text files correctly", async () => { + const mockUri = { + fsPath: "/test/data.json", + toString: () => "file:///test/data.json", + } + vi.mocked(vscode.window.showOpenDialog).mockResolvedValue([mockUri] as any) + vi.mocked(fs.stat).mockResolvedValue({ size: 1000 } as any) + vi.mocked(isImageExtension).mockReturnValue(false) + vi.mocked(fs.readFile).mockResolvedValue('{"test": "data"}') + + const result = await selectFiles() + + expect(result.files).toHaveLength(1) + expect(result.files[0]).toEqual({ + path: "data.json", + content: '{"test": "data"}', + type: "json", + }) + expect(result.images).toHaveLength(0) + }) + + it("should handle multiple files of different types", async () => { + const mockUris = [ + { fsPath: "/test/image.jpg", toString: () => "file:///test/image.jpg" }, + { fsPath: "/test/config.xml", toString: () => "file:///test/config.xml" }, + { fsPath: "/test/readme.md", toString: () => "file:///test/readme.md" }, + ] + vi.mocked(vscode.window.showOpenDialog).mockResolvedValue(mockUris as any) + vi.mocked(fs.stat).mockResolvedValue({ size: 1000 } as any) + vi.mocked(isImageExtension).mockReturnValueOnce(true).mockReturnValueOnce(false).mockReturnValueOnce(false) + vi.mocked(getMimeType).mockReturnValue("image/jpeg") + vi.mocked(fs.readFile).mockResolvedValueOnce("test").mockResolvedValueOnce("# README") + + const result = await selectFiles() + + expect(result.images).toHaveLength(1) + expect(result.files).toHaveLength(2) + expect(result.files[0].type).toBe("xml") + expect(result.files[1].type).toBe("md") + }) + + it("should reject files that are too large", async () => { + const mockUri = { + fsPath: "/test/large.txt", + toString: () => "file:///test/large.txt", + } + vi.mocked(vscode.window.showOpenDialog).mockResolvedValue([mockUri] as any) + vi.mocked(fs.stat).mockResolvedValue({ size: 25 * 1024 * 1024 } as any) // 25MB + + const result = await selectFiles() + + expect(vscode.window.showWarningMessage).toHaveBeenCalledWith("File too large: large.txt (max 20MB)") + expect(result).toEqual({ images: [], files: [] }) + }) + + it("should handle binary files gracefully", async () => { + const mockUri = { + fsPath: "/test/binary.exe", + toString: () => "file:///test/binary.exe", + } + vi.mocked(vscode.window.showOpenDialog).mockResolvedValue([mockUri] as any) + vi.mocked(fs.stat).mockResolvedValue({ size: 1000 } as any) + vi.mocked(isImageExtension).mockReturnValue(false) + + const result = await selectFiles() + + expect(vscode.window.showWarningMessage).toHaveBeenCalledWith("Cannot attach binary file: binary.exe") + expect(result).toEqual({ images: [], files: [] }) + }) + + it("should detect binary content in text files", async () => { + const mockUri = { + fsPath: "/test/fake-text.txt", + toString: () => "file:///test/fake-text.txt", + } + vi.mocked(vscode.window.showOpenDialog).mockResolvedValue([mockUri] as any) + vi.mocked(fs.stat).mockResolvedValue({ size: 1000 } as any) + vi.mocked(isImageExtension).mockReturnValue(false) + vi.mocked(fs.readFile).mockResolvedValue("text with \0 null byte") + + const result = await selectFiles() + + expect(vscode.window.showWarningMessage).toHaveBeenCalledWith("File appears to be binary: fake-text.txt") + expect(result).toEqual({ images: [], files: [] }) + }) + + it("should handle file read errors", async () => { + const mockUri = { + fsPath: "/test/unreadable.txt", + toString: () => "file:///test/unreadable.txt", + } + vi.mocked(vscode.window.showOpenDialog).mockResolvedValue([mockUri] as any) + vi.mocked(fs.stat).mockResolvedValue({ size: 1000 } as any) + vi.mocked(isImageExtension).mockReturnValue(false) + vi.mocked(fs.readFile).mockRejectedValue(new Error("Permission denied")) + + const result = await selectFiles() + + expect(vscode.window.showWarningMessage).toHaveBeenCalledWith("Cannot read file as text: unreadable.txt") + expect(result).toEqual({ images: [], files: [] }) + }) + }) + + describe("convertImageReferencesToDataUrls", () => { + it("should convert image references to data URLs", async () => { + const imageRefs = [ + { + path: "image1.png", + uri: "file:///test/image1.png", + size: 1000, + mimeType: "image/png", + }, + { + path: "image2.jpg", + uri: "file:///test/image2.jpg", + size: 2000, + mimeType: "image/jpeg", + }, + ] + + vi.mocked(fs.readFile) + .mockResolvedValueOnce(Buffer.from("fake-png-data")) + .mockResolvedValueOnce(Buffer.from("fake-jpg-data")) + + const result = await convertImageReferencesToDataUrls(imageRefs) + + expect(result).toHaveLength(2) + expect(result[0]).toBe("") + expect(result[1]).toBe("") + }) + + it("should handle errors when converting images", async () => { + const imageRefs = [ + { + path: "missing.png", + uri: "file:///test/missing.png", + size: 1000, + mimeType: "image/png", + }, + ] + + vi.mocked(fs.readFile).mockRejectedValue(new Error("File not found")) + + const result = await convertImageReferencesToDataUrls(imageRefs) + + expect(vscode.window.showWarningMessage).toHaveBeenCalledWith("Failed to read image: missing.png") + expect(result).toEqual([]) + }) + }) + + describe("selectImages", () => { + it("should only return images from selectFiles as data URLs", async () => { + const mockUris = [ + { fsPath: "/test/image.png", toString: () => "file:///test/image.png" }, + { fsPath: "/test/data.json", toString: () => "file:///test/data.json" }, + ] + vi.mocked(vscode.window.showOpenDialog).mockResolvedValue(mockUris as any) + vi.mocked(fs.stat).mockResolvedValue({ size: 1000 } as any) + vi.mocked(isImageExtension).mockReturnValueOnce(true).mockReturnValueOnce(false) + vi.mocked(getMimeType).mockReturnValue("image/png") + // First call for text file read attempt, second for image conversion + vi.mocked(fs.readFile) + .mockResolvedValueOnce('{"test": "data"}') + .mockResolvedValueOnce(Buffer.from("fake-image")) + + const result = await selectImages() + + expect(result).toHaveLength(1) + expect(result[0]).toBe("") + }) + }) +}) diff --git a/src/integrations/misc/process-files.ts b/src/integrations/misc/process-files.ts new file mode 100644 index 0000000000..2373b7a0e2 --- /dev/null +++ b/src/integrations/misc/process-files.ts @@ -0,0 +1,165 @@ +import * as vscode from "vscode" +import fs from "fs/promises" +import * as path from "path" +import { getMimeType, isImageExtension } from "../../utils/mimeTypes" +import { FileAttachment, FileReference, ProcessedFiles } from "../../shared/FileTypes" + +const MAX_FILE_SIZE = 20 * 1024 * 1024 // 20MB + +// Known binary file extensions to skip upfront +const BINARY_EXTENSIONS = [ + // Executables & Libraries + "exe", + "dll", + "so", + "dylib", + "app", + "deb", + "rpm", + "dmg", + "pkg", + "msi", + // Archives + "zip", + "tar", + "gz", + "bz2", + "7z", + "rar", + "jar", + "war", + "ear", + // Media + "mp3", + "mp4", + "avi", + "mov", + "wmv", + "flv", + "mkv", + "webm", + "ogg", + "wav", + "flac", + // Documents (binary formats) + "pdf", + "doc", + "docx", + "xls", + "xlsx", + "ppt", + "pptx", + "odt", + "ods", + "odp", + // Databases + "db", + "sqlite", + "mdb", + // Other binary formats + "pyc", + "pyo", + "class", + "o", + "a", + "lib", + "node", + "wasm", +] + +export async function selectFiles(): Promise { + const options: vscode.OpenDialogOptions = { + canSelectMany: true, + openLabel: "Select", + filters: { + "All Files": ["*"], + }, + } + + const fileUris = await vscode.window.showOpenDialog(options) + + if (!fileUris || fileUris.length === 0) { + return { images: [], files: [] } + } + + const images: FileReference[] = [] + const files: FileAttachment[] = [] + + await Promise.all( + fileUris.map(async (uri) => { + const filePath = uri.fsPath + const ext = path.extname(filePath).toLowerCase().substring(1) + const fileName = path.basename(filePath) + + // Check file size + const stats = await fs.stat(filePath) + if (stats.size > MAX_FILE_SIZE) { + vscode.window.showWarningMessage(`File too large: ${fileName} (max 20MB)`) + return + } + + if (isImageExtension(ext)) { + // Store image reference instead of loading into memory + const mimeType = getMimeType(filePath) + images.push({ + path: fileName, + uri: uri.toString(), + size: stats.size, + mimeType: mimeType, + }) + } else if (BINARY_EXTENSIONS.includes(ext)) { + // Skip known binary files + vscode.window.showWarningMessage(`Cannot attach binary file: ${fileName}`) + } else { + // Try to read as text file + try { + const content = await fs.readFile(filePath, "utf-8") + // Additional check: if the content has null bytes, it's likely binary + if (content.includes("\0")) { + vscode.window.showWarningMessage(`File appears to be binary: ${fileName}`) + } else { + files.push({ + path: fileName, + content: content, + type: ext || "txt", // Default to 'txt' if no extension + }) + } + } catch (error) { + // File couldn't be read as UTF-8, likely binary + vscode.window.showWarningMessage(`Cannot read file as text: ${fileName}`) + } + } + }), + ) + + return { images, files } +} + +/** + * Convert file references to data URLs when needed + * This should be called only when actually sending the images + */ +export async function convertImageReferencesToDataUrls(images: FileReference[]): Promise { + const dataUrls: string[] = [] + + for (const image of images) { + try { + const uri = vscode.Uri.parse(image.uri) + const buffer = await fs.readFile(uri.fsPath) + const base64 = buffer.toString("base64") + const dataUrl = `data:${image.mimeType};base64,${base64}` + dataUrls.push(dataUrl) + } catch (error) { + vscode.window.showWarningMessage(`Failed to read image: ${image.path}`) + } + } + + return dataUrls +} + +// Keep the original selectImages function for backward compatibility +export async function selectImages(): Promise { + const result = await selectFiles() + // Convert references to data URLs for backward compatibility + return convertImageReferencesToDataUrls(result.images) +} diff --git a/src/integrations/misc/process-images.ts b/src/integrations/misc/process-images.ts deleted file mode 100644 index cf3e201538..0000000000 --- a/src/integrations/misc/process-images.ts +++ /dev/null @@ -1,45 +0,0 @@ -import * as vscode from "vscode" -import fs from "fs/promises" -import * as path from "path" - -export async function selectImages(): Promise { - const options: vscode.OpenDialogOptions = { - canSelectMany: true, - openLabel: "Select", - filters: { - Images: ["png", "jpg", "jpeg", "webp"], // supported by anthropic and openrouter - }, - } - - const fileUris = await vscode.window.showOpenDialog(options) - - if (!fileUris || fileUris.length === 0) { - return [] - } - - return await Promise.all( - fileUris.map(async (uri) => { - const imagePath = uri.fsPath - const buffer = await fs.readFile(imagePath) - const base64 = buffer.toString("base64") - const mimeType = getMimeType(imagePath) - const dataUrl = `data:${mimeType};base64,${base64}` - return dataUrl - }), - ) -} - -function getMimeType(filePath: string): string { - const ext = path.extname(filePath).toLowerCase() - switch (ext) { - case ".png": - return "image/png" - case ".jpeg": - case ".jpg": - return "image/jpeg" - case ".webp": - return "image/webp" - default: - throw new Error(`Unsupported file type: ${ext}`) - } -} diff --git a/src/shared/ExtensionMessage.ts b/src/shared/ExtensionMessage.ts index 953c0c1070..e33a9ad75f 100644 --- a/src/shared/ExtensionMessage.ts +++ b/src/shared/ExtensionMessage.ts @@ -54,6 +54,7 @@ export interface ExtensionMessage { | "action" | "state" | "selectedImages" + | "selectedFiles" | "theme" | "workspaceUpdated" | "invoke" @@ -121,6 +122,7 @@ export interface ExtensionMessage { invoke?: "newChat" | "sendMessage" | "primaryButtonClick" | "secondaryButtonClick" | "setChatBoxMessage" state?: ExtensionState images?: string[] + files?: Array<{ path: string; content?: string; type: string }> filePaths?: string[] openedTabs?: Array<{ label: string diff --git a/src/shared/FileTypes.ts b/src/shared/FileTypes.ts new file mode 100644 index 0000000000..a2753c8099 --- /dev/null +++ b/src/shared/FileTypes.ts @@ -0,0 +1,39 @@ +/** + * Type definitions for file attachments + */ + +/** + * Represents a file attachment with its content + */ +export interface FileAttachment { + /** File name or path */ + path: string + /** File content as string */ + content: string + /** File type/extension without dot */ + type: string +} + +/** + * Represents a file reference for memory-efficient handling + */ +export interface FileReference { + /** File name or path */ + path: string + /** VSCode URI for the file */ + uri: string + /** File size in bytes */ + size: number + /** MIME type of the file */ + mimeType: string +} + +/** + * Result of file selection/processing + */ +export interface ProcessedFiles { + /** Image file references */ + images: FileReference[] + /** Text file attachments with content */ + files: FileAttachment[] +} diff --git a/src/shared/WebviewMessage.ts b/src/shared/WebviewMessage.ts index fa9fb67310..5b450325f9 100644 --- a/src/shared/WebviewMessage.ts +++ b/src/shared/WebviewMessage.ts @@ -52,6 +52,7 @@ export interface WebviewMessage { | "clearTask" | "didShowAnnouncement" | "selectImages" + | "selectFiles" | "exportCurrentTask" | "shareCurrentTask" | "showTaskWithId" @@ -201,6 +202,7 @@ export interface WebviewMessage { askResponse?: ClineAskResponse apiConfiguration?: ProviderSettings images?: string[] + files?: Array<{ path: string; content?: string; type: string }> bool?: boolean value?: number commands?: string[] diff --git a/src/utils/__tests__/mimeTypes.test.ts b/src/utils/__tests__/mimeTypes.test.ts new file mode 100644 index 0000000000..d07456cbfa --- /dev/null +++ b/src/utils/__tests__/mimeTypes.test.ts @@ -0,0 +1,111 @@ +import { describe, it, expect } from "vitest" +import { getMimeType, isImageExtension } from "../mimeTypes" + +describe("mimeTypes", () => { + describe("getMimeType", () => { + it("should return correct MIME types for common image formats", () => { + expect(getMimeType("image.png")).toBe("image/png") + expect(getMimeType("photo.jpg")).toBe("image/jpeg") + expect(getMimeType("photo.jpeg")).toBe("image/jpeg") + expect(getMimeType("icon.gif")).toBe("image/gif") + expect(getMimeType("logo.svg")).toBe("image/svg+xml") + expect(getMimeType("picture.webp")).toBe("image/webp") + expect(getMimeType("bitmap.bmp")).toBe("image/bmp") + expect(getMimeType("icon.ico")).toBe("image/x-icon") + }) + + it("should return correct MIME types for text formats", () => { + expect(getMimeType("file.txt")).toBe("text/plain") + expect(getMimeType("script.js")).toBe("text/javascript") + expect(getMimeType("script.ts")).toBe("text/typescript") + expect(getMimeType("style.css")).toBe("text/css") + expect(getMimeType("page.html")).toBe("text/html") + expect(getMimeType("data.json")).toBe("application/json") + expect(getMimeType("config.xml")).toBe("application/xml") + expect(getMimeType("readme.md")).toBe("text/markdown") + expect(getMimeType("config.yaml")).toBe("text/yaml") + expect(getMimeType("config.yml")).toBe("text/yaml") + }) + + it("should return correct MIME types for programming languages", () => { + expect(getMimeType("app.py")).toBe("text/x-python") + expect(getMimeType("main.java")).toBe("text/x-java") + expect(getMimeType("main.cpp")).toBe("text/x-c++src") + expect(getMimeType("main.c")).toBe("text/x-c") + expect(getMimeType("main.cs")).toBe("text/x-csharp") + expect(getMimeType("main.go")).toBe("text/x-go") + expect(getMimeType("main.rs")).toBe("text/x-rust") + expect(getMimeType("main.rb")).toBe("text/x-ruby") + expect(getMimeType("main.php")).toBe("text/x-php") + expect(getMimeType("main.swift")).toBe("text/x-swift") + expect(getMimeType("main.kt")).toBe("text/x-kotlin") + }) + + it("should handle paths with directories", () => { + expect(getMimeType("/path/to/image.png")).toBe("image/png") + expect(getMimeType("C:\\Users\\test\\document.pdf")).toBe("application/pdf") + expect(getMimeType("./src/main.js")).toBe("text/javascript") + }) + + it("should be case insensitive", () => { + expect(getMimeType("IMAGE.PNG")).toBe("image/png") + expect(getMimeType("Script.JS")).toBe("text/javascript") + expect(getMimeType("DATA.JSON")).toBe("application/json") + }) + + it("should return application/octet-stream for unknown extensions", () => { + expect(getMimeType("file.xyz")).toBe("application/octet-stream") + expect(getMimeType("unknown")).toBe("application/octet-stream") + expect(getMimeType("")).toBe("application/octet-stream") + }) + + it("should handle files without extensions", () => { + expect(getMimeType("README")).toBe("application/octet-stream") + expect(getMimeType("Makefile")).toBe("application/octet-stream") + expect(getMimeType(".gitignore")).toBe("application/octet-stream") + }) + + it("should handle special cases", () => { + expect(getMimeType("archive.tar.gz")).toBe("application/octet-stream") + expect(getMimeType("file.min.js")).toBe("text/javascript") + expect(getMimeType("style.min.css")).toBe("text/css") + }) + }) + + describe("isImageExtension", () => { + it("should return true for common image extensions", () => { + expect(isImageExtension("png")).toBe(true) + expect(isImageExtension("jpg")).toBe(true) + expect(isImageExtension("jpeg")).toBe(true) + expect(isImageExtension("gif")).toBe(true) + expect(isImageExtension("webp")).toBe(true) + expect(isImageExtension("bmp")).toBe(true) + expect(isImageExtension("svg")).toBe(true) + expect(isImageExtension("ico")).toBe(true) + }) + + it("should be case insensitive", () => { + expect(isImageExtension("PNG")).toBe(true) + expect(isImageExtension("Jpg")).toBe(true) + expect(isImageExtension("JPEG")).toBe(true) + }) + + it("should return false for non-image extensions", () => { + expect(isImageExtension("txt")).toBe(false) + expect(isImageExtension("js")).toBe(false) + expect(isImageExtension("pdf")).toBe(false) + expect(isImageExtension("doc")).toBe(false) + expect(isImageExtension("mp4")).toBe(false) + }) + + it("should return false for empty or invalid input", () => { + expect(isImageExtension("")).toBe(false) + expect(isImageExtension(" ")).toBe(false) + }) + + it("should handle extensions with dots", () => { + expect(isImageExtension(".png")).toBe(true) + expect(isImageExtension(".jpg")).toBe(true) + }) + }) +}) diff --git a/src/utils/mimeTypes.ts b/src/utils/mimeTypes.ts new file mode 100644 index 0000000000..fe7f7c9413 --- /dev/null +++ b/src/utils/mimeTypes.ts @@ -0,0 +1,162 @@ +/** + * Shared utility for MIME type detection + */ + +import * as path from "path" + +/** + * Get MIME type for a file based on its extension + * @param filePath - The file path or extension + * @returns The MIME type string + * @throws Error if the file type is not supported + */ +export function getMimeType(filePath: string): string { + const ext = path.extname(filePath).toLowerCase() + + switch (ext) { + // Images + case ".png": + return "image/png" + case ".jpg": + case ".jpeg": + return "image/jpeg" + case ".gif": + return "image/gif" + case ".webp": + return "image/webp" + case ".bmp": + return "image/bmp" + case ".svg": + return "image/svg+xml" + case ".ico": + return "image/x-icon" + + // Text files + case ".txt": + return "text/plain" + case ".json": + return "application/json" + case ".xml": + return "application/xml" + case ".yaml": + case ".yml": + return "text/yaml" + case ".csv": + return "text/csv" + case ".tsv": + return "text/tab-separated-values" + case ".md": + return "text/markdown" + case ".log": + return "text/plain" + case ".ini": + case ".cfg": + case ".conf": + return "text/plain" + + // Code files + case ".js": + return "text/javascript" + case ".ts": + return "text/typescript" + case ".jsx": + return "text/jsx" + case ".tsx": + return "text/tsx" + case ".py": + return "text/x-python" + case ".java": + return "text/x-java" + case ".c": + return "text/x-c" + case ".cpp": + case ".cc": + case ".cxx": + return "text/x-c++src" + case ".cs": + return "text/x-csharp" + case ".go": + return "text/x-go" + case ".rs": + return "text/x-rust" + case ".php": + return "text/x-php" + case ".rb": + return "text/x-ruby" + case ".swift": + return "text/x-swift" + case ".kt": + return "text/x-kotlin" + case ".scala": + return "text/x-scala" + case ".r": + return "text/x-r" + case ".m": + return "text/x-objc" + case ".mm": + return "text/x-objc++src" + case ".h": + case ".hpp": + return "text/x-c++hdr" + case ".sh": + case ".bash": + return "text/x-shellscript" + case ".ps1": + return "text/x-powershell" + case ".bat": + case ".cmd": + return "text/x-bat" + + // Web files + case ".html": + case ".htm": + return "text/html" + case ".css": + return "text/css" + case ".scss": + case ".sass": + return "text/x-scss" + case ".less": + return "text/x-less" + + // Documents + case ".pdf": + return "application/pdf" + case ".doc": + return "application/msword" + case ".docx": + return "application/vnd.openxmlformats-officedocument.wordprocessingml.document" + case ".xls": + return "application/vnd.ms-excel" + case ".xlsx": + return "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet" + case ".ppt": + return "application/vnd.ms-powerpoint" + case ".pptx": + return "application/vnd.openxmlformats-officedocument.presentationml.presentation" + + default: + // For unknown extensions, return a generic binary type + return "application/octet-stream" + } +} + +/** + * Check if a file extension represents an image + * @param extension - The file extension (with or without dot) + * @returns true if the extension is for an image file + */ +export function isImageExtension(extension: string): boolean { + const ext = extension.startsWith(".") ? extension.substring(1) : extension + const imageExtensions = ["png", "jpg", "jpeg", "webp", "gif", "bmp", "svg", "ico"] + return imageExtensions.includes(ext.toLowerCase()) +} + +/** + * Check if a MIME type represents an image + * @param mimeType - The MIME type to check + * @returns true if the MIME type is for an image + */ +export function isImageMimeType(mimeType: string): boolean { + return mimeType.startsWith("image/") +} diff --git a/webview-ui/src/components/chat/ChatTextArea.tsx b/webview-ui/src/components/chat/ChatTextArea.tsx index ee622239e2..56076f4a73 100644 --- a/webview-ui/src/components/chat/ChatTextArea.tsx +++ b/webview-ui/src/components/chat/ChatTextArea.tsx @@ -25,8 +25,9 @@ import Thumbnails from "../common/Thumbnails" import ModeSelector from "./ModeSelector" import { MAX_IMAGES_PER_MESSAGE } from "./ChatView" import ContextMenu from "./ContextMenu" -import { VolumeX, Pin, Check, Image, WandSparkles, SendHorizontal } from "lucide-react" +import { VolumeX, Pin, Check, Paperclip, WandSparkles, SendHorizontal } from "lucide-react" import { IndexingStatusBadge } from "./IndexingStatusBadge" +import FileAttachment from "./FileAttachment" import { cn } from "@/lib/utils" import { usePromptHistory } from "./hooks/usePromptHistory" @@ -38,6 +39,8 @@ interface ChatTextAreaProps { placeholderText: string selectedImages: string[] setSelectedImages: React.Dispatch> + selectedFiles?: Array<{ path: string; content: string; type: string }> + setSelectedFiles?: React.Dispatch>> onSend: () => void onSelectImages: () => void shouldDisableImages: boolean @@ -57,6 +60,8 @@ const ChatTextArea = forwardRef( placeholderText, selectedImages, setSelectedImages, + selectedFiles, + setSelectedFiles, onSend, onSelectImages, shouldDisableImages, @@ -853,6 +858,9 @@ const ChatTextArea = forwardRef( /> )} + {selectedFiles && selectedFiles.length > 0 && setSelectedFiles && ( + + )}
(
- +
diff --git a/webview-ui/src/components/chat/ChatView.tsx b/webview-ui/src/components/chat/ChatView.tsx index 38b8997faf..214fd193f3 100644 --- a/webview-ui/src/components/chat/ChatView.tsx +++ b/webview-ui/src/components/chat/ChatView.tsx @@ -143,6 +143,7 @@ const ChatViewComponent: React.ForwardRefRenderFunction(null) const [sendingDisabled, setSendingDisabled] = useState(false) const [selectedImages, setSelectedImages] = useState([]) + const [selectedFiles, setSelectedFiles] = useState>([]) // we need to hold on to the ask because useEffect > lastMessage will always let us know when an ask comes in and handle it, but by the time handleMessage is called, the last message might not be the ask anymore (it could be a say that followed) const [clineAsk, setClineAsk] = useState(undefined) @@ -528,15 +529,15 @@ const ChatViewComponent: React.ForwardRefRenderFunction { + (text: string, images: string[], files?: Array<{ path: string; content: string; type: string }>) => { text = text.trim() - if (text || images.length > 0) { + if (text || images.length > 0 || (files && files.length > 0)) { // Mark that user has responded - this prevents any pending auto-approvals userRespondedRef.current = true if (messagesRef.current.length === 0) { - vscode.postMessage({ type: "newTask", text, images }) + vscode.postMessage({ type: "newTask", text, images, files }) } else if (clineAskRef.current) { if (clineAskRef.current === "followup") { markFollowUpAsAnswered() @@ -556,7 +557,13 @@ const ChatViewComponent: React.ForwardRefRenderFunction vscode.postMessage({ type: "selectImages" }), []) + const selectImages = useCallback(() => vscode.postMessage({ type: "selectFiles" }), []) + const MAX_ATTACHMENTS_PER_MESSAGE = MAX_IMAGES_PER_MESSAGE const shouldDisableImages = - !model?.supportsImages || sendingDisabled || selectedImages.length >= MAX_IMAGES_PER_MESSAGE + !model?.supportsImages || + sendingDisabled || + selectedImages.length + selectedFiles.length >= MAX_ATTACHMENTS_PER_MESSAGE const handleMessage = useCallback( (e: MessageEvent) => { @@ -719,13 +729,38 @@ const ChatViewComponent: React.ForwardRefRenderFunction 0) { + setSelectedImages((prevImages) => + [...prevImages, ...newFileImages].slice(0, MAX_IMAGES_PER_MESSAGE), + ) + } + if (newFiles.length > 0) { + // Filter out files without content + const validFiles = newFiles.filter( + (file): file is { path: string; content: string; type: string } => + file.content !== undefined, + ) + setSelectedFiles((prevFiles) => [...prevFiles, ...validFiles]) + } + break case "invoke": switch (message.invoke!) { case "newChat": handleChatReset() break case "sendMessage": - handleSendMessage(message.text ?? "", message.images ?? []) + handleSendMessage( + message.text ?? "", + message.images ?? [], + message.files?.filter((f) => f.content !== undefined) as Array<{ + path: string + content: string + type: string + }>, + ) break case "setChatBoxMessage": handleSetChatBoxMessage(message.text ?? "", message.images ?? []) @@ -1286,7 +1321,7 @@ const ChatViewComponent: React.ForwardRefRenderFunction 0)) { - handleSendMessage(inputValue, selectedImages) + handleSendMessage(inputValue, selectedImages, selectedFiles) } }, })) @@ -1749,7 +1784,9 @@ const ChatViewComponent: React.ForwardRefRenderFunction handleSendMessage(inputValue, selectedImages)} + selectedFiles={selectedFiles} + setSelectedFiles={setSelectedFiles} + onSend={() => handleSendMessage(inputValue, selectedImages, selectedFiles)} onSelectImages={selectImages} shouldDisableImages={shouldDisableImages} onHeightChange={() => { diff --git a/webview-ui/src/components/chat/FileAttachment.tsx b/webview-ui/src/components/chat/FileAttachment.tsx new file mode 100644 index 0000000000..a7703dd2f1 --- /dev/null +++ b/webview-ui/src/components/chat/FileAttachment.tsx @@ -0,0 +1,127 @@ +import React, { useState, useRef, useLayoutEffect, memo } from "react" +import { useWindowSize } from "react-use" +import { cn } from "@/lib/utils" +import type { FileAttachment as FileAttachmentType } from "../../../../src/shared/FileTypes" + +interface FileAttachmentProps { + files: FileAttachmentType[] + setFiles?: (files: FileAttachmentType[]) => void + style?: React.CSSProperties + className?: string + onHeightChange?: (height: number) => void +} + +const FileAttachment: React.FC = ({ files, setFiles, style, className, onHeightChange }) => { + const [hoveredIndex, setHoveredIndex] = useState(null) + const containerRef = useRef(null) + const { width } = useWindowSize() + + useLayoutEffect(() => { + if (containerRef.current) { + let height = containerRef.current.clientHeight + // some browsers return 0 for clientHeight + if (!height) { + height = containerRef.current.getBoundingClientRect().height + } + onHeightChange?.(height) + } + setHoveredIndex(null) + }, [files, width, onHeightChange]) + + const handleDelete = (index: number) => { + setFiles?.(files.filter((_, i) => i !== index)) + } + + const isDeletable = setFiles !== undefined + + if (files.length === 0) return null + + return ( +
+ {files.map((file, index) => ( +
setHoveredIndex(index)} + onMouseLeave={() => setHoveredIndex(null)}> + + + {file.path.length > 30 ? `${file.path.substring(0, 27)}...` : file.path} + + {isDeletable && hoveredIndex === index && ( +
handleDelete(index)} + className="absolute -top-1 -right-1 w-4 h-4 rounded-full bg-vscode-badge-background flex justify-center items-center cursor-pointer"> + +
+ )} +
+ ))} +
+ ) +} + +function getFileIcon(type: string): string { + const iconMap: Record = { + // Code files + json: "code", + xml: "code", + yaml: "code", + yml: "code", + js: "code", + ts: "code", + jsx: "code", + tsx: "code", + py: "code", + java: "code", + c: "code", + cpp: "code", + cs: "code", + go: "code", + rs: "code", + php: "code", + rb: "code", + swift: "code", + kt: "code", + scala: "code", + r: "code", + sh: "code", + ps1: "code", + bat: "code", + cmd: "code", + // Text files + txt: "text", + md: "markdown", + log: "text", + ini: "code", + cfg: "code", + conf: "code", + // Data files + csv: "table", + tsv: "table", + // Documents + pdf: "pdf", + doc: "file-text", + docx: "file-text", + xls: "table", + xlsx: "table", + ppt: "file-media", + pptx: "file-media", + // Web files + html: "code", + htm: "code", + css: "code", + scss: "code", + sass: "code", + less: "code", + } + return iconMap[type] || "text" +} + +export default memo(FileAttachment) diff --git a/webview-ui/src/components/chat/__tests__/FileAttachment.spec.tsx b/webview-ui/src/components/chat/__tests__/FileAttachment.spec.tsx new file mode 100644 index 0000000000..f2b655f6fb --- /dev/null +++ b/webview-ui/src/components/chat/__tests__/FileAttachment.spec.tsx @@ -0,0 +1,237 @@ +import { describe, it, expect, vi, beforeEach } from "vitest" +import { render, screen, fireEvent, waitFor } from "@testing-library/react" +import FileAttachment from "../FileAttachment" +import { useWindowSize } from "react-use" + +// Mock react-use +vi.mock("react-use", () => ({ + useWindowSize: vi.fn(() => ({ width: 1024, height: 768 })), +})) + +describe("FileAttachment", () => { + const mockFiles = [ + { path: "test.json", content: '{"test": true}', type: "json" }, + { path: "readme.md", content: "# Test", type: "md" }, + { path: "data.xml", content: "", type: "xml" }, + ] + + beforeEach(() => { + vi.clearAllMocks() + }) + + it("should not render when files array is empty", () => { + const { container } = render() + + expect(container.firstChild).toBeNull() + }) + + it("should render all files", () => { + render() + + expect(screen.getByText("test.json")).toBeInTheDocument() + expect(screen.getByText("readme.md")).toBeInTheDocument() + expect(screen.getByText("data.xml")).toBeInTheDocument() + }) + + it("should display correct icons for file types", () => { + const { container } = render() + + expect(container.querySelector(".codicon-file-code")).toBeInTheDocument() // json + expect(container.querySelector(".codicon-file-markdown")).toBeInTheDocument() // md + // XML also uses file-code icon + const codeIcons = container.querySelectorAll(".codicon-file-code") + expect(codeIcons).toHaveLength(2) // json and xml + }) + + it("should show delete icon on hover", async () => { + const { container } = render() + + // Initially, delete icons should not be visible + const deleteIcons = container.querySelectorAll(".codicon-close") + expect(deleteIcons).toHaveLength(0) + + // Hover over first file + const firstFile = container.querySelector(".file-attachment-item") + fireEvent.mouseEnter(firstFile!) + + // Delete icon should appear + await waitFor(() => { + const deleteIcon = container.querySelector(".codicon-close") + expect(deleteIcon).toBeInTheDocument() + }) + }) + + it("should call setFiles when delete icon is clicked", async () => { + const mockSetFiles = vi.fn() + const { container } = render() + + // Hover over first file + const firstFile = container.querySelector(".file-attachment-item") + fireEvent.mouseEnter(firstFile!) + + // Click delete icon container + await waitFor(() => { + const deleteContainer = firstFile!.querySelector("div[class*='cursor-pointer']") + fireEvent.click(deleteContainer!) + }) + + expect(mockSetFiles).toHaveBeenCalledWith([ + { path: "readme.md", content: "# Test", type: "md" }, + { path: "data.xml", content: "", type: "xml" }, + ]) + }) + + it("should apply custom className and style", () => { + const { container } = render( + , + ) + + const wrapper = container.firstChild as HTMLElement + expect(wrapper).toHaveClass("custom-class") + expect(wrapper).toHaveStyle({ marginTop: "10px" }) + }) + + it("should handle unknown file types with default icon", () => { + const unknownFile = [{ path: "unknown.xyz", content: "data", type: "xyz" }] + const { container } = render() + + expect(container.querySelector(".codicon-file-text")).toBeInTheDocument() + }) + + it("should call onHeightChange when height changes", async () => { + const mockOnHeightChange = vi.fn() + + // Mock getBoundingClientRect + const mockGetBoundingClientRect = vi.fn(() => ({ + height: 100, + width: 200, + top: 0, + left: 0, + bottom: 100, + right: 200, + x: 0, + y: 0, + toJSON: () => {}, + })) + + Object.defineProperty(HTMLElement.prototype, "getBoundingClientRect", { + value: mockGetBoundingClientRect, + configurable: true, + }) + + const { rerender } = render() + + // Wait for initial height calculation + await waitFor(() => { + expect(mockOnHeightChange).toHaveBeenCalledWith(100) + }) + + // Clear mock + mockOnHeightChange.mockClear() + + // Add more files + const moreFiles = [...mockFiles, { path: "new.txt", content: "new", type: "txt" }] + rerender() + + // Should call onHeightChange again + await waitFor(() => { + expect(mockOnHeightChange).toHaveBeenCalled() + }) + }) + + it("should respond to window size changes", async () => { + const mockOnHeightChange = vi.fn() + + // Mock getBoundingClientRect + const mockGetBoundingClientRect = vi.fn(() => ({ + height: 100, + width: 200, + top: 0, + left: 0, + bottom: 100, + right: 200, + x: 0, + y: 0, + toJSON: () => {}, + })) + + Object.defineProperty(HTMLElement.prototype, "getBoundingClientRect", { + value: mockGetBoundingClientRect, + configurable: true, + }) + + // Initial render with width 1024 + const { unmount } = render() + + // Wait for initial render + await waitFor(() => { + expect(mockOnHeightChange).toHaveBeenCalled() + }) + + // Clear mock and unmount + mockOnHeightChange.mockClear() + unmount() + + // Change window size mock + vi.mocked(useWindowSize).mockReturnValue({ width: 800, height: 600 }) + + // Re-render with new window size + render() + + // Should trigger height recalculation due to width change + await waitFor(() => { + expect(mockOnHeightChange).toHaveBeenCalled() + }) + }) + + it("should not show delete icons when setFiles is not provided", async () => { + const { container } = render() + + // Hover over first file + const firstFile = container.querySelector(".file-attachment-item") + fireEvent.mouseEnter(firstFile!) + + // Delete icon should not appear + await waitFor(() => { + const deleteIcon = container.querySelector(".codicon-close") + expect(deleteIcon).not.toBeInTheDocument() + }) + }) + + it("should handle file deletion correctly with hover state", async () => { + const mockSetFiles = vi.fn() + const { container } = render() + + // Hover over second file + const files = container.querySelectorAll(".file-attachment-item") + fireEvent.mouseEnter(files[1]) + + // Click delete icon container + await waitFor(() => { + const deleteContainer = files[1].querySelector("div[class*='cursor-pointer']") + fireEvent.click(deleteContainer!) + }) + + expect(mockSetFiles).toHaveBeenCalledWith([ + { path: "test.json", content: '{"test": true}', type: "json" }, + { path: "data.xml", content: "", type: "xml" }, + ]) + }) + + it("should be memoized and not re-render unnecessarily", () => { + const mockSetFiles = vi.fn() + const { rerender } = render() + + // Re-render with same props + rerender() + + // Component should be memoized, so no additional renders + // This is implicitly tested by React.memo + expect(true).toBe(true) + }) +}) diff --git a/webview-ui/src/i18n/locales/ca/chat.json b/webview-ui/src/i18n/locales/ca/chat.json index 6accf68fcb..0b7d443d68 100644 --- a/webview-ui/src/i18n/locales/ca/chat.json +++ b/webview-ui/src/i18n/locales/ca/chat.json @@ -104,6 +104,7 @@ "selectApiConfig": "Seleccioneu la configuració de l'API", "enhancePrompt": "Millora la sol·licitud amb context addicional", "addImages": "Afegeix imatges al missatge", + "addFiles": "Adjunta fitxers al missatge", "sendMessage": "Envia el missatge", "stopTts": "Atura la síntesi de veu", "typeMessage": "Escriu un missatge...", diff --git a/webview-ui/src/i18n/locales/de/chat.json b/webview-ui/src/i18n/locales/de/chat.json index fcbbdb2b68..a3849af53b 100644 --- a/webview-ui/src/i18n/locales/de/chat.json +++ b/webview-ui/src/i18n/locales/de/chat.json @@ -104,6 +104,7 @@ "selectApiConfig": "API-Konfiguration auswählen", "enhancePrompt": "Prompt mit zusätzlichem Kontext verbessern", "addImages": "Bilder zur Nachricht hinzufügen", + "addFiles": "Dateien an die Nachricht anhängen", "sendMessage": "Nachricht senden", "stopTts": "Text-in-Sprache beenden", "typeMessage": "Nachricht eingeben...", diff --git a/webview-ui/src/i18n/locales/en/chat.json b/webview-ui/src/i18n/locales/en/chat.json index 326f52c74d..045490576e 100644 --- a/webview-ui/src/i18n/locales/en/chat.json +++ b/webview-ui/src/i18n/locales/en/chat.json @@ -119,6 +119,7 @@ }, "enhancePromptDescription": "The 'Enhance Prompt' button helps improve your prompt by providing additional context, clarification, or rephrasing. Try typing a prompt in here and clicking the button again to see how it works.", "addImages": "Add images to message", + "addFiles": "Attach files to message", "sendMessage": "Send message", "stopTts": "Stop text-to-speech", "typeMessage": "Type a message...", diff --git a/webview-ui/src/i18n/locales/es/chat.json b/webview-ui/src/i18n/locales/es/chat.json index 9bea6bd2c4..76a9f3cc8f 100644 --- a/webview-ui/src/i18n/locales/es/chat.json +++ b/webview-ui/src/i18n/locales/es/chat.json @@ -104,6 +104,7 @@ "selectApiConfig": "Seleccionar configuración de API", "enhancePrompt": "Mejorar el mensaje con contexto adicional", "addImages": "Agregar imágenes al mensaje", + "addFiles": "Adjuntar archivos al mensaje", "sendMessage": "Enviar mensaje", "stopTts": "Detener texto a voz", "typeMessage": "Escribe un mensaje...", diff --git a/webview-ui/src/i18n/locales/fr/chat.json b/webview-ui/src/i18n/locales/fr/chat.json index d7748fb15c..5db8fd40ea 100644 --- a/webview-ui/src/i18n/locales/fr/chat.json +++ b/webview-ui/src/i18n/locales/fr/chat.json @@ -104,6 +104,7 @@ "selectApiConfig": "Sélectionner la configuration de l'API", "enhancePrompt": "Améliorer la requête avec un contexte supplémentaire", "addImages": "Ajouter des images au message", + "addFiles": "Joindre des fichiers au message", "sendMessage": "Envoyer le message", "stopTts": "Arrêter la synthèse vocale", "typeMessage": "Écrivez un message...", diff --git a/webview-ui/src/i18n/locales/hi/chat.json b/webview-ui/src/i18n/locales/hi/chat.json index 7979f624e0..b8da93bca2 100644 --- a/webview-ui/src/i18n/locales/hi/chat.json +++ b/webview-ui/src/i18n/locales/hi/chat.json @@ -104,6 +104,7 @@ "selectApiConfig": "एपीआई कॉन्फ़िगरेशन का चयन करें", "enhancePrompt": "अतिरिक्त संदर्भ के साथ प्रॉम्प्ट बढ़ाएँ", "addImages": "संदेश में चित्र जोड़ें", + "addFiles": "संदेश में फ़ाइलें अनुलग्न करें", "sendMessage": "संदेश भेजें", "stopTts": "टेक्स्ट-टू-स्पीच बंद करें", "typeMessage": "एक संदेश लिखें...", diff --git a/webview-ui/src/i18n/locales/id/chat.json b/webview-ui/src/i18n/locales/id/chat.json index 2134b48bfc..ccb98009aa 100644 --- a/webview-ui/src/i18n/locales/id/chat.json +++ b/webview-ui/src/i18n/locales/id/chat.json @@ -125,6 +125,7 @@ "description": "Persona khusus yang menyesuaikan perilaku Roo." }, "addImages": "Tambahkan gambar ke pesan", + "addFiles": "Lampirkan file ke pesan", "sendMessage": "Kirim pesan", "stopTts": "Hentikan text-to-speech", "typeMessage": "Ketik pesan...", diff --git a/webview-ui/src/i18n/locales/it/chat.json b/webview-ui/src/i18n/locales/it/chat.json index ed886e6ef9..e093200756 100644 --- a/webview-ui/src/i18n/locales/it/chat.json +++ b/webview-ui/src/i18n/locales/it/chat.json @@ -104,6 +104,7 @@ "selectApiConfig": "Seleziona la configurazione API", "enhancePrompt": "Migliora prompt con contesto aggiuntivo", "addImages": "Aggiungi immagini al messaggio", + "addFiles": "Allega file al messaggio", "sendMessage": "Invia messaggio", "stopTts": "Interrompi sintesi vocale", "typeMessage": "Scrivi un messaggio...", diff --git a/webview-ui/src/i18n/locales/ja/chat.json b/webview-ui/src/i18n/locales/ja/chat.json index 61c2820299..571fb2fa6e 100644 --- a/webview-ui/src/i18n/locales/ja/chat.json +++ b/webview-ui/src/i18n/locales/ja/chat.json @@ -104,6 +104,7 @@ "selectApiConfig": "API構成を選択", "enhancePrompt": "追加コンテキストでプロンプトを強化", "addImages": "メッセージに画像を追加", + "addFiles": "メッセージにファイルを添付", "sendMessage": "メッセージを送信", "stopTts": "テキスト読み上げを停止", "typeMessage": "メッセージを入力...", diff --git a/webview-ui/src/i18n/locales/ko/chat.json b/webview-ui/src/i18n/locales/ko/chat.json index 9ac8e90644..4ad986b773 100644 --- a/webview-ui/src/i18n/locales/ko/chat.json +++ b/webview-ui/src/i18n/locales/ko/chat.json @@ -104,6 +104,7 @@ "selectApiConfig": "API 구성 선택", "enhancePrompt": "추가 컨텍스트로 프롬프트 향상", "addImages": "메시지에 이미지 추가", + "addFiles": "메시지에 파일 첨부", "sendMessage": "메시지 보내기", "stopTts": "텍스트 음성 변환 중지", "typeMessage": "메시지 입력...", diff --git a/webview-ui/src/i18n/locales/nl/chat.json b/webview-ui/src/i18n/locales/nl/chat.json index 5b2da04160..fb67567c01 100644 --- a/webview-ui/src/i18n/locales/nl/chat.json +++ b/webview-ui/src/i18n/locales/nl/chat.json @@ -111,6 +111,7 @@ "description": "Gespecialiseerde persona's die het gedrag van Roo aanpassen." }, "addImages": "Afbeeldingen toevoegen aan bericht", + "addFiles": "Bestanden aan bericht toevoegen", "sendMessage": "Bericht verzenden", "stopTts": "Stop tekst-naar-spraak", "typeMessage": "Typ een bericht...", diff --git a/webview-ui/src/i18n/locales/pl/chat.json b/webview-ui/src/i18n/locales/pl/chat.json index 5226ef51ea..099d8e2458 100644 --- a/webview-ui/src/i18n/locales/pl/chat.json +++ b/webview-ui/src/i18n/locales/pl/chat.json @@ -104,6 +104,7 @@ "selectApiConfig": "Wybierz konfigurację API", "enhancePrompt": "Ulepsz podpowiedź dodatkowym kontekstem", "addImages": "Dodaj obrazy do wiadomości", + "addFiles": "Dołącz pliki do wiadomości", "sendMessage": "Wyślij wiadomość", "stopTts": "Zatrzymaj syntezę mowy", "typeMessage": "Wpisz wiadomość...", diff --git a/webview-ui/src/i18n/locales/pt-BR/chat.json b/webview-ui/src/i18n/locales/pt-BR/chat.json index eca50fbd33..b846dd3119 100644 --- a/webview-ui/src/i18n/locales/pt-BR/chat.json +++ b/webview-ui/src/i18n/locales/pt-BR/chat.json @@ -104,6 +104,7 @@ "selectApiConfig": "Selecionar configuração da API", "enhancePrompt": "Aprimorar prompt com contexto adicional", "addImages": "Adicionar imagens à mensagem", + "addFiles": "Anexar arquivos à mensagem", "sendMessage": "Enviar mensagem", "stopTts": "Parar conversão de texto em fala", "typeMessage": "Digite uma mensagem...", diff --git a/webview-ui/src/i18n/locales/ru/chat.json b/webview-ui/src/i18n/locales/ru/chat.json index 72ae883703..9bdd17e6b2 100644 --- a/webview-ui/src/i18n/locales/ru/chat.json +++ b/webview-ui/src/i18n/locales/ru/chat.json @@ -111,6 +111,7 @@ "description": "Специализированные персоны, которые настраивают поведение Roo." }, "addImages": "Добавить изображения к сообщению", + "addFiles": "Прикрепить файлы к сообщению", "sendMessage": "Отправить сообщение", "stopTts": "Остановить синтез речи", "typeMessage": "Введите сообщение...", diff --git a/webview-ui/src/i18n/locales/tr/chat.json b/webview-ui/src/i18n/locales/tr/chat.json index 4f108d7ac3..ef5ced35ea 100644 --- a/webview-ui/src/i18n/locales/tr/chat.json +++ b/webview-ui/src/i18n/locales/tr/chat.json @@ -104,6 +104,7 @@ "selectApiConfig": "API yapılandırmasını seçin", "enhancePrompt": "Ek bağlamla istemi geliştir", "addImages": "Mesaja resim ekle", + "addFiles": "Mesaja dosya ekle", "sendMessage": "Mesaj gönder", "stopTts": "Metin okumayı durdur", "typeMessage": "Bir mesaj yazın...", diff --git a/webview-ui/src/i18n/locales/vi/chat.json b/webview-ui/src/i18n/locales/vi/chat.json index 0f0ea71ec3..d50b3c22e5 100644 --- a/webview-ui/src/i18n/locales/vi/chat.json +++ b/webview-ui/src/i18n/locales/vi/chat.json @@ -104,6 +104,7 @@ "selectApiConfig": "Chọn cấu hình API", "enhancePrompt": "Nâng cao yêu cầu với ngữ cảnh bổ sung", "addImages": "Thêm hình ảnh vào tin nhắn", + "addFiles": "Đính kèm tệp vào tin nhắn", "sendMessage": "Gửi tin nhắn", "stopTts": "Dừng chuyển văn bản thành giọng nói", "typeMessage": "Nhập tin nhắn...", diff --git a/webview-ui/src/i18n/locales/zh-CN/chat.json b/webview-ui/src/i18n/locales/zh-CN/chat.json index 6e883b751a..fd680cb978 100644 --- a/webview-ui/src/i18n/locales/zh-CN/chat.json +++ b/webview-ui/src/i18n/locales/zh-CN/chat.json @@ -104,6 +104,7 @@ "selectApiConfig": "选择 API 配置", "enhancePrompt": "增强提示词", "addImages": "添加图片到消息", + "addFiles": "添加文件到消息", "sendMessage": "发送消息", "stopTts": "停止文本转语音", "typeMessage": "输入消息...", diff --git a/webview-ui/src/i18n/locales/zh-TW/chat.json b/webview-ui/src/i18n/locales/zh-TW/chat.json index 75d214db03..128e6589ea 100644 --- a/webview-ui/src/i18n/locales/zh-TW/chat.json +++ b/webview-ui/src/i18n/locales/zh-TW/chat.json @@ -104,6 +104,7 @@ "selectApiConfig": "選取 API 設定", "enhancePrompt": "使用額外內容增強提示", "addImages": "新增圖片到訊息中", + "addFiles": "新增檔案到訊息", "sendMessage": "傳送訊息", "stopTts": "停止文字轉語音", "typeMessage": "輸入訊息...",