| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213 |
- import * as vscode from "vscode"
- import delay from "delay"
- import { ClineProvider } from "../core/webview/ClineProvider"
- import { ContextProxy } from "../core/config/ContextProxy"
- import { registerHumanRelayCallback, unregisterHumanRelayCallback, handleHumanRelayResponse } from "./humanRelay"
- import { handleNewTask } from "./handleTask"
- /**
- * Helper to get the visible ClineProvider instance or log if not found.
- */
- export function getVisibleProviderOrLog(outputChannel: vscode.OutputChannel): ClineProvider | undefined {
- const visibleProvider = ClineProvider.getVisibleInstance()
- if (!visibleProvider) {
- outputChannel.appendLine("Cannot find any visible Roo Code instances.")
- return undefined
- }
- return visibleProvider
- }
- // Store panel references in both modes
- let sidebarPanel: vscode.WebviewView | undefined = undefined
- let tabPanel: vscode.WebviewPanel | undefined = undefined
- /**
- * Get the currently active panel
- * @returns WebviewPanel或WebviewView
- */
- export function getPanel(): vscode.WebviewPanel | vscode.WebviewView | undefined {
- return tabPanel || sidebarPanel
- }
- /**
- * Set panel references
- */
- export function setPanel(
- newPanel: vscode.WebviewPanel | vscode.WebviewView | undefined,
- type: "sidebar" | "tab",
- ): void {
- if (type === "sidebar") {
- sidebarPanel = newPanel as vscode.WebviewView
- tabPanel = undefined
- } else {
- tabPanel = newPanel as vscode.WebviewPanel
- sidebarPanel = undefined
- }
- }
- export type RegisterCommandOptions = {
- context: vscode.ExtensionContext
- outputChannel: vscode.OutputChannel
- provider: ClineProvider
- }
- export const registerCommands = (options: RegisterCommandOptions) => {
- const { context } = options
- for (const [command, callback] of Object.entries(getCommandsMap(options))) {
- context.subscriptions.push(vscode.commands.registerCommand(command, callback))
- }
- }
- const getCommandsMap = ({ context, outputChannel, provider }: RegisterCommandOptions) => {
- return {
- "roo-cline.activationCompleted": () => {},
- "roo-cline.plusButtonClicked": async () => {
- const visibleProvider = getVisibleProviderOrLog(outputChannel)
- if (!visibleProvider) {
- return
- }
- await visibleProvider.removeClineFromStack()
- await visibleProvider.postStateToWebview()
- await visibleProvider.postMessageToWebview({ type: "action", action: "chatButtonClicked" })
- },
- "roo-cline.mcpButtonClicked": () => {
- const visibleProvider = getVisibleProviderOrLog(outputChannel)
- if (!visibleProvider) {
- return
- }
- visibleProvider.postMessageToWebview({ type: "action", action: "mcpButtonClicked" })
- },
- "roo-cline.promptsButtonClicked": () => {
- const visibleProvider = getVisibleProviderOrLog(outputChannel)
- if (!visibleProvider) {
- return
- }
- visibleProvider.postMessageToWebview({ type: "action", action: "promptsButtonClicked" })
- },
- "roo-cline.popoutButtonClicked": () => openClineInNewTab({ context, outputChannel }),
- "roo-cline.openInNewTab": () => openClineInNewTab({ context, outputChannel }),
- "roo-cline.settingsButtonClicked": () => {
- const visibleProvider = getVisibleProviderOrLog(outputChannel)
- if (!visibleProvider) {
- return
- }
- visibleProvider.postMessageToWebview({ type: "action", action: "settingsButtonClicked" })
- },
- "roo-cline.historyButtonClicked": () => {
- const visibleProvider = getVisibleProviderOrLog(outputChannel)
- if (!visibleProvider) {
- return
- }
- visibleProvider.postMessageToWebview({ type: "action", action: "historyButtonClicked" })
- },
- "roo-cline.helpButtonClicked": () => {
- vscode.env.openExternal(vscode.Uri.parse("https://docs.roocode.com"))
- },
- "roo-cline.showHumanRelayDialog": (params: { requestId: string; promptText: string }) => {
- const panel = getPanel()
- if (panel) {
- panel?.webview.postMessage({
- type: "showHumanRelayDialog",
- requestId: params.requestId,
- promptText: params.promptText,
- })
- }
- },
- "roo-cline.registerHumanRelayCallback": registerHumanRelayCallback,
- "roo-cline.unregisterHumanRelayCallback": unregisterHumanRelayCallback,
- "roo-cline.handleHumanRelayResponse": handleHumanRelayResponse,
- "roo-cline.newTask": handleNewTask,
- "roo-cline.setCustomStoragePath": async () => {
- const { promptForCustomStoragePath } = await import("../shared/storagePathManager")
- await promptForCustomStoragePath()
- },
- "roo-cline.focusInput": async () => {
- try {
- const panel = getPanel()
- if (!panel) {
- await vscode.commands.executeCommand("workbench.view.extension.roo-cline-ActivityBar")
- } else if (panel === tabPanel) {
- panel.reveal(vscode.ViewColumn.Active, false)
- } else if (panel === sidebarPanel) {
- await vscode.commands.executeCommand(`${ClineProvider.sideBarId}.focus`)
- provider.postMessageToWebview({ type: "action", action: "focusInput" })
- }
- } catch (error) {
- outputChannel.appendLine(`Error focusing input: ${error}`)
- }
- },
- "roo.acceptInput": () => {
- const visibleProvider = getVisibleProviderOrLog(outputChannel)
- if (!visibleProvider) {
- return
- }
- visibleProvider.postMessageToWebview({ type: "acceptInput" })
- },
- }
- }
- export const openClineInNewTab = async ({ context, outputChannel }: Omit<RegisterCommandOptions, "provider">) => {
- // (This example uses webviewProvider activation event which is necessary to
- // deserialize cached webview, but since we use retainContextWhenHidden, we
- // don't need to use that event).
- // https://github.com/microsoft/vscode-extension-samples/blob/main/webview-sample/src/extension.ts
- const contextProxy = await ContextProxy.getInstance(context)
- const tabProvider = new ClineProvider(context, outputChannel, "editor", contextProxy)
- const lastCol = Math.max(...vscode.window.visibleTextEditors.map((editor) => editor.viewColumn || 0))
- // Check if there are any visible text editors, otherwise open a new group
- // to the right.
- const hasVisibleEditors = vscode.window.visibleTextEditors.length > 0
- if (!hasVisibleEditors) {
- await vscode.commands.executeCommand("workbench.action.newGroupRight")
- }
- const targetCol = hasVisibleEditors ? Math.max(lastCol + 1, 1) : vscode.ViewColumn.Two
- const newPanel = vscode.window.createWebviewPanel(ClineProvider.tabPanelId, "Roo Code", targetCol, {
- enableScripts: true,
- retainContextWhenHidden: true,
- localResourceRoots: [context.extensionUri],
- })
- // Save as tab type panel.
- setPanel(newPanel, "tab")
- // TODO: Use better svg icon with light and dark variants (see
- // https://stackoverflow.com/questions/58365687/vscode-extension-iconpath).
- newPanel.iconPath = {
- light: vscode.Uri.joinPath(context.extensionUri, "assets", "icons", "panel_light.png"),
- dark: vscode.Uri.joinPath(context.extensionUri, "assets", "icons", "panel_dark.png"),
- }
- await tabProvider.resolveWebviewView(newPanel)
- // Handle panel closing events.
- newPanel.onDidDispose(() => {
- setPanel(undefined, "tab")
- })
- // Lock the editor group so clicking on files doesn't open them over the panel.
- await delay(100)
- await vscode.commands.executeCommand("workbench.action.lockEditorGroup")
- return tabProvider
- }
|