Undo button (#76)
This commit is contained in:
@@ -1,17 +1,19 @@
|
|||||||
import type React from "react";
|
import type React from "react";
|
||||||
import type { Message } from "@/ipc/ipc_types";
|
import type { Message } from "@/ipc/ipc_types";
|
||||||
import { forwardRef } from "react";
|
import { forwardRef, useState } from "react";
|
||||||
import ChatMessage from "./ChatMessage";
|
import ChatMessage from "./ChatMessage";
|
||||||
import { SetupBanner } from "../SetupBanner";
|
import { SetupBanner } from "../SetupBanner";
|
||||||
import { useSettings } from "@/hooks/useSettings";
|
import { useSettings } from "@/hooks/useSettings";
|
||||||
import { useStreamChat } from "@/hooks/useStreamChat";
|
import { useStreamChat } from "@/hooks/useStreamChat";
|
||||||
import { selectedChatIdAtom } from "@/atoms/chatAtoms";
|
import { selectedChatIdAtom } from "@/atoms/chatAtoms";
|
||||||
import { useAtom, useAtomValue } from "jotai";
|
import { useAtom, useAtomValue, useSetAtom } from "jotai";
|
||||||
import { RefreshCw } from "lucide-react";
|
import { Loader2, RefreshCw, Undo } from "lucide-react";
|
||||||
import { Button } from "@/components/ui/button";
|
import { Button } from "@/components/ui/button";
|
||||||
import { useVersions } from "@/hooks/useVersions";
|
import { useVersions } from "@/hooks/useVersions";
|
||||||
import { selectedAppIdAtom } from "@/atoms/appAtoms";
|
import { selectedAppIdAtom } from "@/atoms/appAtoms";
|
||||||
import { showError } from "@/lib/toast";
|
import { showError } from "@/lib/toast";
|
||||||
|
import { IpcClient } from "@/ipc/ipc_client";
|
||||||
|
import { chatMessagesAtom } from "@/atoms/chatAtoms";
|
||||||
|
|
||||||
interface MessagesListProps {
|
interface MessagesListProps {
|
||||||
messages: Message[];
|
messages: Message[];
|
||||||
@@ -25,6 +27,10 @@ export const MessagesList = forwardRef<HTMLDivElement, MessagesListProps>(
|
|||||||
const { streamMessage, isStreaming, error, setError } = useStreamChat();
|
const { streamMessage, isStreaming, error, setError } = useStreamChat();
|
||||||
const { isAnyProviderSetup } = useSettings();
|
const { isAnyProviderSetup } = useSettings();
|
||||||
const selectedChatId = useAtomValue(selectedChatIdAtom);
|
const selectedChatId = useAtomValue(selectedChatIdAtom);
|
||||||
|
const setMessages = useSetAtom(chatMessagesAtom);
|
||||||
|
const [isUndoLoading, setIsUndoLoading] = useState(false);
|
||||||
|
const [isRetryLoading, setIsRetryLoading] = useState(false);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex-1 overflow-y-auto p-4" ref={ref}>
|
<div className="flex-1 overflow-y-auto p-4" ref={ref}>
|
||||||
{messages.length > 0 ? (
|
{messages.length > 0 ? (
|
||||||
@@ -40,67 +46,136 @@ export const MessagesList = forwardRef<HTMLDivElement, MessagesListProps>(
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
{messages.length > 0 && !isStreaming && (
|
{messages.length > 0 && !isStreaming && (
|
||||||
<div className="flex max-w-3xl mx-auto">
|
<div className="flex max-w-3xl mx-auto gap-2">
|
||||||
|
{messages[messages.length - 1].role === "assistant" && (
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
size="sm"
|
||||||
|
disabled={isUndoLoading}
|
||||||
|
onClick={async () => {
|
||||||
|
if (!selectedChatId || !appId) {
|
||||||
|
console.error("No chat selected or app ID not available");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (versions.length < 2) {
|
||||||
|
showError("Cannot undo; no previous version");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setIsUndoLoading(true);
|
||||||
|
try {
|
||||||
|
const previousAssistantMessage =
|
||||||
|
messages[messages.length - 3];
|
||||||
|
if (
|
||||||
|
previousAssistantMessage?.role === "assistant" &&
|
||||||
|
previousAssistantMessage?.commitHash
|
||||||
|
) {
|
||||||
|
console.debug("Reverting to previous assistant version");
|
||||||
|
await revertVersion({
|
||||||
|
versionId: previousAssistantMessage.commitHash,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
// Revert to the previous version
|
||||||
|
await revertVersion({
|
||||||
|
versionId: versions[1].oid,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (selectedChatId) {
|
||||||
|
const chat = await IpcClient.getInstance().getChat(
|
||||||
|
selectedChatId
|
||||||
|
);
|
||||||
|
setMessages(chat.messages);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error during undo operation:", error);
|
||||||
|
showError("Failed to undo changes");
|
||||||
|
} finally {
|
||||||
|
setIsUndoLoading(false);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{isUndoLoading ? (
|
||||||
|
<Loader2 size={16} className="mr-1 animate-spin" />
|
||||||
|
) : (
|
||||||
|
<Undo size={16} />
|
||||||
|
)}
|
||||||
|
Undo
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
<Button
|
<Button
|
||||||
variant="ghost"
|
variant="outline"
|
||||||
size="sm"
|
size="sm"
|
||||||
|
disabled={isRetryLoading}
|
||||||
onClick={async () => {
|
onClick={async () => {
|
||||||
if (!selectedChatId) {
|
if (!selectedChatId) {
|
||||||
console.error("No chat selected");
|
console.error("No chat selected");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
// The last message is usually an assistant, but it might not be.
|
|
||||||
const lastVersion = versions[0];
|
setIsRetryLoading(true);
|
||||||
const lastMessage = messages[messages.length - 1];
|
try {
|
||||||
let reverted = false;
|
// The last message is usually an assistant, but it might not be.
|
||||||
if (
|
const lastVersion = versions[0];
|
||||||
lastVersion.oid === lastMessage.commitHash &&
|
const lastMessage = messages[messages.length - 1];
|
||||||
lastMessage.role === "assistant"
|
let reverted = false;
|
||||||
) {
|
if (
|
||||||
if (versions.length < 2) {
|
lastVersion.oid === lastMessage.commitHash &&
|
||||||
showError("Cannot retry message; no previous version");
|
lastMessage.role === "assistant"
|
||||||
|
) {
|
||||||
|
if (versions.length < 2) {
|
||||||
|
showError("Cannot retry message; no previous version");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const previousAssistantMessage =
|
||||||
|
messages[messages.length - 3];
|
||||||
|
if (
|
||||||
|
previousAssistantMessage?.role === "assistant" &&
|
||||||
|
previousAssistantMessage?.commitHash
|
||||||
|
) {
|
||||||
|
console.debug("Reverting to previous assistant version");
|
||||||
|
await revertVersion({
|
||||||
|
versionId: previousAssistantMessage.commitHash,
|
||||||
|
});
|
||||||
|
reverted = true;
|
||||||
|
} else {
|
||||||
|
console.debug("Reverting to previous version");
|
||||||
|
await revertVersion({
|
||||||
|
versionId: versions[1].oid,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Find the last user message
|
||||||
|
const lastUserMessage = [...messages]
|
||||||
|
.reverse()
|
||||||
|
.find((message) => message.role === "user");
|
||||||
|
if (!lastUserMessage) {
|
||||||
|
console.error("No user message found");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const previousAssistantMessage =
|
// If we reverted, we don't need to mark "redo" because
|
||||||
messages[messages.length - 3];
|
// the old message has already been deleted.
|
||||||
if (
|
const redo = !reverted;
|
||||||
previousAssistantMessage?.role === "assistant" &&
|
console.debug("Streaming message with redo", redo);
|
||||||
previousAssistantMessage?.commitHash
|
|
||||||
) {
|
|
||||||
console.debug("Reverting to previous assistant version");
|
|
||||||
await revertVersion({
|
|
||||||
versionId: previousAssistantMessage.commitHash,
|
|
||||||
});
|
|
||||||
reverted = true;
|
|
||||||
} else {
|
|
||||||
console.debug("Reverting to previous version");
|
|
||||||
await revertVersion({
|
|
||||||
versionId: versions[1].oid,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Find the last user message
|
streamMessage({
|
||||||
const lastUserMessage = [...messages]
|
prompt: lastUserMessage.content,
|
||||||
.reverse()
|
chatId: selectedChatId,
|
||||||
.find((message) => message.role === "user");
|
redo,
|
||||||
if (!lastUserMessage) {
|
});
|
||||||
console.error("No user message found");
|
} catch (error) {
|
||||||
return;
|
console.error("Error during retry operation:", error);
|
||||||
|
showError("Failed to retry message");
|
||||||
|
} finally {
|
||||||
|
setIsRetryLoading(false);
|
||||||
}
|
}
|
||||||
// If we reverted, we don't need to mark "redo" because
|
|
||||||
// the old message has already been deleted.
|
|
||||||
const redo = !reverted;
|
|
||||||
console.debug("Streaming message with redo", redo);
|
|
||||||
|
|
||||||
streamMessage({
|
|
||||||
prompt: lastUserMessage.content,
|
|
||||||
chatId: selectedChatId,
|
|
||||||
redo,
|
|
||||||
});
|
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<RefreshCw size={16} />
|
{isRetryLoading ? (
|
||||||
|
<Loader2 size={16} className="mr-1 animate-spin" />
|
||||||
|
) : (
|
||||||
|
<RefreshCw size={16} />
|
||||||
|
)}
|
||||||
Retry
|
Retry
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -494,7 +494,6 @@ export function registerAppHandlers() {
|
|||||||
_,
|
_,
|
||||||
{ appId, previousVersionId }: { appId: number; previousVersionId: string }
|
{ appId, previousVersionId }: { appId: number; previousVersionId: string }
|
||||||
) => {
|
) => {
|
||||||
logger.log(`Reverting to version ${previousVersionId} for app ${appId}`);
|
|
||||||
return withLock(appId, async () => {
|
return withLock(appId, async () => {
|
||||||
const app = await db.query.apps.findFirst({
|
const app = await db.query.apps.findFirst({
|
||||||
where: eq(apps.id, appId),
|
where: eq(apps.id, appId),
|
||||||
|
|||||||
@@ -787,25 +787,31 @@ export class IpcClient {
|
|||||||
|
|
||||||
public async listLocalOllamaModels(): Promise<LocalModel[]> {
|
public async listLocalOllamaModels(): Promise<LocalModel[]> {
|
||||||
try {
|
try {
|
||||||
const response = await this.ipcRenderer.invoke("local-models:list-ollama");
|
const response = await this.ipcRenderer.invoke(
|
||||||
|
"local-models:list-ollama"
|
||||||
|
);
|
||||||
return response?.models || [];
|
return response?.models || [];
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
if (error instanceof Error) {
|
if (error instanceof Error) {
|
||||||
throw new Error(`Failed to fetch Ollama models: ${error.message}`);
|
throw new Error(`Failed to fetch Ollama models: ${error.message}`);
|
||||||
}
|
}
|
||||||
throw new Error('Failed to fetch Ollama models: Unknown error occurred');
|
throw new Error("Failed to fetch Ollama models: Unknown error occurred");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public async listLocalLMStudioModels(): Promise<LocalModel[]> {
|
public async listLocalLMStudioModels(): Promise<LocalModel[]> {
|
||||||
try {
|
try {
|
||||||
const response = await this.ipcRenderer.invoke("local-models:list-lmstudio");
|
const response = await this.ipcRenderer.invoke(
|
||||||
|
"local-models:list-lmstudio"
|
||||||
|
);
|
||||||
return response?.models || [];
|
return response?.models || [];
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
if (error instanceof Error) {
|
if (error instanceof Error) {
|
||||||
throw new Error(`Failed to fetch LM Studio models: ${error.message}`);
|
throw new Error(`Failed to fetch LM Studio models: ${error.message}`);
|
||||||
}
|
}
|
||||||
throw new Error('Failed to fetch LM Studio models: Unknown error occurred');
|
throw new Error(
|
||||||
|
"Failed to fetch LM Studio models: Unknown error occurred"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user