@ai-stack/payloadcms 3.2.26 → 3.68.0-beta.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 (313) hide show
  1. package/{LICENSE.md → LICENSE} +1 -1
  2. package/README.md +218 -229
  3. package/dist/access/checkAccess.d.ts +4 -0
  4. package/dist/access/checkAccess.js +20 -0
  5. package/dist/access/checkAccess.js.map +1 -0
  6. package/dist/ai/core/generateObject.d.ts +7 -0
  7. package/dist/ai/core/generateObject.js +35 -0
  8. package/dist/ai/core/generateObject.js.map +1 -0
  9. package/dist/ai/core/generateText.d.ts +7 -0
  10. package/dist/ai/core/generateText.js +31 -0
  11. package/dist/ai/core/generateText.js.map +1 -0
  12. package/dist/ai/core/index.d.ts +11 -0
  13. package/dist/ai/core/index.js +10 -0
  14. package/dist/ai/core/index.js.map +1 -0
  15. package/dist/ai/core/media/generateMedia.d.ts +7 -0
  16. package/dist/ai/core/media/generateMedia.js +50 -0
  17. package/dist/ai/core/media/generateMedia.js.map +1 -0
  18. package/dist/ai/core/media/image/generateImage.d.ts +6 -0
  19. package/dist/ai/core/media/image/generateImage.js +41 -0
  20. package/dist/ai/core/media/image/generateImage.js.map +1 -0
  21. package/dist/ai/core/media/image/handlers/multimodal.d.ts +7 -0
  22. package/dist/ai/core/media/image/handlers/multimodal.js +100 -0
  23. package/dist/ai/core/media/image/handlers/multimodal.js.map +1 -0
  24. package/dist/ai/core/media/image/handlers/standard.d.ts +7 -0
  25. package/dist/ai/core/media/image/handlers/standard.js +28 -0
  26. package/dist/ai/core/media/image/handlers/standard.js.map +1 -0
  27. package/dist/ai/core/media/image/index.d.ts +2 -0
  28. package/dist/ai/core/media/image/index.js +3 -0
  29. package/dist/ai/core/media/image/index.js.map +1 -0
  30. package/dist/ai/core/media/index.d.ts +2 -0
  31. package/dist/ai/core/media/index.js +3 -0
  32. package/dist/ai/core/media/index.js.map +1 -0
  33. package/dist/ai/core/media/speech/generateSpeech.d.ts +5 -0
  34. package/dist/ai/core/media/speech/generateSpeech.js +55 -0
  35. package/dist/ai/core/media/speech/generateSpeech.js.map +1 -0
  36. package/dist/ai/core/media/speech/index.d.ts +2 -0
  37. package/dist/ai/core/media/speech/index.js +3 -0
  38. package/dist/ai/core/media/speech/index.js.map +1 -0
  39. package/dist/ai/core/media/types.d.ts +74 -0
  40. package/dist/ai/core/media/types.js +5 -0
  41. package/dist/ai/core/media/types.js.map +1 -0
  42. package/dist/ai/core/media/utils.d.ts +11 -0
  43. package/dist/ai/core/media/utils.js +34 -0
  44. package/dist/ai/core/media/utils.js.map +1 -0
  45. package/dist/ai/core/media/video/generateVideo.d.ts +6 -0
  46. package/dist/ai/core/media/video/generateVideo.js +32 -0
  47. package/dist/ai/core/media/video/generateVideo.js.map +1 -0
  48. package/dist/ai/core/media/video/index.d.ts +2 -0
  49. package/dist/ai/core/media/video/index.js +3 -0
  50. package/dist/ai/core/media/video/index.js.map +1 -0
  51. package/dist/ai/core/streamObject.d.ts +7 -0
  52. package/dist/ai/core/streamObject.js +54 -0
  53. package/dist/ai/core/streamObject.js.map +1 -0
  54. package/dist/ai/core/streamText.d.ts +7 -0
  55. package/dist/ai/core/streamText.js +30 -0
  56. package/dist/ai/core/streamText.js.map +1 -0
  57. package/dist/ai/core/types.d.ts +85 -0
  58. package/dist/ai/core/types.js +5 -0
  59. package/dist/ai/core/types.js.map +1 -0
  60. package/dist/ai/index.d.ts +11 -0
  61. package/dist/ai/index.js +25 -0
  62. package/dist/ai/index.js.map +1 -0
  63. package/dist/ai/providers/blocks/anthropic.d.ts +2 -0
  64. package/dist/ai/providers/blocks/anthropic.js +223 -0
  65. package/dist/ai/providers/blocks/anthropic.js.map +1 -0
  66. package/dist/ai/providers/blocks/elevenlabs.d.ts +2 -0
  67. package/dist/ai/providers/blocks/elevenlabs.js +449 -0
  68. package/dist/ai/providers/blocks/elevenlabs.js.map +1 -0
  69. package/dist/ai/providers/blocks/fal.d.ts +2 -0
  70. package/dist/ai/providers/blocks/fal.js +312 -0
  71. package/dist/ai/providers/blocks/fal.js.map +1 -0
  72. package/dist/ai/providers/blocks/google.d.ts +2 -0
  73. package/dist/ai/providers/blocks/google.js +623 -0
  74. package/dist/ai/providers/blocks/google.js.map +1 -0
  75. package/dist/ai/providers/blocks/index.d.ts +2 -0
  76. package/dist/ai/providers/blocks/index.js +18 -0
  77. package/dist/ai/providers/blocks/index.js.map +1 -0
  78. package/dist/ai/providers/blocks/openai-compatible.d.ts +2 -0
  79. package/dist/ai/providers/blocks/openai-compatible.js +308 -0
  80. package/dist/ai/providers/blocks/openai-compatible.js.map +1 -0
  81. package/dist/ai/providers/blocks/openai.d.ts +2 -0
  82. package/dist/ai/providers/blocks/openai.js +600 -0
  83. package/dist/ai/providers/blocks/openai.js.map +1 -0
  84. package/dist/ai/providers/blocks/xai.d.ts +2 -0
  85. package/dist/ai/providers/blocks/xai.js +247 -0
  86. package/dist/ai/providers/blocks/xai.js.map +1 -0
  87. package/dist/ai/providers/icons.d.ts +7 -0
  88. package/dist/ai/providers/icons.js +9 -0
  89. package/dist/ai/providers/icons.js.map +1 -0
  90. package/dist/ai/providers/index.d.ts +2 -0
  91. package/dist/ai/providers/index.js +6 -0
  92. package/dist/ai/providers/index.js.map +1 -0
  93. package/dist/ai/providers/registry.d.ts +40 -0
  94. package/dist/ai/providers/registry.js +267 -0
  95. package/dist/ai/providers/registry.js.map +1 -0
  96. package/dist/ai/providers/types.d.ts +115 -0
  97. package/dist/ai/providers/types.js +4 -0
  98. package/dist/ai/providers/types.js.map +1 -0
  99. package/dist/ai/utils/systemGenerate.d.ts +1 -1
  100. package/dist/ai/utils/systemGenerate.js +19 -19
  101. package/dist/ai/utils/systemGenerate.js.map +1 -1
  102. package/dist/collections/AIJobs.d.ts +2 -0
  103. package/dist/collections/AIJobs.js +81 -0
  104. package/dist/collections/AIJobs.js.map +1 -0
  105. package/dist/collections/AISettings.d.ts +2 -0
  106. package/dist/collections/AISettings.js +279 -0
  107. package/dist/collections/AISettings.js.map +1 -0
  108. package/dist/collections/Instructions.js +224 -50
  109. package/dist/collections/Instructions.js.map +1 -1
  110. package/dist/defaults.d.ts +3 -0
  111. package/dist/defaults.js +3 -0
  112. package/dist/defaults.js.map +1 -1
  113. package/dist/endpoints/buildPromptUtils.d.ts +19 -0
  114. package/dist/endpoints/buildPromptUtils.js +114 -0
  115. package/dist/endpoints/buildPromptUtils.js.map +1 -0
  116. package/dist/endpoints/chat.d.ts +4 -0
  117. package/dist/endpoints/fetchFields.js +0 -7
  118. package/dist/endpoints/fetchFields.js.map +1 -1
  119. package/dist/endpoints/fetchVoices.d.ts +2 -0
  120. package/dist/endpoints/fetchVoices.js +79 -0
  121. package/dist/endpoints/fetchVoices.js.map +1 -0
  122. package/dist/endpoints/index.js +339 -232
  123. package/dist/endpoints/index.js.map +1 -1
  124. package/dist/exports/client.d.ts +9 -0
  125. package/dist/exports/client.js +9 -0
  126. package/dist/exports/client.js.map +1 -1
  127. package/dist/exports/fields.d.ts +1 -0
  128. package/dist/exports/fields.js +1 -0
  129. package/dist/exports/fields.js.map +1 -1
  130. package/dist/fields/ArrayComposeField/ArrayComposeField.d.ts +15 -0
  131. package/dist/fields/ArrayComposeField/ArrayComposeField.js +87 -0
  132. package/dist/fields/ArrayComposeField/ArrayComposeField.js.map +1 -0
  133. package/dist/fields/ArrayComposeField/ArrayComposeField.jsx +73 -0
  134. package/dist/fields/ComposeField/ComposeField.js +2 -2
  135. package/dist/fields/ComposeField/ComposeField.js.map +1 -1
  136. package/dist/fields/ComposeField/ComposeField.jsx +2 -2
  137. package/dist/fields/PromptEditorField/PromptEditorField.js +162 -16
  138. package/dist/fields/PromptEditorField/PromptEditorField.js.map +1 -1
  139. package/dist/fields/PromptEditorField/PromptEditorField.jsx +123 -5
  140. package/dist/index.d.ts +3 -0
  141. package/dist/index.js +1 -0
  142. package/dist/index.js.map +1 -1
  143. package/dist/init.js +63 -65
  144. package/dist/init.js.map +1 -1
  145. package/dist/payload-ai.d.ts +149 -0
  146. package/dist/plugin.js +94 -46
  147. package/dist/plugin.js.map +1 -1
  148. package/dist/providers/InstructionsProvider/InstructionsProvider.js +38 -7
  149. package/dist/providers/InstructionsProvider/InstructionsProvider.js.map +1 -1
  150. package/dist/providers/InstructionsProvider/InstructionsProvider.jsx +30 -4
  151. package/dist/providers/InstructionsProvider/context.d.ts +1 -0
  152. package/dist/providers/InstructionsProvider/context.js +1 -0
  153. package/dist/providers/InstructionsProvider/context.js.map +1 -1
  154. package/dist/providers/InstructionsProvider/useInstructions.js +30 -10
  155. package/dist/providers/InstructionsProvider/useInstructions.js.map +1 -1
  156. package/dist/styles.d.ts +11 -0
  157. package/dist/types/handlebars-async-helpers.d.ts +1 -0
  158. package/dist/types/handlebars-dist-handlebars.d.ts +1 -0
  159. package/dist/types/react-mentions.d.ts +1 -0
  160. package/dist/types.d.ts +6 -16
  161. package/dist/types.js.map +1 -1
  162. package/dist/ui/AIConfigDashboard/index.d.ts +2 -0
  163. package/dist/ui/AIConfigDashboard/index.js +46 -0
  164. package/dist/ui/AIConfigDashboard/index.js.map +1 -0
  165. package/dist/ui/AIConfigDashboard/index.jsx +24 -0
  166. package/dist/ui/ApiKeyStatusIndicator/index.d.ts +6 -0
  167. package/dist/ui/ApiKeyStatusIndicator/index.js +39 -0
  168. package/dist/ui/ApiKeyStatusIndicator/index.js.map +1 -0
  169. package/dist/ui/ApiKeyStatusIndicator/index.jsx +29 -0
  170. package/dist/ui/Compose/Compose.d.ts +2 -2
  171. package/dist/ui/Compose/Compose.js +118 -92
  172. package/dist/ui/Compose/Compose.js.map +1 -1
  173. package/dist/ui/Compose/Compose.jsx +113 -103
  174. package/dist/ui/Compose/ComposePlaceholder.d.ts +7 -0
  175. package/dist/ui/Compose/ComposePlaceholder.js +78 -0
  176. package/dist/ui/Compose/ComposePlaceholder.js.map +1 -0
  177. package/dist/ui/Compose/ComposePlaceholder.jsx +66 -0
  178. package/dist/ui/Compose/UndoRedoActions.d.ts +2 -2
  179. package/dist/ui/Compose/UndoRedoActions.js +11 -6
  180. package/dist/ui/Compose/UndoRedoActions.js.map +1 -1
  181. package/dist/ui/Compose/UndoRedoActions.jsx +8 -6
  182. package/dist/ui/Compose/compose.module.css +57 -17
  183. package/dist/ui/Compose/hooks/menu/itemsMap.js +13 -7
  184. package/dist/ui/Compose/hooks/menu/itemsMap.js.map +1 -1
  185. package/dist/ui/Compose/hooks/menu/useMenu.d.ts +2 -1
  186. package/dist/ui/Compose/hooks/menu/useMenu.js +28 -17
  187. package/dist/ui/Compose/hooks/menu/useMenu.js.map +1 -1
  188. package/dist/ui/Compose/hooks/menu/useMenu.jsx +27 -14
  189. package/dist/ui/Compose/hooks/useActiveFieldTracking.js +69 -10
  190. package/dist/ui/Compose/hooks/useActiveFieldTracking.js.map +1 -1
  191. package/dist/ui/Compose/hooks/useGenerate.d.ts +3 -0
  192. package/dist/ui/Compose/hooks/useGenerate.js +71 -11
  193. package/dist/ui/Compose/hooks/useGenerate.js.map +1 -1
  194. package/dist/ui/Compose/hooks/useHistory.d.ts +0 -1
  195. package/dist/ui/Compose/hooks/useHistory.js +113 -26
  196. package/dist/ui/Compose/hooks/useHistory.js.map +1 -1
  197. package/dist/ui/DynamicModelSelect/index.d.ts +7 -0
  198. package/dist/ui/DynamicModelSelect/index.js +231 -0
  199. package/dist/ui/DynamicModelSelect/index.js.map +1 -0
  200. package/dist/ui/DynamicModelSelect/index.jsx +207 -0
  201. package/dist/ui/DynamicProviderSelect/index.d.ts +7 -0
  202. package/dist/ui/DynamicProviderSelect/index.js +101 -0
  203. package/dist/ui/DynamicProviderSelect/index.js.map +1 -0
  204. package/dist/ui/DynamicProviderSelect/index.jsx +90 -0
  205. package/dist/ui/DynamicVoiceSelect/index.d.ts +7 -0
  206. package/dist/ui/DynamicVoiceSelect/index.js +156 -0
  207. package/dist/ui/DynamicVoiceSelect/index.js.map +1 -0
  208. package/dist/ui/DynamicVoiceSelect/index.jsx +102 -0
  209. package/dist/ui/EncryptedTextField/index.d.ts +8 -0
  210. package/dist/ui/EncryptedTextField/index.js +74 -0
  211. package/dist/ui/EncryptedTextField/index.js.map +1 -0
  212. package/dist/ui/EncryptedTextField/index.jsx +35 -0
  213. package/dist/ui/Icons/LottieAnimation.js +3 -1
  214. package/dist/ui/Icons/LottieAnimation.js.map +1 -1
  215. package/dist/ui/Icons/LottieAnimation.jsx +2 -1
  216. package/dist/ui/ModelRowLabel/index.d.ts +6 -0
  217. package/dist/ui/ModelRowLabel/index.js +41 -0
  218. package/dist/ui/ModelRowLabel/index.js.map +1 -0
  219. package/dist/ui/ModelRowLabel/index.jsx +26 -0
  220. package/dist/ui/ProviderOptionsEditor/index.d.ts +7 -0
  221. package/dist/ui/ProviderOptionsEditor/index.js +291 -0
  222. package/dist/ui/ProviderOptionsEditor/index.js.map +1 -0
  223. package/dist/ui/ProviderOptionsEditor/index.jsx +210 -0
  224. package/dist/ui/VoicesFetcher/index.d.ts +7 -0
  225. package/dist/ui/VoicesFetcher/index.js +118 -0
  226. package/dist/ui/VoicesFetcher/index.js.map +1 -0
  227. package/dist/ui/VoicesFetcher/index.jsx +79 -0
  228. package/dist/utilities/buildSmartPrompt.d.ts +22 -0
  229. package/dist/utilities/buildSmartPrompt.js +143 -0
  230. package/dist/utilities/buildSmartPrompt.js.map +1 -0
  231. package/dist/utilities/encryption.d.ts +2 -0
  232. package/dist/utilities/encryption.js +47 -0
  233. package/dist/utilities/encryption.js.map +1 -0
  234. package/dist/utilities/extractImageData.d.ts +9 -0
  235. package/dist/utilities/extractImageData.js +12 -2
  236. package/dist/utilities/extractImageData.js.map +1 -1
  237. package/dist/utilities/fetchImages.d.ts +14 -0
  238. package/dist/utilities/fetchImages.js +38 -0
  239. package/dist/utilities/fetchImages.js.map +1 -0
  240. package/dist/utilities/fieldToJsonSchema.d.ts +2 -1
  241. package/dist/utilities/fieldToJsonSchema.js +66 -3
  242. package/dist/utilities/fieldToJsonSchema.js.map +1 -1
  243. package/dist/utilities/getFieldBySchemaPath.d.ts +2 -2
  244. package/dist/utilities/getFieldBySchemaPath.js +15 -0
  245. package/dist/utilities/getFieldBySchemaPath.js.map +1 -1
  246. package/dist/utilities/getProviderOptionsFields.d.ts +16 -0
  247. package/dist/utilities/getProviderOptionsFields.js +80 -0
  248. package/dist/utilities/getProviderOptionsFields.js.map +1 -0
  249. package/dist/utilities/isPluginActivated.js +1 -2
  250. package/dist/utilities/isPluginActivated.js.map +1 -1
  251. package/dist/utilities/lexicalToHTML.js.map +1 -1
  252. package/dist/utilities/resolveImageReferences.d.ts +30 -0
  253. package/dist/utilities/resolveImageReferences.js +167 -0
  254. package/dist/utilities/resolveImageReferences.js.map +1 -0
  255. package/dist/utilities/schemaConverter.d.ts +3 -0
  256. package/dist/utilities/schemaConverter.js +93 -0
  257. package/dist/utilities/schemaConverter.js.map +1 -0
  258. package/dist/utilities/setSafeLexicalState.d.ts +1 -3
  259. package/dist/utilities/setSafeLexicalState.js +1 -1
  260. package/dist/utilities/setSafeLexicalState.js.map +1 -1
  261. package/dist/utilities/updateFieldsConfig.js +27 -43
  262. package/dist/utilities/updateFieldsConfig.js.map +1 -1
  263. package/package.json +23 -24
  264. package/dist/ai/models/anthropic/index.d.ts +0 -2
  265. package/dist/ai/models/anthropic/index.js +0 -129
  266. package/dist/ai/models/anthropic/index.js.map +0 -1
  267. package/dist/ai/models/elevenLabs/generateVoice.d.ts +0 -8
  268. package/dist/ai/models/elevenLabs/generateVoice.js +0 -20
  269. package/dist/ai/models/elevenLabs/generateVoice.js.map +0 -1
  270. package/dist/ai/models/elevenLabs/index.d.ts +0 -2
  271. package/dist/ai/models/elevenLabs/index.js +0 -133
  272. package/dist/ai/models/elevenLabs/index.js.map +0 -1
  273. package/dist/ai/models/elevenLabs/voices.d.ts +0 -8
  274. package/dist/ai/models/elevenLabs/voices.js +0 -24
  275. package/dist/ai/models/elevenLabs/voices.js.map +0 -1
  276. package/dist/ai/models/generateObject.d.ts +0 -11
  277. package/dist/ai/models/generateObject.js +0 -22
  278. package/dist/ai/models/generateObject.js.map +0 -1
  279. package/dist/ai/models/google/generateImage.d.ts +0 -9
  280. package/dist/ai/models/google/generateImage.js +0 -27
  281. package/dist/ai/models/google/generateImage.js.map +0 -1
  282. package/dist/ai/models/google/index.d.ts +0 -2
  283. package/dist/ai/models/google/index.js +0 -201
  284. package/dist/ai/models/google/index.js.map +0 -1
  285. package/dist/ai/models/index.d.ts +0 -2
  286. package/dist/ai/models/index.js +0 -13
  287. package/dist/ai/models/index.js.map +0 -1
  288. package/dist/ai/models/openai/generateImage.d.ts +0 -5
  289. package/dist/ai/models/openai/generateImage.js +0 -31
  290. package/dist/ai/models/openai/generateImage.js.map +0 -1
  291. package/dist/ai/models/openai/generateVoice.d.ts +0 -6
  292. package/dist/ai/models/openai/generateVoice.js +0 -19
  293. package/dist/ai/models/openai/generateVoice.js.map +0 -1
  294. package/dist/ai/models/openai/index.d.ts +0 -2
  295. package/dist/ai/models/openai/index.js +0 -428
  296. package/dist/ai/models/openai/index.js.map +0 -1
  297. package/dist/ai/models/openai/openai.d.ts +0 -1
  298. package/dist/ai/models/openai/openai.js +0 -8
  299. package/dist/ai/models/openai/openai.js.map +0 -1
  300. package/dist/ai/utils/editImagesWithOpenAI.d.ts +0 -10
  301. package/dist/ai/utils/editImagesWithOpenAI.js +0 -37
  302. package/dist/ai/utils/editImagesWithOpenAI.js.map +0 -1
  303. package/dist/styles.d.js +0 -2
  304. package/dist/styles.d.js.map +0 -1
  305. package/dist/types/handlebars-async-helpers.d.js +0 -2
  306. package/dist/types/handlebars-async-helpers.d.js.map +0 -1
  307. package/dist/types/handlebars-dist-handlebars.d.js +0 -2
  308. package/dist/types/handlebars-dist-handlebars.d.js.map +0 -1
  309. package/dist/types/react-mentions.d.js +0 -2
  310. package/dist/types/react-mentions.d.js.map +0 -1
  311. package/dist/utilities/getGenerationModels.d.ts +0 -2
  312. package/dist/utilities/getGenerationModels.js +0 -10
  313. package/dist/utilities/getGenerationModels.js.map +0 -1
@@ -0,0 +1,26 @@
1
+ 'use client';
2
+ import { useRowLabel } from '@payloadcms/ui';
3
+ import React from 'react';
4
+ /**
5
+ * Generic Model Row Label component for provider blocks
6
+ * Displays model name with enabled status badge
7
+ */
8
+ export const ModelRowLabel = () => {
9
+ const { data, rowNumber } = useRowLabel();
10
+ const displayName = data?.name || data?.id || `Model ${rowNumber}`;
11
+ const isEnabled = data?.enabled !== false;
12
+ return (<div style={{ alignItems: 'center', display: 'flex', gap: '10px' }}>
13
+ <span style={{ fontWeight: '500' }}>{displayName}</span>
14
+ <span style={{
15
+ backgroundColor: isEnabled ? 'var(--theme-success-100)' : 'var(--theme-elevation-200)',
16
+ borderRadius: '4px',
17
+ color: isEnabled ? 'var(--theme-success-700)' : 'var(--theme-elevation-600)',
18
+ fontSize: '11px',
19
+ fontWeight: '600',
20
+ padding: '2px 8px',
21
+ textTransform: 'uppercase',
22
+ }}>
23
+ {isEnabled ? '●Enabled' : 'Disabled'}
24
+ </span>
25
+ </div>);
26
+ };
@@ -0,0 +1,7 @@
1
+ import React from 'react';
2
+ interface ProviderOptionsEditorProps {
3
+ name?: string;
4
+ path: string;
5
+ }
6
+ export declare const ProviderOptionsEditor: React.FC<ProviderOptionsEditorProps>;
7
+ export {};
@@ -0,0 +1,291 @@
1
+ 'use client';
2
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
3
+ import { RenderFields, useField, useFormFields } from '@payloadcms/ui';
4
+ import React, { useEffect, useMemo, useState } from 'react';
5
+ import { allProviderBlocks } from '../../ai/providers/blocks/index.js';
6
+ /**
7
+ * Find a field by name within a block's fields, searching through tabs
8
+ */ function findFieldInBlock(block, fieldName) {
9
+ const searchFields = (fields)=>{
10
+ for (const field of fields){
11
+ if ('name' in field && field.name === fieldName) {
12
+ return field;
13
+ }
14
+ if (field.type === 'tabs' && 'tabs' in field) {
15
+ for (const tab of field.tabs){
16
+ const found = searchFields(tab.fields);
17
+ if (found) {
18
+ return found;
19
+ }
20
+ }
21
+ }
22
+ if (field.type === 'group' && 'fields' in field) {
23
+ const found = searchFields(field.fields);
24
+ if (found) {
25
+ return found;
26
+ }
27
+ }
28
+ }
29
+ return undefined;
30
+ };
31
+ return searchFields(block.fields);
32
+ }
33
+ /**
34
+ * Get provider options fields for a given provider and use case
35
+ */ function getProviderOptionsFields(providerSlug, useCase) {
36
+ const block = allProviderBlocks.find((b)=>b.slug === providerSlug);
37
+ if (!block) {
38
+ return [];
39
+ }
40
+ const groupName = `${useCase}ProviderOptions`;
41
+ const optionsGroup = findFieldInBlock(block, groupName);
42
+ if (optionsGroup && optionsGroup.type === 'group' && 'fields' in optionsGroup) {
43
+ return optionsGroup.fields;
44
+ }
45
+ return [];
46
+ }
47
+ export const ProviderOptionsEditor = (props)=>{
48
+ const { path } = props;
49
+ // Get parent path to find sibling provider field
50
+ const parentPath = path.split('.').slice(0, -1).join('.');
51
+ const providerField = useFormFields(([fields])=>fields[`${parentPath}.provider`]);
52
+ const provider = providerField?.value;
53
+ // Infer use case from path
54
+ // Handles:
55
+ // - AISettings: 'defaults.text.options' -> useCase is 'text'
56
+ // - Instructions: 'text-settings.providerOptions' -> useCase is 'text'
57
+ const useCase = useMemo(()=>{
58
+ // Check for AISettings paths first (e.g., 'defaults.text.options')
59
+ const pathParts = path.split('.');
60
+ const parentName = pathParts[pathParts.length - 2];
61
+ if ([
62
+ 'image',
63
+ 'text',
64
+ 'tts',
65
+ 'video'
66
+ ].includes(parentName)) {
67
+ return parentName;
68
+ }
69
+ // Check for Instructions paths
70
+ if (path.includes('tts-settings')) {
71
+ return 'tts';
72
+ }
73
+ if (path.includes('image-settings')) {
74
+ return 'image';
75
+ }
76
+ if (path.includes('video-settings')) {
77
+ return 'video';
78
+ }
79
+ return 'text';
80
+ }, [
81
+ path
82
+ ]);
83
+ const { setValue, value } = useField({
84
+ path
85
+ });
86
+ const [aiSettings, setAiSettings] = useState(null);
87
+ // Fetch AI Settings to get current provider defaults
88
+ useEffect(()=>{
89
+ fetch('/api/globals/ai-settings?depth=1').then((res)=>res.json()).then((data)=>setAiSettings(data)).catch((err)=>console.error('Error fetching AI settings:', err));
90
+ }, []);
91
+ // Get the configured default options from AI Settings for this provider
92
+ const configuredDefaults = useMemo(()=>{
93
+ if (!provider || !aiSettings) {
94
+ return null;
95
+ }
96
+ const providerBlock = aiSettings.providers?.find((p)=>p.blockType === provider && p.enabled);
97
+ if (!providerBlock) {
98
+ return null;
99
+ }
100
+ // Get provider options by use case
101
+ const optionsKey = `${useCase}ProviderOptions`;
102
+ return providerBlock[optionsKey] || null;
103
+ }, [
104
+ provider,
105
+ useCase,
106
+ aiSettings
107
+ ]);
108
+ // Get field definitions from provider block
109
+ const fields = useMemo(()=>{
110
+ if (!provider) {
111
+ return [];
112
+ }
113
+ return getProviderOptionsFields(provider, useCase);
114
+ }, [
115
+ provider,
116
+ useCase
117
+ ]);
118
+ // Check if there are any overrides set
119
+ const hasOverrides = useMemo(()=>{
120
+ return value && Object.keys(value).length > 0;
121
+ }, [
122
+ value
123
+ ]);
124
+ if (!provider) {
125
+ return /*#__PURE__*/ _jsx("div", {
126
+ className: "field-type",
127
+ style: {
128
+ padding: '12px 0'
129
+ },
130
+ children: /*#__PURE__*/ _jsx("p", {
131
+ style: {
132
+ color: 'var(--theme-elevation-600)',
133
+ fontSize: '13px',
134
+ margin: 0
135
+ },
136
+ children: "Please select a provider first to configure options."
137
+ })
138
+ });
139
+ }
140
+ if (fields.length === 0) {
141
+ return /*#__PURE__*/ _jsx("div", {
142
+ className: "field-type",
143
+ style: {
144
+ padding: '12px 0'
145
+ },
146
+ children: /*#__PURE__*/ _jsxs("p", {
147
+ style: {
148
+ color: 'var(--theme-elevation-600)',
149
+ fontSize: '13px',
150
+ margin: 0
151
+ },
152
+ children: [
153
+ "No configurable options available for ",
154
+ provider,
155
+ " (",
156
+ useCase,
157
+ ")."
158
+ ]
159
+ })
160
+ });
161
+ }
162
+ return /*#__PURE__*/ _jsxs("div", {
163
+ className: "field-type provider-options-editor",
164
+ children: [
165
+ /*#__PURE__*/ _jsxs("div", {
166
+ style: {
167
+ marginBottom: '16px'
168
+ },
169
+ children: [
170
+ /*#__PURE__*/ _jsx("label", {
171
+ className: "field-label",
172
+ style: {
173
+ display: 'block',
174
+ marginBottom: '8px'
175
+ },
176
+ children: "Provider Options"
177
+ }),
178
+ configuredDefaults && /*#__PURE__*/ _jsxs("div", {
179
+ style: {
180
+ background: 'var(--theme-elevation-50)',
181
+ border: '1px solid var(--theme-elevation-100)',
182
+ borderRadius: '4px',
183
+ fontSize: '12px',
184
+ marginBottom: '12px',
185
+ padding: '12px'
186
+ },
187
+ children: [
188
+ /*#__PURE__*/ _jsxs("div", {
189
+ style: {
190
+ alignItems: 'center',
191
+ display: 'flex',
192
+ justifyContent: 'space-between',
193
+ marginBottom: '8px'
194
+ },
195
+ children: [
196
+ /*#__PURE__*/ _jsx("strong", {
197
+ style: {
198
+ color: 'var(--theme-elevation-800)'
199
+ },
200
+ children: "Defaults from AI Settings"
201
+ }),
202
+ /*#__PURE__*/ _jsx("span", {
203
+ style: {
204
+ background: 'var(--theme-elevation-100)',
205
+ borderRadius: '10px',
206
+ color: 'var(--theme-elevation-500)',
207
+ fontSize: '11px',
208
+ padding: '2px 8px'
209
+ },
210
+ children: "Inherited"
211
+ })
212
+ ]
213
+ }),
214
+ /*#__PURE__*/ _jsx("div", {
215
+ style: {
216
+ color: 'var(--theme-elevation-600)',
217
+ display: 'flex',
218
+ flexWrap: 'wrap',
219
+ gap: '8px'
220
+ },
221
+ children: Object.entries(configuredDefaults).map(([key, val])=>{
222
+ // Skip nested objects for display
223
+ if (typeof val === 'object' && val !== null) {
224
+ return null;
225
+ }
226
+ return /*#__PURE__*/ _jsxs("span", {
227
+ style: {
228
+ background: 'var(--theme-elevation-100)',
229
+ borderRadius: '3px',
230
+ fontSize: '11px',
231
+ padding: '2px 6px'
232
+ },
233
+ children: [
234
+ key,
235
+ ": ",
236
+ /*#__PURE__*/ _jsx("strong", {
237
+ children: String(val)
238
+ })
239
+ ]
240
+ }, key);
241
+ })
242
+ })
243
+ ]
244
+ }),
245
+ /*#__PURE__*/ _jsx("p", {
246
+ style: {
247
+ color: 'var(--theme-elevation-500)',
248
+ fontSize: '12px',
249
+ fontStyle: 'italic',
250
+ marginBottom: '12px'
251
+ },
252
+ children: "Override defaults for this specific field. Empty values inherit from AI Settings."
253
+ })
254
+ ]
255
+ }),
256
+ /*#__PURE__*/ _jsx(RenderFields, {
257
+ fields: fields,
258
+ forceRender: true,
259
+ margins: "small",
260
+ parentIndexPath: "",
261
+ parentPath: path,
262
+ parentSchemaPath: path,
263
+ permissions: true
264
+ }),
265
+ hasOverrides && /*#__PURE__*/ _jsx("button", {
266
+ onClick: ()=>setValue({}),
267
+ onMouseEnter: (e)=>{
268
+ e.currentTarget.style.background = 'var(--theme-elevation-100)';
269
+ },
270
+ onMouseLeave: (e)=>{
271
+ e.currentTarget.style.background = 'transparent';
272
+ },
273
+ style: {
274
+ background: 'transparent',
275
+ border: '1px solid var(--theme-elevation-200)',
276
+ borderRadius: '4px',
277
+ color: 'var(--theme-text)',
278
+ cursor: 'pointer',
279
+ fontSize: '13px',
280
+ marginTop: '12px',
281
+ padding: '8px 16px',
282
+ transition: 'all 0.15s ease'
283
+ },
284
+ type: "button",
285
+ children: "Reset to Defaults"
286
+ })
287
+ ]
288
+ });
289
+ };
290
+
291
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../src/ui/ProviderOptionsEditor/index.tsx"],"sourcesContent":["'use client'\n\nimport { RenderFields, useField, useFormFields } from '@payloadcms/ui'\nimport React, { useEffect, useMemo, useState } from 'react'\n\nimport { allProviderBlocks } from '../../ai/providers/blocks/index.js'\n\ntype UseCase = 'image' | 'text' | 'tts' | 'video'\n\ninterface ProviderOptionsEditorProps {\n name?: string\n path: string\n}\n\n/**\n * Find a field by name within a block's fields, searching through tabs\n */\nfunction findFieldInBlock(block: any, fieldName: string): any | undefined {\n const searchFields = (fields: any[]): any | undefined => {\n for (const field of fields) {\n if ('name' in field && field.name === fieldName) {\n return field\n }\n if (field.type === 'tabs' && 'tabs' in field) {\n for (const tab of field.tabs) {\n const found = searchFields(tab.fields)\n if (found) {\n return found\n }\n }\n }\n if (field.type === 'group' && 'fields' in field) {\n const found = searchFields(field.fields)\n if (found) {\n return found\n }\n }\n }\n return undefined\n }\n\n return searchFields(block.fields)\n}\n\n/**\n * Get provider options fields for a given provider and use case\n */\nfunction getProviderOptionsFields(providerSlug: string, useCase: UseCase): any[] {\n const block = allProviderBlocks.find((b) => b.slug === providerSlug)\n if (!block) {\n return []\n }\n\n const groupName = `${useCase}ProviderOptions`\n const optionsGroup = findFieldInBlock(block, groupName)\n\n if (optionsGroup && optionsGroup.type === 'group' && 'fields' in optionsGroup) {\n return optionsGroup.fields\n }\n\n return []\n}\n\nexport const ProviderOptionsEditor: React.FC<ProviderOptionsEditorProps> = (props) => {\n const { path } = props\n\n // Get parent path to find sibling provider field\n const parentPath = path.split('.').slice(0, -1).join('.')\n const providerField = useFormFields(([fields]) => fields[`${parentPath}.provider`])\n const provider = providerField?.value as string\n\n // Infer use case from path\n // Handles:\n // - AISettings: 'defaults.text.options' -> useCase is 'text'\n // - Instructions: 'text-settings.providerOptions' -> useCase is 'text'\n const useCase: UseCase = useMemo(() => {\n // Check for AISettings paths first (e.g., 'defaults.text.options')\n const pathParts = path.split('.')\n const parentName = pathParts[pathParts.length - 2]\n\n if (['image', 'text', 'tts', 'video'].includes(parentName)) {\n return parentName as UseCase\n }\n\n // Check for Instructions paths\n if (path.includes('tts-settings')) {\n return 'tts'\n }\n if (path.includes('image-settings')) {\n return 'image'\n }\n if (path.includes('video-settings')) {\n return 'video'\n }\n return 'text'\n }, [path])\n\n const { setValue, value } = useField<Record<string, any>>({ path })\n const [aiSettings, setAiSettings] = useState<any>(null)\n\n // Fetch AI Settings to get current provider defaults\n useEffect(() => {\n fetch('/api/globals/ai-settings?depth=1')\n .then((res) => res.json())\n .then((data) => setAiSettings(data))\n .catch((err) => console.error('Error fetching AI settings:', err))\n }, [])\n\n // Get the configured default options from AI Settings for this provider\n const configuredDefaults = useMemo(() => {\n if (!provider || !aiSettings) {\n return null\n }\n\n const providerBlock = aiSettings.providers?.find(\n (p: any) => p.blockType === provider && p.enabled,\n )\n if (!providerBlock) {\n return null\n }\n\n // Get provider options by use case\n const optionsKey = `${useCase}ProviderOptions`\n return providerBlock[optionsKey] || null\n }, [provider, useCase, aiSettings])\n\n // Get field definitions from provider block\n const fields = useMemo(() => {\n if (!provider) {\n return []\n }\n return getProviderOptionsFields(provider, useCase)\n }, [provider, useCase])\n\n // Check if there are any overrides set\n const hasOverrides = useMemo(() => {\n return value && Object.keys(value).length > 0\n }, [value])\n\n if (!provider) {\n return (\n <div className=\"field-type\" style={{ padding: '12px 0' }}>\n <p style={{ color: 'var(--theme-elevation-600)', fontSize: '13px', margin: 0 }}>\n Please select a provider first to configure options.\n </p>\n </div>\n )\n }\n\n if (fields.length === 0) {\n return (\n <div className=\"field-type\" style={{ padding: '12px 0' }}>\n <p style={{ color: 'var(--theme-elevation-600)', fontSize: '13px', margin: 0 }}>\n No configurable options available for {provider} ({useCase}).\n </p>\n </div>\n )\n }\n\n return (\n <div className=\"field-type provider-options-editor\">\n <div style={{ marginBottom: '16px' }}>\n <label className=\"field-label\" style={{ display: 'block', marginBottom: '8px' }}>\n Provider Options\n </label>\n\n {configuredDefaults && (\n <div\n style={{\n background: 'var(--theme-elevation-50)',\n border: '1px solid var(--theme-elevation-100)',\n borderRadius: '4px',\n fontSize: '12px',\n marginBottom: '12px',\n padding: '12px',\n }}\n >\n <div\n style={{\n alignItems: 'center',\n display: 'flex',\n justifyContent: 'space-between',\n marginBottom: '8px',\n }}\n >\n <strong style={{ color: 'var(--theme-elevation-800)' }}>\n Defaults from AI Settings\n </strong>\n <span\n style={{\n background: 'var(--theme-elevation-100)',\n borderRadius: '10px',\n color: 'var(--theme-elevation-500)',\n fontSize: '11px',\n padding: '2px 8px',\n }}\n >\n Inherited\n </span>\n </div>\n <div\n style={{\n color: 'var(--theme-elevation-600)',\n display: 'flex',\n flexWrap: 'wrap',\n gap: '8px',\n }}\n >\n {Object.entries(configuredDefaults).map(([key, val]) => {\n // Skip nested objects for display\n if (typeof val === 'object' && val !== null) {\n return null\n }\n return (\n <span\n key={key}\n style={{\n background: 'var(--theme-elevation-100)',\n borderRadius: '3px',\n fontSize: '11px',\n padding: '2px 6px',\n }}\n >\n {key}: <strong>{String(val)}</strong>\n </span>\n )\n })}\n </div>\n </div>\n )}\n\n <p\n style={{\n color: 'var(--theme-elevation-500)',\n fontSize: '12px',\n fontStyle: 'italic',\n marginBottom: '12px',\n }}\n >\n Override defaults for this specific field. Empty values inherit from AI Settings.\n </p>\n </div>\n\n <RenderFields\n fields={fields}\n forceRender\n margins=\"small\"\n parentIndexPath=\"\"\n parentPath={path}\n parentSchemaPath={path}\n permissions={true}\n />\n\n {hasOverrides && (\n <button\n onClick={() => setValue({})}\n onMouseEnter={(e) => {\n e.currentTarget.style.background = 'var(--theme-elevation-100)'\n }}\n onMouseLeave={(e) => {\n e.currentTarget.style.background = 'transparent'\n }}\n style={{\n background: 'transparent',\n border: '1px solid var(--theme-elevation-200)',\n borderRadius: '4px',\n color: 'var(--theme-text)',\n cursor: 'pointer',\n fontSize: '13px',\n marginTop: '12px',\n padding: '8px 16px',\n transition: 'all 0.15s ease',\n }}\n type=\"button\"\n >\n Reset to Defaults\n </button>\n )}\n </div>\n )\n}\n"],"names":["RenderFields","useField","useFormFields","React","useEffect","useMemo","useState","allProviderBlocks","findFieldInBlock","block","fieldName","searchFields","fields","field","name","type","tab","tabs","found","undefined","getProviderOptionsFields","providerSlug","useCase","find","b","slug","groupName","optionsGroup","ProviderOptionsEditor","props","path","parentPath","split","slice","join","providerField","provider","value","pathParts","parentName","length","includes","setValue","aiSettings","setAiSettings","fetch","then","res","json","data","catch","err","console","error","configuredDefaults","providerBlock","providers","p","blockType","enabled","optionsKey","hasOverrides","Object","keys","div","className","style","padding","color","fontSize","margin","marginBottom","label","display","background","border","borderRadius","alignItems","justifyContent","strong","span","flexWrap","gap","entries","map","key","val","String","fontStyle","forceRender","margins","parentIndexPath","parentSchemaPath","permissions","button","onClick","onMouseEnter","e","currentTarget","onMouseLeave","cursor","marginTop","transition"],"mappings":"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"}
@@ -0,0 +1,210 @@
1
+ 'use client';
2
+ import { RenderFields, useField, useFormFields } from '@payloadcms/ui';
3
+ import React, { useEffect, useMemo, useState } from 'react';
4
+ import { allProviderBlocks } from '../../ai/providers/blocks/index.js';
5
+ /**
6
+ * Find a field by name within a block's fields, searching through tabs
7
+ */
8
+ function findFieldInBlock(block, fieldName) {
9
+ const searchFields = (fields) => {
10
+ for (const field of fields) {
11
+ if ('name' in field && field.name === fieldName) {
12
+ return field;
13
+ }
14
+ if (field.type === 'tabs' && 'tabs' in field) {
15
+ for (const tab of field.tabs) {
16
+ const found = searchFields(tab.fields);
17
+ if (found) {
18
+ return found;
19
+ }
20
+ }
21
+ }
22
+ if (field.type === 'group' && 'fields' in field) {
23
+ const found = searchFields(field.fields);
24
+ if (found) {
25
+ return found;
26
+ }
27
+ }
28
+ }
29
+ return undefined;
30
+ };
31
+ return searchFields(block.fields);
32
+ }
33
+ /**
34
+ * Get provider options fields for a given provider and use case
35
+ */
36
+ function getProviderOptionsFields(providerSlug, useCase) {
37
+ const block = allProviderBlocks.find((b) => b.slug === providerSlug);
38
+ if (!block) {
39
+ return [];
40
+ }
41
+ const groupName = `${useCase}ProviderOptions`;
42
+ const optionsGroup = findFieldInBlock(block, groupName);
43
+ if (optionsGroup && optionsGroup.type === 'group' && 'fields' in optionsGroup) {
44
+ return optionsGroup.fields;
45
+ }
46
+ return [];
47
+ }
48
+ export const ProviderOptionsEditor = (props) => {
49
+ const { path } = props;
50
+ // Get parent path to find sibling provider field
51
+ const parentPath = path.split('.').slice(0, -1).join('.');
52
+ const providerField = useFormFields(([fields]) => fields[`${parentPath}.provider`]);
53
+ const provider = providerField?.value;
54
+ // Infer use case from path
55
+ // Handles:
56
+ // - AISettings: 'defaults.text.options' -> useCase is 'text'
57
+ // - Instructions: 'text-settings.providerOptions' -> useCase is 'text'
58
+ const useCase = useMemo(() => {
59
+ // Check for AISettings paths first (e.g., 'defaults.text.options')
60
+ const pathParts = path.split('.');
61
+ const parentName = pathParts[pathParts.length - 2];
62
+ if (['image', 'text', 'tts', 'video'].includes(parentName)) {
63
+ return parentName;
64
+ }
65
+ // Check for Instructions paths
66
+ if (path.includes('tts-settings')) {
67
+ return 'tts';
68
+ }
69
+ if (path.includes('image-settings')) {
70
+ return 'image';
71
+ }
72
+ if (path.includes('video-settings')) {
73
+ return 'video';
74
+ }
75
+ return 'text';
76
+ }, [path]);
77
+ const { setValue, value } = useField({ path });
78
+ const [aiSettings, setAiSettings] = useState(null);
79
+ // Fetch AI Settings to get current provider defaults
80
+ useEffect(() => {
81
+ fetch('/api/globals/ai-settings?depth=1')
82
+ .then((res) => res.json())
83
+ .then((data) => setAiSettings(data))
84
+ .catch((err) => console.error('Error fetching AI settings:', err));
85
+ }, []);
86
+ // Get the configured default options from AI Settings for this provider
87
+ const configuredDefaults = useMemo(() => {
88
+ if (!provider || !aiSettings) {
89
+ return null;
90
+ }
91
+ const providerBlock = aiSettings.providers?.find((p) => p.blockType === provider && p.enabled);
92
+ if (!providerBlock) {
93
+ return null;
94
+ }
95
+ // Get provider options by use case
96
+ const optionsKey = `${useCase}ProviderOptions`;
97
+ return providerBlock[optionsKey] || null;
98
+ }, [provider, useCase, aiSettings]);
99
+ // Get field definitions from provider block
100
+ const fields = useMemo(() => {
101
+ if (!provider) {
102
+ return [];
103
+ }
104
+ return getProviderOptionsFields(provider, useCase);
105
+ }, [provider, useCase]);
106
+ // Check if there are any overrides set
107
+ const hasOverrides = useMemo(() => {
108
+ return value && Object.keys(value).length > 0;
109
+ }, [value]);
110
+ if (!provider) {
111
+ return (<div className="field-type" style={{ padding: '12px 0' }}>
112
+ <p style={{ color: 'var(--theme-elevation-600)', fontSize: '13px', margin: 0 }}>
113
+ Please select a provider first to configure options.
114
+ </p>
115
+ </div>);
116
+ }
117
+ if (fields.length === 0) {
118
+ return (<div className="field-type" style={{ padding: '12px 0' }}>
119
+ <p style={{ color: 'var(--theme-elevation-600)', fontSize: '13px', margin: 0 }}>
120
+ No configurable options available for {provider} ({useCase}).
121
+ </p>
122
+ </div>);
123
+ }
124
+ return (<div className="field-type provider-options-editor">
125
+ <div style={{ marginBottom: '16px' }}>
126
+ <label className="field-label" style={{ display: 'block', marginBottom: '8px' }}>
127
+ Provider Options
128
+ </label>
129
+
130
+ {configuredDefaults && (<div style={{
131
+ background: 'var(--theme-elevation-50)',
132
+ border: '1px solid var(--theme-elevation-100)',
133
+ borderRadius: '4px',
134
+ fontSize: '12px',
135
+ marginBottom: '12px',
136
+ padding: '12px',
137
+ }}>
138
+ <div style={{
139
+ alignItems: 'center',
140
+ display: 'flex',
141
+ justifyContent: 'space-between',
142
+ marginBottom: '8px',
143
+ }}>
144
+ <strong style={{ color: 'var(--theme-elevation-800)' }}>
145
+ Defaults from AI Settings
146
+ </strong>
147
+ <span style={{
148
+ background: 'var(--theme-elevation-100)',
149
+ borderRadius: '10px',
150
+ color: 'var(--theme-elevation-500)',
151
+ fontSize: '11px',
152
+ padding: '2px 8px',
153
+ }}>
154
+ Inherited
155
+ </span>
156
+ </div>
157
+ <div style={{
158
+ color: 'var(--theme-elevation-600)',
159
+ display: 'flex',
160
+ flexWrap: 'wrap',
161
+ gap: '8px',
162
+ }}>
163
+ {Object.entries(configuredDefaults).map(([key, val]) => {
164
+ // Skip nested objects for display
165
+ if (typeof val === 'object' && val !== null) {
166
+ return null;
167
+ }
168
+ return (<span key={key} style={{
169
+ background: 'var(--theme-elevation-100)',
170
+ borderRadius: '3px',
171
+ fontSize: '11px',
172
+ padding: '2px 6px',
173
+ }}>
174
+ {key}: <strong>{String(val)}</strong>
175
+ </span>);
176
+ })}
177
+ </div>
178
+ </div>)}
179
+
180
+ <p style={{
181
+ color: 'var(--theme-elevation-500)',
182
+ fontSize: '12px',
183
+ fontStyle: 'italic',
184
+ marginBottom: '12px',
185
+ }}>
186
+ Override defaults for this specific field. Empty values inherit from AI Settings.
187
+ </p>
188
+ </div>
189
+
190
+ <RenderFields fields={fields} forceRender margins="small" parentIndexPath="" parentPath={path} parentSchemaPath={path} permissions={true}/>
191
+
192
+ {hasOverrides && (<button onClick={() => setValue({})} onMouseEnter={(e) => {
193
+ e.currentTarget.style.background = 'var(--theme-elevation-100)';
194
+ }} onMouseLeave={(e) => {
195
+ e.currentTarget.style.background = 'transparent';
196
+ }} style={{
197
+ background: 'transparent',
198
+ border: '1px solid var(--theme-elevation-200)',
199
+ borderRadius: '4px',
200
+ color: 'var(--theme-text)',
201
+ cursor: 'pointer',
202
+ fontSize: '13px',
203
+ marginTop: '12px',
204
+ padding: '8px 16px',
205
+ transition: 'all 0.15s ease',
206
+ }} type="button">
207
+ Reset to Defaults
208
+ </button>)}
209
+ </div>);
210
+ };
@@ -0,0 +1,7 @@
1
+ import type { FieldClientComponent } from 'payload';
2
+ /**
3
+ * VoicesFetcher Component
4
+ * Fetches voices from ElevenLabs API (server-side) and populates the voices array field
5
+ * SECURE: API key is never exposed to the client
6
+ */
7
+ export declare const VoicesFetcher: FieldClientComponent;