mirror of
https://github.com/danny-avila/LibreChat.git
synced 2025-09-22 08:12:00 +02:00

* wip: mcp select * refactor: Update useAvailableToolsQuery to support generic data types * feat: Enhance MCPSelect to dynamically load server options and improve MultiSelect component styling * WIP: ephemeral agents * wip: Add null check for MCPSelect and improve MultiSelect focus handling * feat: Pass conversationId prop to MCPSelect in BadgeRow to optimize badge rendering * feat: useApplyNewAgentTemplate hook to manage ephemeral agent upon conversation creation * WIP: eph. agent payload * refactor(OpenAIClient): streamline message processing by replacing content handling with parseTextParts function * feat: enhance applyAgentTemplate function to accept source conversation ID for improved template application * feat(parsers): add skipReasoning parameter to parseTextParts for conditional reasoning handling * WIP: first pass, ephemeral agent backend processing * chore: import order * feat: update loadEphemeralAgent and loadAgent functions to accept model_parameters for enhanced agent configuration * feat: add showMCPServers prop to BadgeRow for conditional rendering of MCPSelect, fix react rule violation * feat: enhance MCPSelect with localized placeholder and custom icon, add renderSelectedValues callback * feat: simplify message processing in AnthropicClient by replacing content handling with parseTextParts function * feat: implement useLocalStorage hook for managing MCP values and update MCPSelect to utilize it * chore: remove chatGPTBrowserSchema from endpoint schemas and update types for improved schema management * chore: remove compactChatGPTSchema from endpoint schemas and update types for better schema management * refactor: rename schemas for clarity and improve schema management * feat: extend model detection to include 'codestral' alongside 'mistral' * feat: add endpointType parameter to buildOptions and initializeClient functions * fix: update condition for handling completion in BaseClient to include agents client * refactor: simplify payload parsing logic in AgentClient and remove unused providerParsers * refactor: change useSetRecoilState to useRecoilState for better state management in MCPSelect component * refactor: streamline chat route handlers by consolidating middleware and improving endpoint structure * style: update MCPSelect and MultiSelect components for improved layout in mobile view * v0.7.790 * feat: add getMessageMapMethod to process message text and content in GoogleClient * chore: include LAST_MCP_ key prefix in clearLocalStorage function for proper teardown on logout
128 lines
4 KiB
TypeScript
128 lines
4 KiB
TypeScript
import React, { useRef } from 'react';
|
|
import {
|
|
Select,
|
|
SelectArrow,
|
|
SelectItem,
|
|
SelectItemCheck,
|
|
SelectLabel,
|
|
SelectPopover,
|
|
SelectProvider,
|
|
} from '@ariakit/react';
|
|
import { cn } from '~/utils';
|
|
|
|
interface MultiSelectProps<T extends string> {
|
|
items: T[];
|
|
label?: string;
|
|
placeholder?: string;
|
|
defaultSelectedValues?: T[];
|
|
onSelectedValuesChange?: (values: T[]) => void;
|
|
renderSelectedValues?: (values: T[], placeholder?: string) => React.ReactNode;
|
|
className?: string;
|
|
itemClassName?: string;
|
|
labelClassName?: string;
|
|
selectClassName?: string;
|
|
selectIcon?: React.ReactNode;
|
|
popoverClassName?: string;
|
|
selectItemsClassName?: string;
|
|
selectedValues: T[];
|
|
setSelectedValues: (values: T[]) => void;
|
|
}
|
|
|
|
function defaultRender<T extends string>(values: T[], placeholder?: string) {
|
|
if (values.length === 0) {
|
|
return placeholder || 'Select...';
|
|
}
|
|
if (values.length === 1) {
|
|
return values[0];
|
|
}
|
|
return `${values.length} items selected`;
|
|
}
|
|
|
|
export default function MultiSelect<T extends string>({
|
|
items,
|
|
label,
|
|
placeholder = 'Select...',
|
|
defaultSelectedValues = [],
|
|
onSelectedValuesChange,
|
|
renderSelectedValues = defaultRender,
|
|
className,
|
|
selectIcon,
|
|
itemClassName,
|
|
labelClassName,
|
|
selectClassName,
|
|
popoverClassName,
|
|
selectItemsClassName,
|
|
selectedValues = [],
|
|
setSelectedValues,
|
|
}: MultiSelectProps<T>) {
|
|
const selectRef = useRef<HTMLButtonElement>(null);
|
|
// const [selectedValues, setSelectedValues] = React.useState<T[]>(defaultSelectedValues);
|
|
|
|
const handleValueChange = (values: T[]) => {
|
|
setSelectedValues(values);
|
|
if (onSelectedValuesChange) {
|
|
onSelectedValuesChange(values);
|
|
}
|
|
};
|
|
|
|
return (
|
|
<div className={cn('h-full', className)}>
|
|
<SelectProvider value={selectedValues} setValue={handleValueChange}>
|
|
{label && (
|
|
<SelectLabel className={cn('mb-1 block text-sm text-text-primary', labelClassName)}>
|
|
{label}
|
|
</SelectLabel>
|
|
)}
|
|
<Select
|
|
ref={selectRef}
|
|
className={cn(
|
|
'flex items-center justify-between gap-2 rounded-xl px-3 py-2 text-sm',
|
|
'bg-surface-tertiary text-text-primary shadow-sm hover:cursor-pointer hover:bg-surface-hover',
|
|
'outline-none focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-white focus-visible:ring-opacity-75',
|
|
selectClassName,
|
|
selectedValues.length > 0 && selectItemsClassName != null && selectItemsClassName,
|
|
)}
|
|
>
|
|
{selectIcon && selectIcon}
|
|
<span className="hidden truncate md:block">
|
|
{renderSelectedValues(selectedValues, placeholder)}
|
|
</span>
|
|
<SelectArrow className="ml-1 hidden stroke-1 text-base opacity-75 md:block" />
|
|
</Select>
|
|
<SelectPopover
|
|
gutter={4}
|
|
sameWidth
|
|
modal
|
|
unmountOnHide
|
|
finalFocus={selectRef}
|
|
className={cn(
|
|
'animate-popover z-50 flex max-h-[300px]',
|
|
'flex-col overflow-auto overscroll-contain rounded-xl',
|
|
'bg-surface-secondary px-1.5 py-1 text-text-primary shadow-lg',
|
|
'border border-border-light',
|
|
'outline-none',
|
|
popoverClassName,
|
|
)}
|
|
>
|
|
{items.map((value) => (
|
|
<SelectItem
|
|
key={value}
|
|
value={value}
|
|
className={cn(
|
|
'flex items-center gap-2 rounded-lg px-2 py-1.5 hover:cursor-pointer',
|
|
'scroll-m-1 outline-none transition-colors',
|
|
'hover:bg-black/[0.075] dark:hover:bg-white/10',
|
|
'data-[active-item]:bg-black/[0.075] dark:data-[active-item]:bg-white/10',
|
|
'w-full min-w-0 text-sm',
|
|
itemClassName,
|
|
)}
|
|
>
|
|
<SelectItemCheck className="text-primary" />
|
|
<span className="truncate">{value}</span>
|
|
</SelectItem>
|
|
))}
|
|
</SelectPopover>
|
|
</SelectProvider>
|
|
</div>
|
|
);
|
|
}
|