@hed-hog/lms 0.0.349 → 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 (409) 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.mcp-tools.d.ts +87 -0
  77. package/dist/class-group/class-group.mcp-tools.d.ts.map +1 -0
  78. package/dist/class-group/class-group.mcp-tools.js +553 -0
  79. package/dist/class-group/class-group.mcp-tools.js.map +1 -0
  80. package/dist/class-group/class-group.module.d.ts.map +1 -1
  81. package/dist/class-group/class-group.module.js +2 -1
  82. package/dist/class-group/class-group.module.js.map +1 -1
  83. package/dist/class-group/class-group.service.d.ts +3 -1
  84. package/dist/class-group/class-group.service.d.ts.map +1 -1
  85. package/dist/class-group/class-group.service.js +45 -2
  86. package/dist/class-group/class-group.service.js.map +1 -1
  87. package/dist/course/course-operations-integration.service.d.ts +40 -0
  88. package/dist/course/course-operations-integration.service.d.ts.map +1 -0
  89. package/dist/course/course-operations-integration.service.js +372 -0
  90. package/dist/course/course-operations-integration.service.js.map +1 -0
  91. package/dist/course/course-structure.controller.d.ts +43 -4
  92. package/dist/course/course-structure.controller.d.ts.map +1 -1
  93. package/dist/course/course-structure.controller.js +22 -0
  94. package/dist/course/course-structure.controller.js.map +1 -1
  95. package/dist/course/course-structure.service.d.ts +42 -1
  96. package/dist/course/course-structure.service.d.ts.map +1 -1
  97. package/dist/course/course-structure.service.js +199 -32
  98. package/dist/course/course-structure.service.js.map +1 -1
  99. package/dist/course/course.controller.d.ts +12 -0
  100. package/dist/course/course.controller.d.ts.map +1 -1
  101. package/dist/course/course.mcp-tools.d.ts +90 -0
  102. package/dist/course/course.mcp-tools.d.ts.map +1 -0
  103. package/dist/course/course.mcp-tools.js +520 -0
  104. package/dist/course/course.mcp-tools.js.map +1 -0
  105. package/dist/course/course.module.d.ts.map +1 -1
  106. package/dist/course/course.module.js +8 -1
  107. package/dist/course/course.module.js.map +1 -1
  108. package/dist/course/course.service.d.ts +15 -1
  109. package/dist/course/course.service.d.ts.map +1 -1
  110. package/dist/course/course.service.js +70 -35
  111. package/dist/course/course.service.js.map +1 -1
  112. package/dist/course/dto/create-course.dto.d.ts +1 -0
  113. package/dist/course/dto/create-course.dto.d.ts.map +1 -1
  114. package/dist/course/dto/create-course.dto.js +7 -0
  115. package/dist/course/dto/create-course.dto.js.map +1 -1
  116. package/dist/course/dto/update-course-resources.dto.d.ts +11 -0
  117. package/dist/course/dto/update-course-resources.dto.d.ts.map +1 -0
  118. package/dist/course/dto/update-course-resources.dto.js +51 -0
  119. package/dist/course/dto/update-course-resources.dto.js.map +1 -0
  120. package/dist/course-lesson-discussion/course-lesson-discussion.controller.d.ts +23 -0
  121. package/dist/course-lesson-discussion/course-lesson-discussion.controller.d.ts.map +1 -0
  122. package/dist/course-lesson-discussion/course-lesson-discussion.controller.js +78 -0
  123. package/dist/course-lesson-discussion/course-lesson-discussion.controller.js.map +1 -0
  124. package/dist/course-lesson-discussion/course-lesson-discussion.mcp-tools.d.ts +22 -0
  125. package/dist/course-lesson-discussion/course-lesson-discussion.mcp-tools.d.ts.map +1 -0
  126. package/dist/course-lesson-discussion/course-lesson-discussion.mcp-tools.js +120 -0
  127. package/dist/course-lesson-discussion/course-lesson-discussion.mcp-tools.js.map +1 -0
  128. package/dist/course-lesson-discussion/course-lesson-discussion.module.d.ts +3 -0
  129. package/dist/course-lesson-discussion/course-lesson-discussion.module.d.ts.map +1 -0
  130. package/dist/course-lesson-discussion/course-lesson-discussion.module.js +26 -0
  131. package/dist/course-lesson-discussion/course-lesson-discussion.module.js.map +1 -0
  132. package/dist/course-lesson-discussion/course-lesson-discussion.service.d.ts +49 -0
  133. package/dist/course-lesson-discussion/course-lesson-discussion.service.d.ts.map +1 -0
  134. package/dist/course-lesson-discussion/course-lesson-discussion.service.js +272 -0
  135. package/dist/course-lesson-discussion/course-lesson-discussion.service.js.map +1 -0
  136. package/dist/course-lesson-discussion/dto/create-course-lesson-discussion-topic.dto.d.ts +6 -0
  137. package/dist/course-lesson-discussion/dto/create-course-lesson-discussion-topic.dto.d.ts.map +1 -0
  138. package/dist/course-lesson-discussion/dto/create-course-lesson-discussion-topic.dto.js +33 -0
  139. package/dist/course-lesson-discussion/dto/create-course-lesson-discussion-topic.dto.js.map +1 -0
  140. package/dist/course-lesson-note/course-lesson-note.controller.d.ts +53 -0
  141. package/dist/course-lesson-note/course-lesson-note.controller.d.ts.map +1 -0
  142. package/dist/course-lesson-note/course-lesson-note.controller.js +93 -0
  143. package/dist/course-lesson-note/course-lesson-note.controller.js.map +1 -0
  144. package/dist/course-lesson-note/course-lesson-note.mcp-tools.d.ts +27 -0
  145. package/dist/course-lesson-note/course-lesson-note.mcp-tools.d.ts.map +1 -0
  146. package/dist/course-lesson-note/course-lesson-note.mcp-tools.js +145 -0
  147. package/dist/course-lesson-note/course-lesson-note.mcp-tools.js.map +1 -0
  148. package/dist/course-lesson-note/course-lesson-note.module.d.ts +3 -0
  149. package/dist/course-lesson-note/course-lesson-note.module.d.ts.map +1 -0
  150. package/dist/course-lesson-note/course-lesson-note.module.js +26 -0
  151. package/dist/course-lesson-note/course-lesson-note.module.js.map +1 -0
  152. package/dist/course-lesson-note/course-lesson-note.service.d.ts +59 -0
  153. package/dist/course-lesson-note/course-lesson-note.service.d.ts.map +1 -0
  154. package/dist/course-lesson-note/course-lesson-note.service.js +195 -0
  155. package/dist/course-lesson-note/course-lesson-note.service.js.map +1 -0
  156. package/dist/course-lesson-note/dto/create-course-lesson-note.dto.d.ts +6 -0
  157. package/dist/course-lesson-note/dto/create-course-lesson-note.dto.d.ts.map +1 -0
  158. package/dist/course-lesson-note/dto/create-course-lesson-note.dto.js +33 -0
  159. package/dist/course-lesson-note/dto/create-course-lesson-note.dto.js.map +1 -0
  160. package/dist/course-lesson-note/dto/update-course-lesson-note.dto.d.ts +6 -0
  161. package/dist/course-lesson-note/dto/update-course-lesson-note.dto.d.ts.map +1 -0
  162. package/dist/course-lesson-note/dto/update-course-lesson-note.dto.js +35 -0
  163. package/dist/course-lesson-note/dto/update-course-lesson-note.dto.js.map +1 -0
  164. package/dist/dashboard/dashboard.mcp-tools.d.ts +10 -0
  165. package/dist/dashboard/dashboard.mcp-tools.d.ts.map +1 -0
  166. package/dist/dashboard/dashboard.mcp-tools.js +46 -0
  167. package/dist/dashboard/dashboard.mcp-tools.js.map +1 -0
  168. package/dist/dashboard/dashboard.module.d.ts.map +1 -1
  169. package/dist/dashboard/dashboard.module.js +2 -1
  170. package/dist/dashboard/dashboard.module.js.map +1 -1
  171. package/dist/enterprise/enterprise.mcp-tools.d.ts +82 -0
  172. package/dist/enterprise/enterprise.mcp-tools.d.ts.map +1 -0
  173. package/dist/enterprise/enterprise.mcp-tools.js +516 -0
  174. package/dist/enterprise/enterprise.mcp-tools.js.map +1 -0
  175. package/dist/enterprise/enterprise.module.d.ts.map +1 -1
  176. package/dist/enterprise/enterprise.module.js +2 -1
  177. package/dist/enterprise/enterprise.module.js.map +1 -1
  178. package/dist/enterprise/training/enterprise-training.module.d.ts.map +1 -1
  179. package/dist/enterprise/training/enterprise-training.module.js +11 -1
  180. package/dist/enterprise/training/enterprise-training.module.js.map +1 -1
  181. package/dist/enterprise/training/training-admin.mcp-tools.d.ts +79 -0
  182. package/dist/enterprise/training/training-admin.mcp-tools.d.ts.map +1 -0
  183. package/dist/enterprise/training/training-admin.mcp-tools.js +620 -0
  184. package/dist/enterprise/training/training-admin.mcp-tools.js.map +1 -0
  185. package/dist/enterprise/training/training-instructor.mcp-tools.d.ts +47 -0
  186. package/dist/enterprise/training/training-instructor.mcp-tools.d.ts.map +1 -0
  187. package/dist/enterprise/training/training-instructor.mcp-tools.js +275 -0
  188. package/dist/enterprise/training/training-instructor.mcp-tools.js.map +1 -0
  189. package/dist/enterprise/training/training-student.controller.d.ts +24 -0
  190. package/dist/enterprise/training/training-student.controller.d.ts.map +1 -1
  191. package/dist/enterprise/training/training-student.controller.js +22 -0
  192. package/dist/enterprise/training/training-student.controller.js.map +1 -1
  193. package/dist/enterprise/training/training-student.mcp-tools.d.ts +27 -0
  194. package/dist/enterprise/training/training-student.mcp-tools.d.ts.map +1 -0
  195. package/dist/enterprise/training/training-student.mcp-tools.js +186 -0
  196. package/dist/enterprise/training/training-student.mcp-tools.js.map +1 -0
  197. package/dist/enterprise/training/training-student.service.d.ts +32 -0
  198. package/dist/enterprise/training/training-student.service.d.ts.map +1 -1
  199. package/dist/enterprise/training/training-student.service.js +138 -0
  200. package/dist/enterprise/training/training-student.service.js.map +1 -1
  201. package/dist/evaluation/evaluation.mcp-tools.d.ts +25 -0
  202. package/dist/evaluation/evaluation.mcp-tools.d.ts.map +1 -0
  203. package/dist/evaluation/evaluation.mcp-tools.js +220 -0
  204. package/dist/evaluation/evaluation.mcp-tools.js.map +1 -0
  205. package/dist/evaluation/evaluation.module.d.ts.map +1 -1
  206. package/dist/evaluation/evaluation.module.js +2 -1
  207. package/dist/evaluation/evaluation.module.js.map +1 -1
  208. package/dist/exam/dto/create-exam-question.dto.d.ts +2 -0
  209. package/dist/exam/dto/create-exam-question.dto.d.ts.map +1 -1
  210. package/dist/exam/dto/create-exam-question.dto.js +10 -0
  211. package/dist/exam/dto/create-exam-question.dto.js.map +1 -1
  212. package/dist/exam/dto/create-exam.dto.d.ts +2 -0
  213. package/dist/exam/dto/create-exam.dto.d.ts.map +1 -1
  214. package/dist/exam/dto/create-exam.dto.js +10 -0
  215. package/dist/exam/dto/create-exam.dto.js.map +1 -1
  216. package/dist/exam/dto/create-question-subject.dto.d.ts +5 -0
  217. package/dist/exam/dto/create-question-subject.dto.d.ts.map +1 -0
  218. package/dist/exam/dto/create-question-subject.dto.js +28 -0
  219. package/dist/exam/dto/create-question-subject.dto.js.map +1 -0
  220. package/dist/exam/exam-attempt.controller.d.ts +4 -0
  221. package/dist/exam/exam-attempt.controller.d.ts.map +1 -1
  222. package/dist/exam/exam-attempt.service.d.ts +7 -1
  223. package/dist/exam/exam-attempt.service.d.ts.map +1 -1
  224. package/dist/exam/exam-attempt.service.js +47 -17
  225. package/dist/exam/exam-attempt.service.js.map +1 -1
  226. package/dist/exam/exam.controller.d.ts +34 -0
  227. package/dist/exam/exam.controller.d.ts.map +1 -1
  228. package/dist/exam/exam.controller.js +27 -0
  229. package/dist/exam/exam.controller.js.map +1 -1
  230. package/dist/exam/exam.mcp-tools.d.ts +62 -0
  231. package/dist/exam/exam.mcp-tools.d.ts.map +1 -0
  232. package/dist/exam/exam.mcp-tools.js +430 -0
  233. package/dist/exam/exam.mcp-tools.js.map +1 -0
  234. package/dist/exam/exam.module.d.ts.map +1 -1
  235. package/dist/exam/exam.module.js +2 -1
  236. package/dist/exam/exam.module.js.map +1 -1
  237. package/dist/exam/exam.service.d.ts +38 -0
  238. package/dist/exam/exam.service.d.ts.map +1 -1
  239. package/dist/exam/exam.service.js +114 -17
  240. package/dist/exam/exam.service.js.map +1 -1
  241. package/dist/index.d.ts +9 -0
  242. package/dist/index.d.ts.map +1 -1
  243. package/dist/index.js +9 -0
  244. package/dist/index.js.map +1 -1
  245. package/dist/instructor/instructor.mcp-tools.d.ts +41 -0
  246. package/dist/instructor/instructor.mcp-tools.d.ts.map +1 -0
  247. package/dist/instructor/instructor.mcp-tools.js +326 -0
  248. package/dist/instructor/instructor.mcp-tools.js.map +1 -0
  249. package/dist/instructor/instructor.module.d.ts.map +1 -1
  250. package/dist/instructor/instructor.module.js +2 -1
  251. package/dist/instructor/instructor.module.js.map +1 -1
  252. package/dist/lms.module.d.ts.map +1 -1
  253. package/dist/lms.module.js +15 -0
  254. package/dist/lms.module.js.map +1 -1
  255. package/dist/realtime/lms-realtime.controller.d.ts +7 -0
  256. package/dist/realtime/lms-realtime.controller.d.ts.map +1 -0
  257. package/dist/realtime/lms-realtime.controller.js +34 -0
  258. package/dist/realtime/lms-realtime.controller.js.map +1 -0
  259. package/dist/realtime/lms-realtime.module.d.ts +3 -0
  260. package/dist/realtime/lms-realtime.module.d.ts.map +1 -0
  261. package/dist/realtime/lms-realtime.module.js +25 -0
  262. package/dist/realtime/lms-realtime.module.js.map +1 -0
  263. package/dist/realtime/lms-realtime.service.d.ts +36 -0
  264. package/dist/realtime/lms-realtime.service.d.ts.map +1 -0
  265. package/dist/realtime/lms-realtime.service.js +59 -0
  266. package/dist/realtime/lms-realtime.service.js.map +1 -0
  267. package/dist/realtime/lms-realtime.subscriber.d.ts +10 -0
  268. package/dist/realtime/lms-realtime.subscriber.d.ts.map +1 -0
  269. package/dist/realtime/lms-realtime.subscriber.js +70 -0
  270. package/dist/realtime/lms-realtime.subscriber.js.map +1 -0
  271. package/dist/reports/reports.mcp-tools.d.ts +10 -0
  272. package/dist/reports/reports.mcp-tools.d.ts.map +1 -0
  273. package/dist/reports/reports.mcp-tools.js +50 -0
  274. package/dist/reports/reports.mcp-tools.js.map +1 -0
  275. package/dist/reports/reports.module.d.ts.map +1 -1
  276. package/dist/reports/reports.module.js +2 -1
  277. package/dist/reports/reports.module.js.map +1 -1
  278. package/dist/training/training.mcp-tools.d.ts +20 -0
  279. package/dist/training/training.mcp-tools.d.ts.map +1 -0
  280. package/dist/training/training.mcp-tools.js +181 -0
  281. package/dist/training/training.mcp-tools.js.map +1 -0
  282. package/dist/training/training.module.d.ts.map +1 -1
  283. package/dist/training/training.module.js +2 -1
  284. package/dist/training/training.module.js.map +1 -1
  285. package/hedhog/data/integration_event_catalog.yaml +69 -0
  286. package/hedhog/data/menu.yaml +34 -0
  287. package/hedhog/data/route.yaml +2351 -0
  288. package/hedhog/frontend/app/_components/class-form-sheet.tsx.ejs +168 -103
  289. package/hedhog/frontend/app/_components/course-form-sheet.tsx.ejs +80 -1
  290. package/hedhog/frontend/app/_components/course-picker.tsx.ejs +228 -0
  291. package/hedhog/frontend/app/_lib/hooks/use-lms-realtime-refresh.ts.ejs +58 -0
  292. package/hedhog/frontend/app/achievements/page.tsx.ejs +844 -0
  293. package/hedhog/frontend/app/bitcodes/page.tsx.ejs +1010 -0
  294. package/hedhog/frontend/app/certificates/issued/page.tsx.ejs +61 -2
  295. package/hedhog/frontend/app/classes/[id]/page.tsx.ejs +7 -0
  296. package/hedhog/frontend/app/classes/page.tsx.ejs +55 -2060
  297. package/hedhog/frontend/app/courses/[id]/_components/CourseRelationsCard.tsx.ejs +28 -0
  298. package/hedhog/frontend/app/courses/[id]/_components/course-edit-types.ts.ejs +1 -0
  299. package/hedhog/frontend/app/courses/[id]/page.tsx.ejs +0 -9
  300. package/hedhog/frontend/app/courses/[id]/structure/_components/course-tree-dnd.tsx.ejs +80 -66
  301. package/hedhog/frontend/app/courses/[id]/structure/_components/editor-course.tsx.ejs +583 -8
  302. package/hedhog/frontend/app/courses/[id]/structure/_components/editor-lesson.tsx.ejs +527 -57
  303. package/hedhog/frontend/app/courses/[id]/structure/_components/mock-data.ts.ejs +1 -1
  304. package/hedhog/frontend/app/courses/[id]/structure/_components/tree-context-menu.tsx.ejs +106 -4
  305. package/hedhog/frontend/app/courses/[id]/structure/_components/types.ts.ejs +2 -1
  306. package/hedhog/frontend/app/courses/[id]/structure/_data/adapters/course-structure.adapter.ts.ejs +11 -1
  307. package/hedhog/frontend/app/courses/[id]/structure/_data/services/course-structure.service.ts.ejs +53 -6
  308. package/hedhog/frontend/app/courses/[id]/structure/_data/types/api-course.types.ts.ejs +13 -2
  309. package/hedhog/frontend/app/courses/page.tsx.ejs +175 -29
  310. package/hedhog/frontend/app/enterprise/_components/enterprise-course-create-sheet.tsx.ejs +3 -0
  311. package/hedhog/frontend/app/enterprise/_components/enterprise-course-edit-sheet.tsx.ejs +7 -0
  312. package/hedhog/frontend/app/exams/[id]/questions/page.tsx.ejs +169 -2
  313. package/hedhog/frontend/app/exams/page.tsx.ejs +77 -22
  314. package/hedhog/frontend/app/instructor-skills/page.tsx.ejs +1 -0
  315. package/hedhog/frontend/app/instructors/page.tsx.ejs +1 -0
  316. package/hedhog/frontend/app/paths/page.tsx.ejs +6 -24
  317. package/hedhog/frontend/app/training/page.tsx.ejs +6 -24
  318. package/hedhog/frontend/messages/en.json +314 -12
  319. package/hedhog/frontend/messages/pt.json +314 -12
  320. package/hedhog/query/triggers.sql +53 -0
  321. package/hedhog/table/achievement.yaml +46 -0
  322. package/hedhog/table/bitcode_wallet.yaml +18 -0
  323. package/hedhog/table/bitcode_wallet_transaction.yaml +22 -0
  324. package/hedhog/table/certificate.yaml +3 -0
  325. package/hedhog/table/course.yaml +4 -0
  326. package/hedhog/table/course_file.yaml +23 -0
  327. package/hedhog/table/course_lesson.yaml +5 -0
  328. package/hedhog/table/course_lesson_discussion_like.yaml +21 -0
  329. package/hedhog/table/course_lesson_discussion_topic.yaml +35 -0
  330. package/hedhog/table/course_lesson_note.yaml +34 -0
  331. package/hedhog/table/exam.yaml +5 -0
  332. package/hedhog/table/learning_path_enrollment.yaml +6 -0
  333. package/hedhog/table/question.yaml +10 -0
  334. package/hedhog/table/question_subject.yaml +17 -0
  335. package/hedhog/table/student_activity_streak.yaml +25 -0
  336. package/package.json +6 -6
  337. package/src/achievement/achievement.controller.ts +60 -0
  338. package/src/achievement/achievement.mcp-tools.ts +108 -0
  339. package/src/achievement/achievement.module.ts +13 -0
  340. package/src/achievement/achievement.service.ts +252 -0
  341. package/src/achievement/dto/create-achievement.dto.ts +50 -0
  342. package/src/achievement/dto/update-achievement.dto.ts +47 -0
  343. package/src/bitcode-wallet/bitcode-wallet.controller.ts +69 -0
  344. package/src/bitcode-wallet/bitcode-wallet.mcp-tools.ts +107 -0
  345. package/src/bitcode-wallet/bitcode-wallet.module.ts +13 -0
  346. package/src/bitcode-wallet/bitcode-wallet.service.ts +361 -0
  347. package/src/bitcode-wallet/dto/create-bitcode-wallet-transaction.dto.ts +27 -0
  348. package/src/bitcode-wallet/dto/create-bitcode-wallet.dto.ts +7 -0
  349. package/src/bitcode-wallet/dto/update-bitcode-wallet-transaction.dto.ts +28 -0
  350. package/src/bitcode-wallet/dto/update-bitcode-wallet.dto.ts +8 -0
  351. package/src/certificate/certificate.controller.ts +20 -11
  352. package/src/certificate/certificate.mcp-tools.ts +131 -0
  353. package/src/certificate/certificate.module.ts +2 -1
  354. package/src/certificate/certificate.service.ts +95 -4
  355. package/src/certificate/dto/update-certificate-public-access.dto.ts +6 -0
  356. package/src/class-group/class-group.mcp-tools.ts +435 -0
  357. package/src/class-group/class-group.module.ts +2 -1
  358. package/src/class-group/class-group.service.ts +51 -1
  359. package/src/course/course-operations-integration.service.ts +520 -0
  360. package/src/course/course-structure.controller.ts +22 -8
  361. package/src/course/course-structure.service.ts +215 -23
  362. package/src/course/course.mcp-tools.ts +409 -0
  363. package/src/course/course.module.ts +8 -1
  364. package/src/course/course.service.ts +106 -27
  365. package/src/course/dto/create-course.dto.ts +8 -0
  366. package/src/course/dto/update-course-resources.dto.ts +39 -0
  367. package/src/course-lesson-discussion/course-lesson-discussion.controller.ts +55 -0
  368. package/src/course-lesson-discussion/course-lesson-discussion.mcp-tools.ts +75 -0
  369. package/src/course-lesson-discussion/course-lesson-discussion.module.ts +13 -0
  370. package/src/course-lesson-discussion/course-lesson-discussion.service.ts +354 -0
  371. package/src/course-lesson-discussion/dto/create-course-lesson-discussion-topic.dto.ts +16 -0
  372. package/src/course-lesson-note/course-lesson-note.controller.ts +68 -0
  373. package/src/course-lesson-note/course-lesson-note.mcp-tools.ts +96 -0
  374. package/src/course-lesson-note/course-lesson-note.module.ts +13 -0
  375. package/src/course-lesson-note/course-lesson-note.service.ts +248 -0
  376. package/src/course-lesson-note/dto/create-course-lesson-note.dto.ts +16 -0
  377. package/src/course-lesson-note/dto/update-course-lesson-note.dto.ts +18 -0
  378. package/src/dashboard/dashboard.mcp-tools.ts +23 -0
  379. package/src/dashboard/dashboard.module.ts +2 -1
  380. package/src/enterprise/enterprise.mcp-tools.ts +403 -0
  381. package/src/enterprise/enterprise.module.ts +2 -1
  382. package/src/enterprise/training/enterprise-training.module.ts +11 -1
  383. package/src/enterprise/training/training-admin.mcp-tools.ts +479 -0
  384. package/src/enterprise/training/training-instructor.mcp-tools.ts +210 -0
  385. package/src/enterprise/training/training-student.controller.ts +17 -1
  386. package/src/enterprise/training/training-student.mcp-tools.ts +136 -0
  387. package/src/enterprise/training/training-student.service.ts +167 -1
  388. package/src/evaluation/evaluation.mcp-tools.ts +155 -0
  389. package/src/evaluation/evaluation.module.ts +2 -1
  390. package/src/exam/dto/create-exam-question.dto.ts +8 -0
  391. package/src/exam/dto/create-exam.dto.ts +8 -0
  392. package/src/exam/dto/create-question-subject.dto.ts +12 -0
  393. package/src/exam/exam-attempt.service.ts +46 -14
  394. package/src/exam/exam.controller.ts +19 -0
  395. package/src/exam/exam.mcp-tools.ts +337 -0
  396. package/src/exam/exam.module.ts +2 -1
  397. package/src/exam/exam.service.ts +121 -0
  398. package/src/index.ts +9 -0
  399. package/src/instructor/instructor.mcp-tools.ts +243 -0
  400. package/src/instructor/instructor.module.ts +2 -1
  401. package/src/lms.module.ts +15 -1
  402. package/src/realtime/lms-realtime.controller.ts +12 -0
  403. package/src/realtime/lms-realtime.module.ts +12 -0
  404. package/src/realtime/lms-realtime.service.ts +98 -0
  405. package/src/realtime/lms-realtime.subscriber.ts +61 -0
  406. package/src/reports/reports.mcp-tools.ts +27 -0
  407. package/src/reports/reports.module.ts +2 -1
  408. package/src/training/training.mcp-tools.ts +128 -0
  409. package/src/training/training.module.ts +2 -1
@@ -0,0 +1,844 @@
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 { Checkbox } from '@/components/ui/checkbox';
22
+ import {
23
+ DropdownMenu,
24
+ DropdownMenuContent,
25
+ DropdownMenuItem,
26
+ DropdownMenuSeparator,
27
+ DropdownMenuTrigger,
28
+ } from '@/components/ui/dropdown-menu';
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 {
39
+ Select,
40
+ SelectContent,
41
+ SelectItem,
42
+ SelectTrigger,
43
+ SelectValue,
44
+ } from '@/components/ui/select';
45
+ import {
46
+ Sheet,
47
+ SheetContent,
48
+ SheetHeader,
49
+ SheetTitle,
50
+ } from '@/components/ui/sheet';
51
+ import { Skeleton } from '@/components/ui/skeleton';
52
+ import {
53
+ Table,
54
+ TableBody,
55
+ TableCell,
56
+ TableHead,
57
+ TableHeader,
58
+ TableRow,
59
+ } from '@/components/ui/table';
60
+ import { Textarea } from '@/components/ui/textarea';
61
+ import { usePersistedPageSize } from '@/hooks/use-persisted-page-size';
62
+ import { cn } from '@/lib/utils';
63
+ import { useApp, useQuery } from '@hed-hog/next-app-provider';
64
+ import { zodResolver } from '@hookform/resolvers/zod';
65
+ import {
66
+ ImageIcon,
67
+ Loader2,
68
+ MoreHorizontal,
69
+ Pencil,
70
+ Plus,
71
+ Trophy,
72
+ Trash2,
73
+ Upload,
74
+ X,
75
+ } from 'lucide-react';
76
+ import { useTranslations } from 'next-intl';
77
+ import Image from 'next/image';
78
+ import { useEffect, useRef, useState } from 'react';
79
+ import { useForm } from 'react-hook-form';
80
+ import { toast } from 'sonner';
81
+ import * as z from 'zod';
82
+
83
+ const ACHIEVEMENT_TYPE_VALUES = [
84
+ 'course_completion',
85
+ 'learning_path_completion',
86
+ 'exam_approval',
87
+ 'attendance',
88
+ 'engagement',
89
+ 'custom',
90
+ ] as const;
91
+
92
+ type AchievementType = (typeof ACHIEVEMENT_TYPE_VALUES)[number];
93
+
94
+ type Achievement = {
95
+ id: number;
96
+ name: string;
97
+ description?: string | null;
98
+ iconFileId?: number | null;
99
+ mainImageFileId?: number | null;
100
+ type: AchievementType;
101
+ bitCodesValue: number;
102
+ canReceiveMultipleTimes: boolean;
103
+ };
104
+
105
+ type AchievementPaginatedResult = {
106
+ data: Achievement[];
107
+ total: number;
108
+ page: number;
109
+ pageSize: number;
110
+ lastPage?: number;
111
+ };
112
+
113
+ const achievementSchema = (t: ReturnType<typeof useTranslations>) =>
114
+ z.object({
115
+ name: z
116
+ .string()
117
+ .min(1, t('form.validation.required'))
118
+ .max(255, t('form.validation.max255')),
119
+ description: z.string().optional(),
120
+ iconFileId: z.number().nullable(),
121
+ mainImageFileId: z.number().nullable(),
122
+ type: z.enum(ACHIEVEMENT_TYPE_VALUES),
123
+ bitCodesValue: z.coerce
124
+ .number()
125
+ .int(t('form.validation.integer'))
126
+ .min(0, t('form.validation.nonNegative')),
127
+ canReceiveMultipleTimes: z.boolean(),
128
+ });
129
+
130
+ type AchievementFormValues = z.infer<ReturnType<typeof achievementSchema>>;
131
+
132
+ function getFileUrl(fileId?: number | null) {
133
+ return fileId && process.env.NEXT_PUBLIC_API_BASE_URL
134
+ ? `${process.env.NEXT_PUBLIC_API_BASE_URL}/file/open/${fileId}`
135
+ : null;
136
+ }
137
+
138
+ function ImageUploadField({
139
+ label,
140
+ description,
141
+ fileId,
142
+ previewUrl,
143
+ uploading,
144
+ uploadLabel,
145
+ replaceLabel,
146
+ removeLabel,
147
+ onPick,
148
+ onRemove,
149
+ inputRef,
150
+ }: {
151
+ label: string;
152
+ description: string;
153
+ fileId?: number | null;
154
+ previewUrl: string | null;
155
+ uploading: boolean;
156
+ uploadLabel: string;
157
+ replaceLabel: string;
158
+ removeLabel: string;
159
+ onPick: (event: React.ChangeEvent<HTMLInputElement>) => Promise<void>;
160
+ onRemove: () => void;
161
+ inputRef: React.RefObject<HTMLInputElement | null>;
162
+ }) {
163
+ return (
164
+ <div className="space-y-2">
165
+ <FormLabel>{label}</FormLabel>
166
+ <div className="flex items-start gap-4">
167
+ {previewUrl ? (
168
+ <Image
169
+ src={previewUrl}
170
+ alt={label}
171
+ width={64}
172
+ height={64}
173
+ unoptimized
174
+ className="size-16 shrink-0 rounded-lg border object-cover"
175
+ />
176
+ ) : (
177
+ <div className="flex size-16 shrink-0 items-center justify-center rounded-lg border border-dashed bg-muted/40">
178
+ <ImageIcon className="size-6 text-muted-foreground/50" />
179
+ </div>
180
+ )}
181
+ <div className="flex flex-col gap-2">
182
+ <input
183
+ ref={inputRef}
184
+ type="file"
185
+ accept="image/*"
186
+ className="hidden"
187
+ onChange={(event) => void onPick(event)}
188
+ />
189
+ <Button
190
+ type="button"
191
+ variant="outline"
192
+ size="sm"
193
+ disabled={uploading}
194
+ onClick={() => inputRef.current?.click()}
195
+ className="gap-2"
196
+ >
197
+ {uploading ? (
198
+ <Loader2 className="size-3.5 animate-spin" />
199
+ ) : (
200
+ <Upload className="size-3.5" />
201
+ )}
202
+ {fileId ? replaceLabel : uploadLabel}
203
+ </Button>
204
+ {fileId ? (
205
+ <Button
206
+ type="button"
207
+ variant="ghost"
208
+ size="sm"
209
+ onClick={onRemove}
210
+ className="gap-2 text-destructive hover:text-destructive"
211
+ >
212
+ <X className="size-3.5" />
213
+ {removeLabel}
214
+ </Button>
215
+ ) : null}
216
+ </div>
217
+ </div>
218
+ <p className="text-xs text-muted-foreground">{description}</p>
219
+ </div>
220
+ );
221
+ }
222
+
223
+ function AchievementFormSheet({
224
+ open,
225
+ onOpenChange,
226
+ achievementToEdit,
227
+ onSaved,
228
+ }: {
229
+ open: boolean;
230
+ onOpenChange: (open: boolean) => void;
231
+ achievementToEdit: Achievement | null;
232
+ onSaved: () => void;
233
+ }) {
234
+ const { request } = useApp();
235
+ const t = useTranslations('lms.AchievementsPage');
236
+ const [isSaving, setIsSaving] = useState(false);
237
+ const [iconUploading, setIconUploading] = useState(false);
238
+ const [mainImageUploading, setMainImageUploading] = useState(false);
239
+ const [iconPreviewUrl, setIconPreviewUrl] = useState<string | null>(null);
240
+ const [mainImagePreviewUrl, setMainImagePreviewUrl] = useState<string | null>(
241
+ null
242
+ );
243
+ const iconInputRef = useRef<HTMLInputElement>(null);
244
+ const mainImageInputRef = useRef<HTMLInputElement>(null);
245
+
246
+ const form = useForm<AchievementFormValues>({
247
+ resolver: zodResolver(achievementSchema(t)),
248
+ defaultValues: {
249
+ name: '',
250
+ description: '',
251
+ iconFileId: null,
252
+ mainImageFileId: null,
253
+ type: 'custom',
254
+ bitCodesValue: 0,
255
+ canReceiveMultipleTimes: false,
256
+ },
257
+ });
258
+
259
+ useEffect(() => {
260
+ if (!open) return;
261
+
262
+ form.reset({
263
+ name: achievementToEdit?.name ?? '',
264
+ description: achievementToEdit?.description ?? '',
265
+ iconFileId: achievementToEdit?.iconFileId ?? null,
266
+ mainImageFileId: achievementToEdit?.mainImageFileId ?? null,
267
+ type: achievementToEdit?.type ?? 'custom',
268
+ bitCodesValue: achievementToEdit?.bitCodesValue ?? 0,
269
+ canReceiveMultipleTimes:
270
+ achievementToEdit?.canReceiveMultipleTimes ?? false,
271
+ });
272
+
273
+ setIconPreviewUrl(getFileUrl(achievementToEdit?.iconFileId) ?? null);
274
+ setMainImagePreviewUrl(getFileUrl(achievementToEdit?.mainImageFileId) ?? null);
275
+ }, [achievementToEdit, form, open]);
276
+
277
+ async function uploadImage(file: File) {
278
+ const formData = new FormData();
279
+ formData.append('file', file);
280
+ const uploadRes = await request<{ id?: number }>({
281
+ url: '/file',
282
+ method: 'POST',
283
+ data: formData,
284
+ headers: { 'Content-Type': 'multipart/form-data' },
285
+ });
286
+ const fileId = uploadRes?.data?.id;
287
+ if (!fileId) throw new Error('invalid file id');
288
+ const openRes = await request<{ url?: string }>({
289
+ url: `/file/open/${fileId}`,
290
+ method: 'PUT',
291
+ });
292
+ return {
293
+ fileId,
294
+ url: openRes?.data?.url ?? getFileUrl(fileId),
295
+ };
296
+ }
297
+
298
+ async function handleIconPick(event: React.ChangeEvent<HTMLInputElement>) {
299
+ const file = event.target.files?.[0];
300
+ if (!file) return;
301
+ setIconUploading(true);
302
+ try {
303
+ const uploaded = await uploadImage(file);
304
+ form.setValue('iconFileId', uploaded.fileId, { shouldDirty: true });
305
+ setIconPreviewUrl(uploaded.url);
306
+ } catch {
307
+ toast.error(t('messages.uploadError'));
308
+ } finally {
309
+ setIconUploading(false);
310
+ if (iconInputRef.current) iconInputRef.current.value = '';
311
+ }
312
+ }
313
+
314
+ async function handleMainImagePick(
315
+ event: React.ChangeEvent<HTMLInputElement>
316
+ ) {
317
+ const file = event.target.files?.[0];
318
+ if (!file) return;
319
+ setMainImageUploading(true);
320
+ try {
321
+ const uploaded = await uploadImage(file);
322
+ form.setValue('mainImageFileId', uploaded.fileId, { shouldDirty: true });
323
+ setMainImagePreviewUrl(uploaded.url);
324
+ } catch {
325
+ toast.error(t('messages.uploadError'));
326
+ } finally {
327
+ setMainImageUploading(false);
328
+ if (mainImageInputRef.current) mainImageInputRef.current.value = '';
329
+ }
330
+ }
331
+
332
+ const onSubmit = async (values: AchievementFormValues) => {
333
+ try {
334
+ setIsSaving(true);
335
+
336
+ const payload = {
337
+ name: values.name,
338
+ description: values.description?.trim() || null,
339
+ iconFileId: values.iconFileId,
340
+ mainImageFileId: values.mainImageFileId,
341
+ type: values.type,
342
+ bitCodesValue: values.bitCodesValue,
343
+ canReceiveMultipleTimes: values.canReceiveMultipleTimes,
344
+ };
345
+
346
+ if (achievementToEdit) {
347
+ await request({
348
+ url: `/lms/achievements/${achievementToEdit.id}`,
349
+ method: 'PATCH',
350
+ data: payload,
351
+ });
352
+ toast.success(t('messages.updateSuccess'));
353
+ } else {
354
+ await request({
355
+ url: '/lms/achievements',
356
+ method: 'POST',
357
+ data: payload,
358
+ });
359
+ toast.success(t('messages.createSuccess'));
360
+ }
361
+
362
+ onSaved();
363
+ onOpenChange(false);
364
+ } catch {
365
+ toast.error(t('messages.saveError'));
366
+ } finally {
367
+ setIsSaving(false);
368
+ }
369
+ };
370
+
371
+ return (
372
+ <Sheet open={open} onOpenChange={onOpenChange}>
373
+ <SheetContent className="w-full overflow-y-auto sm:max-w-2xl">
374
+ <SheetHeader>
375
+ <SheetTitle>
376
+ {achievementToEdit ? t('sheet.editTitle') : t('sheet.createTitle')}
377
+ </SheetTitle>
378
+ </SheetHeader>
379
+
380
+ <Form {...form}>
381
+ <form
382
+ onSubmit={form.handleSubmit(onSubmit)}
383
+ className="flex flex-col gap-5 px-4 py-4"
384
+ >
385
+ <FormField
386
+ control={form.control}
387
+ name="name"
388
+ render={({ field }) => (
389
+ <FormItem>
390
+ <FormLabel>{t('form.name')}</FormLabel>
391
+ <FormControl>
392
+ <Input placeholder={t('form.namePlaceholder')} {...field} />
393
+ </FormControl>
394
+ <FormMessage />
395
+ </FormItem>
396
+ )}
397
+ />
398
+
399
+ <FormField
400
+ control={form.control}
401
+ name="description"
402
+ render={({ field }) => (
403
+ <FormItem>
404
+ <FormLabel>{t('form.description')}</FormLabel>
405
+ <FormControl>
406
+ <Textarea
407
+ rows={4}
408
+ placeholder={t('form.descriptionPlaceholder')}
409
+ {...field}
410
+ value={field.value ?? ''}
411
+ />
412
+ </FormControl>
413
+ <FormMessage />
414
+ </FormItem>
415
+ )}
416
+ />
417
+
418
+ <div className="grid gap-5 md:grid-cols-2">
419
+ <ImageUploadField
420
+ label={t('form.iconImage')}
421
+ description={t('form.iconImageDescription')}
422
+ fileId={form.watch('iconFileId')}
423
+ previewUrl={iconPreviewUrl}
424
+ uploading={iconUploading}
425
+ uploadLabel={t('form.imageUpload')}
426
+ replaceLabel={t('form.imageReplace')}
427
+ removeLabel={t('form.imageRemove')}
428
+ onPick={handleIconPick}
429
+ onRemove={() => {
430
+ form.setValue('iconFileId', null, { shouldDirty: true });
431
+ setIconPreviewUrl(null);
432
+ }}
433
+ inputRef={iconInputRef}
434
+ />
435
+
436
+ <ImageUploadField
437
+ label={t('form.mainImage')}
438
+ description={t('form.mainImageDescription')}
439
+ fileId={form.watch('mainImageFileId')}
440
+ previewUrl={mainImagePreviewUrl}
441
+ uploading={mainImageUploading}
442
+ uploadLabel={t('form.imageUpload')}
443
+ replaceLabel={t('form.imageReplace')}
444
+ removeLabel={t('form.imageRemove')}
445
+ onPick={handleMainImagePick}
446
+ onRemove={() => {
447
+ form.setValue('mainImageFileId', null, { shouldDirty: true });
448
+ setMainImagePreviewUrl(null);
449
+ }}
450
+ inputRef={mainImageInputRef}
451
+ />
452
+ </div>
453
+
454
+ <div className="grid gap-4 md:grid-cols-2">
455
+ <FormField
456
+ control={form.control}
457
+ name="type"
458
+ render={({ field }) => (
459
+ <FormItem>
460
+ <FormLabel>{t('form.type')}</FormLabel>
461
+ <Select onValueChange={field.onChange} value={field.value}>
462
+ <FormControl>
463
+ <SelectTrigger className="w-full">
464
+ <SelectValue
465
+ placeholder={t('form.typePlaceholder')}
466
+ />
467
+ </SelectTrigger>
468
+ </FormControl>
469
+ <SelectContent>
470
+ {ACHIEVEMENT_TYPE_VALUES.map((type) => (
471
+ <SelectItem key={type} value={type}>
472
+ {t(`types.${type}`)}
473
+ </SelectItem>
474
+ ))}
475
+ </SelectContent>
476
+ </Select>
477
+ <FormMessage />
478
+ </FormItem>
479
+ )}
480
+ />
481
+
482
+ <FormField
483
+ control={form.control}
484
+ name="bitCodesValue"
485
+ render={({ field }) => (
486
+ <FormItem>
487
+ <FormLabel>{t('form.bitCodesValue')}</FormLabel>
488
+ <FormControl>
489
+ <Input type="number" min="0" step="1" {...field} />
490
+ </FormControl>
491
+ <FormMessage />
492
+ </FormItem>
493
+ )}
494
+ />
495
+ </div>
496
+
497
+ <FormField
498
+ control={form.control}
499
+ name="canReceiveMultipleTimes"
500
+ render={({ field }) => (
501
+ <FormItem className="flex flex-row items-start gap-3 rounded-lg border p-4">
502
+ <FormControl>
503
+ <Checkbox
504
+ checked={field.value}
505
+ onCheckedChange={(checked) => field.onChange(Boolean(checked))}
506
+ />
507
+ </FormControl>
508
+ <div className="space-y-1">
509
+ <FormLabel className="cursor-pointer">
510
+ {t('form.canReceiveMultipleTimes')}
511
+ </FormLabel>
512
+ <p className="text-sm text-muted-foreground">
513
+ {t('form.canReceiveMultipleTimesDescription')}
514
+ </p>
515
+ </div>
516
+ </FormItem>
517
+ )}
518
+ />
519
+
520
+ <div className="flex justify-end gap-2 pt-2">
521
+ <Button
522
+ type="button"
523
+ variant="outline"
524
+ onClick={() => onOpenChange(false)}
525
+ disabled={isSaving}
526
+ >
527
+ {t('actions.cancel')}
528
+ </Button>
529
+ <Button type="submit" disabled={isSaving}>
530
+ {isSaving ? t('actions.saving') : t('actions.save')}
531
+ </Button>
532
+ </div>
533
+ </form>
534
+ </Form>
535
+ </SheetContent>
536
+ </Sheet>
537
+ );
538
+ }
539
+
540
+ export default function AchievementsPage() {
541
+ const { request } = useApp();
542
+ const t = useTranslations('lms.AchievementsPage');
543
+
544
+ const [page, setPage] = useState(1);
545
+ const [pageSize, setPageSize] = usePersistedPageSize({
546
+ storageKey: 'pagination:lms-achievements:pageSize',
547
+ defaultValue: 12,
548
+ allowedValues: [12, 24, 48],
549
+ });
550
+ const [searchInput, setSearchInput] = useState('');
551
+ const [debouncedSearch, setDebouncedSearch] = useState('');
552
+ const [sheetOpen, setSheetOpen] = useState(false);
553
+ const [achievementToEdit, setAchievementToEdit] =
554
+ useState<Achievement | null>(null);
555
+ const [deleteDialogOpen, setDeleteDialogOpen] = useState(false);
556
+ const [achievementToDelete, setAchievementToDelete] =
557
+ useState<Achievement | null>(null);
558
+ const [isDeleting, setIsDeleting] = useState(false);
559
+
560
+ useEffect(() => {
561
+ const timeout = setTimeout(() => {
562
+ setDebouncedSearch(searchInput.trim());
563
+ setPage(1);
564
+ }, 300);
565
+ return () => clearTimeout(timeout);
566
+ }, [searchInput]);
567
+
568
+ const {
569
+ data: paginate = {
570
+ data: [],
571
+ total: 0,
572
+ page: 1,
573
+ pageSize,
574
+ lastPage: 1,
575
+ },
576
+ isLoading,
577
+ refetch,
578
+ } = useQuery<AchievementPaginatedResult>({
579
+ queryKey: ['lms-achievements', page, pageSize, debouncedSearch],
580
+ queryFn: async () => {
581
+ const params = new URLSearchParams({
582
+ page: String(page),
583
+ pageSize: String(pageSize),
584
+ });
585
+ if (debouncedSearch) params.set('search', debouncedSearch);
586
+
587
+ const response = await request<AchievementPaginatedResult>({
588
+ url: `/lms/achievements?${params.toString()}`,
589
+ method: 'GET',
590
+ });
591
+ return response.data;
592
+ },
593
+ placeholderData: (prev) =>
594
+ prev ?? { data: [], total: 0, page: 1, pageSize, lastPage: 1 },
595
+ });
596
+
597
+ const totalPages = Math.max(
598
+ 1,
599
+ (paginate.lastPage ?? Math.ceil((paginate.total || 0) / pageSize)) || 1
600
+ );
601
+
602
+ useEffect(() => {
603
+ if (page > totalPages) setPage(totalPages);
604
+ }, [page, totalPages]);
605
+
606
+ const handleDeleteConfirm = async () => {
607
+ if (!achievementToDelete) return;
608
+ try {
609
+ setIsDeleting(true);
610
+ await request({
611
+ url: `/lms/achievements/${achievementToDelete.id}`,
612
+ method: 'DELETE',
613
+ });
614
+ toast.success(t('messages.deleteSuccess'));
615
+ setDeleteDialogOpen(false);
616
+ setAchievementToDelete(null);
617
+ await refetch();
618
+ } catch {
619
+ toast.error(t('messages.deleteError'));
620
+ } finally {
621
+ setIsDeleting(false);
622
+ }
623
+ };
624
+
625
+ return (
626
+ <Page>
627
+ <PageHeader
628
+ breadcrumbs={[
629
+ { label: t('breadcrumbs.home'), href: '/' },
630
+ { label: t('breadcrumbs.lms'), href: '/lms' },
631
+ { label: t('breadcrumbs.current') },
632
+ ]}
633
+ title={t('title')}
634
+ description={t('description')}
635
+ actions={[
636
+ {
637
+ label: t('actions.create'),
638
+ onClick: () => {
639
+ setAchievementToEdit(null);
640
+ setSheetOpen(true);
641
+ },
642
+ icon: <Plus className="h-4 w-4" />,
643
+ },
644
+ ]}
645
+ />
646
+
647
+ <SearchBar
648
+ searchQuery={searchInput}
649
+ onSearchChange={(value) => setSearchInput(value)}
650
+ onSearch={() => setPage(1)}
651
+ placeholder={t('filters.searchPlaceholder')}
652
+ />
653
+
654
+ {isLoading ? (
655
+ <div className="space-y-3 p-4">
656
+ {Array.from({ length: 5 }).map((_, index) => (
657
+ <Skeleton key={index} className="h-16 w-full" />
658
+ ))}
659
+ </div>
660
+ ) : paginate.data.length === 0 ? (
661
+ <EmptyState
662
+ icon={<Trophy className="h-12 w-12" />}
663
+ title={t('empty.title')}
664
+ description={t('empty.description')}
665
+ actionLabel={t('actions.create')}
666
+ actionIcon={<Plus className="mr-2 h-4 w-4" />}
667
+ onAction={() => {
668
+ setAchievementToEdit(null);
669
+ setSheetOpen(true);
670
+ }}
671
+ />
672
+ ) : (
673
+ <div className="overflow-x-auto">
674
+ <Table>
675
+ <TableHeader>
676
+ <TableRow>
677
+ <TableHead>{t('table.name')}</TableHead>
678
+ <TableHead>{t('table.type')}</TableHead>
679
+ <TableHead>{t('table.bitCodesValue')}</TableHead>
680
+ <TableHead>{t('table.repeatable')}</TableHead>
681
+ <TableHead>{t('table.images')}</TableHead>
682
+ <TableHead className="w-10" />
683
+ </TableRow>
684
+ </TableHeader>
685
+ <TableBody>
686
+ {paginate.data.map((achievement) => {
687
+ const iconUrl = getFileUrl(achievement.iconFileId);
688
+ const mainImageUrl = getFileUrl(achievement.mainImageFileId);
689
+
690
+ return (
691
+ <TableRow
692
+ key={achievement.id}
693
+ className="cursor-pointer"
694
+ onDoubleClick={() => {
695
+ setAchievementToEdit(achievement);
696
+ setSheetOpen(true);
697
+ }}
698
+ >
699
+ <TableCell>
700
+ <div className="min-w-0">
701
+ <div className="truncate font-medium">
702
+ {achievement.name}
703
+ </div>
704
+ {achievement.description ? (
705
+ <div className="truncate text-xs text-muted-foreground">
706
+ {achievement.description}
707
+ </div>
708
+ ) : null}
709
+ </div>
710
+ </TableCell>
711
+ <TableCell>
712
+ <Badge variant="outline">{t(`types.${achievement.type}`)}</Badge>
713
+ </TableCell>
714
+ <TableCell>{achievement.bitCodesValue}</TableCell>
715
+ <TableCell>
716
+ <span
717
+ className={cn(
718
+ 'text-sm font-medium',
719
+ achievement.canReceiveMultipleTimes
720
+ ? 'text-emerald-600'
721
+ : 'text-muted-foreground'
722
+ )}
723
+ >
724
+ {achievement.canReceiveMultipleTimes
725
+ ? t('repeatable.yes')
726
+ : t('repeatable.no')}
727
+ </span>
728
+ </TableCell>
729
+ <TableCell>
730
+ <div className="flex items-center gap-2">
731
+ {iconUrl ? (
732
+ <Image
733
+ src={iconUrl}
734
+ alt={achievement.name}
735
+ width={32}
736
+ height={32}
737
+ unoptimized
738
+ className="size-8 rounded-md border object-cover"
739
+ />
740
+ ) : (
741
+ <div className="flex size-8 items-center justify-center rounded-md border border-dashed bg-muted/40">
742
+ <ImageIcon className="size-4 text-muted-foreground/50" />
743
+ </div>
744
+ )}
745
+ {mainImageUrl ? (
746
+ <Image
747
+ src={mainImageUrl}
748
+ alt={achievement.name}
749
+ width={48}
750
+ height={32}
751
+ unoptimized
752
+ className="h-8 w-12 rounded-md border object-cover"
753
+ />
754
+ ) : (
755
+ <div className="flex h-8 w-12 items-center justify-center rounded-md border border-dashed bg-muted/40">
756
+ <ImageIcon className="size-4 text-muted-foreground/50" />
757
+ </div>
758
+ )}
759
+ </div>
760
+ </TableCell>
761
+ <TableCell>
762
+ <DropdownMenu>
763
+ <DropdownMenuTrigger asChild>
764
+ <Button variant="ghost" size="icon" className="h-8 w-8">
765
+ <MoreHorizontal className="h-4 w-4" />
766
+ </Button>
767
+ </DropdownMenuTrigger>
768
+ <DropdownMenuContent align="end">
769
+ <DropdownMenuItem
770
+ onClick={() => {
771
+ setAchievementToEdit(achievement);
772
+ setSheetOpen(true);
773
+ }}
774
+ >
775
+ <Pencil className="mr-2 h-4 w-4" />
776
+ {t('actions.edit')}
777
+ </DropdownMenuItem>
778
+ <DropdownMenuSeparator />
779
+ <DropdownMenuItem
780
+ className="text-red-600"
781
+ onClick={() => {
782
+ setAchievementToDelete(achievement);
783
+ setDeleteDialogOpen(true);
784
+ }}
785
+ >
786
+ <Trash2 className="mr-2 h-4 w-4" />
787
+ {t('actions.delete')}
788
+ </DropdownMenuItem>
789
+ </DropdownMenuContent>
790
+ </DropdownMenu>
791
+ </TableCell>
792
+ </TableRow>
793
+ );
794
+ })}
795
+ </TableBody>
796
+ </Table>
797
+ </div>
798
+ )}
799
+
800
+ <PaginationFooter
801
+ currentPage={page}
802
+ pageSize={pageSize}
803
+ totalItems={paginate.total}
804
+ onPageChange={setPage}
805
+ onPageSizeChange={(nextPageSize) => {
806
+ setPageSize(nextPageSize);
807
+ setPage(1);
808
+ }}
809
+ />
810
+
811
+ <AchievementFormSheet
812
+ open={sheetOpen}
813
+ onOpenChange={setSheetOpen}
814
+ achievementToEdit={achievementToEdit}
815
+ onSaved={() => void refetch()}
816
+ />
817
+
818
+ <AlertDialog open={deleteDialogOpen} onOpenChange={setDeleteDialogOpen}>
819
+ <AlertDialogContent>
820
+ <AlertDialogHeader>
821
+ <AlertDialogTitle>{t('deleteDialog.title')}</AlertDialogTitle>
822
+ <AlertDialogDescription>
823
+ {t('deleteDialog.description', {
824
+ name: achievementToDelete?.name ?? '',
825
+ })}
826
+ </AlertDialogDescription>
827
+ </AlertDialogHeader>
828
+ <div className="flex justify-end gap-2">
829
+ <AlertDialogCancel disabled={isDeleting}>
830
+ {t('actions.cancel')}
831
+ </AlertDialogCancel>
832
+ <AlertDialogAction
833
+ onClick={handleDeleteConfirm}
834
+ disabled={isDeleting}
835
+ className="bg-red-600 hover:bg-red-700"
836
+ >
837
+ {isDeleting ? t('actions.deleting') : t('actions.delete')}
838
+ </AlertDialogAction>
839
+ </div>
840
+ </AlertDialogContent>
841
+ </AlertDialog>
842
+ </Page>
843
+ );
844
+ }