πŸ“„ File detail

utils/claudeInChrome/toolRendering.tsx

🧩 .tsxπŸ“ 262 linesπŸ’Ύ 33,862 bytesπŸ“ text
← Back to All Files

🎯 Use case

This file lives under β€œutils/”, which covers cross-cutting helpers (shell, tempfiles, settings, messages, process input, …). On the API surface it exposes ChromeToolName, renderChromeToolResultMessage, and getClaudeInChromeMCPToolOverrides β€” mainly functions, hooks, or classes. Dependencies touch React UI and @modelcontextprotocol. It composes internal code from components, ink, tools, utils, and format (relative imports).

Generated from folder role, exports, dependency roots, and inline comments β€” not hand-reviewed for every path.

🧠 Inline summary

import * as React from 'react'; import { MessageResponse } from '../../components/MessageResponse.js'; import { supportsHyperlinks } from '../../ink/supports-hyperlinks.js'; import { Link, Text } from '../../ink.js'; import { renderToolResultMessage as renderDefaultMCPToolResultMessage } from '../../tools/MCPTool/UI.js';

πŸ“€ Exports (heuristic)

  • ChromeToolName
  • renderChromeToolResultMessage
  • getClaudeInChromeMCPToolOverrides

πŸ“š External import roots

Package roots from from "…" (relative paths omitted).

  • react
  • @modelcontextprotocol

πŸ–₯️ Source preview

import * as React from 'react';
import { MessageResponse } from '../../components/MessageResponse.js';
import { supportsHyperlinks } from '../../ink/supports-hyperlinks.js';
import { Link, Text } from '../../ink.js';
import { renderToolResultMessage as renderDefaultMCPToolResultMessage } from '../../tools/MCPTool/UI.js';
import type { MCPToolResult } from '../../utils/mcpValidation.js';
import { truncateToWidth } from '../format.js';
import { trackClaudeInChromeTabId } from './common.js';
export type { Tool } from '@modelcontextprotocol/sdk/types.js';

/**
 * All tool names from BROWSER_TOOLS in @ant/claude-for-chrome-mcp.
 * Keep in sync with the package's BROWSER_TOOLS array.
 */
export type ChromeToolName = 'javascript_tool' | 'read_page' | 'find' | 'form_input' | 'computer' | 'navigate' | 'resize_window' | 'gif_creator' | 'upload_image' | 'get_page_text' | 'tabs_context_mcp' | 'tabs_create_mcp' | 'update_plan' | 'read_console_messages' | 'read_network_requests' | 'shortcuts_list' | 'shortcuts_execute';
const CHROME_EXTENSION_FOCUS_TAB_URL_BASE = 'https://clau.de/chrome/tab/';
function renderChromeToolUseMessage(input: Record<string, unknown>, toolName: ChromeToolName, verbose: boolean): React.ReactNode {
  const tabId = input.tabId;
  if (typeof tabId === 'number') {
    trackClaudeInChromeTabId(tabId);
  }

  // Build secondary info based on tool type and input
  const secondaryInfo: string[] = [];
  switch (toolName) {
    case 'navigate':
      if (typeof input.url === 'string') {
        try {
          const url = new URL(input.url);
          secondaryInfo.push(url.hostname);
        } catch {
          secondaryInfo.push(truncateToWidth(input.url, 30));
        }
      }
      break;
    case 'find':
      if (typeof input.query === 'string') {
        secondaryInfo.push(`pattern: ${truncateToWidth(input.query, 30)}`);
      }
      break;
    case 'computer':
      if (typeof input.action === 'string') {
        const action = input.action;
        if (action === 'left_click' || action === 'right_click' || action === 'double_click' || action === 'middle_click') {
          if (typeof input.ref === 'string') {
            secondaryInfo.push(`${action} on ${input.ref}`);
          } else if (Array.isArray(input.coordinate)) {
            secondaryInfo.push(`${action} at (${input.coordinate.join(', ')})`);
          } else {
            secondaryInfo.push(action);
          }
        } else if (action === 'type' && typeof input.text === 'string') {
          secondaryInfo.push(`type "${truncateToWidth(input.text, 15)}"`);
        } else if (action === 'key' && typeof input.text === 'string') {
          secondaryInfo.push(`key ${input.text}`);
        } else if (action === 'scroll' && typeof input.scroll_direction === 'string') {
          secondaryInfo.push(`scroll ${input.scroll_direction}`);
        } else if (action === 'wait' && typeof input.duration === 'number') {
          secondaryInfo.push(`wait ${input.duration}s`);
        } else if (action === 'left_click_drag') {
          secondaryInfo.push('drag');
        } else {
          secondaryInfo.push(action);
        }
      }
      break;
    case 'gif_creator':
      if (typeof input.action === 'string') {
        secondaryInfo.push(`${input.action}`);
      }
      break;
    case 'resize_window':
      if (typeof input.width === 'number' && typeof input.height === 'number') {
        secondaryInfo.push(`${input.width}x${input.height}`);
      }
      break;
    case 'read_console_messages':
      if (typeof input.pattern === 'string') {
        secondaryInfo.push(`pattern: ${truncateToWidth(input.pattern, 20)}`);
      }
      if (input.onlyErrors === true) {
        secondaryInfo.push('errors only');
      }
      break;
    case 'read_network_requests':
      if (typeof input.urlPattern === 'string') {
        secondaryInfo.push(`pattern: ${truncateToWidth(input.urlPattern, 20)}`);
      }
      break;
    case 'shortcuts_execute':
      if (typeof input.shortcutId === 'string') {
        secondaryInfo.push(`shortcut_id: ${input.shortcutId}`);
      }
      break;
    case 'javascript_tool':
      // In verbose mode, show the full code
      if (verbose && typeof input.text === 'string') {
        return input.text;
      }
      // In non-verbose mode, return empty string to preserve View Tab layout
      return '';
    case 'tabs_create_mcp':
    case 'tabs_context_mcp':
    case 'form_input':
    case 'shortcuts_list':
    case 'read_page':
    case 'upload_image':
    case 'get_page_text':
    case 'update_plan':
      // These tools don't have meaningful secondary info to show inline.
      // Return empty string (not null) to ensure tool header still renders.
      return '';
  }
  return secondaryInfo.join(', ') || null;
}

/**
 * Renders a clickable "View Tab" link for Claude in Chrome MCP tools.
 * Returns null if:
 * - The tool is not a Claude in Chrome MCP tool
 * - The input doesn't have a valid tabId
 * - Hyperlinks are not supported
 */
function renderChromeViewTabLink(input: unknown): React.ReactNode {
  if (!supportsHyperlinks()) {
    return null;
  }
  if (typeof input !== 'object' || input === null || !('tabId' in input)) {
    return null;
  }
  const tabId = typeof input.tabId === 'number' ? input.tabId : typeof input.tabId === 'string' ? parseInt(input.tabId, 10) : NaN;
  if (isNaN(tabId)) {
    return null;
  }
  const linkUrl = `${CHROME_EXTENSION_FOCUS_TAB_URL_BASE}${tabId}`;
  return <Text>
      {' '}
      <Link url={linkUrl}>
        <Text color="subtle">[View Tab]</Text>
      </Link>
    </Text>;
}

/**
 * Custom tool result message rendering for claude-in-chrome tools.
 * Shows a brief summary for successful results. Errors are handled by
 * the default renderToolUseErrorMessage when is_error is set.
 */
export function renderChromeToolResultMessage(output: MCPToolResult, toolName: ChromeToolName, verbose: boolean): React.ReactNode {
  if (verbose) {
    return renderDefaultMCPToolResultMessage(output, [], {
      verbose
    });
  }
  let summary: string | null = null;
  switch (toolName) {
    case 'navigate':
      summary = 'Navigation completed';
      break;
    case 'tabs_create_mcp':
      summary = 'Tab created';
      break;
    case 'tabs_context_mcp':
      summary = 'Tabs read';
      break;
    case 'form_input':
      summary = 'Input completed';
      break;
    case 'computer':
      summary = 'Action completed';
      break;
    case 'resize_window':
      summary = 'Window resized';
      break;
    case 'find':
      summary = 'Search completed';
      break;
    case 'gif_creator':
      summary = 'GIF action completed';
      break;
    case 'read_console_messages':
      summary = 'Console messages retrieved';
      break;
    case 'read_network_requests':
      summary = 'Network requests retrieved';
      break;
    case 'shortcuts_list':
      summary = 'Shortcuts retrieved';
      break;
    case 'shortcuts_execute':
      summary = 'Shortcut executed';
      break;
    case 'javascript_tool':
      summary = 'Script executed';
      break;
    case 'read_page':
      summary = 'Page read';
      break;
    case 'upload_image':
      summary = 'Image uploaded';
      break;
    case 'get_page_text':
      summary = 'Page text retrieved';
      break;
    case 'update_plan':
      summary = 'Plan updated';
      break;
  }
  if (summary) {
    return <MessageResponse height={1}>
        <Text dimColor>{summary}</Text>
      </MessageResponse>;
  }
  return null;
}

/**
 * Returns tool method overrides for Claude in Chrome MCP tools. Use this to customize
 * rendering for chrome tools in a single spread operation.
 */
export function getClaudeInChromeMCPToolOverrides(toolName: string): {
  userFacingName: (input?: Record<string, unknown>) => string;
  renderToolUseMessage: (input: Record<string, unknown>, options: {
    verbose: boolean;
  }) => React.ReactNode;
  renderToolUseTag: (input: Partial<Record<string, unknown>>) => React.ReactNode;
  renderToolResultMessage: (output: string | MCPToolResult, progressMessagesForMessage: unknown[], options: {
    verbose: boolean;
  }) => React.ReactNode;
} {
  return {
    userFacingName(_input?: Record<string, unknown>) {
      // Trim the _mcp postfix that show up in some of the tool names
      const displayName = toolName.replace(/_mcp$/, '');
      return `Claude in Chrome[${displayName}]`;
    },
    renderToolUseMessage(input: Record<string, unknown>, {
      verbose
    }: {
      verbose: boolean;
    }): React.ReactNode {
      return renderChromeToolUseMessage(input, toolName as ChromeToolName, verbose);
    },
    renderToolUseTag(input: Partial<Record<string, unknown>>): React.ReactNode {
      return renderChromeViewTabLink(input);
    },
    renderToolResultMessage(output: string | MCPToolResult, _progressMessagesForMessage: unknown[], {
      verbose
    }: {
      verbose: boolean;
    }): React.ReactNode {
      if (!isMCPToolResult(output)) {
        return null;
      }
      return renderChromeToolResultMessage(output, toolName as ChromeToolName, verbose);
    }
  };
}
function isMCPToolResult(output: string | MCPToolResult): output is MCPToolResult {
  return typeof output === 'object' && output !== null;
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["React","MessageResponse","supportsHyperlinks","Link","Text","renderToolResultMessage","renderDefaultMCPToolResultMessage","MCPToolResult","truncateToWidth","trackClaudeInChromeTabId","Tool","ChromeToolName","CHROME_EXTENSION_FOCUS_TAB_URL_BASE","renderChromeToolUseMessage","input","Record","toolName","verbose","ReactNode","tabId","secondaryInfo","url","URL","push","hostname","query","action","ref","Array","isArray","coordinate","join","text","scroll_direction","duration","width","height","pattern","onlyErrors","urlPattern","shortcutId","renderChromeViewTabLink","parseInt","NaN","isNaN","linkUrl","renderChromeToolResultMessage","output","summary","getClaudeInChromeMCPToolOverrides","userFacingName","renderToolUseMessage","options","renderToolUseTag","Partial","progressMessagesForMessage","_input","displayName","replace","_progressMessagesForMessage","isMCPToolResult"],"sources":["toolRendering.tsx"],"sourcesContent":["import * as React from 'react'\nimport { MessageResponse } from '../../components/MessageResponse.js'\nimport { supportsHyperlinks } from '../../ink/supports-hyperlinks.js'\nimport { Link, Text } from '../../ink.js'\nimport { renderToolResultMessage as renderDefaultMCPToolResultMessage } from '../../tools/MCPTool/UI.js'\nimport type { MCPToolResult } from '../../utils/mcpValidation.js'\nimport { truncateToWidth } from '../format.js'\nimport { trackClaudeInChromeTabId } from './common.js'\n\nexport type { Tool } from '@modelcontextprotocol/sdk/types.js'\n\n/**\n * All tool names from BROWSER_TOOLS in @ant/claude-for-chrome-mcp.\n * Keep in sync with the package's BROWSER_TOOLS array.\n */\nexport type ChromeToolName =\n  | 'javascript_tool'\n  | 'read_page'\n  | 'find'\n  | 'form_input'\n  | 'computer'\n  | 'navigate'\n  | 'resize_window'\n  | 'gif_creator'\n  | 'upload_image'\n  | 'get_page_text'\n  | 'tabs_context_mcp'\n  | 'tabs_create_mcp'\n  | 'update_plan'\n  | 'read_console_messages'\n  | 'read_network_requests'\n  | 'shortcuts_list'\n  | 'shortcuts_execute'\n\nconst CHROME_EXTENSION_FOCUS_TAB_URL_BASE = 'https://clau.de/chrome/tab/'\n\nfunction renderChromeToolUseMessage(\n  input: Record<string, unknown>,\n  toolName: ChromeToolName,\n  verbose: boolean,\n): React.ReactNode {\n  const tabId = input.tabId\n  if (typeof tabId === 'number') {\n    trackClaudeInChromeTabId(tabId)\n  }\n\n  // Build secondary info based on tool type and input\n  const secondaryInfo: string[] = []\n\n  switch (toolName) {\n    case 'navigate':\n      if (typeof input.url === 'string') {\n        try {\n          const url = new URL(input.url)\n          secondaryInfo.push(url.hostname)\n        } catch {\n          secondaryInfo.push(truncateToWidth(input.url, 30))\n        }\n      }\n      break\n\n    case 'find':\n      if (typeof input.query === 'string') {\n        secondaryInfo.push(`pattern: ${truncateToWidth(input.query, 30)}`)\n      }\n      break\n\n    case 'computer':\n      if (typeof input.action === 'string') {\n        const action = input.action\n        if (\n          action === 'left_click' ||\n          action === 'right_click' ||\n          action === 'double_click' ||\n          action === 'middle_click'\n        ) {\n          if (typeof input.ref === 'string') {\n            secondaryInfo.push(`${action} on ${input.ref}`)\n          } else if (Array.isArray(input.coordinate)) {\n            secondaryInfo.push(`${action} at (${input.coordinate.join(', ')})`)\n          } else {\n            secondaryInfo.push(action)\n          }\n        } else if (action === 'type' && typeof input.text === 'string') {\n          secondaryInfo.push(`type \"${truncateToWidth(input.text, 15)}\"`)\n        } else if (action === 'key' && typeof input.text === 'string') {\n          secondaryInfo.push(`key ${input.text}`)\n        } else if (\n          action === 'scroll' &&\n          typeof input.scroll_direction === 'string'\n        ) {\n          secondaryInfo.push(`scroll ${input.scroll_direction}`)\n        } else if (action === 'wait' && typeof input.duration === 'number') {\n          secondaryInfo.push(`wait ${input.duration}s`)\n        } else if (action === 'left_click_drag') {\n          secondaryInfo.push('drag')\n        } else {\n          secondaryInfo.push(action)\n        }\n      }\n      break\n\n    case 'gif_creator':\n      if (typeof input.action === 'string') {\n        secondaryInfo.push(`${input.action}`)\n      }\n      break\n\n    case 'resize_window':\n      if (typeof input.width === 'number' && typeof input.height === 'number') {\n        secondaryInfo.push(`${input.width}x${input.height}`)\n      }\n      break\n\n    case 'read_console_messages':\n      if (typeof input.pattern === 'string') {\n        secondaryInfo.push(`pattern: ${truncateToWidth(input.pattern, 20)}`)\n      }\n      if (input.onlyErrors === true) {\n        secondaryInfo.push('errors only')\n      }\n      break\n\n    case 'read_network_requests':\n      if (typeof input.urlPattern === 'string') {\n        secondaryInfo.push(`pattern: ${truncateToWidth(input.urlPattern, 20)}`)\n      }\n      break\n\n    case 'shortcuts_execute':\n      if (typeof input.shortcutId === 'string') {\n        secondaryInfo.push(`shortcut_id: ${input.shortcutId}`)\n      }\n      break\n\n    case 'javascript_tool':\n      // In verbose mode, show the full code\n      if (verbose && typeof input.text === 'string') {\n        return input.text\n      }\n      // In non-verbose mode, return empty string to preserve View Tab layout\n      return ''\n\n    case 'tabs_create_mcp':\n    case 'tabs_context_mcp':\n    case 'form_input':\n    case 'shortcuts_list':\n    case 'read_page':\n    case 'upload_image':\n    case 'get_page_text':\n    case 'update_plan':\n      // These tools don't have meaningful secondary info to show inline.\n      // Return empty string (not null) to ensure tool header still renders.\n      return ''\n  }\n\n  return secondaryInfo.join(', ') || null\n}\n\n/**\n * Renders a clickable \"View Tab\" link for Claude in Chrome MCP tools.\n * Returns null if:\n * - The tool is not a Claude in Chrome MCP tool\n * - The input doesn't have a valid tabId\n * - Hyperlinks are not supported\n */\nfunction renderChromeViewTabLink(input: unknown): React.ReactNode {\n  if (!supportsHyperlinks()) {\n    return null\n  }\n  if (typeof input !== 'object' || input === null || !('tabId' in input)) {\n    return null\n  }\n  const tabId =\n    typeof input.tabId === 'number'\n      ? input.tabId\n      : typeof input.tabId === 'string'\n        ? parseInt(input.tabId, 10)\n        : NaN\n  if (isNaN(tabId)) {\n    return null\n  }\n  const linkUrl = `${CHROME_EXTENSION_FOCUS_TAB_URL_BASE}${tabId}`\n  return (\n    <Text>\n      {' '}\n      <Link url={linkUrl}>\n        <Text color=\"subtle\">[View Tab]</Text>\n      </Link>\n    </Text>\n  )\n}\n\n/**\n * Custom tool result message rendering for claude-in-chrome tools.\n * Shows a brief summary for successful results. Errors are handled by\n * the default renderToolUseErrorMessage when is_error is set.\n */\nexport function renderChromeToolResultMessage(\n  output: MCPToolResult,\n  toolName: ChromeToolName,\n  verbose: boolean,\n): React.ReactNode {\n  if (verbose) {\n    return renderDefaultMCPToolResultMessage(output, [], { verbose })\n  }\n\n  let summary: string | null = null\n  switch (toolName) {\n    case 'navigate':\n      summary = 'Navigation completed'\n      break\n    case 'tabs_create_mcp':\n      summary = 'Tab created'\n      break\n    case 'tabs_context_mcp':\n      summary = 'Tabs read'\n      break\n    case 'form_input':\n      summary = 'Input completed'\n      break\n    case 'computer':\n      summary = 'Action completed'\n      break\n    case 'resize_window':\n      summary = 'Window resized'\n      break\n    case 'find':\n      summary = 'Search completed'\n      break\n    case 'gif_creator':\n      summary = 'GIF action completed'\n      break\n    case 'read_console_messages':\n      summary = 'Console messages retrieved'\n      break\n    case 'read_network_requests':\n      summary = 'Network requests retrieved'\n      break\n    case 'shortcuts_list':\n      summary = 'Shortcuts retrieved'\n      break\n    case 'shortcuts_execute':\n      summary = 'Shortcut executed'\n      break\n    case 'javascript_tool':\n      summary = 'Script executed'\n      break\n    case 'read_page':\n      summary = 'Page read'\n      break\n    case 'upload_image':\n      summary = 'Image uploaded'\n      break\n    case 'get_page_text':\n      summary = 'Page text retrieved'\n      break\n    case 'update_plan':\n      summary = 'Plan updated'\n      break\n  }\n\n  if (summary) {\n    return (\n      <MessageResponse height={1}>\n        <Text dimColor>{summary}</Text>\n      </MessageResponse>\n    )\n  }\n\n  return null\n}\n\n/**\n * Returns tool method overrides for Claude in Chrome MCP tools. Use this to customize\n * rendering for chrome tools in a single spread operation.\n */\nexport function getClaudeInChromeMCPToolOverrides(toolName: string): {\n  userFacingName: (input?: Record<string, unknown>) => string\n  renderToolUseMessage: (\n    input: Record<string, unknown>,\n    options: { verbose: boolean },\n  ) => React.ReactNode\n  renderToolUseTag: (input: Partial<Record<string, unknown>>) => React.ReactNode\n  renderToolResultMessage: (\n    output: string | MCPToolResult,\n    progressMessagesForMessage: unknown[],\n    options: { verbose: boolean },\n  ) => React.ReactNode\n} {\n  return {\n    userFacingName(_input?: Record<string, unknown>) {\n      // Trim the _mcp postfix that show up in some of the tool names\n      const displayName = toolName.replace(/_mcp$/, '')\n      return `Claude in Chrome[${displayName}]`\n    },\n    renderToolUseMessage(\n      input: Record<string, unknown>,\n      { verbose }: { verbose: boolean },\n    ): React.ReactNode {\n      return renderChromeToolUseMessage(\n        input,\n        toolName as ChromeToolName,\n        verbose,\n      )\n    },\n    renderToolUseTag(input: Partial<Record<string, unknown>>): React.ReactNode {\n      return renderChromeViewTabLink(input)\n    },\n    renderToolResultMessage(\n      output: string | MCPToolResult,\n      _progressMessagesForMessage: unknown[],\n      { verbose }: { verbose: boolean },\n    ): React.ReactNode {\n      if (!isMCPToolResult(output)) {\n        return null\n      }\n      return renderChromeToolResultMessage(\n        output,\n        toolName as ChromeToolName,\n        verbose,\n      )\n    },\n  }\n}\n\nfunction isMCPToolResult(\n  output: string | MCPToolResult,\n): output is MCPToolResult {\n  return typeof output === 'object' && output !== null\n}\n"],"mappings":"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","ignoreList":[]}