diff --git a/ui/admin/app/components/composed/ComboBox.tsx b/ui/admin/app/components/composed/ComboBox.tsx index 5c2b34c1f..06d819f62 100644 --- a/ui/admin/app/components/composed/ComboBox.tsx +++ b/ui/admin/app/components/composed/ComboBox.tsx @@ -35,7 +35,7 @@ type ComboBoxProps = { onChange: (option: T | null) => void; options: (T | GroupedOption)[]; placeholder?: string; - suggested?: string[]; + renderOption?: (option: T) => ReactNode; value?: T | null; }; @@ -43,7 +43,7 @@ export function ComboBox({ disabled, placeholder, value, - suggested, + renderOption, ...props }: { disabled?: boolean; @@ -58,7 +58,7 @@ export function ComboBox({ @@ -74,7 +74,7 @@ export function ComboBox({
@@ -95,12 +95,9 @@ export function ComboBox({ }} > - {value ? value.name : placeholder}{" "} - {value?.name && suggested?.includes(value.name) && ( - - (Suggested) - - )} + {renderOption && value + ? renderOption(value) + : (value?.name ?? placeholder)} ); @@ -113,7 +110,7 @@ function ComboBoxList({ onChange, options, setOpen, - suggested, + renderOption, value, placeholder = "Filter...", emptyLabel = "No results found.", @@ -160,22 +157,6 @@ function ComboBoxList({ [] as (T | GroupedOption)[] ); - const sortBySuggested = ( - a: T | GroupedOption, - b: T | GroupedOption - ) => { - // Handle nested groups - keep original order - if ("heading" in a || "heading" in b) return 0; - - const aIsSuggested = a.name && suggested?.includes(a.name); - const bIsSuggested = b.name && suggested?.includes(b.name); - - // If both or neither are suggested, maintain original order - if (aIsSuggested === bIsSuggested) return 0; - // Sort suggested items first - return aIsSuggested ? -1 : 1; - }; - const handleValueChange = (value: string) => { setFilteredOptions(filterOptions(options, value)); }; @@ -215,14 +196,11 @@ function ComboBoxList({ function renderGroupedOption(group: GroupedOption) { return ( - {group.value - .slice() // Create a copy to avoid mutating original array - .sort(sortBySuggested) - .map((option) => - "heading" in option - ? renderGroupedOption(option) - : renderUngroupedOption(option) - )} + {group.value.map((option) => + "heading" in option + ? renderGroupedOption(option) + : renderUngroupedOption(option) + )} ); } @@ -239,12 +217,9 @@ function ComboBoxList({ className="justify-between" > - {option.name || option.id}{" "} - {option?.name && suggested?.includes(option.name) && ( - - (Suggested) - - )} + {renderOption + ? renderOption(option) + : (option.name ?? option.id)} {value?.id === option.id && } diff --git a/ui/admin/app/components/model/DefaultModelAliasForm.tsx b/ui/admin/app/components/model/DefaultModelAliasForm.tsx index e5abfa0ba..c5de23562 100644 --- a/ui/admin/app/components/model/DefaultModelAliasForm.tsx +++ b/ui/admin/app/components/model/DefaultModelAliasForm.tsx @@ -180,9 +180,15 @@ export function DefaultModelAliasForm({ } options={getOptionsByUsageAndProvider( activeModelOptions, - usage + usage, + alias )} - suggested={getSuggested(alias)} + renderOption={(option) => + renderDisplayOption( + option, + alias + ) + } value={ field.value ? models?.find( @@ -219,21 +225,33 @@ export function DefaultModelAliasForm({ ); - function getSuggested(alias: ModelAlias) { - return alias && SUGGESTED_MODEL_SELECTIONS[alias] - ? [SUGGESTED_MODEL_SELECTIONS[alias]] - : []; + function renderDisplayOption(option: Model, alias: ModelAlias) { + const suggestion = alias && SUGGESTED_MODEL_SELECTIONS[alias]; + + return ( + + {option.name}{" "} + {suggestion === option.name && ( + (Suggested) + )} + + ); } function getOptionsByUsageAndProvider( modelOptions: Model[] | undefined, - usage: ModelUsage + usage: ModelUsage, + aliasFor: ModelAlias ) { if (!modelOptions) return []; + const suggested = aliasFor && SUGGESTED_MODEL_SELECTIONS[aliasFor]; const usageGroupName = getModelUsageLabel(usage); - const usageModelProviderGroups = - getModelOptionsByModelProvider(modelOptions); + const usageModelProviderGroups = getModelOptionsByModelProvider( + modelOptions, + suggested + ); + const otherModelProviderGroups = getModelOptionsByModelProvider(otherModels); const usageGroup = { @@ -290,7 +308,10 @@ export function DefaultModelAliasFormDialog({ ); } -export function getModelOptionsByModelProvider(models: Model[]) { +export function getModelOptionsByModelProvider( + models: Model[], + suggested?: string +) { const byModelProviderGroups = filterModelsByActive(models).reduce( (acc, model) => { acc[model.modelProvider] = acc[model.modelProvider] || []; @@ -302,9 +323,15 @@ export function getModelOptionsByModelProvider(models: Model[]) { return Object.entries(byModelProviderGroups).map( ([modelProvider, models]) => { - const sorted = models.sort((a, b) => - (a.name ?? "").localeCompare(b.name ?? "") - ); + const sorted = models.sort((a, b) => { + const aIsSuggested = a.name && suggested === a.name; + const bIsSuggested = b.name && suggested === b.name; + + if (aIsSuggested || bIsSuggested) { + return aIsSuggested ? -1 : 1; + } + return (a.name ?? "").localeCompare(b.name ?? ""); + }); return { heading: modelProvider, value: sorted,