From 8329811ef68223b0e09484994e4f6aa7d90e4fdb Mon Sep 17 00:00:00 2001 From: tylerslaton Date: Mon, 21 Oct 2024 16:23:23 -0400 Subject: [PATCH 1/3] feat: overhaul agent edit UX This is a pretty substantial overhaul of the agent edit UX. The main changes are: - A new "Advanced" tab that allows you to edit the prompt - The name and description now look like contentEditable fields (they're not, but they act like them) - The edit agent page has moved the "go back" button to the header - New "Knowledge" suggestion for preview chats - Updated "restart chat" button to be "new thread" along with a companion button to select past threads - Accordion options now have color, descriptions, and larger text. - New Agents now have a name of "New Agent" instead of the fun random names they used to have. Signed-off-by: tylerslaton --- ui/admin/app/components/Typography.tsx | 109 ++++++++++++++---- .../app/components/agent/AdvancedForm.tsx | 65 +++++++++++ ui/admin/app/components/agent/Agent.tsx | 97 ++++++++++++---- ui/admin/app/components/agent/AgentForm.tsx | 29 ++--- ui/admin/app/components/agent/PastThreads.tsx | 107 +++++++++++++++++ ui/admin/app/components/agent/ToolForm.tsx | 46 ++++---- ui/admin/app/components/chat/NoMessages.tsx | 17 ++- .../app/components/form/controlledInputs.tsx | 2 +- ui/admin/app/components/header/HeaderNav.tsx | 21 +++- ui/admin/app/components/thread/ThreadMeta.tsx | 16 ++- ui/admin/app/lib/service/routeQueryParams.ts | 3 + ui/admin/app/routes/_auth.agents.$agent.tsx | 32 +---- ui/admin/app/routes/_auth.agents._index.tsx | 8 +- 13 files changed, 431 insertions(+), 121 deletions(-) create mode 100644 ui/admin/app/components/agent/AdvancedForm.tsx create mode 100644 ui/admin/app/components/agent/PastThreads.tsx diff --git a/ui/admin/app/components/Typography.tsx b/ui/admin/app/components/Typography.tsx index 34a9613af..dc94e683f 100644 --- a/ui/admin/app/components/Typography.tsx +++ b/ui/admin/app/components/Typography.tsx @@ -1,114 +1,183 @@ -import { ReactNode } from "react"; +import React, { ReactNode } from "react"; import { cn } from "~/lib/utils"; -interface TypographyProps { +type TypographyElement = keyof JSX.IntrinsicElements; + +type TypographyProps = { children: ReactNode; className?: string; -} +} & React.JSX.IntrinsicElements[T]; -export function TypographyH1({ children, className }: TypographyProps) { +export function TypographyH1({ + children, + className, + ...props +}: TypographyProps<"h1">) { return (

{children}

); } -export function TypographyH2({ children, className }: TypographyProps) { +export function TypographyH2({ + children, + className, + ...props +}: TypographyProps<"h2">) { return (

{children}

); } -export function TypographyH3({ children, className }: TypographyProps) { +export function TypographyH3({ + children, + className, + ...props +}: TypographyProps<"h3">) { return (

{children}

); } -export function TypographyH4({ children, className }: TypographyProps) { +export function TypographyH4({ + children, + className, + ...props +}: TypographyProps<"h4">) { return (

{children}

); } -export function TypographyP({ children, className }: TypographyProps) { - return

{children}

; +export function TypographyP({ + children, + className, + ...props +}: TypographyProps<"p">) { + return ( +

+ {children} +

+ ); } -export function TypographyBlockquote({ children, className }: TypographyProps) { +export function TypographyBlockquote({ + children, + className, + ...props +}: TypographyProps<"blockquote">) { return ( -
+
{children}
); } -export function TypographyInlineCode({ children, className }: TypographyProps) { +export function TypographyInlineCode({ + children, + className, + ...props +}: TypographyProps<"code">) { return ( {children} ); } -export function TypographyLead({ children, className }: TypographyProps) { +export function TypographyLead({ + children, + className, + ...props +}: TypographyProps<"p">) { return ( -

+

{children}

); } -export function TypographyLarge({ children, className }: TypographyProps) { +export function TypographyLarge({ + children, + className, + ...props +}: TypographyProps<"div">) { return ( -
{children}
+
+ {children} +
); } -export function TypographySmall({ children, className }: TypographyProps) { +export function TypographySmall({ + children, + className, + ...props +}: TypographyProps<"small">) { return ( - + {children} ); } -export function TypographyMuted({ children, className }: TypographyProps) { +export function TypographyMuted({ + children, + className, + ...props +}: TypographyProps<"p">) { return ( -

+

{children}

); diff --git a/ui/admin/app/components/agent/AdvancedForm.tsx b/ui/admin/app/components/agent/AdvancedForm.tsx new file mode 100644 index 000000000..e0f6e1a84 --- /dev/null +++ b/ui/admin/app/components/agent/AdvancedForm.tsx @@ -0,0 +1,65 @@ +import { zodResolver } from "@hookform/resolvers/zod"; +import { useEffect } from "react"; +import { useForm } from "react-hook-form"; +import { z } from "zod"; + +import { Agent } from "~/lib/model/agents"; + +import { ControlledTextarea } from "~/components/form/controlledInputs"; +import { Form } from "~/components/ui/form"; + +const formSchema = z.object({ + prompt: z.string().optional(), +}); + +export type AdvancedFormValues = z.infer; + +type AdvancedFormProps = { + agent: Agent; + onSubmit?: (values: AdvancedFormValues) => void; + onChange?: (values: AdvancedFormValues) => void; +}; + +export function AdvancedForm({ agent, onSubmit, onChange }: AdvancedFormProps) { + const form = useForm({ + resolver: zodResolver(formSchema), + mode: "onChange", + defaultValues: { + prompt: agent.prompt || "", + }, + }); + + useEffect(() => { + if (agent) form.reset({ prompt: agent.prompt || "" }); + }, [agent, form]); + + useEffect(() => { + return form.watch((values) => { + if (!onChange) return; + + const { data, success } = formSchema.safeParse(values); + + if (!success) return; + + onChange(data); + }).unsubscribe; + }, [onChange, form]); + + const handleSubmit = form.handleSubmit((values: AdvancedFormValues) => + onSubmit?.({ ...values }) + ); + + return ( +
+ + + + + ); +} diff --git a/ui/admin/app/components/agent/Agent.tsx b/ui/admin/app/components/agent/Agent.tsx index bffffb5b6..4345fb0bd 100644 --- a/ui/admin/app/components/agent/Agent.tsx +++ b/ui/admin/app/components/agent/Agent.tsx @@ -1,12 +1,15 @@ -import { LibraryIcon, RotateCcw, WrenchIcon } from "lucide-react"; +import { LibraryIcon, PlusIcon, SettingsIcon, WrenchIcon } from "lucide-react"; import { useCallback, useState } from "react"; import { Agent as AgentType } from "~/lib/model/agents"; import { cn } from "~/lib/utils"; import { TypographyP } from "~/components/Typography"; +import { AdvancedForm } from "~/components/agent/AdvancedForm"; import { AgentProvider, useAgent } from "~/components/agent/AgentContext"; import { AgentForm } from "~/components/agent/AgentForm"; +import { PastThreads } from "~/components/agent/PastThreads"; +import { ToolForm } from "~/components/agent/ToolForm"; import { AgentKnowledgePanel } from "~/components/knowledge"; import { Accordion, @@ -18,12 +21,10 @@ import { Button } from "~/components/ui/button"; import { ScrollArea } from "~/components/ui/scroll-area"; import { useDebounce } from "~/hooks/useDebounce"; -import { ToolForm } from "./ToolForm"; - type AgentProps = { agent: AgentType; className?: string; - onRefresh?: () => void; + onRefresh?: (threadId: string | null) => void; }; export function Agent(props: AgentProps) { @@ -52,6 +53,13 @@ function AgentContent({ className, onRefresh }: AgentProps) { const debouncedSetAgentInfo = useDebounce(partialSetAgent, 1000); + const handleThreadSelect = useCallback( + (threadId: string) => { + onRefresh?.(threadId); + }, + [onRefresh] + ); + return (
@@ -62,15 +70,27 @@ function AgentContent({ className, onRefresh }: AgentProps) { />
- - - - - + + + + + Tools +

+ Add tools the allow the agent to perform useful + actions such as searching the web, reading + files, or interacting with other systems. +

- - - - + + + + Knowledge - + +

+ Provide knowledge to the agent in the form of + files, website, or external links in order to + give it context about various topics. +

+ + + + + + Advanced + + + + + +
@@ -102,13 +148,22 @@ function AgentContent({ className, onRefresh }: AgentProps) {
)} - +
+ + +
); diff --git a/ui/admin/app/components/agent/AgentForm.tsx b/ui/admin/app/components/agent/AgentForm.tsx index 8f55c539e..8387a16c5 100644 --- a/ui/admin/app/components/agent/AgentForm.tsx +++ b/ui/admin/app/components/agent/AgentForm.tsx @@ -5,12 +5,10 @@ import { z } from "zod"; import { Agent } from "~/lib/model/agents"; -import { - ControlledInput, - ControlledTextarea, -} from "~/components/form/controlledInputs"; import { Form } from "~/components/ui/form"; +import { ControlledInput } from "../form/controlledInputs"; + const formSchema = z.object({ name: z.string().min(1, { message: "Name is required.", @@ -60,28 +58,19 @@ export function AgentForm({ agent, onSubmit, onChange }: AgentFormProps) { return (
- + - - - - diff --git a/ui/admin/app/components/agent/PastThreads.tsx b/ui/admin/app/components/agent/PastThreads.tsx new file mode 100644 index 000000000..46f634457 --- /dev/null +++ b/ui/admin/app/components/agent/PastThreads.tsx @@ -0,0 +1,107 @@ +import { ChevronUpIcon } from "lucide-react"; +import React, { useState } from "react"; +import useSWR from "swr"; + +import { Thread } from "~/lib/model/threads"; +import { ThreadsService } from "~/lib/service/api/threadsService"; + +import { LoadingSpinner } from "~/components/ui/LoadingSpinner"; +import { Button } from "~/components/ui/button"; +import { + Command, + CommandEmpty, + CommandGroup, + CommandInput, + CommandItem, + CommandList, +} from "~/components/ui/command"; +import { + Popover, + PopoverContent, + PopoverTrigger, +} from "~/components/ui/popover"; + +interface PastThreadsProps { + agentId: string; + onThreadSelect: (threadId: string) => void; +} + +export const PastThreads: React.FC = ({ + agentId, + onThreadSelect, +}) => { + const [open, setOpen] = useState(false); + const { + data: threads, + error, + isLoading, + mutate, + } = useSWR(ThreadsService.getThreadsByAgent.key(agentId), () => + ThreadsService.getThreadsByAgent(agentId) + ); + + const handleOpenChange = (newOpen: boolean) => { + setOpen(newOpen); + if (newOpen) { + mutate(); + } + }; + + const handleThreadSelect = (threadId: string) => { + onThreadSelect(threadId); + setOpen(false); + }; + + return ( + + + + + + + + + No threads found. + {isLoading ? ( +
+ +
+ ) : error ? ( +
+ Failed to load threads +
+ ) : threads && threads.length > 0 ? ( + + {threads.map((thread: Thread) => ( + + handleThreadSelect(thread.id) + } + className="cursor-pointer" + > +
+

+ Thread + + {thread.id} + +

+

+ {new Date( + thread.created + ).toLocaleString()} +

+
+
+ ))} +
+ ) : null} +
+
+
+
+ ); +}; diff --git a/ui/admin/app/components/agent/ToolForm.tsx b/ui/admin/app/components/agent/ToolForm.tsx index 7b2d54e7d..33f308a41 100644 --- a/ui/admin/app/components/agent/ToolForm.tsx +++ b/ui/admin/app/components/agent/ToolForm.tsx @@ -71,27 +71,6 @@ export function ToolForm({ return (
-
- - - - - - - - -
))} +
+ + + + + + + + +
)} diff --git a/ui/admin/app/components/chat/NoMessages.tsx b/ui/admin/app/components/chat/NoMessages.tsx index dba93548d..9d55cf245 100644 --- a/ui/admin/app/components/chat/NoMessages.tsx +++ b/ui/admin/app/components/chat/NoMessages.tsx @@ -1,3 +1,5 @@ +import { BrainCircuit, Handshake, Rocket } from "lucide-react"; + import { useChat } from "~/components/chat/ChatContext"; import { Button } from "~/components/ui/button"; @@ -19,19 +21,26 @@ export function NoMessages() { variant="secondary" onClick={() => handleAddMessage("Hello, how are you?")} > - 👋 Greeting + + Greeting diff --git a/ui/admin/app/components/form/controlledInputs.tsx b/ui/admin/app/components/form/controlledInputs.tsx index 8ac45be95..8fc5e128c 100644 --- a/ui/admin/app/components/form/controlledInputs.tsx +++ b/ui/admin/app/components/form/controlledInputs.tsx @@ -103,7 +103,7 @@ export function ControlledTextarea< {label && {label}} {description && ( - + {description} )} diff --git a/ui/admin/app/components/header/HeaderNav.tsx b/ui/admin/app/components/header/HeaderNav.tsx index b95674073..a46c60999 100644 --- a/ui/admin/app/components/header/HeaderNav.tsx +++ b/ui/admin/app/components/header/HeaderNav.tsx @@ -1,5 +1,5 @@ -import { useLocation, useParams } from "@remix-run/react"; -import { MenuIcon } from "lucide-react"; +import { Link, useLocation, useParams } from "@remix-run/react"; +import { ArrowLeftIcon, MenuIcon } from "lucide-react"; import { $params, $path } from "remix-routes"; import useSWR from "swr"; @@ -98,6 +98,10 @@ function getHeaderContent(route: string) { } const AgentEditContent = () => { + const { from } = + parseQueryParams(window.location.href, QueryParamSchemas.Agents).data || + {}; + const params = useParams(); const { agent: agentId } = $params("/agents/:agent", params); @@ -106,7 +110,18 @@ const AgentEditContent = () => { ({ agentId }) => AgentService.getAgentById(agentId) ); - return <>{agent?.name || "New Agent"}; + return ( +
+ {from && ( + + )} + {agent?.name || "New Agent"} +
+ ); }; const ThreadsContent = () => { diff --git a/ui/admin/app/components/thread/ThreadMeta.tsx b/ui/admin/app/components/thread/ThreadMeta.tsx index 6d055a690..a2b25f8f7 100644 --- a/ui/admin/app/components/thread/ThreadMeta.tsx +++ b/ui/admin/app/components/thread/ThreadMeta.tsx @@ -1,5 +1,6 @@ import { Link } from "@remix-run/react"; import { EditIcon, FileIcon, FilesIcon } from "lucide-react"; +import { $path } from "remix-routes"; import { Agent } from "~/lib/model/agents"; import { KnowledgeFile } from "~/lib/model/knowledge"; @@ -61,7 +62,20 @@ export function ThreadMeta({ asChild > diff --git a/ui/admin/app/lib/service/routeQueryParams.ts b/ui/admin/app/lib/service/routeQueryParams.ts index 7cda42732..5fd02c860 100644 --- a/ui/admin/app/lib/service/routeQueryParams.ts +++ b/ui/admin/app/lib/service/routeQueryParams.ts @@ -8,4 +8,7 @@ export const QueryParamSchemas = { agentId: z.string().optional(), workflowId: z.string().optional(), }), + Agents: z.object({ + from: z.string().optional(), + }), }; diff --git a/ui/admin/app/routes/_auth.agents.$agent.tsx b/ui/admin/app/routes/_auth.agents.$agent.tsx index 2701cba45..60fb41d13 100644 --- a/ui/admin/app/routes/_auth.agents.$agent.tsx +++ b/ui/admin/app/routes/_auth.agents.$agent.tsx @@ -1,7 +1,5 @@ -import { ArrowLeftIcon } from "@radix-ui/react-icons"; import { ClientLoaderFunctionArgs, - Link, redirect, useLoaderData, useNavigate, @@ -15,18 +13,11 @@ import { noop, parseQueryParams } from "~/lib/utils"; import { Agent } from "~/components/agent"; import { Chat, ChatProvider } from "~/components/chat"; -import { Button } from "~/components/ui/button"; import { ResizableHandle, ResizablePanel, ResizablePanelGroup, } from "~/components/ui/resizable"; -import { - Tooltip, - TooltipContent, - TooltipProvider, - TooltipTrigger, -} from "~/components/ui/tooltip"; const paramSchema = z.object({ threadId: z.string().optional(), @@ -57,7 +48,7 @@ export const clientLoader = async ({ }; export default function ChatAgent() { - const { agent, threadId, from } = useLoaderData(); + const { agent, threadId } = useLoaderData(); const navigate = useNavigate(); const updateThreadId = useCallback( @@ -85,26 +76,11 @@ export default function ChatAgent() { className="flex-auto" > - - - - Go Back - - updateThreadId(null)} + onRefresh={(threadId: string | null) => + updateThreadId(threadId) + } /> diff --git a/ui/admin/app/routes/_auth.agents._index.tsx b/ui/admin/app/routes/_auth.agents._index.tsx index 18edf61dd..f7fea7d87 100644 --- a/ui/admin/app/routes/_auth.agents._index.tsx +++ b/ui/admin/app/routes/_auth.agents._index.tsx @@ -5,11 +5,11 @@ import { SquarePen, Trash } from "lucide-react"; import { useMemo } from "react"; import { $path } from "remix-routes"; import useSWR, { preload } from "swr"; +import { z } from "zod"; import { Agent } from "~/lib/model/agents"; import { AgentService } from "~/lib/service/api/agentService"; import { ThreadsService } from "~/lib/service/api/threadsService"; -import { generateRandomName } from "~/lib/service/nameGenerator"; import { timeSince } from "~/lib/utils"; import { TypographyP } from "~/components/Typography"; @@ -23,6 +23,10 @@ import { } from "~/components/ui/tooltip"; import { useAsync } from "~/hooks/useAsync"; +export const agentEditParamSchema = z.object({ + from: z.string().optional(), +}); + export async function clientLoader() { await Promise.all([ preload(AgentService.getAgents.key(), AgentService.getAgents), @@ -74,7 +78,7 @@ export default function Threads() { onClick={() => { AgentService.createAgent({ agent: { - name: generateRandomName(), + name: "New Agent", } as Agent, }).then((agent) => { navigate( From ddeb7e22982d41ea8c199462a35b976240f8bc62 Mon Sep 17 00:00:00 2001 From: tylerslaton Date: Tue, 22 Oct 2024 14:07:27 -0400 Subject: [PATCH 2/3] fix: address feedback for #254 Signed-off-by: tylerslaton --- ui/admin/app/components/agent/Agent.tsx | 8 ++++---- ui/admin/app/components/agent/PastThreads.tsx | 10 ++++++---- ui/admin/app/components/agent/ToolForm.tsx | 2 +- ui/admin/app/components/thread/ThreadMeta.tsx | 14 +++----------- ui/admin/app/lib/service/routeQueryParams.ts | 1 + ui/admin/app/routes/_auth.agents.$agent.tsx | 17 +++-------------- 6 files changed, 18 insertions(+), 34 deletions(-) diff --git a/ui/admin/app/components/agent/Agent.tsx b/ui/admin/app/components/agent/Agent.tsx index 4345fb0bd..47f255d23 100644 --- a/ui/admin/app/components/agent/Agent.tsx +++ b/ui/admin/app/components/agent/Agent.tsx @@ -86,11 +86,11 @@ function AgentContent({ className, onRefresh }: AgentProps) { -

+ Add tools the allow the agent to perform useful actions such as searching the web, reading files, or interacting with other systems. -

+ -

+ Provide knowledge to the agent in the form of files, website, or external links in order to give it context about various topics. -

+
diff --git a/ui/admin/app/components/agent/PastThreads.tsx b/ui/admin/app/components/agent/PastThreads.tsx index 46f634457..83ac6772b 100644 --- a/ui/admin/app/components/agent/PastThreads.tsx +++ b/ui/admin/app/components/agent/PastThreads.tsx @@ -21,6 +21,8 @@ import { PopoverTrigger, } from "~/components/ui/popover"; +import { TypographyH4, TypographyP } from "../Typography"; + interface PastThreadsProps { agentId: string; onThreadSelect: (threadId: string) => void; @@ -83,17 +85,17 @@ export const PastThreads: React.FC = ({ className="cursor-pointer" >
-

+ Thread {thread.id} -

-

+ + {new Date( thread.created ).toLocaleString()} -

+
))} diff --git a/ui/admin/app/components/agent/ToolForm.tsx b/ui/admin/app/components/agent/ToolForm.tsx index 33f308a41..581ba3bb2 100644 --- a/ui/admin/app/components/agent/ToolForm.tsx +++ b/ui/admin/app/components/agent/ToolForm.tsx @@ -109,7 +109,7 @@ export function ToolForm({
- + - - - - - - No threads found. - {isLoading ? ( -
- -
- ) : error ? ( -
- Failed to load threads -
- ) : threads && threads.length > 0 ? ( - - {threads.map((thread: Thread) => ( - - handleThreadSelect(thread.id) - } - className="cursor-pointer" - > -
- - Thread - - {thread.id} - - - - {new Date( - thread.created - ).toLocaleString()} - -
-
- ))} -
- ) : null} -
-
-
-
+ + + + + + + + + Switch threads + + + + + No threads found. + {isLoading ? ( +
+ +
+ ) : error ? ( +
+ Failed to load threads +
+ ) : threads && threads.length > 0 ? ( + + {threads.map((thread: Thread) => ( + + handleThreadSelect( + thread.id + ) + } + className="cursor-pointer" + > +
+ + Thread + + {thread.id} + + + + {new Date( + thread.created + ).toLocaleString()} + +
+
+ ))} +
+ ) : null} +
+
+
+
+
+
); }; diff --git a/ui/admin/app/components/ui/input.tsx b/ui/admin/app/components/ui/input.tsx index 4db24b77c..2612d11ee 100644 --- a/ui/admin/app/components/ui/input.tsx +++ b/ui/admin/app/components/ui/input.tsx @@ -1,18 +1,33 @@ +import { type VariantProps, cva } from "class-variance-authority"; import * as React from "react"; import { cn } from "~/lib/utils"; -export type InputProps = React.InputHTMLAttributes; +const inputVariants = cva( + "flex h-9 w-full rounded-md px-3 bg-transparent border border-input text-sm transition-colors file:border-0 file:bg-transparent file:text-sm file:font-medium placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:cursor-not-allowed disabled:opacity-50", + { + variants: { + variant: { + default: "", + ghost: "shadow-none cursor-pointer hover:border-primary px-0 mb-0 font-bold outline-none border-transparent focus:border-primary", + }, + }, + defaultVariants: { + variant: "default", + }, + } +); + +export interface InputProps + extends React.InputHTMLAttributes, + VariantProps {} const Input = React.forwardRef( - ({ className, type, ...props }, ref) => { + ({ className, variant, type, ...props }, ref) => { return ( @@ -21,4 +36,4 @@ const Input = React.forwardRef( ); Input.displayName = "Input"; -export { Input }; +export { Input, inputVariants }; diff --git a/ui/admin/app/routes/_auth.agents.$agent.tsx b/ui/admin/app/routes/_auth.agents.$agent.tsx index 9dcb5ca4c..63d959f41 100644 --- a/ui/admin/app/routes/_auth.agents.$agent.tsx +++ b/ui/admin/app/routes/_auth.agents.$agent.tsx @@ -6,7 +6,6 @@ import { } from "@remix-run/react"; import { useCallback } from "react"; import { $params, $path } from "remix-routes"; -import { z } from "zod"; import { AgentService } from "~/lib/service/api/agentService"; import { QueryParamSchemas } from "~/lib/service/routeQueryParams"; diff --git a/ui/admin/app/routes/_auth.agents._index.tsx b/ui/admin/app/routes/_auth.agents._index.tsx index f7fea7d87..590b78519 100644 --- a/ui/admin/app/routes/_auth.agents._index.tsx +++ b/ui/admin/app/routes/_auth.agents._index.tsx @@ -5,7 +5,6 @@ import { SquarePen, Trash } from "lucide-react"; import { useMemo } from "react"; import { $path } from "remix-routes"; import useSWR, { preload } from "swr"; -import { z } from "zod"; import { Agent } from "~/lib/model/agents"; import { AgentService } from "~/lib/service/api/agentService"; @@ -23,10 +22,6 @@ import { } from "~/components/ui/tooltip"; import { useAsync } from "~/hooks/useAsync"; -export const agentEditParamSchema = z.object({ - from: z.string().optional(), -}); - export async function clientLoader() { await Promise.all([ preload(AgentService.getAgents.key(), AgentService.getAgents),