Files
enlight-lms/frontend/src/pages/Programs/ProgramDetail.vue
2025-08-19 17:33:20 +05:30

136 lines
3.2 KiB
Vue

<template>
<header
class="sticky top-0 z-10 flex items-center justify-between border-b bg-surface-white px-3 py-2.5 sm:px-5"
>
<Breadcrumbs :items="breadcrumbs" />
</header>
<div v-if="program.data" class="pt-5 px-5 pb-10 mx-auto">
<div class="flex items-center mb-5">
<div class="text-lg font-semibold text-ink-gray-9">
{{ program.data.name }}
</div>
<Tooltip
v-if="program.data.enforce_course_order"
:text="
__(
'Courses must be completed in order. You can only start the next course after completing the previous one.'
)
"
>
<Route
class="size-5 ml-2 hover:bg-surface-gray-3 cursor-pointer p-1 rounded-sm"
/>
</Tooltip>
</div>
<div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-5 mb-5">
<div
v-for="course in program.data.courses"
:key="course.name"
class="relative group"
:class="
(course.eligible && program.data.enforce_course_order) ||
!program.data.enforce_course_order
? 'cursor-pointer'
: 'cursor-default'
"
>
<CourseCard
:course="course"
@click="openCourse(course, program.data.enforce_course_order)"
/>
<div
v-if="!course.eligible && program.data.enforce_course_order"
class="absolute inset-0 flex flex-col items-center justify-center space-y-2 text-ink-white rounded-md invisible group-hover:visible"
:style="{
background: 'radial-gradient(circle, darkgray 0%, lightgray 100%)',
}"
>
<LockKeyhole class="size-5" />
<span class="font-medium text-center leading-5 px-10">
{{ __('Please complete the previous course to unlock this one.') }}
</span>
</div>
</div>
</div>
</div>
</template>
<script setup lang="ts">
import { computed, inject, onMounted } from 'vue'
import {
Breadcrumbs,
call,
createResource,
Tooltip,
usePageMeta,
} from 'frappe-ui'
import { sessionStore } from '@/stores/session'
import { LockKeyhole, Route } from 'lucide-vue-next'
import { useRouter } from 'vue-router'
import CourseCard from '@/components/CourseCard.vue'
const { brand } = sessionStore()
const router = useRouter()
const user = inject<any>('$user')
const props = defineProps<{
programName: string
}>()
onMounted(() => {
checkIfEnrolled()
})
const checkIfEnrolled = () => {
call('frappe.client.get_value', {
doctype: 'LMS Program Member',
filters: {
member: user.data.name,
parent: props.programName,
},
parent: 'LMS Program',
fieldname: 'name',
}).then((data: { name: string }) => {
if (data.name) {
program.reload()
} else {
router.push({ name: 'Programs' })
}
})
}
const program = createResource({
url: 'lms.lms.utils.get_program_details',
params: {
program_name: props.programName,
},
})
const openCourse = (course: any, enforceCourseOrder: boolean) => {
if (!course.eligible && enforceCourseOrder) return
router.push({
name: 'CourseDetail',
params: { courseName: course.name },
})
}
const breadcrumbs = computed(() => {
return [
{ label: __('Programs'), route: { name: 'Programs' } },
{
label: props.programName,
route: {
name: 'ProgramDetail',
params: { programName: props.programName },
},
},
]
})
usePageMeta(() => {
return {
title: props.programName,
icon: brand.favicon,
}
})
</script>