mirror of
https://github.com/danny-avila/LibreChat.git
synced 2025-12-17 00:40:14 +01:00
* 🔧 fix: Simplify event handling in Badge component by always preventing default behavior and stopping propagation on toggle
* feat: show Global agents icon in ModelSelector
* feat: show Global agents icon in ModelSelector's search results
* refactor(Header): remove unused import
Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
* refactor(EndpointModelItem): remove unused import of useGetStartupConfig
---------
Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
97 lines
2.9 KiB
TypeScript
97 lines
2.9 KiB
TypeScript
import type React from 'react';
|
|
|
|
import { X, Plus } from 'lucide-react';
|
|
import { motion } from 'framer-motion';
|
|
import type { ButtonHTMLAttributes } from 'react';
|
|
import type { LucideIcon } from 'lucide-react';
|
|
import { cn } from '~/utils';
|
|
|
|
interface BadgeProps extends ButtonHTMLAttributes<HTMLButtonElement> {
|
|
icon?: LucideIcon;
|
|
label: string;
|
|
id?: string;
|
|
isActive?: boolean;
|
|
isEditing?: boolean;
|
|
isDragging?: boolean;
|
|
isAvailable: boolean;
|
|
isInChat?: boolean;
|
|
onBadgeAction?: () => void;
|
|
onToggle?: () => void;
|
|
}
|
|
|
|
export default function Badge({
|
|
icon: Icon,
|
|
label,
|
|
id,
|
|
isActive = false,
|
|
isEditing = false,
|
|
isDragging = false,
|
|
isAvailable = true,
|
|
isInChat = false,
|
|
onBadgeAction,
|
|
onToggle,
|
|
className,
|
|
...props
|
|
}: BadgeProps) {
|
|
const isMoveable = isEditing && isAvailable;
|
|
const isDisabled = id === '1' && isInChat;
|
|
|
|
const handleClick: React.MouseEventHandler<HTMLButtonElement> = (e) => {
|
|
if (isDisabled) {
|
|
e.preventDefault();
|
|
e.stopPropagation();
|
|
return;
|
|
}
|
|
|
|
if (!isEditing && onToggle) {
|
|
e.preventDefault();
|
|
e.stopPropagation();
|
|
onToggle();
|
|
}
|
|
};
|
|
|
|
return (
|
|
<motion.button
|
|
onClick={handleClick}
|
|
className={cn(
|
|
'group relative inline-flex items-center gap-1.5 rounded-full px-4 py-1.5',
|
|
'border border-border-medium text-sm font-medium transition-shadow md:w-full',
|
|
'size-9 p-2 md:p-3',
|
|
isActive
|
|
? 'bg-surface-active shadow-md'
|
|
: 'bg-surface-chat shadow-sm hover:bg-surface-hover hover:shadow-md',
|
|
'active:scale-95 active:shadow-inner',
|
|
isMoveable && 'cursor-move',
|
|
isDisabled && 'cursor-not-allowed opacity-50 hover:shadow-sm',
|
|
className,
|
|
)}
|
|
animate={{
|
|
scale: isDragging ? 1.1 : 1,
|
|
boxShadow: isDragging ? '0 10px 25px rgba(0,0,0,0.1)' : undefined,
|
|
}}
|
|
whileTap={{ scale: isDragging ? 1.1 : isDisabled ? 1 : 0.97 }}
|
|
transition={{ type: 'tween', duration: 0.1, ease: 'easeOut' }}
|
|
{...props}
|
|
>
|
|
{Icon && <Icon className={cn('relative h-5 w-5 md:h-4 md:w-4', !label && 'mx-auto')} />}
|
|
<span className="relative hidden md:inline">{label}</span>
|
|
|
|
{isEditing && !isDragging && (
|
|
<motion.button
|
|
className="absolute -right-1 -top-1 flex h-6 w-6 items-center justify-center rounded-full bg-surface-secondary-alt text-text-primary shadow-sm md:h-5 md:w-5"
|
|
initial={{ opacity: 0, scale: 0.8 }}
|
|
animate={{ opacity: 1, scale: 1 }}
|
|
exit={{ opacity: 0, scale: 0.8 }}
|
|
whileTap={{ scale: 0.9 }}
|
|
onMouseDown={(e) => e.stopPropagation()}
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
onBadgeAction?.();
|
|
}}
|
|
>
|
|
{isAvailable ? <X className="h-3 w-3" /> : <Plus className="h-3 w-3" />}
|
|
</motion.button>
|
|
)}
|
|
</motion.button>
|
|
);
|
|
}
|