From be2986182c6af3d24ac8b02dea3c5ff986c5049f Mon Sep 17 00:00:00 2001
From: wkyas <14069237+wkyas@user.noreply.gitee.com>
Date: Thu, 20 Mar 2025 14:27:56 +0000
Subject: [PATCH] code

Signed-off-by: wkyas <14069237+wkyas@user.noreply.gitee.com>
---
 __pycache__/config.cpython-311.pyc       | Bin 0 -> 748 bytes
 app/__init__.py                          |  21 +++++
 app/__pycache__/__init__.cpython-311.pyc | Bin 0 -> 1109 bytes
 app/__pycache__/models.cpython-311.pyc   | Bin 0 -> 2791 bytes
 app/__pycache__/routes.cpython-311.pyc   | Bin 0 -> 8071 bytes
 app/assessments.db                       | Bin 0 -> 16384 bytes
 app/models.py                            |  24 +++++
 app/routes.py                            | 110 +++++++++++++++++++++++
 app/templates/create_assessment.html     |  46 ++++++++++
 app/templates/formative_test.html        |  48 ++++++++++
 app/templates/index.html                 |  14 +++
 app/templates/manage_questions.html      |  47 ++++++++++
 app/templates/summaritive_test.html      |  48 ++++++++++
 app/templates/take_assessment.html       |  15 ++++
 assessments.db                           |   0
 config.py                                |   6 ++
 init_db.py                               |  45 ++++++++++
 instance/assessments.db                  | Bin 0 -> 16384 bytes
 requirements.txt                         |   4 +
 run.py                                   |   6 ++
 20 files changed, 434 insertions(+)
 create mode 100644 __pycache__/config.cpython-311.pyc
 create mode 100644 app/__init__.py
 create mode 100644 app/__pycache__/__init__.cpython-311.pyc
 create mode 100644 app/__pycache__/models.cpython-311.pyc
 create mode 100644 app/__pycache__/routes.cpython-311.pyc
 create mode 100644 app/assessments.db
 create mode 100644 app/models.py
 create mode 100644 app/routes.py
 create mode 100644 app/templates/create_assessment.html
 create mode 100644 app/templates/formative_test.html
 create mode 100644 app/templates/index.html
 create mode 100644 app/templates/manage_questions.html
 create mode 100644 app/templates/summaritive_test.html
 create mode 100644 app/templates/take_assessment.html
 create mode 100644 assessments.db
 create mode 100644 config.py
 create mode 100644 init_db.py
 create mode 100644 instance/assessments.db
 create mode 100644 requirements.txt
 create mode 100644 run.py

diff --git a/__pycache__/config.cpython-311.pyc b/__pycache__/config.cpython-311.pyc
new file mode 100644
index 0000000000000000000000000000000000000000..a4422bbabaf3827596ee2f2c7b4d04e954d4cba3
GIT binary patch
literal 748
zcmZ3^%ge<81RC|1(q9AV#~=<2us|7~HGqui3@HpLj5!QZ3``8}3@J=43@OaZm>3vV
z12F_dF{UsEGib8B1abY6p`u`l8Or(G1Y}HSNM!)3Wr$)*WzK@x!LWc8Sp$rn!nlkP
zXa^8OKmlV4QxqE!`j|2Fu|%<_utssDumRo3ev8dHKQApa{T5fSt8<WRNW8aeWR*x|
zeyMJGW=@W7UTRrtk#2fvYH{%`9v8<DM<>T%*Z9yNpDKys!ko;KR4aXb{lwy8pnPs>
zUP-ZDN|KwOCg&}V`1riU+|>B^TU_z+x%nxjIUqJqe0*VPVh&V>DZls@dum=;W>J3L
zE#~ypk|L1LZ;1y7`Z)SHd${^W#v|G2c}osSA|%Ms**o6X-^J6-)7de^)88+6CBtV>
z(0p|O5-wIT<@rU~F-E2a#=3sG1||lE3JNdwu79$7+tck!pRedvP%zdrG<e#-{mHH+
zG0FKUsquNK<v<1~DD(;{i$JjfCO{EZ><J_q;P8f^@D!=(a+Bm1@LUwsy&|Z4LD2nz
zp?ibZ4LOB5S@ZK|<*kUjC}(p;&gOz#)CGsA3oOz%RJAU!C_zw>0L*MnrXo<9gILB7
zVu7UufvjH~Hjq%WD-r^7LD63903<#zGcq#XVBl;3!w)RXjM^U<Fo};Ku`gI8z%~E?
DA=kA@

literal 0
HcmV?d00001

diff --git a/app/__init__.py b/app/__init__.py
new file mode 100644
index 0000000..d2f3793
--- /dev/null
+++ b/app/__init__.py
@@ -0,0 +1,21 @@
+from flask import Flask
+from flask_sqlalchemy import SQLAlchemy
+
+# 定义模块级别的 db 对象
+db = SQLAlchemy()
+
+# 定义模块级别的 app 对象
+app = None
+
+def create_app():
+    global app  # 使用 global 关键字将 app 定义为全局变量
+    app = Flask(__name__)
+    # app.config.from_object('config.Config')
+    app.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:///assessments.db'
+    db.init_app(app)
+    
+    with app.app_context():
+        from . import routes  # 导入路由
+        db.create_all()  # 创建数据库表
+        
+    return app
\ No newline at end of file
diff --git a/app/__pycache__/__init__.cpython-311.pyc b/app/__pycache__/__init__.cpython-311.pyc
new file mode 100644
index 0000000000000000000000000000000000000000..acb3e65bf3c0377ea0a698f531f843adc8d08e78
GIT binary patch
literal 1109
zcmZWo-Afcv6u)=Ac6Qfgqy$9-7ka3Lbl21ukzoBIKN89cTm~-F&S;x6yW7mjY7lI(
zQJUbB^y))d;zMr<MGyHGtmK0X2th98+lC3!Q|FHDuA_%Lzd7gJ^L2lB=3OWhKp=&^
zZ_~pJLO-Zc5wKl3?1k5Bq#_mT$iNAVsm|z(!6sOY7^Jal5V3I*4Vd6mR^e5yfD(er
zPM`>XK=&fp;fHj^ngJy+aceBDCm(3W{CKjO2#Np?3(sK}rcG_YirAHbz_AAYjq0Al
zsMzKzaR@!66<kpnI=n6ZH{rF)*YuipI@?<tqtkn<biQ-qJ$^R=f>@Q)eTMi1bWeBx
zxnyv7^Idv@;x#z$M)k}XrGuAH2eP;vGsjM%99+XBRQJwD$sT{)WIfVTw$>AkMitA_
zEX&Z+w$-lQcN!`0WBoS<Z%&ed_?>uPd}5H?y*+$Dxm?GNU}k1*&58s}N}(eXl2#0j
z5QkN=Sw~1_(o?Bvhk;6BDxI<k^#%b5?AY3z?I4q$&<P|>O|dni=(<VCw`l$F(GcK3
zkDSYxGxD|WSZCXKTdXVA5f1<Q@?rn;$I|B8pYLCV!=3FNvC{J9{--x`GNWoFt>t8B
zA`{{^CGFYyT8C`bya;9KcWi9}<WVUgt%wWJrD(qY`_-Z6{e_XDbY+{1>~X@2(PyJ8
z>^9f1!!;DShLXQ-xplF1+uyw7Z!RN-Z*pN*Y$<xzo)}mTEru4PCF!M9<|~12^lzH?
zr<z<Vb4EKpK0f1d%_9-jN%;}$0?hukCEO>nTO!w2=i8DnV8rL}Q}m}fxsZBSCvd;Y
z$=_qr&)y7C2vVR_tfNCfWYUU%#a<($&g$BAvk6Rk4c0w?GK(?ZMU6%8+C`U(-c=Tm
cP@hjcl?!s70mB!}vt=%br7}7XW+RmT2X+zdumAu6

literal 0
HcmV?d00001

diff --git a/app/__pycache__/models.cpython-311.pyc b/app/__pycache__/models.cpython-311.pyc
new file mode 100644
index 0000000000000000000000000000000000000000..9bea3d9da4400740161b10fce2711b2c9f610f78
GIT binary patch
literal 2791
zcmds3O>7fK6y9C`|HY0;Dgq^qP%B}oAd@(VqAIj$N)t#Tl@O_k0;^4C2piUGGrLwW
z2kJu)ks7H+vWk&>;6SMnM$HLOt4cj`+^zOt&9$I5H&E!Mr@mSL5Sx&2=&7^o&$B!4
z&Ahk!y>Ir%P{_|gqt}05y5r=yKiEa1x$2$GJaleyh(o->CHXm?=a7ILijWlM1fCPP
zFF54<hC?pQFFP8}ccFK`c{S&>t=wSc*}=+XTY16Cw}X`%`KP&9V1tLXVmuYl!UkIv
zx?UvdSeKJ{u@MIQch@gA$H3zzhq*Z(adU#eVaGD$Tx-y8<O_LLgu4R755HQQbGKUC
zSDwZwa@>WP@3VO9hMW~wo4<EQfA_Bbz8(E{U*F&AzcpKXB+!iAvrR0m?=pzI@cez$
z`(*}ex|~wa-Q#H>P2{9Rvf^c&-FOL);uPHUsu@L*78JZO0P|l9Nt%W=Es0hAh>WIU
zA?g)HRZ3z}q<&FMrcg#<{h%mb$w*3VgfJ3O_h%_3lT@i^T-EUsCRC6Sbx-R=R+p$#
z$5(a0Y#P&!#S|fUQ5PjuTLC-&sT9HTl6nSb>6Z0H8QtSHS<-)x0*s!RUrCY6^8-Wi
z!TzcKcp`qZukWv4e|q}M!)K3v__O?NU*F)7qw#0oJ$m}{o%zKS!lH^-=B0Fc9-_vI
zb|jr8j1M+2QfUtWa?PXai>^l3dh!E>6Xk^Ai<`c<<%{P=s)4T6fwhnFv-z`y*`jvi
zLg7Lg8-auwNLYbHZnSlr+&Eu2U!E}ngJxjR3Jk({cdzZ4Es|2Uyyw>S()GtKqkGuw
z9=5uNb5qr>1No7{cyXb$T<$L)tn@z)n}^4&!(+zuS4P*Y*)?l*&E_WALH0M|Z>gR}
zRcuE7H`N2|w^a}FWwqX3s{y+rf^3vE<UwAr^0g^isHQ^H|7nxd=ZF{VH=Y2TBCMiF
zhymjT^}phNOrV|xY4I|_7pYT`u4SiU9i)R<h8aW{Fw&Dw22lok7<2(pmoDoHChxGm
zcNFCA9LOCAAC{0JtC$7}RwP!G+9f$neOKxws*!zc3i}nm%X+;HU~y{S1H1)UgWanm
zYvXx2zfh2GJ8q~2wVW}6LuPQu3J&Ey-<7Q4y{lv`Ta1?WmvtkYFvAHeoXCw;Bk$#f
zLio0Sed0IWh)kG~2`e&@8)ru?WEO1;E1Q}=|DUmPt0p2x#o4Cdo?R4_y?VPLSDS*e
zx!`?n*$uZ-^H+>)-wj@mrN&SuGj@Z>U$OOd>(FbFGqTk@!^@0rcB^>WGVZMFe0W;V
zAkgfySW5%1aE<0QhF=Rb7zT<?SPrladl0@TWfXlX7JUm|Q@>W5A2gZlg&AmbPIkv;
zy_eM2rk;^hO2Lv!{jw(3wVu2WUW5r1b!YTNHMK&0>?=UirDVFD_%UY7fEAYZ5a2C{
z-?=XzE%X!zN+;G`M&~iJ^O)6nEO)vZ-Jd^Lh!salr^}(TPzkLk%|oNsq0uK9BRXS7
zXRPQ<?o72#`yL>E&+Wlmy`|pu4~+0}Gkn|%9|!usY4%7Dl!dQ&u=HX1YUOJqa?*^P
zv?3?B76waXD(0be#ju}&Ym#kAwe83*b35hM`v1rKac#RDCJC5=eY$8Sz;g%B^HnZl
zw4PP&fYEwBce>z9@Hw|VLYZHEVUNAsYSH21CvBhY5T5W}<}=ANpQc5JoBsrSwnq@6
OOE2uPmv6AJ_2XZYZEHvX

literal 0
HcmV?d00001

diff --git a/app/__pycache__/routes.cpython-311.pyc b/app/__pycache__/routes.cpython-311.pyc
new file mode 100644
index 0000000000000000000000000000000000000000..3020cd639e47287a28ac0d8be660de8f076c40fe
GIT binary patch
literal 8071
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

literal 0
HcmV?d00001

diff --git a/app/assessments.db b/app/assessments.db
new file mode 100644
index 0000000000000000000000000000000000000000..849772d88c6bb75561bfacb38769c75f79d59d81
GIT binary patch
literal 16384
zcmeI&-*3`D902euWgx_PCT7_a@fLK0Ot7?tj=jt>yGdlAUDfP~ba0c#@<V#4dvw{e
z{xkj${tf!*qko0hjS9tqzL=Q!y|lUOeZTaMPy5nd51-Z?M#)Kk*tHo^&`l%==m8-J
zA<XwE--9CZ#`Lie1Qyg7<xjpZ^91qwH-z_ZZ14&S2!H?xfB*=900@8p2!H?x{3U_2
ziWpzb<%DyM*~c#R*r-i=tU09Kpx^U;Igaj{hHe?e(knHC%x05JGU2pIwQd>vhDquT
zi`1XhYWGMo5v0xuzaH_SoPMu4Cmzg)JlD<LN4l9QN>Vm7fa5h={o#<d7^yTGHAAn5
z!p!ON5$x`OJmmAYst1NYt0!jlKsS%bW8;WqoOagl+iRFcb-(WSWu~T;C8n`wm_~iq
zI3z*;#gCHNvs5I$no0@h%NK}2G^oW8TF|=yi_v#%{y$}>13DEAGmH(IcF%iDhf`@Y
zT(-wAuGjUi^&Ilw*@Vsz(wRiYKJQCHe06P2IJ>;bCh<ut5;B_fFH&-tJAsgAZ-|9V
zZECk&r$-mw4auzkY%bu>h;L9p00ck)1V8`;KmY_l00ck)1V8`;{%rwGSVEh(b&J0N
zBKpr4AD658&KoLAvXU#vIi)~kWxH7BTVA>-%lIpb;vIxP;SadO8&E(11V8`;KmY_l
z00ck)1V8`;KmY`01vZvq=>8^7yy)<+=N*s8WZ;f?wbLEB%o(_}+3NJ27FF*=V<;O+
z+`KAF{G+T$4CU5V5>=0|4kg~JQ*!K%=nfl>X!FE&J!<h5pX&PWNA%bKV>pHI7ybs|
zBYuf1`~V6FfB*=900@8p2!H?xfB*=900{g~0_j*BB?O$%qbgrPFXkk^h^~?XUqRQl
z<#JwC3Q|c+FO8GLE0dJ+mReBOqkhs#LXY^5=2x9eQ8i7L)8d>F)vax{oY#tKSt_L?
fb4DB|6?mF@SMZZ=UDG2+)C&)InN}(*a!LLHlU-`0

literal 0
HcmV?d00001

diff --git a/app/models.py b/app/models.py
new file mode 100644
index 0000000..801a971
--- /dev/null
+++ b/app/models.py
@@ -0,0 +1,24 @@
+from . import db
+from datetime import datetime
+
+class Question(db.Model):
+    id = db.Column(db.Integer, primary_key=True)
+    text = db.Column(db.String(500), nullable=False)
+    type = db.Column(db.String(50), nullable=False)  # e.g., 'multiple_choice', 'true_false'
+    correct_answer = db.Column(db.String(500), nullable=False)
+    assessment_id = db.Column(db.Integer, db.ForeignKey('assessment.id'), nullable=False)
+
+class Assessment(db.Model):
+    id = db.Column(db.Integer, primary_key=True)
+    title = db.Column(db.String(200), nullable=False)
+    type = db.Column(db.String(50), nullable=False)  # 'formative' or 'summative'
+    deadline = db.Column(db.DateTime, nullable=True)
+    questions = db.relationship('Question', backref='assessment', lazy=True)
+
+class StudentResponse(db.Model):
+    id = db.Column(db.Integer, primary_key=True)
+    student_id = db.Column(db.Integer, nullable=False)
+    question_id = db.Column(db.Integer, db.ForeignKey('question.id'), nullable=False)
+    response = db.Column(db.String(500), nullable=False)
+    is_correct = db.Column(db.Boolean, nullable=False)
+    timestamp = db.Column(db.DateTime, default=datetime.utcnow)
\ No newline at end of file
diff --git a/app/routes.py b/app/routes.py
new file mode 100644
index 0000000..e9d0534
--- /dev/null
+++ b/app/routes.py
@@ -0,0 +1,110 @@
+from flask import request, jsonify, render_template
+from . import app, db
+from .models import Assessment, Question, StudentResponse
+from datetime import datetime
+from sqlalchemy import func
+
+# 定义路由
+@app.route('/')
+def index():
+    return render_template('index.html')
+
+@app.route('/create_assessment', methods=['GET', 'POST'])
+def create_assessment():
+    if request.method == 'POST':
+        data = request.json
+        new_assessment = Assessment(
+            title=data['title'],
+            type=data['type'],
+            deadline=datetime.strptime(data['deadline'], '%Y-%m-%d %H:%M:%S') if data.get('deadline') else None
+        )
+        db.session.add(new_assessment)
+        db.session.commit()
+        return jsonify({"message": "Assessment created", "id": new_assessment.id}), 201
+    else:
+        return render_template('create_assessment.html')
+
+@app.route('/add_question', methods=['POST'])
+def add_question():
+    data = request.json
+    new_question = Question(
+        text=data['text'],
+        type=data['type'],
+        correct_answer=data['correct_answer'],
+        assessment_id=data['assessment_id']
+    )
+    db.session.add(new_question)
+    db.session.commit()
+    return jsonify({"message": "Question added", "id": new_question.id}), 201
+
+# 显示评估选择页面
+@app.route('/take_assessment', methods=['GET'])
+def take_assessment():
+    return render_template('take_assessment.html')
+
+#显示形成性评估页面
+@app.route('/take_assessment/formative_test', methods=['GET'])
+def formative_test():
+    # 查询所有形成性评估
+    assessment = Assessment.query.filter_by(type='formative').first()
+    if assessment:
+        questions = Question.query.filter_by(assessment_id=assessment.id).all()
+        return render_template('formative_test.html', assessment=assessment, questions=questions)
+    else:
+        return "No formative assessment available.", 404
+
+# 显示总结性评估页面
+@app.route('/take_assessment/summative_test', methods=['GET'])
+def summative_test():
+    # 查询所有总结性评估
+    assessment = Assessment.query.filter_by(type='summative').first()
+    if assessment:
+        questions = Question.query.filter_by(assessment_id=assessment.id).all()
+        return render_template('summative_test.html', assessment=assessment, questions=questions)
+    else:
+        return "No summative assessment available.", 404
+
+# @app.route('/take_assessment/<int:assessment_id>', methods=['GET'])
+# def take_assessment(assessment_id):
+#     assessment = Assessment.query.get_or_404(assessment_id)
+#     questions = Question.query.filter_by(assessment_id=assessment_id).all()
+#     return render_template('take_assessment.html', assessment=assessment, questions=questions)
+
+# 管理问题页面
+@app.route('/manage_questions', methods=['GET'])
+def manage_questions():
+    return render_template('manage_questions.html')
+
+@app.route('/submit_response', methods=['POST'])
+def submit_response():
+    data = request.json
+    question = Question.query.get_or_404(data['question_id'])
+    is_correct = data['response'] == question.correct_answer
+    new_response = StudentResponse(
+        student_id=data['student_id'],
+        question_id=data['question_id'],
+        response=data['response'],
+        is_correct=is_correct
+    )
+    db.session.add(new_response)
+    db.session.commit()
+    return jsonify({"message": "Response submitted", "is_correct": is_correct}), 201
+
+@app.route('/get_results/<int:assessment_id>', methods=['GET'])
+def get_results(assessment_id):
+    assessment = Assessment.query.get_or_404(assessment_id)
+    responses = StudentResponse.query.join(Question).filter(Question.assessment_id == assessment_id).all()
+    results = [{"student_id": r.student_id, "question_id": r.question_id, "response": r.response, "is_correct": r.is_correct} for r in responses]
+    
+    # 计算统计信息
+    total_students = db.session.query(StudentResponse.student_id).distinct().count()
+    average_score = db.session.query(func.avg(StudentResponse.is_correct)).filter(Question.assessment_id == assessment_id).scalar()
+    most_incorrect_question = db.session.query(Question.text, func.count(StudentResponse.id)).join(StudentResponse).filter(StudentResponse.is_correct == False).group_by(Question.text).order_by(func.count(StudentResponse.id).desc()).first()
+    
+    statistics = {
+        "total_students": total_students,
+        "average_score": average_score,
+        "most_incorrect_question": most_incorrect_question[0] if most_incorrect_question else None
+    }
+    
+    return jsonify({"results": results, "statistics": statistics}), 200
\ No newline at end of file
diff --git a/app/templates/create_assessment.html b/app/templates/create_assessment.html
new file mode 100644
index 0000000..b03b440
--- /dev/null
+++ b/app/templates/create_assessment.html
@@ -0,0 +1,46 @@
+<!DOCTYPE html>
+<html lang="en">
+<head>
+    <meta charset="UTF-8">
+    <meta name="viewport" content="width=device-width, initial-scale=1.0">
+    <title>Create Assessment</title>
+</head>
+<body>
+    <h1>Create Assessment</h1>
+    <form id="createAssessmentForm">
+        <label for="title">Title:</label>
+        <input type="text" id="title" name="title" required>
+        <br>
+        <label for="type">Type:</label>
+        <select id="type" name="type" required>
+            <option value="formative">Formative</option>
+            <option value="summative">Summative</option>
+        </select>
+        <br>
+        <label for="deadline">Deadline (for summative):</label>
+        <input type="datetime-local" id="deadline" name="deadline">
+        <br>
+        <button type="submit">Create Assessment</button>
+    </form>
+
+    <script>
+        document.getElementById('createAssessmentForm').onsubmit = async function(e) {
+            e.preventDefault();
+            const formData = {
+                title: document.getElementById('title').value,
+                type: document.getElementById('type').value,
+                deadline: document.getElementById('deadline').value
+            };
+            const response = await fetch('/create_assessment', {
+                method: 'POST',
+                headers: {
+                    'Content-Type': 'application/json'
+                },
+                body: JSON.stringify(formData)
+            });
+            const result = await response.json();
+            alert(result.message);
+        };
+    </script>
+</body>
+</html>
\ No newline at end of file
diff --git a/app/templates/formative_test.html b/app/templates/formative_test.html
new file mode 100644
index 0000000..45cd1d4
--- /dev/null
+++ b/app/templates/formative_test.html
@@ -0,0 +1,48 @@
+<!DOCTYPE html>
+<html lang="en">
+<head>
+    <meta charset="UTF-8">
+    <meta name="viewport" content="width=device-width, initial-scale=1.0">
+    <title>Formative Test</title>
+</head>
+<body>
+    <h1>Formative Test: {{ assessment.title }}</h1>
+    <div id="questions">
+        {% if questions %}
+            {% for question in questions %}
+            <div>
+                <p><strong>Question {{ loop.index }}:</strong> {{ question.text }}</p>
+                <input type="text" id="response_{{ question.id }}" placeholder="Your answer">
+            </div>
+            {% endfor %}
+        {% else %}
+            <p>No questions available for this assessment.</p>
+        {% endif %}
+    </div>
+    <button onclick="submitResponses()">Submit</button>
+
+    <script>
+        async function submitResponses() {
+            const assessmentId = {{ assessment.id | tojson | safe }};
+            const studentId = 1; // 假设学生 ID 为 1
+            const questions = document.querySelectorAll('[id^="response_"]');
+            for (const question of questions) {
+                const questionId = question.id.split('_')[1];
+                const response = question.value;
+                await fetch('/submit_response', {
+                    method: 'POST',
+                    headers: {
+                        'Content-Type': 'application/json'
+                    },
+                    body: JSON.stringify({
+                        student_id: studentId,
+                        question_id: questionId,
+                        response: response
+                    })
+                });
+            }
+            alert('Responses submitted!');
+        }
+    </script>
+</body>
+</html>
\ No newline at end of file
diff --git a/app/templates/index.html b/app/templates/index.html
new file mode 100644
index 0000000..d1a8ab1
--- /dev/null
+++ b/app/templates/index.html
@@ -0,0 +1,14 @@
+<!DOCTYPE html>
+<html lang="en">
+<head>
+    <meta charset="UTF-8">
+    <meta name="viewport" content="width=device-width, initial-scale=1.0">
+    <title>Automated Assessment Tool</title>
+</head>
+<body>
+    <h1>Automated Assessment Tool</h1>
+    <button onclick="location.href='/create_assessment'">Create Assessment</button>
+    <button onclick="location.href='/take_assessment'">Take Assessment</button>
+    <button onclick="location.href='/manage_questions'">Manage Questions</button>
+</body>
+</html>
\ No newline at end of file
diff --git a/app/templates/manage_questions.html b/app/templates/manage_questions.html
new file mode 100644
index 0000000..2561cae
--- /dev/null
+++ b/app/templates/manage_questions.html
@@ -0,0 +1,47 @@
+<!DOCTYPE html>
+<html lang="en">
+<head>
+    <meta charset="UTF-8">
+    <meta name="viewport" content="width=device-width, initial-scale=1.0">
+    <title>Manage Questions</title>
+</head>
+<body>
+    <h1>Manage Questions</h1>
+    <form id="addQuestionForm">
+        <label for="question_text">Question Text:</label>
+        <input type="text" id="question_text" name="question_text" required>
+        <br>
+        <label for="question_type">Question Type:</label>
+        <select id="question_type" name="question_type" required>
+            <option value="multiple_choice">Multiple Choice</option>
+            <option value="true_false">True/False</option>
+        </select>
+        <br>
+        <label for="correct_answer">Correct Answer:</label>
+        <input type="text" id="correct_answer" name="correct_answer" required>
+        <br>
+        <button type="submit">Add Question</button>
+    </form>
+
+    <script>
+        document.getElementById('addQuestionForm').onsubmit = async function(e) {
+            e.preventDefault();
+            const formData = {
+                text: document.getElementById('question_text').value,
+                type: document.getElementById('question_type').value,
+                correct_answer: document.getElementById('correct_answer').value,
+                assessment_id: 1 // 假设评估 ID 为 1
+            };
+            const response = await fetch('/add_question', {
+                method: 'POST',
+                headers: {
+                    'Content-Type': 'application/json'
+                },
+                body: JSON.stringify(formData)
+            });
+            const result = await response.json();
+            alert(result.message);
+        };
+    </script>
+</body>
+</html>
\ No newline at end of file
diff --git a/app/templates/summaritive_test.html b/app/templates/summaritive_test.html
new file mode 100644
index 0000000..d2a8ff3
--- /dev/null
+++ b/app/templates/summaritive_test.html
@@ -0,0 +1,48 @@
+<!DOCTYPE html>
+<html lang="en">
+<head>
+    <meta charset="UTF-8">
+    <meta name="viewport" content="width=device-width, initial-scale=1.0">
+    <title>Summative Test</title>
+</head>
+<body>
+    <h1>Summative Test: {{ assessment.title }}</h1>
+    <div id="questions">
+        {% if questions %}
+            {% for question in questions %}
+            <div>
+                <p><strong>Question {{ loop.index }}:</strong> {{ question.text }}</p>
+                <input type="text" id="response_{{ question.id }}" placeholder="Your answer">
+            </div>
+            {% endfor %}
+        {% else %}
+            <p>No questions available for this assessment.</p>
+        {% endif %}
+    </div>
+    <button onclick="submitResponses()">Submit</button>
+
+    <script>
+        async function submitResponses() {
+            const assessmentId = {{ assessment.id | tojson | safe }};
+            const studentId = 1; // 假设学生 ID 为 1
+            const questions = document.querySelectorAll('[id^="response_"]');
+            for (const question of questions) {
+                const questionId = question.id.split('_')[1];
+                const response = question.value;
+                await fetch('/submit_response', {
+                    method: 'POST',
+                    headers: {
+                        'Content-Type': 'application/json'
+                    },
+                    body: JSON.stringify({
+                        student_id: studentId,
+                        question_id: questionId,
+                        response: response
+                    })
+                });
+            }
+            alert('Responses submitted!');
+        }
+    </script>
+</body>
+</html>
\ No newline at end of file
diff --git a/app/templates/take_assessment.html b/app/templates/take_assessment.html
new file mode 100644
index 0000000..6466dc4
--- /dev/null
+++ b/app/templates/take_assessment.html
@@ -0,0 +1,15 @@
+<!DOCTYPE html>
+<html lang="en">
+<head>
+    <meta charset="UTF-8">
+    <meta name="viewport" content="width=device-width, initial-scale=1.0">
+    <title>Take Assessment</title>
+</head>
+<body>
+    <h1>Take Assessment</h1>
+    <div>
+        <button onclick="window.location.href='/take_assessment/formative_test'">Formative Test</button>
+        <button onclick="window.location.href='/take_assessment/summative_test'">Summative Test</button>
+    </div>
+</body>
+</html>
\ No newline at end of file
diff --git a/assessments.db b/assessments.db
new file mode 100644
index 0000000..e69de29
diff --git a/config.py b/config.py
new file mode 100644
index 0000000..badcfa7
--- /dev/null
+++ b/config.py
@@ -0,0 +1,6 @@
+import os
+
+class Config:
+    SECRET_KEY = os.environ.get('SECRET_KEY') or 'you-will-never-guess'
+    SQLALCHEMY_DATABASE_URI = os.environ.get('DATABASE_URL') or 'sqlite:///assessments.db'
+    SQLALCHEMY_TRACK_MODIFICATIONS = False
\ No newline at end of file
diff --git a/init_db.py b/init_db.py
new file mode 100644
index 0000000..1bf902e
--- /dev/null
+++ b/init_db.py
@@ -0,0 +1,45 @@
+from app import create_app, app, db
+from app.models import Assessment, Question
+from datetime import datetime
+
+with app.app_context():
+    # 创建数据库表
+    db.create_all()
+
+    # 创建形成性评估
+    formative_assessment = Assessment(title="Formative Test", type="formative")
+    db.session.add(formative_assessment)
+    db.session.commit()
+
+    # 添加形成性评估的问题
+    formative_questions = [
+        {"text": "What is 2 + 2?", "type": "multiple_choice", "correct_answer": "4"},
+        {"text": "What is the capital of France?", "type": "multiple_choice", "correct_answer": "Paris"},
+        {"text": "Is the sky blue?", "type": "true_false", "correct_answer": "True"},
+        {"text": "What is 5 * 5?", "type": "multiple_choice", "correct_answer": "25"},
+        {"text": "Is water H2O?", "type": "true_false", "correct_answer": "True"}
+    ]
+    for q in formative_questions:
+        question = Question(text=q['text'], type=q['type'], correct_answer=q['correct_answer'], assessment_id=formative_assessment.id)
+        db.session.add(question)
+    db.session.commit()
+
+    # 创建总结性评估
+    summative_assessment = Assessment(title="Summative Test", type="summative", deadline=datetime(2023, 12, 31, 23, 59, 59))
+    db.session.add(summative_assessment)
+    db.session.commit()
+
+    # 添加总结性评估的问题
+    summative_questions = [
+        {"text": "What is 10 - 4?", "type": "multiple_choice", "correct_answer": "6"},
+        {"text": "What is the capital of Germany?", "type": "multiple_choice", "correct_answer": "Berlin"},
+        {"text": "Is the Earth flat?", "type": "true_false", "correct_answer": "False"},
+        {"text": "What is 8 / 2?", "type": "multiple_choice", "correct_answer": "4"},
+        {"text": "Is the sun a star?", "type": "true_false", "correct_answer": "True"}
+    ]
+    for q in summative_questions:
+        question = Question(text=q['text'], type=q['type'], correct_answer=q['correct_answer'], assessment_id=summative_assessment.id)
+        db.session.add(question)
+    db.session.commit()
+
+    print("Database initialized with test data.")
\ No newline at end of file
diff --git a/instance/assessments.db b/instance/assessments.db
new file mode 100644
index 0000000000000000000000000000000000000000..05e765dcf522083f6b2f17d394293f802f140fa3
GIT binary patch
literal 16384
zcmeI#O-sWt7zgmy3d7;sZKuKG$rV&YQLlD0&SC8|R>9M#HbP<RW}BfN9ez@eehO1v
z={i|Gc^UseNb@{tlm3!RuCMGs3VQ5EJx|iUwyx>Ac1)ycnxX2Vs%e?->uI8(HME__
z&lNSmSpLw~nyZ_tL4g1SAOHafKmY;|fB*y_@J|Axx>2fZZ|m<z(!1}9SPp#=%1$KW
zK|hRt^%onQSuP{js@sgFM^ml56!_F~T-IcqoVH8OjcspHc_|$_BmFc~Qw9C7Gb5j_
zhi)z2IJ0=|V0YKdO%TMLhkg`^2TApI+h&%NOUt0AX7G9gI#KI)TNf-@)g^CTSp1I8
z*&Wpa-%MhsZO&RvCyCW2mSqxWr;IbF!LBKdf4@{VN7aQ=rCQbBSH2O`Y}y(*v@}|P
zCB=)J{ipJEASSYThOt4%3*%=IO$?jj@?y2QaWC2H8RUQ2<i3#8$ueV~w?n;D+1SuW
zKZh*KXHN?`qby#e6v%D{u{V7n3YmQ2`Q0ECg{PrxCU<jn=|_P81Rwwb2tWV=5P$##
zAOHafKwypqaQ~m<&Bd=l00Izz00bZa0SG_<0uX=z1jYim|Klh?00Izz00bZa0SG_<
M0uX=z1m<7h3#^&+5&!@I

literal 0
HcmV?d00001

diff --git a/requirements.txt b/requirements.txt
new file mode 100644
index 0000000..3d5ec8a
--- /dev/null
+++ b/requirements.txt
@@ -0,0 +1,4 @@
+Flask==2.0.1
+Flask-SQLAlchemy==2.5.1
+werkzeug==2.0.3
+SQLAlchemy==1.4.41
\ No newline at end of file
diff --git a/run.py b/run.py
new file mode 100644
index 0000000..3cc9827
--- /dev/null
+++ b/run.py
@@ -0,0 +1,6 @@
+from app import create_app
+
+app = create_app()
+
+if __name__ == '__main__':
+    app.run(debug=True)
\ No newline at end of file
-- 
GitLab