Compare commits

...

150 Commits

Author SHA1 Message Date
frappe-pr-bot
d4c0ddb191 chore: update POT file 2025-03-28 16:04:20 +00:00
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
100 changed files with 28711 additions and 9328 deletions

View File

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

View File

@@ -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,
}); });
}); });

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

@@ -0,0 +1,96 @@
/* 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']
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']
OnboardingBanner: typeof import('./src/components/OnboardingBanner.vue')['default']
PageModal: typeof import('./src/components/Modals/PageModal.vue')['default']
PaymentSettings: typeof import('./src/components/PaymentSettings.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

@@ -26,12 +26,13 @@
"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.112", "frappe-ui": "^0.1.122",
"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",
"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",

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) {
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

@@ -62,25 +62,48 @@
</div> </div>
</div> </div>
</div> </div>
<div> <div class="m-2 flex flex-col gap-1">
<TrialBanner <TrialBanner
v-if=" v-if="
userResource.data?.is_system_manager && userResource.data?.is_fc_site userResource.data?.is_system_manager && userResource.data?.is_fc_site
" "
:isSidebarCollapsed="sidebarStore.isSidebarCollapsed" :isSidebarCollapsed="sidebarStore.isSidebarCollapsed"
/> />
<GettingStartedBanner
v-if="showOnboarding && !isOnboardingStepsCompleted"
:isSidebarCollapsed="sidebarStore.isSidebarCollapsed"
appName="learning"
/>
<SidebarLink
v-if="isOnboardingStepsCompleted"
:link="{
label: __('Help'),
}"
:isCollapsed="sidebarStore.isSidebarCollapsed"
@click="
() => {
showHelpModal = minimize ? true : !showHelpModal
minimize = !showHelpModal
}
"
>
<template #icon>
<span class="grid h-5 w-6 flex-shrink-0 place-items-center">
<CircleHelp class="h-4 w-4 stroke-1.5" />
</span>
</template>
</SidebarLink>
<SidebarLink <SidebarLink
:link="{ :link="{
label: sidebarStore.isSidebarCollapsed ? 'Expand' : 'Collapse', label: sidebarStore.isSidebarCollapsed ? 'Expand' : 'Collapse',
}" }"
:isCollapsed="sidebarStore.isSidebarCollapsed" :isCollapsed="sidebarStore.isSidebarCollapsed"
@click="toggleSidebar()" @click="toggleSidebar()"
class="m-2"
> >
<template #icon> <template #icon>
<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">
<CollapseSidebar <CollapseSidebar
class="h-4.5 w-4.5 text-ink-gray-7 duration-300 ease-in-out" class="h-4 w-4 text-ink-gray-7 duration-300 ease-in-out"
:class="{ :class="{
'[transform:rotateY(180deg)]': sidebarStore.isSidebarCollapsed, '[transform:rotateY(180deg)]': sidebarStore.isSidebarCollapsed,
}" }"
@@ -89,6 +112,23 @@
</template> </template>
</SidebarLink> </SidebarLink>
</div> </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"
@@ -102,15 +142,38 @@ 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 } from 'frappe-ui'
import { Button, createResource, TrialBanner } 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,
ChevronRight,
Plus,
CircleHelp,
FolderTree,
FileText,
UserPlus,
Users,
BookText,
} 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()
@@ -123,12 +186,27 @@ const isModerator = ref(false)
const isInstructor = ref(false) const isInstructor = ref(false)
const pageToEdit = ref(null) const pageToEdit = ref(null)
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 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(
{}, {},
{ {
@@ -143,7 +221,7 @@ onMounted(() => {
}, },
} }
) )
}) }
const unreadNotifications = createResource({ const unreadNotifications = createResource({
cache: 'Unread Notifications Count', cache: 'Unread Notifications Count',
@@ -187,7 +265,12 @@ const addQuizzes = () => {
label: 'Quizzes', label: 'Quizzes',
icon: 'CircleHelp', icon: 'CircleHelp',
to: 'Quizzes', to: 'Quizzes',
activeFor: ['Quizzes', 'QuizForm'], activeFor: [
'Quizzes',
'QuizForm',
'QuizSubmissionList',
'QuizSubmission',
],
}) })
} }
} }
@@ -198,7 +281,12 @@ const addAssignments = () => {
label: 'Assignments', label: 'Assignments',
icon: 'Pencil', icon: 'Pencil',
to: 'Assignments', to: 'Assignments',
activeFor: ['Assignments', 'AssignmentForm'], activeFor: [
'Assignments',
'AssignmentForm',
'AssignmentSubmissionList',
'AssignmentSubmission',
],
}) })
} }
} }
@@ -261,16 +349,6 @@ const getSidebarFromStorage = () => {
return useStorage('sidebar_is_collapsed', false) return useStorage('sidebar_is_collapsed', false)
} }
watch(userResource, () => {
if (userResource.data) {
isModerator.value = userResource.data.is_moderator
isInstructor.value = userResource.data.is_instructor
addPrograms()
addQuizzes()
addAssignments()
}
})
const toggleSidebar = () => { const toggleSidebar = () => {
sidebarStore.isSidebarCollapsed = !sidebarStore.isSidebarCollapsed sidebarStore.isSidebarCollapsed = !sidebarStore.isSidebarCollapsed
localStorage.setItem( localStorage.setItem(
@@ -286,4 +364,218 @@ const toggleWebPages = () => {
JSON.stringify(sidebarStore.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, () => {
if (userResource.data) {
isModerator.value = userResource.data.is_moderator
isInstructor.value = userResource.data.is_instructor
addPrograms()
addQuizzes()
addAssignments()
setUpOnboarding()
}
})
</script> </script>

View File

@@ -1,10 +1,13 @@
<template> <template>
<div <div
v-if="assignment.data" v-if="assignment.data"
class="grid grid-cols-[65%,35%] h-full" class="grid grid-cols-2 h-full"
:class="{ 'border rounded-lg': !showTitle }" :class="{ 'border rounded-lg overflow-auto': !showTitle }"
> >
<div class="border-r p-5 overflow-y-auto h-[calc(100vh-3.2rem)]"> <div
class="border-r p-5 overflow-y-auto h-[calc(100vh-3.2rem)]"
:class="{ 'h-full': !showTitle }"
>
<div v-if="showTitle" class="text-lg font-semibold mb-5 text-ink-gray-9"> <div v-if="showTitle" class="text-lg font-semibold mb-5 text-ink-gray-9">
<div v-if="submissionName === 'new'"> <div v-if="submissionName === 'new'">
{{ __('Submission by') }} {{ user.data?.full_name }} {{ __('Submission by') }} {{ user.data?.full_name }}
@@ -50,7 +53,7 @@
!['Pass', 'Fail'].includes(submissionResource.doc?.status) && !['Pass', 'Fail'].includes(submissionResource.doc?.status) &&
submissionResource.doc?.owner == user.data?.name submissionResource.doc?.owner == user.data?.name
" "
class="bg-surface-blue-2 p-3 rounded-md leading-5 text-sm mb-4" 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.") }} {{ __("You've successfully submitted the assignment.") }}
{{ {{
@@ -116,7 +119,7 @@
/> />
</div> </div>
<div v-else> <div v-else>
<div class="text-sm mb-4"> <div class="text-sm mb-2 text-ink-gray-7">
{{ __('Write your answer here') }} {{ __('Write your answer here') }}
</div> </div>
<TextEditor <TextEditor
@@ -138,9 +141,10 @@
<div class="text-sm text-ink-gray-5 font-medium mb-2"> <div class="text-sm text-ink-gray-5 font-medium mb-2">
{{ __('Comments by Evaluator') }}: {{ __('Comments by Evaluator') }}:
</div> </div>
<div class="leading-5"> <div
{{ submissionResource.doc.comments }} class="leading-5 text-ink-gray-9"
</div> v-html="submissionResource.doc.comments"
></div>
</div> </div>
<!-- Grading --> <!-- Grading -->
@@ -198,7 +202,6 @@ const answer = ref(null)
const comments = ref(null) const comments = ref(null)
const router = useRouter() const router = useRouter()
const user = inject('$user') const user = inject('$user')
const showTitle = router.currentRoute.value.name == 'AssignmentSubmission'
const isDirty = ref(false) const isDirty = ref(false)
const props = defineProps({ const props = defineProps({
@@ -210,6 +213,10 @@ const props = defineProps({
type: String, type: String,
default: 'new', default: 'new',
}, },
showTitle: {
type: Boolean,
default: true,
},
}) })
onMounted(() => { onMounted(() => {
@@ -353,6 +360,7 @@ const addNewSubmission = () => {
assignmentID: props.assignmentID, assignmentID: props.assignmentID,
submissionName: data.name, submissionName: data.name,
}, },
query: { fromLesson: router.currentRoute.value.query.fromLesson },
}) })
} else { } else {
markLessonProgress() markLessonProgress()

View File

@@ -1,46 +0,0 @@
<template>
<Assignment
v-if="user.data && submission.data"
:assignmentID="assignmentID"
:submissionName="submission.data?.name || 'new'"
/>
<div v-else class="border rounded-md text-center py-20">
<div>
{{ __('Please login to access the assignment.') }}
</div>
<Button @click="redirectToLogin()" class="mt-2">
<span>
{{ __('Login') }}
</span>
</Button>
</div>
</template>
<script setup>
import { inject, watch } from 'vue'
import { Button, createResource } from 'frappe-ui'
import Assignment from '@/components/Assignment.vue'
const user = inject('$user')
const props = defineProps({
assignmentID: {
type: String,
required: true,
},
})
const submission = createResource({
url: 'frappe.client.get_value',
makeParams(values) {
return {
doctype: 'LMS Assignment Submission',
fieldname: 'name',
filters: {
assignment: props.assignmentID,
member: user.data?.name,
},
}
},
auto: true,
})
</script>

View File

@@ -63,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"

View File

@@ -264,7 +264,8 @@ const students = createResource({
auto: true, auto: true,
onSuccess(data) { onSuccess(data) {
chartData.value = getChartData() chartData.value = getChartData()
showProgressChart.value = data.length && true showProgressChart.value =
data.length && (props.batch?.courses?.length || assessmentCount.value)
}, },
}) })

View File

@@ -1,6 +1,16 @@
<template> <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 <div
v-if=" v-else-if="
certification.data && certification.data &&
certification.data.membership && certification.data.membership &&
certification.data.paid_certificate && certification.data.paid_certificate &&
@@ -25,7 +35,7 @@
</Button> </Button>
</router-link> </router-link>
<router-link <router-link
v-else-if="!certification.data.membership.certficate" v-else-if="!certification.data.membership.certificate"
:to="{ :to="{
name: 'CourseCertification', name: 'CourseCertification',
params: { params: {
@@ -61,7 +71,15 @@ const certification = createResource({
params: { params: {
course: props.courseName, course: props.courseName,
}, },
auto: true, auto: user.data ? true : false,
cache: ['certificationData', user.data?.name], 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> </script>

View File

@@ -16,7 +16,8 @@
{{ __('Featured') }} {{ __('Featured') }}
</Badge> </Badge>
<div <div
v-for="tag in course.tags" v-if="course.tags"
v-for="tag in course.tags?.split(', ')"
class="text-xs bg-white text-gray-800 px-2 py-0.5 rounded-md" class="text-xs bg-white text-gray-800 px-2 py-0.5 rounded-md"
> >
{{ tag }} {{ tag }}

View File

@@ -30,7 +30,7 @@
</span> </span>
</Button> </Button>
</router-link> </router-link>
<CertificationLinks :courseName="course.data.name" /> <CertificationLinks :courseName="course.data.name" class="w-full" />
</div> </div>
<router-link <router-link
v-else-if="course.data.paid_course" v-else-if="course.data.paid_course"

View File

@@ -1,10 +1,17 @@
<template> <template>
<div class="text-base"> <div class="h-full">
<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 text-ink-gray-9"> <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()">
@@ -72,7 +79,7 @@
<div <div
class="outline-lesson pl-8 py-2 pr-4 text-ink-gray-9" class="outline-lesson pl-8 py-2 pr-4 text-ink-gray-9"
:class=" :class="
isActiveLesson(lesson.number) ? 'bg-surface-selected' : '' isActiveLesson(lesson.number) ? 'bg-surface-gray-3' : ''
" "
> >
<router-link <router-link

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,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

@@ -4,7 +4,7 @@
class="youtube-video" class="youtube-video"
:src="getYouTubeVideoSource(youtube.split('/').pop())" :src="getYouTubeVideoSource(youtube.split('/').pop())"
width="100%" width="100%"
height="400" :height="screenSize.width < 640 ? 200 : 400"
frameborder="0" frameborder="0"
allowfullscreen allowfullscreen
></iframe> ></iframe>
@@ -15,7 +15,7 @@
class="youtube-video" class="youtube-video"
:src="getYouTubeVideoSource(block)" :src="getYouTubeVideoSource(block)"
width="100%" width="100%"
height="400" :height="screenSize.width < 640 ? 200 : 400"
frameborder="0" frameborder="0"
allowfullscreen allowfullscreen
></iframe> ></iframe>
@@ -66,6 +66,9 @@
<script setup> <script setup>
import Quiz from '@/components/QuizBlock.vue' import Quiz from '@/components/QuizBlock.vue'
import MarkdownIt from 'markdown-it' import MarkdownIt from 'markdown-it'
import { useScreenSize } from '@/utils/composables'
const screenSize = useScreenSize()
const markdown = new MarkdownIt({ const markdown = new MarkdownIt({
html: true, html: true,

View File

@@ -116,6 +116,7 @@ import { createResource, Avatar, Button, FormControl, Badge } from 'frappe-ui'
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
import { ref, watch, reactive, inject } from 'vue' import { ref, watch, reactive, inject } from 'vue'
import { RefreshCw, Plus, X } from 'lucide-vue-next' import { RefreshCw, Plus, X } from 'lucide-vue-next'
import { useOnboarding } from 'frappe-ui/frappe'
const router = useRouter() const router = useRouter()
const show = defineModel('show') const show = defineModel('show')
@@ -125,6 +126,7 @@ const memberList = ref([])
const hasNextPage = ref(false) const hasNextPage = ref(false)
const showForm = ref(false) const showForm = ref(false)
const dayjs = inject('$dayjs') const dayjs = inject('$dayjs')
const { updateOnboardingStep } = useOnboarding('learning')
const member = reactive({ const member = reactive({
email: '', email: '',
@@ -185,6 +187,7 @@ const newMember = createResource({
auto: false, auto: false,
onSuccess(data) { onSuccess(data) {
show.value = false show.value = false
updateOnboardingStep('invite_students')
router.push({ router.push({
name: 'Profile', name: 'Profile',
params: { params: {

View File

@@ -24,6 +24,7 @@
doctype="Course Evaluator" doctype="Course Evaluator"
v-model="evaluator" v-model="evaluator"
:label="__('Evaluator')" :label="__('Evaluator')"
:onCreate="(value, close) => openSettings(close)"
class="mt-4" class="mt-4"
/> />
</template> </template>
@@ -34,11 +35,15 @@ import { Dialog, createResource } from 'frappe-ui'
import { ref } from 'vue' import { ref } from 'vue'
import Link from '@/components/Controls/Link.vue' import Link from '@/components/Controls/Link.vue'
import { showToast } from '@/utils' import { showToast } from '@/utils'
import { useOnboarding } from 'frappe-ui/frappe'
import { useSettings } from '@/stores/settings'
const show = defineModel() const show = defineModel()
const course = ref(null) const course = ref(null)
const evaluator = ref(null) const evaluator = ref(null)
const courses = defineModel('courses') const courses = defineModel('courses')
const { updateOnboardingStep } = useOnboarding('learning')
const settingsStore = useSettings()
const props = defineProps({ const props = defineProps({
batch: { batch: {
@@ -69,6 +74,7 @@ const addCourse = (close) => {
{ {
onSuccess() { onSuccess() {
courses.value.reload() courses.value.reload()
updateOnboardingStep('add_batch_course')
close() close()
course.value = null course.value = null
evaluator.value = null evaluator.value = null
@@ -79,4 +85,10 @@ const addCourse = (close) => {
} }
) )
} }
const openSettings = (close) => {
close()
settingsStore.activeTab = 'Evaluators'
settingsStore.isSettingsOpen = true
}
</script> </script>

View File

@@ -81,11 +81,11 @@ import { reactive, watch } from 'vue'
import { showToast, getFileSize } from '@/utils/' import { showToast, getFileSize } from '@/utils/'
import { capture } from '@/telemetry' import { capture } from '@/telemetry'
import { FileText, X } from 'lucide-vue-next' import { FileText, X } from 'lucide-vue-next'
import { useSettings } from '@/stores/settings' import { useOnboarding } from 'frappe-ui/frappe'
const show = defineModel() const show = defineModel()
const outline = defineModel('outline') const outline = defineModel('outline')
const settingsStore = useSettings() const { updateOnboardingStep } = useOnboarding('learning')
const props = defineProps({ const props = defineProps({
course: { course: {
@@ -140,14 +140,12 @@ const addChapter = async (close) => {
}, },
onSuccess: (data) => { onSuccess: (data) => {
capture('chapter_created') capture('chapter_created')
updateOnboardingStep('create_first_chapter')
chapterReference.submit( chapterReference.submit(
{ name: data.name }, { name: data.name },
{ {
onSuccess(data) { onSuccess(data) {
cleanChapter() cleanChapter()
/* if (!settingsStore.onboardingDetails.data?.is_onboarded) {
settingsStore.onboardingDetails.reload()
} */
outline.value.reload() outline.value.reload()
showToast( showToast(
__('Success'), __('Success'),

View File

@@ -95,8 +95,8 @@ import {
FormControl, FormControl,
Autocomplete, Autocomplete,
} from 'frappe-ui' } from 'frappe-ui'
import { reactive, inject } from 'vue' import { reactive, inject, onMounted } from 'vue'
import { getTimezones, createToast } from '@/utils/' import { getTimezones, createToast, getUserTimezone } from '@/utils/'
const liveClasses = defineModel('reloadLiveClasses') const liveClasses = defineModel('reloadLiveClasses')
const show = defineModel() const show = defineModel()
@@ -122,6 +122,10 @@ let liveClass = reactive({
host: user.data.name, host: user.data.name,
}) })
onMounted(() => {
liveClass.timezone = getUserTimezone()
})
const getTimezoneOptions = () => { const getTimezoneOptions = () => {
return getTimezones().map((timezone) => { return getTimezones().map((timezone) => {
return { return {

View File

@@ -109,11 +109,13 @@ import { Dialog, FormControl, TextEditor, createResource } from 'frappe-ui'
import { computed, watch, reactive, ref } from 'vue' import { computed, watch, reactive, ref } from 'vue'
import Link from '@/components/Controls/Link.vue' import Link from '@/components/Controls/Link.vue'
import { showToast } from '@/utils' import { showToast } from '@/utils'
import { useOnboarding } from 'frappe-ui/frappe'
const show = defineModel() const show = defineModel()
const quiz = defineModel('quiz') const quiz = defineModel('quiz')
const questionType = ref(null) const questionType = ref(null)
const editMode = ref(false) const editMode = ref(false)
const { updateOnboardingStep } = useOnboarding('learning')
const existingQuestion = reactive({ const existingQuestion = reactive({
question: '', question: '',
@@ -122,7 +124,7 @@ const existingQuestion = reactive({
const question = reactive({ const question = reactive({
question: '', question: '',
type: 'Choices', type: 'Choices',
marks: 0, marks: 1,
}) })
const populateFields = () => { const populateFields = () => {
@@ -261,6 +263,7 @@ const addQuestionRow = (question, close) => {
{ {
onSuccess() { onSuccess() {
show.value = false show.value = false
updateOnboardingStep('create_first_quiz')
showToast(__('Success'), __('Question added successfully'), 'check') showToast(__('Success'), __('Question added successfully'), 'check')
quiz.value.reload() quiz.value.reload()
close() close()

View File

@@ -40,6 +40,12 @@
:description="activeTab.description" :description="activeTab.description"
v-model:show="show" v-model:show="show"
/> />
<Evaluators
v-else-if="activeTab.label === 'Evaluators'"
:label="activeTab.label"
:description="activeTab.description"
v-model:show="show"
/>
<Categories <Categories
v-else-if="activeTab.label === 'Categories'" v-else-if="activeTab.label === 'Categories'"
:label="activeTab.label" :label="activeTab.label"
@@ -78,6 +84,7 @@ import { useSettings } from '@/stores/settings'
import SettingDetails from '../SettingDetails.vue' import SettingDetails from '../SettingDetails.vue'
import SidebarLink from '@/components/SidebarLink.vue' import SidebarLink from '@/components/SidebarLink.vue'
import Members from '@/components/Members.vue' import Members from '@/components/Members.vue'
import Evaluators from '@/components/Evaluators.vue'
import Categories from '@/components/Categories.vue' import Categories from '@/components/Categories.vue'
import BrandSettings from '@/components/BrandSettings.vue' import BrandSettings from '@/components/BrandSettings.vue'
import PaymentSettings from '@/components/PaymentSettings.vue' import PaymentSettings from '@/components/PaymentSettings.vue'
@@ -193,6 +200,11 @@ const tabsStructure = computed(() => {
description: 'Manage the members of your learning system', description: 'Manage the members of your learning system',
icon: 'UserRoundPlus', icon: 'UserRoundPlus',
}, },
{
label: 'Evaluators',
description: 'Manage the evaluators of your learning system',
icon: 'UserCheck',
},
{ {
label: 'Categories', label: 'Categories',
description: 'Manage the members of your learning system', description: 'Manage the members of your learning system',

View File

@@ -29,9 +29,11 @@ import { Dialog, createResource } from 'frappe-ui'
import { ref } from 'vue' import { ref } from 'vue'
import Link from '@/components/Controls/Link.vue' import Link from '@/components/Controls/Link.vue'
import { showToast } from '@/utils' import { showToast } from '@/utils'
import { useOnboarding } from 'frappe-ui/frappe'
const students = defineModel('reloadStudents') const students = defineModel('reloadStudents')
const student = ref() const student = ref()
const { updateOnboardingStep } = useOnboarding('learning')
const show = defineModel() const show = defineModel()
const props = defineProps({ const props = defineProps({
@@ -61,6 +63,7 @@ const addStudent = (close) => {
onSuccess() { onSuccess() {
students.value.reload() students.value.reload()
student.value = null student.value = null
updateOnboardingStep('add_batch_student')
close() close()
}, },
onError(err) { onError(err) {

View File

@@ -0,0 +1,11 @@
<template>
<div class="relative flex h-full flex-col">
<div class="h-full flex-1">
<div class="flex h-screen text-base bg-surface-white">
<div class="w-full overflow-auto" id="scrollContainer">
<slot />
</div>
</div>
</div>
</div>
</template>

View File

@@ -1,7 +1,7 @@
<template> <template>
<div v-if="quiz.data"> <div v-if="quiz.data">
<div <div
class="bg-surface-blue-2 space-y-1 py-2 px-2 mb-4 rounded-md text-sm text-ink-blue-800" class="bg-surface-blue-2 space-y-1 py-2 px-2 mb-4 rounded-md text-sm text-ink-blue-2"
> >
<div class="leading-5"> <div class="leading-5">
{{ {{
@@ -29,7 +29,7 @@
).format(quiz.data.passing_percentage) ).format(quiz.data.passing_percentage)
}} }}
</div> </div>
<div v-if="quiz.data.max_attempts" class="leading-relaxed"> <div v-if="quiz.data.max_attempts" class="leading-5">
{{ {{
__('You can attempt this quiz {0}.').format( __('You can attempt this quiz {0}.').format(
quiz.data.max_attempts == 1 quiz.data.max_attempts == 1
@@ -52,7 +52,7 @@
<div v-if="activeQuestion == 0"> <div v-if="activeQuestion == 0">
<div class="border text-center p-20 rounded-md"> <div class="border text-center p-20 rounded-md">
<div class="font-semibold text-lg"> <div class="font-semibold text-lg text-ink-gray-9">
{{ quiz.data.title }} {{ quiz.data.title }}
</div> </div>
<Button <Button
@@ -67,7 +67,7 @@
{{ __('Start') }} {{ __('Start') }}
</span> </span>
</Button> </Button>
<div v-else> <div v-else class="leading-5 text-ink-gray-7">
{{ {{
__( __(
'You have already exceeded the maximum number of attempts allowed for this quiz.' 'You have already exceeded the maximum number of attempts allowed for this quiz.'
@@ -222,11 +222,14 @@
</div> </div>
</div> </div>
</div> </div>
<div v-else class="border rounded-md p-20 text-center space-y-4"> <div v-else class="border rounded-md p-20 text-center space-y-2">
<div class="text-lg font-semibold"> <div class="text-lg font-semibold text-ink-gray-9">
{{ __('Quiz Summary') }} {{ __('Quiz Summary') }}
</div> </div>
<div v-if="quizSubmission.data.is_open_ended"> <div
v-if="quizSubmission.data.is_open_ended"
class="leading-5 text-ink-gray-7"
>
{{ {{
__( __(
"Your submission has been successfully saved. The instructor will review and grade it shortly, and you'll be notified of your final result." "Your submission has been successfully saved. The instructor will review and grade it shortly, and you'll be notified of your final result."
@@ -613,7 +616,6 @@ const getInstructions = (question) => {
} }
const markLessonProgress = () => { const markLessonProgress = () => {
console.log(router)
if (router.currentRoute.value.name == 'Lesson') { if (router.currentRoute.value.name == 'Lesson') {
call('lms.lms.api.mark_lesson_progress', { call('lms.lms.api.mark_lesson_progress', {
course: router.currentRoute.value.params.courseName, course: router.currentRoute.value.params.courseName,

View File

@@ -2,7 +2,9 @@
<button <button
v-if="link && !link.onlyMobile" v-if="link && !link.onlyMobile"
class="flex h-7 cursor-pointer items-center rounded text-ink-gray-8 duration-300 ease-in-out focus:outline-none focus:transition-none focus-visible:rounded focus-visible:ring-2 focus-visible:ring-outline-gray-3" class="flex h-7 cursor-pointer items-center rounded text-ink-gray-8 duration-300 ease-in-out focus:outline-none focus:transition-none focus-visible:rounded focus-visible:ring-2 focus-visible:ring-outline-gray-3"
:class="isActive ? 'bg-surface-white shadow-sm' : 'hover:bg-surface-gray-2'" :class="
isActive ? 'bg-surface-selected shadow-sm' : 'hover:bg-surface-gray-2'
"
@click="handleClick" @click="handleClick"
> >
<div <div

View File

@@ -18,8 +18,51 @@
<div class="grid grid-cols-3 gap-4"> <div class="grid grid-cols-3 gap-4">
<div v-for="evl in upcoming_evals.data"> <div v-for="evl in upcoming_evals.data">
<div class="border rounded-md p-3"> <div class="border rounded-md p-3">
<div class="font-semibold mb-3"> <div class="flex justify-between mb-3">
{{ evl.course_title }} <span class="font-semibold leading-5">
{{ evl.course_title }}
</span>
<Menu
v-if="evl.date > dayjs().format()"
as="div"
class="relative inline-block text-left"
>
<div>
<MenuButton class="inline-flex w-full justify-center">
<EllipsisVertical class="w-4 h-4 stroke-1.5" />
</MenuButton>
</div>
<transition
enter-active-class="transition duration-100 ease-out"
enter-from-class="transform scale-95 opacity-0"
enter-to-class="transform scale-100 opacity-100"
leave-active-class="transition duration-75 ease-in"
leave-from-class="transform scale-100 opacity-100"
leave-to-class="transform scale-95 opacity-0"
>
<MenuItems
class="absolute mt-2 w-32 rounded-md bg-white shadow-lg p-1.5"
>
<MenuItem v-slot="{ active }">
<Button
variant="ghost"
class="w-full"
@click="cancelEvaluation(evl)"
>
<template #prefix>
<Ban
:active="active"
class="size-4 stroke-1.5"
aria-hidden="true"
/>
</template>
{{ __('Cancel') }}
</Button>
</MenuItem>
</MenuItems>
</transition>
</Menu>
</div> </div>
<div class="flex items-center mb-2"> <div class="flex items-center mb-2">
<Calendar class="w-4 h-4 stroke-1.5" /> <Calendar class="w-4 h-4 stroke-1.5" />
@@ -50,16 +93,6 @@
</template> </template>
{{ __('Join Call') }} {{ __('Join Call') }}
</Button> </Button>
<Button
v-if="evl.date > dayjs().format()"
@click="cancelEvaluation(evl)"
class="w-full"
>
<template #prefix>
<Ban class="w-4 h-4 stroke-1.5" />
</template>
{{ __('Cancel') }}
</Button>
</div> </div>
</div> </div>
</div> </div>
@@ -84,11 +117,13 @@ import {
Clock, Clock,
GraduationCap, GraduationCap,
HeadsetIcon, HeadsetIcon,
EllipsisVertical,
} from 'lucide-vue-next' } from 'lucide-vue-next'
import { inject, ref, getCurrentInstance } from 'vue' import { inject, ref, getCurrentInstance } from 'vue'
import { formatTime } from '../utils' import { formatTime } from '../utils'
import { Button, createResource, call } from 'frappe-ui' import { Button, createResource, call } from 'frappe-ui'
import EvaluationModal from '@/components/Modals/EvaluationModal.vue' import EvaluationModal from '@/components/Modals/EvaluationModal.vue'
import { Menu, MenuButton, MenuItems, MenuItem } from '@headlessui/vue'
const dayjs = inject('$dayjs') const dayjs = inject('$dayjs')
const user = inject('$user') const user = inject('$user')

View File

@@ -82,6 +82,7 @@ import {
User, User,
Settings, Settings,
Sun, Sun,
Zap,
} from 'lucide-vue-next' } from 'lucide-vue-next'
const router = useRouter() const router = useRouter()
@@ -125,98 +126,115 @@ const toggleTheme = () => {
const userDropdownOptions = computed(() => { const userDropdownOptions = computed(() => {
return [ return [
{ {
icon: User, group: '',
label: 'My Profile', items: [
onClick: () => { {
router.push(`/user/${userResource.data?.username}`) icon: User,
}, label: 'My Profile',
condition: () => { onClick: () => {
return isLoggedIn router.push(`/user/${userResource.data?.username}`)
}, },
condition: () => {
return isLoggedIn
},
},
{
icon: theme.value === 'light' ? Moon : Sun,
label: 'Toggle Theme',
onClick: () => {
toggleTheme()
},
},
{
component: markRaw(Apps),
condition: () => {
let cookies = new URLSearchParams(
document.cookie.split('; ').join('&')
)
let system_user = cookies.get('system_user')
if (system_user === 'yes') return true
else return false
},
},
{
icon: Settings,
label: 'Settings',
onClick: () => {
settingsStore.isSettingsOpen = true
},
condition: () => {
return userResource.data?.is_moderator
},
},
{
icon: FrappeCloudIcon,
label: 'Login to Frappe Cloud',
onClick: () => {
$dialog({
title: __('Login to Frappe Cloud?'),
message: __(
'Are you sure you want to login to your Frappe Cloud dashboard?'
),
actions: [
{
label: __('Confirm'),
variant: 'solid',
onClick(close) {
loginToFrappeCloud()
close()
},
},
],
})
},
condition: () => {
return (
userResource.data?.is_system_manager &&
userResource.data?.is_fc_site
)
},
},
],
}, },
{ {
icon: theme.value === 'light' ? Moon : Sun, group: '',
label: 'Toggle Theme', items: [
onClick: () => { {
toggleTheme() icon: Zap,
}, label: 'Powered by Learning',
}, onClick: () => {
{ window.open('https://frappe.io/learning', '_blank')
component: markRaw(Apps), },
condition: () => { },
let cookies = new URLSearchParams(document.cookie.split('; ').join('&')) {
let system_user = cookies.get('system_user') icon: LogOut,
if (system_user === 'yes') return true label: 'Log out',
else return false onClick: () => {
}, logout.submit().then(() => {
}, isLoggedIn = false
{ })
icon: Settings, },
label: 'Settings', condition: () => {
onClick: () => { return isLoggedIn
settingsStore.isSettingsOpen = true },
}, },
condition: () => { {
return userResource.data?.is_moderator icon: LogIn,
}, label: 'Log in',
}, onClick: () => {
{ window.location.href = '/login'
icon: FrappeCloudIcon, },
label: 'Login to Frappe Cloud', condition: () => {
onClick: () => { return !isLoggedIn
$dialog({ },
title: __('Login to Frappe Cloud?'), },
message: __( ],
'Are you sure you want to login to your Frappe Cloud dashboard?'
),
actions: [
{
label: __('Confirm'),
variant: 'solid',
onClick(close) {
loginToFrappeCloud()
close()
},
},
],
})
},
condition: () => {
return (
userResource.data?.is_system_manager && userResource.data?.is_fc_site
)
},
},
{
icon: LogOut,
label: 'Log out',
onClick: () => {
logout.submit().then(() => {
isLoggedIn = false
})
},
condition: () => {
return isLoggedIn
},
},
{
icon: LogIn,
label: 'Log in',
onClick: () => {
window.location.href = '/login'
},
condition: () => {
return !isLoggedIn
},
}, },
] ]
}) })
const loginToFrappeCloud = () => { const loginToFrappeCloud = () => {
let redirect_to = '/dashboard/welcome' let redirect_to = '/dashboard/sites/' + userResource.data.sitename
if (userResource.data?.site_info.is_payment_method_added) {
redirect_to = '/dashboard/sites/' + userResource.data.sitename
}
window.open(`${frappeCloudBaseEndpoint}${redirect_to}`, '_blank') window.open(`${frappeCloudBaseEndpoint}${redirect_to}`, '_blank')
} }
</script> </script>

View File

@@ -1,19 +1,25 @@
<template> <template>
<header <header
v-if="!fromLesson"
class="flex justify-between sticky top-0 z-10 border-b bg-surface-white px-3 py-2.5 sm:px-5" class="flex justify-between sticky top-0 z-10 border-b bg-surface-white px-3 py-2.5 sm:px-5"
> >
<Breadcrumbs :items="breadcrumbs" /> <Breadcrumbs :items="breadcrumbs" />
</header> </header>
<div class="overflow-hidden h-[calc(100vh-3.2rem)]"> <div class="overflow-hidden h-[calc(100vh-3.2rem)]">
<Assignment :assignmentID="assignmentID" :submissionName="submissionName" /> <Assignment
:assignmentID="assignmentID"
:submissionName="submissionName"
:showTitle="!fromLesson"
/>
</div> </div>
</template> </template>
<script setup> <script setup>
import { Breadcrumbs, createResource } from 'frappe-ui' import { Breadcrumbs, createResource } from 'frappe-ui'
import { computed, inject, onMounted } from 'vue' import { computed, inject, onMounted, onBeforeUnmount, ref } from 'vue'
import Assignment from '@/components/Assignment.vue' import Assignment from '@/components/Assignment.vue'
const user = inject('$user') const user = inject('$user')
const fromLesson = ref(false)
const props = defineProps({ const props = defineProps({
assignmentID: { assignmentID: {
@@ -42,6 +48,10 @@ onMounted(() => {
if (!user.data) { if (!user.data) {
window.location.href = '/login' window.location.href = '/login'
} }
if (new URLSearchParams(window.location.search).get('fromLesson')) {
fromLesson.value = true
}
}) })
const breadcrumbs = computed(() => { const breadcrumbs = computed(() => {

View File

@@ -190,11 +190,15 @@
</div> </div>
</div> </div>
</div> </div>
<BulkCertificates v-model="openCertificateDialog" :batch="batch.data" /> <BulkCertificates
v-if="batch.data"
v-model="openCertificateDialog"
:batch="batch.data"
/>
</template> </template>
<script setup> <script setup>
import { computed, inject, ref } from 'vue' import { computed, inject, ref, onMounted, watch } from 'vue'
import { useRouteQuery } from '@vueuse/router' import { useRoute, useRouter } from 'vue-router'
import { Breadcrumbs, Button, createResource, Tabs, Badge } from 'frappe-ui' import { Breadcrumbs, Button, createResource, Tabs, Badge } from 'frappe-ui'
import CourseInstructors from '@/components/CourseInstructors.vue' import CourseInstructors from '@/components/CourseInstructors.vue'
import UserAvatar from '@/components/UserAvatar.vue' import UserAvatar from '@/components/UserAvatar.vue'
@@ -226,52 +230,10 @@ import BatchFeedback from '@/components/BatchFeedback.vue'
const user = inject('$user') const user = inject('$user')
const showAnnouncementModal = ref(false) const showAnnouncementModal = ref(false)
const openCertificateDialog = ref(false) const openCertificateDialog = ref(false)
const route = useRoute()
const router = useRouter()
const tabIndex = ref(0)
const props = defineProps({
batchName: {
type: String,
required: true,
},
})
const batch = createResource({
url: 'lms.lms.utils.get_batch_details',
cache: ['batch', props.batchName],
params: {
batch: props.batchName,
},
auto: true,
})
const breadcrumbs = computed(() => {
let crumbs = [{ label: 'Batches', route: { name: 'Batches' } }]
if (!isStudent.value) {
crumbs.push({
label: 'Details',
route: {
name: 'BatchDetail',
params: {
batchName: batch.data?.name,
},
},
})
}
crumbs.push({
label: batch?.data?.title,
route: { name: 'Batch', params: { batchName: props.batchName } },
})
return crumbs
})
const isStudent = computed(() => {
return (
user?.data &&
batch.data?.students?.length &&
batch.data?.students.includes(user.data.name)
)
})
const tabIndex = useRouteQuery('tab', 0, { transform: Number })
const tabs = computed(() => { const tabs = computed(() => {
let batchTabs = [] let batchTabs = []
batchTabs.push({ batchTabs.push({
@@ -313,6 +275,61 @@ const tabs = computed(() => {
return batchTabs return batchTabs
}) })
const props = defineProps({
batchName: {
type: String,
required: true,
},
})
onMounted(() => {
const hash = route.hash
if (hash) {
tabs.value.forEach((tab, index) => {
if (tab.label?.toLowerCase() === hash.replace('#', '')) {
tabIndex.value = index
}
})
}
})
const batch = createResource({
url: 'lms.lms.utils.get_batch_details',
cache: ['batch', props.batchName],
params: {
batch: props.batchName,
},
auto: true,
})
const breadcrumbs = computed(() => {
let crumbs = [{ label: 'Batches', route: { name: 'Batches' } }]
if (!isStudent.value) {
crumbs.push({
label: 'Details',
route: {
name: 'BatchDetail',
params: {
batchName: batch.data?.name,
},
},
})
}
crumbs.push({
label: batch?.data?.title,
route: { name: 'Batch', params: { batchName: props.batchName } },
})
return crumbs
})
const isStudent = computed(() => {
return (
user?.data &&
batch.data?.students?.length &&
batch.data?.students.includes(user.data.name)
)
})
const redirectToLogin = () => { const redirectToLogin = () => {
window.location.href = `/login?redirect-to=/lms/batches/${props.batchName}` window.location.href = `/login?redirect-to=/lms/batches/${props.batchName}`
} }
@@ -321,6 +338,13 @@ const openAnnouncementModal = () => {
showAnnouncementModal.value = true showAnnouncementModal.value = true
} }
watch(tabIndex, () => {
const tab = tabs.value[tabIndex.value]
if (tab.label != route.hash.replace('#', '')) {
router.push({ ...route, hash: `#${tab.label.toLowerCase()}` })
}
})
const pageMeta = computed(() => { const pageMeta = computed(() => {
return { return {
title: batch.data?.title, title: batch.data?.title,

View File

@@ -271,9 +271,11 @@ import { showToast } from '@/utils'
import { Image } from 'lucide-vue-next' import { Image } from 'lucide-vue-next'
import { capture } from '@/telemetry' import { capture } from '@/telemetry'
import MultiSelect from '@/components/Controls/MultiSelect.vue' import MultiSelect from '@/components/Controls/MultiSelect.vue'
import { useOnboarding } from 'frappe-ui/frappe'
const router = useRouter() const router = useRouter()
const user = inject('$user') const user = inject('$user')
const { updateOnboardingStep } = useOnboarding('learning')
const props = defineProps({ const props = defineProps({
batchName: { batchName: {
@@ -426,6 +428,9 @@ const createNewBatch = () => {
{ {
onSuccess(data) { onSuccess(data) {
capture('batch_created') capture('batch_created')
updateOnboardingStep('create_first_batch', true, false, () => {
localStorage.setItem('firstBatch', data.name)
})
router.push({ router.push({
name: 'BatchDetail', name: 'BatchDetail',
params: { params: {

View File

@@ -22,7 +22,7 @@
<div <div
class="flex flex-col lg:flex-row space-y-4 lg:space-y-0 lg:items-center justify-between mb-5" class="flex flex-col lg:flex-row space-y-4 lg:space-y-0 lg:items-center justify-between mb-5"
> >
<div class="text-lg font-semibold"> <div class="text-lg text-ink-gray-9 font-semibold">
{{ __('All Batches') }} {{ __('All Batches') }}
</div> </div>
<div <div
@@ -72,7 +72,7 @@
</div> </div>
<div <div
v-else-if="!batches.list.loading" v-else-if="!batches.list.loading"
class="flex flex-col items-center justify-center text-sm text-ink-gray-5 italic mt-48" class="flex flex-col items-center justify-center text-sm text-ink-gray-5 mt-48"
> >
<BookOpen class="size-10 mx-auto stroke-1 text-ink-gray-4" /> <BookOpen class="size-10 mx-auto stroke-1 text-ink-gray-4" />
<div class="text-lg font-medium mb-1"> <div class="text-lg font-medium mb-1">
@@ -119,7 +119,8 @@ const currentCategory = ref(null)
const title = ref('') const title = ref('')
const certification = ref(false) const certification = ref(false)
const filters = ref({}) const filters = ref({})
const currentTab = ref(user.data?.is_student ? 'All' : 'Upcoming') const is_student = computed(() => user.data?.is_student)
const currentTab = ref(is_student.value ? 'All' : 'Upcoming')
const orderBy = ref('start_date') const orderBy = ref('start_date')
onMounted(() => { onMounted(() => {
@@ -204,12 +205,12 @@ const updateTabFilter = () => {
if (!user.data) { if (!user.data) {
return return
} }
if (currentTab.value == 'Enrolled' && user.data?.is_student) { if (currentTab.value == 'Enrolled' && is_student.value) {
filters.value['enrolled'] = 1 filters.value['enrolled'] = 1
delete filters.value['start_date'] delete filters.value['start_date']
delete filters.value['published'] delete filters.value['published']
orderBy.value = 'start_date desc' orderBy.value = 'start_date desc'
} else if (user.data?.is_student) { } else if (is_student.value) {
delete filters.value['enrolled'] delete filters.value['enrolled']
} else { } else {
delete filters.value['start_date'] delete filters.value['start_date']
@@ -228,7 +229,7 @@ const updateTabFilter = () => {
} }
const updateStudentFilter = () => { const updateStudentFilter = () => {
if (!user.data || (user.data?.is_student && currentTab.value != 'Enrolled')) { if (!user.data || (is_student.value && currentTab.value != 'Enrolled')) {
filters.value['start_date'] = ['>=', dayjs().format('YYYY-MM-DD')] filters.value['start_date'] = ['>=', dayjs().format('YYYY-MM-DD')]
filters.value['published'] = 1 filters.value['published'] = 1
} }
@@ -250,7 +251,12 @@ const setQueryParams = () => {
} }
}) })
history.replaceState({}, '', `${location.pathname}?${queries.toString()}`) let queryString = ''
if (queries.toString()) {
queryString = `?${queries.toString()}`
}
history.replaceState({}, '', `${location.pathname}${queryString}`)
} }
const updateCategories = (data) => { const updateCategories = (data) => {
@@ -270,30 +276,23 @@ watch(currentTab, () => {
updateBatches() updateBatches()
}) })
const batchType = computed(() => {
let types = [
{ label: __(''), value: null },
{ label: __('Upcoming'), value: 'Upcoming' },
{ label: __('Archived'), value: 'Archived' },
]
if (user.data?.is_moderator) {
types.push({ label: __('Unpublished'), value: 'Unpublished' })
}
return types
})
const batchTabs = computed(() => { const batchTabs = computed(() => {
let tabs = [ let tabs = [
{ {
label: __('All'), label: __('All'),
}, },
] ]
if (user.data?.is_student) {
tabs.push({ label: __('Enrolled') }) if (
} else { user.data?.is_moderator ||
user.data?.is_instructor ||
user.data?.is_evaluator
) {
tabs.push({ label: __('Upcoming') }) tabs.push({ label: __('Upcoming') })
tabs.push({ label: __('Archived') }) tabs.push({ label: __('Archived') })
tabs.push({ label: __('Unpublished') }) tabs.push({ label: __('Unpublished') })
} else if (user.data) {
tabs.push({ label: __('Enrolled') })
} }
return tabs return tabs
}) })

View File

@@ -245,12 +245,10 @@ const paymentLink = createResource({
}) })
const generatePaymentLink = () => { const generatePaymentLink = () => {
console.log('called')
paymentLink.submit( paymentLink.submit(
{}, {},
{ {
validate() { validate() {
console.log('validation start')
if (!billingDetails.source) { if (!billingDetails.source) {
return __('Please let us know where you heard about us from.') return __('Please let us know where you heard about us from.')
} }

View File

@@ -37,6 +37,7 @@
<script setup> <script setup>
import { computed, inject, onMounted, ref } from 'vue' import { computed, inject, onMounted, ref } from 'vue'
import { Breadcrumbs, call, createResource } from 'frappe-ui' import { Breadcrumbs, call, createResource } from 'frappe-ui'
import { useRouter } from 'vue-router'
import UpcomingEvaluations from '@/components/UpcomingEvaluations.vue' import UpcomingEvaluations from '@/components/UpcomingEvaluations.vue'
const courseTitle = ref(null) const courseTitle = ref(null)
@@ -44,6 +45,7 @@ const evaluator = ref(null)
const courses = ref([]) const courses = ref([])
const user = inject('$user') const user = inject('$user')
const dayjs = inject('$dayjs') const dayjs = inject('$dayjs')
const router = useRouter()
const props = defineProps({ const props = defineProps({
courseName: { courseName: {
@@ -53,6 +55,7 @@ const props = defineProps({
}) })
onMounted(() => { onMounted(() => {
fetchEnrollmentDetails()
fetchCourseDetails() fetchCourseDetails()
}) })
@@ -66,10 +69,26 @@ const certificate = createResource({
}, },
fieldname: ['name', 'template', 'issue_date'], fieldname: ['name', 'template', 'issue_date'],
}, },
auto: true,
cache: [user.data?.name, props.courseName], cache: [user.data?.name, props.courseName],
}) })
const fetchEnrollmentDetails = () => {
call('frappe.client.get_value', {
doctype: 'LMS Enrollment',
filters: { member: user.data?.name, course: props.courseName },
fieldname: ['purchased_certificate'],
}).then((data) => {
if (data.purchased_certificate) {
certificate.reload()
} else {
router.push({
name: 'CourseDetail',
params: { courseName: props.courseName },
})
}
})
}
const fetchCourseDetails = () => { const fetchCourseDetails = () => {
call('frappe.client.get_value', { call('frappe.client.get_value', {
doctype: 'LMS Course', doctype: 'LMS Course',

View File

@@ -56,12 +56,12 @@
<CourseInstructors :instructors="course.data.instructors" /> <CourseInstructors :instructors="course.data.instructors" />
</div> </div>
</div> </div>
<div class="flex mt-3 mb-4 w-fit"> <div v-if="course.data.tags" class="flex mt-4 w-fit">
<Badge <Badge
theme="gray" theme="gray"
size="lg" size="lg"
class="mr-2 text-ink-gray-9" class="mr-2 text-ink-gray-9"
v-for="tag in course.data.tags" v-for="tag in course.data.tags.split(', ')"
> >
{{ tag }} {{ tag }}
</Badge> </Badge>
@@ -69,7 +69,7 @@
<CourseCardOverlay :course="course" class="md:hidden mb-4" /> <CourseCardOverlay :course="course" class="md:hidden mb-4" />
<div <div
v-html="course.data.description" v-html="course.data.description"
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" 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 mt-4"
></div> ></div>
<div class="mt-10"> <div class="mt-10">
<CourseOutline <CourseOutline

View File

@@ -1,19 +1,20 @@
<template> <template>
<div class=""> <div class="h-full">
<div class="grid md:grid-cols-[70%,30%] h-full"> <div class="grid md:grid-cols-[70%,30%] h-full">
<div> <div>
<header <header
class="sticky top-0 z-10 flex flex-col md:flex-row md:items-center justify-between border-b bg-surface-white px-3 py-2.5 sm:px-5" class="sticky top-0 z-10 group flex flex-col md:flex-row md:items-center justify-between border-b bg-surface-white px-3 py-2.5 sm:px-5"
> >
<Breadcrumbs class="h-7" :items="breadcrumbs" /> <Breadcrumbs class="h-7" :items="breadcrumbs" />
<div class="flex items-center mt-3 md:mt-0"> <div class="flex items-center mt-3 md:mt-0">
<Button v-if="courseResource.data?.name" @click="trashCourse()"> <Button
<template #prefix> v-if="courseResource.data?.name"
@click="trashCourse()"
class="invisible group-hover:visible"
>
<template #icon>
<Trash2 class="w-4 h-4 stroke-1.5" /> <Trash2 class="w-4 h-4 stroke-1.5" />
</template> </template>
<span>
{{ __('Delete') }}
</span>
</Button> </Button>
<Button variant="solid" @click="submitCourse()" class="ml-2"> <Button variant="solid" @click="submitCourse()" class="ml-2">
<span> <span>
@@ -233,11 +234,11 @@
</div> </div>
</div> </div>
</div> </div>
<div class="border-l pt-5"> <div class="border-l">
<CourseOutline <CourseOutline
v-if="courseResource.data" v-if="courseResource.data"
:courseName="courseResource.data.name" :courseName="courseResource.data.name"
:title="course.title" :title="__('Course Outline')"
:allowEdit="true" :allowEdit="true"
/> />
</div> </div>
@@ -270,6 +271,7 @@ import { useRouter } from 'vue-router'
import CourseOutline from '@/components/CourseOutline.vue' import CourseOutline from '@/components/CourseOutline.vue'
import MultiSelect from '@/components/Controls/MultiSelect.vue' import MultiSelect from '@/components/Controls/MultiSelect.vue'
import { capture } from '@/telemetry' import { capture } from '@/telemetry'
import { useOnboarding } from 'frappe-ui/frappe'
import { useSettings } from '@/stores/settings' import { useSettings } from '@/stores/settings'
const user = inject('$user') const user = inject('$user')
@@ -278,6 +280,7 @@ const router = useRouter()
const instructors = ref([]) const instructors = ref([])
const settingsStore = useSettings() const settingsStore = useSettings()
const app = getCurrentInstance() const app = getCurrentInstance()
const { updateOnboardingStep } = useOnboarding('learning')
const { $dialog } = app.appContext.config.globalProperties const { $dialog } = app.appContext.config.globalProperties
const props = defineProps({ const props = defineProps({
@@ -443,9 +446,9 @@ const submitCourse = () => {
onSuccess(data) { onSuccess(data) {
capture('course_created') capture('course_created')
showToast('Success', 'Course created successfully', 'check') showToast('Success', 'Course created successfully', 'check')
/* if (!settingsStore.onboardingDetails.data?.is_onboarded) { updateOnboardingStep('create_first_course', true, false, () => {
settingsStore.onboardingDetails.reload() localStorage.setItem('firstCourse', data.name)
} */ })
router.push({ router.push({
name: 'CourseForm', name: 'CourseForm',
params: { courseName: data.name }, params: { courseName: data.name },

View File

@@ -1,316 +1,312 @@
<template> <template>
<div v-if="courses.data"> <header
<header class="sticky flex items-center justify-between top-0 z-10 border-b bg-surface-white px-3 py-2.5 sm:px-5"
class="sticky top-0 z-10 flex items-center justify-between border-b bg-surface-white px-3 py-2.5 sm:px-5" >
<Breadcrumbs :items="breadcrumbs" />
<router-link
v-if="user.data?.is_moderator"
:to="{
name: 'CourseForm',
params: { courseName: 'new' },
}"
> >
<Breadcrumbs <Button variant="solid">
class="h-7" <template #prefix>
:items="[{ label: __('Courses'), route: { name: 'Courses' } }]" <Plus class="h-4 w-4 stroke-1.5" />
/> </template>
<div class="flex space-x-2 justify-end"> {{ __('New') }}
<div class="w-40 md:w-44"> </Button>
<FormControl </router-link>
v-if="categories.data?.length" </header>
type="select" <div class="p-5 pb-10">
v-model="currentCategory" <div
:options="categories.data" class="flex flex-col lg:flex-row space-y-4 lg:space-y-0 lg:items-center justify-between mb-5"
:placeholder="__('Category')" >
/> <div class="text-lg text-ink-gray-9 font-semibold">
</div> {{ __('All Courses') }}
<div class="w-28 md:w-36"> </div>
<div
class="flex flex-col space-y-2 lg:space-y-0 lg:flex-row lg:items-center lg:space-x-4"
>
<TabButtons :buttons="courseTabs" v-model="currentTab" />
<FormControl
v-model="certification"
:label="__('Certification')"
type="checkbox"
@change="updateCourses()"
/>
<div class="grid grid-cols-2 gap-2">
<FormControl <FormControl
v-model="title"
:placeholder="__('Search by Title')"
type="text" type="text"
placeholder="Search" class="min-w-40 lg:min-w-0 lg:w-32 xl:w-40"
v-model="searchQuery" @input="updateCourses()"
@input="courses.reload()" />
> <div class="min-w-40 lg:min-w-0 lg:w-32 xl:w-40">
<template #prefix> <Select
<Search v-if="categories.length"
class="w-4 h-4 stroke-1.5 text-ink-gray-5" v-model="currentCategory"
name="search" :options="categories"
/> :placeholder="__('Category')"
</template> @change="updateCourses()"
</FormControl> />
</div>
<router-link
v-if="user.data?.is_moderator || user.data?.is_instructor"
:to="{
name: 'CourseForm',
params: {
courseName: 'new',
},
}"
>
<Button variant="solid">
<template #prefix>
<Plus class="h-4 w-4" />
</template>
{{ __('New') }}
</Button>
</router-link>
</div>
</header>
<div class="">
<Tabs
v-if="hasCourses"
as="div"
v-model="tabIndex"
tablistClass="overflow-x-visible flex-wrap !gap-3 md:flex-nowrap"
:tabs="makeTabs"
>
<template #tab="{ tab, selected }">
<div>
<button
class="group -mb-px flex items-center gap-2 overflow-hidden border-b border-transparent py-2.5 text-base text-ink-gray-5 duration-300 ease-in-out hover:border-outline-gray-3 hover:text-ink-gray-9"
:class="{ 'text-ink-gray-9': selected }"
>
<component v-if="tab.icon" :is="tab.icon" class="h-5" />
{{ __(tab.label) }}
<Badge theme="gray">
{{ tab.count }}
</Badge>
</button>
</div> </div>
</template>
<template #tab-panel="{ tab }">
<div
v-if="tab.courses && tab.courses.value.length"
class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 2xl:grid-cols-4 3xl:grid-cols-5 gap-7 my-5 mx-5"
>
<router-link
v-for="course in tab.courses.value"
:to="
course.membership && course.current_lesson
? {
name: 'Lesson',
params: {
courseName: course.name,
chapterNumber: course.current_lesson.split('-')[0],
lessonNumber: course.current_lesson.split('-')[1],
},
}
: course.membership
? {
name: 'Lesson',
params: {
courseName: course.name,
chapterNumber: 1,
lessonNumber: 1,
},
}
: {
name: 'CourseDetail',
params: { courseName: course.name },
}
"
>
<CourseCard :course="course" />
</router-link>
</div>
<div v-else class="p-5 italic text-ink-gray-4">
{{ __('No {0} courses').format(tab.label.toLowerCase()) }}
</div>
</template>
</Tabs>
<div
v-else-if="
!courses.loading &&
(user.data?.is_moderator || user.data?.is_instructor)
"
class="grid grid-cols-3 p-5"
>
<router-link
:to="{
name: 'CourseForm',
params: {
courseName: 'new',
},
}"
>
<div class="bg-surface-menu-bar py-32 px-5 rounded-md">
<div class="flex flex-col items-center text-center space-y-2">
<Plus
class="size-10 stroke-1 text-ink-gray-8 p-1 rounded-full border bg-surface-white"
/>
<div class="font-medium">
{{ __('Create a Course') }}
</div>
<span class="text-ink-gray-7 text-sm leading-4">
{{ __('You can add chapters and lessons to it.') }}
</span>
</div>
</div>
</router-link>
</div>
<div
v-else-if="!courses.loading && !hasCourses"
class="text-center p-5 text-ink-gray-5 mt-52 w-3/4 md:w-1/2 mx-auto space-y-2"
>
<BookOpen class="size-10 mx-auto stroke-1 text-ink-gray-4" />
<div class="text-xl font-medium">
{{ __('No courses found') }}
</div>
<div class="leading-5">
{{
__(
'There are no courses available at the moment. Keep an eye out, fresh learning experiences are on the way soon!'
)
}}
</div> </div>
</div> </div>
</div> </div>
<div
v-if="courses.data?.length"
class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 2xl:grid-cols-4 gap-5"
>
<router-link
v-for="course in courses.data"
:to="{ name: 'CourseDetail', params: { courseName: course.name } }"
>
<CourseCard :course="course" />
</router-link>
</div>
<div
v-else-if="!courses.list.loading"
class="flex flex-col items-center justify-center text-sm text-ink-gray-5 italic mt-48"
>
<BookOpen class="size-10 mx-auto stroke-1 text-ink-gray-4" />
<div class="text-lg font-medium mb-1">
{{ __('No courses found') }}
</div>
<div class="leading-5 w-2/5 text-center">
{{
__(
'There are no courses matching the criteria. Keep an eye out, fresh learning experiences are on the way soon!'
)
}}
</div>
</div>
<div
v-if="!courses.list.loading && courses.hasNextPage"
class="flex justify-center mt-5"
>
<Button @click="courses.next()">
{{ __('Load More') }}
</Button>
</div>
</div> </div>
</template> </template>
<script setup> <script setup>
import { import {
Badge,
Breadcrumbs, Breadcrumbs,
Button, Button,
call, createListResource,
createResource,
FormControl, FormControl,
Tabs, Select,
TabButtons,
} from 'frappe-ui' } from 'frappe-ui'
import CourseCard from '@/components/CourseCard.vue' import { computed, inject, onMounted, ref, watch } from 'vue'
import { BookOpen, Plus, Search } from 'lucide-vue-next' import { BookOpen, Plus } from 'lucide-vue-next'
import { ref, computed, inject, onMounted, watch } from 'vue'
import { updateDocumentTitle } from '@/utils' import { updateDocumentTitle } from '@/utils'
import { useRouter } from 'vue-router' import CourseCard from '@/components/CourseCard.vue'
import { useSettings } from '@/stores/settings'
const user = inject('$user') const user = inject('$user')
const searchQuery = ref('') const dayjs = inject('$dayjs')
const start = ref(0)
const pageLength = ref(30)
const categories = ref([])
const currentCategory = ref(null) const currentCategory = ref(null)
const hasCourses = ref(false) const title = ref('')
const router = useRouter() const certification = ref(false)
const settings = useSettings() const filters = ref({})
const currentTab = ref('Live')
onMounted(() => { onMounted(() => {
checkLearningPath() setFiltersFromQuery()
let queries = new URLSearchParams(location.search) updateCourses()
if (queries.has('category')) { categories.value = [
currentCategory.value = queries.get('category') {
} label: '',
value: null,
},
]
}) })
const checkLearningPath = () => { const setFiltersFromQuery = () => {
if ( let queries = new URLSearchParams(location.search)
settings.learningPaths.data && title.value = queries.get('title') || ''
(!user.data?.is_moderator || !user.data?.is_instructor) currentCategory.value = queries.get('category') || null
) { certification.value = queries.get('certification') || false
router.push({ name: 'Programs' }) }
const courses = createListResource({
doctype: 'LMS Course',
url: 'lms.lms.utils.get_courses',
cache: ['courses', user.data?.name],
pageLength: pageLength.value,
start: start.value,
onSuccess(data) {
let allCategories = data.map((course) => course.category)
allCategories = allCategories.filter(
(category, index) => allCategories.indexOf(category) === index && category
)
if (categories.value.length <= allCategories.length) {
updateCategories(data)
}
},
})
const updateCourses = () => {
updateFilters()
courses.update({
filters: filters.value,
})
courses.reload()
}
const updateFilters = () => {
updateCategoryFilter()
updateTitleFilter()
updateCertificationFilter()
updateTabFilter()
updateStudentFilter()
setQueryParams()
}
const updateCategoryFilter = () => {
if (currentCategory.value) {
filters.value['category'] = currentCategory.value
} else {
delete filters.value['category']
} }
} }
const courses = createResource({ const updateTitleFilter = () => {
url: 'lms.lms.utils.get_courses', if (title.value) {
cache: ['courses', user.data?.email], filters.value['title'] = ['like', `%${title.value}%`]
auto: true, } else {
delete filters.value['title']
}
}
const updateCertificationFilter = () => {
if (certification.value) {
filters.value['certification'] = 1
} else {
delete filters.value['certification']
}
}
const updateTabFilter = () => {
delete filters.value['live']
delete filters.value['created']
delete filters.value['published_on']
delete filters.value['upcoming']
if (currentTab.value == 'Enrolled' && user.data?.is_student) {
filters.value['enrolled'] = 1
delete filters.value['published']
} else {
delete filters.value['published']
delete filters.value['enrolled']
if (currentTab.value == 'Live') {
filters.value['published'] = 1
filters.value['upcoming'] = 0
filters.value['live'] = 1
} else if (currentTab.value == 'Upcoming') {
filters.value['upcoming'] = 1
filters.value['published'] = 1
} else if (currentTab.value == 'New') {
filters.value['published'] = 1
filters.value['published_on'] = [
'>=',
dayjs().add(-3, 'month').format('YYYY-MM-DD'),
]
} else if (currentTab.value == 'Created') {
filters.value['created'] = 1
}
}
}
const updateStudentFilter = () => {
if (!user.data || (user.data?.is_student && currentTab.value != 'Enrolled')) {
filters.value['published'] = 1
}
}
const setQueryParams = () => {
let queries = new URLSearchParams(location.search)
let filterKeys = {
title: title.value,
category: currentCategory.value,
certification: certification.value,
}
Object.keys(filterKeys).forEach((key) => {
if (filterKeys[key]) {
queries.set(key, filterKeys[key])
} else {
queries.delete(key)
}
})
let queryString = ''
if (queries.toString()) {
queryString = `?${queries.toString()}`
}
history.replaceState({}, '', `${location.pathname}${queryString}`)
}
const updateCategories = (data) => {
data.forEach((course) => {
if (
course.category &&
!categories.value.find((category) => category.value === course.category)
)
categories.value.push({
label: course.category,
value: course.category,
})
})
}
watch(currentTab, () => {
updateCourses()
}) })
const tabIndex = ref(0) const courseTabs = computed(() => {
let tabs let tabs = [
{
const makeTabs = computed(() => { label: __('Live'),
tabs = [] },
addToTabs('Live') {
addToTabs('New') label: __('New'),
addToTabs('Upcoming') },
{
if (user.data) { label: __('Upcoming'),
addToTabs('Enrolled') },
]
if ( if (
user.data.is_moderator || user.data?.is_moderator ||
user.data.is_instructor || user.data?.is_instructor ||
courses.data?.created?.length user.data?.is_evaluator
) { ) {
addToTabs('Created') tabs.push({ label: __('Created') })
} } else if (user.data) {
tabs.push({ label: __('Enrolled') })
if (user.data.is_moderator) {
addToTabs('Under Review')
}
} }
return tabs return tabs
}) })
const addToTabs = (label) => { const breadcrumbs = computed(() => [
let courses = getCourses(label.toLowerCase().split(' ').join('_')) {
tabs.push({ label: __('Courses'),
label, route: { name: 'Courses' },
courses: computed(() => courses),
count: computed(() => courses.length),
})
}
const getCourses = (type) => {
let courseList = courses.data[type]
if (searchQuery.value) {
let query = searchQuery.value.toLowerCase()
courseList = courseList.filter(
(course) =>
course.title.toLowerCase().includes(query) ||
course.short_introduction.toLowerCase().includes(query) ||
course.tags.filter((tag) => tag.toLowerCase().includes(query)).length
)
}
if (currentCategory.value && currentCategory.value != '') {
courseList = courseList.filter(
(course) => course.category == currentCategory.value
)
}
return courseList
}
const categories = createResource({
url: 'lms.lms.api.get_categories',
makeParams() {
return {
doctype: 'LMS Course',
filters: {
published: 1,
},
}
}, },
cache: ['courseCategories'], ])
auto: true,
transform(data) {
data.unshift({
label: '',
value: null,
})
},
})
watch(courses, () => {
if (courses.data) {
Object.keys(courses.data).forEach((section) => {
if (courses.data[section].length) {
hasCourses.value = true
}
})
}
})
watch(
() => currentCategory.value,
() => {
let queries = new URLSearchParams(location.search)
if (currentCategory.value) {
queries.set('category', currentCategory.value)
} else {
queries.delete('category')
}
history.pushState(null, '', `${location.pathname}?${queries.toString()}`)
}
)
const pageMeta = computed(() => { const pageMeta = computed(() => {
return { return {
title: 'Courses', title: 'Courses',
description: 'All Courses divided by categories', description: 'All published courses.',
} }
}) })

View File

@@ -587,11 +587,6 @@ updateDocumentTitle(pageMeta)
line-height: 1.7; line-height: 1.7;
} }
iframe {
border-top: 3px solid theme('colors.gray.700');
border-bottom: 3px solid theme('colors.gray.700');
}
.tc-table { .tc-table {
border-left: 1px solid #e8e8eb; border-left: 1px solid #e8e8eb;
} }

View File

@@ -92,13 +92,13 @@ import LessonHelp from '@/components/LessonHelp.vue'
import { ChevronRight } from 'lucide-vue-next' import { ChevronRight } from 'lucide-vue-next'
import { updateDocumentTitle, createToast, getEditorTools } from '@/utils' import { updateDocumentTitle, createToast, getEditorTools } from '@/utils'
import { capture } from '@/telemetry' import { capture } from '@/telemetry'
import { useSettings } from '@/stores/settings' import { useOnboarding } from 'frappe-ui/frappe'
const editor = ref(null) const editor = ref(null)
const instructorEditor = ref(null) const instructorEditor = ref(null)
const user = inject('$user') const user = inject('$user')
const openInstructorEditor = ref(false) const openInstructorEditor = ref(false)
const settingsStore = useSettings() const { updateOnboardingStep } = useOnboarding('learning')
let autoSaveInterval let autoSaveInterval
let showSuccessMessage = false let showSuccessMessage = false
@@ -395,10 +395,8 @@ const createNewLesson = () => {
{ {
onSuccess() { onSuccess() {
capture('lesson_created') capture('lesson_created')
updateOnboardingStep('create_first_lesson')
showToast('Success', 'Lesson created successfully', 'check') showToast('Success', 'Lesson created successfully', 'check')
/* if (!settingsStore.onboardingDetails.data?.is_onboarded) {
settingsStore.onboardingDetails.reload()
} */
lessonDetails.reload() lessonDetails.reload()
}, },
} }
@@ -623,4 +621,12 @@ iframe {
.tc-table { .tc-table {
border-left: 1px solid #e8e8eb; border-left: 1px solid #e8e8eb;
} }
.ce-toolbox__button[data-tool='markdown'] {
display: none !important;
}
.ce-popover-item[data-item-name='markdown'] {
display: none !important;
}
</style> </style>

View File

@@ -72,7 +72,7 @@ const roles = createResource({
}) })
const updateRole = createResource({ const updateRole = createResource({
url: 'lms.overrides.user.save_role', url: 'lms.lms.api.save_role',
makeParams(values) { makeParams(values) {
return { return {
user: props.profile.data?.name, user: props.profile.data?.name,

View File

@@ -17,7 +17,7 @@
<div v-if="programs.data?.length" class="pt-5 px-5"> <div v-if="programs.data?.length" class="pt-5 px-5">
<div v-for="program in programs.data" class="mb-10"> <div v-for="program in programs.data" class="mb-10">
<div class="flex items-center justify-between"> <div class="flex items-center justify-between">
<div class="text-xl font-semibold"> <div class="text-xl text-ink-gray-9 font-semibold">
{{ program.name }} {{ program.name }}
</div> </div>
<div class="flex items-center space-x-2"> <div class="flex items-center space-x-2">

View File

@@ -1,27 +1,36 @@
<template> <template>
<header <header
v-if="!fromLesson"
class="sticky top-0 z-10 flex items-center justify-between border-b bg-surface-white px-3 py-2.5 sm:px-5" class="sticky top-0 z-10 flex items-center justify-between border-b bg-surface-white px-3 py-2.5 sm:px-5"
> >
<Breadcrumbs :items="breadcrumbs" /> <Breadcrumbs :items="breadcrumbs" />
</header> </header>
<div class="md:w-7/12 md:mx-auto mx-4 py-10"> <div
class="md:w-7/12 md:mx-auto mx-4 py-10"
:class="{ 'pt-4 md:w-full': fromLesson }"
>
<Quiz :quizName="quizID" /> <Quiz :quizName="quizID" />
</div> </div>
</template> </template>
<script setup> <script setup>
import Quiz from '@/components/Quiz.vue' import Quiz from '@/components/Quiz.vue'
import { createResource, Breadcrumbs } from 'frappe-ui' import { createResource, Breadcrumbs } from 'frappe-ui'
import { computed, inject, onMounted } from 'vue' import { computed, inject, onMounted, ref } from 'vue'
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
import { updateDocumentTitle } from '@/utils' import { updateDocumentTitle } from '@/utils'
const user = inject('$user') const user = inject('$user')
const router = useRouter() const router = useRouter()
const fromLesson = ref(false)
onMounted(() => { onMounted(() => {
if (!user.data) { if (!user.data) {
router.push({ name: 'Courses' }) router.push({ name: 'Courses' })
} }
if (new URLSearchParams(window.location.search).get('fromLesson')) {
fromLesson.value = true
}
}) })
const props = defineProps({ const props = defineProps({

View File

@@ -1,10 +1,9 @@
import { Pencil } from 'lucide-vue-next' import { Pencil } from 'lucide-vue-next'
import { createApp, h } from 'vue' import { createApp, h } from 'vue'
import AssessmentPlugin from '@/components/AssessmentPlugin.vue' import AssessmentPlugin from '@/components/AssessmentPlugin.vue'
import AssignmentBlock from '@/components/AssignmentBlock.vue'
import translationPlugin from '../translation' import translationPlugin from '../translation'
import { usersStore } from '@/stores/user' import { usersStore } from '@/stores/user'
import router from '../router' import { call } from 'frappe-ui'
export class Assignment { export class Assignment {
constructor({ data, api, readOnly }) { constructor({ data, api, readOnly }) {
@@ -43,14 +42,18 @@ export class Assignment {
renderAssignment(assignment) { renderAssignment(assignment) {
if (this.readOnly) { if (this.readOnly) {
const app = createApp(AssignmentBlock, {
assignmentID: assignment,
})
app.use(translationPlugin)
app.use(router)
const { userResource } = usersStore() const { userResource } = usersStore()
app.provide('$user', userResource) call('frappe.client.get_value', {
app.mount(this.wrapper) doctype: 'LMS Assignment Submission',
filters: {
assignment: assignment,
member: userResource.data?.name,
},
fieldname: ['name'],
}).then((data) => {
let submission = data.name || 'new'
this.wrapper.innerHTML = `<iframe src="/lms/assignment-submission/${assignment}/${submission}?fromLesson=1" class="w-full h-[500px]"></iframe>`
})
return return
} }
this.wrapper.innerHTML = `<div class='border rounded-md p-10 text-center bg-surface-menu-bar mb-2'> this.wrapper.innerHTML = `<div class='border rounded-md p-10 text-center bg-surface-menu-bar mb-2'>

View File

@@ -1,5 +1,7 @@
import { Code } from "lucide-vue-next" import { Code } from "lucide-vue-next"
import { h, createApp } from "vue" import { h, createApp } from "vue"
import hljs from 'highlight.js/lib/core';
const DEFAULT_THEMES = ['light', 'dark']; const DEFAULT_THEMES = ['light', 'dark'];
const COMMON_LANGUAGES = { const COMMON_LANGUAGES = {
@@ -42,7 +44,6 @@ export class CodeBox {
this.selectInput = document.createElement('input'); this.selectInput = document.createElement('input');
this.selectDropIcon = document.createElement('i'); this.selectDropIcon = document.createElement('i');
this._injectHighlightJSScriptElement();
this._injectHighlightJSCSSElement(); this._injectHighlightJSCSSElement();
this.api.listeners.on(window, 'click', this._closeAllLanguageSelects, true); this.api.listeners.on(window, 'click', this._closeAllLanguageSelects, true);
@@ -150,7 +151,7 @@ export class CodeBox {
} }
_highlightCodeArea(event) { _highlightCodeArea(event) {
window.hljs.highlightBlock(this.codeArea); hljs.highlightBlock(this.codeArea);
} }
_handleCodeAreaPaste(event) { _handleCodeAreaPaste(event) {
@@ -167,7 +168,8 @@ export class CodeBox {
this.codeArea.removeAttribute('class'); this.codeArea.removeAttribute('class');
this.data.language = language[0]; this.data.language = language[0];
this.codeArea.setAttribute('class', `codeBoxTextArea ${this.config.useDefaultTheme} ${this.data.language}`); this.codeArea.setAttribute('class', `codeBoxTextArea ${this.config.useDefaultTheme} ${this.data.language}`);
window.hljs.highlightBlock(this.codeArea);
hljs.highlightElement(this.codeArea);
} }
_closeAllLanguageSelects() { _closeAllLanguageSelects() {
@@ -175,20 +177,6 @@ export class CodeBox {
for (let i = 0, len = selectPreviews.length; i < len; i++) selectPreviews[i].classList.remove('codeBoxShow'); for (let i = 0, len = selectPreviews.length; i < len; i++) selectPreviews[i].classList.remove('codeBoxShow');
} }
_injectHighlightJSScriptElement() {
const highlightJSScriptElement = document.querySelector(`#${this.highlightScriptID}`);
const highlightJSScriptURL = 'https://cdn.jsdelivr.net/gh/highlightjs/cdn-release@9.18.1/build/highlight.min.js';
if (!highlightJSScriptElement) {
const script = document.createElement('script');
const head = document.querySelector('head');
script.setAttribute('src', highlightJSScriptURL);
script.setAttribute('id', this.highlightScriptID);
if (head) head.appendChild(script);
}
else highlightJSScriptElement.setAttribute('src', highlightJSScriptURL);
}
_injectHighlightJSCSSElement() { _injectHighlightJSCSSElement() {
const highlightJSCSSElement = document.querySelector(`#${this.highlightCSSID}`); const highlightJSCSSElement = document.querySelector(`#${this.highlightCSSID}`);
let highlightJSCSSURL = this._getThemeURLFromConfig(); let highlightJSCSSURL = this._getThemeURLFromConfig();

View File

@@ -158,7 +158,10 @@ export function getEditorTools() {
quiz: Quiz, quiz: Quiz,
assignment: Assignment, assignment: Assignment,
upload: Upload, upload: Upload,
markdown: Markdown, markdown: {
class: Markdown,
inlineToolbar: true,
},
image: SimpleImage, image: SimpleImage,
table: { table: {
class: Table, class: Table,
@@ -174,9 +177,6 @@ export function getEditorTools() {
codeBox: { codeBox: {
class: CodeBox, class: CodeBox,
config: { config: {
themeURL:
'https://cdn.jsdelivr.net/gh/highlightjs/cdn-release@9.18.1/build/styles/atom-one-dark.min.css',
themeName: 'atom-one-dark',
useDefaultTheme: 'dark', useDefaultTheme: 'dark',
}, },
}, },
@@ -441,6 +441,22 @@ export function getTimezones() {
] ]
} }
export function getUserTimezone() {
try {
const timezone = Intl.DateTimeFormat().resolvedOptions().timeZone
const supportedTimezones = getTimezones()
if (supportedTimezones.includes(timezone)) {
return timezone // e.g., 'Asia/Calcutta', 'America/New_York', etc.
} else {
throw Error('unsupported timezone')
}
} catch (error) {
console.error('Error getting timezone:', error)
return null
}
}
export function getSidebarLinks() { export function getSidebarLinks() {
return [ return [
{ {

View File

@@ -1,3 +1,6 @@
import { CodeXml } from 'lucide-vue-next'
import { createApp, h } from 'vue'
export class Markdown { export class Markdown {
constructor({ data, api, readOnly, config }) { constructor({ data, api, readOnly, config }) {
this.api = api this.api = api
@@ -18,13 +21,26 @@ export class Markdown {
} }
} }
static get toolbox() {
const app = createApp({
render: () =>
h(CodeXml, { size: 18, strokeWidth: 1.5, color: 'black' }),
})
const div = document.createElement('div')
app.mount(div)
return {
title: '',
icon: div.innerHTML,
}
}
onPaste(event) { onPaste(event) {
const data = { const data = {
text: event.detail.data.innerHTML, text: event.detail.data.innerHTML,
} }
this.data = data this.data = data
window.requestAnimationFrame(() => { window.requestAnimationFrame(() => {
if (!this.wrapper) { if (!this.wrapper) {
return return
@@ -41,15 +57,14 @@ export class Markdown {
render() { render() {
this.wrapper = document.createElement('div') this.wrapper = document.createElement('div')
this.wrapper.classList.add('cdx-block') this.wrapper.classList.add('cdx-block', 'ce-paragraph')
this.wrapper.classList.add('ce-paragraph')
this.wrapper.innerHTML = this.text this.wrapper.innerHTML = this.text
if (!this.readOnly) { if (!this.readOnly) {
this.wrapper.contentEditable = true this.wrapper.contentEditable = true
this.wrapper.innerHTML = this.text this.wrapper.innerHTML = this.text
this.wrapper.addEventListener('keydown', (event) => { this.wrapper.addEventListener('input', (event) => {
let value = event.target.textContent let value = event.target.textContent
if (event.keyCode === 32 && value.startsWith('#')) { if (event.keyCode === 32 && value.startsWith('#')) {
this.convertToHeader(event, value) this.convertToHeader(event, value)
@@ -165,7 +180,7 @@ export class Markdown {
} }
canBeEmbed(line) { canBeEmbed(line) {
return /^https?:\/\/.+/.test(line) return /^https?:\/\/.+/.test(line.trim())
} }
} }

View File

@@ -43,14 +43,7 @@ export class Quiz {
renderQuiz(quiz) { renderQuiz(quiz) {
if (this.readOnly) { if (this.readOnly) {
const app = createApp(QuizBlock, { this.wrapper.innerHTML = `<iframe src="/lms/quiz/${quiz}?fromLesson=1" class="w-full h-[500px]"></iframe>`
quiz: quiz,
})
app.use(translationPlugin)
app.use(router)
const { userResource } = usersStore()
app.provide('$user', userResource)
app.mount(this.wrapper)
return return
} }
this.wrapper.innerHTML = `<div class='border rounded-md p-10 text-center bg-surface-menu-bar mb-2'> this.wrapper.innerHTML = `<div class='border rounded-md p-10 text-center bg-surface-menu-bar mb-2'>

View File

@@ -56,11 +56,11 @@ export class Upload {
app.mount(this.wrapper) app.mount(this.wrapper)
return return
} else if (file.file_type == 'PDF') { } else if (file.file_type == 'PDF') {
this.wrapper.innerHTML = `<iframe src="https://docs.google.com/viewer?url=${ this.wrapper.innerHTML = `<iframe src="${
window.location.origin window.location.origin
}${encodeURI( }${encodeURI(
file.file_url file.file_url
)}&embedded=true" width='100%' height='700px' class="mb-4" type="application/pdf"></iframe>` )}" width='100%' height='700px' class="mb-4" type="application/pdf"></iframe>`
return return
} else { } else {
this.wrapper.innerHTML = `<img class="mb-4" src=${encodeURI( this.wrapper.innerHTML = `<img class="mb-4" src=${encodeURI(

View File

@@ -3,8 +3,10 @@ module.exports = {
content: [ content: [
'./index.html', './index.html',
'./src/**/*.{vue,js,ts,jsx,tsx}', './src/**/*.{vue,js,ts,jsx,tsx}',
'./node_modules/frappe-ui/src/components/**/*.{vue,js,ts,jsx,tsx}', './node_modules/frappe-ui/src/**/*.{vue,js,ts,jsx,tsx}',
'../node_modules/frappe-ui/src/components/**/*.{vue,js,ts,jsx,tsx}', '../node_modules/frappe-ui/src/**/*.{vue,js,ts,jsx,tsx}',
'./node_modules/frappe-ui/frappe/**/*.{vue,js,ts,jsx,tsx}',
'../node_modules/frappe-ui/frappe/**/*.{vue,js,ts,jsx,tsx}',
], ],
theme: { theme: {
extend: { extend: {
@@ -12,7 +14,7 @@ module.exports = {
1.5: '1.5', 1.5: '1.5',
}, },
screens: { screens: {
'2xl': '1536px', '2xl': '1600px',
'3xl': '1920px', '3xl': '1920px',
}, },
}, },

View File

@@ -6,7 +6,17 @@ import frappeui from 'frappe-ui/vite'
// https://vitejs.dev/config/ // https://vitejs.dev/config/
export default defineConfig({ export default defineConfig({
plugins: [ plugins: [
frappeui(), frappeui({
frappeProxy: true,
lucideIcons: true,
jinjaBootData: true,
frappeTypes: {
input: {},
},
buildConfig: {
indexHtmlPath: '../lms/www/lms.html',
},
}),
vue({ vue({
script: { script: {
defineModel: true, defineModel: true,
@@ -15,7 +25,7 @@ export default defineConfig({
}), }),
], ],
server: { server: {
allowedHosts: ['fs', 'bs'], allowedHosts: ['fs', 'onb1'],
}, },
resolve: { resolve: {
alias: { alias: {
@@ -23,28 +33,13 @@ export default defineConfig({
'tailwind.config.js': path.resolve(__dirname, 'tailwind.config.js'), 'tailwind.config.js': path.resolve(__dirname, 'tailwind.config.js'),
}, },
}, },
build: {
outDir: `../lms/public/frontend`,
emptyOutDir: true,
commonjsOptions: {
include: [/tailwind.config.js/, /node_modules/],
},
sourcemap: true,
target: 'es2015',
rollupOptions: {
output: {
manualChunks: {
'frappe-ui': ['frappe-ui'],
},
},
},
},
optimizeDeps: { optimizeDeps: {
include: [ include: [
'feather-icons', 'feather-icons',
'showdown', 'showdown',
'engine.io-client', 'engine.io-client',
'tailwind.config.js', 'tailwind.config.js',
'highlight.js',
], ],
}, },
}) })

File diff suppressed because it is too large Load Diff

View File

@@ -1 +1 @@
__version__ = "2.25.0" __version__ = "2.26.0"

View File

@@ -88,7 +88,6 @@ setup_wizard_requires = "assets/lms/js/setup_wizard.js"
# Override standard doctype classes # Override standard doctype classes
override_doctype_class = { override_doctype_class = {
"User": "lms.overrides.user.CustomUser",
"Web Template": "lms.overrides.web_template.CustomWebTemplate", "Web Template": "lms.overrides.web_template.CustomWebTemplate",
} }
@@ -104,6 +103,10 @@ doc_events = {
}, },
"Discussion Reply": {"after_insert": "lms.lms.utils.handle_notifications"}, "Discussion Reply": {"after_insert": "lms.lms.utils.handle_notifications"},
"Notification Log": {"on_change": "lms.lms.utils.publish_notifications"}, "Notification Log": {"on_change": "lms.lms.utils.publish_notifications"},
"User": {
"validate": "lms.lms.user.validate_username_duplicates",
"after_insert": "lms.lms.user.after_insert",
},
} }
# Scheduled Tasks # Scheduled Tasks
@@ -112,6 +115,7 @@ scheduler_events = {
"hourly": [ "hourly": [
"lms.lms.doctype.lms_certificate_request.lms_certificate_request.schedule_evals", "lms.lms.doctype.lms_certificate_request.lms_certificate_request.schedule_evals",
"lms.lms.api.update_course_statistics", "lms.lms.api.update_course_statistics",
"lms.lms.doctype.lms_certificate_request.lms_certificate_request.mark_eval_as_completed",
], ],
"daily": [ "daily": [
"lms.job.doctype.job_opportunity.job_opportunity.update_job_openings", "lms.job.doctype.job_opportunity.job_opportunity.update_job_openings",
@@ -190,8 +194,8 @@ jinja = {
"lms.lms.utils.get_lesson_index", "lms.lms.utils.get_lesson_index",
"lms.lms.utils.get_lesson_url", "lms.lms.utils.get_lesson_url",
"lms.page_renderers.get_profile_url", "lms.page_renderers.get_profile_url",
"lms.overrides.user.get_palette",
"lms.lms.utils.is_instructor", "lms.lms.utils.is_instructor",
"lms.lms.utils.get_palette",
], ],
"filters": [], "filters": [],
} }
@@ -229,7 +233,6 @@ lms_markdown_macro_renderers = {
page_renderer = [ page_renderer = [
"lms.page_renderers.ProfileRedirectPage", "lms.page_renderers.ProfileRedirectPage",
"lms.page_renderers.ProfilePage", "lms.page_renderers.ProfilePage",
"lms.page_renderers.CoursePage",
"lms.page_renderers.SCORMRenderer", "lms.page_renderers.SCORMRenderer",
] ]
@@ -238,7 +241,7 @@ profile_url_prefix = "/users/"
signup_form_template = "lms.plugins.show_custom_signup" signup_form_template = "lms.plugins.show_custom_signup"
on_session_creation = "lms.overrides.user.on_session_creation" on_login = "lms.lms.user.on_login"
add_to_apps_screen = [ add_to_apps_screen = [
{ {

View File

@@ -4,7 +4,6 @@ from lms.lms.api import give_dicussions_permission
def after_install(): def after_install():
add_pages_to_nav()
create_batch_source() create_batch_source()
give_dicussions_permission() give_dicussions_permission()
@@ -15,37 +14,6 @@ def after_sync():
add_all_roles_to("Administrator") add_all_roles_to("Administrator")
def add_pages_to_nav():
pages = [
{"label": "Explore", "idx": 1},
{"label": "Courses", "url": "/lms/courses", "parent": "Explore", "idx": 2},
{"label": "Batches", "url": "/lms/batches", "parent": "Explore", "idx": 3},
{"label": "Statistics", "url": "/lms/statistics", "parent": "Explore", "idx": 4},
{"label": "Jobs", "url": "/lms/job-openings", "parent": "Explore", "idx": 5},
]
for page in pages:
filters = frappe._dict()
if page.get("url"):
filters["url"] = ["like", "%" + page.get("url") + "%"]
else:
filters["label"] = page.get("label")
if not frappe.db.exists("Top Bar Item", filters):
frappe.get_doc(
{
"doctype": "Top Bar Item",
"label": page.get("label"),
"url": page.get("url"),
"parent_label": page.get("parent"),
"idx": page.get("idx"),
"parent": "Website Settings",
"parenttype": "Website Settings",
"parentfield": "top_bar_items",
}
).save()
def before_uninstall(): def before_uninstall():
delete_custom_fields() delete_custom_fields()
delete_lms_roles() delete_lms_roles()

View File

@@ -12,7 +12,6 @@ from frappe.translate import get_all_translations
from frappe import _ from frappe import _
from frappe.utils import ( from frappe.utils import (
get_datetime, get_datetime,
getdate,
cint, cint,
flt, flt,
now, now,
@@ -178,7 +177,9 @@ def get_user_info():
user.is_instructor = "Course Creator" in user.roles user.is_instructor = "Course Creator" in user.roles
user.is_moderator = "Moderator" in user.roles user.is_moderator = "Moderator" in user.roles
user.is_evaluator = "Batch Evaluator" in user.roles user.is_evaluator = "Batch Evaluator" in user.roles
user.is_student = "LMS Student" in user.roles user.is_student = (
not user.is_instructor and not user.is_moderator and not user.is_evaluator
)
user.is_fc_site = is_fc_site() user.is_fc_site = is_fc_site()
user.is_system_manager = "System Manager" in user.roles user.is_system_manager = "System Manager" in user.roles
if user.is_fc_site and user.is_system_manager: if user.is_fc_site and user.is_system_manager:
@@ -230,6 +231,12 @@ def validate_billing_access(billing_type, name):
access = False access = False
message = _("You are already enrolled for this batch.") message = _("You are already enrolled for this batch.")
seat_count = frappe.get_cached_value("LMS Batch", name, "seat_count")
number_of_students = frappe.db.count("LMS Batch Enrollment", {"batch": name})
if seat_count <= number_of_students:
access = False
message = _("Batch is sold out.")
elif access and billing_type == "certificate": elif access and billing_type == "certificate":
purchased_certificate = frappe.db.exists( purchased_certificate = frappe.db.exists(
"LMS Enrollment", "LMS Enrollment",
@@ -1296,10 +1303,60 @@ def get_certification_details(course):
membership = frappe.db.get_value( membership = frappe.db.get_value(
"LMS Enrollment", "LMS Enrollment",
filters, filters,
["name", "certificate", "purchased_certificate"], ["name", "purchased_certificate"],
as_dict=1, as_dict=1,
) )
paid_certificate = frappe.db.get_value("LMS Course", course, "paid_certificate") paid_certificate = frappe.db.get_value("LMS Course", course, "paid_certificate")
certificate = frappe.db.get_value(
"LMS Certificate",
{"member": frappe.session.user, "course": course},
["name", "template"],
as_dict=1,
)
return {"membership": membership, "paid_certificate": paid_certificate} return {
"membership": membership,
"paid_certificate": paid_certificate,
"certificate": certificate,
}
@frappe.whitelist()
def save_role(user, role, value):
frappe.only_for("Moderator")
if cint(value):
doc = frappe.get_doc(
{
"doctype": "Has Role",
"parent": user,
"role": role,
"parenttype": "User",
"parentfield": "roles",
}
)
doc.save(ignore_permissions=True)
else:
frappe.db.delete("Has Role", {"parent": user, "role": role})
return True
@frappe.whitelist()
def add_an_evaluator(email):
if not frappe.db.exists("User", email):
user = frappe.new_doc("User")
user.update(
{
"email": email,
"first_name": email.split("@")[0].capitalize(),
"enabled": 1,
}
)
user.insert()
user.add_roles("Batch Evaluator")
evaluator = frappe.new_doc("Course Evaluator")
evaluator.evaluator = email
evaluator.insert()
return evaluator

View File

@@ -8,6 +8,11 @@
"engine": "InnoDB", "engine": "InnoDB",
"field_order": [ "field_order": [
"evaluator", "evaluator",
"full_name",
"column_break_casg",
"user_image",
"username",
"section_break_ljse",
"schedule", "schedule",
"unavailability_section", "unavailability_section",
"unavailable_from", "unavailable_from",
@@ -18,8 +23,10 @@
{ {
"fieldname": "evaluator", "fieldname": "evaluator",
"fieldtype": "Link", "fieldtype": "Link",
"in_list_view": 1,
"label": "Evaluator", "label": "Evaluator",
"options": "User", "options": "User",
"reqd": 1,
"unique": 1 "unique": 1
}, },
{ {
@@ -46,11 +53,40 @@
"fieldname": "unavailable_to", "fieldname": "unavailable_to",
"fieldtype": "Date", "fieldtype": "Date",
"label": "To" "label": "To"
},
{
"fetch_from": "evaluator.full_name",
"fieldname": "full_name",
"fieldtype": "Data",
"label": "Full Name",
"read_only": 1
},
{
"fieldname": "column_break_casg",
"fieldtype": "Column Break"
},
{
"fieldname": "section_break_ljse",
"fieldtype": "Section Break"
},
{
"fetch_from": "evaluator.user_image",
"fieldname": "user_image",
"fieldtype": "Attach Image",
"label": "User Image",
"read_only": 1
},
{
"fetch_from": "evaluator.username",
"fieldname": "username",
"fieldtype": "Data",
"label": "Username",
"read_only": 1
} }
], ],
"index_web_pages_for_search": 1, "index_web_pages_for_search": 1,
"links": [], "links": [],
"modified": "2025-02-24 12:17:08.436659", "modified": "2025-03-26 14:02:46.588721",
"modified_by": "Administrator", "modified_by": "Administrator",
"module": "LMS", "module": "LMS",
"name": "Course Evaluator", "name": "Course Evaluator",
@@ -94,7 +130,8 @@
"write": 1 "write": 1
} }
], ],
"row_format": "Dynamic",
"sort_field": "modified", "sort_field": "modified",
"sort_order": "DESC", "sort_order": "DESC",
"states": [] "states": []
} }

View File

@@ -11,9 +11,15 @@ from frappe.utils import get_time, getdate
class CourseEvaluator(Document): class CourseEvaluator(Document):
def validate(self): def validate(self):
self.validate_evaluator_role()
self.validate_time_slots() self.validate_time_slots()
self.validate_unavailability() self.validate_unavailability()
def validate_evaluator_role(self):
roles = frappe.get_roles(self.evaluator)
if "Batch Evaluator" not in roles:
frappe.get_doc("User", self.evaluator).add_roles("Batch Evaluator")
def validate_unavailability(self): def validate_unavailability(self):
if ( if (
self.unavailable_from self.unavailable_from

View File

@@ -408,14 +408,14 @@ def send_batch_start_reminder():
for batch in batches: for batch in batches:
students = frappe.get_all( students = frappe.get_all(
"LMS Batch Enrollment", {"batch": batch}, ["member", "member_name"] "LMS Batch Enrollment", {"batch": batch.name}, ["member", "member_name"]
) )
for student in students: for student in students:
send_mail(batch, student) send_mail(batch, student)
def send_mail(batch, student): def send_mail(batch, student):
subject = _("Batch Start Reminder") subject = _("Your batch {0} is starting tomorrow").format(batch.title)
template = "batch_start_reminder" template = "batch_start_reminder"
args = { args = {

View File

@@ -21,7 +21,7 @@
], ],
"index_web_pages_for_search": 1, "index_web_pages_for_search": 1,
"links": [], "links": [],
"modified": "2024-09-23 19:33:49.593950", "modified": "2025-03-19 12:12:23.723432",
"modified_by": "Administrator", "modified_by": "Administrator",
"module": "LMS", "module": "LMS",
"name": "LMS Category", "name": "LMS Category",
@@ -51,6 +51,26 @@
"role": "Moderator", "role": "Moderator",
"share": 1, "share": 1,
"write": 1 "write": 1
},
{
"email": 1,
"export": 1,
"print": 1,
"read": 1,
"report": 1,
"role": "Course Creator",
"select": 1,
"share": 1
},
{
"email": 1,
"export": 1,
"print": 1,
"read": 1,
"report": 1,
"role": "Batch Evaluator",
"select": 1,
"share": 1
} }
], ],
"sort_field": "modified", "sort_field": "modified",

View File

@@ -4,7 +4,7 @@
frappe.ui.form.on("LMS Certificate Evaluation", { frappe.ui.form.on("LMS Certificate Evaluation", {
refresh: function (frm) { refresh: function (frm) {
if (!frm.is_new() && frm.doc.status == "Pass") { if (!frm.is_new() && frm.doc.status == "Pass") {
frm.add_custom_button(__("Create LMS Certificate"), () => { frm.add_custom_button(__("Create Certificate"), () => {
frappe.model.open_mapped_doc({ frappe.model.open_mapped_doc({
method: "lms.lms.doctype.lms_certificate_evaluation.lms_certificate_evaluation.create_lms_certificate", method: "lms.lms.doctype.lms_certificate_evaluation.lms_certificate_evaluation.create_lms_certificate",
frm: frm, frm: frm,

View File

@@ -3,18 +3,15 @@
frappe.ui.form.on("LMS Certificate Request", { frappe.ui.form.on("LMS Certificate Request", {
refresh: function (frm) { refresh: function (frm) {
if (!frm.is_new()) { if (!frm.is_new() && frm.doc.status == "Upcoming") {
frm.add_custom_button( frm.add_custom_button(__("Conduct Evaluation"), () => {
__("Create LMS Certificate Evaluation"), frappe.model.open_mapped_doc({
() => { method: "lms.lms.doctype.lms_certificate_request.lms_certificate_request.create_lms_certificate_evaluation",
frappe.model.open_mapped_doc({ frm: frm,
method: "lms.lms.doctype.lms_certificate_request.lms_certificate_request.create_lms_certificate_evaluation", });
frm: frm, });
});
}
);
} }
if (!frm.doc.google_meet_link) { if (!frm.doc.google_meet_link && frm.doc.status == "Upcoming") {
frm.add_custom_button(__("Generate Google Meet Link"), () => { frm.add_custom_button(__("Generate Google Meet Link"), () => {
frappe.call({ frappe.call({
method: "lms.lms.doctype.lms_certificate_request.lms_certificate_request.setup_calendar_event", method: "lms.lms.doctype.lms_certificate_request.lms_certificate_request.setup_calendar_event",

View File

@@ -77,6 +77,7 @@ class LMSCertificateRequest(Document):
"member": self.member, "member": self.member,
"course": self.course, "course": self.course,
"name": ["!=", self.name], "name": ["!=", self.name],
"status": "Upcoming",
}, },
["date", "start_time", "course"], ["date", "start_time", "course"],
) )
@@ -150,7 +151,11 @@ def schedule_evals():
timelapse = add_to_date(get_datetime(), hours=-5) timelapse = add_to_date(get_datetime(), hours=-5)
evals = frappe.get_all( evals = frappe.get_all(
"LMS Certificate Request", "LMS Certificate Request",
{"creation": [">=", timelapse], "google_meet_link": ["is", "not set"]}, {
"creation": [">=", timelapse],
"google_meet_link": ["is", "not set"],
"status": "Upcoming",
},
["name", "member", "member_name", "evaluator", "date", "start_time", "end_time"], ["name", "member", "member_name", "evaluator", "date", "start_time", "end_time"],
) )
for eval in evals: for eval in evals:
@@ -254,3 +259,20 @@ def create_lms_certificate_evaluation(source_name, target_doc=None):
target_doc, target_doc,
) )
return doc return doc
def mark_eval_as_completed():
requests = frappe.get_all(
"LMS Certificate Request",
{
"status": "Upcoming",
"date": ["<=", getdate()],
},
["name", "end_time", "date"],
)
for req in requests:
if req.date < getdate():
frappe.db.set_value("LMS Certificate Request", req.name, "status", "Completed")
elif req.date == getdate() and get_time(req.end_time) < get_time(nowtime()):
frappe.db.set_value("LMS Certificate Request", req.name, "status", "Completed")

View File

@@ -242,14 +242,14 @@
{ {
"default": "0", "default": "0",
"fieldname": "enrollments", "fieldname": "enrollments",
"fieldtype": "Data", "fieldtype": "Int",
"label": "Enrollments", "label": "Enrollments",
"read_only": 1 "read_only": 1
}, },
{ {
"default": "0", "default": "0",
"fieldname": "lessons", "fieldname": "lessons",
"fieldtype": "Data", "fieldtype": "Int",
"label": "Lessons", "label": "Lessons",
"read_only": 1 "read_only": 1
}, },
@@ -277,28 +277,20 @@
"is_published_field": "published", "is_published_field": "published",
"links": [ "links": [
{ {
"group": "Chapters", "link_doctype": "LMS Enrollment",
"link_fieldname": "course"
},
{
"link_doctype": "Course Chapter", "link_doctype": "Course Chapter",
"link_fieldname": "course" "link_fieldname": "course"
}, },
{ {
"group": "Batches", "link_doctype": "Course Lesson",
"link_doctype": "LMS Batch Old",
"link_fieldname": "course"
},
{
"group": "Mentors",
"link_doctype": "LMS Course Mentor Mapping",
"link_fieldname": "course"
},
{
"group": "Interests",
"link_doctype": "LMS Course Interest",
"link_fieldname": "course" "link_fieldname": "course"
} }
], ],
"make_attachments_public": 1, "make_attachments_public": 1,
"modified": "2025-02-24 11:50:58.325804", "modified": "2025-03-13 16:01:19.105212",
"modified_by": "Administrator", "modified_by": "Administrator",
"module": "LMS", "module": "LMS",
"name": "LMS Course", "name": "LMS Course",

View File

@@ -84,7 +84,7 @@ def send_live_class_reminder():
def send_mail(live_class, student): def send_mail(live_class, student):
subject = f"Your class on {live_class.title} is tomorrow" subject = _("Your class on {0} is today").format(live_class.title)
template = "live_class_reminder" template = "live_class_reminder"
args = { args = {

View File

@@ -44,13 +44,15 @@
"fieldtype": "Currency", "fieldtype": "Currency",
"in_list_view": 1, "in_list_view": 1,
"label": "Amount", "label": "Amount",
"options": "currency" "options": "currency",
"reqd": 1
}, },
{ {
"fieldname": "currency", "fieldname": "currency",
"fieldtype": "Link", "fieldtype": "Link",
"label": "Currency", "label": "Currency",
"options": "Currency" "options": "Currency",
"reqd": 1
}, },
{ {
"fieldname": "column_break_rqkd", "fieldname": "column_break_rqkd",
@@ -70,7 +72,8 @@
"fieldname": "address", "fieldname": "address",
"fieldtype": "Link", "fieldtype": "Link",
"label": "Address", "label": "Address",
"options": "Address" "options": "Address",
"reqd": 1
}, },
{ {
"default": "0", "default": "0",
@@ -124,13 +127,15 @@
"fieldname": "payment_for_document_type", "fieldname": "payment_for_document_type",
"fieldtype": "Select", "fieldtype": "Select",
"label": "Payment for Document Type", "label": "Payment for Document Type",
"options": "\nLMS Course\nLMS Batch" "options": "\nLMS Course\nLMS Batch",
"reqd": 1
}, },
{ {
"fieldname": "payment_for_document", "fieldname": "payment_for_document",
"fieldtype": "Dynamic Link", "fieldtype": "Dynamic Link",
"label": "Payment for Document", "label": "Payment for Document",
"options": "payment_for_document_type" "options": "payment_for_document_type",
"reqd": 1
}, },
{ {
"fieldname": "source", "fieldname": "source",
@@ -156,7 +161,7 @@
"link_fieldname": "payment" "link_fieldname": "payment"
} }
], ],
"modified": "2025-02-21 18:29:55.436611", "modified": "2025-03-13 15:31:38.019002",
"modified_by": "Administrator", "modified_by": "Administrator",
"module": "LMS", "module": "LMS",
"name": "LMS Payment", "name": "LMS Payment",

View File

@@ -33,9 +33,42 @@ def send_payment_reminder():
) )
for payment in incomplete_payments: for payment in incomplete_payments:
if has_paid_later(payment):
continue
if is_batch_sold_out(payment):
continue
send_mail(payment) send_mail(payment)
def has_paid_later(payment):
return frappe.db.exists(
"LMS Payment",
{
"member": payment.member,
"payment_received": 1,
"payment_for_document": payment.payment_for_document,
"payment_for_document_type": payment.payment_for_document_type,
},
)
def is_batch_sold_out(payment):
if payment.payment_for_document_type == "LMS Batch":
seat_count = frappe.get_cached_value(
"LMS Batch", payment.payment_for_document, "seat_count"
)
number_of_students = frappe.db.count(
"LMS Batch Enrollment", {"batch": payment.payment_for_document}
)
if seat_count <= number_of_students:
return True
return False
def send_mail(payment): def send_mail(payment):
subject = _("Complete Your Enrollment - Don't miss out!") subject = _("Complete Your Enrollment - Don't miss out!")
template = "payment_reminder" template = "payment_reminder"

21
lms/lms/onboarding.py Normal file
View File

@@ -0,0 +1,21 @@
import frappe
def get_first_course():
course = frappe.get_all(
"LMS Course",
fields=["name"],
order_by="creation",
limit=1,
)
return course[0].name if course else None
def get_first_batch():
batch = frappe.get_all(
"LMS Batch",
fields=["name"],
order_by="creation",
limit=1,
)
return batch[0].name if batch else None

91
lms/lms/user.py Normal file
View File

@@ -0,0 +1,91 @@
import frappe
from frappe import _
from frappe.model.naming import append_number_if_name_exists
from frappe.website.utils import cleanup_page_name
from frappe.website.utils import is_signup_disabled
from frappe.utils import random_string, escape_html
from lms.lms.utils import get_country_code
def validate_username_duplicates(doc, method):
while not doc.username or doc.username_exists():
doc.username = append_number_if_name_exists(
doc.doctype, cleanup_page_name(doc.full_name), fieldname="username"
)
if " " in doc.username:
doc.username = doc.username.replace(" ", "")
if len(doc.username) < 4:
doc.username = doc.email.replace("@", "").replace(".", "")
def after_insert(doc, method):
doc.add_roles("LMS Student")
@frappe.whitelist(allow_guest=True)
def sign_up(email, full_name, verify_terms, user_category):
if is_signup_disabled():
frappe.throw(_("Sign Up is disabled"), _("Not Allowed"))
user = frappe.db.get("User", {"email": email})
if user:
if user.enabled:
return 0, _("Already Registered")
else:
return 0, _("Registered but disabled")
else:
if frappe.db.get_creation_count("User", 60) > 300:
frappe.respond_as_web_page(
_("Temporarily Disabled"),
_(
"Too many users signed up recently, so the registration is disabled. Please try back in an hour"
),
http_status_code=429,
)
user = frappe.get_doc(
{
"doctype": "User",
"email": email,
"first_name": escape_html(full_name),
"verify_terms": verify_terms,
"user_category": user_category,
"country": "",
"enabled": 1,
"new_password": random_string(10),
"user_type": "Website User",
}
)
user.flags.ignore_permissions = True
user.flags.ignore_password_policy = True
user.insert()
# set default signup role as per Portal Settings
default_role = frappe.db.get_single_value("Portal Settings", "default_role")
if default_role:
user.add_roles(default_role)
user.add_roles("LMS Student")
set_country_from_ip(None, user.name)
if user.flags.email_sent:
return 1, _("Please check your email for verification")
else:
return 2, _("Please ask your administrator to verify your sign-up")
def set_country_from_ip(login_manager=None, user=None):
if not user and login_manager:
user = login_manager.user
user_country = frappe.db.get_value("User", user, "country")
# if user_country:
# return
frappe.db.set_value("User", user, "country", get_country_code())
return
def on_login(login_manager):
default_app = frappe.db.get_single_value("System Settings", "default_app")
if default_app == "lms":
frappe.local.response["home_page"] = "/lms"

View File

@@ -1,6 +1,7 @@
import re import re
import string import string
import frappe import frappe
import hashlib
import json import json
import razorpay import razorpay
import requests import requests
@@ -532,10 +533,11 @@ def has_course_evaluator_role(member=None):
def has_student_role(member=None): def has_student_role(member=None):
return frappe.db.get_value( roles = frappe.get_roles(member or frappe.session.user)
"Has Role", return (
{"parent": member or frappe.session.user, "role": "LMS Student"}, "Moderator" not in roles
"name", and "Course Creator" not in roles
and "Batch Evaluator" not in roles
) )
@@ -984,17 +986,145 @@ def change_currency(amount, currency, country=None):
@frappe.whitelist(allow_guest=True) @frappe.whitelist(allow_guest=True)
def get_courses(): def get_courses(filters=None, start=0, page_length=20):
"""Returns the list of courses.""" """Returns the list of courses."""
courses = []
course_list = frappe.get_all("LMS Course", pluck="name")
for course in course_list:
courses.append(get_course_details(course))
courses = get_categorized_courses(courses) if not filters:
filters = {}
filters, or_filters, show_featured = update_course_filters(filters)
fields = get_course_fields()
courses = frappe.get_all(
"LMS Course",
filters=filters,
fields=fields,
or_filters=or_filters,
order_by="enrollments desc",
start=start,
page_length=page_length,
)
if show_featured:
courses = get_featured_courses(filters, or_filters, fields) + courses
courses = get_enrollment_details(courses)
courses = get_course_card_details(courses)
return courses return courses
def get_course_card_details(courses):
for course in courses:
course.instructors = get_instructors(course.name)
if course.paid_course and course.published == 1:
course.amount, course.currency = check_multicurrency(
course.course_price, course.currency, None, course.amount_usd
)
course.price = fmt_money(course.amount, 0, course.currency)
return courses
def get_course_or_filters(filters):
or_filters = {}
or_filters.update({"title": filters.get("title")})
or_filters.update({"short_introduction": filters.get("title")})
or_filters.update({"description": filters.get("title")})
or_filters.update({"tags": filters.get("title")})
return or_filters
def update_course_filters(filters):
or_filters = {}
show_featured = False
if filters.get("title"):
or_filters = get_course_or_filters(filters)
del filters["title"]
if filters.get("enrolled"):
enrolled_courses = frappe.get_all(
"LMS Enrollment", {"member": frappe.session.user}, pluck="course"
)
filters.update({"name": ["in", enrolled_courses]})
del filters["enrolled"]
if filters.get("created"):
created_courses = frappe.get_all(
"Course Instructor", {"instructor": frappe.session.user}, pluck="parent"
)
filters.update({"name": ["in", created_courses]})
del filters["created"]
if filters.get("live"):
filters.update({"featured": 0})
show_featured = True
del filters["live"]
if filters.get("certification"):
or_filters.update({"enable_certification": 1})
or_filters.update({"paid_certificate": 1})
del filters["certification"]
return filters, or_filters, show_featured
def get_enrollment_details(courses):
for course in courses:
filters = {
"course": course.name,
"member": frappe.session.user,
}
if frappe.db.exists("LMS Enrollment", filters):
course.membership = frappe.db.get_value(
"LMS Enrollment",
filters,
["name", "course", "current_lesson", "progress", "member"],
as_dict=1,
)
return courses
def get_featured_courses(filters, or_filters, fields):
filters.update({"featured": 1})
featured_courses = frappe.get_all(
"LMS Course",
filters=filters,
fields=fields,
or_filters=or_filters,
order_by="enrollments desc",
)
return featured_courses
def get_course_fields():
return [
"name",
"title",
"tags",
"image",
"short_introduction",
"published",
"upcoming",
"featured",
"disable_self_learning",
"published_on",
"category",
"status",
"paid_course",
"paid_certificate",
"course_price",
"currency",
"amount_usd",
"enable_certification",
"lessons",
"enrollments",
"rating",
]
@frappe.whitelist(allow_guest=True) @frappe.whitelist(allow_guest=True)
def get_course_details(course): def get_course_details(course):
course_details = frappe.db.get_value( course_details = frappe.db.get_value(
@@ -1027,7 +1157,6 @@ def get_course_details(course):
], ],
as_dict=1, as_dict=1,
) )
course_details.tags = course_details.tags.split(",") if course_details.tags else []
course_details.instructors = get_instructors(course_details.name) course_details.instructors = get_instructors(course_details.name)
# course_details.is_instructor = is_instructor(course_details.name) # course_details.is_instructor = is_instructor(course_details.name)
@@ -1760,16 +1889,16 @@ def update_payment_record(doctype, docname):
try: try:
if payment_for_certificate: if payment_for_certificate:
update_certificate_purchase(docname) update_certificate_purchase(docname, data.payment)
elif doctype == "LMS Course": elif doctype == "LMS Course":
enroll_in_course(data.payment, docname) enroll_in_course(docname, data.payment)
else: else:
enroll_in_batch(docname, data.payment) enroll_in_batch(docname, data.payment)
except Exception as e: except Exception as e:
frappe.log_error(frappe.get_traceback(), _("Enrollment Failed")) frappe.log_error(frappe.get_traceback(), _("Enrollment Failed"))
def enroll_in_course(payment_name, course): def enroll_in_course(course, payment_name):
if not frappe.db.exists( if not frappe.db.exists(
"LMS Enrollment", {"member": frappe.session.user, "course": course} "LMS Enrollment", {"member": frappe.session.user, "course": course}
): ):
@@ -1821,12 +1950,14 @@ def enroll_in_batch(batch, payment_name=None):
new_student.save() new_student.save()
def update_certificate_purchase(course): def update_certificate_purchase(course, payment_name):
frappe.db.set_value( frappe.db.set_value(
"LMS Enrollment", "LMS Enrollment",
{"member": frappe.session.user, "course": course}, {"member": frappe.session.user, "course": course},
"purchased_certificate", {
1, "purchased_certificate": 1,
"payment": payment_name,
},
) )
@@ -2009,3 +2140,25 @@ def get_batch_card_details(batches):
batch.price = fmt_money(batch.amount, 0, batch.currency) batch.price = fmt_money(batch.amount, 0, batch.currency)
return batches return batches
def get_palette(full_name):
"""
Returns a color unique to each member for Avatar"""
palette = [
["--orange-avatar-bg", "--orange-avatar-color"],
["--pink-avatar-bg", "--pink-avatar-color"],
["--blue-avatar-bg", "--blue-avatar-color"],
["--green-avatar-bg", "--green-avatar-color"],
["--dark-green-avatar-bg", "--dark-green-avatar-color"],
["--red-avatar-bg", "--red-avatar-color"],
["--yellow-avatar-bg", "--yellow-avatar-color"],
["--purple-avatar-bg", "--purple-avatar-color"],
["--gray-avatar-bg", "--gray-avatar-color0"],
]
encoded_name = str(full_name).encode("utf-8")
hash_name = hashlib.md5(encoded_name).hexdigest()
idx = cint((int(hash_name[4:6], 16) + 1) / 5.33)
return palette[idx % 8]

View File

@@ -1,11 +0,0 @@
{% set member = frappe.get_doc("User", frappe.session.user) %}
<div class="mt-10">
{% if member.get_mentored_courses() | length %}
<div class="course-home-headings"> {{ _("Courses Mentored") }} </div>
<div class="cards-parent">
{% for course in member.get_mentored_courses() %}
{{ widgets.CourseCard(course=course) }}
{% endfor %}
</div>
{% endif %}
</div>

View File

@@ -1,16 +0,0 @@
{
"__unsaved": 1,
"creation": "2021-10-21 11:32:57.411626",
"docstatus": 0,
"doctype": "Web Template",
"fields": [],
"idx": 0,
"modified": "2021-10-21 12:01:56.270656",
"modified_by": "Administrator",
"module": "LMS",
"name": "Courses Mentored",
"owner": "Administrator",
"standard": 1,
"template": "",
"type": "Section"
}

View File

@@ -1,31 +0,0 @@
{% set color = get_palette(member.full_name) %}
<div class="common-card-style member-card">
<div class="d-flex">
{{ widgets.Avatar(member=member, avatar_class=avatar_class) }}
<div class="ml-3 my-auto">
<div class="member-card-title">
{{ member.full_name }}
</div>
{% if member.headline %}
<div> {{ member.headline }} </div>
{% endif %}
{% if member.looking_for_job %}
<div class="indicator-pill green"> {{ _("Open Network") }} </div>
{% endif %}
{% set course_count = get_authored_courses(member.name, True) | length %}
{% set suffix = "Courses" if course_count > 1 else "Course" %}
{% if show_course_count and course_count > 0 %}
<div class="">
Created {{ course_count }} {{ suffix }}
</div>
{% endif %}
</div>
<a class="stretched-link" href="{{ get_profile_url(member.username) }}"></a>
</div>
</div>

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

6136
lms/locale/hr.po Normal file

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

6136
lms/locale/pt.po Normal file

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

6136
lms/locale/th.po Normal file

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

View File

@@ -1,34 +0,0 @@
# Copyright (c) 2021, FOSS United and Contributors
# See license.txt
import unittest
import frappe
from lms.lms.doctype.lms_course.test_lms_course import new_user
class TestCustomUser(unittest.TestCase):
def test_with_basic_username(self):
user = new_user("Username", "test_with_basic_username@example.com")
self.assertEqual(user.username, "username")
def test_without_username(self):
"""The user in this test has the same first name as the user of the test test_with_basic_username.
In such cases frappe makes the username of the second user empty.
The condition in lms app should override this and save a username."""
user = new_user("Username", "test-without-username@example.com")
self.assertTrue(user.username)
def test_with_short_first_name(self):
user = new_user("USN", "test_with_short_first_name@example.com")
self.assertGreaterEqual(len(user.username), 4)
@classmethod
def tearDownClass(cls) -> None:
users = [
"test_with_basic_username@example.com",
"test-without-username@example.com",
"test_with_short_first_name@example.com",
]
frappe.db.delete("User", {"name": ["in", users]})

View File

@@ -1,363 +0,0 @@
import hashlib
import frappe
import requests
from frappe import _
from frappe.core.doctype.user.user import User
from frappe.utils import cint, escape_html, random_string
from frappe.website.utils import is_signup_disabled
from lms.lms.utils import get_average_rating, get_country_code
from frappe.website.utils import cleanup_page_name
from frappe.model.naming import append_number_if_name_exists
from lms.widgets import Widgets
class CustomUser(User):
def validate(self):
super().validate()
self.validate_username_duplicates()
def after_insert(self):
super().after_insert()
self.add_roles("LMS Student")
def validate_username_duplicates(self):
while not self.username or self.username_exists():
self.username = append_number_if_name_exists(
self.doctype, cleanup_page_name(self.full_name), fieldname="username"
)
if " " in self.username:
self.username = self.username.replace(" ", "")
if len(self.username) < 4:
self.username = self.email.replace("@", "").replace(".", "")
def validate_skills(self):
unique_skills = []
for skill in self.skill:
if not skill.skill_name:
return
if not skill.skill_name in unique_skills:
unique_skills.append(skill.skill_name)
else:
frappe.throw(_("Skills must be unique"))
def get_batch_count(self) -> int:
"""Returns the number of batches authored by this user."""
return frappe.db.count(
"LMS Enrollment", {"member": self.name, "member_type": "Mentor"}
)
def get_user_reviews(self):
"""Returns the reviews created by user"""
return frappe.get_all("LMS Course Review", {"owner": self.name})
def get_mentored_courses(self):
"""Returns all courses mentored by this user"""
mentored_courses = []
mapping = frappe.get_all(
"LMS Course Mentor Mapping",
{
"mentor": self.name,
},
["name", "course"],
)
for map in mapping:
if frappe.db.get_value("LMS Course", map.course, "published"):
course = frappe.db.get_value(
"LMS Course",
map.course,
["name", "upcoming", "title", "image", "enable_certification"],
as_dict=True,
)
mentored_courses.append(course)
return mentored_courses
def get_enrolled_courses():
in_progress = []
completed = []
memberships = get_course_membership(None, member_type="Student")
for membership in memberships:
course = frappe.db.get_value(
"LMS Course",
membership.course,
[
"name",
"upcoming",
"title",
"short_introduction",
"image",
"enable_certification",
"paid_course",
"course_price",
"currency",
"published",
"creation",
],
as_dict=True,
)
if not course.published:
continue
course.enrollment_count = frappe.db.count(
"LMS Enrollment", {"course": course.name, "member_type": "Student"}
)
course.avg_rating = get_average_rating(course.name) or 0
progress = cint(membership.progress)
if progress < 100:
in_progress.append(course)
else:
completed.append(course)
in_progress.sort(key=lambda x: x.enrollment_count, reverse=True)
completed.sort(key=lambda x: x.enrollment_count, reverse=True)
return {"in_progress": in_progress, "completed": completed}
def get_course_membership(member=None, member_type=None):
"""Returns all memberships of the user."""
filters = {"member": member or frappe.session.user}
if member_type:
filters["member_type"] = member_type
return frappe.get_all("LMS Enrollment", filters, ["name", "course", "progress"])
def get_authored_courses(member=None, only_published=True):
"""Returns the number of courses authored by this user."""
course_details = []
courses = frappe.get_all(
"Course Instructor", {"instructor": member or frappe.session.user}, ["parent"]
)
for course in courses:
detail = frappe.db.get_value(
"LMS Course",
course.parent,
[
"name",
"upcoming",
"title",
"short_introduction",
"image",
"paid_course",
"course_price",
"currency",
"status",
"published",
"creation",
],
as_dict=True,
)
if only_published and detail and not detail.published:
continue
detail.enrollment_count = frappe.db.count(
"LMS Enrollment", {"course": detail.name, "member_type": "Student"}
)
detail.avg_rating = get_average_rating(detail.name) or 0
course_details.append(detail)
course_details.sort(key=lambda x: x.enrollment_count, reverse=True)
return course_details
def get_palette(full_name):
"""
Returns a color unique to each member for Avatar"""
palette = [
["--orange-avatar-bg", "--orange-avatar-color"],
["--pink-avatar-bg", "--pink-avatar-color"],
["--blue-avatar-bg", "--blue-avatar-color"],
["--green-avatar-bg", "--green-avatar-color"],
["--dark-green-avatar-bg", "--dark-green-avatar-color"],
["--red-avatar-bg", "--red-avatar-color"],
["--yellow-avatar-bg", "--yellow-avatar-color"],
["--purple-avatar-bg", "--purple-avatar-color"],
["--gray-avatar-bg", "--gray-avatar-color0"],
]
encoded_name = str(full_name).encode("utf-8")
hash_name = hashlib.md5(encoded_name).hexdigest()
idx = cint((int(hash_name[4:6], 16) + 1) / 5.33)
return palette[idx % 8]
@frappe.whitelist(allow_guest=True)
def sign_up(email, full_name, verify_terms, user_category):
if is_signup_disabled():
frappe.throw(_("Sign Up is disabled"), _("Not Allowed"))
user = frappe.db.get("User", {"email": email})
if user:
if user.enabled:
return 0, _("Already Registered")
else:
return 0, _("Registered but disabled")
else:
if frappe.db.get_creation_count("User", 60) > 300:
frappe.respond_as_web_page(
_("Temporarily Disabled"),
_(
"Too many users signed up recently, so the registration is disabled. Please try back in an hour"
),
http_status_code=429,
)
user = frappe.get_doc(
{
"doctype": "User",
"email": email,
"first_name": escape_html(full_name),
"verify_terms": verify_terms,
"user_category": user_category,
"country": "",
"enabled": 1,
"new_password": random_string(10),
"user_type": "Website User",
}
)
user.flags.ignore_permissions = True
user.flags.ignore_password_policy = True
user.insert()
# set default signup role as per Portal Settings
default_role = frappe.db.get_value("Portal Settings", None, "default_role")
if default_role:
user.add_roles(default_role)
user.add_roles("LMS Student")
set_country_from_ip(None, user.name)
if user.flags.email_sent:
return 1, _("Please check your email for verification")
else:
return 2, _("Please ask your administrator to verify your sign-up")
def set_country_from_ip(login_manager=None, user=None):
if not user and login_manager:
user = login_manager.user
user_country = frappe.db.get_value("User", user, "country")
# if user_country:
# return
frappe.db.set_value("User", user, "country", get_country_code())
return
def on_session_creation(login_manager):
if frappe.db.get_single_value(
"System Settings", "setup_complete"
) and frappe.db.get_single_value("LMS Settings", "default_home"):
frappe.local.response["home_page"] = "/lms"
@frappe.whitelist()
def search_users(start: int = 0, text: str = ""):
start = cint(start)
search_text = frappe.db.escape(f"%{text}%")
or_filters = get_or_filters(search_text)
count = len(get_users(or_filters, 0, 900000000))
users = get_users(or_filters, start, 24)
user_details = get_user_details(users)
return {"user_details": user_details, "start": start + 24, "count": count}
def get_or_filters(text):
user_fields = [
"first_name",
"last_name",
"full_name",
"email",
"preferred_location",
"dream_companies",
]
education_fields = ["institution_name", "location", "degree_type", "major"]
work_fields = ["title", "company"]
certification_fields = ["certification_name", "organization"]
or_filters = []
if text:
for field in user_fields:
or_filters.append(f"u.{field} like {text}")
for field in education_fields:
or_filters.append(f"ed.{field} like {text}")
for field in work_fields:
or_filters.append(f"we.{field} like {text}")
for field in certification_fields:
or_filters.append(f"c.{field} like {text}")
or_filters.append(f"s.skill_name like {text}")
or_filters.append(f"pf.function like {text}")
or_filters.append(f"pi.industry like {text}")
return "AND ({})".format(" OR ".join(or_filters)) if or_filters else ""
def get_user_details(users):
user_details = []
for user in users:
details = frappe.db.get_value(
"User",
user,
["name", "username", "full_name", "user_image", "headline", "looking_for_job"],
as_dict=True,
)
user_details.append(Widgets().MemberCard(member=details, avatar_class="avatar-large"))
return user_details
def get_users(or_filters, start, page_length):
users = frappe.db.sql(
"""
SELECT DISTINCT u.name
FROM `tabUser` u
LEFT JOIN `tabEducation Detail` ed
ON u.name = ed.parent
LEFT JOIN `tabWork Experience` we
ON u.name = we.parent
LEFT JOIN `tabCertification` c
ON u.name = c.parent
LEFT JOIN `tabSkills` s
ON u.name = s.parent
LEFT JOIN `tabPreferred Function` pf
ON u.name = pf.parent
LEFT JOIN `tabPreferred Industry` pi
ON u.name = pi.parent
WHERE u.enabled = True {or_filters}
ORDER BY u.creation desc
LIMIT {start}, {page_length}
""".format(
or_filters=or_filters, start=start, page_length=page_length
),
as_dict=1,
)
return users
@frappe.whitelist()
def save_role(user, role, value):
frappe.only_for("Moderator")
if cint(value):
doc = frappe.get_doc(
{
"doctype": "Has Role",
"parent": user,
"role": role,
"parenttype": "User",
"parentfield": "roles",
}
)
doc.save(ignore_permissions=True)
else:
frappe.db.delete("Has Role", {"parent": user, "role": role})
return True

View File

@@ -6,6 +6,7 @@ import re
import os import os
import mimetypes import mimetypes
import frappe import frappe
from frappe.utils import get_files_path
from frappe.website.page_renderers.base_renderer import BaseRenderer from frappe.website.page_renderers.base_renderer import BaseRenderer
from frappe.website.page_renderers.document_page import DocumentPage from frappe.website.page_renderers.document_page import DocumentPage
from frappe.website.page_renderers.list_page import ListPage from frappe.website.page_renderers.list_page import ListPage
@@ -112,37 +113,6 @@ def render_portal_page(path, **kwargs):
return page.render() return page.render()
class CoursePage(BaseRenderer):
def __init__(self, path, http_status_code):
super().__init__(path, http_status_code)
self.renderer = None
def can_render(self):
return self.path.startswith("course")
def render(self):
if "learn" in self.path:
prefix = self.path.split("/learn")[0]
course_name = prefix.split("/")[1]
lesson_index = self.path.split("/learn/")[1]
chapter_number = lesson_index.split(".")[0]
lesson_number = lesson_index.split(".")[1]
frappe.flags.redirect_location = (
f"/lms/courses/{course_name}/learn/{chapter_number}-{lesson_number}"
)
return RedirectPage(self.path).render()
elif len(self.path.split("/")) > 1:
course_name = self.path.split("/")[1]
frappe.flags.redirect_location = f"/lms/courses/{course_name}"
return RedirectPage(self.path).render()
else:
frappe.flags.redirect_location = "/lms/courses"
return RedirectPage(self.path).render()
class SCORMRenderer(BaseRenderer): class SCORMRenderer(BaseRenderer):
def can_render(self): def can_render(self):
return "scorm/" in self.path return "scorm/" in self.path
@@ -173,3 +143,23 @@ class SCORMRenderer(BaseRenderer):
) )
response.mimetype = mimetypes.guess_type(index_path)[0] response.mimetype = mimetypes.guess_type(index_path)[0]
return response return response
elif not os.path.exists(path):
chapter_folder = "/".join(self.path.split("/")[:3])
chapter_folder_path = os.path.realpath(
frappe.get_site_path("public", chapter_folder)
)
file = path.split("/")[-1]
correct_file_path = None
for root, dirs, files in os.walk(chapter_folder_path):
if file in files:
correct_file_path = os.path.join(root, file)
break
if correct_file_path:
f = open(correct_file_path, "rb")
response = Response(
wrap_file(frappe.local.request.environ, f), direct_passthrough=True
)
response.mimetype = mimetypes.guess_type(correct_file_path)[0]
return response

View File

@@ -9,23 +9,20 @@
<p> <p>
<b>{{ _("Batch:") }}</b> {{ title }} <b>{{ _("Batch:") }}</b> {{ title }}
</p> </p>
<br>
<p> <p>
<b>{{ _("Start Date:") }}</b> {{ frappe.utils.format_date(start_date, "medium") }} <b>{{ _("Start Date:") }}</b> {{ frappe.utils.format_date(start_date, "long") }}
</p> </p>
<br>
<p> <p>
<b>{{ _("Timings:") }}</b> {{ frappe.utils.format_time(start_time, "hh:mm a") }} <b>{{ _("Timings:") }}</b> {{ frappe.utils.format_time(start_time, "hh:mm a") }}
</p> </p>
<br>
<p> <p>
<b>{{ _("Medium:") }}</b> {{ medium }} <b>{{ _("Medium:") }}</b> {{ medium }}
</p> </p>
<br> <br>
<p> <p>
{{ _("Visit the following link to view your ") }} <a href="/lms/batches/{{ name }}">👉 {{ _("Visit your batch") }}</a>
<a href="/lms/batches/{{ name }}">{{ _("Batch Details") }}</a>
</p> </p>
<br>
<p> <p>
{{ _("If you have any questions or require assistance, feel free to contact us.") }} {{ _("If you have any questions or require assistance, feel free to contact us.") }}
</p> </p>

View File

@@ -9,19 +9,17 @@
<p> <p>
<b>{{ _("Class:") }}</b> {{ title }} <b>{{ _("Class:") }}</b> {{ title }}
</p> </p>
<br>
<p> <p>
<b>{{ _("Date:") }}</b> {{ frappe.utils.format_date(date, "medium") }} <b>{{ _("Date:") }}</b> {{ frappe.utils.format_date(date, "long") }}
</p> </p>
<br>
<p> <p>
<b>{{ _("Timings:") }}</b> {{ frappe.utils.format_time(time, "hh:mm a") }} <b>{{ _("Timings:") }}</b> {{ frappe.utils.format_time(time, "hh:mm a") }}
</p> </p>
<br> <br>
<p> <p>
{{ _("Visit the following link to view your ") }} <a href="/lms/batches/{{ batch_name }}">👉 {{ _("Visit your batch") }}</a>
<a href="/lms/live_classes/{{ batch_name }}">{{ _("Batch Details") }}</a>
</p> </p>
<br>
<p> <p>
{{ _("If you have any questions or require assistance, feel free to contact us.") }} {{ _("If you have any questions or require assistance, feel free to contact us.") }}
</p> </p>

View File

@@ -85,7 +85,7 @@
} }
frappe.call({ frappe.call({
method: "lms.overrides.user.sign_up", method: "lms.lms.user.sign_up",
args: { args: {
"email": email, "email": email,
"full_name": full_name, "full_name": full_name,

View File

@@ -9,9 +9,9 @@ from lms import __version__ as version
setup( setup(
name="lms", name="lms",
version=version, version=version,
description="LMS App", description="Learning Management System",
author="Frappe", author="Jannat",
author_email="school@frappe.io", author_email="jannat@frappe.io",
packages=find_packages(), packages=find_packages(),
zip_safe=False, zip_safe=False,
include_package_data=True, include_package_data=True,

3602
yarn.lock

File diff suppressed because it is too large Load Diff