Skip to content
47 changes: 16 additions & 31 deletions webview-ui/src/components/chat/AutoApproveDropdown.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ import { cn } from "@/lib/utils"
import { useExtensionState } from "@/context/ExtensionStateContext"
import { useAppTranslation } from "@/i18n/TranslationContext"
import { useRooPortal } from "@/components/ui/hooks/useRooPortal"
import { Popover, PopoverContent, PopoverTrigger, StandardTooltip, ToggleSwitch } from "@/components/ui"
import { Popover, PopoverContent, PopoverTrigger, StandardTooltip, ToggleSwitch, Button } from "@/components/ui"
import { AutoApproveSetting, autoApproveSettingsConfig } from "../settings/AutoApproveToggle"
import { useAutoApprovalToggles } from "@/hooks/useAutoApprovalToggles"
import { useAutoApprovalState } from "@/hooks/useAutoApprovalState"
Expand Down Expand Up @@ -228,24 +228,21 @@ export const AutoApproveDropdown = ({ disabled = false, triggerClassName = "" }:
const isEnabled = toggles[key]
return (
<StandardTooltip key={key} content={t(descriptionKey)}>
<button
<Button
variant={isEnabled ? "primary" : "secondary"}
onClick={() => onAutoApproveToggle(key, !isEnabled)}
className={cn(
"flex items-center gap-2 px-2 py-2 rounded text-sm text-left",
"flex items-center gap-2 px-2 py-2 text-sm text-left justify-start h-auto",
"transition-all duration-150",
"opacity-100 hover:opacity-70",
"cursor-pointer",
!effectiveAutoApprovalEnabled &&
"opacity-50 cursor-not-allowed hover:opacity-50",
isEnabled
? "bg-vscode-button-background text-vscode-button-foreground"
: "bg-vscode-button-background/15 text-vscode-foreground hover:bg-vscode-list-hoverBackground",
!isEnabled && "bg-vscode-button-background/15",
)}
disabled={!effectiveAutoApprovalEnabled}
data-testid={`auto-approve-${key}`}>
<span className={`codicon codicon-${icon} text-sm flex-shrink-0`} />
<span className="flex-1 truncate">{t(labelKey)}</span>
</button>
</Button>
</StandardTooltip>
)
})}
Expand All @@ -254,44 +251,32 @@ export const AutoApproveDropdown = ({ disabled = false, triggerClassName = "" }:
{/* Bottom bar with Select All/None buttons */}
<div className="flex flex-row items-center justify-between px-2 py-2 border-t border-vscode-dropdown-border">
<div className="flex flex-row gap-1">
<button
<Button
variant="ghost"
size="sm"
aria-label={t("chat:autoApprove.selectAll")}
onClick={handleSelectAll}
disabled={!effectiveAutoApprovalEnabled}
className={cn(
"relative inline-flex items-center justify-center gap-1",
"bg-transparent border-none px-2 py-1",
"rounded-md text-base font-bold",
"text-vscode-foreground",
"transition-all duration-150",
"hover:opacity-100 hover:bg-[rgba(255,255,255,0.03)]",
"focus:outline-none focus-visible:ring-1 focus-visible:ring-vscode-focusBorder",
"active:bg-[rgba(255,255,255,0.1)]",
"cursor-pointer",
"gap-1 px-2 py-1 text-base font-bold h-auto",
!effectiveAutoApprovalEnabled && "opacity-50 hover:opacity-50 cursor-not-allowed",
)}>
<ListChecks className="w-3.5 h-3.5" />
<span>{t("chat:autoApprove.all")}</span>
</button>
<button
</Button>
<Button
variant="ghost"
size="sm"
aria-label={t("chat:autoApprove.selectNone")}
onClick={handleSelectNone}
disabled={!effectiveAutoApprovalEnabled}
className={cn(
"relative inline-flex items-center justify-center gap-1",
"bg-transparent border-none px-2 py-1",
"rounded-md text-base font-bold",
"text-vscode-foreground",
"transition-all duration-150",
"hover:opacity-100 hover:bg-[rgba(255,255,255,0.03)]",
"focus:outline-none focus-visible:ring-1 focus-visible:ring-vscode-focusBorder",
"active:bg-[rgba(255,255,255,0.1)]",
"cursor-pointer",
"gap-1 px-2 py-1 text-base font-bold h-auto",
!effectiveAutoApprovalEnabled && "opacity-50 hover:opacity-50 cursor-not-allowed",
)}>
<LayoutList className="w-3.5 h-3.5" />
<span>{t("chat:autoApprove.none")}</span>
</button>
</Button>
</div>

<label
Expand Down
Original file line number Diff line number Diff line change
@@ -1,10 +1,10 @@
import React, { memo, useState } from "react"
import { VSCodeButton } from "@vscode/webview-ui-toolkit/react"
import { Trans } from "react-i18next"

import type { ClineMessage } from "@roo-code/types"

import { vscode } from "@src/utils/vscode"
import { Button } from "@src/components/ui"

type AutoApprovedRequestLimitWarningProps = {
message: ClineMessage
Expand Down Expand Up @@ -50,15 +50,15 @@ export const AutoApprovedRequestLimitWarning = memo(({ message }: AutoApprovedRe
<div className="flex justify-between items-center">
<Trans i18nKey={descriptionKey} ns="chat" values={{ count }} />
</div>
<VSCodeButton
<Button
style={{ width: "100%", padding: "6px", borderRadius: "4px" }}
onClick={(e) => {
e.preventDefault()
setButtonClicked(true)
vscode.postMessage({ type: "askResponse", askResponse: "yesButtonClicked" })
}}>
<Trans i18nKey={buttonKey} ns="chat" />
</VSCodeButton>
</Button>
</div>
</>
)
Expand Down
10 changes: 5 additions & 5 deletions webview-ui/src/components/chat/BrowserSessionRow.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -2,14 +2,14 @@ import React, { memo, useEffect, useMemo, useRef, useState } from "react"
import { useSize } from "react-use"
import deepEqual from "fast-deep-equal"
import { useTranslation } from "react-i18next"
import { VSCodeButton } from "@vscode/webview-ui-toolkit/react"

import type { ClineMessage } from "@roo-code/types"

import { BrowserAction, BrowserActionResult, ClineSayBrowserAction } from "@roo/ExtensionMessage"

import { vscode } from "@src/utils/vscode"
import { useExtensionState } from "@src/context/ExtensionStateContext"
import { Button } from "@src/components/ui"

import CodeBlock, { CODE_BLOCK_BG_COLOR } from "../common/CodeBlock"
import { ChatRowContent } from "./ChatRow"
Expand Down Expand Up @@ -372,16 +372,16 @@ const BrowserSessionRow = memo((props: BrowserSessionRowProps) => {
{t("chat:browser.navigation.step", { current: currentPageIndex + 1, total: pages.length })}
</div>
<div style={{ display: "flex", gap: "4px" }}>
<VSCodeButton
<Button
disabled={currentPageIndex === 0 || isBrowsing}
onClick={() => setCurrentPageIndex((i) => i - 1)}>
{t("chat:browser.navigation.previous")}
</VSCodeButton>
<VSCodeButton
</Button>
<Button
disabled={currentPageIndex === pages.length - 1 || isBrowsing}
onClick={() => setCurrentPageIndex((i) => i + 1)}>
{t("chat:browser.navigation.next")}
</VSCodeButton>
</Button>
</div>
</div>
)}
Expand Down
22 changes: 11 additions & 11 deletions webview-ui/src/components/chat/ChatView.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ import { useDeepCompareEffect, useEvent, useMount } from "react-use"
import debounce from "debounce"
import { Virtuoso, type VirtuosoHandle } from "react-virtuoso"
import removeMd from "remove-markdown"
import { VSCodeButton, VSCodeLink } from "@vscode/webview-ui-toolkit/react"
import { VSCodeLink } from "@vscode/webview-ui-toolkit/react"
import useSound from "use-sound"
import { LRUCache } from "lru-cache"
import { Trans } from "react-i18next"
Expand All @@ -30,7 +30,7 @@ import { useExtensionState } from "@src/context/ExtensionStateContext"
import { useSelectedModel } from "@src/components/ui/hooks/useSelectedModel"
import RooHero from "@src/components/welcome/RooHero"
import RooTips from "@src/components/welcome/RooTips"
import { StandardTooltip } from "@src/components/ui"
import { StandardTooltip, Button } from "@src/components/ui"
import { CloudUpsellDialog } from "@src/components/cloud/CloudUpsellDialog"

import TelemetryBanner from "../common/TelemetryBanner"
Expand Down Expand Up @@ -1464,15 +1464,15 @@ const ChatViewComponent: React.ForwardRefRenderFunction<ChatViewRef, ChatViewPro
}`}>
{showScrollToBottom ? (
<StandardTooltip content={t("chat:scrollToBottom")}>
<VSCodeButton
appearance="secondary"
<Button
variant="secondary"
className="flex-[2]"
onClick={() => {
scrollToBottomSmooth()
disableAutoScrollRef.current = false
}}>
<span className="codicon codicon-chevron-down"></span>
</VSCodeButton>
</Button>
</StandardTooltip>
) : (
<>
Expand All @@ -1499,13 +1499,13 @@ const ChatViewComponent: React.ForwardRefRenderFunction<ChatViewRef, ChatViewPro
? t("chat:proceedWhileRunning.tooltip")
: undefined
}>
<VSCodeButton
appearance="primary"
<Button
variant="primary"
disabled={!enableButtons}
className={secondaryButtonText ? "flex-1 mr-[6px]" : "flex-[2] mr-0"}
onClick={() => handlePrimaryButtonClick(inputValue, selectedImages)}>
{primaryButtonText}
</VSCodeButton>
</Button>
</StandardTooltip>
)}
{(secondaryButtonText || isStreaming) && (
Expand All @@ -1521,13 +1521,13 @@ const ChatViewComponent: React.ForwardRefRenderFunction<ChatViewRef, ChatViewPro
? t("chat:terminate.tooltip")
: undefined
}>
<VSCodeButton
appearance="secondary"
<Button
variant="secondary"
disabled={!enableButtons && !(isStreaming && !didClickCancel)}
className={isStreaming ? "flex-[2] ml-0" : "flex-1 ml-[6px]"}
onClick={() => handleSecondaryButtonClick(inputValue, selectedImages)}>
{isStreaming ? t("chat:cancel.title") : secondaryButtonText}
</VSCodeButton>
</Button>
</StandardTooltip>
)}
</>
Expand Down
13 changes: 7 additions & 6 deletions webview-ui/src/components/chat/CodeIndexPopover.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,7 @@ import {
PopoverContent,
Slider,
StandardTooltip,
Button,
} from "@src/components/ui"
import { useRooPortal } from "@src/components/ui/hooks/useRooPortal"
import { useEscapeKey } from "@src/hooks/useEscapeKey"
Expand Down Expand Up @@ -1387,21 +1388,21 @@ export const CodeIndexPopover: React.FC<CodeIndexPopoverProps> = ({
{currentSettings.codebaseIndexEnabled &&
(indexingStatus.systemStatus === "Error" ||
indexingStatus.systemStatus === "Standby") && (
<VSCodeButton
<Button
onClick={() => vscode.postMessage({ type: "startIndexing" })}
disabled={saveStatus === "saving" || hasUnsavedChanges}>
{t("settings:codeIndex.startIndexingButton")}
</VSCodeButton>
</Button>
)}

{currentSettings.codebaseIndexEnabled &&
(indexingStatus.systemStatus === "Indexed" ||
indexingStatus.systemStatus === "Error") && (
<AlertDialog>
<AlertDialogTrigger asChild>
<VSCodeButton appearance="secondary">
<Button variant="secondary">
{t("settings:codeIndex.clearIndexDataButton")}
</VSCodeButton>
</Button>
</AlertDialogTrigger>
<AlertDialogContent>
<AlertDialogHeader>
Expand All @@ -1426,13 +1427,13 @@ export const CodeIndexPopover: React.FC<CodeIndexPopoverProps> = ({
)}
</div>

<VSCodeButton
<Button
onClick={handleSaveSettings}
disabled={!hasUnsavedChanges || saveStatus === "saving"}>
{saveStatus === "saving"
? t("settings:codeIndex.saving")
: t("settings:codeIndex.saveSettings")}
</VSCodeButton>
</Button>
</div>

{/* Save Status Messages */}
Expand Down
6 changes: 2 additions & 4 deletions webview-ui/src/components/chat/FollowUpSuggest.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -108,9 +108,7 @@ export const FollowUpSuggest = ({
const isFirstSuggestion = index === 0

return (
<div
key={`${suggestion.answer}-${ts}`}
className="bg-vscode-editor-background rounded-sm w-full relative group">
<div key={`${suggestion.answer}-${ts}`} className="w-full relative group">
<Button
variant="outline"
className="text-left whitespace-normal break-words w-full h-auto px-3 py-2 justify-start pr-8"
Expand All @@ -133,7 +131,7 @@ export const FollowUpSuggest = ({
)}
<StandardTooltip content={t("chat:followUpSuggest.copyToInput")}>
<div
className="absolute cursor-pointer top-2 right-3 opacity-0 group-hover:opacity-100 transition-opacity"
className="absolute cursor-pointer top-1.5 right-3 opacity-0 group-hover:opacity-100 transition-opacity"
onClick={(e) => {
e.stopPropagation()
// Cancel the auto-approve timer when edit button is clicked
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -135,7 +135,7 @@ export const CheckpointMenu = ({ ts, commitHash, checkpoint, onOpenChange }: Che
) : (
<>
<Button
variant="default"
variant="primary"
onClick={onRestore}
className="grow"
data-testid="confirm-restore-btn">
Expand Down
20 changes: 7 additions & 13 deletions webview-ui/src/components/cloud/CloudView.tsx
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import { useEffect, useRef, useState } from "react"
import { VSCodeButton, VSCodeProgressRing, VSCodeTextField } from "@vscode/webview-ui-toolkit/react"
import { VSCodeProgressRing, VSCodeTextField } from "@vscode/webview-ui-toolkit/react"

import { type CloudUserInfo, type CloudOrganizationMembership, TelemetryEventName } from "@roo-code/types"

Expand Down Expand Up @@ -261,18 +261,12 @@ export const CloudView = ({ userInfo, isAuthenticated, cloudApiUrl, onDone, orga
</div>

<div className="flex flex-col gap-2 mt-4 pl-4">
<VSCodeButton
appearance="secondary"
onClick={handleVisitCloudWebsite}
className="w-full max-w-80">
<Button variant="secondary" onClick={handleVisitCloudWebsite} className="w-full max-w-80">
{t("cloud:visitCloudWebsite")}
</VSCodeButton>
<VSCodeButton
appearance="secondary"
onClick={handleLogoutClick}
className="w-full max-w-80">
</Button>
<Button variant="secondary" onClick={handleLogoutClick} className="w-full max-w-80">
{t("cloud:logOut")}
</VSCodeButton>
</Button>
</div>
</>
) : (
Expand All @@ -281,9 +275,9 @@ export const CloudView = ({ userInfo, isAuthenticated, cloudApiUrl, onDone, orga
<div className={cn(authInProgress && "opacity-50")}>{renderCloudBenefitsContent(t)}</div>

{!authInProgress && (
<VSCodeButton appearance="primary" onClick={handleConnectClick} className="w-full">
<Button variant="primary" onClick={handleConnectClick} className="w-full">
{t("cloud:connect")}
</VSCodeButton>
</Button>
)}

{/* Manual entry section */}
Expand Down
30 changes: 19 additions & 11 deletions webview-ui/src/components/common/VSCodeButtonLink.tsx
Original file line number Diff line number Diff line change
@@ -1,19 +1,27 @@
import React from "react"
import { VSCodeButton } from "@vscode/webview-ui-toolkit/react"
import { Button } from "@src/components/ui"

interface VSCodeButtonLinkProps {
href: string
children: React.ReactNode
appearance?: "primary" | "secondary"
[key: string]: any
}

export const VSCodeButtonLink = ({ href, children, ...props }: VSCodeButtonLinkProps) => (
<a
href={href}
style={{
textDecoration: "none",
color: "inherit",
}}>
<VSCodeButton {...props}>{children}</VSCodeButton>
</a>
)
export const VSCodeButtonLink = ({ href, children, appearance, ...props }: VSCodeButtonLinkProps) => {
// Map appearance to variant for the new Button component
const variant = appearance === "primary" ? "primary" : appearance === "secondary" ? "secondary" : undefined

return (
<a
href={href}
style={{
textDecoration: "none",
color: "inherit",
}}>
<Button variant={variant} {...props}>
{children}
</Button>
</a>
)
}
Loading