@ukhomeoffice/cop-react-form-renderer 6.0.6-peter → 6.6.1

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 (80) hide show
  1. package/dist/components/CheckYourAnswers/CheckYourAnswers.scss +2 -2
  2. package/dist/components/CollectionPage/CollectionPage.js +8 -2
  3. package/dist/components/CollectionSummary/BannerStrip.js +3 -2
  4. package/dist/components/CollectionSummary/BannerStrip.scss +2 -2
  5. package/dist/components/CollectionSummary/BannerStrip.test.js +39 -4
  6. package/dist/components/CollectionSummary/CollectionSummary.js +82 -63
  7. package/dist/components/CollectionSummary/CollectionSummary.scss +1 -1
  8. package/dist/components/CollectionSummary/CollectionSummary.test.js +40 -80
  9. package/dist/components/CollectionSummary/Confirmation.scss +1 -1
  10. package/dist/components/CollectionSummary/RenderListView.js +23 -19
  11. package/dist/components/CollectionSummary/RenderListView.scss +11 -2
  12. package/dist/components/CollectionSummary/RenderListView.test.js +14 -4
  13. package/dist/components/CollectionSummary/SummaryCard.js +61 -40
  14. package/dist/components/CollectionSummary/SummaryCard.scss +2 -1
  15. package/dist/components/CollectionSummary/SummaryCard.test.js +193 -150
  16. package/dist/components/CollectionSummary/SummaryCardDetails.js +70 -13
  17. package/dist/components/CollectionSummary/SummaryCardDetails.scss +45 -8
  18. package/dist/components/CollectionSummary/SummaryCardDetails.test.js +174 -26
  19. package/dist/components/CollectionSummary/SummaryCardValidationContext.js +15 -5
  20. package/dist/components/CollectionSummary/SummaryCardValidationContext.test.js +5 -4
  21. package/dist/components/FormComponent/Collection.js +24 -17
  22. package/dist/components/FormComponent/Collection.test.js +138 -0
  23. package/dist/components/FormComponent/FormComponent.js +12 -0
  24. package/dist/components/FormPage/FormPage.scss +1 -1
  25. package/dist/components/FormRenderer/FormRenderer.js +7 -4
  26. package/dist/components/FormRenderer/FormRenderer.scss +1 -1
  27. package/dist/components/FormRenderer/helpers/index.js +1 -3
  28. package/dist/components/FormRenderer/onPageAction.js +7 -9
  29. package/dist/components/FormRenderer/onPageAction.test.js +18 -9
  30. package/dist/components/SummaryList/SummaryList.scss +2 -2
  31. package/dist/components/TaskList/TaskList.scss +1 -1
  32. package/dist/context/ValidationContext/ValidationContext.js +49 -5
  33. package/dist/context/ValidationContext/ValidationContext.test.js +16 -7
  34. package/dist/hooks/useRefData.js +1 -1
  35. package/dist/utils/CheckYourAnswers/showComponentCYA.js +1 -2
  36. package/dist/utils/CheckYourAnswers/showComponentCYA.test.js +5 -0
  37. package/dist/utils/CollectionPage/addCollectionPageEntry.js +1 -2
  38. package/dist/utils/CollectionPage/addCollectionPageEntry.test.js +4 -24
  39. package/dist/utils/CollectionPage/duplicateCollectionPageEntry.js +22 -2
  40. package/dist/utils/CollectionPage/duplicateCollectionPageEntry.test.js +39 -4
  41. package/dist/utils/CollectionPage/getErrorsForCollection.js +55 -0
  42. package/dist/utils/CollectionPage/getErrorsForCollection.test.js +155 -0
  43. package/dist/utils/CollectionPage/getQuickEditPage.js +14 -5
  44. package/dist/utils/CollectionPage/getQuickEditPage.test.js +14 -29
  45. package/dist/utils/CollectionPage/index.js +2 -0
  46. package/dist/utils/CollectionPage/mergeCollectionPages.js +0 -1
  47. package/dist/utils/CollectionPage/setCollectionPageData.js +9 -4
  48. package/dist/utils/CollectionPage/setCollectionPageData.test.js +18 -0
  49. package/dist/utils/Component/isEditable.js +1 -1
  50. package/dist/utils/Condition/meetsCondition.js +18 -0
  51. package/dist/utils/Condition/meetsCondition.test.js +100 -0
  52. package/dist/utils/Data/getOptions.js +10 -0
  53. package/dist/utils/Data/getOptions.test.js +73 -0
  54. package/dist/utils/Data/nestInRefdataOptions.js +49 -0
  55. package/dist/utils/Data/nestInRefdataOptions.test.js +236 -0
  56. package/dist/utils/Validate/additional/mustBeUniqueInCollection.js +4 -0
  57. package/dist/utils/Validate/additional/mustBeUniqueInCollection.test.js +36 -0
  58. package/dist/utils/Validate/validateContainer.js +3 -1
  59. package/dist/utils/Validate/validateContainer.test.js +33 -0
  60. package/dist/utils/Validate/validateEmail.js +1 -1
  61. package/dist/utils/Validate/validatePage.js +10 -1
  62. package/dist/utils/Validate/validatePage.test.js +69 -0
  63. package/package.json +4 -4
  64. package/dist/components/FormRenderer/clear-uncompleted-routes/test-data/component-used-in-multiple-pages-data.json +0 -4
  65. package/dist/components/FormRenderer/clear-uncompleted-routes/test-data/component-used-in-multiple-pages-form.json +0 -61
  66. package/dist/components/FormRenderer/clear-uncompleted-routes/test-data/data-with-collection-data-removed.json +0 -4
  67. package/dist/components/FormRenderer/clear-uncompleted-routes/test-data/data-with-collections.json +0 -8
  68. package/dist/components/FormRenderer/clear-uncompleted-routes/test-data/data-with-components-removed.json +0 -3
  69. package/dist/components/FormRenderer/clear-uncompleted-routes/test-data/data-with-components.json +0 -5
  70. package/dist/components/FormRenderer/clear-uncompleted-routes/test-data/data-with-entire-collection-removed.json +0 -3
  71. package/dist/components/FormRenderer/clear-uncompleted-routes/test-data/data-with-nested-component-removed.json +0 -10
  72. package/dist/components/FormRenderer/clear-uncompleted-routes/test-data/data-with-nested-components.json +0 -11
  73. package/dist/components/FormRenderer/clear-uncompleted-routes/test-data/form-for-nested-components.json +0 -96
  74. package/dist/components/FormRenderer/clear-uncompleted-routes/test-data/form-with-collections-delete-entire.json +0 -47
  75. package/dist/components/FormRenderer/clear-uncompleted-routes/test-data/form-with-collections.json +0 -46
  76. package/dist/components/FormRenderer/clear-uncompleted-routes/test-data/form-with-components.json +0 -48
  77. package/dist/components/FormRenderer/helpers/clearOutUncompletedRoutes.js +0 -175
  78. package/dist/components/FormRenderer/helpers/clearOutUncompletedRoutes.test.js +0 -113
  79. package/dist/components/FormRenderer/helpers/deleteNodeByPath.js +0 -20
  80. package/dist/components/FormRenderer/helpers/deleteNodeByPath.test.js +0 -56
@@ -3,34 +3,14 @@
3
3
  var _addCollectionPageEntry = _interopRequireDefault(require("./addCollectionPageEntry"));
4
4
  function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
5
5
  describe('Utils.CollectionPage.addCollectionPageEntry', function () {
6
- it('should correctly add a new entry for a top-level collection', function () {
6
+ it('should set active ID as null for a top-level collection', function () {
7
7
  var FORM_DATA = {};
8
8
  (0, _addCollectionPageEntry.default)('testCollection', FORM_DATA);
9
- expect(FORM_DATA.testCollectionActiveId).toBeTruthy();
9
+ expect(FORM_DATA.testCollectionActiveId).toBeFalsy();
10
10
  });
11
- it('should correctly add a new entry for a nested collection', function () {
11
+ it('should set a active ID as null for a nested collection', function () {
12
12
  var FORM_DATA = {};
13
13
  (0, _addCollectionPageEntry.default)('parent.child.grandchild', FORM_DATA);
14
- expect(FORM_DATA.grandchildActiveId).toBeTruthy();
15
- });
16
- it('should correctly set the active ID for a top-level collection', function () {
17
- var FORM_DATA = {};
18
- (0, _addCollectionPageEntry.default)('testCollection', FORM_DATA);
19
- expect(FORM_DATA.testCollectionActiveId).toBe('0');
20
- });
21
- it('should correctly set the active ID for a nested collection', function () {
22
- var FORM_DATA = {};
23
- (0, _addCollectionPageEntry.default)('parent.child.grandchild', FORM_DATA);
24
- expect(FORM_DATA.grandchildActiveId).toBe('0');
25
- });
26
- it('should correctly increment the numbered index for a top-level collection', function () {
27
- var FORM_DATA = {};
28
- (0, _addCollectionPageEntry.default)('testCollection', FORM_DATA);
29
- expect(FORM_DATA.testCollectionNumberedIndex).toBe(1);
30
- });
31
- it('should correctly increment the numbered index for a nested collection', function () {
32
- var FORM_DATA = {};
33
- (0, _addCollectionPageEntry.default)('parent.child.grandchild', FORM_DATA);
34
- expect(FORM_DATA.grandchildNumberedIndex).toBe(1);
14
+ expect(FORM_DATA.grandchildActiveId).toBeFalsy();
35
15
  });
36
16
  });
@@ -12,8 +12,20 @@ function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t =
12
12
  function _defineProperty(obj, key, value) { key = _toPropertyKey(key); if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
13
13
  function _toPropertyKey(arg) { var key = _toPrimitive(arg, "string"); return _typeof(key) === "symbol" ? key : String(key); }
14
14
  function _toPrimitive(input, hint) { if (_typeof(input) !== "object" || input === null) return input; var prim = input[Symbol.toPrimitive]; if (prim !== undefined) { var res = prim.call(input, hint || "default"); if (_typeof(res) !== "object") return res; throw new TypeError("@@toPrimitive must return a primitive value."); } return (hint === "string" ? String : Number)(input); }
15
+ /**
16
+ * Duplicates an entry with the given ID in the given collection, optionally
17
+ * ignoring named fields or adding extras to the duplicate if they're provided.
18
+ *
19
+ * @param {string} collectionName The name of the collection the entry to duplicate is in.
20
+ * @param {object} formData The top-level formData object.
21
+ * @param {string} entryId The ID of the entry to duplicate.
22
+ * @param {array} fieldsToIgnore An array of field names that shouldn't be copied to the duplicate.
23
+ * @param {object} fieldsToAdd An object that will be spread to the new entry.
24
+ * @returns The ID of the duplicate entry.
25
+ */
15
26
  var duplicateCollectionPageEntry = function duplicateCollectionPageEntry(collectionName, formData, entryId) {
16
27
  var fieldsToIgnore = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : [];
28
+ var fieldsToAdd = arguments.length > 4 && arguments[4] !== undefined ? arguments[4] : {};
17
29
  var collectionData = (0, _getCollectionPageData.default)(collectionName, formData);
18
30
  var entryToDuplicate = collectionData === null || collectionData === void 0 ? void 0 : collectionData.find(function (entry) {
19
31
  return entry.id === entryId;
@@ -22,13 +34,21 @@ var duplicateCollectionPageEntry = function duplicateCollectionPageEntry(collect
22
34
  return null;
23
35
  }
24
36
  var newEntryId = Date.now().toString();
25
- var newEntry = _objectSpread(_objectSpread({}, entryToDuplicate), {}, {
37
+ var newEntry = _objectSpread(_objectSpread(_objectSpread({}, entryToDuplicate), fieldsToAdd), {}, {
26
38
  id: newEntryId
27
39
  });
28
40
  fieldsToIgnore.forEach(function (field) {
29
41
  return delete newEntry[field];
30
42
  });
31
- collectionData.push(newEntry);
43
+ var insertionIndex = collectionData.findIndex(function (entry) {
44
+ return entry.id === entryId;
45
+ }) + 1;
46
+ if (insertionIndex === -1) {
47
+ collectionData.push(newEntry);
48
+ } else {
49
+ collectionData.splice(insertionIndex, 0, newEntry);
50
+ }
51
+
32
52
  // eslint-disable-next-line no-param-reassign
33
53
  formData["".concat(collectionName.split('.').pop(), "ActiveId")] = newEntry.id;
34
54
  return newEntryId;
@@ -24,6 +24,26 @@ describe('utils.CollectionPage.duplicateCollectionPageEntry', function () {
24
24
  expect(FORM_DATA["".concat(COLLECTION_NAME, "ActiveId")]).toBeTruthy();
25
25
  expect(FORM_DATA["".concat(COLLECTION_NAME, "ActiveId")] === '1').toBeFalsy();
26
26
  });
27
+ it('should put the new entry under the original one', function () {
28
+ var _FORM_DATA2;
29
+ var FORM_DATA = (_FORM_DATA2 = {}, _defineProperty(_FORM_DATA2, "".concat(COLLECTION_NAME, "ActiveId"), '1'), _defineProperty(_FORM_DATA2, COLLECTION_NAME, [{
30
+ id: '1',
31
+ value: 'value1'
32
+ }, {
33
+ id: '2',
34
+ value: 'value2'
35
+ }, {
36
+ id: '3',
37
+ value: 'value1'
38
+ }]), _FORM_DATA2);
39
+ (0, _duplicateCollectionPageEntry.default)(COLLECTION_NAME, FORM_DATA, '1');
40
+ expect(FORM_DATA[COLLECTION_NAME].length).toEqual(4);
41
+ expect(FORM_DATA[COLLECTION_NAME][1].value).toEqual(FORM_DATA[COLLECTION_NAME][0].value);
42
+ expect(FORM_DATA[COLLECTION_NAME][1].id).toBeTruthy();
43
+ expect(FORM_DATA[COLLECTION_NAME][1].id === '1').toBeFalsy();
44
+ expect(FORM_DATA["".concat(COLLECTION_NAME, "ActiveId")]).toBeTruthy();
45
+ expect(FORM_DATA["".concat(COLLECTION_NAME, "ActiveId")] === '1').toBeFalsy();
46
+ });
27
47
  it('should duplicate entries in nested collections correctly', function () {
28
48
  var FORM_DATA = {
29
49
  parentsActiveId: '2',
@@ -54,8 +74,8 @@ describe('utils.CollectionPage.duplicateCollectionPageEntry', function () {
54
74
  expect((0, _getCollectionPageActiveId.default)('parents.children', FORM_DATA) === '2').toBeFalsy();
55
75
  });
56
76
  it('should ignore fields in the fieldToIgnore array', function () {
57
- var _FORM_DATA2;
58
- var FORM_DATA = (_FORM_DATA2 = {}, _defineProperty(_FORM_DATA2, "".concat(COLLECTION_NAME, "ActiveId"), '1'), _defineProperty(_FORM_DATA2, COLLECTION_NAME, [OBJ]), _FORM_DATA2);
77
+ var _FORM_DATA3;
78
+ var FORM_DATA = (_FORM_DATA3 = {}, _defineProperty(_FORM_DATA3, "".concat(COLLECTION_NAME, "ActiveId"), '1'), _defineProperty(_FORM_DATA3, COLLECTION_NAME, [OBJ]), _FORM_DATA3);
59
79
  ;
60
80
  var FIELDS_TO_IGNORE = ['value'];
61
81
  (0, _duplicateCollectionPageEntry.default)(COLLECTION_NAME, FORM_DATA, '1', FIELDS_TO_IGNORE);
@@ -65,9 +85,24 @@ describe('utils.CollectionPage.duplicateCollectionPageEntry', function () {
65
85
  expect(FORM_DATA[COLLECTION_NAME][1].id).toBeTruthy();
66
86
  expect(FORM_DATA[COLLECTION_NAME][1].value).toBeUndefined();
67
87
  });
88
+ it('should add fields from the fieldsToAdd object', function () {
89
+ var _FORM_DATA4;
90
+ var FORM_DATA = (_FORM_DATA4 = {}, _defineProperty(_FORM_DATA4, "".concat(COLLECTION_NAME, "ActiveId"), '1'), _defineProperty(_FORM_DATA4, COLLECTION_NAME, [OBJ]), _FORM_DATA4);
91
+ ;
92
+ var FIELDS_TO_ADD = {
93
+ addedKey: 'addedValue'
94
+ };
95
+ (0, _duplicateCollectionPageEntry.default)(COLLECTION_NAME, FORM_DATA, '1', [], FIELDS_TO_ADD);
96
+ expect(FORM_DATA[COLLECTION_NAME].length).toEqual(2);
97
+ expect(FORM_DATA[COLLECTION_NAME][0].id).toBeTruthy();
98
+ expect(FORM_DATA[COLLECTION_NAME][0].value).toBeTruthy();
99
+ expect(FORM_DATA[COLLECTION_NAME][1].id).toBeTruthy();
100
+ expect(FORM_DATA[COLLECTION_NAME][1].value).toBeTruthy();
101
+ expect(FORM_DATA[COLLECTION_NAME][1].addedKey).toEqual('addedValue');
102
+ });
68
103
  it('should do nothing if an entry with the given id does not exist', function () {
69
- var _FORM_DATA3, _expect$toEqual;
70
- var FORM_DATA = (_FORM_DATA3 = {}, _defineProperty(_FORM_DATA3, "".concat(COLLECTION_NAME, "ActiveId"), '1'), _defineProperty(_FORM_DATA3, COLLECTION_NAME, [OBJ]), _FORM_DATA3);
104
+ var _FORM_DATA5, _expect$toEqual;
105
+ var FORM_DATA = (_FORM_DATA5 = {}, _defineProperty(_FORM_DATA5, "".concat(COLLECTION_NAME, "ActiveId"), '1'), _defineProperty(_FORM_DATA5, COLLECTION_NAME, [OBJ]), _FORM_DATA5);
71
106
  ;
72
107
  (0, _duplicateCollectionPageEntry.default)(COLLECTION_NAME, FORM_DATA, '0');
73
108
  expect(FORM_DATA).toEqual((_expect$toEqual = {}, _defineProperty(_expect$toEqual, "".concat(COLLECTION_NAME, "ActiveId"), '1'), _defineProperty(_expect$toEqual, COLLECTION_NAME, [OBJ]), _expect$toEqual));
@@ -0,0 +1,55 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.default = void 0;
7
+ var _ = _interopRequireDefault(require(".."));
8
+ function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
9
+ function _typeof(o) { "@babel/helpers - typeof"; return _typeof = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function (o) { return typeof o; } : function (o) { return o && "function" == typeof Symbol && o.constructor === Symbol && o !== Symbol.prototype ? "symbol" : typeof o; }, _typeof(o); }
10
+ function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
11
+ function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { _defineProperty(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; }
12
+ function _defineProperty(obj, key, value) { key = _toPropertyKey(key); if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
13
+ function _toPropertyKey(arg) { var key = _toPrimitive(arg, "string"); return _typeof(key) === "symbol" ? key : String(key); }
14
+ function _toPrimitive(input, hint) { if (_typeof(input) !== "object" || input === null) return input; var prim = input[Symbol.toPrimitive]; if (prim !== undefined) { var res = prim.call(input, hint || "default"); if (_typeof(res) !== "object") return res; throw new TypeError("@@toPrimitive must return a primitive value."); } return (hint === "string" ? String : Number)(input); }
15
+ var getErrorsForCollection = function getErrorsForCollection(config, masterPage, formData, hooks) {
16
+ var activeIds = arguments.length > 4 && arguments[4] !== undefined ? arguments[4] : {};
17
+ if (!masterPage) {
18
+ return [];
19
+ }
20
+ // Get data for current collection
21
+ var collectionData = _.default.CollectionPage.getData(masterPage.collection.name, masterPage.formData) || [];
22
+ var allErrors = [];
23
+
24
+ // Loop over each entry in the collection data
25
+ collectionData.forEach(function (entry) {
26
+ var _masterPage$childPage;
27
+ // eslint-disable-next-line no-param-reassign
28
+ activeIds = _objectSpread(_objectSpread({}, activeIds), {}, _defineProperty({}, "".concat(masterPage.collection.name.split('.').pop(), "ActiveId"), entry.id));
29
+
30
+ // Loop over child pages and validate each one
31
+ var allPagesErrors = (_masterPage$childPage = masterPage.childPages) === null || _masterPage$childPage === void 0 ? void 0 : _masterPage$childPage.flatMap(function (page) {
32
+ var pageWithIds = _objectSpread(_objectSpread({}, page), {}, {
33
+ formData: _objectSpread(_objectSpread({}, formData), activeIds)
34
+ });
35
+
36
+ // Handle nested collections
37
+ if (page.childPages) {
38
+ return getErrorsForCollection(config, pageWithIds, formData, hooks, activeIds);
39
+ }
40
+ var pageErrors = _.default.Validate.page(pageWithIds);
41
+ return hooks.onValidate(pageWithIds, pageErrors);
42
+ });
43
+
44
+ // For each error add the entryId so we know what Summary Card we have to pass it to.
45
+ allErrors = allErrors.concat(allPagesErrors.map(function (e) {
46
+ return _objectSpread(_objectSpread({}, e), {}, {
47
+ entryId: entry.id,
48
+ showFor: config.id,
49
+ raisedBy: config.id
50
+ });
51
+ }));
52
+ });
53
+ return allErrors;
54
+ };
55
+ var _default = exports.default = getErrorsForCollection;
@@ -0,0 +1,155 @@
1
+ "use strict";
2
+
3
+ var _getErrorsForCollection = _interopRequireDefault(require("./getErrorsForCollection"));
4
+ var _models = require("../../models");
5
+ function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
6
+ describe('Utils.CollectionPage.getErrorsForCollection', function () {
7
+ var HOOKS = {
8
+ onValidate: function onValidate(_, errors) {
9
+ return errors;
10
+ }
11
+ };
12
+ var CONFIG = {
13
+ id: 'testId'
14
+ };
15
+ var FORM_DATA = {
16
+ collection: [{
17
+ id: 22,
18
+ item1: 'blue',
19
+ child: [{
20
+ id: 11,
21
+ item4: 'red'
22
+ }]
23
+ }]
24
+ };
25
+ var FORM_DATA_MISSING_TOP = {
26
+ collection: [{
27
+ id: 22,
28
+ item2: 'blue',
29
+ child: [{
30
+ id: 11,
31
+ item4: 'red'
32
+ }]
33
+ }]
34
+ };
35
+ var FORM_DATA_MISSING_NESTED = {
36
+ collection: [{
37
+ id: 22,
38
+ item1: 'blue',
39
+ child: [{
40
+ id: 11,
41
+ item9: 'red'
42
+ }]
43
+ }]
44
+ };
45
+ var FORM_DATA_MISSING_BOTH = {
46
+ collection: [{
47
+ id: 22,
48
+ item9: 'blue',
49
+ child: [{
50
+ id: 11,
51
+ item9: 'red'
52
+ }]
53
+ }]
54
+ };
55
+ var MASTER_PAGE = {
56
+ id: 'masterPage',
57
+ collection: {
58
+ name: 'collection',
59
+ masterPage: true
60
+ },
61
+ formData: FORM_DATA,
62
+ childPages: [{
63
+ id: 'page1',
64
+ collection: {
65
+ name: 'collection'
66
+ },
67
+ formData: FORM_DATA,
68
+ components: [{
69
+ id: 'item1',
70
+ fieldId: 'item1',
71
+ label: 'Item 1',
72
+ required: true,
73
+ type: _models.ComponentTypes.TEXT
74
+ }, {
75
+ id: 'item2',
76
+ fieldId: 'item2',
77
+ label: 'Item 2',
78
+ type: _models.ComponentTypes.TEXT
79
+ }]
80
+ }]
81
+ };
82
+ var MASTER_PAGE_WITH_NESTED = {
83
+ id: 'masterPage',
84
+ collection: {
85
+ name: 'collection',
86
+ masterPage: true
87
+ },
88
+ formData: FORM_DATA,
89
+ childPages: [{
90
+ id: 'page1',
91
+ collection: {
92
+ name: 'collection'
93
+ },
94
+ formData: FORM_DATA,
95
+ components: [{
96
+ id: 'item1',
97
+ fieldId: 'item1',
98
+ label: 'Item 1',
99
+ required: true,
100
+ type: _models.ComponentTypes.TEXT
101
+ }, {
102
+ id: 'item2',
103
+ fieldId: 'item2',
104
+ label: 'Item 2',
105
+ type: _models.ComponentTypes.TEXT
106
+ }]
107
+ }, {
108
+ id: 'page2',
109
+ formData: FORM_DATA,
110
+ collection: {
111
+ name: 'collection.child',
112
+ masterPage: true
113
+ },
114
+ childPages: [{
115
+ id: 'nestedPage',
116
+ formData: FORM_DATA,
117
+ collection: {
118
+ name: 'collection.child'
119
+ },
120
+ components: [{
121
+ id: 'item4',
122
+ fieldId: 'item4',
123
+ label: 'Item 4',
124
+ required: true,
125
+ type: _models.ComponentTypes.TEXT
126
+ }, {
127
+ id: 'item5',
128
+ fieldId: 'item5',
129
+ label: 'Item 5',
130
+ type: _models.ComponentTypes.TEXT
131
+ }]
132
+ }]
133
+ }]
134
+ };
135
+ it('should return nothing for a master page without errors', function () {
136
+ var errors = (0, _getErrorsForCollection.default)(CONFIG, MASTER_PAGE, FORM_DATA, HOOKS);
137
+ expect(errors.length).toEqual(0);
138
+ });
139
+ it('should return erorrs for a master page with errors', function () {
140
+ var errors = (0, _getErrorsForCollection.default)(CONFIG, MASTER_PAGE, FORM_DATA_MISSING_TOP, HOOKS);
141
+ expect(errors.length).toEqual(1);
142
+ });
143
+ it('should return nothing for a master page with a nested master page without errors', function () {
144
+ var errors = (0, _getErrorsForCollection.default)(CONFIG, MASTER_PAGE_WITH_NESTED, FORM_DATA, HOOKS);
145
+ expect(errors.length).toEqual(0);
146
+ });
147
+ it('should return errors for a master page with a nested master page without errors', function () {
148
+ var errors = (0, _getErrorsForCollection.default)(CONFIG, MASTER_PAGE_WITH_NESTED, FORM_DATA_MISSING_NESTED, HOOKS);
149
+ expect(errors.length).toEqual(1);
150
+ });
151
+ it('should return errors for a master page with errors with a nested master page with errors', function () {
152
+ var errors = (0, _getErrorsForCollection.default)(CONFIG, MASTER_PAGE_WITH_NESTED, FORM_DATA_MISSING_BOTH, HOOKS);
153
+ expect(errors.length).toEqual(2);
154
+ });
155
+ });
@@ -6,8 +6,11 @@ Object.defineProperty(exports, "__esModule", {
6
6
  exports.default = void 0;
7
7
  var _meetsAllConditions = _interopRequireDefault(require("../Condition/meetsAllConditions"));
8
8
  var _showFormPage = _interopRequireDefault(require("../FormPage/showFormPage"));
9
+ var _excluded = ["id"];
9
10
  function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
10
11
  function _typeof(o) { "@babel/helpers - typeof"; return _typeof = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function (o) { return typeof o; } : function (o) { return o && "function" == typeof Symbol && o.constructor === Symbol && o !== Symbol.prototype ? "symbol" : typeof o; }, _typeof(o); }
12
+ function _objectWithoutProperties(source, excluded) { if (source == null) return {}; var target = _objectWithoutPropertiesLoose(source, excluded); var key, i; if (Object.getOwnPropertySymbols) { var sourceSymbolKeys = Object.getOwnPropertySymbols(source); for (i = 0; i < sourceSymbolKeys.length; i++) { key = sourceSymbolKeys[i]; if (excluded.indexOf(key) >= 0) continue; if (!Object.prototype.propertyIsEnumerable.call(source, key)) continue; target[key] = source[key]; } } return target; }
13
+ function _objectWithoutPropertiesLoose(source, excluded) { if (source == null) return {}; var target = {}; var sourceKeys = Object.keys(source); var key, i; for (i = 0; i < sourceKeys.length; i++) { key = sourceKeys[i]; if (excluded.indexOf(key) >= 0) continue; target[key] = source[key]; } return target; }
11
14
  function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
12
15
  function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { _defineProperty(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; }
13
16
  function _defineProperty(obj, key, value) { key = _toPropertyKey(key); if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
@@ -47,16 +50,17 @@ function _toPrimitive(input, hint) { if (_typeof(input) !== "object" || input ==
47
50
  * editable or not.
48
51
  *
49
52
  * @param {object} masterPage The master page for the collection.
53
+ * @param {object} formData The top-level formData object.
50
54
  * @param {object} entryData The data for the current collection entry.
51
55
  * @returns Page suitble for rendering in a QuickEdit
52
56
  */
53
- var getQuickEditPage = function getQuickEditPage(masterPage, entryData) {
57
+ var getQuickEditPage = function getQuickEditPage(masterPage, formData, entryData) {
54
58
  if (!masterPage) {
55
59
  return null;
56
60
  }
57
61
  var fullComponents = [];
58
62
  masterPage.childPages.forEach(function (page) {
59
- if (!page.summaryQuickEdit || !(0, _showFormPage.default)(page, _objectSpread(_objectSpread({}, page.formData), entryData))) {
63
+ if (!page.summaryQuickEdit || !(0, _showFormPage.default)(page, _objectSpread(_objectSpread({}, formData), entryData))) {
60
64
  return;
61
65
  }
62
66
  page.summaryQuickEdit.forEach(function (quickEditComponent) {
@@ -78,7 +82,7 @@ var getQuickEditPage = function getQuickEditPage(masterPage, entryData) {
78
82
  // that as a specific test for when the component should be quick
79
83
  // editable - it doesn't get spread to the merged object.
80
84
  if (quickEditComponent.show_when) {
81
- if ((0, _meetsAllConditions.default)(quickEditComponent.show_when, _objectSpread(_objectSpread({}, page.formData), entryData))) {
85
+ if ((0, _meetsAllConditions.default)(quickEditComponent.show_when, _objectSpread(_objectSpread({}, formData), entryData))) {
82
86
  fullComponents.push(_objectSpread(_objectSpread({}, mergedComponent), {}, {
83
87
  show_when: originalComponent.show_when || []
84
88
  }));
@@ -90,7 +94,9 @@ var getQuickEditPage = function getQuickEditPage(masterPage, entryData) {
90
94
  });
91
95
 
92
96
  // Put components and actions onto page object
93
- var pageToReturn = _objectSpread({}, masterPage.childPages[0]);
97
+ var pageToReturn = _objectSpread(_objectSpread({}, masterPage.childPages[0]), {}, {
98
+ formData: formData
99
+ });
94
100
  pageToReturn.components = fullComponents;
95
101
  pageToReturn.actions = [{
96
102
  type: 'save',
@@ -104,7 +110,10 @@ var getQuickEditPage = function getQuickEditPage(masterPage, entryData) {
104
110
  }];
105
111
 
106
112
  // Need to have the collection data at the top level
107
- pageToReturn.formData = _objectSpread(_objectSpread({}, pageToReturn.formData), entryData);
113
+ // Remove ID to stop overwriting top-level form ID.
114
+ var id = entryData.id,
115
+ entryWithNoId = _objectWithoutProperties(entryData, _excluded);
116
+ pageToReturn.formData = _objectSpread(_objectSpread({}, pageToReturn.formData), entryWithNoId);
108
117
  delete pageToReturn.title;
109
118
  delete pageToReturn.collection;
110
119
  return pageToReturn;
@@ -9,6 +9,15 @@ function _defineProperty(obj, key, value) { key = _toPropertyKey(key); if (key i
9
9
  function _toPropertyKey(arg) { var key = _toPrimitive(arg, "string"); return _typeof(key) === "symbol" ? key : String(key); }
10
10
  function _toPrimitive(input, hint) { if (_typeof(input) !== "object" || input === null) return input; var prim = input[Symbol.toPrimitive]; if (prim !== undefined) { var res = prim.call(input, hint || "default"); if (_typeof(res) !== "object") return res; throw new TypeError("@@toPrimitive must return a primitive value."); } return (hint === "string" ? String : Number)(input); }
11
11
  describe('Utils.CollectionPage.getQuickEditPage', function () {
12
+ var FORM_DATA = {
13
+ items: [{
14
+ id: '123',
15
+ testText1: 'value 1',
16
+ testText2: 'value 2',
17
+ conditionText: 'true',
18
+ testText3: 'value 3'
19
+ }]
20
+ };
12
21
  var MASTER_PAGE = {
13
22
  childPages: [{
14
23
  id: 'firstPage',
@@ -44,15 +53,7 @@ describe('Utils.CollectionPage.getQuickEditPage', function () {
44
53
  fieldId: 'testText2',
45
54
  required: true
46
55
  }],
47
- formData: {
48
- items: [{
49
- id: '123',
50
- testText1: 'value 1',
51
- testText2: 'value 2',
52
- conditionText: 'true',
53
- testText3: 'value 3'
54
- }]
55
- },
56
+ formData: FORM_DATA,
56
57
  summaryQuickEdit: [{
57
58
  use: "testText2",
58
59
  show_when: {
@@ -69,15 +70,7 @@ describe('Utils.CollectionPage.getQuickEditPage', function () {
69
70
  fieldId: 'testText3',
70
71
  required: true
71
72
  }],
72
- formData: {
73
- items: [{
74
- id: '123',
75
- testText1: 'value 1',
76
- testText2: 'value 2',
77
- conditionText: 'true',
78
- testText3: 'value 3'
79
- }]
80
- },
73
+ formData: FORM_DATA,
81
74
  summaryQuickEdit: ["testText3"]
82
75
  }]
83
76
  };
@@ -103,15 +96,7 @@ describe('Utils.CollectionPage.getQuickEditPage', function () {
103
96
  fieldId: 'testText3',
104
97
  required: true
105
98
  }],
106
- formData: {
107
- items: [{
108
- id: '123',
109
- testText1: 'value 1',
110
- testText2: 'value 2',
111
- conditionText: 'true',
112
- testText3: 'value 3'
113
- }]
114
- },
99
+ formData: FORM_DATA,
115
100
  actions: [{
116
101
  type: 'save',
117
102
  label: 'Save',
@@ -131,7 +116,7 @@ describe('Utils.CollectionPage.getQuickEditPage', function () {
131
116
  testText3: 'value 3'
132
117
  };
133
118
  it('should return a quick edit page', function () {
134
- var createdPage = (0, _getQuickEditPage.default)(MASTER_PAGE, ENTRY_DATA);
119
+ var createdPage = (0, _getQuickEditPage.default)(MASTER_PAGE, FORM_DATA, ENTRY_DATA);
135
120
  expect(createdPage).toMatchObject(EXPECTED_OUTPUT);
136
121
  });
137
122
  it('should return a quick edit page without components that fail the quick edit show_when', function () {
@@ -144,7 +129,7 @@ describe('Utils.CollectionPage.getQuickEditPage', function () {
144
129
  var CUSTOM_ENTRY_DATA = _objectSpread(_objectSpread({}, ENTRY_DATA), {}, {
145
130
  conditionText: 'false'
146
131
  });
147
- var createdPage = (0, _getQuickEditPage.default)(MASTER_PAGE, CUSTOM_ENTRY_DATA);
132
+ var createdPage = (0, _getQuickEditPage.default)(MASTER_PAGE, FORM_DATA, CUSTOM_ENTRY_DATA);
148
133
  expect(createdPage).toMatchObject(CUSTOM_OUTPUT);
149
134
  });
150
135
  });
@@ -10,6 +10,7 @@ var _duplicateCollectionPageEntry = _interopRequireDefault(require("./duplicateC
10
10
  var _getCollectionPageActiveId = _interopRequireDefault(require("./getCollectionPageActiveId"));
11
11
  var _getCollectionPageActiveIndex = _interopRequireDefault(require("./getCollectionPageActiveIndex"));
12
12
  var _getCollectionPageData = _interopRequireDefault(require("./getCollectionPageData"));
13
+ var _getErrorsForCollection = _interopRequireDefault(require("./getErrorsForCollection"));
13
14
  var _mergeCollectionPages = _interopRequireDefault(require("./mergeCollectionPages"));
14
15
  var _removeCollectionPageEntry = _interopRequireDefault(require("./removeCollectionPageEntry"));
15
16
  var _setCollectionPageData = _interopRequireDefault(require("./setCollectionPageData"));
@@ -23,6 +24,7 @@ var CollectionPage = {
23
24
  getActiveId: _getCollectionPageActiveId.default,
24
25
  getActiveIndex: _getCollectionPageActiveIndex.default,
25
26
  getData: _getCollectionPageData.default,
27
+ getErrorsForCollection: _getErrorsForCollection.default,
26
28
  mergePages: _mergeCollectionPages.default,
27
29
  removeEntry: _removeCollectionPageEntry.default,
28
30
  setData: _setCollectionPageData.default
@@ -32,7 +32,6 @@ var createMasterPage = function createMasterPage(page) {
32
32
  return {
33
33
  id: page.id,
34
34
  name: page.name,
35
- deleteCollectionWhenHidden: page.deleteCollectionWhenHidden,
36
35
  collection: _objectSpread({
37
36
  masterPage: true
38
37
  }, page.collection),
@@ -28,15 +28,20 @@ var setCollectionPageData = function setCollectionPageData(collectionName, newDa
28
28
  var data = formData;
29
29
  nameParts.some(function (name) {
30
30
  var _data$name;
31
- // Current collection doesn't exist, so create it,
32
- // add an entry and set the active id.
31
+ // Current collection doesn't exist, so create it
33
32
  if (!data[name]) {
33
+ data[name] = [];
34
+ }
35
+ // If the entry matching the active ID doesn't exist in the collection, create it.
36
+ if (!data[name].find(function (e) {
37
+ return e.id === formData["".concat(name, "ActiveId")];
38
+ })) {
34
39
  var newEntryId = Date.now().toString();
35
40
  // eslint-disable-next-line no-param-reassign
36
41
  formData["".concat(name, "ActiveId")] = formData["".concat(name, "ActiveId")] || newEntryId;
37
- data[name] = [{
42
+ data[name].push({
38
43
  id: formData["".concat(name, "ActiveId")]
39
- }];
44
+ });
40
45
  }
41
46
  // We're at the target collection's level, so
42
47
  // set the new data.
@@ -160,4 +160,22 @@ describe('Utils.CollectionPage.setCollectionPageData', function () {
160
160
  value: 'newValue'
161
161
  });
162
162
  });
163
+ it('creates a new entry and updates active ID if no matching active ID is found', function () {
164
+ var formData = {};
165
+ var collectionName = 'testCollection';
166
+ var newData = [{
167
+ id: 'dataId',
168
+ value: 'testValue'
169
+ }];
170
+ (0, _setCollectionPageData.default)(collectionName, newData, formData);
171
+
172
+ // Check if formData has been updated
173
+ expect(formData["".concat(collectionName, "ActiveId")]).toBeDefined();
174
+
175
+ // Check if the new entry was added to the collection
176
+ expect(formData[collectionName]).toBeDefined();
177
+ expect(formData[collectionName].length).toBe(1);
178
+ expect(formData[collectionName][0].id).toBe(newData[0].id);
179
+ expect(formData[collectionName][0]).toEqual(expect.objectContaining(newData[0]));
180
+ });
163
181
  });
@@ -7,7 +7,7 @@ exports.default = exports.EDITABLE_TYPES = void 0;
7
7
  var _models = require("../../models");
8
8
  // Local imports
9
9
 
10
- var EDITABLE_TYPES = exports.EDITABLE_TYPES = [_models.ComponentTypes.AUTOCOMPLETE, _models.ComponentTypes.CALCULATION, _models.ComponentTypes.CHECKBOXES, _models.ComponentTypes.DATE, _models.ComponentTypes.EMAIL, _models.ComponentTypes.FILE, _models.ComponentTypes.MULTI_FILE, _models.ComponentTypes.PHONE_NUMBER, _models.ComponentTypes.RADIOS, _models.ComponentTypes.SELECT, _models.ComponentTypes.TEXT, _models.ComponentTypes.TEXT_AREA, _models.ComponentTypes.TIME];
10
+ var EDITABLE_TYPES = exports.EDITABLE_TYPES = [_models.ComponentTypes.AUTOCOMPLETE, _models.ComponentTypes.CALCULATION, _models.ComponentTypes.CHECKBOXES, _models.ComponentTypes.COLLECTION_SUMMARY, _models.ComponentTypes.DATE, _models.ComponentTypes.EMAIL, _models.ComponentTypes.FILE, _models.ComponentTypes.MULTI_FILE, _models.ComponentTypes.PHONE_NUMBER, _models.ComponentTypes.RADIOS, _models.ComponentTypes.SELECT, _models.ComponentTypes.TEXT, _models.ComponentTypes.TEXT_AREA, _models.ComponentTypes.TIME];
11
11
  var isEditable = function isEditable(options) {
12
12
  return EDITABLE_TYPES.includes(options === null || options === void 0 ? void 0 : options.type) || (options === null || options === void 0 ? void 0 : options.isEditable) || false;
13
13
  };
@@ -108,6 +108,24 @@ var meetsCondition = function meetsCondition(condition, value, data) {
108
108
  }
109
109
  return true;
110
110
  }
111
+ case 'includesObjectProp':
112
+ {
113
+ if (Array.isArray(value)) {
114
+ return value.some(function (v) {
115
+ return v[condition.key] === compare;
116
+ });
117
+ }
118
+ return false;
119
+ }
120
+ case '!includesObjectProp':
121
+ {
122
+ if (Array.isArray(value)) {
123
+ return !value.some(function (v) {
124
+ return v[condition.key] === compare;
125
+ });
126
+ }
127
+ return true;
128
+ }
111
129
  case 'includesAllOf':
112
130
  {
113
131
  if (Array.isArray(value)) {