Skip to content
Open
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
114 changes: 78 additions & 36 deletions apps/web/src/components/ChatView.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,9 @@ import { serverConfigQueryOptions, serverQueryKeys } from "~/lib/serverReactQuer
import { isElectron } from "../env";
import { parseDiffRouteSearch, stripDiffSearchParams } from "../diffRouteSearch";
import {
clampCollapsedComposerCursor,
type ComposerTrigger,
collapseExpandedComposerCursor,
detectComposerTrigger,
expandCollapsedComposerCursor,
parseStandaloneComposerSlashCommand,
Expand Down Expand Up @@ -263,7 +265,9 @@ export default function ChatView({ threadId }: ChatViewProps) {
const [attachmentPreviewHandoffByMessageId, setAttachmentPreviewHandoffByMessageId] = useState<
Record<string, string[]>
>({});
const [composerCursor, setComposerCursor] = useState(() => prompt.length);
const [composerCursor, setComposerCursor] = useState(() =>
collapseExpandedComposerCursor(prompt, prompt.length),
);
const [composerTrigger, setComposerTrigger] = useState<ComposerTrigger | null>(() =>
detectComposerTrigger(prompt, prompt.length),
);
Expand Down Expand Up @@ -635,14 +639,15 @@ export default function ChatView({ threadId }: ChatViewProps) {
return;
}
promptRef.current = activePendingProgress.customAnswer;
setComposerCursor(activePendingProgress.customAnswer.length);
const nextCursor = collapseExpandedComposerCursor(
activePendingProgress.customAnswer,
activePendingProgress.customAnswer.length,
);
setComposerCursor(nextCursor);
setComposerTrigger(
detectComposerTrigger(
activePendingProgress.customAnswer,
expandCollapsedComposerCursor(
activePendingProgress.customAnswer,
activePendingProgress.customAnswer.length,
),
expandCollapsedComposerCursor(activePendingProgress.customAnswer, nextCursor),
),
);
setComposerHighlightedItemId(null);
Expand Down Expand Up @@ -1720,7 +1725,7 @@ export default function ChatView({ threadId }: ChatViewProps) {

useEffect(() => {
promptRef.current = prompt;
setComposerCursor((existing) => Math.min(Math.max(0, existing), prompt.length));
setComposerCursor((existing) => clampCollapsedComposerCursor(prompt, existing));
}, [prompt]);

useEffect(() => {
Expand All @@ -1733,7 +1738,7 @@ export default function ChatView({ threadId }: ChatViewProps) {
setSendPhase("idle");
setSendStartedAt(null);
setComposerHighlightedItemId(null);
setComposerCursor(promptRef.current.length);
setComposerCursor(collapseExpandedComposerCursor(promptRef.current, promptRef.current.length));
setComposerTrigger(detectComposerTrigger(promptRef.current, promptRef.current.length));
dragDepthRef.current = 0;
setIsDragOverComposer(false);
Expand Down Expand Up @@ -2418,7 +2423,7 @@ export default function ChatView({ threadId }: ChatViewProps) {
});
promptRef.current = trimmed;
setPrompt(trimmed);
setComposerCursor(trimmed.length);
setComposerCursor(collapseExpandedComposerCursor(trimmed, trimmed.length));
addComposerImagesToDraft(composerImagesSnapshot.map(cloneComposerImageForRetry));
setComposerTrigger(detectComposerTrigger(trimmed, trimmed.length));
}
Expand Down Expand Up @@ -2536,7 +2541,13 @@ export default function ChatView({ threadId }: ChatViewProps) {
);

const onChangeActivePendingUserInputCustomAnswer = useCallback(
(questionId: string, value: string, nextCursor: number, cursorAdjacentToMention: boolean) => {
(
questionId: string,
value: string,
nextCursor: number,
expandedCursor: number,
cursorAdjacentToMention: boolean,
) => {
if (!activePendingUserInput) {
return;
}
Expand All @@ -2553,9 +2564,7 @@ export default function ChatView({ threadId }: ChatViewProps) {
}));
setComposerCursor(nextCursor);
setComposerTrigger(
cursorAdjacentToMention
? null
: detectComposerTrigger(value, expandCollapsedComposerCursor(value, nextCursor)),
cursorAdjacentToMention ? null : detectComposerTrigger(value, expandedCursor),
);
},
[activePendingUserInput],
Expand Down Expand Up @@ -2891,6 +2900,7 @@ export default function ChatView({ threadId }: ChatViewProps) {
return false;
}
const next = replaceTextRange(promptRef.current, rangeStart, rangeEnd, replacement);
const nextCursor = collapseExpandedComposerCursor(next.text, next.cursor);
promptRef.current = next.text;
const activePendingQuestion = activePendingProgress?.activeQuestion;
if (activePendingQuestion && activePendingUserInput) {
Expand All @@ -2907,10 +2917,12 @@ export default function ChatView({ threadId }: ChatViewProps) {
} else {
setPrompt(next.text);
}
setComposerCursor(next.cursor);
setComposerTrigger(detectComposerTrigger(next.text, next.cursor));
setComposerCursor(nextCursor);
setComposerTrigger(
detectComposerTrigger(next.text, expandCollapsedComposerCursor(next.text, nextCursor)),
);
window.requestAnimationFrame(() => {
composerEditorRef.current?.focusAt(next.cursor);
composerEditorRef.current?.focusAt(nextCursor);
});
return true;
},
Expand All @@ -2920,23 +2932,38 @@ export default function ChatView({ threadId }: ChatViewProps) {
const readComposerSnapshot = useCallback((): {
value: string;
cursor: number;
expandedCursor: number;
} => {
const editorSnapshot = composerEditorRef.current?.readSnapshot();
if (editorSnapshot) {
return editorSnapshot;
}
return { value: promptRef.current, cursor: composerCursor };
return {
value: promptRef.current,
cursor: composerCursor,
expandedCursor: expandCollapsedComposerCursor(promptRef.current, composerCursor),
};
}, [composerCursor]);

const extendReplacementRangeForTrailingSpace = (
text: string,
rangeEnd: number,
replacement: string,
): number => {
if (!replacement.endsWith(" ")) {
return rangeEnd;
}
return text[rangeEnd] === " " ? rangeEnd + 1 : rangeEnd;
};

const resolveActiveComposerTrigger = useCallback((): {
snapshot: { value: string; cursor: number };
snapshot: { value: string; cursor: number; expandedCursor: number };
trigger: ComposerTrigger | null;
} => {
const snapshot = readComposerSnapshot();
const expandedCursor = expandCollapsedComposerCursor(snapshot.value, snapshot.cursor);
return {
snapshot,
trigger: detectComposerTrigger(snapshot.value, expandedCursor),
trigger: detectComposerTrigger(snapshot.value, snapshot.expandedCursor),
};
}, [readComposerSnapshot]);

Expand All @@ -2949,13 +2976,18 @@ export default function ChatView({ threadId }: ChatViewProps) {
});
const { snapshot, trigger } = resolveActiveComposerTrigger();
if (!trigger) return;
const expectedToken = snapshot.value.slice(trigger.rangeStart, trigger.rangeEnd);
if (item.type === "path") {
const replacement = `@${item.path} `;
const replacementRangeEnd = extendReplacementRangeForTrailingSpace(
snapshot.value,
trigger.rangeEnd,
replacement,
);
const applied = applyPromptReplacement(
trigger.rangeStart,
trigger.rangeEnd,
`@${item.path} `,
{ expectedText: expectedToken },
replacementRangeEnd,
replacement,
{ expectedText: snapshot.value.slice(trigger.rangeStart, replacementRangeEnd) },
);
if (applied) {
setComposerHighlightedItemId(null);
Expand All @@ -2964,17 +2996,26 @@ export default function ChatView({ threadId }: ChatViewProps) {
}
if (item.type === "slash-command") {
if (item.command === "model") {
const applied = applyPromptReplacement(trigger.rangeStart, trigger.rangeEnd, "/model ", {
expectedText: expectedToken,
});
const replacement = "/model ";
const replacementRangeEnd = extendReplacementRangeForTrailingSpace(
snapshot.value,
trigger.rangeEnd,
replacement,
);
const applied = applyPromptReplacement(
trigger.rangeStart,
replacementRangeEnd,
replacement,
{ expectedText: snapshot.value.slice(trigger.rangeStart, replacementRangeEnd) },
);
if (applied) {
setComposerHighlightedItemId(null);
}
return;
}
void handleInteractionModeChange(item.command === "plan" ? "plan" : "default");
const applied = applyPromptReplacement(trigger.rangeStart, trigger.rangeEnd, "", {
expectedText: expectedToken,
expectedText: snapshot.value.slice(trigger.rangeStart, trigger.rangeEnd),
});
if (applied) {
setComposerHighlightedItemId(null);
Expand All @@ -2983,7 +3024,7 @@ export default function ChatView({ threadId }: ChatViewProps) {
}
onProviderModelSelect(item.provider, item.model);
const applied = applyPromptReplacement(trigger.rangeStart, trigger.rangeEnd, "", {
expectedText: expectedToken,
expectedText: snapshot.value.slice(trigger.rangeStart, trigger.rangeEnd),
});
if (applied) {
setComposerHighlightedItemId(null);
Expand Down Expand Up @@ -3024,12 +3065,18 @@ export default function ChatView({ threadId }: ChatViewProps) {
workspaceEntriesQuery.isFetching);

const onPromptChange = useCallback(
(nextPrompt: string, nextCursor: number, cursorAdjacentToMention: boolean) => {
(
nextPrompt: string,
nextCursor: number,
expandedCursor: number,
cursorAdjacentToMention: boolean,
) => {
if (activePendingProgress?.activeQuestion && activePendingUserInput) {
onChangeActivePendingUserInputCustomAnswer(
activePendingProgress.activeQuestion.id,
nextPrompt,
nextCursor,
expandedCursor,
cursorAdjacentToMention,
);
return;
Expand All @@ -3038,12 +3085,7 @@ export default function ChatView({ threadId }: ChatViewProps) {
setPrompt(nextPrompt);
setComposerCursor(nextCursor);
setComposerTrigger(
cursorAdjacentToMention
? null
: detectComposerTrigger(
nextPrompt,
expandCollapsedComposerCursor(nextPrompt, nextCursor),
),
cursorAdjacentToMention ? null : detectComposerTrigger(nextPrompt, expandedCursor),
);
},
[
Expand Down
Loading
Loading