254 lines
6.0 KiB
Vue
254 lines
6.0 KiB
Vue
<template>
|
|
<div class="text-base">
|
|
<div
|
|
v-if="title && (outline.data?.length || allowEdit)"
|
|
class="grid grid-cols-[70%,30%] mb-4 px-2"
|
|
>
|
|
<div class="font-semibold text-lg leading-5">
|
|
{{ __(title) }}
|
|
</div>
|
|
<Button size="sm" v-if="allowEdit" @click="openChapterModal()">
|
|
{{ __('Add Chapter') }}
|
|
</Button>
|
|
<!-- <span class="font-medium cursor-pointer" @click="expandAllChapters()">
|
|
{{ expandAll ? __("Collapse all chapters") : __("Expand all chapters") }}
|
|
</span> -->
|
|
</div>
|
|
<div
|
|
:class="{
|
|
'shadow rounded-md pt-2 px-2': showOutline && outline.data?.length,
|
|
}"
|
|
>
|
|
<Disclosure
|
|
v-slot="{ open }"
|
|
v-for="(chapter, index) in outline.data"
|
|
:key="chapter.name"
|
|
:defaultOpen="openChapterDetail(chapter.idx)"
|
|
>
|
|
<DisclosureButton ref="" class="flex w-full p-2">
|
|
<ChevronRight
|
|
:class="{
|
|
'rotate-90 transform duration-200': open,
|
|
'duration-200': !open,
|
|
open: index == 1,
|
|
}"
|
|
class="h-4 w-4 text-gray-900 stroke-1 mr-2"
|
|
/>
|
|
<div class="text-base text-left font-medium leading-5">
|
|
{{ chapter.title }}
|
|
</div>
|
|
</DisclosureButton>
|
|
<DisclosurePanel>
|
|
<Draggable
|
|
:list="chapter.lessons"
|
|
:disabled="!allowEdit"
|
|
item-key="name"
|
|
group="items"
|
|
@end="updateOutline"
|
|
:data-chapter="chapter.name"
|
|
>
|
|
<template #item="{ element: lesson }">
|
|
<div class="outline-lesson pl-8 py-2 pr-4">
|
|
<router-link
|
|
:to="{
|
|
name: allowEdit ? 'LessonForm' : 'Lesson',
|
|
params: {
|
|
courseName: courseName,
|
|
chapterNumber: lesson.number.split('.')[0],
|
|
lessonNumber: lesson.number.split('.')[1],
|
|
},
|
|
}"
|
|
>
|
|
<div class="flex items-center text-sm leading-5 group">
|
|
<MonitorPlay
|
|
v-if="lesson.icon === 'icon-youtube'"
|
|
class="h-4 w-4 text-gray-900 stroke-1 mr-2"
|
|
/>
|
|
<HelpCircle
|
|
v-else-if="lesson.icon === 'icon-quiz'"
|
|
class="h-4 w-4 text-gray-900 stroke-1 mr-2"
|
|
/>
|
|
<FileText
|
|
v-else-if="lesson.icon === 'icon-list'"
|
|
class="h-4 w-4 text-gray-900 stroke-1 mr-2"
|
|
/>
|
|
{{ lesson.title }}
|
|
<Trash2
|
|
v-if="allowEdit"
|
|
@click.prevent="trashLesson(lesson.name, chapter.name)"
|
|
class="h-4 w-4 text-red-500 ml-auto invisible group-hover:visible"
|
|
/>
|
|
<Check
|
|
v-if="lesson.is_complete"
|
|
class="h-4 w-4 text-green-700 ml-2"
|
|
/>
|
|
</div>
|
|
</router-link>
|
|
</div>
|
|
</template>
|
|
</Draggable>
|
|
<div v-if="allowEdit" class="flex mt-2 mb-4 pl-8">
|
|
<router-link
|
|
:to="{
|
|
name: 'LessonForm',
|
|
params: {
|
|
courseName: courseName,
|
|
chapterNumber: chapter.idx,
|
|
lessonNumber: chapter.lessons.length + 1,
|
|
},
|
|
}"
|
|
>
|
|
<Button>
|
|
{{ __('Add Lesson') }}
|
|
</Button>
|
|
</router-link>
|
|
<Button class="ml-2" @click="openChapterModal(chapter)">
|
|
{{ __('Edit Chapter') }}
|
|
</Button>
|
|
</div>
|
|
</DisclosurePanel>
|
|
</Disclosure>
|
|
</div>
|
|
</div>
|
|
<ChapterModal
|
|
v-model="showChapterModal"
|
|
v-model:outline="outline"
|
|
:course="courseName"
|
|
:chapterDetail="getCurrentChapter()"
|
|
/>
|
|
</template>
|
|
<script setup>
|
|
import { Button, createResource } from 'frappe-ui'
|
|
import { ref, getCurrentInstance } from 'vue'
|
|
import Draggable from 'vuedraggable'
|
|
import { Disclosure, DisclosureButton, DisclosurePanel } from '@headlessui/vue'
|
|
import {
|
|
ChevronRight,
|
|
MonitorPlay,
|
|
HelpCircle,
|
|
FileText,
|
|
Check,
|
|
Trash2,
|
|
} from 'lucide-vue-next'
|
|
import { useRoute } from 'vue-router'
|
|
import ChapterModal from '@/components/Modals/ChapterModal.vue'
|
|
import { showToast } from '@/utils'
|
|
|
|
const route = useRoute()
|
|
const expandAll = ref(true)
|
|
const showChapterModal = ref(false)
|
|
const currentChapter = ref(null)
|
|
const app = getCurrentInstance()
|
|
const { $dialog } = app.appContext.config.globalProperties
|
|
|
|
const props = defineProps({
|
|
courseName: {
|
|
type: String,
|
|
required: true,
|
|
},
|
|
showOutline: {
|
|
type: Boolean,
|
|
default: false,
|
|
},
|
|
title: {
|
|
type: String,
|
|
default: '',
|
|
},
|
|
allowEdit: {
|
|
type: Boolean,
|
|
default: false,
|
|
},
|
|
getProgress: {
|
|
type: Boolean,
|
|
default: false,
|
|
},
|
|
})
|
|
|
|
const outline = createResource({
|
|
url: 'lms.lms.utils.get_course_outline',
|
|
cache: ['course_outline', props.courseName],
|
|
params: {
|
|
course: props.courseName,
|
|
progress: props.getProgress,
|
|
},
|
|
auto: true,
|
|
})
|
|
|
|
const deleteLesson = createResource({
|
|
url: 'lms.lms.api.delete_lesson',
|
|
makeParams(values) {
|
|
return {
|
|
lesson: values.lesson,
|
|
chapter: values.chapter,
|
|
}
|
|
},
|
|
onSuccess() {
|
|
outline.reload()
|
|
showToast('Success', 'Lesson deleted successfully', 'check')
|
|
},
|
|
})
|
|
|
|
const updateLessonIndex = createResource({
|
|
url: 'lms.lms.api.update_lesson_index',
|
|
makeParams(values) {
|
|
return {
|
|
lesson: values.lesson,
|
|
sourceChapter: values.sourceChapter,
|
|
targetChapter: values.targetChapter,
|
|
idx: values.idx,
|
|
}
|
|
},
|
|
onSuccess() {
|
|
showToast('Success', 'Lesson moved successfully', 'check')
|
|
},
|
|
})
|
|
|
|
const trashLesson = (lessonName, chapterName) => {
|
|
$dialog({
|
|
title: __('Delete Lesson'),
|
|
message: __('Are you sure you want to delete this lesson?'),
|
|
actions: [
|
|
{
|
|
label: __('Delete'),
|
|
theme: 'red',
|
|
variant: 'solid',
|
|
onClick(close) {
|
|
deleteLesson.submit({
|
|
lesson: lessonName,
|
|
chapter: chapterName,
|
|
})
|
|
close()
|
|
},
|
|
},
|
|
],
|
|
})
|
|
}
|
|
|
|
const openChapterDetail = (index) => {
|
|
return index == route.params.chapterNumber || index == 1
|
|
}
|
|
|
|
const openChapterModal = (chapter = null) => {
|
|
currentChapter.value = chapter
|
|
showChapterModal.value = true
|
|
}
|
|
|
|
const getCurrentChapter = () => {
|
|
return currentChapter.value
|
|
}
|
|
|
|
const updateOutline = (e) => {
|
|
updateLessonIndex.submit({
|
|
lesson: e.item.__draggable_context.element.name,
|
|
sourceChapter: e.from.dataset.chapter,
|
|
targetChapter: e.to.dataset.chapter,
|
|
idx: e.newIndex,
|
|
})
|
|
}
|
|
</script>
|
|
<style>
|
|
.outline-lesson:has(.router-link-active) {
|
|
background-color: theme('colors.gray.100');
|
|
}
|
|
</style>
|