From 7561deb7669686c9c477464c1cc462b642863fdc Mon Sep 17 00:00:00 2001 From: Ting Chien Meng Date: Wed, 19 Mar 2025 18:41:43 +0800 Subject: [PATCH] improve group panel --- packages/client/src/components/combobox.tsx | 129 ++++++++++++++++++ .../client/src/components/group-panel.tsx | 73 +++------- .../src/components/ui/switch-button.tsx | 30 ---- 3 files changed, 147 insertions(+), 85 deletions(-) create mode 100644 packages/client/src/components/combobox.tsx delete mode 100644 packages/client/src/components/ui/switch-button.tsx diff --git a/packages/client/src/components/combobox.tsx b/packages/client/src/components/combobox.tsx new file mode 100644 index 00000000000..9aea62505b3 --- /dev/null +++ b/packages/client/src/components/combobox.tsx @@ -0,0 +1,129 @@ +import { useState, useRef, useEffect } from 'react'; +import { Input } from '@/components/ui/input'; +import { Card } from '@/components/ui/card'; +import { Badge } from '@/components/ui/badge'; +import { ChevronDown, X } from 'lucide-react'; +import { formatAgentName } from '@/lib/utils'; + +interface Option { + icon: string; + label: string; +} + +interface MultiSelectComboboxProps { + options: Option[]; + className?: string; + onSelect?: (selected: Option[]) => void; +} + +export default function MultiSelectCombobox({ + options = [], + className = '', + onSelect, +}: MultiSelectComboboxProps) { + const [selected, setSelected] = useState([]); + const [isOpen, setIsOpen] = useState(false); + const comboboxRef = useRef(null); + + useEffect(() => { + const handleClickOutside = (event: MouseEvent) => { + if (comboboxRef.current && !comboboxRef.current.contains(event.target as Node)) { + setIsOpen(false); + } + }; + + document.addEventListener('mousedown', handleClickOutside); + return () => document.removeEventListener('mousedown', handleClickOutside); + }, []); + + const toggleSelection = (option: Option) => { + setSelected((prev) => { + const newSelection = prev.some((item) => item.label === option.label) + ? prev.filter((item) => item.label !== option.label) + : [...prev, option]; + if (onSelect) onSelect(newSelection); + return newSelection; + }); + }; + + const removeSelection = (option: Option) => { + setSelected((prev) => { + const newSelection = prev.filter((item) => item.label !== option.label); + if (onSelect) onSelect(newSelection); + return newSelection; + }); + }; + + const removeExtraSelections = () => { + setSelected((prev) => { + const newSelection = prev.slice(0, 3); // Keep only the first 3 + if (onSelect) onSelect(newSelection); + return newSelection; + }); + }; + + return ( +
+
setIsOpen(!isOpen)} + > +
+ {selected.length > 0 ? ( + <> + {selected.slice(0, 3).map((item, index) => ( + + {item.label} + { + e.stopPropagation(); + removeSelection(item); + }} + /> + + ))} + {selected.length > 3 && ( + { + e.stopPropagation(); + removeExtraSelections(); + }} + > + +{selected.length - 3} more + + )} + + ) : ( + Select options... + )} +
+ +
+ {isOpen && ( + + {options.map((option, index) => ( +
item.label === option.label) ? 'bg-muted' : 'bg-card' + }`} + onClick={() => toggleSelection(option)} + > +
+ {option.icon ? ( + {option.label} + ) : ( + formatAgentName(option.label) + )} +
+ {option.label} +
+ ))} +
+ )} +
+ ); +} diff --git a/packages/client/src/components/group-panel.tsx b/packages/client/src/components/group-panel.tsx index 9681784f592..c37b3c40f30 100644 --- a/packages/client/src/components/group-panel.tsx +++ b/packages/client/src/components/group-panel.tsx @@ -1,10 +1,8 @@ import { Card, CardContent } from './ui/card'; -import { formatAgentName } from '@/lib/utils'; import { Button } from './ui/button'; import { ImageIcon, Loader2, X } from 'lucide-react'; import { useEffect, useRef, useState } from 'react'; import { Input } from './ui/input'; -import { Switch } from './ui/switch-button'; import { apiClient } from '@/lib/api'; import { useQueryClient } from '@tanstack/react-query'; import { useNavigate } from 'react-router'; @@ -12,6 +10,7 @@ import { type Agent, AgentStatus } from '@elizaos/core'; import { UUID } from 'crypto'; import { GROUP_CHAT_SOURCE } from '@/constants'; import { useRooms } from '@/hooks/use-query-hooks'; +import MultiSelectCombobox from './combobox'; interface GroupPanel { agents: Agent[] | undefined; @@ -21,7 +20,7 @@ interface GroupPanel { export default function GroupPanel({ onClose, agents, groupId }: GroupPanel) { const [chatName, setChatName] = useState(``); - const [selectedAgents, setSelectedAgents] = useState<{ [key: string]: boolean }>({}); + const [selectedAgents, setSelectedAgents] = useState([]); const [creating, setCreating] = useState(false); const [deleting, setDeleting] = useState(false); @@ -44,13 +43,6 @@ export default function GroupPanel({ onClose, agents, groupId }: GroupPanel) { } }, [groupId]); - const toggleAgentSelection = (agentId: string) => { - setSelectedAgents((prev) => ({ - ...prev, - [agentId]: !prev[agentId], - })); - }; - const handleFileUpload = (event: React.ChangeEvent) => { const file = event.target.files?.[0]; if (file) { @@ -119,7 +111,7 @@ export default function GroupPanel({ onClose, agents, groupId }: GroupPanel) { {!avatar && } - +
@@ -133,42 +125,19 @@ export default function GroupPanel({ onClose, agents, groupId }: GroupPanel) { />
Invite Agents
-
-
- {agents + agent.status === AgentStatus.ACTIVE) - .map((agent) => { - return ( -
-
-
-
- {agent && agent.settings?.avatar ? ( - Agent Avatar - ) : ( - formatAgentName(agent.name) - )} -
-
-
{agent.name}
-
-
-
- toggleAgentSelection(agent.id as UUID)} - /> -
-
-
- ); - })} -
-
+ .map((agent) => ({ + icon: agent.settings?.avatar || '', + label: agent.name, + id: agent.id, + })) || [] + } + onSelect={(selected) => setSelectedAgents(selected)} + className="w-full" + />
@@ -183,11 +152,7 @@ export default function GroupPanel({ onClose, agents, groupId }: GroupPanel) { } setCreating(true); try { - const selectedAgentIds = Object.keys(selectedAgents).filter( - (agentId) => selectedAgents[agentId] - ); - - if (selectedAgentIds.length > 0) { + if (selectedAgents.length > 0) { if (groupId) { try { await apiClient.deleteGroupChat(groupId); @@ -196,13 +161,11 @@ export default function GroupPanel({ onClose, agents, groupId }: GroupPanel) { } } await apiClient.createGroupChat( - selectedAgentIds, + selectedAgents.map((agent) => agent.id), chatName, serverId, GROUP_CHAT_SOURCE, - { - thumbnail: avatar, - } + { thumbnail: avatar } ); } } catch (error) { diff --git a/packages/client/src/components/ui/switch-button.tsx b/packages/client/src/components/ui/switch-button.tsx deleted file mode 100644 index fc2168ab71f..00000000000 --- a/packages/client/src/components/ui/switch-button.tsx +++ /dev/null @@ -1,30 +0,0 @@ -import * as React from 'react'; -import { cn } from '@/lib/utils'; - -export interface SwitchProps { - checked: boolean; - onChange: () => void; - className?: string; -} - -const Switch: React.FC = ({ checked, onChange, className }) => { - return ( -
-
-
- ); -}; - -export { Switch };