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)