@hed-hog/lms 0.0.361 → 0.0.364

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 (329) hide show
  1. package/dist/bitcode-wallet/bitcode-wallet.service.d.ts +65 -0
  2. package/dist/bitcode-wallet/bitcode-wallet.service.d.ts.map +1 -1
  3. package/dist/bitcode-wallet/bitcode-wallet.service.js +72 -0
  4. package/dist/bitcode-wallet/bitcode-wallet.service.js.map +1 -1
  5. package/dist/bitcode-wallet/dto/create-current-bitcode-wallet-transaction.dto.d.ts +8 -0
  6. package/dist/bitcode-wallet/dto/create-current-bitcode-wallet-transaction.dto.d.ts.map +1 -0
  7. package/dist/bitcode-wallet/dto/create-current-bitcode-wallet-transaction.dto.js +40 -0
  8. package/dist/bitcode-wallet/dto/create-current-bitcode-wallet-transaction.dto.js.map +1 -0
  9. package/dist/class-group/class-group.controller.d.ts +16 -16
  10. package/dist/class-group/class-group.service.d.ts +12 -12
  11. package/dist/course/course-audio-transcription.service.d.ts +3 -2
  12. package/dist/course/course-audio-transcription.service.d.ts.map +1 -1
  13. package/dist/course/course-audio-transcription.service.js +49 -8
  14. package/dist/course/course-audio-transcription.service.js.map +1 -1
  15. package/dist/course/course-lesson.controller.d.ts +4 -0
  16. package/dist/course/course-lesson.controller.d.ts.map +1 -1
  17. package/dist/course/course-lesson.controller.js +10 -0
  18. package/dist/course/course-lesson.controller.js.map +1 -1
  19. package/dist/course/course-structure.controller.d.ts +11 -4
  20. package/dist/course/course-structure.controller.d.ts.map +1 -1
  21. package/dist/course/course-structure.controller.js +14 -0
  22. package/dist/course/course-structure.controller.js.map +1 -1
  23. package/dist/course/course-structure.service.d.ts +15 -1
  24. package/dist/course/course-structure.service.d.ts.map +1 -1
  25. package/dist/course/course-structure.service.js +139 -4
  26. package/dist/course/course-structure.service.js.map +1 -1
  27. package/dist/course/course-video-conversion.service.d.ts +8 -0
  28. package/dist/course/course-video-conversion.service.d.ts.map +1 -1
  29. package/dist/course/course-video-conversion.service.js +87 -51
  30. package/dist/course/course-video-conversion.service.js.map +1 -1
  31. package/dist/course/course.controller.d.ts +73 -1
  32. package/dist/course/course.controller.d.ts.map +1 -1
  33. package/dist/course/course.controller.js +27 -3
  34. package/dist/course/course.controller.js.map +1 -1
  35. package/dist/course/course.module.d.ts.map +1 -1
  36. package/dist/course/course.module.js +2 -0
  37. package/dist/course/course.module.js.map +1 -1
  38. package/dist/course/course.service.d.ts +108 -4
  39. package/dist/course/course.service.d.ts.map +1 -1
  40. package/dist/course/course.service.js +631 -30
  41. package/dist/course/course.service.js.map +1 -1
  42. package/dist/course/dto/cleanup-course-storage.dto.d.ts +6 -0
  43. package/dist/course/dto/cleanup-course-storage.dto.d.ts.map +1 -0
  44. package/dist/course/dto/cleanup-course-storage.dto.js +33 -0
  45. package/dist/course/dto/cleanup-course-storage.dto.js.map +1 -0
  46. package/dist/course/dto/cleanup-upload-history.dto.d.ts +9 -0
  47. package/dist/course/dto/cleanup-upload-history.dto.d.ts.map +1 -0
  48. package/dist/course/dto/cleanup-upload-history.dto.js +36 -0
  49. package/dist/course/dto/cleanup-upload-history.dto.js.map +1 -0
  50. package/dist/course/dto/create-course-bulk-job.dto.d.ts +4 -0
  51. package/dist/course/dto/create-course-bulk-job.dto.d.ts.map +1 -0
  52. package/dist/course/dto/create-course-bulk-job.dto.js +21 -0
  53. package/dist/course/dto/create-course-bulk-job.dto.js.map +1 -0
  54. package/dist/course/lms-bulk-upload-automation.service.d.ts +39 -0
  55. package/dist/course/lms-bulk-upload-automation.service.d.ts.map +1 -0
  56. package/dist/course/lms-bulk-upload-automation.service.js +443 -0
  57. package/dist/course/lms-bulk-upload-automation.service.js.map +1 -0
  58. package/dist/course/lms-bulk-upload-infra.service.d.ts +31 -0
  59. package/dist/course/lms-bulk-upload-infra.service.d.ts.map +1 -0
  60. package/dist/course/lms-bulk-upload-infra.service.js +277 -0
  61. package/dist/course/lms-bulk-upload-infra.service.js.map +1 -0
  62. package/dist/course/lms-bulk-upload.constants.d.ts +4 -0
  63. package/dist/course/lms-bulk-upload.constants.d.ts.map +1 -0
  64. package/dist/course/lms-bulk-upload.constants.js +7 -0
  65. package/dist/course/lms-bulk-upload.constants.js.map +1 -0
  66. package/dist/course/lms-bulk-upload.controller.d.ts +116 -0
  67. package/dist/course/lms-bulk-upload.controller.d.ts.map +1 -1
  68. package/dist/course/lms-bulk-upload.controller.js +71 -2
  69. package/dist/course/lms-bulk-upload.controller.js.map +1 -1
  70. package/dist/course/lms-bulk-upload.service.d.ts +142 -3
  71. package/dist/course/lms-bulk-upload.service.d.ts.map +1 -1
  72. package/dist/course/lms-bulk-upload.service.js +606 -21
  73. package/dist/course/lms-bulk-upload.service.js.map +1 -1
  74. package/dist/course/lms-setting.controller.d.ts +4 -1
  75. package/dist/course/lms-setting.controller.d.ts.map +1 -1
  76. package/dist/course/lms-setting.controller.js +21 -6
  77. package/dist/course/lms-setting.controller.js.map +1 -1
  78. package/dist/enterprise/enterprise.controller.d.ts +20 -20
  79. package/dist/enterprise/enterprise.service.d.ts +20 -20
  80. package/dist/enterprise/training/training-admin.controller.d.ts +11 -11
  81. package/dist/enterprise/training/training-admin.service.d.ts +11 -11
  82. package/dist/enterprise/training/training-instructor.controller.d.ts +2 -2
  83. package/dist/enterprise/training/training-instructor.service.d.ts +2 -2
  84. package/dist/enterprise/training/training-student.controller.d.ts +1 -1
  85. package/dist/enterprise/training/training-student.service.d.ts +1 -1
  86. package/dist/enterprise/training/training-viewer.controller.d.ts +2 -2
  87. package/dist/evaluation/evaluation.controller.d.ts +8 -8
  88. package/dist/evaluation/evaluation.service.d.ts +8 -8
  89. package/dist/lesson-xp-map/dto/create-lesson-xp-map.dto.d.ts +6 -0
  90. package/dist/lesson-xp-map/dto/create-lesson-xp-map.dto.d.ts.map +1 -0
  91. package/dist/lesson-xp-map/dto/create-lesson-xp-map.dto.js +34 -0
  92. package/dist/lesson-xp-map/dto/create-lesson-xp-map.dto.js.map +1 -0
  93. package/dist/lesson-xp-map/dto/create-lesson-xp-segment.dto.d.ts +28 -0
  94. package/dist/lesson-xp-map/dto/create-lesson-xp-segment.dto.d.ts.map +1 -0
  95. package/dist/lesson-xp-map/dto/create-lesson-xp-segment.dto.js +123 -0
  96. package/dist/lesson-xp-map/dto/create-lesson-xp-segment.dto.js.map +1 -0
  97. package/dist/lesson-xp-map/dto/review-lesson-xp-map.dto.d.ts +4 -0
  98. package/dist/lesson-xp-map/dto/review-lesson-xp-map.dto.d.ts.map +1 -0
  99. package/dist/lesson-xp-map/dto/review-lesson-xp-map.dto.js +22 -0
  100. package/dist/lesson-xp-map/dto/review-lesson-xp-map.dto.js.map +1 -0
  101. package/dist/lesson-xp-map/dto/update-lesson-xp-map.dto.d.ts +10 -0
  102. package/dist/lesson-xp-map/dto/update-lesson-xp-map.dto.d.ts.map +1 -0
  103. package/dist/lesson-xp-map/dto/update-lesson-xp-map.dto.js +52 -0
  104. package/dist/lesson-xp-map/dto/update-lesson-xp-map.dto.js.map +1 -0
  105. package/dist/lesson-xp-map/dto/update-lesson-xp-segment.dto.d.ts +15 -0
  106. package/dist/lesson-xp-map/dto/update-lesson-xp-segment.dto.d.ts.map +1 -0
  107. package/dist/lesson-xp-map/dto/update-lesson-xp-segment.dto.js +86 -0
  108. package/dist/lesson-xp-map/dto/update-lesson-xp-segment.dto.js.map +1 -0
  109. package/dist/lesson-xp-map/lesson-xp-ai-calculation.service.d.ts +26 -0
  110. package/dist/lesson-xp-map/lesson-xp-ai-calculation.service.d.ts.map +1 -0
  111. package/dist/lesson-xp-map/lesson-xp-ai-calculation.service.js +304 -0
  112. package/dist/lesson-xp-map/lesson-xp-ai-calculation.service.js.map +1 -0
  113. package/dist/lesson-xp-map/lesson-xp-map.controller.d.ts +87 -0
  114. package/dist/lesson-xp-map/lesson-xp-map.controller.d.ts.map +1 -0
  115. package/dist/lesson-xp-map/lesson-xp-map.controller.js +185 -0
  116. package/dist/lesson-xp-map/lesson-xp-map.controller.js.map +1 -0
  117. package/dist/lesson-xp-map/lesson-xp-map.module.d.ts +3 -0
  118. package/dist/lesson-xp-map/lesson-xp-map.module.d.ts.map +1 -0
  119. package/dist/lesson-xp-map/lesson-xp-map.module.js +34 -0
  120. package/dist/lesson-xp-map/lesson-xp-map.module.js.map +1 -0
  121. package/dist/lesson-xp-map/lesson-xp-map.service.d.ts +84 -0
  122. package/dist/lesson-xp-map/lesson-xp-map.service.d.ts.map +1 -0
  123. package/dist/lesson-xp-map/lesson-xp-map.service.js +353 -0
  124. package/dist/lesson-xp-map/lesson-xp-map.service.js.map +1 -0
  125. package/dist/lesson-xp-map/lesson-xp-segment.controller.d.ts +10 -0
  126. package/dist/lesson-xp-map/lesson-xp-segment.controller.d.ts.map +1 -0
  127. package/dist/lesson-xp-map/lesson-xp-segment.controller.js +63 -0
  128. package/dist/lesson-xp-map/lesson-xp-segment.controller.js.map +1 -0
  129. package/dist/lesson-xp-map/lesson-xp-segment.service.d.ts +27 -0
  130. package/dist/lesson-xp-map/lesson-xp-segment.service.d.ts.map +1 -0
  131. package/dist/lesson-xp-map/lesson-xp-segment.service.js +194 -0
  132. package/dist/lesson-xp-map/lesson-xp-segment.service.js.map +1 -0
  133. package/dist/libraries/lms/tsconfig.tsbuildinfo +1 -0
  134. package/dist/lms.module.d.ts.map +1 -1
  135. package/dist/lms.module.js +17 -2
  136. package/dist/lms.module.js.map +1 -1
  137. package/dist/platforma/dto/update-profile.dto.d.ts +17 -0
  138. package/dist/platforma/dto/update-profile.dto.d.ts.map +1 -0
  139. package/dist/platforma/dto/update-profile.dto.js +87 -0
  140. package/dist/platforma/dto/update-profile.dto.js.map +1 -0
  141. package/dist/platforma/platforma.controller.d.ts +88 -1
  142. package/dist/platforma/platforma.controller.d.ts.map +1 -1
  143. package/dist/platforma/platforma.controller.js +85 -2
  144. package/dist/platforma/platforma.controller.js.map +1 -1
  145. package/dist/platforma/platforma.service.d.ts +27 -0
  146. package/dist/platforma/platforma.service.d.ts.map +1 -0
  147. package/dist/platforma/platforma.service.js +274 -0
  148. package/dist/platforma/platforma.service.js.map +1 -0
  149. package/dist/student-xp/student-xp.controller.d.ts +41 -0
  150. package/dist/student-xp/student-xp.controller.d.ts.map +1 -0
  151. package/dist/student-xp/student-xp.controller.js +114 -0
  152. package/dist/student-xp/student-xp.controller.js.map +1 -0
  153. package/dist/student-xp/student-xp.module.d.ts +3 -0
  154. package/dist/student-xp/student-xp.module.d.ts.map +1 -0
  155. package/dist/student-xp/student-xp.module.js +25 -0
  156. package/dist/student-xp/student-xp.module.js.map +1 -0
  157. package/dist/student-xp/student-xp.service.d.ts +65 -0
  158. package/dist/student-xp/student-xp.service.d.ts.map +1 -0
  159. package/dist/student-xp/student-xp.service.js +197 -0
  160. package/dist/student-xp/student-xp.service.js.map +1 -0
  161. package/dist/xp-catalog/dto/create-xp-area.dto.d.ts +12 -0
  162. package/dist/xp-catalog/dto/create-xp-area.dto.d.ts.map +1 -0
  163. package/dist/xp-catalog/dto/create-xp-area.dto.js +63 -0
  164. package/dist/xp-catalog/dto/create-xp-area.dto.js.map +1 -0
  165. package/dist/xp-catalog/dto/create-xp-learning-type.dto.d.ts +11 -0
  166. package/dist/xp-catalog/dto/create-xp-learning-type.dto.d.ts.map +1 -0
  167. package/dist/xp-catalog/dto/create-xp-learning-type.dto.js +57 -0
  168. package/dist/xp-catalog/dto/create-xp-learning-type.dto.js.map +1 -0
  169. package/dist/xp-catalog/dto/create-xp-skill.dto.d.ts +11 -0
  170. package/dist/xp-catalog/dto/create-xp-skill.dto.d.ts.map +1 -0
  171. package/dist/xp-catalog/dto/create-xp-skill.dto.js +57 -0
  172. package/dist/xp-catalog/dto/create-xp-skill.dto.js.map +1 -0
  173. package/dist/xp-catalog/dto/update-xp-area.dto.d.ts +12 -0
  174. package/dist/xp-catalog/dto/update-xp-area.dto.d.ts.map +1 -0
  175. package/dist/xp-catalog/dto/update-xp-area.dto.js +66 -0
  176. package/dist/xp-catalog/dto/update-xp-area.dto.js.map +1 -0
  177. package/dist/xp-catalog/dto/update-xp-learning-type.dto.d.ts +11 -0
  178. package/dist/xp-catalog/dto/update-xp-learning-type.dto.d.ts.map +1 -0
  179. package/dist/xp-catalog/dto/update-xp-learning-type.dto.js +60 -0
  180. package/dist/xp-catalog/dto/update-xp-learning-type.dto.js.map +1 -0
  181. package/dist/xp-catalog/dto/update-xp-skill.dto.d.ts +11 -0
  182. package/dist/xp-catalog/dto/update-xp-skill.dto.d.ts.map +1 -0
  183. package/dist/xp-catalog/dto/update-xp-skill.dto.js +60 -0
  184. package/dist/xp-catalog/dto/update-xp-skill.dto.js.map +1 -0
  185. package/dist/xp-catalog/xp-area.controller.d.ts +25 -0
  186. package/dist/xp-catalog/xp-area.controller.d.ts.map +1 -0
  187. package/dist/xp-catalog/xp-area.controller.js +105 -0
  188. package/dist/xp-catalog/xp-area.controller.js.map +1 -0
  189. package/dist/xp-catalog/xp-area.service.d.ts +35 -0
  190. package/dist/xp-catalog/xp-area.service.d.ts.map +1 -0
  191. package/dist/xp-catalog/xp-area.service.js +168 -0
  192. package/dist/xp-catalog/xp-area.service.js.map +1 -0
  193. package/dist/xp-catalog/xp-catalog.module.d.ts +3 -0
  194. package/dist/xp-catalog/xp-catalog.module.d.ts.map +1 -0
  195. package/dist/xp-catalog/xp-catalog.module.js +29 -0
  196. package/dist/xp-catalog/xp-catalog.module.js.map +1 -0
  197. package/dist/xp-catalog/xp-learning-type.controller.d.ts +20 -0
  198. package/dist/xp-catalog/xp-learning-type.controller.d.ts.map +1 -0
  199. package/dist/xp-catalog/xp-learning-type.controller.js +96 -0
  200. package/dist/xp-catalog/xp-learning-type.controller.js.map +1 -0
  201. package/dist/xp-catalog/xp-learning-type.service.d.ts +30 -0
  202. package/dist/xp-catalog/xp-learning-type.service.d.ts.map +1 -0
  203. package/dist/xp-catalog/xp-learning-type.service.js +146 -0
  204. package/dist/xp-catalog/xp-learning-type.service.js.map +1 -0
  205. package/dist/xp-catalog/xp-skill.controller.d.ts +26 -0
  206. package/dist/xp-catalog/xp-skill.controller.d.ts.map +1 -0
  207. package/dist/xp-catalog/xp-skill.controller.js +113 -0
  208. package/dist/xp-catalog/xp-skill.controller.js.map +1 -0
  209. package/dist/xp-catalog/xp-skill.service.d.ts +37 -0
  210. package/dist/xp-catalog/xp-skill.service.d.ts.map +1 -0
  211. package/dist/xp-catalog/xp-skill.service.js +174 -0
  212. package/dist/xp-catalog/xp-skill.service.js.map +1 -0
  213. package/hedhog/data/menu.yaml +101 -0
  214. package/hedhog/data/route.yaml +512 -0
  215. package/hedhog/data/setting_group.yaml +1 -1
  216. package/hedhog/data/xp_area.yaml +164 -0
  217. package/hedhog/data/xp_learning_type.yaml +131 -0
  218. package/hedhog/data/xp_skill.yaml +1834 -0
  219. package/hedhog/frontend/app/achievements/page.tsx.ejs +108 -118
  220. package/hedhog/frontend/app/bitcodes/page.tsx.ejs +22 -34
  221. package/hedhog/frontend/app/bulk-upload-sessions/page.tsx.ejs +1453 -0
  222. package/hedhog/frontend/app/certificates/models/page.tsx.ejs +21 -45
  223. package/hedhog/frontend/app/classes/[id]/page.tsx.ejs +40 -74
  224. package/hedhog/frontend/app/classes/page.tsx.ejs +56 -85
  225. package/hedhog/frontend/app/courses/[id]/_components/CourseDangerZoneCard.tsx.ejs +3 -2
  226. package/hedhog/frontend/app/courses/[id]/_components/CourseMediaCard.tsx.ejs +48 -5
  227. package/hedhog/frontend/app/courses/[id]/page.tsx.ejs +4 -4
  228. package/hedhog/frontend/app/courses/[id]/structure/_components/course-operations-tab.tsx.ejs +19 -2
  229. package/hedhog/frontend/app/courses/[id]/structure/_components/course-overview-tab.tsx.ejs +1170 -0
  230. package/hedhog/frontend/app/courses/[id]/structure/_components/course-tree-dnd.tsx.ejs +16 -0
  231. package/hedhog/frontend/app/courses/[id]/structure/_components/course-tree.tsx.ejs +2 -0
  232. package/hedhog/frontend/app/courses/[id]/structure/_components/course-xp-overview-tab.tsx.ejs +623 -0
  233. package/hedhog/frontend/app/courses/[id]/structure/_components/detail-lesson-xp-tab.tsx.ejs +1458 -0
  234. package/hedhog/frontend/app/courses/[id]/structure/_components/detail-lesson.tsx.ejs +55 -2
  235. package/hedhog/frontend/app/courses/[id]/structure/_components/editor-course.tsx.ejs +442 -104
  236. package/hedhog/frontend/app/courses/[id]/structure/_components/editor-lesson.tsx.ejs +296 -49
  237. package/hedhog/frontend/app/courses/[id]/structure/_components/sortable-tree-row.tsx.ejs +3 -0
  238. package/hedhog/frontend/app/courses/[id]/structure/_components/store.ts.ejs +1 -0
  239. package/hedhog/frontend/app/courses/[id]/structure/_components/tree-display-settings-popover.tsx.ejs +101 -85
  240. package/hedhog/frontend/app/courses/[id]/structure/_components/tree-row-lesson.tsx.ejs +21 -1
  241. package/hedhog/frontend/app/courses/[id]/structure/_components/tree-row.tsx.ejs +3 -0
  242. package/hedhog/frontend/app/courses/[id]/structure/_components/use-tree-display-settings.ts.ejs +7 -1
  243. package/hedhog/frontend/app/courses/[id]/structure/_components/xp-premium-pills.tsx.ejs +44 -0
  244. package/hedhog/frontend/app/courses/[id]/structure/_data/use-course-content-overview.ts.ejs +54 -0
  245. package/hedhog/frontend/app/courses/[id]/structure/_data/use-course-structure-mutations.ts.ejs +52 -0
  246. package/hedhog/frontend/app/courses/[id]/structure/_data/use-course-xp-overview.ts.ejs +76 -0
  247. package/hedhog/frontend/app/courses/[id]/structure/_data/use-lesson-xp-map.ts.ejs +128 -0
  248. package/hedhog/frontend/app/courses/[id]/structure/_data/use-transcription-segments.ts.ejs +30 -0
  249. package/hedhog/frontend/app/courses/[id]/structure/_utils/xp-color-config.ts.ejs +115 -0
  250. package/hedhog/frontend/app/courses/_components/CourseDeleteDialog.tsx.ejs +223 -0
  251. package/hedhog/frontend/app/courses/_components/CourseRowActions.tsx.ejs +89 -0
  252. package/hedhog/frontend/app/courses/page.tsx.ejs +400 -230
  253. package/hedhog/frontend/app/enterprise/page.tsx.ejs +39 -63
  254. package/hedhog/frontend/app/exams/[id]/questions/page.tsx.ejs +53 -77
  255. package/hedhog/frontend/app/exams/page.tsx.ejs +54 -90
  256. package/hedhog/frontend/app/instructor-skills/page.tsx.ejs +23 -36
  257. package/hedhog/frontend/app/instructors/page.tsx.ejs +72 -81
  258. package/hedhog/frontend/app/paths/page.tsx.ejs +40 -68
  259. package/hedhog/frontend/app/training/page.tsx.ejs +40 -68
  260. package/hedhog/frontend/app/xp/areas/page.tsx.ejs +782 -0
  261. package/hedhog/frontend/app/xp/learning-types/page.tsx.ejs +690 -0
  262. package/hedhog/frontend/app/xp/skills/page.tsx.ejs +811 -0
  263. package/hedhog/frontend/messages/en.json +386 -3
  264. package/hedhog/frontend/messages/pt.json +386 -3
  265. package/hedhog/table/lesson_xp_map.yaml +50 -0
  266. package/hedhog/table/lesson_xp_segment.yaml +40 -0
  267. package/hedhog/table/lesson_xp_segment_area.yaml +24 -0
  268. package/hedhog/table/lesson_xp_segment_learning_type.yaml +24 -0
  269. package/hedhog/table/lesson_xp_segment_skill.yaml +24 -0
  270. package/hedhog/table/lms_bulk_upload_item.yaml +44 -0
  271. package/hedhog/table/lms_bulk_upload_session.yaml +42 -0
  272. package/hedhog/table/student_area_xp.yaml +30 -0
  273. package/hedhog/table/student_learning_type_xp.yaml +30 -0
  274. package/hedhog/table/student_skill_xp.yaml +30 -0
  275. package/hedhog/table/student_xp_event.yaml +34 -0
  276. package/hedhog/table/xp_area.yaml +39 -0
  277. package/hedhog/table/xp_learning_type.yaml +34 -0
  278. package/hedhog/table/xp_skill.yaml +39 -0
  279. package/package.json +8 -7
  280. package/src/bitcode-wallet/bitcode-wallet.service.ts +113 -0
  281. package/src/bitcode-wallet/dto/create-current-bitcode-wallet-transaction.dto.ts +32 -0
  282. package/src/course/course-audio-transcription.service.ts +58 -21
  283. package/src/course/course-lesson.controller.ts +6 -1
  284. package/src/course/course-structure.controller.ts +10 -0
  285. package/src/course/course-structure.service.ts +174 -1
  286. package/src/course/course-video-conversion.service.ts +113 -75
  287. package/src/course/course.controller.ts +22 -3
  288. package/src/course/course.module.ts +2 -0
  289. package/src/course/course.service.ts +847 -30
  290. package/src/course/dto/cleanup-course-storage.dto.ts +22 -0
  291. package/src/course/dto/cleanup-upload-history.dto.ts +26 -0
  292. package/src/course/dto/create-course-bulk-job.dto.ts +6 -0
  293. package/src/course/lms-bulk-upload-automation.service.ts +560 -0
  294. package/src/course/lms-bulk-upload-infra.service.ts +327 -0
  295. package/src/course/lms-bulk-upload.constants.ts +5 -0
  296. package/src/course/lms-bulk-upload.controller.ts +79 -3
  297. package/src/course/lms-bulk-upload.service.ts +1029 -204
  298. package/src/course/lms-setting.controller.ts +22 -6
  299. package/src/lesson-xp-map/dto/create-lesson-xp-map.dto.ts +17 -0
  300. package/src/lesson-xp-map/dto/create-lesson-xp-segment.dto.ts +102 -0
  301. package/src/lesson-xp-map/dto/review-lesson-xp-map.dto.ts +7 -0
  302. package/src/lesson-xp-map/dto/update-lesson-xp-map.dto.ts +36 -0
  303. package/src/lesson-xp-map/dto/update-lesson-xp-segment.dto.ts +78 -0
  304. package/src/lesson-xp-map/lesson-xp-ai-calculation.service.ts +396 -0
  305. package/src/lesson-xp-map/lesson-xp-map.controller.ts +116 -0
  306. package/src/lesson-xp-map/lesson-xp-map.module.ts +21 -0
  307. package/src/lesson-xp-map/lesson-xp-map.service.ts +442 -0
  308. package/src/lesson-xp-map/lesson-xp-segment.controller.ts +36 -0
  309. package/src/lesson-xp-map/lesson-xp-segment.service.ts +229 -0
  310. package/src/lms.module.ts +17 -2
  311. package/src/platforma/dto/update-profile.dto.ts +59 -0
  312. package/src/platforma/platforma.controller.ts +57 -2
  313. package/src/platforma/platforma.service.ts +268 -0
  314. package/src/student-xp/student-xp.controller.ts +76 -0
  315. package/src/student-xp/student-xp.module.ts +12 -0
  316. package/src/student-xp/student-xp.service.ts +236 -0
  317. package/src/xp-catalog/dto/create-xp-area.dto.ts +40 -0
  318. package/src/xp-catalog/dto/create-xp-learning-type.dto.ts +35 -0
  319. package/src/xp-catalog/dto/create-xp-skill.dto.ts +35 -0
  320. package/src/xp-catalog/dto/update-xp-area.dto.ts +43 -0
  321. package/src/xp-catalog/dto/update-xp-learning-type.dto.ts +38 -0
  322. package/src/xp-catalog/dto/update-xp-skill.dto.ts +38 -0
  323. package/src/xp-catalog/xp-area.controller.ts +64 -0
  324. package/src/xp-catalog/xp-area.service.ts +196 -0
  325. package/src/xp-catalog/xp-catalog.module.ts +16 -0
  326. package/src/xp-catalog/xp-learning-type.controller.ts +59 -0
  327. package/src/xp-catalog/xp-learning-type.service.ts +170 -0
  328. package/src/xp-catalog/xp-skill.controller.ts +71 -0
  329. package/src/xp-catalog/xp-skill.service.ts +205 -0
@@ -0,0 +1,782 @@
1
+ 'use client';
2
+
3
+ import {
4
+ EmptyState,
5
+ Page,
6
+ PageHeader,
7
+ PaginationFooter,
8
+ SearchBar,
9
+ } from '@/components/entity-list';
10
+ import {
11
+ AlertDialog,
12
+ AlertDialogAction,
13
+ AlertDialogCancel,
14
+ AlertDialogContent,
15
+ AlertDialogDescription,
16
+ AlertDialogHeader,
17
+ AlertDialogTitle,
18
+ } from '@/components/ui/alert-dialog';
19
+ import { Badge } from '@/components/ui/badge';
20
+ import { Button } from '@/components/ui/button';
21
+ import {
22
+ Form,
23
+ FormControl,
24
+ FormField,
25
+ FormItem,
26
+ FormLabel,
27
+ FormMessage,
28
+ } from '@/components/ui/form';
29
+ import { IconActionGroup } from '@/components/ui/icon-action-group';
30
+ import { Input } from '@/components/ui/input';
31
+ import { KpiCardsGrid } from '@/components/ui/kpi-cards-grid';
32
+ import { ResizableSheetContent } from '@/components/ui/resizable-sheet-content';
33
+ import {
34
+ Select,
35
+ SelectContent,
36
+ SelectItem,
37
+ SelectTrigger,
38
+ SelectValue,
39
+ } from '@/components/ui/select';
40
+ import { Sheet, SheetHeader, SheetTitle } from '@/components/ui/sheet';
41
+ import { Skeleton } from '@/components/ui/skeleton';
42
+ import {
43
+ Table,
44
+ TableBody,
45
+ TableCell,
46
+ TableHead,
47
+ TableHeader,
48
+ TableRow,
49
+ } from '@/components/ui/table';
50
+ import { Textarea } from '@/components/ui/textarea';
51
+ import { usePersistedPageSize } from '@/hooks/use-persisted-page-size';
52
+ import { cn } from '@/lib/utils';
53
+ import { useApp, useQuery } from '@hed-hog/next-app-provider';
54
+ import { zodResolver } from '@hookform/resolvers/zod';
55
+ import {
56
+ CheckCircle2,
57
+ Layers,
58
+ Pencil,
59
+ Plus,
60
+ Trash2,
61
+ XCircle,
62
+ } from 'lucide-react';
63
+ import { useTranslations } from 'next-intl';
64
+ import { useEffect, useState } from 'react';
65
+ import { useForm } from 'react-hook-form';
66
+ import { toast } from 'sonner';
67
+ import * as z from 'zod';
68
+
69
+ // ─── Types ────────────────────────────────────────────────────────────────────
70
+
71
+ type XpArea = {
72
+ id: number;
73
+ slug: string;
74
+ status: 'active' | 'inactive';
75
+ sortOrder: number;
76
+ color: string | null;
77
+ iconSlug: string | null;
78
+ name: string;
79
+ };
80
+
81
+ type XpAreaDetail = {
82
+ id: number;
83
+ slug: string;
84
+ status: 'active' | 'inactive';
85
+ sortOrder: number;
86
+ color: string | null;
87
+ iconSlug: string | null;
88
+ translations: Record<string, { name: string; description: string | null }>;
89
+ };
90
+
91
+ type PaginatedResult<T> = {
92
+ data: T[];
93
+ total: number;
94
+ page: number;
95
+ pageSize: number;
96
+ lastPage?: number;
97
+ };
98
+
99
+ // ─── Zod schema ───────────────────────────────────────────────────────────────
100
+
101
+ const createSchema = (t: ReturnType<typeof useTranslations>) =>
102
+ z.object({
103
+ slug: z
104
+ .string()
105
+ .min(1, t('form.validation.required'))
106
+ .max(100, t('form.validation.max100'))
107
+ .regex(/^[a-z0-9-]+$/, t('form.validation.slugPattern')),
108
+ namePt: z.string().min(1, t('form.validation.required')),
109
+ nameEn: z.string().min(1, t('form.validation.required')),
110
+ descriptionPt: z.string().optional(),
111
+ descriptionEn: z.string().optional(),
112
+ color: z.string().optional(),
113
+ iconSlug: z.string().max(100).optional(),
114
+ sortOrder: z.coerce.number().int().min(0).default(0),
115
+ status: z.enum(['active', 'inactive']),
116
+ });
117
+
118
+ type AreaFormValues = z.infer<ReturnType<typeof createSchema>>;
119
+
120
+ // ─── AreaFormSheet ─────────────────────────────────────────────────────────────
121
+
122
+ interface AreaFormSheetProps {
123
+ open: boolean;
124
+ onOpenChange: (open: boolean) => void;
125
+ areaToEdit: XpArea | null;
126
+ onSaved: () => void;
127
+ }
128
+
129
+ function AreaFormSheet({
130
+ open,
131
+ onOpenChange,
132
+ areaToEdit,
133
+ onSaved,
134
+ }: AreaFormSheetProps) {
135
+ const { request } = useApp();
136
+ const t = useTranslations('lms.XpAreasPage');
137
+ const [isSaving, setIsSaving] = useState(false);
138
+ const [isLoadingDetail, setIsLoadingDetail] = useState(false);
139
+
140
+ const form = useForm<AreaFormValues>({
141
+ resolver: zodResolver(createSchema(t)),
142
+ defaultValues: {
143
+ slug: '',
144
+ namePt: '',
145
+ nameEn: '',
146
+ descriptionPt: '',
147
+ descriptionEn: '',
148
+ color: '#3b82f6',
149
+ iconSlug: '',
150
+ sortOrder: 0,
151
+ status: 'active',
152
+ },
153
+ });
154
+
155
+ useEffect(() => {
156
+ if (!open) return;
157
+
158
+ if (areaToEdit) {
159
+ setIsLoadingDetail(true);
160
+ request<{ data: XpAreaDetail }>({
161
+ url: `/lms/xp/areas/${areaToEdit.id}`,
162
+ method: 'GET',
163
+ })
164
+ .then((res) => {
165
+ const d = res.data as unknown as XpAreaDetail;
166
+ form.reset({
167
+ slug: d.slug,
168
+ namePt: d.translations?.pt?.name ?? '',
169
+ nameEn: d.translations?.en?.name ?? '',
170
+ descriptionPt: d.translations?.pt?.description ?? '',
171
+ descriptionEn: d.translations?.en?.description ?? '',
172
+ color: d.color ?? '#3b82f6',
173
+ iconSlug: d.iconSlug ?? '',
174
+ sortOrder: d.sortOrder ?? 0,
175
+ status: d.status,
176
+ });
177
+ })
178
+ .catch(() => toast.error(t('messages.saveError')))
179
+ .finally(() => setIsLoadingDetail(false));
180
+ } else {
181
+ form.reset({
182
+ slug: '',
183
+ namePt: '',
184
+ nameEn: '',
185
+ descriptionPt: '',
186
+ descriptionEn: '',
187
+ color: '#3b82f6',
188
+ iconSlug: '',
189
+ sortOrder: 0,
190
+ status: 'active',
191
+ });
192
+ }
193
+ }, [open, areaToEdit]);
194
+
195
+ const onSubmit = async (values: AreaFormValues) => {
196
+ try {
197
+ setIsSaving(true);
198
+ const payload = {
199
+ slug: values.slug,
200
+ namePt: values.namePt,
201
+ nameEn: values.nameEn,
202
+ descriptionPt: values.descriptionPt || null,
203
+ descriptionEn: values.descriptionEn || null,
204
+ color: values.color || null,
205
+ iconSlug: values.iconSlug || null,
206
+ sortOrder: values.sortOrder,
207
+ status: values.status,
208
+ };
209
+
210
+ if (areaToEdit) {
211
+ await request({
212
+ url: `/lms/xp/areas/${areaToEdit.id}`,
213
+ method: 'PATCH',
214
+ data: payload,
215
+ });
216
+ toast.success(t('messages.updateSuccess'));
217
+ } else {
218
+ await request({
219
+ url: '/lms/xp/areas',
220
+ method: 'POST',
221
+ data: payload,
222
+ });
223
+ toast.success(t('messages.createSuccess'));
224
+ }
225
+
226
+ onSaved();
227
+ onOpenChange(false);
228
+ } catch {
229
+ toast.error(t('messages.saveError'));
230
+ } finally {
231
+ setIsSaving(false);
232
+ }
233
+ };
234
+
235
+ return (
236
+ <Sheet open={open} onOpenChange={onOpenChange}>
237
+ <ResizableSheetContent
238
+ sheetId="lms-xp-areas-form-sheet"
239
+ defaultWidth={620}
240
+ minWidth={420}
241
+ maxWidth={900}
242
+ className="overflow-y-auto"
243
+ >
244
+ <SheetHeader>
245
+ <SheetTitle>
246
+ {areaToEdit ? t('sheet.editTitle') : t('sheet.createTitle')}
247
+ </SheetTitle>
248
+ </SheetHeader>
249
+
250
+ {isLoadingDetail ? (
251
+ <div className="space-y-3 p-4">
252
+ {Array.from({ length: 5 }).map((_, i) => (
253
+ <Skeleton key={i} className="h-10 w-full" />
254
+ ))}
255
+ </div>
256
+ ) : (
257
+ <Form {...form}>
258
+ <form
259
+ onSubmit={form.handleSubmit(onSubmit)}
260
+ className="flex flex-col gap-4 px-4 py-4"
261
+ >
262
+ <div className="grid grid-cols-2 gap-4">
263
+ <FormField
264
+ control={form.control}
265
+ name="slug"
266
+ render={({ field }) => (
267
+ <FormItem>
268
+ <FormLabel>{t('form.slug')}</FormLabel>
269
+ <FormControl>
270
+ <Input
271
+ placeholder={t('form.slugPlaceholder')}
272
+ {...field}
273
+ />
274
+ </FormControl>
275
+ <FormMessage />
276
+ </FormItem>
277
+ )}
278
+ />
279
+
280
+ <FormField
281
+ control={form.control}
282
+ name="status"
283
+ render={({ field }) => (
284
+ <FormItem>
285
+ <FormLabel>{t('form.status')}</FormLabel>
286
+ <Select
287
+ onValueChange={field.onChange}
288
+ value={field.value}
289
+ >
290
+ <FormControl>
291
+ <SelectTrigger className="w-full">
292
+ <SelectValue
293
+ placeholder={t('form.statusPlaceholder')}
294
+ />
295
+ </SelectTrigger>
296
+ </FormControl>
297
+ <SelectContent>
298
+ <SelectItem value="active">
299
+ {t('status.active')}
300
+ </SelectItem>
301
+ <SelectItem value="inactive">
302
+ {t('status.inactive')}
303
+ </SelectItem>
304
+ </SelectContent>
305
+ </Select>
306
+ <FormMessage />
307
+ </FormItem>
308
+ )}
309
+ />
310
+ </div>
311
+
312
+ <div className="grid grid-cols-2 gap-4">
313
+ <FormField
314
+ control={form.control}
315
+ name="namePt"
316
+ render={({ field }) => (
317
+ <FormItem>
318
+ <FormLabel>{t('form.namePt')}</FormLabel>
319
+ <FormControl>
320
+ <Input
321
+ placeholder={t('form.namePtPlaceholder')}
322
+ {...field}
323
+ />
324
+ </FormControl>
325
+ <FormMessage />
326
+ </FormItem>
327
+ )}
328
+ />
329
+
330
+ <FormField
331
+ control={form.control}
332
+ name="nameEn"
333
+ render={({ field }) => (
334
+ <FormItem>
335
+ <FormLabel>{t('form.nameEn')}</FormLabel>
336
+ <FormControl>
337
+ <Input
338
+ placeholder={t('form.nameEnPlaceholder')}
339
+ {...field}
340
+ />
341
+ </FormControl>
342
+ <FormMessage />
343
+ </FormItem>
344
+ )}
345
+ />
346
+ </div>
347
+
348
+ <div className="grid grid-cols-2 gap-4">
349
+ <FormField
350
+ control={form.control}
351
+ name="descriptionPt"
352
+ render={({ field }) => (
353
+ <FormItem>
354
+ <FormLabel>{t('form.descriptionPt')}</FormLabel>
355
+ <FormControl>
356
+ <Textarea
357
+ rows={3}
358
+ placeholder={t('form.descriptionPtPlaceholder')}
359
+ {...field}
360
+ />
361
+ </FormControl>
362
+ <FormMessage />
363
+ </FormItem>
364
+ )}
365
+ />
366
+
367
+ <FormField
368
+ control={form.control}
369
+ name="descriptionEn"
370
+ render={({ field }) => (
371
+ <FormItem>
372
+ <FormLabel>{t('form.descriptionEn')}</FormLabel>
373
+ <FormControl>
374
+ <Textarea
375
+ rows={3}
376
+ placeholder={t('form.descriptionEnPlaceholder')}
377
+ {...field}
378
+ />
379
+ </FormControl>
380
+ <FormMessage />
381
+ </FormItem>
382
+ )}
383
+ />
384
+ </div>
385
+
386
+ <div className="grid grid-cols-2 gap-4">
387
+ <FormField
388
+ control={form.control}
389
+ name="color"
390
+ render={({ field }) => (
391
+ <FormItem>
392
+ <FormLabel>{t('form.color')}</FormLabel>
393
+ <FormControl>
394
+ <div className="flex items-center gap-2">
395
+ <input
396
+ type="color"
397
+ value={field.value || '#3b82f6'}
398
+ onChange={(e) => field.onChange(e.target.value)}
399
+ className="h-9 w-10 cursor-pointer rounded border p-0.5"
400
+ />
401
+ <Input
402
+ placeholder="#3b82f6"
403
+ {...field}
404
+ className="flex-1"
405
+ />
406
+ </div>
407
+ </FormControl>
408
+ <FormMessage />
409
+ </FormItem>
410
+ )}
411
+ />
412
+
413
+ <FormField
414
+ control={form.control}
415
+ name="iconSlug"
416
+ render={({ field }) => (
417
+ <FormItem>
418
+ <FormLabel>{t('form.iconSlug')}</FormLabel>
419
+ <FormControl>
420
+ <Input
421
+ placeholder={t('form.iconSlugPlaceholder')}
422
+ {...field}
423
+ />
424
+ </FormControl>
425
+ <FormMessage />
426
+ </FormItem>
427
+ )}
428
+ />
429
+ </div>
430
+
431
+ <FormField
432
+ control={form.control}
433
+ name="sortOrder"
434
+ render={({ field }) => (
435
+ <FormItem className="max-w-[160px]">
436
+ <FormLabel>{t('form.sortOrder')}</FormLabel>
437
+ <FormControl>
438
+ <Input type="number" min={0} {...field} />
439
+ </FormControl>
440
+ <FormMessage />
441
+ </FormItem>
442
+ )}
443
+ />
444
+
445
+ <div className="flex justify-end gap-2 pt-2">
446
+ <Button
447
+ type="button"
448
+ variant="outline"
449
+ onClick={() => onOpenChange(false)}
450
+ disabled={isSaving}
451
+ >
452
+ {t('actions.cancel')}
453
+ </Button>
454
+ <Button type="submit" disabled={isSaving}>
455
+ {isSaving ? t('actions.saving') : t('actions.save')}
456
+ </Button>
457
+ </div>
458
+ </form>
459
+ </Form>
460
+ )}
461
+ </ResizableSheetContent>
462
+ </Sheet>
463
+ );
464
+ }
465
+
466
+ // ─── Page ─────────────────────────────────────────────────────────────────────
467
+
468
+ export default function XpAreasPage() {
469
+ const { request } = useApp();
470
+ const t = useTranslations('lms.XpAreasPage');
471
+
472
+ const [page, setPage] = useState(1);
473
+ const [pageSize, setPageSize] = usePersistedPageSize({
474
+ storageKey: 'pagination:global:pageSize',
475
+ defaultValue: 12,
476
+ allowedValues: [6, 12, 24, 48],
477
+ });
478
+ const [searchInput, setSearchInput] = useState('');
479
+ const [debouncedSearch, setDebouncedSearch] = useState('');
480
+ const [sheetOpen, setSheetOpen] = useState(false);
481
+ const [areaToEdit, setAreaToEdit] = useState<XpArea | null>(null);
482
+ const [deleteDialogOpen, setDeleteDialogOpen] = useState(false);
483
+ const [areaToDelete, setAreaToDelete] = useState<XpArea | null>(null);
484
+ const [isDeleting, setIsDeleting] = useState(false);
485
+
486
+ useEffect(() => {
487
+ const timeout = setTimeout(() => {
488
+ setDebouncedSearch(searchInput.trim());
489
+ setPage(1);
490
+ }, 300);
491
+ return () => clearTimeout(timeout);
492
+ }, [searchInput]);
493
+
494
+ const { data: statsData, refetch: refetchStats } = useQuery<{
495
+ total: number;
496
+ active: number;
497
+ inactive: number;
498
+ }>({
499
+ queryKey: ['lms-xp-areas-stats'],
500
+ queryFn: async () => {
501
+ const response = await request<{
502
+ total: number;
503
+ active: number;
504
+ inactive: number;
505
+ }>({
506
+ url: '/lms/xp/areas/stats',
507
+ method: 'GET',
508
+ });
509
+ return response.data;
510
+ },
511
+ });
512
+
513
+ const {
514
+ data: paginate = { data: [], total: 0, page: 1, pageSize, lastPage: 1 },
515
+ isLoading,
516
+ refetch: refetchList,
517
+ } = useQuery<PaginatedResult<XpArea>>({
518
+ queryKey: ['lms-xp-areas', page, pageSize, debouncedSearch],
519
+ queryFn: async () => {
520
+ const params = new URLSearchParams({
521
+ page: String(page),
522
+ pageSize: String(pageSize),
523
+ });
524
+ if (debouncedSearch) params.set('search', debouncedSearch);
525
+ const response = await request<PaginatedResult<XpArea>>({
526
+ url: `/lms/xp/areas?${params.toString()}`,
527
+ method: 'GET',
528
+ });
529
+ return response.data;
530
+ },
531
+ placeholderData: (prev) =>
532
+ prev ?? { data: [], total: 0, page: 1, pageSize, lastPage: 1 },
533
+ });
534
+
535
+ const totalPages = Math.max(
536
+ 1,
537
+ (paginate.lastPage ?? Math.ceil((paginate.total || 0) / pageSize)) || 1
538
+ );
539
+
540
+ useEffect(() => {
541
+ if (page > totalPages) setPage(totalPages);
542
+ }, [page, totalPages]);
543
+
544
+ const openCreateSheet = () => {
545
+ setAreaToEdit(null);
546
+ setSheetOpen(true);
547
+ };
548
+
549
+ const openEditSheet = (area: XpArea) => {
550
+ setAreaToEdit(area);
551
+ setSheetOpen(true);
552
+ };
553
+
554
+ const handleDeleteConfirm = async () => {
555
+ if (!areaToDelete) return;
556
+ try {
557
+ setIsDeleting(true);
558
+ await request({
559
+ url: `/lms/xp/areas/${areaToDelete.id}`,
560
+ method: 'DELETE',
561
+ });
562
+ toast.success(t('messages.deleteSuccess'));
563
+ setDeleteDialogOpen(false);
564
+ setAreaToDelete(null);
565
+ await Promise.all([refetchList(), refetchStats()]);
566
+ } catch {
567
+ toast.error(t('messages.deleteError'));
568
+ } finally {
569
+ setIsDeleting(false);
570
+ }
571
+ };
572
+
573
+ return (
574
+ <Page>
575
+ <PageHeader
576
+ breadcrumbs={[
577
+ { label: t('breadcrumbs.home'), href: '/' },
578
+ { label: t('breadcrumbs.lms'), href: '/lms' },
579
+ { label: t('breadcrumbs.xp'), href: '/lms/xp' },
580
+ { label: t('breadcrumbs.current') },
581
+ ]}
582
+ title={t('title')}
583
+ description={t('description')}
584
+ actions={[
585
+ {
586
+ label: t('actions.create'),
587
+ onClick: openCreateSheet,
588
+ icon: <Plus className="h-4 w-4" />,
589
+ },
590
+ ]}
591
+ />
592
+
593
+ <KpiCardsGrid
594
+ columns={3}
595
+ items={[
596
+ {
597
+ key: 'total',
598
+ title: t('kpis.total.label'),
599
+ value: statsData?.total ?? 0,
600
+ description: t('kpis.total.description'),
601
+ icon: Layers,
602
+ accentClassName: 'from-sky-500/20 via-blue-400/10 to-transparent',
603
+ iconContainerClassName: 'bg-sky-50 text-sky-700',
604
+ layout: 'compact',
605
+ },
606
+ {
607
+ key: 'active',
608
+ title: t('kpis.active.label'),
609
+ value: statsData?.active ?? 0,
610
+ description: t('kpis.active.description'),
611
+ icon: CheckCircle2,
612
+ accentClassName:
613
+ 'from-emerald-500/20 via-green-400/10 to-transparent',
614
+ iconContainerClassName: 'bg-emerald-50 text-emerald-700',
615
+ layout: 'compact',
616
+ },
617
+ {
618
+ key: 'inactive',
619
+ title: t('kpis.inactive.label'),
620
+ value: statsData?.inactive ?? 0,
621
+ description: t('kpis.inactive.description'),
622
+ icon: XCircle,
623
+ accentClassName:
624
+ 'from-amber-500/20 via-orange-400/10 to-transparent',
625
+ iconContainerClassName: 'bg-amber-50 text-amber-700',
626
+ layout: 'compact',
627
+ },
628
+ ]}
629
+ />
630
+
631
+ <SearchBar
632
+ searchQuery={searchInput}
633
+ onSearchChange={setSearchInput}
634
+ onSearch={() => setPage(1)}
635
+ placeholder={t('filters.searchPlaceholder')}
636
+ />
637
+
638
+ {isLoading ? (
639
+ <div className="space-y-3 p-4">
640
+ {Array.from({ length: 6 }).map((_, i) => (
641
+ <Skeleton key={i} className="h-14 w-full" />
642
+ ))}
643
+ </div>
644
+ ) : paginate.data.length === 0 ? (
645
+ <EmptyState
646
+ icon={<Layers className="h-12 w-12" />}
647
+ title={t('empty.title')}
648
+ description={t('empty.description')}
649
+ actionLabel={t('actions.create')}
650
+ actionIcon={<Plus className="mr-2 h-4 w-4" />}
651
+ onAction={openCreateSheet}
652
+ />
653
+ ) : (
654
+ <div className="overflow-x-auto">
655
+ <Table>
656
+ <TableHeader>
657
+ <TableRow>
658
+ <TableHead className="w-8">{t('table.color')}</TableHead>
659
+ <TableHead>{t('table.name')}</TableHead>
660
+ <TableHead>{t('table.slug')}</TableHead>
661
+ <TableHead>{t('table.status')}</TableHead>
662
+ <TableHead className="w-10" />
663
+ </TableRow>
664
+ </TableHeader>
665
+ <TableBody>
666
+ {paginate.data.map((area) => (
667
+ <TableRow
668
+ key={area.id}
669
+ className="cursor-pointer"
670
+ onDoubleClick={() => openEditSheet(area)}
671
+ >
672
+ <TableCell>
673
+ {area.color ? (
674
+ <div
675
+ className="h-5 w-5 rounded-full border border-border"
676
+ style={{ backgroundColor: area.color }}
677
+ title={area.color}
678
+ />
679
+ ) : (
680
+ <div className="h-5 w-5 rounded-full border border-dashed border-muted-foreground/30" />
681
+ )}
682
+ </TableCell>
683
+ <TableCell className="font-medium">{area.name}</TableCell>
684
+ <TableCell>
685
+ <span className="font-mono text-sm text-muted-foreground">
686
+ {area.slug}
687
+ </span>
688
+ </TableCell>
689
+ <TableCell>
690
+ <Badge
691
+ variant="outline"
692
+ className={cn(
693
+ 'border px-2.5 py-1 text-xs font-medium',
694
+ area.status === 'active'
695
+ ? 'border-green-500/20 bg-green-500/10 text-green-600'
696
+ : 'border-gray-500/20 bg-gray-500/10 text-gray-600'
697
+ )}
698
+ >
699
+ {area.status === 'active'
700
+ ? t('status.active')
701
+ : t('status.inactive')}
702
+ </Badge>
703
+ </TableCell>
704
+ <TableCell>
705
+ <IconActionGroup
706
+ className="ml-auto"
707
+ actions={[
708
+ {
709
+ key: 'edit',
710
+ label: t('actions.edit'),
711
+ icon: <Pencil className="size-4" />,
712
+ onClick: () => openEditSheet(area),
713
+ },
714
+ {
715
+ key: 'delete',
716
+ label: t('actions.delete'),
717
+ icon: <Trash2 className="size-4" />,
718
+ onClick: () => {
719
+ setAreaToDelete(area);
720
+ setDeleteDialogOpen(true);
721
+ },
722
+ destructive: true,
723
+ },
724
+ ]}
725
+ />
726
+ </TableCell>
727
+ </TableRow>
728
+ ))}
729
+ </TableBody>
730
+ </Table>
731
+ </div>
732
+ )}
733
+
734
+ <PaginationFooter
735
+ currentPage={page}
736
+ pageSize={pageSize}
737
+ totalItems={paginate.total}
738
+ onPageChange={setPage}
739
+ onPageSizeChange={(next) => {
740
+ setPageSize(next);
741
+ setPage(1);
742
+ }}
743
+ pageSizeOptions={[6, 12, 24, 48]}
744
+ />
745
+
746
+ <AreaFormSheet
747
+ open={sheetOpen}
748
+ onOpenChange={setSheetOpen}
749
+ areaToEdit={areaToEdit}
750
+ onSaved={() => {
751
+ refetchList();
752
+ refetchStats();
753
+ }}
754
+ />
755
+
756
+ <AlertDialog open={deleteDialogOpen} onOpenChange={setDeleteDialogOpen}>
757
+ <AlertDialogContent>
758
+ <AlertDialogHeader>
759
+ <AlertDialogTitle>{t('deleteDialog.title')}</AlertDialogTitle>
760
+ <AlertDialogDescription>
761
+ {t('deleteDialog.description', {
762
+ slug: areaToDelete?.slug ?? '',
763
+ })}
764
+ </AlertDialogDescription>
765
+ </AlertDialogHeader>
766
+ <div className="flex justify-end gap-2">
767
+ <AlertDialogCancel disabled={isDeleting}>
768
+ {t('actions.cancel')}
769
+ </AlertDialogCancel>
770
+ <AlertDialogAction
771
+ onClick={handleDeleteConfirm}
772
+ disabled={isDeleting}
773
+ className="bg-red-600 hover:bg-red-700"
774
+ >
775
+ {isDeleting ? t('actions.deleting') : t('actions.delete')}
776
+ </AlertDialogAction>
777
+ </div>
778
+ </AlertDialogContent>
779
+ </AlertDialog>
780
+ </Page>
781
+ );
782
+ }