client/app/bundles/course/assessment/pages/AssessmentShow/prompts/DuplicationPrompt.tsx
import { Fragment, useDeferredValue, useMemo, useState } from 'react';
import { useLocation, useNavigate } from 'react-router-dom';
import { ArrowForwardRounded, SearchOffRounded } from '@mui/icons-material';
import {
List,
ListItem,
ListItemButton,
ListItemIcon,
ListItemText,
ListSubheader,
Paper,
Typography,
} from '@mui/material';
import { QuestionData } from 'types/course/assessment/questions';
import KoditsuChip from 'course/assessment/components/Koditsu/KoditsuChip';
import Prompt, { PromptText } from 'lib/components/core/dialogs/Prompt';
import TextField from 'lib/components/core/fields/TextField';
import Link from 'lib/components/core/Link';
import { loadingToast } from 'lib/hooks/toast';
import useTranslation from 'lib/hooks/useTranslation';
import { duplicateQuestion } from '../../../operations/questions';
import translations from '../../../translations';
interface DuplicationPromptProps {
for: QuestionData;
onClose: () => void;
open: boolean;
}
const filter = (
keyword: string,
question: QuestionData,
): QuestionData['duplicationUrls'] => {
if (!keyword) return question.duplicationUrls;
return question.duplicationUrls?.reduce<
NonNullable<QuestionData['duplicationUrls']>
>((targets, tab) => {
const filteredDestinations = tab.destinations.filter((assessment) =>
assessment.title.toLowerCase().includes(keyword.toLowerCase().trim()),
);
if (filteredDestinations.length === 0) return targets;
targets.push({
tab: tab.tab,
destinations: filteredDestinations,
});
return targets;
}, []);
};
interface TargetsListProps {
disabled: boolean;
containing: string;
for: QuestionData;
onSelectTarget: (duplicationUrl: string) => void;
}
const TargetsList = (props: TargetsListProps): JSX.Element => {
const { containing: keyword, for: question } = props;
const { t } = useTranslation();
const targets = useMemo(() => filter(keyword, question), [keyword, question]);
if (!targets || targets.length === 0)
return (
<div className="flex h-full flex-col items-center justify-center p-10 text-neutral-400">
<div className="flex items-center justify-center rounded-full p-4 outline">
<SearchOffRounded className="wh-32" />
</div>
<Typography className="mt-8" variant="h6">
{t(translations.noItemsMatched, { keyword: keyword.trim() })}
</Typography>
<Typography>{t(translations.tryAgain)}</Typography>
</div>
);
return (
<List dense disablePadding>
{targets?.map((tab) => (
<Fragment key={tab.tab}>
<ListSubheader className="bg-neutral-100">{tab.tab}</ListSubheader>
{tab.destinations.map((assessment) => (
<ListItem
key={assessment.duplicationUrl}
className="group"
disablePadding
>
<ListItemButton
disabled={
props.disabled ||
(assessment.isKoditsu && !question.isCompatibleWithKoditsu)
}
onClick={(): void =>
props.onSelectTarget(assessment.duplicationUrl)
}
>
<ListItemText>
{assessment.isKoditsu ? (
<>
{assessment.title}
<KoditsuChip />
</>
) : (
assessment.title
)}
</ListItemText>
<ListItemIcon
className={`min-w-fit ${
props.disabled
? 'invisible'
: 'hoverable:invisible group-hover?:visible'
}`}
>
<ArrowForwardRounded />
</ListItemIcon>
</ListItemButton>
</ListItem>
))}
</Fragment>
))}
</List>
);
};
const DuplicationPrompt = (props: DuplicationPromptProps): JSX.Element => {
const { for: question } = props;
const { t } = useTranslation();
const [duplicating, setDuplicating] = useState(false);
const [keyword, setKeyword] = useState('');
const deferredKeyword = useDeferredValue(keyword);
const navigate = useNavigate();
const { pathname } = useLocation();
const duplicate = async (duplicationUrl: string): Promise<void> => {
setDuplicating(true);
const toast = loadingToast(t(translations.duplicatingQuestion));
try {
const result = await duplicateQuestion(duplicationUrl);
const destinationUrl = result?.destinationUrl;
if (destinationUrl === pathname) {
navigate(0);
toast.success(t(translations.questionDuplicatedRefreshing));
} else {
toast.success(
t(translations.questionDuplicated, {
link: (chunk) => (
<Link href={result?.destinationUrl} opensInNewTab>
{chunk} →
</Link>
),
}),
);
}
props.onClose();
} catch (error) {
const message = (error as Error)?.message;
toast.error(message || t(translations.errorDuplicatingQuestion));
} finally {
setDuplicating(false);
}
};
const targetsList = useMemo(
() => (
<TargetsList
containing={deferredKeyword}
disabled={duplicating}
for={question}
onSelectTarget={duplicate}
/>
),
[deferredKeyword, duplicating, question],
);
return (
<Prompt
contentClassName="space-y-4 flex flex-col h-screen"
onClose={props.onClose}
open={props.open}
title={t(translations.chooseAssessmentToDuplicateInto)}
>
<PromptText>{t(translations.duplicatingThisQuestion)}</PromptText>
<PromptText className="line-clamp-2 pb-7 italic">
{question.title}
</PromptText>
<TextField
autoFocus
className="!mt-8"
disabled={duplicating}
fullWidth
label={t(translations.searchTargetAssessment)}
onChange={(e): void => setKeyword(e.target.value)}
size="small"
trims
value={keyword}
variant="filled"
/>
<Paper className="h-screen overflow-scroll" variant="outlined">
{targetsList}
</Paper>
</Prompt>
);
};
export default DuplicationPrompt;