| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226 |
- import { memo, useRef, useState } from "react"
- import { useWindowSize } from "react-use"
- import { useTranslation } from "react-i18next"
- import { VSCodeBadge } from "@vscode/webview-ui-toolkit/react"
- import { CloudUpload, CloudDownload, FoldVertical } from "lucide-react"
- import type { ClineMessage } from "@roo-code/types"
- import { getModelMaxOutputTokens } from "@roo/api"
- import { formatLargeNumber } from "@src/utils/format"
- import { cn } from "@src/lib/utils"
- import { Button } from "@src/components/ui"
- import { useExtensionState } from "@src/context/ExtensionStateContext"
- import { useSelectedModel } from "@/components/ui/hooks/useSelectedModel"
- import Thumbnails from "../common/Thumbnails"
- import { TaskActions } from "./TaskActions"
- import { ShareButton } from "./ShareButton"
- import { ContextWindowProgress } from "./ContextWindowProgress"
- import { Mention } from "./Mention"
- export interface TaskHeaderProps {
- task: ClineMessage
- tokensIn: number
- tokensOut: number
- doesModelSupportPromptCache: boolean
- cacheWrites?: number
- cacheReads?: number
- totalCost: number
- contextTokens: number
- buttonsDisabled: boolean
- handleCondenseContext: (taskId: string) => void
- onClose: () => void
- }
- const TaskHeader = ({
- task,
- tokensIn,
- tokensOut,
- doesModelSupportPromptCache,
- cacheWrites,
- cacheReads,
- totalCost,
- contextTokens,
- buttonsDisabled,
- handleCondenseContext,
- onClose,
- }: TaskHeaderProps) => {
- const { t } = useTranslation()
- const { apiConfiguration, currentTaskItem } = useExtensionState()
- const { id: modelId, info: model } = useSelectedModel(apiConfiguration)
- const [isTaskExpanded, setIsTaskExpanded] = useState(false)
- const textContainerRef = useRef<HTMLDivElement>(null)
- const textRef = useRef<HTMLDivElement>(null)
- const contextWindow = model?.contextWindow || 1
- const { width: windowWidth } = useWindowSize()
- const condenseButton = (
- <button
- title={t("chat:task.condenseContext")}
- disabled={buttonsDisabled}
- onClick={() => currentTaskItem && handleCondenseContext(currentTaskItem.id)}
- className="shrink-0 min-h-[20px] min-w-[20px] p-[2px] cursor-pointer disabled:cursor-not-allowed opacity-85 hover:opacity-100 bg-transparent border-none rounded-md">
- <FoldVertical size={16} />
- </button>
- )
- return (
- <div className="py-2 px-3">
- <div
- className={cn(
- "rounded-xs p-2.5 flex flex-col gap-1.5 relative z-1 border",
- isTaskExpanded
- ? "border-vscode-panel-border text-vscode-foreground"
- : "border-vscode-panel-border/80 text-vscode-foreground/80",
- )}>
- <div className="flex justify-between items-center gap-2">
- <div
- className="flex items-center cursor-pointer -ml-0.5 select-none grow min-w-0"
- onClick={() => setIsTaskExpanded(!isTaskExpanded)}>
- <div className="flex items-center shrink-0">
- <span className={`codicon codicon-chevron-${isTaskExpanded ? "down" : "right"}`}></span>
- </div>
- <div className="ml-1.5 whitespace-nowrap overflow-hidden text-ellipsis grow min-w-0">
- <span className="font-bold">
- {t("chat:task.title")}
- {!isTaskExpanded && ":"}
- </span>
- {!isTaskExpanded && (
- <span className="ml-1">
- <Mention text={task.text} />
- </span>
- )}
- </div>
- </div>
- <Button
- variant="ghost"
- size="icon"
- onClick={onClose}
- title={t("chat:task.closeAndStart")}
- className="shrink-0 w-5 h-5">
- <span className="codicon codicon-close" />
- </Button>
- </div>
- {/* Collapsed state: Track context and cost if we have any */}
- {!isTaskExpanded && contextWindow > 0 && (
- <div className={`w-full flex flex-row items-center gap-1 h-auto`}>
- <ContextWindowProgress
- contextWindow={contextWindow}
- contextTokens={contextTokens || 0}
- maxTokens={
- model
- ? getModelMaxOutputTokens({ modelId, model, settings: apiConfiguration })
- : undefined
- }
- />
- {condenseButton}
- <ShareButton item={currentTaskItem} disabled={buttonsDisabled} />
- {!!totalCost && <VSCodeBadge>${totalCost.toFixed(2)}</VSCodeBadge>}
- </div>
- )}
- {/* Expanded state: Show task text and images */}
- {isTaskExpanded && (
- <>
- <div
- ref={textContainerRef}
- className="-mt-0.5 text-vscode-font-size overflow-y-auto break-words break-anywhere relative">
- <div
- ref={textRef}
- className="overflow-auto max-h-80 whitespace-pre-wrap break-words break-anywhere"
- style={{
- display: "-webkit-box",
- WebkitLineClamp: "unset",
- WebkitBoxOrient: "vertical",
- }}>
- <Mention text={task.text} />
- </div>
- </div>
- {task.images && task.images.length > 0 && <Thumbnails images={task.images} />}
- <div className="flex flex-col gap-1">
- {isTaskExpanded && contextWindow > 0 && (
- <div
- className={`w-full flex ${windowWidth < 400 ? "flex-col" : "flex-row"} gap-1 h-auto`}>
- <div className="flex items-center gap-1 flex-shrink-0">
- <span className="font-bold" data-testid="context-window-label">
- {t("chat:task.contextWindow")}
- </span>
- </div>
- <ContextWindowProgress
- contextWindow={contextWindow}
- contextTokens={contextTokens || 0}
- maxTokens={
- model
- ? getModelMaxOutputTokens({
- modelId,
- model,
- settings: apiConfiguration,
- })
- : undefined
- }
- />
- {condenseButton}
- </div>
- )}
- <div className="flex justify-between items-center h-[20px]">
- <div className="flex items-center gap-1 flex-wrap">
- <span className="font-bold">{t("chat:task.tokens")}</span>
- {typeof tokensIn === "number" && tokensIn > 0 && (
- <span className="flex items-center gap-0.5">
- <i className="codicon codicon-arrow-up text-xs font-bold" />
- {formatLargeNumber(tokensIn)}
- </span>
- )}
- {typeof tokensOut === "number" && tokensOut > 0 && (
- <span className="flex items-center gap-0.5">
- <i className="codicon codicon-arrow-down text-xs font-bold" />
- {formatLargeNumber(tokensOut)}
- </span>
- )}
- </div>
- {!totalCost && <TaskActions item={currentTaskItem} buttonsDisabled={buttonsDisabled} />}
- </div>
- {doesModelSupportPromptCache &&
- ((typeof cacheReads === "number" && cacheReads > 0) ||
- (typeof cacheWrites === "number" && cacheWrites > 0)) && (
- <div className="flex items-center gap-1 flex-wrap h-[20px]">
- <span className="font-bold">{t("chat:task.cache")}</span>
- {typeof cacheWrites === "number" && cacheWrites > 0 && (
- <span className="flex items-center gap-0.5">
- <CloudUpload size={16} />
- {formatLargeNumber(cacheWrites)}
- </span>
- )}
- {typeof cacheReads === "number" && cacheReads > 0 && (
- <span className="flex items-center gap-0.5">
- <CloudDownload size={16} />
- {formatLargeNumber(cacheReads)}
- </span>
- )}
- </div>
- )}
- {!!totalCost && (
- <div className="flex justify-between items-center h-[20px]">
- <div className="flex items-center gap-1">
- <span className="font-bold">{t("chat:task.apiCost")}</span>
- <span>${totalCost?.toFixed(2)}</span>
- </div>
- <TaskActions item={currentTaskItem} buttonsDisabled={buttonsDisabled} />
- </div>
- )}
- </div>
- </>
- )}
- </div>
- </div>
- )
- }
- export default memo(TaskHeader)
|