@hed-hog/lms 0.0.304 → 0.0.306

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 (458) hide show
  1. package/README.md +413 -401
  2. package/dist/certificate/certificate.controller.d.ts +90 -0
  3. package/dist/certificate/certificate.controller.d.ts.map +1 -0
  4. package/dist/certificate/certificate.controller.js +121 -0
  5. package/dist/certificate/certificate.controller.js.map +1 -0
  6. package/dist/certificate/certificate.module.d.ts +3 -0
  7. package/dist/certificate/certificate.module.d.ts.map +1 -0
  8. package/dist/certificate/certificate.module.js +26 -0
  9. package/dist/certificate/certificate.module.js.map +1 -0
  10. package/dist/certificate/certificate.service.d.ts +115 -0
  11. package/dist/certificate/certificate.service.d.ts.map +1 -0
  12. package/dist/certificate/certificate.service.js +343 -0
  13. package/dist/certificate/certificate.service.js.map +1 -0
  14. package/dist/certificate/dto/create-certificate-template.dto.d.ts +8 -0
  15. package/dist/certificate/dto/create-certificate-template.dto.d.ts.map +1 -0
  16. package/dist/certificate/dto/create-certificate-template.dto.js +44 -0
  17. package/dist/certificate/dto/create-certificate-template.dto.js.map +1 -0
  18. package/dist/certificate/dto/update-certificate-template.dto.d.ts +6 -0
  19. package/dist/certificate/dto/update-certificate-template.dto.d.ts.map +1 -0
  20. package/dist/certificate/dto/update-certificate-template.dto.js +9 -0
  21. package/dist/certificate/dto/update-certificate-template.dto.js.map +1 -0
  22. package/dist/class-group/class-group.controller.d.ts +305 -0
  23. package/dist/class-group/class-group.controller.d.ts.map +1 -0
  24. package/dist/class-group/class-group.controller.js +257 -0
  25. package/dist/class-group/class-group.controller.js.map +1 -0
  26. package/dist/class-group/class-group.module.d.ts +3 -0
  27. package/dist/class-group/class-group.module.d.ts.map +1 -0
  28. package/dist/class-group/class-group.module.js +25 -0
  29. package/dist/class-group/class-group.module.js.map +1 -0
  30. package/dist/class-group/class-group.service.d.ts +354 -0
  31. package/dist/class-group/class-group.service.d.ts.map +1 -0
  32. package/dist/class-group/class-group.service.js +1356 -0
  33. package/dist/class-group/class-group.service.js.map +1 -0
  34. package/dist/class-group/dto/create-class-group.dto.d.ts +33 -0
  35. package/dist/class-group/dto/create-class-group.dto.d.ts.map +1 -0
  36. package/dist/class-group/dto/create-class-group.dto.js +165 -0
  37. package/dist/class-group/dto/create-class-group.dto.js.map +1 -0
  38. package/dist/class-group/dto/create-session.dto.d.ts +22 -0
  39. package/dist/class-group/dto/create-session.dto.d.ts.map +1 -0
  40. package/dist/class-group/dto/create-session.dto.js +117 -0
  41. package/dist/class-group/dto/create-session.dto.js.map +1 -0
  42. package/dist/class-group/dto/enrollment.dto.d.ts +22 -0
  43. package/dist/class-group/dto/enrollment.dto.d.ts.map +1 -0
  44. package/dist/class-group/dto/enrollment.dto.js +89 -0
  45. package/dist/class-group/dto/enrollment.dto.js.map +1 -0
  46. package/dist/class-group/dto/update-class-group.dto.d.ts +6 -0
  47. package/dist/class-group/dto/update-class-group.dto.d.ts.map +1 -0
  48. package/dist/class-group/dto/update-class-group.dto.js +9 -0
  49. package/dist/class-group/dto/update-class-group.dto.js.map +1 -0
  50. package/dist/class-group/dto/update-session.dto.d.ts +7 -0
  51. package/dist/class-group/dto/update-session.dto.d.ts.map +1 -0
  52. package/dist/class-group/dto/update-session.dto.js +24 -0
  53. package/dist/class-group/dto/update-session.dto.js.map +1 -0
  54. package/dist/course/course-structure.controller.d.ts +127 -0
  55. package/dist/course/course-structure.controller.d.ts.map +1 -0
  56. package/dist/course/course-structure.controller.js +115 -0
  57. package/dist/course/course-structure.controller.js.map +1 -0
  58. package/dist/course/course-structure.service.d.ts +142 -0
  59. package/dist/course/course-structure.service.d.ts.map +1 -0
  60. package/dist/course/course-structure.service.js +445 -0
  61. package/dist/course/course-structure.service.js.map +1 -0
  62. package/dist/course/course.controller.d.ts +195 -0
  63. package/dist/course/course.controller.d.ts.map +1 -0
  64. package/dist/course/course.controller.js +104 -0
  65. package/dist/course/course.controller.js.map +1 -0
  66. package/dist/course/course.module.d.ts +3 -0
  67. package/dist/course/course.module.d.ts.map +1 -0
  68. package/dist/course/course.module.js +28 -0
  69. package/dist/course/course.module.js.map +1 -0
  70. package/dist/course/course.service.d.ts +215 -0
  71. package/dist/course/course.service.d.ts.map +1 -0
  72. package/dist/course/course.service.js +743 -0
  73. package/dist/course/course.service.js.map +1 -0
  74. package/dist/course/dto/create-course-structure-lesson.dto.d.ts +24 -0
  75. package/dist/course/dto/create-course-structure-lesson.dto.d.ts.map +1 -0
  76. package/dist/course/dto/create-course-structure-lesson.dto.js +118 -0
  77. package/dist/course/dto/create-course-structure-lesson.dto.js.map +1 -0
  78. package/dist/course/dto/create-course-structure-session.dto.d.ts +7 -0
  79. package/dist/course/dto/create-course-structure-session.dto.d.ts.map +1 -0
  80. package/dist/course/dto/create-course-structure-session.dto.js +40 -0
  81. package/dist/course/dto/create-course-structure-session.dto.js.map +1 -0
  82. package/dist/course/dto/create-course.dto.d.ts +26 -0
  83. package/dist/course/dto/create-course.dto.d.ts.map +1 -0
  84. package/dist/course/dto/create-course.dto.js +138 -0
  85. package/dist/course/dto/create-course.dto.js.map +1 -0
  86. package/dist/course/dto/update-course-structure-lesson.dto.d.ts +6 -0
  87. package/dist/course/dto/update-course-structure-lesson.dto.d.ts.map +1 -0
  88. package/dist/course/dto/update-course-structure-lesson.dto.js +9 -0
  89. package/dist/course/dto/update-course-structure-lesson.dto.js.map +1 -0
  90. package/dist/course/dto/update-course-structure-session.dto.d.ts +6 -0
  91. package/dist/course/dto/update-course-structure-session.dto.d.ts.map +1 -0
  92. package/dist/course/dto/update-course-structure-session.dto.js +9 -0
  93. package/dist/course/dto/update-course-structure-session.dto.js.map +1 -0
  94. package/dist/course/dto/update-course.dto.d.ts +6 -0
  95. package/dist/course/dto/update-course.dto.d.ts.map +1 -0
  96. package/dist/course/dto/update-course.dto.js +9 -0
  97. package/dist/course/dto/update-course.dto.js.map +1 -0
  98. package/dist/dashboard/dashboard.controller.d.ts +101 -0
  99. package/dist/dashboard/dashboard.controller.d.ts.map +1 -0
  100. package/dist/dashboard/dashboard.controller.js +40 -0
  101. package/dist/dashboard/dashboard.controller.js.map +1 -0
  102. package/dist/dashboard/dashboard.module.d.ts +3 -0
  103. package/dist/dashboard/dashboard.module.d.ts.map +1 -0
  104. package/dist/dashboard/dashboard.module.js +25 -0
  105. package/dist/dashboard/dashboard.module.js.map +1 -0
  106. package/dist/dashboard/dashboard.service.d.ts +130 -0
  107. package/dist/dashboard/dashboard.service.d.ts.map +1 -0
  108. package/dist/dashboard/dashboard.service.js +626 -0
  109. package/dist/dashboard/dashboard.service.js.map +1 -0
  110. package/dist/enterprise/dto/add-enterprise-class-group.dto.d.ts +4 -0
  111. package/dist/enterprise/dto/add-enterprise-class-group.dto.d.ts.map +1 -0
  112. package/dist/enterprise/dto/add-enterprise-class-group.dto.js +22 -0
  113. package/dist/enterprise/dto/add-enterprise-class-group.dto.js.map +1 -0
  114. package/dist/enterprise/dto/add-enterprise-course.dto.d.ts +5 -0
  115. package/dist/enterprise/dto/add-enterprise-course.dto.d.ts.map +1 -0
  116. package/dist/enterprise/dto/add-enterprise-course.dto.js +27 -0
  117. package/dist/enterprise/dto/add-enterprise-course.dto.js.map +1 -0
  118. package/dist/enterprise/dto/add-enterprise-student.dto.d.ts +5 -0
  119. package/dist/enterprise/dto/add-enterprise-student.dto.d.ts.map +1 -0
  120. package/dist/enterprise/dto/add-enterprise-student.dto.js +27 -0
  121. package/dist/enterprise/dto/add-enterprise-student.dto.js.map +1 -0
  122. package/dist/enterprise/dto/add-enterprise-user.dto.d.ts +7 -0
  123. package/dist/enterprise/dto/add-enterprise-user.dto.d.ts.map +1 -0
  124. package/dist/enterprise/dto/add-enterprise-user.dto.js +36 -0
  125. package/dist/enterprise/dto/add-enterprise-user.dto.js.map +1 -0
  126. package/dist/enterprise/dto/create-enterprise.dto.d.ts +10 -0
  127. package/dist/enterprise/dto/create-enterprise.dto.d.ts.map +1 -0
  128. package/dist/enterprise/dto/create-enterprise.dto.js +54 -0
  129. package/dist/enterprise/dto/create-enterprise.dto.js.map +1 -0
  130. package/dist/enterprise/dto/update-enterprise-student.dto.d.ts +4 -0
  131. package/dist/enterprise/dto/update-enterprise-student.dto.d.ts.map +1 -0
  132. package/dist/enterprise/dto/update-enterprise-student.dto.js +22 -0
  133. package/dist/enterprise/dto/update-enterprise-student.dto.js.map +1 -0
  134. package/dist/enterprise/dto/update-enterprise-user.dto.d.ts +5 -0
  135. package/dist/enterprise/dto/update-enterprise-user.dto.d.ts.map +1 -0
  136. package/dist/enterprise/dto/update-enterprise-user.dto.js +27 -0
  137. package/dist/enterprise/dto/update-enterprise-user.dto.js.map +1 -0
  138. package/dist/enterprise/dto/update-enterprise.dto.d.ts +6 -0
  139. package/dist/enterprise/dto/update-enterprise.dto.d.ts.map +1 -0
  140. package/dist/enterprise/dto/update-enterprise.dto.js +9 -0
  141. package/dist/enterprise/dto/update-enterprise.dto.js.map +1 -0
  142. package/dist/enterprise/enterprise.controller.d.ts +269 -0
  143. package/dist/enterprise/enterprise.controller.d.ts.map +1 -0
  144. package/dist/enterprise/enterprise.controller.js +311 -0
  145. package/dist/enterprise/enterprise.controller.js.map +1 -0
  146. package/dist/enterprise/enterprise.module.d.ts +3 -0
  147. package/dist/enterprise/enterprise.module.d.ts.map +1 -0
  148. package/dist/enterprise/enterprise.module.js +25 -0
  149. package/dist/enterprise/enterprise.module.js.map +1 -0
  150. package/dist/enterprise/enterprise.service.d.ts +282 -0
  151. package/dist/enterprise/enterprise.service.d.ts.map +1 -0
  152. package/dist/enterprise/enterprise.service.js +627 -0
  153. package/dist/enterprise/enterprise.service.js.map +1 -0
  154. package/dist/evaluation/evaluation.controller.d.ts +56 -0
  155. package/dist/evaluation/evaluation.controller.d.ts.map +1 -0
  156. package/dist/evaluation/evaluation.controller.js +76 -0
  157. package/dist/evaluation/evaluation.controller.js.map +1 -0
  158. package/dist/evaluation/evaluation.module.d.ts +3 -0
  159. package/dist/evaluation/evaluation.module.d.ts.map +1 -0
  160. package/dist/evaluation/evaluation.module.js +25 -0
  161. package/dist/evaluation/evaluation.module.js.map +1 -0
  162. package/dist/evaluation/evaluation.service.d.ts +67 -0
  163. package/dist/evaluation/evaluation.service.d.ts.map +1 -0
  164. package/dist/evaluation/evaluation.service.js +378 -0
  165. package/dist/evaluation/evaluation.service.js.map +1 -0
  166. package/dist/exam/dto/create-exam-question.dto.d.ts +25 -0
  167. package/dist/exam/dto/create-exam-question.dto.d.ts.map +1 -0
  168. package/dist/exam/dto/create-exam-question.dto.js +117 -0
  169. package/dist/exam/dto/create-exam-question.dto.js.map +1 -0
  170. package/dist/exam/dto/create-exam.dto.d.ts +11 -0
  171. package/dist/exam/dto/create-exam.dto.d.ts.map +1 -0
  172. package/dist/exam/dto/create-exam.dto.js +63 -0
  173. package/dist/exam/dto/create-exam.dto.js.map +1 -0
  174. package/dist/exam/dto/reorder-exam-questions.dto.d.ts +4 -0
  175. package/dist/exam/dto/reorder-exam-questions.dto.d.ts.map +1 -0
  176. package/dist/exam/dto/reorder-exam-questions.dto.js +23 -0
  177. package/dist/exam/dto/reorder-exam-questions.dto.js.map +1 -0
  178. package/dist/exam/dto/save-exam-attempt-answers.dto.d.ts +14 -0
  179. package/dist/exam/dto/save-exam-attempt-answers.dto.d.ts.map +1 -0
  180. package/dist/exam/dto/save-exam-attempt-answers.dto.js +68 -0
  181. package/dist/exam/dto/save-exam-attempt-answers.dto.js.map +1 -0
  182. package/dist/exam/dto/start-exam-attempt.dto.d.ts +4 -0
  183. package/dist/exam/dto/start-exam-attempt.dto.d.ts.map +1 -0
  184. package/dist/exam/dto/start-exam-attempt.dto.js +23 -0
  185. package/dist/exam/dto/start-exam-attempt.dto.js.map +1 -0
  186. package/dist/exam/dto/submit-exam-attempt.dto.d.ts +5 -0
  187. package/dist/exam/dto/submit-exam-attempt.dto.d.ts.map +1 -0
  188. package/dist/exam/dto/submit-exam-attempt.dto.js +23 -0
  189. package/dist/exam/dto/submit-exam-attempt.dto.js.map +1 -0
  190. package/dist/exam/dto/update-exam-question.dto.d.ts +6 -0
  191. package/dist/exam/dto/update-exam-question.dto.d.ts.map +1 -0
  192. package/dist/exam/dto/update-exam-question.dto.js +9 -0
  193. package/dist/exam/dto/update-exam-question.dto.js.map +1 -0
  194. package/dist/exam/dto/update-exam.dto.d.ts +6 -0
  195. package/dist/exam/dto/update-exam.dto.d.ts.map +1 -0
  196. package/dist/exam/dto/update-exam.dto.js +9 -0
  197. package/dist/exam/dto/update-exam.dto.js.map +1 -0
  198. package/dist/exam/exam-attempt.controller.d.ts +273 -0
  199. package/dist/exam/exam-attempt.controller.d.ts.map +1 -0
  200. package/dist/exam/exam-attempt.controller.js +84 -0
  201. package/dist/exam/exam-attempt.controller.js.map +1 -0
  202. package/dist/exam/exam-attempt.service.d.ts +302 -0
  203. package/dist/exam/exam-attempt.service.d.ts.map +1 -0
  204. package/dist/exam/exam-attempt.service.js +776 -0
  205. package/dist/exam/exam-attempt.service.js.map +1 -0
  206. package/dist/exam/exam.controller.d.ts +162 -0
  207. package/dist/exam/exam.controller.d.ts.map +1 -0
  208. package/dist/exam/exam.controller.js +158 -0
  209. package/dist/exam/exam.controller.js.map +1 -0
  210. package/dist/exam/exam.module.d.ts +3 -0
  211. package/dist/exam/exam.module.d.ts.map +1 -0
  212. package/dist/exam/exam.module.js +27 -0
  213. package/dist/exam/exam.module.js.map +1 -0
  214. package/dist/exam/exam.service.d.ts +179 -0
  215. package/dist/exam/exam.service.d.ts.map +1 -0
  216. package/dist/exam/exam.service.js +597 -0
  217. package/dist/exam/exam.service.js.map +1 -0
  218. package/dist/index.d.ts +28 -0
  219. package/dist/index.d.ts.map +1 -1
  220. package/dist/index.js +28 -0
  221. package/dist/index.js.map +1 -1
  222. package/dist/instructor/dto/create-instructor.dto.d.ts +10 -0
  223. package/dist/instructor/dto/create-instructor.dto.d.ts.map +1 -0
  224. package/dist/instructor/dto/create-instructor.dto.js +55 -0
  225. package/dist/instructor/dto/create-instructor.dto.js.map +1 -0
  226. package/dist/instructor/dto/update-instructor.dto.d.ts +9 -0
  227. package/dist/instructor/dto/update-instructor.dto.d.ts.map +1 -0
  228. package/dist/instructor/dto/update-instructor.dto.js +51 -0
  229. package/dist/instructor/dto/update-instructor.dto.js.map +1 -0
  230. package/dist/instructor/instructor.controller.d.ts +52 -0
  231. package/dist/instructor/instructor.controller.d.ts.map +1 -0
  232. package/dist/instructor/instructor.controller.js +98 -0
  233. package/dist/instructor/instructor.controller.js.map +1 -0
  234. package/dist/instructor/instructor.module.d.ts +3 -0
  235. package/dist/instructor/instructor.module.d.ts.map +1 -0
  236. package/dist/instructor/instructor.module.js +25 -0
  237. package/dist/instructor/instructor.module.js.map +1 -0
  238. package/dist/instructor/instructor.service.d.ts +79 -0
  239. package/dist/instructor/instructor.service.d.ts.map +1 -0
  240. package/dist/instructor/instructor.service.js +528 -0
  241. package/dist/instructor/instructor.service.js.map +1 -0
  242. package/dist/lms.module.d.ts.map +1 -1
  243. package/dist/lms.module.js +36 -4
  244. package/dist/lms.module.js.map +1 -1
  245. package/dist/reports/reports.controller.d.ts +69 -0
  246. package/dist/reports/reports.controller.d.ts.map +1 -0
  247. package/dist/reports/reports.controller.js +40 -0
  248. package/dist/reports/reports.controller.js.map +1 -0
  249. package/dist/reports/reports.module.d.ts +3 -0
  250. package/dist/reports/reports.module.d.ts.map +1 -0
  251. package/dist/reports/reports.module.js +25 -0
  252. package/dist/reports/reports.module.js.map +1 -0
  253. package/dist/reports/reports.service.d.ts +80 -0
  254. package/dist/reports/reports.service.d.ts.map +1 -0
  255. package/dist/reports/reports.service.js +366 -0
  256. package/dist/reports/reports.service.js.map +1 -0
  257. package/dist/training/dto/create-training.dto.d.ts +19 -0
  258. package/dist/training/dto/create-training.dto.d.ts.map +1 -0
  259. package/dist/training/dto/create-training.dto.js +98 -0
  260. package/dist/training/dto/create-training.dto.js.map +1 -0
  261. package/dist/training/dto/update-training.dto.d.ts +6 -0
  262. package/dist/training/dto/update-training.dto.d.ts.map +1 -0
  263. package/dist/training/dto/update-training.dto.js +9 -0
  264. package/dist/training/dto/update-training.dto.js.map +1 -0
  265. package/dist/training/training.controller.d.ts +195 -0
  266. package/dist/training/training.controller.d.ts.map +1 -0
  267. package/dist/training/training.controller.js +104 -0
  268. package/dist/training/training.controller.js.map +1 -0
  269. package/dist/training/training.module.d.ts +3 -0
  270. package/dist/training/training.module.d.ts.map +1 -0
  271. package/dist/training/training.module.js +25 -0
  272. package/dist/training/training.module.js.map +1 -0
  273. package/dist/training/training.service.d.ts +213 -0
  274. package/dist/training/training.service.d.ts.map +1 -0
  275. package/dist/training/training.service.js +497 -0
  276. package/dist/training/training.service.js.map +1 -0
  277. package/hedhog/data/dashboard.yaml +6 -0
  278. package/hedhog/data/dashboard_component.yaml +153 -0
  279. package/hedhog/data/dashboard_component_role.yaml +97 -0
  280. package/hedhog/data/dashboard_item.yaml +167 -0
  281. package/hedhog/data/dashboard_role.yaml +6 -0
  282. package/hedhog/data/instructor_qualification.yaml +16 -0
  283. package/hedhog/data/menu.yaml +129 -19
  284. package/hedhog/data/role.yaml +25 -1
  285. package/hedhog/data/route.yaml +867 -0
  286. package/hedhog/frontend/app/_components/class-form-sheet.tsx.ejs +1992 -0
  287. package/hedhog/frontend/app/_components/course-form-sheet.tsx.ejs +480 -0
  288. package/hedhog/frontend/app/_components/create-lms-instructor-sheet.tsx.ejs +591 -0
  289. package/hedhog/frontend/app/_components/create-lms-person-sheet.tsx.ejs +164 -0
  290. package/hedhog/frontend/app/_components/create-lms-student-person-sheet.tsx.ejs +120 -0
  291. package/hedhog/frontend/app/_components/lms-class-calendar.tsx.ejs +272 -0
  292. package/hedhog/frontend/app/_components/mobile-calendar.tsx.ejs +277 -0
  293. package/hedhog/frontend/app/_lib/editor/canvasInstance.ts.ejs +48 -0
  294. package/hedhog/frontend/app/_lib/editor/pctHelpers.ts.ejs +50 -0
  295. package/hedhog/frontend/app/_lib/editor/templateSerializer.ts.ejs +268 -0
  296. package/hedhog/frontend/app/_lib/editor/types.ts.ejs +94 -0
  297. package/hedhog/frontend/app/_lib/store/useTemplateStore.ts.ejs +284 -0
  298. package/hedhog/frontend/app/certificates/issued/page.tsx.ejs +638 -0
  299. package/hedhog/frontend/app/certificates/models/CanvasStage.tsx.ejs +916 -0
  300. package/hedhog/frontend/app/certificates/models/LeftPanel.tsx.ejs +200 -0
  301. package/hedhog/frontend/app/certificates/models/RightPanel.tsx.ejs +769 -0
  302. package/hedhog/frontend/app/certificates/models/TemplateEditorPage.tsx.ejs +104 -0
  303. package/hedhog/frontend/app/certificates/models/TopBar.tsx.ejs +354 -0
  304. package/hedhog/frontend/app/certificates/models/editor/page.tsx.ejs +5 -0
  305. package/hedhog/frontend/app/certificates/models/page.tsx.ejs +883 -0
  306. package/hedhog/frontend/app/classes/[id]/_components/event-summary-popover.tsx.ejs +279 -0
  307. package/hedhog/frontend/app/classes/[id]/_components/quick-create-session-popover.tsx.ejs +1027 -0
  308. package/hedhog/frontend/app/classes/[id]/page.tsx.ejs +3130 -993
  309. package/hedhog/frontend/app/classes/page.tsx.ejs +2731 -759
  310. package/hedhog/frontend/app/courses/[id]/_components/CourseCertificateCard.tsx.ejs +80 -0
  311. package/hedhog/frontend/app/courses/[id]/_components/CourseClassificationCard.tsx.ejs +226 -0
  312. package/hedhog/frontend/app/courses/[id]/_components/CourseContentCard.tsx.ejs +71 -0
  313. package/hedhog/frontend/app/courses/[id]/_components/CourseDangerZoneCard.tsx.ejs +42 -0
  314. package/hedhog/frontend/app/courses/[id]/_components/CourseFlagsCard.tsx.ejs +111 -0
  315. package/hedhog/frontend/app/courses/[id]/_components/CourseMainInfoCard.tsx.ejs +113 -0
  316. package/hedhog/frontend/app/courses/[id]/_components/CourseMediaCard.tsx.ejs +215 -0
  317. package/hedhog/frontend/app/courses/[id]/_components/CourseMultiEntityPicker.tsx.ejs +236 -0
  318. package/hedhog/frontend/app/courses/[id]/_components/CourseRelationsCard.tsx.ejs +141 -0
  319. package/hedhog/frontend/app/courses/[id]/_components/CourseSectionCard.tsx.ejs +57 -0
  320. package/hedhog/frontend/app/courses/[id]/_components/CourseSummaryCard.tsx.ejs +60 -0
  321. package/hedhog/frontend/app/courses/[id]/_components/course-edit-types.ts.ejs +33 -0
  322. package/hedhog/frontend/app/courses/[id]/page.tsx.ejs +933 -1103
  323. package/hedhog/frontend/app/courses/[id]/structure/page.tsx.ejs +699 -117
  324. package/hedhog/frontend/app/courses/page.tsx.ejs +1018 -1042
  325. package/hedhog/frontend/app/enterprise/[id]/page.tsx.ejs +317 -0
  326. package/hedhog/frontend/app/enterprise/_components/enterprise-activity-panel.tsx.ejs +88 -0
  327. package/hedhog/frontend/app/enterprise/_components/enterprise-admin-create-sheet.tsx.ejs +318 -0
  328. package/hedhog/frontend/app/enterprise/_components/enterprise-administrators-tab.tsx.ejs +332 -0
  329. package/hedhog/frontend/app/enterprise/_components/enterprise-class-create-sheet.tsx.ejs +58 -0
  330. package/hedhog/frontend/app/enterprise/_components/enterprise-classes-tab.tsx.ejs +390 -0
  331. package/hedhog/frontend/app/enterprise/_components/enterprise-company-identity-card.tsx.ejs +112 -0
  332. package/hedhog/frontend/app/enterprise/_components/enterprise-course-create-sheet.tsx.ejs +183 -0
  333. package/hedhog/frontend/app/enterprise/_components/enterprise-courses-tab.tsx.ejs +363 -0
  334. package/hedhog/frontend/app/enterprise/_components/enterprise-detail-constants.ts.ejs +88 -0
  335. package/hedhog/frontend/app/enterprise/_components/enterprise-detail-sheet.tsx.ejs +548 -0
  336. package/hedhog/frontend/app/enterprise/_components/enterprise-detail-utils.ts.ejs +33 -0
  337. package/hedhog/frontend/app/enterprise/_components/enterprise-mocks.ts.ejs +277 -0
  338. package/hedhog/frontend/app/enterprise/_components/enterprise-person-picker.ts.ejs +31 -0
  339. package/hedhog/frontend/app/enterprise/_components/enterprise-progress-bar.tsx.ejs +21 -0
  340. package/hedhog/frontend/app/enterprise/_components/enterprise-related-tab.tsx.ejs +224 -0
  341. package/hedhog/frontend/app/enterprise/_components/enterprise-sheet.tsx.ejs +397 -0
  342. package/hedhog/frontend/app/enterprise/_components/enterprise-student-create-sheet.tsx.ejs +167 -0
  343. package/hedhog/frontend/app/enterprise/_components/enterprise-students-tab.tsx.ejs +267 -0
  344. package/hedhog/frontend/app/enterprise/_components/enterprise-system-user-picker.ts.ejs +42 -0
  345. package/hedhog/frontend/app/enterprise/_components/enterprise-types.ts.ejs +96 -0
  346. package/hedhog/frontend/app/enterprise/_components/enterprise-user-create-sheet.tsx.ejs +207 -0
  347. package/hedhog/frontend/app/enterprise/_components/enterprise-user-distribution-chart.tsx.ejs +149 -0
  348. package/hedhog/frontend/app/enterprise/page.tsx.ejs +596 -0
  349. package/hedhog/frontend/app/evaluations/page.tsx.ejs +1250 -0
  350. package/hedhog/frontend/app/exams/[id]/attempt/page.tsx.ejs +642 -196
  351. package/hedhog/frontend/app/exams/[id]/page.tsx.ejs +11 -0
  352. package/hedhog/frontend/app/exams/[id]/questions/page.tsx.ejs +1316 -436
  353. package/hedhog/frontend/app/exams/page.tsx.ejs +799 -546
  354. package/hedhog/frontend/app/layout.tsx.ejs +5 -0
  355. package/hedhog/frontend/app/page.tsx.ejs +3 -1220
  356. package/hedhog/frontend/app/reports/courses/page.tsx.ejs +843 -0
  357. package/hedhog/frontend/app/reports/dashboard/page.tsx.ejs +890 -0
  358. package/hedhog/frontend/app/reports/page.tsx.ejs +802 -808
  359. package/hedhog/frontend/app/reports/students/page.tsx.ejs +772 -0
  360. package/hedhog/frontend/app/training/page.tsx.ejs +1873 -628
  361. package/hedhog/frontend/messages/en.json +1606 -111
  362. package/hedhog/frontend/messages/pt.json +1636 -134
  363. package/hedhog/frontend/widgets/active-classes-kpi.tsx.ejs +74 -0
  364. package/hedhog/frontend/widgets/active-courses-kpi.tsx.ejs +74 -0
  365. package/hedhog/frontend/widgets/approval-rate-kpi.tsx.ejs +81 -0
  366. package/hedhog/frontend/widgets/category-distribution-chart.tsx.ejs +119 -0
  367. package/hedhog/frontend/widgets/class-calendar.tsx.ejs +440 -0
  368. package/hedhog/frontend/widgets/completion-rate-kpi.tsx.ejs +81 -0
  369. package/hedhog/frontend/widgets/engagement-chart.tsx.ejs +120 -0
  370. package/hedhog/frontend/widgets/footer-summary.tsx.ejs +80 -0
  371. package/hedhog/frontend/widgets/issued-certificates-kpi.tsx.ejs +74 -0
  372. package/hedhog/frontend/widgets/latest-enrollments.tsx.ejs +166 -0
  373. package/hedhog/frontend/widgets/student-growth-chart.tsx.ejs +89 -0
  374. package/hedhog/frontend/widgets/top-courses-chart.tsx.ejs +104 -0
  375. package/hedhog/frontend/widgets/total-students-kpi.tsx.ejs +78 -0
  376. package/hedhog/frontend/widgets/upcoming-classes.tsx.ejs +152 -0
  377. package/hedhog/table/course.yaml +19 -1
  378. package/hedhog/table/course_class_group.yaml +8 -0
  379. package/hedhog/table/course_class_session.yaml +33 -0
  380. package/hedhog/table/course_instructor.yaml +27 -0
  381. package/hedhog/table/enterprise.yaml +29 -0
  382. package/hedhog/table/enterprise_class_group.yaml +20 -0
  383. package/hedhog/table/enterprise_course.yaml +23 -0
  384. package/hedhog/table/enterprise_student.yaml +24 -0
  385. package/hedhog/table/enterprise_user.yaml +35 -0
  386. package/hedhog/table/instructor_qualification.yaml +26 -0
  387. package/hedhog/table/instructor_qualification_assignment.yaml +22 -0
  388. package/hedhog/table/question.yaml +6 -0
  389. package/package.json +6 -6
  390. package/src/certificate/certificate.controller.ts +83 -0
  391. package/src/certificate/certificate.module.ts +13 -0
  392. package/src/certificate/certificate.service.ts +413 -0
  393. package/src/certificate/dto/create-certificate-template.dto.ts +25 -0
  394. package/src/certificate/dto/update-certificate-template.dto.ts +6 -0
  395. package/src/class-group/class-group.controller.ts +189 -0
  396. package/src/class-group/class-group.module.ts +12 -0
  397. package/src/class-group/class-group.service.ts +1802 -0
  398. package/src/class-group/dto/create-class-group.dto.ts +139 -0
  399. package/src/class-group/dto/create-session.dto.ts +102 -0
  400. package/src/class-group/dto/enrollment.dto.ts +70 -0
  401. package/src/class-group/dto/update-class-group.dto.ts +4 -0
  402. package/src/class-group/dto/update-session.dto.ts +9 -0
  403. package/src/course/course-structure.controller.ts +85 -0
  404. package/src/course/course-structure.service.ts +525 -0
  405. package/src/course/course.controller.ts +69 -0
  406. package/src/course/course.module.ts +15 -0
  407. package/src/course/course.service.ts +920 -0
  408. package/src/course/dto/create-course-structure-lesson.dto.ts +97 -0
  409. package/src/course/dto/create-course-structure-session.dto.ts +22 -0
  410. package/src/course/dto/create-course.dto.ts +111 -0
  411. package/src/course/dto/update-course-structure-lesson.dto.ts +6 -0
  412. package/src/course/dto/update-course-structure-session.dto.ts +6 -0
  413. package/src/course/dto/update-course.dto.ts +4 -0
  414. package/src/dashboard/dashboard.controller.ts +14 -0
  415. package/src/dashboard/dashboard.module.ts +12 -0
  416. package/src/dashboard/dashboard.service.ts +726 -0
  417. package/src/enterprise/dto/add-enterprise-class-group.dto.ts +7 -0
  418. package/src/enterprise/dto/add-enterprise-course.dto.ts +11 -0
  419. package/src/enterprise/dto/add-enterprise-student.dto.ts +16 -0
  420. package/src/enterprise/dto/add-enterprise-user.dto.ts +23 -0
  421. package/src/enterprise/dto/create-enterprise.dto.ts +41 -0
  422. package/src/enterprise/dto/update-enterprise-student.dto.ts +7 -0
  423. package/src/enterprise/dto/update-enterprise-user.dto.ts +11 -0
  424. package/src/enterprise/dto/update-enterprise.dto.ts +4 -0
  425. package/src/enterprise/enterprise.controller.ts +233 -0
  426. package/src/enterprise/enterprise.module.ts +12 -0
  427. package/src/enterprise/enterprise.service.ts +712 -0
  428. package/src/evaluation/evaluation.controller.ts +44 -0
  429. package/src/evaluation/evaluation.module.ts +12 -0
  430. package/src/evaluation/evaluation.service.ts +394 -0
  431. package/src/exam/dto/create-exam-question.dto.ts +103 -0
  432. package/src/exam/dto/create-exam.dto.ts +41 -0
  433. package/src/exam/dto/reorder-exam-questions.dto.ts +8 -0
  434. package/src/exam/dto/save-exam-attempt-answers.dto.ts +55 -0
  435. package/src/exam/dto/start-exam-attempt.dto.ts +8 -0
  436. package/src/exam/dto/submit-exam-attempt.dto.ts +8 -0
  437. package/src/exam/dto/update-exam-question.dto.ts +4 -0
  438. package/src/exam/dto/update-exam.dto.ts +4 -0
  439. package/src/exam/exam-attempt.controller.ts +65 -0
  440. package/src/exam/exam-attempt.service.ts +1008 -0
  441. package/src/exam/exam.controller.ts +102 -0
  442. package/src/exam/exam.module.ts +14 -0
  443. package/src/exam/exam.service.ts +784 -0
  444. package/src/index.ts +29 -0
  445. package/src/instructor/dto/create-instructor.dto.ts +43 -0
  446. package/src/instructor/dto/update-instructor.dto.ts +38 -0
  447. package/src/instructor/instructor.controller.ts +73 -0
  448. package/src/instructor/instructor.module.ts +12 -0
  449. package/src/instructor/instructor.service.ts +646 -0
  450. package/src/lms.module.ts +36 -4
  451. package/src/reports/reports.controller.ts +14 -0
  452. package/src/reports/reports.module.ts +12 -0
  453. package/src/reports/reports.service.ts +485 -0
  454. package/src/training/dto/create-training.dto.ts +81 -0
  455. package/src/training/dto/update-training.dto.ts +4 -0
  456. package/src/training/training.controller.ts +68 -0
  457. package/src/training/training.module.ts +12 -0
  458. package/src/training/training.service.ts +574 -0
@@ -0,0 +1,920 @@
1
+ import { PrismaService } from '@hed-hog/api-prisma';
2
+ import { BadRequestException, Injectable } from '@nestjs/common';
3
+ import { CreateCourseDto } from './dto/create-course.dto';
4
+ import { UpdateCourseDto } from './dto/update-course.dto';
5
+
6
+ type CourseImageTypeSlug = 'course-logo' | 'course-banner';
7
+
8
+ type CourseExtraFields = {
9
+ code?: string | null;
10
+ is_featured?: boolean | null;
11
+ has_certificate?: boolean | null;
12
+ is_listed?: boolean | null;
13
+ offering_type?: 'scheduled' | 'on_demand' | 'blended' | null;
14
+ };
15
+
16
+ type PersistCourseExtrasInput = {
17
+ code?: string;
18
+ isFeatured?: boolean;
19
+ hasCertificate?: boolean;
20
+ isListed?: boolean;
21
+ offeringType?: 'scheduled' | 'on_demand' | 'blended';
22
+ };
23
+
24
+ @Injectable()
25
+ export class CourseService {
26
+ constructor(private readonly prisma: PrismaService) {}
27
+
28
+ private normalizeLevel(value?: string | null) {
29
+ if (!value) return undefined;
30
+ const normalized = String(value).trim().toLowerCase();
31
+ if (normalized === 'beginner' || normalized === 'iniciante') {
32
+ return 'beginner';
33
+ }
34
+ if (normalized === 'intermediate' || normalized === 'intermediario') {
35
+ return 'intermediate';
36
+ }
37
+ if (normalized === 'advanced' || normalized === 'avancado') {
38
+ return 'advanced';
39
+ }
40
+ return undefined;
41
+ }
42
+
43
+ private normalizeStatus(value?: string | null) {
44
+ if (!value) return undefined;
45
+ const normalized = String(value).trim().toLowerCase();
46
+ if (
47
+ normalized === 'published' ||
48
+ normalized === 'active' ||
49
+ normalized === 'ativo'
50
+ ) {
51
+ return 'published';
52
+ }
53
+ if (normalized === 'draft' || normalized === 'rascunho') {
54
+ return 'draft';
55
+ }
56
+ if (normalized === 'archived' || normalized === 'arquivado') {
57
+ return 'archived';
58
+ }
59
+ return undefined;
60
+ }
61
+
62
+ private normalizeOfferingType(value?: string | null) {
63
+ if (!value) return undefined;
64
+
65
+ const normalized = String(value).trim().toLowerCase();
66
+ if (normalized === 'scheduled') {
67
+ return 'scheduled';
68
+ }
69
+ if (
70
+ normalized === 'on_demand' ||
71
+ normalized === 'ondemand' ||
72
+ normalized === 'on-demand'
73
+ ) {
74
+ return 'on_demand';
75
+ }
76
+ if (normalized === 'blended') {
77
+ return 'blended';
78
+ }
79
+
80
+ return undefined;
81
+ }
82
+
83
+ private normalizeOptionalText(value?: string | null) {
84
+ const normalized = value?.trim();
85
+ return normalized ? normalized : undefined;
86
+ }
87
+
88
+ private resolveCourseTitle(title: string | undefined, slug: string) {
89
+ return this.normalizeOptionalText(title) ?? slug;
90
+ }
91
+
92
+ async list(params: {
93
+ page?: number;
94
+ pageSize?: number;
95
+ search?: string;
96
+ status?: string;
97
+ level?: string;
98
+ category?: string;
99
+ }) {
100
+ const page = Math.max(Number(params.page) || 1, 1);
101
+ const pageSize = Math.max(Number(params.pageSize) || 12, 1);
102
+ const skip = (page - 1) * pageSize;
103
+
104
+ const where: any = {};
105
+
106
+ const normalizedStatus = this.normalizeStatus(params.status);
107
+ const normalizedLevel = this.normalizeLevel(params.level);
108
+
109
+ if (normalizedStatus) {
110
+ where.status = normalizedStatus;
111
+ }
112
+ if (normalizedLevel) {
113
+ where.level = normalizedLevel;
114
+ }
115
+ if (params.search) {
116
+ where.OR = [
117
+ { title: { contains: params.search, mode: 'insensitive' } },
118
+ { slug: { contains: params.search, mode: 'insensitive' } },
119
+ ];
120
+ }
121
+ if (params.category) {
122
+ where.course_category = {
123
+ some: { category: { slug: params.category } },
124
+ };
125
+ }
126
+
127
+ const [courses, total] = await Promise.all([
128
+ this.prisma.course.findMany({
129
+ skip,
130
+ take: pageSize,
131
+ where,
132
+ orderBy: { created_at: 'desc' },
133
+ include: {
134
+ course_category: {
135
+ include: {
136
+ category: {
137
+ select: { id: true, slug: true },
138
+ },
139
+ },
140
+ },
141
+ course_image: {
142
+ where: {
143
+ image_type: {
144
+ slug: { in: ['course-logo', 'course-banner'] },
145
+ },
146
+ },
147
+ include: {
148
+ image_type: {
149
+ select: { slug: true },
150
+ },
151
+ file: {
152
+ select: { id: true, filename: true },
153
+ },
154
+ },
155
+ orderBy: [{ is_primary: 'desc' }, { order: 'asc' }],
156
+ },
157
+ _count: {
158
+ select: { course_enrollment: true, course_class_group: true },
159
+ },
160
+ },
161
+ }),
162
+ this.prisma.course.count({ where }),
163
+ ]);
164
+
165
+ const extrasById = await this.getCourseExtras(
166
+ courses.map((course) => course.id),
167
+ );
168
+
169
+ return {
170
+ total,
171
+ page,
172
+ pageSize,
173
+ lastPage: Math.max(1, Math.ceil(total / pageSize)),
174
+ data: courses.map((c) => this.mapCourse(c, undefined, extrasById.get(c.id))),
175
+ };
176
+ }
177
+
178
+ async stats() {
179
+ const [total, published, totalStudents] = await Promise.all([
180
+ this.prisma.course.count(),
181
+ this.prisma.course.count({ where: { status: 'published' } }),
182
+ this.prisma.course_enrollment.count(),
183
+ ]);
184
+
185
+ const featured = await this.getFeaturedCoursesCount();
186
+
187
+ return {
188
+ totalCourses: total,
189
+ publishedCourses: published,
190
+ featuredCourses: featured,
191
+ totalStudents,
192
+ };
193
+ }
194
+
195
+ async getById(id: number) {
196
+ const c = await this.prisma.course.findUnique({
197
+ where: { id },
198
+ include: {
199
+ course_category: {
200
+ include: {
201
+ category: { select: { id: true, slug: true } },
202
+ },
203
+ },
204
+ course_image: {
205
+ where: {
206
+ image_type: {
207
+ slug: { in: ['course-logo', 'course-banner'] },
208
+ },
209
+ },
210
+ include: {
211
+ image_type: {
212
+ select: { slug: true },
213
+ },
214
+ file: {
215
+ select: { id: true, filename: true },
216
+ },
217
+ },
218
+ orderBy: [{ is_primary: 'desc' }, { order: 'asc' }],
219
+ },
220
+ course_instructor: {
221
+ include: {
222
+ instructor: {
223
+ select: {
224
+ id: true,
225
+ person: {
226
+ select: {
227
+ id: true,
228
+ name: true,
229
+ avatar_id: true,
230
+ },
231
+ },
232
+ },
233
+ },
234
+ },
235
+ },
236
+ _count: {
237
+ select: {
238
+ course_enrollment: true,
239
+ course_module: true,
240
+ course_class_group: true,
241
+ certificate: true,
242
+ },
243
+ },
244
+ },
245
+ });
246
+
247
+ if (!c) return null;
248
+
249
+ const extrasById = await this.getCourseExtras([id]);
250
+
251
+ const [lessonCount, sessionCount, certificatesIssued, avgProgress, modules] =
252
+ await Promise.all([
253
+ this.prisma.course_lesson.count({
254
+ where: { course_module: { course_id: id } },
255
+ }),
256
+ this.prisma.course_class_session.count({
257
+ where: { course_class_group: { course_id: id } },
258
+ }),
259
+ this.prisma.certificate.count({
260
+ where: { course_id: id },
261
+ }),
262
+ this.prisma.course_enrollment.aggregate({
263
+ where: { course_id: id },
264
+ _avg: { progress_percent: true },
265
+ }),
266
+ this.prisma.course_module.findMany({
267
+ where: { course_id: id },
268
+ orderBy: { order: 'asc' },
269
+ select: {
270
+ id: true,
271
+ title: true,
272
+ course_lesson: {
273
+ select: {
274
+ course_lesson_progress: {
275
+ select: { progress_percent: true },
276
+ },
277
+ },
278
+ },
279
+ },
280
+ }),
281
+ ]);
282
+
283
+ const progressByModule = modules.map((module) => {
284
+ const values = module.course_lesson.flatMap((lesson) =>
285
+ lesson.course_lesson_progress.map((progress) => progress.progress_percent),
286
+ );
287
+
288
+ const average =
289
+ values.length > 0
290
+ ? Math.round(values.reduce((sum, value) => sum + value, 0) / values.length)
291
+ : 0;
292
+
293
+ return {
294
+ moduleId: module.id,
295
+ moduleTitle: module.title,
296
+ progress: average,
297
+ };
298
+ });
299
+
300
+ return this.mapCourse(c, {
301
+ lessonCount,
302
+ sessionCount,
303
+ certificatesIssued,
304
+ averageCompletion: avgProgress._avg.progress_percent ?? 0,
305
+ progressByModule,
306
+ }, extrasById.get(id));
307
+ }
308
+
309
+ async create(dto: CreateCourseDto) {
310
+ const { categorySlugs, logoFileId, bannerFileId, instructorIds, ...data } = dto;
311
+ const normalizedSlug = data.slug.trim();
312
+ const resolvedCode = this.normalizeCourseCode(data.code ?? normalizedSlug);
313
+ const resolvedTitle = this.resolveCourseTitle(data.title, normalizedSlug);
314
+
315
+ const categories = categorySlugs?.length
316
+ ? await this.prisma.category.findMany({
317
+ where: { slug: { in: categorySlugs } },
318
+ select: { id: true },
319
+ })
320
+ : [];
321
+
322
+ // Validate instructor IDs if provided
323
+ let validInstructorIds: number[] = [];
324
+ if (instructorIds?.length) {
325
+ const instructors = await this.prisma.instructor.findMany({
326
+ where: { id: { in: instructorIds } },
327
+ select: { id: true },
328
+ });
329
+ const validIds = new Set(instructors.map((i) => i.id));
330
+ validInstructorIds = instructorIds.filter((id) => validIds.has(id));
331
+ }
332
+
333
+ const c = await this.prisma.course.create({
334
+ data: {
335
+ slug: normalizedSlug,
336
+ title: resolvedTitle,
337
+ ...(data.description !== undefined && {
338
+ description: data.description,
339
+ }),
340
+ level: this.normalizeLevel(data.level) ?? 'beginner',
341
+ status: this.normalizeStatus(data.status) ?? 'draft',
342
+ ...(data.requirements !== undefined && {
343
+ requirements: data.requirements,
344
+ }),
345
+ ...(data.objectives !== undefined && {
346
+ objectives: data.objectives,
347
+ }),
348
+ ...(data.targetAudience !== undefined && {
349
+ target_audience: data.targetAudience,
350
+ }),
351
+ ...(data.durationHours !== undefined && {
352
+ duration_hours: data.durationHours,
353
+ }),
354
+ ...(data.certificateWorkload !== undefined && {
355
+ certificate_workload: data.certificateWorkload,
356
+ }),
357
+ ...(data.primaryColor !== undefined && {
358
+ primary_color: data.primaryColor,
359
+ }),
360
+ ...(data.primaryContrastColor !== undefined && {
361
+ primary_contrast_color: data.primaryContrastColor,
362
+ }),
363
+ ...(data.secondaryColor !== undefined && {
364
+ secondary_color: data.secondaryColor,
365
+ }),
366
+ ...(data.secondaryContrastColor !== undefined && {
367
+ secondary_contrast_color: data.secondaryContrastColor,
368
+ }),
369
+ ...(categories.length > 0 && {
370
+ course_category: {
371
+ create: categories.map((item) => ({ category_id: item.id })),
372
+ },
373
+ }),
374
+ ...(validInstructorIds.length > 0 && {
375
+ course_instructor: {
376
+ create: validInstructorIds.map((instructorId) => ({
377
+ instructor_id: instructorId,
378
+ role: 'lead',
379
+ })),
380
+ },
381
+ }),
382
+ },
383
+ include: {
384
+ course_category: {
385
+ include: { category: { select: { id: true, slug: true } } },
386
+ },
387
+ course_image: {
388
+ where: {
389
+ image_type: {
390
+ slug: { in: ['course-logo', 'course-banner'] },
391
+ },
392
+ },
393
+ include: {
394
+ image_type: {
395
+ select: { slug: true },
396
+ },
397
+ file: {
398
+ select: { id: true, filename: true },
399
+ },
400
+ },
401
+ orderBy: [{ is_primary: 'desc' }, { order: 'asc' }],
402
+ },
403
+ course_instructor: {
404
+ include: {
405
+ instructor: {
406
+ select: {
407
+ id: true,
408
+ person: {
409
+ select: {
410
+ id: true,
411
+ name: true,
412
+ avatar_id: true,
413
+ },
414
+ },
415
+ },
416
+ },
417
+ },
418
+ },
419
+ _count: { select: { course_enrollment: true } },
420
+ },
421
+ });
422
+
423
+ await this.syncCourseImages(c.id, {
424
+ logoFileId,
425
+ bannerFileId,
426
+ });
427
+
428
+ await this.persistCourseExtras(c.id, {
429
+ ...data,
430
+ ...(resolvedCode && { code: resolvedCode }),
431
+ offeringType: this.normalizeOfferingType(data.offeringType) ?? 'on_demand',
432
+ });
433
+
434
+ const extrasById = await this.getCourseExtras([c.id]);
435
+
436
+ return this.mapCourse(c, undefined, extrasById.get(c.id));
437
+ }
438
+
439
+ async update(id: number, dto: UpdateCourseDto) {
440
+ const { categorySlugs, logoFileId, bannerFileId, instructorIds, ...data } = dto;
441
+ let existingSlug: string | undefined;
442
+
443
+ if (
444
+ data.title !== undefined &&
445
+ this.normalizeOptionalText(data.title) === undefined &&
446
+ data.slug === undefined
447
+ ) {
448
+ const currentCourse = await this.prisma.course.findUnique({
449
+ where: { id },
450
+ select: { slug: true },
451
+ });
452
+
453
+ existingSlug = currentCourse?.slug;
454
+ }
455
+
456
+ const resolvedSlug = data.slug !== undefined ? data.slug.trim() : undefined;
457
+ const resolvedTitle =
458
+ data.title !== undefined
459
+ ? this.resolveCourseTitle(data.title, resolvedSlug ?? existingSlug ?? '')
460
+ : undefined;
461
+
462
+ const categories = categorySlugs?.length
463
+ ? await this.prisma.category.findMany({
464
+ where: { slug: { in: categorySlugs } },
465
+ select: { id: true },
466
+ })
467
+ : [];
468
+
469
+ // Validate instructor IDs if provided
470
+ let validInstructorIds: number[] | undefined;
471
+ if (instructorIds !== undefined) {
472
+ if (instructorIds.length === 0) {
473
+ validInstructorIds = [];
474
+ } else {
475
+ const instructors = await this.prisma.instructor.findMany({
476
+ where: { id: { in: instructorIds } },
477
+ select: { id: true },
478
+ });
479
+ const validIds = new Set(instructors.map((i) => i.id));
480
+ validInstructorIds = instructorIds.filter((id) => validIds.has(id));
481
+ }
482
+ }
483
+
484
+ if (categorySlugs !== undefined) {
485
+ await this.prisma.course_category.deleteMany({
486
+ where: { course_id: id },
487
+ });
488
+ }
489
+
490
+ const c = await this.prisma.course.update({
491
+ where: { id },
492
+ data: {
493
+ ...(resolvedSlug !== undefined && { slug: resolvedSlug }),
494
+ ...(resolvedTitle !== undefined && { title: resolvedTitle }),
495
+ ...(data.description !== undefined && { description: data.description }),
496
+ ...(data.level !== undefined && {
497
+ level: this.normalizeLevel(data.level),
498
+ }),
499
+ ...(data.status !== undefined && {
500
+ status: this.normalizeStatus(data.status),
501
+ }),
502
+ ...(data.requirements !== undefined && {
503
+ requirements: data.requirements,
504
+ }),
505
+ ...(data.objectives !== undefined && {
506
+ objectives: data.objectives,
507
+ }),
508
+ ...(data.targetAudience !== undefined && {
509
+ target_audience: data.targetAudience,
510
+ }),
511
+ ...(data.durationHours !== undefined && {
512
+ duration_hours: data.durationHours,
513
+ }),
514
+ ...(data.certificateWorkload !== undefined && {
515
+ certificate_workload: data.certificateWorkload,
516
+ }),
517
+ ...(data.primaryColor !== undefined && {
518
+ primary_color: data.primaryColor,
519
+ }),
520
+ ...(data.primaryContrastColor !== undefined && {
521
+ primary_contrast_color: data.primaryContrastColor,
522
+ }),
523
+ ...(data.secondaryColor !== undefined && {
524
+ secondary_color: data.secondaryColor,
525
+ }),
526
+ ...(data.secondaryContrastColor !== undefined && {
527
+ secondary_contrast_color: data.secondaryContrastColor,
528
+ }),
529
+ ...(categories.length > 0 && {
530
+ course_category: {
531
+ create: categories.map((item) => ({ category_id: item.id })),
532
+ },
533
+ }),
534
+ },
535
+ include: {
536
+ course_category: {
537
+ include: { category: { select: { id: true, slug: true } } },
538
+ },
539
+ course_image: {
540
+ where: {
541
+ image_type: {
542
+ slug: { in: ['course-logo', 'course-banner'] },
543
+ },
544
+ },
545
+ include: {
546
+ image_type: {
547
+ select: { slug: true },
548
+ },
549
+ file: {
550
+ select: { id: true, filename: true },
551
+ },
552
+ },
553
+ orderBy: [{ is_primary: 'desc' }, { order: 'asc' }],
554
+ },
555
+ course_instructor: {
556
+ include: {
557
+ instructor: {
558
+ select: {
559
+ id: true,
560
+ person: {
561
+ select: {
562
+ id: true,
563
+ name: true,
564
+ avatar_id: true,
565
+ },
566
+ },
567
+ },
568
+ },
569
+ },
570
+ },
571
+ _count: { select: { course_enrollment: true } },
572
+ },
573
+ });
574
+
575
+ // Handle instructor relationships
576
+ if (validInstructorIds !== undefined) {
577
+ await this.prisma.course_instructor.deleteMany({
578
+ where: { course_id: id },
579
+ });
580
+
581
+ if (validInstructorIds.length > 0) {
582
+ await this.prisma.course_instructor.createMany({
583
+ data: validInstructorIds.map((instructorId) => ({
584
+ course_id: id,
585
+ instructor_id: instructorId,
586
+ role: 'lead',
587
+ })),
588
+ });
589
+ }
590
+ }
591
+
592
+ await this.syncCourseImages(id, {
593
+ logoFileId,
594
+ bannerFileId,
595
+ });
596
+
597
+ await this.persistCourseExtras(id, data);
598
+
599
+ const extrasById = await this.getCourseExtras([id]);
600
+
601
+ return this.mapCourse(c, undefined, extrasById.get(id));
602
+ }
603
+
604
+ async remove(id: number) {
605
+ await this.prisma.course.delete({ where: { id } });
606
+ return { success: true };
607
+ }
608
+
609
+ private mapCourse(
610
+ c: any,
611
+ metrics?: {
612
+ lessonCount?: number;
613
+ sessionCount?: number;
614
+ certificatesIssued?: number;
615
+ averageCompletion?: number;
616
+ progressByModule?: {
617
+ moduleId: number;
618
+ moduleTitle: string;
619
+ progress: number;
620
+ }[];
621
+ },
622
+ extras?: CourseExtraFields,
623
+ ) {
624
+ const rawCode = extras?.code ?? c.code ?? c.slug;
625
+ const resolvedTitle = this.normalizeOptionalText(c.title) ?? c.slug;
626
+ const courseImages = Array.isArray(c.course_image) ? c.course_image : [];
627
+ const classCount = c._count?.course_class_group ?? 0;
628
+ const offeringType =
629
+ this.normalizeOfferingType(extras?.offering_type) ??
630
+ (classCount > 0 ? 'scheduled' : 'on_demand');
631
+
632
+ const logoImage = courseImages.find(
633
+ (courseImage: any) => courseImage.image_type?.slug === 'course-logo',
634
+ );
635
+ const bannerImage = courseImages.find(
636
+ (courseImage: any) => courseImage.image_type?.slug === 'course-banner',
637
+ );
638
+
639
+ return {
640
+ id: c.id,
641
+ code:
642
+ rawCode?.toUpperCase().replace(/[^A-Z0-9-]+/g, '-').slice(0, 32) ??
643
+ '',
644
+ slug: c.slug,
645
+ title: resolvedTitle,
646
+ description: c.description ?? '',
647
+ level: c.level,
648
+ status: c.status,
649
+ durationHours: c.duration_hours ?? 0,
650
+ certificateWorkload: c.certificate_workload,
651
+ requirements: c.requirements ?? '',
652
+ objectives: c.objectives ?? '',
653
+ targetAudience: c.target_audience ?? '',
654
+ primaryColor: c.primary_color ?? null,
655
+ primaryContrastColor: c.primary_contrast_color ?? null,
656
+ secondaryColor: c.secondary_color ?? null,
657
+ secondaryContrastColor: c.secondary_contrast_color ?? null,
658
+ logoFileId: logoImage?.file?.id ?? logoImage?.file_id ?? null,
659
+ logoFilename: logoImage?.file?.filename ?? null,
660
+ bannerFileId: bannerImage?.file?.id ?? bannerImage?.file_id ?? null,
661
+ bannerFilename: bannerImage?.file?.filename ?? null,
662
+ categories: (c.course_category ?? []).map(
663
+ (cc: any) => cc.category?.slug ?? '',
664
+ ),
665
+ categoryIds: (c.course_category ?? []).map(
666
+ (cc: any) => cc.category?.id ?? 0,
667
+ ),
668
+ isFeatured: extras?.is_featured ?? c.is_featured ?? false,
669
+ hasCertificate: extras?.has_certificate ?? c.has_certificate ?? false,
670
+ isListed: extras?.is_listed ?? c.is_listed ?? false,
671
+ offeringType,
672
+ enrollmentCount: c._count?.course_enrollment ?? 0,
673
+ moduleCount: c._count?.course_module ?? 0,
674
+ classCount,
675
+ lessonCount: metrics?.lessonCount ?? 0,
676
+ sessionCount: metrics?.sessionCount ?? 0,
677
+ averageCompletion: Math.round(metrics?.averageCompletion ?? 0),
678
+ certificatesIssued:
679
+ metrics?.certificatesIssued ?? c._count?.certificate ?? 0,
680
+ progressByModule: metrics?.progressByModule ?? [],
681
+ instructors: (c.course_instructor ?? []).map((ci: any) => ({
682
+ id: ci.instructor_id,
683
+ name: ci.instructor?.person?.name ?? `Instructor #${ci.instructor_id}`,
684
+ avatarId: ci.instructor?.person?.avatar_id ?? null,
685
+ })),
686
+ instructorIds: (c.course_instructor ?? []).map(
687
+ (ci: any) => ci.instructor_id,
688
+ ),
689
+ createdAt: c.created_at,
690
+ };
691
+ }
692
+
693
+ private async getFeaturedCoursesCount() {
694
+ try {
695
+ const rows = (await this.prisma.$queryRawUnsafe(
696
+ 'SELECT COUNT(*) AS value FROM course WHERE is_featured = true',
697
+ )) as Array<{ value: number | string }>;
698
+
699
+ return Number(rows[0]?.value ?? 0);
700
+ } catch {
701
+ return 0;
702
+ }
703
+ }
704
+
705
+ private async getCourseExtras(ids: number[]) {
706
+ const extrasById = new Map<number, CourseExtraFields>();
707
+
708
+ if (ids.length === 0) {
709
+ return extrasById;
710
+ }
711
+
712
+ const normalizedIds = ids
713
+ .map((id) => Number(id))
714
+ .filter((id) => Number.isInteger(id) && id > 0);
715
+
716
+ if (normalizedIds.length === 0) {
717
+ return extrasById;
718
+ }
719
+
720
+ try {
721
+ const idsCsv = normalizedIds.join(',');
722
+ const rows = (await this.prisma.$queryRawUnsafe(
723
+ `
724
+ SELECT id, code, is_featured, has_certificate, is_listed, offering_type
725
+ FROM course
726
+ WHERE id IN (${idsCsv})
727
+ `,
728
+ )) as Array<{
729
+ id: number;
730
+ code: string | null;
731
+ is_featured: boolean | null;
732
+ has_certificate: boolean | null;
733
+ is_listed: boolean | null;
734
+ offering_type: 'scheduled' | 'on_demand' | 'blended' | null;
735
+ }>;
736
+
737
+ for (const row of rows) {
738
+ extrasById.set(row.id, {
739
+ code: row.code,
740
+ is_featured: row.is_featured,
741
+ has_certificate: row.has_certificate,
742
+ is_listed: row.is_listed,
743
+ offering_type: row.offering_type,
744
+ });
745
+ }
746
+ } catch {
747
+ return extrasById;
748
+ }
749
+
750
+ return extrasById;
751
+ }
752
+
753
+ private async persistCourseExtras(
754
+ id: number,
755
+ data: PersistCourseExtrasInput,
756
+ ) {
757
+ const sets: string[] = [];
758
+ const params: Array<string | boolean | number> = [];
759
+
760
+ if (data.code !== undefined) {
761
+ params.push(data.code);
762
+ sets.push(`code = $${params.length}`);
763
+ }
764
+
765
+ if (data.isFeatured !== undefined) {
766
+ params.push(data.isFeatured);
767
+ sets.push(`is_featured = $${params.length}`);
768
+ }
769
+
770
+ if (data.hasCertificate !== undefined) {
771
+ params.push(data.hasCertificate);
772
+ sets.push(`has_certificate = $${params.length}`);
773
+ }
774
+
775
+ if (data.isListed !== undefined) {
776
+ params.push(data.isListed);
777
+ sets.push(`is_listed = $${params.length}`);
778
+ }
779
+
780
+ if (data.offeringType !== undefined) {
781
+ params.push(data.offeringType);
782
+ sets.push(`offering_type = $${params.length}`);
783
+ }
784
+
785
+ if (sets.length === 0) {
786
+ return;
787
+ }
788
+
789
+ params.push(id);
790
+
791
+ try {
792
+ await this.prisma.$executeRawUnsafe(
793
+ `
794
+ UPDATE course
795
+ SET ${sets.join(', ')}, updated_at = NOW()
796
+ WHERE id = $${params.length}
797
+ `,
798
+ ...params,
799
+ );
800
+ } catch {
801
+ // Columns may not exist yet in environments with outdated schema.
802
+ }
803
+ }
804
+
805
+ private normalizeCourseCode(value?: string | null) {
806
+ const normalized = String(value ?? '')
807
+ .trim()
808
+ .toUpperCase()
809
+ .replace(/[^A-Z0-9-]+/g, '-')
810
+ .replace(/-+/g, '-')
811
+ .replace(/^-+|-+$/g, '')
812
+ .slice(0, 32);
813
+
814
+ return normalized || undefined;
815
+ }
816
+
817
+ private async syncCourseImages(
818
+ courseId: number,
819
+ data: {
820
+ logoFileId?: number;
821
+ bannerFileId?: number;
822
+ },
823
+ ) {
824
+ if (data.logoFileId !== undefined) {
825
+ await this.upsertCourseImage(courseId, data.logoFileId, 'course-logo');
826
+ }
827
+
828
+ if (data.bannerFileId !== undefined) {
829
+ await this.upsertCourseImage(
830
+ courseId,
831
+ data.bannerFileId,
832
+ 'course-banner',
833
+ );
834
+ }
835
+ }
836
+
837
+ private async upsertCourseImage(
838
+ courseId: number,
839
+ fileId: number,
840
+ imageTypeSlug: CourseImageTypeSlug,
841
+ ) {
842
+ const normalizedFileId = Number(fileId);
843
+
844
+ if (!Number.isInteger(normalizedFileId) || normalizedFileId <= 0) {
845
+ throw new BadRequestException('Invalid file id for course image');
846
+ }
847
+
848
+ const file = await this.prisma.file.findUnique({
849
+ where: { id: normalizedFileId },
850
+ select: { id: true },
851
+ });
852
+
853
+ if (!file) {
854
+ throw new BadRequestException('File not found');
855
+ }
856
+
857
+ const imageType = await this.getOrCreateCourseImageType(imageTypeSlug);
858
+ const courseImages = await this.prisma.course_image.findMany({
859
+ where: {
860
+ course_id: courseId,
861
+ image_type_id: imageType.id,
862
+ },
863
+ orderBy: [{ is_primary: 'desc' }, { order: 'asc' }],
864
+ select: { id: true, order: true },
865
+ });
866
+
867
+ if (courseImages.length > 0) {
868
+ const [primaryImage, ...duplicates] = courseImages;
869
+
870
+ await this.prisma.course_image.update({
871
+ where: { id: primaryImage.id },
872
+ data: {
873
+ file_id: normalizedFileId,
874
+ is_primary: true,
875
+ },
876
+ });
877
+
878
+ if (duplicates.length > 0) {
879
+ await this.prisma.course_image.deleteMany({
880
+ where: { id: { in: duplicates.map((item) => item.id) } },
881
+ });
882
+ }
883
+
884
+ return;
885
+ }
886
+
887
+ await this.prisma.course_image.create({
888
+ data: {
889
+ course_id: courseId,
890
+ file_id: normalizedFileId,
891
+ image_type_id: imageType.id,
892
+ order: 0,
893
+ is_primary: true,
894
+ },
895
+ });
896
+ }
897
+
898
+ private async getOrCreateCourseImageType(imageTypeSlug: CourseImageTypeSlug) {
899
+ const existing = await this.prisma.image_type.findFirst({
900
+ where: {
901
+ slug: imageTypeSlug,
902
+ applies_to_course: true,
903
+ },
904
+ select: { id: true },
905
+ });
906
+
907
+ if (existing) {
908
+ return existing;
909
+ }
910
+
911
+ return this.prisma.image_type.create({
912
+ data: {
913
+ name: imageTypeSlug === 'course-logo' ? 'Course Logo' : 'Course Banner',
914
+ slug: imageTypeSlug,
915
+ applies_to_course: true,
916
+ },
917
+ select: { id: true },
918
+ });
919
+ }
920
+ }