@hydralms/components 0.1.3 → 0.3.0

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 (344) hide show
  1. package/dist/StudentProfile-BVfZMbnV.cjs +1 -0
  2. package/dist/StudentProfile-DeMxdrL3.js +3275 -0
  3. package/dist/assessment-toolbar/assessment-toolbar.d.ts +1 -1
  4. package/dist/assessment-toolbar/index.d.ts +5 -1
  5. package/dist/assessment-toolbar/question-header-bar.d.ts +2 -0
  6. package/dist/assessment-toolbar/question-materials-drawer.d.ts +2 -0
  7. package/dist/assessment-toolbar/question-navigator.d.ts +1 -1
  8. package/dist/assessment-toolbar/timer-display.d.ts +1 -1
  9. package/dist/assessment-toolbar/types.d.ts +52 -4
  10. package/dist/assessment-toolbar/use-countdown.d.ts +43 -0
  11. package/dist/common/index.d.ts +3 -1
  12. package/dist/common/pagination.d.ts +26 -0
  13. package/dist/common/stepper.d.ts +6 -0
  14. package/dist/common/types.d.ts +38 -0
  15. package/dist/components.css +1 -1
  16. package/dist/content/attachment-list.d.ts +6 -0
  17. package/dist/content/audio-player.d.ts +22 -0
  18. package/dist/content/code-block.d.ts +30 -0
  19. package/dist/content/content-block.d.ts +1 -1
  20. package/dist/content/embed-block.d.ts +28 -0
  21. package/dist/content/index.d.ts +8 -1
  22. package/dist/content/types.d.ts +63 -0
  23. package/dist/curriculum/course-card.d.ts +51 -0
  24. package/dist/curriculum/curriculum-item.d.ts +1 -1
  25. package/dist/curriculum/index.d.ts +2 -0
  26. package/dist/curriculum/types.d.ts +2 -2
  27. package/dist/index.cjs +1 -1
  28. package/dist/index.d.ts +1 -0
  29. package/dist/index.js +597 -308
  30. package/dist/license/HydraContext.d.ts +16 -0
  31. package/dist/license/ProBadge.d.ts +6 -0
  32. package/dist/license/index.d.ts +7 -0
  33. package/dist/license/tiers.d.ts +3 -0
  34. package/dist/license/useHydraLicense.d.ts +6 -0
  35. package/dist/license/validate.d.ts +13 -0
  36. package/dist/license/withProGate.d.ts +6 -0
  37. package/dist/modules/AssignmentModule/AssignmentModule.d.ts +5 -0
  38. package/dist/modules/AssignmentModule/types.d.ts +69 -0
  39. package/dist/modules/CertificateModule/CertificateModule.d.ts +5 -0
  40. package/dist/modules/CertificateModule/types.d.ts +51 -0
  41. package/dist/modules/CourseCatalogModule/CourseCatalogModule.d.ts +5 -0
  42. package/dist/modules/CourseCatalogModule/types.d.ts +43 -0
  43. package/dist/modules/CoursePlayer/CoursePlayer.d.ts +4 -1
  44. package/dist/modules/DiscussionModule/DiscussionModule.d.ts +5 -0
  45. package/dist/modules/DiscussionModule/types.d.ts +47 -0
  46. package/dist/modules/ExamModule/ExamModule.d.ts +5 -0
  47. package/dist/modules/ExamModule/types.d.ts +55 -0
  48. package/dist/modules/FlashcardLab/FlashcardLab.d.ts +4 -1
  49. package/dist/modules/FlashcardLab/types.d.ts +2 -0
  50. package/dist/modules/GradeCenterModule/GradeCenterModule.d.ts +5 -0
  51. package/dist/modules/GradeCenterModule/types.d.ts +56 -0
  52. package/dist/modules/QuizModule/QuizModule.d.ts +4 -1
  53. package/dist/modules/QuizModule/types.d.ts +10 -14
  54. package/dist/modules/StudentDashboardModule/StudentDashboardModule.d.ts +5 -0
  55. package/dist/modules/StudentDashboardModule/types.d.ts +54 -0
  56. package/dist/modules/StudentProfileModule/StudentProfileModule.d.ts +5 -0
  57. package/dist/modules/StudentProfileModule/types.d.ts +43 -0
  58. package/dist/modules/SurveyModule/SurveyModule.d.ts +5 -0
  59. package/dist/modules/SurveyModule/types.d.ts +51 -0
  60. package/dist/modules/_shared/assessment-intro.d.ts +16 -0
  61. package/dist/modules/_shared/assessment-results.d.ts +23 -0
  62. package/dist/modules/_shared/types.d.ts +10 -0
  63. package/dist/modules/_shared/use-timer.d.ts +9 -0
  64. package/dist/modules/index.d.ts +18 -0
  65. package/dist/modules.cjs +1 -0
  66. package/dist/modules.js +1834 -0
  67. package/dist/progress/achievement-badge.d.ts +6 -0
  68. package/dist/progress/activity-timeline.d.ts +6 -0
  69. package/dist/progress/index.d.ts +4 -1
  70. package/dist/progress/stat-card.d.ts +1 -1
  71. package/dist/progress/streak-badge.d.ts +6 -0
  72. package/dist/progress/types.d.ts +99 -0
  73. package/dist/provider/HydraProvider.d.ts +5 -1
  74. package/dist/questions/choice.d.ts +1 -1
  75. package/dist/questions/confidence-indicator.d.ts +37 -0
  76. package/dist/questions/essay.d.ts +2 -2
  77. package/dist/questions/fill-in-the-blank.d.ts +1 -1
  78. package/dist/questions/hotspot.d.ts +21 -0
  79. package/dist/questions/index.d.ts +11 -1
  80. package/dist/questions/inline-choice.d.ts +21 -0
  81. package/dist/questions/matching.d.ts +22 -0
  82. package/dist/questions/multiple-choice.d.ts +1 -1
  83. package/dist/questions/numeric.d.ts +11 -0
  84. package/dist/questions/ordering.d.ts +12 -0
  85. package/dist/questions/question-renderer.d.ts +1 -1
  86. package/dist/questions/scenario.d.ts +23 -0
  87. package/dist/questions/scoring.d.ts +22 -0
  88. package/dist/questions/spreadsheet.d.ts +29 -0
  89. package/dist/questions/true-false.d.ts +1 -1
  90. package/dist/questions/types.d.ts +106 -1
  91. package/dist/questions/use-drag-reorder.d.ts +17 -0
  92. package/dist/sections/AnnouncementFeed/AnnouncementFeed.d.ts +1 -1
  93. package/dist/sections/AnnouncementFeed/types.d.ts +15 -1
  94. package/dist/sections/AssessmentReview/AssessmentReview.d.ts +1 -1
  95. package/dist/sections/AssessmentReview/types.d.ts +6 -0
  96. package/dist/sections/AssignmentSubmission/AssignmentSubmission.d.ts +1 -1
  97. package/dist/sections/AssignmentSubmission/types.d.ts +6 -0
  98. package/dist/sections/CertificateViewer/CertificateViewer.d.ts +1 -1
  99. package/dist/sections/CertificateViewer/certificate-variants.d.ts +42 -0
  100. package/dist/sections/CertificateViewer/types.d.ts +13 -5
  101. package/dist/sections/CourseCatalog/CourseCatalog.d.ts +2 -0
  102. package/dist/sections/CourseCatalog/types.d.ts +80 -0
  103. package/dist/sections/CourseOutline/CourseOutline.d.ts +1 -1
  104. package/dist/sections/CourseOutline/types.d.ts +6 -0
  105. package/dist/sections/DiscussionThread/DiscussionThread.d.ts +1 -1
  106. package/dist/sections/DiscussionThread/types.d.ts +6 -0
  107. package/dist/sections/EnrollmentWizard/EnrollmentWizard.d.ts +2 -0
  108. package/dist/sections/EnrollmentWizard/types.d.ts +66 -0
  109. package/dist/sections/ExamSession/ExamSession.d.ts +1 -1
  110. package/dist/sections/ExamSession/types.d.ts +12 -1
  111. package/dist/sections/FlashcardStudySession/FlashcardStudySession.d.ts +1 -1
  112. package/dist/sections/FlashcardStudySession/types.d.ts +6 -0
  113. package/dist/sections/ForumBoard/ForumBoard.d.ts +8 -0
  114. package/dist/sections/ForumBoard/types.d.ts +78 -0
  115. package/dist/sections/GradebookTable/GradebookTable.d.ts +1 -1
  116. package/dist/sections/GradebookTable/types.d.ts +14 -0
  117. package/dist/sections/LecturePlayer/LecturePlayer.d.ts +1 -1
  118. package/dist/sections/LecturePlayer/types.d.ts +8 -0
  119. package/dist/sections/LessonPage/LessonPage.d.ts +1 -1
  120. package/dist/sections/LessonPage/types.d.ts +6 -0
  121. package/dist/sections/PracticeQuiz/PracticeQuiz.d.ts +1 -1
  122. package/dist/sections/PracticeQuiz/types.d.ts +6 -0
  123. package/dist/sections/ProgressDashboard/ProgressDashboard.d.ts +1 -1
  124. package/dist/sections/ProgressDashboard/types.d.ts +6 -0
  125. package/dist/sections/QuizSession/QuizSession.d.ts +1 -1
  126. package/dist/sections/QuizSession/types.d.ts +12 -1
  127. package/dist/sections/RequirementsChecklist/RequirementsChecklist.d.ts +8 -0
  128. package/dist/sections/RequirementsChecklist/types.d.ts +43 -0
  129. package/dist/sections/ResourceLibrary/ResourceLibrary.d.ts +1 -1
  130. package/dist/sections/ResourceLibrary/types.d.ts +15 -1
  131. package/dist/sections/RubricView/RubricView.d.ts +9 -0
  132. package/dist/sections/RubricView/types.d.ts +56 -0
  133. package/dist/sections/ScrollableQuiz/ScrollableQuiz.d.ts +1 -1
  134. package/dist/sections/ScrollableQuiz/types.d.ts +6 -0
  135. package/dist/sections/StudentProfile/StudentProfile.d.ts +2 -0
  136. package/dist/sections/StudentProfile/types.d.ts +98 -0
  137. package/dist/sections/SurveyForm/SurveyForm.d.ts +1 -1
  138. package/dist/sections/SurveyForm/types.d.ts +6 -0
  139. package/dist/sections/_shared/merge-answers.d.ts +9 -0
  140. package/dist/sections/_shared/section-shell.d.ts +20 -0
  141. package/dist/sections/_shared/use-assessment-session.d.ts +30 -0
  142. package/dist/sections/index.d.ts +13 -1
  143. package/dist/sections.cjs +1 -1
  144. package/dist/sections.js +282 -1786
  145. package/dist/social/post-card.d.ts +1 -1
  146. package/dist/tabs-BsfVo2Bl.cjs +173 -0
  147. package/dist/tabs-BuY1iNJE.js +22305 -0
  148. package/dist/ui/alert.d.ts +1 -1
  149. package/dist/ui/badge.d.ts +1 -1
  150. package/dist/ui/button.d.ts +1 -1
  151. package/dist/ui/drawer.d.ts +84 -0
  152. package/dist/ui/index.d.ts +5 -0
  153. package/dist/ui/progress.d.ts +1 -1
  154. package/dist/ui/rich-text-editor.d.ts +32 -0
  155. package/dist/ui/rich-text-toolbar.d.ts +8 -0
  156. package/dist/ui/toast.d.ts +43 -0
  157. package/dist/utils/array-utils.d.ts +4 -0
  158. package/dist/utils/debounce.d.ts +5 -1
  159. package/dist/utils/flatten-leaves.d.ts +6 -0
  160. package/dist/utils/format-file-size.d.ts +1 -0
  161. package/dist/utils/format-timestamp.d.ts +1 -0
  162. package/dist/utils/is-empty-html.d.ts +5 -0
  163. package/dist/utils/pick-palette-color.d.ts +19 -0
  164. package/dist/utils/shuffle.d.ts +1 -0
  165. package/dist/utils/string-utils.d.ts +12 -0
  166. package/dist/video/types.d.ts +15 -0
  167. package/dist/video/video-bookmark.d.ts +1 -1
  168. package/dist/video/video-player.d.ts +1 -1
  169. package/dist/video/video-playlist-item.d.ts +1 -1
  170. package/dist/withProGate-BWqcKdPM.js +137 -0
  171. package/dist/withProGate-DX6XqKLp.cjs +1 -0
  172. package/package.json +40 -137
  173. package/src/assessment-toolbar/assessment-toolbar.tsx +54 -49
  174. package/src/assessment-toolbar/index.ts +6 -0
  175. package/src/assessment-toolbar/question-header-bar.tsx +61 -0
  176. package/src/assessment-toolbar/question-materials-drawer.tsx +55 -0
  177. package/src/assessment-toolbar/question-navigator.tsx +13 -36
  178. package/src/assessment-toolbar/timer-display.tsx +6 -5
  179. package/src/assessment-toolbar/types.ts +54 -4
  180. package/src/assessment-toolbar/use-countdown.ts +153 -0
  181. package/src/common/empty-state.tsx +1 -0
  182. package/src/common/index.ts +5 -0
  183. package/src/common/pagination.tsx +135 -0
  184. package/src/common/search-input.tsx +8 -6
  185. package/src/common/stepper.tsx +100 -0
  186. package/src/common/types.ts +41 -0
  187. package/src/content/attachment-list.tsx +92 -0
  188. package/src/content/audio-player.tsx +196 -0
  189. package/src/content/code-block.tsx +113 -0
  190. package/src/content/content-block.tsx +68 -2
  191. package/src/content/embed-block.tsx +78 -0
  192. package/src/content/file-upload-zone.tsx +11 -6
  193. package/src/content/index.ts +9 -0
  194. package/src/content/types.ts +46 -0
  195. package/src/curriculum/course-card.tsx +199 -0
  196. package/src/curriculum/curriculum-item.tsx +9 -5
  197. package/src/curriculum/curriculum-tree.tsx +20 -13
  198. package/src/curriculum/index.ts +2 -0
  199. package/src/curriculum/types.ts +2 -2
  200. package/src/feedback/feedback-banner.tsx +12 -14
  201. package/src/flashcards/flashcard-deck.tsx +1 -9
  202. package/src/flashcards/flashcard.tsx +29 -9
  203. package/src/index.ts +3 -0
  204. package/src/license/HydraContext.tsx +62 -0
  205. package/src/license/ProBadge.tsx +43 -0
  206. package/src/license/index.ts +7 -0
  207. package/src/license/tiers.ts +24 -0
  208. package/src/license/useHydraLicense.ts +10 -0
  209. package/src/license/validate.ts +90 -0
  210. package/src/license/withProGate.tsx +21 -0
  211. package/src/modules/AssignmentModule/AssignmentModule.tsx +314 -0
  212. package/src/modules/AssignmentModule/types.ts +77 -0
  213. package/src/modules/CertificateModule/CertificateModule.tsx +173 -0
  214. package/src/modules/CertificateModule/types.ts +49 -0
  215. package/src/modules/CourseCatalogModule/CourseCatalogModule.tsx +126 -0
  216. package/src/modules/CourseCatalogModule/types.ts +47 -0
  217. package/src/modules/CoursePlayer/CoursePlayer.tsx +80 -69
  218. package/src/modules/DiscussionModule/DiscussionModule.tsx +145 -0
  219. package/src/modules/DiscussionModule/types.ts +54 -0
  220. package/src/modules/ExamModule/ExamModule.tsx +151 -0
  221. package/src/modules/ExamModule/types.ts +57 -0
  222. package/src/modules/FlashcardLab/FlashcardLab.tsx +39 -21
  223. package/src/modules/FlashcardLab/types.ts +2 -0
  224. package/src/modules/GradeCenterModule/GradeCenterModule.tsx +174 -0
  225. package/src/modules/GradeCenterModule/types.ts +65 -0
  226. package/src/modules/QuizModule/QuizModule.tsx +58 -178
  227. package/src/modules/QuizModule/types.ts +10 -15
  228. package/src/modules/StudentDashboardModule/StudentDashboardModule.tsx +117 -0
  229. package/src/modules/StudentDashboardModule/types.ts +56 -0
  230. package/src/modules/StudentProfileModule/StudentProfileModule.tsx +289 -0
  231. package/src/modules/StudentProfileModule/types.ts +45 -0
  232. package/src/modules/SurveyModule/SurveyModule.tsx +185 -0
  233. package/src/modules/SurveyModule/types.ts +53 -0
  234. package/src/modules/_shared/assessment-intro.tsx +75 -0
  235. package/src/modules/_shared/assessment-results.tsx +133 -0
  236. package/src/modules/_shared/types.ts +11 -0
  237. package/src/modules/_shared/use-timer.ts +49 -0
  238. package/src/modules/index.ts +33 -0
  239. package/src/progress/achievement-badge.tsx +52 -0
  240. package/src/progress/activity-timeline.tsx +84 -0
  241. package/src/progress/grade-indicator.tsx +9 -1
  242. package/src/progress/index.ts +7 -0
  243. package/src/progress/progress-ring.tsx +2 -1
  244. package/src/progress/stat-card.tsx +37 -18
  245. package/src/progress/streak-badge.tsx +35 -0
  246. package/src/progress/types.ts +103 -0
  247. package/src/provider/HydraProvider.tsx +15 -6
  248. package/src/questions/choice.tsx +19 -14
  249. package/src/questions/confidence-indicator.tsx +107 -0
  250. package/src/questions/essay.tsx +28 -28
  251. package/src/questions/fill-in-the-blank.tsx +20 -19
  252. package/src/questions/hotspot.tsx +154 -0
  253. package/src/questions/index.ts +18 -0
  254. package/src/questions/inline-choice.tsx +152 -0
  255. package/src/questions/matching.tsx +229 -0
  256. package/src/questions/multiple-choice.tsx +19 -14
  257. package/src/questions/numeric.tsx +106 -0
  258. package/src/questions/ordering.tsx +167 -0
  259. package/src/questions/question-renderer.tsx +24 -2
  260. package/src/questions/scenario.tsx +140 -0
  261. package/src/questions/scoring.ts +201 -0
  262. package/src/questions/spreadsheet.tsx +260 -0
  263. package/src/questions/true-false.tsx +19 -14
  264. package/src/questions/types.ts +123 -1
  265. package/src/questions/use-drag-reorder.ts +80 -0
  266. package/src/sections/AnnouncementFeed/AnnouncementFeed.tsx +66 -23
  267. package/src/sections/AnnouncementFeed/types.ts +15 -1
  268. package/src/sections/AssessmentReview/AssessmentReview.tsx +50 -2
  269. package/src/sections/AssessmentReview/types.ts +6 -0
  270. package/src/sections/AssignmentSubmission/AssignmentSubmission.tsx +44 -6
  271. package/src/sections/AssignmentSubmission/types.ts +6 -0
  272. package/src/sections/CertificateViewer/CertificateViewer.tsx +215 -60
  273. package/src/sections/CertificateViewer/certificate-variants.tsx +170 -0
  274. package/src/sections/CertificateViewer/types.ts +19 -5
  275. package/src/sections/CourseCatalog/CourseCatalog.tsx +220 -0
  276. package/src/sections/CourseCatalog/types.ts +76 -0
  277. package/src/sections/CourseOutline/CourseOutline.tsx +45 -14
  278. package/src/sections/CourseOutline/types.ts +6 -0
  279. package/src/sections/DiscussionThread/DiscussionThread.tsx +55 -11
  280. package/src/sections/DiscussionThread/types.ts +6 -0
  281. package/src/sections/EnrollmentWizard/EnrollmentWizard.tsx +343 -0
  282. package/src/sections/EnrollmentWizard/types.ts +65 -0
  283. package/src/sections/ExamSession/ExamSession.tsx +125 -82
  284. package/src/sections/ExamSession/types.ts +12 -1
  285. package/src/sections/FlashcardStudySession/FlashcardStudySession.tsx +53 -36
  286. package/src/sections/FlashcardStudySession/types.ts +6 -0
  287. package/src/sections/ForumBoard/ForumBoard.tsx +342 -0
  288. package/src/sections/ForumBoard/types.ts +81 -0
  289. package/src/sections/GradebookTable/GradebookTable.tsx +55 -2
  290. package/src/sections/GradebookTable/types.ts +14 -0
  291. package/src/sections/LecturePlayer/LecturePlayer.tsx +63 -37
  292. package/src/sections/LecturePlayer/types.ts +8 -0
  293. package/src/sections/LessonPage/LessonPage.tsx +40 -13
  294. package/src/sections/LessonPage/types.ts +6 -0
  295. package/src/sections/PracticeQuiz/PracticeQuiz.tsx +119 -98
  296. package/src/sections/PracticeQuiz/types.ts +6 -0
  297. package/src/sections/ProgressDashboard/ProgressDashboard.tsx +121 -67
  298. package/src/sections/ProgressDashboard/types.ts +6 -0
  299. package/src/sections/QuizSession/QuizSession.tsx +115 -67
  300. package/src/sections/QuizSession/types.ts +12 -1
  301. package/src/sections/RequirementsChecklist/RequirementsChecklist.tsx +147 -0
  302. package/src/sections/RequirementsChecklist/types.ts +44 -0
  303. package/src/sections/ResourceLibrary/ResourceLibrary.tsx +68 -17
  304. package/src/sections/ResourceLibrary/types.ts +15 -1
  305. package/src/sections/RubricView/RubricView.tsx +174 -0
  306. package/src/sections/RubricView/types.ts +58 -0
  307. package/src/sections/ScrollableQuiz/ScrollableQuiz.tsx +58 -23
  308. package/src/sections/ScrollableQuiz/types.ts +6 -0
  309. package/src/sections/StudentProfile/StudentProfile.tsx +279 -0
  310. package/src/sections/StudentProfile/types.ts +99 -0
  311. package/src/sections/SurveyForm/SurveyForm.tsx +40 -10
  312. package/src/sections/SurveyForm/types.ts +6 -0
  313. package/src/sections/_shared/merge-answers.ts +22 -0
  314. package/src/sections/_shared/section-shell.tsx +64 -0
  315. package/src/sections/_shared/use-assessment-session.ts +125 -0
  316. package/src/sections/index.ts +42 -1
  317. package/src/social/post-card.tsx +8 -19
  318. package/src/social/user-avatar.tsx +10 -5
  319. package/src/styles/globals.css +52 -41
  320. package/src/ui/badge.tsx +8 -0
  321. package/src/ui/drawer.tsx +600 -0
  322. package/src/ui/index.ts +21 -0
  323. package/src/ui/progress.tsx +4 -0
  324. package/src/ui/rich-text-editor.tsx +119 -0
  325. package/src/ui/rich-text-toolbar.tsx +157 -0
  326. package/src/ui/toast.tsx +170 -0
  327. package/src/utils/array-utils.ts +17 -0
  328. package/src/utils/debounce.ts +8 -2
  329. package/src/utils/flatten-leaves.ts +17 -0
  330. package/src/utils/format-file-size.ts +5 -0
  331. package/src/utils/format-timestamp.ts +13 -0
  332. package/src/utils/is-empty-html.ts +7 -0
  333. package/src/utils/pick-palette-color.ts +33 -0
  334. package/src/utils/shuffle.ts +8 -0
  335. package/src/utils/string-utils.ts +30 -0
  336. package/src/video/types.ts +16 -0
  337. package/src/video/video-bookmark.tsx +4 -3
  338. package/src/video/video-chapter-list.tsx +9 -4
  339. package/src/video/video-player.tsx +24 -5
  340. package/src/video/video-playlist-item.tsx +8 -3
  341. package/src/video/video-thumbnail-card.tsx +4 -0
  342. package/src/video/video-transcript.tsx +8 -5
  343. package/dist/table-BrS5cDQu.js +0 -2510
  344. package/dist/table-D6AkBBEo.cjs +0 -1
@@ -0,0 +1,125 @@
1
+ import { useCallback, useEffect, useMemo, useRef, useState } from "react";
2
+ import type { QuestionNavigatorItem } from "../../assessment-toolbar/types";
3
+ import type { QuestionData, QuestionMaterial, SessionAnswer } from "../../questions/types";
4
+ import { mergeSessionAnswers } from "./merge-answers";
5
+
6
+ export interface UseAssessmentSessionOptions {
7
+ questions: QuestionData[];
8
+ initialAnswers?: SessionAnswer[];
9
+ onAnswerChange?: (answers: SessionAnswer[]) => void;
10
+ questionMaterials?: QuestionMaterial[];
11
+ }
12
+
13
+ export function useAssessmentSession({
14
+ questions,
15
+ initialAnswers = [],
16
+ onAnswerChange,
17
+ questionMaterials,
18
+ }: UseAssessmentSessionOptions) {
19
+ const [currentIndex, setCurrentIndex] = useState(0);
20
+ const [sessionAnswers, setSessionAnswers] = useState<SessionAnswer[]>(initialAnswers);
21
+ const [flaggedUids, setFlaggedUids] = useState<Set<string>>(new Set());
22
+ const [materialsOpen, setMaterialsOpen] = useState(false);
23
+ const questionAreaRef = useRef<HTMLDivElement>(null);
24
+ const isFirstRender = useRef(true);
25
+
26
+ // Refs for stable callbacks
27
+ const questionsRef = useRef(questions);
28
+ questionsRef.current = questions;
29
+ const currentIndexRef = useRef(currentIndex);
30
+ currentIndexRef.current = currentIndex;
31
+ const onAnswerChangeRef = useRef(onAnswerChange);
32
+ onAnswerChangeRef.current = onAnswerChange;
33
+
34
+ const currentQuestion = questions[currentIndex];
35
+
36
+ // Focus question area on navigation (skip first render)
37
+ useEffect(() => {
38
+ if (isFirstRender.current) {
39
+ isFirstRender.current = false;
40
+ return;
41
+ }
42
+ questionAreaRef.current?.focus();
43
+ }, [currentIndex]);
44
+
45
+ const currentQuestionAnswers = useMemo(
46
+ () =>
47
+ currentQuestion
48
+ ? sessionAnswers.filter((a) => a.uid === currentQuestion.uid)
49
+ : [],
50
+ [sessionAnswers, currentQuestion],
51
+ );
52
+
53
+ const currentMaterials = useMemo(
54
+ () => questionMaterials?.filter((m) => m.questionUid === currentQuestion?.uid) ?? [],
55
+ [questionMaterials, currentQuestion],
56
+ );
57
+
58
+ const navigatorItems = useMemo<QuestionNavigatorItem[]>(
59
+ () =>
60
+ questions.map((q, idx) => ({
61
+ uid: q.uid,
62
+ sequence: idx,
63
+ isFlagged: flaggedUids.has(q.uid),
64
+ isAnswered: sessionAnswers.some((a) => a.uid === q.uid),
65
+ isSkipped: false,
66
+ })),
67
+ [questions, sessionAnswers, flaggedUids],
68
+ );
69
+
70
+ const handleAnswer = useCallback(
71
+ (rawAnswers: { uid: string; content?: string }[]) => {
72
+ const q = questionsRef.current[currentIndexRef.current];
73
+ if (!q) return;
74
+ setSessionAnswers((prev) =>
75
+ mergeSessionAnswers(prev, q.uid, rawAnswers, onAnswerChangeRef.current),
76
+ );
77
+ },
78
+ [],
79
+ );
80
+
81
+ const handleNavigate = useCallback(
82
+ (uid: string) => {
83
+ const idx = questions.findIndex((q) => q.uid === uid);
84
+ if (idx !== -1) setCurrentIndex(idx);
85
+ },
86
+ [questions],
87
+ );
88
+
89
+ const handleToggleFlag = useCallback((uid: string) => {
90
+ setFlaggedUids((prev) => {
91
+ const next = new Set(prev);
92
+ if (next.has(uid)) next.delete(uid);
93
+ else next.add(uid);
94
+ return next;
95
+ });
96
+ }, []);
97
+
98
+ const goNext = useCallback(() => {
99
+ setCurrentIndex((i) => Math.min(i + 1, questionsRef.current.length - 1));
100
+ }, []);
101
+
102
+ const goPrevious = useCallback(() => {
103
+ setCurrentIndex((i) => Math.max(i - 1, 0));
104
+ }, []);
105
+
106
+ return {
107
+ currentIndex,
108
+ currentQuestion,
109
+ sessionAnswers,
110
+ flaggedUids,
111
+ materialsOpen,
112
+ setMaterialsOpen,
113
+ questionAreaRef,
114
+ currentQuestionAnswers,
115
+ currentMaterials,
116
+ navigatorItems,
117
+ handleAnswer,
118
+ handleNavigate,
119
+ handleToggleFlag,
120
+ goNext,
121
+ goPrevious,
122
+ hasNext: currentIndex < questions.length - 1,
123
+ hasPrevious: currentIndex > 0,
124
+ };
125
+ }
@@ -87,4 +87,45 @@ export type {
87
87
  } from "./ProgressDashboard/types";
88
88
 
89
89
  export { CertificateViewer } from "./CertificateViewer/CertificateViewer";
90
- export type { CertificateViewerProps } from "./CertificateViewer/types";
90
+ export type {
91
+ CertificateViewerProps,
92
+ CertificateVariant,
93
+ } from "./CertificateViewer/types";
94
+
95
+ export { RubricView } from "./RubricView/RubricView";
96
+ export type {
97
+ RubricViewProps,
98
+ RubricCriterion,
99
+ RubricLevel,
100
+ } from "./RubricView/types";
101
+
102
+ export { RequirementsChecklist } from "./RequirementsChecklist/RequirementsChecklist";
103
+ export type {
104
+ RequirementsChecklistProps,
105
+ Requirement,
106
+ } from "./RequirementsChecklist/types";
107
+
108
+ export { ForumBoard } from "./ForumBoard/ForumBoard";
109
+ export type { ForumBoardProps, ForumTopic, ForumSortOrder } from "./ForumBoard/types";
110
+
111
+ export { EnrollmentWizard } from "./EnrollmentWizard/EnrollmentWizard";
112
+ export type {
113
+ EnrollmentWizardProps,
114
+ EnrollmentCourse,
115
+ Prerequisite,
116
+ } from "./EnrollmentWizard/types";
117
+
118
+ export { CourseCatalog } from "./CourseCatalog/CourseCatalog";
119
+ export type {
120
+ CourseCatalogProps,
121
+ CourseInfo,
122
+ } from "./CourseCatalog/types";
123
+
124
+ export { StudentProfile } from "./StudentProfile/StudentProfile";
125
+ export type {
126
+ StudentProfileProps,
127
+ StudentInfo,
128
+ EnrolledCourse,
129
+ ProfileAchievement,
130
+ ProfileCertificate,
131
+ } from "./StudentProfile/types";
@@ -1,29 +1,17 @@
1
+ import { memo } from "react";
1
2
  import { Pin } from "lucide-react";
2
3
  import { UserAvatar } from "./user-avatar";
3
4
  import { Separator } from "../ui/separator";
4
5
  import type { PostCardProps } from "./types";
5
6
  import { cn } from "../lib/utils";
6
-
7
- function formatTimestamp(iso: string): string {
8
- const date = new Date(iso);
9
- const now = new Date();
10
- const diffMs = now.getTime() - date.getTime();
11
- const diffMins = Math.floor(diffMs / 60000);
12
- if (diffMins < 1) return "Just now";
13
- if (diffMins < 60) return `${diffMins}m ago`;
14
- const diffHours = Math.floor(diffMins / 60);
15
- if (diffHours < 24) return `${diffHours}h ago`;
16
- const diffDays = Math.floor(diffHours / 24);
17
- if (diffDays < 7) return `${diffDays}d ago`;
18
- return date.toLocaleDateString();
19
- }
7
+ import { formatTimestamp } from "../utils/format-timestamp";
20
8
 
21
9
  const HIGHLIGHT_BORDERS: Record<string, string> = {
22
10
  pinned: "var(--warning)",
23
11
  answer: "var(--success)",
24
12
  };
25
13
 
26
- export function PostCard({
14
+ export const PostCard = memo(function PostCard({
27
15
  author,
28
16
  content,
29
17
  createdAt,
@@ -73,9 +61,10 @@ export function PostCard({
73
61
  )}
74
62
  {highlight === "pinned" && <Pin size={14} />}
75
63
  </div>
76
- <p className="mt-1 text-sm whitespace-pre-wrap">
77
- {content}
78
- </p>
64
+ <div
65
+ className="mt-1 text-sm [&_p]:mb-1 [&_ul]:list-disc [&_ul]:pl-5 [&_ol]:list-decimal [&_ol]:pl-5 [&_a]:text-primary [&_a]:underline"
66
+ dangerouslySetInnerHTML={{ __html: content }}
67
+ />
79
68
  {actions && (
80
69
  <>
81
70
  <Separator className="my-2" />
@@ -88,4 +77,4 @@ export function PostCard({
88
77
  </div>
89
78
  </div>
90
79
  );
91
- }
80
+ });
@@ -13,7 +13,7 @@ const ROLE_LABELS: Record<string, string> = {
13
13
 
14
14
  const ROLE_COLORS: Record<string, string> = {
15
15
  instructor: "var(--primary)",
16
- ta: "var(--purple)",
16
+ ta: "var(--palette-0)",
17
17
  admin: "var(--destructive)",
18
18
  student: "var(--muted-foreground)",
19
19
  };
@@ -59,6 +59,7 @@ export function UserAvatar({
59
59
  className="size-full object-cover"
60
60
  src={avatarUrl}
61
61
  alt={displayName}
62
+ loading="lazy"
62
63
  onError={() => setImgError(true)}
63
64
  />
64
65
  ) : (
@@ -66,10 +67,14 @@ export function UserAvatar({
66
67
  )}
67
68
  </div>
68
69
  {showBadge && (
69
- <span
70
- className="absolute -bottom-0.5 -right-0.5 size-2.5 rounded-full border-2 border-background"
71
- style={{ background: ROLE_COLORS[role] ?? "var(--muted-foreground)" }}
72
- />
70
+ <>
71
+ <span
72
+ className="absolute -bottom-0.5 -right-0.5 size-2.5 rounded-full border-2 border-background"
73
+ style={{ background: ROLE_COLORS[role] ?? "var(--muted-foreground)" }}
74
+ aria-hidden="true"
75
+ />
76
+ <span className="sr-only">{ROLE_LABELS[role] ?? role}</span>
77
+ </>
73
78
  )}
74
79
  </div>
75
80
  );
@@ -1,40 +1,10 @@
1
- @import "tailwindcss";
2
-
3
- @custom-variant dark (&:is(.dark *));
4
-
5
- @theme inline {
6
- --color-background: var(--background);
7
- --color-foreground: var(--foreground);
8
- --color-card: var(--card);
9
- --color-card-foreground: var(--card-foreground);
10
- --color-popover: var(--popover);
11
- --color-popover-foreground: var(--popover-foreground);
12
- --color-primary: var(--primary);
13
- --color-primary-foreground: var(--primary-foreground);
14
- --color-secondary: var(--secondary);
15
- --color-secondary-foreground: var(--secondary-foreground);
16
- --color-muted: var(--muted);
17
- --color-muted-foreground: var(--muted-foreground);
18
- --color-accent: var(--accent);
19
- --color-accent-foreground: var(--accent-foreground);
20
- --color-destructive: var(--destructive);
21
- --color-destructive-foreground: var(--destructive-foreground);
22
- --color-border: var(--border);
23
- --color-input: var(--input);
24
- --color-ring: var(--ring);
25
- --color-success: var(--success);
26
- --color-success-foreground: var(--success-foreground);
27
- --color-warning: var(--warning);
28
- --color-warning-foreground: var(--warning-foreground);
29
- --color-info: var(--info);
30
- --color-info-foreground: var(--info-foreground);
31
- --color-purple: var(--purple);
32
- --color-teal: var(--teal);
33
- --radius-sm: calc(var(--radius) - 4px);
34
- --radius-md: calc(var(--radius) - 2px);
35
- --radius-lg: var(--radius);
36
- --radius-xl: calc(var(--radius) + 4px);
37
- }
1
+ /*
2
+ * HydraLMS design tokens & base styles.
3
+ * NOTE: This file must NOT contain @import "tailwindcss" or Tailwind
4
+ * directives (@theme, @custom-variant). Those belong in the consumer's
5
+ * CSS entry point, which should also include an @source directive
6
+ * pointing to this library's source files. See the getting-started guide.
7
+ */
38
8
 
39
9
  .hydra-root {
40
10
  --radius: 0.625rem;
@@ -65,8 +35,16 @@
65
35
  --warning-foreground: oklch(0.98 0.016 73);
66
36
  --info: oklch(0.623 0.214 259);
67
37
  --info-foreground: oklch(0.97 0.014 254);
68
- --purple: oklch(0.627 0.265 303);
69
- --teal: oklch(0.697 0.146 182);
38
+
39
+ /* Palette colors for visual variety */
40
+ --palette-0: oklch(0.627 0.265 303);
41
+ --palette-0-foreground: oklch(0.985 0 0);
42
+ --palette-1: oklch(0.680 0.162 195);
43
+ --palette-1-foreground: oklch(0.985 0 0);
44
+ --palette-2: oklch(0.720 0.185 130);
45
+ --palette-2-foreground: oklch(0.985 0 0);
46
+ --palette-3: oklch(0.700 0.180 70);
47
+ --palette-3-foreground: oklch(0.15 0 0);
70
48
 
71
49
  color: var(--foreground);
72
50
  background: var(--background);
@@ -101,8 +79,16 @@
101
79
  --warning-foreground: oklch(0.15 0.05 60);
102
80
  --info: oklch(0.623 0.214 259);
103
81
  --info-foreground: oklch(0.15 0.05 255);
104
- --purple: oklch(0.627 0.265 303);
105
- --teal: oklch(0.697 0.146 182);
82
+
83
+ /* Palette colors for visual variety */
84
+ --palette-0: oklch(0.700 0.200 303);
85
+ --palette-0-foreground: oklch(0.15 0 0);
86
+ --palette-1: oklch(0.730 0.150 195);
87
+ --palette-1-foreground: oklch(0.15 0 0);
88
+ --palette-2: oklch(0.760 0.170 130);
89
+ --palette-2-foreground: oklch(0.15 0 0);
90
+ --palette-3: oklch(0.740 0.165 70);
91
+ --palette-3-foreground: oklch(0.15 0 0);
106
92
  }
107
93
 
108
94
  /* Rich text support for flashcard back content */
@@ -123,3 +109,28 @@
123
109
  .flashcard-back-content p {
124
110
  margin: 0.25em 0;
125
111
  }
112
+
113
+ /* Tiptap rich text editor */
114
+ .tiptap:focus {
115
+ outline: none;
116
+ }
117
+
118
+ .tiptap p.is-editor-empty:first-child::before {
119
+ color: var(--muted-foreground);
120
+ content: attr(data-placeholder);
121
+ float: left;
122
+ height: 0;
123
+ pointer-events: none;
124
+ }
125
+
126
+ /* Respect user preference for reduced motion (WCAG 2.3.3) */
127
+ @media (prefers-reduced-motion: reduce) {
128
+ .hydra-root *,
129
+ .hydra-root *::before,
130
+ .hydra-root *::after {
131
+ animation-duration: 0.01ms !important;
132
+ animation-iteration-count: 1 !important;
133
+ transition-duration: 0.01ms !important;
134
+ scroll-behavior: auto !important;
135
+ }
136
+ }
package/src/ui/badge.tsx CHANGED
@@ -24,6 +24,14 @@ const badgeVariants = cva(
24
24
  "border-transparent bg-info/15 text-info",
25
25
  muted:
26
26
  "border-transparent bg-muted text-muted-foreground",
27
+ palette0:
28
+ "border-transparent bg-palette-0/15 text-palette-0",
29
+ palette1:
30
+ "border-transparent bg-palette-1/15 text-palette-1",
31
+ palette2:
32
+ "border-transparent bg-palette-2/15 text-palette-2",
33
+ palette3:
34
+ "border-transparent bg-palette-3/15 text-palette-3",
27
35
  },
28
36
  },
29
37
  defaultVariants: {