@omniumretail/component-library 1.0.74 → 1.0.76

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 (176) hide show
  1. package/.storybook/main.js +16 -16
  2. package/.storybook/preview-head.html +20 -20
  3. package/.storybook/preview.js +27 -27
  4. package/NPMPUBLISH.md +29 -29
  5. package/README.md +54 -54
  6. package/bitbucket-pipelines.yml +94 -94
  7. package/dist/bundle.js +247 -291
  8. package/dist/types/components/AnalyticsBar/AnalyticsBar.stories.d.ts +5 -5
  9. package/dist/types/components/AnalyticsBar/helpers/codeMutation.d.ts +4 -4
  10. package/dist/types/components/AnalyticsBar/index.d.ts +2 -2
  11. package/dist/types/components/AnalyticsBar/interfaces/analyticsBar.d.ts +12 -12
  12. package/dist/types/components/Button/Button.stories.d.ts +6 -6
  13. package/dist/types/components/Button/index.d.ts +7 -7
  14. package/dist/types/components/Category/Category.stories.d.ts +4 -4
  15. package/dist/types/components/Category/CategoryContent/index.d.ts +7 -7
  16. package/dist/types/components/Category/CategorySidebar/index.d.ts +26 -26
  17. package/dist/types/components/Category/index.d.ts +7 -7
  18. package/dist/types/components/CategoryReadOnly/CategoryReadOnly.stories.d.ts +4 -4
  19. package/dist/types/components/CategoryReadOnly/evaluationOptions.d.ts +10 -10
  20. package/dist/types/components/CategoryReadOnly/index.d.ts +9 -9
  21. package/dist/types/components/CategoryResponse/CategoryResponse.stories.d.ts +4 -4
  22. package/dist/types/components/CategoryResponse/evaluationOptions.d.ts +10 -10
  23. package/dist/types/components/CategoryResponse/index.d.ts +9 -9
  24. package/dist/types/components/DatePicker/DatePicker.stories.d.ts +5 -5
  25. package/dist/types/components/DatePicker/index.d.ts +8 -8
  26. package/dist/types/components/DatePickerTag/DatePickerTag.stories.d.ts +5 -5
  27. package/dist/types/components/DatePickerTag/index.d.ts +6 -6
  28. package/dist/types/components/Footer/Footer.stories.d.ts +5 -5
  29. package/dist/types/components/Footer/index.d.ts +4 -4
  30. package/dist/types/components/Input/Input.stories.d.ts +5 -5
  31. package/dist/types/components/Input/index.d.ts +10 -10
  32. package/dist/types/components/InputCountryCode/index.d.ts +10 -10
  33. package/dist/types/components/InputCountryCode/inputCountryCode.stories.d.ts +5 -5
  34. package/dist/types/components/Label/Label.stories.d.ts +6 -6
  35. package/dist/types/components/Label/index.d.ts +7 -7
  36. package/dist/types/components/Link/Link.stories.d.ts +7 -7
  37. package/dist/types/components/Link/index.d.ts +7 -7
  38. package/dist/types/components/Menu/Menu.stories.d.ts +5 -5
  39. package/dist/types/components/Menu/helpers/codeMutation.d.ts +4 -4
  40. package/dist/types/components/Menu/index.d.ts +3 -2
  41. package/dist/types/components/ModalConfirmation/ModalConfirmation.stories.d.ts +5 -5
  42. package/dist/types/components/ModalConfirmation/ModalStatusList.d.ts +5 -5
  43. package/dist/types/components/ModalConfirmation/index.d.ts +9 -9
  44. package/dist/types/components/ModalWithTable/ModalWithTable.stories.d.ts +5 -5
  45. package/dist/types/components/ModalWithTable/index.d.ts +13 -13
  46. package/dist/types/components/Navigation/Navigation.stories.d.ts +5 -5
  47. package/dist/types/components/Navigation/index.d.ts +10 -10
  48. package/dist/types/components/Notification/Notification.stories.d.ts +5 -5
  49. package/dist/types/components/Notification/index.d.ts +9 -9
  50. package/dist/types/components/Questions/Questions.stories.d.ts +4 -4
  51. package/dist/types/components/Questions/SingleQuestion/index.d.ts +1 -1
  52. package/dist/types/components/Questions/index.d.ts +5 -5
  53. package/dist/types/components/Radio/Radio.stories.d.ts +5 -5
  54. package/dist/types/components/Radio/index.d.ts +10 -10
  55. package/dist/types/components/Select/Select.stories.d.ts +6 -6
  56. package/dist/types/components/Select/index.d.ts +5 -5
  57. package/dist/types/components/Separator/Separator.stories.d.ts +6 -6
  58. package/dist/types/components/Separator/index.d.ts +11 -11
  59. package/dist/types/components/Sidebar/Sidebar.stories.d.ts +6 -6
  60. package/dist/types/components/Sidebar/index.d.ts +15 -15
  61. package/dist/types/components/Switch/Switch.stories.d.ts +5 -5
  62. package/dist/types/components/Switch/index.d.ts +2 -2
  63. package/dist/types/components/Table/Table.stories.d.ts +9 -9
  64. package/dist/types/components/Table/index.d.ts +37 -37
  65. package/dist/types/components/Tag/Tag.stories.d.ts +5 -5
  66. package/dist/types/components/Tag/index.d.ts +8 -8
  67. package/dist/types/components/Upload/Upload.stories.d.ts +4 -4
  68. package/dist/types/components/Upload/index.d.ts +8 -8
  69. package/dist/types/components/UserInfo/UserInfo.stories.d.ts +4 -4
  70. package/dist/types/components/UserInfo/index.d.ts +8 -8
  71. package/dist/types/components/index.d.ts +25 -25
  72. package/dist/types/constants/i18n.d.ts +1 -1
  73. package/dist/types/constants/translationHelper.d.ts +2 -2
  74. package/dist/types/index.d.ts +2 -2
  75. package/package.json +198 -198
  76. package/src/assets/scss/_global.scss +89 -89
  77. package/src/assets/scss/index.scss +2 -2
  78. package/src/components/AnalyticsBar/AnalyticsBar.stories.tsx +236 -236
  79. package/src/components/AnalyticsBar/helpers/codeMutation.tsx +19 -19
  80. package/src/components/AnalyticsBar/index.tsx +76 -76
  81. package/src/components/AnalyticsBar/interfaces/analyticsBar.tsx +13 -13
  82. package/src/components/AnalyticsBar/styles.module.scss +108 -108
  83. package/src/components/Button/Button.stories.tsx +26 -26
  84. package/src/components/Button/index.tsx +24 -24
  85. package/src/components/Button/styles.module.scss +65 -65
  86. package/src/components/Category/Category.stories.tsx +88 -88
  87. package/src/components/Category/CategoryContent/index.tsx +188 -188
  88. package/src/components/Category/CategoryContent/styles.module.scss +51 -51
  89. package/src/components/Category/CategorySidebar/index.tsx +268 -268
  90. package/src/components/Category/CategorySidebar/styles.module.scss +28 -28
  91. package/src/components/Category/index.tsx +76 -76
  92. package/src/components/Category/styles.module.scss +13 -13
  93. package/src/components/CategoryReadOnly/CategoryReadOnly.stories.tsx +270 -270
  94. package/src/components/CategoryReadOnly/evaluationOptions.tsx +81 -81
  95. package/src/components/CategoryReadOnly/index.tsx +254 -254
  96. package/src/components/CategoryReadOnly/styles.module.scss +184 -184
  97. package/src/components/CategoryResponse/CategoryResponse.stories.tsx +251 -251
  98. package/src/components/CategoryResponse/evaluationOptions.tsx +81 -81
  99. package/src/components/CategoryResponse/index.tsx +277 -277
  100. package/src/components/CategoryResponse/styles.module.scss +180 -180
  101. package/src/components/DatePicker/DatePicker.stories.tsx +16 -16
  102. package/src/components/DatePicker/index.tsx +38 -38
  103. package/src/components/DatePicker/styles.module.scss +3 -3
  104. package/src/components/DatePickerTag/DatePickerTag.stories.tsx +19 -19
  105. package/src/components/DatePickerTag/index.tsx +89 -89
  106. package/src/components/DatePickerTag/styles.module.scss +31 -31
  107. package/src/components/Footer/Footer.stories.tsx +14 -14
  108. package/src/components/Footer/index.tsx +38 -38
  109. package/src/components/Footer/styles.module.scss +38 -38
  110. package/src/components/Input/Input.stories.tsx +13 -13
  111. package/src/components/Input/index.tsx +31 -31
  112. package/src/components/Input/styles.module.scss +8 -8
  113. package/src/components/InputCountryCode/index.tsx +75 -75
  114. package/src/components/InputCountryCode/inputCountryCode.stories.tsx +55 -55
  115. package/src/components/InputCountryCode/styles.module.scss +3 -3
  116. package/src/components/Label/Label.stories.tsx +21 -21
  117. package/src/components/Label/index.tsx +19 -19
  118. package/src/components/Label/styles.module.scss +16 -16
  119. package/src/components/Link/Link.stories.tsx +30 -30
  120. package/src/components/Link/index.tsx +21 -21
  121. package/src/components/Link/styles.module.scss +24 -24
  122. package/src/components/Menu/Menu.stories.tsx +178 -178
  123. package/src/components/Menu/helpers/codeMutation.tsx +19 -19
  124. package/src/components/Menu/index.tsx +23 -23
  125. package/src/components/ModalConfirmation/ModalConfirmation.stories.tsx +40 -40
  126. package/src/components/ModalConfirmation/ModalStatusList.tsx +5 -5
  127. package/src/components/ModalConfirmation/index.tsx +70 -70
  128. package/src/components/ModalConfirmation/styles.module.scss +62 -62
  129. package/src/components/ModalWithTable/ModalWithTable.stories.tsx +96 -96
  130. package/src/components/ModalWithTable/index.tsx +232 -232
  131. package/src/components/ModalWithTable/styles.module.scss +77 -77
  132. package/src/components/Navigation/Navigation.stories.tsx +17 -17
  133. package/src/components/Navigation/index.tsx +33 -33
  134. package/src/components/Navigation/styles.module.scss +59 -59
  135. package/src/components/Notification/Notification.stories.tsx +29 -29
  136. package/src/components/Notification/index.tsx +21 -21
  137. package/src/components/Questions/Questions.stories.tsx +37 -37
  138. package/src/components/Questions/SingleQuestion/index.tsx +84 -84
  139. package/src/components/Questions/SingleQuestion/styles.module.scss +81 -81
  140. package/src/components/Questions/index.tsx +78 -78
  141. package/src/components/Radio/Radio.stories.tsx +43 -43
  142. package/src/components/Radio/index.tsx +26 -26
  143. package/src/components/Radio/styles.module.scss +23 -23
  144. package/src/components/Select/Select.stories.tsx +39 -39
  145. package/src/components/Select/index.tsx +27 -27
  146. package/src/components/Select/styles.module.scss +13 -13
  147. package/src/components/Separator/Separator.stories.tsx +22 -22
  148. package/src/components/Separator/index.tsx +27 -27
  149. package/src/components/Separator/styles.module.scss +25 -25
  150. package/src/components/Sidebar/Sidebar.stories.tsx +85 -85
  151. package/src/components/Sidebar/index.tsx +103 -103
  152. package/src/components/Sidebar/styles.module.scss +85 -85
  153. package/src/components/Switch/Switch.stories.tsx +14 -14
  154. package/src/components/Switch/index.tsx +8 -8
  155. package/src/components/Switch/styles.module.scss +7 -7
  156. package/src/components/Table/Table.stories.tsx +222 -222
  157. package/src/components/Table/index.tsx +279 -275
  158. package/src/components/Table/styles.module.scss +75 -75
  159. package/src/components/Tag/Tag.stories.tsx +22 -22
  160. package/src/components/Tag/index.tsx +189 -189
  161. package/src/components/Tag/styles.module.scss +60 -60
  162. package/src/components/Upload/Upload.stories.tsx +45 -45
  163. package/src/components/Upload/index.tsx +91 -91
  164. package/src/components/UserInfo/UserInfo.stories.tsx +38 -37
  165. package/src/components/UserInfo/index.tsx +77 -62
  166. package/src/components/UserInfo/styles.module.scss +30 -29
  167. package/src/components/index.tsx +25 -25
  168. package/src/constants/i18n.ts +25 -25
  169. package/src/constants/translationHelper.ts +7 -7
  170. package/src/index.ts +2 -2
  171. package/src/locales/en.json +86 -86
  172. package/src/locales/es.json +86 -86
  173. package/src/locales/pt.json +86 -86
  174. package/src/types/Global.d.ts +4 -4
  175. package/tsconfig.json +29 -29
  176. package/webpack.config.js +51 -51
@@ -1,277 +1,277 @@
1
- import styles from './styles.module.scss';
2
- import { useEffect, useImperativeHandle, useState } from 'react';
3
- import { Form } from 'antd';
4
- import { useForm } from 'antd/es/form/Form';
5
- import { Button } from '../Button';
6
- import classNames from 'classnames';
7
- import TextArea from 'antd/es/input/TextArea';
8
- import { Select } from '../Select';
9
- import { evaluationOptions } from './evaluationOptions';
10
- import React from 'react';
11
- import { t } from 'i18next';
12
-
13
- interface CategoryResponse {
14
- data: any;
15
- serverReadyData: any;
16
- onNextCategoryAvailabilityChange: (hasNext: boolean) => void;
17
- onPreviousCategoryAvailabilityChange: (hasPrevious: boolean) => void;
18
- };
19
-
20
- type Category = {
21
- Title: string;
22
- Key: string;
23
- Data: any;
24
- Children?: Category[];
25
- };
26
-
27
- const updateCategoryAnswers = (CategoryAnswers: any[], categoryToUpdate: any, updatedQuestions: any) => {
28
- CategoryAnswers.forEach((category: any) => {
29
- if (category.Data.CategoryId === categoryToUpdate.CategoryId) {
30
- category.Data.Questions = category.Data.Questions.map((question: any, index: any) => {
31
- return {
32
- ...question,
33
- Answer: updatedQuestions[index].Answer
34
- };
35
- });
36
- }
37
- else if (category.Children && category.Children.length > 0) {
38
- updateCategoryAnswers(category.Children, categoryToUpdate, updatedQuestions);
39
- }
40
- });
41
- return CategoryAnswers;
42
- };
43
-
44
- const findCategoryWithQuestions = (currentKey: string, categories: Category[], direction: 'next' | 'prev'): Category | null => {
45
- let foundCurrent = false;
46
-
47
- const searchCategories = (categoryList: Category[], direction: 'next' | 'prev'): Category | null => {
48
- const iterable = direction === 'next' ? categoryList : [...categoryList].reverse();
49
-
50
- for (const category of iterable) {
51
- if (foundCurrent) {
52
- if (category.Data.Questions.length > 0) {
53
- return category;
54
- }
55
- } else if (category.Key === currentKey) {
56
- foundCurrent = true;
57
- }
58
-
59
- if (category.Children) {
60
- const result = searchCategories(category.Children, direction);
61
- if (result) {
62
- return result;
63
- }
64
- }
65
- }
66
- return null;
67
- };
68
-
69
- return searchCategories(categories, direction);
70
- };
71
-
72
- const getCategoryObject = (currentKey: string, categories: Category[], direction: 'next' | 'prev'): any => {
73
- const category = findCategoryWithQuestions(currentKey, categories, direction);
74
- return category ? category.Data : null;
75
- };
76
-
77
- const hasCategory = (currentKey: string, categories: Category[], direction: 'next' | 'prev'): boolean => {
78
- const category = findCategoryWithQuestions(currentKey, categories, direction);
79
- return !!category;
80
- };
81
-
82
- const getTitleWithQuestionCount = (category: any): string => {
83
- const questionCount = category?.Data?.Questions?.length;
84
- const answeredQuestions = category?.Data?.Questions?.filter((question: any) => question.Answer !== null && question.Answer !== undefined && question.Answer !== '').length;
85
-
86
- return questionCount
87
- ? `${category.Title} - ${answeredQuestions} / ${questionCount}`
88
- : category.Title ;
89
- };
90
-
91
- export const CategoryResponse = React.forwardRef((props: CategoryResponse, ref) => {
92
- const { data } = props;
93
-
94
- const [currentKey, setCurrentKey] = useState(data.CategoryAnswers[0].Key);
95
- const [localData, setLocalData] = useState<any>(data);
96
- // Setting first set of questions as default open
97
- const [selectedCategory, setSelectedCategory] = useState<any>(data.CategoryAnswers[0]);
98
- const [initialValues, setInitialValues] = useState<any>(data.CategoryAnswers[0].Data);
99
- const [form] = useForm();
100
-
101
-
102
- const updateInitialValues = (data: any) => {
103
- const {
104
- Questions,
105
- OpenAnswer
106
- } = data;
107
-
108
- const initial = {
109
- Questions: Questions.map((question: any) => ({
110
- Subject: question.Subject,
111
- Description: question.Description,
112
- Answer: question.Answer || ""
113
- })),
114
- OpenAnswer: OpenAnswer
115
- };
116
-
117
- setInitialValues(initial);
118
- form.setFieldsValue(initial);
119
- };
120
-
121
- const handleLabelClick = (category: any, index: number) => {
122
- setSelectedCategory(category);
123
- updateInitialValues(category.Data);
124
- setCurrentKey(index);
125
- };
126
-
127
- const onFinish = (values: any) => {
128
- const updatedQuestions = initialValues.Questions.map((question: any, index: number) => {
129
- return {
130
- ...question,
131
- Answer: values.Questions[index].Answer
132
- };
133
- });
134
-
135
- const updatedCategory = updateCategoryAnswers(localData.CategoryAnswers, selectedCategory.Data, updatedQuestions);
136
-
137
- const updatedLocalData = { ...localData, CategoryAnswers: updatedCategory };
138
-
139
- setLocalData(updatedLocalData);
140
- };
141
-
142
- const renderCategories = (categories: any) => {
143
- return categories.map((category: any, index: number) => {
144
- const labelClasses = classNames({
145
- [styles.cursorPointer]: category.Data.Questions.length > 0
146
- }, [styles.label]);
147
-
148
- const indexCheck = category.Data.Questions.length > 0 && category.Key;
149
-
150
- return (
151
- <div
152
- className={`${styles.labelWrapper} ${category.Data.CategoryId === selectedCategory.Data.CategoryId ? styles.active : ""}`}
153
- key={index}
154
- >
155
- <div
156
- className={labelClasses}
157
- data-index={indexCheck}
158
- onClick={() => category.Data.Questions.length > 0 && handleLabelClick(category, indexCheck)}
159
- >
160
- {getTitleWithQuestionCount(category)}
161
- </div>
162
- {category.Children && (
163
- <div className={styles.subCategory}>
164
- {renderCategories(category.Children)}
165
- </div>
166
- )}
167
- </div>
168
- )
169
- });
170
- };
171
-
172
- const handleNextClick = () => {
173
- const nextCategory = findCategoryWithQuestions(currentKey, localData.CategoryAnswers, 'next');
174
- if (nextCategory) {
175
- setCurrentKey(nextCategory.Key);
176
- setSelectedCategory(nextCategory);
177
- }
178
-
179
- const nextCategoryData = getCategoryObject(currentKey, localData.CategoryAnswers, 'next');
180
- setSelectedCategory(nextCategoryData ? { ...nextCategory, data: nextCategoryData } : null);
181
- updateInitialValues(nextCategoryData);
182
- };
183
-
184
- const handlePreviousClick = () => {
185
- const prevCategory = findCategoryWithQuestions(currentKey, localData.CategoryAnswers, 'prev');
186
- if (prevCategory) {
187
- setCurrentKey(prevCategory.Key);
188
- setSelectedCategory(prevCategory);
189
- }
190
-
191
- const prevCategoryData = getCategoryObject(currentKey, localData.CategoryAnswers, 'prev');
192
- setSelectedCategory(prevCategoryData ? { ...prevCategory, data: prevCategoryData } : null);
193
- updateInitialValues(prevCategoryData);
194
- };
195
-
196
- useImperativeHandle(ref, () => ({
197
- handleNextClick,
198
- handlePreviousClick,
199
- }));
200
-
201
- useEffect(() => {
202
- props.serverReadyData(localData);
203
- }, [localData]);
204
-
205
- useEffect(() => {
206
- const hasNext = hasCategory(currentKey, localData.CategoryAnswers, 'next');
207
- props.onNextCategoryAvailabilityChange(hasNext);
208
-
209
- const hasPrevious = hasCategory(currentKey, localData.CategoryAnswers, 'prev');
210
- props.onPreviousCategoryAvailabilityChange(hasPrevious);
211
- }, [currentKey, localData.CategoryAnswers, props]);
212
-
213
- const questionWrapper = classNames({
214
- [styles.questionWrapperOpenAnswer]: selectedCategory.Data.OpenAnswer,
215
- }, [styles.questionWrapper]);
216
-
217
- // This gets the scale we use for each select from the database using a key based variable,if
218
- // any new generalEvaluationLevel is added to the backend we need to update our own without
219
- // deleting any of the previous so we dont mess up the project history.
220
- const selectedEvaluationOption = `scale_${selectedCategory?.Data?.GeneralEvaluationLevel}` as keyof typeof evaluationOptions;
221
-
222
- return (
223
- <div className={styles.categoryResponse}>
224
- <div className={styles.sidebarWrapper}>
225
- <div className={styles.title}>Categorias</div>
226
- {renderCategories(localData.CategoryAnswers)}
227
- </div>
228
- <div className={styles.contentWrapper}>
229
- <Form
230
- initialValues={initialValues}
231
- name="dynamic_form_nest_item"
232
- onFinish={onFinish}
233
- autoComplete="off"
234
- form={form}
235
- onValuesChange={(_, allValues) => onFinish(allValues)}
236
- >
237
- <div className={styles.details}>
238
- <div className={styles.categoryName}>
239
- {selectedCategory?.Data?.CategoryName}
240
- </div>
241
- <div className={styles.categoryDescription}>
242
- {selectedCategory?.Data?.Description}
243
- </div>
244
- </div>
245
- {selectedCategory?.Data?.Questions.map((question: any, index: number) => (
246
- <div className={questionWrapper} key={index}>
247
- <div className={styles.question}>
248
- <div className={styles.subject}>
249
- {question.Subject}
250
- </div>
251
- <div className={styles.description}>
252
- {question.Description}
253
- </div>
254
- </div>
255
- <div className={styles.answer}>
256
- {selectedCategory.Data.OpenAnswer
257
- ? <Form.Item
258
- name={["Questions", index, "Answer"]}
259
- >
260
- <TextArea />
261
- </Form.Item>
262
- : <Form.Item
263
- name={["Questions", index, "Answer"]}
264
- >
265
- <Select options={evaluationOptions[selectedEvaluationOption]}
266
- style={{ minWidth: '100%' }}
267
- />
268
- </Form.Item>
269
- }
270
- </div>
271
- </div>
272
- ))}
273
- </Form>
274
- </div>
275
- </div>
276
- );
277
- });
1
+ import styles from './styles.module.scss';
2
+ import { useEffect, useImperativeHandle, useState } from 'react';
3
+ import { Form } from 'antd';
4
+ import { useForm } from 'antd/es/form/Form';
5
+ import { Button } from '../Button';
6
+ import classNames from 'classnames';
7
+ import TextArea from 'antd/es/input/TextArea';
8
+ import { Select } from '../Select';
9
+ import { evaluationOptions } from './evaluationOptions';
10
+ import React from 'react';
11
+ import { t } from 'i18next';
12
+
13
+ interface CategoryResponse {
14
+ data: any;
15
+ serverReadyData: any;
16
+ onNextCategoryAvailabilityChange: (hasNext: boolean) => void;
17
+ onPreviousCategoryAvailabilityChange: (hasPrevious: boolean) => void;
18
+ };
19
+
20
+ type Category = {
21
+ Title: string;
22
+ Key: string;
23
+ Data: any;
24
+ Children?: Category[];
25
+ };
26
+
27
+ const updateCategoryAnswers = (CategoryAnswers: any[], categoryToUpdate: any, updatedQuestions: any) => {
28
+ CategoryAnswers.forEach((category: any) => {
29
+ if (category.Data.CategoryId === categoryToUpdate.CategoryId) {
30
+ category.Data.Questions = category.Data.Questions.map((question: any, index: any) => {
31
+ return {
32
+ ...question,
33
+ Answer: updatedQuestions[index].Answer
34
+ };
35
+ });
36
+ }
37
+ else if (category.Children && category.Children.length > 0) {
38
+ updateCategoryAnswers(category.Children, categoryToUpdate, updatedQuestions);
39
+ }
40
+ });
41
+ return CategoryAnswers;
42
+ };
43
+
44
+ const findCategoryWithQuestions = (currentKey: string, categories: Category[], direction: 'next' | 'prev'): Category | null => {
45
+ let foundCurrent = false;
46
+
47
+ const searchCategories = (categoryList: Category[], direction: 'next' | 'prev'): Category | null => {
48
+ const iterable = direction === 'next' ? categoryList : [...categoryList].reverse();
49
+
50
+ for (const category of iterable) {
51
+ if (foundCurrent) {
52
+ if (category.Data.Questions.length > 0) {
53
+ return category;
54
+ }
55
+ } else if (category.Key === currentKey) {
56
+ foundCurrent = true;
57
+ }
58
+
59
+ if (category.Children) {
60
+ const result = searchCategories(category.Children, direction);
61
+ if (result) {
62
+ return result;
63
+ }
64
+ }
65
+ }
66
+ return null;
67
+ };
68
+
69
+ return searchCategories(categories, direction);
70
+ };
71
+
72
+ const getCategoryObject = (currentKey: string, categories: Category[], direction: 'next' | 'prev'): any => {
73
+ const category = findCategoryWithQuestions(currentKey, categories, direction);
74
+ return category ? category.Data : null;
75
+ };
76
+
77
+ const hasCategory = (currentKey: string, categories: Category[], direction: 'next' | 'prev'): boolean => {
78
+ const category = findCategoryWithQuestions(currentKey, categories, direction);
79
+ return !!category;
80
+ };
81
+
82
+ const getTitleWithQuestionCount = (category: any): string => {
83
+ const questionCount = category?.Data?.Questions?.length;
84
+ const answeredQuestions = category?.Data?.Questions?.filter((question: any) => question.Answer !== null && question.Answer !== undefined && question.Answer !== '').length;
85
+
86
+ return questionCount
87
+ ? `${category.Title} - ${answeredQuestions} / ${questionCount}`
88
+ : category.Title ;
89
+ };
90
+
91
+ export const CategoryResponse = React.forwardRef((props: CategoryResponse, ref) => {
92
+ const { data } = props;
93
+
94
+ const [currentKey, setCurrentKey] = useState(data.CategoryAnswers[0].Key);
95
+ const [localData, setLocalData] = useState<any>(data);
96
+ // Setting first set of questions as default open
97
+ const [selectedCategory, setSelectedCategory] = useState<any>(data.CategoryAnswers[0]);
98
+ const [initialValues, setInitialValues] = useState<any>(data.CategoryAnswers[0].Data);
99
+ const [form] = useForm();
100
+
101
+
102
+ const updateInitialValues = (data: any) => {
103
+ const {
104
+ Questions,
105
+ OpenAnswer
106
+ } = data;
107
+
108
+ const initial = {
109
+ Questions: Questions.map((question: any) => ({
110
+ Subject: question.Subject,
111
+ Description: question.Description,
112
+ Answer: question.Answer || ""
113
+ })),
114
+ OpenAnswer: OpenAnswer
115
+ };
116
+
117
+ setInitialValues(initial);
118
+ form.setFieldsValue(initial);
119
+ };
120
+
121
+ const handleLabelClick = (category: any, index: number) => {
122
+ setSelectedCategory(category);
123
+ updateInitialValues(category.Data);
124
+ setCurrentKey(index);
125
+ };
126
+
127
+ const onFinish = (values: any) => {
128
+ const updatedQuestions = initialValues.Questions.map((question: any, index: number) => {
129
+ return {
130
+ ...question,
131
+ Answer: values.Questions[index].Answer
132
+ };
133
+ });
134
+
135
+ const updatedCategory = updateCategoryAnswers(localData.CategoryAnswers, selectedCategory.Data, updatedQuestions);
136
+
137
+ const updatedLocalData = { ...localData, CategoryAnswers: updatedCategory };
138
+
139
+ setLocalData(updatedLocalData);
140
+ };
141
+
142
+ const renderCategories = (categories: any) => {
143
+ return categories.map((category: any, index: number) => {
144
+ const labelClasses = classNames({
145
+ [styles.cursorPointer]: category.Data.Questions.length > 0
146
+ }, [styles.label]);
147
+
148
+ const indexCheck = category.Data.Questions.length > 0 && category.Key;
149
+
150
+ return (
151
+ <div
152
+ className={`${styles.labelWrapper} ${category.Data.CategoryId === selectedCategory.Data.CategoryId ? styles.active : ""}`}
153
+ key={index}
154
+ >
155
+ <div
156
+ className={labelClasses}
157
+ data-index={indexCheck}
158
+ onClick={() => category.Data.Questions.length > 0 && handleLabelClick(category, indexCheck)}
159
+ >
160
+ {getTitleWithQuestionCount(category)}
161
+ </div>
162
+ {category.Children && (
163
+ <div className={styles.subCategory}>
164
+ {renderCategories(category.Children)}
165
+ </div>
166
+ )}
167
+ </div>
168
+ )
169
+ });
170
+ };
171
+
172
+ const handleNextClick = () => {
173
+ const nextCategory = findCategoryWithQuestions(currentKey, localData.CategoryAnswers, 'next');
174
+ if (nextCategory) {
175
+ setCurrentKey(nextCategory.Key);
176
+ setSelectedCategory(nextCategory);
177
+ }
178
+
179
+ const nextCategoryData = getCategoryObject(currentKey, localData.CategoryAnswers, 'next');
180
+ setSelectedCategory(nextCategoryData ? { ...nextCategory, data: nextCategoryData } : null);
181
+ updateInitialValues(nextCategoryData);
182
+ };
183
+
184
+ const handlePreviousClick = () => {
185
+ const prevCategory = findCategoryWithQuestions(currentKey, localData.CategoryAnswers, 'prev');
186
+ if (prevCategory) {
187
+ setCurrentKey(prevCategory.Key);
188
+ setSelectedCategory(prevCategory);
189
+ }
190
+
191
+ const prevCategoryData = getCategoryObject(currentKey, localData.CategoryAnswers, 'prev');
192
+ setSelectedCategory(prevCategoryData ? { ...prevCategory, data: prevCategoryData } : null);
193
+ updateInitialValues(prevCategoryData);
194
+ };
195
+
196
+ useImperativeHandle(ref, () => ({
197
+ handleNextClick,
198
+ handlePreviousClick,
199
+ }));
200
+
201
+ useEffect(() => {
202
+ props.serverReadyData(localData);
203
+ }, [localData]);
204
+
205
+ useEffect(() => {
206
+ const hasNext = hasCategory(currentKey, localData.CategoryAnswers, 'next');
207
+ props.onNextCategoryAvailabilityChange(hasNext);
208
+
209
+ const hasPrevious = hasCategory(currentKey, localData.CategoryAnswers, 'prev');
210
+ props.onPreviousCategoryAvailabilityChange(hasPrevious);
211
+ }, [currentKey, localData.CategoryAnswers, props]);
212
+
213
+ const questionWrapper = classNames({
214
+ [styles.questionWrapperOpenAnswer]: selectedCategory.Data.OpenAnswer,
215
+ }, [styles.questionWrapper]);
216
+
217
+ // This gets the scale we use for each select from the database using a key based variable,if
218
+ // any new generalEvaluationLevel is added to the backend we need to update our own without
219
+ // deleting any of the previous so we dont mess up the project history.
220
+ const selectedEvaluationOption = `scale_${selectedCategory?.Data?.GeneralEvaluationLevel}` as keyof typeof evaluationOptions;
221
+
222
+ return (
223
+ <div className={styles.categoryResponse}>
224
+ <div className={styles.sidebarWrapper}>
225
+ <div className={styles.title}>Categorias</div>
226
+ {renderCategories(localData.CategoryAnswers)}
227
+ </div>
228
+ <div className={styles.contentWrapper}>
229
+ <Form
230
+ initialValues={initialValues}
231
+ name="dynamic_form_nest_item"
232
+ onFinish={onFinish}
233
+ autoComplete="off"
234
+ form={form}
235
+ onValuesChange={(_, allValues) => onFinish(allValues)}
236
+ >
237
+ <div className={styles.details}>
238
+ <div className={styles.categoryName}>
239
+ {selectedCategory?.Data?.CategoryName}
240
+ </div>
241
+ <div className={styles.categoryDescription}>
242
+ {selectedCategory?.Data?.Description}
243
+ </div>
244
+ </div>
245
+ {selectedCategory?.Data?.Questions.map((question: any, index: number) => (
246
+ <div className={questionWrapper} key={index}>
247
+ <div className={styles.question}>
248
+ <div className={styles.subject}>
249
+ {question.Subject}
250
+ </div>
251
+ <div className={styles.description}>
252
+ {question.Description}
253
+ </div>
254
+ </div>
255
+ <div className={styles.answer}>
256
+ {selectedCategory.Data.OpenAnswer
257
+ ? <Form.Item
258
+ name={["Questions", index, "Answer"]}
259
+ >
260
+ <TextArea />
261
+ </Form.Item>
262
+ : <Form.Item
263
+ name={["Questions", index, "Answer"]}
264
+ >
265
+ <Select options={evaluationOptions[selectedEvaluationOption]}
266
+ style={{ minWidth: '100%' }}
267
+ />
268
+ </Form.Item>
269
+ }
270
+ </div>
271
+ </div>
272
+ ))}
273
+ </Form>
274
+ </div>
275
+ </div>
276
+ );
277
+ });