@strapi/upload 5.25.0 → 5.27.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 (123) hide show
  1. package/dist/admin/ai/components/AIAssetCard.js +589 -0
  2. package/dist/admin/ai/components/AIAssetCard.js.map +1 -0
  3. package/dist/admin/ai/components/AIAssetCard.mjs +567 -0
  4. package/dist/admin/ai/components/AIAssetCard.mjs.map +1 -0
  5. package/dist/admin/ai/components/AIUploadModal.js +355 -0
  6. package/dist/admin/ai/components/AIUploadModal.js.map +1 -0
  7. package/dist/admin/ai/components/AIUploadModal.mjs +333 -0
  8. package/dist/admin/ai/components/AIUploadModal.mjs.map +1 -0
  9. package/dist/admin/components/AssetDialog/AssetDialog.js +1 -1
  10. package/dist/admin/components/AssetDialog/AssetDialog.js.map +1 -1
  11. package/dist/admin/components/AssetDialog/AssetDialog.mjs +1 -1
  12. package/dist/admin/components/AssetDialog/AssetDialog.mjs.map +1 -1
  13. package/dist/admin/components/EditAssetDialog/EditAssetContent.js +5 -5
  14. package/dist/admin/components/EditAssetDialog/EditAssetContent.js.map +1 -1
  15. package/dist/admin/components/EditAssetDialog/EditAssetContent.mjs +5 -5
  16. package/dist/admin/components/EditAssetDialog/EditAssetContent.mjs.map +1 -1
  17. package/dist/admin/components/UploadAssetDialog/AddAssetStep/FromComputerForm.js +0 -15
  18. package/dist/admin/components/UploadAssetDialog/AddAssetStep/FromComputerForm.js.map +1 -1
  19. package/dist/admin/components/UploadAssetDialog/AddAssetStep/FromComputerForm.mjs +1 -16
  20. package/dist/admin/components/UploadAssetDialog/AddAssetStep/FromComputerForm.mjs.map +1 -1
  21. package/dist/admin/hooks/useAiAvailability.js +22 -0
  22. package/dist/admin/hooks/useAiAvailability.js.map +1 -0
  23. package/dist/admin/hooks/useAiAvailability.mjs +20 -0
  24. package/dist/admin/hooks/useAiAvailability.mjs.map +1 -0
  25. package/dist/admin/hooks/useBulkEdit.js +66 -0
  26. package/dist/admin/hooks/useBulkEdit.js.map +1 -0
  27. package/dist/admin/hooks/useBulkEdit.mjs +64 -0
  28. package/dist/admin/hooks/useBulkEdit.mjs.map +1 -0
  29. package/dist/admin/hooks/useSettings.js +22 -0
  30. package/dist/admin/hooks/useSettings.js.map +1 -0
  31. package/dist/admin/hooks/useSettings.mjs +20 -0
  32. package/dist/admin/hooks/useSettings.mjs.map +1 -0
  33. package/dist/admin/hooks/useUpload.js +25 -14
  34. package/dist/admin/hooks/useUpload.js.map +1 -1
  35. package/dist/admin/hooks/useUpload.mjs +25 -14
  36. package/dist/admin/hooks/useUpload.mjs.map +1 -1
  37. package/dist/admin/package.json.js +5 -5
  38. package/dist/admin/package.json.mjs +5 -5
  39. package/dist/admin/pages/App/MediaLibrary/MediaLibrary.js +11 -4
  40. package/dist/admin/pages/App/MediaLibrary/MediaLibrary.js.map +1 -1
  41. package/dist/admin/pages/App/MediaLibrary/MediaLibrary.mjs +11 -4
  42. package/dist/admin/pages/App/MediaLibrary/MediaLibrary.mjs.map +1 -1
  43. package/dist/admin/pages/SettingsPage/SettingsPage.js +222 -144
  44. package/dist/admin/pages/SettingsPage/SettingsPage.js.map +1 -1
  45. package/dist/admin/pages/SettingsPage/SettingsPage.mjs +225 -147
  46. package/dist/admin/pages/SettingsPage/SettingsPage.mjs.map +1 -1
  47. package/dist/admin/pages/SettingsPage/reducer.js +9 -10
  48. package/dist/admin/pages/SettingsPage/reducer.js.map +1 -1
  49. package/dist/admin/pages/SettingsPage/reducer.mjs +9 -10
  50. package/dist/admin/pages/SettingsPage/reducer.mjs.map +1 -1
  51. package/dist/admin/src/ai/components/AIAssetCard.d.ts +13 -0
  52. package/dist/admin/src/ai/components/AIUploadModal.d.ts +55 -0
  53. package/dist/admin/src/components/EditAssetDialog/EditAssetContent.d.ts +3 -1
  54. package/dist/admin/src/hooks/useAiAvailability.d.ts +4 -0
  55. package/dist/admin/src/hooks/useBulkEdit.d.ts +91 -0
  56. package/dist/admin/src/hooks/useSettings.d.ts +7 -0
  57. package/dist/admin/src/hooks/useUpload.d.ts +1 -1
  58. package/dist/admin/src/pages/SettingsPage/reducer.d.ts +3 -12
  59. package/dist/admin/translations/en.json.js +7 -1
  60. package/dist/admin/translations/en.json.js.map +1 -1
  61. package/dist/admin/translations/en.json.mjs +7 -1
  62. package/dist/admin/translations/en.json.mjs.map +1 -1
  63. package/dist/server/bootstrap.js +2 -1
  64. package/dist/server/bootstrap.js.map +1 -1
  65. package/dist/server/bootstrap.mjs +2 -1
  66. package/dist/server/bootstrap.mjs.map +1 -1
  67. package/dist/server/controllers/admin-upload.js +54 -2
  68. package/dist/server/controllers/admin-upload.js.map +1 -1
  69. package/dist/server/controllers/admin-upload.mjs +56 -4
  70. package/dist/server/controllers/admin-upload.mjs.map +1 -1
  71. package/dist/server/controllers/content-api.js +3 -1
  72. package/dist/server/controllers/content-api.js.map +1 -1
  73. package/dist/server/controllers/content-api.mjs +3 -1
  74. package/dist/server/controllers/content-api.mjs.map +1 -1
  75. package/dist/server/controllers/validation/admin/settings.js +2 -1
  76. package/dist/server/controllers/validation/admin/settings.js.map +1 -1
  77. package/dist/server/controllers/validation/admin/settings.mjs +2 -1
  78. package/dist/server/controllers/validation/admin/settings.mjs.map +1 -1
  79. package/dist/server/controllers/validation/admin/upload.js +8 -0
  80. package/dist/server/controllers/validation/admin/upload.js.map +1 -1
  81. package/dist/server/controllers/validation/admin/upload.mjs +8 -1
  82. package/dist/server/controllers/validation/admin/upload.mjs.map +1 -1
  83. package/dist/server/routes/admin.js +18 -0
  84. package/dist/server/routes/admin.js.map +1 -1
  85. package/dist/server/routes/admin.mjs +18 -0
  86. package/dist/server/routes/admin.mjs.map +1 -1
  87. package/dist/server/services/ai-metadata.js +97 -0
  88. package/dist/server/services/ai-metadata.js.map +1 -0
  89. package/dist/server/services/ai-metadata.mjs +95 -0
  90. package/dist/server/services/ai-metadata.mjs.map +1 -0
  91. package/dist/server/services/index.js +3 -1
  92. package/dist/server/services/index.js.map +1 -1
  93. package/dist/server/services/index.mjs +3 -1
  94. package/dist/server/services/index.mjs.map +1 -1
  95. package/dist/server/services/upload.js.map +1 -1
  96. package/dist/server/services/upload.mjs.map +1 -1
  97. package/dist/server/src/bootstrap.d.ts.map +1 -1
  98. package/dist/server/src/controllers/admin-upload.d.ts +1 -0
  99. package/dist/server/src/controllers/admin-upload.d.ts.map +1 -1
  100. package/dist/server/src/controllers/index.d.ts +1 -0
  101. package/dist/server/src/controllers/index.d.ts.map +1 -1
  102. package/dist/server/src/controllers/validation/admin/settings.d.ts +3 -0
  103. package/dist/server/src/controllers/validation/admin/settings.d.ts.map +1 -1
  104. package/dist/server/src/controllers/validation/admin/upload.d.ts +42 -0
  105. package/dist/server/src/controllers/validation/admin/upload.d.ts.map +1 -1
  106. package/dist/server/src/index.d.ts +13 -1
  107. package/dist/server/src/index.d.ts.map +1 -1
  108. package/dist/server/src/routes/admin.d.ts.map +1 -1
  109. package/dist/server/src/services/ai-metadata.d.ts +13 -0
  110. package/dist/server/src/services/ai-metadata.d.ts.map +1 -0
  111. package/dist/server/src/services/index.d.ts +12 -1
  112. package/dist/server/src/services/index.d.ts.map +1 -1
  113. package/dist/server/src/services/upload.d.ts +2 -1
  114. package/dist/server/src/services/upload.d.ts.map +1 -1
  115. package/dist/server/src/types.d.ts +1 -0
  116. package/dist/server/src/types.d.ts.map +1 -1
  117. package/dist/server/src/utils/index.d.ts +2 -0
  118. package/dist/server/src/utils/index.d.ts.map +1 -1
  119. package/dist/server/utils/index.js.map +1 -1
  120. package/dist/server/utils/index.mjs.map +1 -1
  121. package/dist/shared/contracts/files.d.ts +22 -0
  122. package/dist/shared/contracts/settings.d.ts +2 -0
  123. package/package.json +5 -5
@@ -0,0 +1,355 @@
1
+ 'use strict';
2
+
3
+ var jsxRuntime = require('react/jsx-runtime');
4
+ var React = require('react');
5
+ var strapiAdmin = require('@strapi/admin/strapi-admin');
6
+ var designSystem = require('@strapi/design-system');
7
+ var immer = require('immer');
8
+ var reactIntl = require('react-intl');
9
+ var styledComponents = require('styled-components');
10
+ var AddAssetStep = require('../../components/UploadAssetDialog/AddAssetStep/AddAssetStep.js');
11
+ var useBulkEdit = require('../../hooks/useBulkEdit.js');
12
+ var useUpload = require('../../hooks/useUpload.js');
13
+ require('byte-size');
14
+ require('date-fns');
15
+ var getTrad = require('../../utils/getTrad.js');
16
+ require('qs');
17
+ require('../../constants.js');
18
+ require('../../utils/urlYupSchema.js');
19
+ var AIAssetCard = require('./AIAssetCard.js');
20
+
21
+ function _interopNamespaceDefault(e) {
22
+ var n = Object.create(null);
23
+ if (e) {
24
+ Object.keys(e).forEach(function (k) {
25
+ if (k !== 'default') {
26
+ var d = Object.getOwnPropertyDescriptor(e, k);
27
+ Object.defineProperty(n, k, d.get ? d : {
28
+ enumerable: true,
29
+ get: function () { return e[k]; }
30
+ });
31
+ }
32
+ });
33
+ }
34
+ n.default = e;
35
+ return Object.freeze(n);
36
+ }
37
+
38
+ var React__namespace = /*#__PURE__*/_interopNamespaceDefault(React);
39
+
40
+ /* -------------------------------------------------------------------------------------------------
41
+ * ModalBody
42
+ * -----------------------------------------------------------------------------------------------*/ const StyledModalBody = styledComponents.styled(designSystem.Modal.Body)`
43
+ padding: 0;
44
+ display: flex;
45
+ justify-content: center;
46
+
47
+ [data-radix-scroll-area-viewport] {
48
+ padding-top: ${({ theme })=>theme.spaces[6]};
49
+ padding-bottom: ${({ theme })=>theme.spaces[6]};
50
+ padding-left: ${({ theme })=>theme.spaces[7]};
51
+ padding-right: ${({ theme })=>theme.spaces[7]};
52
+ }
53
+ `;
54
+ const StyledAlert = styledComponents.styled(designSystem.Alert)`
55
+ & > button {
56
+ display: none;
57
+ }
58
+ `;
59
+ const ModalContent = ({ onClose })=>{
60
+ const { formatMessage } = reactIntl.useIntl();
61
+ const state = useAIUploadModalContext('ModalContent', (s)=>s.state);
62
+ const dispatch = useAIUploadModalContext('ModalContent', (s)=>s.dispatch);
63
+ const folderId = useAIUploadModalContext('ModalContent', (s)=>s.folderId);
64
+ const { upload } = useUpload.useUpload();
65
+ const { edit, isLoading: isSaving } = useBulkEdit.useBulkEdit();
66
+ const [isUploading, setIsUploading] = React__namespace.useState(false);
67
+ const [uploadError, setUploadError] = React__namespace.useState(null);
68
+ const handleCaptionChange = (assetId, caption)=>{
69
+ dispatch({
70
+ type: 'set_uploaded_asset_caption',
71
+ payload: {
72
+ id: assetId,
73
+ caption
74
+ }
75
+ });
76
+ };
77
+ const handleAltTextChange = (assetId, altText)=>{
78
+ dispatch({
79
+ type: 'set_uploaded_asset_alt_text',
80
+ payload: {
81
+ id: assetId,
82
+ altText
83
+ }
84
+ });
85
+ };
86
+ const resetState = ()=>{
87
+ dispatch({
88
+ type: 'set_uploaded_assets',
89
+ payload: []
90
+ });
91
+ };
92
+ const handleFinish = async ()=>{
93
+ if (state.hasUnsavedChanges) {
94
+ const assetsToUpdate = state.uploadedAssets.filter((asset)=>(asset.wasCaptionChanged || asset.wasAltTextChanged) && asset.file.id);
95
+ if (assetsToUpdate.length > 0) {
96
+ const updates = assetsToUpdate.map((asset)=>({
97
+ id: asset.file.id,
98
+ fileInfo: {
99
+ name: asset.file.name,
100
+ alternativeText: asset.file.alternativeText ?? null,
101
+ caption: asset.file.caption ?? null,
102
+ folder: typeof asset.file.folder === 'object' && asset.file.folder !== null ? asset.file.folder.id : asset.file.folder
103
+ }
104
+ }));
105
+ try {
106
+ await edit(updates);
107
+ dispatch({
108
+ type: 'clear_unsaved_changes'
109
+ });
110
+ } catch (err) {
111
+ console.error('Failed to save asset changes:', err);
112
+ return; // Don't close modal on error
113
+ }
114
+ }
115
+ }
116
+ resetState();
117
+ onClose();
118
+ };
119
+ const handleCancel = ()=>{
120
+ resetState();
121
+ onClose();
122
+ };
123
+ const handleUpload = async (assets)=>{
124
+ dispatch({
125
+ type: 'set_assets_to_upload_length',
126
+ payload: assets.length
127
+ });
128
+ setUploadError(null);
129
+ setIsUploading(true);
130
+ try {
131
+ const assetsForUpload = assets.map((asset)=>({
132
+ ...asset,
133
+ id: asset.id ? Number(asset.id) : undefined
134
+ }));
135
+ const uploadedFiles = await upload(assetsForUpload, folderId);
136
+ const filesWithFolder = uploadedFiles.map((file)=>({
137
+ ...file,
138
+ // The upload API doesn't populate the folder relation, so we add it manually
139
+ folder: folderId || file.folder
140
+ }));
141
+ dispatch({
142
+ type: 'set_uploaded_assets',
143
+ payload: filesWithFolder
144
+ });
145
+ } catch (error) {
146
+ console.error('Upload failed:', error);
147
+ setUploadError(error instanceof Error ? error : new Error('Upload failed'));
148
+ } finally{
149
+ setIsUploading(false);
150
+ }
151
+ };
152
+ if (state.assetsToUploadLength === 0) {
153
+ return /*#__PURE__*/ jsxRuntime.jsx(designSystem.Modal.Content, {
154
+ children: /*#__PURE__*/ jsxRuntime.jsx(AddAssetStep.AddAssetStep, {
155
+ onClose: onClose,
156
+ onAddAsset: handleUpload
157
+ })
158
+ });
159
+ }
160
+ if (isUploading || state.assetsToUploadLength > 0 && state.uploadedAssets.length === 0 && !uploadError) {
161
+ return /*#__PURE__*/ jsxRuntime.jsxs(designSystem.Modal.Content, {
162
+ children: [
163
+ /*#__PURE__*/ jsxRuntime.jsx(designSystem.Modal.Header, {
164
+ children: /*#__PURE__*/ jsxRuntime.jsx(designSystem.Modal.Title, {
165
+ children: formatMessage({
166
+ id: getTrad.getTrad('ai.modal.uploading.title'),
167
+ defaultMessage: 'Uploading and processing with AI...'
168
+ })
169
+ })
170
+ }),
171
+ /*#__PURE__*/ jsxRuntime.jsx(StyledModalBody, {
172
+ children: /*#__PURE__*/ jsxRuntime.jsx(AIAssetCard.AIAssetCardSkeletons, {
173
+ count: state.assetsToUploadLength
174
+ })
175
+ })
176
+ ]
177
+ });
178
+ }
179
+ const title = formatMessage({
180
+ id: getTrad.getTrad('ai.modal.title'),
181
+ defaultMessage: '{count, plural, one {# asset uploaded} other {# assets uploaded}}, review AI generated metadata'
182
+ }, {
183
+ count: state.uploadedAssets.length
184
+ });
185
+ if (uploadError) {
186
+ return /*#__PURE__*/ jsxRuntime.jsxs(designSystem.Modal.Content, {
187
+ children: [
188
+ /*#__PURE__*/ jsxRuntime.jsx(designSystem.Modal.Header, {
189
+ children: /*#__PURE__*/ jsxRuntime.jsx(designSystem.Modal.Title, {
190
+ children: title
191
+ })
192
+ }),
193
+ /*#__PURE__*/ jsxRuntime.jsx(designSystem.Modal.Body, {
194
+ children: /*#__PURE__*/ jsxRuntime.jsx(StyledAlert, {
195
+ closeLabel: "",
196
+ variant: "danger",
197
+ children: formatMessage({
198
+ id: getTrad.getTrad('ai.modal.error'),
199
+ defaultMessage: 'Could not generate AI metadata for the uploaded files.'
200
+ })
201
+ })
202
+ }),
203
+ /*#__PURE__*/ jsxRuntime.jsxs(designSystem.Modal.Footer, {
204
+ children: [
205
+ /*#__PURE__*/ jsxRuntime.jsx(designSystem.Button, {
206
+ onClick: handleCancel,
207
+ variant: "tertiary",
208
+ children: formatMessage({
209
+ id: 'cancel',
210
+ defaultMessage: 'Cancel'
211
+ })
212
+ }),
213
+ /*#__PURE__*/ jsxRuntime.jsx(designSystem.Button, {
214
+ onClick: handleFinish,
215
+ loading: isSaving,
216
+ children: formatMessage({
217
+ id: 'global.finish',
218
+ defaultMessage: 'Finish'
219
+ })
220
+ })
221
+ ]
222
+ })
223
+ ]
224
+ });
225
+ }
226
+ return /*#__PURE__*/ jsxRuntime.jsxs(designSystem.Modal.Content, {
227
+ children: [
228
+ /*#__PURE__*/ jsxRuntime.jsx(designSystem.Modal.Header, {
229
+ children: /*#__PURE__*/ jsxRuntime.jsx(designSystem.Modal.Title, {
230
+ children: title
231
+ })
232
+ }),
233
+ /*#__PURE__*/ jsxRuntime.jsx(StyledModalBody, {
234
+ children: /*#__PURE__*/ jsxRuntime.jsx(designSystem.Flex, {
235
+ gap: 6,
236
+ direction: "column",
237
+ alignItems: "stretch",
238
+ children: state.uploadedAssets.map(({ file: asset, wasCaptionChanged, wasAltTextChanged })=>/*#__PURE__*/ jsxRuntime.jsx(AIAssetCard.AIAssetCard, {
239
+ asset: asset,
240
+ onCaptionChange: (caption)=>asset.id && handleCaptionChange(asset.id, caption),
241
+ onAltTextChange: (altText)=>asset.id && handleAltTextChange(asset.id, altText),
242
+ wasCaptionChanged: wasCaptionChanged,
243
+ wasAltTextChanged: wasAltTextChanged
244
+ }, asset.id))
245
+ })
246
+ }),
247
+ /*#__PURE__*/ jsxRuntime.jsxs(designSystem.Modal.Footer, {
248
+ children: [
249
+ /*#__PURE__*/ jsxRuntime.jsx(designSystem.Button, {
250
+ onClick: handleCancel,
251
+ variant: "tertiary",
252
+ children: formatMessage({
253
+ id: 'cancel',
254
+ defaultMessage: 'Cancel'
255
+ })
256
+ }),
257
+ /*#__PURE__*/ jsxRuntime.jsx(designSystem.Button, {
258
+ onClick: handleFinish,
259
+ loading: isSaving,
260
+ children: formatMessage({
261
+ id: 'global.finish',
262
+ defaultMessage: 'Finish'
263
+ })
264
+ })
265
+ ]
266
+ })
267
+ ]
268
+ });
269
+ };
270
+ const [AIUploadModalContext, useAIUploadModalContext] = strapiAdmin.createContext('AIUploadModalContext');
271
+ const reducer = (state, action)=>{
272
+ return immer.produce(state, (draft)=>{
273
+ if (action.type === 'set_uploaded_assets') {
274
+ draft.uploadedAssets = action.payload.map((file)=>({
275
+ file,
276
+ wasCaptionChanged: false,
277
+ wasAltTextChanged: false
278
+ }));
279
+ draft.hasUnsavedChanges = false;
280
+ }
281
+ if (action.type === 'set_assets_to_upload_length') {
282
+ draft.assetsToUploadLength = action.payload;
283
+ }
284
+ if (action.type === 'set_uploaded_asset_caption') {
285
+ const asset = draft.uploadedAssets.find((a)=>a.file.id === action.payload.id);
286
+ if (asset && asset.file.caption !== action.payload.caption) {
287
+ asset.file.caption = action.payload.caption;
288
+ asset.wasCaptionChanged = true;
289
+ draft.hasUnsavedChanges = true;
290
+ }
291
+ }
292
+ if (action.type === 'set_uploaded_asset_alt_text') {
293
+ const asset = draft.uploadedAssets.find((a)=>a.file.id === action.payload.id);
294
+ if (asset && asset.file.alternativeText !== action.payload.altText) {
295
+ asset.file.alternativeText = action.payload.altText;
296
+ asset.wasAltTextChanged = true;
297
+ draft.hasUnsavedChanges = true;
298
+ }
299
+ }
300
+ if (action.type === 'remove_uploaded_asset') {
301
+ draft.uploadedAssets = draft.uploadedAssets.filter((a)=>a.file.id !== action.payload.id);
302
+ }
303
+ if (action.type === 'edit_uploaded_asset') {
304
+ const assetIndex = draft.uploadedAssets.findIndex((a)=>a.file.id === action.payload.editedAsset.id);
305
+ if (assetIndex !== -1) {
306
+ draft.uploadedAssets[assetIndex] = {
307
+ file: action.payload.editedAsset,
308
+ wasCaptionChanged: draft.uploadedAssets[assetIndex].wasCaptionChanged,
309
+ wasAltTextChanged: draft.uploadedAssets[assetIndex].wasAltTextChanged
310
+ };
311
+ }
312
+ }
313
+ if (action.type === 'clear_unsaved_changes') {
314
+ draft.hasUnsavedChanges = false;
315
+ draft.uploadedAssets.forEach((asset)=>{
316
+ asset.wasCaptionChanged = false;
317
+ asset.wasAltTextChanged = false;
318
+ });
319
+ }
320
+ });
321
+ };
322
+ const AIUploadModal = ({ open, onClose, folderId = null })=>{
323
+ const [state, dispatch] = React__namespace.useReducer(reducer, {
324
+ uploadedAssets: [],
325
+ assetsToUploadLength: 0,
326
+ hasUnsavedChanges: false
327
+ });
328
+ const handleClose = React__namespace.useCallback(()=>{
329
+ // Reset state when modal closes
330
+ dispatch({
331
+ type: 'set_uploaded_assets',
332
+ payload: []
333
+ });
334
+ onClose();
335
+ }, [
336
+ onClose
337
+ ]);
338
+ return /*#__PURE__*/ jsxRuntime.jsx(AIUploadModalContext, {
339
+ state: state,
340
+ dispatch: dispatch,
341
+ folderId: folderId,
342
+ onClose: handleClose,
343
+ children: /*#__PURE__*/ jsxRuntime.jsx(designSystem.Modal.Root, {
344
+ open: open,
345
+ onOpenChange: handleClose,
346
+ children: /*#__PURE__*/ jsxRuntime.jsx(ModalContent, {
347
+ onClose: handleClose
348
+ })
349
+ })
350
+ });
351
+ };
352
+
353
+ exports.AIUploadModal = AIUploadModal;
354
+ exports.useAIUploadModalContext = useAIUploadModalContext;
355
+ //# sourceMappingURL=AIUploadModal.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"AIUploadModal.js","sources":["../../../../admin/src/ai/components/AIUploadModal.tsx"],"sourcesContent":["import * as React from 'react';\n\nimport { createContext } from '@strapi/admin/strapi-admin';\nimport { Alert, Button, Flex, Modal } from '@strapi/design-system';\nimport { produce } from 'immer';\nimport { useIntl } from 'react-intl';\nimport { styled } from 'styled-components';\n\nimport {\n AddAssetStep,\n FileWithRawFile,\n} from '../../components/UploadAssetDialog/AddAssetStep/AddAssetStep';\nimport { useBulkEdit } from '../../hooks/useBulkEdit';\nimport { useUpload } from '../../hooks/useUpload';\nimport { getTrad } from '../../utils';\n\nimport { AIAssetCard, AIAssetCardSkeletons } from './AIAssetCard';\n\nimport type { File } from '../../../../shared/contracts/files';\n\n/* -------------------------------------------------------------------------------------------------\n * ModalBody\n * -----------------------------------------------------------------------------------------------*/\n\nconst StyledModalBody = styled(Modal.Body)`\n padding: 0;\n display: flex;\n justify-content: center;\n\n [data-radix-scroll-area-viewport] {\n padding-top: ${({ theme }) => theme.spaces[6]};\n padding-bottom: ${({ theme }) => theme.spaces[6]};\n padding-left: ${({ theme }) => theme.spaces[7]};\n padding-right: ${({ theme }) => theme.spaces[7]};\n }\n`;\n\nconst StyledAlert = styled(Alert)`\n & > button {\n display: none;\n }\n`;\n\nconst ModalContent = ({ onClose }: Pick<AIUploadModalProps, 'onClose'>) => {\n const { formatMessage } = useIntl();\n const state = useAIUploadModalContext('ModalContent', (s) => s.state);\n const dispatch = useAIUploadModalContext('ModalContent', (s) => s.dispatch);\n const folderId = useAIUploadModalContext('ModalContent', (s) => s.folderId);\n const { upload } = useUpload();\n const { edit, isLoading: isSaving } = useBulkEdit();\n const [isUploading, setIsUploading] = React.useState(false);\n const [uploadError, setUploadError] = React.useState<Error | null>(null);\n\n const handleCaptionChange = (assetId: number, caption: string) => {\n dispatch({\n type: 'set_uploaded_asset_caption',\n payload: { id: assetId, caption },\n });\n };\n\n const handleAltTextChange = (assetId: number, altText: string) => {\n dispatch({\n type: 'set_uploaded_asset_alt_text',\n payload: { id: assetId, altText },\n });\n };\n\n const resetState = () => {\n dispatch({ type: 'set_uploaded_assets', payload: [] });\n };\n\n const handleFinish = async () => {\n if (state.hasUnsavedChanges) {\n const assetsToUpdate = state.uploadedAssets.filter(\n (asset) => (asset.wasCaptionChanged || asset.wasAltTextChanged) && asset.file.id\n );\n\n if (assetsToUpdate.length > 0) {\n const updates = assetsToUpdate.map((asset) => ({\n id: asset.file.id!,\n fileInfo: {\n name: asset.file.name,\n alternativeText: asset.file.alternativeText ?? null,\n caption: asset.file.caption ?? null,\n folder:\n typeof asset.file.folder === 'object' && asset.file.folder !== null\n ? // @ts-expect-error types are wrong\n asset.file.folder.id\n : asset.file.folder,\n },\n }));\n\n try {\n await edit(updates);\n dispatch({ type: 'clear_unsaved_changes' });\n } catch (err) {\n console.error('Failed to save asset changes:', err);\n return; // Don't close modal on error\n }\n }\n }\n\n resetState();\n onClose();\n };\n\n const handleCancel = () => {\n resetState();\n onClose();\n };\n\n const handleUpload = async (assets: FileWithRawFile[]) => {\n dispatch({ type: 'set_assets_to_upload_length', payload: assets.length });\n setUploadError(null);\n setIsUploading(true);\n\n try {\n const assetsForUpload = assets.map((asset) => ({\n ...asset,\n id: asset.id ? Number(asset.id) : undefined,\n }));\n\n const uploadedFiles = await upload(assetsForUpload, folderId);\n const filesWithFolder = uploadedFiles.map((file: File) => ({\n ...file,\n // The upload API doesn't populate the folder relation, so we add it manually\n folder: folderId || file.folder,\n }));\n dispatch({ type: 'set_uploaded_assets', payload: filesWithFolder });\n } catch (error) {\n console.error('Upload failed:', error);\n setUploadError(error instanceof Error ? error : new Error('Upload failed'));\n } finally {\n setIsUploading(false);\n }\n };\n\n if (state.assetsToUploadLength === 0) {\n return (\n <Modal.Content>\n <AddAssetStep onClose={onClose} onAddAsset={handleUpload} />\n </Modal.Content>\n );\n }\n\n if (\n isUploading ||\n (state.assetsToUploadLength > 0 && state.uploadedAssets.length === 0 && !uploadError)\n ) {\n return (\n <Modal.Content>\n <Modal.Header>\n <Modal.Title>\n {formatMessage({\n id: getTrad('ai.modal.uploading.title'),\n defaultMessage: 'Uploading and processing with AI...',\n })}\n </Modal.Title>\n </Modal.Header>\n <StyledModalBody>\n <AIAssetCardSkeletons count={state.assetsToUploadLength} />\n </StyledModalBody>\n </Modal.Content>\n );\n }\n\n const title = formatMessage(\n {\n id: getTrad('ai.modal.title'),\n defaultMessage:\n '{count, plural, one {# asset uploaded} other {# assets uploaded}}, review AI generated metadata',\n },\n { count: state.uploadedAssets.length }\n );\n\n if (uploadError) {\n return (\n <Modal.Content>\n <Modal.Header>\n <Modal.Title>{title}</Modal.Title>\n </Modal.Header>\n <Modal.Body>\n <StyledAlert closeLabel=\"\" variant=\"danger\">\n {formatMessage({\n id: getTrad('ai.modal.error'),\n defaultMessage: 'Could not generate AI metadata for the uploaded files.',\n })}\n </StyledAlert>\n </Modal.Body>\n <Modal.Footer>\n <Button onClick={handleCancel} variant=\"tertiary\">\n {formatMessage({ id: 'cancel', defaultMessage: 'Cancel' })}\n </Button>\n <Button onClick={handleFinish} loading={isSaving}>\n {formatMessage({ id: 'global.finish', defaultMessage: 'Finish' })}\n </Button>\n </Modal.Footer>\n </Modal.Content>\n );\n }\n\n return (\n <Modal.Content>\n <Modal.Header>\n <Modal.Title>{title}</Modal.Title>\n </Modal.Header>\n\n <StyledModalBody>\n <Flex gap={6} direction=\"column\" alignItems=\"stretch\">\n {state.uploadedAssets.map(({ file: asset, wasCaptionChanged, wasAltTextChanged }) => (\n <AIAssetCard\n key={asset.id}\n asset={asset}\n onCaptionChange={(caption: string) =>\n asset.id && handleCaptionChange(asset.id, caption)\n }\n onAltTextChange={(altText: string) =>\n asset.id && handleAltTextChange(asset.id, altText)\n }\n wasCaptionChanged={wasCaptionChanged}\n wasAltTextChanged={wasAltTextChanged}\n />\n ))}\n </Flex>\n </StyledModalBody>\n\n <Modal.Footer>\n <Button onClick={handleCancel} variant=\"tertiary\">\n {formatMessage({ id: 'cancel', defaultMessage: 'Cancel' })}\n </Button>\n <Button onClick={handleFinish} loading={isSaving}>\n {formatMessage({ id: 'global.finish', defaultMessage: 'Finish' })}\n </Button>\n </Modal.Footer>\n </Modal.Content>\n );\n};\n\n/* -------------------------------------------------------------------------------------------------\n * UploadModal\n * -----------------------------------------------------------------------------------------------*/\n\ninterface AIUploadModalProps {\n open: boolean;\n onClose: () => void;\n folderId?: number | null;\n}\n\ntype State = {\n uploadedAssets: Array<{ file: File; wasCaptionChanged: boolean; wasAltTextChanged: boolean }>;\n assetsToUploadLength: number;\n hasUnsavedChanges: boolean;\n};\n\ntype Action =\n | {\n type: 'set_uploaded_assets';\n payload: File[];\n }\n | {\n type: 'set_assets_to_upload_length';\n payload: number;\n }\n | {\n type: 'set_uploaded_asset_caption';\n payload: { id: number; caption: string };\n }\n | {\n type: 'set_uploaded_asset_alt_text';\n payload: { id: number; altText: string };\n }\n | {\n type: 'remove_uploaded_asset';\n payload: { id: number };\n }\n | {\n type: 'edit_uploaded_asset';\n payload: { editedAsset: File };\n }\n | {\n type: 'clear_unsaved_changes';\n };\n\nconst [AIUploadModalContext, useAIUploadModalContext] = createContext<{\n state: State;\n dispatch: React.Dispatch<Action>;\n folderId: number | null;\n onClose: () => void;\n}>('AIUploadModalContext');\n\nconst reducer = (state: State, action: Action): State => {\n return produce(state, (draft: State) => {\n if (action.type === 'set_uploaded_assets') {\n draft.uploadedAssets = action.payload.map((file) => ({\n file,\n wasCaptionChanged: false,\n wasAltTextChanged: false,\n }));\n draft.hasUnsavedChanges = false;\n }\n\n if (action.type === 'set_assets_to_upload_length') {\n draft.assetsToUploadLength = action.payload;\n }\n\n if (action.type === 'set_uploaded_asset_caption') {\n const asset = draft.uploadedAssets.find((a) => a.file.id === action.payload.id);\n if (asset && asset.file.caption !== action.payload.caption) {\n asset.file.caption = action.payload.caption;\n asset.wasCaptionChanged = true;\n draft.hasUnsavedChanges = true;\n }\n }\n\n if (action.type === 'set_uploaded_asset_alt_text') {\n const asset = draft.uploadedAssets.find((a) => a.file.id === action.payload.id);\n if (asset && asset.file.alternativeText !== action.payload.altText) {\n asset.file.alternativeText = action.payload.altText;\n asset.wasAltTextChanged = true;\n draft.hasUnsavedChanges = true;\n }\n }\n\n if (action.type === 'remove_uploaded_asset') {\n draft.uploadedAssets = draft.uploadedAssets.filter((a) => a.file.id !== action.payload.id);\n }\n\n if (action.type === 'edit_uploaded_asset') {\n const assetIndex = draft.uploadedAssets.findIndex(\n (a) => a.file.id === action.payload.editedAsset.id\n );\n if (assetIndex !== -1) {\n draft.uploadedAssets[assetIndex] = {\n file: action.payload.editedAsset,\n wasCaptionChanged: draft.uploadedAssets[assetIndex].wasCaptionChanged,\n wasAltTextChanged: draft.uploadedAssets[assetIndex].wasAltTextChanged,\n };\n }\n }\n\n if (action.type === 'clear_unsaved_changes') {\n draft.hasUnsavedChanges = false;\n draft.uploadedAssets.forEach((asset) => {\n asset.wasCaptionChanged = false;\n asset.wasAltTextChanged = false;\n });\n }\n });\n};\n\nexport const AIUploadModal = ({ open, onClose, folderId = null }: AIUploadModalProps) => {\n const [state, dispatch] = React.useReducer(reducer, {\n uploadedAssets: [],\n assetsToUploadLength: 0,\n hasUnsavedChanges: false,\n });\n\n const handleClose = React.useCallback(() => {\n // Reset state when modal closes\n dispatch({ type: 'set_uploaded_assets', payload: [] });\n onClose();\n }, [onClose]);\n\n return (\n <AIUploadModalContext\n state={state}\n dispatch={dispatch}\n folderId={folderId}\n onClose={handleClose}\n >\n <Modal.Root open={open} onOpenChange={handleClose}>\n <ModalContent onClose={handleClose} />\n </Modal.Root>\n </AIUploadModalContext>\n );\n};\n\nexport { useAIUploadModalContext };\n"],"names":["StyledModalBody","styled","Modal","Body","theme","spaces","StyledAlert","Alert","ModalContent","onClose","formatMessage","useIntl","state","useAIUploadModalContext","s","dispatch","folderId","upload","useUpload","edit","isLoading","isSaving","useBulkEdit","isUploading","setIsUploading","React","useState","uploadError","setUploadError","handleCaptionChange","assetId","caption","type","payload","id","handleAltTextChange","altText","resetState","handleFinish","hasUnsavedChanges","assetsToUpdate","uploadedAssets","filter","asset","wasCaptionChanged","wasAltTextChanged","file","length","updates","map","fileInfo","name","alternativeText","folder","err","console","error","handleCancel","handleUpload","assets","assetsForUpload","Number","undefined","uploadedFiles","filesWithFolder","Error","assetsToUploadLength","_jsx","Content","AddAssetStep","onAddAsset","_jsxs","Header","Title","getTrad","defaultMessage","AIAssetCardSkeletons","count","title","closeLabel","variant","Footer","Button","onClick","loading","Flex","gap","direction","alignItems","AIAssetCard","onCaptionChange","onAltTextChange","AIUploadModalContext","createContext","reducer","action","produce","draft","find","a","assetIndex","findIndex","editedAsset","forEach","AIUploadModal","open","useReducer","handleClose","useCallback","Root","onOpenChange"],"mappings":"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"}