import streamlit as st import importlib import logging from fpdf import FPDF import uuid # Configure logging logging.basicConfig(level=logging.INFO) # List of available modules with shorter names and icons module_names = { "Bases": "presentation_bases", "Validity": "valid_invalid_numbers", "Conversion": "conversion_bases", "Grouping": "grouping_techniques", "Addition": "addition_bases", "2's Complement": "twos_complement", "Negative Numbers": "negative_binary", "Subtraction": "subtraction_bases", } # Initialize unique session state if 'session_id' not in st.session_state: st.session_state.session_id = str(uuid.uuid4()) # Initialize session state variables if 'questions' not in st.session_state: st.session_state.questions = [] if 'current_index' not in st.session_state: st.session_state.current_index = 0 if 'current_module' not in st.session_state: st.session_state.current_module = None if 'correct_count' not in st.session_state: st.session_state.correct_count = 0 if 'module_correct_count' not in st.session_state: st.session_state.module_correct_count = {name: 0 for name in module_names} if 'module_question_count' not in st.session_state: st.session_state.module_question_count = {name: 0 for name in module_names} if 'selected_answer' not in st.session_state: st.session_state.selected_answer = None if 'pdf_data' not in st.session_state: st.session_state.pdf_data = None def reset_pdf_cache(): st.session_state.pdf_data = None def generate_pdf_report(): pdf = FPDF() pdf.add_page() pdf.set_font("Arial", size=12) pdf.cell(200, 10, txt="Quiz Report", ln=True, align="C") pdf.ln(10) for module in module_names.keys(): pdf.set_text_color(0, 0, 0) pdf.cell(200, 10, txt=f"Module: {module}", ln=True, align="L") pdf.ln(5) correct_count = st.session_state.module_correct_count[module] total_count = st.session_state.module_question_count[module] pdf.cell(200, 10, txt=f"Correct Answers: {correct_count}/{total_count}", ln=True, align="L") pdf.ln(5) for entry in st.session_state.questions: if entry['module'] == module: question, options, selected, correct, explanation = ( entry['question'], entry['options'], entry['selected'], entry['correct'], entry['explanation'] ) pdf.multi_cell(0, 10, f"Q: {question}") for option in options: if option == correct: pdf.set_text_color(0, 128, 0) # Green for correct pdf.multi_cell(0, 10, f"{option}") elif option == selected: pdf.set_text_color(255, 0, 0) # Red for incorrect pdf.multi_cell(0, 10, f"{option}") else: pdf.set_text_color(0, 0, 0) # Default color for others pdf.multi_cell(0, 10, f" {option}") pdf.set_text_color(0, 0, 0) # Reset color pdf.multi_cell(0, 10, f"Explanation: {explanation}") pdf.ln(10) pdf.ln(10) # Add space after each module return pdf.output(dest='S').encode('latin1', 'replace') def load_module(module_name): module_file = module_names[module_name] module = importlib.import_module(f'modules.{module_file}') return module def generate_new_question(module_name): module = load_module(module_name) question, options, correct_answer, explanation = module.generate_question() return { 'module': module_name, 'question': question, 'options': options, 'correct': correct_answer, 'explanation': explanation, 'selected': None, 'answered': False } def navigate_question(direction): if direction == "prev" and st.session_state.current_index > 0: st.session_state.current_index -= 1 elif direction == "next": if st.session_state.current_index < len(st.session_state.questions) - 1: st.session_state.current_index += 1 else: # Generate a new question if at the end of the list new_question = generate_new_question(st.session_state.current_module) st.session_state.questions.append(new_question) st.session_state.current_index += 1 # Streamlit interface st.sidebar.title("Quiz Modules") module_name = st.sidebar.radio("Choose a module:", list(module_names.keys()), index=0) if module_name != st.session_state.current_module: st.session_state.current_module = module_name st.session_state.current_index = 0 st.session_state.questions = [generate_new_question(module_name)] # Load the current module for title and description current_module = load_module(st.session_state.current_module) current_question = st.session_state.questions[st.session_state.current_index] # Display module title and description st.title(current_module.title) st.write(current_module.description) # Button Row: Prev, Next, and PDF Download col1, col2, col3 = st.columns([1, 1, 2]) with col1: st.button("⬅️ Prev", disabled=st.session_state.current_index == 0, on_click=lambda: navigate_question("prev")) with col2: st.button("➡️ Next", on_click=lambda: navigate_question("next")) with col3: if len(st.session_state.questions) > 0: pdf = generate_pdf_report() st.session_state.pdf_data = pdf # Reset PDF cache st.download_button( label="Download PDF Report 📄", data=st.session_state.pdf_data, file_name="quiz_report.pdf", mime="application/pdf" ) # Display the current question st.write(f"**Question {st.session_state.current_index + 1}:** {current_question['question']}") # Option highlighting logic def get_option_style(option): if current_question['answered']: if option == current_question['correct']: return "background-color:#d4edda;padding:10px;border-radius:5px;" # Green background for correct elif option == current_question['selected']: return "background-color:#f8d7da;padding:10px;border-radius:5px;" # Red background for incorrect return "background-color:#f0f0f0;padding:10px;border-radius:5px;" # Gray background by default selected_answer = st.radio( "Choose an answer:", current_question['options'], index=current_question['options'].index(current_question['selected']) if current_question['selected'] else None, key=st.session_state.current_index, format_func=lambda x: f"{x}" ) if st.button("Submit"): if not current_question['answered']: current_question['selected'] = selected_answer current_question['answered'] = True st.session_state.module_question_count[module_name] += 1 if selected_answer == current_question['correct']: st.session_state.correct_count += 1 st.session_state.module_correct_count[module_name] += 1 # Retain and highlight the options after submission for option in current_question['options']: st.markdown(f"
{option}
", unsafe_allow_html=True)