mirror of
https://github.com/danny-avila/LibreChat.git
synced 2025-12-18 17:30:16 +01:00
140 lines
7.5 KiB
TypeScript
140 lines
7.5 KiB
TypeScript
|
|
import { useState } from 'react';
|
||
|
|
import { Settings } from 'lucide-react';
|
||
|
|
import { EModelEndpoint } from 'librechat-data-provider';
|
||
|
|
import type { FC } from 'react';
|
||
|
|
import { useLocalize, useUserKey, useNewConvo, useOriginNavigate } from '~/hooks';
|
||
|
|
import { SetKeyDialog } from '~/components/Input/SetKeyDialog';
|
||
|
|
import { icons } from './Icons';
|
||
|
|
import { cn } from '~/utils';
|
||
|
|
|
||
|
|
type MenuItemProps = {
|
||
|
|
title: string;
|
||
|
|
value: EModelEndpoint;
|
||
|
|
selected: boolean;
|
||
|
|
description?: string;
|
||
|
|
userProvidesKey: boolean;
|
||
|
|
// iconPath: string;
|
||
|
|
// hoverContent?: string;
|
||
|
|
};
|
||
|
|
|
||
|
|
const MenuItem: FC<MenuItemProps> = ({
|
||
|
|
title,
|
||
|
|
value: endpoint,
|
||
|
|
description,
|
||
|
|
selected,
|
||
|
|
userProvidesKey,
|
||
|
|
...rest
|
||
|
|
}) => {
|
||
|
|
const Icon = icons[endpoint] ?? icons.unknown;
|
||
|
|
const [isDialogOpen, setDialogOpen] = useState(false);
|
||
|
|
const { getExpiry } = useUserKey(endpoint);
|
||
|
|
const { newConversation } = useNewConvo();
|
||
|
|
const navigate = useOriginNavigate();
|
||
|
|
const localize = useLocalize();
|
||
|
|
const expiryTime = getExpiry();
|
||
|
|
|
||
|
|
const onSelectEndpoint = (newEndpoint: EModelEndpoint) => {
|
||
|
|
if (!newEndpoint) {
|
||
|
|
return;
|
||
|
|
} else {
|
||
|
|
if (!expiryTime) {
|
||
|
|
setDialogOpen(true);
|
||
|
|
}
|
||
|
|
newConversation({ template: { endpoint: newEndpoint } });
|
||
|
|
navigate('new');
|
||
|
|
}
|
||
|
|
};
|
||
|
|
|
||
|
|
return (
|
||
|
|
<>
|
||
|
|
<div
|
||
|
|
role="menuitem"
|
||
|
|
className="group m-1.5 flex cursor-pointer gap-2 rounded px-5 py-2.5 !pr-3 text-sm !opacity-100 hover:bg-black/5 focus:ring-0 radix-disabled:pointer-events-none radix-disabled:opacity-50 dark:hover:bg-white/5"
|
||
|
|
tabIndex={-1}
|
||
|
|
{...rest}
|
||
|
|
onClick={() => onSelectEndpoint(endpoint)}
|
||
|
|
>
|
||
|
|
<div className="flex grow items-center justify-between gap-2">
|
||
|
|
<div>
|
||
|
|
<div className="flex items-center gap-2">
|
||
|
|
{<Icon size={18} className="icon-md shrink-0 dark:text-white" />}
|
||
|
|
{/* <svg width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg" className="icon-md shrink-0">
|
||
|
|
<path d="M19.3975 1.35498C19.3746 1.15293 19.2037 1.00021 19.0004 1C18.7971 0.999793 18.6259 1.15217 18.6026 1.35417C18.4798 2.41894 18.1627 3.15692 17.6598 3.65983C17.1569 4.16274 16.4189 4.47983 15.3542 4.60264C15.1522 4.62593 14.9998 4.79707 15 5.00041C15.0002 5.20375 15.1529 5.37457 15.355 5.39746C16.4019 5.51605 17.1562 5.83304 17.6716 6.33906C18.1845 6.84269 18.5078 7.57998 18.6016 8.63539C18.6199 8.84195 18.7931 9.00023 19.0005 9C19.2078 8.99977 19.3806 8.84109 19.3985 8.6345C19.4883 7.59673 19.8114 6.84328 20.3273 6.32735C20.8433 5.81142 21.5967 5.48834 22.6345 5.39851C22.8411 5.38063 22.9998 5.20782 23 5.00045C23.0002 4.79308 22.842 4.61992 22.6354 4.60157C21.58 4.50782 20.8427 4.18447 20.3391 3.67157C19.833 3.15623 19.516 2.40192 19.3975 1.35498Z" fill="currentColor"/>
|
||
|
|
<path fillRule="evenodd" clipRule="evenodd" d="M11 3C11.4833 3 11.8974 3.34562 11.9839 3.82111C12.4637 6.46043 13.279 8.23983 14.5196 9.48039C15.7602 10.721 17.5396 11.5363 20.1789 12.0161C20.6544 12.1026 21 12.5167 21 13C21 13.4833 20.6544 13.8974 20.1789 13.9839C17.5396 14.4637 15.7602 15.279 14.5196 16.5196C13.279 17.7602 12.4637 19.5396 11.9839 22.1789C11.8974 22.6544 11.4833 23 11 23C10.5167 23 10.1026 22.6544 10.0161 22.1789C9.53625 19.5396 8.72096 17.7602 7.48039 16.5196C6.23983 15.279 4.46043 14.4637 1.82111 13.9839C1.34562 13.8974 1 13.4833 1 13C1 12.5167 1.34562 12.1026 1.82111 12.0161C4.46043 11.5363 6.23983 10.721 7.48039 9.48039C8.72096 8.23983 9.53625 6.46043 10.0161 3.82111C10.1026 3.34562 10.5167 3 11 3ZM5.66618 13C6.9247 13.5226 7.99788 14.2087 8.89461 15.1054C9.79134 16.0021 10.4774 17.0753 11 18.3338C11.5226 17.0753 12.2087 16.0021 13.1054 15.1054C14.0021 14.2087 15.0753 13.5226 16.3338 13C15.0753 12.4774 14.0021 11.7913 13.1054 10.8946C12.2087 9.99788 11.5226 8.9247 11 7.66618C10.4774 8.9247 9.79134 9.99788 8.89461 10.8946C7.99788 11.7913 6.9247 12.4774 5.66618 13Z" fill="currentColor"/>
|
||
|
|
</svg> */}
|
||
|
|
<div>
|
||
|
|
{title}
|
||
|
|
<div className="text-token-text-tertiary">{description}</div>
|
||
|
|
</div>
|
||
|
|
</div>
|
||
|
|
</div>
|
||
|
|
<div className="flex items-center gap-2">
|
||
|
|
{userProvidesKey ? (
|
||
|
|
<div className="text-token-text-primary" key={`set-key-${endpoint}`}>
|
||
|
|
<button
|
||
|
|
className={cn(
|
||
|
|
'invisible flex gap-x-1 group-hover:visible',
|
||
|
|
selected ? 'visible' : '',
|
||
|
|
expiryTime ? 'w-full rounded-lg p-2 hover:bg-gray-900' : '',
|
||
|
|
)}
|
||
|
|
onClick={(e) => {
|
||
|
|
e.preventDefault();
|
||
|
|
setDialogOpen(true);
|
||
|
|
}}
|
||
|
|
>
|
||
|
|
<div className={cn('invisible group-hover:visible', expiryTime ? 'text-xs' : '')}>
|
||
|
|
{localize('com_endpoint_config_key')}
|
||
|
|
</div>
|
||
|
|
<Settings className={cn(expiryTime ? 'icon-sm' : 'icon-md stroke-1')} />
|
||
|
|
</button>
|
||
|
|
</div>
|
||
|
|
) : null}
|
||
|
|
{selected && (
|
||
|
|
<svg
|
||
|
|
width="24"
|
||
|
|
height="24"
|
||
|
|
viewBox="0 0 24 24"
|
||
|
|
fill="none"
|
||
|
|
xmlns="http://www.w3.org/2000/svg"
|
||
|
|
className="icon-md block group-hover:hidden"
|
||
|
|
>
|
||
|
|
<path
|
||
|
|
fillRule="evenodd"
|
||
|
|
clipRule="evenodd"
|
||
|
|
d="M2 12C2 6.47715 6.47715 2 12 2C17.5228 2 22 6.47715 22 12C22 17.5228 17.5228 22 12 22C6.47715 22 2 17.5228 2 12ZM16.0755 7.93219C16.5272 8.25003 16.6356 8.87383 16.3178 9.32549L11.5678 16.0755C11.3931 16.3237 11.1152 16.4792 10.8123 16.4981C10.5093 16.517 10.2142 16.3973 10.0101 16.1727L7.51006 13.4227C7.13855 13.014 7.16867 12.3816 7.57733 12.0101C7.98598 11.6386 8.61843 11.6687 8.98994 12.0773L10.6504 13.9039L14.6822 8.17451C15 7.72284 15.6238 7.61436 16.0755 7.93219Z"
|
||
|
|
fill="currentColor"
|
||
|
|
/>
|
||
|
|
</svg>
|
||
|
|
)}
|
||
|
|
{(!userProvidesKey || expiryTime) && (
|
||
|
|
<div className="text-token-text-primary hidden gap-x-1 group-hover:flex ">
|
||
|
|
<div className="">New Chat</div>
|
||
|
|
<svg
|
||
|
|
width="24"
|
||
|
|
height="24"
|
||
|
|
viewBox="0 0 24 24"
|
||
|
|
fill="none"
|
||
|
|
xmlns="http://www.w3.org/2000/svg"
|
||
|
|
className="icon-md"
|
||
|
|
>
|
||
|
|
<path
|
||
|
|
fillRule="evenodd"
|
||
|
|
clipRule="evenodd"
|
||
|
|
d="M16.7929 2.79289C18.0118 1.57394 19.9882 1.57394 21.2071 2.79289C22.4261 4.01184 22.4261 5.98815 21.2071 7.20711L12.7071 15.7071C12.5196 15.8946 12.2652 16 12 16H9C8.44772 16 8 15.5523 8 15V12C8 11.7348 8.10536 11.4804 8.29289 11.2929L16.7929 2.79289ZM19.7929 4.20711C19.355 3.7692 18.645 3.7692 18.2071 4.2071L10 12.4142V14H11.5858L19.7929 5.79289C20.2308 5.35499 20.2308 4.64501 19.7929 4.20711ZM6 5C5.44772 5 5 5.44771 5 6V18C5 18.5523 5.44772 19 6 19H18C18.5523 19 19 18.5523 19 18V14C19 13.4477 19.4477 13 20 13C20.5523 13 21 13.4477 21 14V18C21 19.6569 19.6569 21 18 21H6C4.34315 21 3 19.6569 3 18V6C3 4.34314 4.34315 3 6 3H10C10.5523 3 11 3.44771 11 4C11 4.55228 10.5523 5 10 5H6Z"
|
||
|
|
fill="currentColor"
|
||
|
|
/>
|
||
|
|
</svg>
|
||
|
|
</div>
|
||
|
|
)}
|
||
|
|
</div>
|
||
|
|
</div>
|
||
|
|
</div>
|
||
|
|
{userProvidesKey && (
|
||
|
|
<SetKeyDialog open={isDialogOpen} onOpenChange={setDialogOpen} endpoint={endpoint} />
|
||
|
|
)}
|
||
|
|
</>
|
||
|
|
);
|
||
|
|
};
|
||
|
|
|
||
|
|
export default MenuItem;
|