@ai-stack/payloadcms 3.68.0 → 3.76.0-beta.0

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 (154) hide show
  1. package/dist/ai/core/media/image/generateImage.js +2 -6
  2. package/dist/ai/core/media/image/generateImage.js.map +1 -1
  3. package/dist/ai/core/media/image/handlers/multimodal.js +5 -0
  4. package/dist/ai/core/media/image/handlers/multimodal.js.map +1 -1
  5. package/dist/ai/core/streamObject.js +3 -3
  6. package/dist/ai/core/streamObject.js.map +1 -1
  7. package/dist/ai/core/types.d.ts +3 -0
  8. package/dist/ai/core/types.js.map +1 -1
  9. package/dist/ai/prompts.d.ts +1 -2
  10. package/dist/ai/prompts.js +0 -110
  11. package/dist/ai/prompts.js.map +1 -1
  12. package/dist/ai/providers/blocks/anthropic.js +2 -1
  13. package/dist/ai/providers/blocks/anthropic.js.map +1 -1
  14. package/dist/ai/providers/blocks/elevenlabs.js +3 -2
  15. package/dist/ai/providers/blocks/elevenlabs.js.map +1 -1
  16. package/dist/ai/providers/blocks/fal.js +2 -1
  17. package/dist/ai/providers/blocks/fal.js.map +1 -1
  18. package/dist/ai/providers/blocks/google.js +11 -6
  19. package/dist/ai/providers/blocks/google.js.map +1 -1
  20. package/dist/ai/providers/blocks/openai-compatible.js +2 -1
  21. package/dist/ai/providers/blocks/openai-compatible.js.map +1 -1
  22. package/dist/ai/providers/blocks/openai.js +3 -2
  23. package/dist/ai/providers/blocks/openai.js.map +1 -1
  24. package/dist/ai/providers/blocks/xai.js +2 -1
  25. package/dist/ai/providers/blocks/xai.js.map +1 -1
  26. package/dist/ai/providers/icons.d.ts +7 -0
  27. package/dist/ai/providers/icons.js +9 -0
  28. package/dist/ai/providers/icons.js.map +1 -0
  29. package/dist/ai/providers/registry.js +34 -24
  30. package/dist/ai/providers/registry.js.map +1 -1
  31. package/dist/ai/utils/filterEditorSchemaByNodes.d.ts +9 -0
  32. package/dist/ai/utils/filterEditorSchemaByNodes.js +30 -3
  33. package/dist/ai/utils/filterEditorSchemaByNodes.js.map +1 -1
  34. package/dist/ai/utils/nodeToSchemaMap.d.ts +22 -0
  35. package/dist/ai/utils/nodeToSchemaMap.js +72 -0
  36. package/dist/ai/utils/nodeToSchemaMap.js.map +1 -0
  37. package/dist/collections/AIJobs.js +1 -1
  38. package/dist/collections/AIJobs.js.map +1 -1
  39. package/dist/collections/AISettings.js +47 -20
  40. package/dist/collections/AISettings.js.map +1 -1
  41. package/dist/collections/Instructions.js +37 -0
  42. package/dist/collections/Instructions.js.map +1 -1
  43. package/dist/defaults.d.ts +1 -0
  44. package/dist/defaults.js +8 -0
  45. package/dist/defaults.js.map +1 -1
  46. package/dist/endpoints/chat.d.ts +4 -0
  47. package/dist/endpoints/fetchFields.js +10 -0
  48. package/dist/endpoints/fetchFields.js.map +1 -1
  49. package/dist/endpoints/fetchVoices.js +41 -24
  50. package/dist/endpoints/fetchVoices.js.map +1 -1
  51. package/dist/endpoints/index.js +194 -16
  52. package/dist/endpoints/index.js.map +1 -1
  53. package/dist/exports/fields.d.ts +1 -0
  54. package/dist/exports/fields.js +1 -0
  55. package/dist/exports/fields.js.map +1 -1
  56. package/dist/fields/ArrayComposeField/ArrayComposeField.d.ts +15 -0
  57. package/dist/fields/ArrayComposeField/ArrayComposeField.js +87 -0
  58. package/dist/fields/ArrayComposeField/ArrayComposeField.js.map +1 -0
  59. package/dist/fields/ArrayComposeField/ArrayComposeField.jsx +73 -0
  60. package/dist/fields/PromptEditorField/PromptEditorField.js +7 -2
  61. package/dist/fields/PromptEditorField/PromptEditorField.js.map +1 -1
  62. package/dist/fields/PromptEditorField/PromptEditorField.jsx +5 -2
  63. package/dist/index.d.ts +3 -1
  64. package/dist/index.js +2 -1
  65. package/dist/index.js.map +1 -1
  66. package/dist/payload-ai.d.ts +152 -0
  67. package/dist/plugin.js +16 -32
  68. package/dist/plugin.js.map +1 -1
  69. package/dist/providers/InstructionsProvider/InstructionsProvider.js +47 -15
  70. package/dist/providers/InstructionsProvider/InstructionsProvider.js.map +1 -1
  71. package/dist/providers/InstructionsProvider/InstructionsProvider.jsx +39 -16
  72. package/dist/providers/InstructionsProvider/context.d.ts +3 -0
  73. package/dist/providers/InstructionsProvider/context.js +2 -0
  74. package/dist/providers/InstructionsProvider/context.js.map +1 -1
  75. package/dist/providers/InstructionsProvider/useInstructions.js +21 -2
  76. package/dist/providers/InstructionsProvider/useInstructions.js.map +1 -1
  77. package/dist/styles.d.ts +11 -0
  78. package/dist/types/handlebars-async-helpers.d.ts +1 -0
  79. package/dist/types/handlebars-dist-handlebars.d.ts +1 -0
  80. package/dist/types/react-mentions.d.ts +1 -0
  81. package/dist/types.d.ts +34 -5
  82. package/dist/types.js +1 -0
  83. package/dist/types.js.map +1 -1
  84. package/dist/ui/AIConfigDashboard/index.js +198 -22
  85. package/dist/ui/AIConfigDashboard/index.js.map +1 -1
  86. package/dist/ui/AIConfigDashboard/index.jsx +159 -13
  87. package/dist/ui/Compose/Compose.d.ts +1 -0
  88. package/dist/ui/Compose/Compose.js +23 -4
  89. package/dist/ui/Compose/Compose.js.map +1 -1
  90. package/dist/ui/Compose/Compose.jsx +23 -4
  91. package/dist/ui/Compose/UndoRedoActions.d.ts +2 -2
  92. package/dist/ui/Compose/UndoRedoActions.js +8 -5
  93. package/dist/ui/Compose/UndoRedoActions.js.map +1 -1
  94. package/dist/ui/Compose/UndoRedoActions.jsx +6 -5
  95. package/dist/ui/Compose/compose.module.css +56 -16
  96. package/dist/ui/Compose/hooks/menu/itemsMap.js +12 -6
  97. package/dist/ui/Compose/hooks/menu/itemsMap.js.map +1 -1
  98. package/dist/ui/Compose/hooks/menu/useMenu.js +26 -15
  99. package/dist/ui/Compose/hooks/menu/useMenu.js.map +1 -1
  100. package/dist/ui/Compose/hooks/menu/useMenu.jsx +25 -12
  101. package/dist/ui/Compose/hooks/useGenerate.js +26 -174
  102. package/dist/ui/Compose/hooks/useGenerate.js.map +1 -1
  103. package/dist/ui/Compose/hooks/useGenerateUpload.d.ts +11 -0
  104. package/dist/ui/Compose/hooks/useGenerateUpload.js +150 -0
  105. package/dist/ui/Compose/hooks/useGenerateUpload.js.map +1 -0
  106. package/dist/ui/Compose/hooks/useHistory.d.ts +0 -1
  107. package/dist/ui/Compose/hooks/useHistory.js +65 -25
  108. package/dist/ui/Compose/hooks/useHistory.js.map +1 -1
  109. package/dist/ui/Compose/hooks/useStreamingUpdate.d.ts +8 -0
  110. package/dist/ui/Compose/hooks/useStreamingUpdate.js +48 -0
  111. package/dist/ui/Compose/hooks/useStreamingUpdate.js.map +1 -0
  112. package/dist/ui/DynamicVoiceSelect/index.js +63 -11
  113. package/dist/ui/DynamicVoiceSelect/index.js.map +1 -1
  114. package/dist/ui/DynamicVoiceSelect/index.jsx +47 -14
  115. package/dist/ui/EncryptedTextField/index.js +4 -4
  116. package/dist/ui/EncryptedTextField/index.js.map +1 -1
  117. package/dist/ui/EncryptedTextField/index.jsx +4 -4
  118. package/dist/ui/VoicesFetcher/index.js +34 -16
  119. package/dist/ui/VoicesFetcher/index.js.map +1 -1
  120. package/dist/ui/VoicesFetcher/index.jsx +32 -15
  121. package/dist/utilities/buildSmartPrompt.d.ts +22 -0
  122. package/dist/utilities/buildSmartPrompt.js +141 -0
  123. package/dist/utilities/buildSmartPrompt.js.map +1 -0
  124. package/dist/utilities/encryption.js +2 -1
  125. package/dist/utilities/encryption.js.map +1 -1
  126. package/dist/utilities/fieldToJsonSchema.js +32 -3
  127. package/dist/utilities/fieldToJsonSchema.js.map +1 -1
  128. package/dist/utilities/resolveImageReferences.d.ts +3 -1
  129. package/dist/utilities/resolveImageReferences.js +21 -2
  130. package/dist/utilities/resolveImageReferences.js.map +1 -1
  131. package/dist/utilities/seedProperties.d.ts +7 -0
  132. package/dist/utilities/seedProperties.js +100 -0
  133. package/dist/utilities/seedProperties.js.map +1 -0
  134. package/dist/utilities/setSafeLexicalState.js +79 -6
  135. package/dist/utilities/setSafeLexicalState.js.map +1 -1
  136. package/dist/utilities/updateFieldsConfig.d.ts +1 -1
  137. package/dist/utilities/updateFieldsConfig.js +8 -1
  138. package/dist/utilities/updateFieldsConfig.js.map +1 -1
  139. package/package.json +35 -33
  140. package/dist/endpoints/chat.d.js +0 -3
  141. package/dist/endpoints/chat.d.js.map +0 -1
  142. package/dist/init.d.ts +0 -7
  143. package/dist/init.js +0 -135
  144. package/dist/init.js.map +0 -1
  145. package/dist/payload-ai.d.js +0 -3
  146. package/dist/payload-ai.d.js.map +0 -1
  147. package/dist/styles.d.js +0 -2
  148. package/dist/styles.d.js.map +0 -1
  149. package/dist/types/handlebars-async-helpers.d.js +0 -2
  150. package/dist/types/handlebars-async-helpers.d.js.map +0 -1
  151. package/dist/types/handlebars-dist-handlebars.d.js +0 -2
  152. package/dist/types/handlebars-dist-handlebars.d.js.map +0 -1
  153. package/dist/types/react-mentions.d.js +0 -2
  154. package/dist/types/react-mentions.d.js.map +0 -1
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/endpoints/index.ts"],"sourcesContent":["import type { ImagePart } from 'ai'\nimport type { PayloadRequest } from 'payload'\n\nimport * as process from 'node:process'\n\nimport type { Endpoints, PluginConfig } from '../types.js'\n\nimport { checkAccess } from '../access/checkAccess.js'\nimport { filterEditorSchemaByNodes } from '../ai/utils/filterEditorSchemaByNodes.js'\nimport {\n PLUGIN_AI_JOBS_TABLE,\n PLUGIN_API_ENDPOINT_GENERATE,\n PLUGIN_API_ENDPOINT_GENERATE_UPLOAD,\n PLUGIN_API_ENDPOINT_VIDEOGEN_WEBHOOK,\n PLUGIN_INSTRUCTIONS_TABLE,\n PLUGIN_NAME,\n} from '../defaults.js'\nimport { registerEditorHelper } from '../libraries/handlebars/helpers.js'\nimport { replacePlaceholders } from '../libraries/handlebars/replacePlaceholders.js'\nimport { extractImageData } from '../utilities/extractImageData.js'\nimport { type FetchableImage, fetchImages } from '../utilities/fetchImages.js'\nimport { fieldToJsonSchema } from '../utilities/fieldToJsonSchema.js'\nimport { getFieldBySchemaPath } from '../utilities/getFieldBySchemaPath.js'\nimport { resolveImageReferences } from '../utilities/resolveImageReferences.js'\nimport { assignPrompt, extendContextWithPromptFields } from './buildPromptUtils.js'\n\nexport const endpoints: (pluginConfig: PluginConfig) => Endpoints = (pluginConfig) =>\n ({\n textarea: {\n // Text/rich-text generation endpoint using payload.ai.streamObject\n handler: async (req: PayloadRequest) => {\n try {\n // Check authentication and authorization first\n await checkAccess(req, pluginConfig)\n\n const data = await req.json?.()\n\n const { allowedEditorNodes = [], locale = 'en', options } = data\n const { action, actionParams, instructionId } = options\n const contextData = data.doc\n\n if (!instructionId) {\n throw new Error(\n `Instruction ID is required for \"${PLUGIN_NAME}\" to work, please check your configuration, or try again`,\n )\n }\n\n // Verify user has access to the specific instruction\n const instructions = await req.payload.findByID({\n id: instructionId,\n collection: PLUGIN_INSTRUCTIONS_TABLE,\n req, // Pass req to ensure access control is applied\n })\n\n const { collections } = req.payload.config\n const collection = collections.find(\n (collection) => collection.slug === PLUGIN_INSTRUCTIONS_TABLE,\n )\n\n if (!collection) {\n throw new Error('Collection not found')\n }\n\n const { custom: { [PLUGIN_NAME]: { editorConfig = {} } = {} } = {} } = collection.admin\n const { schema: editorSchema = {} } = editorConfig\n const { prompt: promptTemplate = '' } = instructions\n\n let allowedEditorSchema = editorSchema\n if (allowedEditorNodes.length) {\n allowedEditorSchema = filterEditorSchemaByNodes(editorSchema, allowedEditorNodes)\n }\n\n const schemaPath = String(instructions['schema-path'])\n const parts = (schemaPath || '').split('.') || []\n const collectionName = parts[0]\n const fieldName = parts.length > 1 ? parts[parts.length - 1] : ''\n\n registerEditorHelper(req.payload, schemaPath)\n\n const { defaultLocale, locales = [] } = req.payload.config.localization || {}\n const localeData = locales.find((l) => {\n return l.code === locale\n })\n\n let localeInfo = locale\n if (\n localeData &&\n defaultLocale &&\n localeData.label &&\n typeof localeData.label === 'object' &&\n defaultLocale in localeData.label\n ) {\n localeInfo = localeData.label[defaultLocale]\n }\n\n const prompts = await assignPrompt(action, {\n type: String(instructions['field-type']),\n actionParams,\n collection: collectionName,\n context: contextData,\n field: fieldName || '',\n layout: instructions.layout,\n locale: localeInfo,\n pluginConfig,\n systemPrompt: instructions.system,\n template: String(promptTemplate),\n })\n\n if (pluginConfig.debugging) {\n req.payload.logger.info(\n { prompts },\n `— AI Plugin: Executing text prompt on ${schemaPath}`,\n )\n }\n\n // Build per-field JSON schema for structured generation when applicable\n let jsonSchema = allowedEditorSchema\n try {\n const targetCollection = req.payload.config.collections.find(\n (c) => c.slug === collectionName,\n )\n if (targetCollection && fieldName) {\n const targetField = getFieldBySchemaPath(targetCollection, schemaPath)\n const supported = [\n 'text',\n 'textarea',\n 'select',\n 'number',\n 'date',\n 'code',\n 'email',\n 'json',\n ]\n const t = String(targetField?.type || '')\n if (targetField && supported.includes(t)) {\n jsonSchema = fieldToJsonSchema(targetField, { nameOverride: fieldName })\n }\n }\n } catch (e) {\n req.payload.logger.error(e, '— AI Plugin: Error building field JSON schema')\n }\n\n // Get model settings from instruction\n const settingsName =\n instructions['model-id'] === 'richtext'\n ? 'richtext-settings'\n : instructions['model-id'] === 'text'\n ? 'text-settings'\n : undefined\n\n if (!settingsName) {\n throw new Error(`Unsupported model-id: ${instructions['model-id']}`)\n }\n\n const modelSettings = instructions[settingsName] || {}\n\n // Resolve @field:filename references from the prompt\n const { images: resolvedImages, processedPrompt } = await resolveImageReferences(\n prompts.prompt,\n contextData,\n req,\n )\n\n console.log('resolvedImagesL ', resolvedImages)\n\n // Extract hardcoded URLs from the processed prompt\n const hardcodedImages = extractImageData(processedPrompt)\n\n // Combine images\n const allImages = [...hardcodedImages, ...resolvedImages] as FetchableImage[]\n\n let images: ImagePart[] | undefined\n\n if (allImages.length > 0) {\n const imageParts = await fetchImages(req, allImages)\n\n if (imageParts.length > 0) {\n images = imageParts\n }\n }\n\n // Use payload.ai.streamObject directly! 🎉\n const streamResult = await req.payload.ai.streamObject({\n // extractAttachments: modelSettings.extractAttachments as boolean | undefined,\n images,\n maxTokens: modelSettings.maxTokens as number | undefined,\n model: modelSettings.model as string,\n prompt: processedPrompt,\n provider: modelSettings.provider as string,\n schema: jsonSchema,\n system: prompts.system,\n temperature: modelSettings.temperature as number | undefined,\n })\n\n return streamResult\n } catch (error) {\n req.payload.logger.error(error, 'Error generating content: ')\n const message =\n error && typeof error === 'object' && 'message' in error\n ? (error as Error).message\n : String(error)\n return new Response(JSON.stringify({ error: message }), {\n headers: { 'Content-Type': 'application/json' },\n status:\n message.includes('Authentication required') ||\n message.includes('Insufficient permissions')\n ? 401\n : 500,\n })\n }\n },\n method: 'post',\n path: PLUGIN_API_ENDPOINT_GENERATE,\n },\n upload: {\n // Image/video generation endpoint using payload.ai.generateMedia\n handler: async (req: PayloadRequest) => {\n try {\n // Check authentication and authorization first\n await checkAccess(req, pluginConfig)\n\n const data = await req.json?.()\n\n const { collectionSlug, documentId, options } = data\n const { instructionId } = options\n let docData = {}\n\n if (documentId) {\n try {\n docData = await req.payload.findByID({\n id: documentId,\n collection: collectionSlug,\n draft: true,\n req, // Pass req to ensure access control is applied\n })\n } catch (e) {\n req.payload.logger.error(\n e,\n '— AI Plugin: Error fetching document, you should try again after enabling drafts for this collection',\n )\n }\n }\n\n const contextData = {\n ...data.doc,\n ...docData,\n }\n\n let instructions: Record<string, unknown> = { images: [], 'model-id': '', prompt: '' }\n\n if (instructionId) {\n // Verify user has access to the specific instruction\n instructions = await req.payload.findByID({\n id: instructionId,\n collection: PLUGIN_INSTRUCTIONS_TABLE,\n req, // Pass req to ensure access control is applied\n })\n }\n\n const { images: sampleImages = [], prompt: promptTemplate = '' } = instructions\n const schemaPath = String(instructions['schema-path'])\n registerEditorHelper(req.payload, schemaPath)\n\n const extendedContext = extendContextWithPromptFields(\n contextData,\n { type: String(instructions['field-type']), collection: collectionSlug },\n pluginConfig,\n )\n const text = await replacePlaceholders(promptTemplate as string, extendedContext)\n const modelId = instructions['model-id']\n const uploadCollectionSlug = instructions['relation-to']\n\n // Resolve @field:filename references from the prompt\n const { images: resolvedImages, processedPrompt } = await resolveImageReferences(\n text,\n contextData,\n req,\n )\n\n // Extract hardcoded URLs from the processed prompt and merge with resolved images and sample images\n const images = [\n ...extractImageData(processedPrompt),\n ...resolvedImages,\n ...(sampleImages as unknown[]),\n ] as FetchableImage[]\n\n // Process images - convert to ImagePart format using helper\n const editImages: ImagePart[] = await fetchImages(req, images)\n\n if (pluginConfig.debugging) {\n req.payload.logger.info({ text }, `— AI Plugin: Executing media generation`)\n }\n\n // Prepare callback URL for async jobs\n const serverURL =\n req.payload.config?.serverURL ||\n process.env.SERVER_URL ||\n process.env.NEXT_PUBLIC_SERVER_URL\n\n const callbackUrl = serverURL\n ? `${serverURL.replace(/\\/$/, '')}/api${PLUGIN_API_ENDPOINT_VIDEOGEN_WEBHOOK}?instructionId=${instructionId}`\n : undefined\n\n // Get model settings\n const settingsName =\n modelId === 'image'\n ? 'image-settings'\n : modelId === 'video'\n ? 'video-settings'\n : modelId === 'tts'\n ? 'tts-settings'\n : undefined\n if (!settingsName) {\n throw new Error(`Unsupported model-id: ${modelId}`)\n }\n\n const modelSettings = instructions[settingsName] || {}\n\n // Use payload.ai.generateMedia directly! 🎉\n const result = await req.payload.ai.generateMedia({\n callbackUrl,\n images: editImages,\n instructionId,\n model: (modelSettings as Record<string, unknown>).model as string,\n prompt: text,\n provider: (modelSettings as Record<string, unknown>).provider as string,\n ...(modelSettings as Record<string, unknown>),\n })\n\n // If model returned a file immediately, proceed with upload\n if (result && 'file' in result) {\n let assetData: { alt?: string; id: number | string }\n if (typeof pluginConfig.mediaUpload === 'function') {\n assetData = await pluginConfig.mediaUpload(\n result,\n {\n collection: uploadCollectionSlug as string,\n request: req,\n },\n )\n } else {\n assetData = await req.payload.create({\n collection: uploadCollectionSlug as string,\n data: { alt: text },\n file: result.file,\n req, // Pass req to ensure access control is applied\n })\n }\n\n if (!assetData.id) {\n req.payload.logger.error(\n 'Error uploading generated media, is your media upload function correct?',\n )\n throw new Error('Error uploading generated media!')\n }\n\n return new Response(\n JSON.stringify({\n result: {\n id: assetData.id,\n alt: assetData.alt,\n },\n }),\n )\n }\n\n // Otherwise, assume async job launch\n if (result && ('jobId' in result || 'taskId' in result)) {\n const externalTaskId = result.jobId || result.taskId\n const status = result.status || 'queued'\n const progress = result.progress ?? 0\n\n // Create AI Job doc and return only its id\n const createdJob = await req.payload.create({\n collection: PLUGIN_AI_JOBS_TABLE,\n data: {\n instructionId,\n progress,\n status,\n task_id: externalTaskId,\n },\n overrideAccess: true,\n req,\n })\n\n return new Response(JSON.stringify({ job: { id: createdJob.id } }), {\n headers: { 'Content-Type': 'application/json' },\n })\n }\n\n throw new Error('Unexpected model response.')\n } catch (error) {\n req.payload.logger.error(\n // @ts-expect-error\n error?.type || (error as Error).message,\n 'Error generating upload: ',\n )\n const message =\n error && typeof error === 'object' && 'message' in error\n ? (error as Error).message\n : String(error)\n return new Response(JSON.stringify({ error: message }), {\n headers: { 'Content-Type': 'application/json' },\n status:\n message.includes('Authentication required') ||\n message.includes('Insufficient permissions')\n ? 401\n : 500,\n })\n }\n },\n method: 'post',\n path: PLUGIN_API_ENDPOINT_GENERATE_UPLOAD,\n },\n videogenWebhook: {\n handler: async (req: PayloadRequest) => {\n console.log('videogenWebhook --> ', req)\n try {\n const urlAll = new URL(req.url || '')\n const qpSecret = urlAll.searchParams.get('secret') || ''\n const headerSecret = req.headers.get('x-webhook-secret') || ''\n const falSecret = process.env.FAL_WEBHOOK_SECRET\n const legacySecret = process.env.VIDEOGEN_WEBHOOK_SECRET\n const provided = qpSecret || headerSecret\n // TODO: fal is failing because of auth but webhook seem to work\n if (!provided || (falSecret ? provided !== falSecret : provided !== legacySecret)) {\n return new Response('Unauthorized', { status: 401 })\n }\n\n const instructionId = urlAll.searchParams.get('instructionId')\n if (!instructionId) {\n throw new Error('instructionId missing')\n }\n\n const body = await req.json?.()\n // Normalize fal webhook payload\n const status: string | undefined =\n (body && (body.status || body.data?.status || body.response?.status)) || undefined\n const progress: number | undefined =\n (body && (body.progress ?? body.data?.progress ?? body.response?.progress)) ?? undefined\n const requestId: string | undefined =\n (body &&\n (body.taskId ||\n body.request_id ||\n body.gateway_request_id ||\n body.request?.request_id)) ||\n undefined\n const error = body?.error || body?.data?.error || body?.response?.error\n\n // Update AI Job row by task_id (and instructionId)\n const jobSearch = await req.payload.find({\n collection: PLUGIN_AI_JOBS_TABLE,\n depth: 0,\n limit: 1,\n where: {\n and: [\n { task_id: { equals: requestId } },\n { instructionId: { equals: instructionId } },\n ],\n },\n })\n\n const jobDoc = jobSearch.docs?.[0]\n if (jobDoc) {\n await req.payload.update({\n id: jobDoc.id,\n collection: PLUGIN_AI_JOBS_TABLE,\n data: {\n progress,\n status,\n task_id: requestId,\n },\n overrideAccess: true,\n req,\n })\n }\n\n console.log('fal webhook body: ', body)\n\n const videoUrl =\n body?.outputs?.[0]?.url ||\n body?.data?.outputs?.[0]?.url ||\n body?.video?.url ||\n body?.data?.video?.url ||\n body?.response?.video?.url ||\n body?.videos?.[0]?.url ||\n body?.data?.videos?.[0]?.url\n\n if (status === 'completed' && videoUrl) {\n // Fetch the related instruction to get upload collection\n const instructions = await req.payload.findByID({\n id: instructionId,\n collection: PLUGIN_INSTRUCTIONS_TABLE,\n req,\n })\n\n const uploadCollectionSlug = instructions['relation-to']\n\n const videoResp = await fetch(videoUrl)\n if (!videoResp.ok) {\n throw new Error(`Failed to fetch output: ${videoResp.status}`)\n }\n const buffer = Buffer.from(await videoResp.arrayBuffer())\n\n const created = await req.payload.create({\n collection: uploadCollectionSlug,\n data: { alt: 'video generation' },\n file: {\n name: 'video_generation.mp4',\n data: buffer,\n mimetype: 'video/mp4',\n size: buffer.byteLength,\n },\n overrideAccess: true,\n req,\n })\n\n // Persist the result on the AI Job record\n if (jobDoc) {\n await req.payload.update({\n id: jobDoc.id,\n collection: PLUGIN_AI_JOBS_TABLE,\n data: {\n progress: 100,\n result_id: created?.id,\n status: 'completed',\n },\n overrideAccess: true,\n req,\n })\n }\n }\n\n if (status === 'failed' && error) {\n req.payload.logger.error(error, 'Video generation failed: ')\n }\n\n return new Response(JSON.stringify({ ok: true }), {\n headers: { 'Content-Type': 'application/json' },\n })\n } catch (error) {\n req.payload.logger.error(error, 'Error in videogen webhook: ')\n const message =\n error && typeof error === 'object' && 'message' in error\n ? (error as any).message\n : String(error)\n return new Response(JSON.stringify({ error: message }), {\n headers: { 'Content-Type': 'application/json' },\n status: 500,\n })\n }\n },\n method: 'post',\n path: PLUGIN_API_ENDPOINT_VIDEOGEN_WEBHOOK,\n },\n }) satisfies Endpoints\n"],"names":["process","checkAccess","filterEditorSchemaByNodes","PLUGIN_AI_JOBS_TABLE","PLUGIN_API_ENDPOINT_GENERATE","PLUGIN_API_ENDPOINT_GENERATE_UPLOAD","PLUGIN_API_ENDPOINT_VIDEOGEN_WEBHOOK","PLUGIN_INSTRUCTIONS_TABLE","PLUGIN_NAME","registerEditorHelper","replacePlaceholders","extractImageData","fetchImages","fieldToJsonSchema","getFieldBySchemaPath","resolveImageReferences","assignPrompt","extendContextWithPromptFields","endpoints","pluginConfig","textarea","handler","req","data","json","allowedEditorNodes","locale","options","action","actionParams","instructionId","contextData","doc","Error","instructions","payload","findByID","id","collection","collections","config","find","slug","custom","editorConfig","admin","schema","editorSchema","prompt","promptTemplate","allowedEditorSchema","length","schemaPath","String","parts","split","collectionName","fieldName","defaultLocale","locales","localization","localeData","l","code","localeInfo","label","prompts","type","context","field","layout","systemPrompt","system","template","debugging","logger","info","jsonSchema","targetCollection","c","targetField","supported","t","includes","nameOverride","e","error","settingsName","undefined","modelSettings","images","resolvedImages","processedPrompt","console","log","hardcodedImages","allImages","imageParts","streamResult","ai","streamObject","maxTokens","model","provider","temperature","message","Response","JSON","stringify","headers","status","method","path","upload","collectionSlug","documentId","docData","draft","sampleImages","extendedContext","text","modelId","uploadCollectionSlug","editImages","serverURL","env","SERVER_URL","NEXT_PUBLIC_SERVER_URL","callbackUrl","replace","result","generateMedia","assetData","mediaUpload","request","create","alt","file","externalTaskId","jobId","taskId","progress","createdJob","task_id","overrideAccess","job","videogenWebhook","urlAll","URL","url","qpSecret","searchParams","get","headerSecret","falSecret","FAL_WEBHOOK_SECRET","legacySecret","VIDEOGEN_WEBHOOK_SECRET","provided","body","response","requestId","request_id","gateway_request_id","jobSearch","depth","limit","where","and","equals","jobDoc","docs","update","videoUrl","outputs","video","videos","videoResp","fetch","ok","buffer","Buffer","from","arrayBuffer","created","name","mimetype","size","byteLength","result_id"],"mappings":"AAGA,YAAYA,aAAa,eAAc;AAIvC,SAASC,WAAW,QAAQ,2BAA0B;AACtD,SAASC,yBAAyB,QAAQ,2CAA0C;AACpF,SACEC,oBAAoB,EACpBC,4BAA4B,EAC5BC,mCAAmC,EACnCC,oCAAoC,EACpCC,yBAAyB,EACzBC,WAAW,QACN,iBAAgB;AACvB,SAASC,oBAAoB,QAAQ,qCAAoC;AACzE,SAASC,mBAAmB,QAAQ,iDAAgD;AACpF,SAASC,gBAAgB,QAAQ,mCAAkC;AACnE,SAA8BC,WAAW,QAAQ,8BAA6B;AAC9E,SAASC,iBAAiB,QAAQ,oCAAmC;AACrE,SAASC,oBAAoB,QAAQ,uCAAsC;AAC3E,SAASC,sBAAsB,QAAQ,yCAAwC;AAC/E,SAASC,YAAY,EAAEC,6BAA6B,QAAQ,wBAAuB;AAEnF,OAAO,MAAMC,YAAuD,CAACC,eAClE,CAAA;QACCC,UAAU;YACR,mEAAmE;YACnEC,SAAS,OAAOC;gBACd,IAAI;oBACF,+CAA+C;oBAC/C,MAAMrB,YAAYqB,KAAKH;oBAEvB,MAAMI,OAAO,MAAMD,IAAIE,IAAI;oBAE3B,MAAM,EAAEC,qBAAqB,EAAE,EAAEC,SAAS,IAAI,EAAEC,OAAO,EAAE,GAAGJ;oBAC5D,MAAM,EAAEK,MAAM,EAAEC,YAAY,EAAEC,aAAa,EAAE,GAAGH;oBAChD,MAAMI,cAAcR,KAAKS,GAAG;oBAE5B,IAAI,CAACF,eAAe;wBAClB,MAAM,IAAIG,MACR,CAAC,gCAAgC,EAAEzB,YAAY,wDAAwD,CAAC;oBAE5G;oBAEA,qDAAqD;oBACrD,MAAM0B,eAAe,MAAMZ,IAAIa,OAAO,CAACC,QAAQ,CAAC;wBAC9CC,IAAIP;wBACJQ,YAAY/B;wBACZe;oBACF;oBAEA,MAAM,EAAEiB,WAAW,EAAE,GAAGjB,IAAIa,OAAO,CAACK,MAAM;oBAC1C,MAAMF,aAAaC,YAAYE,IAAI,CACjC,CAACH,aAAeA,WAAWI,IAAI,KAAKnC;oBAGtC,IAAI,CAAC+B,YAAY;wBACf,MAAM,IAAIL,MAAM;oBAClB;oBAEA,MAAM,EAAEU,QAAQ,EAAE,CAACnC,YAAY,EAAE,EAAEoC,eAAe,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC,EAAE,GAAGN,WAAWO,KAAK;oBACvF,MAAM,EAAEC,QAAQC,eAAe,CAAC,CAAC,EAAE,GAAGH;oBACtC,MAAM,EAAEI,QAAQC,iBAAiB,EAAE,EAAE,GAAGf;oBAExC,IAAIgB,sBAAsBH;oBAC1B,IAAItB,mBAAmB0B,MAAM,EAAE;wBAC7BD,sBAAsBhD,0BAA0B6C,cAActB;oBAChE;oBAEA,MAAM2B,aAAaC,OAAOnB,YAAY,CAAC,cAAc;oBACrD,MAAMoB,QAAQ,AAACF,CAAAA,cAAc,EAAC,EAAGG,KAAK,CAAC,QAAQ,EAAE;oBACjD,MAAMC,iBAAiBF,KAAK,CAAC,EAAE;oBAC/B,MAAMG,YAAYH,MAAMH,MAAM,GAAG,IAAIG,KAAK,CAACA,MAAMH,MAAM,GAAG,EAAE,GAAG;oBAE/D1C,qBAAqBa,IAAIa,OAAO,EAAEiB;oBAElC,MAAM,EAAEM,aAAa,EAAEC,UAAU,EAAE,EAAE,GAAGrC,IAAIa,OAAO,CAACK,MAAM,CAACoB,YAAY,IAAI,CAAC;oBAC5E,MAAMC,aAAaF,QAAQlB,IAAI,CAAC,CAACqB;wBAC/B,OAAOA,EAAEC,IAAI,KAAKrC;oBACpB;oBAEA,IAAIsC,aAAatC;oBACjB,IACEmC,cACAH,iBACAG,WAAWI,KAAK,IAChB,OAAOJ,WAAWI,KAAK,KAAK,YAC5BP,iBAAiBG,WAAWI,KAAK,EACjC;wBACAD,aAAaH,WAAWI,KAAK,CAACP,cAAc;oBAC9C;oBAEA,MAAMQ,UAAU,MAAMlD,aAAaY,QAAQ;wBACzCuC,MAAMd,OAAOnB,YAAY,CAAC,aAAa;wBACvCL;wBACAS,YAAYkB;wBACZY,SAASrC;wBACTsC,OAAOZ,aAAa;wBACpBa,QAAQpC,aAAaoC,MAAM;wBAC3B5C,QAAQsC;wBACR7C;wBACAoD,cAAcrC,aAAasC,MAAM;wBACjCC,UAAUpB,OAAOJ;oBACnB;oBAEA,IAAI9B,aAAauD,SAAS,EAAE;wBAC1BpD,IAAIa,OAAO,CAACwC,MAAM,CAACC,IAAI,CACrB;4BAAEV;wBAAQ,GACV,CAAC,sCAAsC,EAAEd,WAAW,CAAC;oBAEzD;oBAEA,wEAAwE;oBACxE,IAAIyB,aAAa3B;oBACjB,IAAI;wBACF,MAAM4B,mBAAmBxD,IAAIa,OAAO,CAACK,MAAM,CAACD,WAAW,CAACE,IAAI,CAC1D,CAACsC,IAAMA,EAAErC,IAAI,KAAKc;wBAEpB,IAAIsB,oBAAoBrB,WAAW;4BACjC,MAAMuB,cAAclE,qBAAqBgE,kBAAkB1B;4BAC3D,MAAM6B,YAAY;gCAChB;gCACA;gCACA;gCACA;gCACA;gCACA;gCACA;gCACA;6BACD;4BACD,MAAMC,IAAI7B,OAAO2B,aAAab,QAAQ;4BACtC,IAAIa,eAAeC,UAAUE,QAAQ,CAACD,IAAI;gCACxCL,aAAahE,kBAAkBmE,aAAa;oCAAEI,cAAc3B;gCAAU;4BACxE;wBACF;oBACF,EAAE,OAAO4B,GAAG;wBACV/D,IAAIa,OAAO,CAACwC,MAAM,CAACW,KAAK,CAACD,GAAG;oBAC9B;oBAEA,sCAAsC;oBACtC,MAAME,eACJrD,YAAY,CAAC,WAAW,KAAK,aACzB,sBACAA,YAAY,CAAC,WAAW,KAAK,SAC3B,kBACAsD;oBAER,IAAI,CAACD,cAAc;wBACjB,MAAM,IAAItD,MAAM,CAAC,sBAAsB,EAAEC,YAAY,CAAC,WAAW,CAAC,CAAC;oBACrE;oBAEA,MAAMuD,gBAAgBvD,YAAY,CAACqD,aAAa,IAAI,CAAC;oBAErD,qDAAqD;oBACrD,MAAM,EAAEG,QAAQC,cAAc,EAAEC,eAAe,EAAE,GAAG,MAAM7E,uBACxDmD,QAAQlB,MAAM,EACdjB,aACAT;oBAGFuE,QAAQC,GAAG,CAAC,qBAAqBH;oBAEjC,mDAAmD;oBACnD,MAAMI,kBAAkBpF,iBAAiBiF;oBAEzC,iBAAiB;oBACjB,MAAMI,YAAY;2BAAID;2BAAoBJ;qBAAe;oBAEzD,IAAID;oBAEJ,IAAIM,UAAU7C,MAAM,GAAG,GAAG;wBACxB,MAAM8C,aAAa,MAAMrF,YAAYU,KAAK0E;wBAE1C,IAAIC,WAAW9C,MAAM,GAAG,GAAG;4BACzBuC,SAASO;wBACX;oBACF;oBAEA,2CAA2C;oBAC3C,MAAMC,eAAe,MAAM5E,IAAIa,OAAO,CAACgE,EAAE,CAACC,YAAY,CAAC;wBACrD,+EAA+E;wBAC/EV;wBACAW,WAAWZ,cAAcY,SAAS;wBAClCC,OAAOb,cAAca,KAAK;wBAC1BtD,QAAQ4C;wBACRW,UAAUd,cAAcc,QAAQ;wBAChCzD,QAAQ+B;wBACRL,QAAQN,QAAQM,MAAM;wBACtBgC,aAAaf,cAAce,WAAW;oBACxC;oBAEA,OAAON;gBACT,EAAE,OAAOZ,OAAO;oBACdhE,IAAIa,OAAO,CAACwC,MAAM,CAACW,KAAK,CAACA,OAAO;oBAChC,MAAMmB,UACJnB,SAAS,OAAOA,UAAU,YAAY,aAAaA,QAC/C,AAACA,MAAgBmB,OAAO,GACxBpD,OAAOiC;oBACb,OAAO,IAAIoB,SAASC,KAAKC,SAAS,CAAC;wBAAEtB,OAAOmB;oBAAQ,IAAI;wBACtDI,SAAS;4BAAE,gBAAgB;wBAAmB;wBAC9CC,QACEL,QAAQtB,QAAQ,CAAC,8BACjBsB,QAAQtB,QAAQ,CAAC,8BACb,MACA;oBACR;gBACF;YACF;YACA4B,QAAQ;YACRC,MAAM5G;QACR;QACA6G,QAAQ;YACN,iEAAiE;YACjE5F,SAAS,OAAOC;gBACd,IAAI;oBACF,+CAA+C;oBAC/C,MAAMrB,YAAYqB,KAAKH;oBAEvB,MAAMI,OAAO,MAAMD,IAAIE,IAAI;oBAE3B,MAAM,EAAE0F,cAAc,EAAEC,UAAU,EAAExF,OAAO,EAAE,GAAGJ;oBAChD,MAAM,EAAEO,aAAa,EAAE,GAAGH;oBAC1B,IAAIyF,UAAU,CAAC;oBAEf,IAAID,YAAY;wBACd,IAAI;4BACFC,UAAU,MAAM9F,IAAIa,OAAO,CAACC,QAAQ,CAAC;gCACnCC,IAAI8E;gCACJ7E,YAAY4E;gCACZG,OAAO;gCACP/F;4BACF;wBACF,EAAE,OAAO+D,GAAG;4BACV/D,IAAIa,OAAO,CAACwC,MAAM,CAACW,KAAK,CACtBD,GACA;wBAEJ;oBACF;oBAEA,MAAMtD,cAAc;wBAClB,GAAGR,KAAKS,GAAG;wBACX,GAAGoF,OAAO;oBACZ;oBAEA,IAAIlF,eAAwC;wBAAEwD,QAAQ,EAAE;wBAAE,YAAY;wBAAI1C,QAAQ;oBAAG;oBAErF,IAAIlB,eAAe;wBACjB,qDAAqD;wBACrDI,eAAe,MAAMZ,IAAIa,OAAO,CAACC,QAAQ,CAAC;4BACxCC,IAAIP;4BACJQ,YAAY/B;4BACZe;wBACF;oBACF;oBAEA,MAAM,EAAEoE,QAAQ4B,eAAe,EAAE,EAAEtE,QAAQC,iBAAiB,EAAE,EAAE,GAAGf;oBACnE,MAAMkB,aAAaC,OAAOnB,YAAY,CAAC,cAAc;oBACrDzB,qBAAqBa,IAAIa,OAAO,EAAEiB;oBAElC,MAAMmE,kBAAkBtG,8BACtBc,aACA;wBAAEoC,MAAMd,OAAOnB,YAAY,CAAC,aAAa;wBAAGI,YAAY4E;oBAAe,GACvE/F;oBAEF,MAAMqG,OAAO,MAAM9G,oBAAoBuC,gBAA0BsE;oBACjE,MAAME,UAAUvF,YAAY,CAAC,WAAW;oBACxC,MAAMwF,uBAAuBxF,YAAY,CAAC,cAAc;oBAExD,qDAAqD;oBACrD,MAAM,EAAEwD,QAAQC,cAAc,EAAEC,eAAe,EAAE,GAAG,MAAM7E,uBACxDyG,MACAzF,aACAT;oBAGF,oGAAoG;oBACpG,MAAMoE,SAAS;2BACV/E,iBAAiBiF;2BACjBD;2BACC2B;qBACL;oBAED,4DAA4D;oBAC5D,MAAMK,aAA0B,MAAM/G,YAAYU,KAAKoE;oBAEvD,IAAIvE,aAAauD,SAAS,EAAE;wBAC1BpD,IAAIa,OAAO,CAACwC,MAAM,CAACC,IAAI,CAAC;4BAAE4C;wBAAK,GAAG,CAAC,uCAAuC,CAAC;oBAC7E;oBAEA,sCAAsC;oBACtC,MAAMI,YACJtG,IAAIa,OAAO,CAACK,MAAM,EAAEoF,aACpB5H,QAAQ6H,GAAG,CAACC,UAAU,IACtB9H,QAAQ6H,GAAG,CAACE,sBAAsB;oBAEpC,MAAMC,cAAcJ,YAChB,CAAC,EAAEA,UAAUK,OAAO,CAAC,OAAO,IAAI,IAAI,EAAE3H,qCAAqC,eAAe,EAAEwB,cAAc,CAAC,GAC3G0D;oBAEJ,qBAAqB;oBACrB,MAAMD,eACJkC,YAAY,UACR,mBACAA,YAAY,UACV,mBACAA,YAAY,QACV,iBACAjC;oBACV,IAAI,CAACD,cAAc;wBACjB,MAAM,IAAItD,MAAM,CAAC,sBAAsB,EAAEwF,QAAQ,CAAC;oBACpD;oBAEA,MAAMhC,gBAAgBvD,YAAY,CAACqD,aAAa,IAAI,CAAC;oBAErD,4CAA4C;oBAC5C,MAAM2C,SAAS,MAAM5G,IAAIa,OAAO,CAACgE,EAAE,CAACgC,aAAa,CAAC;wBAChDH;wBACAtC,QAAQiC;wBACR7F;wBACAwE,OAAO,AAACb,cAA0Ca,KAAK;wBACvDtD,QAAQwE;wBACRjB,UAAU,AAACd,cAA0Cc,QAAQ;wBAC7D,GAAId,aAAa;oBACnB;oBAEA,4DAA4D;oBAC5D,IAAIyC,UAAU,UAAUA,QAAQ;wBAC9B,IAAIE;wBACJ,IAAI,OAAOjH,aAAakH,WAAW,KAAK,YAAY;4BAClDD,YAAY,MAAMjH,aAAakH,WAAW,CACxCH,QACA;gCACE5F,YAAYoF;gCACZY,SAAShH;4BACX;wBAEJ,OAAO;4BACL8G,YAAY,MAAM9G,IAAIa,OAAO,CAACoG,MAAM,CAAC;gCACnCjG,YAAYoF;gCACZnG,MAAM;oCAAEiH,KAAKhB;gCAAK;gCAClBiB,MAAMP,OAAOO,IAAI;gCACjBnH;4BACF;wBACF;wBAEA,IAAI,CAAC8G,UAAU/F,EAAE,EAAE;4BACjBf,IAAIa,OAAO,CAACwC,MAAM,CAACW,KAAK,CACtB;4BAEF,MAAM,IAAIrD,MAAM;wBAClB;wBAEA,OAAO,IAAIyE,SACTC,KAAKC,SAAS,CAAC;4BACbsB,QAAQ;gCACN7F,IAAI+F,UAAU/F,EAAE;gCAChBmG,KAAKJ,UAAUI,GAAG;4BACpB;wBACF;oBAEJ;oBAEA,qCAAqC;oBACrC,IAAIN,UAAW,CAAA,WAAWA,UAAU,YAAYA,MAAK,GAAI;wBACvD,MAAMQ,iBAAiBR,OAAOS,KAAK,IAAIT,OAAOU,MAAM;wBACpD,MAAM9B,SAASoB,OAAOpB,MAAM,IAAI;wBAChC,MAAM+B,WAAWX,OAAOW,QAAQ,IAAI;wBAEpC,2CAA2C;wBAC3C,MAAMC,aAAa,MAAMxH,IAAIa,OAAO,CAACoG,MAAM,CAAC;4BAC1CjG,YAAYnC;4BACZoB,MAAM;gCACJO;gCACA+G;gCACA/B;gCACAiC,SAASL;4BACX;4BACAM,gBAAgB;4BAChB1H;wBACF;wBAEA,OAAO,IAAIoF,SAASC,KAAKC,SAAS,CAAC;4BAAEqC,KAAK;gCAAE5G,IAAIyG,WAAWzG,EAAE;4BAAC;wBAAE,IAAI;4BAClEwE,SAAS;gCAAE,gBAAgB;4BAAmB;wBAChD;oBACF;oBAEA,MAAM,IAAI5E,MAAM;gBAClB,EAAE,OAAOqD,OAAO;oBACdhE,IAAIa,OAAO,CAACwC,MAAM,CAACW,KAAK,CACtB,mBAAmB;oBACnBA,OAAOnB,QAAQ,AAACmB,MAAgBmB,OAAO,EACvC;oBAEF,MAAMA,UACJnB,SAAS,OAAOA,UAAU,YAAY,aAAaA,QAC/C,AAACA,MAAgBmB,OAAO,GACxBpD,OAAOiC;oBACb,OAAO,IAAIoB,SAASC,KAAKC,SAAS,CAAC;wBAAEtB,OAAOmB;oBAAQ,IAAI;wBACtDI,SAAS;4BAAE,gBAAgB;wBAAmB;wBAC9CC,QACEL,QAAQtB,QAAQ,CAAC,8BACjBsB,QAAQtB,QAAQ,CAAC,8BACb,MACA;oBACR;gBACF;YACF;YACA4B,QAAQ;YACRC,MAAM3G;QACR;QACA6I,iBAAiB;YACf7H,SAAS,OAAOC;gBACduE,QAAQC,GAAG,CAAC,wBAAwBxE;gBACpC,IAAI;oBACF,MAAM6H,SAAS,IAAIC,IAAI9H,IAAI+H,GAAG,IAAI;oBAClC,MAAMC,WAAWH,OAAOI,YAAY,CAACC,GAAG,CAAC,aAAa;oBACtD,MAAMC,eAAenI,IAAIuF,OAAO,CAAC2C,GAAG,CAAC,uBAAuB;oBAC5D,MAAME,YAAY1J,QAAQ6H,GAAG,CAAC8B,kBAAkB;oBAChD,MAAMC,eAAe5J,QAAQ6H,GAAG,CAACgC,uBAAuB;oBACxD,MAAMC,WAAWR,YAAYG;oBAC7B,gEAAgE;oBAChE,IAAI,CAACK,YAAaJ,CAAAA,YAAYI,aAAaJ,YAAYI,aAAaF,YAAW,GAAI;wBACjF,OAAO,IAAIlD,SAAS,gBAAgB;4BAAEI,QAAQ;wBAAI;oBACpD;oBAEA,MAAMhF,gBAAgBqH,OAAOI,YAAY,CAACC,GAAG,CAAC;oBAC9C,IAAI,CAAC1H,eAAe;wBAClB,MAAM,IAAIG,MAAM;oBAClB;oBAEA,MAAM8H,OAAO,MAAMzI,IAAIE,IAAI;oBAC3B,gCAAgC;oBAChC,MAAMsF,SACJ,AAACiD,QAASA,CAAAA,KAAKjD,MAAM,IAAIiD,KAAKxI,IAAI,EAAEuF,UAAUiD,KAAKC,QAAQ,EAAElD,MAAK,KAAOtB;oBAC3E,MAAMqD,WACJ,AAACkB,CAAAA,QAASA,CAAAA,KAAKlB,QAAQ,IAAIkB,KAAKxI,IAAI,EAAEsH,YAAYkB,KAAKC,QAAQ,EAAEnB,QAAO,CAAC,KAAMrD;oBACjF,MAAMyE,YACJ,AAACF,QACEA,CAAAA,KAAKnB,MAAM,IACVmB,KAAKG,UAAU,IACfH,KAAKI,kBAAkB,IACvBJ,KAAKzB,OAAO,EAAE4B,UAAS,KAC3B1E;oBACF,MAAMF,QAAQyE,MAAMzE,SAASyE,MAAMxI,MAAM+D,SAASyE,MAAMC,UAAU1E;oBAElE,mDAAmD;oBACnD,MAAM8E,YAAY,MAAM9I,IAAIa,OAAO,CAACM,IAAI,CAAC;wBACvCH,YAAYnC;wBACZkK,OAAO;wBACPC,OAAO;wBACPC,OAAO;4BACLC,KAAK;gCACH;oCAAEzB,SAAS;wCAAE0B,QAAQR;oCAAU;gCAAE;gCACjC;oCAAEnI,eAAe;wCAAE2I,QAAQ3I;oCAAc;gCAAE;6BAC5C;wBACH;oBACF;oBAEA,MAAM4I,SAASN,UAAUO,IAAI,EAAE,CAAC,EAAE;oBAClC,IAAID,QAAQ;wBACV,MAAMpJ,IAAIa,OAAO,CAACyI,MAAM,CAAC;4BACvBvI,IAAIqI,OAAOrI,EAAE;4BACbC,YAAYnC;4BACZoB,MAAM;gCACJsH;gCACA/B;gCACAiC,SAASkB;4BACX;4BACAjB,gBAAgB;4BAChB1H;wBACF;oBACF;oBAEAuE,QAAQC,GAAG,CAAC,sBAAsBiE;oBAElC,MAAMc,WACJd,MAAMe,SAAS,CAAC,EAAE,EAAEzB,OACpBU,MAAMxI,MAAMuJ,SAAS,CAAC,EAAE,EAAEzB,OAC1BU,MAAMgB,OAAO1B,OACbU,MAAMxI,MAAMwJ,OAAO1B,OACnBU,MAAMC,UAAUe,OAAO1B,OACvBU,MAAMiB,QAAQ,CAAC,EAAE,EAAE3B,OACnBU,MAAMxI,MAAMyJ,QAAQ,CAAC,EAAE,EAAE3B;oBAE3B,IAAIvC,WAAW,eAAe+D,UAAU;wBACtC,yDAAyD;wBACzD,MAAM3I,eAAe,MAAMZ,IAAIa,OAAO,CAACC,QAAQ,CAAC;4BAC9CC,IAAIP;4BACJQ,YAAY/B;4BACZe;wBACF;wBAEA,MAAMoG,uBAAuBxF,YAAY,CAAC,cAAc;wBAExD,MAAM+I,YAAY,MAAMC,MAAML;wBAC9B,IAAI,CAACI,UAAUE,EAAE,EAAE;4BACjB,MAAM,IAAIlJ,MAAM,CAAC,wBAAwB,EAAEgJ,UAAUnE,MAAM,CAAC,CAAC;wBAC/D;wBACA,MAAMsE,SAASC,OAAOC,IAAI,CAAC,MAAML,UAAUM,WAAW;wBAEtD,MAAMC,UAAU,MAAMlK,IAAIa,OAAO,CAACoG,MAAM,CAAC;4BACvCjG,YAAYoF;4BACZnG,MAAM;gCAAEiH,KAAK;4BAAmB;4BAChCC,MAAM;gCACJgD,MAAM;gCACNlK,MAAM6J;gCACNM,UAAU;gCACVC,MAAMP,OAAOQ,UAAU;4BACzB;4BACA5C,gBAAgB;4BAChB1H;wBACF;wBAEA,0CAA0C;wBAC1C,IAAIoJ,QAAQ;4BACV,MAAMpJ,IAAIa,OAAO,CAACyI,MAAM,CAAC;gCACvBvI,IAAIqI,OAAOrI,EAAE;gCACbC,YAAYnC;gCACZoB,MAAM;oCACJsH,UAAU;oCACVgD,WAAWL,SAASnJ;oCACpByE,QAAQ;gCACV;gCACAkC,gBAAgB;gCAChB1H;4BACF;wBACF;oBACF;oBAEA,IAAIwF,WAAW,YAAYxB,OAAO;wBAChChE,IAAIa,OAAO,CAACwC,MAAM,CAACW,KAAK,CAACA,OAAO;oBAClC;oBAEA,OAAO,IAAIoB,SAASC,KAAKC,SAAS,CAAC;wBAAEuE,IAAI;oBAAK,IAAI;wBAChDtE,SAAS;4BAAE,gBAAgB;wBAAmB;oBAChD;gBACF,EAAE,OAAOvB,OAAO;oBACdhE,IAAIa,OAAO,CAACwC,MAAM,CAACW,KAAK,CAACA,OAAO;oBAChC,MAAMmB,UACJnB,SAAS,OAAOA,UAAU,YAAY,aAAaA,QAC/C,AAACA,MAAcmB,OAAO,GACtBpD,OAAOiC;oBACb,OAAO,IAAIoB,SAASC,KAAKC,SAAS,CAAC;wBAAEtB,OAAOmB;oBAAQ,IAAI;wBACtDI,SAAS;4BAAE,gBAAgB;wBAAmB;wBAC9CC,QAAQ;oBACV;gBACF;YACF;YACAC,QAAQ;YACRC,MAAM1G;QACR;IACF,CAAA,EAAsB"}
1
+ {"version":3,"sources":["../../src/endpoints/index.ts"],"sourcesContent":["import type { ImagePart } from 'ai'\nimport type { Field, PayloadRequest } from 'payload'\n\nimport * as process from 'node:process'\n\nimport type { Endpoints, PluginConfig } from '../types.js'\n\nimport { checkAccess } from '../access/checkAccess.js'\nimport { filterEditorSchemaByNodes } from '../ai/utils/filterEditorSchemaByNodes.js'\nimport {\n PLUGIN_AI_JOBS_TABLE,\n PLUGIN_API_ENDPOINT_GENERATE,\n PLUGIN_API_ENDPOINT_GENERATE_UPLOAD,\n PLUGIN_API_ENDPOINT_VIDEOGEN_WEBHOOK,\n PLUGIN_INSTRUCTIONS_TABLE,\n PLUGIN_NAME,\n} from '../defaults.js'\nimport { registerEditorHelper } from '../libraries/handlebars/helpers.js'\nimport { replacePlaceholders } from '../libraries/handlebars/replacePlaceholders.js'\nimport { buildSmartPrompt, isGenericPrompt } from '../utilities/buildSmartPrompt.js'\nimport { extractImageData } from '../utilities/extractImageData.js'\nimport { type FetchableImage, fetchImages } from '../utilities/fetchImages.js'\nimport { fieldToJsonSchema } from '../utilities/fieldToJsonSchema.js'\nimport { getFieldBySchemaPath } from '../utilities/getFieldBySchemaPath.js'\nimport { resolveImageReferences } from '../utilities/resolveImageReferences.js'\nimport { assignPrompt, extendContextWithPromptFields } from './buildPromptUtils.js'\n\nexport const endpoints: (pluginConfig: PluginConfig) => Endpoints = (pluginConfig) =>\n ({\n textarea: {\n // Text/rich-text generation endpoint using payload.ai.streamObject\n handler: async (req: PayloadRequest) => {\n try {\n // Check authentication and authorization first\n await checkAccess(req, pluginConfig)\n\n const data = await req.json?.()\n\n const { allowedEditorNodes = [], locale = 'en', options } = data\n const { action, actionParams, instructionId } = options\n const contextData = data.doc\n\n if (!instructionId) {\n throw new Error(\n `Instruction ID is required for \"${PLUGIN_NAME}\" to work, please check your configuration, or try again`,\n )\n }\n\n // Verify user has access to the specific instruction\n const instructions = await req.payload.findByID({\n id: instructionId,\n collection: PLUGIN_INSTRUCTIONS_TABLE,\n req, // Pass req to ensure access control is applied\n })\n\n const { collections } = req.payload.config\n const collection = collections.find(\n (collection) => collection.slug === PLUGIN_INSTRUCTIONS_TABLE,\n )\n\n if (!collection) {\n throw new Error('Collection not found')\n }\n\n const { custom: { [PLUGIN_NAME]: { editorConfig = {} } = {} } = {} } = collection.admin\n const { schema: editorSchema = {} } = editorConfig\n let { prompt: promptTemplate = '' } = instructions\n\n // Smart fallback: if prompt is generic, build a contextual prompt from field metadata\n if (isGenericPrompt(promptTemplate)) {\n const schemaPath = String(instructions['schema-path'])\n promptTemplate = buildSmartPrompt({\n documentData: contextData,\n payload: req.payload,\n schemaPath,\n })\n\n if (pluginConfig.debugging) {\n req.payload.logger.info(\n { smartPrompt: promptTemplate },\n `— AI Plugin: Using smart fallback prompt for ${schemaPath}`,\n )\n }\n }\n\n let allowedEditorSchema = editorSchema\n if (allowedEditorNodes.length) {\n allowedEditorSchema = filterEditorSchemaByNodes(editorSchema, allowedEditorNodes)\n // Debug: Log what nodes were received and what definitions remain\n if (pluginConfig.debugging) {\n req.payload.logger.info(\n {\n receivedNodes: allowedEditorNodes,\n remainingDefinitions: Object.keys(allowedEditorSchema.definitions || {}),\n },\n '— AI Plugin: Schema filtering debug',\n )\n }\n }\n\n const schemaPath = String(instructions['schema-path'])\n const parts = (schemaPath || '').split('.') || []\n const collectionName = parts[0]\n const fieldName = parts.length > 1 ? parts[parts.length - 1] : ''\n\n registerEditorHelper(req.payload, schemaPath)\n\n const { defaultLocale, locales = [] } = req.payload.config.localization || {}\n const localeData = locales.find((l) => {\n return l.code === locale\n })\n\n let localeInfo = locale\n if (\n localeData &&\n defaultLocale &&\n localeData.label &&\n typeof localeData.label === 'object' &&\n defaultLocale in localeData.label\n ) {\n localeInfo = localeData.label[defaultLocale]\n }\n\n const prompts = await assignPrompt(action, {\n type: String(instructions['field-type']),\n actionParams,\n collection: collectionName,\n context: contextData,\n field: fieldName || '',\n layout: instructions.layout,\n locale: localeInfo,\n pluginConfig,\n systemPrompt: instructions.system,\n template: String(promptTemplate),\n })\n\n if (pluginConfig.debugging) {\n req.payload.logger.info(\n { prompts },\n `— AI Plugin: Executing text prompt on ${schemaPath}`,\n )\n }\n\n // Build per-field JSON schema for structured generation when applicable\n let jsonSchema = allowedEditorSchema\n let targetField: Field | null | undefined\n\n try {\n const targetCollection = req.payload.config.collections.find(\n (c) => c.slug === collectionName,\n )\n if (targetCollection && fieldName) {\n targetField = getFieldBySchemaPath(targetCollection, schemaPath)\n const supported = [\n 'array',\n 'text',\n 'textarea',\n 'select',\n 'number',\n 'date',\n 'code',\n 'email',\n 'json',\n ]\n const t = String(targetField?.type || '')\n if (targetField && supported.includes(t)) {\n // For array fields, use count from array-settings if available\n if (t === 'array') {\n const arraySettings = (instructions['array-settings'] || {}) as Record<\n string,\n unknown\n >\n const count = (arraySettings.count as number) || 3\n // Override the field's maxRows with the requested count\n const modifiedField = {\n ...targetField,\n maxRows: count,\n minRows: count,\n } as typeof targetField\n jsonSchema = fieldToJsonSchema(modifiedField, { nameOverride: fieldName })\n } else {\n jsonSchema = fieldToJsonSchema(targetField, { nameOverride: fieldName })\n }\n }\n }\n } catch (e) {\n req.payload.logger.error(e, '— AI Plugin: Error building field JSON schema')\n }\n\n // Get model settings from instruction\n const settingsName =\n instructions['model-id'] === 'richtext'\n ? 'richtext-settings'\n : instructions['model-id'] === 'text'\n ? 'text-settings'\n : instructions['model-id'] === 'array'\n ? 'array-settings'\n : undefined\n\n if (!settingsName) {\n throw new Error(`Unsupported model-id: ${instructions['model-id']}`)\n }\n\n const modelSettings = instructions[settingsName] || {}\n\n // Resolve @field:filename references from the prompt\n const { images: resolvedImages, processedPrompt } = await resolveImageReferences(\n prompts.prompt,\n contextData,\n req,\n collectionName,\n )\n\n // Extract hardcoded URLs from the processed prompt\n const hardcodedImages = extractImageData(processedPrompt)\n\n // Combine images\n const allImages = [...hardcodedImages, ...resolvedImages] as FetchableImage[]\n\n let images: ImagePart[] | undefined\n\n if (allImages.length > 0) {\n const imageParts = await fetchImages(req, allImages)\n\n if (imageParts.length > 0) {\n images = imageParts\n }\n }\n\n let promptToUse = processedPrompt\n let systemToUse = prompts.system\n // let messagesToUse: any = undefined\n\n // Execute beforeGenerate hooks\n if (targetField && (targetField as any).custom?.ai?.beforeGenerate) {\n const beforeHooks = (targetField as any).custom.ai.beforeGenerate as Array<\n (args: any) => Promise<any>\n >\n for (const hook of beforeHooks) {\n const result = await hook({\n doc: contextData,\n field: targetField,\n headers: req.headers,\n instructions,\n payload: req.payload,\n prompt: promptToUse,\n req,\n system: systemToUse,\n })\n\n if (result) {\n if (result.prompt) promptToUse = result.prompt\n if (result.system) systemToUse = result.system\n }\n }\n }\n\n const streamResult = await req.payload.ai.streamObject({\n // extractAttachments: modelSettings.extractAttachments as boolean | undefined,\n images,\n maxTokens: modelSettings.maxTokens as number | undefined,\n // messages: messagesToUse,\n model: modelSettings.model as string,\n onFinish: async ({ object }) => {\n if (targetField && (targetField as any).custom?.ai?.afterGenerate) {\n const afterHooks = (targetField as any).custom.ai.afterGenerate as Array<\n (args: any) => Promise<any>\n >\n for (const hook of afterHooks) {\n await hook({\n doc: contextData,\n field: targetField,\n headers: req.headers,\n instructions,\n payload: req.payload,\n req,\n result: object,\n })\n }\n }\n },\n prompt: processedPrompt,\n provider: modelSettings.provider as string,\n providerOptions: {\n openai: {\n strictJsonSchema: true,\n },\n },\n schema: jsonSchema,\n system: systemToUse,\n temperature: modelSettings.temperature as number | undefined,\n })\n\n return streamResult\n } catch (error) {\n req.payload.logger.error(error, 'Error generating content: ')\n const message =\n error && typeof error === 'object' && 'message' in error\n ? (error as Error).message\n : String(error)\n return new Response(JSON.stringify({ error: message }), {\n headers: { 'Content-Type': 'application/json' },\n status:\n message.includes('Authentication required') ||\n message.includes('Insufficient permissions')\n ? 401\n : 500,\n })\n }\n },\n method: 'post',\n path: PLUGIN_API_ENDPOINT_GENERATE,\n },\n upload: {\n // Image/video generation endpoint using payload.ai.generateMedia\n handler: async (req: PayloadRequest) => {\n try {\n // Check authentication and authorization first\n await checkAccess(req, pluginConfig)\n\n const data = await req.json?.()\n\n const { collectionSlug, documentId, options } = data\n const { instructionId } = options\n let docData = {}\n\n if (documentId) {\n try {\n docData = await req.payload.findByID({\n id: documentId,\n collection: collectionSlug as string,\n draft: true,\n req, // Pass req to ensure access control is applied\n })\n } catch (e) {\n req.payload.logger.error(\n e,\n '— AI Plugin: Error fetching document, you should try again after enabling drafts for this collection',\n )\n }\n }\n\n const contextData = {\n ...data.doc,\n ...docData,\n }\n\n let instructions: Record<string, unknown> = { images: [], 'model-id': '', prompt: '' }\n\n if (instructionId) {\n // Verify user has access to the specific instruction\n instructions = await req.payload.findByID({\n id: instructionId,\n collection: PLUGIN_INSTRUCTIONS_TABLE,\n req, // Pass req to ensure access control is applied\n })\n }\n\n let { prompt: promptTemplate = '' } = instructions\n const { images: sampleImages = [] } = instructions\n const schemaPath = String(instructions['schema-path'])\n registerEditorHelper(req.payload, schemaPath)\n\n // Smart fallback: if prompt is generic, build a contextual prompt from field metadata\n if (isGenericPrompt(promptTemplate as string)) {\n promptTemplate = buildSmartPrompt({\n documentData: contextData,\n payload: req.payload,\n schemaPath,\n })\n\n if (pluginConfig.debugging) {\n req.payload.logger.info(\n { smartPrompt: promptTemplate },\n `— AI Plugin: Using smart fallback prompt for ${schemaPath}`,\n )\n }\n }\n\n const extendedContext = extendContextWithPromptFields(\n contextData,\n { type: String(instructions['field-type']), collection: collectionSlug },\n pluginConfig,\n )\n const text = await replacePlaceholders(promptTemplate as string, extendedContext)\n const uploadCollectionSlug = instructions['relation-to']\n\n // Resolve @field:filename references from the prompt\n const { images: resolvedImages, processedPrompt } = await resolveImageReferences(\n text,\n contextData,\n req,\n collectionSlug,\n )\n\n // Extract hardcoded URLs from the processed prompt and merge with resolved images and sample images\n const images = [\n ...extractImageData(processedPrompt),\n ...resolvedImages,\n ...(sampleImages as unknown[]),\n ] as FetchableImage[]\n\n // Process images - convert to ImagePart format using helper\n const editImages: ImagePart[] = await fetchImages(req, images)\n\n let promptToUse = text\n let targetField: Field | null | undefined\n\n try {\n const targetCollection = req.payload.config.collections.find(\n (c) => c.slug === collectionSlug,\n )\n if (targetCollection && schemaPath) {\n targetField = getFieldBySchemaPath(targetCollection, schemaPath)\n }\n } catch (e) {\n req.payload.logger.error(e, '— AI Plugin: Error finding field for hooks')\n }\n\n if (targetField && (targetField as any).custom?.ai?.beforeGenerate) {\n const beforeHooks = (targetField as any).custom.ai.beforeGenerate as Array<\n (args: any) => Promise<any>\n >\n for (const hook of beforeHooks) {\n const result = await hook({\n doc: contextData,\n field: targetField,\n headers: req.headers,\n instructions,\n payload: req.payload,\n prompt: promptToUse,\n req,\n })\n\n if (result) {\n if (result.prompt) {\n promptToUse = result.prompt\n }\n if (result.instructions) {\n instructions = {\n ...instructions,\n ...result.instructions,\n }\n }\n }\n }\n }\n\n if (pluginConfig.debugging) {\n req.payload.logger.info(\n { text: promptToUse },\n `— AI Plugin: Executing media generation`,\n )\n }\n\n // Prepare callback URL for async jobs\n const serverURL =\n req.payload.config?.serverURL ||\n process.env.SERVER_URL ||\n process.env.NEXT_PUBLIC_SERVER_URL\n\n const callbackUrl = serverURL\n ? `${serverURL.replace(/\\/$/, '')}/api${PLUGIN_API_ENDPOINT_VIDEOGEN_WEBHOOK}?instructionId=${instructionId}`\n : undefined\n\n // Get model settings\n // Re-evaluate settings name and settings in case instructions changed\n const modelId = instructions['model-id']\n const settingsName =\n modelId === 'image'\n ? 'image-settings'\n : modelId === 'video'\n ? 'video-settings'\n : modelId === 'tts'\n ? 'tts-settings'\n : undefined\n if (!settingsName) {\n throw new Error(`Unsupported model-id: ${modelId}`)\n }\n\n // Get model settings from instruction\n const instructionSettings = (instructions[settingsName] || {}) as Record<string, unknown>\n\n // Fallback to AISettings global defaults if instruction-level settings are missing\n let globalDefaults: Record<string, unknown> = {}\n if (!instructionSettings.provider || !instructionSettings.model) {\n try {\n const aiSettings = await req.payload.findGlobal({\n slug: 'ai-settings',\n context: { unsafe: true }, // Get decrypted values for internal use\n })\n\n // Map modelId to the corresponding default settings key\n const defaultsKey =\n modelId === 'image'\n ? 'image'\n : modelId === 'video'\n ? 'video'\n : modelId === 'tts'\n ? 'tts'\n : undefined\n\n if (defaultsKey && aiSettings?.defaults?.[defaultsKey]) {\n globalDefaults = aiSettings.defaults[defaultsKey] as Record<string, unknown>\n\n if (pluginConfig.debugging) {\n req.payload.logger.info(\n { globalDefaults },\n `— AI Plugin: Using AISettings defaults for ${modelId}`,\n )\n }\n }\n } catch (e) {\n req.payload.logger.error(e, '— AI Plugin: Error fetching AISettings defaults')\n }\n }\n\n // Merge: instruction settings take priority over global defaults\n // Filter out null/undefined values so they don't overwrite valid defaults\n const filteredInstructionSettings = Object.fromEntries(\n Object.entries(instructionSettings).filter(([_, v]) => v != null),\n )\n const modelSettings = {\n ...globalDefaults,\n ...filteredInstructionSettings,\n }\n\n // Use payload.ai.generateMedia directly! 🎉\n const result = await req.payload.ai.generateMedia({\n callbackUrl,\n images: editImages,\n instructionId,\n model: modelSettings.model as string,\n prompt: promptToUse,\n provider: modelSettings.provider as string,\n ...modelSettings,\n })\n\n if (targetField && (targetField as any).custom?.ai?.afterGenerate) {\n const afterHooks = (targetField as any).custom.ai.afterGenerate as Array<\n (args: any) => Promise<any>\n >\n for (const hook of afterHooks) {\n await hook({\n doc: contextData,\n field: targetField,\n headers: req.headers,\n instructions,\n payload: req.payload,\n req,\n result,\n })\n }\n }\n\n // If model returned a file immediately, proceed with upload\n if (result && 'file' in result) {\n let assetData: { alt?: string; id: number | string }\n if (typeof pluginConfig.mediaUpload === 'function') {\n assetData = await pluginConfig.mediaUpload(result, {\n collection: uploadCollectionSlug as string,\n request: req,\n })\n } else {\n assetData = await req.payload.create({\n collection: uploadCollectionSlug as string,\n data: { alt: text },\n file: result.file,\n req, // Pass req to ensure access control is applied\n })\n }\n\n if (!assetData.id) {\n req.payload.logger.error(\n 'Error uploading generated media, is your media upload function correct?',\n )\n throw new Error('Error uploading generated media!')\n }\n\n return new Response(\n JSON.stringify({\n result: {\n id: assetData.id,\n alt: assetData.alt,\n },\n }),\n )\n }\n\n // Otherwise, assume async job launch\n if (result && ('jobId' in result || 'taskId' in result)) {\n const externalTaskId = result.jobId || result.taskId\n const status = result.status || 'queued'\n const progress = result.progress ?? 0\n\n // Create AI Job doc and return only its id\n const createdJob = await req.payload.create({\n collection: PLUGIN_AI_JOBS_TABLE,\n data: {\n instructionId,\n progress,\n status,\n task_id: externalTaskId,\n },\n overrideAccess: true,\n req,\n })\n\n return new Response(JSON.stringify({ job: { id: createdJob.id } }), {\n headers: { 'Content-Type': 'application/json' },\n })\n }\n\n throw new Error('Unexpected model response.')\n } catch (error) {\n req.payload.logger.error(\n // @ts-expect-error\n error?.type || (error as Error).message,\n 'Error generating upload: ',\n )\n const message =\n error && typeof error === 'object' && 'message' in error\n ? (error as Error).message\n : String(error)\n return new Response(JSON.stringify({ error: message }), {\n headers: { 'Content-Type': 'application/json' },\n status:\n message.includes('Authentication required') ||\n message.includes('Insufficient permissions')\n ? 401\n : 500,\n })\n }\n },\n method: 'post',\n path: PLUGIN_API_ENDPOINT_GENERATE_UPLOAD,\n },\n videogenWebhook: {\n handler: async (req: PayloadRequest) => {\n console.log('videogenWebhook --> ', req)\n try {\n const urlAll = new URL(req.url || '')\n const qpSecret = urlAll.searchParams.get('secret') || ''\n const headerSecret = req.headers.get('x-webhook-secret') || ''\n const falSecret = process.env.FAL_WEBHOOK_SECRET\n const legacySecret = process.env.VIDEOGEN_WEBHOOK_SECRET\n const provided = qpSecret || headerSecret\n // TODO: fal is failing because of auth but webhook seem to work\n if (!provided || (falSecret ? provided !== falSecret : provided !== legacySecret)) {\n return new Response('Unauthorized', { status: 401 })\n }\n\n const instructionId = urlAll.searchParams.get('instructionId')\n if (!instructionId) {\n throw new Error('instructionId missing')\n }\n\n const body = await req.json?.()\n // Normalize fal webhook payload\n const status: string | undefined =\n (body && (body.status || body.data?.status || body.response?.status)) || undefined\n const progress: number | undefined =\n (body && (body.progress ?? body.data?.progress ?? body.response?.progress)) ?? undefined\n const requestId: string | undefined =\n (body &&\n (body.taskId ||\n body.request_id ||\n body.gateway_request_id ||\n body.request?.request_id)) ||\n undefined\n const error = body?.error || body?.data?.error || body?.response?.error\n\n // Update AI Job row by task_id (and instructionId)\n const jobSearch = await req.payload.find({\n collection: PLUGIN_AI_JOBS_TABLE,\n depth: 0,\n limit: 1,\n where: {\n and: [\n { task_id: { equals: requestId } },\n { instructionId: { equals: instructionId } },\n ],\n },\n })\n\n const jobDoc = jobSearch.docs?.[0]\n if (jobDoc) {\n await req.payload.update({\n id: jobDoc.id,\n collection: PLUGIN_AI_JOBS_TABLE,\n data: {\n progress,\n status,\n task_id: requestId,\n },\n overrideAccess: true,\n req,\n })\n }\n\n console.log('fal webhook body: ', body)\n\n const videoUrl =\n body?.outputs?.[0]?.url ||\n body?.data?.outputs?.[0]?.url ||\n body?.video?.url ||\n body?.data?.video?.url ||\n body?.response?.video?.url ||\n body?.videos?.[0]?.url ||\n body?.data?.videos?.[0]?.url\n\n if (status === 'completed' && videoUrl) {\n // Fetch the related instruction to get upload collection\n const instructions = await req.payload.findByID({\n id: instructionId,\n collection: PLUGIN_INSTRUCTIONS_TABLE,\n req,\n })\n\n const uploadCollectionSlug = instructions['relation-to']\n\n const videoResp = await fetch(videoUrl)\n if (!videoResp.ok) {\n throw new Error(`Failed to fetch output: ${videoResp.status}`)\n }\n const buffer = Buffer.from(await videoResp.arrayBuffer())\n\n const created = await req.payload.create({\n collection: uploadCollectionSlug,\n data: { alt: 'video generation' },\n file: {\n name: 'video_generation.mp4',\n data: buffer,\n mimetype: 'video/mp4',\n size: buffer.byteLength,\n },\n overrideAccess: true,\n req,\n })\n\n // Persist the result on the AI Job record\n if (jobDoc) {\n await req.payload.update({\n id: jobDoc.id,\n collection: PLUGIN_AI_JOBS_TABLE,\n data: {\n progress: 100,\n result_id: created?.id,\n status: 'completed',\n },\n overrideAccess: true,\n req,\n })\n }\n }\n\n if (status === 'failed' && error) {\n req.payload.logger.error(error, 'Video generation failed: ')\n }\n\n return new Response(JSON.stringify({ ok: true }), {\n headers: { 'Content-Type': 'application/json' },\n })\n } catch (error) {\n req.payload.logger.error(error, 'Error in videogen webhook: ')\n const message =\n error && typeof error === 'object' && 'message' in error\n ? (error as any).message\n : String(error)\n return new Response(JSON.stringify({ error: message }), {\n headers: { 'Content-Type': 'application/json' },\n status: 500,\n })\n }\n },\n method: 'post',\n path: PLUGIN_API_ENDPOINT_VIDEOGEN_WEBHOOK,\n },\n }) satisfies Endpoints\n"],"names":["process","checkAccess","filterEditorSchemaByNodes","PLUGIN_AI_JOBS_TABLE","PLUGIN_API_ENDPOINT_GENERATE","PLUGIN_API_ENDPOINT_GENERATE_UPLOAD","PLUGIN_API_ENDPOINT_VIDEOGEN_WEBHOOK","PLUGIN_INSTRUCTIONS_TABLE","PLUGIN_NAME","registerEditorHelper","replacePlaceholders","buildSmartPrompt","isGenericPrompt","extractImageData","fetchImages","fieldToJsonSchema","getFieldBySchemaPath","resolveImageReferences","assignPrompt","extendContextWithPromptFields","endpoints","pluginConfig","textarea","handler","req","data","json","allowedEditorNodes","locale","options","action","actionParams","instructionId","contextData","doc","Error","instructions","payload","findByID","id","collection","collections","config","find","slug","custom","editorConfig","admin","schema","editorSchema","prompt","promptTemplate","schemaPath","String","documentData","debugging","logger","info","smartPrompt","allowedEditorSchema","length","receivedNodes","remainingDefinitions","Object","keys","definitions","parts","split","collectionName","fieldName","defaultLocale","locales","localization","localeData","l","code","localeInfo","label","prompts","type","context","field","layout","systemPrompt","system","template","jsonSchema","targetField","targetCollection","c","supported","t","includes","arraySettings","count","modifiedField","maxRows","minRows","nameOverride","e","error","settingsName","undefined","modelSettings","images","resolvedImages","processedPrompt","hardcodedImages","allImages","imageParts","promptToUse","systemToUse","ai","beforeGenerate","beforeHooks","hook","result","headers","streamResult","streamObject","maxTokens","model","onFinish","object","afterGenerate","afterHooks","provider","providerOptions","openai","strictJsonSchema","temperature","message","Response","JSON","stringify","status","method","path","upload","collectionSlug","documentId","docData","draft","sampleImages","extendedContext","text","uploadCollectionSlug","editImages","serverURL","env","SERVER_URL","NEXT_PUBLIC_SERVER_URL","callbackUrl","replace","modelId","instructionSettings","globalDefaults","aiSettings","findGlobal","unsafe","defaultsKey","defaults","filteredInstructionSettings","fromEntries","entries","filter","_","v","generateMedia","assetData","mediaUpload","request","create","alt","file","externalTaskId","jobId","taskId","progress","createdJob","task_id","overrideAccess","job","videogenWebhook","console","log","urlAll","URL","url","qpSecret","searchParams","get","headerSecret","falSecret","FAL_WEBHOOK_SECRET","legacySecret","VIDEOGEN_WEBHOOK_SECRET","provided","body","response","requestId","request_id","gateway_request_id","jobSearch","depth","limit","where","and","equals","jobDoc","docs","update","videoUrl","outputs","video","videos","videoResp","fetch","ok","buffer","Buffer","from","arrayBuffer","created","name","mimetype","size","byteLength","result_id"],"mappings":"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"}
@@ -1,3 +1,4 @@
1
+ export { ArrayComposeField } from '../fields/ArrayComposeField/ArrayComposeField.js';
1
2
  export { ComposeField } from '../fields/ComposeField/ComposeField.js';
2
3
  export { PromptEditorField } from '../fields/PromptEditorField/PromptEditorField.js';
3
4
  export { SelectField } from '../fields/SelectField/SelectField.js';
@@ -1,3 +1,4 @@
1
+ export { ArrayComposeField } from '../fields/ArrayComposeField/ArrayComposeField.js';
1
2
  export { ComposeField } from '../fields/ComposeField/ComposeField.js';
2
3
  export { PromptEditorField } from '../fields/PromptEditorField/PromptEditorField.js';
3
4
  export { SelectField } from '../fields/SelectField/SelectField.js';
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/exports/fields.ts"],"sourcesContent":["export { ComposeField } from '../fields/ComposeField/ComposeField.js'\nexport { PromptEditorField } from '../fields/PromptEditorField/PromptEditorField.js'\nexport { SelectField } from '../fields/SelectField/SelectField.js'\n"],"names":["ComposeField","PromptEditorField","SelectField"],"mappings":"AAAA,SAASA,YAAY,QAAQ,yCAAwC;AACrE,SAASC,iBAAiB,QAAQ,mDAAkD;AACpF,SAASC,WAAW,QAAQ,uCAAsC"}
1
+ {"version":3,"sources":["../../src/exports/fields.ts"],"sourcesContent":["export { ArrayComposeField } from '../fields/ArrayComposeField/ArrayComposeField.js'\nexport { ComposeField } from '../fields/ComposeField/ComposeField.js'\nexport { PromptEditorField } from '../fields/PromptEditorField/PromptEditorField.js'\nexport { SelectField } from '../fields/SelectField/SelectField.js'\n"],"names":["ArrayComposeField","ComposeField","PromptEditorField","SelectField"],"mappings":"AAAA,SAASA,iBAAiB,QAAQ,mDAAkD;AACpF,SAASC,YAAY,QAAQ,yCAAwC;AACrE,SAASC,iBAAiB,QAAQ,mDAAkD;AACpF,SAASC,WAAW,QAAQ,uCAAsC"}
@@ -0,0 +1,15 @@
1
+ import type { ClientField } from 'payload';
2
+ import React from 'react';
3
+ type ArrayComposeFieldProps = {
4
+ [key: string]: any;
5
+ field: ClientField;
6
+ path?: string;
7
+ schemaPath?: string;
8
+ };
9
+ /**
10
+ * ArrayComposeField - A version of ComposeField specifically for array fields.
11
+ * Unlike regular fields, arrays don't have focus events, so this component
12
+ * renders the Compose button immediately (always visible).
13
+ */
14
+ export declare const ArrayComposeField: (props: ArrayComposeFieldProps) => React.JSX.Element;
15
+ export {};
@@ -0,0 +1,87 @@
1
+ 'use client';
2
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
3
+ import { FieldDescription, useDocumentInfo } from '@payloadcms/ui';
4
+ import React from 'react';
5
+ import { createPortal } from 'react-dom';
6
+ import { FieldProvider } from '../../providers/FieldProvider/FieldProvider.js';
7
+ import { useInstructions } from '../../providers/InstructionsProvider/useInstructions.js';
8
+ import { Compose } from '../../ui/Compose/Compose.js';
9
+ import styles from '../../ui/Compose/compose.module.css';
10
+ /**
11
+ * ArrayComposeField - A version of ComposeField specifically for array fields.
12
+ * Unlike regular fields, arrays don't have focus events, so this component
13
+ * renders the Compose button immediately (always visible).
14
+ */ export const ArrayComposeField = (props)=>{
15
+ const { collectionSlug } = useDocumentInfo();
16
+ const finalSchemaPath = props?.schemaPath ?? (collectionSlug ? `${collectionSlug}.${props?.path ?? ''}` : props?.path ?? '');
17
+ const { id: instructionId, disabled, hasInstructions, isConfigAllowed } = useInstructions({
18
+ schemaPath: finalSchemaPath
19
+ });
20
+ // Portal target state
21
+ const [portalTarget, setPortalTarget] = React.useState(null);
22
+ // State to track if the "Add Row" button is present in the DOM
23
+ const [isAddRowPresent, setIsAddRowPresent] = React.useState(true);
24
+ React.useEffect(()=>{
25
+ if (props.field && 'name' in props.field) {
26
+ const fieldId = `field-${props.field.name}`;
27
+ const fieldElement = document.getElementById(fieldId);
28
+ if (fieldElement) {
29
+ fieldElement.classList.add(styles.arrayFieldWrapper);
30
+ // Force relative position via inline style to prevent class application delays/overrides
31
+ fieldElement.style.position = 'relative';
32
+ setPortalTarget(fieldElement);
33
+ // Check initial state
34
+ const checkAddRow = ()=>{
35
+ const btn = fieldElement.querySelector('.array-field__add-row');
36
+ setIsAddRowPresent(!!btn);
37
+ };
38
+ checkAddRow();
39
+ // Observe for changes (e.g. when max rows reached and button is removed)
40
+ const observer = new MutationObserver(checkAddRow);
41
+ observer.observe(fieldElement, {
42
+ childList: true,
43
+ subtree: true
44
+ });
45
+ return ()=>observer.disconnect();
46
+ }
47
+ }
48
+ }, [
49
+ props.field
50
+ ]);
51
+ const adminDescription = props?.field?.admin || {};
52
+ const description = 'description' in adminDescription ? adminDescription.description : '';
53
+ return /*#__PURE__*/ _jsxs(FieldProvider, {
54
+ context: {
55
+ field: props?.field,
56
+ path: props?.path ?? '',
57
+ schemaPath: finalSchemaPath
58
+ },
59
+ children: [
60
+ /*#__PURE__*/ _jsx("div", {
61
+ children: /*#__PURE__*/ _jsx(FieldDescription, {
62
+ description: description,
63
+ path: props?.path
64
+ })
65
+ }),
66
+ hasInstructions && instructionId && !disabled && portalTarget ? /*#__PURE__*/ createPortal(/*#__PURE__*/ _jsx("div", {
67
+ className: `
68
+ ${styles.composePortal}
69
+ ${isAddRowPresent ? styles.composePortalAbsolute : styles.composePortalStatic}
70
+ `,
71
+ children: /*#__PURE__*/ _jsx(Compose, {
72
+ descriptionProps: {
73
+ ...props,
74
+ field: props?.field,
75
+ path: props?.path ?? '',
76
+ schemaPath: finalSchemaPath
77
+ },
78
+ forceVisible: true,
79
+ instructionId: instructionId,
80
+ isConfigAllowed: isConfigAllowed
81
+ })
82
+ }), portalTarget) : null
83
+ ]
84
+ });
85
+ };
86
+
87
+ //# sourceMappingURL=ArrayComposeField.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../src/fields/ArrayComposeField/ArrayComposeField.tsx"],"sourcesContent":["'use client'\n\nimport type { ClientField } from 'payload'\n\nimport { FieldDescription, useDocumentInfo } from '@payloadcms/ui'\nimport React from 'react'\nimport { createPortal } from 'react-dom'\n\nimport { FieldProvider } from '../../providers/FieldProvider/FieldProvider.js'\nimport { useInstructions } from '../../providers/InstructionsProvider/useInstructions.js'\nimport { Compose } from '../../ui/Compose/Compose.js'\nimport styles from '../../ui/Compose/compose.module.css'\n\ntype ArrayComposeFieldProps = {\n [key: string]: any\n field: ClientField\n path?: string\n schemaPath?: string\n}\n\n/**\n * ArrayComposeField - A version of ComposeField specifically for array fields.\n * Unlike regular fields, arrays don't have focus events, so this component\n * renders the Compose button immediately (always visible).\n */\nexport const ArrayComposeField = (props: ArrayComposeFieldProps) => {\n const { collectionSlug } = useDocumentInfo()\n\n const finalSchemaPath =\n props?.schemaPath ??\n (collectionSlug ? `${collectionSlug}.${props?.path ?? ''}` : (props?.path ?? ''))\n\n const {\n id: instructionId,\n disabled,\n hasInstructions,\n isConfigAllowed,\n } = useInstructions({\n schemaPath: finalSchemaPath,\n })\n\n // Portal target state\n const [portalTarget, setPortalTarget] = React.useState<HTMLElement | null>(null)\n\n // State to track if the \"Add Row\" button is present in the DOM\n const [isAddRowPresent, setIsAddRowPresent] = React.useState(true)\n\n React.useEffect(() => {\n if (props.field && 'name' in props.field) {\n const fieldId = `field-${props.field.name}`\n const fieldElement = document.getElementById(fieldId)\n \n if (fieldElement) {\n fieldElement.classList.add(styles.arrayFieldWrapper)\n // Force relative position via inline style to prevent class application delays/overrides\n fieldElement.style.position = 'relative'\n setPortalTarget(fieldElement)\n\n // Check initial state\n const checkAddRow = () => {\n const btn = fieldElement.querySelector('.array-field__add-row')\n setIsAddRowPresent(!!btn)\n }\n \n checkAddRow()\n\n // Observe for changes (e.g. when max rows reached and button is removed)\n const observer = new MutationObserver(checkAddRow)\n observer.observe(fieldElement, { childList: true, subtree: true })\n\n return () => observer.disconnect()\n }\n }\n }, [props.field])\n\n const adminDescription = props?.field?.admin || {}\n const description = 'description' in adminDescription ? adminDescription.description : ''\n\n return (\n <FieldProvider\n context={{\n field: props?.field,\n path: props?.path ?? '',\n schemaPath: finalSchemaPath,\n }}\n >\n <div>\n <FieldDescription description={description as any} path={props?.path as string} />\n </div>\n\n {/* Portal the Compose button to the bottom of the field wrapper */}\n {hasInstructions && instructionId && !disabled && portalTarget\n ? createPortal(\n <div \n className={`\n ${styles.composePortal} \n ${isAddRowPresent ? styles.composePortalAbsolute : styles.composePortalStatic}\n `}\n >\n <Compose\n descriptionProps={{\n ...props,\n field: props?.field,\n path: props?.path ?? '',\n schemaPath: finalSchemaPath,\n }}\n forceVisible={true}\n instructionId={instructionId}\n isConfigAllowed={isConfigAllowed}\n />\n </div>,\n portalTarget,\n )\n : null}\n </FieldProvider>\n )\n}\n"],"names":["FieldDescription","useDocumentInfo","React","createPortal","FieldProvider","useInstructions","Compose","styles","ArrayComposeField","props","collectionSlug","finalSchemaPath","schemaPath","path","id","instructionId","disabled","hasInstructions","isConfigAllowed","portalTarget","setPortalTarget","useState","isAddRowPresent","setIsAddRowPresent","useEffect","field","fieldId","name","fieldElement","document","getElementById","classList","add","arrayFieldWrapper","style","position","checkAddRow","btn","querySelector","observer","MutationObserver","observe","childList","subtree","disconnect","adminDescription","admin","description","context","div","className","composePortal","composePortalAbsolute","composePortalStatic","descriptionProps","forceVisible"],"mappings":"AAAA;;AAIA,SAASA,gBAAgB,EAAEC,eAAe,QAAQ,iBAAgB;AAClE,OAAOC,WAAW,QAAO;AACzB,SAASC,YAAY,QAAQ,YAAW;AAExC,SAASC,aAAa,QAAQ,iDAAgD;AAC9E,SAASC,eAAe,QAAQ,0DAAyD;AACzF,SAASC,OAAO,QAAQ,8BAA6B;AACrD,OAAOC,YAAY,sCAAqC;AASxD;;;;CAIC,GACD,OAAO,MAAMC,oBAAoB,CAACC;IAChC,MAAM,EAAEC,cAAc,EAAE,GAAGT;IAE3B,MAAMU,kBACJF,OAAOG,cACNF,CAAAA,iBAAiB,CAAC,EAAEA,eAAe,CAAC,EAAED,OAAOI,QAAQ,GAAG,CAAC,GAAIJ,OAAOI,QAAQ,EAAE;IAEjF,MAAM,EACJC,IAAIC,aAAa,EACjBC,QAAQ,EACRC,eAAe,EACfC,eAAe,EAChB,GAAGb,gBAAgB;QAClBO,YAAYD;IACd;IAEA,sBAAsB;IACtB,MAAM,CAACQ,cAAcC,gBAAgB,GAAGlB,MAAMmB,QAAQ,CAAqB;IAE3E,+DAA+D;IAC/D,MAAM,CAACC,iBAAiBC,mBAAmB,GAAGrB,MAAMmB,QAAQ,CAAC;IAE7DnB,MAAMsB,SAAS,CAAC;QACd,IAAIf,MAAMgB,KAAK,IAAI,UAAUhB,MAAMgB,KAAK,EAAE;YACxC,MAAMC,UAAU,CAAC,MAAM,EAAEjB,MAAMgB,KAAK,CAACE,IAAI,CAAC,CAAC;YAC3C,MAAMC,eAAeC,SAASC,cAAc,CAACJ;YAE7C,IAAIE,cAAc;gBAChBA,aAAaG,SAAS,CAACC,GAAG,CAACzB,OAAO0B,iBAAiB;gBACnD,yFAAyF;gBACzFL,aAAaM,KAAK,CAACC,QAAQ,GAAG;gBAC9Bf,gBAAgBQ;gBAEhB,sBAAsB;gBACtB,MAAMQ,cAAc;oBAClB,MAAMC,MAAMT,aAAaU,aAAa,CAAC;oBACvCf,mBAAmB,CAAC,CAACc;gBACvB;gBAEAD;gBAEA,yEAAyE;gBACzE,MAAMG,WAAW,IAAIC,iBAAiBJ;gBACtCG,SAASE,OAAO,CAACb,cAAc;oBAAEc,WAAW;oBAAMC,SAAS;gBAAK;gBAEhE,OAAO,IAAMJ,SAASK,UAAU;YAClC;QACF;IACF,GAAG;QAACnC,MAAMgB,KAAK;KAAC;IAEhB,MAAMoB,mBAAmBpC,OAAOgB,OAAOqB,SAAS,CAAC;IACjD,MAAMC,cAAc,iBAAiBF,mBAAmBA,iBAAiBE,WAAW,GAAG;IAEvF,qBACE,MAAC3C;QACC4C,SAAS;YACPvB,OAAOhB,OAAOgB;YACdZ,MAAMJ,OAAOI,QAAQ;YACrBD,YAAYD;QACd;;0BAEA,KAACsC;0BACC,cAAA,KAACjD;oBAAiB+C,aAAaA;oBAAoBlC,MAAMJ,OAAOI;;;YAIjEI,mBAAmBF,iBAAiB,CAACC,YAAYG,6BAC9ChB,2BACE,KAAC8C;gBACCC,WAAW,CAAC;gBACV,EAAE3C,OAAO4C,aAAa,CAAC;gBACvB,EAAE7B,kBAAkBf,OAAO6C,qBAAqB,GAAG7C,OAAO8C,mBAAmB,CAAC;cAChF,CAAC;0BAED,cAAA,KAAC/C;oBACCgD,kBAAkB;wBAChB,GAAG7C,KAAK;wBACRgB,OAAOhB,OAAOgB;wBACdZ,MAAMJ,OAAOI,QAAQ;wBACrBD,YAAYD;oBACd;oBACA4C,cAAc;oBACdxC,eAAeA;oBACfG,iBAAiBA;;gBAGrBC,gBAEF;;;AAGV,EAAC"}
@@ -0,0 +1,73 @@
1
+ 'use client';
2
+ import { FieldDescription, useDocumentInfo } from '@payloadcms/ui';
3
+ import React from 'react';
4
+ import { createPortal } from 'react-dom';
5
+ import { FieldProvider } from '../../providers/FieldProvider/FieldProvider.js';
6
+ import { useInstructions } from '../../providers/InstructionsProvider/useInstructions.js';
7
+ import { Compose } from '../../ui/Compose/Compose.js';
8
+ import styles from '../../ui/Compose/compose.module.css';
9
+ /**
10
+ * ArrayComposeField - A version of ComposeField specifically for array fields.
11
+ * Unlike regular fields, arrays don't have focus events, so this component
12
+ * renders the Compose button immediately (always visible).
13
+ */
14
+ export const ArrayComposeField = (props) => {
15
+ const { collectionSlug } = useDocumentInfo();
16
+ const finalSchemaPath = props?.schemaPath ??
17
+ (collectionSlug ? `${collectionSlug}.${props?.path ?? ''}` : (props?.path ?? ''));
18
+ const { id: instructionId, disabled, hasInstructions, isConfigAllowed, } = useInstructions({
19
+ schemaPath: finalSchemaPath,
20
+ });
21
+ // Portal target state
22
+ const [portalTarget, setPortalTarget] = React.useState(null);
23
+ // State to track if the "Add Row" button is present in the DOM
24
+ const [isAddRowPresent, setIsAddRowPresent] = React.useState(true);
25
+ React.useEffect(() => {
26
+ if (props.field && 'name' in props.field) {
27
+ const fieldId = `field-${props.field.name}`;
28
+ const fieldElement = document.getElementById(fieldId);
29
+ if (fieldElement) {
30
+ fieldElement.classList.add(styles.arrayFieldWrapper);
31
+ // Force relative position via inline style to prevent class application delays/overrides
32
+ fieldElement.style.position = 'relative';
33
+ setPortalTarget(fieldElement);
34
+ // Check initial state
35
+ const checkAddRow = () => {
36
+ const btn = fieldElement.querySelector('.array-field__add-row');
37
+ setIsAddRowPresent(!!btn);
38
+ };
39
+ checkAddRow();
40
+ // Observe for changes (e.g. when max rows reached and button is removed)
41
+ const observer = new MutationObserver(checkAddRow);
42
+ observer.observe(fieldElement, { childList: true, subtree: true });
43
+ return () => observer.disconnect();
44
+ }
45
+ }
46
+ }, [props.field]);
47
+ const adminDescription = props?.field?.admin || {};
48
+ const description = 'description' in adminDescription ? adminDescription.description : '';
49
+ return (<FieldProvider context={{
50
+ field: props?.field,
51
+ path: props?.path ?? '',
52
+ schemaPath: finalSchemaPath,
53
+ }}>
54
+ <div>
55
+ <FieldDescription description={description} path={props?.path}/>
56
+ </div>
57
+
58
+ {/* Portal the Compose button to the bottom of the field wrapper */}
59
+ {hasInstructions && instructionId && !disabled && portalTarget
60
+ ? createPortal(<div className={`
61
+ ${styles.composePortal}
62
+ ${isAddRowPresent ? styles.composePortalAbsolute : styles.composePortalStatic}
63
+ `}>
64
+ <Compose descriptionProps={{
65
+ ...props,
66
+ field: props?.field,
67
+ path: props?.path ?? '',
68
+ schemaPath: finalSchemaPath,
69
+ }} forceVisible={true} instructionId={instructionId} isConfigAllowed={isConfigAllowed}/>
70
+ </div>, portalTarget)
71
+ : null}
72
+ </FieldProvider>);
73
+ };
@@ -156,8 +156,13 @@ export const PromptEditorField = (props)=>{
156
156
  payloadValue
157
157
  ]);
158
158
  const handleChange = useCallback((e)=>{
159
- setLocalValue(e.target.value);
160
- }, []);
159
+ const newValue = e.target.value;
160
+ setLocalValue(newValue);
161
+ // Also update Payload value immediately to prevent loss when Save is clicked
162
+ setValue(newValue);
163
+ }, [
164
+ setValue
165
+ ]);
161
166
  const handleBlur = useCallback(()=>{
162
167
  setValue(localValue);
163
168
  }, [
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../src/fields/PromptEditorField/PromptEditorField.tsx"],"sourcesContent":["'use client'\n\nimport type { TextareaFieldClientProps } from 'payload'\n\nimport { FieldDescription, FieldLabel, useConfig, useField } from '@payloadcms/ui'\nimport React, { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport { Mention, MentionsInput } from 'react-mentions/dist/react-mentions.cjs.js'\n\nimport { useInstructions } from '../../providers/InstructionsProvider/useInstructions.js'\nimport { defaultStyle } from './defaultStyle.js'\n\nexport const PromptEditorField: React.FC<TextareaFieldClientProps> = (props) => {\n const { field, path: pathFromContext } = props\n const { setValue, value: payloadValue } = useField<string>({\n path: pathFromContext,\n })\n\n const [localValue, setLocalValue] = useState(payloadValue || '')\n const hasInitialized = useRef(false)\n\n const { activeCollection, promptEditorSuggestions } = useInstructions()\n const { config } = useConfig()\n\n const suggestions = useMemo(\n () =>\n promptEditorSuggestions.map((suggestion: string) => ({\n id: suggestion,\n display: suggestion,\n })),\n [promptEditorSuggestions],\n )\n\n // Extract document ID from URL if available (to get specific filenames)\n const [documentData, setDocumentData] = useState<null | Record<string, unknown>>(null)\n \n useEffect(() => {\n // Only run in browser\n if (typeof window === 'undefined') {\n return\n }\n\n // Allow time for verify window.location is stable (unlikely to change but good practice)\n const segments = window.location.pathname.split('/')\n const collectionsIndex = segments.indexOf('collections')\n \n if (collectionsIndex > -1 && segments.length > collectionsIndex + 2) {\n const urlCollectionSlug = segments[collectionsIndex + 1]\n const urlId = segments[collectionsIndex + 2]\n\n // Only fetch if we are editing instructions for the same collection we are viewing\n // and we haven't fetched yet (or ID changed)\n if (urlCollectionSlug === activeCollection && urlId && urlId !== 'create') {\n const fetchDocument = async () => {\n try {\n \n const response = await fetch(`${String(config.serverURL)}${String(config.routes.api)}/${String(urlCollectionSlug)}/${String(urlId)}`)\n if (response.ok) {\n const data = await response.json()\n setDocumentData(data)\n }\n } catch (_err) {\n // Ignore error\n }\n }\n void fetchDocument()\n }\n }\n }, [activeCollection, config])\n\n // Extract all upload fields from the current collection schema\n const imageFieldSuggestions = useMemo(() => {\n const suggestions: { display: string; id: string }[] = []\n \n // Use activeCollection from context which holds the target collection slug\n const targetSlug = activeCollection\n \n if (!targetSlug || !config?.collections) {\n return []\n }\n\n const collection = config.collections.find((c) => c.slug === targetSlug)\n if (!collection?.fields) {\n return []\n }\n\n const uploadFields: { hasMany: boolean; name: string }[] = []\n\n // Recursive function to find upload fields\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const findUploadFields = (fields: any[], prefix = ''): void => {\n for (const field of fields) {\n if (field.type === 'upload' && field.name) {\n const fieldPath = prefix ? `${prefix}.${String(field.name)}` : String(field.name)\n uploadFields.push({ name: fieldPath, hasMany: field.hasMany === true })\n }\n // Check nested fields in groups, arrays, etc.\n if (field.fields && Array.isArray(field.fields)) {\n const newPrefix = field.name ? (prefix ? `${prefix}.${String(field.name)}` : String(field.name)) : prefix\n findUploadFields(field.fields, newPrefix)\n }\n }\n }\n\n findUploadFields(collection.fields)\n \n // Add generic field names (base suggestions) - ONLY for single uploads (not hasMany arrays)\n uploadFields.forEach(({ name, hasMany }) => {\n // User requested to hide the array itself for hasMany fields\n if (!hasMany) {\n suggestions.push({ id: name, display: name })\n }\n })\n\n // If we have document data, add specific filename suggestions\n if (documentData) {\n uploadFields.forEach(({ name, hasMany }) => {\n const value = documentData[name] // Note: nested access logic simplified for now\n \n // Helper to extract filename from media doc (which might be ID or object)\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const getFilename = (item: any): null | string => {\n if (typeof item === 'object' && item && (item.filename || item.name)) {\n return item.filename || item.name\n }\n // If it's just an ID, we can't show filename without populating. \n // Assuming compose view usually fetches with depth > 0 or we rely on what we have.\n return null\n }\n\n if (value) {\n if (hasMany && Array.isArray(value)) {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n value.forEach((item: any) => {\n const fname = getFilename(item)\n if (fname) {\n const suggestion = `${name}:${fname}`\n suggestions.push({ id: suggestion, display: suggestion })\n }\n })\n } else if (!hasMany) {\n // Single image - we already added the base name above.\n // We can optionally add the specific filename too if desired, \n // but user request focused on arrays.\n // Adding the specific filename option for Single images too as it's explicit.\n const fname = getFilename(value)\n if (fname) {\n const suggestion = `${name}:${fname}`\n suggestions.push({ id: suggestion, display: suggestion })\n }\n }\n }\n })\n }\n \n return suggestions\n }, [activeCollection, config, documentData])\n\n useEffect(() => {\n if (!hasInitialized.current || payloadValue === '') {\n setLocalValue(payloadValue || '')\n hasInitialized.current = true\n }\n }, [payloadValue])\n\n const handleChange = useCallback((e: React.ChangeEvent<HTMLTextAreaElement>) => {\n setLocalValue(e.target.value)\n }, [])\n\n const handleBlur = useCallback(() => {\n setValue(localValue)\n }, [localValue, setValue])\n\n const displayTransform = useCallback((id: string) => `{{ ${id} }}`, [])\n const imageDisplayTransform = useCallback((id: string) => `@${id}`, [])\n\n return (\n <div className=\"field-type textarea\">\n <FieldLabel label={field.label} />\n <MentionsInput\n onBlur={handleBlur}\n onChange={handleChange}\n placeholder=\"Type {{ for fields }} or @imageField for images. For specific images use @imageField:filename.jpg\"\n style={defaultStyle}\n value={localValue}\n >\n <Mention\n data={suggestions}\n displayTransform={displayTransform}\n markup=\"{{__id__}}\"\n style={{\n backgroundColor: 'var(--theme-elevation-100)',\n padding: '2px 0',\n }}\n trigger=\"{\"\n />\n <Mention\n data={imageFieldSuggestions}\n displayTransform={imageDisplayTransform}\n markup=\"@__id__\"\n style={{\n backgroundColor: 'var(--theme-elevation-150)',\n padding: '2px 0',\n }}\n trigger=\"@\"\n />\n </MentionsInput>\n <FieldDescription description={field?.admin?.description} path=\"\" />\n </div>\n )\n}\n"],"names":["FieldDescription","FieldLabel","useConfig","useField","React","useCallback","useEffect","useMemo","useRef","useState","Mention","MentionsInput","useInstructions","defaultStyle","PromptEditorField","props","field","path","pathFromContext","setValue","value","payloadValue","localValue","setLocalValue","hasInitialized","activeCollection","promptEditorSuggestions","config","suggestions","map","suggestion","id","display","documentData","setDocumentData","window","segments","location","pathname","split","collectionsIndex","indexOf","length","urlCollectionSlug","urlId","fetchDocument","response","fetch","String","serverURL","routes","api","ok","data","json","_err","imageFieldSuggestions","targetSlug","collections","collection","find","c","slug","fields","uploadFields","findUploadFields","prefix","type","name","fieldPath","push","hasMany","Array","isArray","newPrefix","forEach","getFilename","item","filename","fname","current","handleChange","e","target","handleBlur","displayTransform","imageDisplayTransform","div","className","label","onBlur","onChange","placeholder","style","markup","backgroundColor","padding","trigger","description","admin"],"mappings":"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"}
1
+ {"version":3,"sources":["../../../src/fields/PromptEditorField/PromptEditorField.tsx"],"sourcesContent":["'use client'\n\nimport type { TextareaFieldClientProps } from 'payload'\n\nimport { FieldDescription, FieldLabel, useConfig, useField } from '@payloadcms/ui'\nimport React, { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport { Mention, MentionsInput } from 'react-mentions/dist/react-mentions.cjs.js'\n\nimport { useInstructions } from '../../providers/InstructionsProvider/useInstructions.js'\nimport { defaultStyle } from './defaultStyle.js'\n\nexport const PromptEditorField: React.FC<TextareaFieldClientProps> = (props) => {\n const { field, path: pathFromContext } = props\n const { setValue, value: payloadValue } = useField<string>({\n path: pathFromContext,\n })\n\n const [localValue, setLocalValue] = useState(payloadValue || '')\n const hasInitialized = useRef(false)\n\n const { activeCollection, promptEditorSuggestions } = useInstructions()\n const { config } = useConfig()\n\n const suggestions = useMemo(\n () =>\n promptEditorSuggestions.map((suggestion: string) => ({\n id: suggestion,\n display: suggestion,\n })),\n [promptEditorSuggestions],\n )\n\n // Extract document ID from URL if available (to get specific filenames)\n const [documentData, setDocumentData] = useState<null | Record<string, unknown>>(null)\n \n useEffect(() => {\n // Only run in browser\n if (typeof window === 'undefined') {\n return\n }\n\n // Allow time for verify window.location is stable (unlikely to change but good practice)\n const segments = window.location.pathname.split('/')\n const collectionsIndex = segments.indexOf('collections')\n \n if (collectionsIndex > -1 && segments.length > collectionsIndex + 2) {\n const urlCollectionSlug = segments[collectionsIndex + 1]\n const urlId = segments[collectionsIndex + 2]\n\n // Only fetch if we are editing instructions for the same collection we are viewing\n // and we haven't fetched yet (or ID changed)\n if (urlCollectionSlug === activeCollection && urlId && urlId !== 'create') {\n const fetchDocument = async () => {\n try {\n \n const response = await fetch(`${String(config.serverURL)}${String(config.routes.api)}/${String(urlCollectionSlug)}/${String(urlId)}`)\n if (response.ok) {\n const data = await response.json()\n setDocumentData(data)\n }\n } catch (_err) {\n // Ignore error\n }\n }\n void fetchDocument()\n }\n }\n }, [activeCollection, config])\n\n // Extract all upload fields from the current collection schema\n const imageFieldSuggestions = useMemo(() => {\n const suggestions: { display: string; id: string }[] = []\n \n // Use activeCollection from context which holds the target collection slug\n const targetSlug = activeCollection\n \n if (!targetSlug || !config?.collections) {\n return []\n }\n\n const collection = config.collections.find((c) => c.slug === targetSlug)\n if (!collection?.fields) {\n return []\n }\n\n const uploadFields: { hasMany: boolean; name: string }[] = []\n\n // Recursive function to find upload fields\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const findUploadFields = (fields: any[], prefix = ''): void => {\n for (const field of fields) {\n if (field.type === 'upload' && field.name) {\n const fieldPath = prefix ? `${prefix}.${String(field.name)}` : String(field.name)\n uploadFields.push({ name: fieldPath, hasMany: field.hasMany === true })\n }\n // Check nested fields in groups, arrays, etc.\n if (field.fields && Array.isArray(field.fields)) {\n const newPrefix = field.name ? (prefix ? `${prefix}.${String(field.name)}` : String(field.name)) : prefix\n findUploadFields(field.fields, newPrefix)\n }\n }\n }\n\n findUploadFields(collection.fields)\n \n // Add generic field names (base suggestions) - ONLY for single uploads (not hasMany arrays)\n uploadFields.forEach(({ name, hasMany }) => {\n // User requested to hide the array itself for hasMany fields\n if (!hasMany) {\n suggestions.push({ id: name, display: name })\n }\n })\n\n // If we have document data, add specific filename suggestions\n if (documentData) {\n uploadFields.forEach(({ name, hasMany }) => {\n const value = documentData[name] // Note: nested access logic simplified for now\n \n // Helper to extract filename from media doc (which might be ID or object)\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const getFilename = (item: any): null | string => {\n if (typeof item === 'object' && item && (item.filename || item.name)) {\n return item.filename || item.name\n }\n // If it's just an ID, we can't show filename without populating. \n // Assuming compose view usually fetches with depth > 0 or we rely on what we have.\n return null\n }\n\n if (value) {\n if (hasMany && Array.isArray(value)) {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n value.forEach((item: any) => {\n const fname = getFilename(item)\n if (fname) {\n const suggestion = `${name}:${fname}`\n suggestions.push({ id: suggestion, display: suggestion })\n }\n })\n } else if (!hasMany) {\n // Single image - we already added the base name above.\n // We can optionally add the specific filename too if desired, \n // but user request focused on arrays.\n // Adding the specific filename option for Single images too as it's explicit.\n const fname = getFilename(value)\n if (fname) {\n const suggestion = `${name}:${fname}`\n suggestions.push({ id: suggestion, display: suggestion })\n }\n }\n }\n })\n }\n \n return suggestions\n }, [activeCollection, config, documentData])\n\n useEffect(() => {\n if (!hasInitialized.current || payloadValue === '') {\n setLocalValue(payloadValue || '')\n hasInitialized.current = true\n }\n }, [payloadValue])\n\n const handleChange = useCallback((e: React.ChangeEvent<HTMLTextAreaElement>) => {\n const newValue = e.target.value\n setLocalValue(newValue)\n // Also update Payload value immediately to prevent loss when Save is clicked\n setValue(newValue)\n }, [setValue])\n\n const handleBlur = useCallback(() => {\n setValue(localValue)\n }, [localValue, setValue])\n\n const displayTransform = useCallback((id: string) => `{{ ${id} }}`, [])\n const imageDisplayTransform = useCallback((id: string) => `@${id}`, [])\n\n return (\n <div className=\"field-type textarea\">\n <FieldLabel label={field.label} />\n <MentionsInput\n onBlur={handleBlur}\n onChange={handleChange}\n placeholder=\"Type {{ for fields }} or @imageField for images. For specific images use @imageField:filename.jpg\"\n style={defaultStyle}\n value={localValue}\n >\n <Mention\n data={suggestions}\n displayTransform={displayTransform}\n markup=\"{{__id__}}\"\n style={{\n backgroundColor: 'var(--theme-elevation-100)',\n padding: '2px 0',\n }}\n trigger=\"{\"\n />\n <Mention\n data={imageFieldSuggestions}\n displayTransform={imageDisplayTransform}\n markup=\"@__id__\"\n style={{\n backgroundColor: 'var(--theme-elevation-150)',\n padding: '2px 0',\n }}\n trigger=\"@\"\n />\n </MentionsInput>\n <FieldDescription description={field?.admin?.description} path=\"\" />\n </div>\n )\n}\n"],"names":["FieldDescription","FieldLabel","useConfig","useField","React","useCallback","useEffect","useMemo","useRef","useState","Mention","MentionsInput","useInstructions","defaultStyle","PromptEditorField","props","field","path","pathFromContext","setValue","value","payloadValue","localValue","setLocalValue","hasInitialized","activeCollection","promptEditorSuggestions","config","suggestions","map","suggestion","id","display","documentData","setDocumentData","window","segments","location","pathname","split","collectionsIndex","indexOf","length","urlCollectionSlug","urlId","fetchDocument","response","fetch","String","serverURL","routes","api","ok","data","json","_err","imageFieldSuggestions","targetSlug","collections","collection","find","c","slug","fields","uploadFields","findUploadFields","prefix","type","name","fieldPath","push","hasMany","Array","isArray","newPrefix","forEach","getFilename","item","filename","fname","current","handleChange","e","newValue","target","handleBlur","displayTransform","imageDisplayTransform","div","className","label","onBlur","onChange","placeholder","style","markup","backgroundColor","padding","trigger","description","admin"],"mappings":"AAAA;;AAIA,SAASA,gBAAgB,EAAEC,UAAU,EAAEC,SAAS,EAAEC,QAAQ,QAAQ,iBAAgB;AAClF,OAAOC,SAASC,WAAW,EAAEC,SAAS,EAAEC,OAAO,EAAEC,MAAM,EAAEC,QAAQ,QAAQ,QAAO;AAChF,SAASC,OAAO,EAAEC,aAAa,QAAQ,4CAA2C;AAElF,SAASC,eAAe,QAAQ,0DAAyD;AACzF,SAASC,YAAY,QAAQ,oBAAmB;AAEhD,OAAO,MAAMC,oBAAwD,CAACC;IACpE,MAAM,EAAEC,KAAK,EAAEC,MAAMC,eAAe,EAAE,GAAGH;IACzC,MAAM,EAAEI,QAAQ,EAAEC,OAAOC,YAAY,EAAE,GAAGlB,SAAiB;QACzDc,MAAMC;IACR;IAEA,MAAM,CAACI,YAAYC,cAAc,GAAGd,SAASY,gBAAgB;IAC7D,MAAMG,iBAAiBhB,OAAO;IAE9B,MAAM,EAAEiB,gBAAgB,EAAEC,uBAAuB,EAAE,GAAGd;IACtD,MAAM,EAAEe,MAAM,EAAE,GAAGzB;IAEnB,MAAM0B,cAAcrB,QAClB,IACEmB,wBAAwBG,GAAG,CAAC,CAACC,aAAwB,CAAA;gBACnDC,IAAID;gBACJE,SAASF;YACX,CAAA,IACF;QAACJ;KAAwB;IAG3B,wEAAwE;IACxE,MAAM,CAACO,cAAcC,gBAAgB,GAAGzB,SAAyC;IAEjFH,UAAU;QACR,sBAAsB;QACtB,IAAI,OAAO6B,WAAW,aAAa;YACjC;QACF;QAEA,yFAAyF;QACzF,MAAMC,WAAWD,OAAOE,QAAQ,CAACC,QAAQ,CAACC,KAAK,CAAC;QAChD,MAAMC,mBAAmBJ,SAASK,OAAO,CAAC;QAE1C,IAAID,mBAAmB,CAAC,KAAKJ,SAASM,MAAM,GAAGF,mBAAmB,GAAG;YACnE,MAAMG,oBAAoBP,QAAQ,CAACI,mBAAmB,EAAE;YACxD,MAAMI,QAAQR,QAAQ,CAACI,mBAAmB,EAAE;YAE5C,mFAAmF;YACnF,6CAA6C;YAC7C,IAAIG,sBAAsBlB,oBAAoBmB,SAASA,UAAU,UAAU;gBACxE,MAAMC,gBAAgB;oBACpB,IAAI;wBAEF,MAAMC,WAAW,MAAMC,MAAM,CAAC,EAAEC,OAAOrB,OAAOsB,SAAS,EAAE,EAAED,OAAOrB,OAAOuB,MAAM,CAACC,GAAG,EAAE,CAAC,EAAEH,OAAOL,mBAAmB,CAAC,EAAEK,OAAOJ,OAAO,CAAC;wBACpI,IAAIE,SAASM,EAAE,EAAE;4BACf,MAAMC,OAAO,MAAMP,SAASQ,IAAI;4BAChCpB,gBAAgBmB;wBAClB;oBACF,EAAE,OAAOE,MAAM;oBACb,eAAe;oBACjB;gBACF;gBACA,KAAKV;YACR;QACF;IACF,GAAG;QAACpB;QAAkBE;KAAO;IAE7B,+DAA+D;IAC/D,MAAM6B,wBAAwBjD,QAAQ;QACpC,MAAMqB,cAAiD,EAAE;QAEzD,2EAA2E;QAC3E,MAAM6B,aAAahC;QAEnB,IAAI,CAACgC,cAAc,CAAC9B,QAAQ+B,aAAa;YACvC,OAAO,EAAE;QACX;QAEA,MAAMC,aAAahC,OAAO+B,WAAW,CAACE,IAAI,CAAC,CAACC,IAAMA,EAAEC,IAAI,KAAKL;QAC7D,IAAI,CAACE,YAAYI,QAAQ;YACvB,OAAO,EAAE;QACX;QAEA,MAAMC,eAAqD,EAAE;QAE7D,2CAA2C;QAC3C,8DAA8D;QAC9D,MAAMC,mBAAmB,CAACF,QAAeG,SAAS,EAAE;YAClD,KAAK,MAAMlD,SAAS+C,OAAQ;gBAC1B,IAAI/C,MAAMmD,IAAI,KAAK,YAAYnD,MAAMoD,IAAI,EAAE;oBACzC,MAAMC,YAAYH,SAAS,CAAC,EAAEA,OAAO,CAAC,EAAElB,OAAOhC,MAAMoD,IAAI,EAAE,CAAC,GAAGpB,OAAOhC,MAAMoD,IAAI;oBAChFJ,aAAaM,IAAI,CAAC;wBAAEF,MAAMC;wBAAWE,SAASvD,MAAMuD,OAAO,KAAK;oBAAK;gBACvE;gBACA,8CAA8C;gBAC9C,IAAIvD,MAAM+C,MAAM,IAAIS,MAAMC,OAAO,CAACzD,MAAM+C,MAAM,GAAG;oBAC/C,MAAMW,YAAY1D,MAAMoD,IAAI,GAAIF,SAAS,CAAC,EAAEA,OAAO,CAAC,EAAElB,OAAOhC,MAAMoD,IAAI,EAAE,CAAC,GAAGpB,OAAOhC,MAAMoD,IAAI,IAAKF;oBACnGD,iBAAiBjD,MAAM+C,MAAM,EAAEW;gBACjC;YACF;QACF;QAEAT,iBAAiBN,WAAWI,MAAM;QAElC,4FAA4F;QAC5FC,aAAaW,OAAO,CAAC,CAAC,EAAEP,IAAI,EAAEG,OAAO,EAAE;YACrC,6DAA6D;YAC7D,IAAI,CAACA,SAAS;gBACZ3C,YAAY0C,IAAI,CAAC;oBAAEvC,IAAIqC;oBAAMpC,SAASoC;gBAAK;YAC7C;QACF;QAEA,8DAA8D;QAC9D,IAAInC,cAAc;YAChB+B,aAAaW,OAAO,CAAC,CAAC,EAAEP,IAAI,EAAEG,OAAO,EAAE;gBACrC,MAAMnD,QAAQa,YAAY,CAACmC,KAAK,CAAC,+CAA+C;;gBAEhF,0EAA0E;gBAC1E,8DAA8D;gBAC9D,MAAMQ,cAAc,CAACC;oBAClB,IAAI,OAAOA,SAAS,YAAYA,QAASA,CAAAA,KAAKC,QAAQ,IAAID,KAAKT,IAAI,AAAD,GAAI;wBACpE,OAAOS,KAAKC,QAAQ,IAAID,KAAKT,IAAI;oBACnC;oBACA,kEAAkE;oBAClE,mFAAmF;oBACnF,OAAO;gBACV;gBAEA,IAAIhD,OAAO;oBACR,IAAImD,WAAWC,MAAMC,OAAO,CAACrD,QAAQ;wBACnC,8DAA8D;wBAC9DA,MAAMuD,OAAO,CAAC,CAACE;4BACb,MAAME,QAAQH,YAAYC;4BAC1B,IAAIE,OAAO;gCACT,MAAMjD,aAAa,CAAC,EAAEsC,KAAK,CAAC,EAAEW,MAAM,CAAC;gCACrCnD,YAAY0C,IAAI,CAAC;oCAAEvC,IAAID;oCAAYE,SAASF;gCAAW;4BACzD;wBACF;oBACF,OAAO,IAAI,CAACyC,SAAS;wBACnB,uDAAuD;wBACvD,+DAA+D;wBAC/D,sCAAsC;wBACtC,8EAA8E;wBAC9E,MAAMQ,QAAQH,YAAYxD;wBAC1B,IAAI2D,OAAO;4BACP,MAAMjD,aAAa,CAAC,EAAEsC,KAAK,CAAC,EAAEW,MAAM,CAAC;4BACrCnD,YAAY0C,IAAI,CAAC;gCAAEvC,IAAID;gCAAYE,SAASF;4BAAW;wBAC3D;oBACF;gBACH;YACF;QACF;QAEA,OAAOF;IACT,GAAG;QAACH;QAAkBE;QAAQM;KAAa;IAE3C3B,UAAU;QACR,IAAI,CAACkB,eAAewD,OAAO,IAAI3D,iBAAiB,IAAI;YAClDE,cAAcF,gBAAgB;YAC9BG,eAAewD,OAAO,GAAG;QAC3B;IACF,GAAG;QAAC3D;KAAa;IAEjB,MAAM4D,eAAe5E,YAAY,CAAC6E;QAChC,MAAMC,WAAWD,EAAEE,MAAM,CAAChE,KAAK;QAC/BG,cAAc4D;QACd,6EAA6E;QAC7EhE,SAASgE;IACX,GAAG;QAAChE;KAAS;IAEb,MAAMkE,aAAahF,YAAY;QAC7Bc,SAASG;IACX,GAAG;QAACA;QAAYH;KAAS;IAEzB,MAAMmE,mBAAmBjF,YAAY,CAAC0B,KAAe,CAAC,GAAG,EAAEA,GAAG,GAAG,CAAC,EAAE,EAAE;IACtE,MAAMwD,wBAAwBlF,YAAY,CAAC0B,KAAe,CAAC,CAAC,EAAEA,GAAG,CAAC,EAAE,EAAE;IAEtE,qBACE,MAACyD;QAAIC,WAAU;;0BACb,KAACxF;gBAAWyF,OAAO1E,MAAM0E,KAAK;;0BAC9B,MAAC/E;gBACCgF,QAAQN;gBACRO,UAAUX;gBACVY,aAAY;gBACZC,OAAOjF;gBACPO,OAAOE;;kCAEP,KAACZ;wBACC2C,MAAMzB;wBACN0D,kBAAkBA;wBAClBS,QAAO;wBACPD,OAAO;4BACLE,iBAAiB;4BACjBC,SAAS;wBACX;wBACAC,SAAQ;;kCAEV,KAACxF;wBACC2C,MAAMG;wBACN8B,kBAAkBC;wBAClBQ,QAAO;wBACPD,OAAO;4BACLE,iBAAiB;4BACjBC,SAAS;wBACX;wBACAC,SAAQ;;;;0BAGZ,KAAClG;gBAAiBmG,aAAanF,OAAOoF,OAAOD;gBAAalF,MAAK;;;;AAGrE,EAAC"}
@@ -133,8 +133,11 @@ export const PromptEditorField = (props) => {
133
133
  }
134
134
  }, [payloadValue]);
135
135
  const handleChange = useCallback((e) => {
136
- setLocalValue(e.target.value);
137
- }, []);
136
+ const newValue = e.target.value;
137
+ setLocalValue(newValue);
138
+ // Also update Payload value immediately to prevent loss when Save is clicked
139
+ setValue(newValue);
140
+ }, [setValue]);
138
141
  const handleBlur = useCallback(() => {
139
142
  setValue(localValue);
140
143
  }, [localValue, setValue]);
package/dist/index.d.ts CHANGED
@@ -1,4 +1,6 @@
1
1
  export type { GenerateArgs } from './ai/index.js';
2
- export { defaultPrompts, defaultSeedPrompts } from './ai/prompts.js';
2
+ export { defaultPrompts } from './ai/prompts.js';
3
3
  export { PayloadAiPluginLexicalEditorFeature } from './fields/LexicalEditor/feature.server.js';
4
+ export type {} from './payload-ai.d.ts';
4
5
  export { payloadAiPlugin } from './plugin.js';
6
+ export { fieldToJsonSchema } from './utilities/fieldToJsonSchema.js';
package/dist/index.js CHANGED
@@ -1,5 +1,6 @@
1
- export { defaultPrompts, defaultSeedPrompts } from './ai/prompts.js';
1
+ export { defaultPrompts } from './ai/prompts.js';
2
2
  export { PayloadAiPluginLexicalEditorFeature } from './fields/LexicalEditor/feature.server.js';
3
3
  export { payloadAiPlugin } from './plugin.js';
4
+ export { fieldToJsonSchema } from './utilities/fieldToJsonSchema.js';
4
5
 
5
6
  //# sourceMappingURL=index.js.map
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts"],"sourcesContent":["export type { GenerateArgs } from './ai/index.js'\n\nexport { defaultPrompts, defaultSeedPrompts } from './ai/prompts.js'\nexport { PayloadAiPluginLexicalEditorFeature } from './fields/LexicalEditor/feature.server.js'\n\nexport { payloadAiPlugin } from './plugin.js'\n"],"names":["defaultPrompts","defaultSeedPrompts","PayloadAiPluginLexicalEditorFeature","payloadAiPlugin"],"mappings":"AAEA,SAASA,cAAc,EAAEC,kBAAkB,QAAQ,kBAAiB;AACpE,SAASC,mCAAmC,QAAQ,2CAA0C;AAE9F,SAASC,eAAe,QAAQ,cAAa"}
1
+ {"version":3,"sources":["../src/index.ts"],"sourcesContent":["export type { GenerateArgs } from './ai/index.js'\n\nexport { defaultPrompts } from './ai/prompts.js'\n\nexport { PayloadAiPluginLexicalEditorFeature } from './fields/LexicalEditor/feature.server.js'\n// Re-export to ensure payload.ai module augmentation is included\nexport type {} from './payload-ai.d.ts'\n\nexport { payloadAiPlugin } from './plugin.js'\nexport { fieldToJsonSchema } from './utilities/fieldToJsonSchema.js'\n"],"names":["defaultPrompts","PayloadAiPluginLexicalEditorFeature","payloadAiPlugin","fieldToJsonSchema"],"mappings":"AAEA,SAASA,cAAc,QAAQ,kBAAiB;AAEhD,SAASC,mCAAmC,QAAQ,2CAA0C;AAI9F,SAASC,eAAe,QAAQ,cAAa;AAC7C,SAASC,iBAAiB,QAAQ,mCAAkC"}