Compare commits
210 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
b48e007ea8 | ||
|
|
d5e8973866 | ||
|
|
a8c530f98c | ||
|
|
47769ccd62 | ||
|
|
bfc1d9a0a8 | ||
|
|
824484e608 | ||
|
|
d3f7baae4c | ||
|
|
8d961e9b71 | ||
|
|
f22855920c | ||
|
|
18728e3519 | ||
|
|
65dc2838d3 | ||
|
|
be930ce076 | ||
|
|
1ea47a008c | ||
|
|
e0169cff79 | ||
|
|
7c53ac10e2 | ||
|
|
212e0de6e9 | ||
|
|
8e74384b5a | ||
|
|
86e7e68ce1 | ||
|
|
a77999dbb6 | ||
|
|
3288fb0f06 | ||
|
|
a81b384f90 | ||
|
|
75c11d3fcc | ||
|
|
51a6cc035c | ||
|
|
ae8008d05c | ||
|
|
7f44177986 | ||
|
|
d88aaedf3f | ||
|
|
802d4ccb0b | ||
|
|
76a84c7f5d | ||
|
|
40aefba203 | ||
|
|
6cdfb822b4 | ||
|
|
fdacab66f7 | ||
|
|
5cc12e71df | ||
|
|
f5e5fa2f36 | ||
|
|
6022b83b8c | ||
|
|
a01b1657cc | ||
|
|
6b785bd0e6 | ||
|
|
0beffc3083 | ||
|
|
d345d09b13 | ||
|
|
ec75b8cb8f | ||
|
|
503068b0d2 | ||
|
|
60dc9682b4 | ||
|
|
38e1eb8fc7 | ||
|
|
6490bb9258 | ||
|
|
bdac91c48c | ||
|
|
c95366281b | ||
|
|
484a31ab7e | ||
|
|
dc9546955a | ||
|
|
07b6e851cd | ||
|
|
c3a98db6ae | ||
|
|
0bb50a9742 | ||
|
|
76f96bfcf8 | ||
|
|
a2458281fc | ||
|
|
8467bdf19b | ||
|
|
7c28067922 | ||
|
|
a955db05a0 | ||
|
|
a5ab893f05 | ||
|
|
6afc94704a | ||
|
|
bd79e746ed | ||
|
|
fb58ab08cb | ||
|
|
7868925ba2 | ||
|
|
85f69af38f | ||
|
|
63c9068306 | ||
|
|
1fea3fc52d | ||
|
|
1e26e28515 | ||
|
|
8edddaa502 | ||
|
|
5a68a85317 | ||
|
|
655fde109f | ||
|
|
463a1d8c7c | ||
|
|
726ae8ac06 | ||
|
|
6f73be9a0b | ||
|
|
c1fdddbac3 | ||
|
|
e0127d0824 | ||
|
|
9a07882e8e | ||
|
|
2416777df2 | ||
|
|
d811014b86 | ||
|
|
3134ef6392 | ||
|
|
6c3bb3480e | ||
|
|
0b7ff1dff3 | ||
|
|
9ac4efe9dc | ||
|
|
e278e1ed35 | ||
|
|
9db203d74f | ||
|
|
c6366835d2 | ||
|
|
5e8ad81ff3 | ||
|
|
ac24a353b0 | ||
|
|
8a3c681a6f | ||
|
|
2da946236d | ||
|
|
d4641c9135 | ||
|
|
cf710d7be5 | ||
|
|
e56b8928f7 | ||
|
|
66121e6cce | ||
|
|
cd824631bb | ||
|
|
115b72f2f0 | ||
|
|
8d17b35160 | ||
|
|
4c21ce2caa | ||
|
|
0057467acf | ||
|
|
7048b22df0 | ||
|
|
ddc3352b4b | ||
|
|
060a2808de | ||
|
|
d8f8a8e559 | ||
|
|
c471d39ba8 | ||
|
|
55ec813f82 | ||
|
|
727f7b032c | ||
|
|
d1b613c0bb | ||
|
|
c3af65e535 | ||
|
|
d688d5cdd9 | ||
|
|
97543a43eb | ||
|
|
0e6df83961 | ||
|
|
6329d9c917 | ||
|
|
015e228304 | ||
|
|
a9f40d16f0 | ||
|
|
b8da14a32e | ||
|
|
a64b0f734a | ||
|
|
34ba2fb361 | ||
|
|
98ccb15796 | ||
|
|
6c06f7d19b | ||
|
|
86b129a25f | ||
|
|
6e8d4cd8e8 | ||
|
|
1b4622bdb2 | ||
|
|
58d51579e3 | ||
|
|
06706ea41b | ||
|
|
d634a0f784 | ||
|
|
a92159b811 | ||
|
|
7e1e37393c | ||
|
|
d2f9a2cea4 | ||
|
|
5111d83eee | ||
|
|
0dc77343c4 | ||
|
|
cec5913632 | ||
|
|
75d43a1563 | ||
|
|
1ecdbd9e06 | ||
|
|
a90e3d611c | ||
|
|
d49d638253 | ||
|
|
83338a56c0 | ||
|
|
562020de70 | ||
|
|
044907edeb | ||
|
|
cfa1aa87fc | ||
|
|
0ac32ee474 | ||
|
|
de0675f850 | ||
|
|
1c529790f2 | ||
|
|
40bcc4d572 | ||
|
|
58f109e79c | ||
|
|
cb324f6269 | ||
|
|
7cafaf5cbc | ||
|
|
a394952630 | ||
|
|
68e87f20aa | ||
|
|
64ed0b3e94 | ||
|
|
fcaaee958d | ||
|
|
29e356ff86 | ||
|
|
460edc7bc7 | ||
|
|
582c7af12d | ||
|
|
af533a7a2c | ||
|
|
acbede157f | ||
|
|
8e1db293db | ||
|
|
f63a627ff2 | ||
|
|
b1a0556c12 | ||
|
|
0097ede6ed | ||
|
|
b72774e54d | ||
|
|
08261c804f | ||
|
|
3027a9e523 | ||
|
|
c3995952b3 | ||
|
|
ff1642382c | ||
|
|
cfe35e40da | ||
|
|
c3238a9f91 | ||
|
|
58f08bf065 | ||
|
|
d3ac6ea337 | ||
|
|
6649b7955f | ||
|
|
15a53d33e0 | ||
|
|
57f09542a2 | ||
|
|
fa384b391d | ||
|
|
12b138c39f | ||
|
|
420a5f39eb | ||
|
|
12c2666bd1 | ||
|
|
1ecbc2e3f9 | ||
|
|
e1a78382c3 | ||
|
|
dcf5c72cad | ||
|
|
2ebf6be609 | ||
|
|
4ce7019ce6 | ||
|
|
3faf814162 | ||
|
|
52bd9825d8 | ||
|
|
b6028e741c | ||
|
|
4ee1693434 | ||
|
|
cbc7892b25 | ||
|
|
a4fa2ef0b3 | ||
|
|
96de90cb5f | ||
|
|
dfb22c81c3 | ||
|
|
6a70ed18d8 | ||
|
|
629c237349 | ||
|
|
cf014bca3c | ||
|
|
9323d8e17d | ||
|
|
1ba63a2175 | ||
|
|
b5551fd8ba | ||
|
|
fac0038af8 | ||
|
|
ee6685e324 | ||
|
|
0fb18f995c | ||
|
|
61e13aa7cd | ||
|
|
acb8c6c500 | ||
|
|
af838121d9 | ||
|
|
f504841a5c | ||
|
|
be49ba6d04 | ||
|
|
24ffed11fb | ||
|
|
73754bd104 | ||
|
|
2e1aac4931 | ||
|
|
93b3eda05c | ||
|
|
740584d883 | ||
|
|
5e6160149f | ||
|
|
be66c563a8 | ||
|
|
92c380c74b | ||
|
|
e25f161980 | ||
|
|
822603128d | ||
|
|
9dbe8fbb1f | ||
|
|
26f1e228a9 |
BIN
.github/batch.png
vendored
Normal file
BIN
.github/batch.png
vendored
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 1.1 MiB |
BIN
.github/certificate.png
vendored
Normal file
BIN
.github/certificate.png
vendored
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 912 KiB |
2
.github/helper/install_dependencies.sh
vendored
2
.github/helper/install_dependencies.sh
vendored
@@ -5,7 +5,7 @@ echo "Setting Up System Dependencies..."
|
|||||||
|
|
||||||
sudo apt update
|
sudo apt update
|
||||||
sudo apt remove mysql-server mysql-client
|
sudo apt remove mysql-server mysql-client
|
||||||
sudo apt install libcups2-dev redis-server mariadb-client-10.6
|
sudo apt-get install libcups2-dev redis-server mariadb-client
|
||||||
|
|
||||||
install_wkhtmltopdf() {
|
install_wkhtmltopdf() {
|
||||||
wget -q https://github.com/wkhtmltopdf/packaging/releases/download/0.12.6-1/wkhtmltox_0.12.6-1.focal_amd64.deb
|
wget -q https://github.com/wkhtmltopdf/packaging/releases/download/0.12.6-1/wkhtmltox_0.12.6-1.focal_amd64.deb
|
||||||
|
|||||||
BIN
.github/hero.png
vendored
Normal file
BIN
.github/hero.png
vendored
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 2.0 MiB |
BIN
.github/lms-logo.png
vendored
Normal file
BIN
.github/lms-logo.png
vendored
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 5.4 KiB |
BIN
.github/quiz.png
vendored
Normal file
BIN
.github/quiz.png
vendored
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 1.0 MiB |
2
.github/workflows/ui-tests.yml
vendored
2
.github/workflows/ui-tests.yml
vendored
@@ -24,7 +24,7 @@ jobs:
|
|||||||
|
|
||||||
services:
|
services:
|
||||||
mariadb:
|
mariadb:
|
||||||
image: mariadb:10.6
|
image: mariadb:10.8
|
||||||
env:
|
env:
|
||||||
MARIADB_ROOT_PASSWORD: 123
|
MARIADB_ROOT_PASSWORD: 123
|
||||||
ports:
|
ports:
|
||||||
|
|||||||
247
README.md
247
README.md
@@ -1,115 +1,174 @@
|
|||||||
<p align="center">
|
<div align="center" markdown="1">
|
||||||
<a href="https://www.frappelms.com/">
|
|
||||||
<img src="https://frappe.io/files/lms.png" alt="Frappe LMS" width="50px" height="50px">
|
|
||||||
</a>
|
|
||||||
<p align="center">Easy to use, open source, learning management system.</p>
|
|
||||||
</p>
|
|
||||||
|
|
||||||
|
<img src=".github/lms-logo.png" alt="Frappe Learning logo" width="80" height="80"/>
|
||||||
|
<h1>Frappe Learning</h1>
|
||||||
|
|
||||||
|
**Easy to use, open source, Learning Management System**
|
||||||
|
|
||||||
<p align="center">
|

|
||||||
<a href="https://www.producthunt.com/posts/frappe-lms?utm_source=badge-top-post-topic-badge&utm_medium=badge&utm_souce=badge-frappe-lms" target="_blank"><img src="https://api.producthunt.com/widgets/embed-image/v1/top-post-topic-badge.svg?post_id=396079&theme=dark&period=weekly&topic_id=204" alt="Frappe LMS - Easy to use, 100% open source learning management system | Product Hunt" style="width: 250px; height: 54px;" width="250" height="54" /></a>
|
|
||||||
</p>
|
|
||||||
|
|
||||||
|
|
||||||
<div align="center" style="max-height: 40px;">
|
|
||||||
<a href="https://frappecloud.com/lms/signup">
|
|
||||||
<img src=".github/try-on-f-cloud.svg" height="40">
|
|
||||||
</a>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<p align="center">
|
<div align="center">
|
||||||
<a href="https://dashboard.cypress.io/projects/vandxn/runs">
|
<img src=".github/hero.png?v=5" alt="Hero Image" width="72%" />
|
||||||
<img alt="cypress" src="https://img.shields.io/endpoint?url=https://dashboard.cypress.io/badge/simple/vandxn/main&style=flat&logo=cypress">
|
</div>
|
||||||
</a>
|
<br />
|
||||||
<a href="https://github.com/frappe/lms/blob/main/LICENSE">
|
<div align="center">
|
||||||
<img alt="license" src="https://img.shields.io/badge/license-AGPLv3-blue">
|
<a href="https://frappe.io/learning">Website</a>
|
||||||
</a>
|
-
|
||||||
</p>
|
<a href="https://docs.frappe.io/learning">Documentation</a>
|
||||||
|
</div>
|
||||||
|
|
||||||
<img width="1402" alt="Lesson" src="https://frappelms.com/files/banner.png">
|
## Frappe Learning
|
||||||
|
Frappe Learning is an easy-to-use learning system that helps you bring structure to your content.
|
||||||
|
|
||||||
|
### Motivation
|
||||||
|
In 2021, we were looking for a Learning Management System to launch [Mon.School](https://mon.school) for FOSS United. We checked out Moodle, but it didn’t feel right. The forms were unnecessarily lengthy and the UI was confusing. It shouldn't be this hard to create a course right? So I started making a learning system for Mon.School which soon became a product in itself. The aim is to have a simple platform that anyone can use to launch a course of their own and make knowledge sharing easier.
|
||||||
|
|
||||||
|
### Key Features
|
||||||
|
|
||||||
|
- **Structured Learning**: Design a course with a 3-level hierarchy, where your courses have chapters and you can group your lessons within these chapters. This ensures that the context of the lesson is set by the chapter.
|
||||||
|
|
||||||
|
- **Live Classes**: Group learners into batches based on courses and duration. You can then create Zoom live class for these batches right from the app. Learners get to see the list of live classes they have to take as a part of this batch.
|
||||||
|
|
||||||
|
- **Quizzes and Assignments**: Create quizzes where questions can have single-choice, multiple-choice options, or can be open ended. Instructors can also add assignments which learners can submit as PDF's or Documents.
|
||||||
|
|
||||||
|
- **Getting Certified**: Once a learner has completed the course or batch, you can grant them a certificate. The app provides an inbuilt certificate template. You can use this or else create a template of your own and use that instead.
|
||||||
|
|
||||||
<details>
|
<details>
|
||||||
<summary>Show more screenshots</summary>
|
<summary>View Screenshots</summary>
|
||||||
<img width="1520" alt="ss1" src="https://user-images.githubusercontent.com/31363128/210056046-584bc8aa-d28c-4514-b031-73817012837d.png">
|
|
||||||
<img width="830" alt="ss2" src="https://user-images.githubusercontent.com/31363128/210056097-36849182-6db0-43a2-8c62-5333cd2aedf4.png">
|
|
||||||
<img width="941" alt="ss3" src="https://user-images.githubusercontent.com/31363128/210056134-01a7c429-1ef4-434e-9d43-128dda35d7e5.png">
|

|
||||||
|
<div align="center">
|
||||||
|
<sub>
|
||||||
|
Create batches to group your learners
|
||||||
|
</sub>
|
||||||
|
</div>
|
||||||
|
<br>
|
||||||
|
|
||||||
|
|
||||||
|

|
||||||
|
<div align="center">
|
||||||
|
<sub>
|
||||||
|
Evaluate their knowledge by quizzes
|
||||||
|
</sub>
|
||||||
|
</div>
|
||||||
|
<br>
|
||||||
|
|
||||||
|
|
||||||
|

|
||||||
|
<div align="center">
|
||||||
|
<sub>
|
||||||
|
Autenticate their work with certification
|
||||||
|
</sub>
|
||||||
|
</div>
|
||||||
</details>
|
</details>
|
||||||
|
|
||||||
Frappe LMS is an easy-to-use, open-source learning management system. You can use it to create and share online courses. The app has a clear UI that helps students focus only on what's important and assists in distraction-free learning.
|
|
||||||
|
|
||||||
You can create courses and lessons through simple forms. Lessons can be in the form of text, videos, quizzes or a combination of all these. You can keep your students engaged with quizzes to help revise and test the concepts learned. Course Instructors and Students can reach out to each other through the discussions section available for each lesson and get queries resolved.
|
### Under the Hood
|
||||||
|
|
||||||
## Features
|
- [**Frappe Framework**](https://github.com/frappe/frappe): A full-stack web application framework.
|
||||||
- Create online courses. 📚
|
|
||||||
- Add detailed descriptions and preview videos to the course. 🎬
|
|
||||||
- Add videos, quizzes, and assignments to your lessons and make them interesting and interactive 📝
|
|
||||||
- Discussions section below each lesson where instructors and students can interact with each other. 💬
|
|
||||||
- Create batches to group your students based on courses and track their progress 🏛
|
|
||||||
- Statistics dashboard that provides all important numbers at a glimpse. 📈
|
|
||||||
- Job Board where users can post and look for jobs. 💼
|
|
||||||
- People directory with each person's profile page 👨👩👧👦
|
|
||||||
- Set cover image, profile photo, short bio, and other professional information. 🦹🏼♀️
|
|
||||||
- Simple layout that optimizes readability 🤓
|
|
||||||
- Delightful user experience in overall usage ✨
|
|
||||||
|
|
||||||
## Tech Stack
|
- [**Frappe UI**](https://github.com/frappe/frappe-ui): A Vue-based UI library, to provide a modern user interface.
|
||||||
|
|
||||||
Frappe LMS is built on [Frappe Framework](https://frappeframework.com) which is a batteries-included python web framework.
|
## Production Setup
|
||||||
These are some of the tools it's built on:
|
|
||||||
- [Python](https://www.python.org)
|
|
||||||
- [Redis](https://redis.io/)
|
|
||||||
- [MariaDB](https://mariadb.org/)
|
|
||||||
- [Socket.io](https://socket.io/)
|
|
||||||
|
|
||||||
## Local Setup
|
|
||||||
|
|
||||||
### Docker
|
|
||||||
You need Docker, docker-compose, and git setup on your machine. Refer to [Docker documentation](https://docs.docker.com/). After that, run the following commands:
|
|
||||||
```
|
|
||||||
git clone https://github.com/frappe/lms
|
|
||||||
cd apps/lms/docker
|
|
||||||
docker-compose up
|
|
||||||
```
|
|
||||||
|
|
||||||
Wait for some time until the setup script creates a site. After that, you can access `http://localhost:8000` in your browser and the app's login screen should appear.
|
|
||||||
You'll have to go through the setup wizard to set up the website the first time you access it. Log in using the following credentials to complete the setup wizard.
|
|
||||||
|
|
||||||
```
|
|
||||||
Username: Administrator
|
|
||||||
password: admin
|
|
||||||
```
|
|
||||||
|
|
||||||
### Frappe Bench
|
|
||||||
|
|
||||||
Currently, this app depends on the `develop` branch of [frappe](https://github.com/frappe/frappe).
|
|
||||||
|
|
||||||
1. Setup frappe-bench by following [this guide](https://frappeframework.com/docs/v14/user/en/installation)
|
|
||||||
1. In the frappe-bench directory, run `bench start` and keep it running. Open a new terminal session and cd into the `frappe-bench` directory.
|
|
||||||
1. Run the following commands:
|
|
||||||
```sh
|
|
||||||
bench new-site lms.test
|
|
||||||
bench get-app lms
|
|
||||||
bench --site lms.test install-app lms
|
|
||||||
bench --site lms.test add-to-hosts
|
|
||||||
|
|
||||||
1. Now, you can access the site at `http://lms.test:8000`
|
|
||||||
|
|
||||||
|
|
||||||
## Deployment
|
|
||||||
Frappe LMS is an app built on top of the Frappe Framework. So, you can follow any deployment guide for hosting a Frappe Framework-based site.
|
|
||||||
|
|
||||||
### Managed Hosting
|
### Managed Hosting
|
||||||
Frappe LMS can be deployed in a few clicks on [Frappe Cloud](https://frappecloud.com/marketplace/apps/lms).
|
|
||||||
|
|
||||||
### Self-hosting
|
You can try [Frappe Cloud](https://frappecloud.com), a simple, user-friendly and sophisticated [open-source](https://github.com/frappe/press) platform to host Frappe applications with peace of mind.
|
||||||
If you want to self-host, you can follow official [Frappe Bench Installation](https://github.com/frappe/bench#installation) instructions.
|
|
||||||
|
|
||||||
## Bugs and Feature Requests
|
It takes care of installation, setup, upgrades, monitoring, maintenance and support of your Frappe deployments. It is a fully featured developer platform with an ability to manage and control multiple Frappe deployments.
|
||||||
If you find any bugs or have a feature idea for the app, feel free to report them here on [GitHub Issues](https://github.com/frappe/lms/issues). Make sure you share enough information (app screenshots, browser console screenshots, stack traces, etc) for project maintainers.
|
|
||||||
|
|
||||||
## License
|
<div>
|
||||||
Distributed under [GNU AFFERO GENERAL PUBLIC LICENSE](license.txt)
|
<a href="https://frappecloud.com/lms/signup" target="_blank">
|
||||||
|
<picture>
|
||||||
|
<source media="(prefers-color-scheme: dark)" srcset="https://frappe.io/files/try-on-fc-white.png">
|
||||||
|
<img src="https://frappe.io/files/try-on-fc-black.png" alt="Try on Frappe Cloud" height="28" />
|
||||||
|
</picture>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
### Self Hosting
|
||||||
|
|
||||||
|
Follow these steps to set up Frappe Learning in production:
|
||||||
|
|
||||||
|
**Step 1**: Download the easy install script
|
||||||
|
|
||||||
|
```bash
|
||||||
|
wget https://frappe.io/easy-install.py
|
||||||
|
```
|
||||||
|
|
||||||
|
**Step 2**: Run the deployment command
|
||||||
|
|
||||||
|
```bash
|
||||||
|
python3 ./easy-install.py deploy \
|
||||||
|
--project=learning_prod_setup \
|
||||||
|
--email=your_email.example.com \
|
||||||
|
--image=ghcr.io/frappe/learning \
|
||||||
|
--version=stable \
|
||||||
|
--app=learning \
|
||||||
|
--sitename subdomain.domain.tld
|
||||||
|
```
|
||||||
|
|
||||||
|
Replace the following parameters with your values:
|
||||||
|
- `your_email.example.com`: Your email address
|
||||||
|
- `subdomain.domain.tld`: Your domain name where Learning will be hosted
|
||||||
|
|
||||||
|
The script will set up a production-ready instance of Frappe Learning with all the necessary configurations in about 5 minutes.
|
||||||
|
|
||||||
|
## Development Setup
|
||||||
|
|
||||||
|
### Docker
|
||||||
|
|
||||||
|
You need Docker, docker-compose and git setup on your machine. Refer [Docker documentation](https://docs.docker.com/). After that, follow below steps:
|
||||||
|
|
||||||
|
**Step 1**: Setup folder and download the required files
|
||||||
|
|
||||||
|
mkdir frappe-learning
|
||||||
|
cd frappe-learning
|
||||||
|
|
||||||
|
# Download the docker-compose file
|
||||||
|
wget -O docker-compose.yml https://raw.githubusercontent.com/frappe/lms/develop/docker/docker-compose.yml
|
||||||
|
|
||||||
|
# Download the setup script
|
||||||
|
wget -O init.sh https://raw.githubusercontent.com/frappe/lms/develop/docker/init.sh
|
||||||
|
|
||||||
|
**Step 2**: Run the container and daemonize it
|
||||||
|
|
||||||
|
docker compose up -d
|
||||||
|
|
||||||
|
**Step 3**: The site [http://lms.localhost:8000/lms](http://lms.localhost:8000/lms) should now be available. The default credentials are:
|
||||||
|
- Username: Administrator
|
||||||
|
- Password: admin
|
||||||
|
|
||||||
|
### Local
|
||||||
|
|
||||||
|
To setup the repository locally follow the steps mentioned below:
|
||||||
|
|
||||||
|
1. Install bench and setup a `frappe-bench` directory by following the [Installation Steps](https://frappeframework.com/docs/user/en/installation)
|
||||||
|
1. Start the server by running `bench start`
|
||||||
|
1. In a separate terminal window, create a new site by running `bench new-site learning.test`
|
||||||
|
1. Map your site to localhost with the command `bench --site learning.test add-to-hosts`
|
||||||
|
1. Get the Learning app. Run `bench get-app https://github.com/frappe/lms`
|
||||||
|
1. Run `bench --site learning.test install-app lms`.
|
||||||
|
1. Now open the URL `http://learning.test:8000/lms` in your browser, you should see the app running
|
||||||
|
|
||||||
|
## Learn and connect
|
||||||
|
|
||||||
|
- [Telegram Public Group](https://t.me/frappelms)
|
||||||
|
- [Discuss Forum](https://discuss.frappe.io/c/lms/70)
|
||||||
|
- [Documentation](https://docs.frappe.io/learning)
|
||||||
|
- [YouTube](https://www.youtube.com/channel/UCn3bV5kx77HsVwtnlCeEi_A)
|
||||||
|
|
||||||
|
<br>
|
||||||
|
<br>
|
||||||
|
<div align="center" style="padding-top: 0.75rem;">
|
||||||
|
<a href="https://frappe.io" target="_blank">
|
||||||
|
<picture>
|
||||||
|
<source media="(prefers-color-scheme: dark)" srcset="https://frappe.io/files/Frappe-white.png">
|
||||||
|
<img src="https://frappe.io/files/Frappe-black.png" alt="Frappe Technologies" height="28"/>
|
||||||
|
</picture>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
|||||||
@@ -13,6 +13,6 @@ module.exports = defineConfig({
|
|||||||
openMode: 0,
|
openMode: 0,
|
||||||
},
|
},
|
||||||
e2e: {
|
e2e: {
|
||||||
baseUrl: "http://test:8000",
|
baseUrl: "http://lms1:8000",
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ version: "3.7"
|
|||||||
name: lms
|
name: lms
|
||||||
services:
|
services:
|
||||||
mariadb:
|
mariadb:
|
||||||
image: mariadb:10.6
|
image: mariadb:10.8
|
||||||
command:
|
command:
|
||||||
- --character-set-server=utf8mb4
|
- --character-set-server=utf8mb4
|
||||||
- --collation-server=utf8mb4_unicode_ci
|
- --collation-server=utf8mb4_unicode_ci
|
||||||
|
|||||||
@@ -18,17 +18,19 @@
|
|||||||
"@editorjs/nested-list": "^1.4.2",
|
"@editorjs/nested-list": "^1.4.2",
|
||||||
"@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",
|
||||||
"ace-builds": "^1.36.2",
|
"ace-builds": "^1.36.2",
|
||||||
"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.72",
|
"frappe-ui": "^0.1.89",
|
||||||
"lucide-vue-next": "^0.383.0",
|
"lucide-vue-next": "^0.383.0",
|
||||||
"markdown-it": "^14.0.0",
|
"markdown-it": "^14.0.0",
|
||||||
"pinia": "^2.0.33",
|
"pinia": "^2.0.33",
|
||||||
"socket.io-client": "^4.7.2",
|
"socket.io-client": "^4.7.2",
|
||||||
"tailwindcss": "^3.3.3",
|
"tailwindcss": "^3.3.3",
|
||||||
|
"typescript": "^5.7.2",
|
||||||
"vue": "^3.4.23",
|
"vue": "^3.4.23",
|
||||||
"vue-chartjs": "^5.3.0",
|
"vue-chartjs": "^5.3.0",
|
||||||
"vue-draggable-next": "^2.2.1",
|
"vue-draggable-next": "^2.2.1",
|
||||||
|
|||||||
@@ -1,18 +1,18 @@
|
|||||||
<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 bg-gray-50"
|
||||||
:class="isSidebarCollapsed ? 'w-14' : 'w-56'"
|
:class="sidebarStore.isSidebarCollapsed ? 'w-14' : 'w-56'"
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
class="flex flex-col overflow-hidden"
|
class="flex flex-col overflow-hidden"
|
||||||
:class="isSidebarCollapsed ? 'items-center' : ''"
|
:class="sidebarStore.isSidebarCollapsed ? 'items-center' : ''"
|
||||||
>
|
>
|
||||||
<UserDropdown :isCollapsed="isSidebarCollapsed" />
|
<UserDropdown :isCollapsed="sidebarStore.isSidebarCollapsed" />
|
||||||
<div class="flex flex-col" v-if="sidebarSettings.data">
|
<div class="flex flex-col" v-if="sidebarSettings.data">
|
||||||
<SidebarLink
|
<SidebarLink
|
||||||
v-for="link in sidebarLinks"
|
v-for="link in sidebarLinks"
|
||||||
:link="link"
|
:link="link"
|
||||||
:isCollapsed="isSidebarCollapsed"
|
:isCollapsed="sidebarStore.isSidebarCollapsed"
|
||||||
class="mx-2 my-0.5"
|
class="mx-2 my-0.5"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
@@ -22,11 +22,11 @@
|
|||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
class="flex items-center justify-between pr-2 cursor-pointer"
|
class="flex items-center justify-between pr-2 cursor-pointer"
|
||||||
:class="isSidebarCollapsed ? 'pl-3' : 'pl-4'"
|
:class="sidebarStore.isSidebarCollapsed ? 'pl-3' : 'pl-4'"
|
||||||
@click="showWebPages = !showWebPages"
|
@click="showWebPages = !showWebPages"
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
v-if="!isSidebarCollapsed"
|
v-if="!sidebarStore.isSidebarCollapsed"
|
||||||
class="flex items-center text-sm text-gray-600 my-1"
|
class="flex items-center text-sm text-gray-600 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">
|
||||||
@@ -53,7 +53,7 @@
|
|||||||
<SidebarLink
|
<SidebarLink
|
||||||
v-for="link in sidebarSettings.data.web_pages"
|
v-for="link in sidebarSettings.data.web_pages"
|
||||||
:link="link"
|
:link="link"
|
||||||
:isCollapsed="isSidebarCollapsed"
|
:isCollapsed="sidebarStore.isSidebarCollapsed"
|
||||||
class="mx-2 my-0.5"
|
class="mx-2 my-0.5"
|
||||||
:showControls="isModerator ? true : false"
|
:showControls="isModerator ? true : false"
|
||||||
@openModal="openPageModal"
|
@openModal="openPageModal"
|
||||||
@@ -64,17 +64,19 @@
|
|||||||
</div>
|
</div>
|
||||||
<SidebarLink
|
<SidebarLink
|
||||||
:link="{
|
:link="{
|
||||||
label: isSidebarCollapsed ? 'Expand' : 'Collapse',
|
label: sidebarStore.isSidebarCollapsed ? 'Expand' : 'Collapse',
|
||||||
}"
|
}"
|
||||||
:isCollapsed="isSidebarCollapsed"
|
:isCollapsed="sidebarStore.isSidebarCollapsed"
|
||||||
@click="isSidebarCollapsed = !isSidebarCollapsed"
|
@click="toggleSidebar()"
|
||||||
class="m-2"
|
class="m-2"
|
||||||
>
|
>
|
||||||
<template #icon>
|
<template #icon>
|
||||||
<span class="grid h-5 w-6 flex-shrink-0 place-items-center">
|
<span class="grid h-5 w-6 flex-shrink-0 place-items-center">
|
||||||
<CollapseSidebar
|
<CollapseSidebar
|
||||||
class="h-4.5 w-4.5 text-gray-700 duration-300 ease-in-out"
|
class="h-4.5 w-4.5 text-gray-700 duration-300 ease-in-out"
|
||||||
:class="{ '[transform:rotateY(180deg)]': isSidebarCollapsed }"
|
:class="{
|
||||||
|
'[transform:rotateY(180deg)]': sidebarStore.isSidebarCollapsed,
|
||||||
|
}"
|
||||||
/>
|
/>
|
||||||
</span>
|
</span>
|
||||||
</template>
|
</template>
|
||||||
@@ -96,12 +98,15 @@ import { ref, onMounted, inject, watch } from 'vue'
|
|||||||
import { getSidebarLinks } from '../utils'
|
import { getSidebarLinks } from '../utils'
|
||||||
import { usersStore } from '@/stores/user'
|
import { usersStore } from '@/stores/user'
|
||||||
import { sessionStore } from '@/stores/session'
|
import { sessionStore } from '@/stores/session'
|
||||||
|
import { useSidebar } from '@/stores/sidebar'
|
||||||
|
import { 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 { createResource, Button } 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()
|
||||||
const { userResource } = usersStore()
|
const { userResource } = usersStore()
|
||||||
|
let sidebarStore = useSidebar()
|
||||||
const socket = inject('$socket')
|
const socket = inject('$socket')
|
||||||
const unreadCount = ref(0)
|
const unreadCount = ref(0)
|
||||||
const sidebarLinks = ref(getSidebarLinks())
|
const sidebarLinks = ref(getSidebarLinks())
|
||||||
@@ -110,6 +115,7 @@ 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 showWebPages = ref(false)
|
||||||
|
const settingsStore = useSettings()
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
socket.on('publish_lms_notifications', (data) => {
|
socket.on('publish_lms_notifications', (data) => {
|
||||||
@@ -179,6 +185,37 @@ const addQuizzes = () => {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const addPrograms = () => {
|
||||||
|
let activeFor = ['Programs', 'ProgramForm']
|
||||||
|
let index = 1
|
||||||
|
let canAddProgram = false
|
||||||
|
|
||||||
|
if (
|
||||||
|
!isInstructor.value &&
|
||||||
|
!isModerator.value &&
|
||||||
|
settingsStore.learningPaths.data
|
||||||
|
) {
|
||||||
|
sidebarLinks.value = sidebarLinks.value.filter(
|
||||||
|
(link) => link.label !== 'Courses'
|
||||||
|
)
|
||||||
|
activeFor.push('CourseDetail')
|
||||||
|
activeFor.push('Lesson')
|
||||||
|
index = 0
|
||||||
|
canAddProgram = true
|
||||||
|
} else if (isInstructor.value || isModerator.value) {
|
||||||
|
canAddProgram = true
|
||||||
|
}
|
||||||
|
|
||||||
|
if (canAddProgram) {
|
||||||
|
sidebarLinks.value.splice(index, 0, {
|
||||||
|
label: 'Programs',
|
||||||
|
icon: 'Route',
|
||||||
|
to: 'Programs',
|
||||||
|
activeFor: activeFor,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const openPageModal = (link) => {
|
const openPageModal = (link) => {
|
||||||
showPageModal.value = true
|
showPageModal.value = true
|
||||||
pageToEdit.value = link
|
pageToEdit.value = link
|
||||||
@@ -211,8 +248,11 @@ watch(userResource, () => {
|
|||||||
isModerator.value = userResource.data.is_moderator
|
isModerator.value = userResource.data.is_moderator
|
||||||
isInstructor.value = userResource.data.is_instructor
|
isInstructor.value = userResource.data.is_instructor
|
||||||
addQuizzes()
|
addQuizzes()
|
||||||
|
addPrograms()
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
let isSidebarCollapsed = ref(getSidebarFromStorage())
|
const toggleSidebar = () => {
|
||||||
|
sidebarStore.isSidebarCollapsed = !sidebarStore.isSidebarCollapsed
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<template>
|
<template>
|
||||||
<div>
|
<div>
|
||||||
<div class="flex items-center justify-between">
|
<div class="flex items-center justify-between mb-4">
|
||||||
<div class="text-lg font-semibold mb-4">
|
<div class="text-lg font-semibold">
|
||||||
{{ __('Assessments') }}
|
{{ __('Assessments') }}
|
||||||
</div>
|
</div>
|
||||||
<Button v-if="canSeeAddButton()" @click="showModal = true">
|
<Button v-if="canSeeAddButton()" @click="showModal = true">
|
||||||
@@ -38,7 +38,10 @@
|
|||||||
<ListRow :row="row" v-for="row in assessments.data">
|
<ListRow :row="row" v-for="row in assessments.data">
|
||||||
<template #default="{ column, item }">
|
<template #default="{ column, item }">
|
||||||
<ListRowItem :item="row[column.key]" :align="column.align">
|
<ListRowItem :item="row[column.key]" :align="column.align">
|
||||||
<div>
|
<div v-if="column.key == 'assessment_type'">
|
||||||
|
{{ row[column.key] == 'LMS Quiz' ? 'Quiz' : 'Assignment' }}
|
||||||
|
</div>
|
||||||
|
<div v-else>
|
||||||
{{ row[column.key] }}
|
{{ row[column.key] }}
|
||||||
</div>
|
</div>
|
||||||
</ListRowItem>
|
</ListRowItem>
|
||||||
@@ -177,10 +180,12 @@ const getAssessmentColumns = () => {
|
|||||||
{
|
{
|
||||||
label: 'Assessment',
|
label: 'Assessment',
|
||||||
key: 'title',
|
key: 'title',
|
||||||
|
width: '30rem',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: 'Type',
|
label: 'Type',
|
||||||
key: 'assessment_type',
|
key: 'assessment_type',
|
||||||
|
width: '10rem',
|
||||||
},
|
},
|
||||||
]
|
]
|
||||||
|
|
||||||
@@ -189,6 +194,7 @@ const getAssessmentColumns = () => {
|
|||||||
label: 'Status/Score',
|
label: 'Status/Score',
|
||||||
key: 'status',
|
key: 'status',
|
||||||
align: 'center',
|
align: 'center',
|
||||||
|
width: '10rem',
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
return columns
|
return columns
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<template>
|
<template>
|
||||||
<div>
|
<div>
|
||||||
<div class="flex items-center justify-between mb-4">
|
<div class="flex items-center justify-between mb-4">
|
||||||
<div class="text-xl font-semibold">
|
<div class="text-lg font-semibold">
|
||||||
{{ __('Courses') }}
|
{{ __('Courses') }}
|
||||||
</div>
|
</div>
|
||||||
<Button v-if="canSeeAddButton()" @click="openCourseModal()">
|
<Button v-if="canSeeAddButton()" @click="openCourseModal()">
|
||||||
@@ -118,13 +118,13 @@ const getCoursesColumns = () => {
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: 'Lessons',
|
label: 'Lessons',
|
||||||
key: 'lesson_count',
|
key: 'lessons',
|
||||||
align: 'right',
|
align: 'right',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: 'Enrollments',
|
label: 'Enrollments',
|
||||||
align: 'right',
|
align: 'right',
|
||||||
key: 'enrollment_count',
|
key: 'enrollments',
|
||||||
},
|
},
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,12 +1,14 @@
|
|||||||
<template>
|
<template>
|
||||||
<Button class="float-right mb-3" @click="openStudentModal()">
|
<div class="flex items-center justify-between mb-4">
|
||||||
<template #prefix>
|
<div class="text-lg font-semibold">
|
||||||
<Plus class="h-4 w-4" />
|
{{ __('Students') }}
|
||||||
</template>
|
</div>
|
||||||
{{ __('Add') }}
|
<Button @click="openStudentModal()">
|
||||||
</Button>
|
<template #prefix>
|
||||||
<div class="text-lg font-semibold mb-4">
|
<Plus class="h-4 w-4" />
|
||||||
{{ __('Students') }}
|
</template>
|
||||||
|
{{ __('Add') }}
|
||||||
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
<div v-if="students.data?.length">
|
<div v-if="students.data?.length">
|
||||||
<ListView
|
<ListView
|
||||||
@@ -18,12 +20,16 @@
|
|||||||
<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-gray-100 p-2"
|
||||||
>
|
>
|
||||||
<ListHeaderItem :item="item" v-for="item in getStudentColumns()">
|
<ListHeaderItem
|
||||||
|
:item="item"
|
||||||
|
v-for="item in getStudentColumns()"
|
||||||
|
:title="item.label"
|
||||||
|
>
|
||||||
<template #prefix="{ item }">
|
<template #prefix="{ item }">
|
||||||
<component
|
<FeatherIcon
|
||||||
v-if="item.icon"
|
v-if="item.icon"
|
||||||
:is="item.icon"
|
:name="item.icon"
|
||||||
class="h-4 w-4 stroke-1.5 ml-4"
|
class="h-4 w-4 stroke-1.5"
|
||||||
/>
|
/>
|
||||||
</template>
|
</template>
|
||||||
</ListHeaderItem>
|
</ListHeaderItem>
|
||||||
@@ -42,9 +48,22 @@
|
|||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<div>
|
<div v-if="column.key == 'courses'">
|
||||||
{{ row[column.key] }}
|
{{ row[column.key] }}
|
||||||
</div>
|
</div>
|
||||||
|
<div v-else-if="column.icon == 'book-open'">
|
||||||
|
{{ Math.ceil(row.courses[column.key]) }}
|
||||||
|
</div>
|
||||||
|
<div v-else-if="column.icon == 'help-circle'">
|
||||||
|
<Badge
|
||||||
|
v-if="isAssignment(row.assessments[column.key])"
|
||||||
|
:theme="getStatusTheme(row.assessments[column.key])"
|
||||||
|
class="text-xs"
|
||||||
|
>
|
||||||
|
{{ row.assessments[column.key] }}
|
||||||
|
</Badge>
|
||||||
|
<div v-else>{{ parseInt(row.assessments[column.key]) }}</div>
|
||||||
|
</div>
|
||||||
</ListRowItem>
|
</ListRowItem>
|
||||||
</template>
|
</template>
|
||||||
</ListRow>
|
</ListRow>
|
||||||
@@ -74,7 +93,11 @@
|
|||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import {
|
import {
|
||||||
|
Avatar,
|
||||||
|
Badge,
|
||||||
|
Button,
|
||||||
createResource,
|
createResource,
|
||||||
|
FeatherIcon,
|
||||||
ListHeader,
|
ListHeader,
|
||||||
ListHeaderItem,
|
ListHeaderItem,
|
||||||
ListSelectBanner,
|
ListSelectBanner,
|
||||||
@@ -82,8 +105,6 @@ import {
|
|||||||
ListRows,
|
ListRows,
|
||||||
ListView,
|
ListView,
|
||||||
ListRowItem,
|
ListRowItem,
|
||||||
Avatar,
|
|
||||||
Button,
|
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import { Trash2, Plus } from 'lucide-vue-next'
|
import { Trash2, Plus } from 'lucide-vue-next'
|
||||||
import { ref } from 'vue'
|
import { ref } from 'vue'
|
||||||
@@ -109,27 +130,41 @@ const students = createResource({
|
|||||||
})
|
})
|
||||||
|
|
||||||
const getStudentColumns = () => {
|
const getStudentColumns = () => {
|
||||||
return [
|
let columns = [
|
||||||
{
|
{
|
||||||
label: 'Full Name',
|
label: 'Full Name',
|
||||||
key: 'full_name',
|
key: 'full_name',
|
||||||
width: 2,
|
width: '15rem',
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Courses Done',
|
|
||||||
key: 'courses_completed',
|
|
||||||
align: 'center',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Assessments Done',
|
|
||||||
key: 'assessments_completed',
|
|
||||||
align: 'center',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Last Active',
|
|
||||||
key: 'last_active',
|
|
||||||
},
|
},
|
||||||
]
|
]
|
||||||
|
|
||||||
|
if (students.data?.[0].assessments) {
|
||||||
|
Object.keys(students.data?.[0].assessments).forEach((assessment) => {
|
||||||
|
columns.push({
|
||||||
|
label: assessment,
|
||||||
|
key: assessment,
|
||||||
|
width: '10rem',
|
||||||
|
icon: 'help-circle',
|
||||||
|
align: isAssignment(students.data?.[0].assessments[assessment])
|
||||||
|
? 'left'
|
||||||
|
: 'center',
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
if (students.data?.[0].courses) {
|
||||||
|
Object.keys(students.data?.[0].courses).forEach((course) => {
|
||||||
|
columns.push({
|
||||||
|
label: course,
|
||||||
|
key: course,
|
||||||
|
width: '10rem',
|
||||||
|
icon: 'book-open',
|
||||||
|
align: 'center',
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return columns
|
||||||
}
|
}
|
||||||
|
|
||||||
const openStudentModal = () => {
|
const openStudentModal = () => {
|
||||||
@@ -160,4 +195,18 @@ const removeStudents = (selections, unselectAll) => {
|
|||||||
}
|
}
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const getStatusTheme = (status) => {
|
||||||
|
if (status === 'Pass') {
|
||||||
|
return 'green'
|
||||||
|
} else if (status == 'Not Graded') {
|
||||||
|
return 'orange'
|
||||||
|
} else {
|
||||||
|
return 'red'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const isAssignment = (value) => {
|
||||||
|
return isNaN(value)
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -92,7 +92,7 @@
|
|||||||
{{ option.label }}
|
{{ option.label }}
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
v-if="option.label != option.description"
|
v-if="option.description"
|
||||||
class="text-xs text-gray-700"
|
class="text-xs text-gray-700"
|
||||||
v-html="option.description"
|
v-html="option.description"
|
||||||
></div>
|
></div>
|
||||||
|
|||||||
@@ -44,6 +44,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
</Autocomplete>
|
</Autocomplete>
|
||||||
|
<p v-if="description" class="text-sm text-gray-600">{{ description }}</p>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -67,6 +68,10 @@ const props = defineProps({
|
|||||||
type: String,
|
type: String,
|
||||||
default: '',
|
default: '',
|
||||||
},
|
},
|
||||||
|
description: {
|
||||||
|
type: String,
|
||||||
|
default: '',
|
||||||
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
const emit = defineEmits(['update:modelValue', 'change'])
|
const emit = defineEmits(['update:modelValue', 'change'])
|
||||||
@@ -118,7 +123,7 @@ const options = createResource({
|
|||||||
transform: (data) => {
|
transform: (data) => {
|
||||||
return data.map((option) => {
|
return data.map((option) => {
|
||||||
return {
|
return {
|
||||||
label: option.value,
|
label: option.label || option.value,
|
||||||
value: option.value,
|
value: option.value,
|
||||||
description: option.description,
|
description: option.description,
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -59,7 +59,7 @@
|
|||||||
|
|
||||||
<div v-if="course.status != 'Approved'">
|
<div v-if="course.status != 'Approved'">
|
||||||
<Badge
|
<Badge
|
||||||
variant="solid"
|
variant="subtle"
|
||||||
:theme="course.status === 'Under Review' ? 'orange' : 'blue'"
|
:theme="course.status === 'Under Review' ? 'orange' : 'blue'"
|
||||||
size="sm"
|
size="sm"
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -140,7 +140,7 @@ function enrollStudent() {
|
|||||||
showToast(
|
showToast(
|
||||||
__('Please Login'),
|
__('Please Login'),
|
||||||
__('You need to login first to enroll for this course'),
|
__('You need to login first to enroll for this course'),
|
||||||
'circle-warn'
|
'alert-circle'
|
||||||
)
|
)
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
window.location.href = `/login?redirect-to=${window.location.pathname}`
|
window.location.href = `/login?redirect-to=${window.location.pathname}`
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
<template>
|
<template>
|
||||||
<span v-if="instructors.length == 1">
|
<span v-if="instructors?.length == 1">
|
||||||
<router-link
|
<router-link
|
||||||
:to="{
|
:to="{
|
||||||
name: 'Profile',
|
name: 'Profile',
|
||||||
@@ -9,7 +9,7 @@
|
|||||||
{{ instructors[0].full_name }}
|
{{ instructors[0].full_name }}
|
||||||
</router-link>
|
</router-link>
|
||||||
</span>
|
</span>
|
||||||
<span v-if="instructors.length == 2">
|
<span v-if="instructors?.length == 2">
|
||||||
<router-link
|
<router-link
|
||||||
:to="{
|
:to="{
|
||||||
name: 'Profile',
|
name: 'Profile',
|
||||||
@@ -28,7 +28,7 @@
|
|||||||
{{ instructors[1].first_name }}
|
{{ instructors[1].first_name }}
|
||||||
</router-link>
|
</router-link>
|
||||||
</span>
|
</span>
|
||||||
<span v-if="instructors.length > 2">
|
<span v-if="instructors?.length > 2">
|
||||||
<router-link
|
<router-link
|
||||||
:to="{
|
:to="{
|
||||||
name: 'Profile',
|
name: 'Profile',
|
||||||
@@ -37,7 +37,7 @@
|
|||||||
>
|
>
|
||||||
{{ instructors[0].first_name }}
|
{{ instructors[0].first_name }}
|
||||||
</router-link>
|
</router-link>
|
||||||
and {{ instructors.length - 1 }} others
|
and {{ instructors?.length - 1 }} others
|
||||||
</span>
|
</span>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
|
|||||||
@@ -16,7 +16,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
:class="{
|
:class="{
|
||||||
'shadow rounded-md pt-2 px-2': showOutline && outline.data?.length,
|
'shadow rounded-md py-2 px-2': showOutline && outline.data?.length,
|
||||||
}"
|
}"
|
||||||
>
|
>
|
||||||
<Disclosure
|
<Disclosure
|
||||||
@@ -25,21 +25,42 @@
|
|||||||
:key="chapter.name"
|
:key="chapter.name"
|
||||||
:defaultOpen="openChapterDetail(chapter.idx)"
|
:defaultOpen="openChapterDetail(chapter.idx)"
|
||||||
>
|
>
|
||||||
<DisclosureButton ref="" class="flex w-full p-2">
|
<DisclosureButton ref="" class="flex items-center w-full p-2 group">
|
||||||
<ChevronRight
|
<ChevronRight
|
||||||
:class="{
|
:class="{
|
||||||
'rotate-90 transform duration-200': open,
|
'rotate-90 transform duration-200': open,
|
||||||
'duration-200': !open,
|
'duration-200': !open,
|
||||||
|
hidden: chapter.is_scorm_package,
|
||||||
open: index == 1,
|
open: index == 1,
|
||||||
}"
|
}"
|
||||||
class="h-4 w-4 text-gray-900 stroke-1 mr-2"
|
class="h-4 w-4 text-gray-900 stroke-1"
|
||||||
/>
|
/>
|
||||||
<div class="text-base text-left font-medium leading-5">
|
<div
|
||||||
|
class="text-base text-left font-medium leading-5 ml-2"
|
||||||
|
@click="redirectToChapter(chapter)"
|
||||||
|
>
|
||||||
{{ chapter.title }}
|
{{ chapter.title }}
|
||||||
</div>
|
</div>
|
||||||
|
<div class="flex ml-auto space-x-4">
|
||||||
|
<Tooltip :text="__('Edit Chapter')" placement="bottom">
|
||||||
|
<FilePenLine
|
||||||
|
v-if="allowEdit"
|
||||||
|
@click.prevent="openChapterModal(chapter)"
|
||||||
|
class="h-4 w-4 text-gray-900 invisible group-hover:visible"
|
||||||
|
/>
|
||||||
|
</Tooltip>
|
||||||
|
<Tooltip :text="__('Delete Chapter')" placement="bottom">
|
||||||
|
<Trash2
|
||||||
|
v-if="allowEdit"
|
||||||
|
@click.prevent="trashChapter(chapter.name)"
|
||||||
|
class="h-4 w-4 text-red-500 invisible group-hover:visible"
|
||||||
|
/>
|
||||||
|
</Tooltip>
|
||||||
|
</div>
|
||||||
</DisclosureButton>
|
</DisclosureButton>
|
||||||
<DisclosurePanel>
|
<DisclosurePanel v-if="!chapter.is_scorm_package">
|
||||||
<Draggable
|
<Draggable
|
||||||
|
v-if="!chapter.is_scorm_package"
|
||||||
:list="chapter.lessons"
|
:list="chapter.lessons"
|
||||||
:disabled="!allowEdit"
|
:disabled="!allowEdit"
|
||||||
item-key="name"
|
item-key="name"
|
||||||
@@ -89,6 +110,7 @@
|
|||||||
</Draggable>
|
</Draggable>
|
||||||
<div v-if="allowEdit" class="flex mt-2 mb-4 pl-8">
|
<div v-if="allowEdit" class="flex mt-2 mb-4 pl-8">
|
||||||
<router-link
|
<router-link
|
||||||
|
v-if="!chapter.is_scorm_package"
|
||||||
:to="{
|
:to="{
|
||||||
name: 'LessonForm',
|
name: 'LessonForm',
|
||||||
params: {
|
params: {
|
||||||
@@ -102,9 +124,6 @@
|
|||||||
{{ __('Add Lesson') }}
|
{{ __('Add Lesson') }}
|
||||||
</Button>
|
</Button>
|
||||||
</router-link>
|
</router-link>
|
||||||
<Button class="ml-2" @click="openChapterModal(chapter)">
|
|
||||||
{{ __('Edit Chapter') }}
|
|
||||||
</Button>
|
|
||||||
</div>
|
</div>
|
||||||
</DisclosurePanel>
|
</DisclosurePanel>
|
||||||
</Disclosure>
|
</Disclosure>
|
||||||
@@ -118,24 +137,26 @@
|
|||||||
/>
|
/>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { Button, createResource } from 'frappe-ui'
|
import { Button, createResource, Tooltip } from 'frappe-ui'
|
||||||
import { ref, getCurrentInstance } from 'vue'
|
import { getCurrentInstance, inject, ref } from 'vue'
|
||||||
import Draggable from 'vuedraggable'
|
import Draggable from 'vuedraggable'
|
||||||
import { Disclosure, DisclosureButton, DisclosurePanel } from '@headlessui/vue'
|
import { Disclosure, DisclosureButton, DisclosurePanel } from '@headlessui/vue'
|
||||||
import {
|
import {
|
||||||
ChevronRight,
|
|
||||||
MonitorPlay,
|
|
||||||
HelpCircle,
|
|
||||||
FileText,
|
|
||||||
Check,
|
Check,
|
||||||
|
ChevronRight,
|
||||||
|
FileText,
|
||||||
|
FilePenLine,
|
||||||
|
HelpCircle,
|
||||||
|
MonitorPlay,
|
||||||
Trash2,
|
Trash2,
|
||||||
} from 'lucide-vue-next'
|
} from 'lucide-vue-next'
|
||||||
import { useRoute } from 'vue-router'
|
import { useRoute, useRouter } from 'vue-router'
|
||||||
import ChapterModal from '@/components/Modals/ChapterModal.vue'
|
import ChapterModal from '@/components/Modals/ChapterModal.vue'
|
||||||
import { showToast } from '@/utils'
|
import { showToast } from '@/utils'
|
||||||
|
|
||||||
const route = useRoute()
|
const route = useRoute()
|
||||||
const expandAll = ref(true)
|
const router = useRouter()
|
||||||
|
const user = inject('$user')
|
||||||
const showChapterModal = ref(false)
|
const showChapterModal = ref(false)
|
||||||
const currentChapter = ref(null)
|
const currentChapter = ref(null)
|
||||||
const app = getCurrentInstance()
|
const app = getCurrentInstance()
|
||||||
@@ -205,8 +226,10 @@ const updateLessonIndex = createResource({
|
|||||||
|
|
||||||
const trashLesson = (lessonName, chapterName) => {
|
const trashLesson = (lessonName, chapterName) => {
|
||||||
$dialog({
|
$dialog({
|
||||||
title: __('Delete Lesson'),
|
title: __('Delete this lesson?'),
|
||||||
message: __('Are you sure you want to delete this lesson?'),
|
message: __(
|
||||||
|
'Deleting this lesson will permanently remove it from the course. This action cannot be undone. Are you sure you want to continue?'
|
||||||
|
),
|
||||||
actions: [
|
actions: [
|
||||||
{
|
{
|
||||||
label: __('Delete'),
|
label: __('Delete'),
|
||||||
@@ -245,6 +268,61 @@ const updateOutline = (e) => {
|
|||||||
idx: e.newIndex,
|
idx: e.newIndex,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const deleteChapter = createResource({
|
||||||
|
url: 'lms.lms.api.delete_chapter',
|
||||||
|
makeParams(values) {
|
||||||
|
return {
|
||||||
|
chapter: values.chapter,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
onSuccess() {
|
||||||
|
outline.reload()
|
||||||
|
showToast('Success', 'Chapter deleted successfully', 'check')
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const trashChapter = (chapterName) => {
|
||||||
|
$dialog({
|
||||||
|
title: __('Delete this chapter?'),
|
||||||
|
message: __(
|
||||||
|
'Deleting this chapter will also delete all its lessons and permanently remove it from the course. This action cannot be undone. Are you sure you want to continue?'
|
||||||
|
),
|
||||||
|
actions: [
|
||||||
|
{
|
||||||
|
label: __('Delete'),
|
||||||
|
theme: 'red',
|
||||||
|
variant: 'solid',
|
||||||
|
onClick(close) {
|
||||||
|
deleteChapter.submit({ chapter: chapterName })
|
||||||
|
close()
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const redirectToChapter = (chapter) => {
|
||||||
|
if (!chapter.is_scorm_package) return
|
||||||
|
event.preventDefault()
|
||||||
|
if (props.allowEdit) return
|
||||||
|
if (!user.data) {
|
||||||
|
showToast(
|
||||||
|
__('You are not enrolled'),
|
||||||
|
__('Please enroll for this course to view this lesson'),
|
||||||
|
'alert-circle'
|
||||||
|
)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
router.push({
|
||||||
|
name: 'SCORMChapter',
|
||||||
|
params: {
|
||||||
|
courseName: props.courseName,
|
||||||
|
chapterName: chapter.name,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
<style>
|
<style>
|
||||||
.outline-lesson:has(.router-link-active) {
|
.outline-lesson:has(.router-link-active) {
|
||||||
|
|||||||
@@ -9,7 +9,7 @@
|
|||||||
allowfullscreen
|
allowfullscreen
|
||||||
></iframe>
|
></iframe>
|
||||||
</div>
|
</div>
|
||||||
<div v-for="block in content.split('\n\n')">
|
<div v-for="block in content?.split('\n\n')">
|
||||||
<div v-if="block.includes('{{ YouTubeVideo')">
|
<div v-if="block.includes('{{ YouTubeVideo')">
|
||||||
<iframe
|
<iframe
|
||||||
class="youtube-video"
|
class="youtube-video"
|
||||||
|
|||||||
@@ -25,7 +25,7 @@
|
|||||||
@click="openHelpDialog('upload')"
|
@click="openHelpDialog('upload')"
|
||||||
>
|
>
|
||||||
<span class="leading-5">
|
<span class="leading-5">
|
||||||
{{ __('How to upload content from your system?') }}
|
{{ __(contentMap['upload']) }}
|
||||||
</span>
|
</span>
|
||||||
<Info class="w-3 h-3 text-gray-700" />
|
<Info class="w-3 h-3 text-gray-700" />
|
||||||
</div>
|
</div>
|
||||||
@@ -44,7 +44,7 @@
|
|||||||
@click="openHelpDialog('youtube')"
|
@click="openHelpDialog('youtube')"
|
||||||
>
|
>
|
||||||
<span>
|
<span>
|
||||||
{{ __('How to add a YouTube Video?') }}
|
{{ __(contentMap['youtube']) }}
|
||||||
</span>
|
</span>
|
||||||
<Info class="w-3 h-3 text-gray-700" />
|
<Info class="w-3 h-3 text-gray-700" />
|
||||||
</div>
|
</div>
|
||||||
@@ -72,7 +72,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<ExplanationVideos v-model="showExplanation" :type="type" />
|
<ExplanationVideos v-model="showExplanation" :title="title" :type="type" />
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { Info } from 'lucide-vue-next'
|
import { Info } from 'lucide-vue-next'
|
||||||
@@ -81,9 +81,16 @@ import ExplanationVideos from '@/components/Modals/ExplanationVideos.vue'
|
|||||||
|
|
||||||
const showExplanation = ref(false)
|
const showExplanation = ref(false)
|
||||||
const type = ref(null)
|
const type = ref(null)
|
||||||
|
const title = ref(null)
|
||||||
|
const contentMap = {
|
||||||
|
quiz: 'How to add a Quiz?',
|
||||||
|
upload: 'How to upload content from your system?',
|
||||||
|
youtube: 'How to add a YouTube Video?',
|
||||||
|
}
|
||||||
|
|
||||||
const openHelpDialog = (contentType) => {
|
const openHelpDialog = (contentType) => {
|
||||||
type.value = contentType
|
type.value = contentType
|
||||||
|
title.value = contentMap[contentType]
|
||||||
showExplanation.value = true
|
showExplanation.value = true
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -66,8 +66,19 @@
|
|||||||
<div class="text-gray-900">
|
<div class="text-gray-900">
|
||||||
{{ member.full_name }}
|
{{ member.full_name }}
|
||||||
</div>
|
</div>
|
||||||
<div v-if="getRole(member)">
|
<div
|
||||||
{{ getRole(member) }}
|
class="px-1"
|
||||||
|
v-if="member.role && getRole(member.role) !== 'Student'"
|
||||||
|
>
|
||||||
|
<Badge
|
||||||
|
:variant="'subtle'"
|
||||||
|
:ref_for="true"
|
||||||
|
theme="blue"
|
||||||
|
size="sm"
|
||||||
|
label="Badge"
|
||||||
|
>
|
||||||
|
{{ getRole(member.role) }}
|
||||||
|
</Badge>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="text-sm text-gray-700">
|
<div class="text-sm text-gray-700">
|
||||||
@@ -99,7 +110,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { createResource, Avatar, Button, FormControl } from 'frappe-ui'
|
import { createResource, Avatar, Button, FormControl, Badge } from 'frappe-ui'
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
import { ref, watch, reactive, inject } from 'vue'
|
import { ref, watch, reactive, inject } from 'vue'
|
||||||
import { RefreshCw, Plus, X } from 'lucide-vue-next'
|
import { RefreshCw, Plus, X } from 'lucide-vue-next'
|
||||||
|
|||||||
132
frontend/src/components/Modals/BulkCertificates.vue
Normal file
132
frontend/src/components/Modals/BulkCertificates.vue
Normal file
@@ -0,0 +1,132 @@
|
|||||||
|
<template>
|
||||||
|
<Dialog
|
||||||
|
v-model="show"
|
||||||
|
:options="{
|
||||||
|
title: __('Generate Certificates'),
|
||||||
|
size: 'lg',
|
||||||
|
actions: [
|
||||||
|
{
|
||||||
|
label: 'Create',
|
||||||
|
variant: 'solid',
|
||||||
|
onClick: ({ close }) => {
|
||||||
|
generateCertificates(close)
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
}"
|
||||||
|
>
|
||||||
|
<template #body-content>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<FormControl
|
||||||
|
type="select"
|
||||||
|
v-model="details.course"
|
||||||
|
:label="__('Course')"
|
||||||
|
:options="getCourses()"
|
||||||
|
/>
|
||||||
|
<Link
|
||||||
|
v-model="details.evaluator"
|
||||||
|
:label="__('Evaluator')"
|
||||||
|
doctype="Course Evaluator"
|
||||||
|
/>
|
||||||
|
<FormControl
|
||||||
|
type="date"
|
||||||
|
v-model="details.issue_date"
|
||||||
|
:label="__('Issue Date')"
|
||||||
|
/>
|
||||||
|
<FormControl
|
||||||
|
type="date"
|
||||||
|
v-model="details.expiry_date"
|
||||||
|
:label="__('Expiry Date')"
|
||||||
|
/>
|
||||||
|
<Link
|
||||||
|
v-model="details.template"
|
||||||
|
:label="__('Template')"
|
||||||
|
doctype="Print Format"
|
||||||
|
:filters="{
|
||||||
|
doc_type: 'LMS Certificate',
|
||||||
|
}"
|
||||||
|
/>
|
||||||
|
<Switch
|
||||||
|
size="sm"
|
||||||
|
:label="__('Published')"
|
||||||
|
:description="
|
||||||
|
__(
|
||||||
|
'Enabling this will publish the certificate on the certified participants page.'
|
||||||
|
)
|
||||||
|
"
|
||||||
|
v-model="details.published"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</Dialog>
|
||||||
|
</template>
|
||||||
|
<script setup>
|
||||||
|
import { inject, reactive } from 'vue'
|
||||||
|
import { createResource, Dialog, FormControl, Switch } from 'frappe-ui'
|
||||||
|
import Link from '@/components/Controls/Link.vue'
|
||||||
|
import { showToast } from '@/utils'
|
||||||
|
|
||||||
|
const show = defineModel()
|
||||||
|
const dayjs = inject('$dayjs')
|
||||||
|
const details = reactive({
|
||||||
|
issue_date: dayjs().format('YYYY-MM-DD'),
|
||||||
|
expiry_date: null,
|
||||||
|
template: null,
|
||||||
|
evaluator: null,
|
||||||
|
published: true,
|
||||||
|
})
|
||||||
|
|
||||||
|
const props = defineProps({
|
||||||
|
batch: {
|
||||||
|
type: [Object, null],
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const createCertificate = createResource({
|
||||||
|
url: 'frappe.client.insert',
|
||||||
|
makeParams(values) {
|
||||||
|
return {
|
||||||
|
doc: {
|
||||||
|
doctype: 'LMS Certificate',
|
||||||
|
issue_date: details.issue_date,
|
||||||
|
expiry_date: details.expiry_date,
|
||||||
|
template: details.template,
|
||||||
|
published: details.published,
|
||||||
|
course: values.course,
|
||||||
|
batch: values.batch,
|
||||||
|
member: values.member,
|
||||||
|
evaluator: details.evaluator,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const generateCertificates = (close) => {
|
||||||
|
props.batch?.students.forEach((student) => {
|
||||||
|
createCertificate.submit(
|
||||||
|
{
|
||||||
|
course: details.course,
|
||||||
|
batch: props.batch.name,
|
||||||
|
member: student,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
onError(err) {
|
||||||
|
showToast(__('Error'), err.messages?.[0] || err, 'x')
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
})
|
||||||
|
close()
|
||||||
|
showToast(__('Success'), __('Certificates generated successfully'), 'check')
|
||||||
|
}
|
||||||
|
|
||||||
|
const getCourses = () => {
|
||||||
|
return props.batch?.courses.map((course) => {
|
||||||
|
return {
|
||||||
|
label: course.course,
|
||||||
|
value: course.course,
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
</script>
|
||||||
@@ -15,25 +15,77 @@
|
|||||||
}"
|
}"
|
||||||
>
|
>
|
||||||
<template #body-content>
|
<template #body-content>
|
||||||
<FormControl
|
<div class="space-y-4 text-base">
|
||||||
ref="chapterInput"
|
<FormControl label="Title" v-model="chapter.title" :required="true" />
|
||||||
label="Title"
|
<Switch
|
||||||
v-model="chapter.title"
|
size="sm"
|
||||||
class="mb-4"
|
:label="__('SCORM Package')"
|
||||||
:required="true"
|
:description="
|
||||||
/>
|
__(
|
||||||
|
'Enable this only if you want to upload a SCORM package as a chapter.'
|
||||||
|
)
|
||||||
|
"
|
||||||
|
v-model="chapter.is_scorm_package"
|
||||||
|
/>
|
||||||
|
<div v-if="chapter.is_scorm_package">
|
||||||
|
<FileUploader
|
||||||
|
v-if="!chapter.scorm_package"
|
||||||
|
:fileTypes="['.zip']"
|
||||||
|
:validateFile="validateFile"
|
||||||
|
@success="(file) => (chapter.scorm_package = file)"
|
||||||
|
>
|
||||||
|
<template v-slot="{ file, progress, uploading, openFileSelector }">
|
||||||
|
<div class="mb-4">
|
||||||
|
<Button @click="openFileSelector" :loading="uploading">
|
||||||
|
{{
|
||||||
|
uploading ? `Uploading ${progress}%` : 'Upload an zip file'
|
||||||
|
}}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</FileUploader>
|
||||||
|
<div v-else class="">
|
||||||
|
<div class="flex items-center">
|
||||||
|
<div class="border rounded-md p-2 mr-2">
|
||||||
|
<FileText class="h-5 w-5 stroke-1.5 text-gray-700" />
|
||||||
|
</div>
|
||||||
|
<div class="flex flex-col">
|
||||||
|
<span>
|
||||||
|
{{ chapter.scorm_package.file_name }}
|
||||||
|
</span>
|
||||||
|
<span class="text-sm text-gray-500 mt-1">
|
||||||
|
{{ getFileSize(chapter.scorm_package.file_size) }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<X
|
||||||
|
@click="() => (chapter.scorm_package = null)"
|
||||||
|
class="bg-gray-200 rounded-md cursor-pointer stroke-1.5 w-5 h-5 p-1 ml-4"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</template>
|
</template>
|
||||||
</Dialog>
|
</Dialog>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { Dialog, FormControl, createResource } from 'frappe-ui'
|
import {
|
||||||
import { defineModel, reactive, watch, ref } from 'vue'
|
Button,
|
||||||
import { createToast } from '@/utils/'
|
createResource,
|
||||||
|
Dialog,
|
||||||
|
FileUploader,
|
||||||
|
FormControl,
|
||||||
|
Switch,
|
||||||
|
} from 'frappe-ui'
|
||||||
|
import { defineModel, reactive, watch } from 'vue'
|
||||||
|
import { showToast, getFileSize } from '@/utils/'
|
||||||
import { capture } from '@/telemetry'
|
import { capture } from '@/telemetry'
|
||||||
|
import { FileText, X } from 'lucide-vue-next'
|
||||||
|
import { useSettings } from '@/stores/settings'
|
||||||
|
|
||||||
const show = defineModel()
|
const show = defineModel()
|
||||||
const outline = defineModel('outline')
|
const outline = defineModel('outline')
|
||||||
const chapterInput = ref(null)
|
const settingsStore = useSettings()
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
course: {
|
course: {
|
||||||
@@ -47,30 +99,19 @@ const props = defineProps({
|
|||||||
|
|
||||||
const chapter = reactive({
|
const chapter = reactive({
|
||||||
title: '',
|
title: '',
|
||||||
|
is_scorm_package: 0,
|
||||||
|
scorm_package: null,
|
||||||
})
|
})
|
||||||
|
|
||||||
const chapterResource = createResource({
|
const chapterResource = createResource({
|
||||||
url: 'frappe.client.insert',
|
url: 'lms.lms.api.upsert_chapter',
|
||||||
makeParams(values) {
|
makeParams(values) {
|
||||||
return {
|
return {
|
||||||
doc: {
|
title: chapter.title,
|
||||||
doctype: 'Course Chapter',
|
course: props.course,
|
||||||
title: chapter.title,
|
is_scorm_package: chapter.is_scorm_package,
|
||||||
description: chapter.description,
|
scorm_package: chapter.scorm_package,
|
||||||
course: props.course,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
},
|
|
||||||
})
|
|
||||||
|
|
||||||
const chapterEditResource = createResource({
|
|
||||||
url: 'frappe.client.set_value',
|
|
||||||
makeParams(values) {
|
|
||||||
return {
|
|
||||||
doctype: 'Course Chapter',
|
|
||||||
name: props.chapterDetail?.name,
|
name: props.chapterDetail?.name,
|
||||||
fieldname: 'title',
|
|
||||||
value: chapter.title,
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
@@ -90,14 +131,12 @@ const chapterReference = createResource({
|
|||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
const addChapter = (close) => {
|
const addChapter = async (close) => {
|
||||||
chapterResource.submit(
|
chapterResource.submit(
|
||||||
{},
|
{},
|
||||||
{
|
{
|
||||||
validate() {
|
validate() {
|
||||||
if (!chapter.title) {
|
return validateChapter()
|
||||||
return 'Title is required'
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
onSuccess: (data) => {
|
onSuccess: (data) => {
|
||||||
capture('chapter_created')
|
capture('chapter_created')
|
||||||
@@ -105,30 +144,48 @@ const addChapter = (close) => {
|
|||||||
{ name: data.name },
|
{ name: data.name },
|
||||||
{
|
{
|
||||||
onSuccess(data) {
|
onSuccess(data) {
|
||||||
chapter.title = ''
|
cleanChapter()
|
||||||
|
if (!settingsStore.onboardingDetails.data?.is_onboarded) {
|
||||||
|
settingsStore.onboardingDetails.reload()
|
||||||
|
}
|
||||||
outline.value.reload()
|
outline.value.reload()
|
||||||
createToast({
|
showToast(
|
||||||
text: 'Chapter added successfully',
|
__('Success'),
|
||||||
icon: 'check',
|
__('Chapter added successfully'),
|
||||||
iconClasses: 'bg-green-600 text-white rounded-md p-px',
|
'check'
|
||||||
})
|
)
|
||||||
},
|
},
|
||||||
onError(err) {
|
onError(err) {
|
||||||
showError(err)
|
showToast(__('Error'), err.messages?.[0] || err, 'x')
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
close()
|
close()
|
||||||
},
|
},
|
||||||
onError(err) {
|
onError(err) {
|
||||||
showError(err)
|
showToast(__('Error'), err.messages?.[0] || err, 'x')
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const validateChapter = () => {
|
||||||
|
if (!chapter.title) {
|
||||||
|
return __('Title is required')
|
||||||
|
}
|
||||||
|
if (chapter.is_scorm_package && !chapter.scorm_package) {
|
||||||
|
return __('Please upload a SCORM package')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const cleanChapter = () => {
|
||||||
|
chapter.title = ''
|
||||||
|
chapter.is_scorm_package = 0
|
||||||
|
chapter.scorm_package = null
|
||||||
|
}
|
||||||
|
|
||||||
const editChapter = (close) => {
|
const editChapter = (close) => {
|
||||||
chapterEditResource.submit(
|
chapterResource.submit(
|
||||||
{},
|
{},
|
||||||
{
|
{
|
||||||
validate() {
|
validate() {
|
||||||
@@ -138,43 +195,29 @@ const editChapter = (close) => {
|
|||||||
},
|
},
|
||||||
onSuccess() {
|
onSuccess() {
|
||||||
outline.value.reload()
|
outline.value.reload()
|
||||||
createToast({
|
showToast(__('Success'), __('Chapter updated successfully'), 'check')
|
||||||
text: 'Chapter updated successfully',
|
|
||||||
icon: 'check',
|
|
||||||
iconClasses: 'bg-green-600 text-white rounded-md p-px',
|
|
||||||
})
|
|
||||||
close()
|
close()
|
||||||
},
|
},
|
||||||
onError(err) {
|
onError(err) {
|
||||||
showError(err)
|
showToast(__('Error'), err.messages?.[0] || err, 'x')
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
const showError = (err) => {
|
|
||||||
createToast({
|
|
||||||
title: 'Error',
|
|
||||||
text: err.messages?.[0] || err,
|
|
||||||
icon: 'x',
|
|
||||||
iconClasses: 'bg-red-600 text-white rounded-md p-px',
|
|
||||||
position: 'top-center',
|
|
||||||
timeout: 10,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
watch(
|
watch(
|
||||||
() => props.chapterDetail,
|
() => props.chapterDetail,
|
||||||
(newChapter) => {
|
(newChapter) => {
|
||||||
chapter.title = newChapter?.title
|
chapter.title = newChapter?.title
|
||||||
|
chapter.is_scorm_package = newChapter?.is_scorm_package
|
||||||
|
chapter.scorm_package = newChapter?.scorm_package
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
watch(show, () => {
|
const validateFile = (file) => {
|
||||||
if (show.value) {
|
let extension = file.name.split('.').pop().toLowerCase()
|
||||||
setTimeout(() => {
|
if (extension !== 'zip') {
|
||||||
chapterInput.value.$el.querySelector('input').focus()
|
return __('Only zip files are allowed')
|
||||||
}, 100)
|
|
||||||
}
|
}
|
||||||
})
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -3,10 +3,11 @@
|
|||||||
v-model="show"
|
v-model="show"
|
||||||
:options="{
|
:options="{
|
||||||
size: '4xl',
|
size: '4xl',
|
||||||
|
title: title,
|
||||||
}"
|
}"
|
||||||
>
|
>
|
||||||
<template #body>
|
<template #body-content>
|
||||||
<div class="p-4">
|
<div>
|
||||||
<VideoBlock :file="file" />
|
<VideoBlock :file="file" />
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
@@ -24,6 +25,10 @@ const props = defineProps({
|
|||||||
type: [String, null],
|
type: [String, null],
|
||||||
required: true,
|
required: true,
|
||||||
},
|
},
|
||||||
|
title: {
|
||||||
|
type: [String, null],
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
const file = computed(() => {
|
const file = computed(() => {
|
||||||
|
|||||||
@@ -161,25 +161,34 @@ const submitLiveClass = (close) => {
|
|||||||
return createLiveClass.submit(liveClass, {
|
return createLiveClass.submit(liveClass, {
|
||||||
validate() {
|
validate() {
|
||||||
if (!liveClass.title) {
|
if (!liveClass.title) {
|
||||||
return 'Please enter a title.'
|
return __('Please enter a title.')
|
||||||
}
|
}
|
||||||
if (!liveClass.date) {
|
if (!liveClass.date) {
|
||||||
return 'Please select a date.'
|
return __('Please select a date.')
|
||||||
}
|
|
||||||
if (dayjs(liveClass.date).isSameOrBefore(dayjs(), 'day')) {
|
|
||||||
return 'Please select a future date.'
|
|
||||||
}
|
}
|
||||||
if (!liveClass.time) {
|
if (!liveClass.time) {
|
||||||
return 'Please select a time.'
|
return __('Please select a time.')
|
||||||
}
|
|
||||||
if (!valideTime()) {
|
|
||||||
return 'Please enter a valid time in the format HH:mm.'
|
|
||||||
}
|
|
||||||
if (!liveClass.duration) {
|
|
||||||
return 'Please select a duration.'
|
|
||||||
}
|
}
|
||||||
if (!liveClass.timezone) {
|
if (!liveClass.timezone) {
|
||||||
return 'Please select a timezone.'
|
return __('Please select a timezone.')
|
||||||
|
}
|
||||||
|
if (!valideTime()) {
|
||||||
|
return __('Please enter a valid time in the format HH:mm.')
|
||||||
|
}
|
||||||
|
const liveClassDateTime = dayjs(`${liveClass.date}T${liveClass.time}`).tz(
|
||||||
|
liveClass.timezone,
|
||||||
|
true
|
||||||
|
)
|
||||||
|
if (
|
||||||
|
liveClassDateTime.isSameOrBefore(
|
||||||
|
dayjs().tz(liveClass.timezone, false),
|
||||||
|
'minute'
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
return __('Please select a future date and time.')
|
||||||
|
}
|
||||||
|
if (!liveClass.duration) {
|
||||||
|
return __('Please select a duration.')
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
onSuccess() {
|
onSuccess() {
|
||||||
|
|||||||
@@ -12,9 +12,9 @@
|
|||||||
id="existing"
|
id="existing"
|
||||||
value="existing"
|
value="existing"
|
||||||
v-model="questionType"
|
v-model="questionType"
|
||||||
class="w-3 h-3 accent-gray-900"
|
class="w-3 h-3 cursor-pointer"
|
||||||
/>
|
/>
|
||||||
<label for="existing">
|
<label for="existing" class="cursor-pointer">
|
||||||
{{ __('Add an existing question') }}
|
{{ __('Add an existing question') }}
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
@@ -25,9 +25,9 @@
|
|||||||
id="new"
|
id="new"
|
||||||
value="new"
|
value="new"
|
||||||
v-model="questionType"
|
v-model="questionType"
|
||||||
class="w-3 h-3"
|
class="w-3 h-3 cursor-pointer"
|
||||||
/>
|
/>
|
||||||
<label for="new">
|
<label for="new" class="cursor-pointer">
|
||||||
{{ __('Create a new question') }}
|
{{ __('Create a new question') }}
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
@@ -56,12 +56,14 @@
|
|||||||
type="select"
|
type="select"
|
||||||
:options="['Choices', 'User Input', 'Open Ended']"
|
:options="['Choices', 'User Input', 'Open Ended']"
|
||||||
class="pb-2"
|
class="pb-2"
|
||||||
|
:required="true"
|
||||||
/>
|
/>
|
||||||
<div v-if="question.type == 'Choices'" class="divide-y border-t">
|
<div v-if="question.type == 'Choices'" class="divide-y border-t">
|
||||||
<div v-for="n in 4" class="space-y-4 py-2">
|
<div v-for="n in 4" class="space-y-4 py-2">
|
||||||
<FormControl
|
<FormControl
|
||||||
:label="__('Option') + ' ' + n"
|
:label="__('Option') + ' ' + n"
|
||||||
v-model="question[`option_${n}`]"
|
v-model="question[`option_${n}`]"
|
||||||
|
:required="n <= 2 ? true : false"
|
||||||
/>
|
/>
|
||||||
<FormControl
|
<FormControl
|
||||||
:label="__('Explanation')"
|
:label="__('Explanation')"
|
||||||
@@ -82,6 +84,7 @@
|
|||||||
<FormControl
|
<FormControl
|
||||||
:label="__('Possibility') + ' ' + n"
|
:label="__('Possibility') + ' ' + n"
|
||||||
v-model="question[`possibility_${n}`]"
|
v-model="question[`possibility_${n}`]"
|
||||||
|
:required="n == 1 ? true : false"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -127,7 +130,7 @@ const populateFields = () => {
|
|||||||
let counter = 1
|
let counter = 1
|
||||||
fields.forEach((field) => {
|
fields.forEach((field) => {
|
||||||
while (counter <= 4) {
|
while (counter <= 4) {
|
||||||
question[`${field}_${counter}`] = field === 'is_correct' ? false : ''
|
question[`${field}_${counter}`] = field === 'is_correct' ? false : null
|
||||||
counter++
|
counter++
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -108,9 +108,31 @@ const tabsStructure = computed(() => {
|
|||||||
hideLabel: true,
|
hideLabel: true,
|
||||||
items: [
|
items: [
|
||||||
{
|
{
|
||||||
label: 'Members',
|
label: 'General',
|
||||||
description: 'Manage the members of your learning system',
|
icon: 'Wrench',
|
||||||
icon: 'UserRoundPlus',
|
fields: [
|
||||||
|
{
|
||||||
|
label: 'Enable Learning Paths',
|
||||||
|
name: 'enable_learning_paths',
|
||||||
|
description:
|
||||||
|
'This will enforce students to go through programs assigned to them in the correct order.',
|
||||||
|
type: 'checkbox',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Send calendar invite for evaluations',
|
||||||
|
name: 'send_calendar_invite_for_evaluations',
|
||||||
|
description:
|
||||||
|
'If enabled, it sends google calendar invite to the student for evaluations.',
|
||||||
|
type: 'checkbox',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Unsplash Access Key',
|
||||||
|
name: 'unsplash_access_key',
|
||||||
|
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.',
|
||||||
|
type: 'text',
|
||||||
|
},
|
||||||
|
],
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
@@ -156,9 +178,14 @@ const tabsStructure = computed(() => {
|
|||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: 'Settings',
|
label: 'Lists',
|
||||||
hideLabel: true,
|
hideLabel: false,
|
||||||
items: [
|
items: [
|
||||||
|
{
|
||||||
|
label: 'Members',
|
||||||
|
description: 'Manage the members of your learning system',
|
||||||
|
icon: 'UserRoundPlus',
|
||||||
|
},
|
||||||
{
|
{
|
||||||
label: 'Categories',
|
label: 'Categories',
|
||||||
description: 'Manage the members of your learning system',
|
description: 'Manage the members of your learning system',
|
||||||
|
|||||||
@@ -28,6 +28,7 @@
|
|||||||
import { Dialog, createResource } from 'frappe-ui'
|
import { Dialog, createResource } from 'frappe-ui'
|
||||||
import { ref } from 'vue'
|
import { ref } from 'vue'
|
||||||
import Link from '@/components/Controls/Link.vue'
|
import Link from '@/components/Controls/Link.vue'
|
||||||
|
import { showToast } from '@/utils'
|
||||||
|
|
||||||
const students = defineModel('reloadStudents')
|
const students = defineModel('reloadStudents')
|
||||||
const student = ref()
|
const student = ref()
|
||||||
@@ -61,8 +62,11 @@ const addStudent = (close) => {
|
|||||||
{
|
{
|
||||||
onSuccess() {
|
onSuccess() {
|
||||||
students.value.reload()
|
students.value.reload()
|
||||||
close()
|
|
||||||
student.value = null
|
student.value = null
|
||||||
|
close()
|
||||||
|
},
|
||||||
|
onError(err) {
|
||||||
|
showToast(__('Error'), __(err.messages?.[0] || err), 'x')
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|||||||
151
frontend/src/components/OnboardingBanner.vue
Normal file
151
frontend/src/components/OnboardingBanner.vue
Normal file
@@ -0,0 +1,151 @@
|
|||||||
|
<template>
|
||||||
|
<div v-if="showOnboardingBanner && onboardingDetails.data">
|
||||||
|
<Tooltip :text="__('Skip Onboarding')" placement="left">
|
||||||
|
<X
|
||||||
|
class="w-4 h-4 stroke-1 absolute top-2 right-2 cursor-pointer mr-1"
|
||||||
|
@click="skipOnboarding.reload()"
|
||||||
|
/>
|
||||||
|
</Tooltip>
|
||||||
|
<div class="flex items-center justify-evenly bg-gray-100 p-10">
|
||||||
|
<div
|
||||||
|
@click="redirectToCourseForm()"
|
||||||
|
class="flex items-center space-x-2"
|
||||||
|
:class="{
|
||||||
|
'cursor-pointer': !onboardingDetails.data.course_created?.length,
|
||||||
|
}"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
v-if="onboardingDetails.data.course_created?.length"
|
||||||
|
class="py-1 px-1 bg-white rounded-full"
|
||||||
|
>
|
||||||
|
<Check class="h-4 w-4 stroke-2 text-green-600" />
|
||||||
|
</span>
|
||||||
|
<span v-else class="font-semibold bg-white px-2 py-1 rounded-full">
|
||||||
|
1
|
||||||
|
</span>
|
||||||
|
<span class="text-lg font-semibold">
|
||||||
|
{{ __('Create a course') }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
@click="redirectToChapterForm()"
|
||||||
|
class="flex items-center space-x-2"
|
||||||
|
:class="{
|
||||||
|
'cursor-pointer':
|
||||||
|
onboardingDetails.data.course_created?.length &&
|
||||||
|
!onboardingDetails.data.chapter_created?.length,
|
||||||
|
'text-gray-400': !onboardingDetails.data.course_created?.length,
|
||||||
|
}"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
v-if="onboardingDetails.data.chapter_created?.length"
|
||||||
|
class="py-1 px-1 bg-white rounded-full"
|
||||||
|
>
|
||||||
|
<Check class="h-4 w-4 stroke-2 text-green-600" />
|
||||||
|
</span>
|
||||||
|
<span v-else class="font-semibold bg-white px-2 py-1 rounded-full">
|
||||||
|
2
|
||||||
|
</span>
|
||||||
|
<span class="text-lg font-semibold">
|
||||||
|
{{ __('Add a chapter') }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
@click="redirectToLessonForm()"
|
||||||
|
class="flex items-center space-x-2"
|
||||||
|
:class="{
|
||||||
|
'cursor-pointer':
|
||||||
|
onboardingDetails.data.course_created?.length &&
|
||||||
|
onboardingDetails.data.chapter_created?.length,
|
||||||
|
'text-gray-400':
|
||||||
|
!onboardingDetails.data.course_created?.length ||
|
||||||
|
!onboardingDetails.data.chapter_created?.length,
|
||||||
|
}"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
v-if="onboardingDetails.data.lesson_created?.length"
|
||||||
|
class="py-1 px-1 bg-white rounded-full"
|
||||||
|
>
|
||||||
|
<Check class="h-4 w-4 stroke-2 text-green-600" />
|
||||||
|
</span>
|
||||||
|
<span class="font-semibold bg-white px-2 py-1 rounded-full"> 3 </span>
|
||||||
|
<span class="text-lg font-semibold">
|
||||||
|
{{ __('Add a lesson') }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { ref, watch } from 'vue'
|
||||||
|
import { Check, X } from 'lucide-vue-next'
|
||||||
|
import { useRouter } from 'vue-router'
|
||||||
|
import { useSettings } from '@/stores/settings'
|
||||||
|
import { createResource, Tooltip } from 'frappe-ui'
|
||||||
|
|
||||||
|
const showOnboardingBanner = ref(false)
|
||||||
|
const settings = useSettings()
|
||||||
|
const onboardingDetails = settings.onboardingDetails
|
||||||
|
const router = useRouter()
|
||||||
|
|
||||||
|
watch(onboardingDetails, () => {
|
||||||
|
if (!onboardingDetails.data?.is_onboarded) {
|
||||||
|
showOnboardingBanner.value = true
|
||||||
|
} else {
|
||||||
|
showOnboardingBanner.value = false
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
const redirectToCourseForm = () => {
|
||||||
|
if (onboardingDetails.data?.course_created.length) {
|
||||||
|
return
|
||||||
|
} else {
|
||||||
|
router.push({ name: 'CourseForm', params: { courseName: 'new' } })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const redirectToChapterForm = () => {
|
||||||
|
if (!onboardingDetails.data?.course_created.length) {
|
||||||
|
return
|
||||||
|
} else {
|
||||||
|
router.push({
|
||||||
|
name: 'CourseForm',
|
||||||
|
params: {
|
||||||
|
courseName: onboardingDetails.data?.first_course,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const redirectToLessonForm = () => {
|
||||||
|
if (!onboardingDetails.data?.course_created.length) {
|
||||||
|
return
|
||||||
|
} else if (!onboardingDetails.data?.chapter_created.length) {
|
||||||
|
return
|
||||||
|
} else {
|
||||||
|
router.push({
|
||||||
|
name: 'LessonForm',
|
||||||
|
params: {
|
||||||
|
courseName: onboardingDetails.data?.first_course,
|
||||||
|
chapterNumber: 1,
|
||||||
|
lessonNumber: 1,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const skipOnboarding = createResource({
|
||||||
|
url: 'frappe.client.set_value',
|
||||||
|
makeParams() {
|
||||||
|
return {
|
||||||
|
doctype: 'LMS Settings',
|
||||||
|
name: 'LMS Settings',
|
||||||
|
fieldname: 'is_onboarding_complete',
|
||||||
|
value: 1,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
onSuccess(data) {
|
||||||
|
onboardingDetails.reload()
|
||||||
|
},
|
||||||
|
})
|
||||||
|
</script>
|
||||||
@@ -397,6 +397,9 @@ const attempts = createResource({
|
|||||||
watch(
|
watch(
|
||||||
() => quiz.data,
|
() => quiz.data,
|
||||||
() => {
|
() => {
|
||||||
|
if (quiz.data) {
|
||||||
|
populateQuestions()
|
||||||
|
}
|
||||||
if (quiz.data && quiz.data.max_attempts) {
|
if (quiz.data && quiz.data.max_attempts) {
|
||||||
attempts.reload()
|
attempts.reload()
|
||||||
resetQuiz()
|
resetQuiz()
|
||||||
|
|||||||
@@ -29,6 +29,7 @@
|
|||||||
<script setup>
|
<script setup>
|
||||||
import { Button, Badge } from 'frappe-ui'
|
import { Button, Badge } from 'frappe-ui'
|
||||||
import SettingFields from '@/components/SettingFields.vue'
|
import SettingFields from '@/components/SettingFields.vue'
|
||||||
|
import { showToast } from '@/utils'
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
fields: {
|
fields: {
|
||||||
@@ -54,7 +55,14 @@ const update = () => {
|
|||||||
props.data.doc[f.name] = f.value
|
props.data.doc[f.name] = f.value
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
props.data.save.submit()
|
props.data.save.submit(
|
||||||
|
{},
|
||||||
|
{
|
||||||
|
onError(err) {
|
||||||
|
showToast(__('Error'), err.messages?.[0] || err, 'x')
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|||||||
@@ -90,6 +90,7 @@
|
|||||||
:type="field.type"
|
:type="field.type"
|
||||||
:rows="field.rows"
|
:rows="field.rows"
|
||||||
:options="field.options"
|
:options="field.options"
|
||||||
|
:description="field.description"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -100,7 +101,7 @@
|
|||||||
import { FormControl, FileUploader, Button, Switch } from 'frappe-ui'
|
import { FormControl, FileUploader, Button, Switch } from 'frappe-ui'
|
||||||
import { computed } from 'vue'
|
import { computed } from 'vue'
|
||||||
import { getFileSize, validateFile } from '@/utils'
|
import { getFileSize, validateFile } from '@/utils'
|
||||||
import { X, FileText } from 'lucide-vue-next'
|
import { X } from 'lucide-vue-next'
|
||||||
import Link from '@/components/Controls/Link.vue'
|
import Link from '@/components/Controls/Link.vue'
|
||||||
import CodeEditor from '@/components/Controls/CodeEditor.vue'
|
import CodeEditor from '@/components/Controls/CodeEditor.vue'
|
||||||
|
|
||||||
|
|||||||
@@ -7,7 +7,7 @@
|
|||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
class="flex items-center w-full duration-300 ease-in-out group"
|
class="flex items-center w-full duration-300 ease-in-out group"
|
||||||
:class="isCollapsed ? 'p-1' : 'px-2 py-1'"
|
:class="isCollapsed ? 'p-1 relative' : 'px-2 py-1'"
|
||||||
>
|
>
|
||||||
<Tooltip :text="link.label" placement="right">
|
<Tooltip :text="link.label" placement="right">
|
||||||
<slot name="icon">
|
<slot name="icon">
|
||||||
@@ -29,7 +29,15 @@
|
|||||||
>
|
>
|
||||||
{{ __(link.label) }}
|
{{ __(link.label) }}
|
||||||
</span>
|
</span>
|
||||||
<span v-if="link.count" class="!ml-auto block text-xs text-gray-600">
|
<span
|
||||||
|
v-if="link.count"
|
||||||
|
class="!ml-auto block text-xs text-gray-600"
|
||||||
|
:class="
|
||||||
|
isCollapsed && link.count > 9
|
||||||
|
? 'absolute top-[2px] right-0 bg-white'
|
||||||
|
: ''
|
||||||
|
"
|
||||||
|
>
|
||||||
{{ link.count }}
|
{{ link.count }}
|
||||||
</span>
|
</span>
|
||||||
<div
|
<div
|
||||||
|
|||||||
@@ -4,21 +4,29 @@
|
|||||||
class="sticky top-0 z-10 flex items-center justify-between border-b bg-white px-3 py-2.5 sm:px-5"
|
class="sticky top-0 z-10 flex items-center justify-between border-b bg-white px-3 py-2.5 sm:px-5"
|
||||||
>
|
>
|
||||||
<Breadcrumbs class="h-7" :items="breadcrumbs" />
|
<Breadcrumbs class="h-7" :items="breadcrumbs" />
|
||||||
<Button v-if="user.data?.is_moderator" @click="openAnnouncementModal()">
|
<div class="flex items-center space-x-2">
|
||||||
<span>
|
<Button
|
||||||
{{ __('Make an Announcement') }}
|
v-if="user.data?.is_moderator"
|
||||||
</span>
|
@click="openCertificateDialog = true"
|
||||||
<template #suffix>
|
>
|
||||||
<SendIcon class="h-4 stroke-1.5" />
|
{{ __('Generate Certificates') }}
|
||||||
</template>
|
</Button>
|
||||||
</Button>
|
<Button v-if="user.data?.is_moderator" @click="openAnnouncementModal()">
|
||||||
|
<span>
|
||||||
|
{{ __('Make an Announcement') }}
|
||||||
|
</span>
|
||||||
|
<template #suffix>
|
||||||
|
<SendIcon class="h-4 stroke-1.5" />
|
||||||
|
</template>
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
</header>
|
</header>
|
||||||
<div v-if="batch.data" class="grid grid-cols-[70%,30%] h-screen">
|
<div v-if="batch.data" class="grid grid-cols-[70%,30%] h-screen">
|
||||||
<div class="border-r-2">
|
<div class="border-r-2">
|
||||||
<Tabs
|
<Tabs
|
||||||
v-model="tabIndex"
|
v-model="tabIndex"
|
||||||
:tabs="tabs"
|
:tabs="tabs"
|
||||||
tablistClass="overflow-y-hidden sticky top-11 bg-white z-10"
|
tablistClass="overflow-y-hidden bg-white"
|
||||||
>
|
>
|
||||||
<template #tab="{ tab, selected }" class="overflow-x-hidden">
|
<template #tab="{ tab, selected }" class="overflow-x-hidden">
|
||||||
<div>
|
<div>
|
||||||
@@ -169,6 +177,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<BulkCertificates v-model="openCertificateDialog" :batch="batch.data" />
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { Breadcrumbs, Button, createResource, Tabs, Badge } from 'frappe-ui'
|
import { Breadcrumbs, Button, createResource, Tabs, Badge } from 'frappe-ui'
|
||||||
@@ -197,9 +206,11 @@ import Announcements from '@/components/Annoucements.vue'
|
|||||||
import AnnouncementModal from '@/components/Modals/AnnouncementModal.vue'
|
import AnnouncementModal from '@/components/Modals/AnnouncementModal.vue'
|
||||||
import Discussions from '@/components/Discussions.vue'
|
import Discussions from '@/components/Discussions.vue'
|
||||||
import DateRange from '@/components/Common/DateRange.vue'
|
import DateRange from '@/components/Common/DateRange.vue'
|
||||||
|
import BulkCertificates from '@/components/Modals/BulkCertificates.vue'
|
||||||
|
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
const showAnnouncementModal = ref(false)
|
const showAnnouncementModal = ref(false)
|
||||||
|
const openCertificateDialog = ref(false)
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
batchName: {
|
batchName: {
|
||||||
|
|||||||
@@ -252,7 +252,7 @@ import {
|
|||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import Link from '@/components/Controls/Link.vue'
|
import Link from '@/components/Controls/Link.vue'
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
import { showToast } from '../utils'
|
import { showToast } from '@/utils'
|
||||||
import { Image } from 'lucide-vue-next'
|
import { Image } from 'lucide-vue-next'
|
||||||
import { capture } from '@/telemetry'
|
import { capture } from '@/telemetry'
|
||||||
import MultiSelect from '@/components/Controls/MultiSelect.vue'
|
import MultiSelect from '@/components/Controls/MultiSelect.vue'
|
||||||
@@ -345,6 +345,10 @@ const batchDetail = createResource({
|
|||||||
data.instructors.forEach((instructor) => {
|
data.instructors.forEach((instructor) => {
|
||||||
instructors.value.push(instructor.instructor)
|
instructors.value.push(instructor.instructor)
|
||||||
})
|
})
|
||||||
|
} else if (['start_time', 'end_time'].includes(key)) {
|
||||||
|
let [hours, minutes, seconds] = data[key].split(':')
|
||||||
|
hours = hours.length == 1 ? '0' + hours : hours
|
||||||
|
batch[key] = `${hours}:${minutes}`
|
||||||
} else if (Object.hasOwn(batch, key)) batch[key] = data[key]
|
} else if (Object.hasOwn(batch, key)) batch[key] = data[key]
|
||||||
})
|
})
|
||||||
let checkboxes = ['published', 'paid_batch', 'allow_self_enrollment']
|
let checkboxes = ['published', 'paid_batch', 'allow_self_enrollment']
|
||||||
|
|||||||
@@ -70,7 +70,11 @@
|
|||||||
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-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"
|
||||||
></div>
|
></div>
|
||||||
<div class="mt-10">
|
<div class="mt-10">
|
||||||
<CourseOutline :courseName="course.data.name" :showOutline="true" />
|
<CourseOutline
|
||||||
|
:title="__('Course Outline')"
|
||||||
|
:courseName="course.data.name"
|
||||||
|
:showOutline="true"
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
<CourseReviews
|
<CourseReviews
|
||||||
:courseName="course.data.name"
|
:courseName="course.data.name"
|
||||||
|
|||||||
@@ -133,8 +133,8 @@
|
|||||||
</div>
|
</div>
|
||||||
<FormControl
|
<FormControl
|
||||||
v-model="newTag"
|
v-model="newTag"
|
||||||
:placeholder="__('Keywords for the course')"
|
:placeholder="__('Add a keyword and then press enter')"
|
||||||
class="w-52"
|
class="w-72"
|
||||||
@keyup.enter="updateTags()"
|
@keyup.enter="updateTags()"
|
||||||
id="tags"
|
id="tags"
|
||||||
/>
|
/>
|
||||||
@@ -288,6 +288,7 @@ const course = reactive({
|
|||||||
video_link: '',
|
video_link: '',
|
||||||
course_image: null,
|
course_image: null,
|
||||||
tags: '',
|
tags: '',
|
||||||
|
category: '',
|
||||||
published: false,
|
published: false,
|
||||||
published_on: '',
|
published_on: '',
|
||||||
featured: false,
|
featured: false,
|
||||||
@@ -434,6 +435,9 @@ const submitCourse = () => {
|
|||||||
onSuccess(data) {
|
onSuccess(data) {
|
||||||
capture('course_created')
|
capture('course_created')
|
||||||
showToast('Success', 'Course created successfully', 'check')
|
showToast('Success', 'Course created successfully', 'check')
|
||||||
|
if (!settingsStore.onboardingDetails.data?.is_onboarded) {
|
||||||
|
settingsStore.onboardingDetails.reload()
|
||||||
|
}
|
||||||
router.push({
|
router.push({
|
||||||
name: 'CourseForm',
|
name: 'CourseForm',
|
||||||
params: { courseName: data.name },
|
params: { courseName: data.name },
|
||||||
|
|||||||
@@ -30,6 +30,7 @@
|
|||||||
</FormControl>
|
</FormControl>
|
||||||
</div>
|
</div>
|
||||||
<router-link
|
<router-link
|
||||||
|
v-if="user.data?.is_moderator || user.data?.is_instructor"
|
||||||
:to="{
|
:to="{
|
||||||
name: 'CourseForm',
|
name: 'CourseForm',
|
||||||
params: {
|
params: {
|
||||||
@@ -37,7 +38,7 @@
|
|||||||
},
|
},
|
||||||
}"
|
}"
|
||||||
>
|
>
|
||||||
<Button v-if="user.data?.is_moderator" variant="solid">
|
<Button variant="solid">
|
||||||
<template #prefix>
|
<template #prefix>
|
||||||
<Plus class="h-4 w-4" />
|
<Plus class="h-4 w-4" />
|
||||||
</template>
|
</template>
|
||||||
@@ -159,30 +160,45 @@
|
|||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
import {
|
import {
|
||||||
Breadcrumbs,
|
|
||||||
Tabs,
|
|
||||||
Badge,
|
Badge,
|
||||||
|
Breadcrumbs,
|
||||||
Button,
|
Button,
|
||||||
FormControl,
|
call,
|
||||||
createResource,
|
createResource,
|
||||||
|
FormControl,
|
||||||
|
Tabs,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import CourseCard from '@/components/CourseCard.vue'
|
import CourseCard from '@/components/CourseCard.vue'
|
||||||
import { BookOpen, Plus, Search } from 'lucide-vue-next'
|
import { BookOpen, Plus, Search } from 'lucide-vue-next'
|
||||||
import { ref, computed, inject, onMounted, watch } from 'vue'
|
import { ref, computed, inject, onMounted, watch } from 'vue'
|
||||||
import { updateDocumentTitle } from '@/utils'
|
import { updateDocumentTitle } from '@/utils'
|
||||||
|
import { useRouter } from 'vue-router'
|
||||||
|
import { useSettings } from '@/stores/settings'
|
||||||
|
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
const searchQuery = ref('')
|
const searchQuery = ref('')
|
||||||
const currentCategory = ref(null)
|
const currentCategory = ref(null)
|
||||||
const hasCourses = ref(false)
|
const hasCourses = ref(false)
|
||||||
|
const router = useRouter()
|
||||||
|
const settings = useSettings()
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
|
checkLearningPath()
|
||||||
let queries = new URLSearchParams(location.search)
|
let queries = new URLSearchParams(location.search)
|
||||||
if (queries.has('category')) {
|
if (queries.has('category')) {
|
||||||
currentCategory.value = queries.get('category')
|
currentCategory.value = queries.get('category')
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
const checkLearningPath = () => {
|
||||||
|
if (
|
||||||
|
settings.learningPaths.data &&
|
||||||
|
(!user.data?.is_moderator || !user.data?.is_instructor)
|
||||||
|
) {
|
||||||
|
router.push({ name: 'Programs' })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const courses = createResource({
|
const courses = createResource({
|
||||||
url: 'lms.lms.utils.get_courses',
|
url: 'lms.lms.utils.get_courses',
|
||||||
cache: ['courses', user.data?.email],
|
cache: ['courses', user.data?.email],
|
||||||
|
|||||||
@@ -19,8 +19,13 @@
|
|||||||
v-model="job.job_title"
|
v-model="job.job_title"
|
||||||
:label="__('Title')"
|
:label="__('Title')"
|
||||||
class="mb-4"
|
class="mb-4"
|
||||||
|
:required="true"
|
||||||
|
/>
|
||||||
|
<FormControl
|
||||||
|
v-model="job.location"
|
||||||
|
:label="__('Location')"
|
||||||
|
:required="true"
|
||||||
/>
|
/>
|
||||||
<FormControl v-model="job.location" :label="__('Location')" />
|
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<FormControl
|
<FormControl
|
||||||
@@ -29,18 +34,21 @@
|
|||||||
type="select"
|
type="select"
|
||||||
:options="jobTypes"
|
:options="jobTypes"
|
||||||
class="mb-4"
|
class="mb-4"
|
||||||
|
:required="true"
|
||||||
/>
|
/>
|
||||||
<FormControl
|
<FormControl
|
||||||
v-model="job.status"
|
v-model="job.status"
|
||||||
:label="__('Status')"
|
:label="__('Status')"
|
||||||
type="select"
|
type="select"
|
||||||
:options="jobStatuses"
|
:options="jobStatuses"
|
||||||
|
:required="true"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="mt-4">
|
<div class="mt-4">
|
||||||
<label class="block text-gray-600 text-xs mb-1">
|
<label class="block text-gray-600 text-xs mb-1">
|
||||||
{{ __('Description') }}
|
{{ __('Description') }}
|
||||||
|
<span class="text-red-500">*</span>
|
||||||
</label>
|
</label>
|
||||||
<TextEditor
|
<TextEditor
|
||||||
:content="job.description"
|
:content="job.description"
|
||||||
@@ -61,10 +69,12 @@
|
|||||||
v-model="job.company_name"
|
v-model="job.company_name"
|
||||||
:label="__('Company Name')"
|
:label="__('Company Name')"
|
||||||
class="mb-4"
|
class="mb-4"
|
||||||
|
:required="true"
|
||||||
/>
|
/>
|
||||||
<FormControl
|
<FormControl
|
||||||
v-model="job.company_website"
|
v-model="job.company_website"
|
||||||
:label="__('Company Website')"
|
:label="__('Company Website')"
|
||||||
|
:required="true"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
@@ -72,9 +82,11 @@
|
|||||||
v-model="job.company_email_address"
|
v-model="job.company_email_address"
|
||||||
:label="__('Company Email Address')"
|
:label="__('Company Email Address')"
|
||||||
class="mb-4"
|
class="mb-4"
|
||||||
|
:required="true"
|
||||||
/>
|
/>
|
||||||
<label class="block text-gray-600 text-xs mb-1 mt-4">
|
<label class="block text-gray-600 text-xs mb-1 mt-4">
|
||||||
{{ __('Company Logo') }}
|
{{ __('Company Logo') }}
|
||||||
|
<span class="text-red-500">*</span>
|
||||||
</label>
|
</label>
|
||||||
<FileUploader
|
<FileUploader
|
||||||
v-if="!job.image"
|
v-if="!job.image"
|
||||||
|
|||||||
@@ -7,7 +7,22 @@
|
|||||||
class="h-7"
|
class="h-7"
|
||||||
:items="[{ label: __('Jobs'), route: { name: 'Jobs' } }]"
|
:items="[{ label: __('Jobs'), route: { name: 'Jobs' } }]"
|
||||||
/>
|
/>
|
||||||
<div class="flex">
|
<div class="flex space-x-2">
|
||||||
|
<div class="w-40 md:w-44">
|
||||||
|
<FormControl
|
||||||
|
v-model="jobType"
|
||||||
|
type="select"
|
||||||
|
:options="jobTypes"
|
||||||
|
:placeholder="__('Type')"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="w-28 md:w-36">
|
||||||
|
<FormControl type="text" placeholder="Search" v-model="searchQuery">
|
||||||
|
<template #prefix>
|
||||||
|
<Search class="w-4 h-4 stroke-1.5 text-gray-600" name="search" />
|
||||||
|
</template>
|
||||||
|
</FormControl>
|
||||||
|
</div>
|
||||||
<router-link
|
<router-link
|
||||||
v-if="user.data?.name"
|
v-if="user.data?.name"
|
||||||
:to="{
|
:to="{
|
||||||
@@ -26,9 +41,9 @@
|
|||||||
</router-link>
|
</router-link>
|
||||||
</div>
|
</div>
|
||||||
</header>
|
</header>
|
||||||
<div v-if="jobs.data?.length">
|
<div v-if="jobsList?.length">
|
||||||
<div class="divide-y lg:w-3/4 mx-auto p-5">
|
<div class="divide-y lg:w-3/4 mx-auto p-5">
|
||||||
<div v-for="job in jobs.data">
|
<div v-for="job in jobsList">
|
||||||
<router-link
|
<router-link
|
||||||
:to="{
|
:to="{
|
||||||
name: 'JobDetail',
|
name: 'JobDetail',
|
||||||
@@ -47,13 +62,22 @@
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { Button, Breadcrumbs, createResource } from 'frappe-ui'
|
import { Button, Breadcrumbs, createResource, FormControl } from 'frappe-ui'
|
||||||
import { Plus } from 'lucide-vue-next'
|
import { Plus, Search } from 'lucide-vue-next'
|
||||||
import { inject, computed } from 'vue'
|
import { inject, computed, ref, onMounted } from 'vue'
|
||||||
import JobCard from '@/components/JobCard.vue'
|
import JobCard from '@/components/JobCard.vue'
|
||||||
import { updateDocumentTitle } from '@/utils'
|
import { updateDocumentTitle } from '@/utils'
|
||||||
|
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
|
const jobType = ref(null)
|
||||||
|
const searchQuery = ref('')
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
let queries = new URLSearchParams(location.search)
|
||||||
|
if (queries.has('type')) {
|
||||||
|
jobType.value = queries.get('type')
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
const jobs = createResource({
|
const jobs = createResource({
|
||||||
url: 'lms.lms.api.get_job_opportunities',
|
url: 'lms.lms.api.get_job_opportunities',
|
||||||
@@ -68,5 +92,32 @@ const pageMeta = computed(() => {
|
|||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
const jobsList = computed(() => {
|
||||||
|
let jobData = jobs.data
|
||||||
|
if (jobType.value && jobType.value != '') {
|
||||||
|
jobData = jobData.filter((job) => job.type == jobType.value)
|
||||||
|
}
|
||||||
|
if (searchQuery.value) {
|
||||||
|
let query = searchQuery.value.toLowerCase()
|
||||||
|
jobData = jobData.filter(
|
||||||
|
(job) =>
|
||||||
|
job.job_title.toLowerCase().includes(query) ||
|
||||||
|
job.company_name.toLowerCase().includes(query) ||
|
||||||
|
job.location.toLowerCase().includes(query)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
return jobData
|
||||||
|
})
|
||||||
|
|
||||||
|
const jobTypes = computed(() => {
|
||||||
|
return [
|
||||||
|
'',
|
||||||
|
{ label: __('Full Time'), value: 'Full Time' },
|
||||||
|
{ label: __('Part Time'), value: 'Part Time' },
|
||||||
|
{ label: __('Contract'), value: 'Contract' },
|
||||||
|
{ label: __('Freelance'), value: 'Freelance' },
|
||||||
|
]
|
||||||
|
})
|
||||||
|
|
||||||
updateDocumentTitle(pageMeta)
|
updateDocumentTitle(pageMeta)
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -103,7 +103,7 @@
|
|||||||
<span
|
<span
|
||||||
class="h-6 mr-1"
|
class="h-6 mr-1"
|
||||||
:class="{
|
:class="{
|
||||||
'avatar-group overlap': lesson.data.instructors.length > 1,
|
'avatar-group overlap': lesson.data.instructors?.length > 1,
|
||||||
}"
|
}"
|
||||||
>
|
>
|
||||||
<UserAvatar
|
<UserAvatar
|
||||||
@@ -111,7 +111,10 @@
|
|||||||
:user="instructor"
|
:user="instructor"
|
||||||
/>
|
/>
|
||||||
</span>
|
</span>
|
||||||
<CourseInstructors :instructors="lesson.data.instructors" />
|
<CourseInstructors
|
||||||
|
v-if="lesson.data?.instructors"
|
||||||
|
:instructors="lesson.data.instructors"
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
v-if="
|
v-if="
|
||||||
@@ -146,6 +149,7 @@
|
|||||||
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 mt-5"
|
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 mt-5"
|
||||||
>
|
>
|
||||||
<LessonContent
|
<LessonContent
|
||||||
|
v-if="lesson.data?.body"
|
||||||
:content="lesson.data.body"
|
:content="lesson.data.body"
|
||||||
:youtube="lesson.data.youtube"
|
:youtube="lesson.data.youtube"
|
||||||
:quizId="lesson.data.quiz_id"
|
:quizId="lesson.data.quiz_id"
|
||||||
@@ -240,7 +244,10 @@ const lesson = createResource({
|
|||||||
auto: true,
|
auto: true,
|
||||||
onSuccess(data) {
|
onSuccess(data) {
|
||||||
if (Object.keys(data).length === 0) {
|
if (Object.keys(data).length === 0) {
|
||||||
router.push({ name: 'Courses' })
|
router.push({
|
||||||
|
name: 'CourseDetail',
|
||||||
|
params: { courseName: props.courseName },
|
||||||
|
})
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
lessonProgress.value = data.membership?.progress
|
lessonProgress.value = data.membership?.progress
|
||||||
@@ -369,13 +376,13 @@ const checkIfDiscussionsAllowed = () => {
|
|||||||
|
|
||||||
const allowEdit = () => {
|
const allowEdit = () => {
|
||||||
if (user.data?.is_moderator) return true
|
if (user.data?.is_moderator) return true
|
||||||
if (lesson.data?.instructors.includes(user.data?.name)) return true
|
if (lesson.data?.instructors?.includes(user.data?.name)) return true
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
const allowInstructorContent = () => {
|
const allowInstructorContent = () => {
|
||||||
if (user.data?.is_moderator) return true
|
if (user.data?.is_moderator) return true
|
||||||
if (lesson.data?.instructors.includes(user.data?.name)) return true
|
if (lesson.data?.instructors?.includes(user.data?.name)) return true
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -6,7 +6,11 @@
|
|||||||
class="sticky top-0 z-10 flex flex-col md:flex-row md:items-center justify-between border-b overflow-hidden bg-white px-3 py-2.5 sm:px-5"
|
class="sticky top-0 z-10 flex flex-col md:flex-row md:items-center justify-between border-b overflow-hidden bg-white px-3 py-2.5 sm:px-5"
|
||||||
>
|
>
|
||||||
<Breadcrumbs class="text-ellipsis" :items="breadcrumbs" />
|
<Breadcrumbs class="text-ellipsis" :items="breadcrumbs" />
|
||||||
<Button variant="solid" @click="saveLesson()" class="mt-3 md:mt-0">
|
<Button
|
||||||
|
variant="solid"
|
||||||
|
@click="saveLesson({ showSuccessMessage: true })"
|
||||||
|
class="mt-3 md:mt-0"
|
||||||
|
>
|
||||||
{{ __('Save') }}
|
{{ __('Save') }}
|
||||||
</Button>
|
</Button>
|
||||||
</header>
|
</header>
|
||||||
@@ -88,12 +92,15 @@ import LessonHelp from '@/components/LessonHelp.vue'
|
|||||||
import { ChevronRight } from 'lucide-vue-next'
|
import { ChevronRight } from 'lucide-vue-next'
|
||||||
import { updateDocumentTitle, createToast, getEditorTools } from '@/utils'
|
import { updateDocumentTitle, createToast, getEditorTools } from '@/utils'
|
||||||
import { capture } from '@/telemetry'
|
import { capture } from '@/telemetry'
|
||||||
|
import { useSettings } from '@/stores/settings'
|
||||||
|
|
||||||
const editor = ref(null)
|
const editor = ref(null)
|
||||||
const instructorEditor = ref(null)
|
const instructorEditor = ref(null)
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
const openInstructorEditor = ref(false)
|
const openInstructorEditor = ref(false)
|
||||||
|
const settingsStore = useSettings()
|
||||||
let autoSaveInterval
|
let autoSaveInterval
|
||||||
|
let showSuccessMessage = false
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
courseName: {
|
courseName: {
|
||||||
@@ -117,6 +124,7 @@ onMounted(() => {
|
|||||||
capture('lesson_form_opened')
|
capture('lesson_form_opened')
|
||||||
editor.value = renderEditor('content')
|
editor.value = renderEditor('content')
|
||||||
instructorEditor.value = renderEditor('instructor-notes')
|
instructorEditor.value = renderEditor('instructor-notes')
|
||||||
|
window.addEventListener('keydown', keyboardShortcut)
|
||||||
})
|
})
|
||||||
|
|
||||||
const renderEditor = (holder) => {
|
const renderEditor = (holder) => {
|
||||||
@@ -124,6 +132,7 @@ const renderEditor = (holder) => {
|
|||||||
holder: holder,
|
holder: holder,
|
||||||
tools: getEditorTools(true),
|
tools: getEditorTools(true),
|
||||||
autofocus: true,
|
autofocus: true,
|
||||||
|
defaultBlock: 'markdown',
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -186,12 +195,24 @@ const addInstructorNotes = (data) => {
|
|||||||
|
|
||||||
const enableAutoSave = () => {
|
const enableAutoSave = () => {
|
||||||
autoSaveInterval = setInterval(() => {
|
autoSaveInterval = setInterval(() => {
|
||||||
saveLesson()
|
saveLesson({ showSuccessMessage: false })
|
||||||
}, 10000)
|
}, 10000)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const keyboardShortcut = (e) => {
|
||||||
|
if (
|
||||||
|
e.key === 's' &&
|
||||||
|
(e.ctrlKey || e.metaKey) &&
|
||||||
|
!e.target.classList.contains('ProseMirror')
|
||||||
|
) {
|
||||||
|
saveLesson({ showSuccessMessage: true })
|
||||||
|
e.preventDefault()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
onBeforeUnmount(() => {
|
onBeforeUnmount(() => {
|
||||||
clearInterval(autoSaveInterval)
|
clearInterval(autoSaveInterval)
|
||||||
|
window.removeEventListener('keydown', keyboardShortcut)
|
||||||
})
|
})
|
||||||
|
|
||||||
const newLessonResource = createResource({
|
const newLessonResource = createResource({
|
||||||
@@ -343,7 +364,11 @@ const convertToJSON = (lessonData) => {
|
|||||||
return blocks
|
return blocks
|
||||||
}
|
}
|
||||||
|
|
||||||
const saveLesson = () => {
|
const saveLesson = (e) => {
|
||||||
|
showSuccessMessage = false
|
||||||
|
if (typeof e != 'undefined' && e.showSuccessMessage) {
|
||||||
|
showSuccessMessage = true
|
||||||
|
}
|
||||||
editor.value.save().then((outputData) => {
|
editor.value.save().then((outputData) => {
|
||||||
lesson.content = JSON.stringify(outputData)
|
lesson.content = JSON.stringify(outputData)
|
||||||
instructorEditor.value.save().then((outputData) => {
|
instructorEditor.value.save().then((outputData) => {
|
||||||
@@ -371,6 +396,9 @@ const createNewLesson = () => {
|
|||||||
onSuccess() {
|
onSuccess() {
|
||||||
capture('lesson_created')
|
capture('lesson_created')
|
||||||
showToast('Success', 'Lesson created successfully', 'check')
|
showToast('Success', 'Lesson created successfully', 'check')
|
||||||
|
if (!settingsStore.onboardingDetails.data?.is_onboarded) {
|
||||||
|
settingsStore.onboardingDetails.reload()
|
||||||
|
}
|
||||||
lessonDetails.reload()
|
lessonDetails.reload()
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
@@ -392,6 +420,11 @@ const editCurrentLesson = () => {
|
|||||||
validate() {
|
validate() {
|
||||||
return validateLesson()
|
return validateLesson()
|
||||||
},
|
},
|
||||||
|
onSuccess() {
|
||||||
|
showSuccessMessage
|
||||||
|
? showToast('Success', 'Lesson updated successfully', 'check')
|
||||||
|
: ''
|
||||||
|
},
|
||||||
onError(err) {
|
onError(err) {
|
||||||
showToast('Error', err.message, 'x')
|
showToast('Error', err.message, 'x')
|
||||||
},
|
},
|
||||||
|
|||||||
367
frontend/src/pages/ProgramForm.vue
Normal file
367
frontend/src/pages/ProgramForm.vue
Normal file
@@ -0,0 +1,367 @@
|
|||||||
|
<template>
|
||||||
|
<header
|
||||||
|
class="sticky top-0 z-10 flex flex-col md:flex-row md:items-center justify-between border-b bg-white px-3 py-2.5 sm:px-5"
|
||||||
|
>
|
||||||
|
<Breadcrumbs :items="breadbrumbs" />
|
||||||
|
<Button variant="solid" @click="saveProgram()">
|
||||||
|
{{ __('Save') }}
|
||||||
|
</Button>
|
||||||
|
</header>
|
||||||
|
<div v-if="program.doc" class="pt-5 px-5 w-3/4 mx-auto space-y-10">
|
||||||
|
<FormControl v-model="program.doc.title" :label="__('Title')" />
|
||||||
|
|
||||||
|
<!-- Courses -->
|
||||||
|
<div>
|
||||||
|
<div class="flex items-center justify-between mb-2">
|
||||||
|
<div class="text-lg font-semibold">
|
||||||
|
{{ __('Program Courses') }}
|
||||||
|
</div>
|
||||||
|
<Button
|
||||||
|
@click="
|
||||||
|
() => {
|
||||||
|
currentForm = 'course'
|
||||||
|
showDialog = true
|
||||||
|
}
|
||||||
|
"
|
||||||
|
>
|
||||||
|
<template #prefix>
|
||||||
|
<Plus class="w-4 h-4" />
|
||||||
|
</template>
|
||||||
|
{{ __('Add') }}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<ListView
|
||||||
|
:columns="courseColumns"
|
||||||
|
:rows="program.doc.program_courses"
|
||||||
|
row-key="name"
|
||||||
|
:options="{
|
||||||
|
showTooltip: false,
|
||||||
|
}"
|
||||||
|
>
|
||||||
|
<ListHeader
|
||||||
|
class="mb-2 grid items-center space-x-4 rounded bg-gray-100 p-2"
|
||||||
|
>
|
||||||
|
<ListHeaderItem :item="item" v-for="item in courseColumns" />
|
||||||
|
</ListHeader>
|
||||||
|
<ListRows>
|
||||||
|
<Draggable
|
||||||
|
:list="program.doc.program_courses"
|
||||||
|
item-key="name"
|
||||||
|
group="items"
|
||||||
|
@end="updateOrder"
|
||||||
|
class="cursor-move"
|
||||||
|
>
|
||||||
|
<template #item="{ element: row }">
|
||||||
|
<ListRow :row="row" />
|
||||||
|
</template>
|
||||||
|
</Draggable>
|
||||||
|
</ListRows>
|
||||||
|
<ListSelectBanner>
|
||||||
|
<template #actions="{ unselectAll, selections }">
|
||||||
|
<div class="flex gap-2">
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
@click="remove(selections, unselectAll, 'program_courses')"
|
||||||
|
>
|
||||||
|
<Trash2 class="h-4 w-4 stroke-1.5" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</ListSelectBanner>
|
||||||
|
</ListView>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Members -->
|
||||||
|
<div>
|
||||||
|
<div class="flex items-center justify-between mb-2">
|
||||||
|
<div class="text-lg font-semibold">
|
||||||
|
{{ __('Program Members') }}
|
||||||
|
</div>
|
||||||
|
<Button
|
||||||
|
@click="
|
||||||
|
() => {
|
||||||
|
currentForm = 'member'
|
||||||
|
showDialog = true
|
||||||
|
}
|
||||||
|
"
|
||||||
|
>
|
||||||
|
<template #prefix>
|
||||||
|
<Plus class="w-4 h-4" />
|
||||||
|
</template>
|
||||||
|
{{ __('Add') }}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<ListView
|
||||||
|
:columns="memberColumns"
|
||||||
|
:rows="program.doc.program_members"
|
||||||
|
row-key="name"
|
||||||
|
:options="{
|
||||||
|
showTooltip: false,
|
||||||
|
}"
|
||||||
|
>
|
||||||
|
<ListHeader
|
||||||
|
class="mb-2 grid items-center space-x-4 rounded bg-gray-100 p-2"
|
||||||
|
>
|
||||||
|
<ListHeaderItem :item="item" v-for="item in memberColumns" />
|
||||||
|
</ListHeader>
|
||||||
|
<ListRows>
|
||||||
|
<ListRow :row="row" v-for="row in program.doc.program_members" />
|
||||||
|
</ListRows>
|
||||||
|
<ListSelectBanner>
|
||||||
|
<template #actions="{ unselectAll, selections }">
|
||||||
|
<div class="flex gap-2">
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
@click="remove(selections, unselectAll, 'program_members')"
|
||||||
|
>
|
||||||
|
<Trash2 class="h-4 w-4 stroke-1.5" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</ListSelectBanner>
|
||||||
|
</ListView>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Dialog
|
||||||
|
v-model="showDialog"
|
||||||
|
:options="{
|
||||||
|
title:
|
||||||
|
currentForm == 'course'
|
||||||
|
? __('New Program Course')
|
||||||
|
: __('New Program Member'),
|
||||||
|
actions: [
|
||||||
|
{
|
||||||
|
label: __('Add'),
|
||||||
|
variant: 'solid',
|
||||||
|
onClick: () =>
|
||||||
|
currentForm == 'course'
|
||||||
|
? addProgramCourse(close)
|
||||||
|
: addProgramMember(close),
|
||||||
|
},
|
||||||
|
],
|
||||||
|
}"
|
||||||
|
>
|
||||||
|
<template #body-content>
|
||||||
|
<Link
|
||||||
|
v-if="currentForm == 'course'"
|
||||||
|
v-model="course"
|
||||||
|
doctype="LMS Course"
|
||||||
|
:filters="{
|
||||||
|
disable_self_learning: 1,
|
||||||
|
}"
|
||||||
|
:label="__('Program Course')"
|
||||||
|
:description="
|
||||||
|
__(
|
||||||
|
'Only courses for which self learning is disabled can be added to program.'
|
||||||
|
)
|
||||||
|
"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Link
|
||||||
|
v-if="currentForm == 'member'"
|
||||||
|
v-model="member"
|
||||||
|
doctype="User"
|
||||||
|
:filters="{
|
||||||
|
ignore_user_type: 1,
|
||||||
|
}"
|
||||||
|
:label="__('Program Member')"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
</Dialog>
|
||||||
|
</template>
|
||||||
|
<script setup>
|
||||||
|
import {
|
||||||
|
Breadcrumbs,
|
||||||
|
Button,
|
||||||
|
call,
|
||||||
|
createDocumentResource,
|
||||||
|
Dialog,
|
||||||
|
FormControl,
|
||||||
|
ListView,
|
||||||
|
ListRows,
|
||||||
|
ListRow,
|
||||||
|
ListHeader,
|
||||||
|
ListHeaderItem,
|
||||||
|
ListSelectBanner,
|
||||||
|
} from 'frappe-ui'
|
||||||
|
import { computed, ref } from 'vue'
|
||||||
|
import { Plus, Trash2 } from 'lucide-vue-next'
|
||||||
|
import Link from '@/components/Controls/Link.vue'
|
||||||
|
import { showToast } from '@/utils/'
|
||||||
|
import Draggable from 'vuedraggable'
|
||||||
|
import { useRouter } from 'vue-router'
|
||||||
|
|
||||||
|
const showDialog = ref(false)
|
||||||
|
const currentForm = ref(null)
|
||||||
|
const course = ref(null)
|
||||||
|
const member = ref(null)
|
||||||
|
const router = useRouter()
|
||||||
|
|
||||||
|
const props = defineProps({
|
||||||
|
programName: {
|
||||||
|
type: String,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const program = createDocumentResource({
|
||||||
|
doctype: 'LMS Program',
|
||||||
|
name: props.programName,
|
||||||
|
auto: true,
|
||||||
|
cache: ['program', props.programName],
|
||||||
|
})
|
||||||
|
|
||||||
|
const addProgramCourse = () => {
|
||||||
|
program.setValue.submit(
|
||||||
|
{
|
||||||
|
program_courses: [
|
||||||
|
...program.doc.program_courses,
|
||||||
|
{ course: course.value },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
onSuccess(data) {
|
||||||
|
showDialog.value = false
|
||||||
|
course.value = null
|
||||||
|
showToast(__('Success'), __('Course added to program'), 'check')
|
||||||
|
program.reload()
|
||||||
|
},
|
||||||
|
onError(err) {
|
||||||
|
showToast('Error', err.messages?.[0] || err, 'x')
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const addProgramMember = () => {
|
||||||
|
program.setValue.submit(
|
||||||
|
{
|
||||||
|
program_members: [
|
||||||
|
...program.doc.program_members,
|
||||||
|
{ member: member.value },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
onSuccess(data) {
|
||||||
|
showDialog.value = false
|
||||||
|
member.value = null
|
||||||
|
showToast(__('Success'), __('Member added to program'), 'check')
|
||||||
|
program.reload()
|
||||||
|
},
|
||||||
|
onError(err) {
|
||||||
|
showToast('Error', err.messages?.[0] || err, 'x')
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const remove = (selections, unselectAll, doctype) => {
|
||||||
|
selections = Array.from(selections)
|
||||||
|
program.setValue.submit(
|
||||||
|
{
|
||||||
|
[doctype]: program.doc[doctype].filter(
|
||||||
|
(row) => !selections.includes(row.name)
|
||||||
|
),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
onSuccess(data) {
|
||||||
|
unselectAll()
|
||||||
|
showToast(__('Success'), __('Items removed successfully'), 'check')
|
||||||
|
program.reload()
|
||||||
|
},
|
||||||
|
onError(err) {
|
||||||
|
showToast('Error', err.messages?.[0] || err, 'x')
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const updateOrder = (e) => {
|
||||||
|
let sourceIdx = e.from.dataset.idx
|
||||||
|
let targetIdx = e.to.dataset.idx
|
||||||
|
let courses = program.doc.program_courses
|
||||||
|
courses.splice(targetIdx, 0, courses.splice(sourceIdx, 1)[0])
|
||||||
|
|
||||||
|
courses.forEach((course, index) => {
|
||||||
|
course.idx = index + 1
|
||||||
|
})
|
||||||
|
|
||||||
|
program.setValue.submit(
|
||||||
|
{
|
||||||
|
program_courses: courses,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
onSuccess(data) {
|
||||||
|
showToast(__('Success'), __('Course moved successfully'), 'check')
|
||||||
|
program.reload()
|
||||||
|
},
|
||||||
|
onError(err) {
|
||||||
|
showToast('Error', err.messages?.[0] || err, 'x')
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const saveProgram = () => {
|
||||||
|
call('frappe.model.rename_doc.update_document_title', {
|
||||||
|
doctype: 'LMS Program',
|
||||||
|
docname: program.doc.name,
|
||||||
|
name: program.doc.title,
|
||||||
|
}).then((data) => {
|
||||||
|
router.push({ name: 'ProgramForm', params: { programName: data } })
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const courseColumns = computed(() => {
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
label: 'Title',
|
||||||
|
key: 'course_title',
|
||||||
|
width: 3,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'ID',
|
||||||
|
key: 'course',
|
||||||
|
width: 3,
|
||||||
|
},
|
||||||
|
]
|
||||||
|
})
|
||||||
|
|
||||||
|
const memberColumns = computed(() => {
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
label: 'Member',
|
||||||
|
key: 'member',
|
||||||
|
width: 3,
|
||||||
|
align: 'left',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Full Name',
|
||||||
|
key: 'full_name',
|
||||||
|
width: 3,
|
||||||
|
align: 'left',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Progress (%)',
|
||||||
|
key: 'progress',
|
||||||
|
width: 3,
|
||||||
|
align: 'right',
|
||||||
|
},
|
||||||
|
]
|
||||||
|
})
|
||||||
|
|
||||||
|
const breadbrumbs = computed(() => {
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
label: 'Programs',
|
||||||
|
route: { name: 'Programs' },
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: props.programName === 'new' ? 'New Program' : props.programName,
|
||||||
|
},
|
||||||
|
]
|
||||||
|
})
|
||||||
|
</script>
|
||||||
215
frontend/src/pages/Programs.vue
Normal file
215
frontend/src/pages/Programs.vue
Normal file
@@ -0,0 +1,215 @@
|
|||||||
|
<template>
|
||||||
|
<header
|
||||||
|
class="sticky top-0 z-10 flex items-center justify-between border-b bg-white px-3 py-2.5 sm:px-5"
|
||||||
|
>
|
||||||
|
<Breadcrumbs :items="breadbrumbs" />
|
||||||
|
<Button
|
||||||
|
v-if="user.data?.is_moderator || user.data?.is_instructor"
|
||||||
|
@click="showDialog = true"
|
||||||
|
variant="solid"
|
||||||
|
>
|
||||||
|
<template #prefix>
|
||||||
|
<Plus class="h-4 w-4 stroke-1.5" />
|
||||||
|
</template>
|
||||||
|
{{ __('New') }}
|
||||||
|
</Button>
|
||||||
|
</header>
|
||||||
|
<div v-if="programs.data?.length" class="pt-5 px-5">
|
||||||
|
<div v-for="program in programs.data" class="mb-10">
|
||||||
|
<div class="flex items-center justify-between">
|
||||||
|
<div class="text-xl font-semibold">
|
||||||
|
{{ program.name }}
|
||||||
|
</div>
|
||||||
|
<div class="flex items-center space-x-2">
|
||||||
|
<Badge
|
||||||
|
v-if="program.members"
|
||||||
|
variant="subtle"
|
||||||
|
theme="green"
|
||||||
|
size="lg"
|
||||||
|
>
|
||||||
|
{{ program.members }}
|
||||||
|
{{
|
||||||
|
program.members == 1 ? __(singularize('members')) : __('members')
|
||||||
|
}}
|
||||||
|
</Badge>
|
||||||
|
<Badge
|
||||||
|
v-if="program.progress"
|
||||||
|
variant="subtle"
|
||||||
|
theme="blue"
|
||||||
|
size="lg"
|
||||||
|
>
|
||||||
|
{{ program.progress }}{{ __('% completed') }}
|
||||||
|
</Badge>
|
||||||
|
|
||||||
|
<router-link
|
||||||
|
v-if="user.data?.is_moderator || user.data?.is_instructor"
|
||||||
|
:to="{
|
||||||
|
name: 'ProgramForm',
|
||||||
|
params: { programName: program.name },
|
||||||
|
}"
|
||||||
|
>
|
||||||
|
<Button>
|
||||||
|
<template #prefix>
|
||||||
|
<Edit class="h-4 w-4 stroke-1.5" />
|
||||||
|
</template>
|
||||||
|
{{ __('Edit') }}
|
||||||
|
</Button>
|
||||||
|
</router-link>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
v-if="program.courses?.length"
|
||||||
|
class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-5 mt-5"
|
||||||
|
>
|
||||||
|
<div v-for="course in program.courses" class="relative group">
|
||||||
|
<CourseCard
|
||||||
|
:course="course"
|
||||||
|
@click="enrollMember(program.name, course.name)"
|
||||||
|
class="cursor-pointer"
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
v-if="lockCourse(course)"
|
||||||
|
class="absolute inset-0 bg-black-overlay-500 opacity-60 rounded-md"
|
||||||
|
></div>
|
||||||
|
<div
|
||||||
|
v-if="lockCourse(course)"
|
||||||
|
class="absolute inset-0 flex items-center justify-center"
|
||||||
|
>
|
||||||
|
<LockKeyhole class="size-10 text-white" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div v-else class="text-sm italic text-gray-600 mt-4">
|
||||||
|
{{ __('No courses in this program') }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
v-else
|
||||||
|
class="text-center p-5 text-gray-600 mt-52 w-3/4 md:w-1/2 mx-auto space-y-2"
|
||||||
|
>
|
||||||
|
<BookOpen class="size-10 mx-auto stroke-1 text-gray-500" />
|
||||||
|
<div class="text-xl font-medium">
|
||||||
|
{{ __('No programs found') }}
|
||||||
|
</div>
|
||||||
|
<div class="leading-5">
|
||||||
|
{{
|
||||||
|
__(
|
||||||
|
'There are no programs available at the moment. Keep an eye out, fresh learning experiences are on the way soon!'
|
||||||
|
)
|
||||||
|
}}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Dialog
|
||||||
|
v-model="showDialog"
|
||||||
|
:options="{
|
||||||
|
title: __('New Program'),
|
||||||
|
actions: [
|
||||||
|
{
|
||||||
|
label: __('Create'),
|
||||||
|
variant: 'solid',
|
||||||
|
onClick: () => createProgram(close),
|
||||||
|
},
|
||||||
|
],
|
||||||
|
}"
|
||||||
|
>
|
||||||
|
<template #body-content>
|
||||||
|
<FormControl :label="__('Title')" v-model="title" />
|
||||||
|
</template>
|
||||||
|
</Dialog>
|
||||||
|
</template>
|
||||||
|
<script setup>
|
||||||
|
import {
|
||||||
|
Badge,
|
||||||
|
Breadcrumbs,
|
||||||
|
Button,
|
||||||
|
call,
|
||||||
|
createResource,
|
||||||
|
Dialog,
|
||||||
|
FormControl,
|
||||||
|
} from 'frappe-ui'
|
||||||
|
import { computed, inject, onMounted, ref } from 'vue'
|
||||||
|
import { BookOpen, Edit, Plus, LockKeyhole } from 'lucide-vue-next'
|
||||||
|
import CourseCard from '@/components/CourseCard.vue'
|
||||||
|
import { useRouter } from 'vue-router'
|
||||||
|
import { showToast, singularize } from '@/utils'
|
||||||
|
import { useSettings } from '@/stores/settings'
|
||||||
|
|
||||||
|
const user = inject('$user')
|
||||||
|
const showDialog = ref(false)
|
||||||
|
const router = useRouter()
|
||||||
|
const title = ref('')
|
||||||
|
const settings = useSettings()
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
if (
|
||||||
|
!settings.learningPaths.data &&
|
||||||
|
!user.data?.is_moderator &&
|
||||||
|
!user.data?.is_instructor
|
||||||
|
) {
|
||||||
|
router.push({ name: 'Courses' })
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
const programs = createResource({
|
||||||
|
url: 'lms.lms.utils.get_programs',
|
||||||
|
auto: true,
|
||||||
|
cache: 'programs',
|
||||||
|
})
|
||||||
|
|
||||||
|
const createProgram = (close) => {
|
||||||
|
call('frappe.client.insert', {
|
||||||
|
doc: {
|
||||||
|
doctype: 'LMS Program',
|
||||||
|
title: title.value,
|
||||||
|
},
|
||||||
|
}).then((res) => {
|
||||||
|
router.push({ name: 'ProgramForm', params: { programName: res.name } })
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const enrollMember = (program, course) => {
|
||||||
|
call('lms.lms.utils.enroll_in_program_course', {
|
||||||
|
program: program,
|
||||||
|
course: course,
|
||||||
|
})
|
||||||
|
.then((data) => {
|
||||||
|
if (data.current_lesson) {
|
||||||
|
router.push({
|
||||||
|
name: 'Lesson',
|
||||||
|
params: {
|
||||||
|
courseName: course,
|
||||||
|
chapterNumber: data.current_lesson.split('-')[0],
|
||||||
|
lessonNumber: data.current_lesson.split('-')[1],
|
||||||
|
},
|
||||||
|
})
|
||||||
|
} else if (data) {
|
||||||
|
router.push({
|
||||||
|
name: 'Lesson',
|
||||||
|
params: {
|
||||||
|
courseName: course,
|
||||||
|
chapterNumber: 1,
|
||||||
|
lessonNumber: 1,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch((err) => {
|
||||||
|
showToast('Error', err.messages?.[0] || err, 'x')
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const lockCourse = (course) => {
|
||||||
|
if (user.data?.is_moderator || user.data?.is_instructor) return false
|
||||||
|
if (course.membership) return false
|
||||||
|
if (course.eligible) return false
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
const breadbrumbs = computed(() => [
|
||||||
|
{
|
||||||
|
label: 'Programs',
|
||||||
|
},
|
||||||
|
])
|
||||||
|
</script>
|
||||||
@@ -48,6 +48,7 @@
|
|||||||
? __('Title')
|
? __('Title')
|
||||||
: __('Enter a title and save the quiz to proceed')
|
: __('Enter a title and save the quiz to proceed')
|
||||||
"
|
"
|
||||||
|
:required="true"
|
||||||
/>
|
/>
|
||||||
<div v-if="quizDetails.data?.name">
|
<div v-if="quizDetails.data?.name">
|
||||||
<div class="grid grid-cols-2 gap-5 mt-4 mb-8">
|
<div class="grid grid-cols-2 gap-5 mt-4 mb-8">
|
||||||
@@ -141,6 +142,7 @@
|
|||||||
v-slot="{ idx, column, item }"
|
v-slot="{ idx, column, item }"
|
||||||
v-for="row in quiz.questions"
|
v-for="row in quiz.questions"
|
||||||
@click="openQuestionModal(row)"
|
@click="openQuestionModal(row)"
|
||||||
|
class="cursor-pointer"
|
||||||
>
|
>
|
||||||
<ListRowItem :item="item">
|
<ListRowItem :item="item">
|
||||||
<div
|
<div
|
||||||
@@ -204,7 +206,6 @@ import {
|
|||||||
inject,
|
inject,
|
||||||
onBeforeUnmount,
|
onBeforeUnmount,
|
||||||
watch,
|
watch,
|
||||||
isReactive,
|
|
||||||
} from 'vue'
|
} from 'vue'
|
||||||
import { Plus, Trash2 } from 'lucide-vue-next'
|
import { Plus, Trash2 } from 'lucide-vue-next'
|
||||||
import Question from '@/components/Modals/Question.vue'
|
import Question from '@/components/Modals/Question.vue'
|
||||||
|
|||||||
@@ -15,38 +15,45 @@
|
|||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
</header>
|
</header>
|
||||||
<div v-if="submisisonDetails.doc" class="w-1/2 mx-auto py-5 space-y-4">
|
<div v-if="submisisonDetails.doc" class="w-1/2 mx-auto py-5 space-y-5">
|
||||||
<div class="grid grid-cols-2 gap-5">
|
<div class="text-xl font-semibold">
|
||||||
<FormControl
|
{{ submisisonDetails.doc.member_name }}
|
||||||
v-model="submisisonDetails.doc.quiz_title"
|
|
||||||
:label="__('Quiz')"
|
|
||||||
:disabled="true"
|
|
||||||
/>
|
|
||||||
<FormControl
|
|
||||||
v-model="submisisonDetails.doc.member_name"
|
|
||||||
:label="__('Member')"
|
|
||||||
:disabled="true"
|
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
|
<div class="space-y-4 border p-5 rounded-md">
|
||||||
|
<div class="grid grid-cols-2 gap-5">
|
||||||
|
<FormControl
|
||||||
|
v-model="submisisonDetails.doc.quiz_title"
|
||||||
|
:label="__('Quiz')"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
<FormControl
|
||||||
|
v-model="submisisonDetails.doc.member_name"
|
||||||
|
:label="__('Member')"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="grid grid-cols-2 gap-5">
|
<div class="grid grid-cols-2 gap-5">
|
||||||
<FormControl
|
<FormControl
|
||||||
v-model="submisisonDetails.doc.score"
|
v-model="submisisonDetails.doc.score"
|
||||||
:label="__('Score')"
|
:label="__('Score')"
|
||||||
:disabled="true"
|
:disabled="true"
|
||||||
/>
|
/>
|
||||||
<FormControl
|
<FormControl
|
||||||
v-model="submisisonDetails.doc.percentage"
|
v-model="submisisonDetails.doc.percentage"
|
||||||
:label="__('Percentage')"
|
:label="__('Percentage')"
|
||||||
:disabled="true"
|
:disabled="true"
|
||||||
/>
|
/>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div
|
<div
|
||||||
v-for="row in submisisonDetails.doc.result"
|
v-for="row in submisisonDetails.doc.result"
|
||||||
class="border p-5 rounded-md space-y-4"
|
class="border p-5 rounded-md space-y-4"
|
||||||
>
|
>
|
||||||
<div class="font-semibold">{{ row.idx }}. {{ row.question }}</div>
|
<div class="flex space-x-1 font-semibold">
|
||||||
|
<span class="leading-5" v-html="row.question"> </span>
|
||||||
|
</div>
|
||||||
<div v-html="row.answer" class="leading-5"></div>
|
<div v-html="row.answer" class="leading-5"></div>
|
||||||
<div class="grid grid-cols-2 gap-5">
|
<div class="grid grid-cols-2 gap-5">
|
||||||
<FormControl v-model="row.marks" :label="__('Marks')" />
|
<FormControl v-model="row.marks" :label="__('Marks')" />
|
||||||
@@ -67,7 +74,7 @@ import {
|
|||||||
Button,
|
Button,
|
||||||
Badge,
|
Badge,
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import { computed, onMounted, inject } from 'vue'
|
import { computed, onBeforeUnmount, onMounted, inject } from 'vue'
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
import { showToast } from '@/utils'
|
import { showToast } from '@/utils'
|
||||||
|
|
||||||
@@ -77,8 +84,25 @@ const user = inject('$user')
|
|||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
if (!user.data?.is_instructor && !user.data?.is_moderator)
|
if (!user.data?.is_instructor && !user.data?.is_moderator)
|
||||||
router.push({ name: 'Courses' })
|
router.push({ name: 'Courses' })
|
||||||
|
|
||||||
|
window.addEventListener('keydown', keyboardShortcut)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
onBeforeUnmount(() => {
|
||||||
|
window.removeEventListener('keydown', keyboardShortcut)
|
||||||
|
})
|
||||||
|
|
||||||
|
const keyboardShortcut = (e) => {
|
||||||
|
if (
|
||||||
|
e.key === 's' &&
|
||||||
|
(e.ctrlKey || e.metaKey) &&
|
||||||
|
!e.target.classList.contains('ProseMirror')
|
||||||
|
) {
|
||||||
|
saveSubmission()
|
||||||
|
e.preventDefault()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
submission: {
|
submission: {
|
||||||
type: String,
|
type: String,
|
||||||
|
|||||||
@@ -5,6 +5,9 @@
|
|||||||
<Breadcrumbs :items="breadcrumbs" />
|
<Breadcrumbs :items="breadcrumbs" />
|
||||||
</header>
|
</header>
|
||||||
<div v-if="submissions.data?.length" class="md:w-3/4 md:mx-auto py-5 mx-5">
|
<div v-if="submissions.data?.length" class="md:w-3/4 md:mx-auto py-5 mx-5">
|
||||||
|
<div class="text-xl font-semibold mb-5">
|
||||||
|
{{ submissions.data[0].quiz_title }}
|
||||||
|
</div>
|
||||||
<ListView
|
<ListView
|
||||||
:columns="quizColumns"
|
:columns="quizColumns"
|
||||||
:rows="submissions.data"
|
:rows="submissions.data"
|
||||||
@@ -31,12 +34,18 @@
|
|||||||
</router-link>
|
</router-link>
|
||||||
</ListRows>
|
</ListRows>
|
||||||
</ListView>
|
</ListView>
|
||||||
|
<div class="flex justify-center my-5">
|
||||||
|
<Button v-if="submissions.hasNextPage" @click="submissions.next()">
|
||||||
|
{{ __('Load More') }}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import {
|
import {
|
||||||
createListResource,
|
createListResource,
|
||||||
Breadcrumbs,
|
Breadcrumbs,
|
||||||
|
Button,
|
||||||
ListView,
|
ListView,
|
||||||
ListRow,
|
ListRow,
|
||||||
ListRows,
|
ListRows,
|
||||||
@@ -76,12 +85,7 @@ const quizColumns = computed(() => {
|
|||||||
{
|
{
|
||||||
label: __('Member'),
|
label: __('Member'),
|
||||||
key: 'member_name',
|
key: 'member_name',
|
||||||
width: 2,
|
width: 1,
|
||||||
},
|
|
||||||
{
|
|
||||||
label: __('Quiz'),
|
|
||||||
key: 'quiz_title',
|
|
||||||
width: 2,
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: __('Score'),
|
label: __('Score'),
|
||||||
|
|||||||
@@ -46,6 +46,11 @@
|
|||||||
</router-link>
|
</router-link>
|
||||||
</ListRows>
|
</ListRows>
|
||||||
</ListView>
|
</ListView>
|
||||||
|
<div class="flex justify-center my-5">
|
||||||
|
<Button v-if="quizzes.hasNextPage" @click="quizzes.next()">
|
||||||
|
{{ __('Load More') }}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
v-else
|
v-else
|
||||||
@@ -67,13 +72,13 @@
|
|||||||
<script setup>
|
<script setup>
|
||||||
import {
|
import {
|
||||||
Breadcrumbs,
|
Breadcrumbs,
|
||||||
|
Button,
|
||||||
createListResource,
|
createListResource,
|
||||||
ListView,
|
ListView,
|
||||||
ListRows,
|
ListRows,
|
||||||
ListRow,
|
ListRow,
|
||||||
ListHeader,
|
ListHeader,
|
||||||
ListHeaderItem,
|
ListHeaderItem,
|
||||||
Button,
|
|
||||||
} from 'frappe-ui'
|
} from 'frappe-ui'
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
import { computed, inject, onMounted } from 'vue'
|
import { computed, inject, onMounted } from 'vue'
|
||||||
@@ -103,9 +108,6 @@ const quizzes = createListResource({
|
|||||||
auto: true,
|
auto: true,
|
||||||
cache: ['quizzes', user.data?.name],
|
cache: ['quizzes', user.data?.name],
|
||||||
orderBy: 'modified desc',
|
orderBy: 'modified desc',
|
||||||
onSuccess(data) {
|
|
||||||
data.forEach((row) => {})
|
|
||||||
},
|
|
||||||
})
|
})
|
||||||
|
|
||||||
const quizColumns = computed(() => {
|
const quizColumns = computed(() => {
|
||||||
|
|||||||
208
frontend/src/pages/SCORMChapter.vue
Normal file
208
frontend/src/pages/SCORMChapter.vue
Normal file
@@ -0,0 +1,208 @@
|
|||||||
|
<template>
|
||||||
|
<header
|
||||||
|
class="sticky top-0 z-10 flex items-center justify-between border-b bg-white px-3 py-2.5 sm:px-5"
|
||||||
|
>
|
||||||
|
<Breadcrumbs class="h-7" :items="breadcrumbs" />
|
||||||
|
</header>
|
||||||
|
<div
|
||||||
|
v-if="
|
||||||
|
readyToRender &&
|
||||||
|
(enrollment.data?.length ||
|
||||||
|
user.data?.is_moderator ||
|
||||||
|
user.data?.is_instructor)
|
||||||
|
"
|
||||||
|
>
|
||||||
|
<iframe :src="chapter.doc.launch_file" class="w-full h-screen" />
|
||||||
|
</div>
|
||||||
|
<div v-else-if="!enrollment.data?.length">
|
||||||
|
<div class="text-center pt-10 px-5 md:px-0 pb-10">
|
||||||
|
<div class="text-center">
|
||||||
|
<div class="mb-4">
|
||||||
|
{{
|
||||||
|
__(
|
||||||
|
'You are not enrolled in this course. Please enroll to access this lesson.'
|
||||||
|
)
|
||||||
|
}}
|
||||||
|
</div>
|
||||||
|
<Button variant="solid" @click="enrollStudent()">
|
||||||
|
{{ __('Start Learning') }}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
<script setup>
|
||||||
|
import {
|
||||||
|
Breadcrumbs,
|
||||||
|
Button,
|
||||||
|
call,
|
||||||
|
createDocumentResource,
|
||||||
|
createListResource,
|
||||||
|
createResource,
|
||||||
|
} from 'frappe-ui'
|
||||||
|
import { computed, inject, onBeforeMount, ref } from 'vue'
|
||||||
|
import { useSidebar } from '@/stores/sidebar'
|
||||||
|
import { updateDocumentTitle } from '@/utils'
|
||||||
|
|
||||||
|
const sidebarStore = useSidebar()
|
||||||
|
const user = inject('$user')
|
||||||
|
const readyToRender = ref(false)
|
||||||
|
|
||||||
|
const props = defineProps({
|
||||||
|
courseName: {
|
||||||
|
type: String,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
chapterName: {
|
||||||
|
type: String,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
onBeforeMount(() => {
|
||||||
|
sidebarStore.isSidebarCollapsed = true
|
||||||
|
setupSCORMAPI()
|
||||||
|
})
|
||||||
|
|
||||||
|
const chapter = createDocumentResource({
|
||||||
|
doctype: 'Course Chapter',
|
||||||
|
name: props.chapterName,
|
||||||
|
auto: true,
|
||||||
|
cache: ['chapter', props.chapterName],
|
||||||
|
onSuccess(data) {
|
||||||
|
progress.submit()
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const enrollment = createListResource({
|
||||||
|
doctype: 'LMS Enrollment',
|
||||||
|
fields: ['member', 'course'],
|
||||||
|
filters: {
|
||||||
|
course: props.courseName,
|
||||||
|
member: user.data?.name,
|
||||||
|
},
|
||||||
|
auto: true,
|
||||||
|
cache: ['enrollments', props.courseName, user.data?.name],
|
||||||
|
})
|
||||||
|
|
||||||
|
const getDataFromLMS = (key) => {
|
||||||
|
if (key == 'cmi.core.lesson_status') {
|
||||||
|
if (progress.data?.status == 'Complete') {
|
||||||
|
return 'passed'
|
||||||
|
}
|
||||||
|
return 'incomplete'
|
||||||
|
}
|
||||||
|
return ''
|
||||||
|
}
|
||||||
|
|
||||||
|
const saveDataToLMS = (key, value) => {
|
||||||
|
if (key == 'cmi.core.lesson_status' && value == 'passed') {
|
||||||
|
saveProgress()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const saveProgress = () => {
|
||||||
|
call('lms.lms.doctype.course_lesson.course_lesson.save_progress', {
|
||||||
|
lesson: chapter.doc.lessons[0].lesson,
|
||||||
|
course: props.courseName,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const progress = createResource({
|
||||||
|
url: 'frappe.client.get_value',
|
||||||
|
makeParams(values) {
|
||||||
|
return {
|
||||||
|
doctype: 'LMS Course Progress',
|
||||||
|
fieldname: 'status',
|
||||||
|
filters: {
|
||||||
|
member: user.data?.name,
|
||||||
|
lesson: chapter.doc.lessons[0].lesson,
|
||||||
|
chapter: chapter.doc.name,
|
||||||
|
course: chapter.doc?.course,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
},
|
||||||
|
onSuccess(data) {
|
||||||
|
readyToRender.value = true
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const enrollStudent = () => {
|
||||||
|
enrollment.insert.submit(
|
||||||
|
{
|
||||||
|
course: props.courseName,
|
||||||
|
member: user.data?.name,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
onSuccess(data) {
|
||||||
|
window.location.reload()
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const setupSCORMAPI = () => {
|
||||||
|
window.API_1484_11 = {
|
||||||
|
Initialize: () => 'true',
|
||||||
|
Terminate: () => 'true',
|
||||||
|
GetValue: (key) => {
|
||||||
|
console.log(`GET: ${key}`)
|
||||||
|
return getDataFromLMS(key)
|
||||||
|
},
|
||||||
|
SetValue: (key, value) => {
|
||||||
|
console.log(`SET: ${key} to value: ${value}`)
|
||||||
|
|
||||||
|
saveDataToLMS(key, value)
|
||||||
|
return 'true'
|
||||||
|
},
|
||||||
|
Commit: () => 'true',
|
||||||
|
GetLastError: () => '0',
|
||||||
|
GetErrorString: () => '',
|
||||||
|
GetDiagnostic: () => '',
|
||||||
|
}
|
||||||
|
window.API = {
|
||||||
|
LMSInitialize: () => 'true',
|
||||||
|
LMSFinish: () => 'true',
|
||||||
|
LMSGetValue: (key) => {
|
||||||
|
console.log(`GET: ${key}`)
|
||||||
|
return getDataFromLMS(key)
|
||||||
|
},
|
||||||
|
LMSSetValue: (key, value) => {
|
||||||
|
console.log(`SET: ${key} to value: ${value}`)
|
||||||
|
saveDataToLMS(key, value)
|
||||||
|
return 'true'
|
||||||
|
},
|
||||||
|
LMSCommit: () => 'true',
|
||||||
|
LMSGetLastError: () => '0',
|
||||||
|
LMSGetErrorString: () => '',
|
||||||
|
LMSGetDiagnostic: () => '',
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const breadcrumbs = computed(() => {
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
label: 'Courses',
|
||||||
|
route: { name: 'Courses' },
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: chapter.doc?.course_title,
|
||||||
|
route: { name: 'CourseDetail', params: { courseName: props.courseName } },
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: chapter.doc?.title,
|
||||||
|
},
|
||||||
|
]
|
||||||
|
})
|
||||||
|
|
||||||
|
const pageMeta = computed(() => {
|
||||||
|
return {
|
||||||
|
title: chapter?.doc?.title,
|
||||||
|
description: __('This is a chapter in the course {0}').format(
|
||||||
|
chapter?.doc?.course_title
|
||||||
|
),
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
updateDocumentTitle(pageMeta)
|
||||||
|
</script>
|
||||||
@@ -27,6 +27,12 @@ const routes = [
|
|||||||
component: () => import('@/pages/Lesson.vue'),
|
component: () => import('@/pages/Lesson.vue'),
|
||||||
props: true,
|
props: true,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
path: '/courses/:courseName/learn/:chapterName',
|
||||||
|
name: 'SCORMChapter',
|
||||||
|
component: () => import('@/pages/SCORMChapter.vue'),
|
||||||
|
props: true,
|
||||||
|
},
|
||||||
{
|
{
|
||||||
path: '/batches',
|
path: '/batches',
|
||||||
name: 'Batches',
|
name: 'Batches',
|
||||||
@@ -176,6 +182,17 @@ const routes = [
|
|||||||
component: () => import('@/pages/QuizSubmission.vue'),
|
component: () => import('@/pages/QuizSubmission.vue'),
|
||||||
props: true,
|
props: true,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
path: '/programs/:programName',
|
||||||
|
name: 'ProgramForm',
|
||||||
|
component: () => import('@/pages/ProgramForm.vue'),
|
||||||
|
props: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: '/programs',
|
||||||
|
name: 'Programs',
|
||||||
|
component: () => import('@/pages/Programs.vue'),
|
||||||
|
},
|
||||||
]
|
]
|
||||||
|
|
||||||
let router = createRouter({
|
let router = createRouter({
|
||||||
|
|||||||
@@ -1,12 +1,32 @@
|
|||||||
import { defineStore } from 'pinia'
|
import { defineStore } from 'pinia'
|
||||||
import { ref } from 'vue'
|
import { ref } from 'vue'
|
||||||
|
import { createResource } from 'frappe-ui'
|
||||||
|
|
||||||
export const useSettings = defineStore('settings', () => {
|
export const useSettings = defineStore('settings', () => {
|
||||||
const isSettingsOpen = ref(false)
|
const isSettingsOpen = ref(false)
|
||||||
const activeTab = ref(null)
|
const activeTab = ref(null)
|
||||||
|
const learningPaths = createResource({
|
||||||
|
url: 'frappe.client.get_single_value',
|
||||||
|
makeParams(values) {
|
||||||
|
return {
|
||||||
|
doctype: 'LMS Settings',
|
||||||
|
field: 'enable_learning_paths',
|
||||||
|
}
|
||||||
|
},
|
||||||
|
auto: true,
|
||||||
|
cache: ['learningPaths'],
|
||||||
|
})
|
||||||
|
|
||||||
|
const onboardingDetails = createResource({
|
||||||
|
url: 'lms.lms.utils.is_onboarding_complete',
|
||||||
|
auto: true,
|
||||||
|
cache: ['onboardingDetails'],
|
||||||
|
})
|
||||||
|
|
||||||
return {
|
return {
|
||||||
isSettingsOpen,
|
isSettingsOpen,
|
||||||
activeTab,
|
activeTab,
|
||||||
|
learningPaths,
|
||||||
|
onboardingDetails,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|||||||
10
frontend/src/stores/sidebar.js
Normal file
10
frontend/src/stores/sidebar.js
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
import { defineStore } from 'pinia'
|
||||||
|
import { ref } from 'vue'
|
||||||
|
|
||||||
|
export const useSidebar = defineStore('sidebar', () => {
|
||||||
|
const isSidebarCollapsed = ref(false)
|
||||||
|
|
||||||
|
return {
|
||||||
|
isSidebarCollapsed,
|
||||||
|
}
|
||||||
|
})
|
||||||
@@ -5,6 +5,8 @@ import updateLocale from 'dayjs/esm/plugin/updateLocale'
|
|||||||
import isToday from 'dayjs/esm/plugin/isToday'
|
import isToday from 'dayjs/esm/plugin/isToday'
|
||||||
import isSameOrBefore from 'dayjs/esm/plugin/isSameOrBefore'
|
import isSameOrBefore from 'dayjs/esm/plugin/isSameOrBefore'
|
||||||
import isSameOrAfter from 'dayjs/esm/plugin/isSameOrAfter'
|
import isSameOrAfter from 'dayjs/esm/plugin/isSameOrAfter'
|
||||||
|
import utc from 'dayjs/esm/plugin/utc'
|
||||||
|
import timezone from 'dayjs/esm/plugin/timezone'
|
||||||
|
|
||||||
dayjs.extend(updateLocale)
|
dayjs.extend(updateLocale)
|
||||||
dayjs.extend(relativeTime)
|
dayjs.extend(relativeTime)
|
||||||
@@ -12,5 +14,7 @@ dayjs.extend(localizedFormat)
|
|||||||
dayjs.extend(isToday)
|
dayjs.extend(isToday)
|
||||||
dayjs.extend(isSameOrBefore)
|
dayjs.extend(isSameOrBefore)
|
||||||
dayjs.extend(isSameOrAfter)
|
dayjs.extend(isSameOrAfter)
|
||||||
|
dayjs.extend(utc)
|
||||||
|
dayjs.extend(timezone)
|
||||||
|
|
||||||
export default dayjs
|
export default dayjs
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ import { toast } from 'frappe-ui'
|
|||||||
import { useTimeAgo } from '@vueuse/core'
|
import { useTimeAgo } from '@vueuse/core'
|
||||||
import { Quiz } from '@/utils/quiz'
|
import { Quiz } from '@/utils/quiz'
|
||||||
import { Upload } from '@/utils/upload'
|
import { Upload } from '@/utils/upload'
|
||||||
|
import { Markdown } from '@/utils/markdownParser'
|
||||||
import Header from '@editorjs/header'
|
import Header from '@editorjs/header'
|
||||||
import Paragraph from '@editorjs/paragraph'
|
import Paragraph from '@editorjs/paragraph'
|
||||||
import { CodeBox } from '@/utils/code'
|
import { CodeBox } from '@/utils/code'
|
||||||
@@ -11,6 +12,7 @@ import { watch } from 'vue'
|
|||||||
import dayjs from '@/utils/dayjs'
|
import dayjs from '@/utils/dayjs'
|
||||||
import Embed from '@editorjs/embed'
|
import Embed from '@editorjs/embed'
|
||||||
import SimpleImage from '@editorjs/simple-image'
|
import SimpleImage from '@editorjs/simple-image'
|
||||||
|
import Table from '@editorjs/table'
|
||||||
|
|
||||||
export function createToast(options) {
|
export function createToast(options) {
|
||||||
toast({
|
toast({
|
||||||
@@ -93,7 +95,7 @@ export function showToast(title, text, icon, iconClasses = null) {
|
|||||||
if (!iconClasses) {
|
if (!iconClasses) {
|
||||||
if (icon == 'check') {
|
if (icon == 'check') {
|
||||||
iconClasses = 'bg-green-600 text-white rounded-md p-px'
|
iconClasses = 'bg-green-600 text-white rounded-md p-px'
|
||||||
} else if (icon == 'circle-warn') {
|
} else if (icon == 'alert-circle') {
|
||||||
iconClasses = 'bg-yellow-600 text-white rounded-md p-px'
|
iconClasses = 'bg-yellow-600 text-white rounded-md p-px'
|
||||||
} else {
|
} else {
|
||||||
iconClasses = 'bg-red-600 text-white rounded-md p-px'
|
iconClasses = 'bg-red-600 text-white rounded-md p-px'
|
||||||
@@ -146,10 +148,17 @@ export function htmlToText(html) {
|
|||||||
|
|
||||||
export function getEditorTools() {
|
export function getEditorTools() {
|
||||||
return {
|
return {
|
||||||
header: Header,
|
header: {
|
||||||
|
class: Header,
|
||||||
|
config: {
|
||||||
|
placeholder: 'Header',
|
||||||
|
},
|
||||||
|
},
|
||||||
quiz: Quiz,
|
quiz: Quiz,
|
||||||
upload: Upload,
|
upload: Upload,
|
||||||
|
markdown: Markdown,
|
||||||
image: SimpleImage,
|
image: SimpleImage,
|
||||||
|
table: Table,
|
||||||
paragraph: {
|
paragraph: {
|
||||||
class: Paragraph,
|
class: Paragraph,
|
||||||
inlineToolbar: true,
|
inlineToolbar: true,
|
||||||
|
|||||||
152
frontend/src/utils/markdownParser.js
Normal file
152
frontend/src/utils/markdownParser.js
Normal file
@@ -0,0 +1,152 @@
|
|||||||
|
export class Markdown {
|
||||||
|
constructor({ data, api, readOnly, config }) {
|
||||||
|
this.api = api
|
||||||
|
this.data = data || {}
|
||||||
|
this.config = config || {}
|
||||||
|
this.text = data.text || ''
|
||||||
|
this.readOnly = readOnly
|
||||||
|
}
|
||||||
|
|
||||||
|
static get isReadOnlySupported() {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
static get conversionConfig() {
|
||||||
|
return {
|
||||||
|
export: 'text',
|
||||||
|
import: 'text',
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
this.wrapper = document.createElement('div')
|
||||||
|
this.wrapper.classList.add('cdx-block')
|
||||||
|
this.wrapper.classList.add('ce-paragraph')
|
||||||
|
this.wrapper.innerHTML = this.text
|
||||||
|
|
||||||
|
if (!this.readOnly) {
|
||||||
|
this.wrapper.contentEditable = true
|
||||||
|
this.wrapper.innerHTML = this.text
|
||||||
|
|
||||||
|
this.wrapper.addEventListener('keydown', (event) => {
|
||||||
|
const value = event.target.textContent
|
||||||
|
if (event.keyCode === 32 && value.startsWith('#')) {
|
||||||
|
this.convertToHeader(event, value)
|
||||||
|
} else if (event.keyCode === 13) {
|
||||||
|
this.parseContent(event)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
this.wrapper.addEventListener('paste', (event) =>
|
||||||
|
this.handlePaste(event)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.wrapper
|
||||||
|
}
|
||||||
|
|
||||||
|
convertToHeader(event, value) {
|
||||||
|
event.preventDefault()
|
||||||
|
if (['#', '##', '###', '####', '#####', '######'].includes(value)) {
|
||||||
|
let level = value.length
|
||||||
|
event.target.textContent = ''
|
||||||
|
this.convertBlock('header', {
|
||||||
|
level: level,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
parseContent(event) {
|
||||||
|
event.preventDefault()
|
||||||
|
const previousLine = this.wrapper.textContent
|
||||||
|
if (previousLine && this.hasImage(previousLine)) {
|
||||||
|
this.wrapper.textContent = ''
|
||||||
|
this.convertBlock('image')
|
||||||
|
} else if (previousLine && this.hasLink(previousLine)) {
|
||||||
|
const { text, url } = this.extractLink(previousLine)
|
||||||
|
const anchorTag = `<a href="${url}" target="_blank">${text}</a>`
|
||||||
|
this.convertBlock('paragraph', {
|
||||||
|
text: previousLine.replace(/\[.+?\]\(.+?\)/, anchorTag),
|
||||||
|
})
|
||||||
|
} else if (previousLine && previousLine.startsWith('- ')) {
|
||||||
|
this.convertBlock('list', {
|
||||||
|
style: 'unordered',
|
||||||
|
items: [
|
||||||
|
{
|
||||||
|
content: previousLine.replace('- ', ''),
|
||||||
|
},
|
||||||
|
],
|
||||||
|
})
|
||||||
|
} else if (previousLine && previousLine.startsWith('1. ')) {
|
||||||
|
this.convertBlock('list', {
|
||||||
|
style: 'ordered',
|
||||||
|
items: [
|
||||||
|
{
|
||||||
|
content: previousLine.replace('1. ', ''),
|
||||||
|
},
|
||||||
|
],
|
||||||
|
})
|
||||||
|
} else if (previousLine && this.canBeEmbed(previousLine)) {
|
||||||
|
this.wrapper.textContent = ''
|
||||||
|
this.convertBlock('embed', {
|
||||||
|
source: previousLine,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async convertBlock(type, data, index = null) {
|
||||||
|
const currentIndex = this.api.blocks.getCurrentBlockIndex()
|
||||||
|
const currentBlock = this.api.blocks.getBlockByIndex(currentIndex)
|
||||||
|
await this.api.blocks.convert(currentBlock.id, type, data)
|
||||||
|
this.api.caret.focus(true)
|
||||||
|
}
|
||||||
|
|
||||||
|
handlePaste(event) {
|
||||||
|
event.preventDefault()
|
||||||
|
|
||||||
|
const clipboardData = event.clipboardData || window.clipboardData
|
||||||
|
const pastedText = clipboardData.getData('text/plain')
|
||||||
|
const sanitizedText = this.processPastedContent(pastedText)
|
||||||
|
document.execCommand('insertText', false, sanitizedText)
|
||||||
|
}
|
||||||
|
|
||||||
|
processPastedContent(text) {
|
||||||
|
return text.trim()
|
||||||
|
}
|
||||||
|
|
||||||
|
save(blockContent) {
|
||||||
|
return {
|
||||||
|
text: blockContent.innerHTML,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
hasImage(line) {
|
||||||
|
return /!\[.+?\]\(.+?\)/.test(line)
|
||||||
|
}
|
||||||
|
|
||||||
|
extractImage(line) {
|
||||||
|
const match = line.match(/!\[(.+?)\]\((.+?)\)/)
|
||||||
|
if (match) {
|
||||||
|
return { alt: match[1], url: match[2] }
|
||||||
|
}
|
||||||
|
return { alt: '', url: '' }
|
||||||
|
}
|
||||||
|
|
||||||
|
hasLink(line) {
|
||||||
|
return /\[.+?\]\(.+?\)/.test(line)
|
||||||
|
}
|
||||||
|
|
||||||
|
extractLink(line) {
|
||||||
|
const match = line.match(/\[(.+?)\]\((.+?)\)/)
|
||||||
|
if (match) {
|
||||||
|
return { text: match[1], url: match[2] }
|
||||||
|
}
|
||||||
|
return { text: '', url: '' }
|
||||||
|
}
|
||||||
|
|
||||||
|
canBeEmbed(line) {
|
||||||
|
return /^https?:\/\/.+/.test(line)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default Markdown
|
||||||
@@ -60,6 +60,9 @@ export class Quiz {
|
|||||||
}
|
}
|
||||||
|
|
||||||
renderQuizModal() {
|
renderQuizModal() {
|
||||||
|
if (this.readOnly) {
|
||||||
|
return
|
||||||
|
}
|
||||||
const app = createApp(QuizPlugin, {
|
const app = createApp(QuizPlugin, {
|
||||||
onQuizAddition: (quiz) => {
|
onQuizAddition: (quiz) => {
|
||||||
this.data.quiz = quiz
|
this.data.quiz = quiz
|
||||||
|
|||||||
1522
frontend/yarn.lock
1522
frontend/yarn.lock
File diff suppressed because it is too large
Load Diff
@@ -1 +1 @@
|
|||||||
__version__ = "2.11.0"
|
__version__ = "2.17.0"
|
||||||
|
|||||||
@@ -225,6 +225,7 @@ page_renderer = [
|
|||||||
"lms.page_renderers.ProfileRedirectPage",
|
"lms.page_renderers.ProfileRedirectPage",
|
||||||
"lms.page_renderers.ProfilePage",
|
"lms.page_renderers.ProfilePage",
|
||||||
"lms.page_renderers.CoursePage",
|
"lms.page_renderers.CoursePage",
|
||||||
|
"lms.page_renderers.SCORMRenderer",
|
||||||
]
|
]
|
||||||
|
|
||||||
# set this to "/" to have profiles on the top-level
|
# set this to "/" to have profiles on the top-level
|
||||||
|
|||||||
@@ -66,7 +66,9 @@ def delete_lms_roles():
|
|||||||
|
|
||||||
|
|
||||||
def create_course_creator_role():
|
def create_course_creator_role():
|
||||||
if not frappe.db.exists("Role", "Course Creator"):
|
if frappe.db.exists("Role", "Course Creator"):
|
||||||
|
frappe.db.set_value("Role", "Course Creator", "desk_access", 0)
|
||||||
|
else:
|
||||||
role = frappe.get_doc(
|
role = frappe.get_doc(
|
||||||
{
|
{
|
||||||
"doctype": "Role",
|
"doctype": "Role",
|
||||||
@@ -79,7 +81,9 @@ def create_course_creator_role():
|
|||||||
|
|
||||||
|
|
||||||
def create_moderator_role():
|
def create_moderator_role():
|
||||||
if not frappe.db.exists("Role", "Moderator"):
|
if frappe.db.exists("Role", "Moderator"):
|
||||||
|
frappe.db.set_value("Role", "Moderator", "desk_access", 0)
|
||||||
|
else:
|
||||||
role = frappe.get_doc(
|
role = frappe.get_doc(
|
||||||
{
|
{
|
||||||
"doctype": "Role",
|
"doctype": "Role",
|
||||||
@@ -92,7 +96,9 @@ def create_moderator_role():
|
|||||||
|
|
||||||
|
|
||||||
def create_evaluator_role():
|
def create_evaluator_role():
|
||||||
if not frappe.db.exists("Role", "Batch Evaluator"):
|
if frappe.db.exists("Role", "Batch Evaluator"):
|
||||||
|
frappe.db.set_value("Role", "Batch Evaluator", "desk_access", 0)
|
||||||
|
else:
|
||||||
role = frappe.new_doc("Role")
|
role = frappe.new_doc("Role")
|
||||||
role.update(
|
role.update(
|
||||||
{
|
{
|
||||||
@@ -105,7 +111,9 @@ def create_evaluator_role():
|
|||||||
|
|
||||||
|
|
||||||
def create_lms_student_role():
|
def create_lms_student_role():
|
||||||
if not frappe.db.exists("Role", "LMS Student"):
|
if frappe.db.exists("Role", "LMS Student"):
|
||||||
|
frappe.db.set_value("Role", "LMS Student", "desk_access", 0)
|
||||||
|
else:
|
||||||
role = frappe.new_doc("Role")
|
role = frappe.new_doc("Role")
|
||||||
role.update(
|
role.update(
|
||||||
{
|
{
|
||||||
|
|||||||
160
lms/lms/api.py
160
lms/lms/api.py
@@ -3,6 +3,12 @@
|
|||||||
|
|
||||||
import json
|
import json
|
||||||
import frappe
|
import frappe
|
||||||
|
import zipfile
|
||||||
|
import os
|
||||||
|
import re
|
||||||
|
import shutil
|
||||||
|
import requests
|
||||||
|
import xml.etree.ElementTree as ET
|
||||||
from frappe.translate import get_all_translations
|
from frappe.translate import get_all_translations
|
||||||
from frappe import _
|
from frappe import _
|
||||||
from frappe.query_builder import DocType
|
from frappe.query_builder import DocType
|
||||||
@@ -10,6 +16,7 @@ from frappe.query_builder.functions import Count
|
|||||||
from frappe.utils import time_diff, now_datetime, get_datetime, flt
|
from frappe.utils import time_diff, now_datetime, get_datetime, flt
|
||||||
from typing import Optional
|
from typing import Optional
|
||||||
from lms.lms.utils import get_average_rating, get_lesson_count
|
from lms.lms.utils import get_average_rating, get_lesson_count
|
||||||
|
from xml.dom.minidom import parseString
|
||||||
|
|
||||||
|
|
||||||
@frappe.whitelist()
|
@frappe.whitelist()
|
||||||
@@ -585,7 +592,7 @@ def get_categories(doctype, filters):
|
|||||||
def get_members(start=0, search=""):
|
def get_members(start=0, search=""):
|
||||||
"""Get members for the given search term and start index.
|
"""Get members for the given search term and start index.
|
||||||
Args: start (int): Start index for the query.
|
Args: start (int): Start index for the query.
|
||||||
search (str): Search term to filter the results.
|
search (str): Search term to filter the results.
|
||||||
Returns: List of members.
|
Returns: List of members.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@@ -834,8 +841,6 @@ def delete_course(course):
|
|||||||
frappe.delete_doc("Lesson Reference", lesson)
|
frappe.delete_doc("Lesson Reference", lesson)
|
||||||
|
|
||||||
for lesson in lessons:
|
for lesson in lessons:
|
||||||
frappe.db.delete("LMS Course Progress", {"lesson": lesson})
|
|
||||||
|
|
||||||
topics = frappe.get_all(
|
topics = frappe.get_all(
|
||||||
"Discussion Topic",
|
"Discussion Topic",
|
||||||
{"reference_doctype": "Course Lesson", "reference_docname": lesson},
|
{"reference_doctype": "Course Lesson", "reference_docname": lesson},
|
||||||
@@ -855,6 +860,9 @@ def delete_course(course):
|
|||||||
for chapter in chapters:
|
for chapter in chapters:
|
||||||
frappe.delete_doc("Course Chapter", chapter)
|
frappe.delete_doc("Course Chapter", chapter)
|
||||||
|
|
||||||
|
frappe.db.delete("LMS Course Progress", {"course": course})
|
||||||
|
frappe.db.delete("LMS Quiz", {"course": course})
|
||||||
|
frappe.db.delete("LMS Quiz Submission", {"course": course})
|
||||||
frappe.db.delete("LMS Enrollment", {"course": course})
|
frappe.db.delete("LMS Enrollment", {"course": course})
|
||||||
frappe.delete_doc("LMS Course", course)
|
frappe.delete_doc("LMS Course", course)
|
||||||
|
|
||||||
@@ -876,3 +884,149 @@ def give_dicussions_permission():
|
|||||||
"delete": 1,
|
"delete": 1,
|
||||||
}
|
}
|
||||||
).save(ignore_permissions=True)
|
).save(ignore_permissions=True)
|
||||||
|
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
|
def upsert_chapter(title, course, is_scorm_package, scorm_package, name=None):
|
||||||
|
values = frappe._dict(
|
||||||
|
{"title": title, "course": course, "is_scorm_package": is_scorm_package}
|
||||||
|
)
|
||||||
|
|
||||||
|
if is_scorm_package:
|
||||||
|
scorm_package = frappe._dict(scorm_package)
|
||||||
|
extract_path = extract_package(course, title, scorm_package)
|
||||||
|
|
||||||
|
values.update(
|
||||||
|
{
|
||||||
|
"scorm_package": scorm_package.name,
|
||||||
|
"scorm_package_path": extract_path.split("public")[1],
|
||||||
|
"manifest_file": get_manifest_file(extract_path).split("public")[1],
|
||||||
|
"launch_file": get_launch_file(extract_path).split("public")[1],
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
if name:
|
||||||
|
chapter = frappe.get_doc("Course Chapter", name)
|
||||||
|
else:
|
||||||
|
chapter = frappe.new_doc("Course Chapter")
|
||||||
|
|
||||||
|
chapter.update(values)
|
||||||
|
chapter.save()
|
||||||
|
|
||||||
|
if is_scorm_package and not len(chapter.lessons):
|
||||||
|
add_lesson(title, chapter.name, course)
|
||||||
|
|
||||||
|
return chapter
|
||||||
|
|
||||||
|
|
||||||
|
def extract_package(course, title, scorm_package):
|
||||||
|
package = frappe.get_doc("File", scorm_package.name)
|
||||||
|
zip_path = package.get_full_path()
|
||||||
|
# check_for_malicious_code(zip_path)
|
||||||
|
extract_path = frappe.get_site_path("public", "scorm", course, title)
|
||||||
|
zipfile.ZipFile(zip_path).extractall(extract_path)
|
||||||
|
return extract_path
|
||||||
|
|
||||||
|
|
||||||
|
def check_for_malicious_code(zip_path):
|
||||||
|
suspicious_patterns = [
|
||||||
|
# Unsafe inline JavaScript
|
||||||
|
r'on(click|load|mouseover|error|submit|focus|blur|change|keyup|keydown|keypress|resize)=".*?"', # Inline event handlers (e.g., onerror, onclick)
|
||||||
|
r'<script.*?src=["\']http', # External script tags
|
||||||
|
r"eval\(", # Usage of eval()
|
||||||
|
r"Function\(", # Usage of Function constructor
|
||||||
|
r"(btoa|atob)\(", # Base64 encoding/decoding
|
||||||
|
# Dangerous XML patterns
|
||||||
|
r"<!ENTITY", # XXE-related
|
||||||
|
r"<\?xml-stylesheet .*?>", # External stylesheets in XML
|
||||||
|
]
|
||||||
|
|
||||||
|
with zipfile.ZipFile(zip_path, "r") as zf:
|
||||||
|
for file_name in zf.namelist():
|
||||||
|
if file_name.endswith((".html", ".js", ".xml")):
|
||||||
|
with zf.open(file_name) as file:
|
||||||
|
content = file.read().decode("utf-8", errors="ignore")
|
||||||
|
for pattern in suspicious_patterns:
|
||||||
|
if re.search(pattern, content):
|
||||||
|
frappe.throw(
|
||||||
|
_("Suspicious pattern found in {0}: {1}").format(file_name, pattern)
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def get_manifest_file(extract_path):
|
||||||
|
manifest_file = None
|
||||||
|
for root, dirs, files in os.walk(extract_path):
|
||||||
|
for file in files:
|
||||||
|
if file == "imsmanifest.xml":
|
||||||
|
manifest_file = os.path.join(root, file)
|
||||||
|
break
|
||||||
|
if manifest_file:
|
||||||
|
break
|
||||||
|
return manifest_file
|
||||||
|
|
||||||
|
|
||||||
|
def get_launch_file(extract_path):
|
||||||
|
launch_file = None
|
||||||
|
manifest_file = get_manifest_file(extract_path)
|
||||||
|
|
||||||
|
if manifest_file:
|
||||||
|
with open(manifest_file) as file:
|
||||||
|
data = file.read()
|
||||||
|
dom = parseString(data)
|
||||||
|
resource = dom.getElementsByTagName("resource")
|
||||||
|
for res in resource:
|
||||||
|
if (
|
||||||
|
res.getAttribute("adlcp:scormtype") == "sco"
|
||||||
|
or res.getAttribute("adlcp:scormType") == "sco"
|
||||||
|
):
|
||||||
|
launch_file = res.getAttribute("href")
|
||||||
|
break
|
||||||
|
|
||||||
|
if launch_file:
|
||||||
|
launch_file = os.path.join(os.path.dirname(manifest_file), launch_file)
|
||||||
|
|
||||||
|
return launch_file
|
||||||
|
|
||||||
|
|
||||||
|
def add_lesson(title, chapter, course):
|
||||||
|
lesson = frappe.new_doc("Course Lesson")
|
||||||
|
lesson.update(
|
||||||
|
{
|
||||||
|
"title": title,
|
||||||
|
"chapter": chapter,
|
||||||
|
"course": course,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
lesson.insert()
|
||||||
|
|
||||||
|
lesson_reference = frappe.new_doc("Lesson Reference")
|
||||||
|
lesson_reference.update(
|
||||||
|
{
|
||||||
|
"lesson": lesson.name,
|
||||||
|
"parent": chapter,
|
||||||
|
"parenttype": "Course Chapter",
|
||||||
|
"parentfield": "lessons",
|
||||||
|
}
|
||||||
|
)
|
||||||
|
lesson_reference.insert()
|
||||||
|
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
|
def delete_chapter(chapter):
|
||||||
|
chapterInfo = frappe.db.get_value(
|
||||||
|
"Course Chapter", chapter, ["is_scorm_package", "scorm_package_path"], as_dict=True
|
||||||
|
)
|
||||||
|
|
||||||
|
if chapterInfo.is_scorm_package:
|
||||||
|
delete_scorm_package(chapterInfo.scorm_package_path)
|
||||||
|
|
||||||
|
frappe.db.delete("Chapter Reference", {"chapter": chapter})
|
||||||
|
frappe.db.delete("Lesson Reference", {"parent": chapter})
|
||||||
|
frappe.db.delete("Course Lesson", {"chapter": chapter})
|
||||||
|
frappe.db.delete("Course Chapter", chapter)
|
||||||
|
|
||||||
|
|
||||||
|
def delete_scorm_package(scorm_package_path):
|
||||||
|
scorm_package_path = frappe.get_site_path("public", scorm_package_path[1:])
|
||||||
|
if os.path.exists(scorm_package_path):
|
||||||
|
shutil.rmtree(scorm_package_path)
|
||||||
|
|||||||
@@ -8,9 +8,17 @@
|
|||||||
"editable_grid": 1,
|
"editable_grid": 1,
|
||||||
"engine": "InnoDB",
|
"engine": "InnoDB",
|
||||||
"field_order": [
|
"field_order": [
|
||||||
"course",
|
|
||||||
"column_break_3",
|
|
||||||
"title",
|
"title",
|
||||||
|
"column_break_3",
|
||||||
|
"course",
|
||||||
|
"course_title",
|
||||||
|
"scorm_section",
|
||||||
|
"is_scorm_package",
|
||||||
|
"scorm_package",
|
||||||
|
"scorm_package_path",
|
||||||
|
"column_break_dlnw",
|
||||||
|
"manifest_file",
|
||||||
|
"launch_file",
|
||||||
"section_break_5",
|
"section_break_5",
|
||||||
"lessons"
|
"lessons"
|
||||||
],
|
],
|
||||||
@@ -43,6 +51,56 @@
|
|||||||
"fieldtype": "Table",
|
"fieldtype": "Table",
|
||||||
"label": "Lessons",
|
"label": "Lessons",
|
||||||
"options": "Lesson Reference"
|
"options": "Lesson Reference"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"default": "0",
|
||||||
|
"fieldname": "is_scorm_package",
|
||||||
|
"fieldtype": "Check",
|
||||||
|
"label": "Is SCORM Package"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"depends_on": "is_scorm_package",
|
||||||
|
"fieldname": "manifest_file",
|
||||||
|
"fieldtype": "Code",
|
||||||
|
"label": "Manifest File",
|
||||||
|
"read_only": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"depends_on": "is_scorm_package",
|
||||||
|
"fieldname": "launch_file",
|
||||||
|
"fieldtype": "Code",
|
||||||
|
"label": "Launch File",
|
||||||
|
"read_only": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "scorm_section",
|
||||||
|
"fieldtype": "Section Break",
|
||||||
|
"label": "SCORM"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "scorm_package",
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"label": "SCORM Package",
|
||||||
|
"options": "File",
|
||||||
|
"read_only": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "column_break_dlnw",
|
||||||
|
"fieldtype": "Column Break"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"depends_on": "is_scorm_package",
|
||||||
|
"fieldname": "scorm_package_path",
|
||||||
|
"fieldtype": "Code",
|
||||||
|
"label": "SCORM Package Path",
|
||||||
|
"read_only": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fetch_from": "course.title",
|
||||||
|
"fieldname": "course_title",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"label": "Course Title",
|
||||||
|
"read_only": 1
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"index_web_pages_for_search": 1,
|
"index_web_pages_for_search": 1,
|
||||||
@@ -53,7 +111,7 @@
|
|||||||
"link_fieldname": "chapter"
|
"link_fieldname": "chapter"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"modified": "2024-10-29 16:54:20.904683",
|
"modified": "2024-11-15 12:03:31.370943",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "LMS",
|
"module": "LMS",
|
||||||
"name": "Course Chapter",
|
"name": "Course Chapter",
|
||||||
@@ -73,17 +131,14 @@
|
|||||||
"write": 1
|
"write": 1
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"create": 1,
|
|
||||||
"email": 1,
|
"email": 1,
|
||||||
"export": 1,
|
"export": 1,
|
||||||
"if_owner": 1,
|
|
||||||
"print": 1,
|
"print": 1,
|
||||||
"read": 1,
|
"read": 1,
|
||||||
"report": 1,
|
"report": 1,
|
||||||
"role": "LMS Student",
|
"role": "LMS Student",
|
||||||
"select": 1,
|
"select": 1,
|
||||||
"share": 1,
|
"share": 1
|
||||||
"write": 1
|
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"search_fields": "title",
|
"search_fields": "title",
|
||||||
|
|||||||
@@ -8,12 +8,18 @@
|
|||||||
"editable_grid": 1,
|
"editable_grid": 1,
|
||||||
"engine": "InnoDB",
|
"engine": "InnoDB",
|
||||||
"field_order": [
|
"field_order": [
|
||||||
"chapter",
|
|
||||||
"course",
|
|
||||||
"column_break_4",
|
|
||||||
"title",
|
"title",
|
||||||
"include_in_preview",
|
"include_in_preview",
|
||||||
"index_label",
|
"column_break_4",
|
||||||
|
"chapter",
|
||||||
|
"is_scorm_package",
|
||||||
|
"course",
|
||||||
|
"section_break_11",
|
||||||
|
"content",
|
||||||
|
"body",
|
||||||
|
"column_break_cjmf",
|
||||||
|
"instructor_content",
|
||||||
|
"instructor_notes",
|
||||||
"section_break_6",
|
"section_break_6",
|
||||||
"youtube",
|
"youtube",
|
||||||
"column_break_9",
|
"column_break_9",
|
||||||
@@ -22,13 +28,7 @@
|
|||||||
"question",
|
"question",
|
||||||
"column_break_15",
|
"column_break_15",
|
||||||
"file_type",
|
"file_type",
|
||||||
"section_break_11",
|
"column_break_syza",
|
||||||
"content",
|
|
||||||
"body",
|
|
||||||
"column_break_cjmf",
|
|
||||||
"instructor_content",
|
|
||||||
"instructor_notes",
|
|
||||||
"help_section",
|
|
||||||
"help"
|
"help"
|
||||||
],
|
],
|
||||||
"fields": [
|
"fields": [
|
||||||
@@ -59,12 +59,6 @@
|
|||||||
"label": "Title",
|
"label": "Title",
|
||||||
"reqd": 1
|
"reqd": 1
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"fieldname": "index_label",
|
|
||||||
"fieldtype": "Data",
|
|
||||||
"label": "Index Label",
|
|
||||||
"read_only": 1
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"fieldname": "section_break_6",
|
"fieldname": "section_break_6",
|
||||||
"fieldtype": "Section Break",
|
"fieldtype": "Section Break",
|
||||||
@@ -74,14 +68,7 @@
|
|||||||
"fieldname": "body",
|
"fieldname": "body",
|
||||||
"fieldtype": "Markdown Editor",
|
"fieldtype": "Markdown Editor",
|
||||||
"ignore_xss_filter": 1,
|
"ignore_xss_filter": 1,
|
||||||
"label": "Body",
|
"label": "Body"
|
||||||
"reqd": 1
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"fieldname": "help_section",
|
|
||||||
"fieldtype": "Section Break",
|
|
||||||
"hidden": 1,
|
|
||||||
"label": "Help"
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"fieldname": "help",
|
"fieldname": "help",
|
||||||
@@ -158,11 +145,23 @@
|
|||||||
"fieldname": "instructor_content",
|
"fieldname": "instructor_content",
|
||||||
"fieldtype": "Text",
|
"fieldtype": "Text",
|
||||||
"label": "Instructor Content"
|
"label": "Instructor Content"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "column_break_syza",
|
||||||
|
"fieldtype": "Column Break"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"default": "0",
|
||||||
|
"fetch_from": "chapter.is_scorm_package",
|
||||||
|
"fieldname": "is_scorm_package",
|
||||||
|
"fieldtype": "Check",
|
||||||
|
"label": "Is SCORM Package",
|
||||||
|
"read_only": 1
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"index_web_pages_for_search": 1,
|
"index_web_pages_for_search": 1,
|
||||||
"links": [],
|
"links": [],
|
||||||
"modified": "2024-10-08 11:04:54.748773",
|
"modified": "2024-11-14 13:46:56.838659",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "LMS",
|
"module": "LMS",
|
||||||
"name": "Course Lesson",
|
"name": "Course Lesson",
|
||||||
|
|||||||
@@ -52,7 +52,6 @@ class CourseLesson(Document):
|
|||||||
ex.lesson = None
|
ex.lesson = None
|
||||||
ex.course = None
|
ex.course = None
|
||||||
ex.index_ = 0
|
ex.index_ = 0
|
||||||
ex.index_label = ""
|
|
||||||
ex.save(ignore_permissions=True)
|
ex.save(ignore_permissions=True)
|
||||||
|
|
||||||
def check_and_create_folder(self):
|
def check_and_create_folder(self):
|
||||||
@@ -94,15 +93,15 @@ def save_progress(lesson, course):
|
|||||||
|
|
||||||
frappe.db.set_value("LMS Enrollment", membership, "current_lesson", lesson)
|
frappe.db.set_value("LMS Enrollment", membership, "current_lesson", lesson)
|
||||||
|
|
||||||
quiz_completed = get_quiz_progress(lesson)
|
|
||||||
if not quiz_completed:
|
|
||||||
return 0
|
|
||||||
|
|
||||||
if frappe.db.exists(
|
if frappe.db.exists(
|
||||||
"LMS Course Progress", {"lesson": lesson, "member": frappe.session.user}
|
"LMS Course Progress", {"lesson": lesson, "member": frappe.session.user}
|
||||||
):
|
):
|
||||||
return
|
return
|
||||||
|
|
||||||
|
quiz_completed = get_quiz_progress(lesson)
|
||||||
|
if not quiz_completed:
|
||||||
|
return 0
|
||||||
|
|
||||||
frappe.get_doc(
|
frappe.get_doc(
|
||||||
{
|
{
|
||||||
"doctype": "LMS Course Progress",
|
"doctype": "LMS Course Progress",
|
||||||
|
|||||||
@@ -193,13 +193,15 @@
|
|||||||
"depends_on": "paid_batch",
|
"depends_on": "paid_batch",
|
||||||
"fieldname": "amount",
|
"fieldname": "amount",
|
||||||
"fieldtype": "Currency",
|
"fieldtype": "Currency",
|
||||||
"label": "Amount"
|
"label": "Amount",
|
||||||
|
"mandatory_depends_on": "paid_batch"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"depends_on": "paid_batch",
|
"depends_on": "paid_batch",
|
||||||
"fieldname": "currency",
|
"fieldname": "currency",
|
||||||
"fieldtype": "Link",
|
"fieldtype": "Link",
|
||||||
"label": "Currency",
|
"label": "Currency",
|
||||||
|
"mandatory_depends_on": "paid_batch",
|
||||||
"options": "Currency"
|
"options": "Currency"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -328,7 +330,7 @@
|
|||||||
],
|
],
|
||||||
"index_web_pages_for_search": 1,
|
"index_web_pages_for_search": 1,
|
||||||
"links": [],
|
"links": [],
|
||||||
"modified": "2024-07-18 18:06:37.229885",
|
"modified": "2024-11-18 16:28:41.336928",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "LMS",
|
"module": "LMS",
|
||||||
"name": "LMS Batch",
|
"name": "LMS Batch",
|
||||||
|
|||||||
@@ -28,6 +28,7 @@ class LMSBatch(Document):
|
|||||||
self.validate_duplicate_courses()
|
self.validate_duplicate_courses()
|
||||||
self.validate_duplicate_students()
|
self.validate_duplicate_students()
|
||||||
self.validate_payments_app()
|
self.validate_payments_app()
|
||||||
|
self.validate_amount_and_currency()
|
||||||
self.validate_duplicate_assessments()
|
self.validate_duplicate_assessments()
|
||||||
self.validate_membership()
|
self.validate_membership()
|
||||||
self.validate_timetable()
|
self.validate_timetable()
|
||||||
@@ -64,6 +65,10 @@ class LMSBatch(Document):
|
|||||||
if "payments" not in installed_apps:
|
if "payments" not in installed_apps:
|
||||||
frappe.throw(_("Please install the Payments app to create a paid batches."))
|
frappe.throw(_("Please install the Payments app to create a paid batches."))
|
||||||
|
|
||||||
|
def validate_amount_and_currency(self):
|
||||||
|
if self.paid_batch and (not self.amount or not self.currency):
|
||||||
|
frappe.throw(_("Amount and currency are required for paid batches."))
|
||||||
|
|
||||||
def validate_duplicate_assessments(self):
|
def validate_duplicate_assessments(self):
|
||||||
assessments = [row.assessment_name for row in self.assessment]
|
assessments = [row.assessment_name for row in self.assessment]
|
||||||
for assessment in self.assessment:
|
for assessment in self.assessment:
|
||||||
|
|||||||
@@ -114,7 +114,7 @@
|
|||||||
],
|
],
|
||||||
"index_web_pages_for_search": 1,
|
"index_web_pages_for_search": 1,
|
||||||
"links": [],
|
"links": [],
|
||||||
"modified": "2024-09-11 11:37:20.419955",
|
"modified": "2024-09-11 11:37:20.419956",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "LMS",
|
"module": "LMS",
|
||||||
"name": "LMS Certificate",
|
"name": "LMS Certificate",
|
||||||
|
|||||||
@@ -19,6 +19,7 @@ class LMSCourse(Document):
|
|||||||
self.validate_video_link()
|
self.validate_video_link()
|
||||||
self.validate_status()
|
self.validate_status()
|
||||||
self.validate_payments_app()
|
self.validate_payments_app()
|
||||||
|
self.validate_amount_and_currency()
|
||||||
self.image = validate_image(self.image)
|
self.image = validate_image(self.image)
|
||||||
|
|
||||||
def validate_published(self):
|
def validate_published(self):
|
||||||
@@ -51,6 +52,10 @@ class LMSCourse(Document):
|
|||||||
if "payments" not in installed_apps:
|
if "payments" not in installed_apps:
|
||||||
frappe.throw(_("Please install the Payments app to create a paid courses."))
|
frappe.throw(_("Please install the Payments app to create a paid courses."))
|
||||||
|
|
||||||
|
def validate_amount_and_currency(self):
|
||||||
|
if self.paid_course and (not self.course_price and not self.currency):
|
||||||
|
frappe.throw(_("Amount and currency are required for paid courses."))
|
||||||
|
|
||||||
def on_update(self):
|
def on_update(self):
|
||||||
if not self.upcoming and self.has_value_changed("upcoming"):
|
if not self.upcoming and self.has_value_changed("upcoming"):
|
||||||
self.send_email_to_interested_users()
|
self.send_email_to_interested_users()
|
||||||
|
|||||||
@@ -4,6 +4,7 @@
|
|||||||
import frappe
|
import frappe
|
||||||
from frappe import _
|
from frappe import _
|
||||||
from frappe.model.document import Document
|
from frappe.model.document import Document
|
||||||
|
from frappe.utils import ceil
|
||||||
|
|
||||||
|
|
||||||
class LMSEnrollment(Document):
|
class LMSEnrollment(Document):
|
||||||
@@ -11,6 +12,9 @@ class LMSEnrollment(Document):
|
|||||||
self.validate_membership_in_same_batch()
|
self.validate_membership_in_same_batch()
|
||||||
self.validate_membership_in_different_batch_same_course()
|
self.validate_membership_in_different_batch_same_course()
|
||||||
|
|
||||||
|
def on_update(self):
|
||||||
|
self.update_program_progress()
|
||||||
|
|
||||||
def validate_membership_in_same_batch(self):
|
def validate_membership_in_same_batch(self):
|
||||||
filters = {"member": self.member, "course": self.course, "name": ["!=", self.name]}
|
filters = {"member": self.member, "course": self.course, "name": ["!=", self.name]}
|
||||||
if self.batch_old:
|
if self.batch_old:
|
||||||
@@ -55,6 +59,26 @@ class LMSEnrollment(Document):
|
|||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def update_program_progress(self):
|
||||||
|
programs = frappe.get_all(
|
||||||
|
"LMS Program Member", {"member": self.member}, ["parent", "name"]
|
||||||
|
)
|
||||||
|
|
||||||
|
for program in programs:
|
||||||
|
total_progress = 0
|
||||||
|
courses = frappe.get_all(
|
||||||
|
"LMS Program Course", {"parent": program.parent}, pluck="course"
|
||||||
|
)
|
||||||
|
for course in courses:
|
||||||
|
progress = frappe.db.get_value(
|
||||||
|
"LMS Enrollment", {"course": course, "member": self.member}, "progress"
|
||||||
|
)
|
||||||
|
progress = progress or 0
|
||||||
|
total_progress += progress
|
||||||
|
|
||||||
|
average_progress = ceil(total_progress / len(courses))
|
||||||
|
frappe.db.set_value("LMS Program Member", program.name, "progress", average_progress)
|
||||||
|
|
||||||
|
|
||||||
@frappe.whitelist()
|
@frappe.whitelist()
|
||||||
def create_membership(
|
def create_membership(
|
||||||
|
|||||||
0
lms/lms/doctype/lms_program/__init__.py
Normal file
0
lms/lms/doctype/lms_program/__init__.py
Normal file
8
lms/lms/doctype/lms_program/lms_program.js
Normal file
8
lms/lms/doctype/lms_program/lms_program.js
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
// Copyright (c) 2024, Frappe and contributors
|
||||||
|
// For license information, please see license.txt
|
||||||
|
|
||||||
|
// frappe.ui.form.on("LMS Program", {
|
||||||
|
// refresh(frm) {
|
||||||
|
|
||||||
|
// },
|
||||||
|
// });
|
||||||
85
lms/lms/doctype/lms_program/lms_program.json
Normal file
85
lms/lms/doctype/lms_program/lms_program.json
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
{
|
||||||
|
"actions": [],
|
||||||
|
"allow_rename": 1,
|
||||||
|
"autoname": "field:title",
|
||||||
|
"creation": "2024-11-18 12:27:13.283169",
|
||||||
|
"doctype": "DocType",
|
||||||
|
"engine": "InnoDB",
|
||||||
|
"field_order": [
|
||||||
|
"title",
|
||||||
|
"program_courses",
|
||||||
|
"program_members"
|
||||||
|
],
|
||||||
|
"fields": [
|
||||||
|
{
|
||||||
|
"fieldname": "program_courses",
|
||||||
|
"fieldtype": "Table",
|
||||||
|
"label": "Program Courses",
|
||||||
|
"options": "LMS Program Course"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "program_members",
|
||||||
|
"fieldtype": "Table",
|
||||||
|
"label": "Program Members",
|
||||||
|
"options": "LMS Program Member"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "title",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"in_list_view": 1,
|
||||||
|
"label": "Title",
|
||||||
|
"reqd": 1,
|
||||||
|
"unique": 1
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"index_web_pages_for_search": 1,
|
||||||
|
"links": [],
|
||||||
|
"modified": "2024-11-28 22:06:16.742867",
|
||||||
|
"modified_by": "Administrator",
|
||||||
|
"module": "LMS",
|
||||||
|
"name": "LMS Program",
|
||||||
|
"naming_rule": "By fieldname",
|
||||||
|
"owner": "Administrator",
|
||||||
|
"permissions": [
|
||||||
|
{
|
||||||
|
"create": 1,
|
||||||
|
"delete": 1,
|
||||||
|
"email": 1,
|
||||||
|
"export": 1,
|
||||||
|
"print": 1,
|
||||||
|
"read": 1,
|
||||||
|
"report": 1,
|
||||||
|
"role": "System Manager",
|
||||||
|
"share": 1,
|
||||||
|
"write": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"create": 1,
|
||||||
|
"delete": 1,
|
||||||
|
"email": 1,
|
||||||
|
"export": 1,
|
||||||
|
"print": 1,
|
||||||
|
"read": 1,
|
||||||
|
"report": 1,
|
||||||
|
"role": "Moderator",
|
||||||
|
"share": 1,
|
||||||
|
"write": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"create": 1,
|
||||||
|
"delete": 1,
|
||||||
|
"email": 1,
|
||||||
|
"export": 1,
|
||||||
|
"print": 1,
|
||||||
|
"read": 1,
|
||||||
|
"report": 1,
|
||||||
|
"role": "Course Creator",
|
||||||
|
"share": 1,
|
||||||
|
"write": 1
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"sort_field": "creation",
|
||||||
|
"sort_order": "DESC",
|
||||||
|
"states": [],
|
||||||
|
"track_changes": 1
|
||||||
|
}
|
||||||
32
lms/lms/doctype/lms_program/lms_program.py
Normal file
32
lms/lms/doctype/lms_program/lms_program.py
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
# Copyright (c) 2024, Frappe and contributors
|
||||||
|
# For license information, please see license.txt
|
||||||
|
|
||||||
|
import frappe
|
||||||
|
from frappe import _
|
||||||
|
from frappe.model.document import Document
|
||||||
|
|
||||||
|
|
||||||
|
class LMSProgram(Document):
|
||||||
|
def validate(self):
|
||||||
|
self.validate_program_courses()
|
||||||
|
self.validate_program_members()
|
||||||
|
|
||||||
|
def validate_program_courses(self):
|
||||||
|
courses = [row.course for row in self.program_courses]
|
||||||
|
duplicates = {course for course in courses if courses.count(course) > 1}
|
||||||
|
if len(duplicates):
|
||||||
|
frappe.throw(
|
||||||
|
_("Course {0} has already been added to this batch.").format(
|
||||||
|
frappe.bold(next(iter(duplicates)))
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
def validate_program_members(self):
|
||||||
|
members = [row.member for row in self.program_members]
|
||||||
|
duplicates = {member for member in members if members.count(member) > 1}
|
||||||
|
if len(duplicates):
|
||||||
|
frappe.throw(
|
||||||
|
_("Member {0} has already been added to this batch.").format(
|
||||||
|
frappe.bold(next(iter(duplicates)))
|
||||||
|
)
|
||||||
|
)
|
||||||
21
lms/lms/doctype/lms_program/test_lms_program.py
Normal file
21
lms/lms/doctype/lms_program/test_lms_program.py
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
# Copyright (c) 2024, Frappe and Contributors
|
||||||
|
# See license.txt
|
||||||
|
|
||||||
|
# import frappe
|
||||||
|
from frappe.tests import IntegrationTestCase, UnitTestCase
|
||||||
|
|
||||||
|
|
||||||
|
# On IntegrationTestCase, the doctype test records and all
|
||||||
|
# link-field test record depdendencies are recursively loaded
|
||||||
|
# Use these module variables to add/remove to/from that list
|
||||||
|
EXTRA_TEST_RECORD_DEPENDENCIES = [] # eg. ["User"]
|
||||||
|
IGNORE_TEST_RECORD_DEPENDENCIES = [] # eg. ["User"]
|
||||||
|
|
||||||
|
|
||||||
|
class TestLMSProgram(UnitTestCase):
|
||||||
|
"""
|
||||||
|
Unit tests for LMSProgram.
|
||||||
|
Use this class for testing individual functions and methods.
|
||||||
|
"""
|
||||||
|
|
||||||
|
pass
|
||||||
0
lms/lms/doctype/lms_program_course/__init__.py
Normal file
0
lms/lms/doctype/lms_program_course/__init__.py
Normal file
42
lms/lms/doctype/lms_program_course/lms_program_course.json
Normal file
42
lms/lms/doctype/lms_program_course/lms_program_course.json
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
{
|
||||||
|
"actions": [],
|
||||||
|
"allow_rename": 1,
|
||||||
|
"creation": "2024-11-18 12:27:37.030302",
|
||||||
|
"doctype": "DocType",
|
||||||
|
"editable_grid": 1,
|
||||||
|
"engine": "InnoDB",
|
||||||
|
"field_order": [
|
||||||
|
"course",
|
||||||
|
"course_title"
|
||||||
|
],
|
||||||
|
"fields": [
|
||||||
|
{
|
||||||
|
"fieldname": "course",
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"in_list_view": 1,
|
||||||
|
"label": "Course",
|
||||||
|
"options": "LMS Course",
|
||||||
|
"reqd": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fetch_from": "course.title",
|
||||||
|
"fieldname": "course_title",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"in_list_view": 1,
|
||||||
|
"label": "Course Title",
|
||||||
|
"read_only": 1
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"index_web_pages_for_search": 1,
|
||||||
|
"istable": 1,
|
||||||
|
"links": [],
|
||||||
|
"modified": "2024-11-18 12:43:46.800199",
|
||||||
|
"modified_by": "Administrator",
|
||||||
|
"module": "LMS",
|
||||||
|
"name": "LMS Program Course",
|
||||||
|
"owner": "Administrator",
|
||||||
|
"permissions": [],
|
||||||
|
"sort_field": "creation",
|
||||||
|
"sort_order": "DESC",
|
||||||
|
"states": []
|
||||||
|
}
|
||||||
9
lms/lms/doctype/lms_program_course/lms_program_course.py
Normal file
9
lms/lms/doctype/lms_program_course/lms_program_course.py
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
# Copyright (c) 2024, Frappe and contributors
|
||||||
|
# For license information, please see license.txt
|
||||||
|
|
||||||
|
# import frappe
|
||||||
|
from frappe.model.document import Document
|
||||||
|
|
||||||
|
|
||||||
|
class LMSProgramCourse(Document):
|
||||||
|
pass
|
||||||
0
lms/lms/doctype/lms_program_member/__init__.py
Normal file
0
lms/lms/doctype/lms_program_member/__init__.py
Normal file
50
lms/lms/doctype/lms_program_member/lms_program_member.json
Normal file
50
lms/lms/doctype/lms_program_member/lms_program_member.json
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
{
|
||||||
|
"actions": [],
|
||||||
|
"allow_rename": 1,
|
||||||
|
"creation": "2024-11-18 12:29:13.615014",
|
||||||
|
"doctype": "DocType",
|
||||||
|
"editable_grid": 1,
|
||||||
|
"engine": "InnoDB",
|
||||||
|
"field_order": [
|
||||||
|
"member",
|
||||||
|
"full_name",
|
||||||
|
"progress"
|
||||||
|
],
|
||||||
|
"fields": [
|
||||||
|
{
|
||||||
|
"fieldname": "member",
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"in_list_view": 1,
|
||||||
|
"label": "Member",
|
||||||
|
"options": "User",
|
||||||
|
"reqd": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fetch_from": "member.full_name",
|
||||||
|
"fieldname": "full_name",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"in_list_view": 1,
|
||||||
|
"label": "Full Name",
|
||||||
|
"read_only": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"default": "0",
|
||||||
|
"fieldname": "progress",
|
||||||
|
"fieldtype": "Int",
|
||||||
|
"in_list_view": 1,
|
||||||
|
"label": "Progress"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"index_web_pages_for_search": 1,
|
||||||
|
"istable": 1,
|
||||||
|
"links": [],
|
||||||
|
"modified": "2024-11-21 12:51:31.882576",
|
||||||
|
"modified_by": "Administrator",
|
||||||
|
"module": "LMS",
|
||||||
|
"name": "LMS Program Member",
|
||||||
|
"owner": "Administrator",
|
||||||
|
"permissions": [],
|
||||||
|
"sort_field": "creation",
|
||||||
|
"sort_order": "DESC",
|
||||||
|
"states": []
|
||||||
|
}
|
||||||
9
lms/lms/doctype/lms_program_member/lms_program_member.py
Normal file
9
lms/lms/doctype/lms_program_member/lms_program_member.py
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
# Copyright (c) 2024, Frappe and contributors
|
||||||
|
# For license information, please see license.txt
|
||||||
|
|
||||||
|
# import frappe
|
||||||
|
from frappe.model.document import Document
|
||||||
|
|
||||||
|
|
||||||
|
class LMSProgramMember(Document):
|
||||||
|
pass
|
||||||
@@ -16,6 +16,7 @@ class LMSQuestion(Document):
|
|||||||
def validate_correct_answers(question):
|
def validate_correct_answers(question):
|
||||||
if question.type == "Choices":
|
if question.type == "Choices":
|
||||||
validate_duplicate_options(question)
|
validate_duplicate_options(question)
|
||||||
|
validate_minimum_options(question)
|
||||||
validate_correct_options(question)
|
validate_correct_options(question)
|
||||||
elif question.type == "User Input":
|
elif question.type == "User Input":
|
||||||
validate_possible_answer(question)
|
validate_possible_answer(question)
|
||||||
@@ -42,6 +43,11 @@ def validate_correct_options(question):
|
|||||||
frappe.throw(_("At least one option must be correct for this question."))
|
frappe.throw(_("At least one option must be correct for this question."))
|
||||||
|
|
||||||
|
|
||||||
|
def validate_minimum_options(question):
|
||||||
|
if question.type == "Choices" and (not question.option_1 or not question.option_2):
|
||||||
|
frappe.throw(_("Minimum two options are required for multiple choice questions."))
|
||||||
|
|
||||||
|
|
||||||
def validate_possible_answer(question):
|
def validate_possible_answer(question):
|
||||||
possible_answers = []
|
possible_answers = []
|
||||||
possible_answers_fields = [
|
possible_answers_fields = [
|
||||||
|
|||||||
@@ -134,7 +134,6 @@ def quiz_summary(quiz, results):
|
|||||||
result["marks"] = marks
|
result["marks"] = marks
|
||||||
score += marks
|
score += marks
|
||||||
|
|
||||||
del result["question_name"]
|
|
||||||
else:
|
else:
|
||||||
result["is_correct"] = 0
|
result["is_correct"] = 0
|
||||||
is_open_ended = True
|
is_open_ended = True
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import frappe
|
|||||||
from frappe.model.document import Document
|
from frappe.model.document import Document
|
||||||
from frappe.utils import cint
|
from frappe.utils import cint
|
||||||
from frappe import _
|
from frappe import _
|
||||||
|
from frappe.desk.doctype.notification_log.notification_log import make_notification_logs
|
||||||
|
|
||||||
|
|
||||||
class LMSQuizSubmission(Document):
|
class LMSQuizSubmission(Document):
|
||||||
@@ -12,7 +13,11 @@ class LMSQuizSubmission(Document):
|
|||||||
self.validate_marks()
|
self.validate_marks()
|
||||||
self.set_percentage()
|
self.set_percentage()
|
||||||
|
|
||||||
|
def on_update(self):
|
||||||
|
self.notify_member()
|
||||||
|
|
||||||
def validate_marks(self):
|
def validate_marks(self):
|
||||||
|
self.score = 0
|
||||||
for row in self.result:
|
for row in self.result:
|
||||||
if cint(row.marks) > cint(row.marks_out_of):
|
if cint(row.marks) > cint(row.marks_out_of):
|
||||||
frappe.throw(
|
frappe.throw(
|
||||||
@@ -26,3 +31,24 @@ class LMSQuizSubmission(Document):
|
|||||||
def set_percentage(self):
|
def set_percentage(self):
|
||||||
if self.score and self.score_out_of:
|
if self.score and self.score_out_of:
|
||||||
self.percentage = (self.score / self.score_out_of) * 100
|
self.percentage = (self.score / self.score_out_of) * 100
|
||||||
|
|
||||||
|
def notify_member(self):
|
||||||
|
if self.score != 0 and self.has_value_changed("score"):
|
||||||
|
notification = frappe._dict(
|
||||||
|
{
|
||||||
|
"subject": _("You have got a score of {0} for the quiz {1}").format(
|
||||||
|
self.score, self.quiz_title
|
||||||
|
),
|
||||||
|
"email_content": _(
|
||||||
|
"There has been an update on your submission. You have got a score of {0} for the quiz {1}"
|
||||||
|
).format(self.score, self.quiz_title),
|
||||||
|
"document_type": self.doctype,
|
||||||
|
"document_name": self.name,
|
||||||
|
"for_user": self.member,
|
||||||
|
"from_user": "Administrator",
|
||||||
|
"type": "Alert",
|
||||||
|
"link": "",
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
make_notification_logs(notification, [self.member])
|
||||||
|
|||||||
@@ -5,13 +5,15 @@
|
|||||||
"editable_grid": 1,
|
"editable_grid": 1,
|
||||||
"engine": "InnoDB",
|
"engine": "InnoDB",
|
||||||
"field_order": [
|
"field_order": [
|
||||||
|
"general_tab",
|
||||||
"default_home",
|
"default_home",
|
||||||
|
"send_calendar_invite_for_evaluations",
|
||||||
"is_onboarding_complete",
|
"is_onboarding_complete",
|
||||||
"column_break_zdel",
|
"column_break_zdel",
|
||||||
|
"enable_learning_paths",
|
||||||
"unsplash_access_key",
|
"unsplash_access_key",
|
||||||
"livecode_url",
|
"livecode_url",
|
||||||
"section_break_szgq",
|
"section_break_szgq",
|
||||||
"send_calendar_invite_for_evaluations",
|
|
||||||
"show_day_view",
|
"show_day_view",
|
||||||
"column_break_2",
|
"column_break_2",
|
||||||
"show_dashboard",
|
"show_dashboard",
|
||||||
@@ -80,6 +82,7 @@
|
|||||||
{
|
{
|
||||||
"fieldname": "mentor_request_section",
|
"fieldname": "mentor_request_section",
|
||||||
"fieldtype": "Section Break",
|
"fieldtype": "Section Break",
|
||||||
|
"hidden": 1,
|
||||||
"label": "Mentor Request"
|
"label": "Mentor Request"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -127,6 +130,7 @@
|
|||||||
{
|
{
|
||||||
"fieldname": "section_break_szgq",
|
"fieldname": "section_break_szgq",
|
||||||
"fieldtype": "Section Break",
|
"fieldtype": "Section Break",
|
||||||
|
"hidden": 1,
|
||||||
"label": "Batch Settings"
|
"label": "Batch Settings"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -336,12 +340,23 @@
|
|||||||
"fieldname": "payments_app_is_not_installed",
|
"fieldname": "payments_app_is_not_installed",
|
||||||
"fieldtype": "HTML",
|
"fieldtype": "HTML",
|
||||||
"label": "Payments app is not installed"
|
"label": "Payments app is not installed"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"default": "0",
|
||||||
|
"fieldname": "enable_learning_paths",
|
||||||
|
"fieldtype": "Check",
|
||||||
|
"label": "Enable Learning Paths"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "general_tab",
|
||||||
|
"fieldtype": "Tab Break",
|
||||||
|
"label": "General"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"index_web_pages_for_search": 1,
|
"index_web_pages_for_search": 1,
|
||||||
"issingle": 1,
|
"issingle": 1,
|
||||||
"links": [],
|
"links": [],
|
||||||
"modified": "2024-10-01 12:15:49.800242",
|
"modified": "2024-11-20 11:55:05.358421",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "LMS",
|
"module": "LMS",
|
||||||
"name": "LMS Settings",
|
"name": "LMS Settings",
|
||||||
@@ -356,6 +371,13 @@
|
|||||||
"role": "System Manager",
|
"role": "System Manager",
|
||||||
"share": 1,
|
"share": 1,
|
||||||
"write": 1
|
"write": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"email": 1,
|
||||||
|
"print": 1,
|
||||||
|
"read": 1,
|
||||||
|
"role": "LMS Student",
|
||||||
|
"share": 1
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"sort_field": "modified",
|
"sort_field": "modified",
|
||||||
|
|||||||
@@ -1,6 +0,0 @@
|
|||||||
"""Handy module to make access to all doctypes from a single place.
|
|
||||||
"""
|
|
||||||
from .doctype.lms_enrollment.lms_enrollment import (
|
|
||||||
LMSBatchMembership as Membership,
|
|
||||||
)
|
|
||||||
from .doctype.lms_course.lms_course import LMSCourse as Course
|
|
||||||
@@ -1,5 +1,4 @@
|
|||||||
import frappe
|
import frappe
|
||||||
from payments.utils import get_payment_gateway_controller
|
|
||||||
|
|
||||||
|
|
||||||
def get_payment_gateway():
|
def get_payment_gateway():
|
||||||
@@ -7,7 +6,10 @@ def get_payment_gateway():
|
|||||||
|
|
||||||
|
|
||||||
def get_controller(payment_gateway):
|
def get_controller(payment_gateway):
|
||||||
return get_payment_gateway_controller(payment_gateway)
|
if "payments" in frappe.get_installed_apps():
|
||||||
|
from payments.utils import get_payment_gateway_controller
|
||||||
|
|
||||||
|
return get_payment_gateway_controller(payment_gateway)
|
||||||
|
|
||||||
|
|
||||||
def validate_currency(payment_gateway, currency):
|
def validate_currency(payment_gateway, currency):
|
||||||
|
|||||||
189
lms/lms/utils.py
189
lms/lms/utils.py
@@ -6,11 +6,7 @@ import razorpay
|
|||||||
import requests
|
import requests
|
||||||
from frappe import _
|
from frappe import _
|
||||||
from frappe.desk.doctype.dashboard_chart.dashboard_chart import get_result
|
from frappe.desk.doctype.dashboard_chart.dashboard_chart import get_result
|
||||||
from frappe.desk.doctype.notification_log.notification_log import (
|
from frappe.desk.doctype.notification_log.notification_log import make_notification_logs
|
||||||
make_notification_logs,
|
|
||||||
enqueue_create_notification,
|
|
||||||
get_title,
|
|
||||||
)
|
|
||||||
from frappe.desk.search import get_user_groups
|
from frappe.desk.search import get_user_groups
|
||||||
from frappe.desk.notifications import extract_mentions
|
from frappe.desk.notifications import extract_mentions
|
||||||
from frappe.utils import (
|
from frappe.utils import (
|
||||||
@@ -855,7 +851,11 @@ def get_telemetry_boot_info():
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
def is_onboarding_complete():
|
def is_onboarding_complete():
|
||||||
|
if not has_course_moderator_role():
|
||||||
|
return {"is_onboarded": True}
|
||||||
|
|
||||||
course_created = frappe.db.a_row_exists("LMS Course")
|
course_created = frappe.db.a_row_exists("LMS Course")
|
||||||
chapter_created = frappe.db.a_row_exists("Course Chapter")
|
chapter_created = frappe.db.a_row_exists("Course Chapter")
|
||||||
lesson_created = frappe.db.a_row_exists("Course Lesson")
|
lesson_created = frappe.db.a_row_exists("Course Lesson")
|
||||||
@@ -874,26 +874,6 @@ def is_onboarding_complete():
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
def has_submitted_assessment(assessment, type, member=None):
|
|
||||||
if not member:
|
|
||||||
member = frappe.session.user
|
|
||||||
|
|
||||||
doctype = (
|
|
||||||
"LMS Assignment Submission" if type == "LMS Assignment" else "LMS Quiz Submission"
|
|
||||||
)
|
|
||||||
docfield = "assignment" if type == "LMS Assignment" else "quiz"
|
|
||||||
|
|
||||||
filters = {}
|
|
||||||
filters[docfield] = assessment
|
|
||||||
filters["member"] = member
|
|
||||||
return frappe.db.exists(doctype, filters)
|
|
||||||
|
|
||||||
|
|
||||||
def has_graded_assessment(submission):
|
|
||||||
status = frappe.db.get_value("LMS Assignment Submission", submission, "status")
|
|
||||||
return False if status == "Not Graded" else True
|
|
||||||
|
|
||||||
|
|
||||||
def get_evaluator(course, batch):
|
def get_evaluator(course, batch):
|
||||||
evaluator = None
|
evaluator = None
|
||||||
evaluator = frappe.db.get_value(
|
evaluator = frappe.db.get_value(
|
||||||
@@ -1128,11 +1108,20 @@ def get_course_outline(course, progress=False):
|
|||||||
chapter_details = frappe.db.get_value(
|
chapter_details = frappe.db.get_value(
|
||||||
"Course Chapter",
|
"Course Chapter",
|
||||||
chapter.chapter,
|
chapter.chapter,
|
||||||
["name", "title"],
|
["name", "title", "is_scorm_package", "launch_file", "scorm_package"],
|
||||||
as_dict=True,
|
as_dict=True,
|
||||||
)
|
)
|
||||||
chapter_details["idx"] = chapter.idx
|
chapter_details["idx"] = chapter.idx
|
||||||
chapter_details.lessons = get_lessons(course, chapter_details, progress=progress)
|
chapter_details.lessons = get_lessons(course, chapter_details, progress=progress)
|
||||||
|
|
||||||
|
if chapter_details.is_scorm_package:
|
||||||
|
chapter_details.scorm_package = frappe.db.get_value(
|
||||||
|
"File",
|
||||||
|
chapter_details.scorm_package,
|
||||||
|
["file_name", "file_size", "file_url"],
|
||||||
|
as_dict=1,
|
||||||
|
)
|
||||||
|
|
||||||
outline.append(chapter_details)
|
outline.append(chapter_details)
|
||||||
return outline
|
return outline
|
||||||
|
|
||||||
@@ -1146,9 +1135,12 @@ def get_lesson(course, chapter, lesson):
|
|||||||
"Lesson Reference", {"parent": chapter_name, "idx": lesson}, "lesson"
|
"Lesson Reference", {"parent": chapter_name, "idx": lesson}, "lesson"
|
||||||
)
|
)
|
||||||
lesson_details = frappe.db.get_value(
|
lesson_details = frappe.db.get_value(
|
||||||
"Course Lesson", lesson_name, ["include_in_preview", "title"], as_dict=1
|
"Course Lesson",
|
||||||
|
lesson_name,
|
||||||
|
["include_in_preview", "title", "is_scorm_package"],
|
||||||
|
as_dict=1,
|
||||||
)
|
)
|
||||||
if not lesson_details:
|
if not lesson_details or lesson_details.is_scorm_package:
|
||||||
return {}
|
return {}
|
||||||
|
|
||||||
membership = get_membership(course)
|
membership = get_membership(course)
|
||||||
@@ -1447,13 +1439,11 @@ def get_quiz_details(assessment, member):
|
|||||||
@frappe.whitelist()
|
@frappe.whitelist()
|
||||||
def get_batch_students(batch):
|
def get_batch_students(batch):
|
||||||
students = []
|
students = []
|
||||||
|
|
||||||
students_list = frappe.get_all(
|
students_list = frappe.get_all(
|
||||||
"Batch Student", filters={"parent": batch}, fields=["student", "name"]
|
"Batch Student", filters={"parent": batch}, fields=["student", "name"]
|
||||||
)
|
)
|
||||||
|
|
||||||
batch_courses = frappe.get_all("Batch Course", {"parent": batch}, pluck="course")
|
batch_courses = frappe.get_all("Batch Course", {"parent": batch}, ["course", "title"])
|
||||||
|
|
||||||
assessments = frappe.get_all(
|
assessments = frappe.get_all(
|
||||||
"LMS Assessment",
|
"LMS Assessment",
|
||||||
filters={"parent": batch},
|
filters={"parent": batch},
|
||||||
@@ -1471,29 +1461,64 @@ def get_batch_students(batch):
|
|||||||
)
|
)
|
||||||
detail.last_active = format_datetime(detail.last_active, "dd MMM YY")
|
detail.last_active = format_datetime(detail.last_active, "dd MMM YY")
|
||||||
detail.name = student.name
|
detail.name = student.name
|
||||||
students.append(detail)
|
detail.courses = frappe._dict()
|
||||||
|
detail.assessments = frappe._dict()
|
||||||
|
|
||||||
|
""" Iterate through courses and track their progress """
|
||||||
for course in batch_courses:
|
for course in batch_courses:
|
||||||
progress = frappe.db.get_value(
|
progress = frappe.db.get_value(
|
||||||
"LMS Enrollment", {"course": course, "member": student.student}, "progress"
|
"LMS Enrollment", {"course": course.course, "member": student.student}, "progress"
|
||||||
)
|
)
|
||||||
|
detail.courses[course.title] = progress
|
||||||
if progress == 100:
|
if progress == 100:
|
||||||
courses_completed += 1
|
courses_completed += 1
|
||||||
|
|
||||||
detail.courses_completed = courses_completed
|
""" Iterate through assessments and track their progress """
|
||||||
|
|
||||||
for assessment in assessments:
|
for assessment in assessments:
|
||||||
if has_submitted_assessment(
|
title = frappe.db.get_value(
|
||||||
|
assessment.assessment_type, assessment.assessment_name, "title"
|
||||||
|
)
|
||||||
|
status = has_submitted_assessment(
|
||||||
assessment.assessment_name, assessment.assessment_type, student.student
|
assessment.assessment_name, assessment.assessment_type, student.student
|
||||||
):
|
)
|
||||||
|
detail.assessments[title] = status
|
||||||
|
if status not in ["Not Attempted", 0]:
|
||||||
assessments_completed += 1
|
assessments_completed += 1
|
||||||
|
|
||||||
|
detail.courses_completed = courses_completed
|
||||||
detail.assessments_completed = assessments_completed
|
detail.assessments_completed = assessments_completed
|
||||||
|
students.append(detail)
|
||||||
|
|
||||||
return students
|
return students
|
||||||
|
|
||||||
|
|
||||||
|
def has_submitted_assessment(assessment, assessment_type, member=None):
|
||||||
|
if not member:
|
||||||
|
member = frappe.session.user
|
||||||
|
|
||||||
|
if assessment_type == "LMS Assignment":
|
||||||
|
doctype = "LMS Assignment Submission"
|
||||||
|
docfield = "assignment"
|
||||||
|
fields = ["status"]
|
||||||
|
not_attempted = "Not Attempted"
|
||||||
|
elif assessment_type == "LMS Quiz":
|
||||||
|
doctype = "LMS Quiz Submission"
|
||||||
|
docfield = "quiz"
|
||||||
|
fields = ["percentage"]
|
||||||
|
not_attempted = 0
|
||||||
|
|
||||||
|
filters = {}
|
||||||
|
filters[docfield] = assessment
|
||||||
|
filters["member"] = member
|
||||||
|
|
||||||
|
attempt = frappe.db.exists(doctype, filters)
|
||||||
|
if attempt:
|
||||||
|
attempt_details = frappe.db.get_value(doctype, filters, fields)
|
||||||
|
return attempt_details
|
||||||
|
else:
|
||||||
|
return not_attempted
|
||||||
|
|
||||||
|
|
||||||
@frappe.whitelist()
|
@frappe.whitelist()
|
||||||
def get_discussion_topics(doctype, docname, single_thread):
|
def get_discussion_topics(doctype, docname, single_thread):
|
||||||
if single_thread:
|
if single_thread:
|
||||||
@@ -1739,3 +1764,91 @@ def enroll_in_batch(batch, payment_name=None):
|
|||||||
)
|
)
|
||||||
|
|
||||||
student.save(ignore_permissions=True)
|
student.save(ignore_permissions=True)
|
||||||
|
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
|
def get_programs():
|
||||||
|
if (
|
||||||
|
has_course_moderator_role()
|
||||||
|
or has_course_instructor_role()
|
||||||
|
or has_course_evaluator_role()
|
||||||
|
):
|
||||||
|
programs = frappe.get_all("LMS Program", fields=["name"])
|
||||||
|
else:
|
||||||
|
programs = frappe.get_all(
|
||||||
|
"LMS Program Member", {"member": frappe.session.user}, ["parent as name", "progress"]
|
||||||
|
)
|
||||||
|
|
||||||
|
for program in programs:
|
||||||
|
program_courses = frappe.get_all(
|
||||||
|
"LMS Program Course", {"parent": program.name}, ["course"], order_by="idx"
|
||||||
|
)
|
||||||
|
program.courses = []
|
||||||
|
previous_progress = 0
|
||||||
|
for i, course in enumerate(program_courses):
|
||||||
|
details = get_course_details(course.course)
|
||||||
|
if i == 0:
|
||||||
|
details.eligible = True
|
||||||
|
elif previous_progress == 100:
|
||||||
|
details.eligible = True
|
||||||
|
else:
|
||||||
|
details.eligible = False
|
||||||
|
|
||||||
|
previous_progress = details.membership.progress if details.membership else 0
|
||||||
|
program.courses.append(details)
|
||||||
|
|
||||||
|
program.members = frappe.db.count("LMS Program Member", {"parent": program.name})
|
||||||
|
|
||||||
|
return programs
|
||||||
|
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
|
def enroll_in_program_course(program, course):
|
||||||
|
enrollment = frappe.db.exists(
|
||||||
|
"LMS Enrollment", {"member": frappe.session.user, "course": course}
|
||||||
|
)
|
||||||
|
|
||||||
|
if enrollment:
|
||||||
|
enrollment = frappe.db.get_value(
|
||||||
|
"LMS Enrollment", enrollment, ["name", "current_lesson"], as_dict=1
|
||||||
|
)
|
||||||
|
enrollment.current_lesson = get_lesson_index(enrollment.current_lesson)
|
||||||
|
return enrollment
|
||||||
|
|
||||||
|
program_courses = frappe.get_all(
|
||||||
|
"LMS Program Course", {"parent": program}, ["course", "idx"], order_by="idx"
|
||||||
|
)
|
||||||
|
current_course_idx = [
|
||||||
|
program_course.idx
|
||||||
|
for program_course in program_courses
|
||||||
|
if program_course.course == course
|
||||||
|
][0]
|
||||||
|
|
||||||
|
for program_course in program_courses:
|
||||||
|
if program_course.idx < current_course_idx:
|
||||||
|
enrollment = frappe.db.get_value(
|
||||||
|
"LMS Enrollment",
|
||||||
|
{"member": frappe.session.user, "course": program_course.course},
|
||||||
|
["name", "progress"],
|
||||||
|
as_dict=1,
|
||||||
|
)
|
||||||
|
if enrollment and enrollment.progress != 100:
|
||||||
|
frappe.throw(
|
||||||
|
_("Please complete the previous courses in the program to enroll in this course.")
|
||||||
|
)
|
||||||
|
elif not enrollment:
|
||||||
|
frappe.throw(
|
||||||
|
_("Please complete the previous courses in the program to enroll in this course.")
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
continue
|
||||||
|
|
||||||
|
enrollment = frappe.new_doc("LMS Enrollment")
|
||||||
|
enrollment.update(
|
||||||
|
{
|
||||||
|
"member": frappe.session.user,
|
||||||
|
"course": course,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
enrollment.save()
|
||||||
|
return enrollment
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
{
|
{
|
||||||
|
"app": "lms",
|
||||||
"charts": [
|
"charts": [
|
||||||
{
|
{
|
||||||
"chart_name": "New Signups",
|
"chart_name": "New Signups",
|
||||||
@@ -145,7 +146,7 @@
|
|||||||
"type": "Link"
|
"type": "Link"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"modified": "2024-08-09 13:19:06.273056",
|
"modified": "2024-11-21 12:16:25.886431",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "LMS",
|
"module": "LMS",
|
||||||
"name": "LMS",
|
"name": "LMS",
|
||||||
@@ -212,5 +213,6 @@
|
|||||||
"type": "DocType"
|
"type": "DocType"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"title": "LMS"
|
"title": "LMS",
|
||||||
|
"type": "Workspace"
|
||||||
}
|
}
|
||||||
733
lms/locale/ar.po
733
lms/locale/ar.po
File diff suppressed because it is too large
Load Diff
733
lms/locale/bs.po
733
lms/locale/bs.po
File diff suppressed because it is too large
Load Diff
733
lms/locale/de.po
733
lms/locale/de.po
File diff suppressed because it is too large
Load Diff
750
lms/locale/eo.po
750
lms/locale/eo.po
File diff suppressed because it is too large
Load Diff
735
lms/locale/es.po
735
lms/locale/es.po
File diff suppressed because it is too large
Load Diff
757
lms/locale/fa.po
757
lms/locale/fa.po
File diff suppressed because it is too large
Load Diff
735
lms/locale/fr.po
735
lms/locale/fr.po
File diff suppressed because it is too large
Load Diff
733
lms/locale/hu.po
733
lms/locale/hu.po
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user