const quizData = [
{
question: "Столица Беларуси — это…",
options: ["Гродно", "Минск", "Витебск", "Гомель"],
answer: 1,
fact: "Минск — самый большой город страны и её главный культурный центр.",
facts: [
"Минск впервые упоминается в летописях в 1067 году.",
"В городе находится Национальная библиотека с необычной формой ромбокубооктаэдра.",
"В Минске более 2 миллионов жителей — это крупнейший город Беларуси."
],
imagePath: "assets/pasted-20260305-185724-631e7f99.webp",
imageAlt: "Панорама Минска"
},
{
question: "Какие цвета есть на флаге Беларуси?",
options: ["Синий и жёлтый", "Белый и красный", "Красный и зелёный", "Белый и зелёный"],
answer: 2,
fact: "Красный и зелёный — основные цвета флага, а сбоку есть орнамент.",
facts: [
"Красный цвет на флаге символизирует мужество и историческую память.",
"Зелёный цвет связан с природой, лесами и полями страны.",
"Белорусский орнамент на флаге основан на традиционном народном узоре."
],
imagePath: "assets/images/quiz/q02-flag.jpg",
imageAlt: "Флаг Беларуси"
},
{
question: "Самое большое озеро Беларуси называется…",
options: ["Нарочь", "Свитязь", "Дривяты", "Кромань"],
answer: 0,
fact: "Озеро Нарочь — любимое место отдыха и часть Нарачанского парка.",
facts: [
"Площадь озера Нарочь — около 80 км², это крупнейшее озеро Беларуси.",
"Озеро входит в территорию Национального парка «Нарочанский».",
"Нарочь известна чистой водой и популярна для летнего отдыха."
],
imagePath: "assets/pasted-20260305-185926-dd9a5c42.webp",
imageAlt: "Озеро в Беларуси"
},
{
question: "Национальный символ животного Беларуси — это…",
options: ["Зубр", "Лось", "Медведь", "Лиса"],
answer: 0,
fact: "Зубр — редкое и сильное животное, живёт в Беловежской пуще.",
facts: [
"Зубр — самое тяжёлое наземное животное Европы.",
"После почти полного исчезновения популяцию зубров удалось восстановить.",
"В Беловежской пуще находится одна из самых известных популяций зубра."
],
imagePath: "assets/pasted-20260305-193021-7ef32b53.jpg",
imageAlt: "Зубр"
},
{
question: "Как называется главный заповедник Беларуси?",
options: ["Беловежская пуща", "Березинский парк", "Браславские озёра", "Припятский парк"],
answer: 0,
fact: "Беловежская пуща — старинный лес, который охраняется ЮНЕСКО.",
facts: [
"Беловежская пуща — один из самых древних лесов Европы.",
"Часть территории заповедника внесена в список Всемирного наследия ЮНЕСКО.",
"В пуще можно увидеть зубров, оленей и редкие виды птиц."
],
imagePath: "assets/pasted-20260305-190111-63cd8baa.jpg",
imageAlt: "Лес Беловежской пущи"
},
{
question: "Какой город называют «северной столицей» Беларуси?",
options: ["Витебск", "Гомель", "Брест", "Полоцк"],
answer: 0,
fact: "Витебск — родина художника Марка Шагала.",
facts: [
"В Витебске ежегодно проходит международный фестиваль «Славянский базар».",
"Город связан с именем художника Марка Шагала.",
"Витебск стоит на реке Западная Двина и имеет богатую историю."
],
imagePath: "assets/images/quiz/q06-vitebsk.jpg",
imageAlt: "Город Витебск"
},
{
question: "Как называется белорусский струнный ударный музыкальный инструмент?",
options: ["Цимбалы", "Скрипка", "Дудка", "Сопилка"],
answer: 0,
fact: "Цимбалы — это инструмент со струнами, по которым играют маленькими молоточками.",
facts: [
"Цимбалы похожи на большую деревянную трапецию со струнами.",
"По струнам ударяют двумя маленькими молоточками, и получается звонкий звук.",
"Цимбалы часто звучат в белорусской народной музыке."
],
imagePath: "assets/pasted-20260305-190344-ec46e879.jpg",
imageAlt: "Традиционный музыкальный инструмент"
},
{
question: "Какой замок стоит в городе Мир?",
options: ["Мирский замок", "Несвижский замок", "Лидский замок", "Гродненский замок"],
answer: 0,
fact: "Мирский замок — объект Всемирного наследия ЮНЕСКО.",
facts: [
"Мирский замок начали строить в начале XVI века.",
"Замок сочетает элементы готики, ренессанса и барокко.",
"Комплекс Мирского замка входит в список Всемирного наследия ЮНЕСКО."
],
imagePath: "assets/pasted-20260305-193451-5d1a5abc.jpg",
imageAlt: "Мирский замок"
},
{
question: "Главная река Беларуси — это…",
options: ["Западная Двина", "Днепр", "Нёман", "Припять"],
answer: 1,
fact: "Днепр проходит через южные регионы страны и впадает в Чёрное море.",
facts: [
"Днепр — одна из крупнейших рек Европы.",
"Река протекает через Беларусь, Россию и Украину.",
"В Беларуси Днепр важен для судоходства и водоснабжения."
],
imagePath: "assets/pasted-20260305-190627-76b2a1c1.jpg",
imageAlt: "Река Днепр"
},
{
question: "Как называется традиционный белорусский узор на ткани?",
options: ["Орнамент", "Гжель", "Петриковская роспись", "Хохлома"],
answer: 0,
fact: "Белорусский орнамент часто украшает флаг и народные костюмы.",
facts: [
"Орнамент передавался из поколения в поколение в народной вышивке.",
"В узорах часто встречаются ромбы, кресты и растительные элементы.",
"Белорусский орнамент используется в национальной символике и сувенирах."
],
imagePath: "assets/pasted-20260305-190855-a269e4b8.png",
imageAlt: "Белорусский орнамент"
},
{
question: "Самый древний город Беларуси — это…",
options: ["Полоцк", "Минск", "Брест", "Могилёв"],
answer: 0,
fact: "Полоцк впервые упоминается в летописях в 862 году.",
facts: [
"Полоцк — один из старейших городов восточных славян, известен с 862 года.",
"В городе находится Софийский собор — важный исторический памятник.",
"Полоцк был крупным культурным и торговым центром в Средние века."
],
imagePath: "assets/pasted-20260305-193618-2a0dd0c5.jpg",
imageAlt: "Город Полоцк"
},
{
question: "Как называется сладость из мёда и орехов, популярная в Беларуси?",
options: ["Пастила", "Печенье", "Козинаки", "Зефир"],
answer: 2,
fact: "Козинаки часто готовят из мёда и орехов.",
facts: [
"Козинаки делают из орехов или семечек, скреплённых мёдом или сиропом.",
"Эта сладость получается хрустящей и очень питательной.",
"Козинаки часто подают к чаю как традиционное лакомство."
],
imagePath: "assets/pasted-20260305-191522-fbd0d85f.jpg",
imageAlt: "Козинаки"
}
];
const startBtn = document.getElementById("startQuizBtn");
const quizSection = document.getElementById("quiz");
const summarySection = document.getElementById("summary");
const progressEl = document.getElementById("questionProgress");
const progressLabel = document.getElementById("progressLabel");
const progressFill = document.getElementById("progressFill");
const questionText = document.getElementById("questionText");
const optionList = document.getElementById("optionList");
const totalQuestions = document.getElementById("totalQuestions");
const classScore = document.getElementById("classScore");
const revealBtn = document.getElementById("revealBtn");
const nextBtn = document.getElementById("nextBtn");
const answerArea = document.getElementById("answerArea");
const answerImageTrigger = document.getElementById("answerImageTrigger");
const answerImage = document.getElementById("answerImage");
const correctAnswer = document.getElementById("correctAnswer");
const factList = document.getElementById("factList");
const answerResult = document.getElementById("answerResult");
const finalScore = document.getElementById("finalScore");
const summaryList = document.getElementById("summaryList");
const restartBtn = document.getElementById("restartBtn");
const imageLightbox = document.getElementById("imageLightbox");
const lightboxImage = document.getElementById("lightboxImage");
const closeImageLightbox = document.getElementById("closeImageLightbox");
const state = {
index: 0,
score: 0,
revealed: false,
selectedOption: null,
locked: false,
lastAnswerCorrect: false,
optionOrders: []
};
totalQuestions.textContent = quizData.length.toString();
function updateProgressBar() {
const completed = state.index + (state.revealed ? 1 : 0);
const percent = Math.max(0, Math.min(100, Math.round((completed / quizData.length) * 100)));
progressLabel.textContent = `${completed} из ${quizData.length}`;
progressFill.style.width = `${percent}%`;
progressFill.parentElement?.setAttribute("aria-valuenow", String(percent));
}
function shuffleArray(items) {
const arr = items.slice();
for (let i = arr.length - 1; i > 0; i -= 1) {
const j = Math.floor(Math.random() * (i + 1));
[arr[i], arr[j]] = [arr[j], arr[i]];
}
return arr;
}
function buildOptionOrders() {
return quizData.map((item) => shuffleArray(item.options.map((_, idx) => idx)));
}
function renderQuestion() {
const current = quizData[state.index];
const currentOrder = state.optionOrders[state.index] || current.options.map((_, idx) => idx);
progressEl.textContent = `Вопрос ${state.index + 1} из ${quizData.length}`;
updateProgressBar();
questionText.textContent = current.question;
optionList.innerHTML = "";
optionList.classList.toggle("d-none", state.revealed);
optionList.setAttribute("aria-hidden", state.revealed ? "true" : "false");
currentOrder.forEach((originalIdx, displayIdx) => {
const option = current.options[originalIdx];
const li = document.createElement("li");
li.className = "option-item";
li.setAttribute("role", "button");
li.setAttribute("tabindex", "0");
li.dataset.index = String(originalIdx);
if (originalIdx === state.selectedOption) {
li.classList.add("option-item-selected");
}
if (state.revealed) {
if (originalIdx === current.answer) {
li.classList.add("option-item-correct");
} else if (originalIdx === state.selectedOption) {
li.classList.add("option-item-wrong");
}
}
li.innerHTML = `${String.fromCharCode(65 + displayIdx)}${option}`;
optionList.appendChild(li);
});
classScore.textContent = state.score.toString();
answerArea.classList.add("d-none");
answerResult.classList.remove("text-success", "text-danger");
nextBtn.classList.add("d-none");
}
function renderAnswer() {
const current = quizData[state.index];
answerImage.src = current.imagePath;
answerImage.alt = current.imageAlt;
correctAnswer.textContent = current.options[current.answer];
factList.innerHTML = "";
const facts = Array.isArray(current.facts) && current.facts.length ? current.facts.slice(0, 4) : [current.fact];
facts.forEach((fact) => {
const li = document.createElement("li");
li.textContent = fact;
factList.appendChild(li);
});
const isCorrect = state.selectedOption === current.answer;
answerResult.textContent = isCorrect ? "Верно!" : "Неверно. Правильный ответ показан выше.";
answerResult.classList.toggle("text-success", isCorrect);
answerResult.classList.toggle("text-danger", !isCorrect);
answerArea.classList.remove("d-none");
}
function showSummary() {
quizSection.classList.add("d-none");
summarySection.classList.remove("d-none");
finalScore.textContent = `${state.score} из ${quizData.length}`;
summaryList.innerHTML = "";
quizData.forEach((item, idx) => {
const div = document.createElement("div");
div.className = "summary-item";
div.innerHTML = `
Вопрос ${idx + 1}: ${item.question}
Правильный ответ: ${item.options[item.answer]}
${item.fact}
`; summaryList.appendChild(div); }); } startBtn.addEventListener("click", () => { document.body.classList.add("quiz-live"); quizSection.classList.remove("d-none"); summarySection.classList.add("d-none"); state.index = 0; state.score = 0; state.revealed = false; state.selectedOption = null; state.locked = false; state.lastAnswerCorrect = false; state.optionOrders = buildOptionOrders(); renderQuestion(); quizSection.scrollIntoView({ behavior: "smooth" }); }); function goToNextQuestion() { if (state.index < quizData.length - 1) { state.index += 1; state.revealed = false; state.selectedOption = null; state.locked = false; state.lastAnswerCorrect = false; renderQuestion(); } else { showSummary(); summarySection.scrollIntoView({ behavior: "smooth" }); } } restartBtn.addEventListener("click", () => { state.index = 0; state.score = 0; state.revealed = false; state.selectedOption = null; state.locked = false; state.lastAnswerCorrect = false; state.optionOrders = buildOptionOrders(); summarySection.classList.add("d-none"); quizSection.classList.remove("d-none"); renderQuestion(); quizSection.scrollIntoView({ behavior: "smooth" }); }); function playCorrectClickEffect() { const selectedEl = optionList.querySelector(`.option-item[data-index="${state.selectedOption}"]`); if (!selectedEl) return; selectedEl.classList.remove("option-item-correct-pop"); void selectedEl.offsetWidth; selectedEl.classList.add("option-item-correct-pop"); const rect = selectedEl.getBoundingClientRect(); const burstCount = 28; const colors = ["#f59e0b", "#f97316", "#fb7185", "#22c55e", "#facc15"]; for (let i = 0; i < burstCount; i += 1) { const spark = document.createElement("span"); spark.className = "correct-spark"; spark.style.left = `${rect.left + rect.width / 2}px`; spark.style.top = `${rect.top + rect.height / 2}px`; spark.style.background = colors[i % colors.length]; const angle = ((Math.PI * 2) / burstCount) * i + (Math.random() * 0.3 - 0.15); const distance = 88 + Math.random() * 148; const size = 12 + Math.random() * 18; spark.style.setProperty("--dx", `${Math.cos(angle) * distance}px`); spark.style.setProperty("--dy", `${Math.sin(angle) * distance}px`); spark.style.setProperty("--rot", `${Math.random() * 280 - 140}deg`); spark.style.setProperty("--size", `${size}px`); document.body.appendChild(spark); setTimeout(() => spark.remove(), 1250); } } function closeLightbox() { imageLightbox.classList.add("d-none"); document.body.classList.remove("lightbox-open"); } answerImageTrigger.addEventListener("click", () => { if (!answerImage.src) return; lightboxImage.src = answerImage.src; lightboxImage.alt = answerImage.alt; imageLightbox.classList.remove("d-none"); document.body.classList.add("lightbox-open"); }); closeImageLightbox.addEventListener("click", closeLightbox); imageLightbox.addEventListener("click", (event) => { if (event.target === imageLightbox) { closeLightbox(); } }); document.addEventListener("keydown", (event) => { if (event.key === "Escape" && !imageLightbox.classList.contains("d-none")) { closeLightbox(); } }); function selectOption(index) { if (state.revealed || state.locked) return; const current = quizData[state.index]; state.selectedOption = index; state.lastAnswerCorrect = state.selectedOption === current.answer; if (state.lastAnswerCorrect) { playCorrectClickEffect(); state.score += 1; } state.revealed = true; state.locked = true; renderQuestion(); renderAnswer(); nextBtn.classList.remove("d-none"); } optionList.addEventListener("click", (event) => { const option = event.target.closest(".option-item"); if (!option) return; selectOption(Number(option.dataset.index)); }); optionList.addEventListener("keydown", (event) => { if (event.key !== "Enter" && event.key !== " ") return; const option = event.target.closest(".option-item"); if (!option) return; event.preventDefault(); selectOption(Number(option.dataset.index)); }); nextBtn.addEventListener("click", () => { if (!state.revealed) return; goToNextQuestion(); });