Compare commits
215 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
04aff8d149 | ||
|
|
e88bdd818d | ||
|
|
1a5d8ce07e | ||
|
|
8e405bc8eb | ||
|
|
23e2a153c9 | ||
|
|
85a0949488 | ||
|
|
57b6433dc0 | ||
|
|
1b43e1be44 | ||
|
|
d6738b86c9 | ||
|
|
a5325cef44 | ||
|
|
cc917f3d83 | ||
|
|
492917ea40 | ||
|
|
78263185a1 | ||
|
|
ee7aa9d58b | ||
|
|
a7112937de | ||
|
|
a8d4572aef | ||
|
|
45c530e53a | ||
|
|
e0bcce5e6e | ||
|
|
8346ec8525 | ||
|
|
5d1673bad8 | ||
|
|
a33328e11d | ||
|
|
3efa326684 | ||
|
|
196fead1e0 | ||
|
|
b8ce04e9fe | ||
|
|
6369dfd65c | ||
|
|
f4da56adf9 | ||
|
|
0987a91bfc | ||
|
|
9f23a56cf4 | ||
|
|
34a4754767 | ||
|
|
b88de74552 | ||
|
|
45ac682c7f | ||
|
|
b753d366bf | ||
|
|
06c598886e | ||
|
|
52b0b7f8dc | ||
|
|
656b3b2ebe | ||
|
|
6bdfbde23f | ||
|
|
1b9f5eebc0 | ||
|
|
1f37da08b4 | ||
|
|
5bc44e6fe5 | ||
|
|
c70da08078 | ||
|
|
7600fb14e1 | ||
|
|
e2fdf2042e | ||
|
|
8477d6b9ed | ||
|
|
241df63334 | ||
|
|
7131de8a2a | ||
|
|
473a799f58 | ||
|
|
6c9fe85170 | ||
|
|
2c5d2db340 | ||
|
|
6cd2e6e7fb | ||
|
|
a6b094cff9 | ||
|
|
b024a4546c | ||
|
|
519715f8ee | ||
|
|
522de390a7 | ||
|
|
2ffe19cea1 | ||
|
|
124dc10cc3 | ||
|
|
a41338c3a2 | ||
|
|
aa979b96f2 | ||
|
|
f9b2471b32 | ||
|
|
d594f3ac88 | ||
|
|
e5190d4409 | ||
|
|
4f876c2bbc | ||
|
|
4d031ae55e | ||
|
|
89a348b154 | ||
|
|
db62d40c50 | ||
|
|
eff2ae8a73 | ||
|
|
b23d29767f | ||
|
|
7d5a3c3421 | ||
|
|
1054623d9d | ||
|
|
4eba93f47b | ||
|
|
13bcc84e8f | ||
|
|
c726ad3467 | ||
|
|
5e95ff963c | ||
|
|
1ef232e45b | ||
|
|
034654193f | ||
|
|
bddaa26d5a | ||
|
|
b42648fecb | ||
|
|
aa800bf96b | ||
|
|
6575e139b5 | ||
|
|
c5b3460006 | ||
|
|
b1e490765b | ||
|
|
c0f4a09e22 | ||
|
|
8fb5311844 | ||
|
|
12122f1eaf | ||
|
|
e83312289b | ||
|
|
d59f4113c1 | ||
|
|
8e3b70e7c8 | ||
|
|
c25d95b3b6 | ||
|
|
edde95edeb | ||
|
|
066eaea45d | ||
|
|
7ae3cf5d95 | ||
|
|
2fa728d45c | ||
|
|
04cbd6a1d8 | ||
|
|
c6e658e26b | ||
|
|
0692aceda4 | ||
|
|
072bef5847 | ||
|
|
e94a689f83 | ||
|
|
c71a980f78 | ||
|
|
ef7d850dd4 | ||
|
|
1e6a71f36b | ||
|
|
f5ae4120cd | ||
|
|
82331364b7 | ||
|
|
ef3879e419 | ||
|
|
403dbf13e8 | ||
|
|
c8193c0009 | ||
|
|
9c0c69a728 | ||
|
|
4606fc3e2a | ||
|
|
c9bb3ab368 | ||
|
|
99e4b406a4 | ||
|
|
67b9424b9e | ||
|
|
5b60be5f51 | ||
|
|
d88927a6fb | ||
|
|
6616ee3607 | ||
|
|
0dbd8de335 | ||
|
|
9b406e368b | ||
|
|
4449dc43a0 | ||
|
|
554093ab3e | ||
|
|
ac3ed22ae9 | ||
|
|
2ca7b09d1e | ||
|
|
f29c2da9ce | ||
|
|
e23f6ae0fa | ||
|
|
51061273bc | ||
|
|
4a0812dfe9 | ||
|
|
efb694a6e6 | ||
|
|
1dbe2f31d0 | ||
|
|
be9525dbf2 | ||
|
|
a24afad641 | ||
|
|
abd14aa33c | ||
|
|
5b3c0685ac | ||
|
|
2a59d9ff04 | ||
|
|
619dc73bcb | ||
|
|
02edefc158 | ||
|
|
572f5ae585 | ||
|
|
a326866cc9 | ||
|
|
17decf7b71 | ||
|
|
b9784e22ff | ||
|
|
0f600c5b70 | ||
|
|
a606e9c974 | ||
|
|
9e1938095c | ||
|
|
3491eb3881 | ||
|
|
6277340d6b | ||
|
|
0c12ee4452 | ||
|
|
4ec245a119 | ||
|
|
24fa6d17de | ||
|
|
2eedc1032c | ||
|
|
8c3b1b433f | ||
|
|
ae3f0f9a4e | ||
|
|
f4ae601f0d | ||
|
|
2104b86080 | ||
|
|
9724dceb73 | ||
|
|
4c07a4f35d | ||
|
|
6a15697957 | ||
|
|
47f880d8dc | ||
|
|
d5814f5680 | ||
|
|
345a444d73 | ||
|
|
0053ce5602 | ||
|
|
9851757a4e | ||
|
|
55fe25b8cb | ||
|
|
714f8a17c3 | ||
|
|
732e9db9af | ||
|
|
6fbc448a52 | ||
|
|
76fc241778 | ||
|
|
51cbbfdc45 | ||
|
|
279f2f503e | ||
|
|
795d95b482 | ||
|
|
5b5b95c85c | ||
|
|
8490b07c90 | ||
|
|
dee2c51c60 | ||
|
|
4149fa6ce4 | ||
|
|
7a69611f09 | ||
|
|
6692252df9 | ||
|
|
486ce1bdb0 | ||
|
|
cceff77bc2 | ||
|
|
22a9169f87 | ||
|
|
47a30763a0 | ||
|
|
73379a1bd8 | ||
|
|
7cc46629b4 | ||
|
|
67304245ba | ||
|
|
8edd3a1a34 | ||
|
|
e4bc7c8d78 | ||
|
|
a8af78d400 | ||
|
|
0afe3de818 | ||
|
|
3c81aadec6 | ||
|
|
1dfcb035da | ||
|
|
77b24882a9 | ||
|
|
1fd0673257 | ||
|
|
dbda76e0ce | ||
|
|
a9d22521ce | ||
|
|
6da1d9629f | ||
|
|
37b61a7087 | ||
|
|
9b484e6ee9 | ||
|
|
5ef67ef21c | ||
|
|
f902166643 | ||
|
|
8f91466b3d | ||
|
|
fa1621c3d1 | ||
|
|
2acd45feae | ||
|
|
f19e974b9d | ||
|
|
01598ac002 | ||
|
|
9b3906359b | ||
|
|
4224580d6f | ||
|
|
07d30647d8 | ||
|
|
263096fc77 | ||
|
|
b510cbce7f | ||
|
|
0b84dc3266 | ||
|
|
7ee7b95eb5 | ||
|
|
83b8bdde45 | ||
|
|
1b5dd15b90 | ||
|
|
47c224fcad | ||
|
|
1c866f40eb | ||
|
|
1861aabaca | ||
|
|
cd8fb6eb38 | ||
|
|
21d05d3731 | ||
|
|
7c953925f9 | ||
|
|
33a4bbbe47 | ||
|
|
03915ccfbd | ||
|
|
c6d59216fd |
2
.github/workflows/ci.yml
vendored
2
.github/workflows/ci.yml
vendored
@@ -39,7 +39,7 @@ jobs:
|
|||||||
node-version: '18'
|
node-version: '18'
|
||||||
check-latest: true
|
check-latest: true
|
||||||
- name: setup cache for bench
|
- name: setup cache for bench
|
||||||
uses: actions/cache@v2
|
uses: actions/cache@v4
|
||||||
with:
|
with:
|
||||||
path: ~/bench-cache
|
path: ~/bench-cache
|
||||||
key: ${{ runner.os }}
|
key: ${{ runner.os }}
|
||||||
|
|||||||
32
.github/workflows/linters.yml
vendored
32
.github/workflows/linters.yml
vendored
@@ -7,8 +7,27 @@ on:
|
|||||||
branches: [ main ]
|
branches: [ main ]
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
|
commit-lint:
|
||||||
|
name: 'Semantic Commits'
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
if: github.event_name == 'pull_request'
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
fetch-depth: 200
|
||||||
|
- uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node-version: 20
|
||||||
|
check-latest: true
|
||||||
|
|
||||||
|
- name: Check commit titles
|
||||||
|
run: |
|
||||||
|
npm install @commitlint/cli @commitlint/config-conventional
|
||||||
|
npx commitlint --verbose --from ${{ github.event.pull_request.base.sha }} --to ${{ github.event.pull_request.head.sha }}
|
||||||
|
|
||||||
linters:
|
linters:
|
||||||
name: Semantic Commits
|
name: Semgrep Rules
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
if: github.event_name == 'pull_request'
|
if: github.event_name == 'pull_request'
|
||||||
|
|
||||||
@@ -20,8 +39,17 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
python-version: '3.10'
|
python-version: '3.10'
|
||||||
|
|
||||||
|
- name: Cache pip
|
||||||
|
uses: actions/cache@v4
|
||||||
|
with:
|
||||||
|
path: ~/.cache/pip
|
||||||
|
key: ${{ runner.os }}-pip-${{ hashFiles('**/*requirements.txt', '**/pyproject.toml', '**/setup.py') }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-pip-
|
||||||
|
${{ runner.os }}-
|
||||||
|
|
||||||
- name: Install and Run Pre-commit
|
- name: Install and Run Pre-commit
|
||||||
uses: pre-commit/action@v2.0.3
|
uses: pre-commit/action@v3.0.1
|
||||||
|
|
||||||
- name: Download Semgrep rules
|
- name: Download Semgrep rules
|
||||||
run: git clone --depth 1 https://github.com/frappe/semgrep-rules.git frappe-semgrep-rules
|
run: git clone --depth 1 https://github.com/frappe/semgrep-rules.git frappe-semgrep-rules
|
||||||
|
|||||||
6
.github/workflows/ui-tests.yml
vendored
6
.github/workflows/ui-tests.yml
vendored
@@ -58,7 +58,7 @@ jobs:
|
|||||||
echo "127.0.0.1 lms.test" | sudo tee -a /etc/hosts
|
echo "127.0.0.1 lms.test" | sudo tee -a /etc/hosts
|
||||||
|
|
||||||
- name: Cache pip
|
- name: Cache pip
|
||||||
uses: actions/cache@v3
|
uses: actions/cache@v4
|
||||||
with:
|
with:
|
||||||
path: ~/.cache/pip
|
path: ~/.cache/pip
|
||||||
key: ${{ runner.os }}-pip-${{ hashFiles('**/*requirements.txt', '**/pyproject.toml', '**/setup.py') }}
|
key: ${{ runner.os }}-pip-${{ hashFiles('**/*requirements.txt', '**/pyproject.toml', '**/setup.py') }}
|
||||||
@@ -70,7 +70,7 @@ jobs:
|
|||||||
id: yarn-cache-dir-path
|
id: yarn-cache-dir-path
|
||||||
run: echo "dir=$(yarn cache dir)" >> $GITHUB_OUTPUT
|
run: echo "dir=$(yarn cache dir)" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
- uses: actions/cache@v3
|
- uses: actions/cache@v4
|
||||||
id: yarn-cache
|
id: yarn-cache
|
||||||
with:
|
with:
|
||||||
path: ${{ steps.yarn-cache-dir-path.outputs.dir }}
|
path: ${{ steps.yarn-cache-dir-path.outputs.dir }}
|
||||||
@@ -79,7 +79,7 @@ jobs:
|
|||||||
${{ runner.os }}-yarn-ui-
|
${{ runner.os }}-yarn-ui-
|
||||||
|
|
||||||
- name: Cache cypress binary
|
- name: Cache cypress binary
|
||||||
uses: actions/cache@v3
|
uses: actions/cache@v4
|
||||||
with:
|
with:
|
||||||
path: ~/.cache/Cypress
|
path: ~/.cache/Cypress
|
||||||
key: ${{ runner.os }}-cypress
|
key: ${{ runner.os }}-cypress
|
||||||
|
|||||||
26
commitlint.config.js
Normal file
26
commitlint.config.js
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
module.exports = {
|
||||||
|
parserPreset: "conventional-changelog-conventionalcommits",
|
||||||
|
rules: {
|
||||||
|
"subject-empty": [2, "never"],
|
||||||
|
"type-case": [2, "always", "lower-case"],
|
||||||
|
"type-empty": [2, "never"],
|
||||||
|
"type-enum": [
|
||||||
|
2,
|
||||||
|
"always",
|
||||||
|
[
|
||||||
|
"build",
|
||||||
|
"chore",
|
||||||
|
"ci",
|
||||||
|
"docs",
|
||||||
|
"feat",
|
||||||
|
"fix",
|
||||||
|
"perf",
|
||||||
|
"refactor",
|
||||||
|
"revert",
|
||||||
|
"style",
|
||||||
|
"test",
|
||||||
|
"deprecate", // deprecation decision
|
||||||
|
],
|
||||||
|
],
|
||||||
|
},
|
||||||
|
};
|
||||||
@@ -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,
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
Submodule frappe-ui updated: 70bc4760e4...29307e4fff
96
frontend/components.d.ts
vendored
Normal file
96
frontend/components.d.ts
vendored
Normal 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']
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -2,9 +2,9 @@
|
|||||||
<html lang="en">
|
<html lang="en">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="UTF-8" />
|
<meta charset="UTF-8" />
|
||||||
<link rel="icon" href="{{ favicon or '/assets/lms/frontend/favicon.png' }}" />
|
<link rel="icon" href="{{ favicon }}" />
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||||
<title>Frappe Learning</title>
|
<title>{{ title }}</title>
|
||||||
<meta name="title" content="{{ meta.title }}" />
|
<meta name="title" content="{{ meta.title }}" />
|
||||||
<meta name="image" content="{{ meta.image }}" />
|
<meta name="image" content="{{ meta.image }}" />
|
||||||
<meta name="description" content="{{ meta.description }}" />
|
<meta name="description" content="{{ meta.description }}" />
|
||||||
@@ -23,17 +23,6 @@
|
|||||||
<p>
|
<p>
|
||||||
{{ meta.description }}
|
{{ meta.description }}
|
||||||
</p>
|
</p>
|
||||||
<p>
|
|
||||||
The content here is just for seo purposes. The actual content will be loaded in a few seconds.
|
|
||||||
</p>
|
|
||||||
<p>
|
|
||||||
Seo checks if a page has more than 300 words. So, here are some more words to make it more than 300 words.
|
|
||||||
Page descriptions are the HTML meta tags that provide a brief summary of a web page.
|
|
||||||
Search engines use meta descriptions to help identify the page's topic - they don't use them to rank the page, but they do use them to determine whether or not to display the page in search results.
|
|
||||||
Meta descriptions are important because they're often the first thing people see when they're deciding which search result to click on.
|
|
||||||
They're also important because they can help improve your click-through rate (CTR) from search results.
|
|
||||||
A good meta description can entice people to click on your page instead of someone else's.
|
|
||||||
</p>
|
|
||||||
<a href="{{ meta.link }}">Know More</a>
|
<a href="{{ meta.link }}">Know More</a>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -41,9 +30,8 @@
|
|||||||
<div id="popovers"></div>
|
<div id="popovers"></div>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
window.csrf_token = '{{ csrf_token }}'
|
|
||||||
window.setup_complete = '{{ setup_complete }}'
|
|
||||||
document.getElementById('seo-content').style.display = 'none';
|
document.getElementById('seo-content').style.display = 'none';
|
||||||
|
window.csrf_token = '{{ csrf_token }}'
|
||||||
</script>
|
</script>
|
||||||
<script type="module" src="/src/main.js"></script>
|
<script type="module" src="/src/main.js"></script>
|
||||||
</body>
|
</body>
|
||||||
|
|||||||
@@ -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.109",
|
"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",
|
||||||
|
|||||||
4
frontend/public/learning.svg
Normal file
4
frontend/public/learning.svg
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
<svg width="80" height="79" viewBox="0 0 80 79" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M57.1285 0.580383H22.8514C10.2309 0.580383 0 10.5649 0 22.8815V56.3332C0 68.6497 10.2309 78.6343 22.8514 78.6343H57.1285C69.749 78.6343 79.9799 68.6497 79.9799 56.3332V22.8815C79.9799 10.5649 69.749 0.580383 57.1285 0.580383Z" fill="#0E7159"/>
|
||||||
|
<path d="M62.8434 23.6906L60.7869 23.1052C53.6744 21.0702 45.9048 22.4641 39.992 26.8128C35.8502 23.7742 30.7943 22.1854 25.7099 22.2133H17.1406V27.8163H25.7099C29.6232 27.8163 33.508 29.015 36.6787 31.3845L39.992 33.8377L43.3056 31.3845C47.2475 28.4575 52.3032 27.2588 57.1306 28.0393V50.647C51.1035 49.9223 44.9051 51.4834 39.992 55.0795C35.8502 52.0688 30.8515 50.4798 25.7671 50.4798C24.7959 50.4798 23.8247 50.5355 22.8535 50.647V35.0642H17.1406V57.0588H62.8434V23.7185V23.6906Z" fill="white"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 856 B |
@@ -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>
|
||||||
|
|||||||
@@ -62,26 +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?.user_type == 'System User' &&
|
userResource.data?.is_system_manager && userResource.data?.is_fc_site
|
||||||
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,
|
||||||
}"
|
}"
|
||||||
@@ -90,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"
|
||||||
@@ -103,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()
|
||||||
@@ -124,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(
|
||||||
{},
|
{},
|
||||||
{
|
{
|
||||||
@@ -144,7 +221,7 @@ onMounted(() => {
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
})
|
}
|
||||||
|
|
||||||
const unreadNotifications = createResource({
|
const unreadNotifications = createResource({
|
||||||
cache: 'Unread Notifications Count',
|
cache: 'Unread Notifications Count',
|
||||||
@@ -188,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',
|
||||||
|
],
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -199,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',
|
||||||
|
],
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -262,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(
|
||||||
@@ -287,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>
|
||||||
|
|||||||
@@ -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()
|
||||||
|
|||||||
@@ -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>
|
|
||||||
@@ -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"
|
||||||
|
|||||||
@@ -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)
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|||||||
@@ -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>
|
||||||
|
|||||||
@@ -130,7 +130,7 @@ import {
|
|||||||
ComboboxOptions,
|
ComboboxOptions,
|
||||||
ComboboxOption,
|
ComboboxOption,
|
||||||
} from '@headlessui/vue'
|
} from '@headlessui/vue'
|
||||||
import { Popover, Button } from 'frappe-ui'
|
import { Popover } from 'frappe-ui'
|
||||||
import { ChevronDown, X } from 'lucide-vue-next'
|
import { ChevronDown, X } from 'lucide-vue-next'
|
||||||
import { ref, computed, useAttrs, useSlots, watch, nextTick } from 'vue'
|
import { ref, computed, useAttrs, useSlots, watch, nextTick } from 'vue'
|
||||||
|
|
||||||
|
|||||||
@@ -146,7 +146,6 @@ function resetEditor(value: string, resetHistory = false) {
|
|||||||
value = getModelValue()
|
value = getModelValue()
|
||||||
aceEditor?.setValue(value)
|
aceEditor?.setValue(value)
|
||||||
aceEditor?.clearSelection()
|
aceEditor?.clearSelection()
|
||||||
console.log(isDark.value)
|
|
||||||
aceEditor?.setTheme(isDark.value ? 'ace/theme/twilight' : 'ace/theme/chrome')
|
aceEditor?.setTheme(isDark.value ? 'ace/theme/twilight' : 'ace/theme/chrome')
|
||||||
props.autofocus && aceEditor?.focus()
|
props.autofocus && aceEditor?.focus()
|
||||||
if (resetHistory) {
|
if (resetHistory) {
|
||||||
|
|||||||
@@ -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 }}
|
||||||
|
|||||||
@@ -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"
|
||||||
|
|||||||
@@ -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
|
||||||
@@ -135,6 +142,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<ChapterModal
|
<ChapterModal
|
||||||
|
v-if="user.data"
|
||||||
v-model="showChapterModal"
|
v-model="showChapterModal"
|
||||||
v-model:outline="outline"
|
v-model:outline="outline"
|
||||||
:course="courseName"
|
:course="courseName"
|
||||||
|
|||||||
129
frontend/src/components/Evaluators.vue
Normal file
129
frontend/src/components/Evaluators.vue
Normal 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>
|
||||||
@@ -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>
|
|
||||||
23
frontend/src/components/Icons/FrappeCloudIcon.vue
Normal file
23
frontend/src/components/Icons/FrappeCloudIcon.vue
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
<template>
|
||||||
|
<svg
|
||||||
|
width="32"
|
||||||
|
height="32"
|
||||||
|
viewBox="0 0 32 32"
|
||||||
|
fill="none"
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
>
|
||||||
|
<rect
|
||||||
|
x="0.75"
|
||||||
|
y="0.75"
|
||||||
|
width="30.5"
|
||||||
|
height="30.5"
|
||||||
|
rx="6.25"
|
||||||
|
stroke="currentColor"
|
||||||
|
stroke-width="1.5"
|
||||||
|
/>
|
||||||
|
<path
|
||||||
|
d="M24.5011 14.1124C23.3954 12.4873 21.532 11.5477 19.594 11.6747C18.7616 10.1384 17.2211 9.12267 15.4198 9.0084C14.1651 8.93222 12.8979 9.3766 11.9165 10.24C11.2456 10.8367 10.7611 11.5223 10.463 12.2968C10.289 12.7539 9.89151 13.0459 9.46912 13.0459H6.5V15.5852H9.46912C10.9226 15.5852 12.2271 14.6584 12.7737 13.2237C12.9227 12.8301 13.1712 12.4873 13.5439 12.1571C14.0284 11.7255 14.662 11.4969 15.2583 11.535C16.1528 11.5985 16.7863 12.0175 17.1839 12.538C17.6063 13.0205 17.8423 13.7696 17.979 14.5187C18.774 14.2902 19.6437 14.0997 20.476 14.2394C21.1593 14.3536 21.7929 14.7218 22.2525 15.2678C22.327 15.3567 22.4016 15.4456 22.4637 15.5471C23.06 16.4232 23.1718 17.5024 22.7743 18.5689C22.414 19.5592 21.0847 20.4607 19.9791 20.4607H11.3326C10.1524 20.4607 9.18339 19.5592 9.03432 18.4038H6.54969C6.71119 20.9686 8.78585 23 11.3326 23H19.9915C22.1283 23 24.3769 21.451 25.1098 19.4704C25.7931 17.6167 25.5695 15.6614 24.5135 14.0997L24.5011 14.1124Z"
|
||||||
|
fill="currentColor"
|
||||||
|
/>
|
||||||
|
</svg>
|
||||||
|
</template>
|
||||||
16
frontend/src/components/Icons/InviteIcon.vue
Normal file
16
frontend/src/components/Icons/InviteIcon.vue
Normal 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>
|
||||||
@@ -1,36 +1,18 @@
|
|||||||
<template>
|
<template>
|
||||||
<svg
|
<svg
|
||||||
width="118"
|
width="80"
|
||||||
height="118"
|
height="79"
|
||||||
viewBox="0 0 118 118"
|
viewBox="0 0 80 79"
|
||||||
fill="none"
|
fill="none"
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
>
|
>
|
||||||
<path
|
<path
|
||||||
d="M93.9278 0H23.1013C10.3428 0 0 10.3428 0 23.1013V93.9278C0 106.686 10.3428 117.029 23.1013 117.029H93.9278C106.686 117.029 117.029 106.686 117.029 93.9278V23.1013C117.029 10.3428 106.686 0 93.9278 0Z"
|
d="M57.1285 0.580383H22.8514C10.2309 0.580383 0 10.5649 0 22.8815V56.3332C0 68.6497 10.2309 78.6343 22.8514 78.6343H57.1285C69.749 78.6343 79.9799 68.6497 79.9799 56.3332V22.8815C79.9799 10.5649 69.749 0.580383 57.1285 0.580383Z"
|
||||||
fill="url(#paint0_radial_174_336)"
|
fill="#0E7159"
|
||||||
/>
|
/>
|
||||||
<path
|
<path
|
||||||
d="M93.9278 0H23.1013C10.3428 0 0 10.3428 0 23.1013V93.9278C0 106.686 10.3428 117.029 23.1013 117.029H93.9278C106.686 117.029 117.029 106.686 117.029 93.9278V23.1013C117.029 10.3428 106.686 0 93.9278 0Z"
|
d="M62.8434 23.6906L60.7869 23.1052C53.6744 21.0702 45.9048 22.4641 39.992 26.8128C35.8502 23.7742 30.7943 22.1854 25.7099 22.2133H17.1406V27.8163H25.7099C29.6232 27.8163 33.508 29.015 36.6787 31.3845L39.992 33.8377L43.3056 31.3845C47.2475 28.4575 52.3032 27.2588 57.1306 28.0393V50.647C51.1035 49.9223 44.9051 51.4834 39.992 55.0795C35.8502 52.0688 30.8515 50.4798 25.7671 50.4798C24.7959 50.4798 23.8247 50.5355 22.8535 50.647V35.0642H17.1406V57.0588H62.8434V23.7185V23.6906Z"
|
||||||
fill="#0B3D3D"
|
fill="white"
|
||||||
fill-opacity="0.8"
|
|
||||||
/>
|
/>
|
||||||
<path
|
|
||||||
d="M95.1879 33.1294L91.4077 32.0268C80.1721 28.7716 67.9389 30.9242 58.5409 37.7496C52.083 33.0769 43.9975 30.5042 36.1746 30.5042H21.8938V41.0048H36.2796C42.2649 41.0048 48.1978 42.9999 52.923 46.6226L58.5934 50.9279L64.2637 46.6226C70.144 42.1599 77.5469 40.2698 84.7923 41.2673V76.1818C75.5518 75.2367 66.2063 77.7044 58.6459 83.2172C51.0854 77.7044 41.6349 75.2367 32.4994 76.1818V52.8705H21.9988V86.4724H95.3454V33.1294H95.1879Z"
|
|
||||||
fill="#58FF9B"
|
|
||||||
/>
|
|
||||||
<defs>
|
|
||||||
<radialGradient
|
|
||||||
id="paint0_radial_174_336"
|
|
||||||
cx="0"
|
|
||||||
cy="0"
|
|
||||||
r="1"
|
|
||||||
gradientUnits="userSpaceOnUse"
|
|
||||||
gradientTransform="translate(117.24 -101.5) rotate(105.042) scale(226.282)"
|
|
||||||
>
|
|
||||||
<stop offset="0.445162" stop-color="#1F7676" />
|
|
||||||
<stop offset="1" stop-color="#0A4B4B" />
|
|
||||||
</radialGradient>
|
|
||||||
</defs>
|
|
||||||
</svg>
|
</svg>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -1,41 +1,37 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="flex space-x-4 border rounded-md p-2">
|
<div class="border rounded-md p-4">
|
||||||
<img :src="job.company_logo" class="size-10 rounded-full object-contain" />
|
<div class="flex space-x-4">
|
||||||
<div class="flex flex-col space-y-2 flex-1">
|
<img
|
||||||
<div class="flex items-center justify-between">
|
:src="job.company_logo"
|
||||||
<span class="font-semibold text-ink-gray-9">
|
class="size-10 rounded-full object-contain"
|
||||||
{{ job.job_title }}
|
/>
|
||||||
</span>
|
<div class="flex flex-col space-y-1 flex-1">
|
||||||
</div>
|
<div class="flex items-center justify-between">
|
||||||
<div class="flex items-center space-x-2 text-ink-gray-5">
|
<span class="text-lg font-semibold text-ink-gray-9">
|
||||||
<Building2 class="w-4 h-4 stroke-1.5" />
|
{{ job.job_title }}
|
||||||
<span>
|
</span>
|
||||||
|
</div>
|
||||||
|
<div class="text-xs text-ink-gray-5">
|
||||||
{{ job.company_name }}
|
{{ job.company_name }}
|
||||||
</span>
|
</div>
|
||||||
</div>
|
|
||||||
<div class="flex items-center space-x-2 text-ink-gray-5">
|
|
||||||
<MapPin class="w-4 h-4 stroke-1.5" />
|
|
||||||
<span>
|
|
||||||
{{ job.location }}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div class="flex items-center space-x-2 text-ink-gray-5">
|
|
||||||
<Shapes class="w-4 h-4 stroke-1.5" />
|
|
||||||
<span>
|
|
||||||
{{ job.type }}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div class="flex items-center space-x-2 text-ink-gray-5">
|
|
||||||
<Calendar class="w-4 h-4 stroke-1.5" />
|
|
||||||
<span> {{ __('posted') }} {{ dayjs(job.creation).fromNow() }} </span>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="space-x-4 mt-2">
|
||||||
|
<Badge>
|
||||||
|
{{ job.location }}
|
||||||
|
</Badge>
|
||||||
|
<Badge>
|
||||||
|
{{ job.type }}
|
||||||
|
</Badge>
|
||||||
|
<Badge>
|
||||||
|
{{ dayjs(job.creation).fromNow() }}
|
||||||
|
</Badge>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { Building2, Calendar, MapPin, Shapes } from 'lucide-vue-next'
|
|
||||||
import { inject } from 'vue'
|
import { inject } from 'vue'
|
||||||
import { Avatar } from 'frappe-ui'
|
import { Badge } from 'frappe-ui'
|
||||||
|
|
||||||
const dayjs = inject('$dayjs')
|
const dayjs = inject('$dayjs')
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
|
|||||||
@@ -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,
|
||||||
|
|||||||
@@ -116,6 +116,24 @@ 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'
|
||||||
|
|
||||||
|
interface User {
|
||||||
|
data: {
|
||||||
|
email: string
|
||||||
|
name: string
|
||||||
|
enabled: boolean
|
||||||
|
user_image: string
|
||||||
|
full_name: string
|
||||||
|
user_type: ['System User', 'Website User']
|
||||||
|
username: string
|
||||||
|
is_moderator: boolean
|
||||||
|
is_system_manager: boolean
|
||||||
|
is_evaluator: boolean
|
||||||
|
is_instructor: boolean
|
||||||
|
is_fc_site: boolean
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
const show = defineModel('show')
|
const show = defineModel('show')
|
||||||
@@ -125,6 +143,8 @@ 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 user = inject<User | null>('$user')
|
||||||
|
const { updateOnboardingStep } = useOnboarding('learning')
|
||||||
|
|
||||||
const member = reactive({
|
const member = reactive({
|
||||||
email: '',
|
email: '',
|
||||||
@@ -185,6 +205,9 @@ const newMember = createResource({
|
|||||||
auto: false,
|
auto: false,
|
||||||
onSuccess(data) {
|
onSuccess(data) {
|
||||||
show.value = false
|
show.value = false
|
||||||
|
|
||||||
|
if (user?.data?.is_system_manager) updateOnboardingStep('invite_students')
|
||||||
|
|
||||||
router.push({
|
router.push({
|
||||||
name: 'Profile',
|
name: 'Profile',
|
||||||
params: {
|
params: {
|
||||||
|
|||||||
@@ -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>
|
||||||
@@ -31,14 +32,19 @@
|
|||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { Dialog, createResource } from 'frappe-ui'
|
import { Dialog, createResource } from 'frappe-ui'
|
||||||
import { ref } from 'vue'
|
import { ref, inject } 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 user = inject('$user')
|
||||||
const courses = defineModel('courses')
|
const courses = defineModel('courses')
|
||||||
|
const { updateOnboardingStep } = useOnboarding('learning')
|
||||||
|
const settingsStore = useSettings()
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
batch: {
|
batch: {
|
||||||
@@ -68,8 +74,11 @@ const addCourse = (close) => {
|
|||||||
{},
|
{},
|
||||||
{
|
{
|
||||||
onSuccess() {
|
onSuccess() {
|
||||||
courses.value.reload()
|
if (user.data?.is_system_manager)
|
||||||
|
updateOnboardingStep('add_batch_course')
|
||||||
|
|
||||||
close()
|
close()
|
||||||
|
courses.value.reload()
|
||||||
course.value = null
|
course.value = null
|
||||||
evaluator.value = null
|
evaluator.value = null
|
||||||
},
|
},
|
||||||
@@ -79,4 +88,10 @@ const addCourse = (close) => {
|
|||||||
}
|
}
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const openSettings = (close) => {
|
||||||
|
close()
|
||||||
|
settingsStore.activeTab = 'Evaluators'
|
||||||
|
settingsStore.isSettingsOpen = true
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -77,15 +77,16 @@ import {
|
|||||||
FormControl,
|
FormControl,
|
||||||
Switch,
|
Switch,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import { reactive, watch } from 'vue'
|
import { reactive, watch, inject } 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 user = inject('$user')
|
||||||
|
const { updateOnboardingStep } = useOnboarding('learning')
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
course: {
|
course: {
|
||||||
@@ -139,15 +140,15 @@ const addChapter = async (close) => {
|
|||||||
return validateChapter()
|
return validateChapter()
|
||||||
},
|
},
|
||||||
onSuccess: (data) => {
|
onSuccess: (data) => {
|
||||||
|
if (user.data?.is_system_manager)
|
||||||
|
updateOnboardingStep('create_first_chapter')
|
||||||
|
|
||||||
capture('chapter_created')
|
capture('chapter_created')
|
||||||
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'),
|
||||||
|
|||||||
@@ -1,94 +0,0 @@
|
|||||||
<template>
|
|
||||||
<Dialog
|
|
||||||
v-model="show"
|
|
||||||
:options="{
|
|
||||||
size: 'xl',
|
|
||||||
title: __('Login to Frappe Cloud'),
|
|
||||||
actions: [
|
|
||||||
{
|
|
||||||
label: __('Verify'),
|
|
||||||
variant: 'solid',
|
|
||||||
onClick: (close) => {
|
|
||||||
verifyCode(close)
|
|
||||||
},
|
|
||||||
},
|
|
||||||
],
|
|
||||||
}"
|
|
||||||
>
|
|
||||||
<template #body-content>
|
|
||||||
<div>
|
|
||||||
<p>
|
|
||||||
{{ __('We have sent the verificaton code to your email id ') }}
|
|
||||||
<b>{{ props.email }}</b>
|
|
||||||
</p>
|
|
||||||
<FormControl
|
|
||||||
v-model="code"
|
|
||||||
:label="__('Verification Code')"
|
|
||||||
class="mb-4"
|
|
||||||
/>
|
|
||||||
<p>
|
|
||||||
{{ __("Didn't receive the code?") }}
|
|
||||||
<a href="#" @click="resendCode">{{ __('Resend') }}</a>
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
</Dialog>
|
|
||||||
</template>
|
|
||||||
<script setup>
|
|
||||||
import { call, Dialog } from 'frappe-ui'
|
|
||||||
import { showToast } from '@/utils'
|
|
||||||
|
|
||||||
const show = defineModel()
|
|
||||||
const code = ref('')
|
|
||||||
|
|
||||||
const props = defineProps({
|
|
||||||
email: {
|
|
||||||
type: String,
|
|
||||||
required: true,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
|
|
||||||
const verifyCode = (close) => {
|
|
||||||
if (!code.value) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
call(
|
|
||||||
'frappe.integrations.frappe_providers.frappecloud_billing.verify_verification_code',
|
|
||||||
{
|
|
||||||
verification_code: code.value,
|
|
||||||
route: window.route,
|
|
||||||
}
|
|
||||||
)
|
|
||||||
.then((data) => {
|
|
||||||
if (data.message.login_token) {
|
|
||||||
close()
|
|
||||||
window.open(
|
|
||||||
`${frappeCloudBaseEndpoint}/api/method/press.api.developer.saas.login_to_fc?token=${data.message.login_token}`,
|
|
||||||
'_blank'
|
|
||||||
)
|
|
||||||
showToast(
|
|
||||||
__('Frappe Cloud Login Successful'),
|
|
||||||
`<p>${__('You will be redirected to Frappe Cloud soon.')}</p><p>${__(
|
|
||||||
"If you haven't been redirected,"
|
|
||||||
)} <a href="${frappeCloudBaseEndpoint}/api/method/press.api.developer.saas.login_to_fc?token=${
|
|
||||||
data.message.login_token
|
|
||||||
}" target="_blank">${__('Click here to login')}</a></p>`,
|
|
||||||
'check'
|
|
||||||
)
|
|
||||||
} else {
|
|
||||||
showToast(__('Login failed'), __('Please try again'), 'x')
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.catch((err) => {
|
|
||||||
showToast(__('Login failed'), __('Please try again'), 'x')
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
const resendCode = () => {
|
|
||||||
call(
|
|
||||||
'frappe.integrations.frappe_providers.frappecloud_billing.send_verification_code'
|
|
||||||
).catch((err) => {
|
|
||||||
showToast(__('Failed to resend code'), __('Please try again'), 'x')
|
|
||||||
})
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
@@ -39,13 +39,19 @@
|
|||||||
:required="true"
|
:required="true"
|
||||||
/>
|
/>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
<FormControl
|
|
||||||
v-model="liveClass.timezone"
|
<div class="space-y-1.5">
|
||||||
type="select"
|
<label class="block text-ink-gray-5 text-xs" for="batchTimezone">
|
||||||
:options="getTimezoneOptions()"
|
{{ __('Timezone') }}
|
||||||
:label="__('Timezone')"
|
<span class="text-ink-red-3">*</span>
|
||||||
:required="true"
|
</label>
|
||||||
/>
|
<Autocomplete
|
||||||
|
@update:modelValue="(opt) => (liveClass.timezone = opt.value)"
|
||||||
|
:modelValue="liveClass.timezone"
|
||||||
|
:options="getTimezoneOptions()"
|
||||||
|
:required="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<FormControl
|
<FormControl
|
||||||
@@ -83,18 +89,14 @@
|
|||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import {
|
import {
|
||||||
Input,
|
|
||||||
DatePicker,
|
|
||||||
Select,
|
|
||||||
Textarea,
|
|
||||||
Dialog,
|
Dialog,
|
||||||
createResource,
|
createResource,
|
||||||
Tooltip,
|
Tooltip,
|
||||||
FormControl,
|
FormControl,
|
||||||
|
Autocomplete,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import { reactive, inject } from 'vue'
|
import { reactive, inject, onMounted } from 'vue'
|
||||||
import { getTimezones, createToast } from '@/utils/'
|
import { getTimezones, createToast, getUserTimezone } from '@/utils/'
|
||||||
import { Info } from 'lucide-vue-next'
|
|
||||||
|
|
||||||
const liveClasses = defineModel('reloadLiveClasses')
|
const liveClasses = defineModel('reloadLiveClasses')
|
||||||
const show = defineModel()
|
const show = defineModel()
|
||||||
@@ -120,6 +122,10 @@ let liveClass = reactive({
|
|||||||
host: user.data.name,
|
host: user.data.name,
|
||||||
})
|
})
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
liveClass.timezone = getUserTimezone()
|
||||||
|
})
|
||||||
|
|
||||||
const getTimezoneOptions = () => {
|
const getTimezoneOptions = () => {
|
||||||
return getTimezones().map((timezone) => {
|
return getTimezones().map((timezone) => {
|
||||||
return {
|
return {
|
||||||
|
|||||||
@@ -106,23 +106,26 @@
|
|||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { Dialog, FormControl, TextEditor, createResource } from 'frappe-ui'
|
import { Dialog, FormControl, TextEditor, createResource } from 'frappe-ui'
|
||||||
import { computed, watch, reactive, ref } from 'vue'
|
import { computed, watch, reactive, ref, inject } 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 user = inject('$user')
|
||||||
|
const { updateOnboardingStep } = useOnboarding('learning')
|
||||||
|
|
||||||
const existingQuestion = reactive({
|
const existingQuestion = reactive({
|
||||||
question: '',
|
question: '',
|
||||||
marks: 0,
|
marks: 1,
|
||||||
})
|
})
|
||||||
const question = reactive({
|
const question = reactive({
|
||||||
question: '',
|
question: '',
|
||||||
type: 'Choices',
|
type: 'Choices',
|
||||||
marks: 0,
|
marks: 1,
|
||||||
})
|
})
|
||||||
|
|
||||||
const populateFields = () => {
|
const populateFields = () => {
|
||||||
@@ -260,6 +263,9 @@ const addQuestionRow = (question, close) => {
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
onSuccess() {
|
onSuccess() {
|
||||||
|
if (user.data?.is_system_manager)
|
||||||
|
updateOnboardingStep('create_first_quiz')
|
||||||
|
|
||||||
show.value = false
|
show.value = false
|
||||||
showToast(__('Success'), __('Question added successfully'), 'check')
|
showToast(__('Success'), __('Question added successfully'), 'check')
|
||||||
quiz.value.reload()
|
quiz.value.reload()
|
||||||
|
|||||||
@@ -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',
|
||||||
@@ -316,19 +328,26 @@ const tabsStructure = computed(() => {
|
|||||||
icon: 'LogIn',
|
icon: 'LogIn',
|
||||||
fields: [
|
fields: [
|
||||||
{
|
{
|
||||||
label: 'Custom Content',
|
label: 'Identify User Persona',
|
||||||
|
name: 'user_category',
|
||||||
|
type: 'checkbox',
|
||||||
|
description:
|
||||||
|
'Enable this option to identify the user persona during signup.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Disable signup',
|
||||||
|
name: 'disable_signup',
|
||||||
|
type: 'checkbox',
|
||||||
|
description:
|
||||||
|
'New users will have to be manually registered by Admins.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Signup Consent HTML',
|
||||||
name: 'custom_signup_content',
|
name: 'custom_signup_content',
|
||||||
type: 'Code',
|
type: 'Code',
|
||||||
mode: 'htmlmixed',
|
mode: 'htmlmixed',
|
||||||
rows: 10,
|
rows: 10,
|
||||||
},
|
},
|
||||||
{
|
|
||||||
label: 'Ask for Occupation',
|
|
||||||
name: 'user_category',
|
|
||||||
type: 'checkbox',
|
|
||||||
description:
|
|
||||||
'Enable this option to ask users to select their occupation during the signup process.',
|
|
||||||
},
|
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
|
|||||||
@@ -26,12 +26,15 @@
|
|||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { Dialog, createResource } from 'frappe-ui'
|
import { Dialog, createResource } from 'frappe-ui'
|
||||||
import { ref } from 'vue'
|
import { ref, inject } 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 user = inject('$user')
|
||||||
|
const { updateOnboardingStep } = useOnboarding('learning')
|
||||||
const show = defineModel()
|
const show = defineModel()
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
@@ -59,6 +62,9 @@ const addStudent = (close) => {
|
|||||||
{},
|
{},
|
||||||
{
|
{
|
||||||
onSuccess() {
|
onSuccess() {
|
||||||
|
if (user.data?.is_system_manager)
|
||||||
|
updateOnboardingStep('add_batch_student')
|
||||||
|
|
||||||
students.value.reload()
|
students.value.reload()
|
||||||
student.value = null
|
student.value = null
|
||||||
close()
|
close()
|
||||||
|
|||||||
11
frontend/src/components/NoSidebarLayout.vue
Normal file
11
frontend/src/components/NoSidebarLayout.vue
Normal 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>
|
||||||
@@ -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-3"
|
||||||
>
|
>
|
||||||
<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,
|
||||||
|
|||||||
@@ -99,7 +99,7 @@
|
|||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { FormControl, FileUploader, Button, Switch } from 'frappe-ui'
|
import { FormControl, FileUploader, Button, Switch } from 'frappe-ui'
|
||||||
import { computed } from 'vue'
|
import { computed, onMounted } from 'vue'
|
||||||
import { getFileSize, validateFile } from '@/utils'
|
import { getFileSize, validateFile } from '@/utils'
|
||||||
import { X } from 'lucide-vue-next'
|
import { X } from 'lucide-vue-next'
|
||||||
import Link from '@/components/Controls/Link.vue'
|
import Link from '@/components/Controls/Link.vue'
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
@@ -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')
|
||||||
|
|||||||
@@ -36,7 +36,7 @@
|
|||||||
<span v-else> Learning </span>
|
<span v-else> Learning </span>
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
v-if="userResource"
|
v-if="userResource.data"
|
||||||
class="mt-1 text-sm text-ink-gray-7 leading-none"
|
class="mt-1 text-sm text-ink-gray-7 leading-none"
|
||||||
>
|
>
|
||||||
{{ convertToTitleCase(userResource.data?.full_name) }}
|
{{ convertToTitleCase(userResource.data?.full_name) }}
|
||||||
@@ -59,22 +59,21 @@
|
|||||||
v-if="userResource.data?.is_moderator"
|
v-if="userResource.data?.is_moderator"
|
||||||
v-model="showSettingsModal"
|
v-model="showSettingsModal"
|
||||||
/>
|
/>
|
||||||
<FCVerfiyCodeModal v-if="showFCLoginDialog" :email="verificationEmail" />
|
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
import LMSLogo from '@/components/Icons/LMSLogo.vue'
|
import LMSLogo from '@/components/Icons/LMSLogo.vue'
|
||||||
import { sessionStore } from '@/stores/session'
|
import { sessionStore } from '@/stores/session'
|
||||||
import { call, Dropdown } from 'frappe-ui'
|
import { Dropdown } from 'frappe-ui'
|
||||||
import Apps from '@/components/Apps.vue'
|
import Apps from '@/components/Apps.vue'
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
import { convertToTitleCase, showToast } from '@/utils'
|
import { convertToTitleCase } from '@/utils'
|
||||||
import { usersStore } from '@/stores/user'
|
import { usersStore } from '@/stores/user'
|
||||||
import { useSettings } from '@/stores/settings'
|
import { useSettings } from '@/stores/settings'
|
||||||
import { markRaw, watch, ref, onMounted, computed } from 'vue'
|
import { markRaw, watch, ref, onMounted, computed } from 'vue'
|
||||||
import SettingsModal from '@/components/Modals/Settings.vue'
|
|
||||||
import { createDialog } from '@/utils/dialogs'
|
import { createDialog } from '@/utils/dialogs'
|
||||||
import FCVerfiyCodeModal from './Modals/FCVerfiyCodeModal.vue'
|
import SettingsModal from '@/components/Modals/Settings.vue'
|
||||||
|
import FrappeCloudIcon from '@/components/Icons/FrappeCloudIcon.vue'
|
||||||
import {
|
import {
|
||||||
ChevronDown,
|
ChevronDown,
|
||||||
LogIn,
|
LogIn,
|
||||||
@@ -83,7 +82,7 @@ import {
|
|||||||
User,
|
User,
|
||||||
Settings,
|
Settings,
|
||||||
Sun,
|
Sun,
|
||||||
LogInIcon,
|
Zap,
|
||||||
} from 'lucide-vue-next'
|
} from 'lucide-vue-next'
|
||||||
|
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
@@ -93,11 +92,8 @@ const settingsStore = useSettings()
|
|||||||
let { isLoggedIn } = sessionStore()
|
let { isLoggedIn } = sessionStore()
|
||||||
const showSettingsModal = ref(false)
|
const showSettingsModal = ref(false)
|
||||||
const theme = ref('light')
|
const theme = ref('light')
|
||||||
const $dialog = createDialog
|
|
||||||
const frappeCloudBaseEndpoint = 'https://frappecloud.com'
|
const frappeCloudBaseEndpoint = 'https://frappecloud.com'
|
||||||
|
const $dialog = createDialog
|
||||||
const showFCLoginDialog = ref(false)
|
|
||||||
const verificationEmail = ref(null)
|
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
isCollapsed: {
|
isCollapsed: {
|
||||||
@@ -130,120 +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: LogInIcon,
|
},
|
||||||
label: 'Login to Frappe Cloud',
|
condition: () => {
|
||||||
onClick: () => {
|
return !isLoggedIn
|
||||||
initiateRequestForLoginToFrappeCloud()
|
},
|
||||||
},
|
},
|
||||||
condition: () => {
|
],
|
||||||
return (
|
|
||||||
userResource.data?.user_type == 'System User' &&
|
|
||||||
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 initiateRequestForLoginToFrappeCloud = () => {
|
const loginToFrappeCloud = () => {
|
||||||
$dialog({
|
let redirect_to = '/dashboard/sites/' + userResource.data.sitename
|
||||||
title: __('Login to Frappe Cloud?'),
|
window.open(`${frappeCloudBaseEndpoint}${redirect_to}`, '_blank')
|
||||||
message: __(
|
|
||||||
'Are you sure you want to login to your Frappe Cloud dashboard?'
|
|
||||||
),
|
|
||||||
actions: [
|
|
||||||
{
|
|
||||||
label: __('Confirm'),
|
|
||||||
variant: 'solid',
|
|
||||||
onClick(close) {
|
|
||||||
requestLoginToFC()
|
|
||||||
close()
|
|
||||||
},
|
|
||||||
},
|
|
||||||
],
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
const requestLoginToFC = () => {
|
|
||||||
call(
|
|
||||||
'frappe.integrations.frappe_providers.frappecloud_billing.send_verification_code'
|
|
||||||
)
|
|
||||||
.then((data) => {
|
|
||||||
if (data.message.is_user_logged_in) {
|
|
||||||
window.open(
|
|
||||||
`${frappeCloudBaseEndpoint}${data.message.redirect_to}`,
|
|
||||||
'_blank'
|
|
||||||
)
|
|
||||||
return
|
|
||||||
} else {
|
|
||||||
showFCLoginDialog.value = true
|
|
||||||
verificationEmail.value = data.message.email
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.catch((err) => {
|
|
||||||
showToast(
|
|
||||||
__('Failed to login to Frappe Cloud'),
|
|
||||||
__('Please try again'),
|
|
||||||
'x'
|
|
||||||
)
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -63,6 +63,7 @@ import {
|
|||||||
createResource,
|
createResource,
|
||||||
FormControl,
|
FormControl,
|
||||||
TextEditor,
|
TextEditor,
|
||||||
|
usePageMeta,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import {
|
import {
|
||||||
computed,
|
computed,
|
||||||
@@ -72,11 +73,13 @@ import {
|
|||||||
reactive,
|
reactive,
|
||||||
watch,
|
watch,
|
||||||
} from 'vue'
|
} from 'vue'
|
||||||
|
import { sessionStore } from '../stores/session'
|
||||||
import { showToast } from '@/utils'
|
import { showToast } from '@/utils'
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
|
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
|
const { brand } = sessionStore()
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
assignmentID: {
|
assignmentID: {
|
||||||
@@ -188,4 +191,11 @@ const assignmentOptions = computed(() => {
|
|||||||
{ label: 'URL', value: 'URL' },
|
{ label: 'URL', value: 'URL' },
|
||||||
]
|
]
|
||||||
})
|
})
|
||||||
|
|
||||||
|
usePageMeta(() => {
|
||||||
|
return {
|
||||||
|
title: assignment.doc ? assignment.doc.title : __('New Assignment'),
|
||||||
|
icon: brand.favicon,
|
||||||
|
}
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -1,19 +1,27 @@
|
|||||||
<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, usePageMeta } from 'frappe-ui'
|
||||||
import { computed, inject, onMounted } from 'vue'
|
import { computed, inject, onMounted, ref } from 'vue'
|
||||||
|
import { sessionStore } from '../stores/session'
|
||||||
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 { brand } = sessionStore()
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
assignmentID: {
|
assignmentID: {
|
||||||
@@ -42,6 +50,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(() => {
|
||||||
@@ -62,4 +74,11 @@ const breadcrumbs = computed(() => {
|
|||||||
]
|
]
|
||||||
return crumbs
|
return crumbs
|
||||||
})
|
})
|
||||||
|
|
||||||
|
usePageMeta(() => {
|
||||||
|
return {
|
||||||
|
title: title.data?.title,
|
||||||
|
icon: brand.favicon,
|
||||||
|
}
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -84,14 +84,17 @@ import {
|
|||||||
ListRows,
|
ListRows,
|
||||||
ListRow,
|
ListRow,
|
||||||
ListRowItem,
|
ListRowItem,
|
||||||
|
usePageMeta,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import { computed, inject, onMounted, ref, watch } from 'vue'
|
import { computed, inject, onMounted, ref, watch } from 'vue'
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
import { Pencil } from 'lucide-vue-next'
|
import { Pencil } from 'lucide-vue-next'
|
||||||
|
import { sessionStore } from '../stores/session'
|
||||||
import Link from '@/components/Controls/Link.vue'
|
import Link from '@/components/Controls/Link.vue'
|
||||||
|
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
const dayjs = inject('$dayjs')
|
const dayjs = inject('$dayjs')
|
||||||
|
const { brand } = sessionStore()
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
const assignmentID = ref('')
|
const assignmentID = ref('')
|
||||||
const member = ref('')
|
const member = ref('')
|
||||||
@@ -214,4 +217,11 @@ const breadcrumbs = computed(() => {
|
|||||||
},
|
},
|
||||||
]
|
]
|
||||||
})
|
})
|
||||||
|
|
||||||
|
usePageMeta(() => {
|
||||||
|
return {
|
||||||
|
title: __('Assignment Submissions'),
|
||||||
|
icon: brand.favicon,
|
||||||
|
}
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -80,15 +80,18 @@ import {
|
|||||||
createListResource,
|
createListResource,
|
||||||
FormControl,
|
FormControl,
|
||||||
ListView,
|
ListView,
|
||||||
|
usePageMeta,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import { computed, inject, onMounted, ref, watch } from 'vue'
|
import { computed, inject, onMounted, ref, watch } from 'vue'
|
||||||
import { Plus, Pencil } from 'lucide-vue-next'
|
import { Plus, Pencil } from 'lucide-vue-next'
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
|
import { sessionStore } from '../stores/session'
|
||||||
|
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
const dayjs = inject('$dayjs')
|
const dayjs = inject('$dayjs')
|
||||||
const titleFilter = ref('')
|
const titleFilter = ref('')
|
||||||
const typeFilter = ref('')
|
const typeFilter = ref('')
|
||||||
|
const { brand } = sessionStore()
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
@@ -184,4 +187,11 @@ const breadcrumbs = computed(() => [
|
|||||||
route: { name: 'Assignments' },
|
route: { name: 'Assignments' },
|
||||||
},
|
},
|
||||||
])
|
])
|
||||||
|
|
||||||
|
usePageMeta(() => {
|
||||||
|
return {
|
||||||
|
title: __('Assignments'),
|
||||||
|
icon: brand.favicon,
|
||||||
|
}
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -24,10 +24,12 @@
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { createDocumentResource, createResource } from 'frappe-ui'
|
import { createResource, usePageMeta } from 'frappe-ui'
|
||||||
import { computed, inject } from 'vue'
|
import { computed, inject } from 'vue'
|
||||||
|
import { sessionStore } from '../stores/session'
|
||||||
|
|
||||||
const dayjs = inject('$dayjs')
|
const dayjs = inject('$dayjs')
|
||||||
|
const { brand } = sessionStore()
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
badgeName: {
|
badgeName: {
|
||||||
@@ -70,4 +72,11 @@ const breadcrumbs = computed(() => {
|
|||||||
},
|
},
|
||||||
]
|
]
|
||||||
})
|
})
|
||||||
|
|
||||||
|
usePageMeta(() => {
|
||||||
|
return {
|
||||||
|
title: badge.data.badge,
|
||||||
|
icon: brand.favicon,
|
||||||
|
}
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -190,14 +190,23 @@
|
|||||||
</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 {
|
||||||
import CourseInstructors from '@/components/CourseInstructors.vue'
|
Breadcrumbs,
|
||||||
import UserAvatar from '@/components/UserAvatar.vue'
|
Button,
|
||||||
|
createResource,
|
||||||
|
Tabs,
|
||||||
|
Badge,
|
||||||
|
usePageMeta,
|
||||||
|
} from 'frappe-ui'
|
||||||
import {
|
import {
|
||||||
Clock,
|
Clock,
|
||||||
LayoutDashboard,
|
LayoutDashboard,
|
||||||
@@ -210,7 +219,10 @@ import {
|
|||||||
Globe,
|
Globe,
|
||||||
ClipboardPen,
|
ClipboardPen,
|
||||||
} from 'lucide-vue-next'
|
} from 'lucide-vue-next'
|
||||||
import { formatTime, updateDocumentTitle } from '@/utils'
|
import { formatTime } from '@/utils'
|
||||||
|
import { sessionStore } from '@/stores/session'
|
||||||
|
import CourseInstructors from '@/components/CourseInstructors.vue'
|
||||||
|
import UserAvatar from '@/components/UserAvatar.vue'
|
||||||
import BatchDashboard from '@/components/BatchDashboard.vue'
|
import BatchDashboard from '@/components/BatchDashboard.vue'
|
||||||
import BatchCourses from '@/components/BatchCourses.vue'
|
import BatchCourses from '@/components/BatchCourses.vue'
|
||||||
import LiveClass from '@/components/LiveClass.vue'
|
import LiveClass from '@/components/LiveClass.vue'
|
||||||
@@ -226,52 +238,11 @@ 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 { brand } = sessionStore()
|
||||||
|
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)
|
|
||||||
const tabs = computed(() => {
|
const tabs = computed(() => {
|
||||||
let batchTabs = []
|
let batchTabs = []
|
||||||
batchTabs.push({
|
batchTabs.push({
|
||||||
@@ -313,6 +284,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,12 +347,17 @@ const openAnnouncementModal = () => {
|
|||||||
showAnnouncementModal.value = true
|
showAnnouncementModal.value = true
|
||||||
}
|
}
|
||||||
|
|
||||||
const pageMeta = computed(() => {
|
watch(tabIndex, () => {
|
||||||
return {
|
const tab = tabs.value[tabIndex.value]
|
||||||
title: batch.data?.title,
|
if (tab.label != route.hash.replace('#', '')) {
|
||||||
description: batch.data?.description,
|
router.push({ ...route, hash: `#${tab.label.toLowerCase()}` })
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
updateDocumentTitle(pageMeta)
|
usePageMeta(() => {
|
||||||
|
return {
|
||||||
|
title: batch?.data?.title,
|
||||||
|
icon: brand.favicon,
|
||||||
|
}
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -102,8 +102,9 @@
|
|||||||
import { computed, inject } from 'vue'
|
import { computed, inject } from 'vue'
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
import { BookOpen, Clock } from 'lucide-vue-next'
|
import { BookOpen, Clock } from 'lucide-vue-next'
|
||||||
import { formatTime, updateDocumentTitle } from '@/utils'
|
import { formatTime } from '@/utils'
|
||||||
import { Breadcrumbs, createResource } from 'frappe-ui'
|
import { Breadcrumbs, createResource, usePageMeta } from 'frappe-ui'
|
||||||
|
import { sessionStore } from '@/stores/session'
|
||||||
import CourseCard from '@/components/CourseCard.vue'
|
import CourseCard from '@/components/CourseCard.vue'
|
||||||
import BatchOverlay from '@/components/BatchOverlay.vue'
|
import BatchOverlay from '@/components/BatchOverlay.vue'
|
||||||
import DateRange from '../components/Common/DateRange.vue'
|
import DateRange from '../components/Common/DateRange.vue'
|
||||||
@@ -112,6 +113,7 @@ import UserAvatar from '@/components/UserAvatar.vue'
|
|||||||
|
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
|
const { brand } = sessionStore()
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
batchName: {
|
batchName: {
|
||||||
@@ -152,14 +154,12 @@ const breadcrumbs = computed(() => {
|
|||||||
return items
|
return items
|
||||||
})
|
})
|
||||||
|
|
||||||
const pageMeta = computed(() => {
|
usePageMeta(() => {
|
||||||
return {
|
return {
|
||||||
title: batch.data?.title,
|
title: batch?.data?.title,
|
||||||
description: batch.data?.description,
|
icon: brand.favicon,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
updateDocumentTitle(pageMeta)
|
|
||||||
</script>
|
</script>
|
||||||
<style>
|
<style>
|
||||||
.batch-description p {
|
.batch-description p {
|
||||||
|
|||||||
@@ -264,16 +264,21 @@ import {
|
|||||||
Button,
|
Button,
|
||||||
TextEditor,
|
TextEditor,
|
||||||
createResource,
|
createResource,
|
||||||
|
usePageMeta,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import Link from '@/components/Controls/Link.vue'
|
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
import { showToast } from '@/utils'
|
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 { useOnboarding } from 'frappe-ui/frappe'
|
||||||
|
import { sessionStore } from '../stores/session'
|
||||||
import MultiSelect from '@/components/Controls/MultiSelect.vue'
|
import MultiSelect from '@/components/Controls/MultiSelect.vue'
|
||||||
|
import Link from '@/components/Controls/Link.vue'
|
||||||
|
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
|
const { brand } = sessionStore()
|
||||||
|
const { updateOnboardingStep } = useOnboarding('learning')
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
batchName: {
|
batchName: {
|
||||||
@@ -425,6 +430,12 @@ const createNewBatch = () => {
|
|||||||
{},
|
{},
|
||||||
{
|
{
|
||||||
onSuccess(data) {
|
onSuccess(data) {
|
||||||
|
if (user.data?.is_system_manager) {
|
||||||
|
updateOnboardingStep('create_first_batch', true, false, () => {
|
||||||
|
localStorage.setItem('firstBatch', data.name)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
capture('batch_created')
|
capture('batch_created')
|
||||||
router.push({
|
router.push({
|
||||||
name: 'BatchDetail',
|
name: 'BatchDetail',
|
||||||
@@ -500,4 +511,11 @@ const breadcrumbs = computed(() => {
|
|||||||
})
|
})
|
||||||
return crumbs
|
return crumbs
|
||||||
})
|
})
|
||||||
|
|
||||||
|
usePageMeta(() => {
|
||||||
|
return {
|
||||||
|
title: props.batchName == 'new' ? 'New Batch' : batchDetail.data?.title,
|
||||||
|
icon: brand.favicon,
|
||||||
|
}
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -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">
|
||||||
@@ -104,14 +104,16 @@ import {
|
|||||||
FormControl,
|
FormControl,
|
||||||
Select,
|
Select,
|
||||||
TabButtons,
|
TabButtons,
|
||||||
|
usePageMeta,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import { computed, inject, onMounted, ref, watch } from 'vue'
|
import { computed, inject, onMounted, ref, watch } from 'vue'
|
||||||
import { BookOpen, Plus } from 'lucide-vue-next'
|
import { BookOpen, Plus } from 'lucide-vue-next'
|
||||||
import { updateDocumentTitle } from '@/utils'
|
import { sessionStore } from '@/stores/session'
|
||||||
import BatchCard from '@/components/BatchCard.vue'
|
import BatchCard from '@/components/BatchCard.vue'
|
||||||
|
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
const dayjs = inject('$dayjs')
|
const dayjs = inject('$dayjs')
|
||||||
|
const { brand } = sessionStore()
|
||||||
const start = ref(0)
|
const start = ref(0)
|
||||||
const pageLength = ref(20)
|
const pageLength = ref(20)
|
||||||
const categories = ref([])
|
const categories = ref([])
|
||||||
@@ -119,7 +121,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 +207,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 +231,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 +253,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 +278,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
|
||||||
})
|
})
|
||||||
@@ -305,12 +306,10 @@ const breadcrumbs = computed(() => [
|
|||||||
},
|
},
|
||||||
])
|
])
|
||||||
|
|
||||||
const pageMeta = computed(() => {
|
usePageMeta(() => {
|
||||||
return {
|
return {
|
||||||
title: 'Batches',
|
title: __('Batches'),
|
||||||
description: 'All upcoming batches.',
|
icon: brand.favicon,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
updateDocumentTitle(pageMeta)
|
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -151,19 +151,20 @@
|
|||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import {
|
import {
|
||||||
Input,
|
|
||||||
Button,
|
Button,
|
||||||
createResource,
|
createResource,
|
||||||
FormControl,
|
FormControl,
|
||||||
Breadcrumbs,
|
Breadcrumbs,
|
||||||
Tooltip,
|
usePageMeta,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import { reactive, inject, onMounted, computed } from 'vue'
|
import { reactive, inject, onMounted, computed } from 'vue'
|
||||||
|
import { showToast } from '@/utils/'
|
||||||
|
import { sessionStore } from '../stores/session'
|
||||||
import Link from '@/components/Controls/Link.vue'
|
import Link from '@/components/Controls/Link.vue'
|
||||||
import NotPermitted from '@/components/NotPermitted.vue'
|
import NotPermitted from '@/components/NotPermitted.vue'
|
||||||
import { showToast } from '@/utils/'
|
|
||||||
|
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
|
const { brand } = sessionStore()
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
const script = document.createElement('script')
|
const script = document.createElement('script')
|
||||||
@@ -245,12 +246,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.')
|
||||||
}
|
}
|
||||||
@@ -358,4 +357,11 @@ const redirectTo = computed(() => {
|
|||||||
return `/lms/courses/${props.name}/certification`
|
return `/lms/courses/${props.name}/certification`
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
usePageMeta(() => {
|
||||||
|
return {
|
||||||
|
title: __('Billing Details'),
|
||||||
|
icon: brand.favicon,
|
||||||
|
}
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -102,14 +102,17 @@ import {
|
|||||||
createListResource,
|
createListResource,
|
||||||
FormControl,
|
FormControl,
|
||||||
Select,
|
Select,
|
||||||
|
usePageMeta,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import { computed, onMounted, ref } from 'vue'
|
import { computed, onMounted, ref } from 'vue'
|
||||||
import { updateDocumentTitle } from '@/utils'
|
import { updateDocumentTitle } from '@/utils'
|
||||||
import { BookOpen, GraduationCap } from 'lucide-vue-next'
|
import { BookOpen, GraduationCap } from 'lucide-vue-next'
|
||||||
|
import { sessionStore } from '../stores/session'
|
||||||
|
|
||||||
const currentCategory = ref('')
|
const currentCategory = ref('')
|
||||||
const filters = ref({})
|
const filters = ref({})
|
||||||
const nameFilter = ref('')
|
const nameFilter = ref('')
|
||||||
|
const { brand } = sessionStore()
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
updateParticipants()
|
updateParticipants()
|
||||||
@@ -163,13 +166,12 @@ const breadcrumbs = computed(() => [
|
|||||||
},
|
},
|
||||||
])
|
])
|
||||||
|
|
||||||
const pageMeta = computed(() => {
|
usePageMeta(() => {
|
||||||
return {
|
return {
|
||||||
title: 'Certified Participants',
|
title: __('Certified Participants'),
|
||||||
description: 'All participants that have been certified.',
|
icon: brand.favicon,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
updateDocumentTitle(pageMeta)
|
|
||||||
</script>
|
</script>
|
||||||
<style>
|
<style>
|
||||||
.headline {
|
.headline {
|
||||||
|
|||||||
@@ -36,14 +36,18 @@
|
|||||||
</template>
|
</template>
|
||||||
<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, usePageMeta } from 'frappe-ui'
|
||||||
|
import { useRouter } from 'vue-router'
|
||||||
|
import { sessionStore } from '../stores/session'
|
||||||
import UpcomingEvaluations from '@/components/UpcomingEvaluations.vue'
|
import UpcomingEvaluations from '@/components/UpcomingEvaluations.vue'
|
||||||
|
|
||||||
const courseTitle = ref(null)
|
const courseTitle = ref(null)
|
||||||
const evaluator = ref(null)
|
const evaluator = ref(null)
|
||||||
|
const { brand } = sessionStore()
|
||||||
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 +57,7 @@ const props = defineProps({
|
|||||||
})
|
})
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
|
fetchEnrollmentDetails()
|
||||||
fetchCourseDetails()
|
fetchCourseDetails()
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -66,10 +71,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',
|
||||||
@@ -114,4 +135,11 @@ const breadcrumbs = computed(() => [
|
|||||||
label: __('Certification'),
|
label: __('Certification'),
|
||||||
},
|
},
|
||||||
])
|
])
|
||||||
|
|
||||||
|
usePageMeta(() => {
|
||||||
|
return {
|
||||||
|
title: courseTitle.value,
|
||||||
|
icon: brand.favicon,
|
||||||
|
}
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -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
|
||||||
@@ -92,16 +92,24 @@
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { createResource, Breadcrumbs, Badge, Tooltip } from 'frappe-ui'
|
import {
|
||||||
|
createResource,
|
||||||
|
Breadcrumbs,
|
||||||
|
Badge,
|
||||||
|
Tooltip,
|
||||||
|
usePageMeta,
|
||||||
|
} from 'frappe-ui'
|
||||||
import { computed } from 'vue'
|
import { computed } from 'vue'
|
||||||
import { Users, Star } from 'lucide-vue-next'
|
import { Users, Star } from 'lucide-vue-next'
|
||||||
|
import { sessionStore } from '@/stores/session'
|
||||||
import CourseCardOverlay from '@/components/CourseCardOverlay.vue'
|
import CourseCardOverlay from '@/components/CourseCardOverlay.vue'
|
||||||
import CourseOutline from '@/components/CourseOutline.vue'
|
import CourseOutline from '@/components/CourseOutline.vue'
|
||||||
import CourseReviews from '@/components/CourseReviews.vue'
|
import CourseReviews from '@/components/CourseReviews.vue'
|
||||||
import UserAvatar from '@/components/UserAvatar.vue'
|
import UserAvatar from '@/components/UserAvatar.vue'
|
||||||
import { updateDocumentTitle } from '@/utils'
|
|
||||||
import CourseInstructors from '@/components/CourseInstructors.vue'
|
import CourseInstructors from '@/components/CourseInstructors.vue'
|
||||||
|
|
||||||
|
const { brand } = sessionStore()
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
courseName: {
|
courseName: {
|
||||||
type: String,
|
type: String,
|
||||||
@@ -127,14 +135,12 @@ const breadcrumbs = computed(() => {
|
|||||||
return items
|
return items
|
||||||
})
|
})
|
||||||
|
|
||||||
const pageMeta = computed(() => {
|
usePageMeta(() => {
|
||||||
return {
|
return {
|
||||||
title: course?.data?.title,
|
title: course?.data?.title,
|
||||||
description: course?.data?.short_introduction,
|
icon: brand.favicon,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
updateDocumentTitle(pageMeta)
|
|
||||||
</script>
|
</script>
|
||||||
<style>
|
<style>
|
||||||
.avatar-group {
|
.avatar-group {
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
<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
|
||||||
@@ -8,12 +8,9 @@
|
|||||||
<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 v-if="courseResource.data?.name" @click="trashCourse()">
|
||||||
<template #prefix>
|
<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 +230,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>
|
||||||
@@ -252,6 +249,7 @@ import {
|
|||||||
createResource,
|
createResource,
|
||||||
FormControl,
|
FormControl,
|
||||||
FileUploader,
|
FileUploader,
|
||||||
|
usePageMeta,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import {
|
import {
|
||||||
inject,
|
inject,
|
||||||
@@ -263,21 +261,25 @@ import {
|
|||||||
watch,
|
watch,
|
||||||
getCurrentInstance,
|
getCurrentInstance,
|
||||||
} from 'vue'
|
} from 'vue'
|
||||||
import { showToast, updateDocumentTitle } from '@/utils'
|
import { showToast } from '@/utils'
|
||||||
import Link from '@/components/Controls/Link.vue'
|
|
||||||
import { Image, Trash2, X } from 'lucide-vue-next'
|
import { Image, Trash2, X } from 'lucide-vue-next'
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
|
import { capture } from '@/telemetry'
|
||||||
|
import { useOnboarding } from 'frappe-ui/frappe'
|
||||||
|
import { sessionStore } from '../stores/session'
|
||||||
|
import { useSettings } from '@/stores/settings'
|
||||||
|
import Link from '@/components/Controls/Link.vue'
|
||||||
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 { useSettings } from '@/stores/settings'
|
|
||||||
|
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
const newTag = ref('')
|
const newTag = ref('')
|
||||||
|
const { brand } = sessionStore()
|
||||||
const router = useRouter()
|
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({
|
||||||
@@ -398,7 +400,7 @@ const courseResource = createResource({
|
|||||||
'paid_course',
|
'paid_course',
|
||||||
'featured',
|
'featured',
|
||||||
'enable_certification',
|
'enable_certification',
|
||||||
'paid_certifiate',
|
'paid_certificate',
|
||||||
]
|
]
|
||||||
for (let idx in checkboxes) {
|
for (let idx in checkboxes) {
|
||||||
let key = checkboxes[idx]
|
let key = checkboxes[idx]
|
||||||
@@ -441,11 +443,14 @@ const submitCourse = () => {
|
|||||||
} else {
|
} else {
|
||||||
courseCreationResource.submit(course, {
|
courseCreationResource.submit(course, {
|
||||||
onSuccess(data) {
|
onSuccess(data) {
|
||||||
|
if (user.data?.is_system_manager) {
|
||||||
|
updateOnboardingStep('create_first_course', true, false, () => {
|
||||||
|
localStorage.setItem('firstCourse', data.name)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
capture('course_created')
|
capture('course_created')
|
||||||
showToast('Success', 'Course created successfully', 'check')
|
showToast('Success', 'Course created successfully', 'check')
|
||||||
/* if (!settingsStore.onboardingDetails.data?.is_onboarded) {
|
|
||||||
settingsStore.onboardingDetails.reload()
|
|
||||||
} */
|
|
||||||
router.push({
|
router.push({
|
||||||
name: 'CourseForm',
|
name: 'CourseForm',
|
||||||
params: { courseName: data.name },
|
params: { courseName: data.name },
|
||||||
@@ -571,12 +576,10 @@ const breadcrumbs = computed(() => {
|
|||||||
return crumbs
|
return crumbs
|
||||||
})
|
})
|
||||||
|
|
||||||
const pageMeta = computed(() => {
|
usePageMeta(() => {
|
||||||
return {
|
return {
|
||||||
title: 'Create a Course',
|
title: courseResource.data?.title || __('New Course'),
|
||||||
description: 'Create or edit a course for your learning system.',
|
icon: brand.favicon,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
updateDocumentTitle(pageMeta)
|
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -1,318 +1,315 @@
|
|||||||
<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="canCreateCourse()"
|
||||||
|
: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,
|
||||||
|
usePageMeta,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
|
import { computed, inject, onMounted, ref, watch } from 'vue'
|
||||||
|
import { BookOpen, Plus } from 'lucide-vue-next'
|
||||||
|
import { sessionStore } from '@/stores/session'
|
||||||
|
import { canCreateCourse } from '@/utils'
|
||||||
import CourseCard from '@/components/CourseCard.vue'
|
import CourseCard from '@/components/CourseCard.vue'
|
||||||
import { BookOpen, Plus, Search } from 'lucide-vue-next'
|
|
||||||
import { ref, computed, inject, onMounted, watch } from 'vue'
|
|
||||||
import { updateDocumentTitle } from '@/utils'
|
|
||||||
import { useRouter } from 'vue-router'
|
|
||||||
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')
|
||||||
|
const { brand } = sessionStore()
|
||||||
|
|
||||||
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, () => {
|
usePageMeta(() => {
|
||||||
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(() => {
|
|
||||||
return {
|
return {
|
||||||
title: 'Courses',
|
title: __('Courses'),
|
||||||
description: 'All Courses divided by categories',
|
icon: brand.favicon,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
updateDocumentTitle(pageMeta)
|
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -1,39 +0,0 @@
|
|||||||
<template>
|
|
||||||
<div class="max-w-3xl py-12 mx-auto">
|
|
||||||
<Button
|
|
||||||
icon-left="code"
|
|
||||||
@click="$resources.ping.fetch"
|
|
||||||
:loading="$resources.ping.loading"
|
|
||||||
>
|
|
||||||
Click to send 'ping' request
|
|
||||||
</Button>
|
|
||||||
<div>
|
|
||||||
{{ $resources.ping.data }}
|
|
||||||
</div>
|
|
||||||
<pre>{{ $resources.ping }}</pre>
|
|
||||||
|
|
||||||
<Button @click="showDialog = true">Open Dialog</Button>
|
|
||||||
<Dialog title="Title" v-model="showDialog"> Dialog content </Dialog>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script>
|
|
||||||
import { Dialog } from 'frappe-ui'
|
|
||||||
|
|
||||||
export default {
|
|
||||||
name: 'Home',
|
|
||||||
data() {
|
|
||||||
return {
|
|
||||||
showDialog: false,
|
|
||||||
}
|
|
||||||
},
|
|
||||||
resources: {
|
|
||||||
ping: {
|
|
||||||
url: 'ping',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
components: {
|
|
||||||
Dialog,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
@@ -139,14 +139,17 @@ import {
|
|||||||
Button,
|
Button,
|
||||||
TextEditor,
|
TextEditor,
|
||||||
FileUploader,
|
FileUploader,
|
||||||
|
usePageMeta,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import { computed, onMounted, reactive, inject } from 'vue'
|
import { computed, onMounted, reactive, inject } from 'vue'
|
||||||
import { FileText, X } from 'lucide-vue-next'
|
import { FileText, X } from 'lucide-vue-next'
|
||||||
|
import { sessionStore } from '@/stores/session'
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
import { getFileSize, showToast } from '../utils'
|
import { getFileSize, showToast } from '../utils'
|
||||||
|
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
|
const { brand } = sessionStore()
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
jobName: {
|
jobName: {
|
||||||
@@ -319,4 +322,11 @@ const breadcrumbs = computed(() => {
|
|||||||
]
|
]
|
||||||
return crumbs
|
return crumbs
|
||||||
})
|
})
|
||||||
|
|
||||||
|
usePageMeta(() => {
|
||||||
|
return {
|
||||||
|
title: props.jobName == 'new' ? 'New Job' : jobDetail.data?.title,
|
||||||
|
icon: brand.favicon,
|
||||||
|
}
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -16,7 +16,7 @@
|
|||||||
},
|
},
|
||||||
]"
|
]"
|
||||||
/>
|
/>
|
||||||
<div v-if="user.data?.name" class="flex">
|
<div v-if="user.data?.name" class="flex space-x-2">
|
||||||
<router-link
|
<router-link
|
||||||
v-if="user.data.name == job.data?.owner"
|
v-if="user.data.name == job.data?.owner"
|
||||||
:to="{
|
:to="{
|
||||||
@@ -24,13 +24,19 @@
|
|||||||
params: { jobName: job.data?.name },
|
params: { jobName: job.data?.name },
|
||||||
}"
|
}"
|
||||||
>
|
>
|
||||||
<Button class="mr-2">
|
<Button>
|
||||||
<template #prefix>
|
<template #prefix>
|
||||||
<Pencil class="h-4 w-4 stroke-1.5" />
|
<Pencil class="h-4 w-4 stroke-1.5" />
|
||||||
</template>
|
</template>
|
||||||
{{ __('Edit') }}
|
{{ __('Edit') }}
|
||||||
</Button>
|
</Button>
|
||||||
</router-link>
|
</router-link>
|
||||||
|
<Button @click="redirectToWebsite(job.data?.company_website)">
|
||||||
|
<template #prefix>
|
||||||
|
<SquareArrowOutUpRight class="h-4 w-4 stroke-1.5" />
|
||||||
|
</template>
|
||||||
|
{{ __('Visit Website') }}
|
||||||
|
</Button>
|
||||||
<Button
|
<Button
|
||||||
v-if="!jobApplication.data?.length"
|
v-if="!jobApplication.data?.length"
|
||||||
variant="solid"
|
variant="solid"
|
||||||
@@ -56,10 +62,11 @@
|
|||||||
<div class="flex items-center">
|
<div class="flex items-center">
|
||||||
<img
|
<img
|
||||||
:src="job.data.company_logo"
|
:src="job.data.company_logo"
|
||||||
class="w-16 h-16 rounded-lg object-contain mr-4"
|
class="w-16 h-16 rounded-lg object-contain cursor-pointer mr-4"
|
||||||
:alt="job.data.company_name"
|
:alt="job.data.company_name"
|
||||||
|
@click="redirectToWebsite(job.data.company_website)"
|
||||||
/>
|
/>
|
||||||
<div class="text-2xl text-ink-gray-9 font-semibold mb-4">
|
<div class="text-2xl text-ink-gray-9 font-semibold">
|
||||||
{{ job.data.job_title }}
|
{{ job.data.job_title }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -69,7 +76,7 @@
|
|||||||
>
|
>
|
||||||
<div class="flex items-center space-x-4">
|
<div class="flex items-center space-x-4">
|
||||||
<Building2 class="h-4 w-4 text-ink-green-2" />
|
<Building2 class="h-4 w-4 text-ink-green-2" />
|
||||||
<div class="flex flex-col space-y-2 text-ink-gray-7">
|
<div class="flex flex-col space-y-1 text-ink-gray-7">
|
||||||
<span class="text-xs text-ink-gray-5 font-medium uppercase">
|
<span class="text-xs text-ink-gray-5 font-medium uppercase">
|
||||||
{{ __('Organisation') }}
|
{{ __('Organisation') }}
|
||||||
</span>
|
</span>
|
||||||
@@ -80,7 +87,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<div class="flex items-center space-x-4">
|
<div class="flex items-center space-x-4">
|
||||||
<MapPin class="size-4 text-ink-red-3" />
|
<MapPin class="size-4 text-ink-red-3" />
|
||||||
<div class="flex flex-col space-y-2 text-ink-gray-7">
|
<div class="flex flex-col space-y-1 text-ink-gray-7">
|
||||||
<span class="text-xs font-medium uppercase">
|
<span class="text-xs font-medium uppercase">
|
||||||
{{ __('Location') }}
|
{{ __('Location') }}
|
||||||
</span>
|
</span>
|
||||||
@@ -91,7 +98,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<div class="flex items-center space-x-4">
|
<div class="flex items-center space-x-4">
|
||||||
<ClipboardType class="h-4 w-4 text-yellow-500" />
|
<ClipboardType class="h-4 w-4 text-yellow-500" />
|
||||||
<div class="flex flex-col space-y-2 text-ink-gray-7">
|
<div class="flex flex-col space-y-1 text-ink-gray-7">
|
||||||
<span class="text-xs font-medium uppercase">
|
<span class="text-xs font-medium uppercase">
|
||||||
{{ __('Category') }}
|
{{ __('Category') }}
|
||||||
</span>
|
</span>
|
||||||
@@ -102,7 +109,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<div class="flex items-center space-x-4">
|
<div class="flex items-center space-x-4">
|
||||||
<CalendarDays class="h-4 w-4 text-ink-blue-2" />
|
<CalendarDays class="h-4 w-4 text-ink-blue-2" />
|
||||||
<div class="flex flex-col space-y-2 text-ink-gray-7">
|
<div class="flex flex-col space-y-1 text-ink-gray-7">
|
||||||
<span class="text-xs font-medium uppercase">
|
<span class="text-xs font-medium uppercase">
|
||||||
{{ __('Posted on') }}
|
{{ __('Posted on') }}
|
||||||
</span>
|
</span>
|
||||||
@@ -116,7 +123,7 @@
|
|||||||
class="flex items-center space-x-4"
|
class="flex items-center space-x-4"
|
||||||
>
|
>
|
||||||
<SquareUserRound class="h-4 w-4 text-purple-500" />
|
<SquareUserRound class="h-4 w-4 text-purple-500" />
|
||||||
<div class="flex flex-col space-y-2 text-ink-gray-7">
|
<div class="flex flex-col space-y-1 text-ink-gray-7">
|
||||||
<span class="text-xs font-medium uppercase">
|
<span class="text-xs font-medium uppercase">
|
||||||
{{ __('Applications Received') }}
|
{{ __('Applications Received') }}
|
||||||
</span>
|
</span>
|
||||||
@@ -142,9 +149,9 @@
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { Button, Breadcrumbs, createResource } from 'frappe-ui'
|
import { Button, Breadcrumbs, createResource, usePageMeta } from 'frappe-ui'
|
||||||
import { inject, ref, computed } from 'vue'
|
import { inject, ref } from 'vue'
|
||||||
import { updateDocumentTitle } from '@/utils'
|
import { sessionStore } from '../stores/session'
|
||||||
import JobApplicationModal from '@/components/Modals/JobApplicationModal.vue'
|
import JobApplicationModal from '@/components/Modals/JobApplicationModal.vue'
|
||||||
import {
|
import {
|
||||||
MapPin,
|
MapPin,
|
||||||
@@ -154,10 +161,12 @@ import {
|
|||||||
CalendarDays,
|
CalendarDays,
|
||||||
ClipboardType,
|
ClipboardType,
|
||||||
SquareUserRound,
|
SquareUserRound,
|
||||||
|
SquareArrowOutUpRight,
|
||||||
} from 'lucide-vue-next'
|
} from 'lucide-vue-next'
|
||||||
|
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
const dayjs = inject('$dayjs')
|
const dayjs = inject('$dayjs')
|
||||||
|
const { brand } = sessionStore()
|
||||||
const showApplicationModal = ref(false)
|
const showApplicationModal = ref(false)
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
@@ -215,12 +224,14 @@ const redirectToLogin = (job) => {
|
|||||||
window.location.href = `/login?redirect-to=/job-openings/${job}`
|
window.location.href = `/login?redirect-to=/job-openings/${job}`
|
||||||
}
|
}
|
||||||
|
|
||||||
const pageMeta = computed(() => {
|
const redirectToWebsite = (url) => {
|
||||||
|
window.open(url, '_blank')
|
||||||
|
}
|
||||||
|
|
||||||
|
usePageMeta(() => {
|
||||||
return {
|
return {
|
||||||
title: job.data?.job_title,
|
title: job.data?.job_title,
|
||||||
description: job.data?.description,
|
icon: brand.favicon,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
updateDocumentTitle(pageMeta)
|
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -25,7 +25,7 @@
|
|||||||
</router-link>
|
</router-link>
|
||||||
</header>
|
</header>
|
||||||
<div>
|
<div>
|
||||||
<div class="lg:w-3/4 mx-auto p-5">
|
<div v-if="jobs.data?.length" class="p-5">
|
||||||
<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"
|
||||||
>
|
>
|
||||||
@@ -58,10 +58,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div
|
<div class="grid grid-cols-1 lg:grid-cols-3 gap-5">
|
||||||
v-if="jobs.data?.length"
|
|
||||||
class="grid grid-cols-1 lg:grid-cols-2 gap-5"
|
|
||||||
>
|
|
||||||
<router-link
|
<router-link
|
||||||
v-for="job in jobs.data"
|
v-for="job in jobs.data"
|
||||||
:to="{
|
:to="{
|
||||||
@@ -73,22 +70,42 @@
|
|||||||
<JobCard :job="job" />
|
<JobCard :job="job" />
|
||||||
</router-link>
|
</router-link>
|
||||||
</div>
|
</div>
|
||||||
<div v-else class="text-ink-gray-7 italic p-5 w-fit mx-auto">
|
</div>
|
||||||
{{ __('No jobs posted') }}
|
<div
|
||||||
|
v-else
|
||||||
|
class="flex flex-col items-center justify-center text-sm text-ink-gray-5 mt-48"
|
||||||
|
>
|
||||||
|
<Laptop class="size-10 mx-auto stroke-1 text-ink-gray-4" />
|
||||||
|
<div class="text-lg font-medium mb-1">
|
||||||
|
{{ __('No jobs found') }}
|
||||||
|
</div>
|
||||||
|
<div class="leading-5 w-2/5 text-center">
|
||||||
|
{{
|
||||||
|
__(
|
||||||
|
'There are no jobs available at the moment. Open a job opportunity or check here again later.'
|
||||||
|
)
|
||||||
|
}}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { Button, Breadcrumbs, createResource, FormControl } from 'frappe-ui'
|
import {
|
||||||
import { Plus, Search } from 'lucide-vue-next'
|
Button,
|
||||||
|
Breadcrumbs,
|
||||||
|
createResource,
|
||||||
|
FormControl,
|
||||||
|
usePageMeta,
|
||||||
|
} from 'frappe-ui'
|
||||||
|
import { Laptop, Plus, Search } from 'lucide-vue-next'
|
||||||
|
import { sessionStore } from '../stores/session'
|
||||||
import { inject, computed, ref, onMounted } from 'vue'
|
import { inject, computed, ref, onMounted } from 'vue'
|
||||||
import JobCard from '@/components/JobCard.vue'
|
import JobCard from '@/components/JobCard.vue'
|
||||||
import { updateDocumentTitle } from '@/utils'
|
|
||||||
|
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
const jobType = ref(null)
|
const jobType = ref(null)
|
||||||
|
const { brand } = sessionStore()
|
||||||
const searchQuery = ref('')
|
const searchQuery = ref('')
|
||||||
const filters = ref({})
|
const filters = ref({})
|
||||||
const orFilters = ref({})
|
const orFilters = ref({})
|
||||||
@@ -147,12 +164,11 @@ const jobTypes = computed(() => {
|
|||||||
{ label: __('Freelance'), value: 'Freelance' },
|
{ label: __('Freelance'), value: 'Freelance' },
|
||||||
]
|
]
|
||||||
})
|
})
|
||||||
const pageMeta = computed(() => {
|
|
||||||
|
usePageMeta(() => {
|
||||||
return {
|
return {
|
||||||
title: 'Jobs',
|
title: __('Jobs'),
|
||||||
description: 'An open job board for the community',
|
icon: brand.favicon,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
updateDocumentTitle(pageMeta)
|
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -193,14 +193,15 @@
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { createResource, Breadcrumbs, Button } from 'frappe-ui'
|
import { createResource, Breadcrumbs, Button, usePageMeta } from 'frappe-ui'
|
||||||
import { computed, watch, inject, ref, onMounted, onBeforeUnmount } from 'vue'
|
import { computed, watch, inject, ref, onMounted, onBeforeUnmount } from 'vue'
|
||||||
import CourseOutline from '@/components/CourseOutline.vue'
|
import CourseOutline from '@/components/CourseOutline.vue'
|
||||||
import UserAvatar from '@/components/UserAvatar.vue'
|
import UserAvatar from '@/components/UserAvatar.vue'
|
||||||
import { useRouter, useRoute } from 'vue-router'
|
import { useRouter, useRoute } from 'vue-router'
|
||||||
import { ChevronLeft, ChevronRight, GraduationCap } from 'lucide-vue-next'
|
import { ChevronLeft, ChevronRight, GraduationCap } from 'lucide-vue-next'
|
||||||
import Discussions from '@/components/Discussions.vue'
|
import Discussions from '@/components/Discussions.vue'
|
||||||
import { getEditorTools, updateDocumentTitle } from '../utils'
|
import { getEditorTools } from '../utils'
|
||||||
|
import { sessionStore } from '@/stores/session'
|
||||||
import EditorJS from '@editorjs/editorjs'
|
import EditorJS from '@editorjs/editorjs'
|
||||||
import LessonContent from '@/components/LessonContent.vue'
|
import LessonContent from '@/components/LessonContent.vue'
|
||||||
import CourseInstructors from '@/components/CourseInstructors.vue'
|
import CourseInstructors from '@/components/CourseInstructors.vue'
|
||||||
@@ -215,6 +216,7 @@ const editor = ref(null)
|
|||||||
const instructorEditor = ref(null)
|
const instructorEditor = ref(null)
|
||||||
const lessonProgress = ref(0)
|
const lessonProgress = ref(0)
|
||||||
const timer = ref(0)
|
const timer = ref(0)
|
||||||
|
const { brand } = sessionStore()
|
||||||
let timerInterval
|
let timerInterval
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
@@ -419,14 +421,12 @@ const redirectToLogin = () => {
|
|||||||
window.location.href = `/login?redirect-to=/lms/courses/${props.courseName}`
|
window.location.href = `/login?redirect-to=/lms/courses/${props.courseName}`
|
||||||
}
|
}
|
||||||
|
|
||||||
const pageMeta = computed(() => {
|
usePageMeta(() => {
|
||||||
return {
|
return {
|
||||||
title: lesson.data?.title,
|
title: lesson?.data?.title,
|
||||||
description: lesson.data?.course,
|
icon: brand.favicon,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
updateDocumentTitle(pageMeta)
|
|
||||||
</script>
|
</script>
|
||||||
<style>
|
<style>
|
||||||
.avatar-group {
|
.avatar-group {
|
||||||
@@ -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;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -78,7 +78,13 @@
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { Breadcrumbs, Button, createResource, FormControl } from 'frappe-ui'
|
import {
|
||||||
|
Breadcrumbs,
|
||||||
|
Button,
|
||||||
|
createResource,
|
||||||
|
FormControl,
|
||||||
|
usePageMeta,
|
||||||
|
} from 'frappe-ui'
|
||||||
import {
|
import {
|
||||||
computed,
|
computed,
|
||||||
reactive,
|
reactive,
|
||||||
@@ -87,18 +93,20 @@ import {
|
|||||||
ref,
|
ref,
|
||||||
onBeforeUnmount,
|
onBeforeUnmount,
|
||||||
} from 'vue'
|
} from 'vue'
|
||||||
|
import { sessionStore } from '../stores/session'
|
||||||
import EditorJS from '@editorjs/editorjs'
|
import EditorJS from '@editorjs/editorjs'
|
||||||
import LessonHelp from '@/components/LessonHelp.vue'
|
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 { createToast, getEditorTools } from '@/utils'
|
||||||
import { capture } from '@/telemetry'
|
import { capture } from '@/telemetry'
|
||||||
import { useSettings } from '@/stores/settings'
|
import { useOnboarding } from 'frappe-ui/frappe'
|
||||||
|
|
||||||
|
const { brand } = sessionStore()
|
||||||
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
|
||||||
|
|
||||||
@@ -394,11 +402,11 @@ const createNewLesson = () => {
|
|||||||
{ lesson: data.name },
|
{ lesson: data.name },
|
||||||
{
|
{
|
||||||
onSuccess() {
|
onSuccess() {
|
||||||
|
if (user.data?.is_system_manager)
|
||||||
|
updateOnboardingStep('create_first_lesson')
|
||||||
|
|
||||||
capture('lesson_created')
|
capture('lesson_created')
|
||||||
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()
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
@@ -494,14 +502,14 @@ const breadcrumbs = computed(() => {
|
|||||||
return crumbs
|
return crumbs
|
||||||
})
|
})
|
||||||
|
|
||||||
const pageMeta = computed(() => {
|
usePageMeta(() => {
|
||||||
return {
|
return {
|
||||||
title: 'Lesson Editor',
|
title: lessonDetails?.data?.lesson
|
||||||
description: 'Create and edit lessons for your course',
|
? lessonDetails.data.lesson.title
|
||||||
|
: 'New Lesson',
|
||||||
|
icon: brand.favicon,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
updateDocumentTitle(pageMeta)
|
|
||||||
</script>
|
</script>
|
||||||
<style>
|
<style>
|
||||||
.embed-tool__caption,
|
.embed-tool__caption,
|
||||||
@@ -623,4 +631,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>
|
||||||
|
|||||||
@@ -65,12 +65,14 @@ import {
|
|||||||
TabButtons,
|
TabButtons,
|
||||||
Button,
|
Button,
|
||||||
Tooltip,
|
Tooltip,
|
||||||
|
usePageMeta,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
|
import { sessionStore } from '../stores/session'
|
||||||
import { computed, inject, ref, onMounted } from 'vue'
|
import { computed, inject, ref, onMounted } from 'vue'
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
import { X } from 'lucide-vue-next'
|
import { X } from 'lucide-vue-next'
|
||||||
import { updateDocumentTitle } from '@/utils'
|
|
||||||
|
|
||||||
|
const { brand } = sessionStore()
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
const socket = inject('$socket')
|
const socket = inject('$socket')
|
||||||
const activeTab = ref('Unread')
|
const activeTab = ref('Unread')
|
||||||
@@ -145,14 +147,12 @@ const breadcrumbs = computed(() => {
|
|||||||
return crumbs
|
return crumbs
|
||||||
})
|
})
|
||||||
|
|
||||||
const pageMeta = computed(() => {
|
usePageMeta(() => {
|
||||||
return {
|
return {
|
||||||
title: 'Notifications',
|
title: 'Notifications',
|
||||||
description: 'All your notifications in one place.',
|
icon: brand.favicon,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
updateDocumentTitle(pageMeta)
|
|
||||||
</script>
|
</script>
|
||||||
<style>
|
<style>
|
||||||
.notification strong {
|
.notification strong {
|
||||||
|
|||||||
@@ -86,18 +86,24 @@
|
|||||||
/>
|
/>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { Breadcrumbs, createResource, Button, TabButtons } from 'frappe-ui'
|
import {
|
||||||
|
Breadcrumbs,
|
||||||
|
createResource,
|
||||||
|
Button,
|
||||||
|
TabButtons,
|
||||||
|
usePageMeta,
|
||||||
|
} from 'frappe-ui'
|
||||||
import { computed, inject, watch, ref, onMounted, watchEffect } from 'vue'
|
import { computed, inject, watch, ref, onMounted, watchEffect } from 'vue'
|
||||||
import { sessionStore } from '@/stores/session'
|
import { sessionStore } from '@/stores/session'
|
||||||
import { Edit } from 'lucide-vue-next'
|
import { Edit, icons } from 'lucide-vue-next'
|
||||||
import UserAvatar from '@/components/UserAvatar.vue'
|
import UserAvatar from '@/components/UserAvatar.vue'
|
||||||
import { useRoute, useRouter } from 'vue-router'
|
import { useRoute, useRouter } from 'vue-router'
|
||||||
import NoPermission from '@/components/NoPermission.vue'
|
import NoPermission from '@/components/NoPermission.vue'
|
||||||
import { convertToTitleCase, updateDocumentTitle } from '@/utils'
|
import { convertToTitleCase } from '@/utils'
|
||||||
import EditProfile from '@/components/Modals/EditProfile.vue'
|
import EditProfile from '@/components/Modals/EditProfile.vue'
|
||||||
import EditCoverImage from '@/components/Modals/EditCoverImage.vue'
|
import EditCoverImage from '@/components/Modals/EditCoverImage.vue'
|
||||||
|
|
||||||
const { user } = sessionStore()
|
const { user, brand } = sessionStore()
|
||||||
const $user = inject('$user')
|
const $user = inject('$user')
|
||||||
const route = useRoute()
|
const route = useRoute()
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
@@ -215,12 +221,10 @@ const breadcrumbs = computed(() => {
|
|||||||
return crumbs
|
return crumbs
|
||||||
})
|
})
|
||||||
|
|
||||||
const pageMeta = computed(() => {
|
usePageMeta(() => {
|
||||||
return {
|
return {
|
||||||
title: profile.data?.full_name,
|
title: profile.data?.full_name,
|
||||||
description: profile.data?.headline,
|
icon: brand.favicon,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
updateDocumentTitle(pageMeta)
|
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -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,
|
||||||
|
|||||||
@@ -186,14 +186,17 @@ import {
|
|||||||
ListHeader,
|
ListHeader,
|
||||||
ListHeaderItem,
|
ListHeaderItem,
|
||||||
ListSelectBanner,
|
ListSelectBanner,
|
||||||
|
usePageMeta,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import { computed, ref } from 'vue'
|
import { computed, ref } from 'vue'
|
||||||
import { Plus, Trash2 } from 'lucide-vue-next'
|
import { Plus, Trash2 } from 'lucide-vue-next'
|
||||||
import Link from '@/components/Controls/Link.vue'
|
|
||||||
import { showToast } from '@/utils/'
|
import { showToast } from '@/utils/'
|
||||||
import Draggable from 'vuedraggable'
|
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
|
import { sessionStore } from '../stores/session'
|
||||||
|
import Draggable from 'vuedraggable'
|
||||||
|
import Link from '@/components/Controls/Link.vue'
|
||||||
|
|
||||||
|
const { brand } = sessionStore()
|
||||||
const showDialog = ref(false)
|
const showDialog = ref(false)
|
||||||
const currentForm = ref(null)
|
const currentForm = ref(null)
|
||||||
const course = ref(null)
|
const course = ref(null)
|
||||||
@@ -364,4 +367,11 @@ const breadbrumbs = computed(() => {
|
|||||||
},
|
},
|
||||||
]
|
]
|
||||||
})
|
})
|
||||||
|
|
||||||
|
usePageMeta(() => {
|
||||||
|
return {
|
||||||
|
title: program.doc?.title,
|
||||||
|
icon: brand.favicon,
|
||||||
|
}
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -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">
|
||||||
@@ -126,14 +126,17 @@ import {
|
|||||||
createResource,
|
createResource,
|
||||||
Dialog,
|
Dialog,
|
||||||
FormControl,
|
FormControl,
|
||||||
|
usePageMeta,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import { computed, inject, onMounted, ref } from 'vue'
|
import { computed, inject, onMounted, ref } from 'vue'
|
||||||
import { BookOpen, Edit, Plus, LockKeyhole } from 'lucide-vue-next'
|
import { BookOpen, Edit, Plus, LockKeyhole } from 'lucide-vue-next'
|
||||||
import CourseCard from '@/components/CourseCard.vue'
|
import CourseCard from '@/components/CourseCard.vue'
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
|
import { sessionStore } from '../stores/session'
|
||||||
import { showToast } from '@/utils'
|
import { showToast } from '@/utils'
|
||||||
import { useSettings } from '@/stores/settings'
|
import { useSettings } from '@/stores/settings'
|
||||||
|
|
||||||
|
const { brand } = sessionStore()
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
const showDialog = ref(false)
|
const showDialog = ref(false)
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
@@ -210,4 +213,11 @@ const breadbrumbs = computed(() => [
|
|||||||
label: 'Programs',
|
label: 'Programs',
|
||||||
},
|
},
|
||||||
])
|
])
|
||||||
|
|
||||||
|
usePageMeta(() => {
|
||||||
|
return {
|
||||||
|
title: __('Programs'),
|
||||||
|
icon: brand.favicon,
|
||||||
|
}
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -197,6 +197,7 @@ import {
|
|||||||
ListRowItem,
|
ListRowItem,
|
||||||
ListSelectBanner,
|
ListSelectBanner,
|
||||||
Button,
|
Button,
|
||||||
|
usePageMeta,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import {
|
import {
|
||||||
computed,
|
computed,
|
||||||
@@ -207,11 +208,13 @@ import {
|
|||||||
onBeforeUnmount,
|
onBeforeUnmount,
|
||||||
watch,
|
watch,
|
||||||
} from 'vue'
|
} from 'vue'
|
||||||
|
import { sessionStore } from '../stores/session'
|
||||||
import { Plus, Trash2 } from 'lucide-vue-next'
|
import { Plus, Trash2 } from 'lucide-vue-next'
|
||||||
import Question from '@/components/Modals/Question.vue'
|
|
||||||
import { showToast, updateDocumentTitle } from '@/utils'
|
import { showToast, updateDocumentTitle } from '@/utils'
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
|
import Question from '@/components/Modals/Question.vue'
|
||||||
|
|
||||||
|
const { brand } = sessionStore()
|
||||||
const showQuestionModal = ref(false)
|
const showQuestionModal = ref(false)
|
||||||
const currentQuestion = reactive({
|
const currentQuestion = reactive({
|
||||||
question: '',
|
question: '',
|
||||||
@@ -453,12 +456,10 @@ const breadcrumbs = computed(() => {
|
|||||||
return crumbs
|
return crumbs
|
||||||
})
|
})
|
||||||
|
|
||||||
const pageMeta = computed(() => {
|
usePageMeta(() => {
|
||||||
return {
|
return {
|
||||||
title: props.quizID == 'new' ? __('New Quiz') : quizDetails.data?.title,
|
title: props.quizID == 'new' ? __('New Quiz') : quizDetails.data?.title,
|
||||||
description: __('Form to create and edit quizzes'),
|
icon: brand.favicon,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
updateDocumentTitle(pageMeta)
|
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -1,27 +1,37 @@
|
|||||||
<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, usePageMeta } 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 { sessionStore } from '../stores/session'
|
||||||
|
|
||||||
|
const { brand } = sessionStore()
|
||||||
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({
|
||||||
@@ -47,12 +57,10 @@ const breadcrumbs = computed(() => {
|
|||||||
return [{ label: __('Quiz Submission') }, { label: title.data?.title }]
|
return [{ label: __('Quiz Submission') }, { label: title.data?.title }]
|
||||||
})
|
})
|
||||||
|
|
||||||
const pageMeta = computed(() => {
|
usePageMeta(() => {
|
||||||
return {
|
return {
|
||||||
title: title.data?.title,
|
title: `${title.data?.title}`,
|
||||||
description: __('Quiz Submission'),
|
icon: brand.favicon,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
updateDocumentTitle(pageMeta)
|
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -79,11 +79,14 @@ import {
|
|||||||
FormControl,
|
FormControl,
|
||||||
Button,
|
Button,
|
||||||
Badge,
|
Badge,
|
||||||
|
usePageMeta,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import { computed, onBeforeUnmount, onMounted, inject } from 'vue'
|
import { computed, onBeforeUnmount, onMounted, inject } from 'vue'
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
import { showToast } from '@/utils'
|
import { showToast } from '@/utils'
|
||||||
|
import { sessionStore } from '@/stores/session'
|
||||||
|
|
||||||
|
const { brand } = sessionStore()
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
|
|
||||||
@@ -149,4 +152,11 @@ const saveSubmission = () => {
|
|||||||
}
|
}
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
usePageMeta(() => {
|
||||||
|
return {
|
||||||
|
title: `${submisisonDetails.doc.quiz_title}`,
|
||||||
|
icon: brand.favicon,
|
||||||
|
}
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -40,6 +40,18 @@
|
|||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<div
|
||||||
|
v-else
|
||||||
|
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 submissions') }}
|
||||||
|
</div>
|
||||||
|
<div class="leading-5">
|
||||||
|
{{ __('No quiz submissions found. Please check again later.') }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import {
|
import {
|
||||||
@@ -51,10 +63,14 @@ import {
|
|||||||
ListRows,
|
ListRows,
|
||||||
ListHeader,
|
ListHeader,
|
||||||
ListHeaderItem,
|
ListHeaderItem,
|
||||||
|
usePageMeta,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
|
import { BookOpen } from 'lucide-vue-next'
|
||||||
import { computed, onMounted, inject } from 'vue'
|
import { computed, onMounted, inject } from 'vue'
|
||||||
|
import { sessionStore } from '../stores/session'
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
|
|
||||||
|
const { brand } = sessionStore()
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
|
|
||||||
@@ -105,4 +121,11 @@ const quizColumns = computed(() => {
|
|||||||
const breadcrumbs = computed(() => {
|
const breadcrumbs = computed(() => {
|
||||||
return [{ label: __('Quiz Submissions') }]
|
return [{ label: __('Quiz Submissions') }]
|
||||||
})
|
})
|
||||||
|
|
||||||
|
usePageMeta(() => {
|
||||||
|
return {
|
||||||
|
title: __('Quiz Submissions'),
|
||||||
|
icon: brand.favicon,
|
||||||
|
}
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -79,12 +79,14 @@ import {
|
|||||||
ListRow,
|
ListRow,
|
||||||
ListHeader,
|
ListHeader,
|
||||||
ListHeaderItem,
|
ListHeaderItem,
|
||||||
|
usePageMeta,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
import { computed, inject, onMounted } from 'vue'
|
import { computed, inject, onMounted } from 'vue'
|
||||||
import { BookOpen, Plus } from 'lucide-vue-next'
|
import { BookOpen, Plus } from 'lucide-vue-next'
|
||||||
import { updateDocumentTitle } from '@/utils'
|
import { sessionStore } from '@/stores/session'
|
||||||
|
|
||||||
|
const { brand } = sessionStore()
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
|
|
||||||
@@ -143,12 +145,10 @@ const breadcrumbs = computed(() => {
|
|||||||
]
|
]
|
||||||
})
|
})
|
||||||
|
|
||||||
const pageMeta = computed(() => {
|
usePageMeta(() => {
|
||||||
return {
|
return {
|
||||||
title: __('Quizzes'),
|
title: __('Quizzes'),
|
||||||
description: __('List of quizzes'),
|
icon: brand.favicon,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
updateDocumentTitle(pageMeta)
|
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -39,11 +39,13 @@ import {
|
|||||||
createDocumentResource,
|
createDocumentResource,
|
||||||
createListResource,
|
createListResource,
|
||||||
createResource,
|
createResource,
|
||||||
|
usePageMeta,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import { computed, inject, onBeforeMount, ref } from 'vue'
|
import { computed, inject, onBeforeMount, ref } from 'vue'
|
||||||
import { useSidebar } from '@/stores/sidebar'
|
import { useSidebar } from '@/stores/sidebar'
|
||||||
import { updateDocumentTitle } from '@/utils'
|
import { sessionStore } from '../stores/session'
|
||||||
|
|
||||||
|
const { brand } = sessionStore()
|
||||||
const sidebarStore = useSidebar()
|
const sidebarStore = useSidebar()
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
const readyToRender = ref(false)
|
const readyToRender = ref(false)
|
||||||
@@ -195,14 +197,10 @@ const breadcrumbs = computed(() => {
|
|||||||
]
|
]
|
||||||
})
|
})
|
||||||
|
|
||||||
const pageMeta = computed(() => {
|
usePageMeta(() => {
|
||||||
return {
|
return {
|
||||||
title: chapter?.doc?.title,
|
title: chapter.doc?.title,
|
||||||
description: __('This is a chapter in the course {0}').format(
|
icon: brand.favicon,
|
||||||
chapter?.doc?.course_title
|
|
||||||
),
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
updateDocumentTitle(pageMeta)
|
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -117,9 +117,9 @@
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { createResource, Breadcrumbs } from 'frappe-ui'
|
import { createResource, Breadcrumbs, usePageMeta } from 'frappe-ui'
|
||||||
import { computed, inject } from 'vue'
|
import { computed } from 'vue'
|
||||||
import { updateDocumentTitle } from '@/utils'
|
import { sessionStore } from '../stores/session'
|
||||||
import { formatNumber } from '@/utils'
|
import { formatNumber } from '@/utils'
|
||||||
import { Line, Pie } from 'vue-chartjs'
|
import { Line, Pie } from 'vue-chartjs'
|
||||||
import {
|
import {
|
||||||
@@ -154,7 +154,7 @@ import {
|
|||||||
BookOpenCheck,
|
BookOpenCheck,
|
||||||
} from 'lucide-vue-next'
|
} from 'lucide-vue-next'
|
||||||
|
|
||||||
const dayjs = inject('$dayjs')
|
const { brand } = sessionStore()
|
||||||
|
|
||||||
const breadcrumbs = computed(() => {
|
const breadcrumbs = computed(() => {
|
||||||
return [
|
return [
|
||||||
@@ -317,12 +317,10 @@ const chartOptions = (isPie) => {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const pageMeta = computed(() => {
|
usePageMeta(() => {
|
||||||
return {
|
return {
|
||||||
title: 'Statistics',
|
title: __('Statistics'),
|
||||||
description: 'Statistics of the platform',
|
icon: brand.favicon,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
updateDocumentTitle(pageMeta)
|
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -2,10 +2,11 @@ import { defineStore } from 'pinia'
|
|||||||
import { createResource } from 'frappe-ui'
|
import { createResource } from 'frappe-ui'
|
||||||
import { usersStore } from './user'
|
import { usersStore } from './user'
|
||||||
import router from '@/router'
|
import router from '@/router'
|
||||||
import { ref, computed } from 'vue'
|
import { computed, reactive, ref } from 'vue'
|
||||||
|
|
||||||
export const sessionStore = defineStore('lms-session', () => {
|
export const sessionStore = defineStore('lms-session', () => {
|
||||||
let { userResource } = usersStore()
|
let { userResource } = usersStore()
|
||||||
|
const brand = reactive({})
|
||||||
|
|
||||||
function sessionUser() {
|
function sessionUser() {
|
||||||
let cookies = new URLSearchParams(document.cookie.split('; ').join('&'))
|
let cookies = new URLSearchParams(document.cookie.split('; ').join('&'))
|
||||||
@@ -46,7 +47,11 @@ export const sessionStore = defineStore('lms-session', () => {
|
|||||||
cache: 'brand',
|
cache: 'brand',
|
||||||
auto: true,
|
auto: true,
|
||||||
onSuccess(data) {
|
onSuccess(data) {
|
||||||
document.querySelector("link[rel='icon']").href = data.favicon
|
brand.name = data.app_name
|
||||||
|
brand.logo = data.app_logo
|
||||||
|
brand.favicon =
|
||||||
|
data.favicon?.file_url ||
|
||||||
|
'/assets/lms/frontend/public/learning.svg'
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -61,6 +66,7 @@ export const sessionStore = defineStore('lms-session', () => {
|
|||||||
isLoggedIn,
|
isLoggedIn,
|
||||||
login,
|
login,
|
||||||
logout,
|
logout,
|
||||||
|
brand,
|
||||||
branding,
|
branding,
|
||||||
sidebarSettings,
|
sidebarSettings,
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,15 +9,9 @@ export const useSettings = defineStore('settings', () => {
|
|||||||
const activeTab = ref(null)
|
const activeTab = ref(null)
|
||||||
|
|
||||||
const learningPaths = createResource({
|
const learningPaths = createResource({
|
||||||
url: 'frappe.client.get_single_value',
|
url: 'lms.lms.api.is_learning_path_enabled',
|
||||||
makeParams(values) {
|
auto: true,
|
||||||
return {
|
cache: ['learningPath'],
|
||||||
doctype: 'LMS Settings',
|
|
||||||
field: 'enable_learning_paths',
|
|
||||||
}
|
|
||||||
},
|
|
||||||
auto: isLoggedIn ? true : false,
|
|
||||||
cache: ['learningPaths'],
|
|
||||||
})
|
})
|
||||||
|
|
||||||
const allowGuestAccess = createResource({
|
const allowGuestAccess = createResource({
|
||||||
@@ -26,12 +20,6 @@ export const useSettings = defineStore('settings', () => {
|
|||||||
cache: ['allowGuestAccess'],
|
cache: ['allowGuestAccess'],
|
||||||
})
|
})
|
||||||
|
|
||||||
/* const onboardingDetails = createResource({
|
|
||||||
url: 'lms.lms.utils.is_onboarding_complete',
|
|
||||||
auto: isLoggedIn ? true : false,
|
|
||||||
cache: ['onboardingDetails'],
|
|
||||||
}) */
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
isSettingsOpen,
|
isSettingsOpen,
|
||||||
activeTab,
|
activeTab,
|
||||||
|
|||||||
@@ -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'>
|
||||||
|
|||||||
@@ -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();
|
||||||
|
|||||||
@@ -14,6 +14,7 @@ import dayjs from '@/utils/dayjs'
|
|||||||
import Embed from '@editorjs/embed'
|
import Embed from '@editorjs/embed'
|
||||||
import SimpleImage from '@editorjs/simple-image'
|
import SimpleImage from '@editorjs/simple-image'
|
||||||
import Table from '@editorjs/table'
|
import Table from '@editorjs/table'
|
||||||
|
import { usersStore } from '../stores/user'
|
||||||
|
|
||||||
export function createToast(options) {
|
export function createToast(options) {
|
||||||
toast({
|
toast({
|
||||||
@@ -158,7 +159,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 +178,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 +442,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 [
|
||||||
{
|
{
|
||||||
@@ -551,3 +568,8 @@ export const escapeHTML = (text) => {
|
|||||||
(char) => escape_html_mapping[char] || char
|
(char) => escape_html_mapping[char] || char
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const canCreateCourse = () => {
|
||||||
|
const { userResource } = usersStore()
|
||||||
|
return userResource.data?.is_instructor || userResource.data?.is_moderator
|
||||||
|
}
|
||||||
|
|||||||
@@ -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())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -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'>
|
||||||
|
|||||||
@@ -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(
|
||||||
|
|||||||
@@ -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',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -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', 'onb2'],
|
||||||
},
|
},
|
||||||
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',
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -1 +1 @@
|
|||||||
__version__ = "2.24.0"
|
__version__ = "2.27.0"
|
||||||
|
|||||||
11
lms/hooks.py
11
lms/hooks.py
@@ -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 = [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -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()
|
||||||
|
|||||||
@@ -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,
|
||||||
@@ -23,7 +22,10 @@ from frappe.utils import (
|
|||||||
from lms.lms.utils import get_average_rating, get_lesson_count
|
from lms.lms.utils import get_average_rating, get_lesson_count
|
||||||
from xml.dom.minidom import parseString
|
from xml.dom.minidom import parseString
|
||||||
from lms.lms.doctype.course_lesson.course_lesson import save_progress
|
from lms.lms.doctype.course_lesson.course_lesson import save_progress
|
||||||
from frappe.integrations.frappe_providers.frappecloud_billing import is_fc_site
|
from frappe.integrations.frappe_providers.frappecloud_billing import (
|
||||||
|
is_fc_site,
|
||||||
|
current_site_info,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
@frappe.whitelist()
|
@frappe.whitelist()
|
||||||
@@ -175,8 +177,14 @@ 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
|
||||||
|
if user.is_fc_site and user.is_system_manager:
|
||||||
|
user.site_info = current_site_info()
|
||||||
|
user.sitename = frappe.local.site
|
||||||
return user
|
return user
|
||||||
|
|
||||||
|
|
||||||
@@ -223,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",
|
||||||
@@ -267,6 +281,7 @@ def get_job_details(job):
|
|||||||
"type",
|
"type",
|
||||||
"company_name",
|
"company_name",
|
||||||
"company_logo",
|
"company_logo",
|
||||||
|
"company_website",
|
||||||
"name",
|
"name",
|
||||||
"creation",
|
"creation",
|
||||||
"description",
|
"description",
|
||||||
@@ -1244,6 +1259,11 @@ def is_guest_allowed():
|
|||||||
return frappe.get_cached_value("LMS Settings", None, "allow_guest_access")
|
return frappe.get_cached_value("LMS Settings", None, "allow_guest_access")
|
||||||
|
|
||||||
|
|
||||||
|
@frappe.whitelist(allow_guest=True)
|
||||||
|
def is_learning_path_enabled():
|
||||||
|
return frappe.get_cached_value("LMS Settings", None, "enable_learning_paths")
|
||||||
|
|
||||||
|
|
||||||
@frappe.whitelist()
|
@frappe.whitelist()
|
||||||
def cancel_evaluation(evaluation):
|
def cancel_evaluation(evaluation):
|
||||||
evaluation = frappe._dict(evaluation)
|
evaluation = frappe._dict(evaluation)
|
||||||
@@ -1289,10 +1309,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
|
||||||
|
|||||||
@@ -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": []
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
@@ -161,7 +161,7 @@
|
|||||||
],
|
],
|
||||||
"index_web_pages_for_search": 1,
|
"index_web_pages_for_search": 1,
|
||||||
"links": [],
|
"links": [],
|
||||||
"modified": "2024-11-14 13:46:56.838659",
|
"modified": "2025-04-10 15:19:22.400932",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "LMS",
|
"module": "LMS",
|
||||||
"name": "Course Lesson",
|
"name": "Course Lesson",
|
||||||
@@ -189,14 +189,28 @@
|
|||||||
"print": 1,
|
"print": 1,
|
||||||
"read": 1,
|
"read": 1,
|
||||||
"report": 1,
|
"report": 1,
|
||||||
"role": "LMS Student",
|
"role": "Course Creator",
|
||||||
|
"select": 1,
|
||||||
|
"share": 1,
|
||||||
|
"write": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"create": 1,
|
||||||
|
"delete": 1,
|
||||||
|
"email": 1,
|
||||||
|
"export": 1,
|
||||||
|
"print": 1,
|
||||||
|
"read": 1,
|
||||||
|
"report": 1,
|
||||||
|
"role": "Moderator",
|
||||||
"select": 1,
|
"select": 1,
|
||||||
"share": 1,
|
"share": 1,
|
||||||
"write": 1
|
"write": 1
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"row_format": "Dynamic",
|
||||||
"sort_field": "modified",
|
"sort_field": "modified",
|
||||||
"sort_order": "DESC",
|
"sort_order": "DESC",
|
||||||
"states": [],
|
"states": [],
|
||||||
"track_changes": 1
|
"track_changes": 1
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,66 +11,34 @@ import json
|
|||||||
|
|
||||||
|
|
||||||
class CourseLesson(Document):
|
class CourseLesson(Document):
|
||||||
def validate(self):
|
def on_update(self):
|
||||||
# self.check_and_create_folder()
|
|
||||||
self.validate_quiz_id()
|
self.validate_quiz_id()
|
||||||
|
|
||||||
def validate_quiz_id(self):
|
def validate_quiz_id(self):
|
||||||
if self.quiz_id and not frappe.db.exists("LMS Quiz", self.quiz_id):
|
if self.quiz_id and not frappe.db.exists("LMS Quiz", self.quiz_id):
|
||||||
frappe.throw(_("Invalid Quiz ID"))
|
frappe.throw(_("Invalid Quiz ID"))
|
||||||
|
|
||||||
def on_update(self):
|
if self.content:
|
||||||
dynamic_documents = ["Exercise", "Quiz"]
|
self.save_lesson_details_in_quiz(self.content)
|
||||||
for section in dynamic_documents:
|
|
||||||
self.update_lesson_name_in_document(section)
|
|
||||||
|
|
||||||
def update_lesson_name_in_document(self, section):
|
if self.instructor_content:
|
||||||
doctype_map = {"Exercise": "LMS Exercise", "Quiz": "LMS Quiz"}
|
self.save_lesson_details_in_quiz(self.instructor_content)
|
||||||
macros = find_macros(self.body)
|
|
||||||
documents = [value for name, value in macros if name == section]
|
|
||||||
index = 1
|
|
||||||
for name in documents:
|
|
||||||
e = frappe.get_doc(doctype_map[section], name)
|
|
||||||
e.lesson = self.name
|
|
||||||
e.index_ = index
|
|
||||||
e.course = self.course
|
|
||||||
e.save(ignore_permissions=True)
|
|
||||||
index += 1
|
|
||||||
self.update_orphan_documents(doctype_map[section], documents)
|
|
||||||
|
|
||||||
def update_orphan_documents(self, doctype, documents):
|
def save_lesson_details_in_quiz(self, content):
|
||||||
"""Updates the documents that were previously part of this lesson,
|
content = json.loads(self.content)
|
||||||
but not any more.
|
for block in content.get("blocks"):
|
||||||
"""
|
if block.get("type") == "quiz":
|
||||||
linked_documents = {
|
quiz = block.get("data").get("quiz")
|
||||||
row["name"] for row in frappe.get_all(doctype, {"lesson": self.name})
|
if not frappe.db.exists("LMS Quiz", quiz):
|
||||||
}
|
frappe.throw(_("Invalid Quiz ID in content"))
|
||||||
active_documents = set(documents)
|
frappe.db.set_value(
|
||||||
orphan_documents = linked_documents - active_documents
|
"LMS Quiz",
|
||||||
for name in orphan_documents:
|
quiz,
|
||||||
ex = frappe.get_doc(doctype, name)
|
{
|
||||||
ex.lesson = None
|
"course": self.course,
|
||||||
ex.course = None
|
"lesson": self.name,
|
||||||
ex.index_ = 0
|
},
|
||||||
ex.save(ignore_permissions=True)
|
)
|
||||||
|
|
||||||
def check_and_create_folder(self):
|
|
||||||
args = {
|
|
||||||
"doctype": "File",
|
|
||||||
"is_folder": True,
|
|
||||||
"file_name": f"{self.name} {self.course}",
|
|
||||||
}
|
|
||||||
if not frappe.db.exists(args):
|
|
||||||
folder = frappe.get_doc(args)
|
|
||||||
folder.save(ignore_permissions=True)
|
|
||||||
|
|
||||||
def get_exercises(self):
|
|
||||||
if not self.body:
|
|
||||||
return []
|
|
||||||
|
|
||||||
macros = find_macros(self.body)
|
|
||||||
exercises = [value for name, value in macros if name == "Exercise"]
|
|
||||||
return [frappe.get_doc("LMS Exercise", name) for name in exercises]
|
|
||||||
|
|
||||||
|
|
||||||
@frappe.whitelist()
|
@frappe.whitelist()
|
||||||
@@ -102,7 +70,7 @@ def save_progress(lesson, course):
|
|||||||
progress = get_course_progress(course)
|
progress = get_course_progress(course)
|
||||||
capture_progress_for_analytics(progress, course)
|
capture_progress_for_analytics(progress, course)
|
||||||
|
|
||||||
# Had to get doc, as on_change doesn't trigger when you use set_value. The trigger is necesary for badge to get assigned.
|
# Had to get doc, as on_change doesn't trigger when you use set_value. The trigger is necessary for badge to get assigned.
|
||||||
enrollment = frappe.get_doc("LMS Enrollment", membership)
|
enrollment = frappe.get_doc("LMS Enrollment", membership)
|
||||||
enrollment.progress = progress
|
enrollment.progress = progress
|
||||||
enrollment.save()
|
enrollment.save()
|
||||||
|
|||||||
@@ -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 = {
|
||||||
|
|||||||
@@ -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",
|
||||||
|
|||||||
@@ -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,
|
||||||
|
|||||||
@@ -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",
|
||||||
|
|||||||
@@ -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")
|
||||||
|
|||||||
@@ -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",
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user