feat: program restrictions
This commit is contained in:
@@ -99,6 +99,7 @@ import { getSidebarLinks } from '../utils'
|
|||||||
import { usersStore } from '@/stores/user'
|
import { usersStore } from '@/stores/user'
|
||||||
import { sessionStore } from '@/stores/session'
|
import { sessionStore } from '@/stores/session'
|
||||||
import { useSidebar } from '@/stores/sidebar'
|
import { useSidebar } from '@/stores/sidebar'
|
||||||
|
import { useSettings } from '@/stores/settings'
|
||||||
import { 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'
|
||||||
@@ -114,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) => {
|
||||||
@@ -184,12 +186,23 @@ const addQuizzes = () => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const addPrograms = () => {
|
const addPrograms = () => {
|
||||||
if (isInstructor.value || isModerator.value) {
|
if (settingsStore.learningPaths.data) {
|
||||||
sidebarLinks.value.push({
|
let activeFor = ['Programs', 'ProgramForm']
|
||||||
|
let index = 1
|
||||||
|
if (!isInstructor.value && !isModerator.value) {
|
||||||
|
sidebarLinks.value = sidebarLinks.value.filter(
|
||||||
|
(link) => link.label !== 'Courses'
|
||||||
|
)
|
||||||
|
activeFor.push('CourseDetail')
|
||||||
|
activeFor.push('Lesson')
|
||||||
|
index = 0
|
||||||
|
}
|
||||||
|
|
||||||
|
sidebarLinks.value.splice(index, 0, {
|
||||||
label: 'Programs',
|
label: 'Programs',
|
||||||
icon: 'Route',
|
icon: 'Route',
|
||||||
to: 'Programs',
|
to: 'Programs',
|
||||||
activeFor: ['Programs', 'ProgramForm'],
|
activeFor: activeFor,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -303,9 +303,9 @@ const trashChapter = (chapterName) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const redirectToChapter = (chapter) => {
|
const redirectToChapter = (chapter) => {
|
||||||
|
if (!chapter.is_scorm_package) return
|
||||||
event.preventDefault()
|
event.preventDefault()
|
||||||
if (props.allowEdit) return
|
if (props.allowEdit) return
|
||||||
if (!chapter.is_scorm_package) return
|
|
||||||
if (!user.data) {
|
if (!user.data) {
|
||||||
showToast(
|
showToast(
|
||||||
__('You are not enrolled'),
|
__('You are not enrolled'),
|
||||||
|
|||||||
@@ -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()
|
||||||
|
|||||||
@@ -173,12 +173,14 @@ 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 { 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 router = useRouter()
|
||||||
|
const settings = useSettings()
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
checkLearningPath()
|
checkLearningPath()
|
||||||
@@ -189,14 +191,12 @@ onMounted(() => {
|
|||||||
})
|
})
|
||||||
|
|
||||||
const checkLearningPath = () => {
|
const checkLearningPath = () => {
|
||||||
call('frappe.client.get_single_value', {
|
if (
|
||||||
doctype: 'LMS Settings',
|
settings.learningPaths.data &&
|
||||||
field: 'enable_learning_paths',
|
(!user.data?.is_moderator || !user.data?.is_instructor)
|
||||||
}).then((res) => {
|
) {
|
||||||
if (res && !user.data?.is_moderator && !user.data?.is_instructor) {
|
router.push({ name: 'Programs' })
|
||||||
router.push({ name: 'Programs' })
|
}
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const courses = createResource({
|
const courses = createResource({
|
||||||
|
|||||||
@@ -211,8 +211,6 @@ const program = createDocumentResource({
|
|||||||
cache: ['program', props.programName],
|
cache: ['program', props.programName],
|
||||||
})
|
})
|
||||||
|
|
||||||
console.log(program)
|
|
||||||
|
|
||||||
const addProgramCourse = () => {
|
const addProgramCourse = () => {
|
||||||
program.setValue.submit(
|
program.setValue.submit(
|
||||||
{
|
{
|
||||||
@@ -288,7 +286,6 @@ const updateOrder = (e) => {
|
|||||||
course.idx = index + 1
|
course.idx = index + 1
|
||||||
})
|
})
|
||||||
|
|
||||||
console.log(courses)
|
|
||||||
program.setValue.submit(
|
program.setValue.submit(
|
||||||
{
|
{
|
||||||
program_courses: courses,
|
program_courses: courses,
|
||||||
|
|||||||
@@ -15,15 +15,27 @@
|
|||||||
</Button>
|
</Button>
|
||||||
</header>
|
</header>
|
||||||
<div v-if="programs.data?.length" class="pt-5 px-5">
|
<div v-if="programs.data?.length" class="pt-5 px-5">
|
||||||
<div v-for="program in programs.data" class="mb-10">
|
<div v-for="program in programs.data" class="mb-20">
|
||||||
<div class="flex items-center justify-between">
|
<div class="flex items-center justify-between">
|
||||||
<div class="text-xl font-semibold">
|
<div class="text-xl font-semibold">
|
||||||
{{ program.name }}
|
{{ program.name }}
|
||||||
</div>
|
</div>
|
||||||
<div class="flex items-center space-x-2">
|
<div class="flex items-center space-x-2">
|
||||||
<Badge v-if="program.members" variant="subtle" theme="green">
|
<Badge
|
||||||
{{ program.members }} {{ __('Members') }}
|
v-if="program.members"
|
||||||
|
variant="subtle"
|
||||||
|
theme="green"
|
||||||
|
size="lg"
|
||||||
|
>
|
||||||
|
{{ program.members }}
|
||||||
|
{{
|
||||||
|
program.members == 1 ? __(singularize('members')) : __('members')
|
||||||
|
}}
|
||||||
</Badge>
|
</Badge>
|
||||||
|
<Badge variant="subtle" theme="blue" size="lg">
|
||||||
|
{{ program.progress }}{{ __('% completed') }}
|
||||||
|
</Badge>
|
||||||
|
|
||||||
<router-link
|
<router-link
|
||||||
v-if="user.data?.is_moderator || user.data?.is_instructor"
|
v-if="user.data?.is_moderator || user.data?.is_instructor"
|
||||||
:to="{
|
:to="{
|
||||||
@@ -44,35 +56,12 @@
|
|||||||
v-if="program.courses?.length"
|
v-if="program.courses?.length"
|
||||||
class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-5 mt-5"
|
class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-5 mt-5"
|
||||||
>
|
>
|
||||||
<router-link
|
<CourseCard
|
||||||
v-for="course in program.courses"
|
v-for="course in program.courses"
|
||||||
:to="
|
:course="course"
|
||||||
course.membership && course.current_lesson
|
@click="enrollMember(program.name, course.name)"
|
||||||
? {
|
class="cursor-pointer"
|
||||||
name: 'Lesson',
|
/>
|
||||||
params: {
|
|
||||||
courseName: course.name,
|
|
||||||
chapterNumber: course.current_lesson.split('-')[0],
|
|
||||||
lessonNumber: course.current_lesson.split('-')[1],
|
|
||||||
},
|
|
||||||
}
|
|
||||||
: course.membership
|
|
||||||
? {
|
|
||||||
name: 'Lesson',
|
|
||||||
params: {
|
|
||||||
courseName: course.name,
|
|
||||||
chapterNumber: 1,
|
|
||||||
lessonNumber: 1,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
: {
|
|
||||||
name: 'CourseDetail',
|
|
||||||
params: { courseName: course.name },
|
|
||||||
}
|
|
||||||
"
|
|
||||||
>
|
|
||||||
<CourseCard :course="course" />
|
|
||||||
</router-link>
|
|
||||||
</div>
|
</div>
|
||||||
<div v-else class="text-sm italic text-gray-600 mt-4">
|
<div v-else class="text-sm italic text-gray-600 mt-4">
|
||||||
{{ __('No courses in this program') }}
|
{{ __('No courses in this program') }}
|
||||||
@@ -128,6 +117,7 @@ import { computed, inject, ref } from 'vue'
|
|||||||
import { BookOpen, Edit, Plus } from 'lucide-vue-next'
|
import { BookOpen, Edit, Plus } from 'lucide-vue-next'
|
||||||
import CourseCard from '@/components/CourseCard.vue'
|
import CourseCard from '@/components/CourseCard.vue'
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
|
import { showToast, singularize } from '@/utils'
|
||||||
|
|
||||||
const user = inject('$user')
|
const user = inject('$user')
|
||||||
const showDialog = ref(false)
|
const showDialog = ref(false)
|
||||||
@@ -140,8 +130,6 @@ const programs = createResource({
|
|||||||
cache: 'programs',
|
cache: 'programs',
|
||||||
})
|
})
|
||||||
|
|
||||||
console.log(programs)
|
|
||||||
|
|
||||||
const createProgram = (close) => {
|
const createProgram = (close) => {
|
||||||
call('frappe.client.insert', {
|
call('frappe.client.insert', {
|
||||||
doc: {
|
doc: {
|
||||||
@@ -153,6 +141,37 @@ const createProgram = (close) => {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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 breadbrumbs = computed(() => [
|
const breadbrumbs = computed(() => [
|
||||||
{
|
{
|
||||||
label: 'Programs',
|
label: 'Programs',
|
||||||
|
|||||||
@@ -1,12 +1,25 @@
|
|||||||
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'],
|
||||||
|
})
|
||||||
|
|
||||||
return {
|
return {
|
||||||
isSettingsOpen,
|
isSettingsOpen,
|
||||||
activeTab,
|
activeTab,
|
||||||
|
learningPaths,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -438,8 +438,6 @@ export function getSidebarLinks() {
|
|||||||
'Lesson',
|
'Lesson',
|
||||||
'CourseForm',
|
'CourseForm',
|
||||||
'LessonForm',
|
'LessonForm',
|
||||||
'Programs',
|
|
||||||
'ProgramForm',
|
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -93,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",
|
||||||
|
|||||||
@@ -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(
|
||||||
|
|||||||
@@ -11,82 +11,6 @@ class LMSProgram(Document):
|
|||||||
self.validate_program_courses()
|
self.validate_program_courses()
|
||||||
self.validate_program_members()
|
self.validate_program_members()
|
||||||
|
|
||||||
def on_update(self):
|
|
||||||
self.manage_acccess()
|
|
||||||
|
|
||||||
def manage_acccess(self):
|
|
||||||
old_doc = self.get_doc_before_save()
|
|
||||||
|
|
||||||
if not old_doc:
|
|
||||||
return
|
|
||||||
|
|
||||||
previous_courses = [row.course for row in old_doc.program_courses]
|
|
||||||
current_courses = [row.course for row in self.program_courses]
|
|
||||||
|
|
||||||
print("previous_courses", previous_courses)
|
|
||||||
print("current_courses", current_courses)
|
|
||||||
|
|
||||||
previous_members = [row.member for row in old_doc.program_members]
|
|
||||||
current_members = [row.member for row in self.program_members]
|
|
||||||
|
|
||||||
print("previous_members", previous_members)
|
|
||||||
print("current_members", current_members)
|
|
||||||
|
|
||||||
courses_added = [
|
|
||||||
course for course in current_courses if course not in previous_courses
|
|
||||||
]
|
|
||||||
courses_removed = [
|
|
||||||
course for course in previous_courses if course not in current_courses
|
|
||||||
]
|
|
||||||
|
|
||||||
members_added = [
|
|
||||||
member for member in current_members if member not in previous_members
|
|
||||||
]
|
|
||||||
members_removed = [
|
|
||||||
member for member in previous_members if member not in current_members
|
|
||||||
]
|
|
||||||
|
|
||||||
print(courses_removed)
|
|
||||||
print(members_removed)
|
|
||||||
|
|
||||||
if len(courses_added) > 0:
|
|
||||||
self.grant_program_access(current_members, courses_added)
|
|
||||||
|
|
||||||
if len(courses_removed) > 0:
|
|
||||||
print(courses_removed)
|
|
||||||
self.revoke_program_access(current_members, courses_removed)
|
|
||||||
|
|
||||||
if len(members_added) > 0:
|
|
||||||
self.grant_program_access(members_added, current_courses)
|
|
||||||
|
|
||||||
if len(members_removed) > 0:
|
|
||||||
print(members_removed)
|
|
||||||
self.revoke_program_access(members_removed, current_courses)
|
|
||||||
|
|
||||||
def grant_program_access(self, members, courses):
|
|
||||||
for course in courses:
|
|
||||||
for member in members:
|
|
||||||
enrollment = frappe.db.exists(
|
|
||||||
"LMS Enrollment", {"course": course, "member": member}
|
|
||||||
)
|
|
||||||
if not enrollment:
|
|
||||||
enrollment = frappe.new_doc("LMS Enrollment")
|
|
||||||
enrollment.course = course
|
|
||||||
enrollment.member = member
|
|
||||||
enrollment.insert()
|
|
||||||
|
|
||||||
def revoke_program_access(self, members, courses):
|
|
||||||
for course in courses:
|
|
||||||
print(course)
|
|
||||||
for member in members:
|
|
||||||
print(member)
|
|
||||||
enrollment = frappe.db.exists(
|
|
||||||
"LMS Enrollment", {"course": course, "member": member}
|
|
||||||
)
|
|
||||||
print(enrollment)
|
|
||||||
if enrollment:
|
|
||||||
frappe.delete_doc("LMS Enrollment", enrollment)
|
|
||||||
|
|
||||||
def validate_program_courses(self):
|
def validate_program_courses(self):
|
||||||
courses = [row.course for row in self.program_courses]
|
courses = [row.course for row in self.program_courses]
|
||||||
duplicates = {course for course in courses if courses.count(course) > 1}
|
duplicates = {course for course in courses if courses.count(course) > 1}
|
||||||
|
|||||||
@@ -7,7 +7,8 @@
|
|||||||
"engine": "InnoDB",
|
"engine": "InnoDB",
|
||||||
"field_order": [
|
"field_order": [
|
||||||
"member",
|
"member",
|
||||||
"full_name"
|
"full_name",
|
||||||
|
"progress"
|
||||||
],
|
],
|
||||||
"fields": [
|
"fields": [
|
||||||
{
|
{
|
||||||
@@ -25,12 +26,19 @@
|
|||||||
"in_list_view": 1,
|
"in_list_view": 1,
|
||||||
"label": "Full Name",
|
"label": "Full Name",
|
||||||
"read_only": 1
|
"read_only": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"default": "0",
|
||||||
|
"fieldname": "progress",
|
||||||
|
"fieldtype": "Int",
|
||||||
|
"in_list_view": 1,
|
||||||
|
"label": "Progress"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"index_web_pages_for_search": 1,
|
"index_web_pages_for_search": 1,
|
||||||
"istable": 1,
|
"istable": 1,
|
||||||
"links": [],
|
"links": [],
|
||||||
"modified": "2024-11-18 12:44:02.648786",
|
"modified": "2024-11-21 12:51:31.882576",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "LMS",
|
"module": "LMS",
|
||||||
"name": "LMS Program Member",
|
"name": "LMS Program Member",
|
||||||
|
|||||||
@@ -1763,12 +1763,12 @@ def get_programs():
|
|||||||
programs = frappe.get_all("LMS Program", fields=["name"])
|
programs = frappe.get_all("LMS Program", fields=["name"])
|
||||||
else:
|
else:
|
||||||
programs = frappe.get_all(
|
programs = frappe.get_all(
|
||||||
"LMS Program Member", {"member": frappe.session.user}, ["parent as name"]
|
"LMS Program Member", {"member": frappe.session.user}, ["parent as name", "progress"]
|
||||||
)
|
)
|
||||||
|
|
||||||
for program in programs:
|
for program in programs:
|
||||||
program_courses = frappe.get_all(
|
program_courses = frappe.get_all(
|
||||||
"LMS Program Course", {"parent": program.name}, ["course"]
|
"LMS Program Course", {"parent": program.name}, ["course"], order_by="idx"
|
||||||
)
|
)
|
||||||
program.courses = []
|
program.courses = []
|
||||||
for course in program_courses:
|
for course in program_courses:
|
||||||
@@ -1777,3 +1777,55 @@ def get_programs():
|
|||||||
program.members = frappe.db.count("LMS Program Member", {"parent": program.name})
|
program.members = frappe.db.count("LMS Program Member", {"parent": program.name})
|
||||||
|
|
||||||
return programs
|
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"
|
||||||
}
|
}
|
||||||
Reference in New Issue
Block a user