mirror of
https://github.com/willchen96/mike.git
synced 2026-06-14 20:55:13 +02:00
126 lines
5.2 KiB
TypeScript
126 lines
5.2 KiB
TypeScript
"use client";
|
|
|
|
import { useState } from "react";
|
|
import { ChevronDown, Check, AlertCircle } from "lucide-react";
|
|
import {
|
|
DropdownMenu,
|
|
DropdownMenuContent,
|
|
DropdownMenuItem,
|
|
DropdownMenuLabel,
|
|
DropdownMenuSeparator,
|
|
DropdownMenuTrigger,
|
|
} from "@/components/ui/dropdown-menu";
|
|
import { isModelAvailable } from "@/app/lib/modelAvailability";
|
|
import type { ApiKeyState } from "@/app/lib/mikeApi";
|
|
|
|
export interface ModelOption {
|
|
id: string;
|
|
label: string;
|
|
group: "Anthropic" | "Google" | "OpenAI";
|
|
}
|
|
|
|
export const MODELS: ModelOption[] = [
|
|
{ id: "claude-opus-4-7", label: "Claude Opus 4.7", group: "Anthropic" },
|
|
{ id: "claude-sonnet-4-6", label: "Claude Sonnet 4.6", group: "Anthropic" },
|
|
{ id: "gemini-3.1-pro-preview", label: "Gemini 3.1 Pro", group: "Google" },
|
|
{ id: "gemini-3-flash-preview", label: "Gemini 3 Flash", group: "Google" },
|
|
{ id: "gpt-5.5", label: "GPT-5.5", group: "OpenAI" },
|
|
{ id: "gpt-5.4", label: "GPT-5.4", group: "OpenAI" },
|
|
];
|
|
|
|
export const SETTINGS_MODELS: ModelOption[] = [
|
|
...MODELS,
|
|
{ id: "claude-haiku-4-5", label: "Claude Haiku 4.5", group: "Anthropic" },
|
|
{
|
|
id: "gemini-3.1-flash-lite-preview",
|
|
label: "Gemini 3.1 Flash Lite",
|
|
group: "Google",
|
|
},
|
|
{ id: "gpt-5.4-lite", label: "GPT-5.4 Lite", group: "OpenAI" },
|
|
];
|
|
|
|
export const DEFAULT_MODEL_ID = "gemini-3-flash-preview";
|
|
|
|
export const ALLOWED_MODEL_IDS = new Set(MODELS.map((m) => m.id));
|
|
|
|
const GROUP_ORDER: ModelOption["group"][] = ["Anthropic", "Google", "OpenAI"];
|
|
|
|
interface Props {
|
|
value: string;
|
|
onChange: (id: string) => void;
|
|
apiKeys?: ApiKeyState;
|
|
}
|
|
|
|
export function ModelToggle({ value, onChange, apiKeys }: Props) {
|
|
const [isOpen, setIsOpen] = useState(false);
|
|
const selected = MODELS.find((m) => m.id === value);
|
|
const selectedLabel = selected?.label ?? "Model";
|
|
const selectedAvailable = apiKeys
|
|
? isModelAvailable(value, apiKeys)
|
|
: true;
|
|
|
|
return (
|
|
<DropdownMenu onOpenChange={setIsOpen}>
|
|
<DropdownMenuTrigger asChild>
|
|
<button
|
|
type="button"
|
|
className={`flex items-center gap-1.5 rounded-lg px-2 h-8 text-sm transition-colors cursor-pointer text-gray-400 hover:bg-gray-100 hover:text-gray-700 ${isOpen ? "bg-gray-100 text-gray-700" : ""}`}
|
|
title={
|
|
!selectedAvailable
|
|
? "API key missing for selected model"
|
|
: "Choose model"
|
|
}
|
|
>
|
|
{!selectedAvailable && (
|
|
<AlertCircle className="h-3 w-3 shrink-0 text-red-500" />
|
|
)}
|
|
<span className="max-w-[140px] truncate">{selectedLabel}</span>
|
|
<ChevronDown
|
|
className={`h-3 w-3 shrink-0 transition-transform duration-200 ${isOpen ? "rotate-180" : ""}`}
|
|
/>
|
|
</button>
|
|
</DropdownMenuTrigger>
|
|
<DropdownMenuContent className="w-56 z-50" side="top" align="end">
|
|
{GROUP_ORDER.map((group, gi) => {
|
|
const items = MODELS.filter((m) => m.group === group);
|
|
if (items.length === 0) return null;
|
|
return (
|
|
<div key={group}>
|
|
{gi > 0 && <DropdownMenuSeparator />}
|
|
<DropdownMenuLabel className="text-[10px] uppercase tracking-wider text-gray-400">
|
|
{group}
|
|
</DropdownMenuLabel>
|
|
{items.map((m) => {
|
|
const available = apiKeys
|
|
? isModelAvailable(m.id, apiKeys)
|
|
: true;
|
|
return (
|
|
<DropdownMenuItem
|
|
key={m.id}
|
|
className="cursor-pointer"
|
|
onSelect={() => onChange(m.id)}
|
|
>
|
|
<span
|
|
className={`flex-1 ${available ? "" : "text-gray-400"}`}
|
|
>
|
|
{m.label}
|
|
</span>
|
|
{!available && (
|
|
<AlertCircle
|
|
className="h-3.5 w-3.5 text-red-500 ml-1"
|
|
aria-label="API key missing"
|
|
/>
|
|
)}
|
|
{m.id === value && available && (
|
|
<Check className="h-3.5 w-3.5 text-gray-600 ml-1" />
|
|
)}
|
|
</DropdownMenuItem>
|
|
);
|
|
})}
|
|
</div>
|
|
);
|
|
})}
|
|
</DropdownMenuContent>
|
|
</DropdownMenu>
|
|
);
|
|
}
|