fix: game play on quiz
This commit is contained in:
parent
488479befa
commit
55d96c3baf
|
@ -6,6 +6,9 @@ import 'package:quiz_app/app/routes/app_pages.dart';
|
||||||
import 'package:quiz_app/component/notification/pop_up_confirmation.dart';
|
import 'package:quiz_app/component/notification/pop_up_confirmation.dart';
|
||||||
import 'package:quiz_app/data/models/quiz/library_quiz_model.dart';
|
import 'package:quiz_app/data/models/quiz/library_quiz_model.dart';
|
||||||
import 'package:quiz_app/data/models/quiz/question/base_qustion_model.dart';
|
import 'package:quiz_app/data/models/quiz/question/base_qustion_model.dart';
|
||||||
|
import 'package:quiz_app/data/models/quiz/question/fill_in_the_blank_question_model.dart';
|
||||||
|
import 'package:quiz_app/data/models/quiz/question/option_question_model.dart';
|
||||||
|
import 'package:quiz_app/data/models/quiz/question/true_false_question_model.dart';
|
||||||
|
|
||||||
class QuizPlayController extends GetxController {
|
class QuizPlayController extends GetxController {
|
||||||
late final QuizData quizData;
|
late final QuizData quizData;
|
||||||
|
@ -88,25 +91,35 @@ class QuizPlayController extends GetxController {
|
||||||
|
|
||||||
void _submitAnswerIfNeeded() {
|
void _submitAnswerIfNeeded() {
|
||||||
final question = currentQuestion;
|
final question = currentQuestion;
|
||||||
String userAnswer = '';
|
dynamic userAnswer = '';
|
||||||
|
|
||||||
switch (question.type) {
|
if (question is FillInTheBlankQuestion) {
|
||||||
case 'fill_the_blank':
|
userAnswer = answerTextController.text.toString();
|
||||||
userAnswer = answerTextController.text.trim();
|
} else {
|
||||||
break;
|
userAnswer = selectedAnswer.value;
|
||||||
case 'option':
|
|
||||||
case 'true_false':
|
|
||||||
userAnswer = selectedAnswer.value.trim();
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
// answeredQuestions.add(AnsweredQuestion(
|
|
||||||
// index: currentIndex.value,
|
dynamic correctAnswer;
|
||||||
// questionIndex: question.index,
|
if (question is FillInTheBlankQuestion) {
|
||||||
// selectedAnswer: userAnswer,
|
correctAnswer = question.targetAnswer.toLowerCase();
|
||||||
// correctAnswer: question.,
|
userAnswer = userAnswer.toString().toLowerCase();
|
||||||
// isCorrect: userAnswer.toLowerCase() == question.targetAnswer.trim().toLowerCase(),
|
} else if (question is TrueFalseQuestion) {
|
||||||
// duration: currentQuestion.duration - timeLeft.value,
|
correctAnswer = question.targetAnswer;
|
||||||
// ));
|
userAnswer = userAnswer == 'true' || userAnswer == true;
|
||||||
|
} else if (question is OptionQuestion) {
|
||||||
|
correctAnswer = question.targetAnswer;
|
||||||
|
userAnswer = int.tryParse(userAnswer.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
final isCorrect = userAnswer == correctAnswer;
|
||||||
|
|
||||||
|
answeredQuestions.add(AnsweredQuestion(
|
||||||
|
index: currentIndex.value,
|
||||||
|
questionIndex: question.index,
|
||||||
|
selectedAnswer: userAnswer,
|
||||||
|
isCorrect: isCorrect,
|
||||||
|
duration: question.duration - timeLeft.value,
|
||||||
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
void nextQuestion() {
|
void nextQuestion() {
|
||||||
|
@ -138,10 +151,17 @@ class QuizPlayController extends GetxController {
|
||||||
_timer?.cancel();
|
_timer?.cancel();
|
||||||
|
|
||||||
AppDialog.showMessage(Get.context!, "Yeay semua soal selesai");
|
AppDialog.showMessage(Get.context!, "Yeay semua soal selesai");
|
||||||
|
|
||||||
await Future.delayed(Duration(seconds: 2));
|
await Future.delayed(Duration(seconds: 2));
|
||||||
|
|
||||||
|
print(quizData);
|
||||||
|
|
||||||
Get.offAllNamed(
|
Get.offAllNamed(
|
||||||
AppRoutes.resultQuizPage,
|
AppRoutes.resultQuizPage,
|
||||||
arguments: [quizData, answeredQuestions],
|
arguments: {
|
||||||
|
"quiz_data": quizData,
|
||||||
|
"answer_data": answeredQuestions,
|
||||||
|
},
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -157,7 +177,6 @@ class AnsweredQuestion {
|
||||||
final int index;
|
final int index;
|
||||||
final int questionIndex;
|
final int questionIndex;
|
||||||
final dynamic selectedAnswer;
|
final dynamic selectedAnswer;
|
||||||
final dynamic correctAnswer;
|
|
||||||
final bool isCorrect;
|
final bool isCorrect;
|
||||||
final int duration;
|
final int duration;
|
||||||
|
|
||||||
|
@ -165,7 +184,6 @@ class AnsweredQuestion {
|
||||||
required this.index,
|
required this.index,
|
||||||
required this.questionIndex,
|
required this.questionIndex,
|
||||||
required this.selectedAnswer,
|
required this.selectedAnswer,
|
||||||
required this.correctAnswer,
|
|
||||||
required this.isCorrect,
|
required this.isCorrect,
|
||||||
required this.duration,
|
required this.duration,
|
||||||
});
|
});
|
||||||
|
@ -175,7 +193,6 @@ class AnsweredQuestion {
|
||||||
index: json['index'],
|
index: json['index'],
|
||||||
questionIndex: json['question_index'],
|
questionIndex: json['question_index'],
|
||||||
selectedAnswer: json['selectedAnswer'],
|
selectedAnswer: json['selectedAnswer'],
|
||||||
correctAnswer: json['correctAnswer'],
|
|
||||||
isCorrect: json['isCorrect'],
|
isCorrect: json['isCorrect'],
|
||||||
duration: json['duration'],
|
duration: json['duration'],
|
||||||
);
|
);
|
||||||
|
@ -185,7 +202,6 @@ class AnsweredQuestion {
|
||||||
'index': index,
|
'index': index,
|
||||||
'question_index': questionIndex,
|
'question_index': questionIndex,
|
||||||
'selectedAnswer': selectedAnswer,
|
'selectedAnswer': selectedAnswer,
|
||||||
'correctAnswer': correctAnswer,
|
|
||||||
'isCorrect': isCorrect,
|
'isCorrect': isCorrect,
|
||||||
'duration': duration,
|
'duration': duration,
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,5 +1,8 @@
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
|
import 'package:quiz_app/app/const/colors/app_colors.dart';
|
||||||
|
import 'package:quiz_app/component/global_text_field.dart';
|
||||||
|
import 'package:quiz_app/data/models/quiz/question/option_question_model.dart';
|
||||||
import 'package:quiz_app/feature/quiz_play/controller/quiz_play_controller.dart';
|
import 'package:quiz_app/feature/quiz_play/controller/quiz_play_controller.dart';
|
||||||
|
|
||||||
class QuizPlayView extends GetView<QuizPlayController> {
|
class QuizPlayView extends GetView<QuizPlayController> {
|
||||||
|
@ -33,7 +36,7 @@ class QuizPlayView extends GetView<QuizPlayController> {
|
||||||
const SizedBox(height: 12),
|
const SizedBox(height: 12),
|
||||||
_buildQuestionText(),
|
_buildQuestionText(),
|
||||||
const SizedBox(height: 30),
|
const SizedBox(height: 30),
|
||||||
// _buildAnswerSection(),
|
_buildAnswerSection(),
|
||||||
const Spacer(),
|
const Spacer(),
|
||||||
_buildNextButton(),
|
_buildNextButton(),
|
||||||
],
|
],
|
||||||
|
@ -98,44 +101,44 @@ class QuizPlayView extends GetView<QuizPlayController> {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Widget _buildAnswerSection() {
|
Widget _buildAnswerSection() {
|
||||||
// final question = controller.currentQuestion;
|
final question = controller.currentQuestion;
|
||||||
|
|
||||||
// if (question.type == 'option' && question.options != null) {
|
if (question is OptionQuestion) {
|
||||||
// return Column(
|
return Column(
|
||||||
// children: List.generate(question.options!.length, (index) {
|
children: List.generate(question.options.length, (index) {
|
||||||
// final option = question.options![index];
|
final option = question.options[index];
|
||||||
// final isSelected = controller.idxOptionSelected.value == index;
|
final isSelected = controller.idxOptionSelected.value == index;
|
||||||
|
|
||||||
// return Container(
|
return Container(
|
||||||
// margin: const EdgeInsets.only(bottom: 12),
|
margin: const EdgeInsets.only(bottom: 12),
|
||||||
// width: double.infinity,
|
width: double.infinity,
|
||||||
// child: ElevatedButton(
|
child: ElevatedButton(
|
||||||
// style: ElevatedButton.styleFrom(
|
style: ElevatedButton.styleFrom(
|
||||||
// backgroundColor: isSelected ? AppColors.primaryBlue : Colors.white,
|
backgroundColor: isSelected ? AppColors.primaryBlue : Colors.white,
|
||||||
// foregroundColor: isSelected ? Colors.white : Colors.black,
|
foregroundColor: isSelected ? Colors.white : Colors.black,
|
||||||
// side: const BorderSide(color: Colors.grey),
|
side: const BorderSide(color: Colors.grey),
|
||||||
// padding: const EdgeInsets.symmetric(vertical: 14),
|
padding: const EdgeInsets.symmetric(vertical: 14),
|
||||||
// shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(12)),
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(12)),
|
||||||
// ),
|
),
|
||||||
// onPressed: () => controller.selectAnswerOption(index),
|
onPressed: () => controller.selectAnswerOption(index),
|
||||||
// child: Text(option),
|
child: Text(option),
|
||||||
// ),
|
),
|
||||||
// );
|
);
|
||||||
// }),
|
}),
|
||||||
// );
|
);
|
||||||
// } else if (question.type == 'true_false') {
|
} else if (question.type == 'true_false') {
|
||||||
// return Row(
|
return Row(
|
||||||
// mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
||||||
// children: [
|
children: [
|
||||||
// _buildTrueFalseButton('Ya', true, controller.choosenAnswerTOF),
|
_buildTrueFalseButton('Ya', true, controller.choosenAnswerTOF),
|
||||||
// _buildTrueFalseButton('Tidak', false, controller.choosenAnswerTOF),
|
_buildTrueFalseButton('Tidak', false, controller.choosenAnswerTOF),
|
||||||
// ],
|
],
|
||||||
// );
|
);
|
||||||
// } else {
|
} else {
|
||||||
// return GlobalTextField(controller: controller.answerTextController);
|
return GlobalTextField(controller: controller.answerTextController);
|
||||||
// }
|
}
|
||||||
// }
|
}
|
||||||
|
|
||||||
Widget _buildTrueFalseButton(String label, bool value, RxInt choosenAnswer) {
|
Widget _buildTrueFalseButton(String label, bool value, RxInt choosenAnswer) {
|
||||||
return Obx(() {
|
return Obx(() {
|
||||||
|
|
|
@ -32,7 +32,7 @@ class QuizPreviewController extends GetxController {
|
||||||
if (Get.arguments is List<QuestionData>) {
|
if (Get.arguments is List<QuestionData>) {
|
||||||
data = Get.arguments as List<QuestionData>;
|
data = Get.arguments as List<QuestionData>;
|
||||||
} else {
|
} else {
|
||||||
data = []; // Default aman supaya gak crash
|
data = [];
|
||||||
Get.snackbar('Error', 'Data soal tidak ditemukan');
|
Get.snackbar('Error', 'Data soal tidak ditemukan');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,14 +1,12 @@
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
import 'package:quiz_app/app/const/enums/question_type.dart';
|
|
||||||
import 'package:quiz_app/app/routes/app_pages.dart';
|
import 'package:quiz_app/app/routes/app_pages.dart';
|
||||||
import 'package:quiz_app/data/models/quiz/library_quiz_model.dart';
|
import 'package:quiz_app/data/models/quiz/library_quiz_model.dart';
|
||||||
import 'package:quiz_app/data/models/quiz/question_listings_model.dart';
|
import 'package:quiz_app/data/models/quiz/question/base_qustion_model.dart';
|
||||||
import 'package:quiz_app/data/models/quiz/quiestion_data_model.dart';
|
|
||||||
import 'package:quiz_app/feature/quiz_play/controller/quiz_play_controller.dart';
|
import 'package:quiz_app/feature/quiz_play/controller/quiz_play_controller.dart';
|
||||||
|
|
||||||
class QuizResultController extends GetxController {
|
class QuizResultController extends GetxController {
|
||||||
late final QuizData question;
|
late final QuizData question;
|
||||||
late final List<QuestionListing> questions;
|
late final List<BaseQuestionModel> questions;
|
||||||
late final List<AnsweredQuestion> answers;
|
late final List<AnsweredQuestion> answers;
|
||||||
|
|
||||||
RxInt correctAnswers = 0.obs;
|
RxInt correctAnswers = 0.obs;
|
||||||
|
@ -23,11 +21,12 @@ class QuizResultController extends GetxController {
|
||||||
}
|
}
|
||||||
|
|
||||||
void loadData() {
|
void loadData() {
|
||||||
final args = Get.arguments as List<dynamic>;
|
final args = Get.arguments;
|
||||||
|
|
||||||
question = args[0] as QuizData;
|
question = args['quiz_data'] as QuizData;
|
||||||
// questions = question.questionListings;
|
answers = args["answer_data"] as List<AnsweredQuestion>;
|
||||||
answers = args[1] as List<AnsweredQuestion>;
|
|
||||||
|
questions = question.questionListings;
|
||||||
totalQuestions.value = questions.length;
|
totalQuestions.value = questions.length;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -40,52 +39,9 @@ class QuizResultController extends GetxController {
|
||||||
}
|
}
|
||||||
|
|
||||||
String getResultMessage() {
|
String getResultMessage() {
|
||||||
return "Nilai kamu ${scorePercentage.value}";
|
double value = scorePercentage.value;
|
||||||
}
|
String formatted = value % 1 == 0 ? value.toStringAsFixed(0) : value.toStringAsFixed(1);
|
||||||
|
return "Nilai kamu $formatted";
|
||||||
QuestionData mapQuestionListingToQuestionData(QuestionListing questionListing, int index) {
|
|
||||||
// Convert type string ke enum
|
|
||||||
QuestionType? questionType;
|
|
||||||
switch (questionListing.type) {
|
|
||||||
case 'fill_the_blank':
|
|
||||||
questionType = QuestionType.fillTheBlank;
|
|
||||||
break;
|
|
||||||
case 'option':
|
|
||||||
questionType = QuestionType.option;
|
|
||||||
break;
|
|
||||||
case 'true_false':
|
|
||||||
questionType = QuestionType.trueOrFalse;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
questionType = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Convert options ke OptionData
|
|
||||||
List<OptionData>? optionDataList;
|
|
||||||
if (questionListing.options != null) {
|
|
||||||
optionDataList = [];
|
|
||||||
for (int i = 0; i < questionListing.options!.length; i++) {
|
|
||||||
optionDataList.add(OptionData(index: i, text: questionListing.options![i]));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Cari correctAnswerIndex kalau tipe-nya option
|
|
||||||
int? correctAnswerIndex;
|
|
||||||
if (questionType == QuestionType.option && optionDataList != null) {
|
|
||||||
correctAnswerIndex = optionDataList.indexWhere((option) => option.text == questionListing.targetAnswer);
|
|
||||||
if (correctAnswerIndex == -1) {
|
|
||||||
correctAnswerIndex = null; // Kalau tidak ketemu
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return QuestionData(
|
|
||||||
index: index,
|
|
||||||
question: questionListing.question,
|
|
||||||
answer: questionListing.targetAnswer,
|
|
||||||
options: optionDataList,
|
|
||||||
correctAnswerIndex: correctAnswerIndex,
|
|
||||||
type: questionType,
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void onPopInvoke(bool isPop, dynamic value) {
|
void onPopInvoke(bool isPop, dynamic value) {
|
||||||
|
|
|
@ -0,0 +1,167 @@
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:lucide_icons/lucide_icons.dart';
|
||||||
|
import 'package:quiz_app/app/const/colors/app_colors.dart';
|
||||||
|
import 'package:quiz_app/app/const/text/text_style.dart';
|
||||||
|
|
||||||
|
class QuizItemWAComponent extends StatelessWidget {
|
||||||
|
final int index;
|
||||||
|
final String question;
|
||||||
|
final String type;
|
||||||
|
final dynamic userAnswer;
|
||||||
|
final dynamic targetAnswer;
|
||||||
|
final bool isCorrect;
|
||||||
|
final double timeSpent;
|
||||||
|
final List<String>? options;
|
||||||
|
|
||||||
|
const QuizItemWAComponent({
|
||||||
|
super.key,
|
||||||
|
required this.index,
|
||||||
|
required this.question,
|
||||||
|
required this.type,
|
||||||
|
required this.userAnswer,
|
||||||
|
required this.targetAnswer,
|
||||||
|
required this.isCorrect,
|
||||||
|
required this.timeSpent,
|
||||||
|
this.options,
|
||||||
|
});
|
||||||
|
|
||||||
|
bool get isOptionType => type == 'option';
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Container(
|
||||||
|
width: double.infinity,
|
||||||
|
margin: const EdgeInsets.symmetric(horizontal: 16, vertical: 10),
|
||||||
|
padding: const EdgeInsets.all(20),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
color: Colors.white,
|
||||||
|
borderRadius: BorderRadius.circular(16),
|
||||||
|
boxShadow: [
|
||||||
|
BoxShadow(
|
||||||
|
color: Colors.black.withOpacity(0.04),
|
||||||
|
blurRadius: 6,
|
||||||
|
offset: const Offset(0, 2),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
'$index. $question',
|
||||||
|
style: AppTextStyles.title.copyWith(fontSize: 16, fontWeight: FontWeight.w600),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
if (isOptionType && options != null) _buildOptions(),
|
||||||
|
const SizedBox(height: 12),
|
||||||
|
_buildAnswerIndicator(),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
const Divider(height: 24, color: AppColors.shadowPrimary),
|
||||||
|
_buildMetadata(),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget _buildOptions() {
|
||||||
|
return Column(
|
||||||
|
children: options!.asMap().entries.map((entry) {
|
||||||
|
final int optIndex = entry.key;
|
||||||
|
final String text = entry.value;
|
||||||
|
|
||||||
|
final bool isCorrectAnswer = optIndex == targetAnswer;
|
||||||
|
final bool isUserWrongAnswer = optIndex == userAnswer && !isCorrectAnswer;
|
||||||
|
|
||||||
|
Color? backgroundColor;
|
||||||
|
IconData icon = LucideIcons.circle;
|
||||||
|
Color iconColor = AppColors.shadowPrimary;
|
||||||
|
|
||||||
|
if (isCorrectAnswer) {
|
||||||
|
backgroundColor = AppColors.primaryBlue.withOpacity(0.15);
|
||||||
|
icon = LucideIcons.checkCircle2;
|
||||||
|
iconColor = AppColors.primaryBlue;
|
||||||
|
} else if (isUserWrongAnswer) {
|
||||||
|
backgroundColor = Colors.red.withOpacity(0.15);
|
||||||
|
icon = LucideIcons.xCircle;
|
||||||
|
iconColor = Colors.red;
|
||||||
|
}
|
||||||
|
|
||||||
|
return Container(
|
||||||
|
width: double.infinity,
|
||||||
|
margin: const EdgeInsets.symmetric(vertical: 6),
|
||||||
|
padding: const EdgeInsets.symmetric(vertical: 10, horizontal: 12),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
color: backgroundColor,
|
||||||
|
borderRadius: BorderRadius.circular(12),
|
||||||
|
border: Border.all(color: AppColors.shadowPrimary),
|
||||||
|
),
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
Icon(icon, size: 16, color: iconColor),
|
||||||
|
const SizedBox(width: 8),
|
||||||
|
Flexible(
|
||||||
|
child: Text(text, style: AppTextStyles.optionText),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}).toList(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget _buildAnswerIndicator() {
|
||||||
|
final icon = isCorrect ? LucideIcons.checkCircle2 : LucideIcons.xCircle;
|
||||||
|
final color = isCorrect ? AppColors.primaryBlue : Colors.red;
|
||||||
|
|
||||||
|
final String userAnswerText = isOptionType ? options![userAnswer] : userAnswer.toString();
|
||||||
|
final String correctAnswerText = targetAnswer.toString();
|
||||||
|
|
||||||
|
return Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Row(
|
||||||
|
children: [
|
||||||
|
Icon(icon, color: color, size: 18),
|
||||||
|
const SizedBox(width: 8),
|
||||||
|
Text(
|
||||||
|
'Jawabanmu: $userAnswerText',
|
||||||
|
style: AppTextStyles.statValue,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
if (!isCorrect && !isOptionType) ...[
|
||||||
|
const SizedBox(height: 6),
|
||||||
|
Row(
|
||||||
|
children: [
|
||||||
|
const SizedBox(width: 26),
|
||||||
|
Text(
|
||||||
|
'Jawaban benar: $correctAnswerText',
|
||||||
|
style: AppTextStyles.caption,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
],
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget _buildMetadata() {
|
||||||
|
return Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
|
children: [
|
||||||
|
_metaItem(icon: LucideIcons.helpCircle, label: type),
|
||||||
|
_metaItem(icon: LucideIcons.clock3, label: '${timeSpent.toStringAsFixed(1)}s'),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget _metaItem({required IconData icon, required String label}) {
|
||||||
|
return Row(
|
||||||
|
children: [
|
||||||
|
Icon(icon, size: 16, color: AppColors.primaryBlue),
|
||||||
|
const SizedBox(width: 6),
|
||||||
|
Text(label, style: AppTextStyles.caption),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,8 +1,12 @@
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
|
import 'package:lucide_icons/lucide_icons.dart';
|
||||||
import 'package:quiz_app/app/const/colors/app_colors.dart';
|
import 'package:quiz_app/app/const/colors/app_colors.dart';
|
||||||
import 'package:quiz_app/component/widget/question_container_widget.dart';
|
import 'package:quiz_app/data/models/quiz/question/fill_in_the_blank_question_model.dart';
|
||||||
|
import 'package:quiz_app/data/models/quiz/question/option_question_model.dart';
|
||||||
|
import 'package:quiz_app/data/models/quiz/question/true_false_question_model.dart';
|
||||||
import 'package:quiz_app/feature/quiz_result/controller/quiz_result_controller.dart';
|
import 'package:quiz_app/feature/quiz_result/controller/quiz_result_controller.dart';
|
||||||
|
import 'package:quiz_app/feature/quiz_result/view/component/quiz_item_wa_component.dart';
|
||||||
|
|
||||||
class QuizResultView extends GetView<QuizResultController> {
|
class QuizResultView extends GetView<QuizResultController> {
|
||||||
const QuizResultView({super.key});
|
const QuizResultView({super.key});
|
||||||
|
@ -20,21 +24,11 @@ class QuizResultView extends GetView<QuizResultController> {
|
||||||
child: Obx(() => Column(
|
child: Obx(() => Column(
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
_buildCustomAppBar(context),
|
_buildAppBar(context),
|
||||||
const SizedBox(height: 16),
|
const SizedBox(height: 16),
|
||||||
_buildScoreSummary(),
|
_buildScoreSummary(),
|
||||||
const SizedBox(height: 16),
|
const SizedBox(height: 16),
|
||||||
Expanded(
|
_buildQuizList(),
|
||||||
child: ListView.builder(
|
|
||||||
itemCount: controller.questions.length,
|
|
||||||
itemBuilder: (context, index) {
|
|
||||||
return QuestionContainerWidget(
|
|
||||||
question: controller.mapQuestionListingToQuestionData(controller.questions[index], index),
|
|
||||||
answeredQuestion: controller.answers[index],
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
],
|
||||||
)),
|
)),
|
||||||
),
|
),
|
||||||
|
@ -43,14 +37,12 @@ class QuizResultView extends GetView<QuizResultController> {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget _buildCustomAppBar(BuildContext context) {
|
Widget _buildAppBar(BuildContext context) {
|
||||||
return Row(
|
return Row(
|
||||||
children: [
|
children: [
|
||||||
IconButton(
|
IconButton(
|
||||||
icon: const Icon(Icons.arrow_back, color: Colors.black),
|
icon: const Icon(LucideIcons.arrowLeft, color: Colors.black),
|
||||||
onPressed: () {
|
onPressed: () => controller.onPopInvoke(true, null),
|
||||||
controller.onPopInvoke(true, null);
|
|
||||||
},
|
|
||||||
),
|
),
|
||||||
const SizedBox(width: 8),
|
const SizedBox(width: 8),
|
||||||
const Text(
|
const Text(
|
||||||
|
@ -62,6 +54,7 @@ class QuizResultView extends GetView<QuizResultController> {
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget _buildScoreSummary() {
|
Widget _buildScoreSummary() {
|
||||||
|
final score = controller.scorePercentage.value;
|
||||||
return Column(
|
return Column(
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
|
@ -71,7 +64,7 @@ class QuizResultView extends GetView<QuizResultController> {
|
||||||
),
|
),
|
||||||
const SizedBox(height: 8),
|
const SizedBox(height: 8),
|
||||||
LinearProgressIndicator(
|
LinearProgressIndicator(
|
||||||
value: controller.scorePercentage.value / 100,
|
value: score / 100,
|
||||||
minHeight: 10,
|
minHeight: 10,
|
||||||
backgroundColor: AppColors.borderLight,
|
backgroundColor: AppColors.borderLight,
|
||||||
valueColor: const AlwaysStoppedAnimation<Color>(AppColors.primaryBlue),
|
valueColor: const AlwaysStoppedAnimation<Color>(AppColors.primaryBlue),
|
||||||
|
@ -81,10 +74,57 @@ class QuizResultView extends GetView<QuizResultController> {
|
||||||
controller.getResultMessage(),
|
controller.getResultMessage(),
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
fontSize: 16,
|
fontSize: 16,
|
||||||
color: controller.scorePercentage.value >= 80 ? Colors.green : Colors.red,
|
color: score >= 80 ? Colors.green : Colors.red,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Widget _buildQuizList() {
|
||||||
|
return Expanded(
|
||||||
|
child: ListView.builder(
|
||||||
|
itemCount: controller.questions.length,
|
||||||
|
itemBuilder: (context, index) {
|
||||||
|
final answer = controller.answers[index];
|
||||||
|
final question = controller.questions.firstWhere(
|
||||||
|
(q) => q.index == answer.questionIndex,
|
||||||
|
orElse: () => throw Exception("Question not found"),
|
||||||
|
);
|
||||||
|
|
||||||
|
final parsed = _parseAnswer(question, answer.selectedAnswer);
|
||||||
|
|
||||||
|
return QuizItemWAComponent(
|
||||||
|
index: index,
|
||||||
|
isCorrect: answer.isCorrect,
|
||||||
|
question: question.question,
|
||||||
|
targetAnswer: parsed.targetAnswer,
|
||||||
|
userAnswer: parsed.userAnswer,
|
||||||
|
timeSpent: answer.duration.toDouble(),
|
||||||
|
type: question.type,
|
||||||
|
options: parsed.options,
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Helper class for parsed answer details
|
||||||
|
({dynamic userAnswer, dynamic targetAnswer, List<String> options}) _parseAnswer(dynamic question, dynamic selectedAnswer) {
|
||||||
|
switch (question.type) {
|
||||||
|
case 'fill_the_blank':
|
||||||
|
final q = question as FillInTheBlankQuestion;
|
||||||
|
return (userAnswer: selectedAnswer.toString(), targetAnswer: q.targetAnswer, options: []);
|
||||||
|
case 'option':
|
||||||
|
final q = question as OptionQuestion;
|
||||||
|
final parsedAnswer = int.tryParse(selectedAnswer.toString()) ?? -1;
|
||||||
|
return (userAnswer: parsedAnswer, targetAnswer: q.targetAnswer, options: q.options);
|
||||||
|
case 'true_false':
|
||||||
|
final q = question as TrueFalseQuestion;
|
||||||
|
final boolAnswer = selectedAnswer is bool ? selectedAnswer : selectedAnswer.toString().toLowerCase() == 'true';
|
||||||
|
return (userAnswer: boolAnswer, targetAnswer: q.targetAnswer, options: []);
|
||||||
|
default:
|
||||||
|
throw Exception("Unknown question type: ${question.type}");
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue