import gradio as gr import matplotlib as mpl from quiz import BenchmarkQuiz, BENCHMARKS, QuestionData mpl.rcParams["figure.dpi"] = 300 def update_quiz_screen(quiz: BenchmarkQuiz, question_data: QuestionData): quiz_state = quiz.state return { quiz_screen: gr.update(visible=True), question_number: gr.update(value=question_data.question_num), question_text: gr.update(value=question_data.question), answer_input: gr.update( value=quiz_state.user_answers[quiz_state.current_question], choices=question_data.options, visible=BENCHMARKS[quiz_state.benchmark_name]["type"] == "multiple_choice", label=question_data.instruction, ), free_text_input: gr.update( value=quiz_state.user_answers[quiz_state.current_question], visible=BENCHMARKS[quiz_state.benchmark_name]["type"] == "free_text", label=question_data.instruction, ), next_button: gr.update( value=question_data.next_button_text, ), previous_button: gr.update(visible=question_data.previous_button_visibility), } def update_score_screen(plot, results_data): updates = { score_screen: gr.update(visible=True), score_plot: gr.update(value=plot), results_container: gr.update(visible=True), } for i, result in enumerate(results_data): updates[result_cards[i]] = gr.update( visible=True, elem_classes=[ "correct-answer" if result["points"] == 1.0 else "semi-correct-answer" if result["points"] == 0.5 else "incorrect-answer" ], ) emoji = "✅" if result["points"] == 1.0 else "❌" if result["points"] == 0.0 else "🔶" markdown_string = f"### {emoji} Spurning {result['question_num']}" markdown_string += f"\n{result['question']}" markdown_string += ( f"\n\nValkostir:\n" + "\n".join([f"- {option}" for option in result["options"]]) if result["options"] else "" ) markdown_string += f"\n\n**Þitt svar:** {result['user_answer']}" markdown_string += f"\n\n**Rétt svar:** {result['correct_answer']}" markdown_string += f"\n\n**Stig:** {result['points']}" updates[result_cards[i].children[0]] = gr.update(value=markdown_string) return updates def start_quiz_handler(benchmark_name): quiz = BenchmarkQuiz() quiz.start_quiz(benchmark_name) question_data = quiz.update_question() return { start_screen: gr.update(visible=False), score_screen: gr.update(visible=False), **update_quiz_screen(quiz, question_data), quiz_state: quiz, } def next_question_handler(quiz, answer_input, free_text_input): answer = ( answer_input if BENCHMARKS[quiz.state.benchmark_name]["type"] == "multiple_choice" else free_text_input ) result = quiz.next_question(answer) if result["completed"]: return { quiz_screen: gr.update(visible=False), **update_score_screen(result["plot"], result["results_data"]), quiz_state: quiz, } else: return {**update_quiz_screen(quiz, result["question_data"]), quiz_state: quiz} def previous_question_handler(quiz): question_data = quiz.previous_question() return {**update_quiz_screen(quiz, question_data), quiz_state: quiz} def reset_quiz_handler(): return { start_screen: gr.update(visible=True), quiz_screen: gr.update(visible=False), score_screen: gr.update(visible=False), next_button: gr.update(visible=True), } demo = gr.Blocks( theme=gr.themes.Soft(), title="Mælipróf", css=""" .correct-answer, .correct-answer .block { background-color: #d4edda !important; border-color: #c3e6cb !important; } .incorrect-answer, .incorrect-answer .block { background-color: #f8d7da !important; border-color: #f5c6cb !important; } .semi-correct-answer, .semi-correct-answer .block { background-color: #fff3cd !important; border-color: #ffeeba !important; } .correct-answer, .incorrect-answer, .semi-correct-answer { border-radius: 5px; padding: 10px; margin-bottom: 10px; } """, ) with demo: start_screen = gr.Column(visible=True) with start_screen: gr.Markdown("# Veldu mælipróf") benchmark_buttons = { name: gr.Button(info["name"]) for name, info in BENCHMARKS.items() } quiz_screen = gr.Column(visible=False) with quiz_screen: question_number = gr.Markdown() question_text = gr.Markdown() answer_input = gr.Radio(choices=[], visible=False) free_text_input = gr.Textbox(visible=False) with gr.Row(): previous_button = gr.Button("Fyrri") next_button = gr.Button("Næsta") score_screen = gr.Column(visible=False) with score_screen: gr.Markdown(f"## Niðurstöður") score_plot = gr.Plot() reset_btn = gr.Button("Byrja upp á nýtt") results_container = gr.Column(visible=False) with results_container: result_cards = [gr.Group(visible=False) for _ in range(5)] for card in result_cards: with card: gr.Markdown("") quiz_state = gr.State() for benchmark_name, button in benchmark_buttons.items(): button.click( fn=start_quiz_handler, inputs=[gr.State(benchmark_name)], outputs=[ start_screen, quiz_screen, score_screen, question_number, question_text, answer_input, free_text_input, next_button, previous_button, quiz_state, ], ) next_button.click( fn=next_question_handler, inputs=[quiz_state, answer_input, free_text_input], outputs=[ quiz_screen, score_screen, question_number, question_text, answer_input, free_text_input, next_button, previous_button, score_plot, results_container, *result_cards, *[child for card in result_cards for child in card.children], quiz_state, ], ) previous_button.click( fn=previous_question_handler, inputs=[quiz_state], outputs=[ quiz_screen, question_number, question_text, answer_input, free_text_input, next_button, previous_button, quiz_state, ], ) reset_btn.click( fn=reset_quiz_handler, inputs=[], outputs=[start_screen, quiz_screen, score_screen, next_button], ) demo.launch()