454 lines
22 KiB
TypeScript
454 lines
22 KiB
TypeScript
import AutoExpandingTextArea from "@/components/Low/AutoExpandingTextarea";
|
|
import { Card, CardContent } from "@/components/ui/card";
|
|
import { BiQuestionMark } from 'react-icons/bi';
|
|
import { AiOutlineUnorderedList, AiOutlinePlus, AiOutlineDelete } from 'react-icons/ai';
|
|
import { Tooltip } from "react-tooltip";
|
|
import Header from "../../Shared/Header";
|
|
import GenLoader from "../Shared/GenLoader";
|
|
import { useEffect, useState } from "react";
|
|
import useSectionEdit from "../../Hooks/useSectionEdit";
|
|
import useExamEditorStore from "@/stores/examEditor";
|
|
import { InteractiveSpeakingExercise, LevelPart } from "@/interfaces/exam";
|
|
import { BsFileText } from "react-icons/bs";
|
|
import { FaChevronLeft, FaChevronRight } from "react-icons/fa6";
|
|
import { RiVideoLine } from "react-icons/ri";
|
|
import { Module } from "@/interfaces";
|
|
|
|
interface Props {
|
|
sectionId: number;
|
|
exercise: InteractiveSpeakingExercise;
|
|
module?: Module;
|
|
}
|
|
|
|
const InteractiveSpeaking: React.FC<Props> = ({ sectionId, exercise, module = "speaking" }) => {
|
|
const { dispatch } = useExamEditorStore();
|
|
const [local, setLocal] = useState(exercise);
|
|
const [currentVideoIndex, setCurrentVideoIndex] = useState(0);
|
|
|
|
const { generating, genResult, state, levelGenResults, levelGenerating } = useExamEditorStore(
|
|
(state) => state.modules[module].sections.find((section) => section.sectionId === sectionId)!
|
|
);
|
|
|
|
const { editing, setEditing, handleSave, handleDiscard, handleEdit, handlePractice } = useSectionEdit({
|
|
sectionId,
|
|
onSave: () => {
|
|
setEditing(false);
|
|
|
|
if (module === "level") {
|
|
const updatedState = {
|
|
...state,
|
|
exercises: (state as LevelPart).exercises.map((ex) =>
|
|
ex.id === local.id ? local : ex
|
|
)
|
|
};
|
|
dispatch({
|
|
type: "UPDATE_SECTION_STATE",
|
|
payload: { sectionId, update: updatedState, module }
|
|
});
|
|
} else {
|
|
dispatch({
|
|
type: "UPDATE_SECTION_STATE",
|
|
payload: { sectionId, update: local, module }
|
|
});
|
|
}
|
|
|
|
if (genResult) {
|
|
dispatch({
|
|
type: "UPDATE_SECTION_SINGLE_FIELD",
|
|
payload: {
|
|
sectionId,
|
|
module,
|
|
field: "genResult",
|
|
value: undefined
|
|
}
|
|
});
|
|
}
|
|
|
|
const speakingScript = levelGenResults?.find((res) => res.generating === `${local.id}-speakingScript`);
|
|
if (module === "level" && speakingScript) {
|
|
dispatch({
|
|
type: "UPDATE_SECTION_SINGLE_FIELD", payload: {
|
|
sectionId,
|
|
field: "levelGenResults",
|
|
value: levelGenResults.filter((res) => res.generating !== `${local.id}-speakingScript`),
|
|
module
|
|
}
|
|
});
|
|
}
|
|
},
|
|
onDiscard: () => {
|
|
setLocal(exercise);
|
|
},
|
|
onPractice: () => {
|
|
const updatedLocal = { ...local, isPractice: !local.isPractice };
|
|
setLocal(updatedLocal);
|
|
|
|
if (module === "level") {
|
|
const updatedState = {
|
|
...state,
|
|
exercises: (state as LevelPart).exercises.map((ex) =>
|
|
ex.id === local.id ? updatedLocal : ex
|
|
)
|
|
};
|
|
dispatch({
|
|
type: "UPDATE_SECTION_STATE",
|
|
payload: { sectionId, update: updatedState, module }
|
|
});
|
|
} else {
|
|
dispatch({
|
|
type: "UPDATE_SECTION_STATE",
|
|
payload: { sectionId, update: updatedLocal, module }
|
|
});
|
|
}
|
|
},
|
|
});
|
|
|
|
useEffect(() => {
|
|
if (genResult && generating === "speakingScript") {
|
|
const updatedLocal = {
|
|
...local,
|
|
title: genResult.result[0].title,
|
|
prompts: genResult.result[0].prompts.map((item: any) => ({
|
|
text: item || "",
|
|
video_url: ""
|
|
}))
|
|
};
|
|
setEditing(true);
|
|
setLocal(updatedLocal);
|
|
|
|
dispatch({
|
|
type: "UPDATE_SECTION_SINGLE_FIELD",
|
|
payload: {
|
|
sectionId,
|
|
module,
|
|
field: "generating",
|
|
value: undefined
|
|
}
|
|
});
|
|
}
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
}, [genResult, generating]);
|
|
|
|
useEffect(() => {
|
|
if (genResult && generating === "video") {
|
|
const updatedLocal = { ...local, prompts: genResult.result[0].prompts };
|
|
setLocal(updatedLocal);
|
|
|
|
dispatch({
|
|
type: "UPDATE_SECTION_STATE",
|
|
payload: { sectionId, update: updatedLocal, module }
|
|
});
|
|
|
|
|
|
dispatch({
|
|
type: "UPDATE_SECTION_SINGLE_FIELD",
|
|
payload: {
|
|
sectionId,
|
|
module,
|
|
field: "generating",
|
|
value: undefined
|
|
}
|
|
});
|
|
}
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
}, [genResult, generating]);
|
|
|
|
useEffect(() => {
|
|
const speakingScript = levelGenResults?.find((res) => res.generating === `${local.id}-speakingScript`);
|
|
const isGenerating = levelGenerating?.includes(`${local.id}-speakingScript`);
|
|
|
|
if (speakingScript && isGenerating) {
|
|
const updatedLocal = {
|
|
...local,
|
|
title: speakingScript.result[0].title,
|
|
prompts: speakingScript.result[0].prompts.map((item: any) => ({
|
|
text: item || "",
|
|
video_url: ""
|
|
}))
|
|
};
|
|
setEditing(true);
|
|
setLocal(updatedLocal);
|
|
|
|
dispatch({
|
|
type: "UPDATE_SECTION_SINGLE_FIELD", payload: {
|
|
sectionId,
|
|
field: "levelGenerating",
|
|
value: levelGenerating.filter((g) => g !== `${local.id}-speakingScript`),
|
|
module
|
|
}
|
|
});
|
|
}
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
}, [levelGenResults, levelGenerating]);
|
|
|
|
useEffect(() => {
|
|
const speakingVideo = levelGenResults?.find((res) => res.generating === `${local.id}-video`);
|
|
const isGenerating = levelGenerating?.includes(`${local.id}-video`);
|
|
|
|
if (speakingVideo && isGenerating) {
|
|
const updatedLocal = { ...local, prompts: speakingVideo.result[0].prompts };
|
|
setLocal(updatedLocal);
|
|
|
|
const updatedState = {
|
|
...state,
|
|
exercises: (state as LevelPart).exercises.map((ex) =>
|
|
ex.id === local.id ? updatedLocal : ex
|
|
)
|
|
};
|
|
|
|
dispatch({
|
|
type: "UPDATE_SECTION_STATE",
|
|
payload: { sectionId, update: updatedState, module }
|
|
});
|
|
|
|
dispatch({
|
|
type: "UPDATE_SECTION_SINGLE_FIELD", payload: {
|
|
sectionId,
|
|
field: "levelGenerating",
|
|
value: levelGenerating.filter((g) => g !== `${local.id}-video`),
|
|
module
|
|
}
|
|
});
|
|
}
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
}, [levelGenResults, levelGenerating]);
|
|
|
|
const addPrompt = () => {
|
|
setLocal(prev => ({
|
|
...prev,
|
|
prompts: [...prev.prompts, { text: "", video_url: "" }]
|
|
}));
|
|
};
|
|
|
|
const removePrompt = (index: number) => {
|
|
setLocal(prev => ({
|
|
...prev,
|
|
prompts: prev.prompts.filter((_, i) => i !== index)
|
|
}));
|
|
};
|
|
|
|
const updatePrompt = (index: number, text: string) => {
|
|
setLocal(prev => {
|
|
const newPrompts = [...prev.prompts];
|
|
newPrompts[index] = { ...newPrompts[index], text };
|
|
return { ...prev, prompts: newPrompts };
|
|
});
|
|
};
|
|
|
|
const isUnedited = local.prompts.length === 0;
|
|
|
|
useEffect(() => {
|
|
if (genResult && generating === "video") {
|
|
setLocal({ ...local, prompts: genResult.result[0].prompts });
|
|
dispatch({ type: "UPDATE_SECTION_STATE", payload: { sectionId, update: { ...local, prompts: genResult.result[0].prompts }, module: module } });
|
|
dispatch({
|
|
type: "UPDATE_SECTION_SINGLE_FIELD",
|
|
payload: {
|
|
sectionId,
|
|
module: module,
|
|
field: "generating",
|
|
value: undefined
|
|
}
|
|
});
|
|
}
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
}, [genResult, generating]);
|
|
|
|
const handlePrevVideo = () => {
|
|
setCurrentVideoIndex((prev) => (prev > 0 ? prev - 1 : prev));
|
|
};
|
|
|
|
const handleNextVideo = () => {
|
|
setCurrentVideoIndex((prev) =>
|
|
(prev < local.prompts.length - 1 ? prev + 1 : prev)
|
|
);
|
|
};
|
|
|
|
return (
|
|
<>
|
|
<div className='relative pb-4'>
|
|
<Header
|
|
title={`Interactive Speaking Script`}
|
|
description='Generate or write the scripts for the videos.'
|
|
editing={editing}
|
|
handleSave={handleSave}
|
|
handleEdit={handleEdit}
|
|
handleDiscard={handleDiscard}
|
|
handlePractice={handlePractice}
|
|
isEvaluationEnabled={!local.isPractice}
|
|
module="speaking"
|
|
/>
|
|
</div>
|
|
{(generating && generating === "speakingScript") || (levelGenerating.find((g) => g === `${local.id}-speakingScript`)) ? (
|
|
<GenLoader module={module} />
|
|
) : (
|
|
<>
|
|
{editing ? (
|
|
<>
|
|
{local.prompts.every((p) => p.video_url !== "") && (
|
|
<Card>
|
|
<CardContent className="pt-6">
|
|
<div className="flex flex-col items-start gap-3">
|
|
<div className="flex flex-row mb-3 gap-4 w-full justify-between items-center">
|
|
<div className="flex flex-row gap-4">
|
|
<RiVideoLine className="h-5 w-5 text-amber-500 mt-1" />
|
|
<h3 className="font-semibold text-xl">Videos</h3>
|
|
</div>
|
|
<div className="flex items-center gap-4">
|
|
<button
|
|
onClick={handlePrevVideo}
|
|
disabled={currentVideoIndex === 0}
|
|
className={`p-2 rounded-full ${currentVideoIndex === 0
|
|
? 'text-gray-400 cursor-not-allowed'
|
|
: 'text-gray-600 hover:bg-gray-100'
|
|
}`}
|
|
>
|
|
<FaChevronLeft className="w-4 h-4" />
|
|
</button>
|
|
<span className="text-sm text-gray-600">
|
|
{currentVideoIndex + 1} / {local.prompts.length}
|
|
</span>
|
|
<button
|
|
onClick={handleNextVideo}
|
|
disabled={currentVideoIndex === local.prompts.length - 1}
|
|
className={`p-2 rounded-full ${currentVideoIndex === local.prompts.length - 1
|
|
? 'text-gray-400 cursor-not-allowed'
|
|
: 'text-gray-600 hover:bg-gray-100'
|
|
}`}
|
|
>
|
|
<FaChevronRight className="w-4 h-4" />
|
|
</button>
|
|
</div>
|
|
</div>
|
|
<div className="flex flex-col gap-4 w-full items-center">
|
|
<div className="w-full">
|
|
<video
|
|
key={local.prompts[currentVideoIndex].video_url}
|
|
controls
|
|
className="w-full rounded-xl"
|
|
>
|
|
<source src={local.prompts[currentVideoIndex].video_url} />
|
|
</video>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
)}
|
|
{(generating && generating === "video") || levelGenerating.find((g) => g === `${local.id}-video`) &&
|
|
<GenLoader module={module} custom="Generating the videos ... This may take a while ..." />
|
|
}
|
|
<Card>
|
|
<CardContent>
|
|
<div className="flex flex-col py-2 mt-2">
|
|
<h2 className="font-semibold text-xl mb-2">Title</h2>
|
|
<AutoExpandingTextArea
|
|
value={local.title || ''}
|
|
onChange={(text) => setLocal(prev => ({ ...prev, title: text }))}
|
|
className="w-full p-3 border border-gray-300 rounded-md focus:ring-2 focus:ring-blue-500 focus:border-transparent min-h-[80px] transition-all"
|
|
placeholder="Enter the title"
|
|
/>
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
<Card>
|
|
<CardContent>
|
|
<div className="flex items-center mb-4 mt-6">
|
|
<h2 className="font-semibold text-xl">Questions</h2>
|
|
</div>
|
|
|
|
<div className="space-y-5">
|
|
{local.prompts.length === 0 ? (
|
|
<div className="py-12 text-center bg-gray-200 rounded-lg border-2 border-dashed border-gray-400">
|
|
<p className="text-gray-600">No questions added yet</p>
|
|
</div>
|
|
) : (
|
|
local.prompts.map((prompt, index) => (
|
|
<Card key={index}>
|
|
<CardContent>
|
|
<div className="bg-gray-50 rounded-lg pt-4">
|
|
<div className="flex justify-between items-center mb-3">
|
|
<h3 className="font-medium text-gray-700">Question {index + 1}</h3>
|
|
<button
|
|
type="button"
|
|
className="p-1.5 text-gray-400 hover:text-red-500 hover:bg-red-50 rounded-full transition-colors"
|
|
onClick={() => removePrompt(index)}
|
|
>
|
|
<AiOutlineDelete className="h-5 w-5" />
|
|
</button>
|
|
</div>
|
|
<AutoExpandingTextArea
|
|
value={prompt.text}
|
|
onChange={(text) => updatePrompt(index, text)}
|
|
className="w-full p-3 border border-gray-200 rounded-md focus:ring-2 focus:ring-blue-500 focus:border-transparent min-h-[80px] transition-all bg-white"
|
|
placeholder={`Enter question ${index + 1}`}
|
|
/>
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
))
|
|
)}
|
|
</div>
|
|
|
|
<div className="mt-6">
|
|
<button
|
|
type="button"
|
|
onClick={addPrompt}
|
|
className="w-full py-3 px-4 bg-gray-50 border border-gray-200 rounded-lg hover:bg-gray-100 transition-colors flex items-center justify-center gap-2 text-gray-600 font-medium"
|
|
>
|
|
<AiOutlinePlus className="h-5 w-5" />
|
|
Add Question
|
|
</button>
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
</>
|
|
) : isUnedited ? (
|
|
<p className="w-full text-gray-600 px-7 py-8 border-2 bg-white rounded-3xl whitespace-pre-line">
|
|
Generate or edit the questions!
|
|
</p>
|
|
) : (
|
|
<div className="space-y-6">
|
|
<Card>
|
|
<CardContent className="pt-6">
|
|
<div className="flex flex-col items-start gap-3">
|
|
<div className="flex flex-row mb-3 gap-4">
|
|
<BsFileText className="h-5 w-5 text-blue-500 mt-1" />
|
|
<h3 className="font-semibold text-xl">Title</h3>
|
|
</div>
|
|
<div className="w-full px-4 py-3 bg-white shadow-inner rounded-lg border border-gray-100">
|
|
<p className="text-lg">{local.title || 'Untitled'}</p>
|
|
</div>
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
<Card>
|
|
<CardContent className="pt-6">
|
|
<div className="flex flex-col items-start gap-3">
|
|
<div className="flex flex-row mb-3 gap-4">
|
|
<AiOutlineUnorderedList className="h-5 w-5 text-purple-500 mt-1" />
|
|
<h3 className="font-semibold text-xl">Questions</h3>
|
|
</div>
|
|
<div className="w-full space-y-4">
|
|
{local.prompts
|
|
.filter(prompt => prompt.text !== "")
|
|
.map((prompt, index) => (
|
|
<div key={index} className="bg-white shadow-inner rounded-lg border border-gray-100 p-4">
|
|
<h4 className="font-medium text-gray-700 mb-2">Question {index + 1}</h4>
|
|
<p className="text-gray-700">{prompt.text}</p>
|
|
</div>
|
|
))
|
|
}
|
|
</div>
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
</div>
|
|
)}
|
|
</>
|
|
)}
|
|
</>
|
|
);
|
|
};
|
|
|
|
export default InteractiveSpeaking; |