You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
138 lines
3.6 KiB
138 lines
3.6 KiB
"use client";
|
|
|
|
import {
|
|
type ReactNode,
|
|
useCallback,
|
|
useEffect,
|
|
useRef,
|
|
useState,
|
|
} from "react";
|
|
|
|
type QuestionProgressTrackerProps = {
|
|
children: ReactNode;
|
|
total: number;
|
|
};
|
|
|
|
function isQuestionAnswered(question: Element) {
|
|
const explicitAnsweredState = question.getAttribute("data-question-answered");
|
|
|
|
if (explicitAnsweredState === "true") {
|
|
return true;
|
|
}
|
|
|
|
if (explicitAnsweredState === "false") {
|
|
return false;
|
|
}
|
|
|
|
const inputs = Array.from(
|
|
question.querySelectorAll<
|
|
HTMLInputElement | HTMLSelectElement | HTMLTextAreaElement
|
|
>("input, select, textarea"),
|
|
);
|
|
|
|
return inputs.some((input) => {
|
|
if (input instanceof HTMLInputElement) {
|
|
if (input.type === "checkbox" || input.type === "radio") {
|
|
return input.checked;
|
|
}
|
|
|
|
if (input.type === "file") {
|
|
return input.files !== null && input.files.length > 0;
|
|
}
|
|
}
|
|
|
|
return input.value.trim().length > 0;
|
|
});
|
|
}
|
|
|
|
export function QuestionProgressTracker({
|
|
children,
|
|
total: initialTotal,
|
|
}: QuestionProgressTrackerProps) {
|
|
const containerRef = useRef<HTMLDivElement>(null);
|
|
const [answered, setAnswered] = useState(0);
|
|
const [total, setTotal] = useState(initialTotal);
|
|
const safeTotal = Math.max(total, 0);
|
|
const progress = safeTotal > 0 ? (answered / safeTotal) * 100 : 0;
|
|
|
|
const updateProgress = useCallback(() => {
|
|
const container = containerRef.current;
|
|
|
|
if (!container) {
|
|
return;
|
|
}
|
|
|
|
const requiredQuestions = Array.from(
|
|
container.querySelectorAll("[data-question-required='true']"),
|
|
).filter((el) => el.getAttribute("data-question-disabled") !== "true");
|
|
|
|
const nextTotal = requiredQuestions.length;
|
|
const nextAnswered = requiredQuestions.filter(isQuestionAnswered).length;
|
|
|
|
setTotal(nextTotal);
|
|
setAnswered(nextAnswered);
|
|
}, []);
|
|
|
|
useEffect(() => {
|
|
const container = containerRef.current;
|
|
|
|
if (!container) {
|
|
return;
|
|
}
|
|
|
|
const observer = new MutationObserver(updateProgress);
|
|
|
|
observer.observe(container, {
|
|
attributeFilter: ["data-question-answered"],
|
|
attributes: true,
|
|
subtree: true,
|
|
});
|
|
|
|
updateProgress();
|
|
|
|
return () => observer.disconnect();
|
|
}, [updateProgress]);
|
|
|
|
return (
|
|
<div
|
|
ref={containerRef}
|
|
className="flex flex-col"
|
|
onChange={updateProgress}
|
|
onInput={updateProgress}
|
|
>
|
|
<div className="mb-5 h-[68px]" aria-hidden="true" />
|
|
<div className="fixed top-[100px] left-1/2 z-20 w-full max-w-[375px] -translate-x-1/2 px-[17px]">
|
|
<div
|
|
aria-label={`Answered questions: ${answered} of ${safeTotal}`}
|
|
aria-valuemax={safeTotal}
|
|
aria-valuemin={0}
|
|
aria-valuenow={answered}
|
|
role="progressbar"
|
|
className="w-full rounded-none bg-[#F7F1F0] px-[9px] pt-[25px] pb-[16px]"
|
|
>
|
|
<div className="mb-[7px] flex items-center justify-between text-xs leading-none font-normal text-[#747474]">
|
|
<span>fields to complete</span>
|
|
<span>
|
|
{answered} /{safeTotal}
|
|
</span>
|
|
</div>
|
|
<div className="relative h-[6px] rounded-full bg-[#D8D8D8]">
|
|
<div
|
|
className={[
|
|
"absolute top-1/2 h-[10px] -translate-y-1/2 rounded-full bg-[#F2465F] transition-[width] duration-200",
|
|
answered > 0 ? "min-w-[37px]" : "",
|
|
]
|
|
.filter(Boolean)
|
|
.join(" ")}
|
|
style={{ width: `${progress}%` }}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
{children}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export default QuestionProgressTracker;
|