File size: 11,855 Bytes
7ba3b4e c9ca893 bad6a07 e0f928e bbc7ac2 d924141 c9ca893 e0f928e e9df8e8 795b9ba 169cec0 c9ca893 169cec0 c9ca893 9d5f38a d7cd069 08df020 9d5f38a 32f2e18 795b9ba 169cec0 795b9ba bad6a07 c9ca893 bbc7ac2 169cec0 e9df8e8 169cec0 81e422f 169cec0 a26033f 81e422f 169cec0 bbc7ac2 169cec0 9d5f38a bbc7ac2 169cec0 9d5f38a bbc7ac2 169cec0 9d5f38a bbc7ac2 81e422f bbc7ac2 d3582b2 bbc7ac2 169cec0 32f2e18 8e6cd67 32f2e18 c9ca893 58956eb bbc7ac2 c9ca893 58956eb c9ca893 f1969d5 81e422f 2a4a631 f1969d5 c9ca893 81e422f ee53acf e9df8e8 08df020 ce2a304 08df020 169cec0 c9ca893 e9df8e8 58956eb 7b6a611 58956eb e9df8e8 58956eb d6d844d 08df020 e9df8e8 81e422f c9ca893 e9df8e8 054ca33 58956eb ce2a304 58956eb bbc7ac2 bc85220 1f50ad3 bc85220 1f50ad3 bbc7ac2 199933f d3582b2 a39e202 a26033f 58d0b6d d015779 fedff07 d3582b2 10939a2 58d0b6d ce2a304 08df020 58956eb a39e202 08df020 10939a2 199933f c7df340 d015779 81e422f bbc7ac2 |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 |
import streamlit as st
import os
from fpdf import FPDF
import uuid
import importlib
# Initialize session state variables
if 'session_id' not in st.session_state:
st.session_state.session_id = str(uuid.uuid4())
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 = {}
if 'module_question_count' not in st.session_state:
st.session_state.module_question_count = {}
if 'pdf_data' not in st.session_state:
st.session_state.pdf_data = None
if 'selected_answer' not in st.session_state:
st.session_state.selected_answer = None
if 'answered' not in st.session_state:
st.session_state.answered = False
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, data in st.session_state.module_question_count.items():
pdf.set_text_color(0, 0, 0) # Reset to default color
pdf.set_font("Arial", size=12, style='B')
pdf.cell(200, 10, txt=f"Module: {module}", ln=True, align="L")
pdf.ln(5)
for entry in st.session_state.questions:
if entry['module'] == module:
question, options, selected, correct, explanation, step_by_step_solution = (
entry['question'],
entry['options'],
entry['selected'] if entry['selected'] is not None else "Not Answered",
entry['correct_answer'],
entry['explanation'],
entry['step_by_step_solution']
)
# Draw a border around the question and options with a light background
pdf.set_draw_color(0, 0, 0) # Black border
pdf.set_fill_color(240, 240, 240) # Light gray background for question
pdf.set_line_width(0.3)
# Question
pdf.multi_cell(0, 10, f"Q: {question}", border=1, align='L', fill=True)
pdf.ln(5)
# Options with distinct background
for option in options:
if option == correct:
pdf.set_text_color(0, 128, 0) # Green for correct
pdf.set_fill_color(200, 255, 200) # Light green background
pdf.multi_cell(0, 10, f"{option}", border=1, align='L', fill=True)
elif option == selected:
pdf.set_text_color(255, 0, 0) # Red for incorrect
pdf.set_fill_color(255, 200, 200) # Light red background
pdf.multi_cell(0, 10, f"{option}", border=1, align='L', fill=True)
else:
pdf.set_text_color(0, 0, 0) # Default color for others
pdf.set_fill_color(240, 240, 240) # Light gray background
pdf.multi_cell(0, 10, f" {option}", border=1, align='L', fill=True)
pdf.ln(5)
# Explanation with background
pdf.set_text_color(0, 0, 0) # Reset to black
pdf.set_fill_color(230, 230, 250) # Light blue background
pdf.multi_cell(0, 10, f"Explanation: {explanation}", border=1, align='L', fill=True)
pdf.ln(5)
# Step-by-step solution with background
pdf.set_fill_color(250, 235, 215) # Light tan background
pdf.multi_cell(0, 10, "Step-by-Step Solution:", border=1, align='L', fill=True)
for step in step_by_step_solution:
pdf.multi_cell(0, 10, step, border=1, align='L', fill=True)
pdf.ln(10)
pdf.ln(10) # Add space after each module
return pdf.output(dest='S').encode('latin1', 'replace')
def load_modules():
modules = {}
module_dir = "modules"
for filename in os.listdir(module_dir):
if filename.endswith(".py") and filename != "__init__.py":
module_name = filename[:-3]
# Dynamically import the module only when needed
module = importlib.import_module(f"modules.{module_name}")
modules[module_name] = {
"title": getattr(module, "title", module_name),
"description": getattr(module, "description", "No description available."),
"generate_question": module.generate_question # Access the generate_question function
}
return modules
def generate_new_question(module_name, module):
question_data = module['generate_question']()
# Ensure 'answered' is initialized to False and add the 'module' and 'selected' keys
question_data['answered'] = False
question_data['module'] = module_name # Add the module name to the question data
question_data['selected'] = None # Initialize 'selected' to None
# Ensure there are exactly 4 options
if len(question_data['options']) != 4:
st.warning(f"Question in module '{module_name}' does not have 4 options. Found {len(question_data['options'])}.")
return question_data
def navigate_question(direction):
if direction == "prev" and st.session_state.current_index > 0:
st.session_state.current_index -= 1
st.session_state.answered = True
elif direction == "next" and st.session_state.current_index < len(st.session_state.questions) - 1:
st.session_state.current_index += 1
st.session_state.answered = True
# Load all modules dynamically
modules = load_modules()
# Streamlit interface
st.sidebar.title("Quiz Modules")
module_name = st.sidebar.radio("Choose a module:", [modules[module]["title"] for module in modules], index=0)
selected_module = None
for module in modules:
if modules[module]["title"] == module_name:
selected_module = module
break
if selected_module != st.session_state.current_module:
st.session_state.current_module = selected_module
st.session_state.current_index = 0
st.session_state.questions = [generate_new_question(selected_module, modules[selected_module])]
st.session_state.module_question_count[selected_module] = 0
st.session_state.module_correct_count[selected_module] = 0
st.session_state.selected_answer = None
st.session_state.answered = False
# Load the current module's question
current_question = st.session_state.questions[st.session_state.current_index]
# Display module title and description
st.title(modules[selected_module]["title"])
st.write(modules[selected_module]["description"])
# Navigation and PDF report buttons
col1, col2, col3 = st.columns([1, 1, 2])
with col1:
if st.button("⬅️ Prev", disabled=st.session_state.current_index == 0):
navigate_question("prev")
with col2:
if st.button("➡️ Next", disabled=st.session_state.current_index >= len(st.session_state.questions) - 1):
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"
)
# Web Page Presentation Styling
def display_question_with_styles(question_data):
st.markdown(f"""
<style>
.question-box {{
border: 2px solid #000;
padding: 10px;
margin-bottom: 20px;
background-color: #f0f0f0;
}}
.option {{
border: 1px solid #000;
padding: 5px;
margin-bottom: 5px;
}}
.correct {{
background-color: #c8e6c9;
color: green;
}}
.incorrect {{
background-color: #ffcdd2;
color: red;
}}
.explanation-box {{
border: 2px solid #000;
padding: 10px;
background-color: #e3f2fd;
margin-top: 20px;
}}
.step-box {{
border: 1px solid #000;
padding: 5px;
background-color: #fff3e0;
margin-top: 10px;
}}
</style>
""", unsafe_allow_html=True)
st.markdown(f"""
<div class="question-box">
<strong>Q:</strong> {question_data['question']}
</div>
""", unsafe_allow_html=True)
if question_data.get('answered', False):
st.markdown(f"<div>", unsafe_allow_html=True)
for option in question_data['options']:
option_class = ''
if option == question_data['correct_answer']:
option_class = 'correct'
elif option == question_data['selected']:
option_class = 'incorrect'
st.markdown(f"""
<div class="option {option_class}">
{option}
</div>
""", unsafe_allow_html=True)
st.markdown(f"</div>", unsafe_allow_html=True)
st.markdown(f"""
<div class="explanation-box">
<strong>Explanation:</strong> {question_data['explanation']}
</div>
""", unsafe_allow_html=True)
st.markdown(f"""
<div class="step-box">
<strong>Step-by-Step Solution:</strong>
<ul>
""", unsafe_allow_html=True)
for step in question_data['step_by_step_solution']:
st.markdown(f"<li>{step}</li>", unsafe_allow_html=True)
st.markdown("</ul></div>", unsafe_allow_html=True)
# Display the current question with styles
display_question_with_styles(current_question)
# Disable the radio buttons if the question is already answered
disabled_options = current_question['answered']
# Create the form for the question
with st.form(key=f'question_form_{st.session_state.current_index}'):
selected_answer = st.radio(
"Choose an answer:",
options=current_question['options'],
index=current_question['options'].index(current_question['selected']) if current_question['answered'] else None,
disabled=disabled_options, # Disable if the question has been answered
key=f"question_{st.session_state.current_index}_options"
)
submit_button = st.form_submit_button(label="Submit/Next")
# Handle button state and answer submission
if submit_button:
if not current_question['answered']:
if selected_answer is not None:
# Process the answer
current_question['selected'] = selected_answer
current_question['answered'] = True
st.session_state.module_question_count[selected_module] += 1
if selected_answer == current_question['correct_answer']:
st.session_state.correct_count += 1
st.session_state.module_correct_count[selected_module] += 1
# Set answered to true to disable options
st.session_state.questions[st.session_state.current_index]['answered'] = True
st.session_state.selected_answer = selected_answer
else:
# If already answered, move to the next question
new_question = generate_new_question(selected_module, modules[selected_module])
st.session_state.questions.append(new_question)
st.session_state.current_index = len(st.session_state.questions) - 1
st.session_state.answered = False
# Show correct/incorrect feedback after submission
if current_question.get('answered', False):
display_question_with_styles(current_question)
|