File size: 9,054 Bytes
87ed062
 
 
 
 
 
 
 
 
 
 
db3291d
 
 
0165bb9
db3291d
 
 
 
0165bb9
 
 
 
 
 
 
db3291d
 
0165bb9
db3291d
0165bb9
 
 
 
db3291d
 
0165bb9
 
 
 
 
 
 
 
 
 
 
 
 
 
 
db3291d
0165bb9
 
 
 
 
 
 
 
 
 
 
db3291d
 
 
 
 
0165bb9
 
 
 
 
db3291d
0165bb9
db3291d
0165bb9
db3291d
0165bb9
 
 
 
 
 
db3291d
 
0165bb9
 
 
 
 
 
 
db3291d
 
d697ee0
db3291d
 
 
0165bb9
 
 
 
 
 
 
 
db3291d
0165bb9
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
db3291d
0165bb9
 
 
 
db3291d
3b05c91
db3291d
 
 
0165bb9
 
db3291d
0165bb9
db3291d
3b05c91
 
db3291d
3b05c91
 
 
 
 
 
 
 
 
 
 
0165bb9
db3291d
 
0165bb9
db3291d
 
 
 
0165bb9
 
179a824
0165bb9
db3291d
0165bb9
db3291d
 
 
 
 
 
 
 
 
 
 
 
 
0165bb9
db3291d
0165bb9
 
 
 
db3291d
 
 
 
 
 
fad3d57
db3291d
 
 
 
 
 
 
 
 
 
 
179a824
 
db3291d
 
 
 
 
 
fad3d57
db3291d
 
 
 
 
 
 
 
0165bb9
 
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
import gradio as gr
import uuid
import os
import tempfile
import hashlib
from reportlab.lib.pagesizes import A5, landscape
from reportlab.pdfgen import canvas
from reportlab.lib.units import mm
from reportlab.lib.colors import HexColor
from datetime import datetime

# --- Certificate Generator (This function is part of the generated code) ---
# Note: This function is defined here for completeness but is also embedded
# in the generated string.
def generate_certificate(name, score, total, instructor="Instructor"):
    """
    Generates a PDF certificate of completion.
    This entire function will be part of the output code string.
    """
    unique_id = str(uuid.uuid4())
    filename = f"cert_{unique_id}.pdf"
    filepath = os.path.join(tempfile.gettempdir(), filename)

    c = canvas.Canvas(filepath, pagesize=landscape(A5))
    width, height = landscape(A5)

    # Set background and border
    c.setFillColor(HexColor("#fffdf6")) # Creamy background
    c.rect(0, 0, width, height, stroke=0, fill=1)
    c.setStrokeColor(HexColor("#001858")) # Dark blue border
    c.setLineWidth(3)
    margin = 10 * mm
    c.rect(margin, margin, width - 2 * margin, height - 2 * margin)

    # Add text content
    c.setFillColor(HexColor("#001858")) # Dark blue text
    c.setFont("Helvetica-Bold", 24)
    c.drawCentredString(width / 2, height - 60, "Certificate of Completion")

    c.setFont("Helvetica", 14)
    c.drawCentredString(width / 2, height - 100, "This is awarded to")

    c.setFont("Helvetica-Bold", 18)
    c.drawCentredString(width / 2, height - 130, name)

    c.setFont("Helvetica", 14)
    c.drawCentredString(width / 2, height - 160, "For successfully completing the quiz")

    c.setFont("Helvetica", 12)
    c.drawCentredString(width / 2, height - 185, f"Score: {score} / {total}")

    # Footer
    c.setFont("Helvetica-Oblique", 10)
    c.drawString(margin + 10, margin + 20, f"Instructor: {instructor}")
    date_str = datetime.now().strftime("%d %B %Y")
    c.setFont("Helvetica", 10)
    c.drawRightString(width - margin - 10, margin + 20, f"Issued on: {date_str}")

    c.save()
    return filepath

# --- Main Quiz Code Generator ---
def generate_python_code(title, instructor, quiz_type, questions_text):
    """
    Parses instructor input and generates a self-contained Python script
    for a student-facing Gradio quiz app.
    """
    # Parse questions and hash the answers for security
    parsed_questions = []
    for line in questions_text.strip().split("\n"):
        if not line.strip():
            continue
        parts = [p.strip() for p in line.split(",")]

        if quiz_type == "Multiple Choice":
            if len(parts) < 3: continue # Skip malformed lines
            q_text = parts[0]
            options = parts[1:-1]
            answer = parts[-1]
            parsed_questions.append({
                "question": q_text,
                "options": options,
                "answer_hash": hashlib.sha256(answer.lower().encode()).hexdigest()
            })
        else: # Text Answer
            if len(parts) < 2: continue # Skip malformed lines
            q_text = parts[0]
            answer = parts[1]
            parsed_questions.append({
                "question": q_text,
                "answer_hash": hashlib.sha256(answer.lower().encode()).hexdigest()
            })

    # Generate the complete Python code string for the student quiz app
    # f-strings with {{ and }} are used to escape braces for the final code.
    python_code = f'''!pip install reportlab
# --- Generated Quiz App ---
# Copy and paste this entire code block into a single Google Colab cell and run it.

import gradio as gr
import uuid, os, tempfile, hashlib
from reportlab.lib.pagesizes import A5, landscape
from reportlab.pdfgen import canvas
from reportlab.lib.units import mm
from reportlab.lib.colors import HexColor
from datetime import datetime

# Certificate generation function (included for a self-contained script)
def generate_certificate(name, score, total, instructor="{instructor}"):
    unique_id = str(uuid.uuid4())
    filename = f"cert_{{unique_id}}.pdf"
    filepath = os.path.join(tempfile.gettempdir(), filename)
    c = canvas.Canvas(filepath, pagesize=landscape(A5))
    width, height = landscape(A5)
    c.setFillColor(HexColor("#fffdf6"))
    c.rect(0, 0, width, height, stroke=0, fill=1)
    c.setStrokeColor(HexColor("#001858"))
    c.setLineWidth(3)
    margin = 10 * mm
    c.rect(margin, margin, width - 2 * margin, height - 2 * margin)
    c.setFillColor(HexColor("#001858"))
    c.setFont("Helvetica-Bold", 24)
    c.drawCentredString(width / 2, height - 60, "Certificate of Completion")
    c.setFont("Helvetica", 14)
    c.drawCentredString(width / 2, height - 100, "This is awarded to")
    c.setFont("Helvetica-Bold", 18)
    c.drawCentredString(width / 2, height - 130, name)
    c.setFont("Helvetica", 14)
    c.drawCentredString(width / 2, height - 160, "For successfully completing the quiz")
    c.setFont("Helvetica", 12)
    c.drawCentredString(width / 2, height - 185, f"Score: {{score}} / {{total}}")
    c.setFont("Helvetica-Oblique", 10)
    c.drawString(margin + 10, margin + 20, f"Instructor: {{instructor}}")
    date_str = datetime.now().strftime("%d %B %Y")
    c.setFont("Helvetica", 10)
    c.drawRightString(width - margin - 10, margin + 20, f"Issued on: {{date_str}}")
    c.save()
    return filepath

# Quiz data (answers are hashed)
quiz_type = "{quiz_type}"
questions = {parsed_questions}

def eval_quiz(name, *answers):
    """
    Evaluates the student's answers and generates a certificate only if the score is 80% or higher.
    """
    if not name.strip():
        name = "Anonymous"
    score = 0
    for i, ans in enumerate(answers):
        if ans and hashlib.sha256(str(ans).lower().strip().encode()).hexdigest() == questions[i]["answer_hash"]:
            score += 1
    
    total_questions = len(questions)
    passing_threshold = 0.8
    
    result_message = f"Hi {{name}}, your score is: {{score}} / {{total_questions}}."
    cert_path = None # Default to no certificate

    # Check if the score meets the passing threshold
    if total_questions > 0 and (score / total_questions) >= passing_threshold:
        cert_path = generate_certificate(name, score, total_questions, instructor="{instructor}")
        result_message += " Congratulations, you passed and earned a certificate!"
    else:
        result_message += " A score of 80% is required to receive a certificate."

    return result_message, cert_path

# Gradio interface for the student
with gr.Blocks(theme=gr.themes.Soft()) as app:
    gr.Markdown("## {title}")
    
    with gr.Row():
        name = gr.Textbox(label="Enter Your Full Name to Generate Certificate", placeholder="e.g., Ada Lovelace")

    answer_inputs = []
    for q in questions:
        gr.Markdown("**Question:** " + q['question'])
        if quiz_type == "Multiple Choice":
            answer_inputs.append(gr.Radio(choices=q["options"], label="Select your answer"))
        else:
            answer_inputs.append(gr.Textbox(label="Type your answer"))

    submit_btn = gr.Button("Submit Quiz")
    
    with gr.Row():
        result_output = gr.Textbox(label="Your Result")
        certificate_output = gr.File(label="Download Your Certificate")

    submit_btn.click(
        fn=eval_quiz, 
        inputs=[name] + answer_inputs, 
        outputs=[result_output, certificate_output]
    )

app.launch(debug=True)
'''
    return python_code

# --- Instructor Interface ---
with gr.Blocks(theme=gr.themes.Base()) as interface:
    gr.Markdown("# Instructor Quiz Generator")
    gr.Markdown("Create a secure, interactive quiz for your Google Colab notebooks.")

    with gr.Row():
        # Left column for inputs
        with gr.Column(scale=2):
            title = gr.Textbox(label="Quiz Title", placeholder="e.g. Python Basics Quiz")
            instructor = gr.Textbox(label="Instructor Name", placeholder="e.g. Dr. Ada Lovelace")
            quiz_type = gr.Dropdown(
                choices=["Multiple Choice", "Text Answer"], 
                label="Quiz Type", 
                value="Multiple Choice"
            )
            questions = gr.Textbox(
                lines=10, 
                label="Questions & Answers",
                placeholder=(
                    "One question per line. Separate parts with commas.\\n\\n"
                    "MCQ Format: Question,Option1,Option2,CorrectOption\\n"
                    "Text Format: Question,CorrectAnswer"
                )
            )
            generate_btn = gr.Button("πŸš€ Generate Python Quiz Code", variant="primary")

        # Right column for the generated code output
        with gr.Column(scale=1):
            output = gr.Code(label="Generated Python Code for Colab", language="python", lines=22)

    # Link the button to the generation function
    generate_btn.click(
        fn=generate_python_code, 
        inputs=[title, instructor, quiz_type, questions], 
        outputs=output
    )

interface.launch()