@inseefr/lunatic 0.2.2-experimental → 0.2.3-prisme

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 (219) hide show
  1. package/LICENSE +21 -21
  2. package/README.md +25 -23
  3. package/lib/index.js +1191 -475
  4. package/lib/index.js.map +1 -1
  5. package/package.json +174 -172
  6. package/src/components/breadcrumb/breadcrumb.scss +22 -22
  7. package/src/components/checkbox/boolean.js +172 -172
  8. package/src/components/checkbox/checkbox.scss +73 -73
  9. package/src/components/checkbox/group.js +231 -231
  10. package/src/components/checkbox/one.js +11 -11
  11. package/src/components/component-wrapper/controls/component.js +70 -0
  12. package/src/components/component-wrapper/controls/controls.scss +6 -0
  13. package/src/components/component-wrapper/controls/index.js +1 -0
  14. package/src/components/component-wrapper/controls/validators/datepicker.js +44 -0
  15. package/src/components/component-wrapper/controls/validators/index.js +16 -0
  16. package/src/components/component-wrapper/controls/validators/input-number.js +23 -0
  17. package/src/components/{missing-wrapper → component-wrapper}/index.js +0 -0
  18. package/src/components/component-wrapper/missing/component.js +200 -0
  19. package/src/components/component-wrapper/missing/index.js +1 -0
  20. package/src/components/{missing-wrapper → component-wrapper/missing}/missing.scss +32 -32
  21. package/src/components/component-wrapper/wrapper.js +23 -0
  22. package/src/components/components.js +1 -0
  23. package/src/components/datepicker/component.js +26 -11
  24. package/src/components/declarations/component.js +46 -46
  25. package/src/components/declarations/declarations.scss +40 -40
  26. package/src/components/declarations/wrappers/input-declarations-wrapper.js +328 -274
  27. package/src/components/declarations/wrappers/simple-declarations-wrapper.js +54 -54
  28. package/src/components/dropdown/commons/actions.js +65 -56
  29. package/src/components/dropdown/commons/children-to-option.js +9 -9
  30. package/src/components/dropdown/commons/cleaner-callbacks.js +58 -58
  31. package/src/components/dropdown/commons/components/dropdown-container.js +29 -29
  32. package/src/components/dropdown/commons/components/dropdown.js +204 -183
  33. package/src/components/dropdown/commons/event-callbacks/on-mousedown-callback.js +15 -15
  34. package/src/components/dropdown/commons/reducer.js +3 -0
  35. package/src/components/dropdown/commons/tools/index.js +17 -17
  36. package/src/components/dropdown/component.js +135 -135
  37. package/src/components/dropdown/dropdown-edit/dropdown-edit.js +194 -185
  38. package/src/components/dropdown/dropdown-edit/index.js +11 -11
  39. package/src/components/dropdown/dropdown-simple/dropdown.js +173 -164
  40. package/src/components/dropdown/dropdown-simple/index.js +20 -20
  41. package/src/components/dropdown/dropdown.scss +178 -178
  42. package/src/components/icon/icon.scss +15 -15
  43. package/src/components/index.js +2 -0
  44. package/src/components/index.scss +177 -175
  45. package/src/components/input/input-number.js +30 -54
  46. package/src/components/input/input.js +11 -11
  47. package/src/components/input/input.scss +31 -37
  48. package/src/components/loop/component.js +170 -169
  49. package/src/components/loop/loop.scss +13 -13
  50. package/src/components/loop/wrapper.js +1 -1
  51. package/src/components/loop-constructor/block/block.scss +10 -10
  52. package/src/components/loop-constructor/block/component.js +9 -9
  53. package/src/components/loop-constructor/roster/component.js +8 -8
  54. package/src/components/loop-constructor/wrapper/body-component.js +146 -125
  55. package/src/components/modal/component.js +36 -0
  56. package/src/components/modal/index.js +1 -0
  57. package/src/components/modal/modal.scss +33 -0
  58. package/src/components/progress-bar/progress-bar.scss +54 -54
  59. package/src/components/radio/component.js +9 -9
  60. package/src/components/radio/radio.scss +59 -59
  61. package/src/components/sequence/component.js +50 -50
  62. package/src/components/sequence/sequence.scss +10 -10
  63. package/src/components/subsequence/component.js +49 -49
  64. package/src/components/suggester/check-store.js +2 -4
  65. package/src/components/suggester/components/panel/default-option-renderer.js +27 -27
  66. package/src/components/suggester/components/panel/option-container.js +61 -61
  67. package/src/components/suggester/components/panel/panel.js +47 -47
  68. package/src/components/suggester/components/selection/default-label-renderer.js +31 -31
  69. package/src/components/suggester/components/selection/label.js +35 -35
  70. package/src/components/suggester/components/selection/selection.js +50 -50
  71. package/src/components/suggester/components/suggester-content.js +2 -2
  72. package/src/components/suggester/components/suggester.js +128 -88
  73. package/src/components/suggester/components/suggester.scss +101 -101
  74. package/src/components/suggester/default-style.scss +125 -125
  75. package/src/components/suggester/find-best-label/find-best-label.js +51 -49
  76. package/src/components/suggester/find-best-label/index.js +1 -1
  77. package/src/components/suggester/idb-suggester.js +73 -67
  78. package/src/components/suggester/lunatic-suggester.js +140 -137
  79. package/src/components/suggester/searching/create-searching.js +49 -47
  80. package/src/components/suggester/searching/index.js +1 -1
  81. package/src/components/suggester/state-management/actions.js +38 -38
  82. package/src/components/suggester/state-management/reducer/reduce-on-delete-search.js +11 -11
  83. package/src/components/suggester/state-management/reducer/reduce-on-init.js +29 -29
  84. package/src/components/suggester/state-management/reducer/reducer.js +38 -38
  85. package/src/components/suggester/suggester-wrapper.js +127 -121
  86. package/src/components/suggester-loader-widget/loader.js +67 -67
  87. package/src/components/suggester-loader-widget/widget.js +123 -123
  88. package/src/components/table/table.js +173 -171
  89. package/src/components/table/table.scss +26 -26
  90. package/src/components/textarea/component.js +11 -11
  91. package/src/components/textarea/textarea.scss +8 -8
  92. package/src/components/tooltip/tooltip.scss +30 -30
  93. package/src/stories/checkbox-boolean/data.json +78 -78
  94. package/src/stories/datepicker/data.json +3 -1
  95. package/src/stories/dropdown/README.md +44 -44
  96. package/src/stories/dropdown/data.json +98 -98
  97. package/src/stories/dropdown/dropdown.stories.js +89 -89
  98. package/src/stories/progress-bar/progress-bar.stories.js +24 -24
  99. package/src/stories/questionnaire/arithmetic-management.json +47 -0
  100. package/src/stories/questionnaire/arithmetic.json +247 -247
  101. package/src/stories/questionnaire/kish.json +275 -0
  102. package/src/stories/questionnaire/logement-queen.json +23390 -0
  103. package/src/stories/questionnaire/logement-s2.json +46028 -0
  104. package/src/stories/questionnaire/logement.json +20347 -26087
  105. package/src/stories/questionnaire/loop-and-controls.json +481 -0
  106. package/src/stories/questionnaire/questionnaire.stories.js +236 -138
  107. package/src/stories/questionnaire/update-external/data.json +1 -0
  108. package/src/stories/questionnaire/update-external/questionnaire.json +75 -0
  109. package/src/stories/suggester/README.md +46 -46
  110. package/src/stories/suggester/bailleurs-sociaux/fetch-bailleurs.js +15 -15
  111. package/src/stories/suggester/bailleurs-sociaux/index.js +2 -2
  112. package/src/stories/suggester/bailleurs-sociaux/option-bailleur-renderer.js +58 -58
  113. package/src/stories/suggester/bailleurs-sociaux/preloader.svg +51 -51
  114. package/src/stories/suggester/bailleurs-sociaux/theme.scss +22 -22
  115. package/src/stories/suggester/bailleurs-sociaux-2021/fetch-bailleurs.js +12 -0
  116. package/src/stories/suggester/bailleurs-sociaux-2021/index.js +1 -0
  117. package/src/stories/suggester/cog-communes/fetch-cog.js +15 -15
  118. package/src/stories/suggester/data-auto.json +232 -231
  119. package/src/stories/suggester/data-vtl.json +82 -82
  120. package/src/stories/suggester/data.json +169 -136
  121. package/src/stories/suggester/naf-rev2/index.js +2 -2
  122. package/src/stories/suggester/naf-rev2/option-naf-renderer.js +17 -17
  123. package/src/stories/suggester/suggester-workers.stories.js +226 -179
  124. package/src/stories/suggester/suggester.stories.js +138 -133
  125. package/src/stories/utils/orchestrator-split.js +119 -0
  126. package/src/stories/utils/orchestrator.js +119 -108
  127. package/src/tests/components/input-number.spec.js +6 -12
  128. package/src/tests/components/missing-wrapper.spec.js +0 -1
  129. package/src/tests/utils/lib/table/roster.spec.js +25 -25
  130. package/src/tests/utils/to-expose/handler/results/res-input-edited.json +1 -1
  131. package/src/utils/components/dragger/dragger.scss +7 -7
  132. package/src/utils/idb-tools/create-db-opener.js +43 -43
  133. package/src/utils/idb-tools/create-open-db.js +25 -25
  134. package/src/utils/idb-tools/idb-bulk-insert.js +96 -96
  135. package/src/utils/idb-tools/index.js +10 -10
  136. package/src/utils/idb-tools/insert-entity.js +15 -15
  137. package/src/utils/idb-tools/open-db.js +13 -13
  138. package/src/utils/idb-tools/open-or-create-db.js +34 -34
  139. package/src/utils/lib/controls/index.js +1 -0
  140. package/src/utils/lib/controls/utils.js +152 -0
  141. package/src/utils/lib/decorator/title-decorator.js +16 -16
  142. package/src/utils/lib/env.js +2 -2
  143. package/src/utils/lib/index.js +2 -0
  144. package/src/utils/lib/input-number.js +1 -1
  145. package/src/utils/lib/options-positioning.js +9 -9
  146. package/src/utils/lib/pagination/navigation/shared.js +12 -9
  147. package/src/utils/lib/prop-types/lines.js +6 -6
  148. package/src/utils/lib/responses.js +11 -9
  149. package/src/utils/lib/splitting.js +142 -0
  150. package/src/utils/lib/style.js +10 -10
  151. package/src/utils/store-tools/auto-load.js +74 -73
  152. package/src/utils/suggester-workers/append-to-index/append.js +25 -25
  153. package/src/utils/suggester-workers/append-to-index/append.worker.js +16 -16
  154. package/src/utils/suggester-workers/append-to-index/create-append-task.js +45 -43
  155. package/src/utils/suggester-workers/append-to-index/index.js +2 -2
  156. package/src/utils/suggester-workers/append-to-index/prepare-entities.js +61 -61
  157. package/src/utils/suggester-workers/append-to-index/store-messages.js +21 -21
  158. package/src/utils/suggester-workers/commons-tokenizer/create-entity-tokenizer.js +56 -0
  159. package/src/utils/suggester-workers/commons-tokenizer/create-fields-tokenizer.js +56 -0
  160. package/src/utils/suggester-workers/commons-tokenizer/create-filter-stop-words.js +17 -17
  161. package/src/utils/suggester-workers/commons-tokenizer/filters/compose-filters.js +10 -0
  162. package/src/utils/suggester-workers/commons-tokenizer/filters/create-filter-stop-words.js +17 -0
  163. package/src/utils/suggester-workers/commons-tokenizer/filters/create-filter-stop-words.spec.js +14 -0
  164. package/src/utils/suggester-workers/commons-tokenizer/filters/filter-accents.js +12 -0
  165. package/src/utils/suggester-workers/commons-tokenizer/filters/filter-accents.spec.js +12 -0
  166. package/src/utils/suggester-workers/commons-tokenizer/{filter-double.js → filters/filter-double.js} +12 -12
  167. package/src/utils/suggester-workers/commons-tokenizer/filters/filter-double.spec.js +20 -0
  168. package/src/utils/suggester-workers/commons-tokenizer/{filter-length.js → filters/filter-length.js} +7 -7
  169. package/src/utils/suggester-workers/commons-tokenizer/filters/filter-length.spec.js +18 -0
  170. package/src/utils/suggester-workers/commons-tokenizer/{filter-stemmer.js → filters/filter-stemmer.js} +13 -13
  171. package/src/utils/suggester-workers/commons-tokenizer/filters/filter-stemmer.spec.js +12 -0
  172. package/src/utils/suggester-workers/commons-tokenizer/filters/filter-synonyms.js +36 -0
  173. package/src/utils/suggester-workers/commons-tokenizer/filters/filter-synonyms.spec.js +12 -0
  174. package/src/utils/suggester-workers/commons-tokenizer/filters/filter-to-lower.js +10 -0
  175. package/src/utils/suggester-workers/commons-tokenizer/filters/filter-to-lower.spec.js +12 -0
  176. package/src/utils/suggester-workers/commons-tokenizer/filters/index.js +2 -0
  177. package/src/utils/suggester-workers/commons-tokenizer/{stop-words.js → filters/stop-words.js} +118 -118
  178. package/src/utils/suggester-workers/commons-tokenizer/get-regexp-from-pattern.js +8 -8
  179. package/src/utils/suggester-workers/commons-tokenizer/get-stemmer.js +18 -18
  180. package/src/utils/suggester-workers/commons-tokenizer/index.js +9 -8
  181. package/src/utils/suggester-workers/commons-tokenizer/prepare-string-indexation.js +13 -9
  182. package/src/utils/suggester-workers/commons-tokenizer/soft-tokenizer.js +7 -7
  183. package/src/utils/suggester-workers/create-worker.js +56 -0
  184. package/src/utils/suggester-workers/find-best-label/find-best-label.js +39 -39
  185. package/src/utils/suggester-workers/find-best-label/find-best-label.worker.js +40 -40
  186. package/src/utils/suggester-workers/find-best-label/tokenize.js +30 -33
  187. package/src/utils/suggester-workers/find-best-label/tokenize.spec.js +19 -19
  188. package/src/utils/suggester-workers/searching/compute-score.js +33 -33
  189. package/src/utils/suggester-workers/searching/get-db.js +18 -18
  190. package/src/utils/suggester-workers/searching/index.js +1 -1
  191. package/src/utils/suggester-workers/searching/order/create-alphanumeric-orderer.js +20 -20
  192. package/src/utils/suggester-workers/searching/order/index.js +19 -19
  193. package/src/utils/suggester-workers/{query-parser → searching/query-parser}/index.js +2 -2
  194. package/src/utils/suggester-workers/searching/query-parser/query-parser-soft.js +7 -0
  195. package/src/utils/suggester-workers/{query-parser → searching/query-parser}/query-parser-soft.spec.js +24 -24
  196. package/src/utils/suggester-workers/searching/query-parser/query-parser-tokenized.js +34 -0
  197. package/src/utils/suggester-workers/searching/resolve-query-parser.js +27 -27
  198. package/src/utils/suggester-workers/searching/search-in-index.js +17 -17
  199. package/src/utils/suggester-workers/searching/searching.js +70 -70
  200. package/src/utils/suggester-workers/searching/searching.worker.js +11 -11
  201. package/src/utils/to-expose/calculated-variables.js +113 -113
  202. package/src/utils/to-expose/handler.js +149 -112
  203. package/src/utils/to-expose/hooks/filter-components.js +27 -12
  204. package/src/utils/to-expose/hooks/index.js +2 -1
  205. package/src/utils/to-expose/hooks/lunatic-split.js +428 -0
  206. package/src/utils/to-expose/hooks/lunatic.js +284 -187
  207. package/src/utils/to-expose/index.js +1 -1
  208. package/src/utils/to-expose/init-questionnaire.js +164 -164
  209. package/src/utils/to-expose/interpret/vtl.js +18 -18
  210. package/src/utils/to-expose/state.js +66 -58
  211. package/src/components/missing-wrapper/component.js +0 -120
  212. package/src/components/missing-wrapper/wrapper.js +0 -10
  213. package/src/tests/utils/to-expose/hooks/use-lunatic.spec.js +0 -46
  214. package/src/utils/suggester-workers/commons-tokenizer/create-tokenizer.js +0 -103
  215. package/src/utils/suggester-workers/commons-tokenizer/filter-accents-to-lower.js +0 -9
  216. package/src/utils/suggester-workers/commons-tokenizer/filter-synonyms.js +0 -10
  217. package/src/utils/suggester-workers/query-parser/query-parser-soft.js +0 -7
  218. package/src/utils/suggester-workers/query-parser/query-parser-tokenized.js +0 -31
  219. package/src/utils/suggester-workers/query-parser/query-parser-tokenized.spec.js +0 -32
@@ -0,0 +1,152 @@
1
+ import { buildLoopBindings } from '../loops';
2
+ import { interpret } from '../../to-expose/interpret';
3
+ import { splitPage } from '..';
4
+ import { getTypeControls } from '../../../components/component-wrapper/controls/validators';
5
+
6
+ export const getControls = ({
7
+ page,
8
+ components,
9
+ bindings,
10
+ features,
11
+ preferences,
12
+ }) =>
13
+ getControlsFromComponents({
14
+ page,
15
+ components,
16
+ depth: 1,
17
+ bindings,
18
+ features,
19
+ preferences,
20
+ });
21
+
22
+ const getControlsFromComponents = ({
23
+ page,
24
+ depth,
25
+ components,
26
+ bindings,
27
+ features,
28
+ preferences,
29
+ }) => {
30
+ const { currentIteration, currentPageWithoutAnyIteration } = splitPage(
31
+ page,
32
+ depth
33
+ );
34
+ return components.reduce((acc, component) => {
35
+ const {
36
+ controls,
37
+ componentType,
38
+ components: loopComponents,
39
+ page: componentPage,
40
+ paginatedLoop,
41
+ } = component;
42
+ if (componentType === 'Loop') {
43
+ const { bindingDependencies: bindingDependenciesLoop } = component;
44
+ const rootLoopBindings = getReducedBindings(
45
+ bindingDependenciesLoop,
46
+ bindings
47
+ );
48
+ let loopBindings = {};
49
+ if (paginatedLoop) {
50
+ loopBindings = buildLoopBindings(currentIteration - 1)(
51
+ Object.entries(rootLoopBindings)
52
+ );
53
+ return [
54
+ ...acc,
55
+ ...getControlsFromComponents({
56
+ page,
57
+ components: loopComponents,
58
+ bindings: loopBindings,
59
+ features,
60
+ depth: depth + 1,
61
+ preferences,
62
+ }),
63
+ ];
64
+ } else {
65
+ const {
66
+ loopDependencies,
67
+ lines: { min },
68
+ } = component;
69
+ const iterationsStr = loopDependencies
70
+ ? bindings[loopDependencies[0]]
71
+ : min;
72
+ const iterations = parseInt(iterationsStr, 10);
73
+ const ctrls = [...Array(iterations).keys()].reduce((accIt, i) => {
74
+ loopBindings = buildLoopBindings(i)(Object.entries(rootLoopBindings));
75
+ return [
76
+ ...accIt,
77
+ ...getControlsFromComponents({
78
+ page,
79
+ components: loopComponents,
80
+ bindings: loopBindings,
81
+ features,
82
+ depth: depth,
83
+ preferences,
84
+ }),
85
+ ];
86
+ }, []);
87
+ return [...acc, ...ctrls];
88
+ }
89
+ }
90
+ if (currentPageWithoutAnyIteration === componentPage) {
91
+ if (controls) {
92
+ const interpretedControls = controls.reduce(
93
+ (
94
+ iC,
95
+ { id, control, errorMessage, criticality, bindingDependencies }
96
+ ) => {
97
+ const reducedBindings = getReducedBindings(
98
+ bindingDependencies,
99
+ bindings
100
+ );
101
+ const interpretedControl =
102
+ interpret(features)(reducedBindings)(control);
103
+ if (!interpretedControl)
104
+ return [...iC, { id, errorMessage, criticality }];
105
+ return iC;
106
+ },
107
+ []
108
+ );
109
+ if (!['InputNumber', 'Datepicker'].includes(componentType))
110
+ return [...acc, ...interpretedControls];
111
+ else {
112
+ const {
113
+ response: { name },
114
+ } = component;
115
+ const value = bindings[name];
116
+ const typeControls = getTypeControls({
117
+ ...component,
118
+ value,
119
+ preferences,
120
+ });
121
+
122
+ if (!typeControls) return [...acc, ...interpretedControls];
123
+ return [
124
+ ...acc,
125
+ { ...typeControls, criticality: 'LOCK' },
126
+ ...interpretedControls,
127
+ ];
128
+ }
129
+ }
130
+ if (['InputNumber', 'Datepicker'].includes(componentType)) {
131
+ const {
132
+ response: { name },
133
+ } = component;
134
+ const value = bindings[name];
135
+ const typeControls = getTypeControls({
136
+ ...component,
137
+ value,
138
+ preferences,
139
+ });
140
+ if (!typeControls) return [...acc];
141
+ return [...acc, { ...typeControls, criticality: 'LOCK' }];
142
+ }
143
+ }
144
+ return acc;
145
+ }, []);
146
+ };
147
+
148
+ const getReducedBindings = (bindingDependencies, bindings) =>
149
+ (bindingDependencies || []).reduce(
150
+ (accB, bD) => ({ ...accB, [bD]: bindings[bD] }),
151
+ {}
152
+ );
@@ -1,16 +1,16 @@
1
- import React from 'react';
2
-
3
- const titleDecorator = WrappedComponent => {
4
- const TitleComponent = props => {
5
- const { title } = props;
6
- return (
7
- <>
8
- <h1>{title}</h1>
9
- <WrappedComponent />
10
- </>
11
- );
12
- };
13
- return TitleComponent;
14
- };
15
-
16
- export default titleDecorator;
1
+ import React from 'react';
2
+
3
+ const titleDecorator = WrappedComponent => {
4
+ const TitleComponent = props => {
5
+ const { title } = props;
6
+ return (
7
+ <>
8
+ <h1>{title}</h1>
9
+ <WrappedComponent />
10
+ </>
11
+ );
12
+ };
13
+ return TitleComponent;
14
+ };
15
+
16
+ export default titleDecorator;
@@ -1,2 +1,2 @@
1
- export const isDev =
2
- !process.env.NODE_ENV || process.env.NODE_ENV === 'development';
1
+ export const isDev = false;
2
+ // !process.env.NODE_ENV || process.env.NODE_ENV === 'development';
@@ -1,6 +1,7 @@
1
1
  export { getAlphabet } from './alphabet';
2
2
  export * from './array';
3
3
  export * from './checkbox';
4
+ export * from './controls';
4
5
  export { titleDecorator } from './decorator';
5
6
  export * from './env';
6
7
  export * from './input-number';
@@ -17,3 +18,4 @@ export * from './table';
17
18
  export * from './tooltip';
18
19
  export { isFunction } from './function';
19
20
  export { createObjectEvent } from './event';
21
+ export * from './splitting';
@@ -1,6 +1,6 @@
1
1
  export const isNumberValid = (num, decimals = 0) => {
2
2
  // \\ because of escaping, basic regex has only one \
3
3
  if (num.includes('.') && !decimals) return false;
4
- const regex = new RegExp(`^(-)?(0|[1-9]\\d*)(\\.(\\d){0,${decimals}})?$`);
4
+ const regex = new RegExp(`^(-)?(0|[1-9]\\d*)((\\.|,)(\\d){0,${decimals}})?$`);
5
5
  return regex.test(num);
6
6
  };
@@ -1,9 +1,9 @@
1
- export const getItemsPositioningClass = positioning => {
2
- switch (positioning) {
3
- case 'HORIZONTAL':
4
- return 'horizontal-options';
5
- case 'VERTICAL':
6
- default:
7
- return '';
8
- }
9
- };
1
+ export const getItemsPositioningClass = positioning => {
2
+ switch (positioning) {
3
+ case 'HORIZONTAL':
4
+ return 'horizontal-options';
5
+ case 'VERTICAL':
6
+ default:
7
+ return '';
8
+ }
9
+ };
@@ -50,7 +50,8 @@ export const getPage = ({
50
50
  interpret(featuresWithoutMD)(loopVectorialB)(loopV) !== true ||
51
51
  (flow === FLOW_PREVIOUS &&
52
52
  currentIteration === 1 &&
53
- currentComponentIndex === 1)
53
+ currentComponentIndex === 1 &&
54
+ page === currentRootPage)
54
55
  )
55
56
  continue;
56
57
  const iterations = getIterations({
@@ -86,7 +87,6 @@ export const getPage = ({
86
87
  page,
87
88
  it,
88
89
  currentIteration,
89
- loopComponentIndex: componentIndex,
90
90
  currentComponentIndex,
91
91
  loopComponent,
92
92
  })
@@ -112,7 +112,6 @@ export const getPage = ({
112
112
  const block = filteredComponents.filter(
113
113
  (c) => c.page === page && c.componentType === 'Loop'
114
114
  );
115
-
116
115
  if (block.length > 0) {
117
116
  const loopIdOfInterest =
118
117
  flow === FLOW_NEXT ? block[0].id : block[block.length - 1].id;
@@ -168,10 +167,14 @@ const hasToBeExcluded = ({
168
167
  page,
169
168
  it,
170
169
  currentIteration,
171
- loopComponentIndex,
172
170
  currentComponentIndex,
173
171
  loopComponent,
174
172
  }) => {
173
+ const { page: loopPage, depth } = loopComponent;
174
+ const { currentComponentIndex: loopComponentIndex } = splitPage(
175
+ loopPage,
176
+ depth
177
+ );
175
178
  if (flow === FLOW_NEXT)
176
179
  return (
177
180
  (currentRootPage === page &&
@@ -218,28 +221,28 @@ const getIterations = ({ component, bindings, featuresWithoutMD }) => {
218
221
  export const splitPage = (currentPage = '1', depth) => {
219
222
  const currentPageWithDepth = depth
220
223
  ? currentPage
221
- .split('.')
224
+ ?.split('.')
222
225
  .slice(0, depth + 1) // scoped
223
226
  .join('.')
224
227
  : currentPage;
225
228
 
226
229
  const currentPageWithoutIteration = currentPageWithDepth
227
- .split('#')
230
+ ?.split('#')
228
231
  .slice(0, -1)
229
232
  .join('#');
230
233
 
231
234
  const currentPageWithoutAnyIteration = currentPageWithDepth
232
- .split('.')
235
+ ?.split('.')
233
236
  .map((e) => e.split('#')[0])
234
237
  .join('.');
235
238
 
236
239
  const currentRootPage = currentPageWithoutIteration
237
- .split('.')
240
+ ?.split('.')
238
241
  .slice(0, -1)
239
242
  .join('.');
240
243
 
241
244
  const [currentComponentIndex, currentIteration] = currentPageWithDepth
242
- .split('.')
245
+ ?.split('.')
243
246
  .pop()
244
247
  .split('#')
245
248
  .map((c) => parseInt(c, 10));
@@ -1,6 +1,6 @@
1
- import PropTypes from 'prop-types';
2
-
3
- export default PropTypes.shape({
4
- min: PropTypes.oneOfType([PropTypes.string, PropTypes.number]),
5
- max: PropTypes.oneOfType([PropTypes.string, PropTypes.number]),
6
- });
1
+ import PropTypes from 'prop-types';
2
+
3
+ export default PropTypes.shape({
4
+ min: PropTypes.oneOfType([PropTypes.string, PropTypes.number]),
5
+ max: PropTypes.oneOfType([PropTypes.string, PropTypes.number]),
6
+ });
@@ -1,9 +1,11 @@
1
- export const getResponseName = (response) => (response && response.name) || '';
2
-
3
- export const getResponseByPreference = (preferences) => (response) => {
4
- if (!(response && response.values)) return null;
5
- return preferences.reduce((acc, p) => {
6
- const value = response.values[p];
7
- return value !== null ? value : acc;
8
- }, null);
9
- };
1
+ export const getResponseName = (response) => (response && response.name) || '';
2
+
3
+ export const getResponseByPreference =
4
+ (preferences = []) =>
5
+ (response) => {
6
+ if (!(response && response.values)) return null;
7
+ return preferences.reduce((acc, p) => {
8
+ const value = response.values[p];
9
+ return value !== null ? value : acc;
10
+ }, null);
11
+ };
@@ -0,0 +1,142 @@
1
+ const getBindingsDependenciesCalculated = (variables) => {
2
+ if (!variables) return {};
3
+ return variables.reduce((acc, { name, bindingDependencies, shapeFrom }) => {
4
+ if (shapeFrom && bindingDependencies)
5
+ return { ...acc, [name]: [...bindingDependencies, shapeFrom] };
6
+ if (bindingDependencies) return { ...acc, [name]: bindingDependencies };
7
+ if (shapeFrom) return { ...acc, [name]: [shapeFrom] };
8
+ return acc;
9
+ }, {});
10
+ };
11
+
12
+ const getAllDeps = (deps) => (variablesCalcDeps) => {
13
+ if (!deps || !variablesCalcDeps) return [];
14
+ return deps.reduce((acc, dep) => {
15
+ const depsOfDep = variablesCalcDeps[dep];
16
+ if (Array.isArray(depsOfDep)) {
17
+ return [...acc, dep, ...getAllDeps(depsOfDep)(variablesCalcDeps)];
18
+ }
19
+ return [...acc, dep];
20
+ }, []);
21
+ };
22
+
23
+ const getNestedVarsInFilterOrControl = (element) => {
24
+ if (element && Array.isArray(element?.bindingDependencies))
25
+ return element?.bindingDependencies;
26
+ return [];
27
+ };
28
+
29
+ const getNestedVarsInComponent = (component) => {
30
+ const {
31
+ componentType,
32
+ bindingDependencies = [],
33
+ conditionFilter,
34
+ controls = [],
35
+ } = component;
36
+ var bindings = [
37
+ ...bindingDependencies, // bindingDependencies of Component
38
+ ...getNestedVarsInFilterOrControl(conditionFilter), // bindingDependencies of its conditionFilter
39
+ ...controls.reduce(
40
+ (acc, c) => [...acc, ...getNestedVarsInFilterOrControl(c)],
41
+ []
42
+ ), // bindingDependencies of its controls
43
+ ];
44
+
45
+ if (componentType === 'Loop') {
46
+ const { components, loopDependencies } = component;
47
+ if (Array.isArray(loopDependencies))
48
+ bindings = [...bindings, ...loopDependencies];
49
+ if (Array.isArray(components)) {
50
+ bindings = components.reduce(
51
+ (acc, c) => [...acc, ...getNestedVarsInComponent(c)],
52
+ [...bindings]
53
+ );
54
+ }
55
+ }
56
+
57
+ return bindings;
58
+ };
59
+
60
+ const getNestedVars =
61
+ (components = []) =>
62
+ (variables) => {
63
+ const variableCalculatedDependencies =
64
+ getBindingsDependenciesCalculated(variables);
65
+ const depsVarsTemp = components
66
+ .reduce((acc, c) => {
67
+ return [...acc, ...getNestedVarsInComponent(c)];
68
+ }, [])
69
+ .filter((v, i, a) => a.indexOf(v) === i);
70
+ return getAllDeps(depsVarsTemp)(variableCalculatedDependencies).filter(
71
+ (v, i, a) => a.indexOf(v) === i
72
+ );
73
+ };
74
+
75
+ const getUsefullVariablesFromSource = (variables) => (nestedVars) => {
76
+ return variables.filter(({ variableType, name }) => {
77
+ if (variableType === 'CALCULATED' && !nestedVars.includes(name))
78
+ return false;
79
+ if (variableType === 'COLLECTED' && !nestedVars.includes(name))
80
+ return false;
81
+ return true;
82
+ });
83
+ };
84
+
85
+ export const getSplitQuestionnaireSource = (source) => {
86
+ const { components, variables, ...rest } = source;
87
+ var split = [];
88
+ var currentComponents = [];
89
+ var previousPage = null;
90
+ components.map((c) => {
91
+ const { componentType, page } = c;
92
+ // splitting by Sequence or Loop
93
+ if (
94
+ (componentType === 'Sequence' || componentType === 'Loop') &&
95
+ previousPage !== page
96
+ ) {
97
+ if (currentComponents.length > 0) split.push(currentComponents);
98
+ currentComponents = [c];
99
+ } else {
100
+ currentComponents.push(c);
101
+ }
102
+ previousPage = page;
103
+ return null;
104
+ });
105
+ if (currentComponents.length > 0) split.push(currentComponents);
106
+
107
+ return split.reduce((prev, currentSource) => {
108
+ const firstPage = currentSource[0].page;
109
+ const maxPage = currentSource[currentSource.length - 1].page;
110
+ const nestedVars = getNestedVars(currentSource)(variables);
111
+ const newVariables = getUsefullVariablesFromSource(variables)(nestedVars);
112
+
113
+ return [
114
+ ...prev,
115
+ {
116
+ ...rest,
117
+ variables: newVariables,
118
+ firstPage,
119
+ maxPage,
120
+ components: currentSource,
121
+ },
122
+ ];
123
+ }, []);
124
+ };
125
+
126
+ export const getRootPageInSources = (sources) => {
127
+ return sources.map((source) => {
128
+ const { components } = source;
129
+ return components.reduce((acc, { page }) => {
130
+ if (page) return [...acc, page];
131
+ return acc;
132
+ }, []);
133
+ });
134
+ };
135
+
136
+ export const mergeStateData = (oldData, newData) => {
137
+ return {
138
+ COLLECTED: { ...oldData.COLLECTED, ...newData.COLLECTED },
139
+ CALCULATED: { ...oldData.CALCULATED, ...newData.CALCULATED },
140
+ EXTERNAL: { ...oldData.EXTERNAL, ...newData.EXTERNAL },
141
+ };
142
+ };
@@ -1,10 +1,10 @@
1
- import camelCase from 'lodash.camelcase';
2
-
3
- export const buildStyleObject = obj =>
4
- obj
5
- ? Object.entries(obj).reduce((_, [key, value]) => {
6
- if (key.startsWith(':')) _[key] = buildStyleObject(value);
7
- else _[camelCase(key)] = value;
8
- return _;
9
- }, {})
10
- : {};
1
+ import camelCase from 'lodash.camelcase';
2
+
3
+ export const buildStyleObject = obj =>
4
+ obj
5
+ ? Object.entries(obj).reduce((_, [key, value]) => {
6
+ if (key.startsWith(':')) _[key] = buildStyleObject(value);
7
+ else _[camelCase(key)] = value;
8
+ return _;
9
+ }, {})
10
+ : {};
@@ -1,73 +1,74 @@
1
- import openStorage from './open-or-create-store';
2
- import updateStoreInfo from './create/update-store-info';
3
- import cleanStorage from './clear-store-data';
4
- import cleanInfoStorage from './clear-store-info';
5
-
6
- const workerPath =
7
- process.env.LUNATIC_LOADER_WORKER_PATH ||
8
- process.env.REACT_APP_LUNATIC_LOADER_WORKER_PATH ||
9
- 'workers/lunatic-loader-worker-0.1.0.js';
10
-
11
- export const loadSuggesters = (suggesterFetcher) => async (suggesters) => {
12
- Object.entries(suggesters).forEach(async ([name, attrs]) => {
13
- const { version } = attrs;
14
- const db = await openStorage(name, version);
15
- if (db) {
16
- await cleanStorage(db);
17
- await cleanInfoStorage(db);
18
- await updateStoreInfo(db, attrs);
19
- }
20
- });
21
- Object.entries(suggesters).forEach(([name, attrs]) => {
22
- const { url, version, fields, stopWords } = attrs;
23
- const f = suggesterFetcher || fetch;
24
- f(url).then(async (res) => {
25
- const data = await res.json();
26
- const [launch, terminate] = task({ name, fields, stopWords }, version);
27
- await launch(data);
28
- terminate();
29
- });
30
- });
31
- };
32
-
33
- /**
34
- * Only with Worker
35
- */
36
- function task({ name, fields, stopWords }, version, log = () => null) {
37
- const worker = new Worker(workerPath);
38
- let start = false;
39
- let stop = false;
40
-
41
- function launch(entities, post = () => null) {
42
- return new Promise(function (resolve) {
43
- start = true;
44
- worker.postMessage({
45
- name,
46
- version,
47
- fields,
48
- entities,
49
- stopWords,
50
- });
51
- worker.addEventListener('message', function (e) {
52
- const { data } = e;
53
- if (data === 'success') {
54
- if (!stop) {
55
- post();
56
- }
57
- resolve(data);
58
- } else {
59
- log(data);
60
- }
61
- });
62
- });
63
- }
64
-
65
- function terminate() {
66
- if (start) {
67
- stop = true;
68
- worker.terminate();
69
- }
70
- }
71
-
72
- return [launch, terminate];
73
- }
1
+ import openStorage from './open-or-create-store';
2
+ import updateStoreInfo from './create/update-store-info';
3
+ import cleanStorage from './clear-store-data';
4
+ import cleanInfoStorage from './clear-store-info';
5
+ import { createWorker } from '../suggester-workers/create-worker';
6
+
7
+ const workerPath =
8
+ process.env.LUNATIC_LOADER_WORKER_PATH ||
9
+ process.env.REACT_APP_LUNATIC_LOADER_WORKER_PATH ||
10
+ 'workers/lunatic-loader-worker-0.1.0.js';
11
+
12
+ export const loadSuggesters = (suggesterFetcher) => async (suggesters) => {
13
+ Object.entries(suggesters).forEach(async ([name, attrs]) => {
14
+ const { version } = attrs;
15
+ const db = await openStorage(name, version);
16
+ if (db) {
17
+ await cleanStorage(db);
18
+ await cleanInfoStorage(db);
19
+ await updateStoreInfo(db, attrs);
20
+ }
21
+ });
22
+ Object.entries(suggesters).forEach(([name, attrs]) => {
23
+ const { url, version, fields, stopWords } = attrs;
24
+ const f = suggesterFetcher || fetch;
25
+ f(url).then(async (res) => {
26
+ const data = await res.json();
27
+ const [launch, terminate] = task({ name, fields, stopWords }, version);
28
+ await launch(data);
29
+ terminate();
30
+ });
31
+ });
32
+ };
33
+
34
+ /**
35
+ * Only with Worker
36
+ */
37
+ function task({ name, fields, stopWords }, version, log = () => null) {
38
+ const worker = createWorker(workerPath);
39
+ let start = false;
40
+ let stop = false;
41
+
42
+ function launch(entities, post = () => null) {
43
+ return new Promise(function (resolve) {
44
+ start = true;
45
+ worker.postMessage({
46
+ name,
47
+ version,
48
+ fields,
49
+ entities,
50
+ stopWords,
51
+ });
52
+ worker.addEventListener('message', function (e) {
53
+ const { data } = e;
54
+ if (data === 'success') {
55
+ if (!stop) {
56
+ post();
57
+ }
58
+ resolve(data);
59
+ } else {
60
+ log(data);
61
+ }
62
+ });
63
+ });
64
+ }
65
+
66
+ function terminate() {
67
+ if (start) {
68
+ stop = true;
69
+ worker.terminate();
70
+ }
71
+ }
72
+
73
+ return [launch, terminate];
74
+ }