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