Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
34 changes: 34 additions & 0 deletions src/browser/components/ChatInput/draftImagesStorage.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
import { describe, expect, test } from "bun:test";

import {
estimatePersistedImageAttachmentsChars,
parsePersistedImageAttachments,
} from "./draftImagesStorage";

describe("draftImagesStorage", () => {
test("parsePersistedImageAttachments returns [] for non-arrays", () => {
expect(parsePersistedImageAttachments(null)).toEqual([]);
expect(parsePersistedImageAttachments({})).toEqual([]);
expect(parsePersistedImageAttachments("nope")).toEqual([]);
});

test("parsePersistedImageAttachments returns [] for invalid array items", () => {
expect(parsePersistedImageAttachments([{}])).toEqual([]);
expect(
parsePersistedImageAttachments([{ id: "img", url: 123, mediaType: "image/png" }])
).toEqual([]);
});

test("parsePersistedImageAttachments returns attachments for valid items", () => {
expect(
parsePersistedImageAttachments([
{ id: "img-1", url: "", mediaType: "image/png" },
])
).toEqual([{ id: "img-1", url: "", mediaType: "image/png" }]);
});

test("estimatePersistedImageAttachmentsChars matches JSON length", () => {
const images = [{ id: "img-1", url: "", mediaType: "image/png" }];
expect(estimatePersistedImageAttachmentsChars(images)).toBe(JSON.stringify(images).length);
});
});
39 changes: 39 additions & 0 deletions src/browser/components/ChatInput/draftImagesStorage.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
import type { ImageAttachment } from "@/browser/components/ImageAttachments";
import { readPersistedState } from "@/browser/hooks/usePersistedState";

function isRecord(value: unknown): value is Record<string, unknown> {
return typeof value === "object" && value !== null;
}

function isImageAttachment(value: unknown): value is ImageAttachment {
if (!isRecord(value)) return false;
return (
typeof value.id === "string" &&
typeof value.url === "string" &&
typeof value.mediaType === "string"
);
}

export function parsePersistedImageAttachments(raw: unknown): ImageAttachment[] {
if (!Array.isArray(raw)) {
return [];
}

const attachments: ImageAttachment[] = [];
for (const item of raw) {
if (!isImageAttachment(item)) {
return [];
}
attachments.push({ id: item.id, url: item.url, mediaType: item.mediaType });
}

return attachments;
}

export function readPersistedImageAttachments(imagesKey: string): ImageAttachment[] {
return parsePersistedImageAttachments(readPersistedState<unknown>(imagesKey, []));
}

export function estimatePersistedImageAttachmentsChars(images: ImageAttachment[]): number {
return JSON.stringify(images).length;
}
143 changes: 108 additions & 35 deletions src/browser/components/ChatInput/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ import { useSendMessageOptions } from "@/browser/hooks/useSendMessageOptions";
import {
getModelKey,
getInputKey,
getInputImagesKey,
VIM_ENABLED_KEY,
getProjectScopeId,
getPendingScopeId,
Expand Down Expand Up @@ -78,9 +79,16 @@ import { useCreationWorkspace } from "./useCreationWorkspace";
import { useTutorial } from "@/browser/contexts/TutorialContext";
import { useVoiceInput } from "@/browser/hooks/useVoiceInput";
import { VoiceInputButton } from "./VoiceInputButton";
import {
estimatePersistedImageAttachmentsChars,
readPersistedImageAttachments,
} from "./draftImagesStorage";
import { RecordingOverlay } from "./RecordingOverlay";
import { ReviewBlockFromData } from "../shared/ReviewBlock";

// localStorage quotas are environment-dependent and relatively small.
// Be conservative here so we can warn the user before writes start failing.
const MAX_PERSISTED_IMAGE_DRAFT_CHARS = 4_000_000;
type TokenCountReader = () => number;

function createTokenCountResource(promise: Promise<number>): TokenCountReader {
Expand Down Expand Up @@ -132,13 +140,16 @@ const ChatInputInner: React.FC<ChatInputProps> = (props) => {
// Storage keys differ by variant
const storageKeys = (() => {
if (variant === "creation") {
const pendingScopeId = getPendingScopeId(props.projectPath);
return {
inputKey: getInputKey(getPendingScopeId(props.projectPath)),
inputKey: getInputKey(pendingScopeId),
imagesKey: getInputImagesKey(pendingScopeId),
modelKey: getModelKey(getProjectScopeId(props.projectPath)),
};
}
return {
inputKey: getInputKey(props.workspaceId),
imagesKey: getInputImagesKey(props.workspaceId),
modelKey: getModelKey(props.workspaceId),
};
})();
Expand All @@ -150,10 +161,6 @@ const ChatInputInner: React.FC<ChatInputProps> = (props) => {
const [commandSuggestions, setCommandSuggestions] = useState<SlashSuggestion[]>([]);
const [providerNames, setProviderNames] = useState<string[]>([]);
const [toast, setToast] = useState<Toast | null>(null);
const [imageAttachments, setImageAttachments] = useState<ImageAttachment[]>([]);
// Attached reviews come from parent via props (persisted in pendingReviews state)
const attachedReviews = variant === "workspace" ? (props.attachedReviews ?? []) : [];

const pushToast = useCallback(
(nextToast: Omit<Toast, "id">) => {
setToast({ id: Date.now().toString(), ...nextToast });
Expand All @@ -163,6 +170,60 @@ const ChatInputInner: React.FC<ChatInputProps> = (props) => {
const handleToastDismiss = useCallback(() => {
setToast(null);
}, []);

const imageDraftTooLargeToastKeyRef = useRef<string | null>(null);

const [imageAttachments, setImageAttachmentsState] = useState<ImageAttachment[]>(() => {
return readPersistedImageAttachments(storageKeys.imagesKey);
});
const persistImageAttachments = useCallback(
(nextImages: ImageAttachment[]) => {
if (nextImages.length === 0) {
imageDraftTooLargeToastKeyRef.current = null;
updatePersistedState<ImageAttachment[] | undefined>(storageKeys.imagesKey, undefined);
return;
}

const estimatedChars = estimatePersistedImageAttachmentsChars(nextImages);
if (estimatedChars > MAX_PERSISTED_IMAGE_DRAFT_CHARS) {
// Clear persisted value to avoid restoring stale images on restart.
updatePersistedState<ImageAttachment[] | undefined>(storageKeys.imagesKey, undefined);

if (imageDraftTooLargeToastKeyRef.current !== storageKeys.imagesKey) {
imageDraftTooLargeToastKeyRef.current = storageKeys.imagesKey;
pushToast({
type: "error",
message:
"This draft image is too large to save. It will be lost when you switch workspaces or restart.",
duration: 5000,
});
}
return;
}

imageDraftTooLargeToastKeyRef.current = null;
updatePersistedState<ImageAttachment[] | undefined>(storageKeys.imagesKey, nextImages);
},
[storageKeys.imagesKey, pushToast]
);

// Keep image drafts in sync when the storage scope changes (e.g. switching creation projects).
useEffect(() => {
imageDraftTooLargeToastKeyRef.current = null;
setImageAttachmentsState(readPersistedImageAttachments(storageKeys.imagesKey));
}, [storageKeys.imagesKey]);
const setImageAttachments = useCallback(
(value: ImageAttachment[] | ((prev: ImageAttachment[]) => ImageAttachment[])) => {
setImageAttachmentsState((prev) => {
const next = value instanceof Function ? value(prev) : value;
persistImageAttachments(next);
return next;
});
},
[persistImageAttachments]
);
// Attached reviews come from parent via props (persisted in pendingReviews state)
const attachedReviews = variant === "workspace" ? (props.attachedReviews ?? []) : [];
const inputRef = useRef<HTMLTextAreaElement>(null);
const modelSelectorRef = useRef<ModelSelectorRef>(null);

Expand All @@ -181,7 +242,7 @@ const ChatInputInner: React.FC<ChatInputProps> = (props) => {
setInput(draft.text);
setImageAttachments(draft.images);
},
[setInput]
[setInput, setImageAttachments]
);
const preEditDraftRef = useRef<DraftState>({ text: "", images: [] });
const { open } = useSettings();
Expand Down Expand Up @@ -387,14 +448,17 @@ const ChatInputInner: React.FC<ChatInputProps> = (props) => {
);

// Method to restore images to input (used by queued message edit)
const restoreImages = useCallback((images: ImagePart[]) => {
const attachments: ImageAttachment[] = images.map((img, index) => ({
id: `restored-${Date.now()}-${index}`,
url: img.url,
mediaType: img.mediaType,
}));
setImageAttachments(attachments);
}, []);
const restoreImages = useCallback(
(images: ImagePart[]) => {
const attachments: ImageAttachment[] = images.map((img, index) => ({
id: `restored-${Date.now()}-${index}`,
url: img.url,
mediaType: img.mediaType,
}));
setImageAttachments(attachments);
},
[setImageAttachments]
);

// Provide API to parent via callback
useEffect(() => {
Expand Down Expand Up @@ -678,24 +742,30 @@ const ChatInputInner: React.FC<ChatInputProps> = (props) => {
}, [variant, workspaceIdForFocus, focusMessageInput, setChatInputAutoFocusState]);

// Handle paste events to extract images
const handlePaste = useCallback((e: React.ClipboardEvent<HTMLTextAreaElement>) => {
const items = e.clipboardData?.items;
if (!items) return;
const handlePaste = useCallback(
(e: React.ClipboardEvent<HTMLTextAreaElement>) => {
const items = e.clipboardData?.items;
if (!items) return;

const imageFiles = extractImagesFromClipboard(items);
if (imageFiles.length === 0) return;
const imageFiles = extractImagesFromClipboard(items);
if (imageFiles.length === 0) return;

e.preventDefault(); // Prevent default paste behavior for images
e.preventDefault(); // Prevent default paste behavior for images

void processImageFiles(imageFiles).then((attachments) => {
setImageAttachments((prev) => [...prev, ...attachments]);
});
}, []);
void processImageFiles(imageFiles).then((attachments) => {
setImageAttachments((prev) => [...prev, ...attachments]);
});
},
[setImageAttachments]
);

// Handle removing an image attachment
const handleRemoveImage = useCallback((id: string) => {
setImageAttachments((prev) => prev.filter((img) => img.id !== id));
}, []);
const handleRemoveImage = useCallback(
(id: string) => {
setImageAttachments((prev) => prev.filter((img) => img.id !== id));
},
[setImageAttachments]
);

// Handle drag over to allow drop
const handleDragOver = useCallback((e: React.DragEvent<HTMLTextAreaElement>) => {
Expand All @@ -707,16 +777,19 @@ const ChatInputInner: React.FC<ChatInputProps> = (props) => {
}, []);

// Handle drop to extract images
const handleDrop = useCallback((e: React.DragEvent<HTMLTextAreaElement>) => {
e.preventDefault();
const handleDrop = useCallback(
(e: React.DragEvent<HTMLTextAreaElement>) => {
e.preventDefault();

const imageFiles = extractImagesFromDrop(e.dataTransfer);
if (imageFiles.length === 0) return;
const imageFiles = extractImagesFromDrop(e.dataTransfer);
if (imageFiles.length === 0) return;

void processImageFiles(imageFiles).then((attachments) => {
setImageAttachments((prev) => [...prev, ...attachments]);
});
}, []);
void processImageFiles(imageFiles).then((attachments) => {
setImageAttachments((prev) => [...prev, ...attachments]);
});
},
[setImageAttachments]
);

// Handle command selection
const handleCommandSelect = useCallback(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ import type { APIClient } from "@/browser/contexts/API";
import type { DraftWorkspaceSettings } from "@/browser/hooks/useDraftWorkspaceSettings";
import {
getInputKey,
getInputImagesKey,
getModelKey,
getModeKey,
getPendingScopeId,
Expand Down Expand Up @@ -461,10 +462,13 @@ describe("useCreationWorkspace", () => {
expect(readPersistedStateCalls).toContainEqual([projectModeKey, null]);

const modeKey = getModeKey(TEST_WORKSPACE_ID);
const pendingInputKey = getInputKey(getPendingScopeId(TEST_PROJECT_PATH));
const pendingScopeId = getPendingScopeId(TEST_PROJECT_PATH);
const pendingInputKey = getInputKey(pendingScopeId);
const pendingImagesKey = getInputImagesKey(pendingScopeId);
expect(updatePersistedStateCalls).toContainEqual([modeKey, "plan"]);
// Note: thinking level is no longer synced per-workspace, it's stored per-model globally
expect(updatePersistedStateCalls).toContainEqual([pendingInputKey, ""]);
expect(updatePersistedStateCalls).toContainEqual([pendingImagesKey, undefined]);
});

test("handleSend surfaces backend errors and resets state", async () => {
Expand Down
6 changes: 4 additions & 2 deletions src/browser/components/ChatInput/useCreationWorkspace.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import { readPersistedState, updatePersistedState } from "@/browser/hooks/usePer
import { getSendOptionsFromStorage } from "@/browser/utils/messages/sendOptions";
import {
getInputKey,
getInputImagesKey,
getModelKey,
getModeKey,
getPendingScopeId,
Expand Down Expand Up @@ -198,8 +199,9 @@ export function useCreationWorkspace({
// Sync preferences immediately (before switching)
syncCreationPreferences(projectPath, metadata.id);
if (projectPath) {
const pendingInputKey = getInputKey(getPendingScopeId(projectPath));
updatePersistedState(pendingInputKey, "");
const pendingScopeId = getPendingScopeId(projectPath);
updatePersistedState(getInputKey(pendingScopeId), "");
updatePersistedState(getInputImagesKey(pendingScopeId), undefined);
}

// Switch to the workspace IMMEDIATELY after creation to exit splash faster.
Expand Down
50 changes: 50 additions & 0 deletions src/browser/components/ChatInputToast.test.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
import React from "react";
import { afterEach, beforeEach, describe, expect, test } from "bun:test";
import { GlobalWindow } from "happy-dom";
import { cleanup, fireEvent, render, waitFor } from "@testing-library/react";

import { ChatInputToast, type Toast } from "./ChatInputToast";

describe("ChatInputToast", () => {
beforeEach(() => {
globalThis.window = new GlobalWindow() as unknown as Window & typeof globalThis;
globalThis.document = globalThis.window.document;
});

afterEach(() => {
cleanup();
globalThis.window = undefined as unknown as Window & typeof globalThis;
globalThis.document = undefined as unknown as Document;
});

test("resets leaving state when a new toast is shown", async () => {
const toast1: Toast = { id: "toast-1", type: "error", message: "first" };
const toast2: Toast = { id: "toast-2", type: "error", message: "second" };

function Harness() {
const [toast, setToast] = React.useState<Toast | null>(toast1);
return (
<div>
<ChatInputToast toast={toast} onDismiss={() => undefined} />
<button onClick={() => setToast(toast2)}>Next toast</button>
</div>
);
}

const { getByLabelText, getByRole, getByText } = render(<Harness />);

fireEvent.click(getByLabelText("Dismiss"));

await waitFor(() => {
expect(getByRole("alert").className).toContain("toastFadeOut");
});

fireEvent.click(getByText("Next toast"));

await waitFor(() => {
const className = getByRole("alert").className;
expect(className).toContain("toastSlideIn");
expect(className).not.toContain("toastFadeOut");
});
});
});
Loading