mirror of
https://github.com/rzmk/learnhouse.git
synced 2025-12-19 04:19:25 +00:00
148 lines
No EOL
6.4 KiB
TypeScript
148 lines
No EOL
6.4 KiB
TypeScript
'use client';
|
|
import { getAPIUrl } from '@services/config/config';
|
|
import { revalidateTags, swrFetcher } from '@services/utils/ts/requests';
|
|
import React, { useContext, useEffect, useState } from 'react'
|
|
import { DragDropContext, Droppable, Draggable } from 'react-beautiful-dnd';
|
|
import useSWR, { mutate } from 'swr';
|
|
import ChapterElement from './DraggableElements/ChapterElement';
|
|
import PageLoading from '@components/Objects/Loaders/PageLoading';
|
|
import { createChapter, updateCourseOrderStructure } from '@services/courses/chapters';
|
|
import { useRouter } from 'next/navigation';
|
|
import { useCourse, useCourseDispatch } from '@components/Contexts/CourseContext';
|
|
import { Hexagon } from 'lucide-react';
|
|
import Modal from '@components/StyledElements/Modal/Modal';
|
|
import NewChapterModal from '@components/Objects/Modals/Chapters/NewChapter';
|
|
|
|
type EditCourseStructureProps = {
|
|
orgslug: string,
|
|
course_uuid?: string,
|
|
}
|
|
|
|
export type OrderPayload = {
|
|
chapter_order_by_ids: [
|
|
{
|
|
chapter_id: string,
|
|
activities_order_by_ids: [
|
|
{
|
|
activity_id: string
|
|
}
|
|
]
|
|
}
|
|
],
|
|
} | undefined
|
|
|
|
const EditCourseStructure = (props: EditCourseStructureProps) => {
|
|
const router = useRouter();
|
|
// Check window availability
|
|
const [winReady, setwinReady] = useState(false);
|
|
|
|
const dispatchCourse = useCourseDispatch() as any;
|
|
|
|
const [order, setOrder] = useState<OrderPayload>();
|
|
const course = useCourse() as any;
|
|
const course_structure = course ? course.courseStructure : {};
|
|
const course_uuid = course ? course.courseStructure.course_uuid : '';
|
|
|
|
// New Chapter creation
|
|
const [newChapterModal, setNewChapterModal] = useState(false);
|
|
|
|
const closeNewChapterModal = async () => {
|
|
setNewChapterModal(false);
|
|
};
|
|
|
|
// Submit new chapter
|
|
const submitChapter = async (chapter: any) => {
|
|
await createChapter(chapter);
|
|
mutate(`${getAPIUrl()}courses/${course.courseStructure.course_uuid}/meta`);
|
|
await revalidateTags(['courses'], props.orgslug);
|
|
router.refresh();
|
|
setNewChapterModal(false);
|
|
};
|
|
|
|
const updateStructure = (result: any) => {
|
|
const { destination, source, draggableId, type } = result;
|
|
if (!destination) return;
|
|
if (destination.droppableId === source.droppableId && destination.index === source.index) return;
|
|
if (type === 'chapter') {
|
|
const newChapterOrder = Array.from(course_structure.chapters);
|
|
newChapterOrder.splice(source.index, 1);
|
|
newChapterOrder.splice(destination.index, 0, course_structure.chapters[source.index]);
|
|
dispatchCourse({ type: 'setCourseStructure', payload: { ...course_structure, chapters: newChapterOrder } })
|
|
dispatchCourse({ type: 'setIsNotSaved' })
|
|
}
|
|
if (type === 'activity') {
|
|
const newChapterOrder = Array.from(course_structure.chapters);
|
|
const sourceChapter = newChapterOrder.find((chapter: any) => chapter.chapter_uuid === source.droppableId) as any;
|
|
const destinationChapter = newChapterOrder.find((chapter: any) => chapter.chapter_uuid === destination.droppableId) ? newChapterOrder.find((chapter: any) => chapter.chapter_uuid === destination.droppableId) : sourceChapter;
|
|
const activity = sourceChapter.activities.find((activity: any) => activity.activity_uuid === draggableId);
|
|
sourceChapter.activities.splice(source.index, 1);
|
|
destinationChapter.activities.splice(destination.index, 0, activity);
|
|
dispatchCourse({ type: 'setCourseStructure', payload: { ...course_structure, chapters: newChapterOrder } })
|
|
dispatchCourse({ type: 'setIsNotSaved' })
|
|
}
|
|
}
|
|
|
|
useEffect(() => {
|
|
setwinReady(true);
|
|
|
|
}, [props.course_uuid, course_structure, course]);
|
|
|
|
|
|
if (!course) return <PageLoading></PageLoading>
|
|
|
|
return (
|
|
<div className='flex flex-col'>
|
|
{winReady ?
|
|
<DragDropContext onDragEnd={updateStructure}>
|
|
<Droppable type='chapter' droppableId='chapters'>
|
|
{(provided) => (
|
|
<div
|
|
className='space-y-4'
|
|
{...provided.droppableProps}
|
|
ref={provided.innerRef}>
|
|
{course_structure.chapters && course_structure.chapters.map((chapter: any, index: any) => {
|
|
return (
|
|
|
|
<ChapterElement
|
|
key={chapter.chapter_uuid}
|
|
chapterIndex={index}
|
|
orgslug={props.orgslug}
|
|
course_uuid={course_uuid}
|
|
chapter={chapter} />
|
|
)
|
|
})}
|
|
{provided.placeholder}
|
|
</div>
|
|
)}
|
|
</Droppable>
|
|
|
|
{/* New Chapter Modal */}
|
|
<Modal
|
|
isDialogOpen={newChapterModal}
|
|
onOpenChange={setNewChapterModal}
|
|
minHeight="sm"
|
|
dialogContent={<NewChapterModal
|
|
course={course ? course.courseStructure : null}
|
|
closeModal={closeNewChapterModal}
|
|
submitChapter={submitChapter}
|
|
></NewChapterModal>}
|
|
dialogTitle="Create chapter"
|
|
dialogDescription="Add a new chapter to the course"
|
|
dialogTrigger={
|
|
<div className="mt-4 w-44 max-w-screen-2xl mx-auto bg-cyan-800 text-white rounded-xl shadow-sm px-6 items-center flex flex-row h-10">
|
|
<div className='mx-auto flex space-x-2 items-center hover:cursor-pointer'>
|
|
<Hexagon strokeWidth={3} size={16} className="text-white text-sm " />
|
|
<div className='font-bold text-sm'>Add Chapter</div></div>
|
|
</div>
|
|
}
|
|
/>
|
|
</DragDropContext>
|
|
|
|
: <></>}
|
|
</div>
|
|
|
|
|
|
)
|
|
}
|
|
|
|
export default EditCourseStructure |