mirror of
https://github.com/ruvnet/RuVector.git
synced 2026-05-29 19:33:34 +00:00
Major additions: - Complete Next.js studio application with 1600+ components - Docker support (Dockerfile.combined, docker-compose.yml) - GCP deployment documentation and benchmarks - SQL benchmark scripts for performance testing - Sentry integration for monitoring - Comprehensive test suite and mocks Studio features: - Dashboard and admin interfaces - Data visualization components - Authentication and user management - API integration with RuVector backend - Static data and public assets 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
103 lines
3.2 KiB
TypeScript
103 lines
3.2 KiB
TypeScript
import { Check, ChevronsUpDown } from 'lucide-react'
|
|
import { useState } from 'react'
|
|
|
|
import { useIsFeatureEnabled } from 'hooks/misc/useIsFeatureEnabled'
|
|
import { useSelectedOrganizationQuery } from 'hooks/misc/useSelectedOrganization'
|
|
import { useRouter } from 'next/router'
|
|
import {
|
|
Badge,
|
|
Button,
|
|
CommandGroup_Shadcn_,
|
|
CommandItem_Shadcn_,
|
|
CommandList_Shadcn_,
|
|
Command_Shadcn_,
|
|
PopoverContent_Shadcn_,
|
|
PopoverTrigger_Shadcn_,
|
|
Popover_Shadcn_,
|
|
TooltipContent,
|
|
TooltipTrigger,
|
|
Tooltip,
|
|
} from 'ui'
|
|
|
|
interface ModelSelectorProps {
|
|
selectedModel: 'gpt-5' | 'gpt-5-mini'
|
|
onSelectModel: (model: 'gpt-5' | 'gpt-5-mini') => void
|
|
}
|
|
|
|
export const ModelSelector = ({ selectedModel, onSelectModel }: ModelSelectorProps) => {
|
|
const router = useRouter()
|
|
const { data: organization } = useSelectedOrganizationQuery()
|
|
|
|
const [open, setOpen] = useState(false)
|
|
|
|
const canAccessProModels = organization?.plan?.id !== 'free'
|
|
const slug = organization?.slug ?? '_'
|
|
|
|
const upgradeHref = `/org/${slug ?? '_'}/billing?panel=subscriptionPlan&source=ai-assistant-model`
|
|
|
|
const handleSelectModel = (model: 'gpt-5' | 'gpt-5-mini') => {
|
|
if (model === 'gpt-5' && !canAccessProModels) {
|
|
setOpen(false)
|
|
void router.push(upgradeHref)
|
|
return
|
|
}
|
|
|
|
onSelectModel(model)
|
|
setOpen(false)
|
|
}
|
|
|
|
return (
|
|
<Popover_Shadcn_ open={open} onOpenChange={setOpen}>
|
|
<PopoverTrigger_Shadcn_ asChild>
|
|
<Button
|
|
type="default"
|
|
className="text-foreground-light"
|
|
iconRight={<ChevronsUpDown strokeWidth={1} size={12} />}
|
|
>
|
|
{selectedModel}
|
|
</Button>
|
|
</PopoverTrigger_Shadcn_>
|
|
<PopoverContent_Shadcn_ className="p-0 w-44" align="start" side="top">
|
|
<Command_Shadcn_>
|
|
<CommandList_Shadcn_>
|
|
<CommandGroup_Shadcn_>
|
|
<CommandItem_Shadcn_
|
|
value="gpt-5-mini"
|
|
onSelect={() => handleSelectModel('gpt-5-mini')}
|
|
className="flex justify-between"
|
|
>
|
|
<span>gpt-5-mini</span>
|
|
{selectedModel === 'gpt-5-mini' && <Check className="h-3.5 w-3.5" />}
|
|
</CommandItem_Shadcn_>
|
|
<CommandItem_Shadcn_
|
|
value="gpt-5"
|
|
onSelect={() => handleSelectModel('gpt-5')}
|
|
className="flex justify-between"
|
|
>
|
|
<span>gpt-5</span>
|
|
{canAccessProModels ? (
|
|
selectedModel === 'gpt-5' ? (
|
|
<Check className="h-3.5 w-3.5" />
|
|
) : null
|
|
) : (
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<div>
|
|
<Badge role="button" variant="warning">
|
|
Upgrade
|
|
</Badge>
|
|
</div>
|
|
</TooltipTrigger>
|
|
<TooltipContent side="right">
|
|
gpt-5 is available on Pro plans and above
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
)}
|
|
</CommandItem_Shadcn_>
|
|
</CommandGroup_Shadcn_>
|
|
</CommandList_Shadcn_>
|
|
</Command_Shadcn_>
|
|
</PopoverContent_Shadcn_>
|
|
</Popover_Shadcn_>
|
|
)
|
|
}
|