session-context-usage.tsx 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. import { Match, Show, Switch, createMemo } from "solid-js"
  2. import { Tooltip } from "@opencode-ai/ui/tooltip"
  3. import { ProgressCircle } from "@opencode-ai/ui/progress-circle"
  4. import { Button } from "@opencode-ai/ui/button"
  5. import { useParams } from "@solidjs/router"
  6. import { AssistantMessage } from "@opencode-ai/sdk/v2/client"
  7. import { findLast } from "@opencode-ai/util/array"
  8. import { useLayout } from "@/context/layout"
  9. import { useSync } from "@/context/sync"
  10. import { useLanguage } from "@/context/language"
  11. interface SessionContextUsageProps {
  12. variant?: "button" | "indicator"
  13. }
  14. export function SessionContextUsage(props: SessionContextUsageProps) {
  15. const sync = useSync()
  16. const params = useParams()
  17. const layout = useLayout()
  18. const language = useLanguage()
  19. const variant = createMemo(() => props.variant ?? "button")
  20. const sessionKey = createMemo(() => `${params.dir}${params.id ? "/" + params.id : ""}`)
  21. const tabs = createMemo(() => layout.tabs(sessionKey))
  22. const view = createMemo(() => layout.view(sessionKey))
  23. const messages = createMemo(() => (params.id ? (sync.data.message[params.id] ?? []) : []))
  24. const cost = createMemo(() => {
  25. const locale = language.locale()
  26. const total = messages().reduce((sum, x) => sum + (x.role === "assistant" ? x.cost : 0), 0)
  27. return new Intl.NumberFormat(locale, {
  28. style: "currency",
  29. currency: "USD",
  30. }).format(total)
  31. })
  32. const context = createMemo(() => {
  33. const locale = language.locale()
  34. const last = findLast(messages(), (x) => {
  35. if (x.role !== "assistant") return false
  36. const total = x.tokens.input + x.tokens.output + x.tokens.reasoning + x.tokens.cache.read + x.tokens.cache.write
  37. return total > 0
  38. }) as AssistantMessage
  39. if (!last) return
  40. const total =
  41. last.tokens.input + last.tokens.output + last.tokens.reasoning + last.tokens.cache.read + last.tokens.cache.write
  42. const model = sync.data.provider.all.find((x) => x.id === last.providerID)?.models[last.modelID]
  43. return {
  44. tokens: total.toLocaleString(locale),
  45. percentage: model?.limit.context ? Math.round((total / model.limit.context) * 100) : null,
  46. }
  47. })
  48. const openContext = () => {
  49. if (!params.id) return
  50. view().reviewPanel.open()
  51. tabs().open("context")
  52. tabs().setActive("context")
  53. }
  54. const circle = () => (
  55. <div class="p-1">
  56. <ProgressCircle size={16} strokeWidth={2} percentage={context()?.percentage ?? 0} />
  57. </div>
  58. )
  59. const tooltipValue = () => (
  60. <div>
  61. <Show when={context()}>
  62. {(ctx) => (
  63. <>
  64. <div class="flex items-center gap-2">
  65. <span class="text-text-invert-strong">{ctx().tokens}</span>
  66. <span class="text-text-invert-base">{language.t("context.usage.tokens")}</span>
  67. </div>
  68. <div class="flex items-center gap-2">
  69. <span class="text-text-invert-strong">{ctx().percentage ?? 0}%</span>
  70. <span class="text-text-invert-base">{language.t("context.usage.usage")}</span>
  71. </div>
  72. </>
  73. )}
  74. </Show>
  75. <div class="flex items-center gap-2">
  76. <span class="text-text-invert-strong">{cost()}</span>
  77. <span class="text-text-invert-base">{language.t("context.usage.cost")}</span>
  78. </div>
  79. </div>
  80. )
  81. return (
  82. <Show when={params.id}>
  83. <Tooltip value={tooltipValue()} placement="top">
  84. <Switch>
  85. <Match when={variant() === "indicator"}>{circle()}</Match>
  86. <Match when={true}>
  87. <Button
  88. type="button"
  89. variant="ghost"
  90. class="size-6"
  91. onClick={openContext}
  92. aria-label={language.t("context.usage.view")}
  93. >
  94. {circle()}
  95. </Button>
  96. </Match>
  97. </Switch>
  98. </Tooltip>
  99. </Show>
  100. )
  101. }