@redocly/theme 0.59.0-next.1 → 0.59.0-next.11

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (275) hide show
  1. package/LICENSE +7 -1
  2. package/lib/components/Accordion/Accordion.js +17 -7
  3. package/lib/components/Accordion/AccordionBody.js +17 -7
  4. package/lib/components/Admonition/Admonition.js +17 -7
  5. package/lib/components/Badge/Badge.js +17 -7
  6. package/lib/components/Breadcrumbs/Breadcrumb.js +17 -7
  7. package/lib/components/Breadcrumbs/BreadcrumbDropdown.js +17 -7
  8. package/lib/components/Button/Button.js +17 -7
  9. package/lib/components/Buttons/AIAssistantButton.js +23 -9
  10. package/lib/components/Buttons/ConnectMCPButton.d.ts +8 -0
  11. package/lib/components/Buttons/ConnectMCPButton.js +145 -0
  12. package/lib/components/Buttons/CopyButton.js +17 -7
  13. package/lib/components/Buttons/variables.d.ts +1 -0
  14. package/lib/components/Buttons/variables.js +42 -2
  15. package/lib/components/Catalog/Catalog.d.ts +6 -0
  16. package/lib/components/Catalog/Catalog.js +7 -6
  17. package/lib/components/Catalog/CatalogEntities.js +17 -7
  18. package/lib/components/Catalog/CatalogEntity/CatalogEntityGraph/CatalogEntityRelationsGraph.js +17 -7
  19. package/lib/components/Catalog/CatalogEntity/CatalogEntityGraph/CatalogEntityRelationsGraph.lazy.js +17 -7
  20. package/lib/components/Catalog/CatalogEntity/CatalogEntityInfoBar.js +1 -0
  21. package/lib/components/Catalog/CatalogEntity/CatalogEntityMetadata.js +17 -7
  22. package/lib/components/Catalog/CatalogEntity/CatalogEntityRelations/CatalogEntityApiDescriptionRelations.js +1 -1
  23. package/lib/components/Catalog/CatalogEntity/CatalogEntityRelations/CatalogEntityRelations.js +17 -7
  24. package/lib/components/Catalog/CatalogEntity/CatalogEntityRelations/CatalogEntityTeamRelations.js +1 -1
  25. package/lib/components/Catalog/CatalogEntity/CatalogEntitySchema.js +17 -7
  26. package/lib/components/Catalog/CatalogEntityIcon.js +2 -1
  27. package/lib/components/Catalog/CatalogFilter/CatalogFilter.js +4 -0
  28. package/lib/components/Catalog/CatalogFilter/CatalogFilterCheckboxes.js +17 -7
  29. package/lib/components/Catalog/CatalogFilter/CatalogFilterContent.js +17 -7
  30. package/lib/components/Catalog/CatalogFilter/CatalogFilterDateRange.js +17 -7
  31. package/lib/components/Catalog/CatalogFilter/CatalogFilterSelect.js +17 -7
  32. package/lib/components/Catalog/CatalogSortButton.js +17 -7
  33. package/lib/components/Catalog/CatalogTableView/CatalogTableHeaderCell.js +17 -7
  34. package/lib/components/Catalog/CatalogViewModeToggle.js +17 -7
  35. package/lib/components/Catalog/variables.js +1 -1
  36. package/lib/components/CatalogClassic/CatalogClassicActions.js +17 -7
  37. package/lib/components/CatalogClassic/CatalogClassicCard.js +17 -7
  38. package/lib/components/CatalogClassic/CatalogClassicHighlight.js +17 -7
  39. package/lib/components/CatalogClassic/CatalogClassicVirtualizedGroups.js +17 -7
  40. package/lib/components/CodeBlock/CodeBlock.js +17 -7
  41. package/lib/components/CodeBlock/CodeBlockContainer.js +17 -7
  42. package/lib/components/CodeBlock/CodeBlockTabs.js +17 -7
  43. package/lib/components/Dropdown/Dropdown.d.ts +16 -2
  44. package/lib/components/Dropdown/Dropdown.js +22 -12
  45. package/lib/components/Dropdown/DropdownMenuItem.js +17 -7
  46. package/lib/components/Feedback/Comment.js +17 -7
  47. package/lib/components/Feedback/Feedback.js +17 -7
  48. package/lib/components/Feedback/Mood.js +17 -7
  49. package/lib/components/Feedback/Rating.js +17 -7
  50. package/lib/components/Feedback/Reasons.js +17 -7
  51. package/lib/components/Feedback/Scale.js +17 -7
  52. package/lib/components/Feedback/Sentiment.js +17 -7
  53. package/lib/components/Feedback/Stars.js +17 -7
  54. package/lib/components/Filter/FilterContent.js +17 -7
  55. package/lib/components/Filter/FilterInput.js +17 -7
  56. package/lib/components/Image/Image.js +17 -7
  57. package/lib/components/JsonViewer/JsonViewer.js +17 -7
  58. package/lib/components/JsonViewer/helpers.js +17 -7
  59. package/lib/components/LastUpdated/LastUpdated.js +17 -7
  60. package/lib/components/Link/Link.js +17 -7
  61. package/lib/components/Markdown/Markdown.js +17 -7
  62. package/lib/components/Marker/Marker.js +17 -7
  63. package/lib/components/Menu/MenuContainer.js +17 -7
  64. package/lib/components/Menu/MenuItem.js +18 -8
  65. package/lib/components/Menu/MenuMobile.js +17 -7
  66. package/lib/components/Navbar/NavbarItem.js +3 -3
  67. package/lib/components/PageActions/PageActions.js +21 -8
  68. package/lib/components/PageActions/variables.js +2 -0
  69. package/lib/components/PageNavigation/NextButton.js +17 -7
  70. package/lib/components/Panel/Panel.js +17 -7
  71. package/lib/components/Panel/PanelBody.js +17 -7
  72. package/lib/components/Search/FilterFields/SearchFilterFieldSelect.js +17 -7
  73. package/lib/components/Search/FilterFields/SearchFilterFieldTags.js +1 -2
  74. package/lib/components/Search/SearchAiActionButtons.d.ts +10 -0
  75. package/lib/components/Search/SearchAiActionButtons.js +43 -0
  76. package/lib/components/Search/SearchAiConversationInput.d.ts +3 -1
  77. package/lib/components/Search/SearchAiConversationInput.js +56 -14
  78. package/lib/components/Search/SearchAiDialog.d.ts +3 -6
  79. package/lib/components/Search/SearchAiDialog.js +37 -16
  80. package/lib/components/Search/SearchAiMessage.d.ts +9 -5
  81. package/lib/components/Search/SearchAiMessage.js +146 -22
  82. package/lib/components/Search/SearchAiNegativeFeedbackForm.d.ts +8 -0
  83. package/lib/components/Search/SearchAiNegativeFeedbackForm.js +169 -0
  84. package/lib/components/Search/SearchDialog.js +53 -12
  85. package/lib/components/Search/SearchFilter.js +17 -7
  86. package/lib/components/Search/SearchGroups.js +19 -9
  87. package/lib/components/Search/SearchHighlight.js +17 -7
  88. package/lib/components/Search/SearchItem.js +17 -7
  89. package/lib/components/Search/SearchRecent.js +17 -7
  90. package/lib/components/Search/SearchShortcut.js +17 -7
  91. package/lib/components/Search/SearchSuggestedPages.js +17 -7
  92. package/lib/components/Search/SearchTrigger.js +17 -7
  93. package/lib/components/Search/variables.js +36 -64
  94. package/lib/components/Segmented/Segmented.d.ts +1 -8
  95. package/lib/components/Segmented/Segmented.js +20 -8
  96. package/lib/components/Select/Select.js +17 -7
  97. package/lib/components/Select/SelectInput.js +18 -8
  98. package/lib/components/Sidebar/Sidebar.js +17 -7
  99. package/lib/components/SidebarActions/styled.js +17 -7
  100. package/lib/components/SkipContent/SkipContent.js +17 -7
  101. package/lib/components/Switch/Switch.js +17 -7
  102. package/lib/components/TableOfContent/TableOfContent.js +17 -7
  103. package/lib/components/Tag/Tag.d.ts +2 -1
  104. package/lib/components/Tag/Tag.js +67 -18
  105. package/lib/components/Tag/variables.dark.js +135 -36
  106. package/lib/components/Tag/variables.js +78 -61
  107. package/lib/components/Tooltip/Tooltip.js +17 -7
  108. package/lib/components/VersionPicker/VersionPicker.js +17 -7
  109. package/lib/core/constants/index.d.ts +1 -0
  110. package/lib/core/constants/index.js +1 -0
  111. package/lib/core/constants/mcp.d.ts +1 -0
  112. package/lib/core/constants/mcp.js +5 -0
  113. package/lib/core/constants/search.d.ts +5 -4
  114. package/lib/core/constants/search.js +4 -5
  115. package/lib/core/contexts/CodeSnippetContext.js +17 -7
  116. package/lib/core/hooks/index.d.ts +3 -0
  117. package/lib/core/hooks/index.js +3 -0
  118. package/lib/core/hooks/menu/use-nested-menu.js +1 -1
  119. package/lib/core/hooks/search/use-feedback-tooltip.d.ts +6 -0
  120. package/lib/core/hooks/search/use-feedback-tooltip.js +26 -0
  121. package/lib/core/hooks/use-connect-mcp-button.d.ts +13 -0
  122. package/lib/core/hooks/use-connect-mcp-button.js +50 -0
  123. package/lib/core/hooks/use-mcp-config.d.ts +9 -0
  124. package/lib/core/hooks/use-mcp-config.js +27 -0
  125. package/lib/core/hooks/use-page-actions.d.ts +1 -1
  126. package/lib/core/hooks/use-page-actions.js +98 -95
  127. package/lib/core/hooks/use-product-picker.js +2 -1
  128. package/lib/core/hooks/use-tabs.d.ts +3 -2
  129. package/lib/core/hooks/use-tabs.js +115 -57
  130. package/lib/core/hooks/use-telemetry-fallback.d.ts +10 -8
  131. package/lib/core/hooks/use-telemetry-fallback.js +10 -8
  132. package/lib/core/openapi/index.d.ts +1 -0
  133. package/lib/core/styles/dark.js +4 -0
  134. package/lib/core/styles/global.js +5 -0
  135. package/lib/core/templates/Markdown.js +17 -7
  136. package/lib/core/types/hooks.d.ts +6 -3
  137. package/lib/core/types/index.d.ts +1 -0
  138. package/lib/core/types/index.js +1 -0
  139. package/lib/core/types/l10n.d.ts +1 -1
  140. package/lib/core/types/mcp.d.ts +6 -0
  141. package/lib/core/types/mcp.js +3 -0
  142. package/lib/core/types/search.d.ts +11 -4
  143. package/lib/core/types/search.js +6 -0
  144. package/lib/core/types/segmented.d.ts +12 -0
  145. package/lib/core/types/segmented.js +3 -0
  146. package/lib/core/utils/download-code-walkthrough.js +17 -7
  147. package/lib/core/utils/frontmatter-translate.d.ts +6 -0
  148. package/lib/core/utils/frontmatter-translate.js +14 -0
  149. package/lib/core/utils/get-file-icon.js +17 -7
  150. package/lib/core/utils/index.d.ts +2 -0
  151. package/lib/core/utils/index.js +2 -0
  152. package/lib/core/utils/mcp.d.ts +2 -0
  153. package/lib/core/utils/mcp.js +31 -0
  154. package/lib/icons/AiStarsGradientIcon/AiStarsGradientIcon.js +44 -4
  155. package/lib/icons/AiStarsIcon/AiStarsIcon.js +11 -2
  156. package/lib/icons/ConnectIcon/ConnectIcon.d.ts +9 -0
  157. package/lib/icons/ConnectIcon/ConnectIcon.js +17 -0
  158. package/lib/icons/CubeIcon/CubeIcon.d.ts +9 -0
  159. package/lib/icons/CubeIcon/CubeIcon.js +17 -0
  160. package/lib/icons/GenericIcon/GenericIcon.js +17 -7
  161. package/lib/icons/HashtagIcon/HashtagIcon.d.ts +9 -0
  162. package/lib/icons/HashtagIcon/HashtagIcon.js +22 -0
  163. package/lib/icons/RedoclyIcon/RedoclyIcon.js +4 -7
  164. package/lib/icons/Spinner/Spinner.js +17 -7
  165. package/lib/icons/ThumbDownFilledIcon/ThumbDownFilledIcon.d.ts +9 -0
  166. package/lib/icons/ThumbDownFilledIcon/ThumbDownFilledIcon.js +34 -0
  167. package/lib/icons/ThumbUpFilledIcon/ThumbUpFilledIcon.d.ts +9 -0
  168. package/lib/icons/ThumbUpFilledIcon/ThumbUpFilledIcon.js +34 -0
  169. package/lib/icons/VSCodeIcon/VSCodeIcon.d.ts +9 -0
  170. package/lib/icons/VSCodeIcon/VSCodeIcon.js +17 -0
  171. package/lib/index.d.ts +1 -0
  172. package/lib/index.js +18 -7
  173. package/lib/layouts/OIDCForbidden.js +17 -7
  174. package/lib/layouts/ThreePanelLayout.js +17 -7
  175. package/lib/markdoc/components/Cards/Card.js +1 -28
  176. package/lib/markdoc/components/Cards/Cards.js +17 -7
  177. package/lib/markdoc/components/CodeGroup/CodeGroup.js +17 -7
  178. package/lib/markdoc/components/CodeWalkthrough/CodeContainer.js +17 -7
  179. package/lib/markdoc/components/CodeWalkthrough/CodePanel.js +17 -7
  180. package/lib/markdoc/components/CodeWalkthrough/CodePanelHeader.js +17 -7
  181. package/lib/markdoc/components/CodeWalkthrough/CodePanelPreview.js +17 -7
  182. package/lib/markdoc/components/CodeWalkthrough/CodePanelToolbar.js +17 -7
  183. package/lib/markdoc/components/CodeWalkthrough/CodeStep.js +17 -7
  184. package/lib/markdoc/components/CodeWalkthrough/CodeToggle.js +17 -7
  185. package/lib/markdoc/components/CodeWalkthrough/CodeWalkthrough.js +17 -7
  186. package/lib/markdoc/components/CodeWalkthrough/Input.js +17 -7
  187. package/lib/markdoc/components/ConnectMCP/ConnectMCP.d.ts +8 -0
  188. package/lib/markdoc/components/ConnectMCP/ConnectMCP.js +19 -0
  189. package/lib/markdoc/components/Heading/Heading.js +17 -7
  190. package/lib/markdoc/components/HtmlBlock/HtmlBlock.js +17 -7
  191. package/lib/markdoc/components/InlineSvg/InlineSvg.js +17 -7
  192. package/lib/markdoc/components/MarkdocExample/MarkdocExample.js +17 -7
  193. package/lib/markdoc/components/Tabs/TabList.d.ts +3 -1
  194. package/lib/markdoc/components/Tabs/TabList.js +214 -54
  195. package/lib/markdoc/components/Tabs/Tabs.d.ts +2 -1
  196. package/lib/markdoc/components/Tabs/Tabs.js +74 -19
  197. package/lib/markdoc/components/default.d.ts +1 -0
  198. package/lib/markdoc/components/default.js +1 -0
  199. package/lib/markdoc/default.d.ts +110 -1
  200. package/lib/markdoc/default.js +19 -7
  201. package/lib/markdoc/tags/card.js +0 -1
  202. package/lib/markdoc/tags/connect-mcp.d.ts +2 -0
  203. package/lib/markdoc/tags/connect-mcp.js +27 -0
  204. package/package.json +8 -8
  205. package/src/components/Buttons/AIAssistantButton.tsx +6 -2
  206. package/src/components/Buttons/ConnectMCPButton.tsx +180 -0
  207. package/src/components/Buttons/variables.ts +42 -1
  208. package/src/components/Catalog/Catalog.tsx +15 -4
  209. package/src/components/Catalog/CatalogEntity/CatalogEntityInfoBar.tsx +1 -0
  210. package/src/components/Catalog/CatalogEntity/CatalogEntityRelations/CatalogEntityApiDescriptionRelations.tsx +1 -1
  211. package/src/components/Catalog/CatalogEntity/CatalogEntityRelations/CatalogEntityTeamRelations.tsx +1 -1
  212. package/src/components/Catalog/CatalogEntityIcon.tsx +2 -1
  213. package/src/components/Catalog/CatalogFilter/CatalogFilter.tsx +5 -0
  214. package/src/components/Catalog/variables.ts +1 -1
  215. package/src/components/Dropdown/Dropdown.tsx +84 -79
  216. package/src/components/Menu/MenuItem.tsx +1 -0
  217. package/src/components/Navbar/NavbarItem.tsx +6 -5
  218. package/src/components/PageActions/PageActions.tsx +5 -1
  219. package/src/components/PageActions/variables.ts +2 -0
  220. package/src/components/Search/FilterFields/SearchFilterFieldTags.tsx +3 -3
  221. package/src/components/Search/SearchAiActionButtons.tsx +76 -0
  222. package/src/components/Search/SearchAiConversationInput.tsx +61 -18
  223. package/src/components/Search/SearchAiDialog.tsx +52 -23
  224. package/src/components/Search/SearchAiMessage.tsx +172 -43
  225. package/src/components/Search/SearchAiNegativeFeedbackForm.tsx +210 -0
  226. package/src/components/Search/SearchDialog.tsx +49 -13
  227. package/src/components/Search/SearchGroups.tsx +2 -0
  228. package/src/components/Search/variables.ts +36 -64
  229. package/src/components/Segmented/Segmented.tsx +15 -20
  230. package/src/components/Select/SelectInput.tsx +1 -0
  231. package/src/components/Tag/Tag.tsx +36 -20
  232. package/src/components/Tag/variables.dark.ts +135 -36
  233. package/src/components/Tag/variables.ts +78 -61
  234. package/src/core/constants/index.ts +1 -0
  235. package/src/core/constants/mcp.ts +1 -0
  236. package/src/core/constants/search.ts +8 -4
  237. package/src/core/hooks/index.ts +3 -0
  238. package/src/core/hooks/menu/use-nested-menu.ts +2 -2
  239. package/src/core/hooks/search/use-feedback-tooltip.ts +32 -0
  240. package/src/core/hooks/use-connect-mcp-button.ts +79 -0
  241. package/src/core/hooks/use-mcp-config.ts +43 -0
  242. package/src/core/hooks/use-page-actions.ts +148 -126
  243. package/src/core/hooks/use-product-picker.ts +2 -1
  244. package/src/core/hooks/use-tabs.ts +168 -86
  245. package/src/core/hooks/use-telemetry-fallback.ts +10 -8
  246. package/src/core/openapi/index.ts +1 -0
  247. package/src/core/styles/dark.ts +4 -0
  248. package/src/core/styles/global.ts +6 -1
  249. package/src/core/types/hooks.ts +6 -1
  250. package/src/core/types/index.ts +1 -0
  251. package/src/core/types/l10n.ts +14 -0
  252. package/src/core/types/mcp.ts +8 -0
  253. package/src/core/types/search.ts +13 -4
  254. package/src/core/types/segmented.ts +14 -0
  255. package/src/core/utils/frontmatter-translate.ts +9 -0
  256. package/src/core/utils/index.ts +2 -0
  257. package/src/core/utils/mcp.ts +34 -0
  258. package/src/icons/AiStarsGradientIcon/AiStarsGradientIcon.tsx +13 -4
  259. package/src/icons/AiStarsIcon/AiStarsIcon.tsx +11 -2
  260. package/src/icons/ConnectIcon/ConnectIcon.tsx +27 -0
  261. package/src/icons/CubeIcon/CubeIcon.tsx +27 -0
  262. package/src/icons/HashtagIcon/HashtagIcon.tsx +23 -0
  263. package/src/icons/RedoclyIcon/RedoclyIcon.tsx +4 -22
  264. package/src/icons/ThumbDownFilledIcon/ThumbDownFilledIcon.tsx +38 -0
  265. package/src/icons/ThumbUpFilledIcon/ThumbUpFilledIcon.tsx +35 -0
  266. package/src/icons/VSCodeIcon/VSCodeIcon.tsx +29 -0
  267. package/src/index.ts +1 -0
  268. package/src/markdoc/components/Cards/Card.tsx +1 -28
  269. package/src/markdoc/components/ConnectMCP/ConnectMCP.tsx +28 -0
  270. package/src/markdoc/components/Tabs/TabList.tsx +312 -105
  271. package/src/markdoc/components/Tabs/Tabs.tsx +136 -11
  272. package/src/markdoc/components/default.ts +1 -0
  273. package/src/markdoc/default.ts +2 -0
  274. package/src/markdoc/tags/card.ts +0 -1
  275. package/src/markdoc/tags/connect-mcp.ts +25 -0
@@ -1,24 +1,40 @@
1
- import { useMemo } from 'react';
1
+ import { useMemo, useCallback } from 'react';
2
2
 
3
3
  import type { PageProps, UiAccessibleConfig } from '@redocly/config';
4
- import type { PageAction } from '../types';
4
+ import type { PageAction, MCPClientType, McpConnectionParams } from '../types';
5
5
 
6
6
  import { CopyIcon } from '@redocly/theme/icons/CopyIcon/CopyIcon';
7
7
  import { ChatGptIcon } from '@redocly/theme/icons/ChatGptIcon/ChatGptIcon';
8
8
  import { ClaudeIcon } from '@redocly/theme/icons/ClaudeIcon/ClaudeIcon';
9
9
  import { MarkdownFullIcon } from '@redocly/theme/icons/MarkdownFullIcon/MarkdownFullIcon';
10
+ import { VSCodeIcon } from '@redocly/theme/icons/VSCodeIcon/VSCodeIcon';
11
+ import { CursorIcon } from '@redocly/theme/icons/CursorIcon/CursorIcon';
10
12
 
11
13
  import { useThemeHooks } from './use-theme-hooks';
12
14
  import { useThemeConfig } from './use-theme-config';
15
+ import { useMCPConfig } from './use-mcp-config';
13
16
  import { ClipboardService } from '../utils/clipboard-service';
14
17
  import { IS_BROWSER } from '../utils/dom';
15
- import { CursorIcon } from '../../icons/CursorIcon/CursorIcon';
16
-
17
- const DEFAULT_ENABLED_ACTIONS = ['copy', 'view', 'chatgpt', 'claude'] as const;
18
- const CURSOR_URL =
19
- 'cursor://anysphere.cursor-deeplink/mcp/install?name=$NAME&config=$BASE64_ENCODED_CONFIG';
20
-
21
- export type PageActionType = 'copy' | 'view' | 'chatgpt' | 'claude' | 'mcp-cursor';
18
+ import { generateMCPDeepLink } from '../utils/mcp';
19
+
20
+ const DEFAULT_ENABLED_ACTIONS = [
21
+ 'copy',
22
+ 'view',
23
+ 'chatgpt',
24
+ 'claude',
25
+ 'docs-mcp-cursor',
26
+ 'docs-mcp-vscode',
27
+ ] as const;
28
+
29
+ export type PageActionType =
30
+ | 'copy'
31
+ | 'view'
32
+ | 'chatgpt'
33
+ | 'claude'
34
+ | 'docs-mcp-cursor'
35
+ | 'docs-mcp-vscode'
36
+ | 'mcp-cursor'
37
+ | 'mcp-vscode';
22
38
 
23
39
  export function usePageActions(
24
40
  pageSlug: string,
@@ -27,12 +43,12 @@ export function usePageActions(
27
43
  ): PageAction[] {
28
44
  const { useTranslate, usePageData, usePageProps, usePageSharedData } = useThemeHooks();
29
45
  const { translate } = useTranslate();
30
-
31
46
  const themeConfig = useThemeConfig();
32
47
  const pageProps = usePageProps();
33
48
  const openApiSharedData = usePageSharedData<
34
49
  { options: { excludeFromSearch: boolean } } | undefined
35
50
  >('openAPIDocsStore');
51
+ const mcpConfig = useMCPConfig();
36
52
 
37
53
  const shouldHideAllActions = shouldHidePageActions(
38
54
  pageProps,
@@ -41,148 +57,154 @@ export function usePageActions(
41
57
  );
42
58
  const { isPublic } = usePageData() || {};
43
59
 
60
+ const createMCPHandler = useCallback(
61
+ (clientType: MCPClientType, requiresMcpUrl: boolean = false) =>
62
+ () => {
63
+ if (requiresMcpUrl && !mcpUrl) return null;
64
+ if (!requiresMcpUrl && (mcpUrl || mcpConfig.isMcpDisabled)) return null;
65
+
66
+ const config = requiresMcpUrl
67
+ ? { serverName: mcpConfig.serverName, url: mcpUrl || '' }
68
+ : { serverName: mcpConfig.serverName, url: mcpConfig.serverUrl || '' };
69
+
70
+ return createMCPAction({ clientType, mcpConfig: config, translate });
71
+ },
72
+ [mcpUrl, mcpConfig, translate],
73
+ );
74
+
44
75
  const result: PageAction[] = useMemo(() => {
45
76
  if (shouldHideAllActions) {
46
77
  return [];
47
78
  }
48
79
 
49
80
  const origin = IS_BROWSER ? window.location.origin : (globalThis as any)['SSR_HOSTNAME'];
50
-
51
81
  const normalizedSlug = pageSlug.startsWith('/') ? pageSlug : '/' + pageSlug;
52
-
53
82
  const mdPageUrl = new URL(
54
83
  origin + normalizedSlug + (normalizedSlug === '/' ? 'index.html.md' : '.md'),
55
84
  ).toString();
56
85
 
57
- function getExternalAiPromptLink(baseUrl: string): string {
58
- const externalAiPrompt = `Read ${mdPageUrl} and answer questions based on the content.`;
59
-
60
- const url = new URL(baseUrl);
61
- url.searchParams.set('q', externalAiPrompt);
62
-
63
- return url.toString();
64
- }
65
-
66
- return (themeConfig.navigation?.actions?.items || actions || DEFAULT_ENABLED_ACTIONS)
67
- .map((action) => {
68
- function generateMCPConfig(isCursor?: boolean): string {
69
- const jsonConfig = {
70
- 'mcp-server': {
71
- url: mcpUrl,
72
- description: 'MCP Server',
73
- },
74
- };
75
- if (isCursor) {
76
- const url = CURSOR_URL.replace('$NAME', 'mcp-server').replace(
77
- '$BASE64_ENCODED_CONFIG',
78
- btoa(JSON.stringify(jsonConfig['mcp-server'])),
79
- );
80
- return url;
86
+ const actionHandlers: Record<PageActionType, () => PageAction | null> = {
87
+ 'docs-mcp-cursor': createMCPHandler('cursor'),
88
+ 'docs-mcp-vscode': createMCPHandler('vscode'),
89
+ 'mcp-cursor': createMCPHandler('cursor', true),
90
+ 'mcp-vscode': createMCPHandler('vscode', true),
91
+
92
+ copy: () => ({
93
+ buttonText: translate('page.actions.copyButtonText', 'Copy'),
94
+ title: translate('page.actions.copyTitle', 'Copy for LLM'),
95
+ description: translate('page.actions.copyDescription', 'Copy page as Markdown for LLMs'),
96
+ iconComponent: CopyIcon,
97
+ onClick: async () => {
98
+ try {
99
+ const result = await fetch(mdPageUrl);
100
+ if (result.status !== 200) {
101
+ return;
102
+ }
103
+ const text = await result.text();
104
+ ClipboardService.copyCustom(text);
105
+ } catch (error) {
106
+ console.error(error);
81
107
  }
82
-
83
- return JSON.stringify(jsonConfig, null, 2);
108
+ },
109
+ }),
110
+
111
+ view: () => ({
112
+ buttonText: translate('page.actions.viewAsMdButtonText', 'View as Markdown'),
113
+ title: translate('page.actions.viewAsMdTitle', 'View as Markdown'),
114
+ description: translate('page.actions.viewAsMdDescription', 'Open this page as Markdown'),
115
+ iconComponent: MarkdownFullIcon,
116
+ link: mdPageUrl,
117
+ }),
118
+
119
+ chatgpt: () => {
120
+ if (!isPublic) {
121
+ return null;
84
122
  }
85
-
86
- switch (action) {
87
- case 'mcp-cursor':
88
- if (!mcpUrl) {
89
- return null;
90
- }
91
-
92
- return {
93
- buttonText: translate('page.actions.cursorMcpButtonText', 'Connect to Cursor'),
94
- title: translate('page.actions.cursorMcpTitle', 'Connect to Cursor'),
95
- description: translate(
96
- 'page.actions.cursorMcpDescription',
97
- 'Install MCP server on Cursor',
98
- ),
99
- iconComponent: CursorIcon,
100
- onClick: () => {
101
- try {
102
- const url = generateMCPConfig(true);
103
- window.open(url, '_blank');
104
- } catch (error) {
105
- console.error(error);
106
- }
107
- },
108
- };
109
- case 'copy':
110
- return {
111
- buttonText: translate('page.actions.copyButtonText', 'Copy'),
112
- title: translate('page.actions.copyTitle', 'Copy for LLM'),
113
- description: translate(
114
- 'page.actions.copyDescription',
115
- 'Copy page as Markdown for LLMs',
116
- ),
117
- iconComponent: CopyIcon,
118
- onClick: async () => {
119
- try {
120
- const result = await fetch(mdPageUrl);
121
- if (result.status !== 200) {
122
- throw new Error('Failed to fetch markdown content');
123
- }
124
- const text = await result.text();
125
-
126
- ClipboardService.copyCustom(text);
127
- } catch (error) {
128
- console.error(error);
129
- }
130
- },
131
- };
132
- case 'view':
133
- return {
134
- buttonText: translate('page.actions.viewAsMdButtonText', 'View as Markdown'),
135
- title: translate('page.actions.viewAsMdTitle', 'View as Markdown'),
136
- description: translate(
137
- 'page.actions.viewAsMdDescription',
138
- 'Open this page as Markdown',
139
- ),
140
- iconComponent: MarkdownFullIcon,
141
- link: mdPageUrl,
142
- };
143
- case 'chatgpt':
144
- if (!isPublic) {
145
- return null;
146
- }
147
-
148
- return {
149
- buttonText: translate('page.actions.chatGptButtonText', 'Open in ChatGPT'),
150
- title: translate('page.actions.chatGptTitle', 'Open in ChatGPT'),
151
- description: translate(
152
- 'page.actions.chatGptDescription',
153
- 'Get insights from ChatGPT',
154
- ),
155
- iconComponent: ChatGptIcon,
156
- link: getExternalAiPromptLink('https://chat.openai.com'),
157
- };
158
- case 'claude':
159
- if (!isPublic) {
160
- return null;
161
- }
162
-
163
- return {
164
- buttonText: translate('page.actions.claudeButtonText', 'Open in Claude'),
165
- title: translate('page.actions.claudeTitle', 'Open in Claude'),
166
- description: translate('page.actions.claudeDescription', 'Get insights from Claude'),
167
- iconComponent: ClaudeIcon,
168
- link: getExternalAiPromptLink('https://claude.ai/new'),
169
- };
123
+ return {
124
+ buttonText: translate('page.actions.chatGptButtonText', 'Open in ChatGPT'),
125
+ title: translate('page.actions.chatGptTitle', 'Open in ChatGPT'),
126
+ description: translate('page.actions.chatGptDescription', 'Get insights from ChatGPT'),
127
+ iconComponent: ChatGptIcon,
128
+ link: getExternalAiPromptLink('https://chat.openai.com', mdPageUrl),
129
+ };
130
+ },
131
+
132
+ claude: () => {
133
+ if (!isPublic) {
134
+ return null;
170
135
  }
171
- })
172
- .filter((action) => action !== null);
136
+ return {
137
+ buttonText: translate('page.actions.claudeButtonText', 'Open in Claude'),
138
+ title: translate('page.actions.claudeTitle', 'Open in Claude'),
139
+ description: translate('page.actions.claudeDescription', 'Get insights from Claude'),
140
+ iconComponent: ClaudeIcon,
141
+ link: getExternalAiPromptLink('https://claude.ai/new', mdPageUrl),
142
+ };
143
+ },
144
+ };
145
+
146
+ return (themeConfig.navigation?.actions?.items || actions || DEFAULT_ENABLED_ACTIONS)
147
+ .map((action) => actionHandlers[action]?.())
148
+ .filter((action): action is PageAction => action !== null);
173
149
  }, [
174
150
  shouldHideAllActions,
175
151
  pageSlug,
176
152
  themeConfig.navigation?.actions?.items,
177
153
  actions,
178
- mcpUrl,
179
154
  translate,
180
155
  isPublic,
156
+ createMCPHandler,
181
157
  ]);
182
158
 
183
159
  return result;
184
160
  }
185
161
 
162
+ function getExternalAiPromptLink(baseUrl: string, mdPageUrl: string): string {
163
+ const externalAiPrompt = `Read ${mdPageUrl} and answer questions based on the content.`;
164
+ const url = new URL(baseUrl);
165
+ url.searchParams.set('q', externalAiPrompt);
166
+ return url.toString();
167
+ }
168
+
169
+ type CreateMCPActionParams = {
170
+ clientType: MCPClientType;
171
+ mcpConfig: McpConnectionParams;
172
+ translate: (key: string, defaultValue: string) => string;
173
+ };
174
+
175
+ function createMCPAction({ clientType, mcpConfig, translate }: CreateMCPActionParams): PageAction {
176
+ const url = generateMCPDeepLink(clientType, mcpConfig);
177
+ const sharedProps = {
178
+ onClick: () => {
179
+ window.open(url, '_blank');
180
+ },
181
+ };
182
+
183
+ if (clientType === 'cursor') {
184
+ return {
185
+ ...sharedProps,
186
+ buttonText: translate('page.actions.connectMcp.cursor', 'Connect to Cursor'),
187
+ title: translate('page.actions.connectMcp.cursor', 'Connect to Cursor'),
188
+ description: translate(
189
+ 'page.actions.connectMcp.cursorDescription',
190
+ 'Install MCP server on Cursor',
191
+ ),
192
+ iconComponent: CursorIcon,
193
+ };
194
+ }
195
+
196
+ return {
197
+ ...sharedProps,
198
+ buttonText: translate('page.actions.connectMcp.vscode', 'Connect to VS Code'),
199
+ title: translate('page.actions.connectMcp.vscode', 'Connect to VS Code'),
200
+ description: translate(
201
+ 'page.actions.connectMcp.vscodeDescription',
202
+ 'Install MCP server on VS Code',
203
+ ),
204
+ iconComponent: VSCodeIcon,
205
+ };
206
+ }
207
+
186
208
  function shouldHidePageActions(
187
209
  pageProps: PageProps,
188
210
  themeConfig: UiAccessibleConfig,
@@ -1,6 +1,7 @@
1
1
  import { useNavigate } from 'react-router-dom';
2
2
 
3
3
  import { useThemeHooks } from './use-theme-hooks';
4
+ import { withPathPrefix } from '../utils';
4
5
 
5
6
  export function useProductPicker() {
6
7
  const { useCurrentProduct, useProducts, useTelemetry, useLoadAndNavigate } = useThemeHooks();
@@ -12,7 +13,7 @@ export function useProductPicker() {
12
13
  function setProduct(product: typeof currentProduct) {
13
14
  if (!product) return;
14
15
  telemetry.sendProductPickedMessage({ product: product.slug });
15
- loadAndNavigate({ navigate, to: product.link });
16
+ loadAndNavigate({ navigate, to: withPathPrefix(product.link) });
16
17
  }
17
18
  return {
18
19
  currentProduct,