mirror of
https://github.com/anomalyco/opencode.git
synced 2026-02-01 22:48:16 +00:00
234 lines
7.5 KiB
TypeScript
234 lines
7.5 KiB
TypeScript
import { createMemo, createSignal } from "solid-js"
|
|
import { useLocal } from "@tui/context/local"
|
|
import { useSync } from "@tui/context/sync"
|
|
import { map, pipe, flatMap, entries, filter, sortBy, take } from "remeda"
|
|
import { DialogSelect, type DialogSelectRef } from "@tui/ui/dialog-select"
|
|
import { useDialog } from "@tui/ui/dialog"
|
|
import { createDialogProviderOptions, DialogProvider } from "./dialog-provider"
|
|
import { Keybind } from "@/util/keybind"
|
|
import * as fuzzysort from "fuzzysort"
|
|
|
|
export function useConnected() {
|
|
const sync = useSync()
|
|
return createMemo(() =>
|
|
sync.data.provider.some((x) => x.id !== "opencode" || Object.values(x.models).some((y) => y.cost?.input !== 0)),
|
|
)
|
|
}
|
|
|
|
export function DialogModel(props: { providerID?: string }) {
|
|
const local = useLocal()
|
|
const sync = useSync()
|
|
const dialog = useDialog()
|
|
const [ref, setRef] = createSignal<DialogSelectRef<unknown>>()
|
|
const [query, setQuery] = createSignal("")
|
|
|
|
const connected = useConnected()
|
|
const providers = createDialogProviderOptions()
|
|
|
|
const showExtra = createMemo(() => {
|
|
if (!connected()) return false
|
|
if (props.providerID) return false
|
|
return true
|
|
})
|
|
|
|
const options = createMemo(() => {
|
|
const q = query()
|
|
const needle = q.trim()
|
|
const showSections = showExtra() && needle.length === 0
|
|
const favorites = connected() ? local.model.favorite() : []
|
|
const recents = local.model.recent()
|
|
|
|
const recentList = showSections
|
|
? recents.filter(
|
|
(item) => !favorites.some((fav) => fav.providerID === item.providerID && fav.modelID === item.modelID),
|
|
)
|
|
: []
|
|
|
|
const favoriteOptions = showSections
|
|
? favorites.flatMap((item) => {
|
|
const provider = sync.data.provider.find((x) => x.id === item.providerID)
|
|
if (!provider) return []
|
|
const model = provider.models[item.modelID]
|
|
if (!model) return []
|
|
return [
|
|
{
|
|
key: item,
|
|
value: {
|
|
providerID: provider.id,
|
|
modelID: model.id,
|
|
},
|
|
title: model.name ?? item.modelID,
|
|
description: provider.name,
|
|
category: "Favorites",
|
|
disabled: provider.id === "opencode" && model.id.includes("-nano"),
|
|
footer: model.cost?.input === 0 && provider.id === "opencode" ? "Free" : undefined,
|
|
onSelect: () => {
|
|
dialog.clear()
|
|
local.model.set(
|
|
{
|
|
providerID: provider.id,
|
|
modelID: model.id,
|
|
},
|
|
{ recent: true },
|
|
)
|
|
},
|
|
},
|
|
]
|
|
})
|
|
: []
|
|
|
|
const recentOptions = showSections
|
|
? recentList.flatMap((item) => {
|
|
const provider = sync.data.provider.find((x) => x.id === item.providerID)
|
|
if (!provider) return []
|
|
const model = provider.models[item.modelID]
|
|
if (!model) return []
|
|
return [
|
|
{
|
|
key: item,
|
|
value: {
|
|
providerID: provider.id,
|
|
modelID: model.id,
|
|
},
|
|
title: model.name ?? item.modelID,
|
|
description: provider.name,
|
|
category: "Recent",
|
|
disabled: provider.id === "opencode" && model.id.includes("-nano"),
|
|
footer: model.cost?.input === 0 && provider.id === "opencode" ? "Free" : undefined,
|
|
onSelect: () => {
|
|
dialog.clear()
|
|
local.model.set(
|
|
{
|
|
providerID: provider.id,
|
|
modelID: model.id,
|
|
},
|
|
{ recent: true },
|
|
)
|
|
},
|
|
},
|
|
]
|
|
})
|
|
: []
|
|
|
|
const providerOptions = pipe(
|
|
sync.data.provider,
|
|
sortBy(
|
|
(provider) => provider.id !== "opencode",
|
|
(provider) => provider.name,
|
|
),
|
|
flatMap((provider) =>
|
|
pipe(
|
|
provider.models,
|
|
entries(),
|
|
filter(([_, info]) => info.status !== "deprecated"),
|
|
filter(([_, info]) => (props.providerID ? info.providerID === props.providerID : true)),
|
|
map(([model, info]) => {
|
|
const value = {
|
|
providerID: provider.id,
|
|
modelID: model,
|
|
}
|
|
return {
|
|
value,
|
|
title: info.name ?? model,
|
|
description: favorites.some(
|
|
(item) => item.providerID === value.providerID && item.modelID === value.modelID,
|
|
)
|
|
? "(Favorite)"
|
|
: undefined,
|
|
category: connected() ? provider.name : undefined,
|
|
disabled: provider.id === "opencode" && model.includes("-nano"),
|
|
footer: info.cost?.input === 0 && provider.id === "opencode" ? "Free" : undefined,
|
|
onSelect() {
|
|
dialog.clear()
|
|
local.model.set(
|
|
{
|
|
providerID: provider.id,
|
|
modelID: model,
|
|
},
|
|
{ recent: true },
|
|
)
|
|
},
|
|
}
|
|
}),
|
|
filter((x) => {
|
|
if (!showSections) return true
|
|
const value = x.value
|
|
const inFavorites = favorites.some(
|
|
(item) => item.providerID === value.providerID && item.modelID === value.modelID,
|
|
)
|
|
if (inFavorites) return false
|
|
const inRecents = recentList.some(
|
|
(item) => item.providerID === value.providerID && item.modelID === value.modelID,
|
|
)
|
|
if (inRecents) return false
|
|
return true
|
|
}),
|
|
sortBy(
|
|
(x) => x.footer !== "Free",
|
|
(x) => x.title,
|
|
),
|
|
),
|
|
),
|
|
)
|
|
|
|
const popularProviders = !connected()
|
|
? pipe(
|
|
providers(),
|
|
map((option) => {
|
|
return {
|
|
...option,
|
|
category: "Popular providers",
|
|
}
|
|
}),
|
|
take(6),
|
|
)
|
|
: []
|
|
|
|
// Search shows a single merged list (favorites inline)
|
|
if (needle) {
|
|
const filteredProviders = fuzzysort.go(needle, providerOptions, { keys: ["title", "category"] }).map((x) => x.obj)
|
|
const filteredPopular = fuzzysort.go(needle, popularProviders, { keys: ["title"] }).map((x) => x.obj)
|
|
return [...filteredProviders, ...filteredPopular]
|
|
}
|
|
|
|
return [...favoriteOptions, ...recentOptions, ...providerOptions, ...popularProviders]
|
|
})
|
|
|
|
const provider = createMemo(() =>
|
|
props.providerID ? sync.data.provider.find((x) => x.id === props.providerID) : null,
|
|
)
|
|
|
|
const title = createMemo(() => {
|
|
if (provider()) return provider()!.name
|
|
return "Select model"
|
|
})
|
|
|
|
return (
|
|
<DialogSelect
|
|
keybind={[
|
|
{
|
|
keybind: Keybind.parse("ctrl+a")[0],
|
|
title: connected() ? "Connect provider" : "View all providers",
|
|
onTrigger() {
|
|
dialog.replace(() => <DialogProvider />)
|
|
},
|
|
},
|
|
{
|
|
keybind: Keybind.parse("ctrl+f")[0],
|
|
title: "Favorite",
|
|
disabled: !connected(),
|
|
onTrigger: (option) => {
|
|
local.model.toggleFavorite(option.value as { providerID: string; modelID: string })
|
|
},
|
|
},
|
|
]}
|
|
ref={setRef}
|
|
onFilter={setQuery}
|
|
skipFilter={true}
|
|
title={title()}
|
|
current={local.model.current()}
|
|
options={options()}
|
|
/>
|
|
)
|
|
}
|