Compare commits
299 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
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 | ||
|
|
dfb82570ea | ||
|
|
e712d6ae42 | ||
|
|
6ffc953370 | ||
|
|
63bf6a5574 | ||
|
|
1e73fc5751 | ||
|
|
65604a0b88 | ||
|
|
5a1a39f5f5 | ||
|
|
d22576c85c | ||
|
|
b7e5332c38 | ||
|
|
ed8570fb88 | ||
|
|
ce69e6634d | ||
|
|
274db20c60 | ||
|
|
3d72072f1f | ||
|
|
ed156c09d7 | ||
|
|
fda3a1a468 | ||
|
|
c261387635 | ||
|
|
7a2fa4dae8 | ||
|
|
b0c41958d9 | ||
|
|
4f1dcbfb78 | ||
|
|
dc9ed099d0 | ||
|
|
95255d44a9 | ||
|
|
5a94e8df75 | ||
|
|
015e3f8490 | ||
|
|
558601f02b | ||
|
|
461d96a079 | ||
|
|
bacfaf4a71 | ||
|
|
0678def698 | ||
|
|
07b0a0af51 | ||
|
|
f12f6cb720 | ||
|
|
4e6c1478f9 | ||
|
|
f9fd36f77e | ||
|
|
db4c7424b3 | ||
|
|
9311043190 | ||
|
|
03915ccfbd | ||
|
|
c6d59216fd | ||
|
|
a8690e41e6 | ||
|
|
cda42b9ec5 | ||
|
|
21a75fdd6d | ||
|
|
a90a1e9855 | ||
|
|
2a046e2e8b | ||
|
|
bb41656d81 | ||
|
|
a88a107718 | ||
|
|
2d21469f91 | ||
|
|
960ebe4a79 | ||
|
|
46dba0c394 | ||
|
|
ba27e8ca95 | ||
|
|
30574ea0fd | ||
|
|
c3c985c4a1 | ||
|
|
7b3d2d8812 | ||
|
|
d573a9f008 | ||
|
|
85a05f56b2 | ||
|
|
904adfb905 | ||
|
|
b2201c29fd | ||
|
|
fe01f68623 | ||
|
|
531c8ebe94 | ||
|
|
52dfb5a360 | ||
|
|
7e04e7e461 | ||
|
|
bce47f606d | ||
|
|
4dc1fdfdd8 | ||
|
|
9a852b52bc | ||
|
|
71a57b1fc0 | ||
|
|
d634598db1 | ||
|
|
6377d682a4 | ||
|
|
6e1acfdc24 | ||
|
|
30ec1dfd7c | ||
|
|
3d209024dd | ||
|
|
9ce64a037d | ||
|
|
43117bc035 | ||
|
|
2af704043e | ||
|
|
fa14ffdcba | ||
|
|
492b715ea0 | ||
|
|
d452e20b8a | ||
|
|
6b634c15d9 | ||
|
|
eeaec3369f | ||
|
|
ce1eece90d | ||
|
|
030bff6592 | ||
|
|
65de46a59e | ||
|
|
974f67aefe | ||
|
|
e374ae3229 | ||
|
|
8b1058e577 | ||
|
|
aaa2eea5e6 | ||
|
|
54047e3c2c | ||
|
|
50fe94e47b | ||
|
|
6999f6641a | ||
|
|
c2b12aa65f | ||
|
|
1a731b6908 | ||
|
|
837d050628 | ||
|
|
8b00bec49c | ||
|
|
9ade643af0 | ||
|
|
a29b92a886 | ||
|
|
e2c28e211f | ||
|
|
65f5b6a0a4 | ||
|
|
905e240fb9 | ||
|
|
75cea1ab78 | ||
|
|
dd3da3dd49 | ||
|
|
5ab9131629 | ||
|
|
8f1c9612b7 | ||
|
|
15a12d2518 | ||
|
|
e83734e0e4 | ||
|
|
f2a95af45c | ||
|
|
1bb61d0c1d | ||
|
|
51fb4f2296 | ||
|
|
5f0f625c0f | ||
|
|
ea7b803905 | ||
|
|
76af3921dd | ||
|
|
e2f999fc31 | ||
|
|
f63d57c4a9 | ||
|
|
ee73790127 | ||
|
|
1c3e84e9bb | ||
|
|
451a151ce0 | ||
|
|
1ac4f819ec | ||
|
|
526eba0129 | ||
|
|
8638e0a1f9 | ||
|
|
69c1093c93 | ||
|
|
74cd0a4d40 | ||
|
|
e28fc3bee6 | ||
|
|
879dfac111 | ||
|
|
b6cfcd797b | ||
|
|
2ea73888f0 | ||
|
|
f43331967c | ||
|
|
9da1249e51 | ||
|
|
2342dfe452 | ||
|
|
e24d22c348 | ||
|
|
533d9545de | ||
|
|
03c0c3c821 | ||
|
|
05be628afb | ||
|
|
cb2dc3e645 | ||
|
|
25f3d2fb9f | ||
|
|
db39a6416c | ||
|
|
48e0787344 | ||
|
|
838de2f692 | ||
|
|
1953d89e3c | ||
|
|
d0898d4c75 | ||
|
|
f01bb1aecb | ||
|
|
bbdbda4942 | ||
|
|
7741696011 | ||
|
|
2d4567bfbd | ||
|
|
8f643dae27 | ||
|
|
81e287ffe5 | ||
|
|
5543aa5e02 | ||
|
|
b5a7b4cd2c | ||
|
|
8857ce8146 | ||
|
|
bfbc5f600f | ||
|
|
a8fa42db00 | ||
|
|
4ee2bfcf32 | ||
|
|
ab98884f77 | ||
|
|
dbf443300b | ||
|
|
dbf44a7a85 | ||
|
|
2818c95795 | ||
|
|
27a13a6151 | ||
|
|
9f974786f2 | ||
|
|
2f2f41ac3c | ||
|
|
d5d30f683a | ||
|
|
56007aa4ba | ||
|
|
d489e08718 | ||
|
|
16b9356944 | ||
|
|
ba26826896 | ||
|
|
49631b6e56 | ||
|
|
ae2bffc56d | ||
|
|
47e51c4787 | ||
|
|
06ef289427 | ||
|
|
4190f39993 | ||
|
|
26a22375c8 | ||
|
|
0c174caf86 | ||
|
|
661748adc1 | ||
|
|
73f24339e3 | ||
|
|
9775d7425c | ||
|
|
3ff6c96273 | ||
|
|
f9706f10e1 | ||
|
|
e9a20c61d5 | ||
|
|
f3ee1a84dd | ||
|
|
381ca43c01 | ||
|
|
8cc16dc51b | ||
|
|
4337603e33 | ||
|
|
5c39acb745 | ||
|
|
1b584f0b88 | ||
|
|
68a28ef6d4 | ||
|
|
867df7f2c7 | ||
|
|
c18e84bb8e | ||
|
|
3fc1fd9dbc | ||
|
|
bc284c327c | ||
|
|
85961c76fb | ||
|
|
1c11a5964b | ||
|
|
4d1ba4ea3f | ||
|
|
6d3e24fce9 | ||
|
|
de37ec5704 | ||
|
|
745592432c | ||
|
|
cf47965e8c | ||
|
|
3d64872352 | ||
|
|
b89ad4204c | ||
|
|
71e9ba849d | ||
|
|
1d412175c6 | ||
|
|
b282a37a04 | ||
|
|
5f6d0bcf25 | ||
|
|
74c2d5eb06 | ||
|
|
4618d3b30e | ||
|
|
9e32e8f499 | ||
|
|
f47e2e758b | ||
|
|
9e03e30bd8 | ||
|
|
6be0e6bfca | ||
|
|
7bbdedf5f4 | ||
|
|
e942e6a2f5 | ||
|
|
6162df7013 | ||
|
|
a28227ad75 | ||
|
|
ed8baf3327 | ||
|
|
1ac5de96f9 | ||
|
|
15dd4c4350 | ||
|
|
c986089e77 | ||
|
|
17dc77f061 | ||
|
|
189f353de0 | ||
|
|
845e7174f0 | ||
|
|
8c6e4ad3ee | ||
|
|
5dfddc890c | ||
|
|
1ebabc23d3 | ||
|
|
1bf8c1c763 | ||
|
|
c5a59b6370 | ||
|
|
4a5a777478 | ||
|
|
4fd7dcd5b2 | ||
|
|
55920d9e3f | ||
|
|
6d0c3c9cd8 | ||
|
|
7b20c3fe03 | ||
|
|
efbe35c836 | ||
|
|
e591cd74ab | ||
|
|
669b9c73be | ||
|
|
52e1dd6d33 | ||
|
|
828e195b81 | ||
|
|
145342bb72 | ||
|
|
58abfd004d | ||
|
|
9dc8322270 | ||
|
|
4f0a6a7d57 | ||
|
|
2fb8ae00b9 | ||
|
|
63da1e384d | ||
|
|
34685ebdb2 | ||
|
|
215ae941e1 | ||
|
|
9d1211e872 | ||
|
|
cd4f2b1039 | ||
|
|
9881b7b498 | ||
|
|
28a687f6bf | ||
|
|
bd43ed0e88 | ||
|
|
17b59ce4e5 | ||
|
|
7acc1864c8 | ||
|
|
5a6fdfcbc3 | ||
|
|
23d465d4a1 | ||
|
|
27ae014fcb | ||
|
|
b4c7338b76 | ||
|
|
0d1464c5e9 | ||
|
|
f4421d362c | ||
|
|
5c8378f2d4 | ||
|
|
8401e86acb |
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 }}
|
||||||
|
|||||||
2
.github/workflows/ui-tests.yml
vendored
2
.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') }}
|
||||||
|
|||||||
@@ -13,6 +13,6 @@ module.exports = defineConfig({
|
|||||||
openMode: 0,
|
openMode: 0,
|
||||||
},
|
},
|
||||||
e2e: {
|
e2e: {
|
||||||
baseUrl: "http://lms1:8000",
|
baseUrl: "http://testui:8000",
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ describe("Course Creation", () => {
|
|||||||
cy.visit("/lms/courses");
|
cy.visit("/lms/courses");
|
||||||
|
|
||||||
// Create a course
|
// Create a course
|
||||||
cy.get("header").children().last().children().last().click();
|
cy.get("button").contains("New").click();
|
||||||
cy.wait(1000);
|
cy.wait(1000);
|
||||||
cy.url().should("include", "/courses/new/edit");
|
cy.url().should("include", "/courses/new/edit");
|
||||||
|
|
||||||
@@ -84,9 +84,8 @@ describe("Course Creation", () => {
|
|||||||
cy.wait(1000);
|
cy.wait(1000);
|
||||||
|
|
||||||
cy.get("label").contains("Title").type("Test Lesson");
|
cy.get("label").contains("Title").type("Test Lesson");
|
||||||
|
|
||||||
cy.get("#content .ce-block").type(
|
cy.get("#content .ce-block").type(
|
||||||
"This is an extremely big paragraph that is meant to test the UI. This is a very long paragraph. It contains more than once sentence. Its meant to be this long as this is a UI test. Its unbearably long and I'm not sure why I'm typing this much. I'm just going to keep typing until I feel like its long enough. I think its long enough now. I'm going to stop typing now."
|
"{enter}This is an extremely big paragraph that is meant to test the UI. This is a very long paragraph. It contains more than once sentence. Its meant to be this long as this is a UI test. Its unbearably long and I'm not sure why I'm typing this much. I'm just going to keep typing until I feel like its long enough. I think its long enough now. I'm going to stop typing now."
|
||||||
);
|
);
|
||||||
cy.button("Save").click();
|
cy.button("Save").click();
|
||||||
|
|
||||||
|
|||||||
@@ -37,6 +37,7 @@ Cypress.Commands.add("login", (email, password) => {
|
|||||||
url: "/api/method/login",
|
url: "/api/method/login",
|
||||||
method: "POST",
|
method: "POST",
|
||||||
body: { usr: email, pwd: password },
|
body: { usr: email, pwd: password },
|
||||||
|
timeout: 60000,
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
Submodule frappe-ui updated: 8cd9b06a5e...70bc4760e4
@@ -2,7 +2,7 @@
|
|||||||
<html lang="en">
|
<html lang="en">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="UTF-8" />
|
<meta charset="UTF-8" />
|
||||||
<link rel="icon" href="/favicon.png" />
|
<link rel="icon" href="{{ favicon or '/assets/lms/frontend/favicon.png' }}" />
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||||
<title>Frappe Learning</title>
|
<title>Frappe Learning</title>
|
||||||
<meta name="title" content="{{ meta.title }}" />
|
<meta name="title" content="{{ meta.title }}" />
|
||||||
@@ -42,6 +42,7 @@
|
|||||||
|
|
||||||
<script>
|
<script>
|
||||||
window.csrf_token = '{{ csrf_token }}'
|
window.csrf_token = '{{ csrf_token }}'
|
||||||
|
window.setup_complete = '{{ setup_complete }}'
|
||||||
document.getElementById('seo-content').style.display = 'none';
|
document.getElementById('seo-content').style.display = 'none';
|
||||||
</script>
|
</script>
|
||||||
<script type="module" src="/src/main.js"></script>
|
<script type="module" src="/src/main.js"></script>
|
||||||
|
|||||||
@@ -19,13 +19,14 @@
|
|||||||
"@editorjs/paragraph": "^2.11.3",
|
"@editorjs/paragraph": "^2.11.3",
|
||||||
"@editorjs/simple-image": "^1.6.0",
|
"@editorjs/simple-image": "^1.6.0",
|
||||||
"@editorjs/table": "^2.4.2",
|
"@editorjs/table": "^2.4.2",
|
||||||
|
"@vueuse/router": "^12.7.0",
|
||||||
"ace-builds": "^1.36.2",
|
"ace-builds": "^1.36.2",
|
||||||
"apexcharts": "^4.3.0",
|
"apexcharts": "^4.3.0",
|
||||||
"chart.js": "^4.4.1",
|
"chart.js": "^4.4.1",
|
||||||
"codemirror-editor-vue3": "^2.8.0",
|
"codemirror-editor-vue3": "^2.8.0",
|
||||||
"dayjs": "^1.11.6",
|
"dayjs": "^1.11.6",
|
||||||
"feather-icons": "^4.28.0",
|
"feather-icons": "^4.28.0",
|
||||||
"frappe-ui": "^0.1.89",
|
"frappe-ui": "^0.1.112",
|
||||||
"lucide-vue-next": "^0.383.0",
|
"lucide-vue-next": "^0.383.0",
|
||||||
"markdown-it": "^14.0.0",
|
"markdown-it": "^14.0.0",
|
||||||
"pinia": "^2.0.33",
|
"pinia": "^2.0.33",
|
||||||
|
|||||||
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@@ -5,7 +5,7 @@
|
|||||||
<div class="flex items-center justify-between mb-2">
|
<div class="flex items-center justify-between mb-2">
|
||||||
<div class="flex items-center">
|
<div class="flex items-center">
|
||||||
<Avatar :label="comm.sender_full_name" size="lg" />
|
<Avatar :label="comm.sender_full_name" size="lg" />
|
||||||
<div class="ml-2">
|
<div class="ml-2 text-ink-gray-7">
|
||||||
{{ comm.sender_full_name }}
|
{{ comm.sender_full_name }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -14,13 +14,13 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
class="prose prose-sm bg-gray-50 !min-w-full px-4 py-2 rounded-md"
|
class="prose prose-sm bg-surface-menu-bar !min-w-full px-4 py-2 rounded-md"
|
||||||
v-html="comm.content"
|
v-html="comm.content"
|
||||||
></div>
|
></div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div v-else class="text-sm italic text-gray-600">
|
<div v-else class="text-sm italic text-ink-gray-5">
|
||||||
{{ __('No announcements') }}
|
{{ __('No announcements') }}
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<template>
|
<template>
|
||||||
<div
|
<div
|
||||||
class="flex h-full flex-col justify-between transition-all duration-300 ease-in-out bg-gray-50"
|
class="flex h-full flex-col justify-between transition-all duration-300 ease-in-out border-r bg-surface-menu-bar"
|
||||||
:class="sidebarStore.isSidebarCollapsed ? 'w-14' : 'w-56'"
|
:class="sidebarStore.isSidebarCollapsed ? 'w-14' : 'w-56'"
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
@@ -23,16 +23,16 @@
|
|||||||
<div
|
<div
|
||||||
class="flex items-center justify-between pr-2 cursor-pointer"
|
class="flex items-center justify-between pr-2 cursor-pointer"
|
||||||
:class="sidebarStore.isSidebarCollapsed ? 'pl-3' : 'pl-4'"
|
:class="sidebarStore.isSidebarCollapsed ? 'pl-3' : 'pl-4'"
|
||||||
@click="showWebPages = !showWebPages"
|
@click="toggleWebPages"
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
v-if="!sidebarStore.isSidebarCollapsed"
|
v-if="!sidebarStore.isSidebarCollapsed"
|
||||||
class="flex items-center text-sm text-gray-600 my-1"
|
class="flex items-center text-sm text-ink-gray-5 my-1"
|
||||||
>
|
>
|
||||||
<span class="grid h-5 w-6 flex-shrink-0 place-items-center">
|
<span class="grid h-5 w-6 flex-shrink-0 place-items-center">
|
||||||
<ChevronRight
|
<ChevronRight
|
||||||
class="h-4 w-4 stroke-1.5 text-gray-900 transition-all duration-300 ease-in-out"
|
class="h-4 w-4 stroke-1.5 text-ink-gray-9 transition-all duration-300 ease-in-out"
|
||||||
:class="{ 'rotate-90': showWebPages }"
|
:class="{ 'rotate-90': !sidebarStore.isWebpagesCollapsed }"
|
||||||
/>
|
/>
|
||||||
</span>
|
</span>
|
||||||
<span class="ml-2">
|
<span class="ml-2">
|
||||||
@@ -41,14 +41,14 @@
|
|||||||
</div>
|
</div>
|
||||||
<Button v-if="isModerator" variant="ghost" @click="openPageModal()">
|
<Button v-if="isModerator" variant="ghost" @click="openPageModal()">
|
||||||
<template #icon>
|
<template #icon>
|
||||||
<Plus class="h-4 w-4 text-gray-700 stroke-1.5" />
|
<Plus class="h-4 w-4 text-ink-gray-7 stroke-1.5" />
|
||||||
</template>
|
</template>
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
v-if="sidebarSettings.data?.web_pages?.length"
|
v-if="sidebarSettings.data?.web_pages?.length"
|
||||||
class="flex flex-col transition-all duration-300 ease-in-out"
|
class="flex flex-col transition-all duration-300 ease-in-out"
|
||||||
:class="showWebPages ? 'block' : 'hidden'"
|
:class="!sidebarStore.isWebpagesCollapsed ? 'block' : 'hidden'"
|
||||||
>
|
>
|
||||||
<SidebarLink
|
<SidebarLink
|
||||||
v-for="link in sidebarSettings.data.web_pages"
|
v-for="link in sidebarSettings.data.web_pages"
|
||||||
@@ -62,25 +62,33 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<SidebarLink
|
<div>
|
||||||
:link="{
|
<TrialBanner
|
||||||
label: sidebarStore.isSidebarCollapsed ? 'Expand' : 'Collapse',
|
v-if="
|
||||||
}"
|
userResource.data?.is_system_manager && userResource.data?.is_fc_site
|
||||||
:isCollapsed="sidebarStore.isSidebarCollapsed"
|
"
|
||||||
@click="toggleSidebar()"
|
:isSidebarCollapsed="sidebarStore.isSidebarCollapsed"
|
||||||
class="m-2"
|
/>
|
||||||
>
|
<SidebarLink
|
||||||
<template #icon>
|
:link="{
|
||||||
<span class="grid h-5 w-6 flex-shrink-0 place-items-center">
|
label: sidebarStore.isSidebarCollapsed ? 'Expand' : 'Collapse',
|
||||||
<CollapseSidebar
|
}"
|
||||||
class="h-4.5 w-4.5 text-gray-700 duration-300 ease-in-out"
|
:isCollapsed="sidebarStore.isSidebarCollapsed"
|
||||||
:class="{
|
@click="toggleSidebar()"
|
||||||
'[transform:rotateY(180deg)]': sidebarStore.isSidebarCollapsed,
|
class="m-2"
|
||||||
}"
|
>
|
||||||
/>
|
<template #icon>
|
||||||
</span>
|
<span class="grid h-5 w-6 flex-shrink-0 place-items-center">
|
||||||
</template>
|
<CollapseSidebar
|
||||||
</SidebarLink>
|
class="h-4.5 w-4.5 text-ink-gray-7 duration-300 ease-in-out"
|
||||||
|
:class="{
|
||||||
|
'[transform:rotateY(180deg)]': sidebarStore.isSidebarCollapsed,
|
||||||
|
}"
|
||||||
|
/>
|
||||||
|
</span>
|
||||||
|
</template>
|
||||||
|
</SidebarLink>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<PageModal
|
<PageModal
|
||||||
v-model="showPageModal"
|
v-model="showPageModal"
|
||||||
@@ -101,7 +109,7 @@ import { sessionStore } from '@/stores/session'
|
|||||||
import { useSidebar } from '@/stores/sidebar'
|
import { useSidebar } from '@/stores/sidebar'
|
||||||
import { useSettings } from '@/stores/settings'
|
import { useSettings } from '@/stores/settings'
|
||||||
import { ChevronRight, Plus } from 'lucide-vue-next'
|
import { ChevronRight, Plus } from 'lucide-vue-next'
|
||||||
import { createResource, Button } from 'frappe-ui'
|
import { Button, createResource, TrialBanner } from 'frappe-ui'
|
||||||
import PageModal from '@/components/Modals/PageModal.vue'
|
import PageModal from '@/components/Modals/PageModal.vue'
|
||||||
|
|
||||||
const { user, sidebarSettings } = sessionStore()
|
const { user, sidebarSettings } = sessionStore()
|
||||||
@@ -114,7 +122,6 @@ const showPageModal = ref(false)
|
|||||||
const isModerator = ref(false)
|
const isModerator = ref(false)
|
||||||
const isInstructor = ref(false)
|
const isInstructor = ref(false)
|
||||||
const pageToEdit = ref(null)
|
const pageToEdit = ref(null)
|
||||||
const showWebPages = ref(false)
|
|
||||||
const settingsStore = useSettings()
|
const settingsStore = useSettings()
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
@@ -266,5 +273,17 @@ watch(userResource, () => {
|
|||||||
|
|
||||||
const toggleSidebar = () => {
|
const toggleSidebar = () => {
|
||||||
sidebarStore.isSidebarCollapsed = !sidebarStore.isSidebarCollapsed
|
sidebarStore.isSidebarCollapsed = !sidebarStore.isSidebarCollapsed
|
||||||
|
localStorage.setItem(
|
||||||
|
'isSidebarCollapsed',
|
||||||
|
JSON.stringify(sidebarStore.isSidebarCollapsed)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const toggleWebPages = () => {
|
||||||
|
sidebarStore.isWebpagesCollapsed = !sidebarStore.isWebpagesCollapsed
|
||||||
|
localStorage.setItem(
|
||||||
|
'isWebpagesCollapsed',
|
||||||
|
JSON.stringify(sidebarStore.isWebpagesCollapsed)
|
||||||
|
)
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
<template #target="{ togglePopover }">
|
<template #target="{ togglePopover }">
|
||||||
<button
|
<button
|
||||||
:class="[
|
:class="[
|
||||||
'group w-full flex h-7 items-center justify-between rounded px-2 text-base text-gray-800 hover:bg-gray-100',
|
'group w-full flex h-7 items-center justify-between rounded px-2 text-base text-ink-gray-8 hover:bg-surface-gray-2',
|
||||||
]"
|
]"
|
||||||
@click.prevent="togglePopover()"
|
@click.prevent="togglePopover()"
|
||||||
>
|
>
|
||||||
@@ -18,15 +18,15 @@
|
|||||||
</template>
|
</template>
|
||||||
<template #body>
|
<template #body>
|
||||||
<div
|
<div
|
||||||
class="grid grid-cols-3 justify-between mx-3 p-2 rounded-lg border border-gray-100 bg-white shadow-xl"
|
class="grid grid-cols-3 justify-between mx-3 p-2 rounded-lg border border-gray-100 bg-surface-white shadow-xl"
|
||||||
>
|
>
|
||||||
<div v-for="app in apps.data" key="name">
|
<div v-for="app in apps.data" key="name">
|
||||||
<a
|
<a
|
||||||
:href="app.route"
|
:href="app.route"
|
||||||
class="flex flex-col gap-1.5 rounded justify-center items-center py-2 px-3 hover:bg-gray-100"
|
class="flex flex-col gap-1.5 rounded justify-center items-center py-2 px-3 hover:bg-surface-gray-2"
|
||||||
>
|
>
|
||||||
<img class="size-8" :src="app.logo" />
|
<img class="size-8" :src="app.logo" />
|
||||||
<div class="text-sm" @click="app.onClick">
|
<div class="text-sm text-ink-gray-7" @click="app.onClick">
|
||||||
{{ app.title }}
|
{{ app.title }}
|
||||||
</div>
|
</div>
|
||||||
</a>
|
</a>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<template>
|
<template>
|
||||||
<div>
|
<div>
|
||||||
<div class="flex items-center justify-between mb-4">
|
<div class="flex items-center justify-between mb-4">
|
||||||
<div class="text-lg font-semibold">
|
<div class="text-lg font-semibold text-ink-gray-9">
|
||||||
{{ __('Assessments') }}
|
{{ __('Assessments') }}
|
||||||
</div>
|
</div>
|
||||||
<Button v-if="canSeeAddButton()" @click="showModal = true">
|
<Button v-if="canSeeAddButton()" @click="showModal = true">
|
||||||
@@ -23,7 +23,7 @@
|
|||||||
}"
|
}"
|
||||||
>
|
>
|
||||||
<ListHeader
|
<ListHeader
|
||||||
class="mb-2 grid items-center space-x-4 rounded bg-gray-100 p-2"
|
class="mb-2 grid items-center space-x-4 rounded bg-surface-gray-2 p-2"
|
||||||
>
|
>
|
||||||
<ListHeaderItem :item="item" v-for="item in getAssessmentColumns()">
|
<ListHeaderItem :item="item" v-for="item in getAssessmentColumns()">
|
||||||
<template #prefix="{ item }">
|
<template #prefix="{ item }">
|
||||||
@@ -71,7 +71,7 @@
|
|||||||
</ListSelectBanner>
|
</ListSelectBanner>
|
||||||
</ListView>
|
</ListView>
|
||||||
</div>
|
</div>
|
||||||
<div v-else class="text-sm italic text-gray-600">
|
<div v-else class="text-sm italic text-ink-gray-5">
|
||||||
{{ __('No Assessments') }}
|
{{ __('No Assessments') }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,11 +1,11 @@
|
|||||||
<template>
|
<template>
|
||||||
<div
|
<div
|
||||||
v-if="assignment.data"
|
v-if="assignment.data"
|
||||||
class="grid grid-cols-[68%,32%] h-full"
|
class="grid grid-cols-[65%,35%] h-full"
|
||||||
:class="{ 'border rounded-lg': !showTitle }"
|
:class="{ 'border rounded-lg': !showTitle }"
|
||||||
>
|
>
|
||||||
<div class="border-r p-5 overflow-y-auto h-[calc(100vh-3.2rem)]">
|
<div class="border-r p-5 overflow-y-auto h-[calc(100vh-3.2rem)]">
|
||||||
<div v-if="showTitle" class="text-lg font-semibold mb-5">
|
<div v-if="showTitle" class="text-lg font-semibold mb-5 text-ink-gray-9">
|
||||||
<div v-if="submissionName === 'new'">
|
<div v-if="submissionName === 'new'">
|
||||||
{{ __('Submission by') }} {{ user.data?.full_name }}
|
{{ __('Submission by') }} {{ user.data?.full_name }}
|
||||||
</div>
|
</div>
|
||||||
@@ -13,19 +13,19 @@
|
|||||||
{{ __('Submission by') }} {{ submissionResource.doc?.member_name }}
|
{{ __('Submission by') }} {{ submissionResource.doc?.member_name }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="text-sm text-gray-600 font-medium mb-2">
|
<div class="text-sm text-ink-gray-7 font-medium mb-2">
|
||||||
{{ __('Question') }}:
|
{{ __('Question') }}:
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
v-html="assignment.data.question"
|
v-html="assignment.data.question"
|
||||||
class="ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-gray-300 prose-th:border-gray-300 prose-td:relative prose-th:relative prose-th:bg-gray-100 prose-sm max-w-none !whitespace-normal"
|
class="ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-outline-gray-2 prose-th:border-outline-gray-2 prose-td:relative prose-th:relative prose-th:bg-surface-gray-2 prose-sm max-w-none !whitespace-normal"
|
||||||
></div>
|
></div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="flex flex-col">
|
<div class="flex flex-col">
|
||||||
<div class="p-5">
|
<div class="p-5">
|
||||||
<div class="flex items-center justify-between mb-4">
|
<div class="flex items-center justify-between mb-4">
|
||||||
<div class="font-semibold">
|
<div class="font-semibold text-ink-gray-9">
|
||||||
{{ __('Submission') }}
|
{{ __('Submission') }}
|
||||||
</div>
|
</div>
|
||||||
<div class="flex items-center space-x-2">
|
<div class="flex items-center space-x-2">
|
||||||
@@ -50,7 +50,7 @@
|
|||||||
!['Pass', 'Fail'].includes(submissionResource.doc?.status) &&
|
!['Pass', 'Fail'].includes(submissionResource.doc?.status) &&
|
||||||
submissionResource.doc?.owner == user.data?.name
|
submissionResource.doc?.owner == user.data?.name
|
||||||
"
|
"
|
||||||
class="bg-blue-100 p-3 rounded-md leading-5 text-sm mb-4"
|
class="bg-surface-blue-2 p-3 rounded-md leading-5 text-sm mb-4"
|
||||||
>
|
>
|
||||||
{{ __("You've successfully submitted the assignment.") }}
|
{{ __("You've successfully submitted the assignment.") }}
|
||||||
{{
|
{{
|
||||||
@@ -61,7 +61,7 @@
|
|||||||
{{ __('Feel free to make edits to your submission if needed.') }}
|
{{ __('Feel free to make edits to your submission if needed.') }}
|
||||||
</div>
|
</div>
|
||||||
<div v-if="showUploader()">
|
<div v-if="showUploader()">
|
||||||
<div class="text-xs text-gray-600 mt-1 mb-2">
|
<div class="text-xs text-ink-gray-5 mt-1 mb-2">
|
||||||
{{ __('Add your assignment as {0}').format(assignment.data.type) }}
|
{{ __('Add your assignment as {0}').format(assignment.data.type) }}
|
||||||
</div>
|
</div>
|
||||||
<FileUploader
|
<FileUploader
|
||||||
@@ -81,32 +81,32 @@
|
|||||||
</template>
|
</template>
|
||||||
</FileUploader>
|
</FileUploader>
|
||||||
<div v-else>
|
<div v-else>
|
||||||
<div class="flex items-center">
|
<div class="flex text-ink-gray-7">
|
||||||
<div class="border rounded-md p-2 mr-2">
|
<div class="border self-start rounded-md p-2 mr-2">
|
||||||
<FileText class="h-5 w-5 stroke-1.5 text-gray-700" />
|
<FileText class="h-5 w-5 stroke-1.5" />
|
||||||
</div>
|
</div>
|
||||||
<a
|
<a
|
||||||
:href="submissionFile.file_url"
|
:href="submissionFile.file_url"
|
||||||
target="_blank"
|
target="_blank"
|
||||||
class="flex flex-col cursor-pointer !no-underline"
|
class="flex flex-col cursor-pointer !no-underline"
|
||||||
>
|
>
|
||||||
<span>
|
<span class="text-sm leading-5">
|
||||||
{{ submissionFile.file_name }}
|
{{ submissionFile.file_name }}
|
||||||
</span>
|
</span>
|
||||||
<span class="text-sm text-gray-500 mt-1">
|
<span class="text-sm text-ink-gray-5 mt-1">
|
||||||
{{ getFileSize(submissionFile.file_size) }}
|
{{ getFileSize(submissionFile.file_size) }}
|
||||||
</span>
|
</span>
|
||||||
</a>
|
</a>
|
||||||
<X
|
<X
|
||||||
v-if="canModifyAssignment"
|
v-if="canModifyAssignment"
|
||||||
@click="removeSubmission()"
|
@click="removeSubmission()"
|
||||||
class="bg-gray-200 rounded-md cursor-pointer stroke-1.5 w-5 h-5 p-1 ml-4"
|
class="bg-surface-gray-3 rounded-md cursor-pointer stroke-1.5 w-5 h-5 p-1 ml-4"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div v-else-if="assignment.data.type == 'URL'">
|
<div v-else-if="assignment.data.type == 'URL'">
|
||||||
<div class="text-xs text-gray-600 mb-1">
|
<div class="text-xs text-ink-gray-5 mb-1">
|
||||||
{{ __('Enter a URL') }}
|
{{ __('Enter a URL') }}
|
||||||
</div>
|
</div>
|
||||||
<FormControl
|
<FormControl
|
||||||
@@ -124,7 +124,7 @@
|
|||||||
@change="(val) => (answer = val)"
|
@change="(val) => (answer = val)"
|
||||||
:editable="true"
|
:editable="true"
|
||||||
:fixedMenu="true"
|
:fixedMenu="true"
|
||||||
editorClass="prose-sm max-w-none border-b border-x bg-gray-100 rounded-b-md py-1 px-2 min-h-[7rem]"
|
editorClass="prose-sm max-w-none border-b border-x bg-surface-gray-2 rounded-b-md py-1 px-2 min-h-[7rem]"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -133,9 +133,9 @@
|
|||||||
user.data?.name == submissionResource.doc?.owner &&
|
user.data?.name == submissionResource.doc?.owner &&
|
||||||
submissionResource.doc?.comments
|
submissionResource.doc?.comments
|
||||||
"
|
"
|
||||||
class="mt-8 p-3 bg-blue-100 rounded-md"
|
class="mt-8 p-3 bg-surface-blue-2 rounded-md"
|
||||||
>
|
>
|
||||||
<div class="text-sm text-gray-600 font-medium mb-2">
|
<div class="text-sm text-ink-gray-5 font-medium mb-2">
|
||||||
{{ __('Comments by Evaluator') }}:
|
{{ __('Comments by Evaluator') }}:
|
||||||
</div>
|
</div>
|
||||||
<div class="leading-5">
|
<div class="leading-5">
|
||||||
@@ -145,7 +145,7 @@
|
|||||||
|
|
||||||
<!-- Grading -->
|
<!-- Grading -->
|
||||||
<div v-if="canGradeSubmission" class="mt-8 space-y-4">
|
<div v-if="canGradeSubmission" class="mt-8 space-y-4">
|
||||||
<div class="font-semibold mb-2">
|
<div class="font-semibold mb-2 text-ink-gray-9">
|
||||||
{{ __('Grading') }}
|
{{ __('Grading') }}
|
||||||
</div>
|
</div>
|
||||||
<FormControl
|
<FormControl
|
||||||
@@ -155,12 +155,23 @@
|
|||||||
type="select"
|
type="select"
|
||||||
:options="submissionStatusOptions"
|
:options="submissionStatusOptions"
|
||||||
/>
|
/>
|
||||||
<FormControl
|
<div>
|
||||||
v-if="submissionResource.doc"
|
<div class="text-sm text-ink-gray-5 mb-1">
|
||||||
v-model="submissionResource.doc.comments"
|
{{ __('Comments') }}
|
||||||
:label="__('Comments')"
|
</div>
|
||||||
type="textarea"
|
<TextEditor
|
||||||
/>
|
:content="comments"
|
||||||
|
@change="
|
||||||
|
(val) => {
|
||||||
|
comments = val
|
||||||
|
isDirty = true
|
||||||
|
}
|
||||||
|
"
|
||||||
|
:editable="true"
|
||||||
|
:fixedMenu="true"
|
||||||
|
editorClass="prose-sm max-w-none border-b border-x bg-surface-gray-2 rounded-b-md py-1 px-2 min-h-[7rem]"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -184,6 +195,7 @@ import { useRouter } from 'vue-router'
|
|||||||
|
|
||||||
const submissionFile = ref(null)
|
const submissionFile = ref(null)
|
||||||
const answer = ref(null)
|
const answer = ref(null)
|
||||||
|
const comments = ref(null)
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
const showTitle = router.currentRoute.value.name == 'AssignmentSubmission'
|
const showTitle = router.currentRoute.value.name == 'AssignmentSubmission'
|
||||||
@@ -281,7 +293,9 @@ watch(submissionResource, () => {
|
|||||||
if (submissionResource.doc.answer) {
|
if (submissionResource.doc.answer) {
|
||||||
answer.value = submissionResource.doc.answer
|
answer.value = submissionResource.doc.answer
|
||||||
}
|
}
|
||||||
|
if (submissionResource.doc.comments) {
|
||||||
|
comments.value = submissionResource.doc.comments
|
||||||
|
}
|
||||||
if (submissionResource.isDirty) {
|
if (submissionResource.isDirty) {
|
||||||
isDirty.value = true
|
isDirty.value = true
|
||||||
} else if (showUploader() && !submissionFile.value) {
|
} else if (showUploader() && !submissionFile.value) {
|
||||||
@@ -306,10 +320,14 @@ const submitAssignment = () => {
|
|||||||
submissionResource.doc && submissionResource.doc.owner != user.data?.name
|
submissionResource.doc && submissionResource.doc.owner != user.data?.name
|
||||||
? user.data?.name
|
? user.data?.name
|
||||||
: null
|
: null
|
||||||
|
|
||||||
submissionResource.setValue.submit(
|
submissionResource.setValue.submit(
|
||||||
{
|
{
|
||||||
...submissionResource.doc,
|
...submissionResource.doc,
|
||||||
|
assignment_attachment: submissionFile.value?.file_url,
|
||||||
evaluator: evaluator,
|
evaluator: evaluator,
|
||||||
|
comments: comments.value,
|
||||||
|
answer: answer.value,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
onSuccess(data) {
|
onSuccess(data) {
|
||||||
@@ -351,6 +369,7 @@ const addNewSubmission = () => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const saveSubmission = (file) => {
|
const saveSubmission = (file) => {
|
||||||
|
isDirty.value = true
|
||||||
submissionFile.value = file
|
submissionFile.value = file
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -401,6 +420,7 @@ const validateFile = (file) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const removeSubmission = () => {
|
const removeSubmission = () => {
|
||||||
|
isDirty.value = true
|
||||||
submissionFile.value = null
|
submissionFile.value = null
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -9,8 +9,8 @@
|
|||||||
<div class="flex items-center space-x-2 shadow rounded-lg p-1 w-1/2">
|
<div class="flex items-center space-x-2 shadow rounded-lg p-1 w-1/2">
|
||||||
<Button variant="ghost" @click="togglePlay">
|
<Button variant="ghost" @click="togglePlay">
|
||||||
<template #icon>
|
<template #icon>
|
||||||
<Play v-if="!isPlaying" class="w-4 h-4 text-gray-900" />
|
<Play v-if="!isPlaying" class="w-4 h-4 text-ink-gray-9" />
|
||||||
<Pause v-else class="w-4 h-4 text-gray-900" />
|
<Pause v-else class="w-4 h-4 text-ink-gray-9" />
|
||||||
</template>
|
</template>
|
||||||
</Button>
|
</Button>
|
||||||
<input
|
<input
|
||||||
@@ -22,13 +22,13 @@
|
|||||||
@input="changeCurrentTime"
|
@input="changeCurrentTime"
|
||||||
class="duration-slider w-full h-1"
|
class="duration-slider w-full h-1"
|
||||||
/>
|
/>
|
||||||
<span class="text-xs text-gray-900 font-medium">
|
<span class="text-xs text-ink-gray-9 font-medium">
|
||||||
{{ formatTime(currentTime) }} / {{ formatTime(duration) }}
|
{{ formatTime(currentTime) }} / {{ formatTime(duration) }}
|
||||||
</span>
|
</span>
|
||||||
<Button variant="ghost" @click="toggleMute">
|
<Button variant="ghost" @click="toggleMute">
|
||||||
<template #icon>
|
<template #icon>
|
||||||
<Volume2 v-if="!isMuted" class="w-4 h-4 text-gray-900" />
|
<Volume2 v-if="!isMuted" class="w-4 h-4 text-ink-gray-9" />
|
||||||
<VolumeX v-else class="w-4 h-4 text-gray-900" />
|
<VolumeX v-else class="w-4 h-4 text-ink-gray-9" />
|
||||||
</template>
|
</template>
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,50 +1,52 @@
|
|||||||
<template>
|
<template>
|
||||||
<div
|
<div
|
||||||
class="flex flex-col shadow hover:bg-gray-100 rounded-md p-4 h-full"
|
class="flex flex-col border-2 hover:bg-surface-gray-2 rounded-md p-4 h-full"
|
||||||
style="min-height: 150px"
|
style="min-height: 150px"
|
||||||
>
|
>
|
||||||
<div class="text-lg leading-5 font-semibold mb-2">
|
<div class="text-lg leading-5 font-semibold mb-2 text-ink-gray-9">
|
||||||
{{ batch.title }}
|
{{ batch.title }}
|
||||||
</div>
|
</div>
|
||||||
<Badge
|
<div
|
||||||
v-if="batch.seat_count && batch.seats_left > 0"
|
v-if="batch.seat_count && batch.seats_left > 0"
|
||||||
theme="green"
|
class="text-xs bg-green-100 text-green-700 self-start px-2 py-0.5 rounded-md"
|
||||||
class="self-start mb-2"
|
|
||||||
>
|
>
|
||||||
{{ batch.seats_left }}
|
{{ batch.seats_left }}
|
||||||
<span v-if="batch.seats_left > 1">{{ __('Seats Left') }}</span
|
<span v-if="batch.seats_left > 1">
|
||||||
><span v-else-if="batch.seats_left == 1">{{ __('Seat Left') }}</span>
|
{{ __('Seats Left') }}
|
||||||
</Badge>
|
</span>
|
||||||
<Badge
|
<span v-else-if="batch.seats_left == 1">
|
||||||
|
{{ __('Seat Left') }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
v-else-if="batch.seat_count && batch.seats_left <= 0"
|
v-else-if="batch.seat_count && batch.seats_left <= 0"
|
||||||
theme="red"
|
class="text-xs bg-red-100 text-red-700 self-start px-2 py-0.5 rounded-md"
|
||||||
class="self-start mb-2"
|
|
||||||
>
|
>
|
||||||
{{ __('Sold Out') }}
|
{{ __('Sold Out') }}
|
||||||
</Badge>
|
</div>
|
||||||
<div class="short-introduction text-sm text-gray-700">
|
<div class="short-introduction text-sm text-ink-gray-7">
|
||||||
{{ batch.description }}
|
{{ batch.description }}
|
||||||
</div>
|
</div>
|
||||||
<div v-if="batch.amount" class="font-semibold mb-4">
|
<div v-if="batch.amount" class="font-semibold text-ink-gray-9 mb-4">
|
||||||
{{ batch.price }}
|
{{ batch.price }}
|
||||||
</div>
|
</div>
|
||||||
<div class="flex flex-col space-y-2 mt-auto">
|
<div class="flex flex-col space-y-2 mt-auto">
|
||||||
<DateRange
|
<DateRange
|
||||||
:startDate="batch.start_date"
|
:startDate="batch.start_date"
|
||||||
:endDate="batch.end_date"
|
:endDate="batch.end_date"
|
||||||
class="text-sm text-gray-700"
|
class="text-sm text-ink-gray-7"
|
||||||
/>
|
/>
|
||||||
<div class="flex items-center text-sm text-gray-700">
|
<div class="flex items-center text-sm text-ink-gray-7">
|
||||||
<Clock class="h-4 w-4 stroke-1.5 mr-2 text-gray-700" />
|
<Clock class="h-4 w-4 stroke-1.5 mr-2 text-ink-gray-7" />
|
||||||
<span>
|
<span>
|
||||||
{{ formatTime(batch.start_time) }} - {{ formatTime(batch.end_time) }}
|
{{ formatTime(batch.start_time) }} - {{ formatTime(batch.end_time) }}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
v-if="batch.timezone"
|
v-if="batch.timezone"
|
||||||
class="flex items-center text-sm text-gray-700"
|
class="flex items-center text-sm text-ink-gray-7"
|
||||||
>
|
>
|
||||||
<Globe class="h-4 w-4 stroke-1.5 mr-2 text-gray-600" />
|
<Globe class="h-4 w-4 stroke-1.5 mr-2 text-ink-gray-5" />
|
||||||
<span>
|
<span>
|
||||||
{{ batch.timezone }}
|
{{ batch.timezone }}
|
||||||
</span>
|
</span>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<template>
|
<template>
|
||||||
<div>
|
<div>
|
||||||
<div class="flex items-center justify-between mb-4">
|
<div class="flex items-center justify-between mb-4">
|
||||||
<div class="text-lg font-semibold">
|
<div class="text-lg font-semibold text-ink-gray-9">
|
||||||
{{ __('Courses') }}
|
{{ __('Courses') }}
|
||||||
</div>
|
</div>
|
||||||
<Button v-if="canSeeAddButton()" @click="openCourseModal()">
|
<Button v-if="canSeeAddButton()" @click="openCourseModal()">
|
||||||
@@ -18,6 +18,7 @@
|
|||||||
row-key="batch_course"
|
row-key="batch_course"
|
||||||
:options="{
|
:options="{
|
||||||
showTooltip: false,
|
showTooltip: false,
|
||||||
|
selectable: user.data?.is_student ? false : true,
|
||||||
getRowRoute: (row) => ({
|
getRowRoute: (row) => ({
|
||||||
name: 'CourseDetail',
|
name: 'CourseDetail',
|
||||||
params: { courseName: row.name },
|
params: { courseName: row.name },
|
||||||
@@ -25,7 +26,7 @@
|
|||||||
}"
|
}"
|
||||||
>
|
>
|
||||||
<ListHeader
|
<ListHeader
|
||||||
class="mb-2 grid items-center space-x-4 rounded bg-gray-100 p-2"
|
class="mb-2 grid items-center space-x-4 rounded bg-surface-gray-2 p-2"
|
||||||
>
|
>
|
||||||
<ListHeaderItem :item="item" v-for="item in getCoursesColumns()">
|
<ListHeaderItem :item="item" v-for="item in getCoursesColumns()">
|
||||||
<template #prefix="{ item }">
|
<template #prefix="{ item }">
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
<div v-if="user.data?.is_student">
|
<div v-if="user.data?.is_student">
|
||||||
<div
|
<div
|
||||||
v-if="feedbackList.data?.length"
|
v-if="feedbackList.data?.length"
|
||||||
class="bg-blue-100 text-blue-700 p-2 rounded-md mb-5"
|
class="bg-surface-blue-2 text-blue-700 p-2 rounded-md mb-5"
|
||||||
>
|
>
|
||||||
{{ __('Thank you for providing your feedback!') }}
|
{{ __('Thank you for providing your feedback!') }}
|
||||||
</div>
|
</div>
|
||||||
@@ -61,13 +61,13 @@
|
|||||||
}"
|
}"
|
||||||
>
|
>
|
||||||
<ListHeader
|
<ListHeader
|
||||||
class="mb-2 grid items-center space-x-4 rounded bg-gray-100 p-2"
|
class="mb-2 grid items-center space-x-4 rounded bg-surface-gray-2 p-2"
|
||||||
></ListHeader>
|
></ListHeader>
|
||||||
<ListRows>
|
<ListRows>
|
||||||
<ListRow
|
<ListRow
|
||||||
:row="row"
|
:row="row"
|
||||||
v-for="row in feedbackList.data"
|
v-for="row in feedbackList.data"
|
||||||
class="group cursor-pointer"
|
class="group cursor-pointer feedback-list"
|
||||||
>
|
>
|
||||||
<template #default="{ column, item }">
|
<template #default="{ column, item }">
|
||||||
<ListRowItem
|
<ListRowItem
|
||||||
@@ -78,7 +78,7 @@
|
|||||||
<template #prefix>
|
<template #prefix>
|
||||||
<div v-if="column.key == 'member_name'">
|
<div v-if="column.key == 'member_name'">
|
||||||
<Avatar
|
<Avatar
|
||||||
class="flex items-center"
|
class="flex"
|
||||||
:image="row['member_image']"
|
:image="row['member_image']"
|
||||||
:label="item"
|
:label="item"
|
||||||
size="sm"
|
size="sm"
|
||||||
@@ -97,7 +97,7 @@
|
|||||||
</ListRows>
|
</ListRows>
|
||||||
</ListView>
|
</ListView>
|
||||||
</div>
|
</div>
|
||||||
<div v-else class="text-sm italic text-center text-gray-700 mt-5">
|
<div v-else class="text-sm italic text-center text-ink-gray-7 mt-5">
|
||||||
{{ __('No feedback received yet.') }}
|
{{ __('No feedback received yet.') }}
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
@@ -237,3 +237,9 @@ const feedbackColumns = computed(() => {
|
|||||||
]
|
]
|
||||||
})
|
})
|
||||||
</script>
|
</script>
|
||||||
|
<style>
|
||||||
|
.feedback-list > button > div {
|
||||||
|
align-items: start;
|
||||||
|
padding: 0.15rem 0;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|||||||
@@ -1,25 +1,31 @@
|
|||||||
<template>
|
<template>
|
||||||
<div v-if="batch.data" class="shadow rounded-md p-5 lg:w-72">
|
<div v-if="batch.data" class="border-2 rounded-md p-5 lg:w-72">
|
||||||
<Badge
|
<div
|
||||||
v-if="batch.data.seat_count && seats_left > 0"
|
v-if="batch.data.seat_count && seats_left > 0"
|
||||||
theme="green"
|
class="text-xs bg-green-100 text-green-700 float-right px-2 py-0.5 rounded-md"
|
||||||
class="self-start mb-2 float-right"
|
|
||||||
>
|
>
|
||||||
{{ seats_left }} <span v-if="seats_left > 1">{{ __('Seats Left') }}</span
|
{{ seats_left }}
|
||||||
><span v-else-if="seats_left == 1">{{ __('Seat Left') }}</span>
|
<span v-if="seats_left > 1">
|
||||||
</Badge>
|
{{ __('Seats Left') }}
|
||||||
<Badge
|
</span>
|
||||||
|
<span v-else-if="seats_left == 1">
|
||||||
|
{{ __('Seat Left') }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
v-else-if="batch.data.seat_count && seats_left <= 0"
|
v-else-if="batch.data.seat_count && seats_left <= 0"
|
||||||
theme="red"
|
class="text-xs bg-red-100 text-red-700 float-right px-2 py-0.5 rounded-md"
|
||||||
class="self-start mb-2 float-right"
|
|
||||||
>
|
>
|
||||||
{{ __('Sold Out') }}
|
{{ __('Sold Out') }}
|
||||||
</Badge>
|
</div>
|
||||||
<div v-if="batch.data.amount" class="text-lg font-semibold mb-3">
|
<div
|
||||||
|
v-if="batch.data.amount"
|
||||||
|
class="text-lg font-semibold mb-3 text-ink-gray-9"
|
||||||
|
>
|
||||||
{{ formatNumberIntoCurrency(batch.data.amount, batch.data.currency) }}
|
{{ formatNumberIntoCurrency(batch.data.amount, batch.data.currency) }}
|
||||||
</div>
|
</div>
|
||||||
<div class="flex items-center mb-3">
|
<div class="flex items-center mb-3 text-ink-gray-7">
|
||||||
<BookOpen class="h-4 w-4 stroke-1.5 mr-2 text-gray-700" />
|
<BookOpen class="h-4 w-4 stroke-1.5 mr-2" />
|
||||||
<span> {{ batch.data.courses.length }} {{ __('Courses') }} </span>
|
<span> {{ batch.data.courses.length }} {{ __('Courses') }} </span>
|
||||||
</div>
|
</div>
|
||||||
<DateRange
|
<DateRange
|
||||||
@@ -27,15 +33,15 @@
|
|||||||
:endDate="batch.data.end_date"
|
:endDate="batch.data.end_date"
|
||||||
class="mb-3"
|
class="mb-3"
|
||||||
/>
|
/>
|
||||||
<div class="flex items-center mb-3">
|
<div class="flex items-center mb-3 text-ink-gray-7">
|
||||||
<Clock class="h-4 w-4 stroke-1.5 mr-2 text-gray-700" />
|
<Clock class="h-4 w-4 stroke-1.5 mr-2" />
|
||||||
<span>
|
<span>
|
||||||
{{ formatTime(batch.data.start_time) }} -
|
{{ formatTime(batch.data.start_time) }} -
|
||||||
{{ formatTime(batch.data.end_time) }}
|
{{ formatTime(batch.data.end_time) }}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<div v-if="batch.data.timezone" class="flex items-center">
|
<div v-if="batch.data.timezone" class="flex items-center text-ink-gray-7">
|
||||||
<Globe class="h-4 w-4 stroke-1.5 mr-2 text-gray-700" />
|
<Globe class="h-4 w-4 stroke-1.5 mr-2" />
|
||||||
<span>
|
<span>
|
||||||
{{ batch.data.timezone }}
|
{{ batch.data.timezone }}
|
||||||
</span>
|
</span>
|
||||||
@@ -63,7 +69,11 @@
|
|||||||
name: batch.data.name,
|
name: batch.data.name,
|
||||||
},
|
},
|
||||||
}"
|
}"
|
||||||
v-else-if="batch.data.paid_batch && batch.data.seats_left"
|
v-else-if="
|
||||||
|
batch.data.paid_batch &&
|
||||||
|
batch.data.seats_left > 0 &&
|
||||||
|
batch.data.accept_enrollments
|
||||||
|
"
|
||||||
>
|
>
|
||||||
<Button v-if="!isStudent" class="w-full mt-4" variant="solid">
|
<Button v-if="!isStudent" class="w-full mt-4" variant="solid">
|
||||||
<span>
|
<span>
|
||||||
@@ -74,7 +84,11 @@
|
|||||||
<Button
|
<Button
|
||||||
variant="solid"
|
variant="solid"
|
||||||
class="w-full mt-2"
|
class="w-full mt-2"
|
||||||
v-else-if="batch.data.allow_self_enrollment && batch.data.seats_left"
|
v-else-if="
|
||||||
|
batch.data.allow_self_enrollment &&
|
||||||
|
batch.data.seats_left &&
|
||||||
|
batch.data.accept_enrollments
|
||||||
|
"
|
||||||
@click="enrollInBatch()"
|
@click="enrollInBatch()"
|
||||||
>
|
>
|
||||||
{{ __('Enroll Now') }}
|
{{ __('Enroll Now') }}
|
||||||
@@ -106,6 +120,7 @@ import { useRouter } from 'vue-router'
|
|||||||
|
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
|
const dayjs = inject('$dayjs')
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
batch: {
|
batch: {
|
||||||
|
|||||||
@@ -1,65 +1,87 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="">
|
<div class="">
|
||||||
<div class="w-full flex items-center justify-between pb-4">
|
<div class="w-full flex items-center justify-between pb-4">
|
||||||
<div class="font-medium text-gray-600">
|
<div class="font-medium text-ink-gray-7">
|
||||||
{{ __('Statistics') }}
|
{{ __('Statistics') }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="grid grid-cols-3 gap-5 mb-8">
|
<div class="grid grid-cols-4 gap-5 mb-8">
|
||||||
<div class="flex items-center shadow py-2 px-3 rounded-md">
|
<div
|
||||||
<div class="p-2 rounded-md bg-gray-100 mr-3">
|
class="flex items-center border py-2 px-3 rounded-md text-ink-gray-7"
|
||||||
<User class="w-5 h-5 stroke-1.5 text-gray-700" />
|
>
|
||||||
|
<div class="p-2 rounded-md bg-surface-gray-2 mr-3">
|
||||||
|
<User class="w-5 h-5 stroke-1.5" />
|
||||||
</div>
|
</div>
|
||||||
<div class="flex items-center space-x-2">
|
<div class="flex items-center space-x-2">
|
||||||
<span class="font-semibold">
|
<span class="font-semibold">
|
||||||
{{ students.data?.length }}
|
{{ students.data?.length }}
|
||||||
</span>
|
</span>
|
||||||
<span class="text-gray-700">
|
<span class="">
|
||||||
{{ __('Students') }}
|
{{ __('Students') }}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="flex items-center shadow py-2 px-3 rounded-md">
|
<div
|
||||||
<div class="p-2 rounded-md bg-gray-100 mr-3">
|
class="flex items-center border py-2 px-3 rounded-md text-ink-gray-7"
|
||||||
<BookOpen class="w-5 h-5 stroke-1.5 text-gray-700" />
|
>
|
||||||
|
<div class="p-2 rounded-md bg-surface-gray-2 mr-3">
|
||||||
|
<GraduationCap class="w-5 h-5 stroke-1.5" />
|
||||||
|
</div>
|
||||||
|
<div class="flex items-center space-x-2">
|
||||||
|
<span class="font-semibold">
|
||||||
|
{{ certificationCount.data }}
|
||||||
|
</span>
|
||||||
|
<span class="">
|
||||||
|
{{ __('Certified') }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div
|
||||||
|
class="flex items-center border py-2 px-3 rounded-md text-ink-gray-7"
|
||||||
|
>
|
||||||
|
<div class="p-2 rounded-md bg-surface-gray-2 mr-3">
|
||||||
|
<BookOpen class="w-5 h-5 stroke-1.5" />
|
||||||
</div>
|
</div>
|
||||||
<div class="flex items-center space-x-2">
|
<div class="flex items-center space-x-2">
|
||||||
<span class="font-semibold">
|
<span class="font-semibold">
|
||||||
{{ batch.courses?.length }}
|
{{ batch.courses?.length }}
|
||||||
</span>
|
</span>
|
||||||
<span class="text-gray-700">
|
<span>
|
||||||
{{ __('Courses') }}
|
{{ __('Courses') }}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="flex items-center shadow py-2 px-3 rounded-md">
|
<div
|
||||||
<div class="p-2 rounded-md bg-gray-100 mr-3">
|
class="flex items-center border py-2 px-3 rounded-md text-ink-gray-7"
|
||||||
<ShieldCheck class="w-5 h-5 stroke-1.5 text-gray-700" />
|
>
|
||||||
|
<div class="p-2 rounded-md bg-surface-gray-2 mr-3">
|
||||||
|
<ShieldCheck class="w-5 h-5 stroke-1.5" />
|
||||||
</div>
|
</div>
|
||||||
<div class="flex items-center space-x-2">
|
<div class="flex items-center space-x-2">
|
||||||
<span class="font-semibold">
|
<span class="font-semibold">
|
||||||
{{ assessmentCount }}
|
{{ assessmentCount }}
|
||||||
</span>
|
</span>
|
||||||
<span class="text-gray-700">
|
<span>
|
||||||
{{ __('Assessments') }}
|
{{ __('Assessments') }}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div v-if="showProgressChart" class="mb-8">
|
<div v-if="showProgressChart" class="mb-8">
|
||||||
<div class="text-gray-600 font-medium">
|
<div class="text-ink-gray-7 font-medium">
|
||||||
{{ __('Progress') }}
|
{{ __('Progress') }}
|
||||||
</div>
|
</div>
|
||||||
<ApexChart
|
<ApexChart
|
||||||
:options="chartOptions"
|
:options="chartOptions"
|
||||||
:series="chartData"
|
:series="chartData"
|
||||||
type="bar"
|
type="bar"
|
||||||
height="200"
|
:height="chartData[0].data.length * 30 + 100"
|
||||||
/>
|
/>
|
||||||
<div
|
<div
|
||||||
class="flex items-center justify-center text-sm text-gray-700 space-x-4"
|
class="flex items-center justify-center text-sm text-ink-gray-7 space-x-4"
|
||||||
>
|
>
|
||||||
<div class="flex items-center space-x-2">
|
<div class="flex items-center space-x-2">
|
||||||
<div
|
<div
|
||||||
@@ -85,7 +107,7 @@
|
|||||||
|
|
||||||
<div>
|
<div>
|
||||||
<div class="flex items-center justify-between mb-4">
|
<div class="flex items-center justify-between mb-4">
|
||||||
<div class="text-gray-600 font-medium">
|
<div class="text-ink-gray-7 font-medium">
|
||||||
{{ __('Students') }}
|
{{ __('Students') }}
|
||||||
</div>
|
</div>
|
||||||
<Button @click="openStudentModal()">
|
<Button @click="openStudentModal()">
|
||||||
@@ -106,7 +128,7 @@
|
|||||||
}"
|
}"
|
||||||
>
|
>
|
||||||
<ListHeader
|
<ListHeader
|
||||||
class="mb-2 grid items-center space-x-4 rounded bg-gray-100 p-2"
|
class="mb-2 grid items-center space-x-4 rounded bg-surface-gray-2 p-2"
|
||||||
>
|
>
|
||||||
<ListHeaderItem
|
<ListHeaderItem
|
||||||
:item="item"
|
:item="item"
|
||||||
@@ -173,7 +195,7 @@
|
|||||||
</ListSelectBanner>
|
</ListSelectBanner>
|
||||||
</ListView>
|
</ListView>
|
||||||
</div>
|
</div>
|
||||||
<div v-else class="text-sm italic text-gray-600">
|
<div v-else class="text-sm italic text-ink-gray-5">
|
||||||
{{ __('There are no students in this batch.') }}
|
{{ __('There are no students in this batch.') }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -204,7 +226,7 @@ import {
|
|||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import {
|
import {
|
||||||
BookOpen,
|
BookOpen,
|
||||||
Clipboard,
|
GraduationCap,
|
||||||
Plus,
|
Plus,
|
||||||
ShieldCheck,
|
ShieldCheck,
|
||||||
Trash2,
|
Trash2,
|
||||||
@@ -242,7 +264,7 @@ const students = createResource({
|
|||||||
auto: true,
|
auto: true,
|
||||||
onSuccess(data) {
|
onSuccess(data) {
|
||||||
chartData.value = getChartData()
|
chartData.value = getChartData()
|
||||||
showProgressChart.value = true
|
showProgressChart.value = data.length && true
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -285,7 +307,7 @@ const deleteStudents = createResource({
|
|||||||
url: 'lms.lms.api.delete_documents',
|
url: 'lms.lms.api.delete_documents',
|
||||||
makeParams(values) {
|
makeParams(values) {
|
||||||
return {
|
return {
|
||||||
doctype: 'Batch Student',
|
doctype: 'LMS Batch Enrollment',
|
||||||
documents: values.students,
|
documents: values.students,
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@@ -309,7 +331,9 @@ const removeStudents = (selections, unselectAll) => {
|
|||||||
const getChartData = () => {
|
const getChartData = () => {
|
||||||
let categories = {}
|
let categories = {}
|
||||||
|
|
||||||
Object.keys(students.data?.[0].courses).forEach((course) => {
|
if (!students.data?.length) return []
|
||||||
|
|
||||||
|
Object.keys(students.data[0].courses).forEach((course) => {
|
||||||
categories[course] = {
|
categories[course] = {
|
||||||
value: 0,
|
value: 0,
|
||||||
type: 'course',
|
type: 'course',
|
||||||
@@ -333,7 +357,7 @@ const getChartData = () => {
|
|||||||
})
|
})
|
||||||
|
|
||||||
Object.keys(student.assessments).forEach((assessment) => {
|
Object.keys(student.assessments).forEach((assessment) => {
|
||||||
if (student.assessments[assessment] === 100) {
|
if (student.assessments[assessment].result === 'Pass') {
|
||||||
categories[assessment].value += 1
|
categories[assessment].value += 1
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
@@ -402,6 +426,17 @@ watch(students, () => {
|
|||||||
assessmentCount.value = Object.keys(students.data?.[0].assessments).length
|
assessmentCount.value = Object.keys(students.data?.[0].assessments).length
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
const certificationCount = createResource({
|
||||||
|
url: 'frappe.client.get_count',
|
||||||
|
params: {
|
||||||
|
doctype: 'LMS Certificate',
|
||||||
|
filters: {
|
||||||
|
batch_name: props.batch.name,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
auto: true,
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
<style>
|
<style>
|
||||||
.apexcharts-legend {
|
.apexcharts-legend {
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
<div class="flex flex-col justify-between min-h-0">
|
<div class="flex flex-col justify-between min-h-0">
|
||||||
<div>
|
<div>
|
||||||
<div class="flex items-center justify-between">
|
<div class="flex items-center justify-between">
|
||||||
<div class="font-semibold mb-1">
|
<div class="font-semibold mb-1 text-ink-gray-9">
|
||||||
{{ __(label) }}
|
{{ __(label) }}
|
||||||
</div>
|
</div>
|
||||||
<Badge
|
<Badge
|
||||||
@@ -12,7 +12,7 @@
|
|||||||
theme="orange"
|
theme="orange"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div class="text-xs text-gray-600">
|
<div class="text-xs text-ink-gray-5">
|
||||||
{{ __(description) }}
|
{{ __(description) }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="flex flex-col min-h-0">
|
<div class="flex flex-col min-h-0">
|
||||||
<div class="flex items-center justify-between">
|
<div class="flex items-center justify-between">
|
||||||
<div class="text-xl font-semibold mb-1">
|
<div class="text-xl font-semibold mb-5 text-ink-gray-9">
|
||||||
{{ label }}
|
{{ label }}
|
||||||
</div>
|
</div>
|
||||||
<Button @click="() => showCategoryForm()">
|
<Button @click="() => showCategoryForm()">
|
||||||
@@ -28,12 +28,12 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="overflow-y-scroll">
|
<div class="overflow-y-scroll">
|
||||||
<div class="text-base divide-y">
|
<div class="text-base divide-y space-y-2">
|
||||||
<FormControl
|
<FormControl
|
||||||
:value="cat.category"
|
:value="cat.category"
|
||||||
type="text"
|
type="text"
|
||||||
v-for="cat in categories.data"
|
v-for="cat in categories.data"
|
||||||
class="form-control"
|
class=""
|
||||||
@change.stop="(e) => update(cat.name, e.target.value)"
|
@change.stop="(e) => update(cat.name, e.target.value)"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
@@ -128,24 +128,3 @@ const update = (name, value) => {
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
<style>
|
|
||||||
.form-control input {
|
|
||||||
padding: 1.25rem 0;
|
|
||||||
border-color: transparent;
|
|
||||||
background: white;
|
|
||||||
}
|
|
||||||
|
|
||||||
.form-control input:focus {
|
|
||||||
outline: transparent;
|
|
||||||
background: white;
|
|
||||||
box-shadow: none;
|
|
||||||
border-color: transparent;
|
|
||||||
}
|
|
||||||
|
|
||||||
.form-control input:hover {
|
|
||||||
outline: transparent;
|
|
||||||
background: white;
|
|
||||||
box-shadow: none;
|
|
||||||
border-color: transparent;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
|
|||||||
67
frontend/src/components/CertificationLinks.vue
Normal file
67
frontend/src/components/CertificationLinks.vue
Normal file
@@ -0,0 +1,67 @@
|
|||||||
|
<template>
|
||||||
|
<div
|
||||||
|
v-if="
|
||||||
|
certification.data &&
|
||||||
|
certification.data.membership &&
|
||||||
|
certification.data.paid_certificate &&
|
||||||
|
user.data?.is_student
|
||||||
|
"
|
||||||
|
>
|
||||||
|
<router-link
|
||||||
|
v-if="!certification.data.membership.purchased_certificate"
|
||||||
|
:to="{
|
||||||
|
name: 'Billing',
|
||||||
|
params: {
|
||||||
|
type: 'certificate',
|
||||||
|
name: courseName,
|
||||||
|
},
|
||||||
|
}"
|
||||||
|
>
|
||||||
|
<Button class="w-full">
|
||||||
|
<template #prefix>
|
||||||
|
<GraduationCap class="size-4 stroke-1.5" />
|
||||||
|
</template>
|
||||||
|
{{ __('Get Certified') }}
|
||||||
|
</Button>
|
||||||
|
</router-link>
|
||||||
|
<router-link
|
||||||
|
v-else-if="!certification.data.membership.certficate"
|
||||||
|
:to="{
|
||||||
|
name: 'CourseCertification',
|
||||||
|
params: {
|
||||||
|
courseName: courseName,
|
||||||
|
},
|
||||||
|
}"
|
||||||
|
>
|
||||||
|
<Button class="w-full">
|
||||||
|
<template #prefix>
|
||||||
|
<GraduationCap class="size-4 stroke-1.5" />
|
||||||
|
</template>
|
||||||
|
{{ __('Get Certified') }}
|
||||||
|
</Button>
|
||||||
|
</router-link>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
<script setup>
|
||||||
|
import { Button, createResource } from 'frappe-ui'
|
||||||
|
import { inject } from 'vue'
|
||||||
|
import { GraduationCap } from 'lucide-vue-next'
|
||||||
|
|
||||||
|
const user = inject('$user')
|
||||||
|
|
||||||
|
const props = defineProps({
|
||||||
|
courseName: {
|
||||||
|
type: String,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const certification = createResource({
|
||||||
|
url: 'lms.lms.api.get_certification_details',
|
||||||
|
params: {
|
||||||
|
course: props.courseName,
|
||||||
|
},
|
||||||
|
auto: true,
|
||||||
|
cache: ['certificationData', user.data?.name],
|
||||||
|
})
|
||||||
|
</script>
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="flex items-center">
|
<div class="flex items-center text-ink-gray-7">
|
||||||
<Calendar class="h-4 w-4 stroke-1.5 mr-2 text-gray-700" />
|
<Calendar class="h-4 w-4 stroke-1.5 mr-2" />
|
||||||
<span>
|
<span>
|
||||||
{{ getFormattedDateRange(props.startDate, props.endDate) }}
|
{{ getFormattedDateRange(props.startDate, props.endDate) }}
|
||||||
</span>
|
</span>
|
||||||
|
|||||||
@@ -17,7 +17,7 @@
|
|||||||
>
|
>
|
||||||
{{ displayValue(selectedValue) }}
|
{{ displayValue(selectedValue) }}
|
||||||
</span>
|
</span>
|
||||||
<span class="text-base leading-5 text-gray-500" v-else>
|
<span class="text-base leading-5 text-ink-gray-4" v-else>
|
||||||
{{ placeholder || '' }}
|
{{ placeholder || '' }}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
@@ -28,7 +28,9 @@
|
|||||||
</template>
|
</template>
|
||||||
<template #body="{ isOpen }">
|
<template #body="{ isOpen }">
|
||||||
<div v-show="isOpen">
|
<div v-show="isOpen">
|
||||||
<div class="mt-1 rounded-lg bg-white py-1 text-base shadow-2xl">
|
<div
|
||||||
|
class="mt-1 rounded-lg bg-surface-white py-1 text-base shadow-2xl"
|
||||||
|
>
|
||||||
<div class="relative px-1.5 pt-0.5">
|
<div class="relative px-1.5 pt-0.5">
|
||||||
<ComboboxInput
|
<ComboboxInput
|
||||||
ref="search"
|
ref="search"
|
||||||
@@ -62,7 +64,7 @@
|
|||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
v-if="group.group && !group.hideLabel"
|
v-if="group.group && !group.hideLabel"
|
||||||
class="px-2.5 py-1.5 text-sm font-medium text-gray-500"
|
class="px-2.5 py-1.5 text-sm font-medium text-ink-gray-4"
|
||||||
>
|
>
|
||||||
{{ group.group }}
|
{{ group.group }}
|
||||||
</div>
|
</div>
|
||||||
@@ -76,7 +78,7 @@
|
|||||||
<li
|
<li
|
||||||
:class="[
|
:class="[
|
||||||
'flex items-center rounded px-2.5 py-2 text-base',
|
'flex items-center rounded px-2.5 py-2 text-base',
|
||||||
{ 'bg-gray-100': active },
|
{ 'bg-surface-gray-2': active },
|
||||||
]"
|
]"
|
||||||
>
|
>
|
||||||
<slot
|
<slot
|
||||||
@@ -93,7 +95,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
v-if="option.description"
|
v-if="option.description"
|
||||||
class="text-xs text-gray-700"
|
class="text-xs text-ink-gray-7"
|
||||||
v-html="option.description"
|
v-html="option.description"
|
||||||
></div>
|
></div>
|
||||||
</div>
|
</div>
|
||||||
@@ -103,7 +105,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<li
|
<li
|
||||||
v-if="groups.length == 0"
|
v-if="groups.length == 0"
|
||||||
class="mt-1.5 rounded-md px-2.5 py-1.5 text-base text-gray-600"
|
class="mt-1.5 rounded-md px-2.5 py-1.5 text-base text-ink-gray-5"
|
||||||
>
|
>
|
||||||
No results found
|
No results found
|
||||||
</li>
|
</li>
|
||||||
@@ -128,7 +130,7 @@ import {
|
|||||||
ComboboxOptions,
|
ComboboxOptions,
|
||||||
ComboboxOption,
|
ComboboxOption,
|
||||||
} from '@headlessui/vue'
|
} from '@headlessui/vue'
|
||||||
import { Popover, Button } from 'frappe-ui'
|
import { Popover } from 'frappe-ui'
|
||||||
import { ChevronDown, X } from 'lucide-vue-next'
|
import { ChevronDown, X } from 'lucide-vue-next'
|
||||||
import { ref, computed, useAttrs, useSlots, watch, nextTick } from 'vue'
|
import { ref, computed, useAttrs, useSlots, watch, nextTick } from 'vue'
|
||||||
|
|
||||||
@@ -243,7 +245,7 @@ watch(showOptions, (val) => {
|
|||||||
})
|
})
|
||||||
|
|
||||||
const textColor = computed(() => {
|
const textColor = computed(() => {
|
||||||
return props.disabled ? 'text-gray-600' : 'text-gray-800'
|
return props.disabled ? 'text-ink-gray-5' : 'text-ink-gray-8'
|
||||||
})
|
})
|
||||||
|
|
||||||
const inputClasses = computed(() => {
|
const inputClasses = computed(() => {
|
||||||
@@ -264,12 +266,14 @@ const inputClasses = computed(() => {
|
|||||||
let variant = props.disabled ? 'disabled' : props.variant
|
let variant = props.disabled ? 'disabled' : props.variant
|
||||||
let variantClasses = {
|
let variantClasses = {
|
||||||
subtle:
|
subtle:
|
||||||
'border border-gray-100 bg-gray-100 placeholder-gray-500 hover:border-gray-200 hover:bg-gray-200 focus:bg-white focus:border-gray-500 focus:shadow-sm focus:ring-0 focus-visible:ring-2 focus-visible:ring-gray-400',
|
'border border-gray-100 bg-surface-gray-2 placeholder-ink-gray-4 hover:border-outline-gray-modals hover:bg-surface-gray-3 focus:bg-surface-white focus:border-outline-gray-4 focus:shadow-sm focus:ring-0 focus-visible:ring-2 focus-visible:ring-outline-gray-3',
|
||||||
outline:
|
outline:
|
||||||
'border border-gray-300 bg-white placeholder-gray-500 hover:border-gray-400 hover:shadow-sm focus:bg-white focus:border-gray-500 focus:shadow-sm focus:ring-0 focus-visible:ring-2 focus-visible:ring-gray-400',
|
'border border-outline-gray-2 bg-surface-white placeholder-ink-gray-4 hover:border-outline-gray-3 hover:shadow-sm focus:bg-surface-white focus:border-outline-gray-4 focus:shadow-sm focus:ring-0 focus-visible:ring-2 focus-visible:ring-outline-gray-3',
|
||||||
disabled: [
|
disabled: [
|
||||||
'border bg-gray-50 placeholder-gray-400',
|
'border bg-surface-menu-bar placeholder-ink-gray-3',
|
||||||
props.variant === 'outline' ? 'border-gray-300' : 'border-transparent',
|
props.variant === 'outline'
|
||||||
|
? 'border-outline-gray-2'
|
||||||
|
: 'border-transparent',
|
||||||
],
|
],
|
||||||
}[variant]
|
}[variant]
|
||||||
|
|
||||||
|
|||||||
@@ -5,7 +5,7 @@
|
|||||||
height: height,
|
height: height,
|
||||||
}"
|
}"
|
||||||
>
|
>
|
||||||
<span class="text-xs" v-if="label">
|
<span class="text-xs text-ink-gray-7" v-if="label">
|
||||||
{{ label }}
|
{{ label }}
|
||||||
</span>
|
</span>
|
||||||
<div
|
<div
|
||||||
@@ -13,7 +13,7 @@
|
|||||||
class="h-auto flex-1 overflow-hidden overscroll-none !rounded border border-outline-gray-2 bg-surface-gray-2 dark:bg-gray-900"
|
class="h-auto flex-1 overflow-hidden overscroll-none !rounded border border-outline-gray-2 bg-surface-gray-2 dark:bg-gray-900"
|
||||||
/>
|
/>
|
||||||
<span
|
<span
|
||||||
class="mt-1 text-xs text-gray-600"
|
class="mt-1 text-xs text-ink-gray-5"
|
||||||
v-show="description"
|
v-show="description"
|
||||||
v-html="description"
|
v-html="description"
|
||||||
></span>
|
></span>
|
||||||
@@ -27,7 +27,6 @@
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { useDark } from '@vueuse/core'
|
|
||||||
import ace from 'ace-builds'
|
import ace from 'ace-builds'
|
||||||
import 'ace-builds/src-min-noconflict/ext-searchbox'
|
import 'ace-builds/src-min-noconflict/ext-searchbox'
|
||||||
import 'ace-builds/src-min-noconflict/theme-chrome'
|
import 'ace-builds/src-min-noconflict/theme-chrome'
|
||||||
@@ -35,9 +34,7 @@ import 'ace-builds/src-min-noconflict/theme-twilight'
|
|||||||
import { PropType, onMounted, ref, watch } from 'vue'
|
import { PropType, onMounted, ref, watch } from 'vue'
|
||||||
import { Button } from 'frappe-ui'
|
import { Button } from 'frappe-ui'
|
||||||
|
|
||||||
const isDark = useDark({
|
const isDark = ref(false)
|
||||||
attribute: 'data-theme',
|
|
||||||
})
|
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
modelValue: {
|
modelValue: {
|
||||||
@@ -82,6 +79,7 @@ const editor = ref<HTMLElement | null>(null)
|
|||||||
let aceEditor = null as ace.Ace.Editor | null
|
let aceEditor = null as ace.Ace.Editor | null
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
|
isDark.value = localStorage.getItem('theme') === 'dark'
|
||||||
setupEditor()
|
setupEditor()
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -148,6 +146,7 @@ function resetEditor(value: string, resetHistory = false) {
|
|||||||
value = getModelValue()
|
value = getModelValue()
|
||||||
aceEditor?.setValue(value)
|
aceEditor?.setValue(value)
|
||||||
aceEditor?.clearSelection()
|
aceEditor?.clearSelection()
|
||||||
|
console.log(isDark.value)
|
||||||
aceEditor?.setTheme(isDark.value ? 'ace/theme/twilight' : 'ace/theme/chrome')
|
aceEditor?.setTheme(isDark.value ? 'ace/theme/twilight' : 'ace/theme/chrome')
|
||||||
props.autofocus && aceEditor?.focus()
|
props.autofocus && aceEditor?.focus()
|
||||||
if (resetHistory) {
|
if (resetHistory) {
|
||||||
@@ -156,6 +155,7 @@ function resetEditor(value: string, resetHistory = false) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
watch(isDark, () => {
|
watch(isDark, () => {
|
||||||
|
console.log(isDark.value)
|
||||||
aceEditor?.setTheme(isDark.value ? 'ace/theme/twilight' : 'ace/theme/chrome')
|
aceEditor?.setTheme(isDark.value ? 'ace/theme/twilight' : 'ace/theme/chrome')
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -175,30 +175,3 @@ watch(
|
|||||||
|
|
||||||
defineExpose({ resetEditor })
|
defineExpose({ resetEditor })
|
||||||
</script>
|
</script>
|
||||||
<style scoped>
|
|
||||||
.editor .ace_editor {
|
|
||||||
height: 100%;
|
|
||||||
width: 100%;
|
|
||||||
border-radius: 5px;
|
|
||||||
overscroll-behavior: none;
|
|
||||||
}
|
|
||||||
.editor :deep(.ace_scrollbar-h) {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
.editor :deep(.ace_search) {
|
|
||||||
@apply dark:bg-gray-800 dark:text-gray-200;
|
|
||||||
@apply dark:border-gray-800;
|
|
||||||
}
|
|
||||||
.editor :deep(.ace_searchbtn) {
|
|
||||||
@apply dark:bg-gray-800 dark:text-gray-200;
|
|
||||||
@apply dark:border-gray-800;
|
|
||||||
}
|
|
||||||
.editor :deep(.ace_button) {
|
|
||||||
@apply dark:bg-gray-800 dark:text-gray-200;
|
|
||||||
}
|
|
||||||
|
|
||||||
.editor :deep(.ace_search_field) {
|
|
||||||
@apply dark:bg-gray-900 dark:text-gray-200;
|
|
||||||
@apply dark:border-gray-800;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="space-y-1.5">
|
<div class="space-y-1.5">
|
||||||
<label class="block text-xs text-gray-600">
|
<label class="block text-xs text-ink-gray-5">
|
||||||
{{ label }}
|
{{ label }}
|
||||||
</label>
|
</label>
|
||||||
<div class="w-full">
|
<div class="w-full">
|
||||||
@@ -8,22 +8,22 @@
|
|||||||
<template #target="{ togglePopover }">
|
<template #target="{ togglePopover }">
|
||||||
<button
|
<button
|
||||||
@click="openPopover(togglePopover)"
|
@click="openPopover(togglePopover)"
|
||||||
class="flex w-full items-center space-x-2 focus:outline-none bg-gray-100 rounded h-7 py-1.5 px-2 hover:bg-gray-200 focus:bg-white border border-gray-100 hover:border-gray-200 focus:border-gray-500"
|
class="flex w-full items-center space-x-2 focus:outline-none bg-surface-gray-2 rounded h-7 py-1.5 px-2 hover:bg-surface-gray-3 focus:bg-surface-white border border-gray-100 hover:border-outline-gray-modals focus:border-outline-gray-4"
|
||||||
>
|
>
|
||||||
<component
|
<component
|
||||||
v-if="selectedIcon"
|
v-if="selectedIcon"
|
||||||
class="w-4 h-4 text-gray-700 stroke-1.5"
|
class="w-4 h-4 text-ink-gray-7 stroke-1.5"
|
||||||
:is="icons[selectedIcon]"
|
:is="icons[selectedIcon]"
|
||||||
/>
|
/>
|
||||||
<component
|
<component
|
||||||
v-else
|
v-else
|
||||||
class="w-4 h-4 text-gray-700 stroke-1.5"
|
class="w-4 h-4 text-ink-gray-7 stroke-1.5"
|
||||||
:is="icons.Folder"
|
:is="icons.Folder"
|
||||||
/>
|
/>
|
||||||
<span v-if="selectedIcon">
|
<span v-if="selectedIcon">
|
||||||
{{ selectedIcon }}
|
{{ selectedIcon }}
|
||||||
</span>
|
</span>
|
||||||
<span v-else class="text-gray-600">
|
<span v-else class="text-ink-gray-5">
|
||||||
{{ __('Choose an icon') }}
|
{{ __('Choose an icon') }}
|
||||||
</span>
|
</span>
|
||||||
</button>
|
</button>
|
||||||
@@ -40,7 +40,7 @@
|
|||||||
<div v-for="(iconComponent, iconName) in filteredIcons">
|
<div v-for="(iconComponent, iconName) in filteredIcons">
|
||||||
<component
|
<component
|
||||||
:is="iconComponent"
|
:is="iconComponent"
|
||||||
class="h-4 w-4 stroke-1.5 text-gray-700 cursor-pointer"
|
class="h-4 w-4 stroke-1.5 text-ink-gray-7 cursor-pointer"
|
||||||
@click="setIcon(iconName, close)"
|
@click="setIcon(iconName, close)"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
<div class="space-y-1.5">
|
<div class="space-y-1.5">
|
||||||
<label class="block" :class="labelClasses" v-if="attrs.label">
|
<label class="block" :class="labelClasses" v-if="attrs.label">
|
||||||
{{ attrs.label }}
|
{{ attrs.label }}
|
||||||
<span class="text-red-500" v-if="attrs.required">*</span>
|
<span class="text-ink-red-3" v-if="attrs.required">*</span>
|
||||||
</label>
|
</label>
|
||||||
<Autocomplete
|
<Autocomplete
|
||||||
ref="autocomplete"
|
ref="autocomplete"
|
||||||
@@ -56,7 +56,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
</Autocomplete>
|
</Autocomplete>
|
||||||
<p v-if="description" class="text-sm text-gray-600">{{ description }}</p>
|
<p v-if="description" class="text-sm text-ink-gray-5">{{ description }}</p>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -163,7 +163,7 @@ const labelClasses = computed(() => {
|
|||||||
sm: 'text-xs',
|
sm: 'text-xs',
|
||||||
md: 'text-base',
|
md: 'text-base',
|
||||||
}[attrs.size || 'sm'],
|
}[attrs.size || 'sm'],
|
||||||
'text-gray-600',
|
'text-ink-gray-5',
|
||||||
]
|
]
|
||||||
})
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
<div>
|
<div>
|
||||||
<label class="block mb-1" :class="labelClasses" v-if="label">
|
<label class="block mb-1" :class="labelClasses" v-if="label">
|
||||||
{{ label }}
|
{{ label }}
|
||||||
<span class="text-red-500" v-if="required">*</span>
|
<span class="text-ink-red-3" v-if="required">*</span>
|
||||||
</label>
|
</label>
|
||||||
<div class="grid grid-cols-3 gap-1">
|
<div class="grid grid-cols-3 gap-1">
|
||||||
<Button
|
<Button
|
||||||
@@ -41,7 +41,9 @@
|
|||||||
</template>
|
</template>
|
||||||
<template #body="{ isOpen }">
|
<template #body="{ isOpen }">
|
||||||
<div v-show="isOpen">
|
<div v-show="isOpen">
|
||||||
<div class="mt-1 rounded-lg bg-white py-1 text-base shadow-2xl">
|
<div
|
||||||
|
class="mt-1 rounded-lg bg-surface-white py-1 text-base shadow-2xl"
|
||||||
|
>
|
||||||
<ComboboxOptions
|
<ComboboxOptions
|
||||||
class="my-1 max-h-[12rem] overflow-y-auto px-1.5"
|
class="my-1 max-h-[12rem] overflow-y-auto px-1.5"
|
||||||
static
|
static
|
||||||
@@ -55,14 +57,14 @@
|
|||||||
<li
|
<li
|
||||||
:class="[
|
:class="[
|
||||||
'flex cursor-pointer items-center rounded px-2 py-1 text-base',
|
'flex cursor-pointer items-center rounded px-2 py-1 text-base',
|
||||||
{ 'bg-gray-100': active },
|
{ 'bg-surface-gray-2': active },
|
||||||
]"
|
]"
|
||||||
>
|
>
|
||||||
<div class="flex flex-col gap-1 p-1">
|
<div class="flex flex-col gap-1 p-1">
|
||||||
<div class="text-base font-medium">
|
<div class="text-base font-medium">
|
||||||
{{ option.description }}
|
{{ option.description }}
|
||||||
</div>
|
</div>
|
||||||
<div class="text-sm text-gray-600">
|
<div class="text-sm text-ink-gray-5">
|
||||||
{{ option.value }}
|
{{ option.value }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -240,7 +242,7 @@ const labelClasses = computed(() => {
|
|||||||
sm: 'text-xs',
|
sm: 'text-xs',
|
||||||
md: 'text-base',
|
md: 'text-base',
|
||||||
}[props.size || 'sm'],
|
}[props.size || 'sm'],
|
||||||
'text-gray-600',
|
'text-ink-gray-5',
|
||||||
]
|
]
|
||||||
})
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="space-y-1">
|
<div class="space-y-1">
|
||||||
<label class="block text-xs text-gray-600" v-if="props.label">
|
<label class="block text-xs text-ink-gray-5" v-if="props.label">
|
||||||
{{ props.label }}
|
{{ props.label }}
|
||||||
</label>
|
</label>
|
||||||
<div class="flex text-center">
|
<div class="flex text-center">
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<template>
|
<template>
|
||||||
<div
|
<div
|
||||||
v-if="course.title"
|
v-if="course.title"
|
||||||
class="flex flex-col h-full rounded-md shadow-md text-base overflow-auto"
|
class="flex flex-col h-full rounded-md border-2 overflow-auto"
|
||||||
style="min-height: 350px"
|
style="min-height: 350px"
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
@@ -15,14 +15,13 @@
|
|||||||
<Badge v-if="course.featured" variant="subtle" theme="green" size="md">
|
<Badge v-if="course.featured" variant="subtle" theme="green" size="md">
|
||||||
{{ __('Featured') }}
|
{{ __('Featured') }}
|
||||||
</Badge>
|
</Badge>
|
||||||
<Badge
|
<div
|
||||||
variant="subtle"
|
v-if="course.tags"
|
||||||
theme="gray"
|
v-for="tag in course.tags?.split(', ')"
|
||||||
size="md"
|
class="text-xs bg-white text-gray-800 px-2 py-0.5 rounded-md"
|
||||||
v-for="tag in course.tags"
|
|
||||||
>
|
>
|
||||||
{{ tag }}
|
{{ tag }}
|
||||||
</Badge>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div v-if="!course.image" class="image-placeholder">
|
<div v-if="!course.image" class="image-placeholder">
|
||||||
{{ course.title[0] }}
|
{{ course.title[0] }}
|
||||||
@@ -32,8 +31,8 @@
|
|||||||
<div class="flex items-center justify-between mb-2">
|
<div class="flex items-center justify-between mb-2">
|
||||||
<div v-if="course.lessons">
|
<div v-if="course.lessons">
|
||||||
<Tooltip :text="__('Lessons')">
|
<Tooltip :text="__('Lessons')">
|
||||||
<span class="flex items-center">
|
<span class="flex items-center text-ink-gray-7">
|
||||||
<BookOpen class="h-4 w-4 stroke-1.5 text-gray-700 mr-1" />
|
<BookOpen class="h-4 w-4 stroke-1.5 mr-1" />
|
||||||
{{ course.lessons }}
|
{{ course.lessons }}
|
||||||
</span>
|
</span>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
@@ -41,8 +40,8 @@
|
|||||||
|
|
||||||
<div v-if="course.enrollments">
|
<div v-if="course.enrollments">
|
||||||
<Tooltip :text="__('Enrolled Students')">
|
<Tooltip :text="__('Enrolled Students')">
|
||||||
<span class="flex items-center">
|
<span class="flex items-center text-ink-gray-7">
|
||||||
<Users class="h-4 w-4 stroke-1.5 text-gray-700 mr-1" />
|
<Users class="h-4 w-4 stroke-1. mr-1" />
|
||||||
{{ course.enrollments }}
|
{{ course.enrollments }}
|
||||||
</span>
|
</span>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
@@ -50,8 +49,8 @@
|
|||||||
|
|
||||||
<div v-if="course.rating">
|
<div v-if="course.rating">
|
||||||
<Tooltip :text="__('Average Rating')">
|
<Tooltip :text="__('Average Rating')">
|
||||||
<span class="flex items-center">
|
<span class="flex items-center text-ink-gray-7">
|
||||||
<Star class="h-4 w-4 stroke-1.5 text-gray-700 mr-1" />
|
<Star class="h-4 w-4 stroke-1.5 mr-1" />
|
||||||
{{ course.rating }}
|
{{ course.rating }}
|
||||||
</span>
|
</span>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
@@ -68,11 +67,11 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="text-xl font-semibold leading-6">
|
<div class="text-xl font-semibold leading-6 text-ink-gray-9">
|
||||||
{{ course.title }}
|
{{ course.title }}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="short-introduction text-gray-700 text-sm">
|
<div class="short-introduction text-ink-gray-7 text-sm">
|
||||||
{{ course.short_introduction }}
|
{{ course.short_introduction }}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -81,7 +80,10 @@
|
|||||||
:progress="course.membership.progress"
|
:progress="course.membership.progress"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<div v-if="user && course.membership" class="text-sm mb-4">
|
<div
|
||||||
|
v-if="user && course.membership"
|
||||||
|
class="text-sm text-ink-gray-7 mt-2 mb-4"
|
||||||
|
>
|
||||||
{{ Math.ceil(course.membership.progress) }}% completed
|
{{ Math.ceil(course.membership.progress) }}% completed
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -99,9 +101,15 @@
|
|||||||
<CourseInstructors :instructors="course.instructors" />
|
<CourseInstructors :instructors="course.instructors" />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="font-semibold">
|
<div v-if="course.paid_course" class="font-semibold">
|
||||||
{{ course.price }}
|
{{ course.price }}
|
||||||
</div>
|
</div>
|
||||||
|
<div
|
||||||
|
v-if="course.paid_certificate || course.enable_certification"
|
||||||
|
class="text-xs text-ink-blue-3 bg-surface-blue-1 py-0.5 px-1 rounded-md"
|
||||||
|
>
|
||||||
|
{{ __('Certification') }}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,35 +1,37 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="shadow rounded-md min-w-80">
|
<div class="border-2 rounded-md min-w-80">
|
||||||
<iframe
|
<iframe
|
||||||
v-if="course.data.video_link"
|
v-if="course.data.video_link"
|
||||||
:src="video_link"
|
:src="video_link"
|
||||||
class="rounded-t-md min-h-56 w-full"
|
class="rounded-t-md min-h-56 w-full"
|
||||||
/>
|
/>
|
||||||
<div class="p-5">
|
<div class="p-5">
|
||||||
<div v-if="course.data.price" class="text-2xl font-semibold mb-3">
|
<div v-if="course.data.paid_course" class="text-2xl font-semibold mb-3">
|
||||||
{{ course.data.price }}
|
{{ course.data.price }}
|
||||||
</div>
|
</div>
|
||||||
<router-link
|
<div v-if="course.data.membership" class="space-y-2">
|
||||||
v-if="course.data.membership"
|
<router-link
|
||||||
:to="{
|
:to="{
|
||||||
name: 'Lesson',
|
name: 'Lesson',
|
||||||
params: {
|
params: {
|
||||||
courseName: course.name,
|
courseName: course.name,
|
||||||
chapterNumber: course.data.current_lesson
|
chapterNumber: course.data.current_lesson
|
||||||
? course.data.current_lesson.split('-')[0]
|
? course.data.current_lesson.split('-')[0]
|
||||||
: 1,
|
: 1,
|
||||||
lessonNumber: course.data.current_lesson
|
lessonNumber: course.data.current_lesson
|
||||||
? course.data.current_lesson.split('-')[1]
|
? course.data.current_lesson.split('-')[1]
|
||||||
: 1,
|
: 1,
|
||||||
},
|
},
|
||||||
}"
|
}"
|
||||||
>
|
>
|
||||||
<Button variant="solid" size="md" class="w-full">
|
<Button variant="solid" size="md" class="w-full">
|
||||||
<span>
|
<span>
|
||||||
{{ __('Continue Learning') }}
|
{{ __('Continue Learning') }}
|
||||||
</span>
|
</span>
|
||||||
</Button>
|
</Button>
|
||||||
</router-link>
|
</router-link>
|
||||||
|
<CertificationLinks :courseName="course.data.name" />
|
||||||
|
</div>
|
||||||
<router-link
|
<router-link
|
||||||
v-else-if="course.data.paid_course"
|
v-else-if="course.data.paid_course"
|
||||||
:to="{
|
:to="{
|
||||||
@@ -48,7 +50,7 @@
|
|||||||
</router-link>
|
</router-link>
|
||||||
<div
|
<div
|
||||||
v-else-if="course.data.disable_self_learning"
|
v-else-if="course.data.disable_self_learning"
|
||||||
class="bg-blue-100 text-blue-900 text-sm rounded-md py-1 px-3"
|
class="bg-surface-blue-2 text-blue-900 text-sm rounded-md py-1 px-3"
|
||||||
>
|
>
|
||||||
{{ __('Contact the Administrator to enroll for this course.') }}
|
{{ __('Contact the Administrator to enroll for this course.') }}
|
||||||
</div>
|
</div>
|
||||||
@@ -88,39 +90,61 @@
|
|||||||
</Button>
|
</Button>
|
||||||
</router-link>
|
</router-link>
|
||||||
<div class="space-y-4">
|
<div class="space-y-4">
|
||||||
<div class="mt-8 font-medium">
|
<div class="mt-8 font-medium text-ink-gray-9">
|
||||||
{{ __('This course has:') }}
|
{{ __('This course has:') }}
|
||||||
</div>
|
</div>
|
||||||
<div class="flex items-center">
|
<div class="flex items-center text-ink-gray-9">
|
||||||
<BookOpen class="h-4 w-4 stroke-1.5 text-gray-600" />
|
<BookOpen class="h-4 w-4 stroke-1.5" />
|
||||||
<span class="ml-2">
|
<span class="ml-2">
|
||||||
{{ course.data.lessons }} {{ __('Lessons') }}
|
{{ course.data.lessons }} {{ __('Lessons') }}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<div class="flex items-center">
|
<div class="flex items-center text-ink-gray-9">
|
||||||
<Users class="h-4 w-4 stroke-1.5 text-gray-600" />
|
<Users class="h-4 w-4 stroke-1.5" />
|
||||||
<span class="ml-2">
|
<span class="ml-2">
|
||||||
{{ formatAmount(course.data.enrollments) }}
|
{{ formatAmount(course.data.enrollments) }}
|
||||||
{{ __('Enrolled Students') }}
|
{{ __('Enrolled Students') }}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<div v-if="parseInt(course.data.rating) > 0" class="flex items-center">
|
<div
|
||||||
|
v-if="parseInt(course.data.rating) > 0"
|
||||||
|
class="flex items-center text-ink-gray-9"
|
||||||
|
>
|
||||||
<Star class="h-4 w-4 stroke-1.5 fill-orange-500 text-gray-50" />
|
<Star class="h-4 w-4 stroke-1.5 fill-orange-500 text-gray-50" />
|
||||||
<span class="ml-2">
|
<span class="ml-2">
|
||||||
{{ course.data.rating }} {{ __('Rating') }}
|
{{ course.data.rating }} {{ __('Rating') }}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
|
<div
|
||||||
|
v-if="course.data.enable_certification"
|
||||||
|
class="flex items-center font-semibold text-ink-gray-9"
|
||||||
|
>
|
||||||
|
<GraduationCap class="h-4 w-4 stroke-2" />
|
||||||
|
<span class="ml-2">
|
||||||
|
{{ __('Certificate of Completion') }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
v-if="course.data.paid_certificate"
|
||||||
|
class="flex items-center font-semibold text-ink-gray-9"
|
||||||
|
>
|
||||||
|
<GraduationCap class="h-4 w-4 stroke-2" />
|
||||||
|
<span class="ml-2">
|
||||||
|
{{ __('Paid Certificate after Evaluation') }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { BookOpen, Users, Star } from 'lucide-vue-next'
|
import { BookOpen, Users, Star, GraduationCap } from 'lucide-vue-next'
|
||||||
import { computed, inject } from 'vue'
|
import { computed, inject } from 'vue'
|
||||||
import { Button, createResource } from 'frappe-ui'
|
import { Button, createResource, Tooltip } from 'frappe-ui'
|
||||||
import { showToast, formatAmount } from '@/utils/'
|
import { showToast, formatAmount } from '@/utils/'
|
||||||
import { capture } from '@/telemetry'
|
import { capture } from '@/telemetry'
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
|
import CertificationLinks from '@/components/CertificationLinks.vue'
|
||||||
|
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
|
|||||||
@@ -1,44 +1,46 @@
|
|||||||
<template>
|
<template>
|
||||||
<span v-if="instructors?.length == 1">
|
<div class="text-ink-gray-7">
|
||||||
<router-link
|
<span v-if="instructors?.length == 1">
|
||||||
:to="{
|
<router-link
|
||||||
name: 'Profile',
|
:to="{
|
||||||
params: { username: instructors[0].username },
|
name: 'Profile',
|
||||||
}"
|
params: { username: instructors[0].username },
|
||||||
>
|
}"
|
||||||
{{ instructors[0].full_name }}
|
>
|
||||||
</router-link>
|
{{ instructors[0].full_name }}
|
||||||
</span>
|
</router-link>
|
||||||
<span v-if="instructors?.length == 2">
|
</span>
|
||||||
<router-link
|
<span v-if="instructors?.length == 2">
|
||||||
:to="{
|
<router-link
|
||||||
name: 'Profile',
|
:to="{
|
||||||
params: { username: instructors[0].username },
|
name: 'Profile',
|
||||||
}"
|
params: { username: instructors[0].username },
|
||||||
>
|
}"
|
||||||
{{ instructors[0].first_name }}
|
>
|
||||||
</router-link>
|
{{ instructors[0].first_name }}
|
||||||
and
|
</router-link>
|
||||||
<router-link
|
and
|
||||||
:to="{
|
<router-link
|
||||||
name: 'Profile',
|
:to="{
|
||||||
params: { username: instructors[1].username },
|
name: 'Profile',
|
||||||
}"
|
params: { username: instructors[1].username },
|
||||||
>
|
}"
|
||||||
{{ instructors[1].first_name }}
|
>
|
||||||
</router-link>
|
{{ instructors[1].first_name }}
|
||||||
</span>
|
</router-link>
|
||||||
<span v-if="instructors?.length > 2">
|
</span>
|
||||||
<router-link
|
<span v-if="instructors?.length > 2">
|
||||||
:to="{
|
<router-link
|
||||||
name: 'Profile',
|
:to="{
|
||||||
params: { username: instructors[0].username },
|
name: 'Profile',
|
||||||
}"
|
params: { username: instructors[0].username },
|
||||||
>
|
}"
|
||||||
{{ instructors[0].first_name }}
|
>
|
||||||
</router-link>
|
{{ instructors[0].first_name }}
|
||||||
and {{ instructors?.length - 1 }} others
|
</router-link>
|
||||||
</span>
|
and {{ instructors?.length - 1 }} others
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
|
|||||||
@@ -4,7 +4,7 @@
|
|||||||
v-if="title && (outline.data?.length || allowEdit)"
|
v-if="title && (outline.data?.length || allowEdit)"
|
||||||
class="grid grid-cols-[70%,30%] mb-4 px-2"
|
class="grid grid-cols-[70%,30%] mb-4 px-2"
|
||||||
>
|
>
|
||||||
<div class="font-semibold text-lg leading-5">
|
<div class="font-semibold text-lg leading-5 text-ink-gray-9">
|
||||||
{{ __(title) }}
|
{{ __(title) }}
|
||||||
</div>
|
</div>
|
||||||
<Button size="sm" v-if="allowEdit" @click="openChapterModal()">
|
<Button size="sm" v-if="allowEdit" @click="openChapterModal()">
|
||||||
@@ -16,7 +16,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
:class="{
|
:class="{
|
||||||
'shadow rounded-md py-2 px-2': showOutline && outline.data?.length,
|
'border-2 rounded-md py-2 px-2': showOutline && outline.data?.length,
|
||||||
}"
|
}"
|
||||||
>
|
>
|
||||||
<Disclosure
|
<Disclosure
|
||||||
@@ -33,10 +33,10 @@
|
|||||||
hidden: chapter.is_scorm_package,
|
hidden: chapter.is_scorm_package,
|
||||||
open: index == 1,
|
open: index == 1,
|
||||||
}"
|
}"
|
||||||
class="h-4 w-4 text-gray-900 stroke-1"
|
class="h-4 w-4 text-ink-gray-9 stroke-1"
|
||||||
/>
|
/>
|
||||||
<div
|
<div
|
||||||
class="text-base text-left font-medium leading-5 ml-2"
|
class="text-base text-left text-ink-gray-9 font-medium leading-5 ml-2"
|
||||||
@click="redirectToChapter(chapter)"
|
@click="redirectToChapter(chapter)"
|
||||||
>
|
>
|
||||||
{{ chapter.title }}
|
{{ chapter.title }}
|
||||||
@@ -46,14 +46,14 @@
|
|||||||
<FilePenLine
|
<FilePenLine
|
||||||
v-if="allowEdit"
|
v-if="allowEdit"
|
||||||
@click.prevent="openChapterModal(chapter)"
|
@click.prevent="openChapterModal(chapter)"
|
||||||
class="h-4 w-4 text-gray-900 invisible group-hover:visible"
|
class="h-4 w-4 text-ink-gray-9 invisible group-hover:visible"
|
||||||
/>
|
/>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
<Tooltip :text="__('Delete Chapter')" placement="bottom">
|
<Tooltip :text="__('Delete Chapter')" placement="bottom">
|
||||||
<Trash2
|
<Trash2
|
||||||
v-if="allowEdit"
|
v-if="allowEdit"
|
||||||
@click.prevent="trashChapter(chapter.name)"
|
@click.prevent="trashChapter(chapter.name)"
|
||||||
class="h-4 w-4 text-red-500 invisible group-hover:visible"
|
class="h-4 w-4 text-ink-red-3 invisible group-hover:visible"
|
||||||
/>
|
/>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
</div>
|
</div>
|
||||||
@@ -69,7 +69,12 @@
|
|||||||
:data-chapter="chapter.name"
|
:data-chapter="chapter.name"
|
||||||
>
|
>
|
||||||
<template #item="{ element: lesson }">
|
<template #item="{ element: lesson }">
|
||||||
<div class="outline-lesson pl-8 py-2 pr-4">
|
<div
|
||||||
|
class="outline-lesson pl-8 py-2 pr-4 text-ink-gray-9"
|
||||||
|
:class="
|
||||||
|
isActiveLesson(lesson.number) ? 'bg-surface-selected' : ''
|
||||||
|
"
|
||||||
|
>
|
||||||
<router-link
|
<router-link
|
||||||
:to="{
|
:to="{
|
||||||
name: allowEdit ? 'LessonForm' : 'Lesson',
|
name: allowEdit ? 'LessonForm' : 'Lesson',
|
||||||
@@ -83,21 +88,21 @@
|
|||||||
<div class="flex items-center text-sm leading-5 group">
|
<div class="flex items-center text-sm leading-5 group">
|
||||||
<MonitorPlay
|
<MonitorPlay
|
||||||
v-if="lesson.icon === 'icon-youtube'"
|
v-if="lesson.icon === 'icon-youtube'"
|
||||||
class="h-4 w-4 text-gray-900 stroke-1 mr-2"
|
class="h-4 w-4 stroke-1 mr-2"
|
||||||
/>
|
/>
|
||||||
<HelpCircle
|
<HelpCircle
|
||||||
v-else-if="lesson.icon === 'icon-quiz'"
|
v-else-if="lesson.icon === 'icon-quiz'"
|
||||||
class="h-4 w-4 text-gray-900 stroke-1 mr-2"
|
class="h-4 w-4 stroke-1 mr-2"
|
||||||
/>
|
/>
|
||||||
<FileText
|
<FileText
|
||||||
v-else-if="lesson.icon === 'icon-list'"
|
v-else-if="lesson.icon === 'icon-list'"
|
||||||
class="h-4 w-4 text-gray-900 stroke-1 mr-2"
|
class="h-4 w-4 text-ink-gray-9 stroke-1 mr-2"
|
||||||
/>
|
/>
|
||||||
{{ lesson.title }}
|
{{ lesson.title }}
|
||||||
<Trash2
|
<Trash2
|
||||||
v-if="allowEdit"
|
v-if="allowEdit"
|
||||||
@click.prevent="trashLesson(lesson.name, chapter.name)"
|
@click.prevent="trashLesson(lesson.name, chapter.name)"
|
||||||
class="h-4 w-4 text-red-500 ml-auto invisible group-hover:visible"
|
class="h-4 w-4 text-ink-red-3 ml-auto invisible group-hover:visible"
|
||||||
/>
|
/>
|
||||||
<Check
|
<Check
|
||||||
v-if="lesson.is_complete"
|
v-if="lesson.is_complete"
|
||||||
@@ -323,9 +328,11 @@ const redirectToChapter = (chapter) => {
|
|||||||
},
|
},
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
</script>
|
|
||||||
<style>
|
const isActiveLesson = (lessonNumber) => {
|
||||||
.outline-lesson:has(.router-link-active) {
|
return (
|
||||||
background-color: theme('colors.gray.100');
|
route.params.chapterNumber == lessonNumber.split('.')[0] &&
|
||||||
|
route.params.lessonNumber == lessonNumber.split('.')[1]
|
||||||
|
)
|
||||||
}
|
}
|
||||||
</style>
|
</script>
|
||||||
|
|||||||
@@ -7,7 +7,7 @@
|
|||||||
>
|
>
|
||||||
{{ __('Write a Review') }}
|
{{ __('Write a Review') }}
|
||||||
</Button>
|
</Button>
|
||||||
<div class="flex items-center font-semibold text-2xl">
|
<div class="flex items-center font-semibold text-2xl text-ink-gray-9">
|
||||||
{{ __('Student Reviews') }}
|
{{ __('Student Reviews') }}
|
||||||
</div>
|
</div>
|
||||||
<div class="grid gap-8 mt-10">
|
<div class="grid gap-8 mt-10">
|
||||||
@@ -28,17 +28,17 @@
|
|||||||
params: { username: review.owner_details.username },
|
params: { username: review.owner_details.username },
|
||||||
}"
|
}"
|
||||||
>
|
>
|
||||||
<span class="text-lg font-medium mr-4">
|
<span class="text-lg font-medium mr-4 text-ink-gray-7">
|
||||||
{{ review.owner_details.full_name }}
|
{{ review.owner_details.full_name }}
|
||||||
</span>
|
</span>
|
||||||
</router-link>
|
</router-link>
|
||||||
<span>
|
<span class="text-ink-gray-7">
|
||||||
{{ review.creation }}
|
{{ review.creation }}
|
||||||
</span>
|
</span>
|
||||||
<div class="flex mt-2">
|
<div class="flex mt-2">
|
||||||
<Star
|
<Star
|
||||||
v-for="index in 5"
|
v-for="index in 5"
|
||||||
class="h-5 w-5 text-gray-100 bg-gray-200 rounded-sm mr-2"
|
class="h-5 w-5 text-ink-gray-1 rounded-sm mr-2"
|
||||||
:class="
|
:class="
|
||||||
index <= Math.ceil(review.rating)
|
index <= Math.ceil(review.rating)
|
||||||
? 'fill-orange-500'
|
? 'fill-orange-500'
|
||||||
@@ -48,7 +48,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div v-if="review.review" class="mt-4 leading-5">
|
<div v-if="review.review" class="mt-4 leading-5 text-ink-gray-7">
|
||||||
{{ review.review }}
|
{{ review.review }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -6,7 +6,7 @@
|
|||||||
<div v-if="course.chapters.length">
|
<div v-if="course.chapters.length">
|
||||||
{{ course.chapters }}
|
{{ course.chapters }}
|
||||||
</div>
|
</div>
|
||||||
<div v-else class="border bg-white rounded-md p-5 text-center mt-4">
|
<div v-else class="border bg-surface-white rounded-md p-5 text-center mt-4">
|
||||||
<div>
|
<div>
|
||||||
{{
|
{{
|
||||||
__(
|
__(
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="relative flex h-full flex-col">
|
<div class="relative flex h-full flex-col">
|
||||||
<div class="h-full flex-1">
|
<div class="h-full flex-1">
|
||||||
<div class="flex h-screen text-base">
|
<div class="flex h-screen text-base bg-surface-white">
|
||||||
<div
|
<div
|
||||||
class="relative block min-h-0 flex-shrink-0 overflow-hidden hover:overflow-auto"
|
class="relative block min-h-0 flex-shrink-0 overflow-hidden hover:overflow-auto"
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -3,10 +3,10 @@
|
|||||||
<div v-if="!singleThread" class="flex items-center mb-5">
|
<div v-if="!singleThread" class="flex items-center mb-5">
|
||||||
<Button variant="outline" @click="showTopics = true">
|
<Button variant="outline" @click="showTopics = true">
|
||||||
<template #icon>
|
<template #icon>
|
||||||
<ChevronLeft class="w-5 h-5 stroke-1.5 text-gray-700" />
|
<ChevronLeft class="w-5 h-5 stroke-1.5 text-ink-gray-7" />
|
||||||
</template>
|
</template>
|
||||||
</Button>
|
</Button>
|
||||||
<span class="text-lg font-semibold ml-2">
|
<span class="text-lg font-semibold ml-2 text-ink-gray-9">
|
||||||
{{ topic.title }}
|
{{ topic.title }}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
@@ -17,7 +17,7 @@
|
|||||||
:class="{ 'border-b': index + 1 != replies.data.length }"
|
:class="{ 'border-b': index + 1 != replies.data.length }"
|
||||||
>
|
>
|
||||||
<div class="flex items-center justify-between mb-2">
|
<div class="flex items-center justify-between mb-2">
|
||||||
<div class="flex items-center">
|
<div class="flex items-center text-ink-gray-5">
|
||||||
<UserAvatar :user="reply.user" class="mr-2" />
|
<UserAvatar :user="reply.user" class="mr-2" />
|
||||||
<span>
|
<span>
|
||||||
{{ reply.user.full_name }}
|
{{ reply.user.full_name }}
|
||||||
@@ -63,7 +63,7 @@
|
|||||||
:fixedMenu="reply.editable || false"
|
:fixedMenu="reply.editable || false"
|
||||||
:editorClass="
|
:editorClass="
|
||||||
reply.editable
|
reply.editable
|
||||||
? 'ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-gray-300 prose-th:border-gray-300 prose-td:relative prose-th:relative prose-th:bg-gray-100 prose-sm max-w-none'
|
? 'ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-outline-gray-2 prose-th:border-outline-gray-2 prose-td:relative prose-th:relative prose-th:bg-surface-gray-2 prose-sm max-w-none'
|
||||||
: 'prose-sm'
|
: 'prose-sm'
|
||||||
"
|
"
|
||||||
/>
|
/>
|
||||||
@@ -71,13 +71,14 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<TextEditor
|
<TextEditor
|
||||||
|
v-if="renderEditor"
|
||||||
class="mt-5"
|
class="mt-5"
|
||||||
:content="newReply"
|
:content="newReply"
|
||||||
:mentions="mentionUsers"
|
:mentions="mentionUsers"
|
||||||
@change="(val) => (newReply = val)"
|
@change="(val) => (newReply = val)"
|
||||||
placeholder="Type your reply here..."
|
placeholder="Type your reply here..."
|
||||||
:fixedMenu="true"
|
:fixedMenu="true"
|
||||||
editorClass="ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-gray-300 prose-th:border-gray-300 prose-td:relative prose-th:relative prose-th:bg-gray-100 prose-sm max-w-none border border-gray-300 rounded-b-md min-h-[7rem] py-1 px-2"
|
editorClass="ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-outline-gray-2 prose-th:border-outline-gray-2 prose-td:relative prose-th:relative prose-th:bg-surface-gray-2 prose-sm max-w-none border border-outline-gray-2 rounded-b-md min-h-[7rem] py-1 px-2"
|
||||||
/>
|
/>
|
||||||
<div class="flex justify-between mt-2">
|
<div class="flex justify-between mt-2">
|
||||||
<span> </span>
|
<span> </span>
|
||||||
@@ -94,7 +95,7 @@ import { createResource, TextEditor, Button, Dropdown } from 'frappe-ui'
|
|||||||
import { timeAgo } from '../utils'
|
import { timeAgo } from '../utils'
|
||||||
import UserAvatar from '@/components/UserAvatar.vue'
|
import UserAvatar from '@/components/UserAvatar.vue'
|
||||||
import { ChevronLeft, MoreHorizontal } from 'lucide-vue-next'
|
import { ChevronLeft, MoreHorizontal } from 'lucide-vue-next'
|
||||||
import { ref, inject, onMounted, computed } from 'vue'
|
import { ref, inject, onMounted } from 'vue'
|
||||||
import { createToast } from '../utils'
|
import { createToast } from '../utils'
|
||||||
|
|
||||||
const showTopics = defineModel('showTopics')
|
const showTopics = defineModel('showTopics')
|
||||||
@@ -102,6 +103,8 @@ const newReply = ref('')
|
|||||||
const socket = inject('$socket')
|
const socket = inject('$socket')
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
const allUsers = inject('$allUsers')
|
const allUsers = inject('$allUsers')
|
||||||
|
const mentionUsers = ref([])
|
||||||
|
const renderEditor = ref(false)
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
topic: {
|
topic: {
|
||||||
@@ -124,6 +127,7 @@ onMounted(() => {
|
|||||||
socket.on('delete_message', (data) => {
|
socket.on('delete_message', (data) => {
|
||||||
replies.reload()
|
replies.reload()
|
||||||
})
|
})
|
||||||
|
fetchMentionUsers()
|
||||||
})
|
})
|
||||||
|
|
||||||
const replies = createResource({
|
const replies = createResource({
|
||||||
@@ -150,15 +154,26 @@ const newReplyResource = createResource({
|
|||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
const mentionUsers = computed(() => {
|
const fetchMentionUsers = () => {
|
||||||
let users = Object.values(allUsers.data).map((user) => {
|
if (user.data?.is_student) {
|
||||||
return {
|
renderEditor.value = true
|
||||||
value: user.name,
|
} else {
|
||||||
label: user.full_name,
|
allUsers.reload(
|
||||||
}
|
{},
|
||||||
})
|
{
|
||||||
return users
|
onSuccess(data) {
|
||||||
})
|
mentionUsers.value = Object.values(data).map((user) => {
|
||||||
|
return {
|
||||||
|
value: user.name,
|
||||||
|
label: user.full_name,
|
||||||
|
}
|
||||||
|
})
|
||||||
|
renderEditor.value = true
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const postReply = () => {
|
const postReply = () => {
|
||||||
newReplyResource.submit(
|
newReplyResource.submit(
|
||||||
@@ -178,7 +193,7 @@ const postReply = () => {
|
|||||||
title: 'Error',
|
title: 'Error',
|
||||||
text: err.messages?.[0] || err,
|
text: err.messages?.[0] || err,
|
||||||
icon: 'x',
|
icon: 'x',
|
||||||
iconClasses: 'bg-red-600 text-white rounded-md p-px',
|
iconClasses: 'bg-surface-red-5 text-ink-white rounded-md p-px',
|
||||||
position: 'top-center',
|
position: 'top-center',
|
||||||
timeout: 10,
|
timeout: 10,
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
<Button v-if="!singleThread" class="float-right" @click="openTopicModal()">
|
<Button v-if="!singleThread" class="float-right" @click="openTopicModal()">
|
||||||
{{ __('New {0}').format(singularize(title)) }}
|
{{ __('New {0}').format(singularize(title)) }}
|
||||||
</Button>
|
</Button>
|
||||||
<div class="text-xl font-semibold">
|
<div class="text-xl font-semibold text-ink-gray-9">
|
||||||
{{ __(title) }}
|
{{ __(title) }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -16,10 +16,10 @@
|
|||||||
>
|
>
|
||||||
<UserAvatar :user="topic.user" size="2xl" class="mr-4" />
|
<UserAvatar :user="topic.user" size="2xl" class="mr-4" />
|
||||||
<div>
|
<div>
|
||||||
<div class="text-lg font-semibold mb-1">
|
<div class="text-lg font-semibold mb-1 text-ink-gray-7">
|
||||||
{{ topic.title }}
|
{{ topic.title }}
|
||||||
</div>
|
</div>
|
||||||
<div class="flex items-center">
|
<div class="flex items-center text-ink-gray-5">
|
||||||
<span>
|
<span>
|
||||||
{{ topic.user.full_name }}
|
{{ topic.user.full_name }}
|
||||||
</span>
|
</span>
|
||||||
@@ -44,12 +44,12 @@
|
|||||||
v-else
|
v-else
|
||||||
class="flex flex-col items-center justify-center border-2 border-dashed mt-5 py-8 rounded-md"
|
class="flex flex-col items-center justify-center border-2 border-dashed mt-5 py-8 rounded-md"
|
||||||
>
|
>
|
||||||
<MessageSquareText class="w-7 h-7 text-gray-500 stroke-1.5 mr-2" />
|
<MessageSquareText class="w-7 h-7 text-ink-gray-4 stroke-1.5 mr-2" />
|
||||||
<div class="">
|
<div class="">
|
||||||
<div v-if="emptyStateTitle" class="font-medium mb-2">
|
<div v-if="emptyStateTitle" class="font-medium mb-2">
|
||||||
{{ __(emptyStateTitle) }}
|
{{ __(emptyStateTitle) }}
|
||||||
</div>
|
</div>
|
||||||
<div class="text-gray-600">
|
<div class="text-ink-gray-5">
|
||||||
{{ __(emptyStateText) }}
|
{{ __(emptyStateText) }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
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>
|
||||||
@@ -1,71 +1,41 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="flex rounded p-1 lg:px-2 lg:py-4 hover:bg-gray-100">
|
<div class="flex space-x-4 border rounded-md p-2">
|
||||||
<div class="flex w-3/5 md:w-2/5">
|
<img :src="job.company_logo" class="size-10 rounded-full object-contain" />
|
||||||
<img
|
<div class="flex flex-col space-y-2 flex-1">
|
||||||
:src="job.company_logo"
|
<div class="flex items-center justify-between">
|
||||||
class="w-12 h-12 rounded-lg object-contain mr-4"
|
<span class="font-semibold text-ink-gray-9">
|
||||||
:alt="job.company_name"
|
|
||||||
/>
|
|
||||||
<div>
|
|
||||||
<div class="font-medium mb-1">
|
|
||||||
{{ job.job_title }}
|
{{ job.job_title }}
|
||||||
</div>
|
|
||||||
<div class="text-gray-700">
|
|
||||||
{{ job.company_name }}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="flex justify-end w-1/5 text-gray-700">
|
|
||||||
{{ job.location.replace(',', '').split(' ')[0] }}
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="flex justify-end w-1/5 text-gray-700 text-right hidden md:block"
|
|
||||||
>
|
|
||||||
{{ job.type }}
|
|
||||||
</div>
|
|
||||||
<div class="flex justify-end w-1/5 text-sm text-gray-700 text-right">
|
|
||||||
{{ dayjs(job.creation).format('DD MMM YYYY') }}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<!-- <div class="flex flex-col shadow rounded-md p-4 h-full">
|
|
||||||
<div class="flex justify-between">
|
|
||||||
<div>
|
|
||||||
<div class="text-xl font-semibold mb-2">
|
|
||||||
{{ job.job_title }}
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
{{ __("posted by") }}
|
|
||||||
<span class="font-medium">
|
|
||||||
{{ job.company_name }}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<img
|
|
||||||
:src="job.company_logo"
|
|
||||||
class="w-12 h-12 rounded-lg object-contain"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div class="flex justify-between mt-8">
|
|
||||||
<div class="flex items-center">
|
|
||||||
<Badge :label="job.type" theme="green" size="lg" class="mr-4"/>
|
|
||||||
<Badge :label="job.location" theme="gray" size="lg">
|
|
||||||
<template #prefix>
|
|
||||||
<MapPin class="h-4 w-4 stroke-1.5" />
|
|
||||||
</template>
|
|
||||||
</Badge>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<span class="font-medium">
|
|
||||||
{{ dayjs(job.creation).format('DD MMM YYYY') }}
|
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="flex items-center space-x-2 text-ink-gray-5">
|
||||||
|
<Building2 class="w-4 h-4 stroke-1.5" />
|
||||||
|
<span>
|
||||||
|
{{ job.company_name }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div class="flex items-center space-x-2 text-ink-gray-5">
|
||||||
|
<MapPin class="w-4 h-4 stroke-1.5" />
|
||||||
|
<span>
|
||||||
|
{{ job.location }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div class="flex items-center space-x-2 text-ink-gray-5">
|
||||||
|
<Shapes class="w-4 h-4 stroke-1.5" />
|
||||||
|
<span>
|
||||||
|
{{ job.type }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div class="flex items-center space-x-2 text-ink-gray-5">
|
||||||
|
<Calendar class="w-4 h-4 stroke-1.5" />
|
||||||
|
<span> {{ __('posted') }} {{ dayjs(job.creation).fromNow() }} </span>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div> -->
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { MapPin } from 'lucide-vue-next'
|
import { Building2, Calendar, MapPin, Shapes } from 'lucide-vue-next'
|
||||||
import { Badge } from 'frappe-ui'
|
|
||||||
import { inject } from 'vue'
|
import { inject } from 'vue'
|
||||||
|
import { Avatar } from 'frappe-ui'
|
||||||
|
|
||||||
const dayjs = inject('$dayjs')
|
const dayjs = inject('$dayjs')
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
|
|||||||
@@ -1,12 +1,12 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="space-y-5">
|
<div class="space-y-5 text-ink-gray-9">
|
||||||
<div class="space-y-2">
|
<div class="space-y-2">
|
||||||
<div class="flex items-center text-sm font-medium space-x-2">
|
<div class="flex items-center text-sm font-medium space-x-2">
|
||||||
<span>
|
<span>
|
||||||
{{ __('What does include in preview mean?') }}
|
{{ __('What does include in preview mean?') }}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<div class="text-xs text-gray-600 mb-1 leading-5">
|
<div class="text-xs text-ink-gray-5 mb-1 leading-5">
|
||||||
{{
|
{{
|
||||||
__(
|
__(
|
||||||
'If Include in Preview is enabled for a lesson then the lesson will also be accessible to non logged in users.'
|
'If Include in Preview is enabled for a lesson then the lesson will also be accessible to non logged in users.'
|
||||||
@@ -23,9 +23,9 @@
|
|||||||
<span>
|
<span>
|
||||||
{{ __('How to add a Quiz?') }}
|
{{ __('How to add a Quiz?') }}
|
||||||
</span>
|
</span>
|
||||||
<Info class="w-3 h-3 text-gray-700" />
|
<Info class="w-3 h-3 text-ink-gray-7" />
|
||||||
</div>
|
</div>
|
||||||
<div class="text-xs text-gray-600 mb-1 leading-5">
|
<div class="text-xs text-ink-gray-5 mb-1 leading-5">
|
||||||
{{
|
{{
|
||||||
__(
|
__(
|
||||||
'Click on the add icon in the editor and select Quiz from the menu. It opens up a dialog, where you can either select a quiz from the list or create a new quiz. When you select the Create New option it redirects you to the quiz creation page.'
|
'Click on the add icon in the editor and select Quiz from the menu. It opens up a dialog, where you can either select a quiz from the list or create a new quiz. When you select the Create New option it redirects you to the quiz creation page.'
|
||||||
@@ -42,9 +42,9 @@
|
|||||||
<span class="leading-5">
|
<span class="leading-5">
|
||||||
{{ __(contentMap['upload']) }}
|
{{ __(contentMap['upload']) }}
|
||||||
</span>
|
</span>
|
||||||
<Info class="w-3 h-3 text-gray-700" />
|
<Info class="w-3 h-3 text-ink-gray-7" />
|
||||||
</div>
|
</div>
|
||||||
<div class="text-xs text-gray-600 mb-1 leading-5">
|
<div class="text-xs text-ink-gray-5 mb-1 leading-5">
|
||||||
{{
|
{{
|
||||||
__(
|
__(
|
||||||
'To upload Image, Video, Audio or PDF from your system, click on the add icon and select upload from the menu. Then choose the file you want to add to the lesson and it gets added to your lesson.'
|
'To upload Image, Video, Audio or PDF from your system, click on the add icon and select upload from the menu. Then choose the file you want to add to the lesson and it gets added to your lesson.'
|
||||||
@@ -61,9 +61,9 @@
|
|||||||
<span>
|
<span>
|
||||||
{{ __(contentMap['youtube']) }}
|
{{ __(contentMap['youtube']) }}
|
||||||
</span>
|
</span>
|
||||||
<Info class="w-3 h-3 text-gray-700" />
|
<Info class="w-3 h-3 text-ink-gray-7" />
|
||||||
</div>
|
</div>
|
||||||
<div class="text-xs text-gray-600 mb-1 leading-5">
|
<div class="text-xs text-ink-gray-5 mb-1 leading-5">
|
||||||
{{
|
{{
|
||||||
__(
|
__(
|
||||||
'Copy the URL of the video from YouTube and paste it in the editor.'
|
'Copy the URL of the video from YouTube and paste it in the editor.'
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="flex items-center justify-between mb-5">
|
<div class="flex items-center justify-between mb-5">
|
||||||
<div class="text-lg font-semibold">
|
<div class="text-lg font-semibold text-ink-gray-9">
|
||||||
{{ __('Live Class') }}
|
{{ __('Live Class') }}
|
||||||
</div>
|
</div>
|
||||||
<Button v-if="user.data.is_moderator" @click="openLiveClassModal">
|
<Button v-if="user.data.is_moderator" @click="openLiveClassModal">
|
||||||
@@ -15,9 +15,9 @@
|
|||||||
<div v-if="liveClasses.data?.length" class="grid grid-cols-2 gap-5">
|
<div v-if="liveClasses.data?.length" class="grid grid-cols-2 gap-5">
|
||||||
<div
|
<div
|
||||||
v-for="cls in liveClasses.data"
|
v-for="cls in liveClasses.data"
|
||||||
class="flex flex-col border rounded-md h-full text-gray-700 p-3"
|
class="flex flex-col border rounded-md h-full text-ink-gray-7 p-3"
|
||||||
>
|
>
|
||||||
<div class="font-semibold text-gray-900 text-lg mb-1">
|
<div class="font-semibold text-ink-gray-9 text-lg mb-1">
|
||||||
{{ cls.title }}
|
{{ cls.title }}
|
||||||
</div>
|
</div>
|
||||||
<div class="short-introduction">
|
<div class="short-introduction">
|
||||||
@@ -38,13 +38,13 @@
|
|||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
v-if="cls.date >= dayjs().format('YYYY-MM-DD')"
|
v-if="cls.date >= dayjs().format('YYYY-MM-DD')"
|
||||||
class="flex items-center space-x-2 text-gray-900 mt-auto"
|
class="flex items-center space-x-2 text-ink-gray-9 mt-auto"
|
||||||
>
|
>
|
||||||
<a
|
<a
|
||||||
v-if="user.data?.is_moderator || user.data?.is_evaluator"
|
v-if="user.data?.is_moderator || user.data?.is_evaluator"
|
||||||
:href="cls.start_url"
|
:href="cls.start_url"
|
||||||
target="_blank"
|
target="_blank"
|
||||||
class="w-1/2 cursor-pointer inline-flex items-center justify-center gap-2 transition-colors focus:outline-none text-gray-800 bg-gray-100 hover:bg-gray-200 active:bg-gray-300 focus-visible:ring focus-visible:ring-gray-400 h-7 text-base px-2 rounded"
|
class="w-1/2 cursor-pointer inline-flex items-center justify-center gap-2 transition-colors focus:outline-none text-ink-gray-8 bg-surface-gray-2 hover:bg-surface-gray-3 active:bg-surface-gray-4 focus-visible:ring focus-visible:ring-outline-gray-3 h-7 text-base px-2 rounded"
|
||||||
>
|
>
|
||||||
<Monitor class="h-4 w-4 stroke-1.5" />
|
<Monitor class="h-4 w-4 stroke-1.5" />
|
||||||
{{ __('Start') }}
|
{{ __('Start') }}
|
||||||
@@ -52,7 +52,7 @@
|
|||||||
<a
|
<a
|
||||||
:href="cls.join_url"
|
:href="cls.join_url"
|
||||||
target="_blank"
|
target="_blank"
|
||||||
class="w-full cursor-pointer inline-flex items-center justify-center gap-2 transition-colors focus:outline-none text-gray-800 bg-gray-100 hover:bg-gray-200 active:bg-gray-300 focus-visible:ring focus-visible:ring-gray-400 h-7 text-base px-2 rounded"
|
class="w-full cursor-pointer inline-flex items-center justify-center gap-2 transition-colors focus:outline-none text-ink-gray-8 bg-surface-gray-2 hover:bg-surface-gray-3 active:bg-surface-gray-4 focus-visible:ring focus-visible:ring-outline-gray-3 h-7 text-base px-2 rounded"
|
||||||
>
|
>
|
||||||
<Video class="h-4 w-4 stroke-1.5" />
|
<Video class="h-4 w-4 stroke-1.5" />
|
||||||
{{ __('Join') }}
|
{{ __('Join') }}
|
||||||
@@ -67,7 +67,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div v-else class="text-sm italic text-gray-600">
|
<div v-else class="text-sm italic text-ink-gray-5">
|
||||||
{{ __('No live classes scheduled') }}
|
{{ __('No live classes scheduled') }}
|
||||||
</div>
|
</div>
|
||||||
<LiveClassModal
|
<LiveClassModal
|
||||||
|
|||||||
@@ -2,10 +2,10 @@
|
|||||||
<div class="flex min-h-0 flex-col text-base">
|
<div class="flex min-h-0 flex-col text-base">
|
||||||
<div class="flex items-center justify-between">
|
<div class="flex items-center justify-between">
|
||||||
<div>
|
<div>
|
||||||
<div class="text-xl font-semibold mb-1">
|
<div class="text-xl font-semibold mb-1 text-ink-gray-9">
|
||||||
{{ __(label) }}
|
{{ __(label) }}
|
||||||
</div>
|
</div>
|
||||||
<!-- <div class="text-xs text-gray-600">
|
<!-- <div class="text-xs text-ink-gray-5">
|
||||||
{{ __(description) }}
|
{{ __(description) }}
|
||||||
</div> -->
|
</div> -->
|
||||||
</div>
|
</div>
|
||||||
@@ -36,7 +36,7 @@
|
|||||||
<FormControl
|
<FormControl
|
||||||
v-model="member.first_name"
|
v-model="member.first_name"
|
||||||
:placeholder="__('First Name')"
|
:placeholder="__('First Name')"
|
||||||
type="test"
|
type="text"
|
||||||
class="w-full"
|
class="w-full"
|
||||||
/>
|
/>
|
||||||
<Button @click="addMember()" variant="subtle">
|
<Button @click="addMember()" variant="subtle">
|
||||||
@@ -63,7 +63,7 @@
|
|||||||
/>
|
/>
|
||||||
<div class="space-y-1">
|
<div class="space-y-1">
|
||||||
<div class="flex">
|
<div class="flex">
|
||||||
<div class="text-gray-900">
|
<div class="text-ink-gray-9">
|
||||||
{{ member.full_name }}
|
{{ member.full_name }}
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
@@ -81,12 +81,14 @@
|
|||||||
</Badge>
|
</Badge>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="text-sm text-gray-700">
|
<div class="text-sm text-ink-gray-7">
|
||||||
{{ member.name }}
|
{{ member.name }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="flex items-center justify-center text-gray-700 text-sm">
|
<div
|
||||||
|
class="flex items-center justify-center text-ink-gray-7 text-sm"
|
||||||
|
>
|
||||||
<div v-if="member.last_active">
|
<div v-if="member.last_active">
|
||||||
{{ dayjs(member.last_active).format('DD MMM, YYYY HH:mm a') }}
|
{{ dayjs(member.last_active).format('DD MMM, YYYY HH:mm a') }}
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -5,7 +5,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
v-if="sidebarSettings.data"
|
v-if="sidebarSettings.data"
|
||||||
class="fixed flex items-center justify-around border-t border-gray-300 bottom-0 z-10 w-full bg-white standalone:pb-4"
|
class="fixed flex items-center justify-around border-t border-outline-gray-2 bottom-0 z-10 w-full bg-surface-white standalone:pb-4"
|
||||||
:style="{
|
:style="{
|
||||||
gridTemplateColumns: `repeat(${
|
gridTemplateColumns: `repeat(${
|
||||||
sidebarLinks.length + 1
|
sidebarLinks.length + 1
|
||||||
@@ -22,7 +22,7 @@
|
|||||||
<component
|
<component
|
||||||
:is="icons[tab.icon]"
|
:is="icons[tab.icon]"
|
||||||
class="h-6 w-6 stroke-1.5"
|
class="h-6 w-6 stroke-1.5"
|
||||||
:class="[isActive(tab) ? 'text-gray-900' : 'text-gray-600']"
|
:class="[isActive(tab) ? 'text-ink-gray-9' : 'text-ink-gray-5']"
|
||||||
/>
|
/>
|
||||||
</button>
|
</button>
|
||||||
<Popover
|
<Popover
|
||||||
@@ -33,7 +33,7 @@
|
|||||||
<template #target>
|
<template #target>
|
||||||
<component
|
<component
|
||||||
:is="icons['List']"
|
:is="icons['List']"
|
||||||
class="h-6 w-6 stroke-1.5 text-gray-600"
|
class="h-6 w-6 stroke-1.5 text-ink-gray-5"
|
||||||
/>
|
/>
|
||||||
</template>
|
</template>
|
||||||
<template #body-main>
|
<template #body-main>
|
||||||
@@ -46,7 +46,7 @@
|
|||||||
>
|
>
|
||||||
<component
|
<component
|
||||||
:is="icons[link.icon]"
|
:is="icons[link.icon]"
|
||||||
class="h-4 w-4 stroke-1.5 text-gray-600"
|
class="h-4 w-4 stroke-1.5 text-ink-gray-5"
|
||||||
/>
|
/>
|
||||||
<div>
|
<div>
|
||||||
{{ link.label }}
|
{{ link.label }}
|
||||||
|
|||||||
@@ -16,26 +16,26 @@
|
|||||||
<template #body-content>
|
<template #body-content>
|
||||||
<div class="flex flex-col gap-4">
|
<div class="flex flex-col gap-4">
|
||||||
<div class="">
|
<div class="">
|
||||||
<div class="mb-1.5 text-sm text-gray-600">
|
<div class="mb-1.5 text-sm text-ink-gray-5">
|
||||||
{{ __('Subject') }}
|
{{ __('Subject') }}
|
||||||
<span class="text-red-500">*</span>
|
<span class="text-ink-red-3">*</span>
|
||||||
</div>
|
</div>
|
||||||
<Input type="text" v-model="announcement.subject" />
|
<Input type="text" v-model="announcement.subject" />
|
||||||
</div>
|
</div>
|
||||||
<div class="">
|
<div class="">
|
||||||
<div class="mb-1.5 text-sm text-gray-600">
|
<div class="mb-1.5 text-sm text-ink-gray-5">
|
||||||
{{ __('Reply To') }}
|
{{ __('Reply To') }}
|
||||||
</div>
|
</div>
|
||||||
<Input type="text" v-model="announcement.replyTo" />
|
<Input type="text" v-model="announcement.replyTo" />
|
||||||
</div>
|
</div>
|
||||||
<div class="mb-4">
|
<div class="mb-4">
|
||||||
<div class="mb-1.5 text-sm text-gray-600">
|
<div class="mb-1.5 text-sm text-ink-gray-5">
|
||||||
{{ __('Announcement') }}
|
{{ __('Announcement') }}
|
||||||
</div>
|
</div>
|
||||||
<TextEditor
|
<TextEditor
|
||||||
:bubbleMenu="true"
|
:fixedMenu="true"
|
||||||
@change="(val) => (announcement.announcement = val)"
|
@change="(val) => (announcement.announcement = val)"
|
||||||
editorClass="prose-sm py-2 px-2 min-h-[200px] border-gray-300 hover:border-gray-400 rounded-md bg-gray-200"
|
editorClass="prose-sm py-2 px-2 min-h-[200px] border-outline-gray-2 hover:border-outline-gray-3 rounded-b-md bg-surface-gray-3"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -102,7 +102,7 @@ const makeAnnouncement = (close) => {
|
|||||||
)
|
)
|
||||||
},
|
},
|
||||||
onError(err) {
|
onError(err) {
|
||||||
showToast(__('Error'), __(err.messages?.[0] || err), 'check')
|
showToast(__('Error'), __(err.messages?.[0] || err), 'alert-circle')
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -31,7 +31,7 @@
|
|||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { Dialog, createResource } from 'frappe-ui'
|
import { Dialog, createResource } from 'frappe-ui'
|
||||||
import { ref, defineModel } from 'vue'
|
import { ref } from 'vue'
|
||||||
import Link from '@/components/Controls/Link.vue'
|
import Link from '@/components/Controls/Link.vue'
|
||||||
import { showToast } from '@/utils'
|
import { showToast } from '@/utils'
|
||||||
|
|
||||||
|
|||||||
@@ -18,7 +18,7 @@
|
|||||||
{{ student.progress }}% {{ __('Complete') }}
|
{{ student.progress }}% {{ __('Complete') }}
|
||||||
</Badge>
|
</Badge>
|
||||||
</div>
|
</div>
|
||||||
<div class="text-sm text-gray-700">
|
<div class="text-sm text-ink-gray-7">
|
||||||
{{ student.email }}
|
{{ student.email }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -32,25 +32,44 @@
|
|||||||
{{ __('Assessment') }}
|
{{ __('Assessment') }}
|
||||||
</span>
|
</span>
|
||||||
<span>
|
<span>
|
||||||
{{ __('Progress') }}
|
{{ __('Percentage/Status') }}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<div
|
<router-link
|
||||||
v-for="assessment in Object.keys(student.assessments)"
|
v-for="assessment in Object.keys(student.assessments)"
|
||||||
class="flex items-center text-gray-700 font-medium"
|
class="flex items-center text-ink-gray-7 font-medium"
|
||||||
|
:to="{
|
||||||
|
name:
|
||||||
|
student.assessments[assessment].type == 'LMS Assignment'
|
||||||
|
? 'AssignmentSubmission'
|
||||||
|
: '',
|
||||||
|
params:
|
||||||
|
student.assessments[assessment].type == 'LMS Assignment'
|
||||||
|
? {
|
||||||
|
assignmentID:
|
||||||
|
student.assessments[assessment].assessment,
|
||||||
|
submissionName:
|
||||||
|
student.assessments[assessment].submission,
|
||||||
|
}
|
||||||
|
: {},
|
||||||
|
}"
|
||||||
>
|
>
|
||||||
<span class="flex-1">
|
<span class="flex-1">
|
||||||
{{ assessment }}
|
{{ assessment }}
|
||||||
</span>
|
</span>
|
||||||
<span v-if="isAssignment(student.assessments[assessment])">
|
<span v-if="isAssignment(student.assessments[assessment].status)">
|
||||||
<Badge :theme="getStatusTheme(student.assessments[assessment])">
|
<Badge
|
||||||
{{ student.assessments[assessment] }}
|
:theme="
|
||||||
|
getStatusTheme(student.assessments[assessment].status)
|
||||||
|
"
|
||||||
|
>
|
||||||
|
{{ student.assessments[assessment].status }}
|
||||||
</Badge>
|
</Badge>
|
||||||
</span>
|
</span>
|
||||||
<span v-else>
|
<span v-else>
|
||||||
{{ student.assessments[assessment] }}
|
{{ student.assessments[assessment].status }}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</router-link>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Courses -->
|
<!-- Courses -->
|
||||||
@@ -65,7 +84,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
v-for="course in Object.keys(student.courses)"
|
v-for="course in Object.keys(student.courses)"
|
||||||
class="flex items-center text-gray-700 font-medium"
|
class="flex items-center text-ink-gray-7 font-medium"
|
||||||
>
|
>
|
||||||
<span class="flex-1">
|
<span class="flex-1">
|
||||||
{{ course }}
|
{{ course }}
|
||||||
@@ -78,7 +97,7 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Heatmap -->
|
<!-- Heatmap -->
|
||||||
<StudentHeatmap :member="student.email" :base_days="120" />
|
<StudentHeatmap :member="student.email" :days="120" />
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
</Dialog>
|
</Dialog>
|
||||||
|
|||||||
@@ -17,12 +17,6 @@
|
|||||||
>
|
>
|
||||||
<template #body-content>
|
<template #body-content>
|
||||||
<div class="space-y-4">
|
<div class="space-y-4">
|
||||||
<FormControl
|
|
||||||
type="select"
|
|
||||||
v-model="details.course"
|
|
||||||
:label="__('Course')"
|
|
||||||
:options="getCourses()"
|
|
||||||
/>
|
|
||||||
<Link
|
<Link
|
||||||
v-model="details.evaluator"
|
v-model="details.evaluator"
|
||||||
:label="__('Evaluator')"
|
:label="__('Evaluator')"
|
||||||
@@ -38,6 +32,12 @@
|
|||||||
v-model="details.expiry_date"
|
v-model="details.expiry_date"
|
||||||
:label="__('Expiry Date')"
|
:label="__('Expiry Date')"
|
||||||
/>
|
/>
|
||||||
|
<FormControl
|
||||||
|
type="select"
|
||||||
|
v-model="details.course"
|
||||||
|
:label="__('Course')"
|
||||||
|
:options="getCourses()"
|
||||||
|
/>
|
||||||
<Link
|
<Link
|
||||||
v-model="details.template"
|
v-model="details.template"
|
||||||
:label="__('Template')"
|
:label="__('Template')"
|
||||||
@@ -94,7 +94,7 @@ const createCertificate = createResource({
|
|||||||
template: details.template,
|
template: details.template,
|
||||||
published: details.published,
|
published: details.published,
|
||||||
course: values.course,
|
course: values.course,
|
||||||
batch: values.batch,
|
batch_name: values.batch,
|
||||||
member: values.member,
|
member: values.member,
|
||||||
evaluator: details.evaluator,
|
evaluator: details.evaluator,
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -47,19 +47,19 @@
|
|||||||
<div v-else class="">
|
<div v-else class="">
|
||||||
<div class="flex items-center">
|
<div class="flex items-center">
|
||||||
<div class="border rounded-md p-2 mr-2">
|
<div class="border rounded-md p-2 mr-2">
|
||||||
<FileText class="h-5 w-5 stroke-1.5 text-gray-700" />
|
<FileText class="h-5 w-5 stroke-1.5 text-ink-gray-7" />
|
||||||
</div>
|
</div>
|
||||||
<div class="flex flex-col">
|
<div class="flex flex-col">
|
||||||
<span>
|
<span>
|
||||||
{{ chapter.scorm_package.file_name }}
|
{{ chapter.scorm_package.file_name }}
|
||||||
</span>
|
</span>
|
||||||
<span class="text-sm text-gray-500 mt-1">
|
<span class="text-sm text-ink-gray-4 mt-1">
|
||||||
{{ getFileSize(chapter.scorm_package.file_size) }}
|
{{ getFileSize(chapter.scorm_package.file_size) }}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<X
|
<X
|
||||||
@click="() => (chapter.scorm_package = null)"
|
@click="() => (chapter.scorm_package = null)"
|
||||||
class="bg-gray-200 rounded-md cursor-pointer stroke-1.5 w-5 h-5 p-1 ml-4"
|
class="bg-surface-gray-3 rounded-md cursor-pointer stroke-1.5 w-5 h-5 p-1 ml-4"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -77,7 +77,7 @@ import {
|
|||||||
FormControl,
|
FormControl,
|
||||||
Switch,
|
Switch,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import { defineModel, reactive, watch } from 'vue'
|
import { reactive, watch } from 'vue'
|
||||||
import { showToast, getFileSize } from '@/utils/'
|
import { showToast, getFileSize } from '@/utils/'
|
||||||
import { capture } from '@/telemetry'
|
import { capture } from '@/telemetry'
|
||||||
import { FileText, X } from 'lucide-vue-next'
|
import { FileText, X } from 'lucide-vue-next'
|
||||||
@@ -145,9 +145,9 @@ const addChapter = async (close) => {
|
|||||||
{
|
{
|
||||||
onSuccess(data) {
|
onSuccess(data) {
|
||||||
cleanChapter()
|
cleanChapter()
|
||||||
if (!settingsStore.onboardingDetails.data?.is_onboarded) {
|
/* if (!settingsStore.onboardingDetails.data?.is_onboarded) {
|
||||||
settingsStore.onboardingDetails.reload()
|
settingsStore.onboardingDetails.reload()
|
||||||
}
|
} */
|
||||||
outline.value.reload()
|
outline.value.reload()
|
||||||
showToast(
|
showToast(
|
||||||
__('Success'),
|
__('Success'),
|
||||||
|
|||||||
@@ -18,7 +18,7 @@
|
|||||||
<FormControl v-model="topic.title" :label="__('Title')" type="text" />
|
<FormControl v-model="topic.title" :label="__('Title')" type="text" />
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<div class="mb-1.5 text-sm text-gray-600">
|
<div class="mb-1.5 text-sm text-ink-gray-5">
|
||||||
{{ __('Details') }}
|
{{ __('Details') }}
|
||||||
</div>
|
</div>
|
||||||
<TextEditor
|
<TextEditor
|
||||||
@@ -26,7 +26,7 @@
|
|||||||
@change="(val) => (topic.reply = val)"
|
@change="(val) => (topic.reply = val)"
|
||||||
:editable="true"
|
:editable="true"
|
||||||
:fixedMenu="true"
|
:fixedMenu="true"
|
||||||
editorClass="prose-sm max-w-none border-b border-x bg-gray-100 rounded-b-md py-1 px-2 min-h-[7rem]"
|
editorClass="prose-sm max-w-none border-b border-x bg-surface-gray-2 rounded-b-md py-1 px-2 min-h-[7rem]"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -35,7 +35,7 @@
|
|||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { Dialog, FormControl, TextEditor, createResource } from 'frappe-ui'
|
import { Dialog, FormControl, TextEditor, createResource } from 'frappe-ui'
|
||||||
import { reactive, defineModel } from 'vue'
|
import { reactive } from 'vue'
|
||||||
import { showToast, singularize } from '@/utils'
|
import { showToast, singularize } from '@/utils'
|
||||||
|
|
||||||
const topics = defineModel('reloadTopics')
|
const topics = defineModel('reloadTopics')
|
||||||
|
|||||||
@@ -5,7 +5,7 @@
|
|||||||
</template>
|
</template>
|
||||||
<template #body>
|
<template #body>
|
||||||
<div
|
<div
|
||||||
class="absolute left-1/2 mt-3 w-96 max-w-lg -translate-x-1/2 transform rounded-lg bg-white px-4 sm:px-0 lg:max-w-3xl"
|
class="absolute left-1/2 mt-3 w-96 max-w-lg -translate-x-1/2 transform rounded-lg bg-surface-white px-4 sm:px-0 lg:max-w-3xl"
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
class="overflow-hidden rounded-lg p-3 shadow-2xl ring-1 ring-black ring-opacity-5"
|
class="overflow-hidden rounded-lg p-3 shadow-2xl ring-1 ring-black ring-opacity-5"
|
||||||
@@ -35,7 +35,7 @@
|
|||||||
</FileUploader>
|
</FileUploader>
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
class="relative mt-2 grid w-[25.5rem] gap-2 bg-white lg:grid-cols-2"
|
class="relative mt-2 grid w-[25.5rem] gap-2 bg-surface-white lg:grid-cols-2"
|
||||||
>
|
>
|
||||||
<button
|
<button
|
||||||
v-for="image in images.data"
|
v-for="image in images.data"
|
||||||
@@ -53,7 +53,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
v-if="images.data"
|
v-if="images.data"
|
||||||
class="mt-2 text-center text-sm text-gray-500"
|
class="mt-2 text-center text-sm text-ink-gray-4"
|
||||||
>
|
>
|
||||||
{{ __('Image search powered by') }}
|
{{ __('Image search powered by') }}
|
||||||
<a class="underline" target="_blank" href="https://unsplash.com">
|
<a class="underline" target="_blank" href="https://unsplash.com">
|
||||||
|
|||||||
@@ -33,24 +33,24 @@
|
|||||||
</template>
|
</template>
|
||||||
</FileUploader>
|
</FileUploader>
|
||||||
<div v-else class="mb-4">
|
<div v-else class="mb-4">
|
||||||
<div class="text-xs text-gray-600 mb-1">
|
<div class="text-xs text-ink-gray-5 mb-1">
|
||||||
{{ __('Profile Image') }}
|
{{ __('Profile Image') }}
|
||||||
</div>
|
</div>
|
||||||
<div class="flex items-center">
|
<div class="flex items-center">
|
||||||
<div class="border rounded-md p-2 mr-2">
|
<div class="border rounded-md p-2 mr-2">
|
||||||
<FileText class="h-5 w-5 stroke-1.5 text-gray-700" />
|
<FileText class="h-5 w-5 stroke-1.5 text-ink-gray-7" />
|
||||||
</div>
|
</div>
|
||||||
<div class="text-base flex flex-col">
|
<div class="text-base flex flex-col">
|
||||||
<span>
|
<span>
|
||||||
{{ profile.image.file_name }}
|
{{ profile.image.file_name }}
|
||||||
</span>
|
</span>
|
||||||
<span class="text-sm text-gray-500 mt-1">
|
<span class="text-sm text-ink-gray-4 mt-1">
|
||||||
{{ getFileSize(profile.image.file_size) }}
|
{{ getFileSize(profile.image.file_size) }}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<X
|
<X
|
||||||
@click="removeImage()"
|
@click="removeImage()"
|
||||||
class="bg-gray-200 rounded-md cursor-pointer stroke-1.5 w-5 h-5 p-1 ml-4"
|
class="bg-surface-gray-3 rounded-md cursor-pointer stroke-1.5 w-5 h-5 p-1 ml-4"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -71,14 +71,14 @@
|
|||||||
/>
|
/>
|
||||||
|
|
||||||
<div class="mb-4">
|
<div class="mb-4">
|
||||||
<div class="mb-1.5 text-sm text-gray-600">
|
<div class="mb-1.5 text-sm text-ink-gray-5">
|
||||||
{{ __('Bio') }}
|
{{ __('Bio') }}
|
||||||
</div>
|
</div>
|
||||||
<TextEditor
|
<TextEditor
|
||||||
:fixedMenu="true"
|
:fixedMenu="true"
|
||||||
@change="(val) => (profile.bio = val)"
|
@change="(val) => (profile.bio = val)"
|
||||||
:content="profile.bio"
|
:content="profile.bio"
|
||||||
editorClass="prose-sm py-2 px-2 min-h-[200px] border-gray-300 hover:border-gray-400 rounded-md bg-gray-200"
|
editorClass="prose-sm py-2 px-2 min-h-[200px] border-outline-gray-2 hover:border-outline-gray-3 rounded-md bg-surface-gray-3"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -94,7 +94,7 @@ import {
|
|||||||
createResource,
|
createResource,
|
||||||
TextEditor,
|
TextEditor,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import { reactive, watch, defineModel } from 'vue'
|
import { reactive, watch } from 'vue'
|
||||||
import { FileText, X } from 'lucide-vue-next'
|
import { FileText, X } from 'lucide-vue-next'
|
||||||
import { getFileSize, showToast, escapeHTML } from '@/utils'
|
import { getFileSize, showToast, escapeHTML } from '@/utils'
|
||||||
|
|
||||||
|
|||||||
@@ -16,25 +16,33 @@
|
|||||||
<template #body-content>
|
<template #body-content>
|
||||||
<div class="flex flex-col gap-4">
|
<div class="flex flex-col gap-4">
|
||||||
<div>
|
<div>
|
||||||
<div class="mb-1.5 text-sm text-gray-600">
|
<div class="mb-1.5 text-sm text-ink-gray-5">
|
||||||
{{ __('Course') }}
|
{{ __('Course') }}
|
||||||
</div>
|
</div>
|
||||||
<Select v-model="evaluation.course" :options="getCourses()" />
|
<Select v-model="evaluation.course" :options="getCourses()" />
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<div class="mb-1.5 text-sm text-gray-600">
|
<div class="mb-1.5 text-sm text-ink-gray-5">
|
||||||
{{ __('Date') }}
|
{{ __('Date') }}
|
||||||
</div>
|
</div>
|
||||||
<FormControl type="date" v-model="evaluation.date" />
|
<FormControl
|
||||||
|
type="date"
|
||||||
|
v-model="evaluation.date"
|
||||||
|
:min="
|
||||||
|
dayjs()
|
||||||
|
.add(dayjs.duration({ days: 1 }))
|
||||||
|
.format('YYYY-MM-DD')
|
||||||
|
"
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div v-if="slots.data?.length">
|
<div v-if="slots.data?.length">
|
||||||
<div class="mb-1.5 text-sm text-gray-600">
|
<div class="mb-1.5 text-sm text-ink-gray-5">
|
||||||
{{ __('Select a slot') }}
|
{{ __('Select a slot') }}
|
||||||
</div>
|
</div>
|
||||||
<div class="grid grid-cols-2 gap-2">
|
<div class="grid grid-cols-2 gap-2">
|
||||||
<div v-for="slot in slots.data">
|
<div v-for="slot in slots.data">
|
||||||
<div
|
<div
|
||||||
class="text-base text-center border rounded-md bg-gray-200 p-2 cursor-pointer"
|
class="text-base text-center border rounded-md bg-surface-gray-3 p-2 cursor-pointer"
|
||||||
@click="saveSlot(slot)"
|
@click="saveSlot(slot)"
|
||||||
:class="{
|
:class="{
|
||||||
'border-gray-900': evaluation.start_time == slot.start_time,
|
'border-gray-900': evaluation.start_time == slot.start_time,
|
||||||
@@ -48,7 +56,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
v-else-if="evaluation.course && evaluation.date"
|
v-else-if="evaluation.course && evaluation.date"
|
||||||
class="text-sm italic text-red-600"
|
class="text-sm italic text-ink-red-4"
|
||||||
>
|
>
|
||||||
{{ __('No slots available for this date.') }}
|
{{ __('No slots available for this date.') }}
|
||||||
</div>
|
</div>
|
||||||
@@ -58,7 +66,7 @@
|
|||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { Dialog, createResource, Select, FormControl } from 'frappe-ui'
|
import { Dialog, createResource, Select, FormControl } from 'frappe-ui'
|
||||||
import { defineModel, reactive, watch, inject } from 'vue'
|
import { reactive, watch, inject } from 'vue'
|
||||||
import { createToast, formatTime } from '@/utils/'
|
import { createToast, formatTime } from '@/utils/'
|
||||||
|
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
@@ -143,7 +151,7 @@ function submitEvaluation(close) {
|
|||||||
title: unavailabilityMessage ? __('Evaluator is Unavailable') : '',
|
title: unavailabilityMessage ? __('Evaluator is Unavailable') : '',
|
||||||
text: message,
|
text: message,
|
||||||
icon: unavailabilityMessage ? 'alert-circle' : 'x',
|
icon: unavailabilityMessage ? 'alert-circle' : 'x',
|
||||||
iconClasses: 'bg-yellow-600 text-white rounded-md p-px',
|
iconClasses: 'bg-yellow-600 text-ink-white rounded-md p-px',
|
||||||
position: 'top-center',
|
position: 'top-center',
|
||||||
timeout: 10,
|
timeout: 10,
|
||||||
})
|
})
|
||||||
@@ -161,6 +169,11 @@ const getCourses = () => {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (courses.length == 1) {
|
||||||
|
evaluation.course = courses[0].value
|
||||||
|
}
|
||||||
|
|
||||||
return courses
|
return courses
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -12,7 +12,7 @@
|
|||||||
{{ event.title }}
|
{{ event.title }}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="flex flex-col space-y-4 text-sm text-gray-800">
|
<div class="flex flex-col space-y-4 text-sm text-ink-gray-8">
|
||||||
<Tooltip :text="__('Email ID')">
|
<Tooltip :text="__('Email ID')">
|
||||||
<div class="flex items-center space-x-2 w-fit">
|
<div class="flex items-center space-x-2 w-fit">
|
||||||
<User class="h-4 w-4 stroke-1.5" />
|
<User class="h-4 w-4 stroke-1.5" />
|
||||||
|
|||||||
@@ -48,13 +48,13 @@
|
|||||||
</div>
|
</div>
|
||||||
<div v-else class="flex items-center">
|
<div v-else class="flex items-center">
|
||||||
<div class="border rounded-md p-2 mr-2">
|
<div class="border rounded-md p-2 mr-2">
|
||||||
<FileText class="h-5 w-5 stroke-1.5 text-gray-700" />
|
<FileText class="h-5 w-5 stroke-1.5 text-ink-gray-7" />
|
||||||
</div>
|
</div>
|
||||||
<div class="flex flex-col">
|
<div class="flex flex-col">
|
||||||
<span>
|
<span>
|
||||||
{{ resume.file_name }}
|
{{ resume.file_name }}
|
||||||
</span>
|
</span>
|
||||||
<span class="text-sm text-gray-500 mt-1">
|
<span class="text-sm text-ink-gray-4 mt-1">
|
||||||
{{ getFileSize(resume.file_size) }}
|
{{ getFileSize(resume.file_size) }}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
@@ -66,7 +66,7 @@
|
|||||||
<script setup>
|
<script setup>
|
||||||
import { Dialog, FileUploader, Button, createResource } from 'frappe-ui'
|
import { Dialog, FileUploader, Button, createResource } from 'frappe-ui'
|
||||||
import { FileText } from 'lucide-vue-next'
|
import { FileText } from 'lucide-vue-next'
|
||||||
import { ref, inject, defineModel } from 'vue'
|
import { ref, inject } from 'vue'
|
||||||
import { createToast, getFileSize } from '@/utils/'
|
import { createToast, getFileSize } from '@/utils/'
|
||||||
|
|
||||||
const resume = ref(null)
|
const resume = ref(null)
|
||||||
@@ -116,7 +116,7 @@ const submitResume = (close) => {
|
|||||||
title: 'Success',
|
title: 'Success',
|
||||||
text: 'Your application has been submitted',
|
text: 'Your application has been submitted',
|
||||||
icon: 'check',
|
icon: 'check',
|
||||||
iconClasses: 'bg-green-600 text-white rounded-md p-px',
|
iconClasses: 'bg-surface-green-3 text-ink-white rounded-md p-px',
|
||||||
})
|
})
|
||||||
application.value.reload()
|
application.value.reload()
|
||||||
close()
|
close()
|
||||||
@@ -126,7 +126,7 @@ const submitResume = (close) => {
|
|||||||
title: 'Error',
|
title: 'Error',
|
||||||
text: err.messages?.[0] || err,
|
text: err.messages?.[0] || err,
|
||||||
icon: 'x',
|
icon: 'x',
|
||||||
iconClasses: 'bg-red-600 text-white rounded-md p-px',
|
iconClasses: 'bg-surface-red-5 text-ink-white rounded-md p-px',
|
||||||
position: 'top-center',
|
position: 'top-center',
|
||||||
timeout: 10,
|
timeout: 10,
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -39,13 +39,19 @@
|
|||||||
:required="true"
|
:required="true"
|
||||||
/>
|
/>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
<FormControl
|
|
||||||
v-model="liveClass.timezone"
|
<div class="space-y-1.5">
|
||||||
type="select"
|
<label class="block text-ink-gray-5 text-xs" for="batchTimezone">
|
||||||
:options="getTimezoneOptions()"
|
{{ __('Timezone') }}
|
||||||
:label="__('Timezone')"
|
<span class="text-ink-red-3">*</span>
|
||||||
:required="true"
|
</label>
|
||||||
/>
|
<Autocomplete
|
||||||
|
@update:modelValue="(opt) => (liveClass.timezone = opt.value)"
|
||||||
|
:modelValue="liveClass.timezone"
|
||||||
|
:options="getTimezoneOptions()"
|
||||||
|
:required="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<FormControl
|
<FormControl
|
||||||
@@ -83,18 +89,14 @@
|
|||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import {
|
import {
|
||||||
Input,
|
|
||||||
DatePicker,
|
|
||||||
Select,
|
|
||||||
Textarea,
|
|
||||||
Dialog,
|
Dialog,
|
||||||
createResource,
|
createResource,
|
||||||
Tooltip,
|
Tooltip,
|
||||||
FormControl,
|
FormControl,
|
||||||
|
Autocomplete,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import { reactive, inject } from 'vue'
|
import { reactive, inject, onMounted } from 'vue'
|
||||||
import { getTimezones, createToast } from '@/utils/'
|
import { getTimezones, createToast, getUserTimezone } from '@/utils/'
|
||||||
import { Info } from 'lucide-vue-next'
|
|
||||||
|
|
||||||
const liveClasses = defineModel('reloadLiveClasses')
|
const liveClasses = defineModel('reloadLiveClasses')
|
||||||
const show = defineModel()
|
const show = defineModel()
|
||||||
@@ -120,6 +122,10 @@ let liveClass = reactive({
|
|||||||
host: user.data.name,
|
host: user.data.name,
|
||||||
})
|
})
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
liveClass.timezone = getUserTimezone()
|
||||||
|
})
|
||||||
|
|
||||||
const getTimezoneOptions = () => {
|
const getTimezoneOptions = () => {
|
||||||
return getTimezones().map((timezone) => {
|
return getTimezones().map((timezone) => {
|
||||||
return {
|
return {
|
||||||
@@ -200,7 +206,7 @@ const submitLiveClass = (close) => {
|
|||||||
title: 'Error',
|
title: 'Error',
|
||||||
text: err.messages?.[0] || err,
|
text: err.messages?.[0] || err,
|
||||||
icon: 'x',
|
icon: 'x',
|
||||||
iconClasses: 'bg-red-600 text-white rounded-md p-px',
|
iconClasses: 'bg-surface-red-5 text-ink-white rounded-md p-px',
|
||||||
position: 'top-center',
|
position: 'top-center',
|
||||||
timeout: 10,
|
timeout: 10,
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -4,7 +4,7 @@
|
|||||||
<div class="space-y-4">
|
<div class="space-y-4">
|
||||||
<div
|
<div
|
||||||
v-if="!editMode"
|
v-if="!editMode"
|
||||||
class="flex items-center text-xs text-gray-700 space-x-5"
|
class="flex items-center text-xs text-ink-gray-7 space-x-5"
|
||||||
>
|
>
|
||||||
<div class="flex items-center space-x-2">
|
<div class="flex items-center space-x-2">
|
||||||
<input
|
<input
|
||||||
@@ -34,7 +34,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<div v-if="questionType == 'new' || editMode" class="space-y-2">
|
<div v-if="questionType == 'new' || editMode" class="space-y-2">
|
||||||
<div>
|
<div>
|
||||||
<label class="block text-xs text-gray-600 mb-1">
|
<label class="block text-xs text-ink-gray-5 mb-1">
|
||||||
{{ __('Question') }}
|
{{ __('Question') }}
|
||||||
</label>
|
</label>
|
||||||
<TextEditor
|
<TextEditor
|
||||||
@@ -42,7 +42,7 @@
|
|||||||
@change="(val) => (question.question = val)"
|
@change="(val) => (question.question = val)"
|
||||||
:editable="true"
|
:editable="true"
|
||||||
:fixedMenu="true"
|
:fixedMenu="true"
|
||||||
editorClass="prose-sm max-w-none border-b border-x bg-gray-100 rounded-b-md py-1 px-2 min-h-[7rem]"
|
editorClass="prose-sm max-w-none border-b border-x bg-surface-gray-2 rounded-b-md py-1 px-2 min-h-[7rem]"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<FormControl
|
<FormControl
|
||||||
|
|||||||
@@ -16,13 +16,13 @@
|
|||||||
<template #body-content>
|
<template #body-content>
|
||||||
<div class="flex flex-col gap-4">
|
<div class="flex flex-col gap-4">
|
||||||
<div>
|
<div>
|
||||||
<div class="mb-1.5 text-sm text-gray-600">
|
<div class="mb-1.5 text-sm text-ink-gray-5">
|
||||||
{{ __('Rating') }}
|
{{ __('Rating') }}
|
||||||
</div>
|
</div>
|
||||||
<Rating v-model="review.rating" />
|
<Rating v-model="review.rating" />
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<div class="mb-1.5 text-sm text-gray-600">
|
<div class="mb-1.5 text-sm text-ink-gray-5">
|
||||||
{{ __('Review') }}
|
{{ __('Review') }}
|
||||||
</div>
|
</div>
|
||||||
<Textarea type="text" size="md" rows="5" v-model="review.review" />
|
<Textarea type="text" size="md" rows="5" v-model="review.review" />
|
||||||
@@ -33,7 +33,7 @@
|
|||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { Dialog, Textarea, createResource } from 'frappe-ui'
|
import { Dialog, Textarea, createResource } from 'frappe-ui'
|
||||||
import { defineModel, reactive } from 'vue'
|
import { reactive } from 'vue'
|
||||||
import Rating from '@/components/Controls/Rating.vue'
|
import Rating from '@/components/Controls/Rating.vue'
|
||||||
import { createToast } from '@/utils/'
|
import { createToast } from '@/utils/'
|
||||||
|
|
||||||
@@ -81,7 +81,7 @@ function submitReview(close) {
|
|||||||
createToast({
|
createToast({
|
||||||
text: err.messages?.[0] || err,
|
text: err.messages?.[0] || err,
|
||||||
icon: 'x',
|
icon: 'x',
|
||||||
iconClasses: 'text-red-600 bg-red-300',
|
iconClasses: 'text-ink-red-4 bg-surface-red-4',
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -2,14 +2,14 @@
|
|||||||
<Dialog v-model="show" :options="{ size: '4xl' }">
|
<Dialog v-model="show" :options="{ size: '4xl' }">
|
||||||
<template #body>
|
<template #body>
|
||||||
<div class="flex h-[calc(100vh_-_8rem)]">
|
<div class="flex h-[calc(100vh_-_8rem)]">
|
||||||
<div class="flex w-52 shrink-0 flex-col bg-gray-50 p-2">
|
<div class="flex w-52 shrink-0 flex-col bg-surface-gray-2 p-2">
|
||||||
<h1 class="mb-3 px-2 pt-2 text-lg font-semibold">
|
<h1 class="mb-3 px-2 pt-2 text-lg font-semibold text-ink-gray-9">
|
||||||
{{ __('Settings') }}
|
{{ __('Settings') }}
|
||||||
</h1>
|
</h1>
|
||||||
<div v-for="tab in tabs" :key="tab.label">
|
<div v-for="tab in tabs" :key="tab.label">
|
||||||
<div
|
<div
|
||||||
v-if="!tab.hideLabel"
|
v-if="!tab.hideLabel"
|
||||||
class="mb-2 mt-3 flex cursor-pointer gap-1.5 px-1 text-base font-medium text-gray-600 transition-all duration-300 ease-in-out"
|
class="mb-2 mt-3 flex cursor-pointer gap-1.5 px-1 text-base font-medium text-ink-gray-5 transition-all duration-300 ease-in-out"
|
||||||
>
|
>
|
||||||
<span>{{ __(tab.label) }}</span>
|
<span>{{ __(tab.label) }}</span>
|
||||||
</div>
|
</div>
|
||||||
@@ -21,8 +21,8 @@
|
|||||||
class="w-full"
|
class="w-full"
|
||||||
:class="
|
:class="
|
||||||
activeTab?.label == item.label
|
activeTab?.label == item.label
|
||||||
? 'bg-white shadow-sm'
|
? 'bg-surface-selected shadow-sm'
|
||||||
: 'hover:bg-gray-100'
|
: 'hover:bg-surface-gray-2'
|
||||||
"
|
"
|
||||||
@click="activeTab = item"
|
@click="activeTab = item"
|
||||||
/>
|
/>
|
||||||
@@ -32,7 +32,7 @@
|
|||||||
<div
|
<div
|
||||||
v-if="activeTab && data.doc"
|
v-if="activeTab && data.doc"
|
||||||
:key="activeTab.label"
|
:key="activeTab.label"
|
||||||
class="flex flex-1 flex-col px-10 py-8"
|
class="flex flex-1 flex-col px-10 py-8 bg-surface-modal"
|
||||||
>
|
>
|
||||||
<Members
|
<Members
|
||||||
v-if="activeTab.label === 'Members'"
|
v-if="activeTab.label === 'Members'"
|
||||||
@@ -118,6 +118,13 @@ const tabsStructure = computed(() => {
|
|||||||
'This will enforce students to go through programs assigned to them in the correct order.',
|
'This will enforce students to go through programs assigned to them in the correct order.',
|
||||||
type: 'checkbox',
|
type: 'checkbox',
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
label: 'Allow Guest Access',
|
||||||
|
name: 'allow_guest_access',
|
||||||
|
description:
|
||||||
|
'If enabled, users can access the course and batch lists without logging in.',
|
||||||
|
type: 'checkbox',
|
||||||
|
},
|
||||||
{
|
{
|
||||||
label: 'Send calendar invite for evaluations',
|
label: 'Send calendar invite for evaluations',
|
||||||
name: 'send_calendar_invite_for_evaluations',
|
name: 'send_calendar_invite_for_evaluations',
|
||||||
@@ -130,7 +137,7 @@ const tabsStructure = computed(() => {
|
|||||||
name: 'unsplash_access_key',
|
name: 'unsplash_access_key',
|
||||||
description:
|
description:
|
||||||
'Optional. If this is set, students can pick a cover image from the unsplash library for their profile page. https://unsplash.com/documentation#getting-started.',
|
'Optional. If this is set, students can pick a cover image from the unsplash library for their profile page. https://unsplash.com/documentation#getting-started.',
|
||||||
type: 'text',
|
type: 'password',
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -46,11 +46,9 @@ const studentResource = createResource({
|
|||||||
makeParams(values) {
|
makeParams(values) {
|
||||||
return {
|
return {
|
||||||
doc: {
|
doc: {
|
||||||
doctype: 'Batch Student',
|
doctype: 'LMS Batch Enrollment',
|
||||||
parent: props.batch,
|
batch: props.batch,
|
||||||
parenttype: 'LMS Batch',
|
member: student.value,
|
||||||
parentfield: 'students',
|
|
||||||
student: student.value,
|
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
<div class="border rounded-md w-1/3 mx-auto my-32">
|
<div class="border rounded-md w-1/3 mx-auto my-32">
|
||||||
<div class="border-b px-5 py-3 font-medium">
|
<div class="border-b px-5 py-3 font-medium">
|
||||||
<span
|
<span
|
||||||
class="inline-flex items-center before:bg-red-600 before:w-2 before:h-2 before:rounded-md before:mr-2"
|
class="inline-flex items-center before:bg-surface-red-5 before:w-2 before:h-2 before:rounded-md before:mr-2"
|
||||||
></span>
|
></span>
|
||||||
{{ __('Not Permitted') }}
|
{{ __('Not Permitted') }}
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user