Compare commits

...

370 Commits

Author SHA1 Message Date
Frappe PR Bot
795d95b482 chore(release): Bumped to Version 2.26.0 2025-03-05 13:04:57 +00:00
Jannat Patel
5b5b95c85c Merge pull request #1363 from pateljannat/scorm-issue-js-files
fix: scorm files getting wrong path
2025-03-05 17:03:40 +05:30
Jannat Patel
8490b07c90 fix: scorm files getting wrong path 2025-03-05 16:31:14 +05:30
Jannat Patel
dee2c51c60 Merge pull request #1359 from pateljannat/evaluation-validation-issue
fix: allow scheduling evals if future eval has been cancelled
2025-03-04 17:47:42 +05:30
Jannat Patel
4149fa6ce4 fix: renamed evaluation and certification buttons 2025-03-04 17:38:43 +05:30
Jannat Patel
7a69611f09 Merge pull request #1358 from pateljannat/payment-reminder-issue
fix: don't send payment reminder if member has already paid later
2025-03-04 17:34:30 +05:30
Jannat Patel
6692252df9 fix: allow scheduling evals if furture eval has been calcelled 2025-03-04 17:33:39 +05:30
Jannat Patel
486ce1bdb0 Merge pull request #1357 from pateljannat/course-certification-filter
refactor: course list fetching and filters
2025-03-04 17:27:01 +05:30
Jannat Patel
cceff77bc2 fix: don't send payment reminder if member has already paid later 2025-03-04 17:24:03 +05:30
Jannat Patel
22a9169f87 fix: show progress bar for enrolled courses 2025-03-04 17:14:01 +05:30
Jannat Patel
47a30763a0 refactor: course list fetching and filters 2025-03-04 17:02:47 +05:30
Jannat Patel
73379a1bd8 Merge pull request #1354 from pateljannat/dont-override-user
fix: reverting user doctype override
2025-03-04 13:08:12 +05:30
Jannat Patel
7cc46629b4 test: increased login request timeout in ui tests 2025-03-04 12:53:33 +05:30
Jannat Patel
67304245ba test: increased login request timeout in ui tests 2025-03-04 12:39:22 +05:30
Jannat Patel
8edd3a1a34 chore: upgrading actions/cache to v4 for ui tests 2025-03-04 11:43:07 +05:30
Jannat Patel
e4bc7c8d78 Merge pull request #1356 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-03-04 10:22:56 +05:30
Jannat Patel
a8af78d400 chore: Esperanto translations 2025-03-04 02:59:49 +05:30
Jannat Patel
0afe3de818 chore: Bosnian translations 2025-03-04 02:59:48 +05:30
Jannat Patel
3c81aadec6 chore: Persian translations 2025-03-04 02:59:47 +05:30
Jannat Patel
1dfcb035da chore: Chinese Simplified translations 2025-03-04 02:59:45 +05:30
Jannat Patel
77b24882a9 chore: Turkish translations 2025-03-04 02:59:44 +05:30
Jannat Patel
1fd0673257 chore: Swedish translations 2025-03-04 02:59:42 +05:30
Jannat Patel
dbda76e0ce chore: Russian translations 2025-03-04 02:59:40 +05:30
Jannat Patel
a9d22521ce chore: Polish translations 2025-03-04 02:59:39 +05:30
Jannat Patel
6da1d9629f chore: Hungarian translations 2025-03-04 02:59:37 +05:30
Jannat Patel
37b61a7087 chore: German translations 2025-03-04 02:59:35 +05:30
Jannat Patel
9b484e6ee9 chore: Arabic translations 2025-03-04 02:59:34 +05:30
Jannat Patel
5ef67ef21c chore: Spanish translations 2025-03-04 02:59:32 +05:30
Jannat Patel
f902166643 chore: French translations 2025-03-04 02:59:31 +05:30
Md Hussain Nagaria
8f91466b3d Merge pull request #1355 from frappe/enhance-timezone
feat: autofill timezone based on user timezone
2025-03-03 22:43:24 +05:30
Hussain Nagaria
fa1621c3d1 feat: autofill client timezone based on user timezone 2025-03-03 22:42:43 +05:30
Jannat Patel
2acd45feae fix: reverting user doctype override 2025-03-03 19:54:41 +05:30
Jannat Patel
f19e974b9d Merge pull request #1353 from pateljannat/mark-eval-request-complete
feat: mark evaluation requests as complete
2025-03-03 17:14:42 +05:30
Jannat Patel
01598ac002 feat: mark evaluation requests as complete 2025-03-03 17:01:10 +05:30
Frappe PR Bot
9b3906359b chore(release): Bumped to Version 2.25.0 2025-03-03 10:20:45 +00:00
Jannat Patel
4224580d6f Merge pull request #1351 from pateljannat/billing-flow-changes
fix: redirect to FC dashboard when login to FC
2025-03-03 14:59:21 +05:30
Jannat Patel
07d30647d8 chore: upgrading actions/cache to v4 for ci tests 2025-03-03 13:58:39 +05:30
Jannat Patel
263096fc77 fix: changed frappe cloud login flow 2025-03-03 13:53:16 +05:30
Jannat Patel
b510cbce7f Merge pull request #1347 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-03-03 10:13:26 +05:30
Jannat Patel
0b84dc3266 Merge pull request #1346 from frappe/pot_develop_2025-02-28
chore: update POT file
2025-03-03 10:13:13 +05:30
Md Hussain Nagaria
7ee7b95eb5 Merge pull request #1350 from frappe/tz-autocomplete
feat: timezone autocomplete in live class & misc fixes
2025-03-03 06:35:07 +05:30
Hussain Nagaria
83b8bdde45 fix: transform tabIndex query param to number 2025-03-03 06:22:19 +05:30
Hussain Nagaria
1b5dd15b90 feat(LiveClass): timezone autocomplete field 2025-03-03 06:19:06 +05:30
Hussain Nagaria
47c224fcad chore: remove unused imports 2025-03-03 06:01:02 +05:30
Jannat Patel
1c866f40eb chore: Persian translations 2025-03-01 02:00:07 +05:30
frappe-pr-bot
1861aabaca chore: update POT file 2025-02-28 16:04:26 +00:00
Jannat Patel
cd8fb6eb38 Merge pull request #1342 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-02-28 10:42:18 +05:30
Jannat Patel
21d05d3731 chore: Bosnian translations 2025-02-28 01:36:22 +05:30
Jannat Patel
7c953925f9 chore: Bosnian translations 2025-02-27 01:40:12 +05:30
Jannat Patel
33a4bbbe47 chore: Persian translations 2025-02-27 01:40:10 +05:30
Frappe PR Bot
dfb82570ea chore(release): Bumped to Version 2.24.0 2025-02-26 04:50:44 +00:00
Jannat Patel
e712d6ae42 Merge pull request #1334 from pateljannat/paid-certificate-on-courses
feat: paid certifications on courses
2025-02-25 14:47:07 +05:30
Jannat Patel
6ffc953370 test: removed course expiry from test 2025-02-25 14:33:53 +05:30
Jannat Patel
63bf6a5574 fix: polished the course certification flow 2025-02-25 12:46:35 +05:30
Jannat Patel
1e73fc5751 Merge pull request #1338 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-02-25 10:50:30 +05:30
Jannat Patel
65604a0b88 chore: Esperanto translations 2025-02-25 01:05:38 +05:30
Jannat Patel
5a1a39f5f5 chore: Bosnian translations 2025-02-25 01:05:36 +05:30
Jannat Patel
d22576c85c chore: Persian translations 2025-02-25 01:05:35 +05:30
Jannat Patel
b7e5332c38 chore: Chinese Simplified translations 2025-02-25 01:05:33 +05:30
Jannat Patel
ed8570fb88 chore: Turkish translations 2025-02-25 01:05:32 +05:30
Jannat Patel
ce69e6634d chore: Swedish translations 2025-02-25 01:05:31 +05:30
Jannat Patel
274db20c60 chore: Russian translations 2025-02-25 01:05:29 +05:30
Jannat Patel
3d72072f1f chore: Polish translations 2025-02-25 01:05:28 +05:30
Jannat Patel
ed156c09d7 chore: Hungarian translations 2025-02-25 01:05:27 +05:30
Jannat Patel
fda3a1a468 chore: German translations 2025-02-25 01:05:25 +05:30
Jannat Patel
c261387635 chore: Arabic translations 2025-02-25 01:05:24 +05:30
Jannat Patel
7a2fa4dae8 chore: Spanish translations 2025-02-25 01:05:22 +05:30
Jannat Patel
b0c41958d9 chore: French translations 2025-02-25 01:05:21 +05:30
Jannat Patel
4f1dcbfb78 feat: eval and certification flow with purchased certificate 2025-02-24 19:15:26 +05:30
Jannat Patel
dc9ed099d0 Merge pull request #1335 from frappe/pot_develop_2025-02-21
chore: update POT file
2025-02-24 10:38:54 +05:30
Md Hussain Nagaria
95255d44a9 feat(batch): track active tab in URL/route (#1337)
* chore: remove defineModel imports

* it is a compiler macro now, so no longer needs to be imported

* feat(batch): track active tab in URL/route

Fixes #1336

* style: lint
2025-02-22 22:30:14 +05:30
Hussain Nagaria
5a94e8df75 style: lint 2025-02-22 22:23:40 +05:30
Hussain Nagaria
015e3f8490 feat(batch): track active tab in URL/route
Fixes #1336
2025-02-22 22:21:26 +05:30
Hussain Nagaria
558601f02b chore: remove defineModel imports
* it is a compiler macro now, so no longer needs to be imported
2025-02-22 21:58:50 +05:30
frappe-pr-bot
461d96a079 chore: update POT file 2025-02-21 16:04:10 +00:00
Jannat Patel
bacfaf4a71 feat: paid certifications on courses 2025-02-21 19:12:20 +05:30
Jannat Patel
0678def698 Merge pull request #1330 from pateljannat/markdown-links
fix: link issue in lesson
2025-02-20 16:37:33 +05:30
Jannat Patel
07b0a0af51 test: fixed lesson content test 2025-02-20 16:31:09 +05:30
Jannat Patel
f12f6cb720 fix: link issue in lesson 2025-02-20 15:08:14 +05:30
Jannat Patel
4e6c1478f9 Merge pull request #1328 from pateljannat/reschedule-evals
feat: cancel evaluations
2025-02-20 10:34:08 +05:30
Jannat Patel
f9fd36f77e feat: cancel evaluations 2025-02-19 22:29:24 +05:30
Jannat Patel
db4c7424b3 Merge pull request #1327 from pateljannat/issues-79
fix: misc batch issues
2025-02-19 16:51:42 +05:30
Jannat Patel
9311043190 fix: misc batch issues 2025-02-19 16:40:11 +05:30
Jannat Patel
03915ccfbd fix: only system managers should login to FC 2025-02-19 15:39:13 +05:30
Jannat Patel
c6d59216fd fix: redirect to FC dashboard when login to FC 2025-02-19 15:35:34 +05:30
Frappe PR Bot
a8690e41e6 chore(release): Bumped to Version 2.23.0 2025-02-19 05:29:30 +00:00
Jannat Patel
cda42b9ec5 Merge pull request #1325 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-02-19 08:49:19 +05:30
Jannat Patel
21a75fdd6d chore: Bosnian translations 2025-02-18 23:05:44 +05:30
Jannat Patel
a90a1e9855 chore: Persian translations 2025-02-18 23:05:43 +05:30
Jannat Patel
2a046e2e8b chore: German translations 2025-02-18 23:05:40 +05:30
Jannat Patel
bb41656d81 Merge branch 'develop' of https://github.com/frappe/lms into develop 2025-02-18 19:12:23 +05:30
Jannat Patel
a88a107718 fix: batch confirmation email template 2025-02-18 19:12:04 +05:30
Jannat Patel
2d21469f91 Merge pull request #1324 from pateljannat/issues-78
fix: redirect users to the batch page after login
2025-02-18 18:29:51 +05:30
Jannat Patel
960ebe4a79 fix: redirect users to the batch page after login 2025-02-18 18:10:33 +05:30
Jannat Patel
46dba0c394 Merge pull request #1323 from pateljannat/batch-reminders
feat: batch start and live class reminder
2025-02-18 17:34:44 +05:30
Jannat Patel
ba27e8ca95 fix: send live class reminder on the day of the class 2025-02-18 17:26:40 +05:30
Jannat Patel
30574ea0fd feat: batch start and live class reminder 2025-02-18 17:22:52 +05:30
Jannat Patel
c3c985c4a1 Merge pull request #1322 from pateljannat/certification-batches
feat: filter certification batches
2025-02-18 17:05:16 +05:30
Jannat Patel
7b3d2d8812 feat: filter certification batches 2025-02-18 15:57:55 +05:30
Jannat Patel
d573a9f008 Merge pull request #1320 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-02-18 12:41:05 +05:30
Jannat Patel
85a05f56b2 chore: Esperanto translations 2025-02-17 22:33:51 +05:30
Jannat Patel
904adfb905 chore: Bosnian translations 2025-02-17 22:33:50 +05:30
Jannat Patel
b2201c29fd chore: Persian translations 2025-02-17 22:33:48 +05:30
Jannat Patel
fe01f68623 chore: Chinese Simplified translations 2025-02-17 22:33:47 +05:30
Jannat Patel
531c8ebe94 chore: Turkish translations 2025-02-17 22:33:45 +05:30
Jannat Patel
52dfb5a360 chore: Swedish translations 2025-02-17 22:33:44 +05:30
Jannat Patel
7e04e7e461 chore: Russian translations 2025-02-17 22:33:42 +05:30
Jannat Patel
bce47f606d chore: Polish translations 2025-02-17 22:33:41 +05:30
Jannat Patel
4dc1fdfdd8 chore: Hungarian translations 2025-02-17 22:33:40 +05:30
Jannat Patel
9a852b52bc chore: German translations 2025-02-17 22:33:38 +05:30
Jannat Patel
71a57b1fc0 chore: Arabic translations 2025-02-17 22:33:37 +05:30
Jannat Patel
d634598db1 chore: Spanish translations 2025-02-17 22:33:35 +05:30
Jannat Patel
6377d682a4 chore: French translations 2025-02-17 22:33:33 +05:30
Jannat Patel
6e1acfdc24 Merge pull request #1316 from FahidLatheef/fix/quiz-maximum-attempts
fix: fixed bug in which user can submit quiz over the maximum limit allowed
2025-02-17 19:59:57 +05:30
Jannat Patel
30ec1dfd7c Merge pull request #1319 from pateljannat/assignment-grading-comment-field
feat: assignment comments is now text editor
2025-02-17 19:56:22 +05:30
Jannat Patel
3d209024dd fix: height of batch page 2025-02-17 19:45:45 +05:30
Jannat Patel
9ce64a037d fix: increased column width for grading 2025-02-17 19:41:24 +05:30
Jannat Patel
43117bc035 feat:assignment comments is now text editor 2025-02-17 19:28:50 +05:30
Jannat Patel
2af704043e Merge pull request #1318 from pateljannat/batch-email-template
feat: batch specific email templates
2025-02-17 18:36:05 +05:30
Jannat Patel
fa14ffdcba feat: batch specific email templates 2025-02-17 18:17:50 +05:30
Jannat Patel
492b715ea0 Merge pull request #1317 from pateljannat/trial-signup
feat: billing banner for FC trial sites
2025-02-17 16:00:46 +05:30
Jannat Patel
d452e20b8a feat: show trial banner only if fc site 2025-02-17 15:39:15 +05:30
Jannat Patel
6b634c15d9 feat: billing banner for FC trial sites 2025-02-17 15:07:31 +05:30
Jannat Patel
eeaec3369f Merge pull request #1313 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-02-17 11:10:39 +05:30
Jannat Patel
ce1eece90d Merge pull request #1312 from frappe/pot_develop_2025-02-14
chore: update POT file
2025-02-17 11:05:48 +05:30
Jannat Patel
030bff6592 chore: Bosnian translations 2025-02-16 22:32:22 +05:30
Jannat Patel
65de46a59e chore: Swedish translations 2025-02-16 22:32:18 +05:30
Fahid Latheef Alungal
974f67aefe fix: validate if submission exceeds the allowed limit in backend 2025-02-16 19:29:03 +05:30
Fahid Latheef Alungal
e374ae3229 fix: fixed spelling nextQuetion -> nextQuestion 2025-02-16 18:28:48 +05:30
Fahid Latheef Alungal
8b1058e577 fix: fixed issue in which submissions are not reflected gracefully until page reload
ListView throws error if initialized without emptyState which was causing the component to not reload when number of submissions was 0.
2025-02-16 18:27:38 +05:30
Fahid Latheef Alungal
aaa2eea5e6 fix: fixed incomplete router initialization in Quiz.vue which was allowing user to submit quiz multiple times 2025-02-16 18:19:14 +05:30
Fahid Latheef Alungal
54047e3c2c fix: fix spelling typo Maximun Attempts -> Maximum Attempts 2025-02-16 16:10:14 +05:30
Fahid Latheef Alungal
50fe94e47b fix: fix yarn dev not working due to const variable re-assignment
It was causing this error

  ✘ [ERROR] Cannot assign to "isLoggedIn" because it is a constant

    src/router.js:230:2:
      230 │     isLoggedIn = false
          ╵     ~~~~~~~~~~

  The symbol "isLoggedIn" was declared a constant here:

    src/router.js:222:7:
      222 │   const { isLoggedIn } = sessionStore()
          ╵         ^
2025-02-16 16:08:35 +05:30
Jannat Patel
6999f6641a chore: Bosnian translations 2025-02-15 22:29:52 +05:30
frappe-pr-bot
c2b12aa65f chore: update POT file 2025-02-14 16:04:13 +00:00
Jannat Patel
1a731b6908 Merge pull request #1311 from pateljannat/issues-77
fix: students should have access private batch if enrolled
2025-02-14 20:21:54 +05:30
Jannat Patel
837d050628 fix: students should be able to access private batch if they are enrolled 2025-02-14 20:10:32 +05:30
Jannat Patel
8b00bec49c fix: students should be able to access private batch if they are enrolled 2025-02-14 20:04:37 +05:30
Jannat Patel
9ade643af0 Merge pull request #1310 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-02-14 17:06:53 +05:30
Jannat Patel
a29b92a886 chore: Bosnian translations 2025-02-13 22:17:06 +05:30
Jannat Patel
e2c28e211f Merge pull request #1309 from pateljannat/issues-76
fix: misc batch issues
2025-02-13 21:26:17 +05:30
Jannat Patel
65f5b6a0a4 fix: delete unused custom fields from web form 2025-02-13 17:23:57 +05:30
Frappe PR Bot
905e240fb9 chore(release): Bumped to Version 2.22.0 2025-02-13 11:52:52 +00:00
Jannat Patel
75cea1ab78 fix: delete unused custom fields from web form 2025-02-13 17:21:14 +05:30
Jannat Patel
dd3da3dd49 Merge pull request #1305 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-02-13 17:06:23 +05:30
Jannat Patel
5ab9131629 fix: misc batch issues 2025-02-13 16:57:21 +05:30
Jannat Patel
8f1c9612b7 fix: payment reminder template issue 2025-02-13 08:57:13 +05:30
Jannat Patel
15a12d2518 Merge pull request #1306 from pateljannat/issues-75
fix: misc ui fixes
2025-02-13 07:36:38 +05:30
Jannat Patel
e83734e0e4 fix: misc ui fixes 2025-02-12 22:54:23 +05:30
Jannat Patel
f2a95af45c chore: Bosnian translations 2025-02-12 22:14:17 +05:30
Jannat Patel
1bb61d0c1d chore: Persian translations 2025-02-12 22:14:15 +05:30
Jannat Patel
51fb4f2296 chore: Swedish translations 2025-02-12 22:14:12 +05:30
Jannat Patel
5f0f625c0f chore: Spanish translations 2025-02-12 22:14:05 +05:30
Jannat Patel
ea7b803905 fix: set email sent in batch enrollment 2025-02-12 13:34:28 +05:30
Jannat Patel
76af3921dd fix: set email sent in batch enrollment 2025-02-12 13:34:00 +05:30
Jannat Patel
e2f999fc31 Merge pull request #1304 from pateljannat/issues-74
fix: misc batch flow changes
2025-02-12 13:15:14 +05:30
Jannat Patel
f63d57c4a9 fix: when checking for duplicates, ignore same document 2025-02-12 13:09:22 +05:30
Jannat Patel
ee73790127 fix: announcements should to go one student at a time 2025-02-12 12:46:57 +05:30
Jannat Patel
1c3e84e9bb fix: misc batch flow changes 2025-02-12 12:24:31 +05:30
Jannat Patel
451a151ce0 Merge pull request #1302 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-02-12 11:02:49 +05:30
Jannat Patel
1ac4f819ec chore: Bosnian translations 2025-02-11 22:17:23 +05:30
Jannat Patel
526eba0129 chore: Persian translations 2025-02-11 22:17:21 +05:30
Jannat Patel
8638e0a1f9 Merge pull request #1301 from pateljannat/dark-mode
feat: dark mode
2025-02-11 18:40:08 +05:30
Jannat Patel
69c1093c93 fix: badge color on batch overlay 2025-02-11 18:30:13 +05:30
Jannat Patel
74cd0a4d40 fix: added get certified button on certified participants list 2025-02-11 18:29:29 +05:30
Jannat Patel
e28fc3bee6 fix: calendar background and assignment text color 2025-02-11 16:28:25 +05:30
Jannat Patel
879dfac111 Merge pull request #1300 from pateljannat/issues-73
fix: misc batch enrollment issues
2025-02-11 15:02:14 +05:30
Jannat Patel
b6cfcd797b fix: pluck only member for to validate batch course membership 2025-02-11 14:55:44 +05:30
Jannat Patel
2ea73888f0 fix: changed naming for LMS Payment 2025-02-11 14:49:04 +05:30
Jannat Patel
f43331967c fix: misc batch enrollment issues 2025-02-11 13:28:49 +05:30
Jannat Patel
9da1249e51 Merge pull request #1298 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-02-11 12:40:58 +05:30
Jannat Patel
2342dfe452 fix: dark mode for course and batch tags 2025-02-11 10:52:37 +05:30
Jannat Patel
e24d22c348 chore: Esperanto translations 2025-02-10 21:39:15 +05:30
Jannat Patel
533d9545de chore: Bosnian translations 2025-02-10 21:39:14 +05:30
Jannat Patel
03c0c3c821 chore: Persian translations 2025-02-10 21:39:12 +05:30
Jannat Patel
05be628afb chore: Chinese Simplified translations 2025-02-10 21:39:10 +05:30
Jannat Patel
cb2dc3e645 chore: Turkish translations 2025-02-10 21:39:09 +05:30
Jannat Patel
25f3d2fb9f chore: Swedish translations 2025-02-10 21:39:07 +05:30
Jannat Patel
db39a6416c chore: Russian translations 2025-02-10 21:39:05 +05:30
Jannat Patel
48e0787344 chore: Polish translations 2025-02-10 21:39:04 +05:30
Jannat Patel
838de2f692 chore: Hungarian translations 2025-02-10 21:39:02 +05:30
Jannat Patel
1953d89e3c chore: German translations 2025-02-10 21:39:00 +05:30
Jannat Patel
d0898d4c75 chore: Arabic translations 2025-02-10 21:38:59 +05:30
Jannat Patel
f01bb1aecb chore: Spanish translations 2025-02-10 21:38:57 +05:30
Jannat Patel
bbdbda4942 chore: French translations 2025-02-10 21:38:56 +05:30
Jannat Patel
7741696011 Merge branch 'develop' of https://github.com/frappe/lms into dark-mode 2025-02-10 19:18:26 +05:30
Jannat Patel
2d4567bfbd Merge pull request #1297 from pateljannat/issues-72
fix: check for duplicates before creating batch enrollment in patch
2025-02-10 18:03:31 +05:30
Jannat Patel
8f643dae27 style: fix formatting 2025-02-10 17:37:08 +05:30
Jannat Patel
81e287ffe5 fix: check for duplicates before creating batch enrollment in patch 2025-02-10 17:31:11 +05:30
Jannat Patel
5543aa5e02 chore: resolved conflicts 2025-02-10 16:48:02 +05:30
Jannat Patel
b5a7b4cd2c Merge pull request #1296 from pateljannat/batch-students-refactor
refactor: LMS Batch Enrollment to store batch students
2025-02-10 16:42:09 +05:30
Jannat Patel
8857ce8146 fix: set confirmation_email_sent after sending email 2025-02-10 16:30:31 +05:30
Jannat Patel
bfbc5f600f Merge pull request #1295 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-02-10 16:16:57 +05:30
Jannat Patel
a8fa42db00 Merge pull request #1294 from frappe/pot_develop_2025-02-07
chore: update POT file
2025-02-10 16:16:44 +05:30
Jannat Patel
4ee2bfcf32 style: formattin 2025-02-10 16:16:11 +05:30
Jannat Patel
ab98884f77 refactor: replaced Batch Student child table with LMS Batch Enrollment doctype 2025-02-10 16:15:28 +05:30
Jannat Patel
dbf443300b feat: dark mode 2025-02-10 10:51:21 +05:30
Jannat Patel
dbf44a7a85 chore: Bosnian translations 2025-02-09 21:42:25 +05:30
Jannat Patel
2818c95795 chore: Swedish translations 2025-02-09 21:42:20 +05:30
Jannat Patel
27a13a6151 chore: Bosnian translations 2025-02-08 21:41:18 +05:30
Jannat Patel
9f974786f2 chore: Persian translations 2025-02-08 21:41:17 +05:30
Jannat Patel
2f2f41ac3c chore: Bosnian translations 2025-02-07 21:41:18 +05:30
frappe-pr-bot
d5d30f683a chore: update POT file 2025-02-07 16:03:56 +00:00
Jannat Patel
56007aa4ba Merge pull request #1293 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-02-07 14:47:50 +05:30
Jannat Patel
d489e08718 chore: Bosnian translations 2025-02-06 21:26:13 +05:30
Jannat Patel
16b9356944 Merge pull request #1292 from pateljannat/hide-public-pages
feat: configuration to allow guest access
2025-02-06 12:41:45 +05:30
Jannat Patel
ba26826896 feat: configuration to allow guest access 2025-02-06 12:14:24 +05:30
Jannat Patel
49631b6e56 Merge pull request #1291 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-02-06 11:11:23 +05:30
Jannat Patel
ae2bffc56d chore: Bosnian translations 2025-02-05 21:27:59 +05:30
Jannat Patel
47e51c4787 chore: Swedish translations 2025-02-05 21:27:55 +05:30
Jannat Patel
06ef289427 Merge pull request #1286 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-02-05 14:49:32 +05:30
Jannat Patel
4190f39993 chore: Bosnian translations 2025-02-04 20:35:56 +05:30
Jannat Patel
26a22375c8 chore: Persian translations 2025-02-04 20:35:54 +05:30
Jannat Patel
0c174caf86 Merge pull request #1285 from pateljannat/jobs-order
fix: misc issues
2025-02-04 12:14:38 +05:30
Jannat Patel
661748adc1 style: improved formatting 2025-02-04 11:27:21 +05:30
Jannat Patel
73f24339e3 Merge pull request #1284 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-02-04 11:19:37 +05:30
Jannat Patel
9775d7425c fix: changed heatmap base days back to 200 2025-02-04 11:19:13 +05:30
Jannat Patel
3ff6c96273 fix: top aligned all batch feedback rows 2025-02-04 11:15:39 +05:30
Jannat Patel
f9706f10e1 feat: reminder notification for incomplete payments 2025-02-04 09:55:10 +05:30
Jannat Patel
e9a20c61d5 chore: Bosnian translations 2025-02-03 20:01:03 +05:30
Jannat Patel
f3ee1a84dd chore: Persian translations 2025-02-03 20:01:01 +05:30
Jannat Patel
381ca43c01 Merge pull request #1269 from FahidLatheef/feat/persistent-sidebar
ui: added persistent Sidebar Collapsibility and Sidebar Webpages
2025-02-03 11:16:57 +05:30
Jannat Patel
8cc16dc51b Merge pull request #1279 from frappe/pot_develop_2025-01-31
chore: update POT file
2025-02-03 11:14:46 +05:30
Jannat Patel
4337603e33 Merge pull request #1280 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-02-03 11:14:35 +05:30
Jannat Patel
5c39acb745 chore: Bosnian translations 2025-02-02 19:51:26 +05:30
Jannat Patel
1b584f0b88 chore: Bosnian translations 2025-02-01 19:30:44 +05:30
frappe-pr-bot
68a28ef6d4 chore: update POT file 2025-01-31 16:04:05 +00:00
Jannat Patel
867df7f2c7 Merge pull request #1278 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-01-31 20:32:09 +05:30
Jannat Patel
c18e84bb8e chore: Bosnian translations 2025-01-31 18:45:28 +05:30
Jannat Patel
3fc1fd9dbc Merge pull request #1272 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-01-29 19:30:00 +05:30
Jannat Patel
bc284c327c chore: Esperanto translations 2025-01-28 17:47:24 +05:30
Jannat Patel
85961c76fb chore: Bosnian translations 2025-01-28 17:47:23 +05:30
Jannat Patel
1c11a5964b chore: Persian translations 2025-01-28 17:47:21 +05:30
Jannat Patel
4d1ba4ea3f chore: Chinese Simplified translations 2025-01-28 17:47:20 +05:30
Jannat Patel
6d3e24fce9 chore: Turkish translations 2025-01-28 17:47:18 +05:30
Jannat Patel
de37ec5704 chore: Swedish translations 2025-01-28 17:47:17 +05:30
Jannat Patel
745592432c chore: Russian translations 2025-01-28 17:47:16 +05:30
Jannat Patel
cf47965e8c chore: Polish translations 2025-01-28 17:47:14 +05:30
Jannat Patel
3d64872352 chore: Hungarian translations 2025-01-28 17:47:13 +05:30
Jannat Patel
b89ad4204c chore: German translations 2025-01-28 17:47:12 +05:30
Jannat Patel
71e9ba849d chore: Arabic translations 2025-01-28 17:47:10 +05:30
Jannat Patel
1d412175c6 chore: Spanish translations 2025-01-28 17:47:09 +05:30
Jannat Patel
b282a37a04 chore: French translations 2025-01-28 17:47:07 +05:30
Jannat Patel
5f6d0bcf25 Merge pull request #1270 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-01-27 21:05:52 +05:30
Jannat Patel
74c2d5eb06 Merge pull request #1268 from frappe/pot_develop_2025-01-24
chore: update POT file
2025-01-27 21:05:36 +05:30
Jannat Patel
4618d3b30e chore: French translations 2025-01-27 17:22:08 +05:30
Jannat Patel
9e32e8f499 chore: Persian translations 2025-01-25 17:22:38 +05:30
Jannat Patel
f47e2e758b chore: Turkish translations 2025-01-25 17:22:37 +05:30
Fahid Latheef Alungal
9e03e30bd8 ui: added persistent Sidebar Collapsibility and Sidebar Webpages 2025-01-25 11:20:09 +05:30
frappe-pr-bot
6be0e6bfca chore: update POT file 2025-01-24 16:04:10 +00:00
Jannat Patel
7bbdedf5f4 fix: margins beneath progress bar in course card 2025-01-23 18:34:15 +05:30
Jannat Patel
e942e6a2f5 Merge pull request #1267 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-01-23 10:34:41 +05:30
Jannat Patel
6162df7013 chore: Turkish translations 2025-01-22 16:48:09 +05:30
Jannat Patel
a28227ad75 Merge pull request #1265 from pateljannat/issues-70
fix: jobs page rendering issue for guest users
2025-01-22 12:54:27 +05:30
Frappe PR Bot
ed8baf3327 chore(release): Bumped to Version 2.21.0 2025-01-22 07:09:58 +00:00
Jannat Patel
1ac5de96f9 fix: jobs page rendering issue for guest users 2025-01-22 12:38:28 +05:30
Jannat Patel
15dd4c4350 Merge pull request #1261 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-01-22 10:53:00 +05:30
Jannat Patel
c986089e77 chore: Persian translations 2025-01-21 16:41:39 +05:30
Jannat Patel
17dc77f061 chore: Turkish translations 2025-01-21 16:41:38 +05:30
Jannat Patel
189f353de0 chore: Esperanto translations 2025-01-20 16:43:19 +05:30
Jannat Patel
845e7174f0 chore: Bosnian translations 2025-01-20 16:43:18 +05:30
Jannat Patel
8c6e4ad3ee chore: Persian translations 2025-01-20 16:43:17 +05:30
Jannat Patel
5dfddc890c chore: Chinese Simplified translations 2025-01-20 16:43:15 +05:30
Jannat Patel
1ebabc23d3 chore: Turkish translations 2025-01-20 16:43:14 +05:30
Jannat Patel
1bf8c1c763 chore: Swedish translations 2025-01-20 16:43:12 +05:30
Jannat Patel
c5a59b6370 chore: Russian translations 2025-01-20 16:43:11 +05:30
Jannat Patel
4a5a777478 chore: Polish translations 2025-01-20 16:43:09 +05:30
Jannat Patel
4fd7dcd5b2 chore: Hungarian translations 2025-01-20 16:43:08 +05:30
Jannat Patel
55920d9e3f chore: German translations 2025-01-20 16:43:07 +05:30
Jannat Patel
6d0c3c9cd8 chore: Arabic translations 2025-01-20 16:43:05 +05:30
Jannat Patel
7b20c3fe03 chore: Spanish translations 2025-01-20 16:43:04 +05:30
Jannat Patel
efbe35c836 chore: French translations 2025-01-20 16:43:02 +05:30
Jannat Patel
e591cd74ab Merge pull request #1260 from frappe/pot_develop_2025-01-17
chore: update POT file
2025-01-20 09:57:38 +05:30
Jannat Patel
669b9c73be Merge pull request #1257 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-01-20 09:57:26 +05:30
frappe-pr-bot
52e1dd6d33 chore: update POT file 2025-01-17 16:04:20 +00:00
Jannat Patel
828e195b81 Merge pull request #1259 from pateljannat/issues-69
perf: misc performance improvements
2025-01-17 17:57:28 +05:30
Jannat Patel
145342bb72 perf: misc performance improvements 2025-01-17 17:17:02 +05:30
Jannat Patel
58abfd004d Merge pull request #1256 from pateljannat/issues-68
fix: changed the naming for certificate and job opportunity
2025-01-17 14:30:56 +05:30
Jannat Patel
9dc8322270 fix: don't check assignment submission status if doc is new 2025-01-17 14:24:14 +05:30
Jannat Patel
4f0a6a7d57 chore: removed print statement 2025-01-17 14:18:07 +05:30
Jannat Patel
2fb8ae00b9 chore: Chinese Simplified translations 2025-01-17 14:11:26 +05:30
Jannat Patel
63da1e384d fix: changed the naming for certificate and job opportunity 2025-01-17 13:00:35 +05:30
Jannat Patel
34685ebdb2 Merge pull request #1255 from pateljannat/batch-url
refactor: changed batch naming to be a slug of the title
2025-01-17 10:54:31 +05:30
Jannat Patel
215ae941e1 Merge pull request #1254 from pateljannat/jobs-page-responsive
fix: improved jobs page ui
2025-01-17 10:36:48 +05:30
Jannat Patel
9d1211e872 fix: changed batch naming to be a slug of the title 2025-01-17 10:35:26 +05:30
Jannat Patel
cd4f2b1039 fix: clarified the posting date 2025-01-17 10:21:49 +05:30
Jannat Patel
9881b7b498 fix: improved jobs page ui 2025-01-17 10:15:58 +05:30
Jannat Patel
28a687f6bf Merge pull request #1252 from pateljannat/refactor-certified-participants-page
refactor: improved ui and performance for certified participants page
2025-01-16 17:01:28 +05:30
Jannat Patel
bd43ed0e88 fix: responsive design for certified participants page 2025-01-16 16:49:03 +05:30
Jannat Patel
17b59ce4e5 refactor: improved ui and performance for certified participants page 2025-01-16 16:39:48 +05:30
Jannat Patel
7acc1864c8 Merge pull request #1251 from pateljannat/issues-67
fix: misc issues
2025-01-16 13:07:23 +05:30
Jannat Patel
5a6fdfcbc3 fix: simplfied logic to filter current day batches 2025-01-16 12:57:13 +05:30
Jannat Patel
23d465d4a1 fix: batch enrolled filter logic 2025-01-16 12:52:17 +05:30
Jannat Patel
27ae014fcb fix: course is no longer mandatory to generate a certificate 2025-01-16 12:35:13 +05:30
Jannat Patel
b4c7338b76 fix: batch listing for current day batches 2025-01-16 11:43:56 +05:30
Jannat Patel
0d1464c5e9 Merge pull request #1249 from pateljannat/batches-responsive
fix: batch list responsive cards
2025-01-15 16:30:03 +05:30
Jannat Patel
f4421d362c fix: batch list responsive cards 2025-01-15 16:15:04 +05:30
Jannat Patel
5c8378f2d4 fix: changed sorting order of batch list 2025-01-15 12:31:23 +05:30
Jannat Patel
8401e86acb feat: batch tabs for moderators 2025-01-15 11:17:07 +05:30
Frappe PR Bot
e16101813c chore(release): Bumped to Version 2.20.0 2025-01-15 05:39:34 +00:00
Jannat Patel
bbd3ac6451 Merge pull request #1246 from pateljannat/batch-refactor
refactor: improved performance and ui batch list
2025-01-15 10:57:11 +05:30
Jannat Patel
c6a26e5260 fix: amount rounding issue 2025-01-14 18:45:57 +05:30
Jannat Patel
a87fda6b84 Merge pull request #1245 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-01-14 17:48:38 +05:30
Jannat Patel
b42c635cdb refactor: improved performance and ui batch list 2025-01-14 17:41:46 +05:30
Jannat Patel
a9c6b71e19 chore: Persian translations 2025-01-14 12:05:13 +05:30
Jannat Patel
282441e0e7 chore: Swedish translations 2025-01-14 12:05:10 +05:30
Jannat Patel
6020d5f5c2 Merge pull request #1244 from pateljannat/issues-65
fix: removed delivery parameter from batch feedback
2025-01-14 11:59:53 +05:30
Jannat Patel
9a395cbda0 Merge pull request #1243 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-01-14 11:29:40 +05:30
Jannat Patel
61e41180dd fix: removed delivery parameter from batch feedback 2025-01-14 11:29:13 +05:30
Jannat Patel
26bde996ac chore: Esperanto translations 2025-01-13 12:01:25 +05:30
Jannat Patel
6f78ac06c2 chore: Bosnian translations 2025-01-13 12:01:24 +05:30
Jannat Patel
8e498f4fbe chore: Persian translations 2025-01-13 12:01:22 +05:30
Jannat Patel
8105e606c9 chore: Chinese Simplified translations 2025-01-13 12:01:21 +05:30
Jannat Patel
7df6e5fe64 chore: Turkish translations 2025-01-13 12:01:19 +05:30
Jannat Patel
909c9b446b chore: Swedish translations 2025-01-13 12:01:18 +05:30
Jannat Patel
29639d59c3 chore: Russian translations 2025-01-13 12:01:16 +05:30
Jannat Patel
a13dac6dd4 chore: Polish translations 2025-01-13 12:01:15 +05:30
Jannat Patel
31257e588f chore: Hungarian translations 2025-01-13 12:01:13 +05:30
Jannat Patel
52ab419040 chore: German translations 2025-01-13 12:01:12 +05:30
Jannat Patel
7dbc35977f chore: Arabic translations 2025-01-13 12:01:10 +05:30
Jannat Patel
ce9aafadd9 chore: Spanish translations 2025-01-13 12:01:08 +05:30
Jannat Patel
13da79488f chore: French translations 2025-01-13 12:01:07 +05:30
Jannat Patel
2c999e2037 Merge pull request #1242 from frappe/pot_develop_2025-01-10
chore: update POT file
2025-01-13 11:47:52 +05:30
Jannat Patel
c096c176e3 Merge pull request #1241 from pateljannat/batch-feedback
feat: batch feedback
2025-01-13 11:47:00 +05:30
Jannat Patel
8fe0b62bb3 feat: batch feedback for moderators 2025-01-13 11:31:18 +05:30
frappe-pr-bot
e3b53efd2c chore: update POT file 2025-01-10 16:04:43 +00:00
Jannat Patel
2ecb93e925 feat: show submitted feedback as readonly 2025-01-10 19:05:59 +05:30
Jannat Patel
5d14d6f1aa chore: merged conflicts 2025-01-10 11:03:44 +05:30
Jannat Patel
4869bba7bb Merge pull request #1239 from pateljannat/issues-64
fix: made course list responsive for bigger screen sizes
2025-01-09 18:53:00 +05:30
Jannat Patel
ecc12d783a fix: list and table formatting in lesson 2025-01-09 17:07:57 +05:30
Jannat Patel
54b7f811f7 fix: made course list responsive for bigger screen sizes 2025-01-09 12:24:21 +05:30
Frappe PR Bot
bb6e97992b chore(release): Bumped to Version 2.19.0 2025-01-08 14:21:07 +00:00
Jannat Patel
64fac451f3 Merge pull request #1236 from FahidLatheef/fix/days_diff_function_name
fix: fixed typo in spelling in frappe.utils.date_diff import
2025-01-08 12:43:10 +05:30
Jannat Patel
e45b33a809 feat: batch feedback 2025-01-08 11:22:07 +05:30
Jannat Patel
eb6b72515e Merge pull request #1235 from FahidLatheef/fix/assignment-popup-on-edit-quiz
fix: fix issue where assignment form is popped up on add quiz button in Lesson Edit form
2025-01-08 10:45:42 +05:30
Fahid Latheef A
0550d3aea3 fix: fixed typo in spelling in frappe.utils.date_diff import 2025-01-07 21:07:23 +05:30
Fahid Latheef A
f6577acbff refactor: fixed linting issue 2025-01-07 20:41:54 +05:30
Fahid Latheef A
09c494f38a Added quiz type prop for AssessmentPlugin component 2025-01-07 20:29:07 +05:30
Fahid Latheef A
6c600d747e Added assignement type prop for AssessmentPlugin component 2025-01-07 20:27:39 +05:30
Jannat Patel
9dcfc347d9 Merge pull request #1234 from pateljannat/batch-dashboard-23
feat: student activities display in a heatmap
2025-01-07 19:31:20 +05:30
Jannat Patel
fb40b627fc feat: show student progress heatmap on moderators dashboard 2025-01-07 18:15:59 +05:30
Jannat Patel
c597f96375 Merge pull request #1233 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-01-07 10:27:39 +05:30
Jannat Patel
f1961ab614 chore: Esperanto translations 2025-01-07 09:39:40 +05:30
Jannat Patel
c2c7b7b250 chore: Bosnian translations 2025-01-07 09:39:38 +05:30
Jannat Patel
c20c272f8e chore: Persian translations 2025-01-07 09:39:37 +05:30
Jannat Patel
85e4115306 chore: Chinese Simplified translations 2025-01-07 09:39:36 +05:30
Jannat Patel
10c2bc589a chore: Turkish translations 2025-01-07 09:39:34 +05:30
Jannat Patel
a30244cb4a chore: Swedish translations 2025-01-07 09:39:33 +05:30
Jannat Patel
5691fcdca4 chore: Russian translations 2025-01-07 09:39:31 +05:30
Jannat Patel
f5848207e2 chore: Polish translations 2025-01-07 09:39:30 +05:30
Jannat Patel
ad224161d8 chore: Hungarian translations 2025-01-07 09:39:28 +05:30
Jannat Patel
5837a1ffab chore: German translations 2025-01-07 09:39:27 +05:30
Jannat Patel
1cfd7cdb98 chore: Arabic translations 2025-01-07 09:39:25 +05:30
Jannat Patel
56a4aa2a3f chore: Spanish translations 2025-01-07 09:39:24 +05:30
Jannat Patel
d91d2ded77 chore: French translations 2025-01-07 09:39:22 +05:30
Jannat Patel
6a48d44b14 Merge pull request #1232 from pateljannat/issues-63
fix: misc issues
2025-01-06 16:25:21 +05:30
Jannat Patel
31c5d423d0 fix: misc issues 2025-01-06 16:00:48 +05:30
Jannat Patel
79177b5f5b feat: students heatmap 2025-01-06 15:42:44 +05:30
Jannat Patel
74658b2054 Merge pull request #1231 from pateljannat/refactor-batch-list
refactor: fetch minimal information for batch cards
2025-01-06 12:44:28 +05:30
Jannat Patel
052fffccef refactor: badge page data 2025-01-06 12:36:44 +05:30
Jannat Patel
bd2b558154 refactor: fetch minimal information for batch cards 2025-01-06 12:05:29 +05:30
Jannat Patel
65ee6b62ea Merge pull request #1230 from pateljannat/issues-62
refactor: duration field in quiz should be in minutes
2025-01-06 11:24:13 +05:30
Jannat Patel
26266a22e8 fix: add description to indicate that duration should be in minutes 2025-01-06 11:02:46 +05:30
Jannat Patel
e52ca63075 refactor: duration field in quiz should be in minutes 2025-01-06 11:01:01 +05:30
Jannat Patel
4d8b2eb5b4 Merge pull request #1229 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-01-06 10:41:42 +05:30
Jannat Patel
2d81a1ce31 Merge pull request #1227 from frappe/pot_develop_2025-01-03
chore: update POT file
2025-01-06 10:41:30 +05:30
Jannat Patel
052a85fbc0 chore: Swedish translations 2025-01-06 09:43:51 +05:30
frappe-pr-bot
fa0e84c671 chore: update POT file 2025-01-03 16:04:23 +00:00
Jannat Patel
4759736571 Merge pull request #1226 from pateljannat/issues-61
fix: misc batch issues
2025-01-03 17:57:42 +05:30
Jannat Patel
f77686feaa fix: misc batch issues 2025-01-03 17:37:22 +05:30
252 changed files with 21437 additions and 11212 deletions

View File

@@ -39,7 +39,7 @@ jobs:
node-version: '18' node-version: '18'
check-latest: true check-latest: true
- name: setup cache for bench - name: setup cache for bench
uses: actions/cache@v2 uses: actions/cache@v4
with: with:
path: ~/bench-cache path: ~/bench-cache
key: ${{ runner.os }} key: ${{ runner.os }}

View File

@@ -58,7 +58,7 @@ jobs:
echo "127.0.0.1 lms.test" | sudo tee -a /etc/hosts echo "127.0.0.1 lms.test" | sudo tee -a /etc/hosts
- name: Cache pip - name: Cache pip
uses: actions/cache@v3 uses: actions/cache@v4
with: with:
path: ~/.cache/pip path: ~/.cache/pip
key: ${{ runner.os }}-pip-${{ hashFiles('**/*requirements.txt', '**/pyproject.toml', '**/setup.py') }} key: ${{ runner.os }}-pip-${{ hashFiles('**/*requirements.txt', '**/pyproject.toml', '**/setup.py') }}

View File

@@ -13,6 +13,6 @@ module.exports = defineConfig({
openMode: 0, openMode: 0,
}, },
e2e: { e2e: {
baseUrl: "http://lms1:8000", baseUrl: "http://testui:8000",
}, },
}); });

View File

@@ -5,7 +5,7 @@ describe("Course Creation", () => {
cy.visit("/lms/courses"); cy.visit("/lms/courses");
// Create a course // Create a course
cy.get("header").children().last().children().last().click(); cy.get("button").contains("New").click();
cy.wait(1000); cy.wait(1000);
cy.url().should("include", "/courses/new/edit"); cy.url().should("include", "/courses/new/edit");
@@ -84,9 +84,8 @@ describe("Course Creation", () => {
cy.wait(1000); cy.wait(1000);
cy.get("label").contains("Title").type("Test Lesson"); cy.get("label").contains("Title").type("Test Lesson");
cy.get("#content .ce-block").type( cy.get("#content .ce-block").type(
"This is an extremely big paragraph that is meant to test the UI. This is a very long paragraph. It contains more than once sentence. Its meant to be this long as this is a UI test. Its unbearably long and I'm not sure why I'm typing this much. I'm just going to keep typing until I feel like its long enough. I think its long enough now. I'm going to stop typing now." "{enter}This is an extremely big paragraph that is meant to test the UI. This is a very long paragraph. It contains more than once sentence. Its meant to be this long as this is a UI test. Its unbearably long and I'm not sure why I'm typing this much. I'm just going to keep typing until I feel like its long enough. I think its long enough now. I'm going to stop typing now."
); );
cy.button("Save").click(); cy.button("Save").click();

View File

@@ -37,6 +37,7 @@ Cypress.Commands.add("login", (email, password) => {
url: "/api/method/login", url: "/api/method/login",
method: "POST", method: "POST",
body: { usr: email, pwd: password }, body: { usr: email, pwd: password },
timeout: 60000,
}); });
}); });

View File

@@ -2,7 +2,7 @@
<html lang="en"> <html lang="en">
<head> <head>
<meta charset="UTF-8" /> <meta charset="UTF-8" />
<link rel="icon" href="/favicon.png" /> <link rel="icon" href="{{ favicon or '/assets/lms/frontend/favicon.png' }}" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" /> <meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Frappe Learning</title> <title>Frappe Learning</title>
<meta name="title" content="{{ meta.title }}" /> <meta name="title" content="{{ meta.title }}" />
@@ -42,6 +42,7 @@
<script> <script>
window.csrf_token = '{{ csrf_token }}' window.csrf_token = '{{ csrf_token }}'
window.setup_complete = '{{ setup_complete }}'
document.getElementById('seo-content').style.display = 'none'; document.getElementById('seo-content').style.display = 'none';
</script> </script>
<script type="module" src="/src/main.js"></script> <script type="module" src="/src/main.js"></script>

View File

@@ -19,13 +19,14 @@
"@editorjs/paragraph": "^2.11.3", "@editorjs/paragraph": "^2.11.3",
"@editorjs/simple-image": "^1.6.0", "@editorjs/simple-image": "^1.6.0",
"@editorjs/table": "^2.4.2", "@editorjs/table": "^2.4.2",
"@vueuse/router": "^12.7.0",
"ace-builds": "^1.36.2", "ace-builds": "^1.36.2",
"apexcharts": "^4.3.0", "apexcharts": "^4.3.0",
"chart.js": "^4.4.1", "chart.js": "^4.4.1",
"codemirror-editor-vue3": "^2.8.0", "codemirror-editor-vue3": "^2.8.0",
"dayjs": "^1.11.6", "dayjs": "^1.11.6",
"feather-icons": "^4.28.0", "feather-icons": "^4.28.0",
"frappe-ui": "^0.1.89", "frappe-ui": "^0.1.112",
"lucide-vue-next": "^0.383.0", "lucide-vue-next": "^0.383.0",
"markdown-it": "^14.0.0", "markdown-it": "^14.0.0",
"pinia": "^2.0.33", "pinia": "^2.0.33",

View File

@@ -5,7 +5,7 @@
<div class="flex items-center justify-between mb-2"> <div class="flex items-center justify-between mb-2">
<div class="flex items-center"> <div class="flex items-center">
<Avatar :label="comm.sender_full_name" size="lg" /> <Avatar :label="comm.sender_full_name" size="lg" />
<div class="ml-2"> <div class="ml-2 text-ink-gray-7">
{{ comm.sender_full_name }} {{ comm.sender_full_name }}
</div> </div>
</div> </div>
@@ -14,13 +14,13 @@
</div> </div>
</div> </div>
<div <div
class="prose prose-sm bg-gray-50 !min-w-full px-4 py-2 rounded-md" class="prose prose-sm bg-surface-menu-bar !min-w-full px-4 py-2 rounded-md"
v-html="comm.content" v-html="comm.content"
></div> ></div>
</div> </div>
</div> </div>
</div> </div>
<div v-else class="text-sm italic text-gray-600"> <div v-else class="text-sm italic text-ink-gray-5">
{{ __('No announcements') }} {{ __('No announcements') }}
</div> </div>
</template> </template>

View File

@@ -1,6 +1,6 @@
<template> <template>
<div <div
class="flex h-full flex-col justify-between transition-all duration-300 ease-in-out bg-gray-50" class="flex h-full flex-col justify-between transition-all duration-300 ease-in-out border-r bg-surface-menu-bar"
:class="sidebarStore.isSidebarCollapsed ? 'w-14' : 'w-56'" :class="sidebarStore.isSidebarCollapsed ? 'w-14' : 'w-56'"
> >
<div <div
@@ -23,16 +23,16 @@
<div <div
class="flex items-center justify-between pr-2 cursor-pointer" class="flex items-center justify-between pr-2 cursor-pointer"
:class="sidebarStore.isSidebarCollapsed ? 'pl-3' : 'pl-4'" :class="sidebarStore.isSidebarCollapsed ? 'pl-3' : 'pl-4'"
@click="showWebPages = !showWebPages" @click="toggleWebPages"
> >
<div <div
v-if="!sidebarStore.isSidebarCollapsed" v-if="!sidebarStore.isSidebarCollapsed"
class="flex items-center text-sm text-gray-600 my-1" class="flex items-center text-sm text-ink-gray-5 my-1"
> >
<span class="grid h-5 w-6 flex-shrink-0 place-items-center"> <span class="grid h-5 w-6 flex-shrink-0 place-items-center">
<ChevronRight <ChevronRight
class="h-4 w-4 stroke-1.5 text-gray-900 transition-all duration-300 ease-in-out" class="h-4 w-4 stroke-1.5 text-ink-gray-9 transition-all duration-300 ease-in-out"
:class="{ 'rotate-90': showWebPages }" :class="{ 'rotate-90': !sidebarStore.isWebpagesCollapsed }"
/> />
</span> </span>
<span class="ml-2"> <span class="ml-2">
@@ -41,14 +41,14 @@
</div> </div>
<Button v-if="isModerator" variant="ghost" @click="openPageModal()"> <Button v-if="isModerator" variant="ghost" @click="openPageModal()">
<template #icon> <template #icon>
<Plus class="h-4 w-4 text-gray-700 stroke-1.5" /> <Plus class="h-4 w-4 text-ink-gray-7 stroke-1.5" />
</template> </template>
</Button> </Button>
</div> </div>
<div <div
v-if="sidebarSettings.data?.web_pages?.length" v-if="sidebarSettings.data?.web_pages?.length"
class="flex flex-col transition-all duration-300 ease-in-out" class="flex flex-col transition-all duration-300 ease-in-out"
:class="showWebPages ? 'block' : 'hidden'" :class="!sidebarStore.isWebpagesCollapsed ? 'block' : 'hidden'"
> >
<SidebarLink <SidebarLink
v-for="link in sidebarSettings.data.web_pages" v-for="link in sidebarSettings.data.web_pages"
@@ -62,25 +62,33 @@
</div> </div>
</div> </div>
</div> </div>
<SidebarLink <div>
:link="{ <TrialBanner
label: sidebarStore.isSidebarCollapsed ? 'Expand' : 'Collapse', v-if="
}" userResource.data?.is_system_manager && userResource.data?.is_fc_site
:isCollapsed="sidebarStore.isSidebarCollapsed" "
@click="toggleSidebar()" :isSidebarCollapsed="sidebarStore.isSidebarCollapsed"
class="m-2" />
> <SidebarLink
<template #icon> :link="{
<span class="grid h-5 w-6 flex-shrink-0 place-items-center"> label: sidebarStore.isSidebarCollapsed ? 'Expand' : 'Collapse',
<CollapseSidebar }"
class="h-4.5 w-4.5 text-gray-700 duration-300 ease-in-out" :isCollapsed="sidebarStore.isSidebarCollapsed"
:class="{ @click="toggleSidebar()"
'[transform:rotateY(180deg)]': sidebarStore.isSidebarCollapsed, class="m-2"
}" >
/> <template #icon>
</span> <span class="grid h-5 w-6 flex-shrink-0 place-items-center">
</template> <CollapseSidebar
</SidebarLink> class="h-4.5 w-4.5 text-ink-gray-7 duration-300 ease-in-out"
:class="{
'[transform:rotateY(180deg)]': sidebarStore.isSidebarCollapsed,
}"
/>
</span>
</template>
</SidebarLink>
</div>
</div> </div>
<PageModal <PageModal
v-model="showPageModal" v-model="showPageModal"
@@ -101,7 +109,7 @@ import { sessionStore } from '@/stores/session'
import { useSidebar } from '@/stores/sidebar' import { useSidebar } from '@/stores/sidebar'
import { useSettings } from '@/stores/settings' import { useSettings } from '@/stores/settings'
import { ChevronRight, Plus } from 'lucide-vue-next' import { ChevronRight, Plus } from 'lucide-vue-next'
import { createResource, Button } from 'frappe-ui' import { Button, createResource, TrialBanner } from 'frappe-ui'
import PageModal from '@/components/Modals/PageModal.vue' import PageModal from '@/components/Modals/PageModal.vue'
const { user, sidebarSettings } = sessionStore() const { user, sidebarSettings } = sessionStore()
@@ -114,7 +122,6 @@ const showPageModal = ref(false)
const isModerator = ref(false) const isModerator = ref(false)
const isInstructor = ref(false) const isInstructor = ref(false)
const pageToEdit = ref(null) const pageToEdit = ref(null)
const showWebPages = ref(false)
const settingsStore = useSettings() const settingsStore = useSettings()
onMounted(() => { onMounted(() => {
@@ -266,5 +273,17 @@ watch(userResource, () => {
const toggleSidebar = () => { const toggleSidebar = () => {
sidebarStore.isSidebarCollapsed = !sidebarStore.isSidebarCollapsed sidebarStore.isSidebarCollapsed = !sidebarStore.isSidebarCollapsed
localStorage.setItem(
'isSidebarCollapsed',
JSON.stringify(sidebarStore.isSidebarCollapsed)
)
}
const toggleWebPages = () => {
sidebarStore.isWebpagesCollapsed = !sidebarStore.isWebpagesCollapsed
localStorage.setItem(
'isWebpagesCollapsed',
JSON.stringify(sidebarStore.isWebpagesCollapsed)
)
} }
</script> </script>

View File

@@ -3,7 +3,7 @@
<template #target="{ togglePopover }"> <template #target="{ togglePopover }">
<button <button
:class="[ :class="[
'group w-full flex h-7 items-center justify-between rounded px-2 text-base text-gray-800 hover:bg-gray-100', 'group w-full flex h-7 items-center justify-between rounded px-2 text-base text-ink-gray-8 hover:bg-surface-gray-2',
]" ]"
@click.prevent="togglePopover()" @click.prevent="togglePopover()"
> >
@@ -18,15 +18,15 @@
</template> </template>
<template #body> <template #body>
<div <div
class="grid grid-cols-3 justify-between mx-3 p-2 rounded-lg border border-gray-100 bg-white shadow-xl" class="grid grid-cols-3 justify-between mx-3 p-2 rounded-lg border border-gray-100 bg-surface-white shadow-xl"
> >
<div v-for="app in apps.data" key="name"> <div v-for="app in apps.data" key="name">
<a <a
:href="app.route" :href="app.route"
class="flex flex-col gap-1.5 rounded justify-center items-center py-2 px-3 hover:bg-gray-100" class="flex flex-col gap-1.5 rounded justify-center items-center py-2 px-3 hover:bg-surface-gray-2"
> >
<img class="size-8" :src="app.logo" /> <img class="size-8" :src="app.logo" />
<div class="text-sm" @click="app.onClick"> <div class="text-sm text-ink-gray-7" @click="app.onClick">
{{ app.title }} {{ app.title }}
</div> </div>
</a> </a>

View File

@@ -1,7 +1,7 @@
<template> <template>
<div> <div>
<div class="flex items-center justify-between mb-4"> <div class="flex items-center justify-between mb-4">
<div class="text-lg font-semibold"> <div class="text-lg font-semibold text-ink-gray-9">
{{ __('Assessments') }} {{ __('Assessments') }}
</div> </div>
<Button v-if="canSeeAddButton()" @click="showModal = true"> <Button v-if="canSeeAddButton()" @click="showModal = true">
@@ -11,7 +11,7 @@
{{ __('Add') }} {{ __('Add') }}
</Button> </Button>
</div> </div>
<div v-if="assessments.data?.length"> <div v-if="assessments.data?.length" class="text-sm">
<ListView <ListView
:columns="getAssessmentColumns()" :columns="getAssessmentColumns()"
:rows="assessments.data" :rows="assessments.data"
@@ -23,7 +23,7 @@
}" }"
> >
<ListHeader <ListHeader
class="mb-2 grid items-center space-x-4 rounded bg-gray-100 p-2" class="mb-2 grid items-center space-x-4 rounded bg-surface-gray-2 p-2"
> >
<ListHeaderItem :item="item" v-for="item in getAssessmentColumns()"> <ListHeaderItem :item="item" v-for="item in getAssessmentColumns()">
<template #prefix="{ item }"> <template #prefix="{ item }">
@@ -71,7 +71,7 @@
</ListSelectBanner> </ListSelectBanner>
</ListView> </ListView>
</div> </div>
<div v-else class="text-sm italic text-gray-600"> <div v-else class="text-sm italic text-ink-gray-5">
{{ __('No Assessments') }} {{ __('No Assessments') }}
</div> </div>
</div> </div>

View File

@@ -1,11 +1,11 @@
<template> <template>
<div <div
v-if="assignment.data" v-if="assignment.data"
class="grid grid-cols-[68%,32%] h-full" class="grid grid-cols-[65%,35%] h-full"
:class="{ 'border rounded-lg': !showTitle }" :class="{ 'border rounded-lg': !showTitle }"
> >
<div class="border-r p-5 overflow-y-auto h-[calc(100vh-3.2rem)]"> <div class="border-r p-5 overflow-y-auto h-[calc(100vh-3.2rem)]">
<div v-if="showTitle" class="text-lg font-semibold mb-5"> <div v-if="showTitle" class="text-lg font-semibold mb-5 text-ink-gray-9">
<div v-if="submissionName === 'new'"> <div v-if="submissionName === 'new'">
{{ __('Submission by') }} {{ user.data?.full_name }} {{ __('Submission by') }} {{ user.data?.full_name }}
</div> </div>
@@ -13,19 +13,19 @@
{{ __('Submission by') }} {{ submissionResource.doc?.member_name }} {{ __('Submission by') }} {{ submissionResource.doc?.member_name }}
</div> </div>
</div> </div>
<div class="text-sm text-gray-600 font-medium mb-2"> <div class="text-sm text-ink-gray-7 font-medium mb-2">
{{ __('Question') }}: {{ __('Question') }}:
</div> </div>
<div <div
v-html="assignment.data.question" v-html="assignment.data.question"
class="ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-gray-300 prose-th:border-gray-300 prose-td:relative prose-th:relative prose-th:bg-gray-100 prose-sm max-w-none !whitespace-normal" class="ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-outline-gray-2 prose-th:border-outline-gray-2 prose-td:relative prose-th:relative prose-th:bg-surface-gray-2 prose-sm max-w-none !whitespace-normal"
></div> ></div>
</div> </div>
<div class="flex flex-col"> <div class="flex flex-col">
<div class="p-5"> <div class="p-5">
<div class="flex items-center justify-between mb-4"> <div class="flex items-center justify-between mb-4">
<div class="font-semibold"> <div class="font-semibold text-ink-gray-9">
{{ __('Submission') }} {{ __('Submission') }}
</div> </div>
<div class="flex items-center space-x-2"> <div class="flex items-center space-x-2">
@@ -50,7 +50,7 @@
!['Pass', 'Fail'].includes(submissionResource.doc?.status) && !['Pass', 'Fail'].includes(submissionResource.doc?.status) &&
submissionResource.doc?.owner == user.data?.name submissionResource.doc?.owner == user.data?.name
" "
class="bg-blue-100 p-3 rounded-md leading-5 text-sm mb-4" class="bg-surface-blue-2 p-3 rounded-md leading-5 text-sm mb-4"
> >
{{ __("You've successfully submitted the assignment.") }} {{ __("You've successfully submitted the assignment.") }}
{{ {{
@@ -61,7 +61,7 @@
{{ __('Feel free to make edits to your submission if needed.') }} {{ __('Feel free to make edits to your submission if needed.') }}
</div> </div>
<div v-if="showUploader()"> <div v-if="showUploader()">
<div class="text-xs text-gray-600 mt-1 mb-2"> <div class="text-xs text-ink-gray-5 mt-1 mb-2">
{{ __('Add your assignment as {0}').format(assignment.data.type) }} {{ __('Add your assignment as {0}').format(assignment.data.type) }}
</div> </div>
<FileUploader <FileUploader
@@ -81,32 +81,32 @@
</template> </template>
</FileUploader> </FileUploader>
<div v-else> <div v-else>
<div class="flex items-center"> <div class="flex text-ink-gray-7">
<div class="border rounded-md p-2 mr-2"> <div class="border self-start rounded-md p-2 mr-2">
<FileText class="h-5 w-5 stroke-1.5 text-gray-700" /> <FileText class="h-5 w-5 stroke-1.5" />
</div> </div>
<a <a
:href="submissionFile.file_url" :href="submissionFile.file_url"
target="_blank" target="_blank"
class="flex flex-col cursor-pointer !no-underline" class="flex flex-col cursor-pointer !no-underline"
> >
<span> <span class="text-sm leading-5">
{{ submissionFile.file_name }} {{ submissionFile.file_name }}
</span> </span>
<span class="text-sm text-gray-500 mt-1"> <span class="text-sm text-ink-gray-5 mt-1">
{{ getFileSize(submissionFile.file_size) }} {{ getFileSize(submissionFile.file_size) }}
</span> </span>
</a> </a>
<X <X
v-if="canModifyAssignment" v-if="canModifyAssignment"
@click="removeSubmission()" @click="removeSubmission()"
class="bg-gray-200 rounded-md cursor-pointer stroke-1.5 w-5 h-5 p-1 ml-4" class="bg-surface-gray-3 rounded-md cursor-pointer stroke-1.5 w-5 h-5 p-1 ml-4"
/> />
</div> </div>
</div> </div>
</div> </div>
<div v-else-if="assignment.data.type == 'URL'"> <div v-else-if="assignment.data.type == 'URL'">
<div class="text-xs text-gray-600 mb-1"> <div class="text-xs text-ink-gray-5 mb-1">
{{ __('Enter a URL') }} {{ __('Enter a URL') }}
</div> </div>
<FormControl <FormControl
@@ -124,7 +124,7 @@
@change="(val) => (answer = val)" @change="(val) => (answer = val)"
:editable="true" :editable="true"
:fixedMenu="true" :fixedMenu="true"
editorClass="prose-sm max-w-none border-b border-x bg-gray-100 rounded-b-md py-1 px-2 min-h-[7rem]" editorClass="prose-sm max-w-none border-b border-x bg-surface-gray-2 rounded-b-md py-1 px-2 min-h-[7rem]"
/> />
</div> </div>
@@ -133,9 +133,9 @@
user.data?.name == submissionResource.doc?.owner && user.data?.name == submissionResource.doc?.owner &&
submissionResource.doc?.comments submissionResource.doc?.comments
" "
class="mt-8 p-3 bg-blue-100 rounded-md" class="mt-8 p-3 bg-surface-blue-2 rounded-md"
> >
<div class="text-sm text-gray-600 font-medium mb-2"> <div class="text-sm text-ink-gray-5 font-medium mb-2">
{{ __('Comments by Evaluator') }}: {{ __('Comments by Evaluator') }}:
</div> </div>
<div class="leading-5"> <div class="leading-5">
@@ -145,7 +145,7 @@
<!-- Grading --> <!-- Grading -->
<div v-if="canGradeSubmission" class="mt-8 space-y-4"> <div v-if="canGradeSubmission" class="mt-8 space-y-4">
<div class="font-semibold mb-2"> <div class="font-semibold mb-2 text-ink-gray-9">
{{ __('Grading') }} {{ __('Grading') }}
</div> </div>
<FormControl <FormControl
@@ -155,12 +155,23 @@
type="select" type="select"
:options="submissionStatusOptions" :options="submissionStatusOptions"
/> />
<FormControl <div>
v-if="submissionResource.doc" <div class="text-sm text-ink-gray-5 mb-1">
v-model="submissionResource.doc.comments" {{ __('Comments') }}
:label="__('Comments')" </div>
type="textarea" <TextEditor
/> :content="comments"
@change="
(val) => {
comments = val
isDirty = true
}
"
:editable="true"
:fixedMenu="true"
editorClass="prose-sm max-w-none border-b border-x bg-surface-gray-2 rounded-b-md py-1 px-2 min-h-[7rem]"
/>
</div>
</div> </div>
</div> </div>
</div> </div>
@@ -184,6 +195,7 @@ import { useRouter } from 'vue-router'
const submissionFile = ref(null) const submissionFile = ref(null)
const answer = ref(null) const answer = ref(null)
const comments = ref(null)
const router = useRouter() const router = useRouter()
const user = inject('$user') const user = inject('$user')
const showTitle = router.currentRoute.value.name == 'AssignmentSubmission' const showTitle = router.currentRoute.value.name == 'AssignmentSubmission'
@@ -281,7 +293,9 @@ watch(submissionResource, () => {
if (submissionResource.doc.answer) { if (submissionResource.doc.answer) {
answer.value = submissionResource.doc.answer answer.value = submissionResource.doc.answer
} }
if (submissionResource.doc.comments) {
comments.value = submissionResource.doc.comments
}
if (submissionResource.isDirty) { if (submissionResource.isDirty) {
isDirty.value = true isDirty.value = true
} else if (showUploader() && !submissionFile.value) { } else if (showUploader() && !submissionFile.value) {
@@ -306,10 +320,14 @@ const submitAssignment = () => {
submissionResource.doc && submissionResource.doc.owner != user.data?.name submissionResource.doc && submissionResource.doc.owner != user.data?.name
? user.data?.name ? user.data?.name
: null : null
submissionResource.setValue.submit( submissionResource.setValue.submit(
{ {
...submissionResource.doc, ...submissionResource.doc,
assignment_attachment: submissionFile.value?.file_url,
evaluator: evaluator, evaluator: evaluator,
comments: comments.value,
answer: answer.value,
}, },
{ {
onSuccess(data) { onSuccess(data) {
@@ -351,6 +369,7 @@ const addNewSubmission = () => {
} }
const saveSubmission = (file) => { const saveSubmission = (file) => {
isDirty.value = true
submissionFile.value = file submissionFile.value = file
} }
@@ -401,6 +420,7 @@ const validateFile = (file) => {
} }
const removeSubmission = () => { const removeSubmission = () => {
isDirty.value = true
submissionFile.value = null submissionFile.value = null
} }

View File

@@ -9,8 +9,8 @@
<div class="flex items-center space-x-2 shadow rounded-lg p-1 w-1/2"> <div class="flex items-center space-x-2 shadow rounded-lg p-1 w-1/2">
<Button variant="ghost" @click="togglePlay"> <Button variant="ghost" @click="togglePlay">
<template #icon> <template #icon>
<Play v-if="!isPlaying" class="w-4 h-4 text-gray-900" /> <Play v-if="!isPlaying" class="w-4 h-4 text-ink-gray-9" />
<Pause v-else class="w-4 h-4 text-gray-900" /> <Pause v-else class="w-4 h-4 text-ink-gray-9" />
</template> </template>
</Button> </Button>
<input <input
@@ -22,13 +22,13 @@
@input="changeCurrentTime" @input="changeCurrentTime"
class="duration-slider w-full h-1" class="duration-slider w-full h-1"
/> />
<span class="text-xs text-gray-900 font-medium"> <span class="text-xs text-ink-gray-9 font-medium">
{{ formatTime(currentTime) }} / {{ formatTime(duration) }} {{ formatTime(currentTime) }} / {{ formatTime(duration) }}
</span> </span>
<Button variant="ghost" @click="toggleMute"> <Button variant="ghost" @click="toggleMute">
<template #icon> <template #icon>
<Volume2 v-if="!isMuted" class="w-4 h-4 text-gray-900" /> <Volume2 v-if="!isMuted" class="w-4 h-4 text-ink-gray-9" />
<VolumeX v-else class="w-4 h-4 text-gray-900" /> <VolumeX v-else class="w-4 h-4 text-ink-gray-9" />
</template> </template>
</Button> </Button>
</div> </div>

View File

@@ -1,50 +1,52 @@
<template> <template>
<div <div
class="flex flex-col shadow hover:bg-gray-100 rounded-md p-4 h-full" class="flex flex-col border-2 hover:bg-surface-gray-2 rounded-md p-4 h-full"
style="min-height: 150px" style="min-height: 150px"
> >
<div class="text-lg leading-5 font-semibold mb-2"> <div class="text-lg leading-5 font-semibold mb-2 text-ink-gray-9">
{{ batch.title }} {{ batch.title }}
</div> </div>
<Badge <div
v-if="batch.seat_count && batch.seats_left > 0" v-if="batch.seat_count && batch.seats_left > 0"
theme="green" class="text-xs bg-green-100 text-green-700 self-start px-2 py-0.5 rounded-md"
class="self-start mb-2"
> >
{{ batch.seats_left }} {{ batch.seats_left }}
<span v-if="batch.seats_left > 1">{{ __('Seats Left') }}</span <span v-if="batch.seats_left > 1">
><span v-else-if="batch.seats_left == 1">{{ __('Seat Left') }}</span> {{ __('Seats Left') }}
</Badge> </span>
<Badge <span v-else-if="batch.seats_left == 1">
{{ __('Seat Left') }}
</span>
</div>
<div
v-else-if="batch.seat_count && batch.seats_left <= 0" v-else-if="batch.seat_count && batch.seats_left <= 0"
theme="red" class="text-xs bg-red-100 text-red-700 self-start px-2 py-0.5 rounded-md"
class="self-start mb-2"
> >
{{ __('Sold Out') }} {{ __('Sold Out') }}
</Badge> </div>
<div class="short-introduction text-sm text-gray-700"> <div class="short-introduction text-sm text-ink-gray-7">
{{ batch.description }} {{ batch.description }}
</div> </div>
<div v-if="batch.amount" class="font-semibold mb-4"> <div v-if="batch.amount" class="font-semibold text-ink-gray-9 mb-4">
{{ batch.price }} {{ batch.price }}
</div> </div>
<div class="flex flex-col space-y-2 mt-auto"> <div class="flex flex-col space-y-2 mt-auto">
<DateRange <DateRange
:startDate="batch.start_date" :startDate="batch.start_date"
:endDate="batch.end_date" :endDate="batch.end_date"
class="text-sm text-gray-700" class="text-sm text-ink-gray-7"
/> />
<div class="flex items-center text-sm text-gray-700"> <div class="flex items-center text-sm text-ink-gray-7">
<Clock class="h-4 w-4 stroke-1.5 mr-2 text-gray-700" /> <Clock class="h-4 w-4 stroke-1.5 mr-2 text-ink-gray-7" />
<span> <span>
{{ formatTime(batch.start_time) }} - {{ formatTime(batch.end_time) }} {{ formatTime(batch.start_time) }} - {{ formatTime(batch.end_time) }}
</span> </span>
</div> </div>
<div <div
v-if="batch.timezone" v-if="batch.timezone"
class="flex items-center text-sm text-gray-700" class="flex items-center text-sm text-ink-gray-7"
> >
<Globe class="h-4 w-4 stroke-1.5 mr-2 text-gray-600" /> <Globe class="h-4 w-4 stroke-1.5 mr-2 text-ink-gray-5" />
<span> <span>
{{ batch.timezone }} {{ batch.timezone }}
</span> </span>

View File

@@ -1,7 +1,7 @@
<template> <template>
<div> <div>
<div class="flex items-center justify-between mb-4"> <div class="flex items-center justify-between mb-4">
<div class="text-lg font-semibold"> <div class="text-lg font-semibold text-ink-gray-9">
{{ __('Courses') }} {{ __('Courses') }}
</div> </div>
<Button v-if="canSeeAddButton()" @click="openCourseModal()"> <Button v-if="canSeeAddButton()" @click="openCourseModal()">
@@ -18,6 +18,7 @@
row-key="batch_course" row-key="batch_course"
:options="{ :options="{
showTooltip: false, showTooltip: false,
selectable: user.data?.is_student ? false : true,
getRowRoute: (row) => ({ getRowRoute: (row) => ({
name: 'CourseDetail', name: 'CourseDetail',
params: { courseName: row.name }, params: { courseName: row.name },
@@ -25,7 +26,7 @@
}" }"
> >
<ListHeader <ListHeader
class="mb-2 grid items-center space-x-4 rounded bg-gray-100 p-2" class="mb-2 grid items-center space-x-4 rounded bg-surface-gray-2 p-2"
> >
<ListHeaderItem :item="item" v-for="item in getCoursesColumns()"> <ListHeaderItem :item="item" v-for="item in getCoursesColumns()">
<template #prefix="{ item }"> <template #prefix="{ item }">

View File

@@ -1,17 +1,18 @@
<template> <template>
<div> <div class="space-y-10">
<UpcomingEvaluations <UpcomingEvaluations
:batch="batch.data.name" :batch="batch.data.name"
:endDate="batch.data.evaluation_end_date" :endDate="batch.data.evaluation_end_date"
:courses="batch.data.courses" :courses="batch.data.courses"
:isStudent="isStudent"
/> />
<Assessments :batch="batch.data.name" /> <Assessments :batch="batch.data.name" />
<StudentHeatmap />
</div> </div>
</template> </template>
<script setup> <script setup>
import UpcomingEvaluations from '@/components/UpcomingEvaluations.vue' import UpcomingEvaluations from '@/components/UpcomingEvaluations.vue'
import Assessments from '@/components/Assessments.vue' import Assessments from '@/components/Assessments.vue'
import StudentHeatmap from '@/components/StudentHeatmap.vue'
const props = defineProps({ const props = defineProps({
batch: { batch: {

View File

@@ -0,0 +1,245 @@
<template>
<div v-if="user.data?.is_student">
<div
v-if="feedbackList.data?.length"
class="bg-surface-blue-2 text-blue-700 p-2 rounded-md mb-5"
>
{{ __('Thank you for providing your feedback!') }}
</div>
<div v-else class="flex justify-between items-center mb-5">
<div class="text-lg font-semibold">
{{ __('Help Us Improve') }}
</div>
<Button @click="submitFeedback()">
{{ __('Submit') }}
</Button>
</div>
<div class="space-y-8">
<div class="flex items-center justify-between">
<Rating
v-for="key in ratingKeys"
v-model="feedback[key]"
:label="__(convertToTitleCase(key))"
:readonly="readOnly"
/>
</div>
<FormControl
v-model="feedback.feedback"
type="textarea"
:label="__('Feedback')"
:rows="7"
:readonly="readOnly"
/>
</div>
</div>
<div v-else-if="feedbackList.data?.length">
<div class="text-lg font-semibold mb-5">
{{ __('Average of Feedback Received') }}
</div>
<div class="flex items-center justify-between mb-10">
<Rating
v-for="key in ratingKeys"
v-model="average[key]"
:label="__(convertToTitleCase(key))"
:readonly="true"
/>
</div>
<div class="text-lg font-semibold mb-5">
{{ __('All Feedback') }}
</div>
<ListView
:columns="feedbackColumns"
:rows="feedbackList.data"
row-key="name"
:options="{
showTooltip: false,
rowHeight: 'h-16',
selectable: false,
}"
>
<ListHeader
class="mb-2 grid items-center space-x-4 rounded bg-surface-gray-2 p-2"
></ListHeader>
<ListRows>
<ListRow
:row="row"
v-for="row in feedbackList.data"
class="group cursor-pointer feedback-list"
>
<template #default="{ column, item }">
<ListRowItem
:item="row[column.key]"
:align="column.align"
class="text-sm"
>
<template #prefix>
<div v-if="column.key == 'member_name'">
<Avatar
class="flex"
:image="row['member_image']"
:label="item"
size="sm"
/>
</div>
</template>
<div v-if="ratingKeys.includes(column.key)">
<Rating v-model="row[column.key]" :readonly="true" />
</div>
<div v-else class="leading-5">
{{ row[column.key] }}
</div>
</ListRowItem>
</template>
</ListRow>
</ListRows>
</ListView>
</div>
<div v-else class="text-sm italic text-center text-ink-gray-7 mt-5">
{{ __('No feedback received yet.') }}
</div>
</template>
<script setup>
import { computed, inject, onMounted, reactive, ref, watch } from 'vue'
import { convertToTitleCase } from '@/utils'
import {
Avatar,
Button,
createListResource,
FormControl,
ListView,
ListHeader,
ListHeaderItem,
ListRows,
ListRow,
ListRowItem,
Rating,
} from 'frappe-ui'
const user = inject('$user')
const ratingKeys = ['content', 'instructors', 'value']
const readOnly = ref(false)
const average = reactive({})
const feedback = reactive({})
const props = defineProps({
batch: {
type: String,
required: true,
},
})
onMounted(() => {
let filters = {
batch: props.batch,
}
if (user.data?.is_student) {
filters['member'] = user.data?.name
}
feedbackList.update({
filters: filters,
})
feedbackList.reload()
})
const feedbackList = createListResource({
doctype: 'LMS Batch Feedback',
filters: {
batch: props.batch,
},
fields: [
'content',
'instructors',
'value',
'feedback',
'name',
'member',
'member_name',
'member_image',
],
cache: ['feedbackList', props.batch, user.data?.name],
})
watch(
() => feedbackList.data,
() => {
if (feedbackList.data.length) {
let data = feedbackList.data
readOnly.value = true
ratingKeys.forEach((key) => {
average[key] = 0
})
data.forEach((row) => {
Object.keys(row).forEach((key) => {
if (ratingKeys.includes(key)) row[key] = row[key] * 5
feedback[key] = row[key]
})
ratingKeys.forEach((key) => {
average[key] += row[key]
})
})
Object.keys(average).forEach((key) => {
average[key] = average[key] / data.length
})
}
}
)
const submitFeedback = () => {
ratingKeys.forEach((key) => {
feedback[key] = feedback[key] / 5
})
feedbackList.insert.submit(
{
member: user.data?.name,
batch: props.batch,
...feedback,
},
{
onSuccess: () => {
feedbackList.reload()
},
}
)
}
const feedbackColumns = computed(() => {
return [
{
label: 'Member',
key: 'member_name',
width: '10rem',
},
{
label: 'Feedback',
key: 'feedback',
width: '15rem',
},
{
label: 'Content',
key: 'content',
width: '9rem',
},
{
label: 'Instructors',
key: 'instructors',
width: '9rem',
},
{
label: 'Value',
key: 'value',
width: '9rem',
},
]
})
</script>
<style>
.feedback-list > button > div {
align-items: start;
padding: 0.15rem 0;
}
</style>

View File

@@ -1,25 +1,31 @@
<template> <template>
<div v-if="batch.data" class="shadow rounded-md p-5 lg:w-72"> <div v-if="batch.data" class="border-2 rounded-md p-5 lg:w-72">
<Badge <div
v-if="batch.data.seat_count && seats_left > 0" v-if="batch.data.seat_count && seats_left > 0"
theme="green" class="text-xs bg-green-100 text-green-700 float-right px-2 py-0.5 rounded-md"
class="self-start mb-2 float-right"
> >
{{ seats_left }} <span v-if="seats_left > 1">{{ __('Seats Left') }}</span {{ seats_left }}
><span v-else-if="seats_left == 1">{{ __('Seat Left') }}</span> <span v-if="seats_left > 1">
</Badge> {{ __('Seats Left') }}
<Badge </span>
<span v-else-if="seats_left == 1">
{{ __('Seat Left') }}
</span>
</div>
<div
v-else-if="batch.data.seat_count && seats_left <= 0" v-else-if="batch.data.seat_count && seats_left <= 0"
theme="red" class="text-xs bg-red-100 text-red-700 float-right px-2 py-0.5 rounded-md"
class="self-start mb-2 float-right"
> >
{{ __('Sold Out') }} {{ __('Sold Out') }}
</Badge> </div>
<div v-if="batch.data.amount" class="text-lg font-semibold mb-3"> <div
v-if="batch.data.amount"
class="text-lg font-semibold mb-3 text-ink-gray-9"
>
{{ formatNumberIntoCurrency(batch.data.amount, batch.data.currency) }} {{ formatNumberIntoCurrency(batch.data.amount, batch.data.currency) }}
</div> </div>
<div class="flex items-center mb-3"> <div class="flex items-center mb-3 text-ink-gray-7">
<BookOpen class="h-4 w-4 stroke-1.5 mr-2 text-gray-700" /> <BookOpen class="h-4 w-4 stroke-1.5 mr-2" />
<span> {{ batch.data.courses.length }} {{ __('Courses') }} </span> <span> {{ batch.data.courses.length }} {{ __('Courses') }} </span>
</div> </div>
<DateRange <DateRange
@@ -27,15 +33,15 @@
:endDate="batch.data.end_date" :endDate="batch.data.end_date"
class="mb-3" class="mb-3"
/> />
<div class="flex items-center mb-3"> <div class="flex items-center mb-3 text-ink-gray-7">
<Clock class="h-4 w-4 stroke-1.5 mr-2 text-gray-700" /> <Clock class="h-4 w-4 stroke-1.5 mr-2" />
<span> <span>
{{ formatTime(batch.data.start_time) }} - {{ formatTime(batch.data.start_time) }} -
{{ formatTime(batch.data.end_time) }} {{ formatTime(batch.data.end_time) }}
</span> </span>
</div> </div>
<div v-if="batch.data.timezone" class="flex items-center"> <div v-if="batch.data.timezone" class="flex items-center text-ink-gray-7">
<Globe class="h-4 w-4 stroke-1.5 mr-2 text-gray-700" /> <Globe class="h-4 w-4 stroke-1.5 mr-2" />
<span> <span>
{{ batch.data.timezone }} {{ batch.data.timezone }}
</span> </span>
@@ -63,7 +69,11 @@
name: batch.data.name, name: batch.data.name,
}, },
}" }"
v-else-if="batch.data.paid_batch && batch.data.seats_left" v-else-if="
batch.data.paid_batch &&
batch.data.seats_left > 0 &&
batch.data.accept_enrollments
"
> >
<Button v-if="!isStudent" class="w-full mt-4" variant="solid"> <Button v-if="!isStudent" class="w-full mt-4" variant="solid">
<span> <span>
@@ -74,7 +84,11 @@
<Button <Button
variant="solid" variant="solid"
class="w-full mt-2" class="w-full mt-2"
v-else-if="batch.data.allow_self_enrollment && batch.data.seats_left" v-else-if="
batch.data.allow_self_enrollment &&
batch.data.seats_left &&
batch.data.accept_enrollments
"
@click="enrollInBatch()" @click="enrollInBatch()"
> >
{{ __('Enroll Now') }} {{ __('Enroll Now') }}
@@ -106,6 +120,7 @@ import { useRouter } from 'vue-router'
const router = useRouter() const router = useRouter()
const user = inject('$user') const user = inject('$user')
const dayjs = inject('$dayjs')
const props = defineProps({ const props = defineProps({
batch: { batch: {

View File

@@ -1,75 +1,102 @@
<template> <template>
<div class=""> <div class="">
<div class="w-full flex items-center justify-between pb-4"> <div class="w-full flex items-center justify-between pb-4">
<div class="font-medium text-gray-600"> <div class="font-medium text-ink-gray-7">
{{ __('Statistics') }} {{ __('Statistics') }}
</div> </div>
</div> </div>
<div class="grid grid-cols-3 gap-5 mb-8"> <div class="grid grid-cols-4 gap-5 mb-8">
<div class="flex items-center shadow py-2 px-3 rounded-md"> <div
<div class="p-2 rounded-md bg-gray-100 mr-3"> class="flex items-center border py-2 px-3 rounded-md text-ink-gray-7"
<User class="w-18 h-18 stroke-1.5 text-gray-700" /> >
<div class="p-2 rounded-md bg-surface-gray-2 mr-3">
<User class="w-5 h-5 stroke-1.5" />
</div> </div>
<div class="flex flex-col"> <div class="flex items-center space-x-2">
<span class="text-xl font-semibold mb-1"> <span class="font-semibold">
{{ students.data?.length }} {{ students.data?.length }}
</span> </span>
<span class="text-gray-700"> <span class="">
{{ __('Students') }} {{ __('Students') }}
</span> </span>
</div> </div>
</div> </div>
<div class="flex items-center shadow py-2 px-3 rounded-md"> <div
<div class="p-2 rounded-md bg-gray-100 mr-3"> class="flex items-center border py-2 px-3 rounded-md text-ink-gray-7"
<BookOpen class="w-18 h-18 stroke-1.5 text-gray-700" /> >
<div class="p-2 rounded-md bg-surface-gray-2 mr-3">
<GraduationCap class="w-5 h-5 stroke-1.5" />
</div> </div>
<div class="flex flex-col"> <div class="flex items-center space-x-2">
<span class="text-xl font-semibold mb-1"> <span class="font-semibold">
{{ certificationCount.data }}
</span>
<span class="">
{{ __('Certified') }}
</span>
</div>
</div>
<div
class="flex items-center border py-2 px-3 rounded-md text-ink-gray-7"
>
<div class="p-2 rounded-md bg-surface-gray-2 mr-3">
<BookOpen class="w-5 h-5 stroke-1.5" />
</div>
<div class="flex items-center space-x-2">
<span class="font-semibold">
{{ batch.courses?.length }} {{ batch.courses?.length }}
</span> </span>
<span class="text-gray-700"> <span>
{{ __('Courses') }} {{ __('Courses') }}
</span> </span>
</div> </div>
</div> </div>
<div class="flex items-center shadow py-2 px-3 rounded-md"> <div
<div class="p-2 rounded-md bg-gray-100 mr-3"> class="flex items-center border py-2 px-3 rounded-md text-ink-gray-7"
<ShieldCheck class="w-18 h-18 stroke-1.5 text-gray-700" /> >
<div class="p-2 rounded-md bg-surface-gray-2 mr-3">
<ShieldCheck class="w-5 h-5 stroke-1.5" />
</div> </div>
<div class="flex flex-col"> <div class="flex items-center space-x-2">
<span class="text-xl font-semibold mb-1"> <span class="font-semibold">
{{ assessmentCount }} {{ assessmentCount }}
</span> </span>
<span class="text-gray-700"> <span>
{{ __('Assessments') }} {{ __('Assessments') }}
</span> </span>
</div> </div>
</div> </div>
</div> </div>
<div class="mb-8"> <div v-if="showProgressChart" class="mb-8">
<div class="text-gray-600 font-medium"> <div class="text-ink-gray-7 font-medium">
{{ __('Progress') }} {{ __('Progress') }}
</div> </div>
<ApexChart <ApexChart
v-if="showProgressChart"
:options="chartOptions" :options="chartOptions"
:series="chartData" :series="chartData"
type="bar" type="bar"
height="350" :height="chartData[0].data.length * 30 + 100"
/> />
<div <div
class="flex items-center justify-center text-sm text-gray-700 space-x-4" class="flex items-center justify-center text-sm text-ink-gray-7 space-x-4"
> >
<div class="flex items-center space-x-2"> <div class="flex items-center space-x-2">
<div class="w-3 h-3" style="background-color: #0f736b"></div> <div
class="w-3 h-3 rounded-sm"
:style="{ 'background-color': theme.colors.green[600] }"
></div>
<div> <div>
{{ __('Courses') }} {{ __('Courses') }}
</div> </div>
</div> </div>
<div class="flex items-center space-x-2"> <div class="flex items-center space-x-2">
<div class="w-3 h-3" style="background-color: #0070cc"></div> <div
class="w-3 h-3 rounded-sm"
:style="{ 'background-color': theme.colors.blue[600] }"
></div>
<div> <div>
{{ __('Assessments') }} {{ __('Assessments') }}
</div> </div>
@@ -80,7 +107,7 @@
<div> <div>
<div class="flex items-center justify-between mb-4"> <div class="flex items-center justify-between mb-4">
<div class="text-gray-600 font-medium"> <div class="text-ink-gray-7 font-medium">
{{ __('Students') }} {{ __('Students') }}
</div> </div>
<Button @click="openStudentModal()"> <Button @click="openStudentModal()">
@@ -101,7 +128,7 @@
}" }"
> >
<ListHeader <ListHeader
class="mb-2 grid items-center space-x-4 rounded bg-gray-100 p-2" class="mb-2 grid items-center space-x-4 rounded bg-surface-gray-2 p-2"
> >
<ListHeaderItem <ListHeaderItem
:item="item" :item="item"
@@ -125,7 +152,11 @@
@click="openStudentProgressModal(row)" @click="openStudentProgressModal(row)"
> >
<template #default="{ column, item }"> <template #default="{ column, item }">
<ListRowItem :item="row[column.key]" :align="column.align"> <ListRowItem
:item="row[column.key]"
:align="column.align"
class="text-sm"
>
<template #prefix> <template #prefix>
<div v-if="column.key == 'full_name'"> <div v-if="column.key == 'full_name'">
<Avatar <Avatar
@@ -141,16 +172,7 @@
class="flex items-center space-x-4 w-full" class="flex items-center space-x-4 w-full"
> >
<ProgressBar :progress="row[column.key]" size="sm" /> <ProgressBar :progress="row[column.key]" size="sm" />
</div> <div class="text-xs">{{ row[column.key] }}%</div>
<div
v-else-if="column.key == 'copy'"
class="invisible group-hover:visible"
>
<Button variant="ghost" @click="copyEmail(row)">
<template #icon>
<Clipboard class="h-4 w-4 stroke-1.5" />
</template>
</Button>
</div> </div>
<div v-else> <div v-else>
{{ row[column.key] }} {{ row[column.key] }}
@@ -173,7 +195,7 @@
</ListSelectBanner> </ListSelectBanner>
</ListView> </ListView>
</div> </div>
<div v-else class="text-sm italic text-gray-600"> <div v-else class="text-sm italic text-ink-gray-5">
{{ __('There are no students in this batch.') }} {{ __('There are no students in this batch.') }}
</div> </div>
</div> </div>
@@ -204,7 +226,7 @@ import {
} from 'frappe-ui' } from 'frappe-ui'
import { import {
BookOpen, BookOpen,
Clipboard, GraduationCap,
Plus, Plus,
ShieldCheck, ShieldCheck,
Trash2, Trash2,
@@ -216,6 +238,7 @@ import { showToast } from '@/utils'
import ProgressBar from '@/components/ProgressBar.vue' import ProgressBar from '@/components/ProgressBar.vue'
import BatchStudentProgress from '@/components/Modals/BatchStudentProgress.vue' import BatchStudentProgress from '@/components/Modals/BatchStudentProgress.vue'
import ApexChart from 'vue3-apexcharts' import ApexChart from 'vue3-apexcharts'
import { theme } from '@/utils/theme'
const showStudentModal = ref(false) const showStudentModal = ref(false)
const showStudentProgressModal = ref(false) const showStudentProgressModal = ref(false)
@@ -241,7 +264,7 @@ const students = createResource({
auto: true, auto: true,
onSuccess(data) { onSuccess(data) {
chartData.value = getChartData() chartData.value = getChartData()
showProgressChart.value = true showProgressChart.value = data.length && true
}, },
}) })
@@ -256,20 +279,16 @@ const getStudentColumns = () => {
{ {
label: 'Progress', label: 'Progress',
key: 'progress', key: 'progress',
width: '10rem', width: '15rem',
icon: 'activity', icon: 'activity',
}, },
{ {
label: 'Last Active', label: 'Last Active',
key: 'last_active', key: 'last_active',
width: '15rem', width: '10rem',
align: 'center', align: 'center',
icon: 'clock', icon: 'clock',
}, },
{
label: '',
key: 'copy',
},
] ]
return columns return columns
@@ -288,7 +307,7 @@ const deleteStudents = createResource({
url: 'lms.lms.api.delete_documents', url: 'lms.lms.api.delete_documents',
makeParams(values) { makeParams(values) {
return { return {
doctype: 'Batch Student', doctype: 'LMS Batch Enrollment',
documents: values.students, documents: values.students,
} }
}, },
@@ -312,7 +331,9 @@ const removeStudents = (selections, unselectAll) => {
const getChartData = () => { const getChartData = () => {
let categories = {} let categories = {}
Object.keys(students.data?.[0].courses).forEach((course) => { if (!students.data?.length) return []
Object.keys(students.data[0].courses).forEach((course) => {
categories[course] = { categories[course] = {
value: 0, value: 0,
type: 'course', type: 'course',
@@ -336,7 +357,7 @@ const getChartData = () => {
}) })
Object.keys(student.assessments).forEach((assessment) => { Object.keys(student.assessments).forEach((assessment) => {
if (student.assessments[assessment] === 100) { if (student.assessments[assessment].result === 'Pass') {
categories[assessment].value += 1 categories[assessment].value += 1
} }
}) })
@@ -352,8 +373,8 @@ const getChartData = () => {
} }
const getChartOptions = (categories) => { const getChartOptions = (categories) => {
const courseColor = '#0F736B' const courseColor = theme.colors.green[700]
const assessmentColor = '#0070CC' const assessmentColor = theme.colors.blue[700]
const maxY = const maxY =
students.data?.length % 5 students.data?.length % 5
? students.data?.length + (5 - (students.data?.length % 5)) ? students.data?.length + (5 - (students.data?.length % 5))
@@ -362,7 +383,6 @@ const getChartOptions = (categories) => {
return { return {
chart: { chart: {
type: 'bar', type: 'bar',
height: 50,
toolbar: { toolbar: {
show: false, show: false,
}, },
@@ -370,9 +390,10 @@ const getChartOptions = (categories) => {
plotOptions: { plotOptions: {
bar: { bar: {
distributed: true, distributed: true,
borderRadius: 0, borderRadius: 3,
borderRadiusApplication: 'end',
horizontal: true, horizontal: true,
barHeight: '30%', barHeight: '40%',
}, },
}, },
colors: Object.values(categories).map((item) => colors: Object.values(categories).map((item) =>
@@ -386,7 +407,7 @@ const getChartOptions = (categories) => {
}, },
rotate: 0, rotate: 0,
formatter: function (value) { formatter: function (value) {
return value.length > 20 ? `${value.substring(0, 20)}...` : value // Trim long labels return value.length > 30 ? `${value.substring(0, 30)}...` : value
}, },
}, },
}, },
@@ -395,20 +416,27 @@ const getChartOptions = (categories) => {
min: 0, min: 0,
stepSize: 10, stepSize: 10,
tickAmount: maxY / 5, tickAmount: maxY / 5,
/* reversed: true */
}, },
} }
} }
const copyEmail = (row) => {
navigator.clipboard.writeText(row.email)
showToast(__('Success'), __('Email copied to clipboard'), 'check')
}
watch(students, () => { watch(students, () => {
if (students.data?.length) { if (students.data?.length) {
assessmentCount.value = Object.keys(students.data?.[0].assessments).length assessmentCount.value = Object.keys(students.data?.[0].assessments).length
} }
}) })
const certificationCount = createResource({
url: 'frappe.client.get_count',
params: {
doctype: 'LMS Certificate',
filters: {
batch_name: props.batch.name,
},
},
auto: true,
})
</script> </script>
<style> <style>
.apexcharts-legend { .apexcharts-legend {

View File

@@ -2,7 +2,7 @@
<div class="flex flex-col justify-between min-h-0"> <div class="flex flex-col justify-between min-h-0">
<div> <div>
<div class="flex items-center justify-between"> <div class="flex items-center justify-between">
<div class="font-semibold mb-1"> <div class="font-semibold mb-1 text-ink-gray-9">
{{ __(label) }} {{ __(label) }}
</div> </div>
<Badge <Badge
@@ -12,7 +12,7 @@
theme="orange" theme="orange"
/> />
</div> </div>
<div class="text-xs text-gray-600"> <div class="text-xs text-ink-gray-5">
{{ __(description) }} {{ __(description) }}
</div> </div>
</div> </div>

View File

@@ -1,7 +1,7 @@
<template> <template>
<div class="flex flex-col min-h-0"> <div class="flex flex-col min-h-0">
<div class="flex items-center justify-between"> <div class="flex items-center justify-between">
<div class="text-xl font-semibold mb-1"> <div class="text-xl font-semibold mb-5 text-ink-gray-9">
{{ label }} {{ label }}
</div> </div>
<Button @click="() => showCategoryForm()"> <Button @click="() => showCategoryForm()">
@@ -28,12 +28,12 @@
</div> </div>
<div class="overflow-y-scroll"> <div class="overflow-y-scroll">
<div class="text-base divide-y"> <div class="text-base divide-y space-y-2">
<FormControl <FormControl
:value="cat.category" :value="cat.category"
type="text" type="text"
v-for="cat in categories.data" v-for="cat in categories.data"
class="form-control" class=""
@change.stop="(e) => update(cat.name, e.target.value)" @change.stop="(e) => update(cat.name, e.target.value)"
/> />
</div> </div>
@@ -128,24 +128,3 @@ const update = (name, value) => {
) )
} }
</script> </script>
<style>
.form-control input {
padding: 1.25rem 0;
border-color: transparent;
background: white;
}
.form-control input:focus {
outline: transparent;
background: white;
box-shadow: none;
border-color: transparent;
}
.form-control input:hover {
outline: transparent;
background: white;
box-shadow: none;
border-color: transparent;
}
</style>

View File

@@ -0,0 +1,67 @@
<template>
<div
v-if="
certification.data &&
certification.data.membership &&
certification.data.paid_certificate &&
user.data?.is_student
"
>
<router-link
v-if="!certification.data.membership.purchased_certificate"
:to="{
name: 'Billing',
params: {
type: 'certificate',
name: courseName,
},
}"
>
<Button class="w-full">
<template #prefix>
<GraduationCap class="size-4 stroke-1.5" />
</template>
{{ __('Get Certified') }}
</Button>
</router-link>
<router-link
v-else-if="!certification.data.membership.certficate"
:to="{
name: 'CourseCertification',
params: {
courseName: courseName,
},
}"
>
<Button class="w-full">
<template #prefix>
<GraduationCap class="size-4 stroke-1.5" />
</template>
{{ __('Get Certified') }}
</Button>
</router-link>
</div>
</template>
<script setup>
import { Button, createResource } from 'frappe-ui'
import { inject } from 'vue'
import { GraduationCap } from 'lucide-vue-next'
const user = inject('$user')
const props = defineProps({
courseName: {
type: String,
required: true,
},
})
const certification = createResource({
url: 'lms.lms.api.get_certification_details',
params: {
course: props.courseName,
},
auto: true,
cache: ['certificationData', user.data?.name],
})
</script>

View File

@@ -1,6 +1,6 @@
<template> <template>
<div class="flex items-center"> <div class="flex items-center text-ink-gray-7">
<Calendar class="h-4 w-4 stroke-1.5 mr-2 text-gray-700" /> <Calendar class="h-4 w-4 stroke-1.5 mr-2" />
<span> <span>
{{ getFormattedDateRange(props.startDate, props.endDate) }} {{ getFormattedDateRange(props.startDate, props.endDate) }}
</span> </span>

View File

@@ -17,7 +17,7 @@
> >
{{ displayValue(selectedValue) }} {{ displayValue(selectedValue) }}
</span> </span>
<span class="text-base leading-5 text-gray-500" v-else> <span class="text-base leading-5 text-ink-gray-4" v-else>
{{ placeholder || '' }} {{ placeholder || '' }}
</span> </span>
</div> </div>
@@ -28,7 +28,9 @@
</template> </template>
<template #body="{ isOpen }"> <template #body="{ isOpen }">
<div v-show="isOpen"> <div v-show="isOpen">
<div class="mt-1 rounded-lg bg-white py-1 text-base shadow-2xl"> <div
class="mt-1 rounded-lg bg-surface-white py-1 text-base shadow-2xl"
>
<div class="relative px-1.5 pt-0.5"> <div class="relative px-1.5 pt-0.5">
<ComboboxInput <ComboboxInput
ref="search" ref="search"
@@ -62,7 +64,7 @@
> >
<div <div
v-if="group.group && !group.hideLabel" v-if="group.group && !group.hideLabel"
class="px-2.5 py-1.5 text-sm font-medium text-gray-500" class="px-2.5 py-1.5 text-sm font-medium text-ink-gray-4"
> >
{{ group.group }} {{ group.group }}
</div> </div>
@@ -76,7 +78,7 @@
<li <li
:class="[ :class="[
'flex items-center rounded px-2.5 py-2 text-base', 'flex items-center rounded px-2.5 py-2 text-base',
{ 'bg-gray-100': active }, { 'bg-surface-gray-2': active },
]" ]"
> >
<slot <slot
@@ -93,7 +95,7 @@
</div> </div>
<div <div
v-if="option.description" v-if="option.description"
class="text-xs text-gray-700" class="text-xs text-ink-gray-7"
v-html="option.description" v-html="option.description"
></div> ></div>
</div> </div>
@@ -103,7 +105,7 @@
</div> </div>
<li <li
v-if="groups.length == 0" v-if="groups.length == 0"
class="mt-1.5 rounded-md px-2.5 py-1.5 text-base text-gray-600" class="mt-1.5 rounded-md px-2.5 py-1.5 text-base text-ink-gray-5"
> >
No results found No results found
</li> </li>
@@ -128,7 +130,7 @@ import {
ComboboxOptions, ComboboxOptions,
ComboboxOption, ComboboxOption,
} from '@headlessui/vue' } from '@headlessui/vue'
import { Popover, Button } from 'frappe-ui' import { Popover } from 'frappe-ui'
import { ChevronDown, X } from 'lucide-vue-next' import { ChevronDown, X } from 'lucide-vue-next'
import { ref, computed, useAttrs, useSlots, watch, nextTick } from 'vue' import { ref, computed, useAttrs, useSlots, watch, nextTick } from 'vue'
@@ -243,7 +245,7 @@ watch(showOptions, (val) => {
}) })
const textColor = computed(() => { const textColor = computed(() => {
return props.disabled ? 'text-gray-600' : 'text-gray-800' return props.disabled ? 'text-ink-gray-5' : 'text-ink-gray-8'
}) })
const inputClasses = computed(() => { const inputClasses = computed(() => {
@@ -264,12 +266,14 @@ const inputClasses = computed(() => {
let variant = props.disabled ? 'disabled' : props.variant let variant = props.disabled ? 'disabled' : props.variant
let variantClasses = { let variantClasses = {
subtle: subtle:
'border border-gray-100 bg-gray-100 placeholder-gray-500 hover:border-gray-200 hover:bg-gray-200 focus:bg-white focus:border-gray-500 focus:shadow-sm focus:ring-0 focus-visible:ring-2 focus-visible:ring-gray-400', 'border border-gray-100 bg-surface-gray-2 placeholder-ink-gray-4 hover:border-outline-gray-modals hover:bg-surface-gray-3 focus:bg-surface-white focus:border-outline-gray-4 focus:shadow-sm focus:ring-0 focus-visible:ring-2 focus-visible:ring-outline-gray-3',
outline: outline:
'border border-gray-300 bg-white placeholder-gray-500 hover:border-gray-400 hover:shadow-sm focus:bg-white focus:border-gray-500 focus:shadow-sm focus:ring-0 focus-visible:ring-2 focus-visible:ring-gray-400', 'border border-outline-gray-2 bg-surface-white placeholder-ink-gray-4 hover:border-outline-gray-3 hover:shadow-sm focus:bg-surface-white focus:border-outline-gray-4 focus:shadow-sm focus:ring-0 focus-visible:ring-2 focus-visible:ring-outline-gray-3',
disabled: [ disabled: [
'border bg-gray-50 placeholder-gray-400', 'border bg-surface-menu-bar placeholder-ink-gray-3',
props.variant === 'outline' ? 'border-gray-300' : 'border-transparent', props.variant === 'outline'
? 'border-outline-gray-2'
: 'border-transparent',
], ],
}[variant] }[variant]

View File

@@ -5,7 +5,7 @@
height: height, height: height,
}" }"
> >
<span class="text-xs" v-if="label"> <span class="text-xs text-ink-gray-7" v-if="label">
{{ label }} {{ label }}
</span> </span>
<div <div
@@ -13,7 +13,7 @@
class="h-auto flex-1 overflow-hidden overscroll-none !rounded border border-outline-gray-2 bg-surface-gray-2 dark:bg-gray-900" class="h-auto flex-1 overflow-hidden overscroll-none !rounded border border-outline-gray-2 bg-surface-gray-2 dark:bg-gray-900"
/> />
<span <span
class="mt-1 text-xs text-gray-600" class="mt-1 text-xs text-ink-gray-5"
v-show="description" v-show="description"
v-html="description" v-html="description"
></span> ></span>
@@ -27,7 +27,6 @@
</div> </div>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { useDark } from '@vueuse/core'
import ace from 'ace-builds' import ace from 'ace-builds'
import 'ace-builds/src-min-noconflict/ext-searchbox' import 'ace-builds/src-min-noconflict/ext-searchbox'
import 'ace-builds/src-min-noconflict/theme-chrome' import 'ace-builds/src-min-noconflict/theme-chrome'
@@ -35,9 +34,7 @@ import 'ace-builds/src-min-noconflict/theme-twilight'
import { PropType, onMounted, ref, watch } from 'vue' import { PropType, onMounted, ref, watch } from 'vue'
import { Button } from 'frappe-ui' import { Button } from 'frappe-ui'
const isDark = useDark({ const isDark = ref(false)
attribute: 'data-theme',
})
const props = defineProps({ const props = defineProps({
modelValue: { modelValue: {
@@ -82,6 +79,7 @@ const editor = ref<HTMLElement | null>(null)
let aceEditor = null as ace.Ace.Editor | null let aceEditor = null as ace.Ace.Editor | null
onMounted(() => { onMounted(() => {
isDark.value = localStorage.getItem('theme') === 'dark'
setupEditor() setupEditor()
}) })
@@ -148,6 +146,7 @@ function resetEditor(value: string, resetHistory = false) {
value = getModelValue() value = getModelValue()
aceEditor?.setValue(value) aceEditor?.setValue(value)
aceEditor?.clearSelection() aceEditor?.clearSelection()
console.log(isDark.value)
aceEditor?.setTheme(isDark.value ? 'ace/theme/twilight' : 'ace/theme/chrome') aceEditor?.setTheme(isDark.value ? 'ace/theme/twilight' : 'ace/theme/chrome')
props.autofocus && aceEditor?.focus() props.autofocus && aceEditor?.focus()
if (resetHistory) { if (resetHistory) {
@@ -156,6 +155,7 @@ function resetEditor(value: string, resetHistory = false) {
} }
watch(isDark, () => { watch(isDark, () => {
console.log(isDark.value)
aceEditor?.setTheme(isDark.value ? 'ace/theme/twilight' : 'ace/theme/chrome') aceEditor?.setTheme(isDark.value ? 'ace/theme/twilight' : 'ace/theme/chrome')
}) })
@@ -175,30 +175,3 @@ watch(
defineExpose({ resetEditor }) defineExpose({ resetEditor })
</script> </script>
<style scoped>
.editor .ace_editor {
height: 100%;
width: 100%;
border-radius: 5px;
overscroll-behavior: none;
}
.editor :deep(.ace_scrollbar-h) {
display: none;
}
.editor :deep(.ace_search) {
@apply dark:bg-gray-800 dark:text-gray-200;
@apply dark:border-gray-800;
}
.editor :deep(.ace_searchbtn) {
@apply dark:bg-gray-800 dark:text-gray-200;
@apply dark:border-gray-800;
}
.editor :deep(.ace_button) {
@apply dark:bg-gray-800 dark:text-gray-200;
}
.editor :deep(.ace_search_field) {
@apply dark:bg-gray-900 dark:text-gray-200;
@apply dark:border-gray-800;
}
</style>

View File

@@ -1,6 +1,6 @@
<template> <template>
<div class="space-y-1.5"> <div class="space-y-1.5">
<label class="block text-xs text-gray-600"> <label class="block text-xs text-ink-gray-5">
{{ label }} {{ label }}
</label> </label>
<div class="w-full"> <div class="w-full">
@@ -8,22 +8,22 @@
<template #target="{ togglePopover }"> <template #target="{ togglePopover }">
<button <button
@click="openPopover(togglePopover)" @click="openPopover(togglePopover)"
class="flex w-full items-center space-x-2 focus:outline-none bg-gray-100 rounded h-7 py-1.5 px-2 hover:bg-gray-200 focus:bg-white border border-gray-100 hover:border-gray-200 focus:border-gray-500" class="flex w-full items-center space-x-2 focus:outline-none bg-surface-gray-2 rounded h-7 py-1.5 px-2 hover:bg-surface-gray-3 focus:bg-surface-white border border-gray-100 hover:border-outline-gray-modals focus:border-outline-gray-4"
> >
<component <component
v-if="selectedIcon" v-if="selectedIcon"
class="w-4 h-4 text-gray-700 stroke-1.5" class="w-4 h-4 text-ink-gray-7 stroke-1.5"
:is="icons[selectedIcon]" :is="icons[selectedIcon]"
/> />
<component <component
v-else v-else
class="w-4 h-4 text-gray-700 stroke-1.5" class="w-4 h-4 text-ink-gray-7 stroke-1.5"
:is="icons.Folder" :is="icons.Folder"
/> />
<span v-if="selectedIcon"> <span v-if="selectedIcon">
{{ selectedIcon }} {{ selectedIcon }}
</span> </span>
<span v-else class="text-gray-600"> <span v-else class="text-ink-gray-5">
{{ __('Choose an icon') }} {{ __('Choose an icon') }}
</span> </span>
</button> </button>
@@ -40,7 +40,7 @@
<div v-for="(iconComponent, iconName) in filteredIcons"> <div v-for="(iconComponent, iconName) in filteredIcons">
<component <component
:is="iconComponent" :is="iconComponent"
class="h-4 w-4 stroke-1.5 text-gray-700 cursor-pointer" class="h-4 w-4 stroke-1.5 text-ink-gray-7 cursor-pointer"
@click="setIcon(iconName, close)" @click="setIcon(iconName, close)"
/> />
</div> </div>

View File

@@ -2,7 +2,7 @@
<div class="space-y-1.5"> <div class="space-y-1.5">
<label class="block" :class="labelClasses" v-if="attrs.label"> <label class="block" :class="labelClasses" v-if="attrs.label">
{{ attrs.label }} {{ attrs.label }}
<span class="text-red-500" v-if="attrs.required">*</span> <span class="text-ink-red-3" v-if="attrs.required">*</span>
</label> </label>
<Autocomplete <Autocomplete
ref="autocomplete" ref="autocomplete"
@@ -56,7 +56,7 @@
</div> </div>
</template> </template>
</Autocomplete> </Autocomplete>
<p v-if="description" class="text-sm text-gray-600">{{ description }}</p> <p v-if="description" class="text-sm text-ink-gray-5">{{ description }}</p>
</div> </div>
</template> </template>
@@ -163,7 +163,7 @@ const labelClasses = computed(() => {
sm: 'text-xs', sm: 'text-xs',
md: 'text-base', md: 'text-base',
}[attrs.size || 'sm'], }[attrs.size || 'sm'],
'text-gray-600', 'text-ink-gray-5',
] ]
}) })
</script> </script>

View File

@@ -2,7 +2,7 @@
<div> <div>
<label class="block mb-1" :class="labelClasses" v-if="label"> <label class="block mb-1" :class="labelClasses" v-if="label">
{{ label }} {{ label }}
<span class="text-red-500" v-if="required">*</span> <span class="text-ink-red-3" v-if="required">*</span>
</label> </label>
<div class="grid grid-cols-3 gap-1"> <div class="grid grid-cols-3 gap-1">
<Button <Button
@@ -41,7 +41,9 @@
</template> </template>
<template #body="{ isOpen }"> <template #body="{ isOpen }">
<div v-show="isOpen"> <div v-show="isOpen">
<div class="mt-1 rounded-lg bg-white py-1 text-base shadow-2xl"> <div
class="mt-1 rounded-lg bg-surface-white py-1 text-base shadow-2xl"
>
<ComboboxOptions <ComboboxOptions
class="my-1 max-h-[12rem] overflow-y-auto px-1.5" class="my-1 max-h-[12rem] overflow-y-auto px-1.5"
static static
@@ -55,14 +57,14 @@
<li <li
:class="[ :class="[
'flex cursor-pointer items-center rounded px-2 py-1 text-base', 'flex cursor-pointer items-center rounded px-2 py-1 text-base',
{ 'bg-gray-100': active }, { 'bg-surface-gray-2': active },
]" ]"
> >
<div class="flex flex-col gap-1 p-1"> <div class="flex flex-col gap-1 p-1">
<div class="text-base font-medium"> <div class="text-base font-medium">
{{ option.description }} {{ option.description }}
</div> </div>
<div class="text-sm text-gray-600"> <div class="text-sm text-ink-gray-5">
{{ option.value }} {{ option.value }}
</div> </div>
</div> </div>
@@ -240,7 +242,7 @@ const labelClasses = computed(() => {
sm: 'text-xs', sm: 'text-xs',
md: 'text-base', md: 'text-base',
}[props.size || 'sm'], }[props.size || 'sm'],
'text-gray-600', 'text-ink-gray-5',
] ]
}) })
</script> </script>

View File

@@ -1,6 +1,6 @@
<template> <template>
<div class="space-y-1"> <div class="space-y-1">
<label class="block text-xs text-gray-600" v-if="props.label"> <label class="block text-xs text-ink-gray-5" v-if="props.label">
{{ props.label }} {{ props.label }}
</label> </label>
<div class="flex text-center"> <div class="flex text-center">

View File

@@ -1,7 +1,7 @@
<template> <template>
<div <div
v-if="course.title" v-if="course.title"
class="flex flex-col h-full rounded-md shadow-md text-base overflow-auto" class="flex flex-col h-full rounded-md border-2 overflow-auto"
style="min-height: 350px" style="min-height: 350px"
> >
<div <div
@@ -15,14 +15,13 @@
<Badge v-if="course.featured" variant="subtle" theme="green" size="md"> <Badge v-if="course.featured" variant="subtle" theme="green" size="md">
{{ __('Featured') }} {{ __('Featured') }}
</Badge> </Badge>
<Badge <div
variant="subtle" v-if="course.tags"
theme="gray" v-for="tag in course.tags?.split(', ')"
size="md" class="text-xs bg-white text-gray-800 px-2 py-0.5 rounded-md"
v-for="tag in course.tags"
> >
{{ tag }} {{ tag }}
</Badge> </div>
</div> </div>
<div v-if="!course.image" class="image-placeholder"> <div v-if="!course.image" class="image-placeholder">
{{ course.title[0] }} {{ course.title[0] }}
@@ -32,8 +31,8 @@
<div class="flex items-center justify-between mb-2"> <div class="flex items-center justify-between mb-2">
<div v-if="course.lessons"> <div v-if="course.lessons">
<Tooltip :text="__('Lessons')"> <Tooltip :text="__('Lessons')">
<span class="flex items-center"> <span class="flex items-center text-ink-gray-7">
<BookOpen class="h-4 w-4 stroke-1.5 text-gray-700 mr-1" /> <BookOpen class="h-4 w-4 stroke-1.5 mr-1" />
{{ course.lessons }} {{ course.lessons }}
</span> </span>
</Tooltip> </Tooltip>
@@ -41,8 +40,8 @@
<div v-if="course.enrollments"> <div v-if="course.enrollments">
<Tooltip :text="__('Enrolled Students')"> <Tooltip :text="__('Enrolled Students')">
<span class="flex items-center"> <span class="flex items-center text-ink-gray-7">
<Users class="h-4 w-4 stroke-1.5 text-gray-700 mr-1" /> <Users class="h-4 w-4 stroke-1. mr-1" />
{{ course.enrollments }} {{ course.enrollments }}
</span> </span>
</Tooltip> </Tooltip>
@@ -50,8 +49,8 @@
<div v-if="course.rating"> <div v-if="course.rating">
<Tooltip :text="__('Average Rating')"> <Tooltip :text="__('Average Rating')">
<span class="flex items-center"> <span class="flex items-center text-ink-gray-7">
<Star class="h-4 w-4 stroke-1.5 text-gray-700 mr-1" /> <Star class="h-4 w-4 stroke-1.5 mr-1" />
{{ course.rating }} {{ course.rating }}
</span> </span>
</Tooltip> </Tooltip>
@@ -68,11 +67,11 @@
</div> </div>
</div> </div>
<div class="text-xl font-semibold leading-6"> <div class="text-xl font-semibold leading-6 text-ink-gray-9">
{{ course.title }} {{ course.title }}
</div> </div>
<div class="short-introduction text-gray-700 text-sm"> <div class="short-introduction text-ink-gray-7 text-sm">
{{ course.short_introduction }} {{ course.short_introduction }}
</div> </div>
@@ -81,7 +80,10 @@
:progress="course.membership.progress" :progress="course.membership.progress"
/> />
<div v-if="user && course.membership" class="text-sm mb-4"> <div
v-if="user && course.membership"
class="text-sm text-ink-gray-7 mt-2 mb-4"
>
{{ Math.ceil(course.membership.progress) }}% completed {{ Math.ceil(course.membership.progress) }}% completed
</div> </div>
@@ -99,9 +101,15 @@
<CourseInstructors :instructors="course.instructors" /> <CourseInstructors :instructors="course.instructors" />
</div> </div>
<div class="font-semibold"> <div v-if="course.paid_course" class="font-semibold">
{{ course.price }} {{ course.price }}
</div> </div>
<div
v-if="course.paid_certificate || course.enable_certification"
class="text-xs text-ink-blue-3 bg-surface-blue-1 py-0.5 px-1 rounded-md"
>
{{ __('Certification') }}
</div>
</div> </div>
</div> </div>
</div> </div>

View File

@@ -1,35 +1,37 @@
<template> <template>
<div class="shadow rounded-md min-w-80"> <div class="border-2 rounded-md min-w-80">
<iframe <iframe
v-if="course.data.video_link" v-if="course.data.video_link"
:src="video_link" :src="video_link"
class="rounded-t-md min-h-56 w-full" class="rounded-t-md min-h-56 w-full"
/> />
<div class="p-5"> <div class="p-5">
<div v-if="course.data.price" class="text-2xl font-semibold mb-3"> <div v-if="course.data.paid_course" class="text-2xl font-semibold mb-3">
{{ course.data.price }} {{ course.data.price }}
</div> </div>
<router-link <div v-if="course.data.membership" class="space-y-2">
v-if="course.data.membership" <router-link
:to="{ :to="{
name: 'Lesson', name: 'Lesson',
params: { params: {
courseName: course.name, courseName: course.name,
chapterNumber: course.data.current_lesson chapterNumber: course.data.current_lesson
? course.data.current_lesson.split('-')[0] ? course.data.current_lesson.split('-')[0]
: 1, : 1,
lessonNumber: course.data.current_lesson lessonNumber: course.data.current_lesson
? course.data.current_lesson.split('-')[1] ? course.data.current_lesson.split('-')[1]
: 1, : 1,
}, },
}" }"
> >
<Button variant="solid" size="md" class="w-full"> <Button variant="solid" size="md" class="w-full">
<span> <span>
{{ __('Continue Learning') }} {{ __('Continue Learning') }}
</span> </span>
</Button> </Button>
</router-link> </router-link>
<CertificationLinks :courseName="course.data.name" />
</div>
<router-link <router-link
v-else-if="course.data.paid_course" v-else-if="course.data.paid_course"
:to="{ :to="{
@@ -48,7 +50,7 @@
</router-link> </router-link>
<div <div
v-else-if="course.data.disable_self_learning" v-else-if="course.data.disable_self_learning"
class="bg-blue-100 text-blue-900 text-sm rounded-md py-1 px-3" class="bg-surface-blue-2 text-blue-900 text-sm rounded-md py-1 px-3"
> >
{{ __('Contact the Administrator to enroll for this course.') }} {{ __('Contact the Administrator to enroll for this course.') }}
</div> </div>
@@ -88,39 +90,61 @@
</Button> </Button>
</router-link> </router-link>
<div class="space-y-4"> <div class="space-y-4">
<div class="mt-8 font-medium"> <div class="mt-8 font-medium text-ink-gray-9">
{{ __('This course has:') }} {{ __('This course has:') }}
</div> </div>
<div class="flex items-center"> <div class="flex items-center text-ink-gray-9">
<BookOpen class="h-4 w-4 stroke-1.5 text-gray-600" /> <BookOpen class="h-4 w-4 stroke-1.5" />
<span class="ml-2"> <span class="ml-2">
{{ course.data.lessons }} {{ __('Lessons') }} {{ course.data.lessons }} {{ __('Lessons') }}
</span> </span>
</div> </div>
<div class="flex items-center"> <div class="flex items-center text-ink-gray-9">
<Users class="h-4 w-4 stroke-1.5 text-gray-600" /> <Users class="h-4 w-4 stroke-1.5" />
<span class="ml-2"> <span class="ml-2">
{{ formatAmount(course.data.enrollments) }} {{ formatAmount(course.data.enrollments) }}
{{ __('Enrolled Students') }} {{ __('Enrolled Students') }}
</span> </span>
</div> </div>
<div v-if="parseInt(course.data.rating) > 0" class="flex items-center"> <div
v-if="parseInt(course.data.rating) > 0"
class="flex items-center text-ink-gray-9"
>
<Star class="h-4 w-4 stroke-1.5 fill-orange-500 text-gray-50" /> <Star class="h-4 w-4 stroke-1.5 fill-orange-500 text-gray-50" />
<span class="ml-2"> <span class="ml-2">
{{ course.data.rating }} {{ __('Rating') }} {{ course.data.rating }} {{ __('Rating') }}
</span> </span>
</div> </div>
<div
v-if="course.data.enable_certification"
class="flex items-center font-semibold text-ink-gray-9"
>
<GraduationCap class="h-4 w-4 stroke-2" />
<span class="ml-2">
{{ __('Certificate of Completion') }}
</span>
</div>
<div
v-if="course.data.paid_certificate"
class="flex items-center font-semibold text-ink-gray-9"
>
<GraduationCap class="h-4 w-4 stroke-2" />
<span class="ml-2">
{{ __('Paid Certificate after Evaluation') }}
</span>
</div>
</div> </div>
</div> </div>
</div> </div>
</template> </template>
<script setup> <script setup>
import { BookOpen, Users, Star } from 'lucide-vue-next' import { BookOpen, Users, Star, GraduationCap } from 'lucide-vue-next'
import { computed, inject } from 'vue' import { computed, inject } from 'vue'
import { Button, createResource } from 'frappe-ui' import { Button, createResource, Tooltip } from 'frappe-ui'
import { showToast, formatAmount } from '@/utils/' import { showToast, formatAmount } from '@/utils/'
import { capture } from '@/telemetry' import { capture } from '@/telemetry'
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
import CertificationLinks from '@/components/CertificationLinks.vue'
const router = useRouter() const router = useRouter()
const user = inject('$user') const user = inject('$user')

View File

@@ -1,44 +1,46 @@
<template> <template>
<span v-if="instructors?.length == 1"> <div class="text-ink-gray-7">
<router-link <span v-if="instructors?.length == 1">
:to="{ <router-link
name: 'Profile', :to="{
params: { username: instructors[0].username }, name: 'Profile',
}" params: { username: instructors[0].username },
> }"
{{ instructors[0].full_name }} >
</router-link> {{ instructors[0].full_name }}
</span> </router-link>
<span v-if="instructors?.length == 2"> </span>
<router-link <span v-if="instructors?.length == 2">
:to="{ <router-link
name: 'Profile', :to="{
params: { username: instructors[0].username }, name: 'Profile',
}" params: { username: instructors[0].username },
> }"
{{ instructors[0].first_name }} >
</router-link> {{ instructors[0].first_name }}
and </router-link>
<router-link and
:to="{ <router-link
name: 'Profile', :to="{
params: { username: instructors[1].username }, name: 'Profile',
}" params: { username: instructors[1].username },
> }"
{{ instructors[1].first_name }} >
</router-link> {{ instructors[1].first_name }}
</span> </router-link>
<span v-if="instructors?.length > 2"> </span>
<router-link <span v-if="instructors?.length > 2">
:to="{ <router-link
name: 'Profile', :to="{
params: { username: instructors[0].username }, name: 'Profile',
}" params: { username: instructors[0].username },
> }"
{{ instructors[0].first_name }} >
</router-link> {{ instructors[0].first_name }}
and {{ instructors?.length - 1 }} others </router-link>
</span> and {{ instructors?.length - 1 }} others
</span>
</div>
</template> </template>
<script setup> <script setup>
const props = defineProps({ const props = defineProps({

View File

@@ -4,7 +4,7 @@
v-if="title && (outline.data?.length || allowEdit)" v-if="title && (outline.data?.length || allowEdit)"
class="grid grid-cols-[70%,30%] mb-4 px-2" class="grid grid-cols-[70%,30%] mb-4 px-2"
> >
<div class="font-semibold text-lg leading-5"> <div class="font-semibold text-lg leading-5 text-ink-gray-9">
{{ __(title) }} {{ __(title) }}
</div> </div>
<Button size="sm" v-if="allowEdit" @click="openChapterModal()"> <Button size="sm" v-if="allowEdit" @click="openChapterModal()">
@@ -16,7 +16,7 @@
</div> </div>
<div <div
:class="{ :class="{
'shadow rounded-md py-2 px-2': showOutline && outline.data?.length, 'border-2 rounded-md py-2 px-2': showOutline && outline.data?.length,
}" }"
> >
<Disclosure <Disclosure
@@ -33,10 +33,10 @@
hidden: chapter.is_scorm_package, hidden: chapter.is_scorm_package,
open: index == 1, open: index == 1,
}" }"
class="h-4 w-4 text-gray-900 stroke-1" class="h-4 w-4 text-ink-gray-9 stroke-1"
/> />
<div <div
class="text-base text-left font-medium leading-5 ml-2" class="text-base text-left text-ink-gray-9 font-medium leading-5 ml-2"
@click="redirectToChapter(chapter)" @click="redirectToChapter(chapter)"
> >
{{ chapter.title }} {{ chapter.title }}
@@ -46,14 +46,14 @@
<FilePenLine <FilePenLine
v-if="allowEdit" v-if="allowEdit"
@click.prevent="openChapterModal(chapter)" @click.prevent="openChapterModal(chapter)"
class="h-4 w-4 text-gray-900 invisible group-hover:visible" class="h-4 w-4 text-ink-gray-9 invisible group-hover:visible"
/> />
</Tooltip> </Tooltip>
<Tooltip :text="__('Delete Chapter')" placement="bottom"> <Tooltip :text="__('Delete Chapter')" placement="bottom">
<Trash2 <Trash2
v-if="allowEdit" v-if="allowEdit"
@click.prevent="trashChapter(chapter.name)" @click.prevent="trashChapter(chapter.name)"
class="h-4 w-4 text-red-500 invisible group-hover:visible" class="h-4 w-4 text-ink-red-3 invisible group-hover:visible"
/> />
</Tooltip> </Tooltip>
</div> </div>
@@ -69,7 +69,12 @@
:data-chapter="chapter.name" :data-chapter="chapter.name"
> >
<template #item="{ element: lesson }"> <template #item="{ element: lesson }">
<div class="outline-lesson pl-8 py-2 pr-4"> <div
class="outline-lesson pl-8 py-2 pr-4 text-ink-gray-9"
:class="
isActiveLesson(lesson.number) ? 'bg-surface-selected' : ''
"
>
<router-link <router-link
:to="{ :to="{
name: allowEdit ? 'LessonForm' : 'Lesson', name: allowEdit ? 'LessonForm' : 'Lesson',
@@ -83,21 +88,21 @@
<div class="flex items-center text-sm leading-5 group"> <div class="flex items-center text-sm leading-5 group">
<MonitorPlay <MonitorPlay
v-if="lesson.icon === 'icon-youtube'" v-if="lesson.icon === 'icon-youtube'"
class="h-4 w-4 text-gray-900 stroke-1 mr-2" class="h-4 w-4 stroke-1 mr-2"
/> />
<HelpCircle <HelpCircle
v-else-if="lesson.icon === 'icon-quiz'" v-else-if="lesson.icon === 'icon-quiz'"
class="h-4 w-4 text-gray-900 stroke-1 mr-2" class="h-4 w-4 stroke-1 mr-2"
/> />
<FileText <FileText
v-else-if="lesson.icon === 'icon-list'" v-else-if="lesson.icon === 'icon-list'"
class="h-4 w-4 text-gray-900 stroke-1 mr-2" class="h-4 w-4 text-ink-gray-9 stroke-1 mr-2"
/> />
{{ lesson.title }} {{ lesson.title }}
<Trash2 <Trash2
v-if="allowEdit" v-if="allowEdit"
@click.prevent="trashLesson(lesson.name, chapter.name)" @click.prevent="trashLesson(lesson.name, chapter.name)"
class="h-4 w-4 text-red-500 ml-auto invisible group-hover:visible" class="h-4 w-4 text-ink-red-3 ml-auto invisible group-hover:visible"
/> />
<Check <Check
v-if="lesson.is_complete" v-if="lesson.is_complete"
@@ -323,9 +328,11 @@ const redirectToChapter = (chapter) => {
}, },
}) })
} }
</script>
<style> const isActiveLesson = (lessonNumber) => {
.outline-lesson:has(.router-link-active) { return (
background-color: theme('colors.gray.100'); route.params.chapterNumber == lessonNumber.split('.')[0] &&
route.params.lessonNumber == lessonNumber.split('.')[1]
)
} }
</style> </script>

View File

@@ -7,7 +7,7 @@
> >
{{ __('Write a Review') }} {{ __('Write a Review') }}
</Button> </Button>
<div class="flex items-center font-semibold text-2xl"> <div class="flex items-center font-semibold text-2xl text-ink-gray-9">
{{ __('Student Reviews') }} {{ __('Student Reviews') }}
</div> </div>
<div class="grid gap-8 mt-10"> <div class="grid gap-8 mt-10">
@@ -28,17 +28,17 @@
params: { username: review.owner_details.username }, params: { username: review.owner_details.username },
}" }"
> >
<span class="text-lg font-medium mr-4"> <span class="text-lg font-medium mr-4 text-ink-gray-7">
{{ review.owner_details.full_name }} {{ review.owner_details.full_name }}
</span> </span>
</router-link> </router-link>
<span> <span class="text-ink-gray-7">
{{ review.creation }} {{ review.creation }}
</span> </span>
<div class="flex mt-2"> <div class="flex mt-2">
<Star <Star
v-for="index in 5" v-for="index in 5"
class="h-5 w-5 text-gray-100 bg-gray-200 rounded-sm mr-2" class="h-5 w-5 text-ink-gray-1 rounded-sm mr-2"
:class=" :class="
index <= Math.ceil(review.rating) index <= Math.ceil(review.rating)
? 'fill-orange-500' ? 'fill-orange-500'
@@ -48,7 +48,7 @@
</div> </div>
</div> </div>
</div> </div>
<div v-if="review.review" class="mt-4 leading-5"> <div v-if="review.review" class="mt-4 leading-5 text-ink-gray-7">
{{ review.review }} {{ review.review }}
</div> </div>
</div> </div>

View File

@@ -6,7 +6,7 @@
<div v-if="course.chapters.length"> <div v-if="course.chapters.length">
{{ course.chapters }} {{ course.chapters }}
</div> </div>
<div v-else class="border bg-white rounded-md p-5 text-center mt-4"> <div v-else class="border bg-surface-white rounded-md p-5 text-center mt-4">
<div> <div>
{{ {{
__( __(

View File

@@ -1,7 +1,7 @@
<template> <template>
<div class="relative flex h-full flex-col"> <div class="relative flex h-full flex-col">
<div class="h-full flex-1"> <div class="h-full flex-1">
<div class="flex h-screen text-base"> <div class="flex h-screen text-base bg-surface-white">
<div <div
class="relative block min-h-0 flex-shrink-0 overflow-hidden hover:overflow-auto" class="relative block min-h-0 flex-shrink-0 overflow-hidden hover:overflow-auto"
> >

View File

@@ -3,10 +3,10 @@
<div v-if="!singleThread" class="flex items-center mb-5"> <div v-if="!singleThread" class="flex items-center mb-5">
<Button variant="outline" @click="showTopics = true"> <Button variant="outline" @click="showTopics = true">
<template #icon> <template #icon>
<ChevronLeft class="w-5 h-5 stroke-1.5 text-gray-700" /> <ChevronLeft class="w-5 h-5 stroke-1.5 text-ink-gray-7" />
</template> </template>
</Button> </Button>
<span class="text-lg font-semibold ml-2"> <span class="text-lg font-semibold ml-2 text-ink-gray-9">
{{ topic.title }} {{ topic.title }}
</span> </span>
</div> </div>
@@ -17,7 +17,7 @@
:class="{ 'border-b': index + 1 != replies.data.length }" :class="{ 'border-b': index + 1 != replies.data.length }"
> >
<div class="flex items-center justify-between mb-2"> <div class="flex items-center justify-between mb-2">
<div class="flex items-center"> <div class="flex items-center text-ink-gray-5">
<UserAvatar :user="reply.user" class="mr-2" /> <UserAvatar :user="reply.user" class="mr-2" />
<span> <span>
{{ reply.user.full_name }} {{ reply.user.full_name }}
@@ -63,7 +63,7 @@
:fixedMenu="reply.editable || false" :fixedMenu="reply.editable || false"
:editorClass=" :editorClass="
reply.editable reply.editable
? 'ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-gray-300 prose-th:border-gray-300 prose-td:relative prose-th:relative prose-th:bg-gray-100 prose-sm max-w-none' ? 'ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-outline-gray-2 prose-th:border-outline-gray-2 prose-td:relative prose-th:relative prose-th:bg-surface-gray-2 prose-sm max-w-none'
: 'prose-sm' : 'prose-sm'
" "
/> />
@@ -71,13 +71,14 @@
</div> </div>
<TextEditor <TextEditor
v-if="renderEditor"
class="mt-5" class="mt-5"
:content="newReply" :content="newReply"
:mentions="mentionUsers" :mentions="mentionUsers"
@change="(val) => (newReply = val)" @change="(val) => (newReply = val)"
placeholder="Type your reply here..." placeholder="Type your reply here..."
:fixedMenu="true" :fixedMenu="true"
editorClass="ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-gray-300 prose-th:border-gray-300 prose-td:relative prose-th:relative prose-th:bg-gray-100 prose-sm max-w-none border border-gray-300 rounded-b-md min-h-[7rem] py-1 px-2" editorClass="ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-outline-gray-2 prose-th:border-outline-gray-2 prose-td:relative prose-th:relative prose-th:bg-surface-gray-2 prose-sm max-w-none border border-outline-gray-2 rounded-b-md min-h-[7rem] py-1 px-2"
/> />
<div class="flex justify-between mt-2"> <div class="flex justify-between mt-2">
<span> </span> <span> </span>
@@ -94,7 +95,7 @@ import { createResource, TextEditor, Button, Dropdown } from 'frappe-ui'
import { timeAgo } from '../utils' import { timeAgo } from '../utils'
import UserAvatar from '@/components/UserAvatar.vue' import UserAvatar from '@/components/UserAvatar.vue'
import { ChevronLeft, MoreHorizontal } from 'lucide-vue-next' import { ChevronLeft, MoreHorizontal } from 'lucide-vue-next'
import { ref, inject, onMounted, computed } from 'vue' import { ref, inject, onMounted } from 'vue'
import { createToast } from '../utils' import { createToast } from '../utils'
const showTopics = defineModel('showTopics') const showTopics = defineModel('showTopics')
@@ -102,6 +103,8 @@ const newReply = ref('')
const socket = inject('$socket') const socket = inject('$socket')
const user = inject('$user') const user = inject('$user')
const allUsers = inject('$allUsers') const allUsers = inject('$allUsers')
const mentionUsers = ref([])
const renderEditor = ref(false)
const props = defineProps({ const props = defineProps({
topic: { topic: {
@@ -124,6 +127,7 @@ onMounted(() => {
socket.on('delete_message', (data) => { socket.on('delete_message', (data) => {
replies.reload() replies.reload()
}) })
fetchMentionUsers()
}) })
const replies = createResource({ const replies = createResource({
@@ -150,15 +154,26 @@ const newReplyResource = createResource({
}, },
}) })
const mentionUsers = computed(() => { const fetchMentionUsers = () => {
let users = Object.values(allUsers.data).map((user) => { if (user.data?.is_student) {
return { renderEditor.value = true
value: user.name, } else {
label: user.full_name, allUsers.reload(
} {},
}) {
return users onSuccess(data) {
}) mentionUsers.value = Object.values(data).map((user) => {
return {
value: user.name,
label: user.full_name,
}
})
renderEditor.value = true
},
}
)
}
}
const postReply = () => { const postReply = () => {
newReplyResource.submit( newReplyResource.submit(
@@ -178,7 +193,7 @@ const postReply = () => {
title: 'Error', title: 'Error',
text: err.messages?.[0] || err, text: err.messages?.[0] || err,
icon: 'x', icon: 'x',
iconClasses: 'bg-red-600 text-white rounded-md p-px', iconClasses: 'bg-surface-red-5 text-ink-white rounded-md p-px',
position: 'top-center', position: 'top-center',
timeout: 10, timeout: 10,
}) })

View File

@@ -3,7 +3,7 @@
<Button v-if="!singleThread" class="float-right" @click="openTopicModal()"> <Button v-if="!singleThread" class="float-right" @click="openTopicModal()">
{{ __('New {0}').format(singularize(title)) }} {{ __('New {0}').format(singularize(title)) }}
</Button> </Button>
<div class="text-xl font-semibold"> <div class="text-xl font-semibold text-ink-gray-9">
{{ __(title) }} {{ __(title) }}
</div> </div>
</div> </div>
@@ -16,10 +16,10 @@
> >
<UserAvatar :user="topic.user" size="2xl" class="mr-4" /> <UserAvatar :user="topic.user" size="2xl" class="mr-4" />
<div> <div>
<div class="text-lg font-semibold mb-1"> <div class="text-lg font-semibold mb-1 text-ink-gray-7">
{{ topic.title }} {{ topic.title }}
</div> </div>
<div class="flex items-center"> <div class="flex items-center text-ink-gray-5">
<span> <span>
{{ topic.user.full_name }} {{ topic.user.full_name }}
</span> </span>
@@ -44,12 +44,12 @@
v-else v-else
class="flex flex-col items-center justify-center border-2 border-dashed mt-5 py-8 rounded-md" class="flex flex-col items-center justify-center border-2 border-dashed mt-5 py-8 rounded-md"
> >
<MessageSquareText class="w-7 h-7 text-gray-500 stroke-1.5 mr-2" /> <MessageSquareText class="w-7 h-7 text-ink-gray-4 stroke-1.5 mr-2" />
<div class=""> <div class="">
<div v-if="emptyStateTitle" class="font-medium mb-2"> <div v-if="emptyStateTitle" class="font-medium mb-2">
{{ __(emptyStateTitle) }} {{ __(emptyStateTitle) }}
</div> </div>
<div class="text-gray-600"> <div class="text-ink-gray-5">
{{ __(emptyStateText) }} {{ __(emptyStateText) }}
</div> </div>
</div> </div>

View File

@@ -0,0 +1,23 @@
<template>
<svg
width="32"
height="32"
viewBox="0 0 32 32"
fill="none"
xmlns="http://www.w3.org/2000/svg"
>
<rect
x="0.75"
y="0.75"
width="30.5"
height="30.5"
rx="6.25"
stroke="currentColor"
stroke-width="1.5"
/>
<path
d="M24.5011 14.1124C23.3954 12.4873 21.532 11.5477 19.594 11.6747C18.7616 10.1384 17.2211 9.12267 15.4198 9.0084C14.1651 8.93222 12.8979 9.3766 11.9165 10.24C11.2456 10.8367 10.7611 11.5223 10.463 12.2968C10.289 12.7539 9.89151 13.0459 9.46912 13.0459H6.5V15.5852H9.46912C10.9226 15.5852 12.2271 14.6584 12.7737 13.2237C12.9227 12.8301 13.1712 12.4873 13.5439 12.1571C14.0284 11.7255 14.662 11.4969 15.2583 11.535C16.1528 11.5985 16.7863 12.0175 17.1839 12.538C17.6063 13.0205 17.8423 13.7696 17.979 14.5187C18.774 14.2902 19.6437 14.0997 20.476 14.2394C21.1593 14.3536 21.7929 14.7218 22.2525 15.2678C22.327 15.3567 22.4016 15.4456 22.4637 15.5471C23.06 16.4232 23.1718 17.5024 22.7743 18.5689C22.414 19.5592 21.0847 20.4607 19.9791 20.4607H11.3326C10.1524 20.4607 9.18339 19.5592 9.03432 18.4038H6.54969C6.71119 20.9686 8.78585 23 11.3326 23H19.9915C22.1283 23 24.3769 21.451 25.1098 19.4704C25.7931 17.6167 25.5695 15.6614 24.5135 14.0997L24.5011 14.1124Z"
fill="currentColor"
/>
</svg>
</template>

View File

@@ -1,71 +1,41 @@
<template> <template>
<div class="flex rounded p-1 lg:px-2 lg:py-2.5 hover:bg-gray-100"> <div class="flex space-x-4 border rounded-md p-2">
<div class="flex w-3/5 md:w-2/5"> <img :src="job.company_logo" class="size-10 rounded-full object-contain" />
<img <div class="flex flex-col space-y-2 flex-1">
:src="job.company_logo" <div class="flex items-center justify-between">
class="w-12 h-12 rounded-lg object-contain mr-4" <span class="font-semibold text-ink-gray-9">
:alt="job.company_name"
/>
<div>
<div class="font-medium mb-1">
{{ job.job_title }} {{ job.job_title }}
</div>
<div class="text-gray-700">
{{ job.company_name }}
</div>
</div>
</div>
<div class="flex justify-end w-1/5 text-gray-700">
{{ job.location.replace(',', '').split(' ')[0] }}
</div>
<div
class="flex justify-end w-1/5 text-gray-700 text-right hidden md:block"
>
{{ job.type }}
</div>
<div class="flex justify-end w-1/5 text-sm text-gray-700 text-right">
{{ dayjs(job.creation).format('DD MMM YYYY') }}
</div>
</div>
<!-- <div class="flex flex-col shadow rounded-md p-4 h-full">
<div class="flex justify-between">
<div>
<div class="text-xl font-semibold mb-2">
{{ job.job_title }}
</div>
<div>
{{ __("posted by") }}
<span class="font-medium">
{{ job.company_name }}
</span>
</div>
</div>
<img
:src="job.company_logo"
class="w-12 h-12 rounded-lg object-contain"
/>
</div>
<div class="flex justify-between mt-8">
<div class="flex items-center">
<Badge :label="job.type" theme="green" size="lg" class="mr-4"/>
<Badge :label="job.location" theme="gray" size="lg">
<template #prefix>
<MapPin class="h-4 w-4 stroke-1.5" />
</template>
</Badge>
</div>
<div>
<span class="font-medium">
{{ dayjs(job.creation).format('DD MMM YYYY') }}
</span> </span>
</div> </div>
<div class="flex items-center space-x-2 text-ink-gray-5">
<Building2 class="w-4 h-4 stroke-1.5" />
<span>
{{ job.company_name }}
</span>
</div>
<div class="flex items-center space-x-2 text-ink-gray-5">
<MapPin class="w-4 h-4 stroke-1.5" />
<span>
{{ job.location }}
</span>
</div>
<div class="flex items-center space-x-2 text-ink-gray-5">
<Shapes class="w-4 h-4 stroke-1.5" />
<span>
{{ job.type }}
</span>
</div>
<div class="flex items-center space-x-2 text-ink-gray-5">
<Calendar class="w-4 h-4 stroke-1.5" />
<span> {{ __('posted') }} {{ dayjs(job.creation).fromNow() }} </span>
</div>
</div> </div>
</div> --> </div>
</template> </template>
<script setup> <script setup>
import { MapPin } from 'lucide-vue-next' import { Building2, Calendar, MapPin, Shapes } from 'lucide-vue-next'
import { Badge } from 'frappe-ui'
import { inject } from 'vue' import { inject } from 'vue'
import { Avatar } from 'frappe-ui'
const dayjs = inject('$dayjs') const dayjs = inject('$dayjs')
const props = defineProps({ const props = defineProps({

View File

@@ -1,12 +1,12 @@
<template> <template>
<div class="space-y-5"> <div class="space-y-5 text-ink-gray-9">
<div class="space-y-2"> <div class="space-y-2">
<div class="flex items-center text-sm font-medium space-x-2"> <div class="flex items-center text-sm font-medium space-x-2">
<span> <span>
{{ __('What does include in preview mean?') }} {{ __('What does include in preview mean?') }}
</span> </span>
</div> </div>
<div class="text-xs text-gray-600 mb-1 leading-5"> <div class="text-xs text-ink-gray-5 mb-1 leading-5">
{{ {{
__( __(
'If Include in Preview is enabled for a lesson then the lesson will also be accessible to non logged in users.' 'If Include in Preview is enabled for a lesson then the lesson will also be accessible to non logged in users.'
@@ -23,9 +23,9 @@
<span> <span>
{{ __('How to add a Quiz?') }} {{ __('How to add a Quiz?') }}
</span> </span>
<Info class="w-3 h-3 text-gray-700" /> <Info class="w-3 h-3 text-ink-gray-7" />
</div> </div>
<div class="text-xs text-gray-600 mb-1 leading-5"> <div class="text-xs text-ink-gray-5 mb-1 leading-5">
{{ {{
__( __(
'Click on the add icon in the editor and select Quiz from the menu. It opens up a dialog, where you can either select a quiz from the list or create a new quiz. When you select the Create New option it redirects you to the quiz creation page.' 'Click on the add icon in the editor and select Quiz from the menu. It opens up a dialog, where you can either select a quiz from the list or create a new quiz. When you select the Create New option it redirects you to the quiz creation page.'
@@ -42,9 +42,9 @@
<span class="leading-5"> <span class="leading-5">
{{ __(contentMap['upload']) }} {{ __(contentMap['upload']) }}
</span> </span>
<Info class="w-3 h-3 text-gray-700" /> <Info class="w-3 h-3 text-ink-gray-7" />
</div> </div>
<div class="text-xs text-gray-600 mb-1 leading-5"> <div class="text-xs text-ink-gray-5 mb-1 leading-5">
{{ {{
__( __(
'To upload Image, Video, Audio or PDF from your system, click on the add icon and select upload from the menu. Then choose the file you want to add to the lesson and it gets added to your lesson.' 'To upload Image, Video, Audio or PDF from your system, click on the add icon and select upload from the menu. Then choose the file you want to add to the lesson and it gets added to your lesson.'
@@ -61,9 +61,9 @@
<span> <span>
{{ __(contentMap['youtube']) }} {{ __(contentMap['youtube']) }}
</span> </span>
<Info class="w-3 h-3 text-gray-700" /> <Info class="w-3 h-3 text-ink-gray-7" />
</div> </div>
<div class="text-xs text-gray-600 mb-1 leading-5"> <div class="text-xs text-ink-gray-5 mb-1 leading-5">
{{ {{
__( __(
'Copy the URL of the video from YouTube and paste it in the editor.' 'Copy the URL of the video from YouTube and paste it in the editor.'

View File

@@ -1,6 +1,6 @@
<template> <template>
<div class="flex items-center justify-between mb-5"> <div class="flex items-center justify-between mb-5">
<div class="text-lg font-semibold"> <div class="text-lg font-semibold text-ink-gray-9">
{{ __('Live Class') }} {{ __('Live Class') }}
</div> </div>
<Button v-if="user.data.is_moderator" @click="openLiveClassModal"> <Button v-if="user.data.is_moderator" @click="openLiveClassModal">
@@ -15,49 +15,59 @@
<div v-if="liveClasses.data?.length" class="grid grid-cols-2 gap-5"> <div v-if="liveClasses.data?.length" class="grid grid-cols-2 gap-5">
<div <div
v-for="cls in liveClasses.data" v-for="cls in liveClasses.data"
class="flex flex-col border rounded-md h-full text-sm text-gray-700 p-3" class="flex flex-col border rounded-md h-full text-ink-gray-7 p-3"
> >
<div class="font-semibold text-gray-900 text-lg mb-4"> <div class="font-semibold text-ink-gray-9 text-lg mb-1">
{{ cls.title }} {{ cls.title }}
</div> </div>
<div class="leading-5 text-gray-700 text-sm mb-4"> <div class="short-introduction">
{{ cls.description }} {{ cls.description }}
</div> </div>
<div class="flex items-center mb-2"> <div class="space-y-3">
<Calendar class="w-4 h-4 stroke-1.5 text-gray-700" /> <div class="flex items-center space-x-2">
<span class="ml-2"> <Calendar class="w-4 h-4 stroke-1.5" />
{{ dayjs(cls.date).format('DD MMMM YYYY') }} <span>
</span> {{ dayjs(cls.date).format('DD MMMM YYYY') }}
</div> </span>
<div class="flex items-center mb-5"> </div>
<Clock class="w-4 h-4 stroke-1.5" /> <div class="flex items-center space-x-2">
<span class="ml-2"> <Clock class="w-4 h-4 stroke-1.5" />
{{ formatTime(cls.time) }} <span>
</span> {{ formatTime(cls.time) }}
</div> </span>
<div class="flex items-center space-x-2 text-gray-900 mt-auto"> </div>
<a <div
v-if="user.data?.is_moderator || user.data?.is_evaluator" v-if="cls.date >= dayjs().format('YYYY-MM-DD')"
:href="cls.start_url" class="flex items-center space-x-2 text-ink-gray-9 mt-auto"
target="_blank"
class="w-1/2 cursor-pointer inline-flex items-center justify-center gap-2 transition-colors focus:outline-none text-gray-800 bg-gray-100 hover:bg-gray-200 active:bg-gray-300 focus-visible:ring focus-visible:ring-gray-400 h-7 text-base px-2 rounded"
> >
<Monitor class="h-4 w-4 stroke-1.5" /> <a
{{ __('Start') }} v-if="user.data?.is_moderator || user.data?.is_evaluator"
</a> :href="cls.start_url"
<a target="_blank"
v-if="cls.date <= dayjs().format('YYYY-MM-DD')" class="w-1/2 cursor-pointer inline-flex items-center justify-center gap-2 transition-colors focus:outline-none text-ink-gray-8 bg-surface-gray-2 hover:bg-surface-gray-3 active:bg-surface-gray-4 focus-visible:ring focus-visible:ring-outline-gray-3 h-7 text-base px-2 rounded"
:href="cls.join_url" >
target="_blank" <Monitor class="h-4 w-4 stroke-1.5" />
class="w-full cursor-pointer inline-flex items-center justify-center gap-2 transition-colors focus:outline-none text-gray-800 bg-gray-100 hover:bg-gray-200 active:bg-gray-300 focus-visible:ring focus-visible:ring-gray-400 h-7 text-base px-2 rounded" {{ __('Start') }}
> </a>
<Video class="h-4 w-4 stroke-1.5" /> <a
{{ __('Join') }} :href="cls.join_url"
</a> target="_blank"
class="w-full cursor-pointer inline-flex items-center justify-center gap-2 transition-colors focus:outline-none text-ink-gray-8 bg-surface-gray-2 hover:bg-surface-gray-3 active:bg-surface-gray-4 focus-visible:ring focus-visible:ring-outline-gray-3 h-7 text-base px-2 rounded"
>
<Video class="h-4 w-4 stroke-1.5" />
{{ __('Join') }}
</a>
</div>
<div v-else class="flex items-center space-x-2 text-yellow-700">
<Info class="w-4 h-4 stroke-1.5" />
<span>
{{ __('This class has ended') }}
</span>
</div>
</div> </div>
</div> </div>
</div> </div>
<div v-else class="text-sm italic text-gray-600"> <div v-else class="text-sm italic text-ink-gray-5">
{{ __('No live classes scheduled') }} {{ __('No live classes scheduled') }}
</div> </div>
<LiveClassModal <LiveClassModal
@@ -68,7 +78,7 @@
</template> </template>
<script setup> <script setup>
import { createListResource, Button } from 'frappe-ui' import { createListResource, Button } from 'frappe-ui'
import { Plus, Clock, Calendar, Video, Monitor } from 'lucide-vue-next' import { Plus, Clock, Calendar, Video, Monitor, Info } from 'lucide-vue-next'
import { inject } from 'vue' import { inject } from 'vue'
import LiveClassModal from '@/components/Modals/LiveClassModal.vue' import LiveClassModal from '@/components/Modals/LiveClassModal.vue'
import { ref } from 'vue' import { ref } from 'vue'
@@ -107,3 +117,15 @@ const openLiveClassModal = () => {
showLiveClassModal.value = true showLiveClassModal.value = true
} }
</script> </script>
<style>
.short-introduction {
display: -webkit-box;
-webkit-line-clamp: 2;
-webkit-box-orient: vertical;
text-overflow: ellipsis;
width: 100%;
overflow: hidden;
margin: 0.25rem 0 1.5rem;
line-height: 1.5;
}
</style>

View File

@@ -2,10 +2,10 @@
<div class="flex min-h-0 flex-col text-base"> <div class="flex min-h-0 flex-col text-base">
<div class="flex items-center justify-between"> <div class="flex items-center justify-between">
<div> <div>
<div class="text-xl font-semibold mb-1"> <div class="text-xl font-semibold mb-1 text-ink-gray-9">
{{ __(label) }} {{ __(label) }}
</div> </div>
<!-- <div class="text-xs text-gray-600"> <!-- <div class="text-xs text-ink-gray-5">
{{ __(description) }} {{ __(description) }}
</div> --> </div> -->
</div> </div>
@@ -36,7 +36,7 @@
<FormControl <FormControl
v-model="member.first_name" v-model="member.first_name"
:placeholder="__('First Name')" :placeholder="__('First Name')"
type="test" type="text"
class="w-full" class="w-full"
/> />
<Button @click="addMember()" variant="subtle"> <Button @click="addMember()" variant="subtle">
@@ -63,7 +63,7 @@
/> />
<div class="space-y-1"> <div class="space-y-1">
<div class="flex"> <div class="flex">
<div class="text-gray-900"> <div class="text-ink-gray-9">
{{ member.full_name }} {{ member.full_name }}
</div> </div>
<div <div
@@ -81,12 +81,14 @@
</Badge> </Badge>
</div> </div>
</div> </div>
<div class="text-sm text-gray-700"> <div class="text-sm text-ink-gray-7">
{{ member.name }} {{ member.name }}
</div> </div>
</div> </div>
</div> </div>
<div class="flex items-center justify-center text-gray-700 text-sm"> <div
class="flex items-center justify-center text-ink-gray-7 text-sm"
>
<div v-if="member.last_active"> <div v-if="member.last_active">
{{ dayjs(member.last_active).format('DD MMM, YYYY HH:mm a') }} {{ dayjs(member.last_active).format('DD MMM, YYYY HH:mm a') }}
</div> </div>

View File

@@ -5,7 +5,7 @@
</div> </div>
<div <div
v-if="sidebarSettings.data" v-if="sidebarSettings.data"
class="fixed flex items-center justify-around border-t border-gray-300 bottom-0 z-10 w-full bg-white standalone:pb-4" class="fixed flex items-center justify-around border-t border-outline-gray-2 bottom-0 z-10 w-full bg-surface-white standalone:pb-4"
:style="{ :style="{
gridTemplateColumns: `repeat(${ gridTemplateColumns: `repeat(${
sidebarLinks.length + 1 sidebarLinks.length + 1
@@ -22,7 +22,7 @@
<component <component
:is="icons[tab.icon]" :is="icons[tab.icon]"
class="h-6 w-6 stroke-1.5" class="h-6 w-6 stroke-1.5"
:class="[isActive(tab) ? 'text-gray-900' : 'text-gray-600']" :class="[isActive(tab) ? 'text-ink-gray-9' : 'text-ink-gray-5']"
/> />
</button> </button>
<Popover <Popover
@@ -33,7 +33,7 @@
<template #target> <template #target>
<component <component
:is="icons['List']" :is="icons['List']"
class="h-6 w-6 stroke-1.5 text-gray-600" class="h-6 w-6 stroke-1.5 text-ink-gray-5"
/> />
</template> </template>
<template #body-main> <template #body-main>
@@ -46,7 +46,7 @@
> >
<component <component
:is="icons[link.icon]" :is="icons[link.icon]"
class="h-4 w-4 stroke-1.5 text-gray-600" class="h-4 w-4 stroke-1.5 text-ink-gray-5"
/> />
<div> <div>
{{ link.label }} {{ link.label }}

View File

@@ -16,26 +16,26 @@
<template #body-content> <template #body-content>
<div class="flex flex-col gap-4"> <div class="flex flex-col gap-4">
<div class=""> <div class="">
<div class="mb-1.5 text-sm text-gray-600"> <div class="mb-1.5 text-sm text-ink-gray-5">
{{ __('Subject') }} {{ __('Subject') }}
<span class="text-red-500">*</span> <span class="text-ink-red-3">*</span>
</div> </div>
<Input type="text" v-model="announcement.subject" /> <Input type="text" v-model="announcement.subject" />
</div> </div>
<div class=""> <div class="">
<div class="mb-1.5 text-sm text-gray-600"> <div class="mb-1.5 text-sm text-ink-gray-5">
{{ __('Reply To') }} {{ __('Reply To') }}
</div> </div>
<Input type="text" v-model="announcement.replyTo" /> <Input type="text" v-model="announcement.replyTo" />
</div> </div>
<div class="mb-4"> <div class="mb-4">
<div class="mb-1.5 text-sm text-gray-600"> <div class="mb-1.5 text-sm text-ink-gray-5">
{{ __('Announcement') }} {{ __('Announcement') }}
</div> </div>
<TextEditor <TextEditor
:bubbleMenu="true" :fixedMenu="true"
@change="(val) => (announcement.announcement = val)" @change="(val) => (announcement.announcement = val)"
editorClass="prose-sm py-2 px-2 min-h-[200px] border-gray-300 hover:border-gray-400 rounded-md bg-gray-200" editorClass="prose-sm py-2 px-2 min-h-[200px] border-outline-gray-2 hover:border-outline-gray-3 rounded-b-md bg-surface-gray-3"
/> />
</div> </div>
</div> </div>
@@ -102,7 +102,7 @@ const makeAnnouncement = (close) => {
) )
}, },
onError(err) { onError(err) {
showToast(__('Error'), __(err.messages?.[0] || err), 'check') showToast(__('Error'), __(err.messages?.[0] || err), 'alert-circle')
}, },
} }
) )

View File

@@ -31,7 +31,7 @@
</template> </template>
<script setup> <script setup>
import { Dialog, createResource } from 'frappe-ui' import { Dialog, createResource } from 'frappe-ui'
import { ref, defineModel } from 'vue' import { ref } from 'vue'
import Link from '@/components/Controls/Link.vue' import Link from '@/components/Controls/Link.vue'
import { showToast } from '@/utils' import { showToast } from '@/utils'

View File

@@ -1,7 +1,12 @@
<template> <template>
<Dialog v-model="show" :options="{}"> <Dialog
v-model="show"
:options="{
size: 'xl',
}"
>
<template #body> <template #body>
<div class="p-5 space-y-8 text-base"> <div class="p-5 space-y-10 text-base">
<div class="flex items-center space-x-2"> <div class="flex items-center space-x-2">
<Avatar :image="student.user_image" size="3xl" /> <Avatar :image="student.user_image" size="3xl" />
<div class="space-y-1"> <div class="space-y-1">
@@ -13,51 +18,64 @@
{{ student.progress }}% {{ __('Complete') }} {{ student.progress }}% {{ __('Complete') }}
</Badge> </Badge>
</div> </div>
<div class="text-sm text-gray-700"> <div class="text-sm text-ink-gray-7">
{{ student.email }} {{ student.email }}
</div> </div>
</div> </div>
</div> </div>
<!-- Assessments --> <div class="space-y-8">
<div> <!-- Assessments -->
<div> <div class="space-y-2 text-sm">
<div <div class="flex items-center border-b pb-1 font-medium">
class="grid grid-cols-[70%,30%] border-b pl-2 pb-1 mb-2 text-xs text-gray-700 font-medium" <span class="flex-1">
>
<span>
{{ __('Assessment') }} {{ __('Assessment') }}
</span> </span>
<span> <span>
{{ __('Progress') }} {{ __('Percentage/Status') }}
</span> </span>
</div> </div>
<div <router-link
v-for="assessment in Object.keys(student.assessments)" v-for="assessment in Object.keys(student.assessments)"
class="grid grid-cols-[70%,30%] pl-2 mb-2 text-gray-700 font-medium" class="flex items-center text-ink-gray-7 font-medium"
:to="{
name:
student.assessments[assessment].type == 'LMS Assignment'
? 'AssignmentSubmission'
: '',
params:
student.assessments[assessment].type == 'LMS Assignment'
? {
assignmentID:
student.assessments[assessment].assessment,
submissionName:
student.assessments[assessment].submission,
}
: {},
}"
> >
<span> <span class="flex-1">
{{ assessment }} {{ assessment }}
</span> </span>
<span v-if="isAssignment(student.assessments[assessment])"> <span v-if="isAssignment(student.assessments[assessment].status)">
<Badge :theme="getStatusTheme(student.assessments[assessment])"> <Badge
{{ student.assessments[assessment] }} :theme="
getStatusTheme(student.assessments[assessment].status)
"
>
{{ student.assessments[assessment].status }}
</Badge> </Badge>
</span> </span>
<span v-else> <span v-else>
{{ student.assessments[assessment] }} {{ student.assessments[assessment].status }}
</span> </span>
</div> </router-link>
</div> </div>
</div>
<!-- Courses --> <!-- Courses -->
<div> <div class="space-y-2 text-sm">
<div> <div class="flex items-center border-b pb-1 font-medium">
<div <span class="flex-1">
class="grid grid-cols-[70%,30%] mb-2 text-xs text-gray-700 border-b pl-2 pb-1 font-medium"
>
<span>
{{ __('Courses') }} {{ __('Courses') }}
</span> </span>
<span> <span>
@@ -66,9 +84,9 @@
</div> </div>
<div <div
v-for="course in Object.keys(student.courses)" v-for="course in Object.keys(student.courses)"
class="grid grid-cols-[70%,30%] pl-2 mb-2 text-gray-700 font-medium" class="flex items-center text-ink-gray-7 font-medium"
> >
<span> <span class="flex-1">
{{ course }} {{ course }}
</span> </span>
<span> <span>
@@ -78,16 +96,15 @@
</div> </div>
</div> </div>
<!-- <span class="mt-4"> <!-- Heatmap -->
{{ student }} <StudentHeatmap :member="student.email" :days="120" />
</span> -->
</div> </div>
</template> </template>
</Dialog> </Dialog>
</template> </template>
<script setup> <script setup>
import { Avatar, Badge, Dialog } from 'frappe-ui' import { Avatar, Badge, Dialog } from 'frappe-ui'
import ProgressBar from '@/components/ProgressBar.vue' import StudentHeatmap from '@/components/StudentHeatmap.vue'
const show = defineModel() const show = defineModel()
const props = defineProps({ const props = defineProps({

View File

@@ -17,12 +17,6 @@
> >
<template #body-content> <template #body-content>
<div class="space-y-4"> <div class="space-y-4">
<FormControl
type="select"
v-model="details.course"
:label="__('Course')"
:options="getCourses()"
/>
<Link <Link
v-model="details.evaluator" v-model="details.evaluator"
:label="__('Evaluator')" :label="__('Evaluator')"
@@ -38,6 +32,12 @@
v-model="details.expiry_date" v-model="details.expiry_date"
:label="__('Expiry Date')" :label="__('Expiry Date')"
/> />
<FormControl
type="select"
v-model="details.course"
:label="__('Course')"
:options="getCourses()"
/>
<Link <Link
v-model="details.template" v-model="details.template"
:label="__('Template')" :label="__('Template')"
@@ -94,7 +94,7 @@ const createCertificate = createResource({
template: details.template, template: details.template,
published: details.published, published: details.published,
course: values.course, course: values.course,
batch: values.batch, batch_name: values.batch,
member: values.member, member: values.member,
evaluator: details.evaluator, evaluator: details.evaluator,
}, },

View File

@@ -47,19 +47,19 @@
<div v-else class=""> <div v-else class="">
<div class="flex items-center"> <div class="flex items-center">
<div class="border rounded-md p-2 mr-2"> <div class="border rounded-md p-2 mr-2">
<FileText class="h-5 w-5 stroke-1.5 text-gray-700" /> <FileText class="h-5 w-5 stroke-1.5 text-ink-gray-7" />
</div> </div>
<div class="flex flex-col"> <div class="flex flex-col">
<span> <span>
{{ chapter.scorm_package.file_name }} {{ chapter.scorm_package.file_name }}
</span> </span>
<span class="text-sm text-gray-500 mt-1"> <span class="text-sm text-ink-gray-4 mt-1">
{{ getFileSize(chapter.scorm_package.file_size) }} {{ getFileSize(chapter.scorm_package.file_size) }}
</span> </span>
</div> </div>
<X <X
@click="() => (chapter.scorm_package = null)" @click="() => (chapter.scorm_package = null)"
class="bg-gray-200 rounded-md cursor-pointer stroke-1.5 w-5 h-5 p-1 ml-4" class="bg-surface-gray-3 rounded-md cursor-pointer stroke-1.5 w-5 h-5 p-1 ml-4"
/> />
</div> </div>
</div> </div>
@@ -77,7 +77,7 @@ import {
FormControl, FormControl,
Switch, Switch,
} from 'frappe-ui' } from 'frappe-ui'
import { defineModel, reactive, watch } from 'vue' import { reactive, watch } from 'vue'
import { showToast, getFileSize } from '@/utils/' import { showToast, getFileSize } from '@/utils/'
import { capture } from '@/telemetry' import { capture } from '@/telemetry'
import { FileText, X } from 'lucide-vue-next' import { FileText, X } from 'lucide-vue-next'
@@ -145,9 +145,9 @@ const addChapter = async (close) => {
{ {
onSuccess(data) { onSuccess(data) {
cleanChapter() cleanChapter()
if (!settingsStore.onboardingDetails.data?.is_onboarded) { /* if (!settingsStore.onboardingDetails.data?.is_onboarded) {
settingsStore.onboardingDetails.reload() settingsStore.onboardingDetails.reload()
} } */
outline.value.reload() outline.value.reload()
showToast( showToast(
__('Success'), __('Success'),

View File

@@ -18,7 +18,7 @@
<FormControl v-model="topic.title" :label="__('Title')" type="text" /> <FormControl v-model="topic.title" :label="__('Title')" type="text" />
</div> </div>
<div> <div>
<div class="mb-1.5 text-sm text-gray-600"> <div class="mb-1.5 text-sm text-ink-gray-5">
{{ __('Details') }} {{ __('Details') }}
</div> </div>
<TextEditor <TextEditor
@@ -26,7 +26,7 @@
@change="(val) => (topic.reply = val)" @change="(val) => (topic.reply = val)"
:editable="true" :editable="true"
:fixedMenu="true" :fixedMenu="true"
editorClass="prose-sm max-w-none border-b border-x bg-gray-100 rounded-b-md py-1 px-2 min-h-[7rem]" editorClass="prose-sm max-w-none border-b border-x bg-surface-gray-2 rounded-b-md py-1 px-2 min-h-[7rem]"
/> />
</div> </div>
</div> </div>
@@ -35,7 +35,7 @@
</template> </template>
<script setup> <script setup>
import { Dialog, FormControl, TextEditor, createResource } from 'frappe-ui' import { Dialog, FormControl, TextEditor, createResource } from 'frappe-ui'
import { reactive, defineModel } from 'vue' import { reactive } from 'vue'
import { showToast, singularize } from '@/utils' import { showToast, singularize } from '@/utils'
const topics = defineModel('reloadTopics') const topics = defineModel('reloadTopics')

View File

@@ -5,7 +5,7 @@
</template> </template>
<template #body> <template #body>
<div <div
class="absolute left-1/2 mt-3 w-96 max-w-lg -translate-x-1/2 transform rounded-lg bg-white px-4 sm:px-0 lg:max-w-3xl" class="absolute left-1/2 mt-3 w-96 max-w-lg -translate-x-1/2 transform rounded-lg bg-surface-white px-4 sm:px-0 lg:max-w-3xl"
> >
<div <div
class="overflow-hidden rounded-lg p-3 shadow-2xl ring-1 ring-black ring-opacity-5" class="overflow-hidden rounded-lg p-3 shadow-2xl ring-1 ring-black ring-opacity-5"
@@ -35,7 +35,7 @@
</FileUploader> </FileUploader>
</div> </div>
<div <div
class="relative mt-2 grid w-[25.5rem] gap-2 bg-white lg:grid-cols-2" class="relative mt-2 grid w-[25.5rem] gap-2 bg-surface-white lg:grid-cols-2"
> >
<button <button
v-for="image in images.data" v-for="image in images.data"
@@ -53,7 +53,7 @@
</div> </div>
<div <div
v-if="images.data" v-if="images.data"
class="mt-2 text-center text-sm text-gray-500" class="mt-2 text-center text-sm text-ink-gray-4"
> >
{{ __('Image search powered by') }} {{ __('Image search powered by') }}
<a class="underline" target="_blank" href="https://unsplash.com"> <a class="underline" target="_blank" href="https://unsplash.com">

View File

@@ -33,24 +33,24 @@
</template> </template>
</FileUploader> </FileUploader>
<div v-else class="mb-4"> <div v-else class="mb-4">
<div class="text-xs text-gray-600 mb-1"> <div class="text-xs text-ink-gray-5 mb-1">
{{ __('Profile Image') }} {{ __('Profile Image') }}
</div> </div>
<div class="flex items-center"> <div class="flex items-center">
<div class="border rounded-md p-2 mr-2"> <div class="border rounded-md p-2 mr-2">
<FileText class="h-5 w-5 stroke-1.5 text-gray-700" /> <FileText class="h-5 w-5 stroke-1.5 text-ink-gray-7" />
</div> </div>
<div class="text-base flex flex-col"> <div class="text-base flex flex-col">
<span> <span>
{{ profile.image.file_name }} {{ profile.image.file_name }}
</span> </span>
<span class="text-sm text-gray-500 mt-1"> <span class="text-sm text-ink-gray-4 mt-1">
{{ getFileSize(profile.image.file_size) }} {{ getFileSize(profile.image.file_size) }}
</span> </span>
</div> </div>
<X <X
@click="removeImage()" @click="removeImage()"
class="bg-gray-200 rounded-md cursor-pointer stroke-1.5 w-5 h-5 p-1 ml-4" class="bg-surface-gray-3 rounded-md cursor-pointer stroke-1.5 w-5 h-5 p-1 ml-4"
/> />
</div> </div>
</div> </div>
@@ -71,14 +71,14 @@
/> />
<div class="mb-4"> <div class="mb-4">
<div class="mb-1.5 text-sm text-gray-600"> <div class="mb-1.5 text-sm text-ink-gray-5">
{{ __('Bio') }} {{ __('Bio') }}
</div> </div>
<TextEditor <TextEditor
:fixedMenu="true" :fixedMenu="true"
@change="(val) => (profile.bio = val)" @change="(val) => (profile.bio = val)"
:content="profile.bio" :content="profile.bio"
editorClass="prose-sm py-2 px-2 min-h-[200px] border-gray-300 hover:border-gray-400 rounded-md bg-gray-200" editorClass="prose-sm py-2 px-2 min-h-[200px] border-outline-gray-2 hover:border-outline-gray-3 rounded-md bg-surface-gray-3"
/> />
</div> </div>
</div> </div>
@@ -94,9 +94,9 @@ import {
createResource, createResource,
TextEditor, TextEditor,
} from 'frappe-ui' } from 'frappe-ui'
import { reactive, watch, defineModel } from 'vue' import { reactive, watch } from 'vue'
import { FileText, X } from 'lucide-vue-next' import { FileText, X } from 'lucide-vue-next'
import { getFileSize, showToast } from '@/utils' import { getFileSize, showToast, escapeHTML } from '@/utils'
const reloadProfile = defineModel('reloadProfile') const reloadProfile = defineModel('reloadProfile')
@@ -131,6 +131,7 @@ const imageResource = createResource({
const updateProfile = createResource({ const updateProfile = createResource({
url: 'frappe.client.set_value', url: 'frappe.client.set_value',
makeParams(values) { makeParams(values) {
profile.bio = escapeHTML(profile.bio)
return { return {
doctype: 'User', doctype: 'User',
name: props.profile.data.name, name: props.profile.data.name,

View File

@@ -16,25 +16,33 @@
<template #body-content> <template #body-content>
<div class="flex flex-col gap-4"> <div class="flex flex-col gap-4">
<div> <div>
<div class="mb-1.5 text-sm text-gray-600"> <div class="mb-1.5 text-sm text-ink-gray-5">
{{ __('Course') }} {{ __('Course') }}
</div> </div>
<Select v-model="evaluation.course" :options="getCourses()" /> <Select v-model="evaluation.course" :options="getCourses()" />
</div> </div>
<div> <div>
<div class="mb-1.5 text-sm text-gray-600"> <div class="mb-1.5 text-sm text-ink-gray-5">
{{ __('Date') }} {{ __('Date') }}
</div> </div>
<FormControl type="date" v-model="evaluation.date" /> <FormControl
type="date"
v-model="evaluation.date"
:min="
dayjs()
.add(dayjs.duration({ days: 1 }))
.format('YYYY-MM-DD')
"
/>
</div> </div>
<div v-if="slots.data?.length"> <div v-if="slots.data?.length">
<div class="mb-1.5 text-sm text-gray-600"> <div class="mb-1.5 text-sm text-ink-gray-5">
{{ __('Select a slot') }} {{ __('Select a slot') }}
</div> </div>
<div class="grid grid-cols-2 gap-2"> <div class="grid grid-cols-2 gap-2">
<div v-for="slot in slots.data"> <div v-for="slot in slots.data">
<div <div
class="text-base text-center border rounded-md bg-gray-200 p-2 cursor-pointer" class="text-base text-center border rounded-md bg-surface-gray-3 p-2 cursor-pointer"
@click="saveSlot(slot)" @click="saveSlot(slot)"
:class="{ :class="{
'border-gray-900': evaluation.start_time == slot.start_time, 'border-gray-900': evaluation.start_time == slot.start_time,
@@ -48,7 +56,7 @@
</div> </div>
<div <div
v-else-if="evaluation.course && evaluation.date" v-else-if="evaluation.course && evaluation.date"
class="text-sm italic text-red-600" class="text-sm italic text-ink-red-4"
> >
{{ __('No slots available for this date.') }} {{ __('No slots available for this date.') }}
</div> </div>
@@ -58,7 +66,7 @@
</template> </template>
<script setup> <script setup>
import { Dialog, createResource, Select, FormControl } from 'frappe-ui' import { Dialog, createResource, Select, FormControl } from 'frappe-ui'
import { defineModel, reactive, watch, inject } from 'vue' import { reactive, watch, inject } from 'vue'
import { createToast, formatTime } from '@/utils/' import { createToast, formatTime } from '@/utils/'
const user = inject('$user') const user = inject('$user')
@@ -143,7 +151,7 @@ function submitEvaluation(close) {
title: unavailabilityMessage ? __('Evaluator is Unavailable') : '', title: unavailabilityMessage ? __('Evaluator is Unavailable') : '',
text: message, text: message,
icon: unavailabilityMessage ? 'alert-circle' : 'x', icon: unavailabilityMessage ? 'alert-circle' : 'x',
iconClasses: 'bg-yellow-600 text-white rounded-md p-px', iconClasses: 'bg-yellow-600 text-ink-white rounded-md p-px',
position: 'top-center', position: 'top-center',
timeout: 10, timeout: 10,
}) })
@@ -161,6 +169,11 @@ const getCourses = () => {
}) })
} }
} }
if (courses.length == 1) {
evaluation.course = courses[0].value
}
return courses return courses
} }

View File

@@ -12,7 +12,7 @@
{{ event.title }} {{ event.title }}
</div> </div>
<div class="flex flex-col space-y-4 text-sm text-gray-800"> <div class="flex flex-col space-y-4 text-sm text-ink-gray-8">
<Tooltip :text="__('Email ID')"> <Tooltip :text="__('Email ID')">
<div class="flex items-center space-x-2 w-fit"> <div class="flex items-center space-x-2 w-fit">
<User class="h-4 w-4 stroke-1.5" /> <User class="h-4 w-4 stroke-1.5" />

View File

@@ -48,13 +48,13 @@
</div> </div>
<div v-else class="flex items-center"> <div v-else class="flex items-center">
<div class="border rounded-md p-2 mr-2"> <div class="border rounded-md p-2 mr-2">
<FileText class="h-5 w-5 stroke-1.5 text-gray-700" /> <FileText class="h-5 w-5 stroke-1.5 text-ink-gray-7" />
</div> </div>
<div class="flex flex-col"> <div class="flex flex-col">
<span> <span>
{{ resume.file_name }} {{ resume.file_name }}
</span> </span>
<span class="text-sm text-gray-500 mt-1"> <span class="text-sm text-ink-gray-4 mt-1">
{{ getFileSize(resume.file_size) }} {{ getFileSize(resume.file_size) }}
</span> </span>
</div> </div>
@@ -66,7 +66,7 @@
<script setup> <script setup>
import { Dialog, FileUploader, Button, createResource } from 'frappe-ui' import { Dialog, FileUploader, Button, createResource } from 'frappe-ui'
import { FileText } from 'lucide-vue-next' import { FileText } from 'lucide-vue-next'
import { ref, inject, defineModel } from 'vue' import { ref, inject } from 'vue'
import { createToast, getFileSize } from '@/utils/' import { createToast, getFileSize } from '@/utils/'
const resume = ref(null) const resume = ref(null)
@@ -116,7 +116,7 @@ const submitResume = (close) => {
title: 'Success', title: 'Success',
text: 'Your application has been submitted', text: 'Your application has been submitted',
icon: 'check', icon: 'check',
iconClasses: 'bg-green-600 text-white rounded-md p-px', iconClasses: 'bg-surface-green-3 text-ink-white rounded-md p-px',
}) })
application.value.reload() application.value.reload()
close() close()
@@ -126,7 +126,7 @@ const submitResume = (close) => {
title: 'Error', title: 'Error',
text: err.messages?.[0] || err, text: err.messages?.[0] || err,
icon: 'x', icon: 'x',
iconClasses: 'bg-red-600 text-white rounded-md p-px', iconClasses: 'bg-surface-red-5 text-ink-white rounded-md p-px',
position: 'top-center', position: 'top-center',
timeout: 10, timeout: 10,
}) })

View File

@@ -39,13 +39,19 @@
:required="true" :required="true"
/> />
</Tooltip> </Tooltip>
<FormControl
v-model="liveClass.timezone" <div class="space-y-1.5">
type="select" <label class="block text-ink-gray-5 text-xs" for="batchTimezone">
:options="getTimezoneOptions()" {{ __('Timezone') }}
:label="__('Timezone')" <span class="text-ink-red-3">*</span>
:required="true" </label>
/> <Autocomplete
@update:modelValue="(opt) => (liveClass.timezone = opt.value)"
:modelValue="liveClass.timezone"
:options="getTimezoneOptions()"
:required="true"
/>
</div>
</div> </div>
<div> <div>
<FormControl <FormControl
@@ -83,18 +89,14 @@
</template> </template>
<script setup> <script setup>
import { import {
Input,
DatePicker,
Select,
Textarea,
Dialog, Dialog,
createResource, createResource,
Tooltip, Tooltip,
FormControl, FormControl,
Autocomplete,
} from 'frappe-ui' } from 'frappe-ui'
import { reactive, inject } from 'vue' import { reactive, inject, onMounted } from 'vue'
import { getTimezones, createToast } from '@/utils/' import { getTimezones, createToast, getUserTimezone } from '@/utils/'
import { Info } from 'lucide-vue-next'
const liveClasses = defineModel('reloadLiveClasses') const liveClasses = defineModel('reloadLiveClasses')
const show = defineModel() const show = defineModel()
@@ -120,6 +122,10 @@ let liveClass = reactive({
host: user.data.name, host: user.data.name,
}) })
onMounted(() => {
liveClass.timezone = getUserTimezone()
})
const getTimezoneOptions = () => { const getTimezoneOptions = () => {
return getTimezones().map((timezone) => { return getTimezones().map((timezone) => {
return { return {
@@ -200,7 +206,7 @@ const submitLiveClass = (close) => {
title: 'Error', title: 'Error',
text: err.messages?.[0] || err, text: err.messages?.[0] || err,
icon: 'x', icon: 'x',
iconClasses: 'bg-red-600 text-white rounded-md p-px', iconClasses: 'bg-surface-red-5 text-ink-white rounded-md p-px',
position: 'top-center', position: 'top-center',
timeout: 10, timeout: 10,
}) })

View File

@@ -4,7 +4,7 @@
<div class="space-y-4"> <div class="space-y-4">
<div <div
v-if="!editMode" v-if="!editMode"
class="flex items-center text-xs text-gray-700 space-x-5" class="flex items-center text-xs text-ink-gray-7 space-x-5"
> >
<div class="flex items-center space-x-2"> <div class="flex items-center space-x-2">
<input <input
@@ -34,7 +34,7 @@
</div> </div>
<div v-if="questionType == 'new' || editMode" class="space-y-2"> <div v-if="questionType == 'new' || editMode" class="space-y-2">
<div> <div>
<label class="block text-xs text-gray-600 mb-1"> <label class="block text-xs text-ink-gray-5 mb-1">
{{ __('Question') }} {{ __('Question') }}
</label> </label>
<TextEditor <TextEditor
@@ -42,7 +42,7 @@
@change="(val) => (question.question = val)" @change="(val) => (question.question = val)"
:editable="true" :editable="true"
:fixedMenu="true" :fixedMenu="true"
editorClass="prose-sm max-w-none border-b border-x bg-gray-100 rounded-b-md py-1 px-2 min-h-[7rem]" editorClass="prose-sm max-w-none border-b border-x bg-surface-gray-2 rounded-b-md py-1 px-2 min-h-[7rem]"
/> />
</div> </div>
<FormControl <FormControl

View File

@@ -16,13 +16,13 @@
<template #body-content> <template #body-content>
<div class="flex flex-col gap-4"> <div class="flex flex-col gap-4">
<div> <div>
<div class="mb-1.5 text-sm text-gray-600"> <div class="mb-1.5 text-sm text-ink-gray-5">
{{ __('Rating') }} {{ __('Rating') }}
</div> </div>
<Rating v-model="review.rating" /> <Rating v-model="review.rating" />
</div> </div>
<div> <div>
<div class="mb-1.5 text-sm text-gray-600"> <div class="mb-1.5 text-sm text-ink-gray-5">
{{ __('Review') }} {{ __('Review') }}
</div> </div>
<Textarea type="text" size="md" rows="5" v-model="review.review" /> <Textarea type="text" size="md" rows="5" v-model="review.review" />
@@ -33,7 +33,7 @@
</template> </template>
<script setup> <script setup>
import { Dialog, Textarea, createResource } from 'frappe-ui' import { Dialog, Textarea, createResource } from 'frappe-ui'
import { defineModel, reactive } from 'vue' import { reactive } from 'vue'
import Rating from '@/components/Controls/Rating.vue' import Rating from '@/components/Controls/Rating.vue'
import { createToast } from '@/utils/' import { createToast } from '@/utils/'
@@ -81,7 +81,7 @@ function submitReview(close) {
createToast({ createToast({
text: err.messages?.[0] || err, text: err.messages?.[0] || err,
icon: 'x', icon: 'x',
iconClasses: 'text-red-600 bg-red-300', iconClasses: 'text-ink-red-4 bg-surface-red-4',
}) })
}, },
}) })

View File

@@ -2,14 +2,14 @@
<Dialog v-model="show" :options="{ size: '4xl' }"> <Dialog v-model="show" :options="{ size: '4xl' }">
<template #body> <template #body>
<div class="flex h-[calc(100vh_-_8rem)]"> <div class="flex h-[calc(100vh_-_8rem)]">
<div class="flex w-52 shrink-0 flex-col bg-gray-50 p-2"> <div class="flex w-52 shrink-0 flex-col bg-surface-gray-2 p-2">
<h1 class="mb-3 px-2 pt-2 text-lg font-semibold"> <h1 class="mb-3 px-2 pt-2 text-lg font-semibold text-ink-gray-9">
{{ __('Settings') }} {{ __('Settings') }}
</h1> </h1>
<div v-for="tab in tabs" :key="tab.label"> <div v-for="tab in tabs" :key="tab.label">
<div <div
v-if="!tab.hideLabel" v-if="!tab.hideLabel"
class="mb-2 mt-3 flex cursor-pointer gap-1.5 px-1 text-base font-medium text-gray-600 transition-all duration-300 ease-in-out" class="mb-2 mt-3 flex cursor-pointer gap-1.5 px-1 text-base font-medium text-ink-gray-5 transition-all duration-300 ease-in-out"
> >
<span>{{ __(tab.label) }}</span> <span>{{ __(tab.label) }}</span>
</div> </div>
@@ -21,8 +21,8 @@
class="w-full" class="w-full"
:class=" :class="
activeTab?.label == item.label activeTab?.label == item.label
? 'bg-white shadow-sm' ? 'bg-surface-selected shadow-sm'
: 'hover:bg-gray-100' : 'hover:bg-surface-gray-2'
" "
@click="activeTab = item" @click="activeTab = item"
/> />
@@ -32,7 +32,7 @@
<div <div
v-if="activeTab && data.doc" v-if="activeTab && data.doc"
:key="activeTab.label" :key="activeTab.label"
class="flex flex-1 flex-col px-10 py-8" class="flex flex-1 flex-col px-10 py-8 bg-surface-modal"
> >
<Members <Members
v-if="activeTab.label === 'Members'" v-if="activeTab.label === 'Members'"
@@ -118,6 +118,13 @@ const tabsStructure = computed(() => {
'This will enforce students to go through programs assigned to them in the correct order.', 'This will enforce students to go through programs assigned to them in the correct order.',
type: 'checkbox', type: 'checkbox',
}, },
{
label: 'Allow Guest Access',
name: 'allow_guest_access',
description:
'If enabled, users can access the course and batch lists without logging in.',
type: 'checkbox',
},
{ {
label: 'Send calendar invite for evaluations', label: 'Send calendar invite for evaluations',
name: 'send_calendar_invite_for_evaluations', name: 'send_calendar_invite_for_evaluations',
@@ -130,7 +137,7 @@ const tabsStructure = computed(() => {
name: 'unsplash_access_key', name: 'unsplash_access_key',
description: description:
'Optional. If this is set, students can pick a cover image from the unsplash library for their profile page. https://unsplash.com/documentation#getting-started.', 'Optional. If this is set, students can pick a cover image from the unsplash library for their profile page. https://unsplash.com/documentation#getting-started.',
type: 'text', type: 'password',
}, },
], ],
}, },

View File

@@ -46,11 +46,9 @@ const studentResource = createResource({
makeParams(values) { makeParams(values) {
return { return {
doc: { doc: {
doctype: 'Batch Student', doctype: 'LMS Batch Enrollment',
parent: props.batch, batch: props.batch,
parenttype: 'LMS Batch', member: student.value,
parentfield: 'students',
student: student.value,
}, },
} }
}, },

Some files were not shown because too many files have changed in this diff Show More