101 lines
3.3 KiB
TypeScript
101 lines
3.3 KiB
TypeScript
import { Match, Show, Switch, createMemo } from "solid-js"
|
|
import { Tooltip } from "@opencode-ai/ui/tooltip"
|
|
import { ProgressCircle } from "@opencode-ai/ui/progress-circle"
|
|
import { Button } from "@opencode-ai/ui/button"
|
|
import { useParams } from "@solidjs/router"
|
|
|
|
import { useLayout } from "@/context/layout"
|
|
import { useSync } from "@/context/sync"
|
|
import { useLanguage } from "@/context/language"
|
|
import { getSessionContextMetrics } from "@/components/session/session-context-metrics"
|
|
|
|
interface SessionContextUsageProps {
|
|
variant?: "button" | "indicator"
|
|
}
|
|
|
|
export function SessionContextUsage(props: SessionContextUsageProps) {
|
|
const sync = useSync()
|
|
const params = useParams()
|
|
const layout = useLayout()
|
|
const language = useLanguage()
|
|
|
|
const variant = createMemo(() => props.variant ?? "button")
|
|
const sessionKey = createMemo(() => `${params.dir}${params.id ? "/" + params.id : ""}`)
|
|
const tabs = createMemo(() => layout.tabs(sessionKey))
|
|
const view = createMemo(() => layout.view(sessionKey))
|
|
const messages = createMemo(() => (params.id ? (sync.data.message[params.id] ?? []) : []))
|
|
|
|
const usd = createMemo(
|
|
() =>
|
|
new Intl.NumberFormat(language.locale(), {
|
|
style: "currency",
|
|
currency: "USD",
|
|
}),
|
|
)
|
|
|
|
const metrics = createMemo(() => getSessionContextMetrics(messages(), sync.data.provider.all))
|
|
const context = createMemo(() => metrics().context)
|
|
const cost = createMemo(() => {
|
|
return usd().format(metrics().totalCost)
|
|
})
|
|
|
|
const openContext = () => {
|
|
if (!params.id) return
|
|
if (!view().reviewPanel.opened()) view().reviewPanel.open()
|
|
layout.fileTree.open()
|
|
layout.fileTree.setTab("all")
|
|
tabs().open("context")
|
|
tabs().setActive("context")
|
|
}
|
|
|
|
const circle = () => (
|
|
<div class="flex items-center justify-center">
|
|
<ProgressCircle size={16} strokeWidth={2} percentage={context()?.usage ?? 0} />
|
|
</div>
|
|
)
|
|
|
|
const tooltipValue = () => (
|
|
<div>
|
|
<Show when={context()}>
|
|
{(ctx) => (
|
|
<>
|
|
<div class="flex items-center gap-2">
|
|
<span class="text-text-invert-strong">{ctx().total.toLocaleString(language.locale())}</span>
|
|
<span class="text-text-invert-base">{language.t("context.usage.tokens")}</span>
|
|
</div>
|
|
<div class="flex items-center gap-2">
|
|
<span class="text-text-invert-strong">{ctx().usage ?? 0}%</span>
|
|
<span class="text-text-invert-base">{language.t("context.usage.usage")}</span>
|
|
</div>
|
|
</>
|
|
)}
|
|
</Show>
|
|
<div class="flex items-center gap-2">
|
|
<span class="text-text-invert-strong">{cost()}</span>
|
|
<span class="text-text-invert-base">{language.t("context.usage.cost")}</span>
|
|
</div>
|
|
</div>
|
|
)
|
|
|
|
return (
|
|
<Show when={params.id}>
|
|
<Tooltip value={tooltipValue()} placement="top">
|
|
<Switch>
|
|
<Match when={variant() === "indicator"}>{circle()}</Match>
|
|
<Match when={true}>
|
|
<Button
|
|
type="button"
|
|
variant="ghost"
|
|
class="size-6"
|
|
onClick={openContext}
|
|
aria-label={language.t("context.usage.view")}
|
|
>
|
|
{circle()}
|
|
</Button>
|
|
</Match>
|
|
</Switch>
|
|
</Tooltip>
|
|
</Show>
|
|
)
|
|
}
|