@payloadcms/plugin-mcp 4.0.0-canary.1 → 4.0.0-canary.2

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 (278) hide show
  1. package/dist/collection/getAccessField.d.ts.map +1 -1
  2. package/dist/collection/getAccessField.js +24 -9
  3. package/dist/collection/getAccessField.js.map +1 -1
  4. package/dist/collection/index.d.ts.map +1 -1
  5. package/dist/collection/index.js +71 -26
  6. package/dist/collection/index.js.map +1 -1
  7. package/dist/components/APIKeyField/index.client.d.ts +9 -0
  8. package/dist/components/APIKeyField/index.client.d.ts.map +1 -0
  9. package/dist/components/APIKeyField/index.client.js +85 -0
  10. package/dist/components/APIKeyField/index.client.js.map +1 -0
  11. package/dist/components/APIKeyField/index.css +105 -0
  12. package/dist/components/APIKeysEmptyState/index.client.d.ts +4 -0
  13. package/dist/components/APIKeysEmptyState/index.client.d.ts.map +1 -0
  14. package/dist/components/APIKeysEmptyState/index.client.js +21 -0
  15. package/dist/components/APIKeysEmptyState/index.client.js.map +1 -0
  16. package/dist/components/AccessField/index.client.d.ts.map +1 -1
  17. package/dist/components/AccessField/index.client.js +139 -197
  18. package/dist/components/AccessField/index.client.js.map +1 -1
  19. package/dist/components/AccessField/index.css +50 -44
  20. package/dist/components/SettingsMenu/index.client.d.ts +8 -0
  21. package/dist/components/SettingsMenu/index.client.d.ts.map +1 -0
  22. package/dist/components/SettingsMenu/index.client.js +29 -0
  23. package/dist/components/SettingsMenu/index.client.js.map +1 -0
  24. package/dist/endpoint/access.js +16 -0
  25. package/dist/endpoint/access.js.map +1 -1
  26. package/dist/exports/client.d.ts +3 -0
  27. package/dist/exports/client.d.ts.map +1 -1
  28. package/dist/exports/client.js +3 -0
  29. package/dist/exports/client.js.map +1 -1
  30. package/dist/index.d.ts.map +1 -1
  31. package/dist/index.js +46 -3
  32. package/dist/index.js.map +1 -1
  33. package/dist/translations/index.d.ts +11 -0
  34. package/dist/translations/index.d.ts.map +1 -0
  35. package/dist/translations/index.js +92 -0
  36. package/dist/translations/index.js.map +1 -0
  37. package/dist/translations/languages/ar.d.ts +31 -0
  38. package/dist/translations/languages/ar.d.ts.map +1 -0
  39. package/dist/translations/languages/ar.js +34 -0
  40. package/dist/translations/languages/ar.js.map +1 -0
  41. package/dist/translations/languages/az.d.ts +31 -0
  42. package/dist/translations/languages/az.d.ts.map +1 -0
  43. package/dist/translations/languages/az.js +34 -0
  44. package/dist/translations/languages/az.js.map +1 -0
  45. package/dist/translations/languages/bg.d.ts +31 -0
  46. package/dist/translations/languages/bg.d.ts.map +1 -0
  47. package/dist/translations/languages/bg.js +34 -0
  48. package/dist/translations/languages/bg.js.map +1 -0
  49. package/dist/translations/languages/bnBd.d.ts +31 -0
  50. package/dist/translations/languages/bnBd.d.ts.map +1 -0
  51. package/dist/translations/languages/bnBd.js +34 -0
  52. package/dist/translations/languages/bnBd.js.map +1 -0
  53. package/dist/translations/languages/bnIn.d.ts +31 -0
  54. package/dist/translations/languages/bnIn.d.ts.map +1 -0
  55. package/dist/translations/languages/bnIn.js +34 -0
  56. package/dist/translations/languages/bnIn.js.map +1 -0
  57. package/dist/translations/languages/ca.d.ts +31 -0
  58. package/dist/translations/languages/ca.d.ts.map +1 -0
  59. package/dist/translations/languages/ca.js +34 -0
  60. package/dist/translations/languages/ca.js.map +1 -0
  61. package/dist/translations/languages/cs.d.ts +31 -0
  62. package/dist/translations/languages/cs.d.ts.map +1 -0
  63. package/dist/translations/languages/cs.js +34 -0
  64. package/dist/translations/languages/cs.js.map +1 -0
  65. package/dist/translations/languages/da.d.ts +31 -0
  66. package/dist/translations/languages/da.d.ts.map +1 -0
  67. package/dist/translations/languages/da.js +34 -0
  68. package/dist/translations/languages/da.js.map +1 -0
  69. package/dist/translations/languages/de.d.ts +31 -0
  70. package/dist/translations/languages/de.d.ts.map +1 -0
  71. package/dist/translations/languages/de.js +34 -0
  72. package/dist/translations/languages/de.js.map +1 -0
  73. package/dist/translations/languages/en.d.ts +31 -0
  74. package/dist/translations/languages/en.d.ts.map +1 -0
  75. package/dist/translations/languages/en.js +34 -0
  76. package/dist/translations/languages/en.js.map +1 -0
  77. package/dist/translations/languages/es.d.ts +31 -0
  78. package/dist/translations/languages/es.d.ts.map +1 -0
  79. package/dist/translations/languages/es.js +34 -0
  80. package/dist/translations/languages/es.js.map +1 -0
  81. package/dist/translations/languages/et.d.ts +31 -0
  82. package/dist/translations/languages/et.d.ts.map +1 -0
  83. package/dist/translations/languages/et.js +34 -0
  84. package/dist/translations/languages/et.js.map +1 -0
  85. package/dist/translations/languages/fa.d.ts +31 -0
  86. package/dist/translations/languages/fa.d.ts.map +1 -0
  87. package/dist/translations/languages/fa.js +34 -0
  88. package/dist/translations/languages/fa.js.map +1 -0
  89. package/dist/translations/languages/fr.d.ts +31 -0
  90. package/dist/translations/languages/fr.d.ts.map +1 -0
  91. package/dist/translations/languages/fr.js +34 -0
  92. package/dist/translations/languages/fr.js.map +1 -0
  93. package/dist/translations/languages/he.d.ts +31 -0
  94. package/dist/translations/languages/he.d.ts.map +1 -0
  95. package/dist/translations/languages/he.js +34 -0
  96. package/dist/translations/languages/he.js.map +1 -0
  97. package/dist/translations/languages/hr.d.ts +31 -0
  98. package/dist/translations/languages/hr.d.ts.map +1 -0
  99. package/dist/translations/languages/hr.js +34 -0
  100. package/dist/translations/languages/hr.js.map +1 -0
  101. package/dist/translations/languages/hu.d.ts +31 -0
  102. package/dist/translations/languages/hu.d.ts.map +1 -0
  103. package/dist/translations/languages/hu.js +34 -0
  104. package/dist/translations/languages/hu.js.map +1 -0
  105. package/dist/translations/languages/hy.d.ts +31 -0
  106. package/dist/translations/languages/hy.d.ts.map +1 -0
  107. package/dist/translations/languages/hy.js +34 -0
  108. package/dist/translations/languages/hy.js.map +1 -0
  109. package/dist/translations/languages/id.d.ts +31 -0
  110. package/dist/translations/languages/id.d.ts.map +1 -0
  111. package/dist/translations/languages/id.js +34 -0
  112. package/dist/translations/languages/id.js.map +1 -0
  113. package/dist/translations/languages/is.d.ts +31 -0
  114. package/dist/translations/languages/is.d.ts.map +1 -0
  115. package/dist/translations/languages/is.js +34 -0
  116. package/dist/translations/languages/is.js.map +1 -0
  117. package/dist/translations/languages/it.d.ts +31 -0
  118. package/dist/translations/languages/it.d.ts.map +1 -0
  119. package/dist/translations/languages/it.js +34 -0
  120. package/dist/translations/languages/it.js.map +1 -0
  121. package/dist/translations/languages/ja.d.ts +31 -0
  122. package/dist/translations/languages/ja.d.ts.map +1 -0
  123. package/dist/translations/languages/ja.js +34 -0
  124. package/dist/translations/languages/ja.js.map +1 -0
  125. package/dist/translations/languages/ko.d.ts +31 -0
  126. package/dist/translations/languages/ko.d.ts.map +1 -0
  127. package/dist/translations/languages/ko.js +34 -0
  128. package/dist/translations/languages/ko.js.map +1 -0
  129. package/dist/translations/languages/lt.d.ts +31 -0
  130. package/dist/translations/languages/lt.d.ts.map +1 -0
  131. package/dist/translations/languages/lt.js +34 -0
  132. package/dist/translations/languages/lt.js.map +1 -0
  133. package/dist/translations/languages/lv.d.ts +31 -0
  134. package/dist/translations/languages/lv.d.ts.map +1 -0
  135. package/dist/translations/languages/lv.js +34 -0
  136. package/dist/translations/languages/lv.js.map +1 -0
  137. package/dist/translations/languages/my.d.ts +31 -0
  138. package/dist/translations/languages/my.d.ts.map +1 -0
  139. package/dist/translations/languages/my.js +34 -0
  140. package/dist/translations/languages/my.js.map +1 -0
  141. package/dist/translations/languages/nb.d.ts +31 -0
  142. package/dist/translations/languages/nb.d.ts.map +1 -0
  143. package/dist/translations/languages/nb.js +34 -0
  144. package/dist/translations/languages/nb.js.map +1 -0
  145. package/dist/translations/languages/nl.d.ts +31 -0
  146. package/dist/translations/languages/nl.d.ts.map +1 -0
  147. package/dist/translations/languages/nl.js +34 -0
  148. package/dist/translations/languages/nl.js.map +1 -0
  149. package/dist/translations/languages/pl.d.ts +31 -0
  150. package/dist/translations/languages/pl.d.ts.map +1 -0
  151. package/dist/translations/languages/pl.js +34 -0
  152. package/dist/translations/languages/pl.js.map +1 -0
  153. package/dist/translations/languages/pt.d.ts +31 -0
  154. package/dist/translations/languages/pt.d.ts.map +1 -0
  155. package/dist/translations/languages/pt.js +34 -0
  156. package/dist/translations/languages/pt.js.map +1 -0
  157. package/dist/translations/languages/ro.d.ts +31 -0
  158. package/dist/translations/languages/ro.d.ts.map +1 -0
  159. package/dist/translations/languages/ro.js +34 -0
  160. package/dist/translations/languages/ro.js.map +1 -0
  161. package/dist/translations/languages/rs.d.ts +31 -0
  162. package/dist/translations/languages/rs.d.ts.map +1 -0
  163. package/dist/translations/languages/rs.js +34 -0
  164. package/dist/translations/languages/rs.js.map +1 -0
  165. package/dist/translations/languages/rsLatin.d.ts +31 -0
  166. package/dist/translations/languages/rsLatin.d.ts.map +1 -0
  167. package/dist/translations/languages/rsLatin.js +34 -0
  168. package/dist/translations/languages/rsLatin.js.map +1 -0
  169. package/dist/translations/languages/ru.d.ts +31 -0
  170. package/dist/translations/languages/ru.d.ts.map +1 -0
  171. package/dist/translations/languages/ru.js +34 -0
  172. package/dist/translations/languages/ru.js.map +1 -0
  173. package/dist/translations/languages/sk.d.ts +31 -0
  174. package/dist/translations/languages/sk.d.ts.map +1 -0
  175. package/dist/translations/languages/sk.js +34 -0
  176. package/dist/translations/languages/sk.js.map +1 -0
  177. package/dist/translations/languages/sl.d.ts +31 -0
  178. package/dist/translations/languages/sl.d.ts.map +1 -0
  179. package/dist/translations/languages/sl.js +34 -0
  180. package/dist/translations/languages/sl.js.map +1 -0
  181. package/dist/translations/languages/sv.d.ts +31 -0
  182. package/dist/translations/languages/sv.d.ts.map +1 -0
  183. package/dist/translations/languages/sv.js +34 -0
  184. package/dist/translations/languages/sv.js.map +1 -0
  185. package/dist/translations/languages/ta.d.ts +31 -0
  186. package/dist/translations/languages/ta.d.ts.map +1 -0
  187. package/dist/translations/languages/ta.js +34 -0
  188. package/dist/translations/languages/ta.js.map +1 -0
  189. package/dist/translations/languages/th.d.ts +31 -0
  190. package/dist/translations/languages/th.d.ts.map +1 -0
  191. package/dist/translations/languages/th.js +34 -0
  192. package/dist/translations/languages/th.js.map +1 -0
  193. package/dist/translations/languages/tr.d.ts +31 -0
  194. package/dist/translations/languages/tr.d.ts.map +1 -0
  195. package/dist/translations/languages/tr.js +34 -0
  196. package/dist/translations/languages/tr.js.map +1 -0
  197. package/dist/translations/languages/uk.d.ts +31 -0
  198. package/dist/translations/languages/uk.d.ts.map +1 -0
  199. package/dist/translations/languages/uk.js +34 -0
  200. package/dist/translations/languages/uk.js.map +1 -0
  201. package/dist/translations/languages/vi.d.ts +31 -0
  202. package/dist/translations/languages/vi.d.ts.map +1 -0
  203. package/dist/translations/languages/vi.js +34 -0
  204. package/dist/translations/languages/vi.js.map +1 -0
  205. package/dist/translations/languages/zh.d.ts +31 -0
  206. package/dist/translations/languages/zh.d.ts.map +1 -0
  207. package/dist/translations/languages/zh.js +34 -0
  208. package/dist/translations/languages/zh.js.map +1 -0
  209. package/dist/translations/languages/zhTw.d.ts +31 -0
  210. package/dist/translations/languages/zhTw.d.ts.map +1 -0
  211. package/dist/translations/languages/zhTw.js +34 -0
  212. package/dist/translations/languages/zhTw.js.map +1 -0
  213. package/dist/translations/types.d.ts +32 -0
  214. package/dist/translations/types.d.ts.map +1 -0
  215. package/dist/translations/types.js +3 -0
  216. package/dist/translations/types.js.map +1 -0
  217. package/dist/types.d.ts +6 -1
  218. package/dist/types.d.ts.map +1 -1
  219. package/dist/types.js.map +1 -1
  220. package/package.json +7 -6
  221. package/src/collection/getAccessField.ts +33 -13
  222. package/src/collection/index.ts +63 -22
  223. package/src/components/APIKeyField/index.client.tsx +73 -0
  224. package/src/components/APIKeyField/index.css +105 -0
  225. package/src/components/APIKeysEmptyState/index.client.tsx +36 -0
  226. package/src/components/AccessField/index.client.tsx +163 -211
  227. package/src/components/AccessField/index.css +50 -44
  228. package/src/components/SettingsMenu/index.client.tsx +34 -0
  229. package/src/endpoint/access.ts +12 -0
  230. package/src/exports/client.ts +3 -0
  231. package/src/index.ts +49 -3
  232. package/src/translations/index.ts +108 -0
  233. package/src/translations/languages/ar.ts +35 -0
  234. package/src/translations/languages/az.ts +35 -0
  235. package/src/translations/languages/bg.ts +35 -0
  236. package/src/translations/languages/bnBd.ts +35 -0
  237. package/src/translations/languages/bnIn.ts +35 -0
  238. package/src/translations/languages/ca.ts +35 -0
  239. package/src/translations/languages/cs.ts +35 -0
  240. package/src/translations/languages/da.ts +35 -0
  241. package/src/translations/languages/de.ts +35 -0
  242. package/src/translations/languages/en.ts +35 -0
  243. package/src/translations/languages/es.ts +35 -0
  244. package/src/translations/languages/et.ts +35 -0
  245. package/src/translations/languages/fa.ts +35 -0
  246. package/src/translations/languages/fr.ts +35 -0
  247. package/src/translations/languages/he.ts +35 -0
  248. package/src/translations/languages/hr.ts +35 -0
  249. package/src/translations/languages/hu.ts +35 -0
  250. package/src/translations/languages/hy.ts +35 -0
  251. package/src/translations/languages/id.ts +35 -0
  252. package/src/translations/languages/is.ts +35 -0
  253. package/src/translations/languages/it.ts +35 -0
  254. package/src/translations/languages/ja.ts +35 -0
  255. package/src/translations/languages/ko.ts +35 -0
  256. package/src/translations/languages/lt.ts +35 -0
  257. package/src/translations/languages/lv.ts +35 -0
  258. package/src/translations/languages/my.ts +35 -0
  259. package/src/translations/languages/nb.ts +35 -0
  260. package/src/translations/languages/nl.ts +35 -0
  261. package/src/translations/languages/pl.ts +35 -0
  262. package/src/translations/languages/pt.ts +35 -0
  263. package/src/translations/languages/ro.ts +35 -0
  264. package/src/translations/languages/rs.ts +35 -0
  265. package/src/translations/languages/rsLatin.ts +35 -0
  266. package/src/translations/languages/ru.ts +35 -0
  267. package/src/translations/languages/sk.ts +35 -0
  268. package/src/translations/languages/sl.ts +35 -0
  269. package/src/translations/languages/sv.ts +35 -0
  270. package/src/translations/languages/ta.ts +35 -0
  271. package/src/translations/languages/th.ts +35 -0
  272. package/src/translations/languages/tr.ts +35 -0
  273. package/src/translations/languages/uk.ts +35 -0
  274. package/src/translations/languages/vi.ts +35 -0
  275. package/src/translations/languages/zh.ts +35 -0
  276. package/src/translations/languages/zhTw.ts +35 -0
  277. package/src/translations/types.ts +34 -0
  278. package/src/types.ts +6 -1
@@ -1,9 +1,14 @@
1
1
  'use client';
2
- import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
3
- import { CheckboxInput, Collapsible, useField } from '@payloadcms/ui';
4
- import React from 'react';
2
+ import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-runtime";
3
+ import { CheckboxInput, Collapsible, Tabs, useField, useTranslation } from '@payloadcms/ui';
4
+ import React, { useState } from 'react';
5
5
  import './index.css';
6
6
  const baseClass = 'mcp-access-field';
7
+ const LEAF_GROUP_KEYS = [
8
+ 'operations',
9
+ 'auth',
10
+ 'custom'
11
+ ];
7
12
  /** Drop a key from an object and return a new object — or `undefined` if it'd be empty. */ const without = (obj, key)=>{
8
13
  if (!obj || !(key in obj)) {
9
14
  return obj;
@@ -19,14 +24,19 @@ export const AccessField = ({ path, pluginConfig })=>{
19
24
  const { setValue, value } = useField({
20
25
  path
21
26
  });
27
+ const { t } = useTranslation();
28
+ const [activeTab, setActiveTab] = useState(null);
22
29
  const access = value ?? {};
23
- // Bucket items for rendering. (Bucketing is cheap and runs once per render;
24
- // memoizing would mean managing inputs/refs for marginal benefit.)
30
+ const leafGroupLabels = {
31
+ auth: t('plugin-mcp:authentication'),
32
+ custom: t('general:custom'),
33
+ operations: t('plugin-mcp:operations')
34
+ };
25
35
  const collectionsBySlug = {};
26
36
  const globalsBySlug = {};
27
- const tools = [];
28
37
  const prompts = [];
29
38
  const resources = [];
39
+ const tools = [];
30
40
  for (const item of pluginConfig.items){
31
41
  switch(item.type){
32
42
  case 'collectionTool':
@@ -90,216 +100,148 @@ export const AccessField = ({ path, pluginConfig })=>{
90
100
  setValue(setKey(access, scope, bucket));
91
101
  }
92
102
  };
93
- const collectionSlugs = Object.keys(collectionsBySlug);
94
- const globalSlugs = Object.keys(globalsBySlug);
95
- return /*#__PURE__*/ _jsxs("div", {
96
- className: baseClass,
97
- children: [
98
- collectionSlugs.length > 0 && /*#__PURE__*/ _jsxs("section", {
99
- className: `${baseClass}__section`,
100
- children: [
101
- /*#__PURE__*/ _jsxs("header", {
102
- className: `${baseClass}__section-header`,
103
- children: [
104
- /*#__PURE__*/ _jsx("h4", {
105
- children: "Collection-level permissions"
106
- }),
107
- /*#__PURE__*/ _jsx("p", {
108
- children: "Allow MCP clients to perform the following actions within these collections:"
109
- })
110
- ]
111
- }),
112
- collectionSlugs.map((slug)=>{
113
- const leaves = collectionsBySlug[slug];
114
- return /*#__PURE__*/ _jsx(Collapsible, {
115
- actions: /*#__PURE__*/ _jsx(GroupActions, {
116
- onSetAll: (allow)=>setAllScoped('collections', slug, leaves, allow)
117
- }),
118
- className: `${baseClass}__group`,
119
- header: /*#__PURE__*/ _jsx("span", {
120
- className: `${baseClass}__group-label`,
121
- children: titleCase(slug)
122
- }),
123
- initCollapsed: true,
124
- children: /*#__PURE__*/ _jsx("ul", {
125
- className: `${baseClass}__list`,
126
- children: leaves.map((leaf)=>/*#__PURE__*/ _jsx("li", {
127
- children: /*#__PURE__*/ _jsx(CheckboxInput, {
128
- checked: isScopedAllowed('collections', slug, leaf.configKey),
129
- id: `${path}.collections.${slug}.${leaf.configKey}`,
130
- label: leaf.label,
131
- onToggle: (e)=>toggleScoped('collections', slug, leaf.configKey, e.target.checked),
132
- tooltip: leaf.description
133
- })
134
- }, leaf.configKey))
135
- })
136
- }, `collection-${slug}`);
137
- })
138
- ]
103
+ const renderLeaf = (leaf, id, checked, onToggle)=>/*#__PURE__*/ _jsxs("li", {
104
+ className: `${baseClass}__leaf`,
105
+ children: [
106
+ /*#__PURE__*/ _jsx(CheckboxInput, {
107
+ checked: checked,
108
+ id: id,
109
+ label: leaf.label,
110
+ onToggle: (e)=>onToggle(e.target.checked)
111
+ }),
112
+ leaf.description && /*#__PURE__*/ _jsx("p", {
113
+ className: `${baseClass}__leaf-description`,
114
+ children: leaf.description
115
+ })
116
+ ]
117
+ }, leaf.configKey);
118
+ const renderCard = ({ id, isLeafAllowed, label, leaves, onSetAll, onToggleLeaf })=>{
119
+ const allowedCount = leaves.filter(isLeafAllowed).length;
120
+ const groups = LEAF_GROUP_KEYS.map((key)=>({
121
+ key,
122
+ label: leafGroupLabels[key],
123
+ leaves: leaves.filter((leaf)=>(leaf.group ?? 'custom') === key)
124
+ })).filter((group)=>group.leaves.length > 0);
125
+ const hasGroupLabels = groups.length > 1;
126
+ return /*#__PURE__*/ _jsx(Collapsible, {
127
+ className: `${baseClass}__card`,
128
+ header: // Keep header clicks on the checkbox from also toggling the collapsible.
129
+ /*#__PURE__*/ _jsx("span", {
130
+ className: `${baseClass}__card-checkbox`,
131
+ onClick: (e)=>e.stopPropagation(),
132
+ role: "presentation",
133
+ children: /*#__PURE__*/ _jsx(CheckboxInput, {
134
+ checked: allowedCount === leaves.length,
135
+ id: `${id}._all`,
136
+ label: label,
137
+ onToggle: ()=>onSetAll(allowedCount < leaves.length),
138
+ partialChecked: allowedCount > 0 && allowedCount < leaves.length
139
+ })
139
140
  }),
140
- globalSlugs.length > 0 && /*#__PURE__*/ _jsxs("section", {
141
- className: `${baseClass}__section`,
142
- children: [
143
- /*#__PURE__*/ _jsxs("header", {
144
- className: `${baseClass}__section-header`,
141
+ initCollapsed: true,
142
+ children: /*#__PURE__*/ _jsx("div", {
143
+ className: `${baseClass}__card-groups`,
144
+ children: groups.map((group)=>/*#__PURE__*/ _jsxs("div", {
145
+ className: `${baseClass}__leaf-group`,
145
146
  children: [
146
- /*#__PURE__*/ _jsx("h4", {
147
- children: "Global-level permissions"
147
+ hasGroupLabels && /*#__PURE__*/ _jsx("p", {
148
+ className: `${baseClass}__leaf-group-label`,
149
+ children: group.label
148
150
  }),
149
- /*#__PURE__*/ _jsx("p", {
150
- children: "Allow MCP clients to perform the following actions on these globals:"
151
+ /*#__PURE__*/ _jsx("ul", {
152
+ className: `${baseClass}__list`,
153
+ children: group.leaves.map((leaf)=>renderLeaf(leaf, `${id}.${leaf.configKey}`, isLeafAllowed(leaf), (allow)=>onToggleLeaf(leaf, allow)))
151
154
  })
152
155
  ]
156
+ }, group.key))
157
+ })
158
+ }, id);
159
+ };
160
+ const renderScope = (scope, bySlug)=>Object.entries(bySlug).map(([slug, leaves])=>renderCard({
161
+ id: `${path}.${scope}.${slug}`,
162
+ isLeafAllowed: (leaf)=>isScopedAllowed(scope, slug, leaf.configKey),
163
+ label: titleCase(slug),
164
+ leaves,
165
+ onSetAll: (allow)=>setAllScoped(scope, slug, leaves, allow),
166
+ onToggleLeaf: (leaf, allow)=>toggleScoped(scope, slug, leaf.configKey, allow)
167
+ }));
168
+ const renderFlat = (scope, label, leaves)=>leaves.length > 0 && renderCard({
169
+ id: `${path}.${scope}`,
170
+ isLeafAllowed: (leaf)=>isFlatAllowed(scope, leaf.configKey),
171
+ label,
172
+ leaves,
173
+ onSetAll: (allow)=>setAllFlat(scope, leaves, allow),
174
+ onToggleLeaf: (leaf, allow)=>toggleFlat(scope, leaf.configKey, allow)
175
+ });
176
+ const tabs = [
177
+ ...Object.keys(collectionsBySlug).length > 0 ? [
178
+ {
179
+ key: 'collections',
180
+ label: t('general:collections')
181
+ }
182
+ ] : [],
183
+ ...Object.keys(globalsBySlug).length > 0 ? [
184
+ {
185
+ key: 'globals',
186
+ label: t('general:globals')
187
+ }
188
+ ] : [],
189
+ ...prompts.length > 0 || resources.length > 0 || tools.length > 0 ? [
190
+ {
191
+ key: 'server',
192
+ label: t('plugin-mcp:server')
193
+ }
194
+ ] : []
195
+ ];
196
+ if (tabs.length === 0) {
197
+ return null;
198
+ }
199
+ const currentTab = activeTab ?? tabs[0].key;
200
+ return /*#__PURE__*/ _jsxs("div", {
201
+ className: baseClass,
202
+ children: [
203
+ /*#__PURE__*/ _jsxs("header", {
204
+ className: `${baseClass}__header`,
205
+ children: [
206
+ /*#__PURE__*/ _jsx("h4", {
207
+ children: t('plugin-mcp:permissions')
153
208
  }),
154
- globalSlugs.map((slug)=>{
155
- const leaves = globalsBySlug[slug];
156
- return /*#__PURE__*/ _jsx(Collapsible, {
157
- actions: /*#__PURE__*/ _jsx(GroupActions, {
158
- onSetAll: (allow)=>setAllScoped('globals', slug, leaves, allow)
159
- }),
160
- className: `${baseClass}__group`,
161
- header: /*#__PURE__*/ _jsx("span", {
162
- className: `${baseClass}__group-label`,
163
- children: titleCase(slug)
164
- }),
165
- initCollapsed: true,
166
- children: /*#__PURE__*/ _jsx("ul", {
167
- className: `${baseClass}__list`,
168
- children: leaves.map((leaf)=>/*#__PURE__*/ _jsx("li", {
169
- children: /*#__PURE__*/ _jsx(CheckboxInput, {
170
- checked: isScopedAllowed('globals', slug, leaf.configKey),
171
- id: `${path}.globals.${slug}.${leaf.configKey}`,
172
- label: leaf.label,
173
- onToggle: (e)=>toggleScoped('globals', slug, leaf.configKey, e.target.checked),
174
- tooltip: leaf.description
175
- })
176
- }, leaf.configKey))
177
- })
178
- }, `global-${slug}`);
209
+ /*#__PURE__*/ _jsx("p", {
210
+ children: t('plugin-mcp:permissionsDescription')
179
211
  })
180
212
  ]
181
213
  }),
182
- (tools.length > 0 || prompts.length > 0 || resources.length > 0) && /*#__PURE__*/ _jsxs("section", {
183
- className: `${baseClass}__section`,
214
+ /*#__PURE__*/ _jsxs("div", {
215
+ className: `${baseClass}__tabbed-content`,
184
216
  children: [
185
- /*#__PURE__*/ _jsxs("header", {
186
- className: `${baseClass}__section-header`,
217
+ /*#__PURE__*/ _jsx(Tabs, {
218
+ className: `${baseClass}__tabs`,
219
+ onChange: setActiveTab,
220
+ tabs: tabs.map((tab)=>({
221
+ label: tab.label,
222
+ value: tab.key
223
+ })),
224
+ value: currentTab
225
+ }),
226
+ /*#__PURE__*/ _jsxs("div", {
227
+ className: `${baseClass}__cards`,
187
228
  children: [
188
- /*#__PURE__*/ _jsx("h4", {
189
- children: "Project-level permissions"
190
- }),
191
- /*#__PURE__*/ _jsx("p", {
192
- children: "Cross-cutting tools, prompts, and resources not scoped to a single collection."
229
+ currentTab === 'collections' && renderScope('collections', collectionsBySlug),
230
+ currentTab === 'globals' && renderScope('globals', globalsBySlug),
231
+ currentTab === 'server' && /*#__PURE__*/ _jsxs(_Fragment, {
232
+ children: [
233
+ renderFlat('prompts', t('plugin-mcp:prompts'), prompts),
234
+ renderFlat('resources', t('plugin-mcp:resources'), resources),
235
+ renderFlat('tools', t('plugin-mcp:tools'), tools)
236
+ ]
193
237
  })
194
238
  ]
195
- }),
196
- tools.length > 0 && /*#__PURE__*/ _jsx(Collapsible, {
197
- actions: /*#__PURE__*/ _jsx(GroupActions, {
198
- onSetAll: (allow)=>setAllFlat('tools', tools, allow)
199
- }),
200
- className: `${baseClass}__group`,
201
- header: /* TODO: needs i18n once design is finalized */ /*#__PURE__*/ _jsx("span", {
202
- className: `${baseClass}__group-label`,
203
- children: "Tools"
204
- }),
205
- initCollapsed: true,
206
- children: /*#__PURE__*/ _jsx("ul", {
207
- className: `${baseClass}__list`,
208
- children: tools.map((leaf)=>/*#__PURE__*/ _jsx("li", {
209
- children: /*#__PURE__*/ _jsx(CheckboxInput, {
210
- checked: isFlatAllowed('tools', leaf.configKey),
211
- id: `${path}.tools.${leaf.configKey}`,
212
- label: leaf.label,
213
- onToggle: (e)=>toggleFlat('tools', leaf.configKey, e.target.checked),
214
- tooltip: leaf.description
215
- })
216
- }, leaf.configKey))
217
- })
218
- }),
219
- prompts.length > 0 && /*#__PURE__*/ _jsx(Collapsible, {
220
- actions: /*#__PURE__*/ _jsx(GroupActions, {
221
- onSetAll: (allow)=>setAllFlat('prompts', prompts, allow)
222
- }),
223
- className: `${baseClass}__group`,
224
- header: /* TODO: needs i18n once design is finalized */ /*#__PURE__*/ _jsx("span", {
225
- className: `${baseClass}__group-label`,
226
- children: "Prompts"
227
- }),
228
- initCollapsed: true,
229
- children: /*#__PURE__*/ _jsx("ul", {
230
- className: `${baseClass}__list`,
231
- children: prompts.map((leaf)=>/*#__PURE__*/ _jsx("li", {
232
- children: /*#__PURE__*/ _jsx(CheckboxInput, {
233
- checked: isFlatAllowed('prompts', leaf.configKey),
234
- id: `${path}.prompts.${leaf.configKey}`,
235
- label: leaf.label,
236
- onToggle: (e)=>toggleFlat('prompts', leaf.configKey, e.target.checked),
237
- tooltip: leaf.description
238
- })
239
- }, leaf.configKey))
240
- })
241
- }),
242
- resources.length > 0 && /*#__PURE__*/ _jsx(Collapsible, {
243
- actions: /*#__PURE__*/ _jsx(GroupActions, {
244
- onSetAll: (allow)=>setAllFlat('resources', resources, allow)
245
- }),
246
- className: `${baseClass}__group`,
247
- header: /* TODO: needs i18n once design is finalized */ /*#__PURE__*/ _jsx("span", {
248
- className: `${baseClass}__group-label`,
249
- children: "Resources"
250
- }),
251
- initCollapsed: true,
252
- children: /*#__PURE__*/ _jsx("ul", {
253
- className: `${baseClass}__list`,
254
- children: resources.map((leaf)=>/*#__PURE__*/ _jsx("li", {
255
- children: /*#__PURE__*/ _jsx(CheckboxInput, {
256
- checked: isFlatAllowed('resources', leaf.configKey),
257
- id: `${path}.resources.${leaf.configKey}`,
258
- label: leaf.label,
259
- onToggle: (e)=>toggleFlat('resources', leaf.configKey, e.target.checked),
260
- tooltip: leaf.description
261
- })
262
- }, leaf.configKey))
263
- })
264
239
  })
265
240
  ]
266
241
  })
267
242
  ]
268
243
  });
269
244
  };
270
- const GroupActions = ({ onSetAll })=>// TODO: button labels + aria-labels need i18n once design is finalized
271
- /*#__PURE__*/ _jsxs("div", {
272
- className: `${baseClass}__group-actions`,
273
- children: [
274
- /*#__PURE__*/ _jsx("button", {
275
- "aria-label": "Select all",
276
- className: `${baseClass}__action`,
277
- onClick: (e)=>{
278
- e.stopPropagation();
279
- onSetAll(true);
280
- },
281
- title: "Select all",
282
- type: "button",
283
- children: "all"
284
- }),
285
- /*#__PURE__*/ _jsx("span", {
286
- "aria-hidden": true,
287
- className: `${baseClass}__action-sep`,
288
- children: "/"
289
- }),
290
- /*#__PURE__*/ _jsx("button", {
291
- "aria-label": "Clear all",
292
- className: `${baseClass}__action`,
293
- onClick: (e)=>{
294
- e.stopPropagation();
295
- onSetAll(false);
296
- },
297
- title: "Clear all",
298
- type: "button",
299
- children: "none"
300
- })
301
- ]
302
- });
303
245
  const titleCase = (slug)=>slug.replace(/(^|[-_])(.)/g, (_, sep, ch)=>sep ? ` ${ch.toUpperCase()}` : ch.toUpperCase());
304
246
 
305
247
  //# sourceMappingURL=index.client.js.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../src/components/AccessField/index.client.tsx"],"sourcesContent":["'use client'\n\nimport type { JSONFieldClientProps } from 'payload'\n\nimport { CheckboxInput, Collapsible, useField } from '@payloadcms/ui'\nimport React from 'react'\n\nimport type { ClientMCPPluginConfig, MCPAPIKeysDocAccessTree } from '../../types.js'\n\nimport './index.css'\n\nconst baseClass = 'mcp-access-field'\n\ntype ClientItem = ClientMCPPluginConfig['items'][number]\ntype ScopeKey = 'collections' | 'globals'\ntype FlatKey = 'prompts' | 'resources' | 'tools'\n\ntype Props = {\n pluginConfig: ClientMCPPluginConfig\n} & JSONFieldClientProps\n\n/** Drop a key from an object and return a new object — or `undefined` if it'd be empty. */\nconst without = <T extends Record<string, unknown>>(\n obj: T | undefined,\n key: string,\n): T | undefined => {\n if (!obj || !(key in obj)) {\n return obj\n }\n const { [key]: _omitted, ...rest } = obj\n return Object.keys(rest).length === 0 ? undefined : (rest as T)\n}\n\nconst setKey = <T extends Record<string, unknown>>(\n obj: T | undefined,\n key: string,\n value: unknown,\n): T => ({ ...(obj ?? {}), [key]: value }) as T\n\nexport const AccessField: React.FC<Props> = ({ path, pluginConfig }) => {\n const { setValue, value } = useField<MCPAPIKeysDocAccessTree>({ path })\n const access = value ?? {}\n\n // Bucket items for rendering. (Bucketing is cheap and runs once per render;\n // memoizing would mean managing inputs/refs for marginal benefit.)\n const collectionsBySlug: Record<string, ClientItem[]> = {}\n const globalsBySlug: Record<string, ClientItem[]> = {}\n const tools: ClientItem[] = []\n const prompts: ClientItem[] = []\n const resources: ClientItem[] = []\n for (const item of pluginConfig.items) {\n switch (item.type) {\n case 'collectionTool':\n ;(collectionsBySlug[item.collectionSlug!] ??= []).push(item)\n break\n case 'globalTool':\n ;(globalsBySlug[item.globalSlug!] ??= []).push(item)\n break\n case 'prompt':\n prompts.push(item)\n break\n case 'resource':\n resources.push(item)\n break\n case 'tool':\n tools.push(item)\n break\n }\n }\n\n const isScopedAllowed = (scope: ScopeKey, slug: string, configKey: string): boolean =>\n access[scope]?.[slug]?.[configKey] !== false\n\n const isFlatAllowed = (scope: FlatKey, configKey: string): boolean =>\n access[scope]?.[configKey] !== false\n\n const toggleScoped = (scope: ScopeKey, slug: string, configKey: string, allow: boolean) => {\n if (allow) {\n const slugBucket = without(access[scope]?.[slug], configKey)\n const scopeBucket = slugBucket\n ? setKey(access[scope], slug, slugBucket)\n : without(access[scope], slug)\n setValue(scopeBucket ? setKey(access, scope, scopeBucket) : (without(access, scope) ?? {}))\n } else {\n setValue(\n setKey(\n access,\n scope,\n setKey(access[scope], slug, setKey(access[scope]?.[slug], configKey, false)),\n ),\n )\n }\n }\n\n const toggleFlat = (scope: FlatKey, configKey: string, allow: boolean) => {\n if (allow) {\n const bucket = without(access[scope], configKey)\n setValue(bucket ? setKey(access, scope, bucket) : (without(access, scope) ?? {}))\n } else {\n setValue(setKey(access, scope, setKey(access[scope], configKey, false)))\n }\n }\n\n const setAllScoped = (scope: ScopeKey, slug: string, leaves: ClientItem[], allow: boolean) => {\n if (allow) {\n const scopeBucket = without(access[scope], slug)\n setValue(scopeBucket ? setKey(access, scope, scopeBucket) : (without(access, scope) ?? {}))\n } else {\n const slugBucket = leaves.reduce<Record<string, boolean>>(\n (acc, leaf) => ({ ...acc, [leaf.configKey]: false }),\n {},\n )\n setValue(setKey(access, scope, setKey(access[scope], slug, slugBucket)))\n }\n }\n\n const setAllFlat = (scope: FlatKey, leaves: ClientItem[], allow: boolean) => {\n if (allow) {\n setValue(without(access, scope) ?? {})\n } else {\n const bucket = leaves.reduce<Record<string, boolean>>(\n (acc, leaf) => ({ ...acc, [leaf.configKey]: false }),\n {},\n )\n setValue(setKey(access, scope, bucket))\n }\n }\n\n const collectionSlugs = Object.keys(collectionsBySlug)\n const globalSlugs = Object.keys(globalsBySlug)\n\n return (\n <div className={baseClass}>\n {collectionSlugs.length > 0 && (\n <section className={`${baseClass}__section`}>\n <header className={`${baseClass}__section-header`}>\n {/* TODO: needs i18n once design is finalized */}\n <h4>Collection-level permissions</h4>\n {/* TODO: needs i18n once design is finalized */}\n <p>Allow MCP clients to perform the following actions within these collections:</p>\n </header>\n {collectionSlugs.map((slug) => {\n const leaves = collectionsBySlug[slug]!\n return (\n <Collapsible\n actions={\n <GroupActions\n onSetAll={(allow) => setAllScoped('collections', slug, leaves, allow)}\n />\n }\n className={`${baseClass}__group`}\n header={<span className={`${baseClass}__group-label`}>{titleCase(slug)}</span>}\n initCollapsed\n key={`collection-${slug}`}\n >\n <ul className={`${baseClass}__list`}>\n {leaves.map((leaf) => (\n <li key={leaf.configKey}>\n <CheckboxInput\n checked={isScopedAllowed('collections', slug, leaf.configKey)}\n id={`${path}.collections.${slug}.${leaf.configKey}`}\n label={leaf.label}\n onToggle={(e) =>\n toggleScoped('collections', slug, leaf.configKey, e.target.checked)\n }\n tooltip={leaf.description}\n />\n </li>\n ))}\n </ul>\n </Collapsible>\n )\n })}\n </section>\n )}\n\n {globalSlugs.length > 0 && (\n <section className={`${baseClass}__section`}>\n <header className={`${baseClass}__section-header`}>\n {/* TODO: needs i18n once design is finalized */}\n <h4>Global-level permissions</h4>\n {/* TODO: needs i18n once design is finalized */}\n <p>Allow MCP clients to perform the following actions on these globals:</p>\n </header>\n {globalSlugs.map((slug) => {\n const leaves = globalsBySlug[slug]!\n return (\n <Collapsible\n actions={\n <GroupActions\n onSetAll={(allow) => setAllScoped('globals', slug, leaves, allow)}\n />\n }\n className={`${baseClass}__group`}\n header={<span className={`${baseClass}__group-label`}>{titleCase(slug)}</span>}\n initCollapsed\n key={`global-${slug}`}\n >\n <ul className={`${baseClass}__list`}>\n {leaves.map((leaf) => (\n <li key={leaf.configKey}>\n <CheckboxInput\n checked={isScopedAllowed('globals', slug, leaf.configKey)}\n id={`${path}.globals.${slug}.${leaf.configKey}`}\n label={leaf.label}\n onToggle={(e) =>\n toggleScoped('globals', slug, leaf.configKey, e.target.checked)\n }\n tooltip={leaf.description}\n />\n </li>\n ))}\n </ul>\n </Collapsible>\n )\n })}\n </section>\n )}\n\n {(tools.length > 0 || prompts.length > 0 || resources.length > 0) && (\n <section className={`${baseClass}__section`}>\n <header className={`${baseClass}__section-header`}>\n {/* TODO: needs i18n once design is finalized */}\n <h4>Project-level permissions</h4>\n {/* TODO: needs i18n once design is finalized */}\n <p>Cross-cutting tools, prompts, and resources not scoped to a single collection.</p>\n </header>\n {tools.length > 0 && (\n <Collapsible\n actions={<GroupActions onSetAll={(allow) => setAllFlat('tools', tools, allow)} />}\n className={`${baseClass}__group`}\n header={\n /* TODO: needs i18n once design is finalized */\n <span className={`${baseClass}__group-label`}>Tools</span>\n }\n initCollapsed\n >\n <ul className={`${baseClass}__list`}>\n {tools.map((leaf) => (\n <li key={leaf.configKey}>\n <CheckboxInput\n checked={isFlatAllowed('tools', leaf.configKey)}\n id={`${path}.tools.${leaf.configKey}`}\n label={leaf.label}\n onToggle={(e) => toggleFlat('tools', leaf.configKey, e.target.checked)}\n tooltip={leaf.description}\n />\n </li>\n ))}\n </ul>\n </Collapsible>\n )}\n {prompts.length > 0 && (\n <Collapsible\n actions={<GroupActions onSetAll={(allow) => setAllFlat('prompts', prompts, allow)} />}\n className={`${baseClass}__group`}\n header={\n /* TODO: needs i18n once design is finalized */\n <span className={`${baseClass}__group-label`}>Prompts</span>\n }\n initCollapsed\n >\n <ul className={`${baseClass}__list`}>\n {prompts.map((leaf) => (\n <li key={leaf.configKey}>\n <CheckboxInput\n checked={isFlatAllowed('prompts', leaf.configKey)}\n id={`${path}.prompts.${leaf.configKey}`}\n label={leaf.label}\n onToggle={(e) => toggleFlat('prompts', leaf.configKey, e.target.checked)}\n tooltip={leaf.description}\n />\n </li>\n ))}\n </ul>\n </Collapsible>\n )}\n {resources.length > 0 && (\n <Collapsible\n actions={\n <GroupActions onSetAll={(allow) => setAllFlat('resources', resources, allow)} />\n }\n className={`${baseClass}__group`}\n header={\n /* TODO: needs i18n once design is finalized */\n <span className={`${baseClass}__group-label`}>Resources</span>\n }\n initCollapsed\n >\n <ul className={`${baseClass}__list`}>\n {resources.map((leaf) => (\n <li key={leaf.configKey}>\n <CheckboxInput\n checked={isFlatAllowed('resources', leaf.configKey)}\n id={`${path}.resources.${leaf.configKey}`}\n label={leaf.label}\n onToggle={(e) => toggleFlat('resources', leaf.configKey, e.target.checked)}\n tooltip={leaf.description}\n />\n </li>\n ))}\n </ul>\n </Collapsible>\n )}\n </section>\n )}\n </div>\n )\n}\n\nconst GroupActions: React.FC<{ onSetAll: (allow: boolean) => void }> = ({ onSetAll }) => (\n // TODO: button labels + aria-labels need i18n once design is finalized\n <div className={`${baseClass}__group-actions`}>\n <button\n aria-label=\"Select all\"\n className={`${baseClass}__action`}\n onClick={(e) => {\n e.stopPropagation()\n onSetAll(true)\n }}\n title=\"Select all\"\n type=\"button\"\n >\n all\n </button>\n <span aria-hidden className={`${baseClass}__action-sep`}>\n /\n </span>\n <button\n aria-label=\"Clear all\"\n className={`${baseClass}__action`}\n onClick={(e) => {\n e.stopPropagation()\n onSetAll(false)\n }}\n title=\"Clear all\"\n type=\"button\"\n >\n none\n </button>\n </div>\n)\n\nconst titleCase = (slug: string): string =>\n slug.replace(/(^|[-_])(.)/g, (_, sep: string, ch: string) =>\n sep ? ` ${ch.toUpperCase()}` : ch.toUpperCase(),\n )\n"],"names":["CheckboxInput","Collapsible","useField","React","baseClass","without","obj","key","_omitted","rest","Object","keys","length","undefined","setKey","value","AccessField","path","pluginConfig","setValue","access","collectionsBySlug","globalsBySlug","tools","prompts","resources","item","items","type","collectionSlug","push","globalSlug","isScopedAllowed","scope","slug","configKey","isFlatAllowed","toggleScoped","allow","slugBucket","scopeBucket","toggleFlat","bucket","setAllScoped","leaves","reduce","acc","leaf","setAllFlat","collectionSlugs","globalSlugs","div","className","section","header","h4","p","map","actions","GroupActions","onSetAll","span","titleCase","initCollapsed","ul","li","checked","id","label","onToggle","e","target","tooltip","description","button","aria-label","onClick","stopPropagation","title","aria-hidden","replace","_","sep","ch","toUpperCase"],"mappings":"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"}
1
+ {"version":3,"sources":["../../../src/components/AccessField/index.client.tsx"],"sourcesContent":["'use client'\n\nimport type { JSONFieldClientProps } from 'payload'\n\nimport { CheckboxInput, Collapsible, Tabs, useField, useTranslation } from '@payloadcms/ui'\nimport React, { useState } from 'react'\n\nimport type {\n PluginMCPTranslationKeys,\n PluginMCPTranslations,\n} from '../../translations/index.js'\nimport type { ClientMCPPluginConfig, MCPAPIKeysDocAccessTree } from '../../types.js'\n\nimport './index.css'\n\nconst baseClass = 'mcp-access-field'\n\ntype ClientItem = ClientMCPPluginConfig['items'][number]\ntype ScopeKey = 'collections' | 'globals'\ntype FlatKey = 'prompts' | 'resources' | 'tools'\ntype TabKey = 'collections' | 'globals' | 'server'\n\nconst LEAF_GROUP_KEYS = ['operations', 'auth', 'custom'] as const\n\ntype LeafGroupKey = (typeof LEAF_GROUP_KEYS)[number]\n\ntype Props = {\n pluginConfig: ClientMCPPluginConfig\n} & JSONFieldClientProps\n\n/** Drop a key from an object and return a new object — or `undefined` if it'd be empty. */\nconst without = <T extends Record<string, unknown>>(\n obj: T | undefined,\n key: string,\n): T | undefined => {\n if (!obj || !(key in obj)) {\n return obj\n }\n const { [key]: _omitted, ...rest } = obj\n return Object.keys(rest).length === 0 ? undefined : (rest as T)\n}\n\nconst setKey = <T extends Record<string, unknown>>(\n obj: T | undefined,\n key: string,\n value: unknown,\n): T => ({ ...(obj ?? {}), [key]: value }) as T\n\nexport const AccessField: React.FC<Props> = ({ path, pluginConfig }) => {\n const { setValue, value } = useField<MCPAPIKeysDocAccessTree>({ path })\n const { t } = useTranslation<PluginMCPTranslations, PluginMCPTranslationKeys>()\n const [activeTab, setActiveTab] = useState<null | TabKey>(null)\n const access = value ?? {}\n const leafGroupLabels: Record<LeafGroupKey, string> = {\n auth: t('plugin-mcp:authentication'),\n custom: t('general:custom'),\n operations: t('plugin-mcp:operations'),\n }\n\n const collectionsBySlug: Record<string, ClientItem[]> = {}\n const globalsBySlug: Record<string, ClientItem[]> = {}\n const prompts: ClientItem[] = []\n const resources: ClientItem[] = []\n const tools: ClientItem[] = []\n\n for (const item of pluginConfig.items) {\n switch (item.type) {\n case 'collectionTool':\n ;(collectionsBySlug[item.collectionSlug!] ??= []).push(item)\n break\n case 'globalTool':\n ;(globalsBySlug[item.globalSlug!] ??= []).push(item)\n break\n case 'prompt':\n prompts.push(item)\n break\n case 'resource':\n resources.push(item)\n break\n case 'tool':\n tools.push(item)\n break\n }\n }\n\n const isScopedAllowed = (scope: ScopeKey, slug: string, configKey: string): boolean =>\n access[scope]?.[slug]?.[configKey] !== false\n\n const isFlatAllowed = (scope: FlatKey, configKey: string): boolean =>\n access[scope]?.[configKey] !== false\n\n const toggleScoped = (scope: ScopeKey, slug: string, configKey: string, allow: boolean) => {\n if (allow) {\n const slugBucket = without(access[scope]?.[slug], configKey)\n const scopeBucket = slugBucket\n ? setKey(access[scope], slug, slugBucket)\n : without(access[scope], slug)\n setValue(scopeBucket ? setKey(access, scope, scopeBucket) : (without(access, scope) ?? {}))\n } else {\n setValue(\n setKey(\n access,\n scope,\n setKey(access[scope], slug, setKey(access[scope]?.[slug], configKey, false)),\n ),\n )\n }\n }\n\n const toggleFlat = (scope: FlatKey, configKey: string, allow: boolean) => {\n if (allow) {\n const bucket = without(access[scope], configKey)\n setValue(bucket ? setKey(access, scope, bucket) : (without(access, scope) ?? {}))\n } else {\n setValue(setKey(access, scope, setKey(access[scope], configKey, false)))\n }\n }\n\n const setAllScoped = (scope: ScopeKey, slug: string, leaves: ClientItem[], allow: boolean) => {\n if (allow) {\n const scopeBucket = without(access[scope], slug)\n setValue(scopeBucket ? setKey(access, scope, scopeBucket) : (without(access, scope) ?? {}))\n } else {\n const slugBucket = leaves.reduce<Record<string, boolean>>(\n (acc, leaf) => ({ ...acc, [leaf.configKey]: false }),\n {},\n )\n setValue(setKey(access, scope, setKey(access[scope], slug, slugBucket)))\n }\n }\n\n const setAllFlat = (scope: FlatKey, leaves: ClientItem[], allow: boolean) => {\n if (allow) {\n setValue(without(access, scope) ?? {})\n } else {\n const bucket = leaves.reduce<Record<string, boolean>>(\n (acc, leaf) => ({ ...acc, [leaf.configKey]: false }),\n {},\n )\n setValue(setKey(access, scope, bucket))\n }\n }\n\n const renderLeaf = (\n leaf: ClientItem,\n id: string,\n checked: boolean,\n onToggle: (allow: boolean) => void,\n ) => (\n <li className={`${baseClass}__leaf`} key={leaf.configKey}>\n <CheckboxInput\n checked={checked}\n id={id}\n label={leaf.label}\n onToggle={(e) => onToggle(e.target.checked)}\n />\n {leaf.description && <p className={`${baseClass}__leaf-description`}>{leaf.description}</p>}\n </li>\n )\n\n const renderCard = ({\n id,\n isLeafAllowed,\n label,\n leaves,\n onSetAll,\n onToggleLeaf,\n }: {\n id: string\n isLeafAllowed: (leaf: ClientItem) => boolean\n label: string\n leaves: ClientItem[]\n onSetAll: (allow: boolean) => void\n onToggleLeaf: (leaf: ClientItem, allow: boolean) => void\n }) => {\n const allowedCount = leaves.filter(isLeafAllowed).length\n const groups = LEAF_GROUP_KEYS.map((key) => ({\n key,\n label: leafGroupLabels[key],\n leaves: leaves.filter((leaf) => (leaf.group ?? 'custom') === key),\n })).filter((group) => group.leaves.length > 0)\n const hasGroupLabels = groups.length > 1\n\n return (\n <Collapsible\n className={`${baseClass}__card`}\n header={\n // Keep header clicks on the checkbox from also toggling the collapsible.\n <span\n className={`${baseClass}__card-checkbox`}\n onClick={(e) => e.stopPropagation()}\n role=\"presentation\"\n >\n <CheckboxInput\n checked={allowedCount === leaves.length}\n id={`${id}._all`}\n label={label}\n onToggle={() => onSetAll(allowedCount < leaves.length)}\n partialChecked={allowedCount > 0 && allowedCount < leaves.length}\n />\n </span>\n }\n initCollapsed\n key={id}\n >\n <div className={`${baseClass}__card-groups`}>\n {groups.map((group) => (\n <div className={`${baseClass}__leaf-group`} key={group.key}>\n {hasGroupLabels && <p className={`${baseClass}__leaf-group-label`}>{group.label}</p>}\n <ul className={`${baseClass}__list`}>\n {group.leaves.map((leaf) =>\n renderLeaf(leaf, `${id}.${leaf.configKey}`, isLeafAllowed(leaf), (allow) =>\n onToggleLeaf(leaf, allow),\n ),\n )}\n </ul>\n </div>\n ))}\n </div>\n </Collapsible>\n )\n }\n\n const renderScope = (scope: ScopeKey, bySlug: Record<string, ClientItem[]>) =>\n Object.entries(bySlug).map(([slug, leaves]) =>\n renderCard({\n id: `${path}.${scope}.${slug}`,\n isLeafAllowed: (leaf) => isScopedAllowed(scope, slug, leaf.configKey),\n label: titleCase(slug),\n leaves,\n onSetAll: (allow) => setAllScoped(scope, slug, leaves, allow),\n onToggleLeaf: (leaf, allow) => toggleScoped(scope, slug, leaf.configKey, allow),\n }),\n )\n\n const renderFlat = (scope: FlatKey, label: string, leaves: ClientItem[]) =>\n leaves.length > 0 &&\n renderCard({\n id: `${path}.${scope}`,\n isLeafAllowed: (leaf) => isFlatAllowed(scope, leaf.configKey),\n label,\n leaves,\n onSetAll: (allow) => setAllFlat(scope, leaves, allow),\n onToggleLeaf: (leaf, allow) => toggleFlat(scope, leaf.configKey, allow),\n })\n\n const tabs: Array<{ key: TabKey; label: string }> = [\n ...(Object.keys(collectionsBySlug).length > 0\n ? [{ key: 'collections' as const, label: t('general:collections') }]\n : []),\n ...(Object.keys(globalsBySlug).length > 0\n ? [{ key: 'globals' as const, label: t('general:globals') }]\n : []),\n ...(prompts.length > 0 || resources.length > 0 || tools.length > 0\n ? [{ key: 'server' as const, label: t('plugin-mcp:server') }]\n : []),\n ]\n if (tabs.length === 0) {\n return null\n }\n\n const currentTab = activeTab ?? tabs[0]!.key\n\n return (\n <div className={baseClass}>\n <header className={`${baseClass}__header`}>\n <h4>{t('plugin-mcp:permissions')}</h4>\n <p>{t('plugin-mcp:permissionsDescription')}</p>\n </header>\n <div className={`${baseClass}__tabbed-content`}>\n <Tabs\n className={`${baseClass}__tabs`}\n onChange={setActiveTab}\n tabs={tabs.map((tab) => ({\n label: tab.label,\n value: tab.key,\n }))}\n value={currentTab}\n />\n <div className={`${baseClass}__cards`}>\n {currentTab === 'collections' && renderScope('collections', collectionsBySlug)}\n {currentTab === 'globals' && renderScope('globals', globalsBySlug)}\n {currentTab === 'server' && (\n <>\n {renderFlat('prompts', t('plugin-mcp:prompts'), prompts)}\n {renderFlat('resources', t('plugin-mcp:resources'), resources)}\n {renderFlat('tools', t('plugin-mcp:tools'), tools)}\n </>\n )}\n </div>\n </div>\n </div>\n )\n}\n\nconst titleCase = (slug: string): string =>\n slug.replace(/(^|[-_])(.)/g, (_, sep: string, ch: string) =>\n sep ? ` ${ch.toUpperCase()}` : ch.toUpperCase(),\n )\n"],"names":["CheckboxInput","Collapsible","Tabs","useField","useTranslation","React","useState","baseClass","LEAF_GROUP_KEYS","without","obj","key","_omitted","rest","Object","keys","length","undefined","setKey","value","AccessField","path","pluginConfig","setValue","t","activeTab","setActiveTab","access","leafGroupLabels","auth","custom","operations","collectionsBySlug","globalsBySlug","prompts","resources","tools","item","items","type","collectionSlug","push","globalSlug","isScopedAllowed","scope","slug","configKey","isFlatAllowed","toggleScoped","allow","slugBucket","scopeBucket","toggleFlat","bucket","setAllScoped","leaves","reduce","acc","leaf","setAllFlat","renderLeaf","id","checked","onToggle","li","className","label","e","target","description","p","renderCard","isLeafAllowed","onSetAll","onToggleLeaf","allowedCount","filter","groups","map","group","hasGroupLabels","header","span","onClick","stopPropagation","role","partialChecked","initCollapsed","div","ul","renderScope","bySlug","entries","titleCase","renderFlat","tabs","currentTab","h4","onChange","tab","replace","_","sep","ch","toUpperCase"],"mappings":"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"}
@@ -3,32 +3,22 @@
3
3
  display: flex;
4
4
  flex-direction: column;
5
5
  gap: var(--base);
6
+ /* Breathing room below the API-key section's divider - the fields
7
+ container itself adds no gap between these two fields. */
8
+ padding-block-start: var(--spacer-4);
9
+ /* This custom field renders straight into `render-fields` (no `.field-type`
10
+ wrapper), so it must carry the standard inter-field margin itself -
11
+ otherwise the next field (e.g. an overrideApiKeyCollection field) butts
12
+ against the last card. */
13
+ margin-block-end: var(--spacer-3);
6
14
  }
7
15
 
8
- .mcp-access-field__section {
9
- display: flex;
10
- flex-direction: column;
11
- gap: calc(var(--base) / 2);
12
- padding-bottom: var(--base);
13
- /* Full-bleed separator: extend past the sidebar's horizontal gutter so the
14
- border-block-end spans the entire sidebar width. */
15
- margin-inline-start: calc(-1 * var(--sidebar-gutter-h-left, var(--gutter-h)));
16
- margin-inline-end: calc(-1 * var(--sidebar-gutter-h-right, var(--gutter-h)));
17
- padding-inline-start: var(--sidebar-gutter-h-left, var(--gutter-h));
18
- padding-inline-end: var(--sidebar-gutter-h-right, var(--gutter-h));
19
- border-block-end: 1px solid var(--theme-elevation-100);
20
- }
21
-
22
- .mcp-access-field__section:last-child {
23
- border-block-end: none;
24
- }
25
-
26
- .mcp-access-field__section-header {
16
+ .mcp-access-field__header {
27
17
  display: flex;
28
18
  flex-direction: column;
29
19
  }
30
20
 
31
- .mcp-access-field__section-header h4 {
21
+ .mcp-access-field__header h4 {
32
22
  margin: 0;
33
23
  color: var(--color-text);
34
24
  font-family: var(--text-heading-small-font-family);
@@ -38,7 +28,7 @@
38
28
  letter-spacing: var(--text-heading-small-letter-spacing);
39
29
  }
40
30
 
41
- .mcp-access-field__section-header p {
31
+ .mcp-access-field__header p {
42
32
  margin: 0;
43
33
  color: var(--color-text-secondary);
44
34
  font-family: var(--text-body-medium-font-family);
@@ -48,38 +38,46 @@
48
38
  letter-spacing: var(--text-body-medium-letter-spacing);
49
39
  }
50
40
 
51
- .mcp-access-field__group {
52
- margin-block-end: calc(var(--base) / 2);
41
+ .mcp-access-field__tabbed-content {
42
+ display: flex;
43
+ flex-direction: column;
53
44
  }
54
45
 
55
- .mcp-access-field__group-label {
56
- font-weight: 600;
46
+ .mcp-access-field__cards {
47
+ display: flex;
48
+ flex-direction: column;
49
+ gap: calc(var(--base) / 2);
57
50
  }
58
51
 
59
- .mcp-access-field__group-actions {
60
- display: flex;
61
- align-items: center;
62
- gap: calc(var(--base) / 6);
52
+ /* The collapsible header is pointer-events: none so clicks reach the toggle
53
+ overlay; re-enable events on the checkbox and lift it above the toggle. */
54
+ .mcp-access-field__card-checkbox {
55
+ position: relative;
56
+ z-index: 1;
57
+ display: inline-flex;
58
+ pointer-events: auto;
63
59
  }
64
60
 
65
- .mcp-access-field__action {
66
- appearance: none;
67
- background: none;
68
- border: none;
69
- color: var(--theme-elevation-500);
70
- cursor: pointer;
71
- font-size: 0.75rem;
72
- padding: 0;
61
+ .mcp-access-field__leaf-group {
62
+ padding-block: calc(var(--base) / 2);
63
+ border-block-end: 1px solid var(--theme-elevation-100);
64
+ }
65
+
66
+ .mcp-access-field__leaf-group:first-child {
67
+ padding-block-start: 0;
73
68
  }
74
69
 
75
- .mcp-access-field__action:hover {
76
- color: var(--theme-text);
77
- text-decoration: underline;
70
+ .mcp-access-field__leaf-group:last-child {
71
+ border-block-end: none;
72
+ padding-block-end: 0;
78
73
  }
79
74
 
80
- .mcp-access-field__action-sep {
81
- color: var(--theme-elevation-300);
82
- font-size: 0.75rem;
75
+ .mcp-access-field__leaf-group-label {
76
+ margin: 0 0 calc(var(--base) / 2);
77
+ color: var(--color-text-secondary);
78
+ font-family: var(--text-body-medium-font-family);
79
+ font-size: var(--text-body-medium-font-size);
80
+ line-height: var(--text-body-medium-line-height);
83
81
  }
84
82
 
85
83
  .mcp-access-field__list {
@@ -88,6 +86,14 @@
88
86
  padding: 0;
89
87
  display: flex;
90
88
  flex-direction: column;
91
- gap: calc(var(--base) / 3);
89
+ gap: calc(var(--base) / 2);
90
+ }
91
+
92
+ .mcp-access-field__leaf-description {
93
+ margin: calc(var(--base) / 6) 0 0;
94
+ color: var(--color-text-secondary);
95
+ font-family: var(--text-body-medium-font-family);
96
+ font-size: var(--text-body-medium-font-size);
97
+ line-height: var(--text-body-medium-line-height);
92
98
  }
93
99
  }
@@ -0,0 +1,8 @@
1
+ import React from 'react';
2
+ /**
3
+ * Entry in the user menu's Settings sub-popup linking to the API keys
4
+ * collection. The collection is excluded from the main nav
5
+ * (`admin.group: false`) and managed from here instead.
6
+ */
7
+ export declare const MCPSettingsMenu: React.FC;
8
+ //# sourceMappingURL=index.client.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.client.d.ts","sourceRoot":"","sources":["../../../src/components/SettingsMenu/index.client.tsx"],"names":[],"mappings":"AAIA,OAAO,KAAK,MAAM,OAAO,CAAA;AAOzB;;;;GAIG;AACH,eAAO,MAAM,eAAe,EAAE,KAAK,CAAC,EAiBnC,CAAA"}