@commercetools-frontend-extensions/delete-resources-modal 1.1.11 → 1.2.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.
@@ -11,7 +11,7 @@ import { defineMessages, useIntl, FormattedMessage, IntlProvider } from 'react-i
11
11
  import { useApplicationContext } from '@commercetools-frontend/application-shell-connectors';
12
12
  import _slicedToArray from '@babel/runtime-corejs3/helpers/esm/slicedToArray';
13
13
  import React, { useContext, useState, useEffect } from 'react';
14
- import { deleteImportContainer, EnabledDropArea, FileDroppedArea, ActiveDragDropArea, useImportContainerUpload, getRowCount, toBytes, MAX_FILE_SIZE_MB, hasSingleKeyColumn, MAX_ROW_COUNT, isAbortError, HttpError, FileDropArea as FileDropArea$1, UploadingModal, mapFileUploadErrorsToUploadFileErrorRows, processUploadedFile } from '@commercetools-frontend-extensions/operations';
14
+ import { deleteImportContainer, EnabledDropArea, FileDroppedArea, ActiveDragDropArea, useFileUpload, getRowCount, toBytes, hasSingleKeyColumn, countUniqueResourcesInCsv, isAbortError, HttpError, IMPORT_MAX_FILE_SIZE_MB, IMPORT_LEGACY_MAX_FILE_SIZE_MB, IMPORT_MAX_ITEM_COUNT, IMPORT_LEGACY_MAX_ROW_COUNT, FileDropArea as FileDropArea$1, UploadingModal, mapFileUploadErrorsToUploadFileErrorRows, processFileImportJob, processUploadedFile } from '@commercetools-frontend-extensions/operations';
15
15
  import { jsx, jsxs } from '@emotion/react/jsx-runtime';
16
16
  import { FormDialog, InfoDialog } from '@commercetools-frontend/application-components';
17
17
  import { Masking } from '@commercetools-frontend/fullstory';
@@ -25,60 +25,85 @@ import _possibleConstructorReturn from '@babel/runtime-corejs3/helpers/esm/possi
25
25
  import _getPrototypeOf from '@babel/runtime-corejs3/helpers/esm/getPrototypeOf';
26
26
  import _inherits from '@babel/runtime-corejs3/helpers/esm/inherits';
27
27
  import _wrapNativeSuper from '@babel/runtime-corejs3/helpers/esm/wrapNativeSuper';
28
+ import { useFeatureToggle } from '@commercetools-frontend/application-shell';
28
29
  import { reportErrorToSentry } from '@commercetools-frontend/sentry';
29
30
  import _sliceInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/slice';
30
31
  import '@emotion/react';
31
32
  import { parseChunkImport, mapLocaleToIntlLocale } from '@commercetools-frontend/i18n';
32
33
 
33
- function ownKeys$6(e, r) { var t = _Object$keys(e); if (_Object$getOwnPropertySymbols) { var o = _Object$getOwnPropertySymbols(e); r && (o = _filterInstanceProperty(o).call(o, function (r) { return _Object$getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
34
- function _objectSpread$6(e) { for (var r = 1; r < arguments.length; r++) { var _context, _context2; var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? _forEachInstanceProperty(_context = ownKeys$6(Object(t), !0)).call(_context, function (r) { _defineProperty(e, r, t[r]); }) : _Object$getOwnPropertyDescriptors ? _Object$defineProperties(e, _Object$getOwnPropertyDescriptors(t)) : _forEachInstanceProperty(_context2 = ownKeys$6(Object(t))).call(_context2, function (r) { _Object$defineProperty(e, r, _Object$getOwnPropertyDescriptor(t, r)); }); } return e; }
34
+ function ownKeys$7(e, r) { var t = _Object$keys(e); if (_Object$getOwnPropertySymbols) { var o = _Object$getOwnPropertySymbols(e); r && (o = _filterInstanceProperty(o).call(o, function (r) { return _Object$getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
35
+ function _objectSpread$7(e) { for (var r = 1; r < arguments.length; r++) { var _context, _context2; var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? _forEachInstanceProperty(_context = ownKeys$7(Object(t), !0)).call(_context, function (r) { _defineProperty(e, r, t[r]); }) : _Object$getOwnPropertyDescriptors ? _Object$defineProperties(e, _Object$getOwnPropertyDescriptors(t)) : _forEachInstanceProperty(_context2 = ownKeys$7(Object(t))).call(_context2, function (r) { _Object$defineProperty(e, r, _Object$getOwnPropertyDescriptor(t, r)); }); } return e; }
35
36
  function reducer(state, action) {
36
- if (action.type === 'setCurrentStep') return _objectSpread$6(_objectSpread$6({}, state), {}, {
37
+ if (action.type === 'setCurrentStep') return _objectSpread$7(_objectSpread$7({}, state), {}, {
37
38
  currentStep: action.currentStep,
38
39
  progress: 0
39
40
  });
40
- if (action.type === 'setContainerKey') return _objectSpread$6(_objectSpread$6({}, state), {}, {
41
+ if (action.type === 'setContainerKey') return _objectSpread$7(_objectSpread$7({}, state), {}, {
41
42
  containerKey: action.containerKey
42
43
  });
43
- if (action.type === 'cancelImport') return _objectSpread$6(_objectSpread$6({}, state), {}, {
44
+ if (action.type === 'cancelImport') return _objectSpread$7(_objectSpread$7({}, state), {}, {
44
45
  currentStep: 'upload',
45
46
  containerKey: undefined,
46
47
  droppedFile: undefined,
47
48
  dropAreaState: 'ready-for-drop',
48
- progress: 0
49
+ progress: 0,
50
+ jobId: undefined,
51
+ totalResourceCount: undefined,
52
+ validationProcessed: undefined,
53
+ isValidating: false
49
54
  });
50
- if (action.type === 'uploadNewFile') return _objectSpread$6(_objectSpread$6({}, state), {}, {
55
+ if (action.type === 'uploadNewFile') return _objectSpread$7(_objectSpread$7({}, state), {}, {
51
56
  currentStep: 'upload',
52
57
  containerKey: undefined,
53
58
  droppedFile: undefined,
54
59
  dropAreaState: 'ready-for-drop',
55
- progress: 0
60
+ progress: 0,
61
+ jobId: undefined,
62
+ totalResourceCount: undefined,
63
+ validationProcessed: undefined,
64
+ isValidating: false
56
65
  });
57
66
  if (action.type === 'setDroppedFile') {
58
- return _objectSpread$6(_objectSpread$6({}, state), {}, {
67
+ return _objectSpread$7(_objectSpread$7({}, state), {}, {
59
68
  droppedFile: action.droppedFile
60
69
  });
61
70
  }
62
71
  if (action.type === 'setAbortController') {
63
- return _objectSpread$6(_objectSpread$6({}, state), {}, {
72
+ return _objectSpread$7(_objectSpread$7({}, state), {}, {
64
73
  abortController: action.abortController
65
74
  });
66
75
  }
67
76
  if (action.type === 'setProgress') {
68
- return _objectSpread$6(_objectSpread$6({}, state), {}, {
77
+ return _objectSpread$7(_objectSpread$7({}, state), {}, {
69
78
  progress: action.progress
70
79
  });
71
80
  }
72
81
  if (action.type === 'setUploadFileResponse') {
73
- return _objectSpread$6(_objectSpread$6({}, state), {}, {
82
+ return _objectSpread$7(_objectSpread$7({}, state), {}, {
74
83
  uploadFileResponse: action.uploadFileResponse
75
84
  });
76
85
  }
77
86
  if (action.type === 'setFileUploadErrors') {
78
- return _objectSpread$6(_objectSpread$6({}, state), {}, {
87
+ return _objectSpread$7(_objectSpread$7({}, state), {}, {
79
88
  fileUploadErrors: action.fileUploadErrors
80
89
  });
81
90
  }
91
+ if (action.type === 'setJobId') {
92
+ return _objectSpread$7(_objectSpread$7({}, state), {}, {
93
+ jobId: action.jobId
94
+ });
95
+ }
96
+ if (action.type === 'setTotalResourceCount') {
97
+ return _objectSpread$7(_objectSpread$7({}, state), {}, {
98
+ totalResourceCount: action.totalResourceCount
99
+ });
100
+ }
101
+ if (action.type === 'setValidationProgress') {
102
+ return _objectSpread$7(_objectSpread$7({}, state), {}, {
103
+ validationProcessed: action.processed,
104
+ isValidating: action.isValidating
105
+ });
106
+ }
82
107
  throw new Error(getUnknownActionError(action));
83
108
  }
84
109
  function getUnknownActionError(actionType) {
@@ -120,12 +145,22 @@ const initialActions = {
120
145
  },
121
146
  setProgress(_progress) {
122
147
  /**/
148
+ },
149
+ setJobId(_jobId) {
150
+ /**/
151
+ },
152
+ setTotalResourceCount(_totalResourceCount) {
153
+ /**/
154
+ },
155
+ setValidationProgress(_params) {
156
+ /**/
123
157
  }
124
158
  };
125
159
 
126
160
  const DeleteResourcesContext = /*#__PURE__*/React.createContext({
127
161
  state: initialState,
128
162
  actions: initialActions,
163
+ resourceType: 'product',
129
164
  onClose: () => {}
130
165
  });
131
166
  const DeleteResourcesProvider = props => {
@@ -175,6 +210,37 @@ const DeleteResourcesProvider = props => {
175
210
  uploadFileResponse
176
211
  });
177
212
  };
213
+ const setJobId = jobId => dispatch({
214
+ type: 'setJobId',
215
+ jobId
216
+ });
217
+ const setTotalResourceCount = totalResourceCount => dispatch({
218
+ type: 'setTotalResourceCount',
219
+ totalResourceCount
220
+ });
221
+ const setValidationProgress = _ref => {
222
+ let processed = _ref.processed,
223
+ isValidating = _ref.isValidating;
224
+ return dispatch({
225
+ type: 'setValidationProgress',
226
+ processed,
227
+ isValidating
228
+ });
229
+ };
230
+ const actions = {
231
+ setCurrentStep,
232
+ cancelImport,
233
+ uploadNewFile,
234
+ setDroppedFile,
235
+ setUploadFileResponse,
236
+ setFileUploadErrors,
237
+ setContainerKey,
238
+ setAbortController,
239
+ setProgress,
240
+ setJobId,
241
+ setTotalResourceCount,
242
+ setValidationProgress
243
+ };
178
244
  const handleClose = function () {
179
245
  let options = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
180
246
  if (state.abortController) state.abortController.abort();
@@ -190,17 +256,7 @@ const DeleteResourcesProvider = props => {
190
256
  resourceType: props.resourceType,
191
257
  state,
192
258
  onClose: handleClose,
193
- actions: {
194
- setCurrentStep,
195
- cancelImport,
196
- uploadNewFile,
197
- setDroppedFile,
198
- setUploadFileResponse,
199
- setFileUploadErrors,
200
- setContainerKey,
201
- setAbortController,
202
- setProgress
203
- }
259
+ actions
204
260
  },
205
261
  children: props.children
206
262
  });
@@ -252,7 +308,7 @@ function getDropAreaState(flags) {
252
308
  return 'ready-for-drop';
253
309
  }
254
310
 
255
- var messages$4 = defineMessages({
311
+ var messages$5 = defineMessages({
256
312
  dragAndDropCSV: {
257
313
  id: 'DeleteResourcesModal.dragAndDropCSV',
258
314
  description: 'The message telling the user to drag and drop CSV file in the file drop area',
@@ -345,7 +401,7 @@ const useDeleteResourcesContext = () => {
345
401
  return context;
346
402
  };
347
403
 
348
- var messages$3 = defineMessages({
404
+ var messages$4 = defineMessages({
349
405
  fileSizeExceededTitle: {
350
406
  id: 'DeleteResourcesModal.sizeExceededTitle',
351
407
  defaultMessage: 'File size exceeded'
@@ -401,6 +457,22 @@ var messages$3 = defineMessages({
401
457
  }
402
458
  });
403
459
 
460
+ function ownKeys$6(e, r) { var t = _Object$keys(e); if (_Object$getOwnPropertySymbols) { var o = _Object$getOwnPropertySymbols(e); r && (o = _filterInstanceProperty(o).call(o, function (r) { return _Object$getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
461
+ function _objectSpread$6(e) { for (var r = 1; r < arguments.length; r++) { var _context, _context2; var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? _forEachInstanceProperty(_context = ownKeys$6(Object(t), !0)).call(_context, function (r) { _defineProperty(e, r, t[r]); }) : _Object$getOwnPropertyDescriptors ? _Object$defineProperties(e, _Object$getOwnPropertyDescriptors(t)) : _forEachInstanceProperty(_context2 = ownKeys$6(Object(t))).call(_context2, function (r) { _Object$defineProperty(e, r, _Object$getOwnPropertyDescriptor(t, r)); }); } return e; }
462
+ const FILE_IMPORT_JOB_FLOW = 'fileImportJobFlow';
463
+ const DEFAULT_SHORT_LIVED_FLAGS = {
464
+ [FILE_IMPORT_JOB_FLOW]: false
465
+ };
466
+ const DEFAULT_LONG_LIVED_FLAGS = {};
467
+ _objectSpread$6(_objectSpread$6({}, DEFAULT_SHORT_LIVED_FLAGS), DEFAULT_LONG_LIVED_FLAGS);
468
+
469
+ const DOCUMENTATION_LINKS = {
470
+ product: 'https://docs.commercetools.com/merchant-center/product-list#bulk-delete-via-csv-import'
471
+ };
472
+ const TEMPLATE_DOWNLOAD_LINKS = {
473
+ product: 'https://docs.commercetools.com/merchant-center/downloads/product_delete_import_template.csv'
474
+ };
475
+
404
476
  const useUpload = () => {
405
477
  const intl = useIntl();
406
478
  const _useApplicationContex = useApplicationContext(context => ({
@@ -412,38 +484,37 @@ const useUpload = () => {
412
484
  state = _useDeleteResourcesCo.state,
413
485
  actions = _useDeleteResourcesCo.actions;
414
486
  const showNotification = useShowNotification();
415
- const _useImportContainerUp = useImportContainerUpload({
416
- projectKey: projectKey
487
+ const isFileImportJobFlowEnabled = useFeatureToggle(FILE_IMPORT_JOB_FLOW);
488
+ const _useFileUpload = useFileUpload({
489
+ projectKey: projectKey,
490
+ useJobBasedFlow: isFileImportJobFlowEnabled
417
491
  }),
418
- upload = _useImportContainerUp.upload;
419
- React.useEffect(() => {
420
- return () => {
421
- if (state.abortController) state.abortController.abort();
422
- };
423
- }, [state.abortController]);
492
+ upload = _useFileUpload.upload;
424
493
  const isFileValid = async file => {
425
494
  const errors = [];
495
+ const maxFileSizeMB = isFileImportJobFlowEnabled ? IMPORT_MAX_FILE_SIZE_MB : IMPORT_LEGACY_MAX_FILE_SIZE_MB;
496
+ const maxItemCount = isFileImportJobFlowEnabled ? IMPORT_MAX_ITEM_COUNT : IMPORT_LEGACY_MAX_ROW_COUNT;
426
497
  const rowCount = await getRowCount(file);
427
498
  if (rowCount < 1) errors.push({
428
- title: intl.formatMessage(messages$3.dropAreaNotEnoughRowsTitle),
429
- description: intl.formatMessage(messages$3.dropAreaNotEnoughRowsDescription)
499
+ title: intl.formatMessage(messages$4.dropAreaNotEnoughRowsTitle),
500
+ description: intl.formatMessage(messages$4.dropAreaNotEnoughRowsDescription)
430
501
  });
431
- if (file.size > toBytes(MAX_FILE_SIZE_MB)) errors.push({
432
- title: intl.formatMessage(messages$3.fileSizeExceededTitle),
433
- description: intl.formatMessage(messages$3.fileSizeExceededDescription, {
434
- fileSizeLimit: intl.formatNumber(MAX_FILE_SIZE_MB)
502
+ if (file.size > toBytes(maxFileSizeMB)) errors.push({
503
+ title: intl.formatMessage(messages$4.fileSizeExceededTitle),
504
+ description: intl.formatMessage(messages$4.fileSizeExceededDescription, {
505
+ fileSizeLimit: intl.formatNumber(maxFileSizeMB)
435
506
  })
436
507
  });
437
508
  if (!(await hasSingleKeyColumn(file))) {
438
509
  errors.push({
439
- title: intl.formatMessage(messages$3.csvFormatErrorTitle),
440
- description: intl.formatMessage(messages$3.csvFormatErrorDescription)
510
+ title: intl.formatMessage(messages$4.csvFormatErrorTitle),
511
+ description: intl.formatMessage(messages$4.csvFormatErrorDescription)
441
512
  });
442
513
  }
443
- if (rowCount > MAX_ROW_COUNT) errors.push({
444
- title: intl.formatMessage(messages$3.rowLimitExceededTitle),
445
- description: intl.formatMessage(messages$3.rowLimitExceededDescription, {
446
- rowLimit: intl.formatNumber(MAX_ROW_COUNT)
514
+ if (rowCount > maxItemCount) errors.push({
515
+ title: intl.formatMessage(messages$4.rowLimitExceededTitle),
516
+ description: intl.formatMessage(messages$4.rowLimitExceededDescription, {
517
+ rowLimit: intl.formatNumber(maxItemCount)
447
518
  })
448
519
  });
449
520
  if (errors.length > 0) {
@@ -451,6 +522,10 @@ const useUpload = () => {
451
522
  actions.setCurrentStep('upload-error');
452
523
  return false;
453
524
  }
525
+ if (isFileImportJobFlowEnabled) {
526
+ const resourceCount = await countUniqueResourcesInCsv(file);
527
+ actions.setTotalResourceCount(resourceCount);
528
+ }
454
529
  return true;
455
530
  };
456
531
  const handleUploadError = error => {
@@ -459,18 +534,25 @@ const useUpload = () => {
459
534
  return;
460
535
  }
461
536
  if (error instanceof HttpError) {
537
+ actions.setValidationProgress({
538
+ processed: 0,
539
+ isValidating: false
540
+ });
462
541
  if (error.errorData?.code === 'MISSING_KEY_ERROR') {
463
542
  actions.setFileUploadErrors([{
464
- title: intl.formatMessage(messages$3.missingRequiredField),
465
- description: intl.formatMessage(messages$3.missingKeyError)
543
+ title: intl.formatMessage(messages$4.missingRequiredField),
544
+ description: intl.formatMessage(messages$4.missingKeyError)
466
545
  }]);
467
546
  actions.setCurrentStep('upload-error');
547
+ } else if (error.errorData?.invalid > 0) {
548
+ actions.setUploadFileResponse(error.errorData);
549
+ actions.setCurrentStep('upload-error');
468
550
  } else {
469
551
  actions.cancelImport();
470
552
  showNotification({
471
553
  kind: 'error',
472
554
  domain: DOMAINS.PAGE,
473
- text: intl.formatMessage(messages$3.unexpectedError)
555
+ text: intl.formatMessage(messages$4.unexpectedError)
474
556
  });
475
557
  }
476
558
  } else {
@@ -492,25 +574,51 @@ const useUpload = () => {
492
574
  if (!state.droppedFile) return;
493
575
  const canUpload = await isFileValid(state.droppedFile);
494
576
  if (!canUpload) return;
577
+ const abortController = new AbortController();
578
+ actions.setAbortController(abortController);
495
579
  actions.setCurrentStep('uploading');
496
- try {
497
- const xhr = await upload({
498
- file: state.droppedFile,
499
- resourceType,
500
- onSuccess: (fileUploadResponse, importContainerKey) => {
501
- actions.setContainerKey(importContainerKey);
502
- actions.setUploadFileResponse(fileUploadResponse);
503
- actions.setCurrentStep('upload-preview');
504
- },
505
- onProgress: progress => {
506
- actions.setProgress(progress);
507
- },
508
- onError: handleUploadError
509
- });
510
- actions.setAbortController(xhr);
511
- } catch (error) {
512
- handleUploadError(error);
513
- }
580
+ await upload({
581
+ file: state.droppedFile,
582
+ resourceType,
583
+ settings: {
584
+ format: 'CSV'
585
+ },
586
+ abortSignal: abortController.signal,
587
+ onSuccess: result => {
588
+ actions.setValidationProgress({
589
+ processed: 0,
590
+ isValidating: false
591
+ });
592
+ actions.setContainerKey(result.containerKey);
593
+ if (result.jobId) {
594
+ actions.setJobId(result.jobId);
595
+ }
596
+ const fileUploadResponse = {
597
+ results: result.summary.results,
598
+ invalid: result.summary.invalid,
599
+ valid: result.summary.valid,
600
+ fileName: state.droppedFile?.name || '',
601
+ itemsCount: result.summary.total,
602
+ rowsCount: result.summary.total,
603
+ columnsCount: result.summary.fieldsCount,
604
+ fields: result.summary.fields,
605
+ ignoredFields: result.summary.ignoredFields
606
+ };
607
+ actions.setUploadFileResponse(fileUploadResponse);
608
+ actions.setCurrentStep('upload-preview');
609
+ },
610
+ onProgress: progress => {
611
+ actions.setProgress(progress);
612
+ },
613
+ onValidationProgress: job => {
614
+ const processed = job.summary?.total ?? 0;
615
+ actions.setValidationProgress({
616
+ processed,
617
+ isValidating: true
618
+ });
619
+ },
620
+ onError: handleUploadError
621
+ });
514
622
  };
515
623
  return {
516
624
  handleUploadAndValidation
@@ -531,9 +639,9 @@ const FileDropArea = () => {
531
639
  }, [actions]);
532
640
  const handleDropRejected = React.useCallback(errorType => {
533
641
  const errorMessageMap = {
534
- 'too-many-files': messages$4.tooManyFilesError,
535
- 'invalid-type': messages$4.fileFormatNotSupported,
536
- generic: messages$4.genericError
642
+ 'too-many-files': messages$5.tooManyFilesError,
643
+ 'invalid-type': messages$5.fileFormatNotSupported,
644
+ generic: messages$5.genericError
537
645
  };
538
646
  showNotification({
539
647
  kind: 'error',
@@ -557,10 +665,10 @@ const FileDropArea = () => {
557
665
  isFileDropped: Boolean(state.droppedFile),
558
666
  fileName: state.droppedFile?.name,
559
667
  messages: {
560
- dragAndDropCSV: intl.formatMessage(messages$4.dragAndDropCSV),
561
- or: intl.formatMessage(messages$4.or),
562
- browseButton: intl.formatMessage(messages$4.browseButton),
563
- chooseFile: intl.formatMessage(messages$4.chooseFile)
668
+ dragAndDropCSV: intl.formatMessage(messages$5.dragAndDropCSV),
669
+ or: intl.formatMessage(messages$5.or),
670
+ browseButton: intl.formatMessage(messages$5.browseButton),
671
+ chooseFile: intl.formatMessage(messages$5.chooseFile)
564
672
  }
565
673
  })
566
674
  });
@@ -709,7 +817,7 @@ var sharedMessages = defineMessages({
709
817
  }
710
818
  });
711
819
 
712
- var messages$2 = defineMessages({
820
+ var messages$3 = defineMessages({
713
821
  fileUploadInstruction: {
714
822
  id: 'DeleteResourcesModal.fileUploadInstruction',
715
823
  description: 'Instructions for uploading a CSV file with resource keys to delete',
@@ -727,13 +835,6 @@ var messages$2 = defineMessages({
727
835
  }
728
836
  });
729
837
 
730
- const DOCUMENTATION_LINKS = {
731
- product: 'https://docs.commercetools.com/merchant-center/product-list#bulk-delete-via-csv-import'
732
- };
733
- const TEMPLATE_DOWNLOAD_LINKS = {
734
- product: 'https://docs.commercetools.com/merchant-center/downloads/product_delete_import_template.csv'
735
- };
736
-
737
838
  function resourceTypeToMessage(resourceType) {
738
839
  const resourceTypeMap = {
739
840
  category: sharedMessages.category,
@@ -791,23 +892,26 @@ const Instructions = () => {
791
892
  const intl = useIntl();
792
893
  const _useDeleteResourcesCo = useDeleteResourcesContext(),
793
894
  resourceType = _useDeleteResourcesCo.resourceType;
895
+ const isFileImportJobFlowEnabled = useFeatureToggle(FILE_IMPORT_JOB_FLOW);
794
896
  const documentTemplateLink = TEMPLATE_DOWNLOAD_LINKS[resourceType];
795
897
  const documentationLink = DOCUMENTATION_LINKS[resourceType];
898
+ const maxFileSizeMB = isFileImportJobFlowEnabled ? IMPORT_MAX_FILE_SIZE_MB : IMPORT_LEGACY_MAX_FILE_SIZE_MB;
899
+ const maxItemCount = isFileImportJobFlowEnabled ? IMPORT_MAX_ITEM_COUNT : IMPORT_LEGACY_MAX_ROW_COUNT;
796
900
  return jsxs(Spacings.Stack, {
797
901
  scale: "m",
798
902
  children: [jsxs(Text.Body, {
799
- children: [intl.formatMessage(messages$2.fileUploadInstruction, {
903
+ children: [intl.formatMessage(messages$3.fileUploadInstruction, {
800
904
  resourceType: intl.formatMessage(resourceTypeToMessage(resourceType))
801
- }), ' ', intl.formatMessage(messages$2.unpublishNote, {
905
+ }), ' ', intl.formatMessage(messages$3.unpublishNote, {
802
906
  b: msg => jsx("b", {
803
907
  children: msg
804
908
  }, "bold")
805
909
  })]
806
910
  }), jsx(Text.Body, {
807
- intlMessage: _objectSpread$5(_objectSpread$5({}, messages$2.fileUploadInstructionForMaxFileSize), {}, {
911
+ intlMessage: _objectSpread$5(_objectSpread$5({}, messages$3.fileUploadInstructionForMaxFileSize), {}, {
808
912
  values: {
809
- maxFileSize: intl.formatNumber(MAX_FILE_SIZE_MB),
810
- maxRowCount: intl.formatNumber(MAX_ROW_COUNT),
913
+ maxFileSize: intl.formatNumber(maxFileSizeMB),
914
+ maxRowCount: intl.formatNumber(maxItemCount),
811
915
  csvTemplateLink: msg => jsx(Link, {
812
916
  tone: "secondary",
813
917
  isExternal: true,
@@ -837,7 +941,7 @@ const Upload = () => {
837
941
  size: 16,
838
942
  isOpen: true,
839
943
  title: intl.formatMessage(sharedMessages.deleteModalTitle),
840
- labelPrimary: intl.formatMessage(messages$4.uploadAndPreview),
944
+ labelPrimary: intl.formatMessage(messages$5.uploadAndPreview),
841
945
  onPrimaryButtonClick: handleUploadAndValidation,
842
946
  isPrimaryButtonDisabled: !Boolean(state.droppedFile),
843
947
  onSecondaryButtonClick: () => onClose({
@@ -855,19 +959,34 @@ const Upload = () => {
855
959
  });
856
960
  };
857
961
 
962
+ var messages$2 = defineMessages({
963
+ validatingResources: {
964
+ id: 'DeleteResourcesModal.Uploading.validatingResources',
965
+ defaultMessage: 'Validating {processed} / {total} {resourceType}'
966
+ }
967
+ });
968
+
858
969
  const Uploading = () => {
859
970
  const intl = useIntl();
860
971
  const _useDeleteResourcesCo = useDeleteResourcesContext(),
972
+ resourceType = _useDeleteResourcesCo.resourceType,
861
973
  state = _useDeleteResourcesCo.state,
862
974
  actions = _useDeleteResourcesCo.actions,
863
975
  onClose = _useDeleteResourcesCo.onClose;
864
976
  if (!state.droppedFile?.name) return null;
977
+ const showValidationProgress = state.isValidating && state.totalResourceCount && (state.validationProcessed ?? 0) > 0;
978
+ const statusMessage = showValidationProgress ? intl.formatMessage(messages$2.validatingResources, {
979
+ processed: intl.formatNumber(state.validationProcessed ?? 0),
980
+ total: intl.formatNumber(state.totalResourceCount ?? 0),
981
+ resourceType: intl.formatMessage(resourceTypeToPluralMessage(resourceType))
982
+ }) : undefined;
865
983
  return jsx(UploadingModal, {
866
984
  isOpen: true,
867
985
  title: intl.formatMessage(sharedMessages.deleteModalTitle),
868
986
  fileName: state.droppedFile.name,
869
987
  fileSize: state.droppedFile.size,
870
988
  progress: state.progress,
989
+ statusMessage: statusMessage,
871
990
  cancelLabel: intl.formatMessage(sharedMessages.cancel),
872
991
  onCancel: () => {
873
992
  actions.setCurrentStep('upload');
@@ -960,7 +1079,7 @@ const UploadErrorsModal = () => {
960
1079
  state = _useDeleteResourcesCo.state,
961
1080
  actions = _useDeleteResourcesCo.actions,
962
1081
  onClose = _useDeleteResourcesCo.onClose;
963
- const errorsCount = state.fileUploadErrors.length;
1082
+ const errorsCount = state.fileUploadErrors?.length ?? 0;
964
1083
  return jsx(InfoDialog, {
965
1084
  size: 16,
966
1085
  isOpen: true,
@@ -980,7 +1099,7 @@ const UploadErrorsModal = () => {
980
1099
  })
981
1100
  })
982
1101
  }), jsx(UploadErrorTable, {
983
- rows: mapFileUploadErrorsToUploadFileErrorRows(state.fileUploadErrors)
1102
+ rows: mapFileUploadErrorsToUploadFileErrorRows(state.fileUploadErrors ?? [])
984
1103
  }), jsxs(Spacings.Inline, {
985
1104
  scale: "m",
986
1105
  justifyContent: "flex-end",
@@ -1061,7 +1180,7 @@ var _ref2 = process.env.NODE_ENV === "production" ? {
1061
1180
  styles: "width:360px"
1062
1181
  } : {
1063
1182
  name: "g36yzl-ImportRunningTextNotification",
1064
- styles: "width:360px;label:ImportRunningTextNotification;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["delete-resources-confirmation-modal.tsx"],"names":[],"mappings":"AAiCY","file":"delete-resources-confirmation-modal.tsx","sourcesContent":["import { useIntl } from 'react-intl'\nimport { useShowNotification } from '@commercetools-frontend/actions-global'\nimport { InfoDialog } from '@commercetools-frontend/application-components'\nimport { useApplicationContext } from '@commercetools-frontend/application-shell-connectors'\nimport { DOMAINS } from '@commercetools-frontend/constants'\nimport { Masking as FullStoryMasking } from '@commercetools-frontend/fullstory'\nimport {\n  Link,\n  PrimaryButton,\n  SecondaryButton,\n  Spacings,\n  Text,\n  WarningIcon,\n} from '@commercetools-frontend/ui-kit'\nimport { processUploadedFile } from '@commercetools-frontend-extensions/operations'\nimport { css } from '@emotion/react'\nimport messages from './messages'\nimport {\n  resourceTypeToMessage,\n  resourceTypeToPluralMessage,\n} from '../../@helpers'\nimport { useDeleteResourcesContext } from '../../@hooks'\nimport sharedMessages from '../../messages'\n\nconst getNewLine = () => <br key=\"break\" />\nconst getImportLogsLink = (msg, projectKey) => (\n  <Link to={`/${projectKey}/operations/import/logs`} key=\"link\">\n    {msg}\n  </Link>\n)\n\nconst ImportRunningTextNotification = (props: { projectKey: string }) => (\n  <div\n    css={css`\n      width: 360px;\n    `}\n  >\n    <Text.Body\n      intlMessage={{\n        ...messages.importStarted,\n        values: {\n          newline: getNewLine,\n          logsLink: (msg) => getImportLogsLink(msg, props.projectKey),\n          b: getBold,\n        },\n      }}\n    />\n  </div>\n)\n\nconst ImportPreparingTextNotification = (props: { projectKey: string }) => (\n  <div\n    css={css`\n      width: 360px;\n    `}\n  >\n    <Text.Body\n      intlMessage={{\n        ...messages.preparingImport,\n        values: {\n          newline: getNewLine,\n          logsLink: (msg) => getImportLogsLink(msg, props.projectKey),\n          b: getBold,\n        },\n      }}\n    />\n  </div>\n)\n\nexport const DeleteResourcesConfirmationModal = () => {\n  const intl = useIntl()\n  const { resourceType, state, onClose } = useDeleteResourcesContext()\n  const { projectKey, projectName } = useApplicationContext((context) => ({\n    projectKey: context.project?.key,\n    projectName: context.project?.name,\n  }))\n\n  const showNotification = useShowNotification()\n\n  const onStartImport = async () => {\n    try {\n      const canConfirm =\n        projectKey && state.containerKey && state.droppedFile?.name\n      if (!canConfirm) return\n      onClose()\n      showNotification(\n        {\n          kind: 'info',\n          domain: DOMAINS.SIDE,\n          // @ts-ignore\n          text: <ImportPreparingTextNotification projectKey={projectKey} />,\n        },\n        {\n          dismissAfter: 5000,\n        }\n      )\n      await processUploadedFile({\n        projectKey,\n        importContainerKey: state.containerKey!,\n        resourceType,\n        action: 'delete',\n      })\n      showNotification(\n        {\n          kind: 'info',\n          domain: DOMAINS.SIDE,\n          // @ts-ignore\n          text: <ImportRunningTextNotification projectKey={projectKey} />,\n        },\n        {\n          dismissAfter: 5000,\n        }\n      )\n    } catch (error) {\n      onClose({ shouldDeleteImportContainer: true })\n      showNotification({\n        kind: 'error',\n        domain: DOMAINS.PAGE,\n        text: String(error),\n      })\n    }\n  }\n\n  return (\n    <InfoDialog\n      title={intl.formatMessage(sharedMessages.deleteModalTitle)}\n      isOpen\n      onClose={() => onClose({ shouldDeleteImportContainer: true })}\n      size={16}\n    >\n      <FullStoryMasking.Unmask>\n        <Spacings.Stack scale=\"xl\">\n          <Spacings.Stack scale=\"m\">\n            <div>\n              <Text.Body\n                intlMessage={{\n                  ...messages.confirmationText,\n                  values: {\n                    projectName: projectName,\n                    resourceTypePlural: intl.formatMessage(\n                      resourceTypeToPluralMessage(resourceType)\n                    ),\n                    resourceType: intl.formatMessage(\n                      resourceTypeToMessage(resourceType)\n                    ),\n                    count: intl.formatNumber(\n                      state.uploadFileResponse?.rowsCount\n                    ),\n                    b: getBold,\n                  },\n                }}\n              />\n              <Text.Body\n                intlMessage={{\n                  ...messages.confirmationQuestion,\n                  values: {\n                    resourceTypePlural: intl.formatMessage(\n                      resourceTypeToPluralMessage(resourceType)\n                    ),\n                  },\n                }}\n              />\n            </div>\n            <Spacings.Inline scale=\"s\" alignItems=\"center\">\n              <WarningIcon size=\"40\" color=\"warning\" />\n              <Text.Body intlMessage={messages.cannotBeUndoneNote} />\n            </Spacings.Inline>\n          </Spacings.Stack>\n          <Spacings.Inline scale=\"m\" justifyContent=\"flex-end\">\n            <SecondaryButton\n              data-test-id=\"confirmation-dialog-cancel\"\n              label={intl.formatMessage(sharedMessages.cancel)}\n              onClick={() => onClose({ shouldDeleteImportContainer: true })}\n            />\n            <PrimaryButton\n              tone=\"critical\"\n              data-test-id=\"confirmation-dialog-confirm\"\n              label={intl.formatMessage(sharedMessages.startBulkDelete)}\n              onClick={onStartImport}\n            />\n          </Spacings.Inline>\n        </Spacings.Stack>\n      </FullStoryMasking.Unmask>\n    </InfoDialog>\n  )\n}\n\nfunction getBold(msg) {\n  return (\n    <Text.Body as=\"span\" isBold key={crypto.randomUUID()}>\n      {msg}\n    </Text.Body>\n  )\n}\n"]} */",
1183
+ styles: "width:360px;label:ImportRunningTextNotification;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["delete-resources-confirmation-modal.tsx"],"names":[],"mappings":"AAoCY","file":"delete-resources-confirmation-modal.tsx","sourcesContent":["import { useIntl } from 'react-intl'\nimport { useShowNotification } from '@commercetools-frontend/actions-global'\nimport { InfoDialog } from '@commercetools-frontend/application-components'\nimport { useApplicationContext } from '@commercetools-frontend/application-shell-connectors'\nimport { DOMAINS } from '@commercetools-frontend/constants'\nimport { Masking as FullStoryMasking } from '@commercetools-frontend/fullstory'\nimport {\n  Link,\n  PrimaryButton,\n  SecondaryButton,\n  Spacings,\n  Text,\n  WarningIcon,\n} from '@commercetools-frontend/ui-kit'\nimport {\n  processUploadedFile,\n  processFileImportJob,\n} from '@commercetools-frontend-extensions/operations'\nimport { css } from '@emotion/react'\nimport messages from './messages'\nimport {\n  resourceTypeToMessage,\n  resourceTypeToPluralMessage,\n} from '../../@helpers'\nimport { useDeleteResourcesContext } from '../../@hooks'\nimport sharedMessages from '../../messages'\n\nconst getNewLine = () => <br key=\"break\" />\nconst getImportLogsLink = (msg, projectKey) => (\n  <Link to={`/${projectKey}/operations/import/logs`} key=\"link\">\n    {msg}\n  </Link>\n)\n\nconst ImportRunningTextNotification = (props: { projectKey: string }) => (\n  <div\n    css={css`\n      width: 360px;\n    `}\n  >\n    <Text.Body\n      intlMessage={{\n        ...messages.importStarted,\n        values: {\n          newline: getNewLine,\n          logsLink: (msg) => getImportLogsLink(msg, props.projectKey),\n          b: getBold,\n        },\n      }}\n    />\n  </div>\n)\n\nconst ImportPreparingTextNotification = (props: { projectKey: string }) => (\n  <div\n    css={css`\n      width: 360px;\n    `}\n  >\n    <Text.Body\n      intlMessage={{\n        ...messages.preparingImport,\n        values: {\n          newline: getNewLine,\n          logsLink: (msg) => getImportLogsLink(msg, props.projectKey),\n          b: getBold,\n        },\n      }}\n    />\n  </div>\n)\n\nexport const DeleteResourcesConfirmationModal = () => {\n  const intl = useIntl()\n  const { resourceType, state, onClose } = useDeleteResourcesContext()\n  const { projectKey, projectName } = useApplicationContext((context) => ({\n    projectKey: context.project?.key,\n    projectName: context.project?.name,\n  }))\n\n  const showNotification = useShowNotification()\n\n  const onStartImport = async () => {\n    try {\n      const canConfirm =\n        projectKey && state.containerKey && state.droppedFile?.name\n      if (!canConfirm) return\n      onClose()\n      showNotification(\n        {\n          kind: 'info',\n          domain: DOMAINS.SIDE,\n          // @ts-ignore\n          text: <ImportPreparingTextNotification projectKey={projectKey} />,\n        },\n        {\n          dismissAfter: 5000,\n        }\n      )\n\n      if (state.jobId) {\n        await processFileImportJob({\n          projectKey,\n          importContainerKey: state.containerKey!,\n          jobId: state.jobId,\n          resourceType,\n          action: 'delete',\n        })\n      } else {\n        await processUploadedFile({\n          projectKey,\n          importContainerKey: state.containerKey!,\n          resourceType,\n          action: 'delete',\n        })\n      }\n      showNotification(\n        {\n          kind: 'info',\n          domain: DOMAINS.SIDE,\n          // @ts-ignore\n          text: <ImportRunningTextNotification projectKey={projectKey} />,\n        },\n        {\n          dismissAfter: 5000,\n        }\n      )\n    } catch (error) {\n      onClose({ shouldDeleteImportContainer: true })\n      showNotification({\n        kind: 'error',\n        domain: DOMAINS.PAGE,\n        text: String(error),\n      })\n    }\n  }\n\n  return (\n    <InfoDialog\n      title={intl.formatMessage(sharedMessages.deleteModalTitle)}\n      isOpen\n      onClose={() => onClose({ shouldDeleteImportContainer: true })}\n      size={16}\n    >\n      <FullStoryMasking.Unmask>\n        <Spacings.Stack scale=\"xl\">\n          <Spacings.Stack scale=\"m\">\n            <div>\n              <Text.Body\n                intlMessage={{\n                  ...messages.confirmationText,\n                  values: {\n                    projectName: projectName,\n                    resourceTypePlural: intl.formatMessage(\n                      resourceTypeToPluralMessage(resourceType)\n                    ),\n                    resourceType: intl.formatMessage(\n                      resourceTypeToMessage(resourceType)\n                    ),\n                    count: intl.formatNumber(\n                      state.uploadFileResponse?.rowsCount ?? 0\n                    ),\n                    b: getBold,\n                  },\n                }}\n              />\n              <Text.Body\n                intlMessage={{\n                  ...messages.confirmationQuestion,\n                  values: {\n                    resourceTypePlural: intl.formatMessage(\n                      resourceTypeToPluralMessage(resourceType)\n                    ),\n                  },\n                }}\n              />\n            </div>\n            <Spacings.Inline scale=\"s\" alignItems=\"center\">\n              <WarningIcon size=\"40\" color=\"warning\" />\n              <Text.Body intlMessage={messages.cannotBeUndoneNote} />\n            </Spacings.Inline>\n          </Spacings.Stack>\n          <Spacings.Inline scale=\"m\" justifyContent=\"flex-end\">\n            <SecondaryButton\n              data-test-id=\"confirmation-dialog-cancel\"\n              label={intl.formatMessage(sharedMessages.cancel)}\n              onClick={() => onClose({ shouldDeleteImportContainer: true })}\n            />\n            <PrimaryButton\n              tone=\"critical\"\n              data-test-id=\"confirmation-dialog-confirm\"\n              label={intl.formatMessage(sharedMessages.startBulkDelete)}\n              onClick={onStartImport}\n            />\n          </Spacings.Inline>\n        </Spacings.Stack>\n      </FullStoryMasking.Unmask>\n    </InfoDialog>\n  )\n}\n\nfunction getBold(msg) {\n  return (\n    <Text.Body as=\"span\" isBold key={crypto.randomUUID()}>\n      {msg}\n    </Text.Body>\n  )\n}\n"]} */",
1065
1184
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
1066
1185
  };
1067
1186
  const ImportRunningTextNotification = props => jsx("div", {
@@ -1081,7 +1200,7 @@ var _ref = process.env.NODE_ENV === "production" ? {
1081
1200
  styles: "width:360px"
1082
1201
  } : {
1083
1202
  name: "mjwslu-ImportPreparingTextNotification",
1084
- styles: "width:360px;label:ImportPreparingTextNotification;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["delete-resources-confirmation-modal.tsx"],"names":[],"mappings":"AAoDY","file":"delete-resources-confirmation-modal.tsx","sourcesContent":["import { useIntl } from 'react-intl'\nimport { useShowNotification } from '@commercetools-frontend/actions-global'\nimport { InfoDialog } from '@commercetools-frontend/application-components'\nimport { useApplicationContext } from '@commercetools-frontend/application-shell-connectors'\nimport { DOMAINS } from '@commercetools-frontend/constants'\nimport { Masking as FullStoryMasking } from '@commercetools-frontend/fullstory'\nimport {\n  Link,\n  PrimaryButton,\n  SecondaryButton,\n  Spacings,\n  Text,\n  WarningIcon,\n} from '@commercetools-frontend/ui-kit'\nimport { processUploadedFile } from '@commercetools-frontend-extensions/operations'\nimport { css } from '@emotion/react'\nimport messages from './messages'\nimport {\n  resourceTypeToMessage,\n  resourceTypeToPluralMessage,\n} from '../../@helpers'\nimport { useDeleteResourcesContext } from '../../@hooks'\nimport sharedMessages from '../../messages'\n\nconst getNewLine = () => <br key=\"break\" />\nconst getImportLogsLink = (msg, projectKey) => (\n  <Link to={`/${projectKey}/operations/import/logs`} key=\"link\">\n    {msg}\n  </Link>\n)\n\nconst ImportRunningTextNotification = (props: { projectKey: string }) => (\n  <div\n    css={css`\n      width: 360px;\n    `}\n  >\n    <Text.Body\n      intlMessage={{\n        ...messages.importStarted,\n        values: {\n          newline: getNewLine,\n          logsLink: (msg) => getImportLogsLink(msg, props.projectKey),\n          b: getBold,\n        },\n      }}\n    />\n  </div>\n)\n\nconst ImportPreparingTextNotification = (props: { projectKey: string }) => (\n  <div\n    css={css`\n      width: 360px;\n    `}\n  >\n    <Text.Body\n      intlMessage={{\n        ...messages.preparingImport,\n        values: {\n          newline: getNewLine,\n          logsLink: (msg) => getImportLogsLink(msg, props.projectKey),\n          b: getBold,\n        },\n      }}\n    />\n  </div>\n)\n\nexport const DeleteResourcesConfirmationModal = () => {\n  const intl = useIntl()\n  const { resourceType, state, onClose } = useDeleteResourcesContext()\n  const { projectKey, projectName } = useApplicationContext((context) => ({\n    projectKey: context.project?.key,\n    projectName: context.project?.name,\n  }))\n\n  const showNotification = useShowNotification()\n\n  const onStartImport = async () => {\n    try {\n      const canConfirm =\n        projectKey && state.containerKey && state.droppedFile?.name\n      if (!canConfirm) return\n      onClose()\n      showNotification(\n        {\n          kind: 'info',\n          domain: DOMAINS.SIDE,\n          // @ts-ignore\n          text: <ImportPreparingTextNotification projectKey={projectKey} />,\n        },\n        {\n          dismissAfter: 5000,\n        }\n      )\n      await processUploadedFile({\n        projectKey,\n        importContainerKey: state.containerKey!,\n        resourceType,\n        action: 'delete',\n      })\n      showNotification(\n        {\n          kind: 'info',\n          domain: DOMAINS.SIDE,\n          // @ts-ignore\n          text: <ImportRunningTextNotification projectKey={projectKey} />,\n        },\n        {\n          dismissAfter: 5000,\n        }\n      )\n    } catch (error) {\n      onClose({ shouldDeleteImportContainer: true })\n      showNotification({\n        kind: 'error',\n        domain: DOMAINS.PAGE,\n        text: String(error),\n      })\n    }\n  }\n\n  return (\n    <InfoDialog\n      title={intl.formatMessage(sharedMessages.deleteModalTitle)}\n      isOpen\n      onClose={() => onClose({ shouldDeleteImportContainer: true })}\n      size={16}\n    >\n      <FullStoryMasking.Unmask>\n        <Spacings.Stack scale=\"xl\">\n          <Spacings.Stack scale=\"m\">\n            <div>\n              <Text.Body\n                intlMessage={{\n                  ...messages.confirmationText,\n                  values: {\n                    projectName: projectName,\n                    resourceTypePlural: intl.formatMessage(\n                      resourceTypeToPluralMessage(resourceType)\n                    ),\n                    resourceType: intl.formatMessage(\n                      resourceTypeToMessage(resourceType)\n                    ),\n                    count: intl.formatNumber(\n                      state.uploadFileResponse?.rowsCount\n                    ),\n                    b: getBold,\n                  },\n                }}\n              />\n              <Text.Body\n                intlMessage={{\n                  ...messages.confirmationQuestion,\n                  values: {\n                    resourceTypePlural: intl.formatMessage(\n                      resourceTypeToPluralMessage(resourceType)\n                    ),\n                  },\n                }}\n              />\n            </div>\n            <Spacings.Inline scale=\"s\" alignItems=\"center\">\n              <WarningIcon size=\"40\" color=\"warning\" />\n              <Text.Body intlMessage={messages.cannotBeUndoneNote} />\n            </Spacings.Inline>\n          </Spacings.Stack>\n          <Spacings.Inline scale=\"m\" justifyContent=\"flex-end\">\n            <SecondaryButton\n              data-test-id=\"confirmation-dialog-cancel\"\n              label={intl.formatMessage(sharedMessages.cancel)}\n              onClick={() => onClose({ shouldDeleteImportContainer: true })}\n            />\n            <PrimaryButton\n              tone=\"critical\"\n              data-test-id=\"confirmation-dialog-confirm\"\n              label={intl.formatMessage(sharedMessages.startBulkDelete)}\n              onClick={onStartImport}\n            />\n          </Spacings.Inline>\n        </Spacings.Stack>\n      </FullStoryMasking.Unmask>\n    </InfoDialog>\n  )\n}\n\nfunction getBold(msg) {\n  return (\n    <Text.Body as=\"span\" isBold key={crypto.randomUUID()}>\n      {msg}\n    </Text.Body>\n  )\n}\n"]} */",
1203
+ styles: "width:360px;label:ImportPreparingTextNotification;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["delete-resources-confirmation-modal.tsx"],"names":[],"mappings":"AAuDY","file":"delete-resources-confirmation-modal.tsx","sourcesContent":["import { useIntl } from 'react-intl'\nimport { useShowNotification } from '@commercetools-frontend/actions-global'\nimport { InfoDialog } from '@commercetools-frontend/application-components'\nimport { useApplicationContext } from '@commercetools-frontend/application-shell-connectors'\nimport { DOMAINS } from '@commercetools-frontend/constants'\nimport { Masking as FullStoryMasking } from '@commercetools-frontend/fullstory'\nimport {\n  Link,\n  PrimaryButton,\n  SecondaryButton,\n  Spacings,\n  Text,\n  WarningIcon,\n} from '@commercetools-frontend/ui-kit'\nimport {\n  processUploadedFile,\n  processFileImportJob,\n} from '@commercetools-frontend-extensions/operations'\nimport { css } from '@emotion/react'\nimport messages from './messages'\nimport {\n  resourceTypeToMessage,\n  resourceTypeToPluralMessage,\n} from '../../@helpers'\nimport { useDeleteResourcesContext } from '../../@hooks'\nimport sharedMessages from '../../messages'\n\nconst getNewLine = () => <br key=\"break\" />\nconst getImportLogsLink = (msg, projectKey) => (\n  <Link to={`/${projectKey}/operations/import/logs`} key=\"link\">\n    {msg}\n  </Link>\n)\n\nconst ImportRunningTextNotification = (props: { projectKey: string }) => (\n  <div\n    css={css`\n      width: 360px;\n    `}\n  >\n    <Text.Body\n      intlMessage={{\n        ...messages.importStarted,\n        values: {\n          newline: getNewLine,\n          logsLink: (msg) => getImportLogsLink(msg, props.projectKey),\n          b: getBold,\n        },\n      }}\n    />\n  </div>\n)\n\nconst ImportPreparingTextNotification = (props: { projectKey: string }) => (\n  <div\n    css={css`\n      width: 360px;\n    `}\n  >\n    <Text.Body\n      intlMessage={{\n        ...messages.preparingImport,\n        values: {\n          newline: getNewLine,\n          logsLink: (msg) => getImportLogsLink(msg, props.projectKey),\n          b: getBold,\n        },\n      }}\n    />\n  </div>\n)\n\nexport const DeleteResourcesConfirmationModal = () => {\n  const intl = useIntl()\n  const { resourceType, state, onClose } = useDeleteResourcesContext()\n  const { projectKey, projectName } = useApplicationContext((context) => ({\n    projectKey: context.project?.key,\n    projectName: context.project?.name,\n  }))\n\n  const showNotification = useShowNotification()\n\n  const onStartImport = async () => {\n    try {\n      const canConfirm =\n        projectKey && state.containerKey && state.droppedFile?.name\n      if (!canConfirm) return\n      onClose()\n      showNotification(\n        {\n          kind: 'info',\n          domain: DOMAINS.SIDE,\n          // @ts-ignore\n          text: <ImportPreparingTextNotification projectKey={projectKey} />,\n        },\n        {\n          dismissAfter: 5000,\n        }\n      )\n\n      if (state.jobId) {\n        await processFileImportJob({\n          projectKey,\n          importContainerKey: state.containerKey!,\n          jobId: state.jobId,\n          resourceType,\n          action: 'delete',\n        })\n      } else {\n        await processUploadedFile({\n          projectKey,\n          importContainerKey: state.containerKey!,\n          resourceType,\n          action: 'delete',\n        })\n      }\n      showNotification(\n        {\n          kind: 'info',\n          domain: DOMAINS.SIDE,\n          // @ts-ignore\n          text: <ImportRunningTextNotification projectKey={projectKey} />,\n        },\n        {\n          dismissAfter: 5000,\n        }\n      )\n    } catch (error) {\n      onClose({ shouldDeleteImportContainer: true })\n      showNotification({\n        kind: 'error',\n        domain: DOMAINS.PAGE,\n        text: String(error),\n      })\n    }\n  }\n\n  return (\n    <InfoDialog\n      title={intl.formatMessage(sharedMessages.deleteModalTitle)}\n      isOpen\n      onClose={() => onClose({ shouldDeleteImportContainer: true })}\n      size={16}\n    >\n      <FullStoryMasking.Unmask>\n        <Spacings.Stack scale=\"xl\">\n          <Spacings.Stack scale=\"m\">\n            <div>\n              <Text.Body\n                intlMessage={{\n                  ...messages.confirmationText,\n                  values: {\n                    projectName: projectName,\n                    resourceTypePlural: intl.formatMessage(\n                      resourceTypeToPluralMessage(resourceType)\n                    ),\n                    resourceType: intl.formatMessage(\n                      resourceTypeToMessage(resourceType)\n                    ),\n                    count: intl.formatNumber(\n                      state.uploadFileResponse?.rowsCount ?? 0\n                    ),\n                    b: getBold,\n                  },\n                }}\n              />\n              <Text.Body\n                intlMessage={{\n                  ...messages.confirmationQuestion,\n                  values: {\n                    resourceTypePlural: intl.formatMessage(\n                      resourceTypeToPluralMessage(resourceType)\n                    ),\n                  },\n                }}\n              />\n            </div>\n            <Spacings.Inline scale=\"s\" alignItems=\"center\">\n              <WarningIcon size=\"40\" color=\"warning\" />\n              <Text.Body intlMessage={messages.cannotBeUndoneNote} />\n            </Spacings.Inline>\n          </Spacings.Stack>\n          <Spacings.Inline scale=\"m\" justifyContent=\"flex-end\">\n            <SecondaryButton\n              data-test-id=\"confirmation-dialog-cancel\"\n              label={intl.formatMessage(sharedMessages.cancel)}\n              onClick={() => onClose({ shouldDeleteImportContainer: true })}\n            />\n            <PrimaryButton\n              tone=\"critical\"\n              data-test-id=\"confirmation-dialog-confirm\"\n              label={intl.formatMessage(sharedMessages.startBulkDelete)}\n              onClick={onStartImport}\n            />\n          </Spacings.Inline>\n        </Spacings.Stack>\n      </FullStoryMasking.Unmask>\n    </InfoDialog>\n  )\n}\n\nfunction getBold(msg) {\n  return (\n    <Text.Body as=\"span\" isBold key={crypto.randomUUID()}>\n      {msg}\n    </Text.Body>\n  )\n}\n"]} */",
1085
1204
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
1086
1205
  };
1087
1206
  const ImportPreparingTextNotification = props => jsx("div", {
@@ -1124,12 +1243,22 @@ const DeleteResourcesConfirmationModal = () => {
1124
1243
  }, {
1125
1244
  dismissAfter: 5000
1126
1245
  });
1127
- await processUploadedFile({
1128
- projectKey,
1129
- importContainerKey: state.containerKey,
1130
- resourceType,
1131
- action: 'delete'
1132
- });
1246
+ if (state.jobId) {
1247
+ await processFileImportJob({
1248
+ projectKey,
1249
+ importContainerKey: state.containerKey,
1250
+ jobId: state.jobId,
1251
+ resourceType,
1252
+ action: 'delete'
1253
+ });
1254
+ } else {
1255
+ await processUploadedFile({
1256
+ projectKey,
1257
+ importContainerKey: state.containerKey,
1258
+ resourceType,
1259
+ action: 'delete'
1260
+ });
1261
+ }
1133
1262
  showNotification({
1134
1263
  kind: 'info',
1135
1264
  domain: DOMAINS.SIDE,
@@ -1170,7 +1299,7 @@ const DeleteResourcesConfirmationModal = () => {
1170
1299
  projectName: projectName,
1171
1300
  resourceTypePlural: intl.formatMessage(resourceTypeToPluralMessage(resourceType)),
1172
1301
  resourceType: intl.formatMessage(resourceTypeToMessage(resourceType)),
1173
- count: intl.formatNumber(state.uploadFileResponse?.rowsCount),
1302
+ count: intl.formatNumber(state.uploadFileResponse?.rowsCount ?? 0),
1174
1303
  b: getBold$1
1175
1304
  }
1176
1305
  })
@@ -1259,7 +1388,7 @@ function UploadPreviewModal() {
1259
1388
  values: {
1260
1389
  resourceType: intl.formatMessage(resourceTypeToMessage(resourceType)),
1261
1390
  resourceTypePlural: intl.formatMessage(resourceTypeToPluralMessage(resourceType)),
1262
- count: intl.formatNumber(state.uploadFileResponse?.rowsCount),
1391
+ count: intl.formatNumber(state.uploadFileResponse?.rowsCount ?? 0),
1263
1392
  b: getBold
1264
1393
  }
1265
1394
  })