Compare commits

...

872 Commits

Author SHA1 Message Date
Frappe PR Bot
015aff9c4b chore(release): Bumped to Version 2.28.0 2025-05-07 07:48:16 +00:00
Jannat Patel
567bfc41e0 Merge pull request #1489 from pateljannat/issues-102
fix: misc fixes
2025-05-07 12:51:11 +05:30
Jannat Patel
90d77e9ffb fix: improved question form for quiz 2025-05-07 12:34:38 +05:30
Jannat Patel
2b33ba1984 fix: dark mode issues 2025-05-07 11:50:34 +05:30
Jannat Patel
1918f0c5d5 Merge branch 'develop' of https://github.com/frappe/lms into issues-102 2025-05-07 11:50:07 +05:30
Jannat Patel
91d79de723 fix: submission list access from assignment form 2025-05-06 20:02:34 +05:30
Jannat Patel
62b05f2377 fix: route to course from the course card widget 2025-05-06 19:25:53 +05:30
Jannat Patel
b628ec4c57 Merge pull request #1488 from pateljannat/simplify-persona-form
chore: simplified the persona form
2025-05-06 19:23:28 +05:30
Jannat Patel
494394f084 Merge pull request #1487 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-05-06 19:15:24 +05:30
Jannat Patel
e99b4b183c chore: simplified the persona form 2025-05-06 19:14:41 +05:30
Jannat Patel
9186353654 chore: Esperanto translations 2025-05-06 18:24:57 +05:30
Jannat Patel
bd2a7b9095 chore: Chinese Simplified translations 2025-05-06 18:24:55 +05:30
Jannat Patel
42b70e7a94 chore: Serbian (Latin) translations 2025-05-06 18:24:54 +05:30
Jannat Patel
7f913203a1 chore: Bosnian translations 2025-05-06 18:24:53 +05:30
Jannat Patel
9b94958840 chore: Croatian translations 2025-05-06 18:24:51 +05:30
Jannat Patel
2070e93379 chore: Thai translations 2025-05-06 18:24:50 +05:30
Jannat Patel
772f4d938f chore: Persian translations 2025-05-06 18:24:48 +05:30
Jannat Patel
531f3af203 chore: Portuguese, Brazilian translations 2025-05-06 18:24:47 +05:30
Jannat Patel
ed522341c1 chore: Turkish translations 2025-05-06 18:24:46 +05:30
Jannat Patel
ee59c5068e chore: Swedish translations 2025-05-06 18:24:44 +05:30
Jannat Patel
ebe3abd05b chore: Russian translations 2025-05-06 18:24:43 +05:30
Jannat Patel
358dd4dddc chore: Portuguese translations 2025-05-06 18:24:41 +05:30
Jannat Patel
3d924d3631 chore: Polish translations 2025-05-06 18:24:40 +05:30
Jannat Patel
0bed316a40 chore: Hungarian translations 2025-05-06 18:24:39 +05:30
Jannat Patel
24b5937793 chore: German translations 2025-05-06 18:24:37 +05:30
Jannat Patel
c5b5876700 chore: Arabic translations 2025-05-06 18:24:36 +05:30
Jannat Patel
0f969e952d chore: Spanish translations 2025-05-06 18:24:34 +05:30
Jannat Patel
43ba512fd5 chore: French translations 2025-05-06 18:24:33 +05:30
Jannat Patel
8aadbffe8c Merge pull request #1483 from pateljannat/issues-101
fix: misc fixes
2025-05-06 10:49:47 +05:30
Jannat Patel
be7e7bc6fd refactor: extracted function that enables plyr as a utility 2025-05-06 10:35:05 +05:30
Jannat Patel
3a10d4bdc0 fix: alignment of information on batch details 2025-05-06 10:17:07 +05:30
Jannat Patel
fc03ecd1b3 fix: made programs breadcrumb translatable 2025-05-06 09:27:48 +05:30
Jannat Patel
c7b10f0e83 Merge pull request #1482 from frappe/pot_develop_2025-05-02
chore: update POT file
2025-05-06 08:50:56 +05:30
Jannat Patel
6a94ce5e1c Merge pull request #1480 from pateljannat/evaluator-list-issue
fix: evaluator list in settings
2025-05-06 08:50:44 +05:30
frappe-pr-bot
59859a8e2f chore: update POT file 2025-05-02 16:04:11 +00:00
Jannat Patel
f51a8aae39 fix: evaluator list in settings 2025-05-02 15:03:54 +05:30
Jannat Patel
bd5b8c5e0e Merge pull request #1478 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-05-02 15:02:35 +05:30
Jannat Patel
67e7744566 Merge pull request #1479 from pateljannat/read-only
feat: read only mode
2025-04-30 18:33:22 +05:30
Jannat Patel
65a6663c31 fix: when moving between lessons in zen mode, ensure discussions remain hidden 2025-04-30 18:19:45 +05:30
Jannat Patel
603e80fd26 feat: read only mode 2025-04-30 18:03:00 +05:30
Jannat Patel
de4ee6bbe6 chore: Persian translations 2025-04-30 15:28:31 +05:30
Jannat Patel
a8aa242280 chore: Portuguese, Brazilian translations 2025-04-30 15:28:30 +05:30
Jannat Patel
0d32c2a9d9 Merge branch 'develop' of https://github.com/frappe/lms into read-only 2025-04-29 18:40:42 +05:30
Jannat Patel
6d5a02e2a8 feat: read only mode 2025-04-29 18:39:22 +05:30
Jannat Patel
67f3cbaaa8 Merge pull request #1475 from pateljannat/issues-100
fix: hide start learning button if self enrollment is disabled
2025-04-29 16:23:51 +05:30
Jannat Patel
f17504e1a0 fix: hide start learning button if self enrollment is disabled 2025-04-29 15:51:03 +05:30
Jannat Patel
b1a9af5de8 Merge pull request #1474 from pateljannat/issues-99
fix: check parenttype when fetching instructors
2025-04-29 11:27:56 +05:30
Jannat Patel
913bf553ae refactor: simplified the condition to check is user is instructor 2025-04-29 11:03:58 +05:30
Jannat Patel
356dcc42bf fix: check parenttype when fetching instructors 2025-04-29 10:48:52 +05:30
Jannat Patel
8c006f24ce Merge pull request #1465 from nextchamp-saqib/refactor-charts
refactor: use charts from `frappe-ui`
2025-04-28 18:19:16 +05:30
Jannat Patel
6f2f0092f0 refactor: dynamic data for statistics charts 2025-04-28 18:05:43 +05:30
Jannat Patel
56afc4c614 Merge pull request #1471 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-04-28 15:34:43 +05:30
Jannat Patel
0a3b9f8f9a chore: Esperanto translations 2025-04-28 14:01:56 +05:30
Jannat Patel
9b0623f4a4 chore: Bosnian translations 2025-04-28 14:01:55 +05:30
Jannat Patel
c13ef17a86 chore: Croatian translations 2025-04-28 14:01:54 +05:30
Jannat Patel
d5ac2f521f chore: Thai translations 2025-04-28 14:01:52 +05:30
Jannat Patel
037af18114 chore: Persian translations 2025-04-28 14:01:51 +05:30
Jannat Patel
92299458f5 chore: Portuguese, Brazilian translations 2025-04-28 14:01:49 +05:30
Jannat Patel
3272f2a4cf chore: Turkish translations 2025-04-28 14:01:48 +05:30
Jannat Patel
6a6dfdd82c chore: Swedish translations 2025-04-28 14:01:46 +05:30
Jannat Patel
fa27452983 chore: Russian translations 2025-04-28 14:01:45 +05:30
Jannat Patel
8df5ec41d5 chore: Portuguese translations 2025-04-28 14:01:43 +05:30
Jannat Patel
55aad3a742 chore: Polish translations 2025-04-28 14:01:42 +05:30
Jannat Patel
e46890d87e chore: Hungarian translations 2025-04-28 14:01:41 +05:30
Jannat Patel
3a36e10fce chore: German translations 2025-04-28 14:01:39 +05:30
Jannat Patel
cc30c6d271 chore: Arabic translations 2025-04-28 14:01:38 +05:30
Jannat Patel
5e75ff7fb7 chore: Spanish translations 2025-04-28 14:01:36 +05:30
Jannat Patel
80681a1f8b chore: French translations 2025-04-28 14:01:34 +05:30
Jannat Patel
5954e10155 chore: Serbian (Latin) translations 2025-04-28 14:01:32 +05:30
Jannat Patel
78c43b7a10 chore: Chinese Simplified translations 2025-04-28 14:01:31 +05:30
Jannat Patel
8c6f8bf97b Merge pull request #1470 from pateljannat/issues-98
fix: don't allow billing page access if batch has started
2025-04-28 12:38:13 +05:30
Jannat Patel
f220438257 fix: remove borders for iframe on lesson form 2025-04-28 11:16:04 +05:30
Jannat Patel
bbd06752d3 Merge pull request #1468 from frappe/pot_develop_2025-04-25
chore: update POT file
2025-04-28 10:36:31 +05:30
Jannat Patel
e34df2ce95 fix: don't allow billing page access if batch has started 2025-04-28 10:35:14 +05:30
frappe-pr-bot
b197c08716 chore: update POT file 2025-04-25 16:04:29 +00:00
Jannat Patel
aeb6c0f433 Merge branch 'develop' of https://github.com/frappe/lms into refactor-charts 2025-04-25 18:27:50 +05:30
Jannat Patel
8f32767267 Merge pull request #1461 from frappe/zen-mode
feat: Zen Mode
2025-04-25 18:21:31 +05:30
Jannat Patel
afd43b9a9a Merge branch 'develop' of https://github.com/frappe/lms into zen-mode 2025-04-25 17:49:13 +05:30
Jannat Patel
5893e02c48 fix: reduced the size of play button in video block 2025-04-25 17:49:07 +05:30
Jannat Patel
66d3325e3c Merge pull request #1467 from harshpwctech/develop
feat: Embedding for CloudflareStream
2025-04-25 17:29:35 +05:30
Jannat Patel
e513993a0d feat: show and hide discussions in zen mode 2025-04-25 17:28:54 +05:30
safe user
ddbdf42265 feat: Embedding Cloudflare Stream 2025-04-25 09:28:44 +00:00
safe user
badaa33ddb feat: Embedding for CloudflareStream 2025-04-25 08:38:00 +00:00
safe user
befa3d7a6d feat: Added embedding for CloudflareStream 2025-04-25 08:38:00 +00:00
Jannat Patel
513f1e8b86 fix: improved lesson locked state 2025-04-25 08:38:00 +00:00
Jannat Patel
4128f0fb73 chore: fixed settings 2025-04-25 08:38:00 +00:00
Jannat Patel
3d81a63410 ci: skip persona form for ui tests 2025-04-25 08:38:00 +00:00
Jannat Patel
c0ba44cacc fix: check persona_captured after details get saved 2025-04-25 08:38:00 +00:00
Jannat Patel
deba027457 chore: identify user persona 2025-04-25 08:38:00 +00:00
Jannat Patel
47089d286e chore: Serbian (Latin) translations 2025-04-25 08:38:00 +00:00
Jannat Patel
6c50292a66 fix: tags spacing on course cards 2025-04-25 08:38:00 +00:00
Jannat Patel
1f23f06926 fix: allow fullscreen on vimeo and adjust video height on mobile devices 2025-04-25 08:38:00 +00:00
Jannat Patel
63319d32e8 fix: detect editor change to enable plyr on newly added videos 2025-04-25 11:18:19 +05:30
Jannat Patel
66f28ef7a6 Merge branch 'develop' of https://github.com/frappe/lms into zen-mode 2025-04-25 10:20:38 +05:30
Jannat Patel
4e4eccd909 Merge pull request #1466 from pateljannat/issues-97
fix: country details in job page and form
2025-04-25 10:19:32 +05:30
Jannat Patel
c21fe99368 fix: country details in job page and form 2025-04-25 10:13:17 +05:30
Jannat Patel
53ea91e945 feat: plyr for vimeo 2025-04-25 10:05:32 +05:30
Jannat Patel
7cde05b58a Merge branch 'develop' of https://github.com/frappe/lms into zen-mode 2025-04-24 18:42:44 +05:30
Jannat Patel
0fc9b35307 Merge pull request #1464 from pateljannat/issues-96
feat: redesigned job list
2025-04-24 18:35:20 +05:30
Jannat Patel
4a36826af0 fix: if student applied for a job show that on the details page 2025-04-24 18:28:06 +05:30
Jannat Patel
26a278c5f4 feat: country filter in job list 2025-04-24 18:22:00 +05:30
Saqib Ansari
66a4d79730 refactor: use charts from frappe-ui 2025-04-24 16:19:12 +05:30
Jannat Patel
097d541391 feat: redesigned job list 2025-04-24 14:20:51 +05:30
Jannat Patel
788ef9b106 Merge pull request #1463 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-04-24 14:12:16 +05:30
Jannat Patel
a38e1163af chore: Chinese Simplified translations 2025-04-24 13:37:52 +05:30
Jannat Patel
a633ff5174 fix: check if youtube or vimeo video exists before enabling plyr 2025-04-24 12:06:57 +05:30
Jannat Patel
6b412106de feat: redesigned video block 2025-04-23 17:06:29 +05:30
Jannat Patel
93b5cb6161 feat: zen mode 2025-04-23 11:44:39 +05:30
Jannat Patel
4b80fbe5eb Merge pull request #1460 from pateljannat/issues-95
fix: improved lesson locked state
2025-04-22 18:13:01 +05:30
Jannat Patel
52775aae60 fix: improved lesson locked state 2025-04-22 18:05:07 +05:30
Jannat Patel
0430178b3e Merge pull request #1459 from pateljannat/user-persona
chore: identify user persona
2025-04-22 17:47:20 +05:30
Jannat Patel
470123c77a chore: fixed settings 2025-04-22 16:17:29 +05:30
Jannat Patel
66d4798db3 Merge branch 'develop' of https://github.com/frappe/lms into user-persona 2025-04-22 15:54:55 +05:30
Jannat Patel
cc39395a12 Merge pull request #1458 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-04-22 15:54:42 +05:30
Jannat Patel
3aeb9cf0b1 ci: skip persona form for ui tests 2025-04-22 15:32:01 +05:30
Jannat Patel
f1b383f0b7 fix: check persona_captured after details get saved 2025-04-22 15:02:21 +05:30
Jannat Patel
e2896b7bf0 chore: Serbian (Latin) translations 2025-04-22 12:52:49 +05:30
Jannat Patel
780dfb8966 Merge branch 'develop' of https://github.com/frappe/lms into user-persona 2025-04-21 18:07:12 +05:30
Jannat Patel
ac47ab3f8a Merge pull request #1456 from pateljannat/issues-94
fix: allow fullscreen on vimeo and adjust video height on mobile devices
2025-04-21 16:36:06 +05:30
Jannat Patel
bfc1488860 fix: tags spacing on course cards 2025-04-21 16:16:54 +05:30
Jannat Patel
726f733434 fix: allow fullscreen on vimeo and adjust video height on mobile devices 2025-04-21 15:41:51 +05:30
Jannat Patel
0c97e31101 Merge pull request #1455 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-04-21 15:07:47 +05:30
Jannat Patel
ec2b0718e6 chore: Portuguese translations 2025-04-21 12:09:04 +05:30
Jannat Patel
720056268c chore: Serbian (Latin) translations 2025-04-21 12:09:03 +05:30
Jannat Patel
345992eda4 chore: Esperanto translations 2025-04-21 12:09:02 +05:30
Jannat Patel
e3e6b35eb7 chore: Croatian translations 2025-04-21 12:09:00 +05:30
Jannat Patel
701ea950de chore: Thai translations 2025-04-21 12:08:59 +05:30
Jannat Patel
4b78865823 chore: Portuguese, Brazilian translations 2025-04-21 12:08:58 +05:30
Jannat Patel
5b2bdf4cf6 chore: Bosnian translations 2025-04-21 12:08:57 +05:30
Jannat Patel
a677b7fd3a chore: Persian translations 2025-04-21 12:08:55 +05:30
Jannat Patel
9cbd3db022 chore: Chinese Simplified translations 2025-04-21 12:08:54 +05:30
Jannat Patel
5f52d2c2c7 chore: Turkish translations 2025-04-21 12:08:53 +05:30
Jannat Patel
b8c403aa5d chore: Swedish translations 2025-04-21 12:08:52 +05:30
Jannat Patel
2c6863e18e chore: Russian translations 2025-04-21 12:08:50 +05:30
Jannat Patel
e7a462c685 chore: Polish translations 2025-04-21 12:08:49 +05:30
Jannat Patel
0cf671ae3b chore: Hungarian translations 2025-04-21 12:08:48 +05:30
Jannat Patel
dfc6f5bfb4 chore: German translations 2025-04-21 12:08:47 +05:30
Jannat Patel
64b9be7e42 chore: Arabic translations 2025-04-21 12:08:45 +05:30
Jannat Patel
7412a8761c chore: Spanish translations 2025-04-21 12:08:44 +05:30
Jannat Patel
65cdeabc77 chore: French translations 2025-04-21 12:08:42 +05:30
Jannat Patel
a507d4464d Merge pull request #1454 from pateljannat/issues-93
feat: meta image and keywords from settings
2025-04-21 11:05:39 +05:30
Jannat Patel
9143cc39d9 test: find the course image label and attach course image to its sibling input 2025-04-21 10:53:22 +05:30
Jannat Patel
e821755721 test: attach course image if selector is hidden 2025-04-21 10:33:24 +05:30
Jannat Patel
d081688fc9 Merge pull request #1453 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-04-21 10:16:30 +05:30
Jannat Patel
cdc7ee698c Merge pull request #1452 from frappe/pot_develop_2025-04-18
chore: update POT file
2025-04-21 09:55:00 +05:30
Jannat Patel
0d0a9c872c chore: Chinese Simplified translations 2025-04-20 11:58:27 +05:30
Jannat Patel
30953cce66 chore: German translations 2025-04-20 11:58:22 +05:30
Jannat Patel
f6008cf46a fix: don't show course count on batch details if there are no courses 2025-04-19 12:46:31 +05:30
Jannat Patel
eb0587f726 feat: meta image and keywords from settings 2025-04-19 12:37:24 +05:30
frappe-pr-bot
ba56ac87c5 chore: update POT file 2025-04-18 16:04:27 +00:00
Jannat Patel
5800ac67c4 chore: identify user persona 2025-04-18 18:05:57 +05:30
Jannat Patel
73941a159a Merge pull request #1451 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-04-18 16:16:35 +05:30
Jannat Patel
d1fe8b203a chore: Thai translations 2025-04-18 11:52:30 +05:30
Jannat Patel
8b8dbc1053 chore: Chinese Simplified translations 2025-04-18 11:52:28 +05:30
Jannat Patel
57e477b17c Merge pull request #1450 from pateljannat/issues-92
fix: moved powered by learning to the bottom of sidebar
2025-04-17 22:41:59 +05:30
Jannat Patel
1a1924de3e fix: removed styles on attachments as they were overriding the styles on webform 2025-04-17 22:32:25 +05:30
Jannat Patel
3bea19c8ad fix: moved powered by learning to the bottom of sidebar 2025-04-17 22:31:47 +05:30
Jannat Patel
cd47b62765 Merge pull request #1449 from pateljannat/assignment-issues
refactor: enhanced assignment form
2025-04-17 22:09:01 +05:30
Jannat Patel
ffeaad324e fix: removed assignment submission email notification 2025-04-17 22:03:07 +05:30
Jannat Patel
4504dd810d fix: make assignment modal scrollable if the questions is very long 2025-04-17 21:52:55 +05:30
Jannat Patel
60ad86f79c refactor: enhanced assignment form 2025-04-17 21:42:43 +05:30
Jannat Patel
f63294699a Merge pull request #1446 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-04-17 17:37:12 +05:30
Jannat Patel
650594d9ea Merge pull request #1447 from pateljannat/certification-redesign
fix: improved UI of certified participants page
2025-04-17 12:48:27 +05:30
Jannat Patel
7c22d5c774 chore: changed release workflow to run on the 15th of every month 2025-04-17 12:38:24 +05:30
Jannat Patel
73a501908d fix: mobile view for certified members page 2025-04-17 12:34:38 +05:30
Jannat Patel
31836e5c9e chore: Persian translations 2025-04-17 11:49:28 +05:30
Jannat Patel
31adab94b3 chore: merged conflicts 2025-04-16 21:38:00 +05:30
Jannat Patel
4e02044eb4 Merge pull request #1441 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-04-16 18:47:37 +05:30
Jannat Patel
f245cf2c5d fix: meta title for pages that don't have a default title 2025-04-16 17:09:09 +05:30
Jannat Patel
1b49cc1408 fix: meta title for pages that don't have a title 2025-04-16 17:02:52 +05:30
Jannat Patel
bd384a9b59 chore: Portuguese translations 2025-04-16 11:50:58 +05:30
Jannat Patel
48eb2ff405 fix: registration button should be visible if batch starts today but a few hours are left 2025-04-16 11:18:17 +05:30
Jannat Patel
dcacda984f fix: mobile view for certified member list 2025-04-16 10:59:06 +05:30
Jannat Patel
8186e9e1d2 Merge pull request #1438 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-04-15 20:14:15 +05:30
Jannat Patel
b5b93917d1 chore: Croatian translations 2025-04-15 11:50:27 +05:30
Jannat Patel
1ffdadbde3 chore: Bosnian translations 2025-04-15 11:50:25 +05:30
Jannat Patel
4506603ea1 chore: Swedish translations 2025-04-15 11:50:23 +05:30
Jannat Patel
fdf8b85f88 Merge branch 'develop' of https://github.com/frappe/lms into certification-redesign 2025-04-14 22:50:47 +05:30
Jannat Patel
340264ce41 Merge pull request #1436 from pateljannat/issues-91
fix: misc issues
2025-04-14 22:49:25 +05:30
Jannat Patel
d6187b3d63 fix: corrected grammar in payments app error message 2025-04-14 22:39:07 +05:30
Jannat Patel
b6577133a9 fix: misc issues 2025-04-14 22:28:06 +05:30
Jannat Patel
2d410eac37 Merge pull request #1430 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-04-14 16:24:43 +05:30
Jannat Patel
e63e71f2bf chore: Portuguese translations 2025-04-14 11:31:24 +05:30
Jannat Patel
ba743e0480 chore: Serbian (Latin) translations 2025-04-14 11:31:23 +05:30
Jannat Patel
2f26b15524 chore: Esperanto translations 2025-04-14 11:31:21 +05:30
Jannat Patel
5841ed0e70 chore: Croatian translations 2025-04-14 11:31:20 +05:30
Jannat Patel
d217dff4b9 chore: Thai translations 2025-04-14 11:31:18 +05:30
Jannat Patel
2746606db1 chore: Portuguese, Brazilian translations 2025-04-14 11:31:17 +05:30
Jannat Patel
2d321780d0 chore: Bosnian translations 2025-04-14 11:31:16 +05:30
Jannat Patel
c26108586f chore: Persian translations 2025-04-14 11:31:14 +05:30
Jannat Patel
7f30d9c3dc chore: Chinese Simplified translations 2025-04-14 11:31:12 +05:30
Jannat Patel
816b40bdc6 chore: Turkish translations 2025-04-14 11:31:11 +05:30
Jannat Patel
09688315cb chore: Swedish translations 2025-04-14 11:31:10 +05:30
Jannat Patel
c709535442 chore: Russian translations 2025-04-14 11:31:08 +05:30
Jannat Patel
08e2d804fa chore: Polish translations 2025-04-14 11:31:07 +05:30
Jannat Patel
b4fb07b435 chore: Hungarian translations 2025-04-14 11:31:06 +05:30
Jannat Patel
d119ae6409 chore: German translations 2025-04-14 11:31:04 +05:30
Jannat Patel
cf26fc4530 chore: Arabic translations 2025-04-14 11:31:03 +05:30
Jannat Patel
f50a7704c9 chore: Spanish translations 2025-04-14 11:31:02 +05:30
Jannat Patel
facec8393c chore: French translations 2025-04-14 11:31:00 +05:30
Jannat Patel
172e8872ef Merge pull request #1428 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-04-14 10:57:06 +05:30
Jannat Patel
b7755b844a Merge pull request #1427 from frappe/pot_develop_2025-04-11
chore: update POT file
2025-04-14 10:56:54 +05:30
Jannat Patel
7e77d29edb chore: Portuguese translations 2025-04-13 11:08:49 +05:30
Jannat Patel
3b84ef6968 fix: seo description for job openings 2025-04-13 10:26:59 +05:30
Jannat Patel
2dd8192dcb feat: list of certified participants 2025-04-13 10:14:00 +05:30
Jannat Patel
cafb499a79 chore: Portuguese translations 2025-04-12 10:51:50 +05:30
Jannat Patel
f952267396 chore: Arabic translations 2025-04-12 10:51:38 +05:30
frappe-pr-bot
6913b71c69 chore: update POT file 2025-04-11 16:04:23 +00:00
Jannat Patel
c485b03b83 chore: merged conflicts 2025-04-11 19:01:46 +05:30
Jannat Patel
e1f35c86db fix: empty meta info issue 2025-04-11 18:43:34 +05:30
Jannat Patel
cfbe60b731 fix: improved UI of certified participants page 2025-04-11 18:36:12 +05:30
Jannat Patel
a21020e226 Merge pull request #1425 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-04-11 13:21:11 +05:30
Jannat Patel
28d18102f0 chore: Serbian (Latin) translations 2025-04-11 10:37:08 +05:30
Jannat Patel
f5e78b7fdb chore: Persian translations 2025-04-11 10:37:04 +05:30
Jannat Patel
d420b2dae5 chore: Turkish translations 2025-04-11 10:37:01 +05:30
Jannat Patel
3cce9107d0 chore: Polish translations 2025-04-11 10:36:59 +05:30
Jannat Patel
a5248eb92b chore: Hungarian translations 2025-04-11 10:36:57 +05:30
Jannat Patel
1acf734229 chore: German translations 2025-04-11 10:36:56 +05:30
Jannat Patel
cc170ecb20 chore: French translations 2025-04-11 10:36:53 +05:30
Jannat Patel
b7f40d16a4 Merge pull request #1424 from pateljannat/seo-description
feat: SEO Meta Description
2025-04-10 17:42:58 +05:30
Jannat Patel
7e6cb727bd feat: seo description field 2025-04-10 17:19:38 +05:30
Jannat Patel
eeaa835bef fix: redirect to course list from course from if user is not moderator and instructor 2025-04-10 16:36:22 +05:30
Frappe PR Bot
04aff8d149 chore(release): Bumped to Version 2.27.0 2025-04-10 10:38:37 +00:00
Jannat Patel
e88bdd818d Merge pull request #1422 from pateljannat/issues-89
fix: don't update onboarding status if user is not system manager
2025-04-10 15:55:32 +05:30
Jannat Patel
1a5d8ce07e fix: don't update onboarding status if user is not system manager 2025-04-10 15:37:59 +05:30
Jannat Patel
8e405bc8eb Merge pull request #1416 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-04-09 15:01:24 +05:30
Jannat Patel
23e2a153c9 chore: German translations 2025-04-09 10:03:41 +05:30
Jannat Patel
85a0949488 Merge pull request #1415 from pateljannat/jobs-improvements
fix: new ui for job list
2025-04-08 22:53:30 +05:30
Jannat Patel
57b6433dc0 fix: new ui for job list 2025-04-08 22:39:42 +05:30
Jannat Patel
1b43e1be44 fix: added back csrf_token 2025-04-08 21:33:52 +05:30
Jannat Patel
d6738b86c9 Merge pull request #1414 from pateljannat/seo-improvements
fix: seo improvements
2025-04-08 21:09:54 +05:30
Jannat Patel
a5325cef44 chore: renamed lint jobs 2025-04-08 21:03:30 +05:30
Jannat Patel
cc917f3d83 chore: bumped up pre commit action version to 3.0. 2025-04-08 21:00:50 +05:30
Jannat Patel
492917ea40 chore: added commit lint rules 2025-04-08 20:53:44 +05:30
Jannat Patel
78263185a1 chore: cached pip for linters 2025-04-08 20:31:42 +05:30
Jannat Patel
ee7aa9d58b feat: fetch meta tags from website route meta 2025-04-08 19:37:16 +05:30
Jannat Patel
a7112937de fix: changed course and batch meta description 2025-04-08 18:21:28 +05:30
Jannat Patel
a8d4572aef fix: add app_name as document title 2025-04-08 18:12:21 +05:30
Jannat Patel
45c530e53a Merge pull request #1413 from pateljannat/improve-page-meta
fix: persistent favicon for all pages
2025-04-08 11:52:23 +05:30
Jannat Patel
e0bcce5e6e fix: show learning logo as favicon if its missing in website settings 2025-04-08 11:45:51 +05:30
Jannat Patel
8346ec8525 fix: import usePageMeta for QuizSubmission 2025-04-08 11:13:46 +05:30
Jannat Patel
5d1673bad8 fix: persistent favicon for all pages 2025-04-08 10:57:02 +05:30
Jannat Patel
a33328e11d Merge pull request #1412 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-04-08 10:27:35 +05:30
Jannat Patel
3efa326684 chore: Esperanto translations 2025-04-08 09:28:34 +05:30
Jannat Patel
196fead1e0 chore: Croatian translations 2025-04-08 09:28:33 +05:30
Jannat Patel
b8ce04e9fe chore: Thai translations 2025-04-08 09:28:31 +05:30
Jannat Patel
6369dfd65c chore: Portuguese, Brazilian translations 2025-04-08 09:28:30 +05:30
Jannat Patel
f4da56adf9 chore: Bosnian translations 2025-04-08 09:28:29 +05:30
Jannat Patel
0987a91bfc chore: Persian translations 2025-04-08 09:28:27 +05:30
Jannat Patel
9f23a56cf4 chore: Chinese Simplified translations 2025-04-08 09:28:26 +05:30
Jannat Patel
34a4754767 chore: Turkish translations 2025-04-08 09:28:25 +05:30
Jannat Patel
b88de74552 chore: Swedish translations 2025-04-08 09:28:24 +05:30
Jannat Patel
45ac682c7f chore: Russian translations 2025-04-08 09:28:22 +05:30
Jannat Patel
b753d366bf chore: Polish translations 2025-04-08 09:28:21 +05:30
Jannat Patel
06c598886e chore: Hungarian translations 2025-04-08 09:28:20 +05:30
Jannat Patel
52b0b7f8dc chore: German translations 2025-04-08 09:28:18 +05:30
Jannat Patel
656b3b2ebe chore: Arabic translations 2025-04-08 09:28:17 +05:30
Jannat Patel
6bdfbde23f chore: Spanish translations 2025-04-08 09:28:16 +05:30
Jannat Patel
1b9f5eebc0 chore: French translations 2025-04-08 09:28:15 +05:30
Jannat Patel
1f37da08b4 Merge pull request #1411 from pateljannat/disable-signup-settings
feat: signups can now be enabled/disabled from portal settings
2025-04-07 18:22:57 +05:30
Jannat Patel
5bc44e6fe5 feat: signups can now be enabled/disabled from portal settings 2025-04-07 18:15:30 +05:30
Jannat Patel
c70da08078 Merge pull request #1410 from pateljannat/quiz-issue
fix: save lesson details in quiz
2025-04-07 15:19:15 +05:30
Jannat Patel
7600fb14e1 Merge pull request #1409 from frappe/pot_develop_2025-04-04
chore: update POT file
2025-04-07 15:10:55 +05:30
Jannat Patel
e2fdf2042e Merge pull request #1406 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-04-07 15:10:38 +05:30
Jannat Patel
8477d6b9ed fix: save lesson details in quiz 2025-04-07 15:09:31 +05:30
Jannat Patel
241df63334 chore: Persian translations 2025-04-07 09:31:08 +05:30
Jannat Patel
7131de8a2a chore: Persian translations 2025-04-06 09:34:34 +05:30
frappe-pr-bot
473a799f58 chore: update POT file 2025-04-04 16:04:26 +00:00
Jannat Patel
6c9fe85170 chore: Portuguese, Brazilian translations 2025-04-01 09:07:24 +05:30
Jannat Patel
2c5d2db340 chore: Chinese Simplified translations 2025-03-29 08:33:34 +05:30
Jannat Patel
6cd2e6e7fb Merge pull request #1403 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-03-28 10:11:02 +05:30
Jannat Patel
a6b094cff9 chore: Chinese Simplified translations 2025-03-28 08:29:45 +05:30
Jannat Patel
b024a4546c Merge pull request #1401 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-03-27 09:42:07 +05:30
Jannat Patel
519715f8ee chore: Chinese Simplified translations 2025-03-27 08:19:36 +05:30
Jannat Patel
522de390a7 Merge pull request #1399 from pateljannat/onboarding-ui
feat: onboarding
2025-03-26 22:45:52 +05:30
Jannat Patel
2ffe19cea1 chore: removed frappe-ui from workspaces 2025-03-26 22:36:13 +05:30
Jannat Patel
124dc10cc3 chore: fixed linters 2025-03-26 22:15:47 +05:30
Jannat Patel
a41338c3a2 fix: onboarding step improvements 2025-03-26 22:13:08 +05:30
Jannat Patel
aa979b96f2 feat: onboarding 2025-03-26 13:08:06 +05:30
Jannat Patel
f9b2471b32 Merge pull request #1397 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-03-25 13:55:56 +05:30
Jannat Patel
d594f3ac88 chore: Chinese Simplified translations 2025-03-25 07:52:09 +05:30
Jannat Patel
e5190d4409 Merge pull request #1394 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-03-24 15:13:10 +05:30
Jannat Patel
4f876c2bbc Merge pull request #1396 from pateljannat/assignment-in-course-issue
fix: assignment and quiz rendering issue in courses
2025-03-24 15:10:19 +05:30
Jannat Patel
4d031ae55e fix: dark mode issues for assignment 2025-03-24 14:57:46 +05:30
Jannat Patel
89a348b154 fix: assignment and quiz rendering issue in courses 2025-03-24 13:42:24 +05:30
Jannat Patel
db62d40c50 chore: Croatian translations 2025-03-23 07:01:35 +05:30
Jannat Patel
eff2ae8a73 chore: Thai translations 2025-03-23 07:01:33 +05:30
Jannat Patel
b23d29767f chore: Portuguese, Brazilian translations 2025-03-23 07:01:32 +05:30
Jannat Patel
7d5a3c3421 chore: Esperanto translations 2025-03-23 07:01:31 +05:30
Jannat Patel
1054623d9d chore: Bosnian translations 2025-03-23 07:01:30 +05:30
Jannat Patel
4eba93f47b chore: Persian translations 2025-03-23 07:01:28 +05:30
Jannat Patel
13bcc84e8f chore: Chinese Simplified translations 2025-03-23 07:01:27 +05:30
Jannat Patel
c726ad3467 chore: Turkish translations 2025-03-23 07:01:26 +05:30
Jannat Patel
5e95ff963c chore: Swedish translations 2025-03-23 07:01:24 +05:30
Jannat Patel
1ef232e45b chore: Russian translations 2025-03-23 07:01:23 +05:30
Jannat Patel
034654193f chore: Polish translations 2025-03-23 07:01:22 +05:30
Jannat Patel
bddaa26d5a chore: Hungarian translations 2025-03-23 07:01:19 +05:30
Jannat Patel
b42648fecb chore: German translations 2025-03-23 07:01:18 +05:30
Jannat Patel
aa800bf96b chore: Arabic translations 2025-03-23 07:01:17 +05:30
Jannat Patel
6575e139b5 chore: Spanish translations 2025-03-23 07:01:15 +05:30
Jannat Patel
c5b3460006 chore: French translations 2025-03-23 07:01:14 +05:30
Jannat Patel
b1e490765b Merge pull request #1393 from frappe/pot_develop_2025-03-21
chore: update POT file
2025-03-22 13:22:38 +05:30
Jannat Patel
c0f4a09e22 fix: removed page_renderer for courses 2025-03-22 11:18:03 +05:30
frappe-pr-bot
8fb5311844 chore: update POT file 2025-03-21 16:04:11 +00:00
Jannat Patel
12122f1eaf Merge pull request #1391 from pateljannat/issues-88
fix: removed user info from assignment block
2025-03-21 12:46:51 +05:30
Jannat Patel
e83312289b fix: removed user info from assignment block 2025-03-21 12:34:49 +05:30
Jannat Patel
d59f4113c1 Merge pull request #1389 from pateljannat/issues-87
fix: course tags issue when getting course details
2025-03-21 06:00:06 +05:30
Jannat Patel
8e3b70e7c8 fix: course tags issue when getting course details 2025-03-21 05:53:24 +05:30
Jannat Patel
c25d95b3b6 Merge pull request #1386 from pateljannat/issues-85
fix: misc issues
2025-03-20 12:29:25 +05:30
Jannat Patel
edde95edeb chore: fixed linters 2025-03-20 12:22:13 +05:30
Jannat Patel
066eaea45d fix: redirection to FC site without checking payment method 2025-03-20 11:57:08 +05:30
Jannat Patel
7ae3cf5d95 fix: check seats of a batch at the time of billing 2025-03-20 11:03:08 +05:30
Jannat Patel
2fa728d45c Merge pull request #1383 from NihalRoshanCK/develop
change the text color according to the theme
2025-03-19 22:42:47 +05:30
Jannat Patel
04cbd6a1d8 chore: use vite plugins from frappe-ui 2025-03-19 22:26:58 +05:30
Jannat Patel
c6e658e26b fix: show tabs and featured courses on list for guest users 2025-03-19 11:04:29 +05:30
Jannat Patel
0692aceda4 fix: don't allow billing page access if batch is sold out 2025-03-19 10:45:47 +05:30
Nihal Roshan
072bef5847 change the text color according to the theme 2025-03-18 10:15:52 +00:00
Jannat Patel
e94a689f83 Merge pull request #1382 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-03-17 15:55:35 +05:30
Jannat Patel
c71a980f78 chore: Chinese Simplified translations 2025-03-17 05:54:46 +05:30
Jannat Patel
ef7d850dd4 chore: Persian translations 2025-03-16 05:49:18 +05:30
Jannat Patel
1e6a71f36b chore: Croatian translations 2025-03-15 05:16:33 +05:30
Jannat Patel
f5ae4120cd chore: Thai translations 2025-03-15 05:16:32 +05:30
Jannat Patel
82331364b7 chore: Portuguese, Brazilian translations 2025-03-15 05:16:31 +05:30
Jannat Patel
ef3879e419 chore: Esperanto translations 2025-03-15 05:16:30 +05:30
Jannat Patel
403dbf13e8 chore: Bosnian translations 2025-03-15 05:16:28 +05:30
Jannat Patel
c8193c0009 chore: Persian translations 2025-03-15 05:16:27 +05:30
Jannat Patel
9c0c69a728 chore: Chinese Simplified translations 2025-03-15 05:16:26 +05:30
Jannat Patel
4606fc3e2a chore: Turkish translations 2025-03-15 05:16:24 +05:30
Jannat Patel
c9bb3ab368 chore: Swedish translations 2025-03-15 05:16:23 +05:30
Jannat Patel
99e4b406a4 chore: Russian translations 2025-03-15 05:16:22 +05:30
Jannat Patel
67b9424b9e chore: Polish translations 2025-03-15 05:16:20 +05:30
Jannat Patel
5b60be5f51 chore: Hungarian translations 2025-03-15 05:16:19 +05:30
Jannat Patel
d88927a6fb chore: German translations 2025-03-15 05:16:18 +05:30
Jannat Patel
6616ee3607 chore: Arabic translations 2025-03-15 05:16:16 +05:30
Jannat Patel
0dbd8de335 chore: Spanish translations 2025-03-15 05:16:15 +05:30
Jannat Patel
9b406e368b chore: French translations 2025-03-15 05:16:14 +05:30
Jannat Patel
4449dc43a0 Merge pull request #1381 from frappe/pot_develop_2025-03-14
chore: update POT file
2025-03-14 21:43:26 +05:30
frappe-pr-bot
554093ab3e chore: update POT file 2025-03-14 16:04:06 +00:00
Jannat Patel
ac3ed22ae9 Merge pull request #1380 from pateljannat/issues-84
fix: moved evaluation cancel button in a menu
2025-03-13 22:17:39 +05:30
Jannat Patel
2ca7b09d1e fix: made address amount and currency mandatory in LMS Payment 2025-03-13 22:01:26 +05:30
Jannat Patel
f29c2da9ce fix: moved evaluation cancel button in a menu 2025-03-13 22:00:39 +05:30
Jannat Patel
e23f6ae0fa Merge pull request #1378 from pateljannat/issues-83
fix: batch reminder email subject and content
2025-03-13 08:27:49 +05:30
Jannat Patel
51061273bc fix: show view certificate link on course page, if already certified 2025-03-13 06:08:39 +05:30
Jannat Patel
4a0812dfe9 fix: batch reminder email subject and content 2025-03-13 06:08:00 +05:30
Md Hussain Nagaria
efb694a6e6 Merge pull request #1377 from frappe/state-enhancement
fix: misc
2025-03-12 14:48:54 +05:30
Hussain Nagaria
1dbe2f31d0 fix: linter 2025-03-12 14:40:45 +05:30
Hussain Nagaria
be9525dbf2 fix: empty query string with trailing ?
Fixes #1376
2025-03-12 14:37:50 +05:30
Hussain Nagaria
a24afad641 chore: more dead code 2025-03-12 14:36:07 +05:30
Hussain Nagaria
abd14aa33c chore: remove dead code 2025-03-12 14:31:53 +05:30
Hussain Nagaria
5b3c0685ac feat: track current tab in batches and courses page 2025-03-12 14:08:25 +05:30
Jannat Patel
2a59d9ff04 Merge pull request #1374 from pateljannat/issues-82
fix: check enrollment on course certification page
2025-03-12 11:03:20 +05:30
Jannat Patel
619dc73bcb fix: show created tab to users with moderator or instructor role 2025-03-12 10:50:20 +05:30
Jannat Patel
02edefc158 fix: check enrollment on course certification page 2025-03-12 10:49:44 +05:30
Jannat Patel
572f5ae585 Merge pull request #1370 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-03-11 10:39:34 +05:30
Jannat Patel
a326866cc9 chore: Croatian translations 2025-03-11 04:16:36 +05:30
Jannat Patel
17decf7b71 chore: Thai translations 2025-03-11 04:16:34 +05:30
Jannat Patel
b9784e22ff chore: Portuguese, Brazilian translations 2025-03-11 04:16:33 +05:30
Jannat Patel
0f600c5b70 chore: Esperanto translations 2025-03-11 04:16:32 +05:30
Jannat Patel
a606e9c974 chore: Bosnian translations 2025-03-11 04:16:30 +05:30
Jannat Patel
9e1938095c chore: Persian translations 2025-03-11 04:16:29 +05:30
Jannat Patel
3491eb3881 chore: Chinese Simplified translations 2025-03-11 04:16:28 +05:30
Jannat Patel
6277340d6b chore: Turkish translations 2025-03-11 04:16:26 +05:30
Jannat Patel
0c12ee4452 chore: Swedish translations 2025-03-11 04:16:25 +05:30
Jannat Patel
4ec245a119 chore: Russian translations 2025-03-11 04:16:23 +05:30
Jannat Patel
24fa6d17de chore: Polish translations 2025-03-11 04:16:22 +05:30
Jannat Patel
2eedc1032c chore: Hungarian translations 2025-03-11 04:16:20 +05:30
Jannat Patel
8c3b1b433f chore: German translations 2025-03-11 04:16:19 +05:30
Jannat Patel
ae3f0f9a4e chore: Arabic translations 2025-03-11 04:16:18 +05:30
Jannat Patel
f4ae601f0d chore: Spanish translations 2025-03-11 04:16:16 +05:30
Jannat Patel
2104b86080 chore: French translations 2025-03-11 04:16:15 +05:30
Jannat Patel
9724dceb73 Merge pull request #1368 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-03-10 11:24:09 +05:30
Jannat Patel
4c07a4f35d Merge pull request #1366 from frappe/pot_develop_2025-03-07
chore: update POT file
2025-03-10 11:23:56 +05:30
Jannat Patel
6a15697957 chore: Croatian translations 2025-03-10 03:59:01 +05:30
frappe-pr-bot
47f880d8dc chore: update POT file 2025-03-07 16:04:14 +00:00
Jannat Patel
d5814f5680 Merge pull request #1365 from pateljannat/issues-81
fix: youtube embed issue
2025-03-07 12:32:26 +05:30
Jannat Patel
345a444d73 fix: youtube embed issue 2025-03-07 12:18:52 +05:30
Jannat Patel
0053ce5602 Merge pull request #1364 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-03-06 16:14:45 +05:30
Jannat Patel
9851757a4e chore: Croatian translations 2025-03-06 03:44:37 +05:30
Jannat Patel
55fe25b8cb chore: Thai translations 2025-03-06 03:44:36 +05:30
Jannat Patel
714f8a17c3 chore: Portuguese, Brazilian translations 2025-03-06 03:44:35 +05:30
Jannat Patel
732e9db9af chore: Bosnian translations 2025-03-06 03:44:33 +05:30
Jannat Patel
6fbc448a52 chore: Persian translations 2025-03-06 03:44:32 +05:30
Jannat Patel
76fc241778 chore: Polish translations 2025-03-06 03:44:27 +05:30
Jannat Patel
51cbbfdc45 chore: German translations 2025-03-06 03:44:25 +05:30
Jannat Patel
279f2f503e chore: Arabic translations 2025-03-06 03:44:24 +05:30
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
Frappe PR Bot
34548b93f4 chore(release): Bumped to Version 2.18.0 2025-01-02 14:31:37 +00:00
Jannat Patel
f438d33f75 Merge pull request #1224 from pateljannat/issues-60
fix: quiz api issue
2025-01-02 20:00:36 +05:30
Jannat Patel
be1c0de4c6 fix: quiz api issue 2025-01-02 19:27:35 +05:30
Jannat Patel
ae5ea9a8aa Merge pull request #1223 from pateljannat/assignments-in-courses
feat: assignments in courses
2025-01-02 15:45:32 +05:30
Jannat Patel
eeb7fb1f78 fix: correct path for assignment plugin 2025-01-02 15:32:43 +05:30
Jannat Patel
3f32d5bb3b feat: notification to student on submission update 2025-01-02 15:22:32 +05:30
Jannat Patel
12019ca37d Merge pull request #1219 from frappe/l10n_develop2
chore: sync translations from crowdin
2025-01-02 15:20:15 +05:30
Jannat Patel
4d133b2f99 fix: assignment dirty state and comments view to student 2025-01-02 14:53:38 +05:30
Jannat Patel
e733226b0c chore: Persian translations 2025-01-01 09:00:19 +05:30
Jannat Patel
2ed583a0c3 fix: assignment submission ux improvements 2024-12-31 23:06:55 +05:30
Jannat Patel
048cee654e fix: mark lesson progress when quiz and assignment are submitted 2024-12-31 13:15:25 +05:30
Jannat Patel
1293294593 feat: assignment in lesson 2024-12-31 12:20:01 +05:30
Jannat Patel
a1947a3106 Merge pull request #1215 from frappe/l10n_develop2
chore: sync translations from crowdin
2024-12-31 12:00:34 +05:30
Jannat Patel
eff6cd6bbe chore: Esperanto translations 2024-12-31 08:59:24 +05:30
Jannat Patel
d784ac5699 chore: Bosnian translations 2024-12-31 08:59:23 +05:30
Jannat Patel
9acad5157b chore: Persian translations 2024-12-31 08:59:21 +05:30
Jannat Patel
94459efa3f chore: Chinese Simplified translations 2024-12-31 08:59:20 +05:30
Jannat Patel
e88bc6a5ce chore: Turkish translations 2024-12-31 08:59:19 +05:30
Jannat Patel
55a7ab54e9 chore: Swedish translations 2024-12-31 08:59:17 +05:30
Jannat Patel
0c324c87cc chore: Russian translations 2024-12-31 08:59:16 +05:30
Jannat Patel
31e8befa11 chore: Polish translations 2024-12-31 08:59:14 +05:30
Jannat Patel
86ab7a6d97 chore: Hungarian translations 2024-12-31 08:59:13 +05:30
Jannat Patel
14bdfb2d98 chore: German translations 2024-12-31 08:59:11 +05:30
Jannat Patel
0036e585da chore: Arabic translations 2024-12-31 08:59:10 +05:30
Jannat Patel
cba2343fc0 chore: Spanish translations 2024-12-31 08:59:08 +05:30
Jannat Patel
864eebce2f chore: French translations 2024-12-31 08:59:07 +05:30
Jannat Patel
156d36fb5e chore: merged conflicts 2024-12-30 18:21:47 +05:30
Jannat Patel
068718aa8a Merge pull request #1214 from pateljannat/issues-59
fix: progress issue in batches
2024-12-30 18:08:09 +05:30
Jannat Patel
10219abfd6 fix: progress issue in batches 2024-12-30 17:51:14 +05:30
Jannat Patel
2ec231a3d0 Merge pull request #1213 from frappe/pot_develop_2024-12-27
chore: update POT file
2024-12-30 11:34:01 +05:30
frappe-pr-bot
78f29b3aff chore: update POT file 2024-12-27 16:04:15 +00:00
Jannat Patel
7f768e81f4 feat: assignment grading 2024-12-26 18:16:46 +05:30
Jannat Patel
aa1460eda1 Merge pull request #1211 from frappe/l10n_develop2
chore: sync translations from crowdin
2024-12-26 14:57:32 +05:30
Jannat Patel
85f85063ac feat: assignment submission list 2024-12-26 11:28:32 +05:30
Jannat Patel
0a7ce3c5d8 chore: Esperanto translations 2024-12-25 07:47:58 +05:30
Jannat Patel
8468d0e3db chore: Bosnian translations 2024-12-25 07:47:56 +05:30
Jannat Patel
059ac27f0b chore: Persian translations 2024-12-25 07:47:55 +05:30
Jannat Patel
a96f8836b1 chore: Chinese Simplified translations 2024-12-25 07:47:54 +05:30
Jannat Patel
4018116136 chore: Turkish translations 2024-12-25 07:47:52 +05:30
Jannat Patel
aa083c8a40 chore: Swedish translations 2024-12-25 07:47:51 +05:30
Jannat Patel
8752243e9c chore: Russian translations 2024-12-25 07:47:50 +05:30
Jannat Patel
1d028e81c4 chore: Polish translations 2024-12-25 07:47:48 +05:30
Jannat Patel
2752d3e42c chore: Hungarian translations 2024-12-25 07:47:47 +05:30
Jannat Patel
aa074ef762 chore: German translations 2024-12-25 07:47:45 +05:30
Jannat Patel
bae75cd2f6 chore: Arabic translations 2024-12-25 07:47:44 +05:30
Jannat Patel
81a714b5a2 chore: Spanish translations 2024-12-25 07:47:41 +05:30
Jannat Patel
10cd44c22f chore: French translations 2024-12-25 07:47:40 +05:30
Jannat Patel
a44f59c362 feat: assignments list and form 2024-12-24 21:48:45 +05:30
Jannat Patel
8d372fcab4 Merge pull request #1204 from frappe/l10n_develop2
chore: sync translations from crowdin
2024-12-24 10:52:10 +05:30
Jannat Patel
97d6c518b5 Merge pull request #1203 from frappe/pot_develop_2024-12-20
chore: update POT file
2024-12-24 10:52:00 +05:30
Jannat Patel
f331c48e1d Merge pull request #1201 from pateljannat/batch-dashboard-2
feat: batch student progress modal
2024-12-23 18:36:44 +05:30
Jannat Patel
9d0b10058d fix: show dashboard to evaluators too 2024-12-23 17:39:37 +05:30
Jannat Patel
4ccd3ba71e fix: legends 2024-12-23 17:19:48 +05:30
Jannat Patel
7a6f5a868c Merge branch 'develop' of https://github.com/frappe/lms into batch-dashboard-2 2024-12-23 12:49:32 +05:30
Jannat Patel
0fae11d031 docs: updated self hosting steps in README 2024-12-23 12:46:02 +05:30
Jannat Patel
8a9725c990 ci: updated the credentials for building docker image 2024-12-23 12:29:21 +05:30
Jannat Patel
d0189b0e3a ci: updated the credentials for building docker image 2024-12-23 12:28:31 +05:30
Jannat Patel
c6853cc95e Merge pull request #1208 from pateljannat/issues-58
ci: added back arch for building docker image
2024-12-23 12:14:36 +05:30
Jannat Patel
f28f37fb2c ci: added back arch for building docker image 2024-12-23 12:14:00 +05:30
Jannat Patel
7dbbe9dba4 Merge pull request #1206 from pateljannat/issues-57
fix: markdown embed and paste issue
2024-12-23 11:49:55 +05:30
Jannat Patel
b625d9b099 fix: markdown embed and paste issue 2024-12-23 11:33:09 +05:30
Jannat Patel
a85c81a4b4 chore: Bosnian translations 2024-12-23 07:45:35 +05:30
Jannat Patel
1677a4a32b chore: Persian translations 2024-12-23 07:45:33 +05:30
Jannat Patel
776d46f5a2 chore: Chinese Simplified translations 2024-12-23 07:45:32 +05:30
Jannat Patel
6384eeaa13 chore: Turkish translations 2024-12-23 07:45:30 +05:30
Jannat Patel
fdc0befcee chore: Russian translations 2024-12-23 07:45:27 +05:30
Jannat Patel
f2c28eb695 chore: Polish translations 2024-12-23 07:45:26 +05:30
Jannat Patel
4095916991 chore: Hungarian translations 2024-12-23 07:45:25 +05:30
Jannat Patel
551703364a chore: German translations 2024-12-23 07:45:23 +05:30
Jannat Patel
4a2fae023c chore: Arabic translations 2024-12-23 07:45:22 +05:30
Jannat Patel
fca206120e chore: Spanish translations 2024-12-23 07:45:21 +05:30
Jannat Patel
65b2199065 chore: French translations 2024-12-23 07:45:19 +05:30
Jannat Patel
9d03a52bf9 chore: Swedish translations 2024-12-21 07:21:24 +05:30
frappe-pr-bot
c8aa44dfcb chore: update POT file 2024-12-20 16:04:18 +00:00
Jannat Patel
7fcbe85ab9 Merge pull request #1202 from pateljannat/docker-production-image
ci: container image for production setup
2024-12-20 13:50:19 +05:30
Jannat Patel
de0dea7df8 ci: container image for production setup 2024-12-20 13:27:07 +05:30
Jannat Patel
43cf7d04b8 feat: batch dashboard for instructors 2024-12-20 13:12:40 +05:30
Jannat Patel
4d18580482 feat: batch student progress modal 2024-12-19 23:00:28 +05:30
Frappe PR Bot
b48e007ea8 chore(release): Bumped to Version 2.17.0 2024-12-18 14:51:51 +00:00
Jannat Patel
d5e8973866 Merge pull request #1196 from frappe/l10n_develop2
chore: sync translations from crowdin
2024-12-18 14:18:51 +05:30
Jannat Patel
a8c530f98c chore: Esperanto translations 2024-12-18 06:07:10 +05:30
Jannat Patel
47769ccd62 Merge pull request #1195 from pateljannat/issues-56
feat: load more in quiz list
2024-12-17 18:33:39 +05:30
Jannat Patel
bfc1d9a0a8 feat: load more in quiz list 2024-12-17 17:48:49 +05:30
Jannat Patel
824484e608 Merge pull request #1194 from pateljannat/issues-55
fix: markdown parser link issue
2024-12-17 16:57:31 +05:30
Jannat Patel
d3f7baae4c fix: markdown parser link issue 2024-12-17 16:35:30 +05:30
Jannat Patel
8d961e9b71 Merge pull request #1193 from pateljannat/issues-54
feat: load more in quiz submissions
2024-12-17 12:48:53 +05:30
Jannat Patel
f22855920c Merge pull request #1192 from frappe/l10n_develop2
chore: sync translations from crowdin
2024-12-17 12:24:18 +05:30
Jannat Patel
18728e3519 Merge pull request #1186 from frappe/pot_develop_2024-12-13
chore: update POT file
2024-12-17 12:24:06 +05:30
Jannat Patel
65dc2838d3 feat: load more in quiz submissions 2024-12-17 12:23:44 +05:30
Jannat Patel
be930ce076 chore: Persian translations 2024-12-17 05:37:38 +05:30
Jannat Patel
1ea47a008c Merge pull request #1191 from pateljannat/scormcontent-issue
fix: scormcontent package load issue
2024-12-16 19:25:32 +05:30
Jannat Patel
e0169cff79 fix: scormcontent package load issue 2024-12-16 19:12:15 +05:30
Jannat Patel
7c53ac10e2 Merge pull request #1189 from pateljannat/lesson-md-parser
feat: markdown parser for lessons
2024-12-16 18:29:45 +05:30
Jannat Patel
212e0de6e9 chore: resolved conflicts 2024-12-16 18:13:49 +05:30
Jannat Patel
8e74384b5a Merge branch 'develop' of https://github.com/frappe/lms into lesson-md-parser 2024-12-16 18:12:17 +05:30
Jannat Patel
86e7e68ce1 chore: removed unused packages 2024-12-16 18:12:13 +05:30
Jannat Patel
a77999dbb6 Merge pull request #1190 from pateljannat/quiz-marks-issue
fix: delete quiz and submission before deleting course
2024-12-16 18:10:23 +05:30
Jannat Patel
3288fb0f06 chore: replace mariadb-client-10.6 with mariadb-client for ui tests 2024-12-16 17:56:57 +05:30
Jannat Patel
a81b384f90 fix: mariadb dependency installation 2024-12-16 17:30:00 +05:30
Jannat Patel
75c11d3fcc fix: course category was not reflecting on course form 2024-12-16 17:21:12 +05:30
Jannat Patel
51a6cc035c fix: delete quiz and submission before deleting course 2024-12-16 17:14:30 +05:30
Jannat Patel
ae8008d05c chore: bumped up mariadb image version 2024-12-16 17:00:55 +05:30
Jannat Patel
7f44177986 feat: markdown parser for links and lists 2024-12-16 16:41:55 +05:30
Jannat Patel
d88aaedf3f Merge branch 'develop' of https://github.com/frappe/lms into lesson-md-parser 2024-12-16 16:40:32 +05:30
frappe-pr-bot
802d4ccb0b chore: update POT file 2024-12-13 16:04:40 +00:00
Jannat Patel
76a84c7f5d Merge pull request #1183 from pateljannat/batch-dashboard
feat: show student course and assessment progress on batch page
2024-12-13 12:11:59 +05:30
Jannat Patel
40aefba203 fix: styling of batch list headers 2024-12-13 12:03:00 +05:30
Jannat Patel
6cdfb822b4 fix: batch time issue 2024-12-13 11:45:54 +05:30
Jannat Patel
fdacab66f7 feat: show student course and assessment progress on batch page 2024-12-13 10:44:56 +05:30
Jannat Patel
5cc12e71df Merge pull request #1182 from pateljannat/fix-readme-2
docs: updated readme header, footer and screenshots
2024-12-12 16:33:24 +05:30
Jannat Patel
f5e5fa2f36 docs: fixed screenshot captions 2024-12-12 16:18:54 +05:30
Jannat Patel
6022b83b8c docs: removed extra space under screenshots 2024-12-12 16:06:30 +05:30
Jannat Patel
a01b1657cc docs: added captions to README screenshots 2024-12-12 16:05:39 +05:30
Jannat Patel
6b785bd0e6 docs: updated readme header, footer and screenshots 2024-12-12 15:59:57 +05:30
Jannat Patel
0beffc3083 Merge pull request #1181 from pateljannat/fix-readme
fix: readme
2024-12-11 12:49:37 +05:30
Jannat Patel
d345d09b13 fix: readme 2024-12-11 12:49:11 +05:30
Jannat Patel
38e1eb8fc7 feat: markdown parser for lessons 2024-12-11 11:57:35 +05:30
303 changed files with 68652 additions and 17305 deletions

BIN
.github/batch.png vendored Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.1 MiB

BIN
.github/batches.png vendored

Binary file not shown.

Before

Width:  |  Height:  |  Size: 572 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 307 KiB

After

Width:  |  Height:  |  Size: 912 KiB

View File

@@ -5,7 +5,7 @@ echo "Setting Up System Dependencies..."
sudo apt update sudo apt update
sudo apt remove mysql-server mysql-client sudo apt remove mysql-server mysql-client
sudo apt install libcups2-dev redis-server mariadb-client-10.6 sudo apt-get install libcups2-dev redis-server mariadb-client
install_wkhtmltopdf() { install_wkhtmltopdf() {
wget -q https://github.com/wkhtmltopdf/packaging/releases/download/0.12.6-1/wkhtmltox_0.12.6-1.focal_amd64.deb wget -q https://github.com/wkhtmltopdf/packaging/releases/download/0.12.6-1/wkhtmltox_0.12.6-1.focal_amd64.deb

BIN
.github/hero.png vendored

Binary file not shown.

Before

Width:  |  Height:  |  Size: 842 KiB

After

Width:  |  Height:  |  Size: 2.0 MiB

BIN
.github/quiz.png vendored

Binary file not shown.

Before

Width:  |  Height:  |  Size: 578 KiB

After

Width:  |  Height:  |  Size: 1.0 MiB

64
.github/workflows/build.yml vendored Normal file
View File

@@ -0,0 +1,64 @@
name: Build Container Image
on:
workflow_dispatch:
push:
branches:
- main
tags:
- "*"
jobs:
build:
name: Build
runs-on: ubuntu-latest
strategy:
matrix:
arch: [amd64, arm64]
permissions:
packages: write
steps:
- name: Checkout Entire Repository
uses: actions/checkout@v4
- name: Set up QEMU
uses: docker/setup-qemu-action@v3
- name: Set up Docker Buildx
uses: docker/setup-buildx-action@v3
with:
platforms: linux/${{ matrix.arch }}
- name: Login to GitHub Container Registry
uses: docker/login-action@v2
with:
registry: ghcr.io
username: ${{ github.actor }}
password: ${{ secrets.GITHUB_TOKEN }}
- name: Set Branch
run: |
export APPS_JSON='[{"url": "https://github.com/frappe/lms","branch": "main"}]'
echo "APPS_JSON_BASE64=$(echo $APPS_JSON | base64 -w 0)" >> $GITHUB_ENV
echo "FRAPPE_BRANCH=version-15" >> $GITHUB_ENV
- name: Set Image Tag
run: |
echo "IMAGE_TAG=stable" >> $GITHUB_ENV
- uses: actions/checkout@v4
with:
repository: frappe/frappe_docker
path: builds
- name: Build and push
uses: docker/build-push-action@v6
with:
push: true
context: builds
file: builds/images/layered/Containerfile
tags: >
ghcr.io/${{ github.repository }}:${{ github.ref_name }},
ghcr.io/${{ github.repository }}:${{ env.IMAGE_TAG }}
build-args: |
"FRAPPE_BRANCH=${{ env.FRAPPE_BRANCH }}"
"APPS_JSON_BASE64=${{ env.APPS_JSON_BASE64 }}"

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

@@ -7,8 +7,27 @@ on:
branches: [ main ] branches: [ main ]
jobs: jobs:
commit-lint:
name: 'Semantic Commits'
runs-on: ubuntu-latest
if: github.event_name == 'pull_request'
steps:
- uses: actions/checkout@v4
with:
fetch-depth: 200
- uses: actions/setup-node@v4
with:
node-version: 20
check-latest: true
- name: Check commit titles
run: |
npm install @commitlint/cli @commitlint/config-conventional
npx commitlint --verbose --from ${{ github.event.pull_request.base.sha }} --to ${{ github.event.pull_request.head.sha }}
linters: linters:
name: Semantic Commits name: Semgrep Rules
runs-on: ubuntu-latest runs-on: ubuntu-latest
if: github.event_name == 'pull_request' if: github.event_name == 'pull_request'
@@ -20,8 +39,17 @@ jobs:
with: with:
python-version: '3.10' python-version: '3.10'
- name: Cache pip
uses: actions/cache@v4
with:
path: ~/.cache/pip
key: ${{ runner.os }}-pip-${{ hashFiles('**/*requirements.txt', '**/pyproject.toml', '**/setup.py') }}
restore-keys: |
${{ runner.os }}-pip-
${{ runner.os }}-
- name: Install and Run Pre-commit - name: Install and Run Pre-commit
uses: pre-commit/action@v2.0.3 uses: pre-commit/action@v3.0.1
- name: Download Semgrep rules - name: Download Semgrep rules
run: git clone --depth 1 https://github.com/frappe/semgrep-rules.git frappe-semgrep-rules run: git clone --depth 1 https://github.com/frappe/semgrep-rules.git frappe-semgrep-rules

View File

@@ -1,8 +1,7 @@
name: Create weekly release name: Create weekly release
on: on:
schedule: schedule:
# 13:00 UTC -> 7pm IST on every Wednesday - cron: '30 4 15 * *'
- cron: '30 4 * * 3'
workflow_dispatch: workflow_dispatch:
jobs: jobs:

View File

@@ -24,7 +24,7 @@ jobs:
services: services:
mariadb: mariadb:
image: mariadb:10.6 image: mariadb:10.8
env: env:
MARIADB_ROOT_PASSWORD: 123 MARIADB_ROOT_PASSWORD: 123
ports: ports:
@@ -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') }}
@@ -70,7 +70,7 @@ jobs:
id: yarn-cache-dir-path id: yarn-cache-dir-path
run: echo "dir=$(yarn cache dir)" >> $GITHUB_OUTPUT run: echo "dir=$(yarn cache dir)" >> $GITHUB_OUTPUT
- uses: actions/cache@v3 - uses: actions/cache@v4
id: yarn-cache id: yarn-cache
with: with:
path: ${{ steps.yarn-cache-dir-path.outputs.dir }} path: ${{ steps.yarn-cache-dir-path.outputs.dir }}
@@ -79,7 +79,7 @@ jobs:
${{ runner.os }}-yarn-ui- ${{ runner.os }}-yarn-ui-
- name: Cache cypress binary - name: Cache cypress binary
uses: actions/cache@v3 uses: actions/cache@v4
with: with:
path: ~/.cache/Cypress path: ~/.cache/Cypress
key: ${{ runner.os }}-cypress key: ${{ runner.os }}-cypress
@@ -100,6 +100,7 @@ jobs:
bench --site lms.test execute frappe.utils.install.complete_setup_wizard bench --site lms.test execute frappe.utils.install.complete_setup_wizard
bench --site lms.test execute frappe.tests.ui_test_helpers.create_test_user bench --site lms.test execute frappe.tests.ui_test_helpers.create_test_user
bench --site lms.test set-password frappe@example.com admin bench --site lms.test set-password frappe@example.com admin
bench --site lms.test execute lms.lms.utils.persona_captured
- name: cypress pre-requisites - name: cypress pre-requisites
run: | run: |

View File

@@ -1,11 +1,10 @@
<div align="center" markdown="1"> <div align="center" markdown="1">
<img src=".github/lms-logo.png" alt="Frappe Learning logo" width="100"/> <img src=".github/lms-logo.png" alt="Frappe Learning logo" width="80" height="80"/>
<h1>Frappe Learning</h1> <h1>Frappe Learning</h1>
**Easy to use, open source, Learning Management System** **Easy to use, open source, Learning Management System**
![GitHub release (latest by date)](https://img.shields.io/github/v/release/frappe/lms)
![Tests](https://img.shields.io/endpoint?url=https://dashboard.cypress.io/badge/simple/vandxn/main&style=flat&logo=cypress) ![Tests](https://img.shields.io/endpoint?url=https://dashboard.cypress.io/badge/simple/vandxn/main&style=flat&logo=cypress)
</div> </div>
@@ -24,10 +23,10 @@
## Frappe Learning ## Frappe Learning
Frappe Learning is an easy-to-use learning system that helps you bring structure to your content. Frappe Learning is an easy-to-use learning system that helps you bring structure to your content.
## Motivation ### Motivation
In 2021, we were looking for a Learning Management System to launch [Mon.School](https://mon.school) for FOSS United. We checked out Moodle, but it didnt feel right. The forms were unnecessarily lengthy and the UI was confusing. It shouldn't be this hard to create a course right? So I started making a learning system for Mon.School which soon became a product in itself. The aim is to have a simple platform that anyone can use to launch a course of their own and make knowledge sharing easier. In 2021, we were looking for a Learning Management System to launch [Mon.School](https://mon.school) for FOSS United. We checked out Moodle, but it didnt feel right. The forms were unnecessarily lengthy and the UI was confusing. It shouldn't be this hard to create a course right? So I started making a learning system for Mon.School which soon became a product in itself. The aim is to have a simple platform that anyone can use to launch a course of their own and make knowledge sharing easier.
## Key Features ### Key Features
- **Structured Learning**: Design a course with a 3-level hierarchy, where your courses have chapters and you can group your lessons within these chapters. This ensures that the context of the lesson is set by the chapter. - **Structured Learning**: Design a course with a 3-level hierarchy, where your courses have chapters and you can group your lessons within these chapters. This ensures that the context of the lesson is set by the chapter.
@@ -37,24 +36,42 @@ In 2021, we were looking for a Learning Management System to launch [Mon.School]
- **Getting Certified**: Once a learner has completed the course or batch, you can grant them a certificate. The app provides an inbuilt certificate template. You can use this or else create a template of your own and use that instead. - **Getting Certified**: Once a learner has completed the course or batch, you can grant them a certificate. The app provides an inbuilt certificate template. You can use this or else create a template of your own and use that instead.
### Batches to group learners <details>
<summary>View Screenshots</summary>
![Batch](.github/batches.png)
### Quiz to evaluate them ![Batch](.github/batch.png)
<div align="center">
<sub>
Create batches to group your learners
</sub>
</div>
<br>
![Quiz](.github/quiz.png) ![Quiz](.github/quiz.png)
<div align="center">
<sub>
Evaluate their knowledge by quizzes
</sub>
</div>
<br>
### Certificate to authenticate their knowledge
![Cerficicate](.github/certificate.png) ![Cerficicate](.github/certificate.png)
<div align="center">
<sub>
Autenticate their work with certification
</sub>
</div>
</details>
## Under the Hood
- [**Frappe Framework**](https://github.com/frappe/frappe): A full-stack web application framework written in Python and Javascript. The framework provides a robust foundation for building web applications, including a database abstraction layer, user authentication, and a REST API. ### Under the Hood
- [**Frappe UI**](https://github.com/frappe/frappe-ui): A Vue-based UI library, to provide a modern user interface. The Frappe UI library provides a variety of components that can be used to build single-page applications on top of the Frappe Framework. - [**Frappe Framework**](https://github.com/frappe/frappe): A full-stack web application framework.
- [**Frappe UI**](https://github.com/frappe/frappe-ui): A Vue-based UI library, to provide a modern user interface.
## Production Setup ## Production Setup
@@ -89,15 +106,15 @@ wget https://frappe.io/easy-install.py
python3 ./easy-install.py deploy \ python3 ./easy-install.py deploy \
--project=learning_prod_setup \ --project=learning_prod_setup \
--email=your_email.example.com \ --email=your_email.example.com \
--image=ghcr.io/frappe/learning \ --image=ghcr.io/frappe/lms \
--version=stable \ --version=stable \
--app=learning \ --app=lms \
--sitename subdomain.domain.tld --sitename subdomain.domain.tld
``` ```
Replace the following parameters with your values: Replace the following parameters with your values:
- `your_email.example.com`: Your email address - `your_email.example.com`: Your email address
- `subdomain.domain.tld`: Your domain name where Insights will be hosted - `subdomain.domain.tld`: Your domain name where Learning will be hosted
The script will set up a production-ready instance of Frappe Learning with all the necessary configurations in about 5 minutes. The script will set up a production-ready instance of Frappe Learning with all the necessary configurations in about 5 minutes.
@@ -113,16 +130,16 @@ You need Docker, docker-compose and git setup on your machine. Refer [Docker doc
cd frappe-learning cd frappe-learning
# Download the docker-compose file # Download the docker-compose file
wget -O docker-compose.yml https://raw.githubusercontent.com/frappe/insights/develop/docker/docker-compose.yml wget -O docker-compose.yml https://raw.githubusercontent.com/frappe/lms/develop/docker/docker-compose.yml
# Download the setup script # Download the setup script
wget -O init.sh https://raw.githubusercontent.com/frappe/insights/develop/docker/init.sh wget -O init.sh https://raw.githubusercontent.com/frappe/lms/develop/docker/init.sh
**Step 2**: Run the container and daemonize it **Step 2**: Run the container and daemonize it
docker compose up -d docker compose up -d
**Step 3**: The site [http://lms.localhost:8000/insights](http://lms.localhost:8000/lms) should now be available. The default credentials are: **Step 3**: The site [http://lms.localhost:8000/lms](http://lms.localhost:8000/lms) should now be available. The default credentials are:
- Username: Administrator - Username: Administrator
- Password: admin - Password: admin
@@ -134,7 +151,7 @@ To setup the repository locally follow the steps mentioned below:
1. Start the server by running `bench start` 1. Start the server by running `bench start`
1. In a separate terminal window, create a new site by running `bench new-site learning.test` 1. In a separate terminal window, create a new site by running `bench new-site learning.test`
1. Map your site to localhost with the command `bench --site learning.test add-to-hosts` 1. Map your site to localhost with the command `bench --site learning.test add-to-hosts`
1. Get the Insights app. Run `bench get-app https://github.com/frappe/lms` 1. Get the Learning app. Run `bench get-app https://github.com/frappe/lms`
1. Run `bench --site learning.test install-app lms`. 1. Run `bench --site learning.test install-app lms`.
1. Now open the URL `http://learning.test:8000/lms` in your browser, you should see the app running 1. Now open the URL `http://learning.test:8000/lms` in your browser, you should see the app running
@@ -145,7 +162,8 @@ To setup the repository locally follow the steps mentioned below:
- [Documentation](https://docs.frappe.io/learning) - [Documentation](https://docs.frappe.io/learning)
- [YouTube](https://www.youtube.com/channel/UCn3bV5kx77HsVwtnlCeEi_A) - [YouTube](https://www.youtube.com/channel/UCn3bV5kx77HsVwtnlCeEi_A)
<h2></h2> <br>
<br>
<div align="center" style="padding-top: 0.75rem;"> <div align="center" style="padding-top: 0.75rem;">
<a href="https://frappe.io" target="_blank"> <a href="https://frappe.io" target="_blank">
<picture> <picture>

26
commitlint.config.js Normal file
View File

@@ -0,0 +1,26 @@
module.exports = {
parserPreset: "conventional-changelog-conventionalcommits",
rules: {
"subject-empty": [2, "never"],
"type-case": [2, "always", "lower-case"],
"type-empty": [2, "never"],
"type-enum": [
2,
"always",
[
"build",
"chore",
"ci",
"docs",
"feat",
"fix",
"perf",
"refactor",
"revert",
"style",
"test",
"deprecate", // deprecation decision
],
],
},
};

View File

@@ -13,6 +13,6 @@ module.exports = defineConfig({
openMode: 0, openMode: 0,
}, },
e2e: { e2e: {
baseUrl: "http://lms1:8000", baseUrl: "http://pertest: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");
@@ -19,12 +19,16 @@ describe("Course Creation", () => {
); );
cy.fixture("profile.png", "base64").then((fileContent) => { cy.fixture("profile.png", "base64").then((fileContent) => {
cy.get('input[type="file"]').attachFile({ cy.get("div")
fileContent, .contains("Course Image")
fileName: "profile.png", .siblings("div")
mimeType: "image/png", .children('input[type="file"]')
encoding: "base64", .attachFile({
}); fileContent,
fileName: "profile.png",
mimeType: "image/png",
encoding: "base64",
});
}); });
cy.get("label") cy.get("label")
@@ -84,9 +88,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,9 @@ 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,
retryOnStatusCodeFailure: true,
retryOnNetworkFailure: true,
}); });
}); });

View File

@@ -2,7 +2,7 @@ version: "3.7"
name: lms name: lms
services: services:
mariadb: mariadb:
image: mariadb:10.6 image: mariadb:10.8
command: command:
- --character-set-server=utf8mb4 - --character-set-server=utf8mb4
- --collation-server=utf8mb4_unicode_ci - --collation-server=utf8mb4_unicode_ci

Submodule frappe-ui deleted from 8cd9b06a5e

97
frontend/components.d.ts vendored Normal file
View File

@@ -0,0 +1,97 @@
/* eslint-disable */
// @ts-nocheck
// Generated by unplugin-vue-components
// Read more: https://github.com/vuejs/core/pull/3399
// biome-ignore lint: disable
export {}
/* prettier-ignore */
declare module 'vue' {
export interface GlobalComponents {
Annoucements: typeof import('./src/components/Annoucements.vue')['default']
AnnouncementModal: typeof import('./src/components/Modals/AnnouncementModal.vue')['default']
Apps: typeof import('./src/components/Apps.vue')['default']
AppSidebar: typeof import('./src/components/AppSidebar.vue')['default']
AssessmentModal: typeof import('./src/components/Modals/AssessmentModal.vue')['default']
AssessmentPlugin: typeof import('./src/components/AssessmentPlugin.vue')['default']
Assessments: typeof import('./src/components/Assessments.vue')['default']
Assignment: typeof import('./src/components/Assignment.vue')['default']
AssignmentForm: typeof import('./src/components/Modals/AssignmentForm.vue')['default']
AudioBlock: typeof import('./src/components/AudioBlock.vue')['default']
Autocomplete: typeof import('./src/components/Controls/Autocomplete.vue')['default']
BatchCard: typeof import('./src/components/BatchCard.vue')['default']
BatchCourseModal: typeof import('./src/components/Modals/BatchCourseModal.vue')['default']
BatchCourses: typeof import('./src/components/BatchCourses.vue')['default']
BatchDashboard: typeof import('./src/components/BatchDashboard.vue')['default']
BatchFeedback: typeof import('./src/components/BatchFeedback.vue')['default']
BatchOverlay: typeof import('./src/components/BatchOverlay.vue')['default']
BatchStudentProgress: typeof import('./src/components/Modals/BatchStudentProgress.vue')['default']
BatchStudents: typeof import('./src/components/BatchStudents.vue')['default']
BrandSettings: typeof import('./src/components/BrandSettings.vue')['default']
BulkCertificates: typeof import('./src/components/Modals/BulkCertificates.vue')['default']
Categories: typeof import('./src/components/Categories.vue')['default']
CertificationLinks: typeof import('./src/components/CertificationLinks.vue')['default']
ChapterModal: typeof import('./src/components/Modals/ChapterModal.vue')['default']
CodeEditor: typeof import('./src/components/Controls/CodeEditor.vue')['default']
CollapseSidebar: typeof import('./src/components/Icons/CollapseSidebar.vue')['default']
CourseCard: typeof import('./src/components/CourseCard.vue')['default']
CourseCardOverlay: typeof import('./src/components/CourseCardOverlay.vue')['default']
CourseInstructors: typeof import('./src/components/CourseInstructors.vue')['default']
CourseOutline: typeof import('./src/components/CourseOutline.vue')['default']
CourseReviews: typeof import('./src/components/CourseReviews.vue')['default']
CreateOutline: typeof import('./src/components/CreateOutline.vue')['default']
DateRange: typeof import('./src/components/Common/DateRange.vue')['default']
DesktopLayout: typeof import('./src/components/DesktopLayout.vue')['default']
DiscussionModal: typeof import('./src/components/Modals/DiscussionModal.vue')['default']
DiscussionReplies: typeof import('./src/components/DiscussionReplies.vue')['default']
Discussions: typeof import('./src/components/Discussions.vue')['default']
EditCoverImage: typeof import('./src/components/Modals/EditCoverImage.vue')['default']
EditProfile: typeof import('./src/components/Modals/EditProfile.vue')['default']
EvaluationModal: typeof import('./src/components/Modals/EvaluationModal.vue')['default']
Evaluators: typeof import('./src/components/Evaluators.vue')['default']
Event: typeof import('./src/components/Modals/Event.vue')['default']
ExplanationVideos: typeof import('./src/components/Modals/ExplanationVideos.vue')['default']
FrappeCloudIcon: typeof import('./src/components/Icons/FrappeCloudIcon.vue')['default']
IconPicker: typeof import('./src/components/Controls/IconPicker.vue')['default']
IndicatorIcon: typeof import('./src/components/Icons/IndicatorIcon.vue')['default']
InviteIcon: typeof import('./src/components/Icons/InviteIcon.vue')['default']
JobApplicationModal: typeof import('./src/components/Modals/JobApplicationModal.vue')['default']
JobCard: typeof import('./src/components/JobCard.vue')['default']
LessonContent: typeof import('./src/components/LessonContent.vue')['default']
LessonHelp: typeof import('./src/components/LessonHelp.vue')['default']
Link: typeof import('./src/components/Controls/Link.vue')['default']
LiveClass: typeof import('./src/components/LiveClass.vue')['default']
LiveClassModal: typeof import('./src/components/Modals/LiveClassModal.vue')['default']
LMSLogo: typeof import('./src/components/Icons/LMSLogo.vue')['default']
Members: typeof import('./src/components/Members.vue')['default']
MobileLayout: typeof import('./src/components/MobileLayout.vue')['default']
MultiSelect: typeof import('./src/components/Controls/MultiSelect.vue')['default']
NoPermission: typeof import('./src/components/NoPermission.vue')['default']
NoSidebarLayout: typeof import('./src/components/NoSidebarLayout.vue')['default']
NotPermitted: typeof import('./src/components/NotPermitted.vue')['default']
PageModal: typeof import('./src/components/Modals/PageModal.vue')['default']
PaymentSettings: typeof import('./src/components/PaymentSettings.vue')['default']
Play: typeof import('./src/components/Icons/Play.vue')['default']
ProgressBar: typeof import('./src/components/ProgressBar.vue')['default']
Question: typeof import('./src/components/Modals/Question.vue')['default']
Quiz: typeof import('./src/components/Quiz.vue')['default']
QuizBlock: typeof import('./src/components/QuizBlock.vue')['default']
Rating: typeof import('./src/components/Controls/Rating.vue')['default']
ReviewModal: typeof import('./src/components/Modals/ReviewModal.vue')['default']
RouterLink: typeof import('vue-router')['RouterLink']
RouterView: typeof import('vue-router')['RouterView']
SettingDetails: typeof import('./src/components/SettingDetails.vue')['default']
SettingFields: typeof import('./src/components/SettingFields.vue')['default']
Settings: typeof import('./src/components/Modals/Settings.vue')['default']
SidebarLink: typeof import('./src/components/SidebarLink.vue')['default']
StudentHeatmap: typeof import('./src/components/StudentHeatmap.vue')['default']
StudentModal: typeof import('./src/components/Modals/StudentModal.vue')['default']
Tags: typeof import('./src/components/Tags.vue')['default']
UnsplashImageBrowser: typeof import('./src/components/UnsplashImageBrowser.vue')['default']
UpcomingEvaluations: typeof import('./src/components/UpcomingEvaluations.vue')['default']
UploadPlugin: typeof import('./src/components/UploadPlugin.vue')['default']
UserAvatar: typeof import('./src/components/UserAvatar.vue')['default']
UserDropdown: typeof import('./src/components/UserDropdown.vue')['default']
VideoBlock: typeof import('./src/components/VideoBlock.vue')['default']
}
}

View File

@@ -2,9 +2,9 @@
<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 }}" />
<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>{{ title }}</title>
<meta name="title" content="{{ meta.title }}" /> <meta name="title" content="{{ meta.title }}" />
<meta name="image" content="{{ meta.image }}" /> <meta name="image" content="{{ meta.image }}" />
<meta name="description" content="{{ meta.description }}" /> <meta name="description" content="{{ meta.description }}" />
@@ -23,25 +23,10 @@
<p> <p>
{{ meta.description }} {{ meta.description }}
</p> </p>
<p>
The content here is just for seo purposes. The actual content will be loaded in a few seconds.
</p>
<p>
Seo checks if a page has more than 300 words. So, here are some more words to make it more than 300 words.
Page descriptions are the HTML meta tags that provide a brief summary of a web page.
Search engines use meta descriptions to help identify the page's topic - they don't use them to rank the page, but they do use them to determine whether or not to display the page in search results.
Meta descriptions are important because they're often the first thing people see when they're deciding which search result to click on.
They're also important because they can help improve your click-through rate (CTR) from search results.
A good meta description can entice people to click on your page instead of someone else's.
</p>
<a href="{{ meta.link }}">Know More</a> <a href="{{ meta.link }}">Know More</a>
</div> </div>
</div> </div>
<div id="modals"></div>
<div id="popovers"></div>
<script> <script>
window.csrf_token = '{{ csrf_token }}'
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,22 +19,27 @@
"@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",
"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.134",
"highlight.js": "^11.11.1",
"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",
"plyr": "^3.7.8",
"socket.io-client": "^4.7.2", "socket.io-client": "^4.7.2",
"tailwindcss": "^3.3.3", "tailwindcss": "3.4.15",
"typescript": "^5.7.2", "typescript": "^5.7.2",
"vue": "^3.4.23", "vue": "^3.4.23",
"vue-chartjs": "^5.3.0", "vue-chartjs": "^5.3.0",
"vue-draggable-next": "^2.2.1", "vue-draggable-next": "^2.2.1",
"vue-router": "^4.0.12", "vue-router": "^4.0.12",
"vue3-apexcharts": "^1.8.0",
"vuedraggable": "4.1.0" "vuedraggable": "4.1.0"
}, },
"devDependencies": { "devDependencies": {

View File

@@ -0,0 +1,4 @@
<svg width="80" height="79" viewBox="0 0 80 79" fill="none" xmlns="http://www.w3.org/2000/svg">
<path d="M57.1285 0.580383H22.8514C10.2309 0.580383 0 10.5649 0 22.8815V56.3332C0 68.6497 10.2309 78.6343 22.8514 78.6343H57.1285C69.749 78.6343 79.9799 68.6497 79.9799 56.3332V22.8815C79.9799 10.5649 69.749 0.580383 57.1285 0.580383Z" fill="#0E7159"/>
<path d="M62.8434 23.6906L60.7869 23.1052C53.6744 21.0702 45.9048 22.4641 39.992 26.8128C35.8502 23.7742 30.7943 22.1854 25.7099 22.2133H17.1406V27.8163H25.7099C29.6232 27.8163 33.508 29.015 36.6787 31.3845L39.992 33.8377L43.3056 31.3845C47.2475 28.4575 52.3032 27.2588 57.1306 28.0393V50.647C51.1035 49.9223 44.9051 51.4834 39.992 55.0795C35.8502 52.0688 30.8515 50.4798 25.7671 50.4798C24.7959 50.4798 23.8247 50.5355 22.8535 50.647V35.0642H17.1406V57.0588H62.8434V23.7185V23.6906Z" fill="white"/>
</svg>

After

Width:  |  Height:  |  Size: 856 B

View File

@@ -8,18 +8,34 @@
<script setup> <script setup>
import { Toasts } from 'frappe-ui' import { Toasts } from 'frappe-ui'
import { Dialogs } from '@/utils/dialogs' import { Dialogs } from '@/utils/dialogs'
import { computed, onMounted, onUnmounted } from 'vue' import { computed, onMounted, onUnmounted, ref } from 'vue'
import { useScreenSize } from './utils/composables' import { useScreenSize } from './utils/composables'
import DesktopLayout from './components/DesktopLayout.vue' import DesktopLayout from './components/DesktopLayout.vue'
import MobileLayout from './components/MobileLayout.vue' import MobileLayout from './components/MobileLayout.vue'
import NoSidebarLayout from './components/NoSidebarLayout.vue'
import { stopSession } from '@/telemetry' import { stopSession } from '@/telemetry'
import { init as initTelemetry } from '@/telemetry' import { init as initTelemetry } from '@/telemetry'
import { usersStore } from '@/stores/user' import { usersStore } from '@/stores/user'
import { useRouter } from 'vue-router'
const screenSize = useScreenSize() const screenSize = useScreenSize()
let { userResource } = usersStore() let { userResource } = usersStore()
const router = useRouter()
const noSidebar = ref(false)
router.beforeEach((to, from, next) => {
if (to.query.fromLesson || to.path === '/persona') {
noSidebar.value = true
} else {
noSidebar.value = false
}
next()
})
const Layout = computed(() => { const Layout = computed(() => {
if (noSidebar.value) {
return NoSidebarLayout
}
if (screenSize.width < 640) { if (screenSize.width < 640) {
return MobileLayout return MobileLayout
} else { } else {
@@ -28,11 +44,11 @@ const Layout = computed(() => {
}) })
onMounted(async () => { onMounted(async () => {
if (!userResource.data) return if (userResource.data) await initTelemetry()
await initTelemetry()
}) })
onUnmounted(() => { onUnmounted(() => {
noSidebar.value = false
stopSession() stopSession()
}) })
</script> </script>

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,32 +23,36 @@
<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">
{{ __('More') }} {{ __('More') }}
</span> </span>
</div> </div>
<Button v-if="isModerator" variant="ghost" @click="openPageModal()"> <Button
v-if="isModerator && !readOnlyMode"
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 +66,109 @@
</div> </div>
</div> </div>
</div> </div>
<SidebarLink <div class="m-2 flex flex-col gap-1">
:link="{ <div
label: sidebarStore.isSidebarCollapsed ? 'Expand' : 'Collapse', v-if="readOnlyMode && !sidebarStore.isSidebarCollapsed"
}" class="z-10 m-2 bg-surface-modal py-2.5 px-3 text-xs text-ink-gray-7 leading-5 rounded-md"
:isCollapsed="sidebarStore.isSidebarCollapsed" >
@click="toggleSidebar()" {{
class="m-2" __(
> 'This site is being updated. You will not be able to make any changes. Full access will be restored shortly.'
<template #icon> )
<span class="grid h-5 w-6 flex-shrink-0 place-items-center"> }}
</div>
<TrialBanner
v-if="
userResource.data?.is_system_manager && userResource.data?.is_fc_site
"
:isSidebarCollapsed="sidebarStore.isSidebarCollapsed"
/>
<GettingStartedBanner
v-if="showOnboarding && !isOnboardingStepsCompleted"
:isSidebarCollapsed="sidebarStore.isSidebarCollapsed"
appName="learning"
/>
<div
class="flex items-center mt-4"
:class="
sidebarStore.isSidebarCollapsed ? 'flex-col space-y-3' : 'flex-row'
"
>
<div
class="flex items-center flex-1"
:class="
sidebarStore.isSidebarCollapsed
? 'flex-col space-y-3'
: 'flex-row space-x-3'
"
>
<Tooltip v-if="readOnlyMode && sidebarStore.isSidebarCollapsed">
<CircleAlert
class="size-4 stroke-1.5 text-ink-gray-7 cursor-pointer"
/>
<template #body>
<div
class="max-w-[30ch] rounded bg-surface-gray-7 px-2 py-1 text-center text-p-xs text-ink-white shadow-xl"
>
{{
__(
'This site is being updated. You will not be able to make any changes. Full access will be restored shortly.'
)
}}
</div>
</template>
</Tooltip>
<Tooltip :text="__('Powered by Learning')">
<Zap
class="size-4 stroke-1.5 text-ink-gray-7 cursor-pointer"
@click="redirectToWebsite()"
/>
</Tooltip>
<Tooltip :text="__('Help')">
<CircleHelp
class="size-4 stroke-1.5 text-ink-gray-7 cursor-pointer"
@click="
() => {
showHelpModal = minimize ? true : !showHelpModal
minimize = !showHelpModal
}
"
/>
</Tooltip>
</div>
<Tooltip
:text="
sidebarStore.isSidebarCollapsed ? __('Expand') : __('Collapse')
"
>
<CollapseSidebar <CollapseSidebar
class="h-4.5 w-4.5 text-gray-700 duration-300 ease-in-out" class="size-4 text-ink-gray-7 duration-300 stroke-1.5 ease-in-out cursor-pointer"
:class="{ :class="{
'[transform:rotateY(180deg)]': sidebarStore.isSidebarCollapsed, '[transform:rotateY(180deg)]': sidebarStore.isSidebarCollapsed,
}" }"
@click="toggleSidebar()"
/> />
</span> </Tooltip>
</template> </div>
</SidebarLink> </div>
<HelpModal
v-if="showOnboarding && showHelpModal"
v-model="showHelpModal"
v-model:articles="articles"
appName="learning"
title="Frappe Learning"
:logo="LMSLogo"
:afterSkip="(step) => capture('onboarding_step_skipped_' + step)"
:afterSkipAll="() => capture('onboarding_steps_skipped')"
:afterReset="(step) => capture('onboarding_step_reset_' + step)"
:afterResetAll="() => capture('onboarding_steps_reset')"
docsLink="https://docs.frappe.io/learning"
/>
<IntermediateStepModal
v-model="showIntermediateModal"
:currentStep="currentStep"
/>
</div> </div>
<PageModal <PageModal
v-model="showPageModal" v-model="showPageModal"
@@ -94,15 +182,40 @@ import UserDropdown from '@/components/UserDropdown.vue'
import CollapseSidebar from '@/components/Icons/CollapseSidebar.vue' import CollapseSidebar from '@/components/Icons/CollapseSidebar.vue'
import SidebarLink from '@/components/SidebarLink.vue' import SidebarLink from '@/components/SidebarLink.vue'
import { useStorage } from '@vueuse/core' import { useStorage } from '@vueuse/core'
import { ref, onMounted, inject, watch } from 'vue' import { ref, onMounted, inject, watch, reactive, markRaw, h } from 'vue'
import { getSidebarLinks } from '../utils' import { getSidebarLinks } from '../utils'
import { usersStore } from '@/stores/user' import { usersStore } from '@/stores/user'
import { sessionStore } from '@/stores/session' 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 { Button, createResource, Tooltip } from 'frappe-ui'
import { createResource, Button } from 'frappe-ui'
import PageModal from '@/components/Modals/PageModal.vue' import PageModal from '@/components/Modals/PageModal.vue'
import { capture } from '@/telemetry'
import LMSLogo from '@/components/Icons/LMSLogo.vue'
import { useRouter } from 'vue-router'
import InviteIcon from './Icons/InviteIcon.vue'
import {
BookOpen,
CircleAlert,
ChevronRight,
Plus,
CircleHelp,
FolderTree,
FileText,
UserPlus,
Users,
BookText,
Zap,
} from 'lucide-vue-next'
import {
TrialBanner,
HelpModal,
GettingStartedBanner,
useOnboarding,
showHelpModal,
minimize,
IntermediateStepModal,
} from 'frappe-ui/frappe'
const { user, sidebarSettings } = sessionStore() const { user, sidebarSettings } = sessionStore()
const { userResource } = usersStore() const { userResource } = usersStore()
@@ -114,14 +227,29 @@ 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()
const showOnboarding = ref(false)
const showIntermediateModal = ref(false)
const currentStep = ref({})
const router = useRouter()
let onboardingDetails
let isOnboardingStepsCompleted = false
const readOnlyMode = window.read_only_mode
const iconProps = {
strokeWidth: 1.5,
width: 16,
height: 16,
}
onMounted(() => { onMounted(() => {
addNotifications()
setSidebarLinks()
socket.on('publish_lms_notifications', (data) => { socket.on('publish_lms_notifications', (data) => {
unreadNotifications.reload() unreadNotifications.reload()
}) })
addNotifications() })
const setSidebarLinks = () => {
sidebarSettings.reload( sidebarSettings.reload(
{}, {},
{ {
@@ -136,7 +264,7 @@ onMounted(() => {
}, },
} }
) )
}) }
const unreadNotifications = createResource({ const unreadNotifications = createResource({
cache: 'Unread Notifications Count', cache: 'Unread Notifications Count',
@@ -180,7 +308,28 @@ const addQuizzes = () => {
label: 'Quizzes', label: 'Quizzes',
icon: 'CircleHelp', icon: 'CircleHelp',
to: 'Quizzes', to: 'Quizzes',
activeFor: ['Quizzes', 'QuizForm'], activeFor: [
'Quizzes',
'QuizForm',
'QuizSubmissionList',
'QuizSubmission',
],
})
}
}
const addAssignments = () => {
if (isInstructor.value || isModerator.value) {
sidebarLinks.value.push({
label: 'Assignments',
icon: 'Pencil',
to: 'Assignments',
activeFor: [
'Assignments',
'AssignmentForm',
'AssignmentSubmissionList',
'AssignmentSubmission',
],
}) })
} }
} }
@@ -243,16 +392,237 @@ const getSidebarFromStorage = () => {
return useStorage('sidebar_is_collapsed', false) return useStorage('sidebar_is_collapsed', false)
} }
const toggleSidebar = () => {
sidebarStore.isSidebarCollapsed = !sidebarStore.isSidebarCollapsed
localStorage.setItem(
'isSidebarCollapsed',
JSON.stringify(sidebarStore.isSidebarCollapsed)
)
}
const toggleWebPages = () => {
sidebarStore.isWebpagesCollapsed = !sidebarStore.isWebpagesCollapsed
localStorage.setItem(
'isWebpagesCollapsed',
JSON.stringify(sidebarStore.isWebpagesCollapsed)
)
}
const getFirstCourse = async () => {
let firstCourse = localStorage.getItem('firstCourse')
if (firstCourse) return firstCourse
return await call('lms.lms.onboarding.get_first_course')
}
const getFirstBatch = async () => {
let firstBatch = localStorage.getItem('firstBatch')
if (firstBatch) return firstBatch
return await call('lms.lms.onboarding.get_first_batch')
}
const steps = reactive([
{
name: 'create_first_course',
title: __('Create your first course'),
icon: markRaw(h(BookOpen, iconProps)),
completed: false,
onClick: () => {
minimize.value = true
router.push({
name: 'Courses',
})
},
},
{
name: 'create_first_chapter',
title: __('Add your first chapter'),
icon: markRaw(h(FolderTree, iconProps)),
completed: false,
onClick: async () => {
minimize.value = true
let course = await getFirstCourse()
if (course) {
router.push({ name: 'CourseForm', params: { courseName: course } })
} else {
router.push({ name: 'CourseForm' })
}
},
},
{
name: 'create_first_lesson',
title: __('Add your first lesson'),
icon: markRaw(h(FileText, iconProps)),
completed: false,
onClick: async () => {
minimize.value = true
let course = await getFirstCourse()
if (course) {
router.push({
name: 'CourseForm',
params: { courseName: course },
})
} else {
router.push({ name: 'Courses' })
}
},
},
{
name: 'create_first_quiz',
title: __('Create your first quiz'),
icon: markRaw(h(CircleHelp, iconProps)),
completed: false,
onClick: () => {
minimize.value = true
router.push({ name: 'Quizzes' })
},
},
{
name: 'invite_students',
title: __('Invite your team and students'),
icon: markRaw(h(InviteIcon, iconProps)),
completed: false,
onClick: () => {
minimize.value = true
settingsStore.activeTab = 'Members'
settingsStore.isSettingsOpen = true
},
},
{
name: 'create_first_batch',
title: __('Create your first batch'),
icon: markRaw(h(Users, iconProps)),
completed: false,
onClick: () => {
minimize.value = true
router.push({ name: 'Batches' })
},
},
{
name: 'add_batch_student',
title: __('Add students to your batch'),
icon: markRaw(h(UserPlus, iconProps)),
completed: false,
onClick: async () => {
minimize.value = true
let batch = await getFirstBatch()
if (batch) {
router.push({
name: 'Batch',
params: {
batchName: batch,
},
})
} else {
router.push({ name: 'Batch' })
}
},
},
{
name: 'add_batch_course',
title: __('Add courses to your batch'),
icon: markRaw(h(BookText, iconProps)),
completed: false,
onClick: async () => {
minimize.value = true
let batch = await getFirstBatch()
if (batch) {
router.push({
name: 'Batch',
params: {
batchName: batch,
},
hash: '#courses',
})
} else {
router.push({ name: 'Batch' })
}
},
},
])
const articles = ref([
{
title: __('Introduction'),
opened: false,
subArticles: [
{ name: 'introduction', title: __('Introduction') },
{ name: 'setting-up', title: __('Setting up') },
],
},
{
title: __('Creating a course'),
opened: false,
subArticles: [
{ name: 'create-a-course', title: __('Create a course') },
{ name: 'add-a-chapter', title: __('Add a chapter') },
{ name: 'add-a-lesson', title: __('Add a lesson') },
],
},
{
title: __('Creating a batch'),
opened: false,
subArticles: [
{ name: 'create-a-batch', title: __('Create a batch') },
{ name: 'create-a-live-class', title: __('Create a live class') },
],
},
{
title: __('Assessments'),
opened: false,
subArticles: [
{ name: 'quizzes', title: __('Quizzes') },
{ name: 'assignments', title: __('Assignments') },
],
},
{
title: __('Certification'),
opened: false,
subArticles: [
{ name: 'issue-a-certificate', title: __('Issue a Certificate') },
{
name: 'custom-certificate-templates',
title: __('Custom Certificate Templates'),
},
],
},
{
title: __('Monetization'),
opened: false,
subArticles: [
{
name: 'setting-up-payment-gateway',
title: __('Setting up payment gateway'),
},
],
},
{
title: __('Settings'),
opened: false,
subArticles: [{ name: 'roles', title: __('Roles') }],
},
])
const setUpOnboarding = () => {
if (userResource.data?.is_system_manager) {
onboardingDetails = useOnboarding('learning')
onboardingDetails.setUp(steps)
isOnboardingStepsCompleted = onboardingDetails.isOnboardingStepsCompleted
showOnboarding.value = true
}
}
watch(userResource, () => { watch(userResource, () => {
if (userResource.data) { if (userResource.data) {
isModerator.value = userResource.data.is_moderator isModerator.value = userResource.data.is_moderator
isInstructor.value = userResource.data.is_instructor isInstructor.value = userResource.data.is_instructor
addQuizzes()
addPrograms() addPrograms()
addQuizzes()
addAssignments()
setUpOnboarding()
} }
}) })
const toggleSidebar = () => { const redirectToWebsite = () => {
sidebarStore.isSidebarCollapsed = !sidebarStore.isSidebarCollapsed window.open('https://frappe.io/learning', '_blank')
} }
</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-7 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

@@ -0,0 +1,75 @@
<template>
<Dialog
v-model="show"
:options="{
size: 'xl',
}"
>
<template #body>
<div class="p-5 space-y-4">
<div v-if="type == 'quiz'" class="text-lg font-semibold">
{{ __('Add a quiz to your lesson') }}
</div>
<div v-else class="text-lg font-semibold">
{{ __('Add an assignment to your lesson') }}
</div>
<div>
<Link
v-if="type == 'quiz'"
v-model="quiz"
doctype="LMS Quiz"
:label="__('Select a quiz')"
:onCreate="(value, close) => redirectToForm()"
/>
<Link
v-else
v-model="assignment"
doctype="LMS Assignment"
:label="__('Select an assignment')"
:onCreate="(value, close) => redirectToForm()"
/>
</div>
<div class="flex justify-end space-x-2">
<Button variant="solid" @click="addAssessment()">
{{ __('Save') }}
</Button>
</div>
</div>
</template>
</Dialog>
</template>
<script setup>
import { Dialog, Button } from 'frappe-ui'
import { onMounted, ref, nextTick } from 'vue'
import Link from '@/components/Controls/Link.vue'
const show = ref(false)
const quiz = ref(null)
const assignment = ref(null)
const props = defineProps({
type: {
type: String,
required: true,
},
onAddition: {
type: Function,
required: true,
},
})
onMounted(async () => {
await nextTick()
show.value = true
})
const addAssessment = () => {
props.onAddition(props.type == 'quiz' ? quiz.value : assignment.value)
show.value = false
}
const redirectToForm = () => {
if (props.type == 'quiz') window.open('/lms/quizzes/new', '_blank')
else window.open('/lms/assignments/new', '_blank')
}
</script>

View File

@@ -1,17 +1,17 @@
<template> <template>
<div> <div>
<div class="flex items-center justify-between"> <div class="flex items-center justify-between mb-4">
<div class="text-lg font-semibold mb-4"> <div class="text-lg font-semibold text-ink-gray-9">
{{ __('Assessments') }} {{ __('Assessments') }}
</div> </div>
<Button v-if="canSeeAddButton()" @click="showModal = true"> <Button v-if="canAddAssessments()" @click="showModal = true">
<template #prefix> <template #prefix>
<Plus class="h-4 w-4" /> <Plus class="h-4 w-4" />
</template> </template>
{{ __('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"
@@ -19,10 +19,11 @@
:options="{ :options="{
showTooltip: false, showTooltip: false,
getRowRoute: (row) => getRowRoute(row), getRowRoute: (row) => getRowRoute(row),
selectable: user.data?.is_student ? false : true,
}" }"
> >
<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 }">
@@ -38,7 +39,18 @@
<ListRow :row="row" v-for="row in assessments.data"> <ListRow :row="row" v-for="row in assessments.data">
<template #default="{ column, item }"> <template #default="{ column, item }">
<ListRowItem :item="row[column.key]" :align="column.align"> <ListRowItem :item="row[column.key]" :align="column.align">
<div> <div v-if="column.key == 'assessment_type'">
{{ row[column.key] == 'LMS Quiz' ? 'Quiz' : 'Assignment' }}
</div>
<div v-else-if="column.key == 'title'">
{{ row[column.key] }}
</div>
<div v-else-if="isNaN(row[column.key])">
<Badge :theme="getStatusTheme(row[column.key])">
{{ row[column.key] }}
</Badge>
</div>
<div v-else>
{{ row[column.key] }} {{ row[column.key] }}
</div> </div>
</ListRowItem> </ListRowItem>
@@ -59,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>
@@ -80,6 +92,7 @@ import {
ListSelectBanner, ListSelectBanner,
createResource, createResource,
Button, Button,
Badge,
} from 'frappe-ui' } from 'frappe-ui'
import { inject, ref } from 'vue' import { inject, ref } from 'vue'
import AssessmentModal from '@/components/Modals/AssessmentModal.vue' import AssessmentModal from '@/components/Modals/AssessmentModal.vue'
@@ -87,6 +100,7 @@ import { Plus, Trash2 } from 'lucide-vue-next'
const user = inject('$user') const user = inject('$user')
const showModal = ref(false) const showModal = ref(false)
const readOnlyMode = window.read_only_mode
const props = defineProps({ const props = defineProps({
batch: { batch: {
@@ -145,7 +159,7 @@ const getRowRoute = (row) => {
return { return {
name: 'AssignmentSubmission', name: 'AssignmentSubmission',
params: { params: {
assignmentName: row.assessment_name, assignmentID: row.assessment_name,
submissionName: row.submission.name, submissionName: row.submission.name,
}, },
} }
@@ -153,7 +167,7 @@ const getRowRoute = (row) => {
return { return {
name: 'AssignmentSubmission', name: 'AssignmentSubmission',
params: { params: {
assignmentName: row.assessment_name, assignmentID: row.assessment_name,
submissionName: 'new', submissionName: 'new',
}, },
} }
@@ -168,7 +182,8 @@ const getRowRoute = (row) => {
} }
} }
const canSeeAddButton = () => { const canAddAssessments = () => {
if (readOnlyMode) return false
return user.data?.is_moderator || user.data?.is_evaluator return user.data?.is_moderator || user.data?.is_evaluator
} }
@@ -177,20 +192,33 @@ const getAssessmentColumns = () => {
{ {
label: 'Assessment', label: 'Assessment',
key: 'title', key: 'title',
width: '25rem',
}, },
{ {
label: 'Type', label: 'Type',
key: 'assessment_type', key: 'assessment_type',
width: '15rem',
}, },
] ]
if (!user.data?.is_moderator) { if (!user.data?.is_moderator) {
columns.push({ columns.push({
label: 'Status/Score', label: 'Status/Percentage',
key: 'status', key: 'status',
align: 'center', align: 'left',
width: '10rem',
}) })
} }
return columns return columns
} }
const getStatusTheme = (status) => {
if (status === 'Pass') {
return 'green'
} else if (status === 'Not Graded') {
return 'orange'
} else {
return 'red'
}
}
</script> </script>

View File

@@ -0,0 +1,476 @@
<template>
<div
v-if="assignment.data"
class="grid grid-cols-2 h-full"
:class="{ 'border rounded-lg overflow-auto': !showTitle }"
>
<div
class="border-r p-5 overflow-y-auto h-[calc(100vh-3.2rem)]"
:class="{ 'h-full': !showTitle }"
>
<div v-if="showTitle" class="text-lg font-semibold mb-5 text-ink-gray-9">
<div v-if="submissionName === 'new'">
{{ __('Submission by') }} {{ user.data?.full_name }}
</div>
<div v-else>
{{ __('Submission by') }} {{ submissionResource.doc?.member_name }}
</div>
</div>
<div class="text-sm text-ink-gray-7 font-medium mb-2">
{{ __('Question') }}:
</div>
<div
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-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 class="flex flex-col">
<div class="p-5">
<div class="flex items-center justify-between mb-4">
<div class="font-semibold text-ink-gray-9">
{{ __('Submission') }}
</div>
<div class="flex items-center space-x-2">
<Badge v-if="isDirty" theme="orange">
{{ __('Not Saved') }}
</Badge>
<Badge
v-else-if="submissionResource.doc?.status"
:theme="statusTheme"
size="lg"
>
{{ submissionResource.doc?.status }}
</Badge>
<Button variant="solid" @click="submitAssignment()">
{{ __('Save') }}
</Button>
</div>
</div>
<div
v-if="
submissionName != 'new' &&
!['Pass', 'Fail'].includes(submissionResource.doc?.status) &&
submissionResource.doc?.owner == user.data?.name
"
class="bg-surface-blue-2 text-ink-blue-2 p-3 rounded-md leading-5 text-sm mb-4"
>
{{ __("You've successfully submitted the assignment.") }}
{{
__(
"Once the moderator grades your submission, you'll find the details here."
)
}}
{{ __('Feel free to make edits to your submission if needed.') }}
</div>
<div v-if="showUploader()">
<div class="text-xs text-ink-gray-5 mt-1 mb-2">
{{ __('Add your assignment as {0}').format(assignment.data.type) }}
</div>
<FileUploader
v-if="!submissionFile"
:fileTypes="getType()"
:validateFile="validateFile"
@success="(file) => saveSubmission(file)"
>
<template #default="{ uploading, progress, openFileSelector }">
<Button @click="openFileSelector" :loading="uploading">
{{
uploading
? __('Uploading {0}%').format(progress)
: __('Upload File')
}}
</Button>
</template>
</FileUploader>
<div v-else>
<div class="flex text-ink-gray-7">
<div class="border self-start rounded-md p-2 mr-2">
<FileText class="h-5 w-5 stroke-1.5" />
</div>
<a
:href="submissionFile.file_url"
target="_blank"
class="flex flex-col cursor-pointer !no-underline"
>
<span class="text-sm leading-5">
{{ submissionFile.file_name }}
</span>
<span class="text-sm text-ink-gray-5 mt-1">
{{ getFileSize(submissionFile.file_size) }}
</span>
</a>
<X
v-if="canModifyAssignment"
@click="removeSubmission()"
class="bg-surface-gray-3 rounded-md cursor-pointer stroke-1.5 w-5 h-5 p-1 ml-4"
/>
</div>
</div>
</div>
<div v-else-if="assignment.data.type == 'URL'">
<div class="text-xs text-ink-gray-5 mb-1">
{{ __('Enter a URL') }}
</div>
<FormControl
v-model="answer"
type="text"
:readonly="!canModifyAssignment"
/>
</div>
<div v-else>
<div class="text-sm mb-2 text-ink-gray-7">
{{ __('Write your answer here') }}
</div>
<TextEditor
:content="answer"
@change="(val) => (answer = val)"
: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
v-if="
user.data?.name == submissionResource.doc?.owner &&
submissionResource.doc?.comments
"
class="mt-8 p-3 bg-surface-blue-2 rounded-md"
>
<div class="text-sm text-ink-gray-5 font-medium mb-2">
{{ __('Comments by Evaluator') }}:
</div>
<div
class="leading-5 text-ink-gray-9"
v-html="submissionResource.doc.comments"
></div>
</div>
<!-- Grading -->
<div v-if="canGradeSubmission" class="mt-8 space-y-4">
<div class="font-semibold mb-2 text-ink-gray-9">
{{ __('Grading') }}
</div>
<FormControl
v-if="submissionResource.doc"
v-model="submissionResource.doc.status"
:label="__('Grade')"
type="select"
:options="submissionStatusOptions"
/>
<div>
<div class="text-sm text-ink-gray-5 mb-1">
{{ __('Comments') }}
</div>
<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>
</template>
<script setup>
import {
Badge,
Button,
call,
createResource,
createDocumentResource,
FileUploader,
FormControl,
TextEditor,
} from 'frappe-ui'
import { computed, inject, onMounted, onBeforeUnmount, ref, watch } from 'vue'
import { FileText, X } from 'lucide-vue-next'
import { showToast, getFileSize } from '@/utils'
import { useRouter } from 'vue-router'
const submissionFile = ref(null)
const answer = ref(null)
const comments = ref(null)
const router = useRouter()
const user = inject('$user')
const isDirty = ref(false)
const props = defineProps({
assignmentID: {
type: String,
required: true,
},
submissionName: {
type: String,
default: 'new',
},
showTitle: {
type: Boolean,
default: true,
},
})
onMounted(() => {
window.addEventListener('keydown', keyboardShortcut)
})
const keyboardShortcut = (e) => {
if (e.key === 's' && (e.ctrlKey || e.metaKey)) {
submitAssignment()
e.preventDefault()
}
}
onBeforeUnmount(() => {
window.removeEventListener('keydown', keyboardShortcut)
})
const assignment = createResource({
url: 'frappe.client.get',
params: {
doctype: 'LMS Assignment',
name: props.assignmentID,
},
auto: true,
onSuccess(data) {
if (props.submissionName != 'new') {
submissionResource.reload()
}
},
})
const newSubmission = createResource({
url: 'frappe.client.insert',
makeParams(values) {
let doc = {
doctype: 'LMS Assignment Submission',
assignment: props.assignmentID,
member: user.data?.name,
}
if (showUploader()) {
doc.assignment_attachment = submissionFile.value.file_url
} else {
doc.answer = answer.value
}
return {
doc: doc,
}
},
})
const imageResource = createResource({
url: 'lms.lms.api.get_file_info',
makeParams(values) {
return {
file_url: values.image,
}
},
auto: false,
onSuccess(data) {
submissionFile.value = data
},
})
const submissionResource = createDocumentResource({
doctype: 'LMS Assignment Submission',
name: props.submissionName,
onError(err) {
showToast(__('Error'), __(err.messages?.[0] || err), 'x')
},
auto: false,
cache: [user.data?.name, props.assignmentID],
})
watch(submissionResource, () => {
if (submissionResource.doc) {
if (submissionResource.doc.assignment_attachment) {
imageResource.reload({
image: submissionResource.doc.assignment_attachment,
})
}
if (submissionResource.doc.answer) {
answer.value = submissionResource.doc.answer
}
if (submissionResource.doc.comments) {
comments.value = submissionResource.doc.comments
}
if (submissionResource.isDirty) {
isDirty.value = true
} else if (showUploader() && !submissionFile.value) {
isDirty.value = true
} else if (!showUploader() && !answer.value) {
isDirty.value = true
} else {
isDirty.value = false
}
}
})
watch(submissionFile, () => {
if (props.submissionName == 'new' && submissionFile.value) {
isDirty.value = true
}
})
const submitAssignment = () => {
if (props.submissionName != 'new') {
let evaluator =
submissionResource.doc && submissionResource.doc.owner != user.data?.name
? user.data?.name
: null
submissionResource.setValue.submit(
{
...submissionResource.doc,
assignment_attachment: submissionFile.value?.file_url,
evaluator: evaluator,
comments: comments.value,
answer: answer.value,
},
{
onSuccess(data) {
showToast(__('Success'), __('Changes saved successfully'), 'check')
},
}
)
} else {
addNewSubmission()
}
}
const addNewSubmission = () => {
newSubmission.submit(
{},
{
onSuccess(data) {
showToast('Success', 'Assignment submitted successfully.', 'check')
if (router.currentRoute.value.name == 'AssignmentSubmission') {
router.push({
name: 'AssignmentSubmission',
params: {
assignmentID: props.assignmentID,
submissionName: data.name,
},
query: { fromLesson: router.currentRoute.value.query.fromLesson },
})
} else {
markLessonProgress()
router.go()
}
submissionResource.name = data.name
submissionResource.reload()
},
onError(err) {
showToast('Error', err.messages?.[0] || err, 'x')
},
}
)
}
const saveSubmission = (file) => {
isDirty.value = true
submissionFile.value = file
}
const markLessonProgress = () => {
if (router.currentRoute.value.name == 'Lesson') {
let courseName = router.currentRoute.value.params.courseName
let chapterNumber = router.currentRoute.value.params.chapterNumber
let lessonNumber = router.currentRoute.value.params.lessonNumber
call('lms.lms.api.mark_lesson_progress', {
course: courseName,
chapter_number: chapterNumber,
lesson_number: lessonNumber,
})
}
}
const getType = () => {
const type = assignment.data?.type
if (type == 'Image') {
return ['image/*']
} else if (type == 'Document') {
return [
'.doc',
'.docx',
'.xml',
'application/msword',
'application/vnd.openxmlformats-officedocument.wordprocessingml.document',
]
} else if (type == 'PDF') {
return ['.pdf']
}
}
const validateFile = (file) => {
let type = assignment.data?.type
let extension = file.name.split('.').pop().toLowerCase()
if (type == 'Image' && !['jpg', 'jpeg', 'png'].includes(extension)) {
return 'Only image file is allowed.'
} else if (
type == 'Document' &&
!['doc', 'docx', 'xml'].includes(extension)
) {
return 'Only document file is allowed.'
} else if (type == 'PDF' && !['pdf'].includes(extension)) {
return 'Only PDF file is allowed.'
}
}
const removeSubmission = () => {
isDirty.value = true
submissionFile.value = null
}
const canGradeSubmission = computed(() => {
return (
(user.data?.is_moderator ||
user.data?.is_evaluator ||
user.data?.is_instructor) &&
props.submissionName != 'new' &&
router.currentRoute.value.name == 'AssignmentSubmission'
)
})
const canModifyAssignment = computed(() => {
return (
!submissionResource.doc ||
(submissionResource.doc?.owner == user.data?.name &&
submissionResource.doc?.status == 'Not Graded')
)
})
const submissionStatusOptions = computed(() => {
return [
{ label: 'Not Graded', value: 'Not Graded' },
{ label: 'Pass', value: 'Pass' },
{ label: 'Fail', value: 'Fail' },
]
})
const statusTheme = computed(() => {
if (!submissionResource.doc) {
return 'orange'
} else if (submissionResource.doc.status == 'Pass') {
return 'green'
} else if (submissionResource.doc.status == 'Not Graded') {
return 'blue'
} else {
return 'red'
}
})
const showUploader = () => {
return ['PDF', 'Image', 'Document'].includes(assignment.data?.type)
}
</script>

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 hover:border-outline-gray-4 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-xl 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 }">
@@ -62,6 +63,9 @@
</ListSelectBanner> </ListSelectBanner>
</ListView> </ListView>
</div> </div>
<div v-else class="text-sm italic text-ink-gray-5">
{{ __('No courses added') }}
</div>
<BatchCourseModal <BatchCourseModal
v-model="showCourseModal" v-model="showCourseModal"
:batch="batch" :batch="batch"
@@ -85,6 +89,7 @@ import {
} from 'frappe-ui' } from 'frappe-ui'
import { Plus, Trash2 } from 'lucide-vue-next' import { Plus, Trash2 } from 'lucide-vue-next'
import { showToast } from '@/utils' import { showToast } from '@/utils'
const readOnlyMode = window.read_only_mode
const showCourseModal = ref(false) const showCourseModal = ref(false)
const user = inject('$user') const user = inject('$user')
@@ -118,13 +123,13 @@ const getCoursesColumns = () => {
}, },
{ {
label: 'Lessons', label: 'Lessons',
key: 'lesson_count', key: 'lessons',
align: 'right', align: 'right',
}, },
{ {
label: 'Enrollments', label: 'Enrollments',
align: 'right', align: 'right',
key: 'enrollment_count', key: 'enrollments',
}, },
] ]
} }
@@ -155,6 +160,9 @@ const removeCourses = (selections, unselectAll) => {
} }
const canSeeAddButton = () => { const canSeeAddButton = () => {
if (readOnlyMode) {
return false
}
return user.data?.is_moderator || user.data?.is_evaluator return user.data?.is_moderator || user.data?.is_evaluator
} }
</script> </script>

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,244 @@
<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,
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,34 @@
<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
<BookOpen class="h-4 w-4 stroke-1.5 mr-2 text-gray-700" /> v-if="batch.data.courses.length"
class="flex items-center mb-3 text-ink-gray-7"
>
<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,73 +36,83 @@
: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>
</div> </div>
<router-link <div v-if="!readOnlyMode">
v-if="isModerator || isStudent" <router-link
:to="{ v-if="isModerator || isStudent"
name: 'Batch', :to="{
params: { name: 'Batch',
batchName: batch.data.name, params: {
}, batchName: batch.data.name,
}" },
> }"
<Button variant="solid" class="w-full mt-4"> >
<span> <Button variant="solid" class="w-full mt-4">
{{ isModerator ? __('Manage Batch') : __('Visit Batch') }} <span>
</span> {{ isModerator ? __('Manage Batch') : __('Visit Batch') }}
</span>
</Button>
</router-link>
<router-link
:to="{
name: 'Billing',
params: {
type: 'batch',
name: batch.data.name,
},
}"
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">
<span>
{{ __('Register Now') }}
</span>
</Button>
</router-link>
<Button
variant="solid"
class="w-full mt-2"
v-else-if="
batch.data.allow_self_enrollment &&
batch.data.seats_left &&
batch.data.accept_enrollments
"
@click="enrollInBatch()"
>
{{ __('Enroll Now') }}
</Button> </Button>
</router-link> <router-link
<router-link v-if="isModerator"
:to="{ :to="{
name: 'Billing', name: 'BatchForm',
params: { params: {
type: 'batch', batchName: batch.data.name,
name: batch.data.name, },
}, }"
}" >
v-else-if="batch.data.paid_batch && batch.data.seats_left" <Button class="w-full mt-2">
> <span>
<Button v-if="!isStudent" class="w-full mt-4" variant="solid"> {{ __('Edit') }}
<span> </span>
{{ __('Register Now') }} </Button>
</span> </router-link>
</Button> </div>
</router-link>
<Button
variant="solid"
class="w-full mt-2"
v-else-if="batch.data.allow_self_enrollment && batch.data.seats_left"
@click="enrollInBatch()"
>
{{ __('Enroll Now') }}
</Button>
<router-link
v-if="isModerator"
:to="{
name: 'BatchForm',
params: {
batchName: batch.data.name,
},
}"
>
<Button class="w-full mt-2">
<span>
{{ __('Edit') }}
</span>
</Button>
</router-link>
</div> </div>
</template> </template>
<script setup> <script setup>
@@ -106,6 +125,7 @@ import { useRouter } from 'vue-router'
const router = useRouter() const router = useRouter()
const user = inject('$user') const user = inject('$user')
const readOnlyMode = window.read_only_mode
const props = defineProps({ const props = defineProps({
batch: { batch: {

View File

@@ -1,80 +1,221 @@
<template> <template>
<Button class="float-right mb-3" @click="openStudentModal()"> <div class="">
<template #prefix> <div class="w-full flex items-center justify-between pb-4">
<Plus class="h-4 w-4" /> <div class="font-medium text-ink-gray-7">
</template> {{ __('Statistics') }}
{{ __('Add') }} </div>
</Button> </div>
<div class="text-lg font-semibold mb-4"> <div class="grid grid-cols-4 gap-5 mb-8">
{{ __('Students') }} <div
</div> class="flex items-center border py-2 px-3 rounded-md text-ink-gray-7"
<div v-if="students.data?.length">
<ListView
:columns="getStudentColumns()"
:rows="students.data"
row-key="name"
:options="{ showTooltip: false }"
>
<ListHeader
class="mb-2 grid items-center space-x-4 rounded bg-gray-100 p-2"
> >
<ListHeaderItem :item="item" v-for="item in getStudentColumns()"> <div class="p-2 rounded-md bg-surface-gray-2 mr-3">
<template #prefix="{ item }"> <User class="w-5 h-5 stroke-1.5" />
<component </div>
v-if="item.icon" <div class="flex items-center space-x-2">
:is="item.icon" <span class="font-semibold">
class="h-4 w-4 stroke-1.5 ml-4" {{ students.data?.length }}
/> </span>
</template> <span class="">
</ListHeaderItem> {{ __('Students') }}
</ListHeader> </span>
<ListRows> </div>
<ListRow :row="row" v-for="row in students.data"> </div>
<template #default="{ column, item }">
<ListRowItem :item="row[column.key]" :align="column.align"> <div
<template #prefix> class="flex items-center border py-2 px-3 rounded-md text-ink-gray-7"
<div v-if="column.key == 'full_name'"> >
<Avatar <div class="p-2 rounded-md bg-surface-gray-2 mr-3">
class="flex items-center" <GraduationCap class="w-5 h-5 stroke-1.5" />
:image="row['user_image']" </div>
:label="item" <div class="flex items-center space-x-2">
size="sm" <span class="font-semibold">
/> {{ certificationCount.data }}
</div> </span>
</template> <span class="">
<div> {{ __('Certified') }}
{{ row[column.key] }} </span>
</div> </div>
</ListRowItem> </div>
</template>
</ListRow> <div
</ListRows> class="flex items-center border py-2 px-3 rounded-md text-ink-gray-7"
<ListSelectBanner> >
<template #actions="{ unselectAll, selections }"> <div class="p-2 rounded-md bg-surface-gray-2 mr-3">
<div class="flex gap-2"> <BookOpen class="w-5 h-5 stroke-1.5" />
<Button </div>
variant="ghost" <div class="flex items-center space-x-2">
@click="removeStudents(selections, unselectAll)" <span class="font-semibold">
> {{ batch.courses?.length }}
<Trash2 class="h-4 w-4 stroke-1.5" /> </span>
</Button> <span>
{{ __('Courses') }}
</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">
<ShieldCheck class="w-5 h-5 stroke-1.5" />
</div>
<div class="flex items-center space-x-2">
<span class="font-semibold">
{{ assessmentCount }}
</span>
<span>
{{ __('Assessments') }}
</span>
</div>
</div>
</div>
<div v-if="showProgressChart" class="mb-8">
<div class="text-ink-gray-7 font-medium">
{{ __('Progress') }}
</div>
<ApexChart
:options="chartOptions"
:series="chartData"
type="bar"
:height="chartData[0].data.length * 30 + 100"
/>
<div
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="w-3 h-3 rounded-sm"
:style="{ 'background-color': theme.colors.green[600] }"
></div>
<div>
{{ __('Courses') }}
</div> </div>
</div>
<div class="flex items-center space-x-2">
<div
class="w-3 h-3 rounded-sm"
:style="{ 'background-color': theme.colors.blue[600] }"
></div>
<div>
{{ __('Assessments') }}
</div>
</div>
</div>
</div>
</div>
<div>
<div class="flex items-center justify-between mb-4">
<div class="text-ink-gray-7 font-medium">
{{ __('Students') }}
</div>
<Button v-if="!readOnlyMode" @click="openStudentModal()">
<template #prefix>
<Plus class="h-4 w-4" />
</template> </template>
</ListSelectBanner> {{ __('Add') }}
</ListView> </Button>
</div> </div>
<div v-else class="text-sm italic text-gray-600">
{{ __('There are no students in this batch.') }} <div v-if="students.data?.length">
<ListView
:columns="getStudentColumns()"
:rows="students.data"
row-key="name"
:options="{
showTooltip: false,
}"
>
<ListHeader
class="mb-2 grid items-center space-x-4 rounded bg-surface-gray-2 p-2"
>
<ListHeaderItem
:item="item"
v-for="item in getStudentColumns()"
:title="item.label"
>
<template #prefix="{ item }">
<FeatherIcon
v-if="item.icon"
:name="item.icon"
class="h-4 w-4 stroke-1.5"
/>
</template>
</ListHeaderItem>
</ListHeader>
<ListRows>
<ListRow
:row="row"
v-for="row in students.data"
class="group cursor-pointer"
@click="openStudentProgressModal(row)"
>
<template #default="{ column, item }">
<ListRowItem
:item="row[column.key]"
:align="column.align"
class="text-sm"
>
<template #prefix>
<div v-if="column.key == 'full_name'">
<Avatar
class="flex items-center"
:image="row['user_image']"
:label="item"
size="sm"
/>
</div>
</template>
<div
v-if="column.key == 'progress'"
class="flex items-center space-x-4 w-full"
>
<ProgressBar :progress="row[column.key]" size="sm" />
<div class="text-xs">{{ row[column.key] }}%</div>
</div>
<div v-else>
{{ row[column.key] }}
</div>
</ListRowItem>
</template>
</ListRow>
</ListRows>
<ListSelectBanner>
<template #actions="{ unselectAll, selections }">
<div class="flex gap-2">
<Button
variant="ghost"
@click="removeStudents(selections, unselectAll)"
>
<Trash2 class="h-4 w-4 stroke-1.5" />
</Button>
</div>
</template>
</ListSelectBanner>
</ListView>
</div>
<div v-else class="text-sm italic text-ink-gray-5">
{{ __('There are no students in this batch.') }}
</div>
</div> </div>
<StudentModal <StudentModal
:batch="props.batch" :batch="props.batch.name"
v-model="showStudentModal" v-model="showStudentModal"
v-model:reloadStudents="students" v-model:reloadStudents="students"
/> />
<BatchStudentProgress
:student="selectedStudent"
v-model="showStudentProgressModal"
/>
</template> </template>
<script setup> <script setup>
import { import {
Avatar,
Button,
createResource, createResource,
FeatherIcon,
ListHeader, ListHeader,
ListHeaderItem, ListHeaderItem,
ListSelectBanner, ListSelectBanner,
@@ -82,65 +223,93 @@ import {
ListRows, ListRows,
ListView, ListView,
ListRowItem, ListRowItem,
Avatar,
Button,
} from 'frappe-ui' } from 'frappe-ui'
import { Trash2, Plus } from 'lucide-vue-next' import {
import { ref } from 'vue' BookOpen,
GraduationCap,
Plus,
ShieldCheck,
Trash2,
User,
} from 'lucide-vue-next'
import { ref, watch } from 'vue'
import StudentModal from '@/components/Modals/StudentModal.vue' import StudentModal from '@/components/Modals/StudentModal.vue'
import { showToast } from '@/utils' import { showToast } from '@/utils'
import ProgressBar from '@/components/ProgressBar.vue'
import BatchStudentProgress from '@/components/Modals/BatchStudentProgress.vue'
import ApexChart from 'vue3-apexcharts'
import { theme } from '@/utils/theme'
const showStudentModal = ref(false) const showStudentModal = ref(false)
const showStudentProgressModal = ref(false)
const selectedStudent = ref(null)
const chartData = ref(null)
const chartOptions = ref(null)
const showProgressChart = ref(false)
const assessmentCount = ref(0)
const readOnlyMode = window.read_only_mode
const props = defineProps({ const props = defineProps({
batch: { batch: {
type: String, type: Object,
default: null, default: null,
}, },
}) })
const students = createResource({ const students = createResource({
url: 'lms.lms.utils.get_batch_students', url: 'lms.lms.utils.get_batch_students',
cache: ['students', props.batch], cache: ['students', props.batch.name],
params: { params: {
batch: props.batch, batch: props.batch?.name,
}, },
auto: true, auto: true,
onSuccess(data) {
chartData.value = getChartData()
showProgressChart.value =
data.length && (props.batch?.courses?.length || assessmentCount.value)
},
}) })
const getStudentColumns = () => { const getStudentColumns = () => {
return [ let columns = [
{ {
label: 'Full Name', label: 'Full Name',
key: 'full_name', key: 'full_name',
width: 2, width: '20rem',
icon: 'user',
}, },
{ {
label: 'Courses Done', label: 'Progress',
key: 'courses_completed', key: 'progress',
align: 'center', width: '15rem',
}, icon: 'activity',
{
label: 'Assessments Done',
key: 'assessments_completed',
align: 'center',
}, },
{ {
label: 'Last Active', label: 'Last Active',
key: 'last_active', key: 'last_active',
width: '10rem',
align: 'center',
icon: 'clock',
}, },
] ]
return columns
} }
const openStudentModal = () => { const openStudentModal = () => {
showStudentModal.value = true showStudentModal.value = true
} }
const openStudentProgressModal = (row) => {
showStudentProgressModal.value = true
selectedStudent.value = row
}
const deleteStudents = createResource({ 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,
} }
}, },
@@ -160,4 +329,119 @@ const removeStudents = (selections, unselectAll) => {
} }
) )
} }
const getChartData = () => {
let categories = {}
if (!students.data?.length) return []
Object.keys(students.data[0].courses).forEach((course) => {
categories[course] = {
value: 0,
type: 'course',
label: course,
}
})
Object.keys(students.data?.[0].assessments).forEach((assessment) => {
categories[assessment] = {
value: 0,
type: 'assessment',
label: assessment,
}
})
students.data.forEach((student) => {
Object.keys(student.courses).forEach((course) => {
if (student.courses[course] === 100) {
categories[course].value += 1
}
})
Object.keys(student.assessments).forEach((assessment) => {
if (student.assessments[assessment].result === 'Pass') {
categories[assessment].value += 1
}
})
})
chartOptions.value = getChartOptions(categories)
return [
{
name: __('Completed by Students'),
data: Object.values(categories).map((item) => item.value),
},
]
}
const getChartOptions = (categories) => {
const courseColor = theme.colors.green[700]
const assessmentColor = theme.colors.blue[700]
const maxY =
students.data?.length % 5
? students.data?.length + (5 - (students.data?.length % 5))
: students.data?.length
return {
chart: {
type: 'bar',
toolbar: {
show: false,
},
},
plotOptions: {
bar: {
distributed: true,
borderRadius: 3,
borderRadiusApplication: 'end',
horizontal: true,
barHeight: '40%',
},
},
colors: Object.values(categories).map((item) =>
item.type === 'course' ? courseColor : assessmentColor
),
xaxis: {
categories: Object.values(categories).map((item) => item.label),
labels: {
style: {
fontSize: '10px',
},
rotate: 0,
formatter: function (value) {
return value.length > 30 ? `${value.substring(0, 30)}...` : value
},
},
},
yaxis: {
max: maxY,
min: 0,
stepSize: 10,
tickAmount: maxY / 5,
/* reversed: true */
},
}
}
watch(students, () => {
if (students.data?.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>
.apexcharts-legend {
display: none !important;
}
</style>

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,85 @@
<template>
<Button
v-if="certification.data && certification.data.certificate"
@click="downloadCertificate"
class=""
>
<template #prefix>
<GraduationCap class="size-4 stroke-1.5" />
</template>
{{ __('View Certificate') }}
</Button>
<div
v-else-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.certificate"
: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: user.data ? true : false,
cache: ['certificationData', user.data?.name],
})
const downloadCertificate = () => {
window.open(
`/api/method/frappe.utils.print_format.download_pdf?doctype=LMS+Certificate&name=${
certification.data.certificate.name
}&format=${encodeURIComponent(certification.data.certificate.template)}`
)
}
</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,7 @@
</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 border-2">
<div class="relative px-1.5 pt-0.5"> <div class="relative px-1.5 pt-0.5">
<ComboboxInput <ComboboxInput
ref="search" ref="search"
@@ -47,7 +47,7 @@
class="absolute right-1.5 inline-flex h-7 w-7 items-center justify-center" class="absolute right-1.5 inline-flex h-7 w-7 items-center justify-center"
@click="selectedValue = null" @click="selectedValue = null"
> >
<X class="h-4 w-4 stroke-1.5" /> <X class="h-4 w-4 stroke-1.5 text-ink-gray-7" />
</button> </button>
</div> </div>
<ComboboxOptions <ComboboxOptions
@@ -62,7 +62,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 +76,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
@@ -87,13 +87,13 @@
name="item-label" name="item-label"
v-bind="{ active, selected, option }" v-bind="{ active, selected, option }"
> >
<div class="flex flex-col space-y-1"> <div class="flex flex-col space-y-1 text-ink-gray-8">
<div> <div>
{{ option.label }} {{ option.label }}
</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 +103,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 +128,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 +243,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 +264,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()
}) })
@@ -156,6 +154,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 +174,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"
@@ -29,8 +29,8 @@
<slot name="item-label" v-bind="{ active, selected, option }" /> <slot name="item-label" v-bind="{ active, selected, option }" />
</template> </template>
<template v-if="attrs.onCreate" #footer="{ value, close }"> <template #footer="{ value, close }">
<div> <div v-if="attrs.onCreate">
<Button <Button
variant="ghost" variant="ghost"
class="w-full !justify-start" class="w-full !justify-start"
@@ -42,9 +42,21 @@
</template> </template>
</Button> </Button>
</div> </div>
<div>
<Button
variant="ghost"
class="w-full !justify-start"
:label="__('Clear')"
@click="() => clearValue(close)"
>
<template #prefix>
<X class="h-4 w-4 stroke-1.5" />
</template>
</Button>
</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>
@@ -52,7 +64,7 @@
import Autocomplete from '@/components/Controls/Autocomplete.vue' import Autocomplete from '@/components/Controls/Autocomplete.vue'
import { watchDebounced } from '@vueuse/core' import { watchDebounced } from '@vueuse/core'
import { createResource, Button } from 'frappe-ui' import { createResource, Button } from 'frappe-ui'
import { Plus } from 'lucide-vue-next' import { Plus, X } from 'lucide-vue-next'
import { useAttrs, computed, ref } from 'vue' import { useAttrs, computed, ref } from 'vue'
const props = defineProps({ const props = defineProps({
@@ -75,9 +87,7 @@ const props = defineProps({
}) })
const emit = defineEmits(['update:modelValue', 'change']) const emit = defineEmits(['update:modelValue', 'change'])
const attrs = useAttrs() const attrs = useAttrs()
const valuePropPassed = computed(() => 'value' in attrs) const valuePropPassed = computed(() => 'value' in attrs)
const value = computed({ const value = computed({
@@ -131,7 +141,7 @@ const options = createResource({
}, },
}) })
function reload(val) { const reload = (val) => {
options.update({ options.update({
params: { params: {
txt: val, txt: val,
@@ -142,13 +152,18 @@ function reload(val) {
options.reload() options.reload()
} }
const clearValue = (close) => {
emit(valuePropPassed.value ? 'change' : 'update:modelValue', '')
close()
}
const labelClasses = computed(() => { const labelClasses = computed(() => {
return [ return [
{ {
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,9 +2,9 @@
<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-2">
<Button <Button
ref="emails" ref="emails"
v-for="value in values" v-for="value in values"
@@ -12,7 +12,7 @@
:label="value" :label="value"
theme="gray" theme="gray"
variant="subtle" variant="subtle"
class="rounded-md" class="rounded-md word-break-all"
@keydown.delete.capture.stop="removeLastValue" @keydown.delete.capture.stop="removeLastValue"
> >
<template #suffix> <template #suffix>
@@ -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 border-2"
>
<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 text-ink-gray-8">
{{ 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
@@ -9,20 +9,23 @@
:class="{ 'default-image': !course.image }" :class="{ 'default-image': !course.image }"
:style="{ backgroundImage: 'url(\'' + encodeURI(course.image) + '\')' }" :style="{ backgroundImage: 'url(\'' + encodeURI(course.image) + '\')' }"
> >
<div <div class="flex items-center flex-wrap relative top-4 px-2 w-fit">
class="flex items-center flex-wrap space-x-1 relative top-4 px-2 w-fit" <Badge
> v-if="course.featured"
<Badge v-if="course.featured" variant="subtle" theme="green" size="md"> variant="subtle"
theme="green"
size="md"
class="mb-1 mr-1"
>
{{ __('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 mb-1 mr-1"
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 +35,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 +44,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 +53,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>
@@ -59,7 +62,7 @@
<div v-if="course.status != 'Approved'"> <div v-if="course.status != 'Approved'">
<Badge <Badge
variant="solid" variant="subtle"
:theme="course.status === 'Under Review' ? 'orange' : 'blue'" :theme="course.status === 'Under Review' ? 'orange' : 'blue'"
size="sm" size="sm"
> >
@@ -68,11 +71,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 +84,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 +105,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,125 +1,160 @@
<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="!readOnlyMode">
v-if="course.data.membership" <div v-if="course.data.membership" class="space-y-2">
:to="{ <router-link
name: 'Lesson', :to="{
params: { name: 'Lesson',
courseName: course.name, params: {
chapterNumber: course.data.current_lesson courseName: course.name,
? course.data.current_lesson.split('-')[0] chapterNumber: course.data.current_lesson
: 1, ? course.data.current_lesson.split('-')[0]
lessonNumber: course.data.current_lesson : 1,
? course.data.current_lesson.split('-')[1] lessonNumber: course.data.current_lesson
: 1, ? course.data.current_lesson.split('-')[1]
}, : 1,
}" },
> }"
<Button variant="solid" size="md" class="w-full"> >
<Button variant="solid" size="md" class="w-full">
<span>
{{ __('Continue Learning') }}
</span>
</Button>
</router-link>
<CertificationLinks :courseName="course.data.name" class="w-full" />
</div>
<router-link
v-else-if="course.data.paid_course"
:to="{
name: 'Billing',
params: {
type: 'course',
name: course.data.name,
},
}"
>
<Button variant="solid" size="md" class="w-full">
<span>
{{ __('Buy this course') }}
</span>
</Button>
</router-link>
<Badge
v-else-if="course.data.disable_self_learning"
theme="blue"
size="lg"
>
{{ __('Contact the Administrator to enroll for this course.') }}
</Badge>
<Button
v-else
@click="enrollStudent()"
variant="solid"
class="w-full"
size="md"
>
<span> <span>
{{ __('Continue Learning') }} {{ __('Start Learning') }}
</span> </span>
</Button> </Button>
</router-link> <Button
<router-link v-if="canGetCertificate"
v-else-if="course.data.paid_course" @click="fetchCertificate()"
:to="{ variant="subtle"
name: 'Billing', class="w-full mt-2"
params: { size="md"
type: 'course', >
name: course.data.name, {{ __('Get Certificate') }}
},
}"
>
<Button variant="solid" size="md" class="w-full">
<span>
{{ __('Buy this course') }}
</span>
</Button> </Button>
</router-link> <router-link
<div v-if="user?.data?.is_moderator || is_instructor()"
v-else-if="course.data.disable_self_learning" :to="{
class="bg-blue-100 text-blue-900 text-sm rounded-md py-1 px-3" name: 'CourseForm',
> params: {
{{ __('Contact the Administrator to enroll for this course.') }} courseName: course.data.name,
},
}"
>
<Button variant="subtle" class="w-full mt-2" size="md">
<span>
{{ __('Edit') }}
</span>
</Button>
</router-link>
</div> </div>
<Button <div class="space-y-4">
v-else <div
@click="enrollStudent()" class="font-medium text-ink-gray-9"
variant="solid" :class="{ 'mt-8': !readOnlyMode }"
class="w-full" >
size="md" {{ __('This course has:') }}
> </div>
<span> <div class="flex items-center text-ink-gray-9">
{{ __('Start Learning') }} <BookOpen class="h-4 w-4 stroke-1.5" />
</span> <span class="ml-2">
</Button> {{ course.data.lessons }} {{ __('Lessons') }}
<Button
v-if="canGetCertificate"
@click="fetchCertificate()"
variant="subtle"
class="w-full mt-2"
size="md"
>
{{ __('Get Certificate') }}
</Button>
<router-link
v-if="user?.data?.is_moderator || is_instructor()"
:to="{
name: 'CourseForm',
params: {
courseName: course.data.name,
},
}"
>
<Button variant="subtle" class="w-full mt-2" size="md">
<span>
{{ __('Edit') }}
</span> </span>
</Button> </div>
</router-link> <div class="flex items-center text-ink-gray-9">
<div class="mt-8 mb-4 font-medium"> <Users class="h-4 w-4 stroke-1.5" />
{{ __('This course has:') }} <span class="ml-2">
</div> {{ formatAmount(course.data.enrollments) }}
<div class="flex items-center mb-3"> {{ __('Enrolled Students') }}
<BookOpen class="h-5 w-5 stroke-1.5 text-gray-600" /> </span>
<span class="ml-2"> </div>
{{ course.data.lessons }} {{ __('Lessons') }} <div
</span> v-if="parseInt(course.data.rating) > 0"
</div> class="flex items-center text-ink-gray-9"
<div class="flex items-center mb-3"> >
<Users class="h-5 w-5 stroke-1.5 text-gray-600" /> <Star class="h-4 w-4 stroke-1.5 fill-orange-500 text-gray-50" />
<span class="ml-2"> <span class="ml-2">
{{ formatAmount(course.data.enrollments) }} {{ course.data.rating }} {{ __('Rating') }}
{{ __('Enrolled Students') }} </span>
</span> </div>
</div> <div
<div class="flex items-center"> v-if="course.data.enable_certification"
<Star class="h-5 w-5 stroke-1.5 fill-orange-500 text-gray-50" /> class="flex items-center font-semibold text-ink-gray-9"
<span class="ml-2"> {{ course.data.rating }} {{ __('Rating') }} </span> >
<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 { Badge, Button, createResource } 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')
const readOnlyMode = window.read_only_mode
const props = defineProps({ const props = defineProps({
course: { course: {
@@ -144,7 +179,7 @@ function enrollStudent() {
) )
setTimeout(() => { setTimeout(() => {
window.location.href = `/login?redirect-to=${window.location.pathname}` window.location.href = `/login?redirect-to=${window.location.pathname}`
}, 2000) }, 1000)
} else { } else {
const enrollStudentResource = createResource({ const enrollStudentResource = createResource({
url: 'lms.lms.doctype.lms_enrollment.lms_enrollment.create_membership', url: 'lms.lms.doctype.lms_enrollment.lms_enrollment.create_membership',

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

@@ -1,22 +1,26 @@
<template> <template>
<div class="text-base"> <div class="">
<div <div
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="flex items-center justify-between space-x-2 mb-4 px-2"
:class="{
'sticky top-0 z-10 bg-surface-white border-b px-3 py-2.5 sm:px-5':
allowEdit,
}"
> >
<div class="font-semibold text-lg leading-5"> <div
class="font-semibold text-lg leading-5 text-ink-gray-9"
:class="{ 'font-medium text-p-base': allowEdit }"
>
{{ __(title) }} {{ __(title) }}
</div> </div>
<Button size="sm" v-if="allowEdit" @click="openChapterModal()"> <Button size="sm" v-if="allowEdit" @click="openChapterModal()">
{{ __('Add Chapter') }} {{ __('Add Chapter') }}
</Button> </Button>
<!-- <span class="font-medium cursor-pointer" @click="expandAllChapters()">
{{ expandAll ? __("Collapse all chapters") : __("Expand all chapters") }}
</span> -->
</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 +37,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 +50,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 +73,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-gray-3' : ''
"
>
<router-link <router-link
:to="{ :to="{
name: allowEdit ? 'LessonForm' : 'Lesson', name: allowEdit ? 'LessonForm' : 'Lesson',
@@ -83,21 +92,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"
@@ -130,6 +139,7 @@
</div> </div>
</div> </div>
<ChapterModal <ChapterModal
v-if="user.data"
v-model="showChapterModal" v-model="showChapterModal"
v-model:outline="outline" v-model:outline="outline"
:course="courseName" :course="courseName"
@@ -323,9 +333,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 }}
@@ -27,7 +27,9 @@
</span> </span>
</div> </div>
<Dropdown <Dropdown
v-if="user.data.name == reply.owner && !reply.editable" v-if="
user.data.name == reply.owner && !reply.editable && !readOnlyMode
"
:options="[ :options="[
{ {
label: 'Edit', label: 'Edit',
@@ -63,7 +65,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,15 +73,16 @@
</div> </div>
<TextEditor <TextEditor
v-if="renderEditor && !readOnlyMode"
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 v-if="!readOnlyMode" class="flex justify-between mt-2">
<span> </span> <span> </span>
<Button @click="postReply()"> <Button @click="postReply()">
<span> <span>
@@ -94,7 +97,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 +105,9 @@ 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 readOnlyMode = window.read_only_mode
const props = defineProps({ const props = defineProps({
topic: { topic: {
@@ -124,6 +130,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 +157,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 +196,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

@@ -1,9 +1,13 @@
<template> <template>
<div> <div>
<Button v-if="!singleThread" class="float-right" @click="openTopicModal()"> <Button
v-if="!singleThread && !readOnlyMode"
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 +20,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 +48,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>
@@ -77,6 +81,7 @@ const currentTopic = ref(null)
const socket = inject('$socket') const socket = inject('$socket')
const user = inject('$user') const user = inject('$user')
const showTopicModal = ref(false) const showTopicModal = ref(false)
const readOnlyMode = window.read_only_mode
const props = defineProps({ const props = defineProps({
title: { title: {

View File

@@ -0,0 +1,129 @@
<template>
<div>
<div class="flex items-center justify-between mb-4">
<div>
<div class="text-xl font-semibold mb-1 text-ink-gray-9">
{{ __(label) }}
</div>
<!-- <div class="text-xs text-ink-gray-5">
{{ __(description) }}
</div> -->
</div>
<div class="flex item-center space-x-2">
<FormControl
v-model="search"
:placeholder="__('Search')"
type="text"
:debounce="300"
/>
<Button @click="() => (showForm = !showForm)">
<template #icon>
<Plus v-if="!showForm" class="h-3 w-3 stroke-1.5" />
<X v-else class="h-3 w-3 stroke-1.5" />
</template>
</Button>
</div>
</div>
<!-- Form to add new member -->
<div v-if="showForm" class="flex items-center space-x-2 my-4">
<FormControl
v-model="email"
:placeholder="__('Email')"
type="email"
class="w-full"
/>
<Button @click="addEvaluator()" variant="subtle">
{{ __('Add') }}
</Button>
</div>
<div class="divide-y">
<div
v-for="evaluator in evaluators.data"
@click="openProfile(evaluator.username)"
class="cursor-pointer"
>
<div class="flex items-center justify-between py-3">
<div class="flex items-center space-x-3">
<Avatar
:image="evaluator.user_image"
:label="evaluator.full_name"
size="lg"
/>
<div>
<div class="text-base font-semibold text-ink-gray-9">
{{ evaluator.full_name }}
</div>
<div class="text-xs text-ink-gray-5">
{{ evaluator.evaluator }}
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</template>
<script setup lang="ts">
import { createResource, Button, FormControl, call, Avatar } from 'frappe-ui'
import { ref, watch } from 'vue'
import { Plus, X } from 'lucide-vue-next'
import { useRouter } from 'vue-router'
const show = defineModel('show')
const search = ref('')
const showForm = ref(false)
const email = ref('')
const router = useRouter()
const props = defineProps({
label: {
type: String,
required: true,
},
description: {
type: String,
default: '',
},
show: {
type: Boolean,
},
})
const evaluators = createResource({
url: 'frappe.client.get_list',
makeParams: () => {
return {
doctype: 'Course Evaluator',
fields: ['evaluator', 'full_name', 'user_image', 'username'],
filters: search.value ? { evaluator: ['like', `%${search.value}%`] } : {},
}
},
auto: true,
})
const addEvaluator = () => {
call('lms.lms.api.add_an_evaluator', {
email: email.value,
}).then((data) => {
showForm.value = false
email.value = ''
evaluators.reload()
})
}
watch(search, () => {
evaluators.reload()
})
const openProfile = (username) => {
show.value = false
router.push({
name: 'Profile',
params: {
username: username,
},
})
}
</script>

View File

@@ -1,23 +0,0 @@
<template>
<svg
width="18"
height="18"
viewBox="0 0 18 18"
fill="none"
xmlns="http://www.w3.org/2000/svg"
>
<g clip-path="url(#clip0_1584_1676)">
<path
fill-rule="evenodd"
clip-rule="evenodd"
d="M3.17474 0.625C2.34632 0.625 1.67474 1.29657 1.67474 2.125V7.475C1.67474 8.30343 2.34632 8.975 3.17474 8.975H14.8247C15.6532 8.975 16.3247 8.30343 16.3247 7.475V2.125C16.3247 1.29657 15.6532 0.625 14.8247 0.625H3.17474ZM2.67474 2.125C2.67474 1.84886 2.8986 1.625 3.17474 1.625H14.8247C15.1009 1.625 15.3247 1.84886 15.3247 2.125V7.475C15.3247 7.75114 15.1009 7.975 14.8247 7.975H3.17474C2.8986 7.975 2.67474 7.75114 2.67474 7.475V2.125ZM4.27478 10.0749C3.99864 10.0749 3.77478 10.2987 3.77478 10.5749V12.6749C3.77478 12.951 3.99864 13.1749 4.27478 13.1749C4.55092 13.1749 4.77478 12.951 4.77478 12.6749V11.0749H6.92478V12.6749C6.92478 12.951 7.14864 13.1749 7.42478 13.1749C7.70092 13.1749 7.92478 12.951 7.92478 12.6749V10.5749C7.92478 10.2987 7.70092 10.0749 7.42478 10.0749H4.27478ZM10.0749 10.5749C10.0749 10.2987 10.2987 10.0749 10.5749 10.0749H13.7249C14.001 10.0749 14.2249 10.2987 14.2249 10.5749V12.6749C14.2249 12.951 14.001 13.1749 13.7249 13.1749C13.4487 13.1749 13.2249 12.951 13.2249 12.6749V11.0749H11.0749V12.6749C11.0749 12.951 10.851 13.1749 10.5749 13.1749C10.2987 13.1749 10.0749 12.951 10.0749 12.6749V10.5749ZM1.125 14.275C0.848858 14.275 0.625 14.4988 0.625 14.775V16.875C0.625 17.1511 0.848858 17.375 1.125 17.375C1.40114 17.375 1.625 17.1511 1.625 16.875V15.275H3.775V16.875C3.775 17.1511 3.99886 17.375 4.275 17.375C4.55114 17.375 4.775 17.1511 4.775 16.875V14.775C4.775 14.4988 4.55114 14.275 4.275 14.275H1.125ZM13.2252 14.775C13.2252 14.4988 13.4491 14.275 13.7252 14.275H16.8752C17.1514 14.275 17.3752 14.4988 17.3752 14.775V16.875C17.3752 17.1511 17.1514 17.375 16.8752 17.375C16.5991 17.375 16.3752 17.1511 16.3752 16.875V15.275H14.2252V16.875C14.2252 17.1511 14.0014 17.375 13.7252 17.375C13.4491 17.375 13.2252 17.1511 13.2252 16.875V14.775ZM7.42511 14.275C7.14897 14.275 6.92511 14.4988 6.92511 14.775V16.875C6.92511 17.1511 7.14897 17.375 7.42511 17.375C7.70125 17.375 7.92511 17.1511 7.92511 16.875V15.275H10.0751V16.875C10.0751 17.1511 10.299 17.375 10.5751 17.375C10.8513 17.375 11.0751 17.1511 11.0751 16.875V14.775C11.0751 14.4988 10.8513 14.275 10.5751 14.275H7.42511Z"
fill="#525252"
/>
</g>
<defs>
<clipPath id="clip0_1584_1676">
<rect width="18" height="18" fill="white" />
</clipPath>
</defs>
</svg>
</template>

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

@@ -0,0 +1,16 @@
<template>
<svg
width="16"
height="16"
viewBox="0 0 16 16"
fill="none"
xmlns="http://www.w3.org/2000/svg"
>
<path
fill-rule="evenodd"
clip-rule="evenodd"
d="M13.5 0C13.7761 0 14 0.223858 14 0.5V2H15.5C15.7761 2 16 2.22386 16 2.5C16 2.77614 15.7761 3 15.5 3H14V4.5C14 4.77614 13.7761 5 13.5 5C13.2239 5 13 4.77614 13 4.5V3H11.5C11.2239 3 11 2.77614 11 2.5C11 2.22386 11.2239 2 11.5 2H13V0.5C13 0.223858 13.2239 0 13.5 0ZM7.9998 2C4.6862 2 2 4.6862 2 7.9998C2 9.49431 2.54643 10.8612 3.45041 11.9116C4.18218 10.8499 5.63104 9.51974 7.99595 9.50011L8.0001 9.50008C9.89267 9.50009 11.5613 10.456 12.5506 11.91C13.4537 10.8598 13.9996 9.49355 13.9996 7.9998C13.9996 7.72366 14.2235 7.4998 14.4996 7.4998C14.7757 7.4998 14.9996 7.72366 14.9996 7.9998C14.9996 11.8657 11.8657 14.9996 7.9998 14.9996C4.13392 14.9996 1 11.8657 1 7.9998C1 4.13392 4.13392 1 7.9998 1C8.27594 1 8.4998 1.22386 8.4998 1.5C8.4998 1.77614 8.27594 2 7.9998 2ZM11.8227 12.6242C11.0281 11.3487 9.61378 10.5008 8.00216 10.5001C5.94811 10.518 4.73746 11.7366 4.17676 12.6241C5.21484 13.4833 6.54702 13.9996 7.9998 13.9996C9.45251 13.9996 10.7846 13.4833 11.8227 12.6242ZM8 4.5C7.0335 4.5 6.25 5.2835 6.25 6.25C6.25 7.2165 7.0335 8 8 8C8.9665 8 9.75 7.2165 9.75 6.25C9.75 5.2835 8.9665 4.5 8 4.5ZM5.25 6.25C5.25 4.73122 6.48122 3.5 8 3.5C9.51878 3.5 10.75 4.73122 10.75 6.25C10.75 7.76878 9.51878 9 8 9C6.48122 9 5.25 7.76878 5.25 6.25Z"
fill="currentColor"
/>
</svg>
</template>

View File

@@ -1,36 +1,18 @@
<template> <template>
<svg <svg
width="118" width="80"
height="118" height="79"
viewBox="0 0 118 118" viewBox="0 0 80 79"
fill="none" fill="none"
xmlns="http://www.w3.org/2000/svg" xmlns="http://www.w3.org/2000/svg"
> >
<path <path
d="M93.9278 0H23.1013C10.3428 0 0 10.3428 0 23.1013V93.9278C0 106.686 10.3428 117.029 23.1013 117.029H93.9278C106.686 117.029 117.029 106.686 117.029 93.9278V23.1013C117.029 10.3428 106.686 0 93.9278 0Z" d="M57.1285 0.580383H22.8514C10.2309 0.580383 0 10.5649 0 22.8815V56.3332C0 68.6497 10.2309 78.6343 22.8514 78.6343H57.1285C69.749 78.6343 79.9799 68.6497 79.9799 56.3332V22.8815C79.9799 10.5649 69.749 0.580383 57.1285 0.580383Z"
fill="url(#paint0_radial_174_336)" fill="#0E7159"
/> />
<path <path
d="M93.9278 0H23.1013C10.3428 0 0 10.3428 0 23.1013V93.9278C0 106.686 10.3428 117.029 23.1013 117.029H93.9278C106.686 117.029 117.029 106.686 117.029 93.9278V23.1013C117.029 10.3428 106.686 0 93.9278 0Z" d="M62.8434 23.6906L60.7869 23.1052C53.6744 21.0702 45.9048 22.4641 39.992 26.8128C35.8502 23.7742 30.7943 22.1854 25.7099 22.2133H17.1406V27.8163H25.7099C29.6232 27.8163 33.508 29.015 36.6787 31.3845L39.992 33.8377L43.3056 31.3845C47.2475 28.4575 52.3032 27.2588 57.1306 28.0393V50.647C51.1035 49.9223 44.9051 51.4834 39.992 55.0795C35.8502 52.0688 30.8515 50.4798 25.7671 50.4798C24.7959 50.4798 23.8247 50.5355 22.8535 50.647V35.0642H17.1406V57.0588H62.8434V23.7185V23.6906Z"
fill="#0B3D3D" fill="white"
fill-opacity="0.8"
/> />
<path
d="M95.1879 33.1294L91.4077 32.0268C80.1721 28.7716 67.9389 30.9242 58.5409 37.7496C52.083 33.0769 43.9975 30.5042 36.1746 30.5042H21.8938V41.0048H36.2796C42.2649 41.0048 48.1978 42.9999 52.923 46.6226L58.5934 50.9279L64.2637 46.6226C70.144 42.1599 77.5469 40.2698 84.7923 41.2673V76.1818C75.5518 75.2367 66.2063 77.7044 58.6459 83.2172C51.0854 77.7044 41.6349 75.2367 32.4994 76.1818V52.8705H21.9988V86.4724H95.3454V33.1294H95.1879Z"
fill="#58FF9B"
/>
<defs>
<radialGradient
id="paint0_radial_174_336"
cx="0"
cy="0"
r="1"
gradientUnits="userSpaceOnUse"
gradientTransform="translate(117.24 -101.5) rotate(105.042) scale(226.282)"
>
<stop offset="0.445162" stop-color="#1F7676" />
<stop offset="1" stop-color="#0A4B4B" />
</radialGradient>
</defs>
</svg> </svg>
</template> </template>

View File

@@ -0,0 +1,16 @@
<template>
<svg
width="20"
height="20"
viewBox="0 0 68 75"
fill="none"
xmlns="http://www.w3.org/2000/svg"
>
<path
fill-rule="evenodd"
clip-rule="evenodd"
d="M0 6.78182C0 1.60212 5.5742 -1.65958 10.09 0.879521L64.09 31.2545C68.6916 33.8443 68.6916 40.4693 64.09 43.0595L10.09 73.4345C5.5744 75.9736 0 72.7119 0 67.5322V6.78182ZM26.2695 38.5201C26.2695 37.3248 25.2265 37.9342 26.2695 38.5201C27.332 39.1178 27.332 37.9225 26.2695 38.5201Z"
fill="white"
/>
</svg>
</template>

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