@hed-hog/lms 0.0.347 → 0.0.350

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 (414) hide show
  1. package/dist/achievement/achievement.controller.d.ts +62 -0
  2. package/dist/achievement/achievement.controller.d.ts.map +1 -0
  3. package/dist/achievement/achievement.controller.js +90 -0
  4. package/dist/achievement/achievement.controller.js.map +1 -0
  5. package/dist/achievement/achievement.mcp-tools.d.ts +19 -0
  6. package/dist/achievement/achievement.mcp-tools.d.ts.map +1 -0
  7. package/dist/achievement/achievement.mcp-tools.js +157 -0
  8. package/dist/achievement/achievement.mcp-tools.js.map +1 -0
  9. package/dist/achievement/achievement.module.d.ts +3 -0
  10. package/dist/achievement/achievement.module.d.ts.map +1 -0
  11. package/dist/achievement/achievement.module.js +26 -0
  12. package/dist/achievement/achievement.module.js.map +1 -0
  13. package/dist/achievement/achievement.service.d.ts +72 -0
  14. package/dist/achievement/achievement.service.d.ts.map +1 -0
  15. package/dist/achievement/achievement.service.js +200 -0
  16. package/dist/achievement/achievement.service.js.map +1 -0
  17. package/dist/achievement/dto/create-achievement.dto.d.ts +12 -0
  18. package/dist/achievement/dto/create-achievement.dto.d.ts.map +1 -0
  19. package/dist/achievement/dto/create-achievement.dto.js +60 -0
  20. package/dist/achievement/dto/create-achievement.dto.js.map +1 -0
  21. package/dist/achievement/dto/update-achievement.dto.d.ts +11 -0
  22. package/dist/achievement/dto/update-achievement.dto.d.ts.map +1 -0
  23. package/dist/achievement/dto/update-achievement.dto.js +57 -0
  24. package/dist/achievement/dto/update-achievement.dto.js.map +1 -0
  25. package/dist/bitcode-wallet/bitcode-wallet.controller.d.ts +114 -0
  26. package/dist/bitcode-wallet/bitcode-wallet.controller.d.ts.map +1 -0
  27. package/dist/bitcode-wallet/bitcode-wallet.controller.js +102 -0
  28. package/dist/bitcode-wallet/bitcode-wallet.controller.js.map +1 -0
  29. package/dist/bitcode-wallet/bitcode-wallet.mcp-tools.d.ts +25 -0
  30. package/dist/bitcode-wallet/bitcode-wallet.mcp-tools.d.ts.map +1 -0
  31. package/dist/bitcode-wallet/bitcode-wallet.mcp-tools.js +160 -0
  32. package/dist/bitcode-wallet/bitcode-wallet.mcp-tools.js.map +1 -0
  33. package/dist/bitcode-wallet/bitcode-wallet.module.d.ts +3 -0
  34. package/dist/bitcode-wallet/bitcode-wallet.module.d.ts.map +1 -0
  35. package/dist/bitcode-wallet/bitcode-wallet.module.js +26 -0
  36. package/dist/bitcode-wallet/bitcode-wallet.module.js.map +1 -0
  37. package/dist/bitcode-wallet/bitcode-wallet.service.d.ts +127 -0
  38. package/dist/bitcode-wallet/bitcode-wallet.service.d.ts.map +1 -0
  39. package/dist/bitcode-wallet/bitcode-wallet.service.js +264 -0
  40. package/dist/bitcode-wallet/bitcode-wallet.service.js.map +1 -0
  41. package/dist/bitcode-wallet/dto/create-bitcode-wallet-transaction.dto.d.ts +8 -0
  42. package/dist/bitcode-wallet/dto/create-bitcode-wallet-transaction.dto.d.ts.map +1 -0
  43. package/dist/bitcode-wallet/dto/create-bitcode-wallet-transaction.dto.js +33 -0
  44. package/dist/bitcode-wallet/dto/create-bitcode-wallet-transaction.dto.js.map +1 -0
  45. package/dist/bitcode-wallet/dto/create-bitcode-wallet.dto.d.ts +4 -0
  46. package/dist/bitcode-wallet/dto/create-bitcode-wallet.dto.d.ts.map +1 -0
  47. package/dist/bitcode-wallet/dto/create-bitcode-wallet.dto.js +22 -0
  48. package/dist/bitcode-wallet/dto/create-bitcode-wallet.dto.js.map +1 -0
  49. package/dist/bitcode-wallet/dto/update-bitcode-wallet-transaction.dto.d.ts +7 -0
  50. package/dist/bitcode-wallet/dto/update-bitcode-wallet-transaction.dto.d.ts.map +1 -0
  51. package/dist/bitcode-wallet/dto/update-bitcode-wallet-transaction.dto.js +35 -0
  52. package/dist/bitcode-wallet/dto/update-bitcode-wallet-transaction.dto.js.map +1 -0
  53. package/dist/bitcode-wallet/dto/update-bitcode-wallet.dto.d.ts +4 -0
  54. package/dist/bitcode-wallet/dto/update-bitcode-wallet.dto.d.ts.map +1 -0
  55. package/dist/bitcode-wallet/dto/update-bitcode-wallet.dto.js +23 -0
  56. package/dist/bitcode-wallet/dto/update-bitcode-wallet.dto.js.map +1 -0
  57. package/dist/certificate/certificate.controller.d.ts +22 -0
  58. package/dist/certificate/certificate.controller.d.ts.map +1 -1
  59. package/dist/certificate/certificate.controller.js +12 -0
  60. package/dist/certificate/certificate.controller.js.map +1 -1
  61. package/dist/certificate/certificate.mcp-tools.d.ts +24 -0
  62. package/dist/certificate/certificate.mcp-tools.d.ts.map +1 -0
  63. package/dist/certificate/certificate.mcp-tools.js +188 -0
  64. package/dist/certificate/certificate.mcp-tools.js.map +1 -0
  65. package/dist/certificate/certificate.module.d.ts.map +1 -1
  66. package/dist/certificate/certificate.module.js +2 -1
  67. package/dist/certificate/certificate.module.js.map +1 -1
  68. package/dist/certificate/certificate.service.d.ts +25 -2
  69. package/dist/certificate/certificate.service.d.ts.map +1 -1
  70. package/dist/certificate/certificate.service.js +87 -2
  71. package/dist/certificate/certificate.service.js.map +1 -1
  72. package/dist/certificate/dto/update-certificate-public-access.dto.d.ts +4 -0
  73. package/dist/certificate/dto/update-certificate-public-access.dto.d.ts.map +1 -0
  74. package/dist/certificate/dto/update-certificate-public-access.dto.js +21 -0
  75. package/dist/certificate/dto/update-certificate-public-access.dto.js.map +1 -0
  76. package/dist/class-group/class-group.controller.d.ts +3 -3
  77. package/dist/class-group/class-group.mcp-tools.d.ts +87 -0
  78. package/dist/class-group/class-group.mcp-tools.d.ts.map +1 -0
  79. package/dist/class-group/class-group.mcp-tools.js +553 -0
  80. package/dist/class-group/class-group.mcp-tools.js.map +1 -0
  81. package/dist/class-group/class-group.module.d.ts.map +1 -1
  82. package/dist/class-group/class-group.module.js +2 -1
  83. package/dist/class-group/class-group.module.js.map +1 -1
  84. package/dist/class-group/class-group.service.d.ts +6 -4
  85. package/dist/class-group/class-group.service.d.ts.map +1 -1
  86. package/dist/class-group/class-group.service.js +45 -2
  87. package/dist/class-group/class-group.service.js.map +1 -1
  88. package/dist/course/course-operations-integration.service.d.ts +40 -0
  89. package/dist/course/course-operations-integration.service.d.ts.map +1 -0
  90. package/dist/course/course-operations-integration.service.js +372 -0
  91. package/dist/course/course-operations-integration.service.js.map +1 -0
  92. package/dist/course/course-structure.controller.d.ts +43 -4
  93. package/dist/course/course-structure.controller.d.ts.map +1 -1
  94. package/dist/course/course-structure.controller.js +22 -0
  95. package/dist/course/course-structure.controller.js.map +1 -1
  96. package/dist/course/course-structure.service.d.ts +42 -1
  97. package/dist/course/course-structure.service.d.ts.map +1 -1
  98. package/dist/course/course-structure.service.js +199 -32
  99. package/dist/course/course-structure.service.js.map +1 -1
  100. package/dist/course/course.controller.d.ts +12 -0
  101. package/dist/course/course.controller.d.ts.map +1 -1
  102. package/dist/course/course.mcp-tools.d.ts +90 -0
  103. package/dist/course/course.mcp-tools.d.ts.map +1 -0
  104. package/dist/course/course.mcp-tools.js +520 -0
  105. package/dist/course/course.mcp-tools.js.map +1 -0
  106. package/dist/course/course.module.d.ts.map +1 -1
  107. package/dist/course/course.module.js +8 -1
  108. package/dist/course/course.module.js.map +1 -1
  109. package/dist/course/course.service.d.ts +15 -1
  110. package/dist/course/course.service.d.ts.map +1 -1
  111. package/dist/course/course.service.js +70 -35
  112. package/dist/course/course.service.js.map +1 -1
  113. package/dist/course/dto/create-course.dto.d.ts +1 -0
  114. package/dist/course/dto/create-course.dto.d.ts.map +1 -1
  115. package/dist/course/dto/create-course.dto.js +7 -0
  116. package/dist/course/dto/create-course.dto.js.map +1 -1
  117. package/dist/course/dto/update-course-resources.dto.d.ts +11 -0
  118. package/dist/course/dto/update-course-resources.dto.d.ts.map +1 -0
  119. package/dist/course/dto/update-course-resources.dto.js +51 -0
  120. package/dist/course/dto/update-course-resources.dto.js.map +1 -0
  121. package/dist/course-lesson-discussion/course-lesson-discussion.controller.d.ts +23 -0
  122. package/dist/course-lesson-discussion/course-lesson-discussion.controller.d.ts.map +1 -0
  123. package/dist/course-lesson-discussion/course-lesson-discussion.controller.js +78 -0
  124. package/dist/course-lesson-discussion/course-lesson-discussion.controller.js.map +1 -0
  125. package/dist/course-lesson-discussion/course-lesson-discussion.mcp-tools.d.ts +22 -0
  126. package/dist/course-lesson-discussion/course-lesson-discussion.mcp-tools.d.ts.map +1 -0
  127. package/dist/course-lesson-discussion/course-lesson-discussion.mcp-tools.js +120 -0
  128. package/dist/course-lesson-discussion/course-lesson-discussion.mcp-tools.js.map +1 -0
  129. package/dist/course-lesson-discussion/course-lesson-discussion.module.d.ts +3 -0
  130. package/dist/course-lesson-discussion/course-lesson-discussion.module.d.ts.map +1 -0
  131. package/dist/course-lesson-discussion/course-lesson-discussion.module.js +26 -0
  132. package/dist/course-lesson-discussion/course-lesson-discussion.module.js.map +1 -0
  133. package/dist/course-lesson-discussion/course-lesson-discussion.service.d.ts +49 -0
  134. package/dist/course-lesson-discussion/course-lesson-discussion.service.d.ts.map +1 -0
  135. package/dist/course-lesson-discussion/course-lesson-discussion.service.js +272 -0
  136. package/dist/course-lesson-discussion/course-lesson-discussion.service.js.map +1 -0
  137. package/dist/course-lesson-discussion/dto/create-course-lesson-discussion-topic.dto.d.ts +6 -0
  138. package/dist/course-lesson-discussion/dto/create-course-lesson-discussion-topic.dto.d.ts.map +1 -0
  139. package/dist/course-lesson-discussion/dto/create-course-lesson-discussion-topic.dto.js +33 -0
  140. package/dist/course-lesson-discussion/dto/create-course-lesson-discussion-topic.dto.js.map +1 -0
  141. package/dist/course-lesson-note/course-lesson-note.controller.d.ts +53 -0
  142. package/dist/course-lesson-note/course-lesson-note.controller.d.ts.map +1 -0
  143. package/dist/course-lesson-note/course-lesson-note.controller.js +93 -0
  144. package/dist/course-lesson-note/course-lesson-note.controller.js.map +1 -0
  145. package/dist/course-lesson-note/course-lesson-note.mcp-tools.d.ts +27 -0
  146. package/dist/course-lesson-note/course-lesson-note.mcp-tools.d.ts.map +1 -0
  147. package/dist/course-lesson-note/course-lesson-note.mcp-tools.js +145 -0
  148. package/dist/course-lesson-note/course-lesson-note.mcp-tools.js.map +1 -0
  149. package/dist/course-lesson-note/course-lesson-note.module.d.ts +3 -0
  150. package/dist/course-lesson-note/course-lesson-note.module.d.ts.map +1 -0
  151. package/dist/course-lesson-note/course-lesson-note.module.js +26 -0
  152. package/dist/course-lesson-note/course-lesson-note.module.js.map +1 -0
  153. package/dist/course-lesson-note/course-lesson-note.service.d.ts +59 -0
  154. package/dist/course-lesson-note/course-lesson-note.service.d.ts.map +1 -0
  155. package/dist/course-lesson-note/course-lesson-note.service.js +195 -0
  156. package/dist/course-lesson-note/course-lesson-note.service.js.map +1 -0
  157. package/dist/course-lesson-note/dto/create-course-lesson-note.dto.d.ts +6 -0
  158. package/dist/course-lesson-note/dto/create-course-lesson-note.dto.d.ts.map +1 -0
  159. package/dist/course-lesson-note/dto/create-course-lesson-note.dto.js +33 -0
  160. package/dist/course-lesson-note/dto/create-course-lesson-note.dto.js.map +1 -0
  161. package/dist/course-lesson-note/dto/update-course-lesson-note.dto.d.ts +6 -0
  162. package/dist/course-lesson-note/dto/update-course-lesson-note.dto.d.ts.map +1 -0
  163. package/dist/course-lesson-note/dto/update-course-lesson-note.dto.js +35 -0
  164. package/dist/course-lesson-note/dto/update-course-lesson-note.dto.js.map +1 -0
  165. package/dist/dashboard/dashboard.mcp-tools.d.ts +10 -0
  166. package/dist/dashboard/dashboard.mcp-tools.d.ts.map +1 -0
  167. package/dist/dashboard/dashboard.mcp-tools.js +46 -0
  168. package/dist/dashboard/dashboard.mcp-tools.js.map +1 -0
  169. package/dist/dashboard/dashboard.module.d.ts.map +1 -1
  170. package/dist/dashboard/dashboard.module.js +2 -1
  171. package/dist/dashboard/dashboard.module.js.map +1 -1
  172. package/dist/enterprise/enterprise.mcp-tools.d.ts +82 -0
  173. package/dist/enterprise/enterprise.mcp-tools.d.ts.map +1 -0
  174. package/dist/enterprise/enterprise.mcp-tools.js +516 -0
  175. package/dist/enterprise/enterprise.mcp-tools.js.map +1 -0
  176. package/dist/enterprise/enterprise.module.d.ts.map +1 -1
  177. package/dist/enterprise/enterprise.module.js +2 -1
  178. package/dist/enterprise/enterprise.module.js.map +1 -1
  179. package/dist/enterprise/training/enterprise-training.module.d.ts.map +1 -1
  180. package/dist/enterprise/training/enterprise-training.module.js +11 -1
  181. package/dist/enterprise/training/enterprise-training.module.js.map +1 -1
  182. package/dist/enterprise/training/training-admin.controller.d.ts +2 -2
  183. package/dist/enterprise/training/training-admin.mcp-tools.d.ts +79 -0
  184. package/dist/enterprise/training/training-admin.mcp-tools.d.ts.map +1 -0
  185. package/dist/enterprise/training/training-admin.mcp-tools.js +620 -0
  186. package/dist/enterprise/training/training-admin.mcp-tools.js.map +1 -0
  187. package/dist/enterprise/training/training-admin.service.d.ts +2 -2
  188. package/dist/enterprise/training/training-instructor.mcp-tools.d.ts +47 -0
  189. package/dist/enterprise/training/training-instructor.mcp-tools.d.ts.map +1 -0
  190. package/dist/enterprise/training/training-instructor.mcp-tools.js +275 -0
  191. package/dist/enterprise/training/training-instructor.mcp-tools.js.map +1 -0
  192. package/dist/enterprise/training/training-student.controller.d.ts +24 -0
  193. package/dist/enterprise/training/training-student.controller.d.ts.map +1 -1
  194. package/dist/enterprise/training/training-student.controller.js +22 -0
  195. package/dist/enterprise/training/training-student.controller.js.map +1 -1
  196. package/dist/enterprise/training/training-student.mcp-tools.d.ts +27 -0
  197. package/dist/enterprise/training/training-student.mcp-tools.d.ts.map +1 -0
  198. package/dist/enterprise/training/training-student.mcp-tools.js +186 -0
  199. package/dist/enterprise/training/training-student.mcp-tools.js.map +1 -0
  200. package/dist/enterprise/training/training-student.service.d.ts +32 -0
  201. package/dist/enterprise/training/training-student.service.d.ts.map +1 -1
  202. package/dist/enterprise/training/training-student.service.js +138 -0
  203. package/dist/enterprise/training/training-student.service.js.map +1 -1
  204. package/dist/evaluation/evaluation.controller.d.ts +4 -4
  205. package/dist/evaluation/evaluation.mcp-tools.d.ts +25 -0
  206. package/dist/evaluation/evaluation.mcp-tools.d.ts.map +1 -0
  207. package/dist/evaluation/evaluation.mcp-tools.js +220 -0
  208. package/dist/evaluation/evaluation.mcp-tools.js.map +1 -0
  209. package/dist/evaluation/evaluation.module.d.ts.map +1 -1
  210. package/dist/evaluation/evaluation.module.js +2 -1
  211. package/dist/evaluation/evaluation.module.js.map +1 -1
  212. package/dist/evaluation/evaluation.service.d.ts +4 -4
  213. package/dist/exam/dto/create-exam-question.dto.d.ts +2 -0
  214. package/dist/exam/dto/create-exam-question.dto.d.ts.map +1 -1
  215. package/dist/exam/dto/create-exam-question.dto.js +10 -0
  216. package/dist/exam/dto/create-exam-question.dto.js.map +1 -1
  217. package/dist/exam/dto/create-exam.dto.d.ts +2 -0
  218. package/dist/exam/dto/create-exam.dto.d.ts.map +1 -1
  219. package/dist/exam/dto/create-exam.dto.js +10 -0
  220. package/dist/exam/dto/create-exam.dto.js.map +1 -1
  221. package/dist/exam/dto/create-question-subject.dto.d.ts +5 -0
  222. package/dist/exam/dto/create-question-subject.dto.d.ts.map +1 -0
  223. package/dist/exam/dto/create-question-subject.dto.js +28 -0
  224. package/dist/exam/dto/create-question-subject.dto.js.map +1 -0
  225. package/dist/exam/exam-attempt.controller.d.ts +4 -0
  226. package/dist/exam/exam-attempt.controller.d.ts.map +1 -1
  227. package/dist/exam/exam-attempt.service.d.ts +7 -1
  228. package/dist/exam/exam-attempt.service.d.ts.map +1 -1
  229. package/dist/exam/exam-attempt.service.js +47 -17
  230. package/dist/exam/exam-attempt.service.js.map +1 -1
  231. package/dist/exam/exam.controller.d.ts +34 -0
  232. package/dist/exam/exam.controller.d.ts.map +1 -1
  233. package/dist/exam/exam.controller.js +27 -0
  234. package/dist/exam/exam.controller.js.map +1 -1
  235. package/dist/exam/exam.mcp-tools.d.ts +62 -0
  236. package/dist/exam/exam.mcp-tools.d.ts.map +1 -0
  237. package/dist/exam/exam.mcp-tools.js +430 -0
  238. package/dist/exam/exam.mcp-tools.js.map +1 -0
  239. package/dist/exam/exam.module.d.ts.map +1 -1
  240. package/dist/exam/exam.module.js +2 -1
  241. package/dist/exam/exam.module.js.map +1 -1
  242. package/dist/exam/exam.service.d.ts +38 -0
  243. package/dist/exam/exam.service.d.ts.map +1 -1
  244. package/dist/exam/exam.service.js +114 -17
  245. package/dist/exam/exam.service.js.map +1 -1
  246. package/dist/index.d.ts +9 -0
  247. package/dist/index.d.ts.map +1 -1
  248. package/dist/index.js +9 -0
  249. package/dist/index.js.map +1 -1
  250. package/dist/instructor/instructor.mcp-tools.d.ts +41 -0
  251. package/dist/instructor/instructor.mcp-tools.d.ts.map +1 -0
  252. package/dist/instructor/instructor.mcp-tools.js +326 -0
  253. package/dist/instructor/instructor.mcp-tools.js.map +1 -0
  254. package/dist/instructor/instructor.module.d.ts.map +1 -1
  255. package/dist/instructor/instructor.module.js +2 -1
  256. package/dist/instructor/instructor.module.js.map +1 -1
  257. package/dist/lms.module.d.ts.map +1 -1
  258. package/dist/lms.module.js +15 -0
  259. package/dist/lms.module.js.map +1 -1
  260. package/dist/realtime/lms-realtime.controller.d.ts +7 -0
  261. package/dist/realtime/lms-realtime.controller.d.ts.map +1 -0
  262. package/dist/realtime/lms-realtime.controller.js +34 -0
  263. package/dist/realtime/lms-realtime.controller.js.map +1 -0
  264. package/dist/realtime/lms-realtime.module.d.ts +3 -0
  265. package/dist/realtime/lms-realtime.module.d.ts.map +1 -0
  266. package/dist/realtime/lms-realtime.module.js +25 -0
  267. package/dist/realtime/lms-realtime.module.js.map +1 -0
  268. package/dist/realtime/lms-realtime.service.d.ts +36 -0
  269. package/dist/realtime/lms-realtime.service.d.ts.map +1 -0
  270. package/dist/realtime/lms-realtime.service.js +59 -0
  271. package/dist/realtime/lms-realtime.service.js.map +1 -0
  272. package/dist/realtime/lms-realtime.subscriber.d.ts +10 -0
  273. package/dist/realtime/lms-realtime.subscriber.d.ts.map +1 -0
  274. package/dist/realtime/lms-realtime.subscriber.js +70 -0
  275. package/dist/realtime/lms-realtime.subscriber.js.map +1 -0
  276. package/dist/reports/reports.mcp-tools.d.ts +10 -0
  277. package/dist/reports/reports.mcp-tools.d.ts.map +1 -0
  278. package/dist/reports/reports.mcp-tools.js +50 -0
  279. package/dist/reports/reports.mcp-tools.js.map +1 -0
  280. package/dist/reports/reports.module.d.ts.map +1 -1
  281. package/dist/reports/reports.module.js +2 -1
  282. package/dist/reports/reports.module.js.map +1 -1
  283. package/dist/training/training.mcp-tools.d.ts +20 -0
  284. package/dist/training/training.mcp-tools.d.ts.map +1 -0
  285. package/dist/training/training.mcp-tools.js +181 -0
  286. package/dist/training/training.mcp-tools.js.map +1 -0
  287. package/dist/training/training.module.d.ts.map +1 -1
  288. package/dist/training/training.module.js +2 -1
  289. package/dist/training/training.module.js.map +1 -1
  290. package/hedhog/data/integration_event_catalog.yaml +69 -0
  291. package/hedhog/data/menu.yaml +34 -0
  292. package/hedhog/data/route.yaml +2351 -0
  293. package/hedhog/frontend/app/_components/class-form-sheet.tsx.ejs +168 -103
  294. package/hedhog/frontend/app/_components/course-form-sheet.tsx.ejs +80 -1
  295. package/hedhog/frontend/app/_components/course-picker.tsx.ejs +228 -0
  296. package/hedhog/frontend/app/_lib/hooks/use-lms-realtime-refresh.ts.ejs +58 -0
  297. package/hedhog/frontend/app/achievements/page.tsx.ejs +844 -0
  298. package/hedhog/frontend/app/bitcodes/page.tsx.ejs +1010 -0
  299. package/hedhog/frontend/app/certificates/issued/page.tsx.ejs +61 -2
  300. package/hedhog/frontend/app/classes/[id]/page.tsx.ejs +7 -0
  301. package/hedhog/frontend/app/classes/page.tsx.ejs +55 -2060
  302. package/hedhog/frontend/app/courses/[id]/_components/CourseRelationsCard.tsx.ejs +28 -0
  303. package/hedhog/frontend/app/courses/[id]/_components/course-edit-types.ts.ejs +1 -0
  304. package/hedhog/frontend/app/courses/[id]/page.tsx.ejs +0 -9
  305. package/hedhog/frontend/app/courses/[id]/structure/_components/course-tree-dnd.tsx.ejs +80 -66
  306. package/hedhog/frontend/app/courses/[id]/structure/_components/editor-course.tsx.ejs +583 -8
  307. package/hedhog/frontend/app/courses/[id]/structure/_components/editor-lesson.tsx.ejs +527 -57
  308. package/hedhog/frontend/app/courses/[id]/structure/_components/mock-data.ts.ejs +1 -1
  309. package/hedhog/frontend/app/courses/[id]/structure/_components/tree-context-menu.tsx.ejs +106 -4
  310. package/hedhog/frontend/app/courses/[id]/structure/_components/types.ts.ejs +2 -1
  311. package/hedhog/frontend/app/courses/[id]/structure/_data/adapters/course-structure.adapter.ts.ejs +11 -1
  312. package/hedhog/frontend/app/courses/[id]/structure/_data/services/course-structure.service.ts.ejs +53 -6
  313. package/hedhog/frontend/app/courses/[id]/structure/_data/types/api-course.types.ts.ejs +13 -2
  314. package/hedhog/frontend/app/courses/page.tsx.ejs +175 -29
  315. package/hedhog/frontend/app/enterprise/_components/enterprise-course-create-sheet.tsx.ejs +3 -0
  316. package/hedhog/frontend/app/enterprise/_components/enterprise-course-edit-sheet.tsx.ejs +7 -0
  317. package/hedhog/frontend/app/exams/[id]/questions/page.tsx.ejs +169 -2
  318. package/hedhog/frontend/app/exams/page.tsx.ejs +77 -22
  319. package/hedhog/frontend/app/instructor-skills/page.tsx.ejs +1 -0
  320. package/hedhog/frontend/app/instructors/page.tsx.ejs +1 -0
  321. package/hedhog/frontend/app/paths/page.tsx.ejs +6 -24
  322. package/hedhog/frontend/app/training/page.tsx.ejs +6 -24
  323. package/hedhog/frontend/messages/en.json +314 -12
  324. package/hedhog/frontend/messages/pt.json +314 -12
  325. package/hedhog/query/triggers.sql +53 -0
  326. package/hedhog/table/achievement.yaml +46 -0
  327. package/hedhog/table/bitcode_wallet.yaml +18 -0
  328. package/hedhog/table/bitcode_wallet_transaction.yaml +22 -0
  329. package/hedhog/table/certificate.yaml +3 -0
  330. package/hedhog/table/course.yaml +4 -0
  331. package/hedhog/table/course_file.yaml +23 -0
  332. package/hedhog/table/course_lesson.yaml +5 -0
  333. package/hedhog/table/course_lesson_discussion_like.yaml +21 -0
  334. package/hedhog/table/course_lesson_discussion_topic.yaml +35 -0
  335. package/hedhog/table/course_lesson_note.yaml +34 -0
  336. package/hedhog/table/exam.yaml +5 -0
  337. package/hedhog/table/learning_path_enrollment.yaml +6 -0
  338. package/hedhog/table/question.yaml +10 -0
  339. package/hedhog/table/question_subject.yaml +17 -0
  340. package/hedhog/table/student_activity_streak.yaml +25 -0
  341. package/package.json +7 -7
  342. package/src/achievement/achievement.controller.ts +60 -0
  343. package/src/achievement/achievement.mcp-tools.ts +108 -0
  344. package/src/achievement/achievement.module.ts +13 -0
  345. package/src/achievement/achievement.service.ts +252 -0
  346. package/src/achievement/dto/create-achievement.dto.ts +50 -0
  347. package/src/achievement/dto/update-achievement.dto.ts +47 -0
  348. package/src/bitcode-wallet/bitcode-wallet.controller.ts +69 -0
  349. package/src/bitcode-wallet/bitcode-wallet.mcp-tools.ts +107 -0
  350. package/src/bitcode-wallet/bitcode-wallet.module.ts +13 -0
  351. package/src/bitcode-wallet/bitcode-wallet.service.ts +361 -0
  352. package/src/bitcode-wallet/dto/create-bitcode-wallet-transaction.dto.ts +27 -0
  353. package/src/bitcode-wallet/dto/create-bitcode-wallet.dto.ts +7 -0
  354. package/src/bitcode-wallet/dto/update-bitcode-wallet-transaction.dto.ts +28 -0
  355. package/src/bitcode-wallet/dto/update-bitcode-wallet.dto.ts +8 -0
  356. package/src/certificate/certificate.controller.ts +20 -11
  357. package/src/certificate/certificate.mcp-tools.ts +131 -0
  358. package/src/certificate/certificate.module.ts +2 -1
  359. package/src/certificate/certificate.service.ts +95 -4
  360. package/src/certificate/dto/update-certificate-public-access.dto.ts +6 -0
  361. package/src/class-group/class-group.mcp-tools.ts +435 -0
  362. package/src/class-group/class-group.module.ts +2 -1
  363. package/src/class-group/class-group.service.ts +51 -1
  364. package/src/course/course-operations-integration.service.ts +520 -0
  365. package/src/course/course-structure.controller.ts +22 -8
  366. package/src/course/course-structure.service.ts +215 -23
  367. package/src/course/course.mcp-tools.ts +409 -0
  368. package/src/course/course.module.ts +8 -1
  369. package/src/course/course.service.ts +106 -27
  370. package/src/course/dto/create-course.dto.ts +8 -0
  371. package/src/course/dto/update-course-resources.dto.ts +39 -0
  372. package/src/course-lesson-discussion/course-lesson-discussion.controller.ts +55 -0
  373. package/src/course-lesson-discussion/course-lesson-discussion.mcp-tools.ts +75 -0
  374. package/src/course-lesson-discussion/course-lesson-discussion.module.ts +13 -0
  375. package/src/course-lesson-discussion/course-lesson-discussion.service.ts +354 -0
  376. package/src/course-lesson-discussion/dto/create-course-lesson-discussion-topic.dto.ts +16 -0
  377. package/src/course-lesson-note/course-lesson-note.controller.ts +68 -0
  378. package/src/course-lesson-note/course-lesson-note.mcp-tools.ts +96 -0
  379. package/src/course-lesson-note/course-lesson-note.module.ts +13 -0
  380. package/src/course-lesson-note/course-lesson-note.service.ts +248 -0
  381. package/src/course-lesson-note/dto/create-course-lesson-note.dto.ts +16 -0
  382. package/src/course-lesson-note/dto/update-course-lesson-note.dto.ts +18 -0
  383. package/src/dashboard/dashboard.mcp-tools.ts +23 -0
  384. package/src/dashboard/dashboard.module.ts +2 -1
  385. package/src/enterprise/enterprise.mcp-tools.ts +403 -0
  386. package/src/enterprise/enterprise.module.ts +2 -1
  387. package/src/enterprise/training/enterprise-training.module.ts +11 -1
  388. package/src/enterprise/training/training-admin.mcp-tools.ts +479 -0
  389. package/src/enterprise/training/training-instructor.mcp-tools.ts +210 -0
  390. package/src/enterprise/training/training-student.controller.ts +17 -1
  391. package/src/enterprise/training/training-student.mcp-tools.ts +136 -0
  392. package/src/enterprise/training/training-student.service.ts +167 -1
  393. package/src/evaluation/evaluation.mcp-tools.ts +155 -0
  394. package/src/evaluation/evaluation.module.ts +2 -1
  395. package/src/exam/dto/create-exam-question.dto.ts +8 -0
  396. package/src/exam/dto/create-exam.dto.ts +8 -0
  397. package/src/exam/dto/create-question-subject.dto.ts +12 -0
  398. package/src/exam/exam-attempt.service.ts +46 -14
  399. package/src/exam/exam.controller.ts +19 -0
  400. package/src/exam/exam.mcp-tools.ts +337 -0
  401. package/src/exam/exam.module.ts +2 -1
  402. package/src/exam/exam.service.ts +121 -0
  403. package/src/index.ts +9 -0
  404. package/src/instructor/instructor.mcp-tools.ts +243 -0
  405. package/src/instructor/instructor.module.ts +2 -1
  406. package/src/lms.module.ts +15 -1
  407. package/src/realtime/lms-realtime.controller.ts +12 -0
  408. package/src/realtime/lms-realtime.module.ts +12 -0
  409. package/src/realtime/lms-realtime.service.ts +98 -0
  410. package/src/realtime/lms-realtime.subscriber.ts +61 -0
  411. package/src/reports/reports.mcp-tools.ts +27 -0
  412. package/src/reports/reports.module.ts +2 -1
  413. package/src/training/training.mcp-tools.ts +128 -0
  414. package/src/training/training.module.ts +2 -1
@@ -0,0 +1,1010 @@
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
+ DropdownMenu,
23
+ DropdownMenuContent,
24
+ DropdownMenuItem,
25
+ DropdownMenuSeparator,
26
+ DropdownMenuTrigger,
27
+ } from '@/components/ui/dropdown-menu';
28
+ import { EntityPicker } from '@/components/ui/entity-picker';
29
+ import {
30
+ Form,
31
+ FormControl,
32
+ FormField,
33
+ FormItem,
34
+ FormLabel,
35
+ FormMessage,
36
+ } from '@/components/ui/form';
37
+ import { Input } from '@/components/ui/input';
38
+ import { KpiCardsGrid, type KpiCardItem } from '@/components/ui/kpi-cards-grid';
39
+ import {
40
+ Select,
41
+ SelectContent,
42
+ SelectItem,
43
+ SelectTrigger,
44
+ SelectValue,
45
+ } from '@/components/ui/select';
46
+ import {
47
+ Sheet,
48
+ SheetContent,
49
+ SheetDescription,
50
+ SheetHeader,
51
+ SheetTitle,
52
+ } from '@/components/ui/sheet';
53
+ import { Skeleton } from '@/components/ui/skeleton';
54
+ import {
55
+ Table,
56
+ TableBody,
57
+ TableCell,
58
+ TableHead,
59
+ TableHeader,
60
+ TableRow,
61
+ } from '@/components/ui/table';
62
+ import { Textarea } from '@/components/ui/textarea';
63
+ import { usePersistedPageSize } from '@/hooks/use-persisted-page-size';
64
+ import { cn } from '@/lib/utils';
65
+ import { useApp, useQuery } from '@hed-hog/next-app-provider';
66
+ import { zodResolver } from '@hookform/resolvers/zod';
67
+ import {
68
+ ArrowDownLeft,
69
+ ArrowUpRight,
70
+ Coins,
71
+ Loader2,
72
+ MoreHorizontal,
73
+ Pencil,
74
+ Plus,
75
+ Trash2,
76
+ Wallet,
77
+ } from 'lucide-react';
78
+ import { useTranslations } from 'next-intl';
79
+ import { useEffect, useMemo, useState } from 'react';
80
+ import { useForm } from 'react-hook-form';
81
+ import { toast } from 'sonner';
82
+ import * as z from 'zod';
83
+ import {
84
+ loadPersonOptions,
85
+ type PersonPickerOption,
86
+ } from '../enterprise/_components/enterprise-person-picker';
87
+
88
+ type TransactionType = 'credit' | 'debit';
89
+
90
+ type BitcodeWalletSummary = {
91
+ walletCount: number;
92
+ totalBalance: number;
93
+ transactionCount: number;
94
+ totalCredits: number;
95
+ totalDebits: number;
96
+ };
97
+
98
+ type BitcodeWallet = {
99
+ id: number;
100
+ personId: number;
101
+ personName: string;
102
+ currentBalance: number;
103
+ transactionCount: number;
104
+ totalCredits: number;
105
+ totalDebits: number;
106
+ createdAt?: string | null;
107
+ updatedAt?: string | null;
108
+ };
109
+
110
+ type BitcodeWalletTransaction = {
111
+ id: number;
112
+ walletId: number;
113
+ type: TransactionType;
114
+ amount: number;
115
+ description?: string | null;
116
+ createdAt?: string | null;
117
+ updatedAt?: string | null;
118
+ };
119
+
120
+ type BitcodeWalletDetail = BitcodeWallet & {
121
+ transactions: BitcodeWalletTransaction[];
122
+ };
123
+
124
+ type BitcodeWalletPaginatedResult = {
125
+ data: BitcodeWallet[];
126
+ total: number;
127
+ page: number;
128
+ pageSize: number;
129
+ lastPage: number;
130
+ summary: BitcodeWalletSummary;
131
+ };
132
+
133
+ const transactionTypeValues: TransactionType[] = ['credit', 'debit'];
134
+
135
+ const walletSchema = (t: ReturnType<typeof useTranslations>) =>
136
+ z.object({
137
+ personId: z.coerce
138
+ .number()
139
+ .int()
140
+ .min(1, t('form.validation.personRequired')),
141
+ });
142
+
143
+ const transactionSchema = (t: ReturnType<typeof useTranslations>) =>
144
+ z.object({
145
+ type: z.enum(transactionTypeValues as any),
146
+ amount: z.coerce
147
+ .number()
148
+ .int(t('transactions.form.validation.integer'))
149
+ .min(1, t('transactions.form.validation.positive')),
150
+ description: z
151
+ .string()
152
+ .max(2000, t('transactions.form.validation.max2000'))
153
+ .optional(),
154
+ });
155
+
156
+ type WalletFormValues = z.infer<ReturnType<typeof walletSchema>>;
157
+ type TransactionFormValues = z.infer<ReturnType<typeof transactionSchema>>;
158
+
159
+ function formatBitcodes(value: number) {
160
+ return new Intl.NumberFormat('pt-BR').format(Number(value ?? 0));
161
+ }
162
+
163
+ function formatDateTime(value?: string | null) {
164
+ if (!value) return '—';
165
+ const date = new Date(value);
166
+ if (Number.isNaN(date.getTime())) return '—';
167
+ return new Intl.DateTimeFormat('pt-BR', {
168
+ dateStyle: 'short',
169
+ timeStyle: 'short',
170
+ }).format(date);
171
+ }
172
+
173
+ function getErrorMessage(error: unknown, fallback: string) {
174
+ if (
175
+ typeof error === 'object' &&
176
+ error !== null &&
177
+ 'response' in error &&
178
+ typeof (error as { response?: unknown }).response === 'object' &&
179
+ (error as { response?: { data?: unknown } }).response !== null
180
+ ) {
181
+ const response = (
182
+ error as {
183
+ response?: { data?: { message?: unknown } };
184
+ }
185
+ ).response;
186
+ const message = response?.data?.message;
187
+ if (typeof message === 'string' && message.trim()) {
188
+ return message;
189
+ }
190
+ }
191
+
192
+ return fallback;
193
+ }
194
+
195
+ function TransactionTypeBadge({
196
+ type,
197
+ t,
198
+ }: {
199
+ type: TransactionType;
200
+ t: ReturnType<typeof useTranslations>;
201
+ }) {
202
+ const isCredit = type === 'credit';
203
+
204
+ return (
205
+ <Badge
206
+ variant="outline"
207
+ className={cn(
208
+ isCredit
209
+ ? 'border-emerald-200 bg-emerald-50 text-emerald-700'
210
+ : 'border-rose-200 bg-rose-50 text-rose-700'
211
+ )}
212
+ >
213
+ {isCredit ? (
214
+ <ArrowUpRight className="mr-1 size-3.5" />
215
+ ) : (
216
+ <ArrowDownLeft className="mr-1 size-3.5" />
217
+ )}
218
+ {t(`transactions.types.${type}`)}
219
+ </Badge>
220
+ );
221
+ }
222
+
223
+ function WalletFormSheet({
224
+ open,
225
+ onOpenChange,
226
+ walletToEdit,
227
+ onSaved,
228
+ }: {
229
+ open: boolean;
230
+ onOpenChange: (open: boolean) => void;
231
+ walletToEdit: BitcodeWallet | null;
232
+ onSaved: () => void;
233
+ }) {
234
+ const { request } = useApp();
235
+ const t = useTranslations('lms.BitcodesPage');
236
+ const walletId = walletToEdit?.id ?? null;
237
+
238
+ const [isSavingWallet, setIsSavingWallet] = useState(false);
239
+ const [isSavingTransaction, setIsSavingTransaction] = useState(false);
240
+
241
+ const walletForm = useForm<WalletFormValues>({
242
+ resolver: zodResolver(walletSchema(t)),
243
+ defaultValues: {
244
+ personId: walletToEdit?.personId ?? 0,
245
+ },
246
+ });
247
+
248
+ const transactionForm = useForm<TransactionFormValues>({
249
+ resolver: zodResolver(transactionSchema(t)),
250
+ defaultValues: {
251
+ type: 'credit',
252
+ amount: 1,
253
+ description: '',
254
+ },
255
+ });
256
+
257
+ const {
258
+ data: walletDetail,
259
+ isLoading: isLoadingWalletDetail,
260
+ refetch: refetchWalletDetail,
261
+ } = useQuery<BitcodeWalletDetail | null>({
262
+ queryKey: ['lms-bitcode-wallet-detail', walletId, open],
263
+ enabled: open && Boolean(walletId),
264
+ queryFn: async () => {
265
+ if (!walletId) return null;
266
+ const response = await request<BitcodeWalletDetail>({
267
+ url: `/lms/bitcode-wallets/${walletId}`,
268
+ method: 'GET',
269
+ });
270
+ return response.data;
271
+ },
272
+ });
273
+
274
+ useEffect(() => {
275
+ if (!open) return;
276
+
277
+ walletForm.reset({
278
+ personId: walletToEdit?.personId ?? 0,
279
+ });
280
+ }, [open, walletForm, walletToEdit]);
281
+
282
+ useEffect(() => {
283
+ if (!open) return;
284
+ if (!walletDetail) {
285
+ transactionForm.reset({
286
+ type: 'credit',
287
+ amount: 1,
288
+ description: '',
289
+ });
290
+ return;
291
+ }
292
+
293
+ walletForm.reset({
294
+ personId: walletDetail.personId,
295
+ });
296
+ }, [open, walletDetail, transactionForm, walletForm]);
297
+
298
+ const handleSaveWallet = async (values: WalletFormValues) => {
299
+ try {
300
+ setIsSavingWallet(true);
301
+
302
+ if (walletToEdit) {
303
+ await request({
304
+ url: `/lms/bitcode-wallets/${walletToEdit.id}`,
305
+ method: 'PATCH',
306
+ data: values,
307
+ });
308
+ toast.success(t('messages.walletUpdated'));
309
+ } else {
310
+ await request({
311
+ url: '/lms/bitcode-wallets',
312
+ method: 'POST',
313
+ data: values,
314
+ });
315
+ toast.success(t('messages.walletCreated'));
316
+ }
317
+
318
+ await onSaved();
319
+ onOpenChange(false);
320
+ } catch (error: unknown) {
321
+ toast.error(getErrorMessage(error, t('messages.walletSaveError')));
322
+ } finally {
323
+ setIsSavingWallet(false);
324
+ }
325
+ };
326
+
327
+ const resetTransactionForm = () => {
328
+ transactionForm.reset({
329
+ type: 'credit',
330
+ amount: 1,
331
+ description: '',
332
+ });
333
+ };
334
+
335
+ const handleSaveTransaction = async (values: TransactionFormValues) => {
336
+ if (!walletId) return;
337
+
338
+ try {
339
+ setIsSavingTransaction(true);
340
+
341
+ const payload = {
342
+ type: values.type,
343
+ amount: values.amount,
344
+ description: values.description?.trim() || null,
345
+ };
346
+
347
+ await request({
348
+ url: `/lms/bitcode-wallets/${walletId}/transactions`,
349
+ method: 'POST',
350
+ data: payload,
351
+ });
352
+ toast.success(t('messages.transactionCreated'));
353
+
354
+ resetTransactionForm();
355
+ await refetchWalletDetail();
356
+ await onSaved();
357
+ } catch (error: unknown) {
358
+ toast.error(getErrorMessage(error, t('messages.transactionSaveError')));
359
+ } finally {
360
+ setIsSavingTransaction(false);
361
+ }
362
+ };
363
+
364
+ return (
365
+ <>
366
+ <Sheet open={open} onOpenChange={onOpenChange}>
367
+ <SheetContent className="w-full overflow-y-auto sm:max-w-4xl">
368
+ <SheetHeader>
369
+ <SheetTitle>
370
+ {walletToEdit ? t('sheet.editTitle') : t('sheet.createTitle')}
371
+ </SheetTitle>
372
+ <SheetDescription>
373
+ {walletToEdit
374
+ ? t('sheet.editDescription')
375
+ : t('sheet.createDescription')}
376
+ </SheetDescription>
377
+ </SheetHeader>
378
+
379
+ <div className="space-y-6 px-4 py-4">
380
+ <Form {...walletForm}>
381
+ <form
382
+ onSubmit={walletForm.handleSubmit(handleSaveWallet)}
383
+ className="space-y-4 rounded-xl border p-4"
384
+ >
385
+ <div className="flex items-center justify-between gap-3">
386
+ <div>
387
+ <h3 className="font-semibold">{t('form.title')}</h3>
388
+ <p className="text-sm text-muted-foreground">
389
+ {t('form.description')}
390
+ </p>
391
+ </div>
392
+ {walletDetail ? (
393
+ <Badge variant="secondary">
394
+ {t('form.currentBalance', {
395
+ value: formatBitcodes(walletDetail.currentBalance),
396
+ })}
397
+ </Badge>
398
+ ) : null}
399
+ </div>
400
+
401
+ <FormField
402
+ control={walletForm.control}
403
+ name="personId"
404
+ render={({ field }) => (
405
+ <FormItem>
406
+ <FormLabel>{t('form.person')}</FormLabel>
407
+ <FormControl>
408
+ <EntityPicker<PersonPickerOption>
409
+ value={field.value || null}
410
+ valueType="number"
411
+ placeholder={t('form.personPlaceholder')}
412
+ searchPlaceholder={t('form.personSearchPlaceholder')}
413
+ entityLabel={t('form.personEntityLabel')}
414
+ noResultsLabel={t('form.personEmpty')}
415
+ clearable
416
+ initialSelectedLabel={walletToEdit?.personName ?? ''}
417
+ showCreateButton={false}
418
+ loadOptions={(args) =>
419
+ loadPersonOptions(request, args)
420
+ }
421
+ getOptionLabel={(option) => option.name}
422
+ getOptionDescription={(option) =>
423
+ option.email ?? undefined
424
+ }
425
+ onChange={(value) =>
426
+ field.onChange(Number(value ?? 0))
427
+ }
428
+ />
429
+ </FormControl>
430
+ <FormMessage />
431
+ </FormItem>
432
+ )}
433
+ />
434
+
435
+ <div className="flex justify-end gap-2">
436
+ <Button
437
+ type="button"
438
+ variant="outline"
439
+ onClick={() => onOpenChange(false)}
440
+ disabled={isSavingWallet}
441
+ >
442
+ {t('actions.cancel')}
443
+ </Button>
444
+ <Button type="submit" disabled={isSavingWallet}>
445
+ {isSavingWallet
446
+ ? t('actions.saving')
447
+ : t('actions.saveWallet')}
448
+ </Button>
449
+ </div>
450
+ </form>
451
+ </Form>
452
+
453
+ {walletToEdit ? (
454
+ isLoadingWalletDetail ? (
455
+ <div className="space-y-3">
456
+ <Skeleton className="h-28 w-full" />
457
+ <Skeleton className="h-52 w-full" />
458
+ </div>
459
+ ) : walletDetail ? (
460
+ <div className="space-y-4">
461
+ <KpiCardsGrid
462
+ columns={3}
463
+ items={[
464
+ {
465
+ key: 'balance',
466
+ title: t('transactions.kpis.balance'),
467
+ value: formatBitcodes(walletDetail.currentBalance),
468
+ description: t('transactions.kpis.balanceDescription'),
469
+ icon: Coins,
470
+ },
471
+ {
472
+ key: 'credits',
473
+ title: t('transactions.kpis.credits'),
474
+ value: formatBitcodes(walletDetail.totalCredits),
475
+ description: t('transactions.kpis.creditsDescription'),
476
+ icon: ArrowUpRight,
477
+ },
478
+ {
479
+ key: 'debits',
480
+ title: t('transactions.kpis.debits'),
481
+ value: formatBitcodes(walletDetail.totalDebits),
482
+ description: t('transactions.kpis.debitsDescription'),
483
+ icon: ArrowDownLeft,
484
+ },
485
+ ]}
486
+ />
487
+
488
+ <Form {...transactionForm}>
489
+ <form
490
+ onSubmit={transactionForm.handleSubmit(
491
+ handleSaveTransaction
492
+ )}
493
+ className="space-y-4 rounded-xl border p-4"
494
+ >
495
+ <div className="flex items-center justify-between gap-3">
496
+ <div>
497
+ <h3 className="font-semibold">
498
+ {t('transactions.form.createTitle')}
499
+ </h3>
500
+ <p className="text-sm text-muted-foreground">
501
+ {t('transactions.form.description')}
502
+ </p>
503
+ </div>
504
+ </div>
505
+
506
+ <div className="grid gap-4 md:grid-cols-[180px_160px_1fr]">
507
+ <FormField
508
+ control={transactionForm.control}
509
+ name="type"
510
+ render={({ field }) => (
511
+ <FormItem>
512
+ <FormLabel>
513
+ {t('transactions.form.type')}
514
+ </FormLabel>
515
+ <Select
516
+ onValueChange={field.onChange}
517
+ value={field.value}
518
+ >
519
+ <FormControl>
520
+ <SelectTrigger className="w-full">
521
+ <SelectValue
522
+ placeholder={t(
523
+ 'transactions.form.typePlaceholder'
524
+ )}
525
+ />
526
+ </SelectTrigger>
527
+ </FormControl>
528
+ <SelectContent>
529
+ {transactionTypeValues.map((type) => (
530
+ <SelectItem key={type} value={type}>
531
+ {t(`transactions.types.${type}`)}
532
+ </SelectItem>
533
+ ))}
534
+ </SelectContent>
535
+ </Select>
536
+ <FormMessage />
537
+ </FormItem>
538
+ )}
539
+ />
540
+
541
+ <FormField
542
+ control={transactionForm.control}
543
+ name="amount"
544
+ render={({ field }) => (
545
+ <FormItem>
546
+ <FormLabel>
547
+ {t('transactions.form.amount')}
548
+ </FormLabel>
549
+ <FormControl>
550
+ <Input
551
+ type="number"
552
+ min="1"
553
+ step="1"
554
+ {...field}
555
+ />
556
+ </FormControl>
557
+ <FormMessage />
558
+ </FormItem>
559
+ )}
560
+ />
561
+
562
+ <FormField
563
+ control={transactionForm.control}
564
+ name="description"
565
+ render={({ field }) => (
566
+ <FormItem>
567
+ <FormLabel>
568
+ {t('transactions.form.descriptionLabel')}
569
+ </FormLabel>
570
+ <FormControl>
571
+ <Textarea
572
+ rows={3}
573
+ placeholder={t(
574
+ 'transactions.form.descriptionPlaceholder'
575
+ )}
576
+ {...field}
577
+ value={field.value ?? ''}
578
+ />
579
+ </FormControl>
580
+ <FormMessage />
581
+ </FormItem>
582
+ )}
583
+ />
584
+ </div>
585
+
586
+ <div className="flex justify-end gap-2">
587
+ <Button
588
+ type="button"
589
+ variant="outline"
590
+ onClick={resetTransactionForm}
591
+ disabled={isSavingTransaction}
592
+ >
593
+ {t('actions.clear')}
594
+ </Button>
595
+ <Button type="submit" disabled={isSavingTransaction}>
596
+ {isSavingTransaction ? (
597
+ <>
598
+ <Loader2 className="mr-2 size-4 animate-spin" />
599
+ {t('actions.saving')}
600
+ </>
601
+ ) : (
602
+ t('transactions.form.createAction')
603
+ )}
604
+ </Button>
605
+ </div>
606
+ </form>
607
+ </Form>
608
+
609
+ <div className="rounded-xl border">
610
+ <div className="flex items-center justify-between gap-3 border-b px-4 py-3">
611
+ <div>
612
+ <h3 className="font-semibold">
613
+ {t('transactions.table.title')}
614
+ </h3>
615
+ <p className="text-sm text-muted-foreground">
616
+ {t('transactions.table.description', {
617
+ count: walletDetail.transactions.length,
618
+ })}
619
+ </p>
620
+ </div>
621
+ </div>
622
+
623
+ {walletDetail.transactions.length === 0 ? (
624
+ <div className="px-4 py-10 text-center text-sm text-muted-foreground">
625
+ {t('transactions.empty')}
626
+ </div>
627
+ ) : (
628
+ <div className="overflow-x-auto">
629
+ <Table>
630
+ <TableHeader>
631
+ <TableRow>
632
+ <TableHead>
633
+ {t('transactions.table.type')}
634
+ </TableHead>
635
+ <TableHead>
636
+ {t('transactions.table.amount')}
637
+ </TableHead>
638
+ <TableHead>
639
+ {t('transactions.table.descriptionLabel')}
640
+ </TableHead>
641
+ <TableHead>
642
+ {t('transactions.table.createdAt')}
643
+ </TableHead>
644
+ </TableRow>
645
+ </TableHeader>
646
+ <TableBody>
647
+ {walletDetail.transactions.map((transaction) => (
648
+ <TableRow key={transaction.id}>
649
+ <TableCell>
650
+ <TransactionTypeBadge
651
+ type={transaction.type}
652
+ t={t}
653
+ />
654
+ </TableCell>
655
+ <TableCell
656
+ className={cn(
657
+ 'font-medium',
658
+ transaction.type === 'credit'
659
+ ? 'text-emerald-600'
660
+ : 'text-rose-600'
661
+ )}
662
+ >
663
+ {transaction.type === 'credit' ? '+' : '-'}
664
+ {formatBitcodes(transaction.amount)}
665
+ </TableCell>
666
+ <TableCell>
667
+ {transaction.description?.trim() || '—'}
668
+ </TableCell>
669
+ <TableCell>
670
+ {formatDateTime(transaction.createdAt)}
671
+ </TableCell>
672
+ </TableRow>
673
+ ))}
674
+ </TableBody>
675
+ </Table>
676
+ </div>
677
+ )}
678
+ </div>
679
+ </div>
680
+ ) : null
681
+ ) : null}
682
+ </div>
683
+ </SheetContent>
684
+ </Sheet>
685
+ </>
686
+ );
687
+ }
688
+
689
+ export default function BitcodesPage() {
690
+ const { request } = useApp();
691
+ const t = useTranslations('lms.BitcodesPage');
692
+
693
+ const [page, setPage] = useState(1);
694
+ const [pageSize, setPageSize] = usePersistedPageSize({
695
+ storageKey: 'pagination:lms-bitcodes:pageSize',
696
+ defaultValue: 12,
697
+ allowedValues: [12, 24, 48],
698
+ });
699
+ const [searchInput, setSearchInput] = useState('');
700
+ const [debouncedSearch, setDebouncedSearch] = useState('');
701
+ const [sheetOpen, setSheetOpen] = useState(false);
702
+ const [walletToEdit, setWalletToEdit] = useState<BitcodeWallet | null>(null);
703
+ const [deleteDialogOpen, setDeleteDialogOpen] = useState(false);
704
+ const [walletToDelete, setWalletToDelete] = useState<BitcodeWallet | null>(
705
+ null
706
+ );
707
+ const [isDeletingWallet, setIsDeletingWallet] = useState(false);
708
+
709
+ useEffect(() => {
710
+ const timeout = setTimeout(() => {
711
+ setDebouncedSearch(searchInput.trim());
712
+ setPage(1);
713
+ }, 300);
714
+ return () => clearTimeout(timeout);
715
+ }, [searchInput]);
716
+
717
+ const {
718
+ data: paginate = {
719
+ data: [],
720
+ total: 0,
721
+ page: 1,
722
+ pageSize,
723
+ lastPage: 1,
724
+ summary: {
725
+ walletCount: 0,
726
+ totalBalance: 0,
727
+ transactionCount: 0,
728
+ totalCredits: 0,
729
+ totalDebits: 0,
730
+ },
731
+ },
732
+ isLoading,
733
+ refetch,
734
+ } = useQuery<BitcodeWalletPaginatedResult>({
735
+ queryKey: ['lms-bitcode-wallets', page, pageSize, debouncedSearch],
736
+ queryFn: async () => {
737
+ const params = new URLSearchParams({
738
+ page: String(page),
739
+ pageSize: String(pageSize),
740
+ });
741
+ if (debouncedSearch) params.set('search', debouncedSearch);
742
+
743
+ const response = await request<BitcodeWalletPaginatedResult>({
744
+ url: `/lms/bitcode-wallets?${params.toString()}`,
745
+ method: 'GET',
746
+ });
747
+ return response.data;
748
+ },
749
+ placeholderData: (prev) =>
750
+ prev ?? {
751
+ data: [],
752
+ total: 0,
753
+ page: 1,
754
+ pageSize,
755
+ lastPage: 1,
756
+ summary: {
757
+ walletCount: 0,
758
+ totalBalance: 0,
759
+ transactionCount: 0,
760
+ totalCredits: 0,
761
+ totalDebits: 0,
762
+ },
763
+ },
764
+ });
765
+
766
+ const totalPages = Math.max(
767
+ 1,
768
+ paginate.lastPage || Math.ceil((paginate.total || 0) / pageSize) || 1
769
+ );
770
+
771
+ useEffect(() => {
772
+ if (page > totalPages) setPage(totalPages);
773
+ }, [page, totalPages]);
774
+
775
+ const kpiItems = useMemo<KpiCardItem[]>(
776
+ () => [
777
+ {
778
+ key: 'wallets',
779
+ title: t('kpis.wallets'),
780
+ value: formatBitcodes(paginate.summary.walletCount),
781
+ description: t('kpis.walletsDescription'),
782
+ icon: Wallet,
783
+ },
784
+ {
785
+ key: 'balance',
786
+ title: t('kpis.balance'),
787
+ value: formatBitcodes(paginate.summary.totalBalance),
788
+ description: t('kpis.balanceDescription'),
789
+ icon: Coins,
790
+ },
791
+ {
792
+ key: 'transactions',
793
+ title: t('kpis.transactions'),
794
+ value: formatBitcodes(paginate.summary.transactionCount),
795
+ description: t('kpis.transactionsDescription'),
796
+ icon: ArrowUpRight,
797
+ },
798
+ {
799
+ key: 'movement',
800
+ title: t('kpis.movement'),
801
+ value: `${formatBitcodes(paginate.summary.totalCredits)} / ${formatBitcodes(
802
+ paginate.summary.totalDebits
803
+ )}`,
804
+ description: t('kpis.movementDescription'),
805
+ icon: ArrowDownLeft,
806
+ },
807
+ ],
808
+ [paginate.summary, t]
809
+ );
810
+
811
+ const handleDeleteWallet = async () => {
812
+ if (!walletToDelete) return;
813
+
814
+ try {
815
+ setIsDeletingWallet(true);
816
+ await request({
817
+ url: `/lms/bitcode-wallets/${walletToDelete.id}`,
818
+ method: 'DELETE',
819
+ });
820
+ toast.success(t('messages.walletDeleted'));
821
+ setDeleteDialogOpen(false);
822
+ setWalletToDelete(null);
823
+ await refetch();
824
+ } catch (error: unknown) {
825
+ toast.error(getErrorMessage(error, t('messages.walletDeleteError')));
826
+ } finally {
827
+ setIsDeletingWallet(false);
828
+ }
829
+ };
830
+
831
+ return (
832
+ <Page>
833
+ <PageHeader
834
+ breadcrumbs={[
835
+ { label: t('breadcrumbs.home'), href: '/' },
836
+ { label: t('breadcrumbs.lms'), href: '/lms' },
837
+ { label: t('breadcrumbs.current') },
838
+ ]}
839
+ title={t('title')}
840
+ description={t('description')}
841
+ actions={[
842
+ {
843
+ label: t('actions.createWallet'),
844
+ onClick: () => {
845
+ setWalletToEdit(null);
846
+ setSheetOpen(true);
847
+ },
848
+ icon: <Plus className="h-4 w-4" />,
849
+ },
850
+ ]}
851
+ />
852
+
853
+ <KpiCardsGrid items={kpiItems} />
854
+
855
+ <SearchBar
856
+ searchQuery={searchInput}
857
+ onSearchChange={(value) => setSearchInput(value)}
858
+ onSearch={() => setPage(1)}
859
+ placeholder={t('filters.searchPlaceholder')}
860
+ />
861
+
862
+ {isLoading ? (
863
+ <div className="space-y-3 p-4">
864
+ {Array.from({ length: 5 }).map((_, index) => (
865
+ <Skeleton key={index} className="h-16 w-full" />
866
+ ))}
867
+ </div>
868
+ ) : paginate.data.length === 0 ? (
869
+ <EmptyState
870
+ icon={<Wallet className="h-12 w-12" />}
871
+ title={t('empty.title')}
872
+ description={t('empty.description')}
873
+ actionLabel={t('actions.createWallet')}
874
+ actionIcon={<Plus className="mr-2 h-4 w-4" />}
875
+ onAction={() => {
876
+ setWalletToEdit(null);
877
+ setSheetOpen(true);
878
+ }}
879
+ />
880
+ ) : (
881
+ <div className="overflow-x-auto">
882
+ <Table>
883
+ <TableHeader>
884
+ <TableRow>
885
+ <TableHead>{t('table.person')}</TableHead>
886
+ <TableHead>{t('table.balance')}</TableHead>
887
+ <TableHead>{t('table.credits')}</TableHead>
888
+ <TableHead>{t('table.debits')}</TableHead>
889
+ <TableHead>{t('table.transactions')}</TableHead>
890
+ <TableHead>{t('table.updatedAt')}</TableHead>
891
+ <TableHead className="w-10" />
892
+ </TableRow>
893
+ </TableHeader>
894
+ <TableBody>
895
+ {paginate.data.map((wallet) => (
896
+ <TableRow
897
+ key={wallet.id}
898
+ className="cursor-pointer"
899
+ onDoubleClick={() => {
900
+ setWalletToEdit(wallet);
901
+ setSheetOpen(true);
902
+ }}
903
+ >
904
+ <TableCell>
905
+ <div className="min-w-0">
906
+ <div className="truncate font-medium">
907
+ {wallet.personName}
908
+ </div>
909
+ <div className="truncate text-xs text-muted-foreground">
910
+ {t('table.personId', { id: wallet.personId })}
911
+ </div>
912
+ </div>
913
+ </TableCell>
914
+ <TableCell className="font-semibold">
915
+ {formatBitcodes(wallet.currentBalance)}
916
+ </TableCell>
917
+ <TableCell className="text-emerald-600">
918
+ +{formatBitcodes(wallet.totalCredits)}
919
+ </TableCell>
920
+ <TableCell className="text-rose-600">
921
+ -{formatBitcodes(wallet.totalDebits)}
922
+ </TableCell>
923
+ <TableCell>
924
+ {formatBitcodes(wallet.transactionCount)}
925
+ </TableCell>
926
+ <TableCell>{formatDateTime(wallet.updatedAt)}</TableCell>
927
+ <TableCell>
928
+ <DropdownMenu>
929
+ <DropdownMenuTrigger asChild>
930
+ <Button variant="ghost" size="icon" className="h-8 w-8">
931
+ <MoreHorizontal className="h-4 w-4" />
932
+ </Button>
933
+ </DropdownMenuTrigger>
934
+ <DropdownMenuContent align="end">
935
+ <DropdownMenuItem
936
+ onClick={() => {
937
+ setWalletToEdit(wallet);
938
+ setSheetOpen(true);
939
+ }}
940
+ >
941
+ <Pencil className="mr-2 h-4 w-4" />
942
+ {t('actions.edit')}
943
+ </DropdownMenuItem>
944
+ <DropdownMenuSeparator />
945
+ <DropdownMenuItem
946
+ className="text-red-600"
947
+ onClick={() => {
948
+ setWalletToDelete(wallet);
949
+ setDeleteDialogOpen(true);
950
+ }}
951
+ >
952
+ <Trash2 className="mr-2 h-4 w-4" />
953
+ {t('actions.delete')}
954
+ </DropdownMenuItem>
955
+ </DropdownMenuContent>
956
+ </DropdownMenu>
957
+ </TableCell>
958
+ </TableRow>
959
+ ))}
960
+ </TableBody>
961
+ </Table>
962
+ </div>
963
+ )}
964
+
965
+ <PaginationFooter
966
+ currentPage={page}
967
+ pageSize={pageSize}
968
+ totalItems={paginate.total}
969
+ onPageChange={setPage}
970
+ onPageSizeChange={(nextPageSize) => {
971
+ setPageSize(nextPageSize);
972
+ setPage(1);
973
+ }}
974
+ pageSizeOptions={[6, 12, 24]}
975
+ />
976
+
977
+ <WalletFormSheet
978
+ open={sheetOpen}
979
+ onOpenChange={setSheetOpen}
980
+ walletToEdit={walletToEdit}
981
+ onSaved={() => refetch()}
982
+ />
983
+
984
+ <AlertDialog open={deleteDialogOpen} onOpenChange={setDeleteDialogOpen}>
985
+ <AlertDialogContent>
986
+ <AlertDialogHeader>
987
+ <AlertDialogTitle>{t('deleteDialog.title')}</AlertDialogTitle>
988
+ <AlertDialogDescription>
989
+ {t('deleteDialog.description', {
990
+ name: walletToDelete?.personName ?? '',
991
+ })}
992
+ </AlertDialogDescription>
993
+ </AlertDialogHeader>
994
+ <div className="flex justify-end gap-2">
995
+ <AlertDialogCancel disabled={isDeletingWallet}>
996
+ {t('actions.cancel')}
997
+ </AlertDialogCancel>
998
+ <AlertDialogAction
999
+ onClick={handleDeleteWallet}
1000
+ disabled={isDeletingWallet}
1001
+ className="bg-red-600 hover:bg-red-700"
1002
+ >
1003
+ {isDeletingWallet ? t('actions.deleting') : t('actions.delete')}
1004
+ </AlertDialogAction>
1005
+ </div>
1006
+ </AlertDialogContent>
1007
+ </AlertDialog>
1008
+ </Page>
1009
+ );
1010
+ }