Как можно настроить прогресс-бар для ваших вопросов с возможностью каждому объекту присваивать 10% и достижения 100% при всем ответах? Учитывая структуру, где ответы считаются пустыми объектами с choices: [], nextQuestions: [].
Для создания прогресс-бара, меняющегося в зависимости от ответов на вопросы опросника при использовании Alpine.js, можно следующим образом:
1. Сначала создадим структуру данных для опросника. Например, вот как может выглядеть структура объекта с вопросами и ответами:
```javascript
{
questions: [
{
questionText: "Ваш любимый цвет?",
choices: ["Синий", "Зеленый", "Красный"],
selectedChoice: null
},
{
questionText: "Ваш любимый вид спорта?",
choices: ["Футбол", "Баскетбол", "Теннис"],
selectedChoice: null
},
// Добавьте другие вопросы здесь
],
progress: 0
}
```
2. Далее создадим прогресс-бар в HTML, который будет отображать прогресс опроса:
```html
<div x-data="surveyProgress()">
<div>
<span x-text="'Progress: ' + progress + '%'"></span>
<div>
<div class="bg-gray-200 w-full h-8">
<div class="bg-blue-500" x-bind:style="'width: ' + progress + '%'"></div>
</div>
</div>
</div>
<template x-for="(question, index) in questions" :key="index">
<div>
<h3 x-text="question.questionText"></h3>
<template x-for="(choice, choiceIndex) in question.choices" :key="choiceIndex">
<input type="radio" x-model="question.selectedChoice" :value="choice">
<label x-text="choice"></label>
</template>
</div>
</template>
</div>
```
3. Далее добавим Alpine.js скрипт, который будет управлять прогресс-баром и обработкой ответов на вопросы:
```javascript
function surveyProgress() {
return {
questions: [
// Вставьте структуру объекта с вопросами и ответами из шага 1
],
get progress() {
let totalQuestions = this.questions.length;
let answeredQuestions = this.questions.filter(question => question.selectedChoice !== null).length;
return Math.round((answeredQuestions / totalQuestions) * 100);
}
};
}
```
Теперь прогресс-бар будет автоматически обновляться в зависимости от того, сколько вопросов уже отвечено. Каждый ответ на вопрос будет увеличивать прогресс на 10%.
Добавить счётчик полученных ответов, делать ему +/- 1 при переходах к следующему/предыдущему вопросу. <br/> <br/> При расчёте прогресса проверять, что текущий вопрос последний (т.е., список возможных следующих вопросов пуст), в зависимости от результата проверки возвращать чего там вам надо - сразу 100 или умножаете счётчик ответов на 10 (кстати, уверены, что больше девяти вопросов задано не будет?). <br/> <br/> UPD. Как избавиться от этой ерунды с умножением на 10 - давайте будем вычислять максимальное количество вопросов, которые ещё могут быть заданы. Тогда текущий прогресс будет выражаться отношением количества отвеченных вопросов к сумме отвеченных и оставшихся. <br/> <br/> А ещё давайте избавимся от поиска вопросов по их id - для этого следующие/предыдущие вопросы надо хранить в виде индексов вместо id, или вместо массива вопросов сделать объект, где ключами будут id. <br/> <br/> <a href="https://jsfiddle.net/6jrfs70e/2/" rel="nofollow">Как это может выглядеть</a> : <br/> <br/> Рекурсия есть: <br/> <br/> <pre><code class="javascript">getMaxQuestionsLeft(question, depth) {
return Math.max(
depth,
...question.nextQuestions.map(n => {
return this.getMaxQuestionsLeft(this.questions[n], depth + 1);
})
);
},
get progress() {
const ac = this.answersCount;
return Math.floor(ac / (ac + this.getMaxQuestionsLeft(this.question, 0)) * 100);
},</code></pre> <br/> Или, рекурсии нет: <br/> <br/> <pre><code class="javascript">get maxQuestionsLeft() {
let result = 0;
for (const stack = [ [ this.question, 0 ] ]; stack.length;) {
const [ q, d ] = stack.pop();
result = Math.max(result, d);
stack.push(...q.nextQuestions.map(n => [ this.questions[n], -~d ]));
}
return result;
},
get progress() {
const { answersCount: ac } = this;
return ac / (ac + this.maxQuestionsLeft) * 100 | 0;
},</code></pre>