Composable model picker with reasoning effort levels, search, and runtime integration.
A picker that lets users switch between AI models and choose a reasoning effort (thinking) level. It is built on Popover + Command, so search, provider grouping, and filtering compose in without being built in. The default export integrates with assistant-ui's ModelContext system, so the selection reaches your backend on every request with no extra wiring.
Getting Started
Add model-selector
npx shadcn@latest add https://r.assistant-ui.com/model-selector.jsonMain Component
npm install @assistant-ui/react class-variance-authority"use client";import { memo, useCallback, useEffect, useMemo, useRef, useState, createContext, useContext, type ComponentPropsWithoutRef, type KeyboardEvent, type ReactNode,} from "react";import { cva, type VariantProps } from "class-variance-authority";import { CheckIcon, ChevronDownIcon } from "lucide-react";import { useAui } from "@assistant-ui/react";import { cn } from "@/lib/utils";import { Popover, PopoverContent, PopoverTrigger,} from "@/components/ui/popover";import { Command, CommandEmpty, CommandGroup, CommandInput, CommandItem, CommandList, CommandSeparator,} from "@/components/ui/command";export type ModelSelectorEffortOption = { id: string; name: string;};export const DEFAULT_EFFORT_OPTIONS: readonly ModelSelectorEffortOption[] = [ { id: "low", name: "Low" }, { id: "medium", name: "Medium" }, { id: "high", name: "High" },];export type ModelOption = { id: string; name: string; description?: string; icon?: ReactNode; disabled?: boolean; /** Extra terms matched by ModelSelector.Search, in addition to id and name. */ keywords?: readonly string[]; /** * Reasoning effort levels the model supports. Pass `true` for the default * low/medium/high levels, or a custom list. Omit for models without * configurable reasoning. */ efforts?: boolean | readonly ModelSelectorEffortOption[];};function getModelEfforts( model: ModelOption | undefined,): readonly ModelSelectorEffortOption[] | undefined { if (!model?.efforts) return undefined; return model.efforts === true ? DEFAULT_EFFORT_OPTIONS : model.efforts;}function resolveEffort( efforts: readonly ModelSelectorEffortOption[] | undefined, effort: string | undefined,): string | undefined { if (effort === undefined) return undefined; return efforts?.some((e) => e.id === effort) ? effort : undefined;}/** * Returns the effort id if the given model supports it, otherwise undefined. * Effort selection is kept sticky across model switches; this resolves what * actually applies to the current model. */export function resolveModelEffort( models: readonly ModelOption[], modelId: string | undefined, effort: string | undefined,): string | undefined { return resolveEffort( getModelEfforts(models.find((m) => m.id === modelId)), effort, );}function useControllableState<T>({ prop, defaultProp, onChange,}: { prop: T | undefined; defaultProp: T | undefined; onChange: ((next: T) => void) | undefined;}) { const [internal, setInternal] = useState(defaultProp); const isControlled = prop !== undefined; const value = isControlled ? prop : internal; // Read onChange through a ref so inline callbacks don't recreate the setter // (and with it the memoized context value) every render. const onChangeRef = useRef(onChange); useEffect(() => { onChangeRef.current = onChange; }); const setValue = useCallback( (next: T) => { if (!isControlled) setInternal(next); onChangeRef.current?.(next); }, [isControlled], ); return [value, setValue] as const;}type ModelSelectorContextValue = { models: readonly ModelOption[]; value: string | undefined; setValue: (value: string) => void; /** The model matching `value`, derived once for all sub-components. */ selectedModel: ModelOption | undefined; /** The selected model's effort levels, undefined when not configurable. */ efforts: readonly ModelSelectorEffortOption[] | undefined; /** Effort resolved against the selected model's supported levels. */ effort: string | undefined; setEffort: (effort: string) => void; setOpen: (open: boolean) => void;};const ModelSelectorContext = createContext<ModelSelectorContextValue | null>( null,);function useModelSelectorContext() { const ctx = useContext(ModelSelectorContext); if (!ctx) { throw new Error( "ModelSelector sub-components must be used within ModelSelector.Root", ); } return ctx;}/** * The selected model's effort levels and the active selection. Use it to build * a custom effort UI inside ModelSelector.Content (e.g. a slider or a shadcn * DropdownMenu) when the built-in ModelSelector.Effort layout doesn't fit. * `efforts` is undefined for models without configurable reasoning. */export function useModelSelectorEfforts(): { efforts: readonly ModelSelectorEffortOption[] | undefined; effort: string | undefined; setEffort: (effort: string) => void;} { const { efforts, effort, setEffort } = useModelSelectorContext(); return { efforts, effort, setEffort };}export type ModelSelectorRootProps = { models: readonly ModelOption[]; value?: string; defaultValue?: string; onValueChange?: (value: string) => void; effort?: string; defaultEffort?: string; onEffortChange?: (effort: string) => void; open?: boolean; defaultOpen?: boolean; onOpenChange?: (open: boolean) => void; children: ReactNode;};function ModelSelectorRoot({ models, value: valueProp, defaultValue, onValueChange, effort: effortProp, defaultEffort, onEffortChange, open: openProp, defaultOpen, onOpenChange, children,}: ModelSelectorRootProps) { const [value, setValue] = useControllableState({ prop: valueProp, defaultProp: defaultValue ?? models[0]?.id, onChange: onValueChange, }); const [effort, setEffort] = useControllableState({ prop: effortProp, defaultProp: defaultEffort, onChange: onEffortChange, }); const [open, setOpen] = useControllableState({ prop: openProp, defaultProp: defaultOpen ?? false, onChange: onOpenChange, }); const selectedModel = models.find((m) => m.id === value); const efforts = getModelEfforts(selectedModel); const activeEffort = resolveEffort(efforts, effort); const contextValue = useMemo( () => ({ models, value, setValue, selectedModel, efforts, effort: activeEffort, setEffort, setOpen, }), [ models, value, setValue, selectedModel, efforts, activeEffort, setEffort, setOpen, ], ); return ( <ModelSelectorContext.Provider value={contextValue}> {/* `?? false` narrows away undefined for exactOptionalPropertyTypes consumers. */} <Popover open={open ?? false} onOpenChange={setOpen}> {children} </Popover> </ModelSelectorContext.Provider> );}export const modelSelectorTriggerVariants = cva( "focus-visible:ring-ring/50 flex w-fit items-center justify-between gap-2 overflow-hidden rounded-md text-sm whitespace-nowrap transition-colors outline-none focus-visible:ring-2 disabled:cursor-not-allowed disabled:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4", { variants: { variant: { outline: "border-input hover:bg-accent hover:text-accent-foreground border bg-transparent", ghost: "hover:bg-accent hover:text-accent-foreground", muted: "bg-secondary text-secondary-foreground hover:bg-secondary/80", }, size: { default: "h-9 px-3 py-2", sm: "h-8 px-2.5 py-1.5 text-xs", lg: "h-10 px-4 py-2.5", }, }, defaultVariants: { variant: "outline", size: "default", }, },);export type ModelSelectorTriggerProps = ComponentPropsWithoutRef< typeof PopoverTrigger> & VariantProps<typeof modelSelectorTriggerVariants>;function ModelSelectorTrigger({ className, variant, size, children, ...props}: ModelSelectorTriggerProps) { return ( <PopoverTrigger data-slot="model-selector-trigger" data-variant={variant ?? "outline"} data-size={size ?? "default"} role="combobox" className={cn(modelSelectorTriggerVariants({ variant, size }), className)} {...props} > {children ?? <ModelSelectorValue />} <ChevronDownIcon className="size-4 opacity-50" /> </PopoverTrigger> );}export type ModelSelectorValueProps = { placeholder?: ReactNode; /** Show the active effort level next to the model name. */ showEffort?: boolean; className?: string;};function ModelIcon({ children }: { children: ReactNode }) { return ( <span className="flex size-4 shrink-0 items-center justify-center [&_svg]:size-4"> {children} </span> );}function ModelSelectorValue({ placeholder = "Select model", showEffort = true, className,}: ModelSelectorValueProps) { const { selectedModel, efforts, effort } = useModelSelectorContext(); if (!selectedModel) { return ( <span data-slot="model-selector-value" className={cn("text-muted-foreground", className)} > {placeholder} </span> ); } const effortName = showEffort && effort !== undefined ? efforts?.find((e) => e.id === effort)?.name : undefined; return ( <span data-slot="model-selector-value" className={cn("flex min-w-0 items-center gap-2", className)} > {selectedModel.icon && <ModelIcon>{selectedModel.icon}</ModelIcon>} <span className="truncate font-medium">{selectedModel.name}</span> {effortName && ( <span className="text-muted-foreground truncate">{effortName}</span> )} </span> );}export type ModelSelectorContentProps = ComponentPropsWithoutRef< typeof PopoverContent>;function ModelSelectorContent({ className, align = "start", sideOffset = 6, children, ...props}: ModelSelectorContentProps) { const { value } = useModelSelectorContext(); return ( <PopoverContent data-slot="model-selector-content" align={align} sideOffset={sideOffset} className={cn( "bg-popover/95 w-72 min-w-(--radix-popover-trigger-width) overflow-hidden rounded-xl p-0 shadow-lg backdrop-blur-sm", className, )} {...props} > {/* Seeding cmdk with the selected id makes it the active item, which cmdk scrolls into view when the popover opens. */} <Command className="bg-transparent" {...(value !== undefined ? { defaultValue: value } : {})} > {children ?? ( <> <ModelSelectorList /> <ModelSelectorEffort /> </> )} </Command> </PopoverContent> );}export type ModelSelectorSearchProps = ComponentPropsWithoutRef< typeof CommandInput>;function ModelSelectorSearch({ placeholder = "Search models...", ...props}: ModelSelectorSearchProps) { return ( <CommandInput data-slot="model-selector-search" placeholder={placeholder} {...props} /> );}export type ModelSelectorListProps = ComponentPropsWithoutRef< typeof CommandList>;function ModelSelectorList({ className, children, ...props}: ModelSelectorListProps) { const { models } = useModelSelectorContext(); return ( <CommandList data-slot="model-selector-list" className={cn( "[-ms-overflow-style:none] [scrollbar-width:none] [&::-webkit-scrollbar]:hidden", className, )} {...props} > {children ?? ( <> <ModelSelectorEmpty /> <CommandGroup> {models.map((model) => ( <ModelSelectorItem key={model.id} model={model} /> ))} </CommandGroup> </> )} </CommandList> );}export type ModelSelectorEmptyProps = ComponentPropsWithoutRef< typeof CommandEmpty>;function ModelSelectorEmpty({ children, ...props }: ModelSelectorEmptyProps) { return ( <CommandEmpty data-slot="model-selector-empty" {...props}> {children ?? "No models found."} </CommandEmpty> );}export type ModelSelectorGroupProps = ComponentPropsWithoutRef< typeof CommandGroup>;function ModelSelectorGroup(props: ModelSelectorGroupProps) { return <CommandGroup data-slot="model-selector-group" {...props} />;}export type ModelSelectorSeparatorProps = ComponentPropsWithoutRef< typeof CommandSeparator>;function ModelSelectorSeparator(props: ModelSelectorSeparatorProps) { return <CommandSeparator data-slot="model-selector-separator" {...props} />;}export type ModelSelectorItemProps = Omit< ComponentPropsWithoutRef<typeof CommandItem>, "value"> & { model: ModelOption;};function ModelSelectorItem({ model, className, children, onSelect, ...props}: ModelSelectorItemProps) { const { value, setValue, setOpen } = useModelSelectorContext(); const isSelected = value === model.id; return ( <CommandItem data-slot="model-selector-item" value={model.id} keywords={[model.name, ...(model.keywords ?? [])]} {...(model.disabled ? { disabled: true } : undefined)} onSelect={(selectedValue) => { setValue(model.id); setOpen(false); onSelect?.(selectedValue); }} className={cn("relative gap-2 rounded-lg py-2 ps-3 pe-9", className)} {...props} > {children ?? ( <> {model.icon && <ModelIcon>{model.icon}</ModelIcon>} <span className="flex min-w-0 flex-col"> <span className="truncate font-medium">{model.name}</span> {model.description && ( <span className="text-muted-foreground truncate text-xs"> {model.description} </span> )} </span> </> )} {isSelected && ( <span className="absolute end-3 flex size-4 items-center justify-center"> <CheckIcon className="size-4" /> </span> )} </CommandItem> );}export type ModelSelectorEffortProps = ComponentPropsWithoutRef<"div"> & { label?: ReactNode;};function ModelSelectorEffort({ label = "Thinking", className, onKeyDown, ...props}: ModelSelectorEffortProps) { const { efforts, effort, setEffort } = useModelSelectorEfforts(); if (!efforts?.length) return null; return ( <div data-slot="model-selector-effort" className={cn( "flex items-center justify-between gap-3 border-t px-3 py-2", className, )} onKeyDown={(e: KeyboardEvent<HTMLDivElement>) => { // cmdk's root keydown handler claims Enter to select the highlighted // model; stop it from seeing Enter so the focused toggle activates. if (e.key === "Enter") e.stopPropagation(); onKeyDown?.(e); }} {...props} > <span className="text-muted-foreground text-xs">{label}</span> <div role="group" aria-label={typeof label === "string" ? label : "Reasoning effort"} className="flex items-center gap-0.5" > {efforts.map((option) => { const isActive = option.id === effort; return ( <button key={option.id} type="button" aria-pressed={isActive} data-state={isActive ? "on" : "off"} onClick={() => setEffort(option.id)} className={cn( "focus-visible:ring-ring/50 rounded-md px-2 py-1 text-xs transition-colors outline-none focus-visible:ring-2", isActive ? "bg-accent text-accent-foreground font-medium" : "text-muted-foreground hover:text-foreground", )} > {option.name} </button> ); })} </div> </div> );}export type ModelSelectorProps = Omit<ModelSelectorRootProps, "children"> & VariantProps<typeof modelSelectorTriggerVariants> & { /** Render a search input above the model list. */ searchable?: boolean; className?: string; contentClassName?: string; };/** Registers the selection with assistant-ui's ModelContext system. The * context's effort is already resolved against the selected model. */function ModelSelectorModelContext() { const { value, effort } = useModelSelectorContext(); const api = useAui(); useEffect(() => { if (value === undefined) return; const config = { config: { modelName: value, ...(effort !== undefined ? { reasoningEffort: effort } : undefined), }, }; return api.modelContext().register({ getModelContext: () => config, }); }, [api, value, effort]); return null;}const ModelSelectorImpl = ({ searchable, variant, size, className, contentClassName, ...rootProps}: ModelSelectorProps) => { return ( <ModelSelectorRoot {...rootProps}> <ModelSelectorModelContext /> <ModelSelectorTrigger variant={variant} size={size} className={className} /> <ModelSelectorContent className={contentClassName}> {searchable && <ModelSelectorSearch />} <ModelSelectorList /> <ModelSelectorEffort /> </ModelSelectorContent> </ModelSelectorRoot> );};type ModelSelectorComponent = typeof ModelSelectorImpl & { displayName?: string; Root: typeof ModelSelectorRoot; Trigger: typeof ModelSelectorTrigger; Value: typeof ModelSelectorValue; Content: typeof ModelSelectorContent; Search: typeof ModelSelectorSearch; List: typeof ModelSelectorList; Empty: typeof ModelSelectorEmpty; Group: typeof ModelSelectorGroup; Separator: typeof ModelSelectorSeparator; Item: typeof ModelSelectorItem; Effort: typeof ModelSelectorEffort;};const ModelSelector = memo( ModelSelectorImpl,) as unknown as ModelSelectorComponent;ModelSelector.displayName = "ModelSelector";ModelSelector.Root = ModelSelectorRoot;ModelSelector.Trigger = ModelSelectorTrigger;ModelSelector.Value = ModelSelectorValue;ModelSelector.Content = ModelSelectorContent;ModelSelector.Search = ModelSelectorSearch;ModelSelector.List = ModelSelectorList;ModelSelector.Empty = ModelSelectorEmpty;ModelSelector.Group = ModelSelectorGroup;ModelSelector.Separator = ModelSelectorSeparator;ModelSelector.Item = ModelSelectorItem;ModelSelector.Effort = ModelSelectorEffort;export { ModelSelector, ModelSelectorRoot, ModelSelectorTrigger, ModelSelectorValue, ModelSelectorContent, ModelSelectorSearch, ModelSelectorList, ModelSelectorEmpty, ModelSelectorGroup, ModelSelectorSeparator, ModelSelectorItem, ModelSelectorEffort,};Use in your application
Place the ModelSelector inside your thread component, typically in the composer area. Each model needs an id and a display name; everything else is optional:
import { ModelSelector } from "@/components/assistant-ui/model-selector";
const ComposerAction: FC = () => {
return (
<div className="flex items-center gap-1">
<ModelSelector
models={[
{ id: "gpt-5.4-nano", name: "GPT-5.4 Nano", description: "Fast and efficient" },
{ id: "gpt-5.4-mini", name: "GPT-5.4 Mini", description: "Balanced performance" },
{ id: "gpt-5.5", name: "GPT-5.5", description: "Most capable", efforts: true },
]}
defaultValue="gpt-5.4-nano"
defaultEffort="medium"
size="sm"
/>
</div>
);
};Read the selection in your API route
The selected model's id arrives as config.modelName, and the effort level as config.reasoningEffort:
export async function POST(req: Request) {
const { messages, config } = await req.json();
const result = streamText({
model: openai(config?.modelName ?? "gpt-5.4-nano"),
providerOptions: {
openai:
config?.reasoningEffort !== undefined
? { reasoningEffort: config.reasoningEffort }
: {},
},
messages: await convertToModelMessages(messages),
});
return result.toUIMessageStreamResponse();
}config.reasoningEffort is only present when the selected model supports the chosen level, so the route only forwards it when it exists. See How It Works.
Reasoning Efforts
A model that declares efforts shows a "Thinking" row at the bottom of the popover. efforts: true enables the default Low / Medium / High levels; pass a list of { id, name } objects to define your own:
{
id: "gpt-5.5",
name: "GPT-5.5",
efforts: [
{ id: "minimal", name: "Minimal" },
{ id: "high", name: "High" },
],
}Omit efforts for models without configurable reasoning. The row is hidden while such a model is selected.
Sticky Selection
The effort selection survives model switches. Switching to a model that doesn't support the current level omits reasoningEffort from the request instead of resetting the user's choice, and the level applies again when the user switches back. The exported resolveModelEffort helper applies the same rule if you build your own runtime integration around ModelSelector.Root; see resolveModelEffort.
Custom Effort UI
ModelSelector.Effort lays the levels out as horizontal segments, which overflows the popover width once a model has more than a few. For those cases, or for a different layout such as a slider or a sub-dropdown, build your own control with the useModelSelectorEfforts hook. It exposes the selected model's levels and the active selection:
import { useModelSelectorEfforts } from "@/components/assistant-ui/model-selector";
import {
DropdownMenu,
DropdownMenuContent,
DropdownMenuRadioGroup,
DropdownMenuRadioItem,
DropdownMenuTrigger,
} from "@/components/ui/dropdown-menu";
function EffortDropdown() {
const { efforts, effort, setEffort } = useModelSelectorEfforts();
if (!efforts?.length) return null;
return (
<div className="flex items-center justify-between gap-3 border-t px-3 py-2">
<span className="text-muted-foreground text-xs">Thinking</span>
<DropdownMenu>
<DropdownMenuTrigger className="text-xs">
{efforts.find((e) => e.id === effort)?.name ?? "Select"}
</DropdownMenuTrigger>
<DropdownMenuContent align="end">
<DropdownMenuRadioGroup value={effort} onValueChange={setEffort}>
{efforts.map((option) => (
<DropdownMenuRadioItem key={option.id} value={option.id}>
{option.name}
</DropdownMenuRadioItem>
))}
</DropdownMenuRadioGroup>
</DropdownMenuContent>
</DropdownMenu>
</div>
);
}Render it inside ModelSelector.Content in place of ModelSelector.Effort. The same hook supports any shape that reads the levels and writes the selection.
Search
Search is opt-in. Pass searchable to the default component:
<ModelSelector models={models} searchable />Or compose ModelSelector.Search into a custom layout. Matching runs against each model's id, name, and keywords; add the provider name to keywords so typing "openai" finds its models.
Composition
All parts are exported individually. The default popover content is List + Effort; replace it to add search, provider groups, or anything else:
import {
ModelSelectorRoot,
ModelSelectorTrigger,
ModelSelectorContent,
ModelSelectorSearch,
ModelSelectorList,
ModelSelectorEmpty,
ModelSelectorGroup,
ModelSelectorItem,
ModelSelectorEffort,
} from "@/components/assistant-ui/model-selector";
<ModelSelectorRoot
models={models}
value={modelId}
onValueChange={setModelId}
effort={effort}
onEffortChange={setEffort}
>
<ModelSelectorTrigger variant="outline" />
<ModelSelectorContent>
<ModelSelectorSearch placeholder="Search models..." />
<ModelSelectorList>
<ModelSelectorEmpty />
<ModelSelectorGroup heading="OpenAI">
{openaiModels.map((model) => (
<ModelSelectorItem key={model.id} model={model} />
))}
</ModelSelectorGroup>
<ModelSelectorGroup heading="Anthropic">
{anthropicModels.map((model) => (
<ModelSelectorItem key={model.id} model={model} />
))}
</ModelSelectorGroup>
</ModelSelectorList>
<ModelSelectorEffort label="Thinking" />
</ModelSelectorContent>
</ModelSelectorRoot>| Component | Description |
|---|---|
ModelSelector | Default export with runtime integration |
ModelSelector.Root | Presentational root (no runtime, controlled state) |
ModelSelector.Trigger | CVA-styled trigger showing the current selection |
ModelSelector.Value | Selected model name, icon, and active effort |
ModelSelector.Content | Popover content wrapping a Command |
ModelSelector.Search | Search input that filters the list |
ModelSelector.List | List of model items (renders all models by default) |
ModelSelector.Empty | Empty state shown when search has no matches |
ModelSelector.Group | Labeled group of items (e.g. by provider) |
ModelSelector.Separator | Divider between groups or items |
ModelSelector.Item | Individual model option |
ModelSelector.Effort | Thinking level row for the selected model |
ModelSelector.List is a Command list, so filtering and keyboard navigation work across groups automatically. Custom sorting is plain code: order the models before rendering items.
ModelSelector.Content wraps a Command whose root keydown handler claims
Enter to select the highlighted model. Interactive elements
composed inside it (filter chips, custom effort controls) should stop
propagation for Enter in their own onKeyDown so the focused control
activates instead. ModelSelector.Effort already does this.
Variants
Use the variant prop to change the trigger's visual style.
<ModelSelector variant="outline" /> // Border (default)
<ModelSelector variant="ghost" /> // No background
<ModelSelector variant="muted" /> // Solid background| Variant | Description |
|---|---|
outline | Border with transparent background (default) |
ghost | No background, subtle hover |
muted | Solid secondary background |
Sizes
Use the size prop to control the trigger dimensions.
<ModelSelector size="sm" /> // Compact (h-8, text-xs)
<ModelSelector size="default" /> // Standard (h-9)
<ModelSelector size="lg" /> // Large (h-10)How It Works
The default ModelSelector export registers the selection with assistant-ui's ModelContext system:
- The component calls
aui.modelContext().register()withconfig.modelName, plusconfig.reasoningEffortwhen the selected model supports the chosen level - The
AssistantChatTransportincludesconfigin the request body of every chat request - Your API route reads
config.modelNameandconfig.reasoningEffort
This works out of the box with @assistant-ui/react-ai-sdk. ModelSelector.Root performs no registration; it is purely presentational, with controlled and uncontrolled props for the value, effort, and open state.
API Reference
ModelSelector
ModelSelectorPropsmodels: ModelOption[]Array of available models to display.
defaultValue?: stringInitial model ID for uncontrolled usage. Defaults to the first model, captured on first render; if models loads asynchronously, control the value instead.
value?: stringControlled selected model ID.
onValueChange?: (value: string) => voidCallback when selected model changes.
defaultEffort?: stringInitial effort level ID for uncontrolled usage.
effort?: stringControlled effort level ID.
onEffortChange?: (effort: string) => voidCallback when effort level changes.
searchable: boolean= falseRender a search input above the model list.
variant: "outline" | "ghost" | "muted"= "outline"Visual style of the trigger button.
size: "sm" | "default" | "lg"= "default"Size of the trigger button.
contentClassName?: stringAdditional class name for the dropdown content.
ModelOption
ModelOptionid: stringUnique identifier sent to the backend as modelName.
name: stringDisplay name shown in trigger and dropdown.
description?: stringOptional subtitle shown below the model name.
icon?: React.ReactNodeOptional icon displayed before the model name.
disabled?: booleanDisable selection of this model.
keywords?: string[]Extra search terms matched by ModelSelector.Search (e.g. the provider name).
efforts?: boolean | ModelSelectorEffortOption[]Reasoning effort levels. true enables the default Low/Medium/High; pass a custom { id, name } list to override. Omit for models without configurable reasoning.
useModelSelectorEfforts
const { efforts, effort, setEffort } = useModelSelectorEfforts();The selected model's effort levels and the active selection, for building a custom effort UI inside ModelSelector.Content. efforts is undefined for models without configurable reasoning.
resolveModelEffort
resolveModelEffort(models, modelId, effort); // => string | undefinedReturns the effort ID when the given model supports it, otherwise undefined. This is the sticky selection rule the default component applies before registering the selection.
Related
- Model Context: How registered context (instructions, tools, config) reaches your backend