Building new test form
Added CRUD for tests
This commit is contained in:
		
							
								
								
									
										5
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -146,4 +146,7 @@ out/ | |||||||
| ref-test/testing.py | ref-test/testing.py | ||||||
|  |  | ||||||
| # Ignore Encryption Keyfile | # Ignore Encryption Keyfile | ||||||
| .encryption.key | .encryption.key | ||||||
|  |  | ||||||
|  | # Ignore Font Binaries | ||||||
|  | **/fonts/ | ||||||
| @@ -1,6 +1,7 @@ | |||||||
| from flask_wtf import FlaskForm | from flask_wtf import FlaskForm | ||||||
| from wtforms import StringField, PasswordField, BooleanField | from wtforms import StringField, PasswordField, BooleanField, DateField, SelectField | ||||||
| from wtforms.validators import InputRequired, Email, Length, EqualTo, Optional | from wtforms.validators import InputRequired, Email, Length, EqualTo, Optional, ValidationError | ||||||
|  | from datetime import date, timedelta | ||||||
|  |  | ||||||
| class LoginForm(FlaskForm): | class LoginForm(FlaskForm): | ||||||
|     username = StringField('Username', validators=[InputRequired(), Length(min=4, max=15)]) |     username = StringField('Username', validators=[InputRequired(), Length(min=4, max=15)]) | ||||||
| @@ -41,4 +42,14 @@ class UpdateAccountForm(FlaskForm): | |||||||
|     password_confirm = PasswordField('Current Password', validators=[InputRequired(), Length(min=6, max=30, message='The password must be between 6 and 20 characters long.')]) |     password_confirm = PasswordField('Current Password', validators=[InputRequired(), Length(min=6, max=30, message='The password must be between 6 and 20 characters long.')]) | ||||||
|     email = StringField('Email Address', validators=[Optional(), Email(message='You must enter a valid email address.'), Length(max=50)]) |     email = StringField('Email Address', validators=[Optional(), Email(message='You must enter a valid email address.'), Length(max=50)]) | ||||||
|     password = PasswordField('Change Password', validators=[Optional(),Length(min=6, max=30, message='The password must be between 6 and 20 characters long.')]) |     password = PasswordField('Change Password', validators=[Optional(),Length(min=6, max=30, message='The password must be between 6 and 20 characters long.')]) | ||||||
|     password_reenter = PasswordField('Re-Enter New Password', validators=[EqualTo('password', message='Passwords do not match.')]) |     password_reenter = PasswordField('Re-Enter New Password', validators=[EqualTo('password', message='Passwords do not match.')]) | ||||||
|  |  | ||||||
|  | class CreateTest(FlaskForm): | ||||||
|  |     time_options = [ | ||||||
|  |         ('none', 'None'), | ||||||
|  |         ('60', '1 hour'), | ||||||
|  |         ('90', '1 hour 30 minutes'), | ||||||
|  |         ('120', '2 hours') | ||||||
|  |     ] | ||||||
|  |     expiry = DateField('Expiry Date', format="%Y-%m-%d", validators=[InputRequired()], default = date.today() + timedelta(days=1) ) | ||||||
|  |     time_limit = SelectField('Time Limit', choices=time_options) | ||||||
							
								
								
									
										45
									
								
								ref-test/admin/models.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								ref-test/admin/models.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | |||||||
|  | import secrets | ||||||
|  | from datetime import datetime | ||||||
|  | from uuid import uuid4 | ||||||
|  | from flask import flash, jsonify | ||||||
|  |  | ||||||
|  | from main import db | ||||||
|  | from security import encrypt, decrypt | ||||||
|  |  | ||||||
|  | class Test: | ||||||
|  |     def __init__(self, _id=None, expiry=None, time_limit=None, creator=None): | ||||||
|  |         self._id = _id | ||||||
|  |         self.expiry = expiry | ||||||
|  |         self.time_limit = None if time_limit == 'none' or time_limit == '' else time_limit | ||||||
|  |         self.creator = creator | ||||||
|  |      | ||||||
|  |     def create(self): | ||||||
|  |         test = { | ||||||
|  |             '_id': self._id, | ||||||
|  |             'date': datetime.today(), | ||||||
|  |             'expiry': self.expiry, | ||||||
|  |             'time_limit': self.time_limit, | ||||||
|  |             'creator': encrypt(self.creator), | ||||||
|  |             'test_code': secrets.token_hex(6).upper() | ||||||
|  |         } | ||||||
|  |         if db.tests.insert_one(test): | ||||||
|  |             flash(f'Created a new exam with Exam Code <strong>{self.render_test_code(test["test_code"])}</strong>.', 'success') | ||||||
|  |             return jsonify({'success': test}), 200 | ||||||
|  |         return jsonify({'error': f'Could not create exam. An error occurred.'}), 400 | ||||||
|  |  | ||||||
|  |     def add_user_code(self, user_code, time_adjustment): | ||||||
|  |         code = { | ||||||
|  |             '_id': uuid4().hex, | ||||||
|  |             'user_code': user_code, | ||||||
|  |             'time_adjustment': time_adjustment | ||||||
|  |         } | ||||||
|  |         if db.tests.find_one_and_update({'_id': self._id}, {'$push': {'time_adjustments': code}},upsert=False): | ||||||
|  |             return jsonify({'success': code}) | ||||||
|  |         else: | ||||||
|  |             return jsonify({'error': 'An error occurred.'}), 400 | ||||||
|  |  | ||||||
|  |     def render_test_code(self, test_code): | ||||||
|  |         return '—'.join([test_code[:4], test_code[4:8], test_code[8:]]) | ||||||
|  |      | ||||||
|  |     def parse_test_code(self, test_code): | ||||||
|  |         return test_code.replace('—', '') | ||||||
| @@ -27,7 +27,7 @@ body { | |||||||
|     margin: auto; |     margin: auto; | ||||||
| } | } | ||||||
|  |  | ||||||
| .form-signin-heading { | .form-heading { | ||||||
|     margin-bottom: 2rem; |     margin-bottom: 2rem; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -65,6 +65,10 @@ body { | |||||||
|     border-bottom: 2px solid #585858; |     border-bottom: 2px solid #585858; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .form-label-group input:active, .form-label-group input:focus { | ||||||
|  |     background-color: transparent; | ||||||
|  | } | ||||||
|  |  | ||||||
| .form-label-group input::-webkit-input-placeholder { | .form-label-group input::-webkit-input-placeholder { | ||||||
|     color: transparent; |     color: transparent; | ||||||
| } | } | ||||||
| @@ -149,6 +153,50 @@ table.dataTable { | |||||||
|     width: 100%; |     width: 100%; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .alert-db-empty { | ||||||
|  |     width: 100%; | ||||||
|  |     max-width: 720px; | ||||||
|  |     font-size: 14pt; | ||||||
|  |     margin: 20px auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-date-input, .form-select-input { | ||||||
|  |     position: relative; | ||||||
|  |     margin: 2rem 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-date-input input, | ||||||
|  | .form-date-input label, .form-select-input select, .form-select-input label { | ||||||
|  |     padding: var(--input-padding-y) var(--input-padding-x); | ||||||
|  |     font-size: 16pt; | ||||||
|  |     width: 100%; | ||||||
|  |     background-color: transparent; | ||||||
|  |     border: none; | ||||||
|  |     border-bottom: 2px solid #585858; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .datepicker::-webkit-calendar-picker-indicator {     | ||||||
|  |     border: 1px; | ||||||
|  |     border-color: gray; | ||||||
|  |     border-radius: 10%; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-date-input label, .form-select-input label { | ||||||
|  |     /* position: absolute; */ | ||||||
|  |     /* top: 0; | ||||||
|  |     left: 0; */ | ||||||
|  |     display: block; | ||||||
|  |     width: 100%; | ||||||
|  |     margin-bottom: 0; /* Override default `<label>` margin */ | ||||||
|  |     line-height: 1.5; | ||||||
|  |     color: #495057; | ||||||
|  |     cursor: text; /* Match the input under the label */ | ||||||
|  |     border: 1px solid transparent; | ||||||
|  |     border-radius: .25rem; | ||||||
|  |     transition: all .1s ease-in-out; | ||||||
|  |     z-index: -1; | ||||||
|  | } | ||||||
|  |  | ||||||
| /* Fallback for Edge | /* Fallback for Edge | ||||||
| -------------------------------------------------- */ | -------------------------------------------------- */ | ||||||
| @supports (-ms-ime-align: auto) { | @supports (-ms-ime-align: auto) { | ||||||
|   | |||||||
| @@ -350,6 +350,49 @@ $('form[name=form-update-account]').submit(function(event) { | |||||||
|     event.preventDefault(); |     event.preventDefault(); | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  | $('form[name=form-create-test]').submit(function(event) { | ||||||
|  |      | ||||||
|  |     var $form = $(this); | ||||||
|  |     var alert = document.getElementById('alert-box'); | ||||||
|  |     var data = $form.serialize(); | ||||||
|  |     console.log(data) | ||||||
|  |     alert.innerHTML = '' | ||||||
|  |  | ||||||
|  |     $.ajax({ | ||||||
|  |         url: window.location.pathname, | ||||||
|  |         type: 'POST', | ||||||
|  |         data: data, | ||||||
|  |         dataType: 'json', | ||||||
|  |         success: function(response) { | ||||||
|  |             window.location.reload(); | ||||||
|  |         }, | ||||||
|  |         error: function(response) { | ||||||
|  |             console.log(response.responseJSON) | ||||||
|  |             if (typeof response.responseJSON.error === 'string' || response.responseJSON.error instanceof String) { | ||||||
|  |                 alert.innerHTML = alert.innerHTML + ` | ||||||
|  |                 <div class="alert alert-danger alert-dismissible fade show" role="alert"> | ||||||
|  |                     <i class="bi bi-exclamation-triangle-fill" title="Danger"></i> | ||||||
|  |                     ${response.responseJSON.error} | ||||||
|  |                     <button type="button" class="btn-close" data-bs-dismiss="alert" aria-label="Close"></button> | ||||||
|  |                 </div> | ||||||
|  |                 `; | ||||||
|  |             } else if (response.responseJSON.error instanceof Array) { | ||||||
|  |                 for (var i = 0; i < response.responseJSON.error.length; i ++) { | ||||||
|  |                     alert.innerHTML = alert.innerHTML + ` | ||||||
|  |                     <div class="alert alert-danger alert-dismissible fade show" role="alert"> | ||||||
|  |                         <i class="bi bi-exclamation-triangle-fill" title="Danger"></i> | ||||||
|  |                         ${response.responseJSON.error[i]} | ||||||
|  |                         <button type="button" class="btn-close" data-bs-dismiss="alert" aria-label="Close"></button> | ||||||
|  |                     </div> | ||||||
|  |                     `; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     event.preventDefault(); | ||||||
|  | }); | ||||||
|  |  | ||||||
| // Dismiss Cookie Alert | // Dismiss Cookie Alert | ||||||
| $('#dismiss-cookie-alert').click(function(event){ | $('#dismiss-cookie-alert').click(function(event){ | ||||||
|  |  | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ | |||||||
|     <div class="form-container"> |     <div class="form-container"> | ||||||
|         <form name="form-login" class="form-signin"> |         <form name="form-login" class="form-signin"> | ||||||
|             {% include "admin/components/server-alerts.html" %} |             {% include "admin/components/server-alerts.html" %} | ||||||
|             <h2 class="form-signin-heading">Log In</h2> |             <h2 class="form">Log In</h2> | ||||||
|             {{ form.hidden_tag() }} |             {{ form.hidden_tag() }} | ||||||
|             <div class="form-label-group"> |             <div class="form-label-group"> | ||||||
|                 {{ form.username(class_="form-control", autofocus=true, placeholder="Enter Username") }} |                 {{ form.username(class_="form-control", autofocus=true, placeholder="Enter Username") }} | ||||||
|   | |||||||
| @@ -42,9 +42,6 @@ | |||||||
|             integrity="sha256-/xUj+3OJU5yExlq6GSYGSHk7tPXikynS7ogEvDej/m4=" |             integrity="sha256-/xUj+3OJU5yExlq6GSYGSHk7tPXikynS7ogEvDej/m4=" | ||||||
|             crossorigin="anonymous"> |             crossorigin="anonymous"> | ||||||
|         </script> |         </script> | ||||||
|         <script> |  | ||||||
|             window.jQuery || document.write('<script src="js/jquery-3.6.0.min.js"><\/script>') |  | ||||||
|         </script> |  | ||||||
|         <script |         <script | ||||||
|             src="https://cdn.jsdelivr.net/npm/@popperjs/core@2.10.2/dist/umd/popper.min.js" |             src="https://cdn.jsdelivr.net/npm/@popperjs/core@2.10.2/dist/umd/popper.min.js" | ||||||
|             integrity="sha384-7+zCNj/IqJ95wo16oMtfsKbZ9ccEh31eOz1HGyDuCQ6wgnyJNSYdrPa03rtR1zdB" |             integrity="sha384-7+zCNj/IqJ95wo16oMtfsKbZ9ccEh31eOz1HGyDuCQ6wgnyJNSYdrPa03rtR1zdB" | ||||||
|   | |||||||
							
								
								
									
										27
									
								
								ref-test/admin/templates/admin/components/datatable.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								ref-test/admin/templates/admin/components/datatable.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | |||||||
|  | {% extends "admin/components/base.html" %} | ||||||
|  | {% block datatable_css %} | ||||||
|  |     <link rel="stylesheet" type="text/css" href="https://cdn.datatables.net/1.11.3/css/dataTables.bootstrap5.min.css"/> | ||||||
|  |     <link rel="stylesheet" type="text/css" href="https://cdn.datatables.net/buttons/2.0.1/css/buttons.bootstrap5.min.css"/> | ||||||
|  |     <link rel="stylesheet" type="text/css" href="https://cdn.datatables.net/colreorder/1.5.5/css/colReorder.bootstrap5.min.css"/> | ||||||
|  |     <link rel="stylesheet" type="text/css" href="https://cdn.datatables.net/fixedheader/3.2.0/css/fixedHeader.bootstrap5.min.css"/> | ||||||
|  |     <link rel="stylesheet" type="text/css" href="https://cdn.datatables.net/keytable/2.6.4/css/keyTable.bootstrap5.min.css"/> | ||||||
|  |     <link rel="stylesheet" type="text/css" href="https://cdn.datatables.net/responsive/2.2.9/css/responsive.bootstrap5.min.css"/> | ||||||
|  | {% endblock %} | ||||||
|  | {% block datatable_scripts %} | ||||||
|  | <script type="text/javascript" src="https://cdnjs.cloudflare.com/ajax/libs/jszip/2.5.0/jszip.min.js"></script> | ||||||
|  | <script type="text/javascript" src="https://cdnjs.cloudflare.com/ajax/libs/pdfmake/0.1.36/pdfmake.min.js"></script> | ||||||
|  | <script type="text/javascript" src="https://cdnjs.cloudflare.com/ajax/libs/pdfmake/0.1.36/vfs_fonts.js"></script> | ||||||
|  | <script type="text/javascript" src="https://cdn.datatables.net/1.11.3/js/jquery.dataTables.min.js"></script> | ||||||
|  | <script type="text/javascript" src="https://cdn.datatables.net/1.11.3/js/dataTables.bootstrap5.min.js"></script> | ||||||
|  | <script type="text/javascript" src="https://cdn.datatables.net/buttons/2.0.1/js/dataTables.buttons.min.js"></script> | ||||||
|  | <script type="text/javascript" src="https://cdn.datatables.net/buttons/2.0.1/js/buttons.bootstrap5.min.js"></script> | ||||||
|  | <script type="text/javascript" src="https://cdn.datatables.net/buttons/2.0.1/js/buttons.colVis.min.js"></script> | ||||||
|  | <script type="text/javascript" src="https://cdn.datatables.net/buttons/2.0.1/js/buttons.html5.min.js"></script> | ||||||
|  | <script type="text/javascript" src="https://cdn.datatables.net/buttons/2.0.1/js/buttons.print.min.js"></script> | ||||||
|  | <script type="text/javascript" src="https://cdn.datatables.net/colreorder/1.5.5/js/dataTables.colReorder.min.js"></script> | ||||||
|  | <script type="text/javascript" src="https://cdn.datatables.net/fixedheader/3.2.0/js/dataTables.fixedHeader.min.js"></script> | ||||||
|  | <script type="text/javascript" src="https://cdn.datatables.net/keytable/2.6.4/js/dataTables.keyTable.min.js"></script> | ||||||
|  | <script type="text/javascript" src="https://cdn.datatables.net/responsive/2.2.9/js/dataTables.responsive.min.js"></script> | ||||||
|  | <script type="text/javascript" src="https://cdn.datatables.net/responsive/2.2.9/js/responsive.bootstrap5.js"></script> | ||||||
|  |     {% block custom_data_script %}{% endblock %} | ||||||
|  | {% endblock %} | ||||||
| @@ -4,26 +4,26 @@ | |||||||
|         {% for category, message in messages %} |         {% for category, message in messages %} | ||||||
|             {% if category == "error" %} |             {% if category == "error" %} | ||||||
|                 <div class="alert alert-danger alert-dismissible fade show" role="alert"> |                 <div class="alert alert-danger alert-dismissible fade show" role="alert"> | ||||||
|                     <i class="bi bi-exclamation-triangle-fill" title="Danger"></i> |                     <i class="bi bi-exclamation-triangle-fill" title="Error" aria-title="Error"></i> | ||||||
|                     {{ message|safe }} |                     {{ message|safe }} | ||||||
|                     <button type="button" class="btn-close" data-bs-dismiss="alert" aria-label="Close"></button> |                     <button type="button" class="btn-close" data-bs-dismiss="alert" aria-label="Close"></button> | ||||||
|                 </div> |                 </div> | ||||||
|             {% elif category == "success" %} |             {% elif category == "success" %} | ||||||
|                 <div class="alert alert-success alert-dismissible fade show" role="alert"> |                 <div class="alert alert-success alert-dismissible fade show" role="alert"> | ||||||
|                     <i class="bi bi-check2-circle" title="Success"></i> |                     <i class="bi bi-check2-circle" title="Success" aria-title="Success"></i> | ||||||
|                     {{ message|safe }} |                     {{ message|safe }} | ||||||
|                     <button type="button" class="btn-close" data-bs-dismiss="alert" aria-label="Close"></button> |                     <button type="button" class="btn-close" data-bs-dismiss="alert" aria-label="Close"></button> | ||||||
|                 </div> |                 </div> | ||||||
|             {% elif category == "warning" %} |             {% elif category == "warning" %} | ||||||
|                 <div class="alert alert-warning alert-dismissible fade show" role="alert"> |                 <div class="alert alert-warning alert-dismissible fade show" role="alert"> | ||||||
|                     <i class="bi bi-info-circle-fill" title="Warning"></i> |                     <i class="bi bi-info-circle-fill" aria-title="Warning" title="Warning"></i> | ||||||
|                     {{ message|safe }} |                     {{ message|safe }} | ||||||
|                     <button type="button" class="btn-close" data-bs-dismiss="alert" aria-label="Close"></button> |                     <button type="button" class="btn-close" data-bs-dismiss="alert" aria-label="Close"></button> | ||||||
|                 </div> |                 </div> | ||||||
|             {% elif category == "cookie_alert" %} |             {% elif category == "cookie_alert" %} | ||||||
|                 {% if not cookie_flash_flag.value %} |                 {% if not cookie_flash_flag.value %} | ||||||
|                     <div class="alert alert-primary alert-dismissible fade show" id="cookie-alert" role="alert"> |                     <div class="alert alert-primary alert-dismissible fade show" id="cookie-alert" role="alert"> | ||||||
|                         <i class="bi bi-info-circle-fill" title="Alert"></i> |                         <i class="bi bi-info-circle-fill" title="Cookie Alert" aria-title="Cookie Alert"></i> | ||||||
|                         {{ message|safe }} |                         {{ message|safe }} | ||||||
|                         <button type="button" id="dismiss-cookie-alert" class="btn btn-success" data-bs-dismiss="alert" aria-label="Close">Accept</button> |                         <button type="button" id="dismiss-cookie-alert" class="btn btn-success" data-bs-dismiss="alert" aria-label="Close">Accept</button> | ||||||
|                     </div> |                     </div> | ||||||
|   | |||||||
| @@ -1,13 +1,5 @@ | |||||||
| {% extends "admin/components/base.html" %} | {% extends "admin/components/datatable.html" %} | ||||||
| {% block title %} SKA Referee Test | Manage Users {% endblock %} | {% block title %} SKA Referee Test | Manage Users {% endblock %} | ||||||
| {% block datatable_css %} |  | ||||||
|     <link rel="stylesheet" type="text/css" href="https://cdn.datatables.net/1.11.3/css/dataTables.bootstrap5.min.css"/> |  | ||||||
|     <link rel="stylesheet" type="text/css" href="https://cdn.datatables.net/buttons/2.0.1/css/buttons.bootstrap5.min.css"/> |  | ||||||
|     <link rel="stylesheet" type="text/css" href="https://cdn.datatables.net/colreorder/1.5.5/css/colReorder.bootstrap5.min.css"/> |  | ||||||
|     <link rel="stylesheet" type="text/css" href="https://cdn.datatables.net/fixedheader/3.2.0/css/fixedHeader.bootstrap5.min.css"/> |  | ||||||
|     <link rel="stylesheet" type="text/css" href="https://cdn.datatables.net/keytable/2.6.4/css/keyTable.bootstrap5.min.css"/> |  | ||||||
|     <link rel="stylesheet" type="text/css" href="https://cdn.datatables.net/responsive/2.2.9/css/responsive.bootstrap5.min.css"/> |  | ||||||
| {% endblock %} |  | ||||||
| {% block content %} | {% block content %} | ||||||
|     <h1>Manage Users</h1> |     <h1>Manage Users</h1> | ||||||
|  |  | ||||||
| @@ -120,22 +112,7 @@ | |||||||
|         </form> |         </form> | ||||||
|     </div> |     </div> | ||||||
| {% endblock %} | {% endblock %} | ||||||
| {% block datatable_scripts %} | {% block custom_data_script %} | ||||||
| <script type="text/javascript" src="https://cdnjs.cloudflare.com/ajax/libs/jszip/2.5.0/jszip.min.js"></script> |  | ||||||
| <script type="text/javascript" src="https://cdnjs.cloudflare.com/ajax/libs/pdfmake/0.1.36/pdfmake.min.js"></script> |  | ||||||
| <script type="text/javascript" src="https://cdnjs.cloudflare.com/ajax/libs/pdfmake/0.1.36/vfs_fonts.js"></script> |  | ||||||
| <script type="text/javascript" src="https://cdn.datatables.net/1.11.3/js/jquery.dataTables.min.js"></script> |  | ||||||
| <script type="text/javascript" src="https://cdn.datatables.net/1.11.3/js/dataTables.bootstrap5.min.js"></script> |  | ||||||
| <script type="text/javascript" src="https://cdn.datatables.net/buttons/2.0.1/js/dataTables.buttons.min.js"></script> |  | ||||||
| <script type="text/javascript" src="https://cdn.datatables.net/buttons/2.0.1/js/buttons.bootstrap5.min.js"></script> |  | ||||||
| <script type="text/javascript" src="https://cdn.datatables.net/buttons/2.0.1/js/buttons.colVis.min.js"></script> |  | ||||||
| <script type="text/javascript" src="https://cdn.datatables.net/buttons/2.0.1/js/buttons.html5.min.js"></script> |  | ||||||
| <script type="text/javascript" src="https://cdn.datatables.net/buttons/2.0.1/js/buttons.print.min.js"></script> |  | ||||||
| <script type="text/javascript" src="https://cdn.datatables.net/colreorder/1.5.5/js/dataTables.colReorder.min.js"></script> |  | ||||||
| <script type="text/javascript" src="https://cdn.datatables.net/fixedheader/3.2.0/js/dataTables.fixedHeader.min.js"></script> |  | ||||||
| <script type="text/javascript" src="https://cdn.datatables.net/keytable/2.6.4/js/dataTables.keyTable.min.js"></script> |  | ||||||
| <script type="text/javascript" src="https://cdn.datatables.net/responsive/2.2.9/js/dataTables.responsive.min.js"></script> |  | ||||||
| <script type="text/javascript" src="https://cdn.datatables.net/responsive/2.2.9/js/responsive.bootstrap5.js"></script> |  | ||||||
|     <script> |     <script> | ||||||
|         $(document).ready(function() { |         $(document).ready(function() { | ||||||
|             $('#user-table').DataTable({ |             $('#user-table').DataTable({ | ||||||
|   | |||||||
| @@ -1 +1,138 @@ | |||||||
| {% extends "admin/components/base.html" %} | {% extends "admin/components/datatable.html" %} | ||||||
|  | {% block title %} SKA Referee Test | Manage Exams {% endblock %} | ||||||
|  | {% block content %} | ||||||
|  |     <h1>Manage Exams</h1> | ||||||
|  |  | ||||||
|  |     {% if tests %} | ||||||
|  |         <table id="test-table" class="table table-striped" style="width:100%"> | ||||||
|  |             <thead> | ||||||
|  |                 <tr> | ||||||
|  |                     <th data-proority="1"> | ||||||
|  |                         Date Created | ||||||
|  |                     </th> | ||||||
|  |                     <th data-priority="2"> | ||||||
|  |                         Exam Code | ||||||
|  |                     </th> | ||||||
|  |                     <th data-priority="1"> | ||||||
|  |                         Expiry Date | ||||||
|  |                     </th> | ||||||
|  |                     <th data-proority="4"> | ||||||
|  |                         Time Limit | ||||||
|  |                     </th> | ||||||
|  |                     <th data-priority="3"> | ||||||
|  |                         Created By | ||||||
|  |                     </th> | ||||||
|  |                     <th data-priority="1"> | ||||||
|  |                         Actions | ||||||
|  |                     </th> | ||||||
|  |                 </tr> | ||||||
|  |             </thead> | ||||||
|  |             <tbody> | ||||||
|  |                 {% for test in tests %} | ||||||
|  |                     <tr class="user-table-row"> | ||||||
|  |                         <td> | ||||||
|  |                             {{ test.date.date() }} | ||||||
|  |                         </td> | ||||||
|  |                         <td> | ||||||
|  |                             {{ '—'.join([test.test_code[:4], test.test_code[4:8], test.test_code[8:]]) }} | ||||||
|  |                         </td> | ||||||
|  |                         <td> | ||||||
|  |                             {{ test.expiry }} | ||||||
|  |                         </td> | ||||||
|  |                         <td> | ||||||
|  |                             {% if test.time_limit == None -%} | ||||||
|  |                                 None | ||||||
|  |                             {% elif test.time_limit == '60' -%} | ||||||
|  |                                 1 hour | ||||||
|  |                             {% elif test.time_limit == '90' -%} | ||||||
|  |                                 1 hour 30 min | ||||||
|  |                             {% elif test.time_limit == '120' -%} | ||||||
|  |                                 2 hours | ||||||
|  |                             {% else -%} | ||||||
|  |                                 {{ test.time_limit }} | ||||||
|  |                             {% endif %} | ||||||
|  |                         </td> | ||||||
|  |                         <td> | ||||||
|  |                             {{ test.creator }} | ||||||
|  |                         </td> | ||||||
|  |                         <td class="test-row-actions"> | ||||||
|  |                             <a | ||||||
|  |                                 href="#" | ||||||
|  |                                 class="btn btn-primary" | ||||||
|  |                                 title="Edit Exam" | ||||||
|  |                             > | ||||||
|  |                                 <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" fill="currentColor" class="bi bi-person-lines-fill" viewBox="0 0 16 16"> | ||||||
|  |                                     <path d="M6 8a3 3 0 1 0 0-6 3 3 0 0 0 0 6zm-5 6s-1 0-1-1 1-4 6-4 6 3 6 4-1 1-1 1H1zM11 3.5a.5.5 0 0 1 .5-.5h4a.5.5 0 0 1 0 1h-4a.5.5 0 0 1-.5-.5zm.5 2.5a.5.5 0 0 0 0 1h4a.5.5 0 0 0 0-1h-4zm2 3a.5.5 0 0 0 0 1h2a.5.5 0 0 0 0-1h-2zm0 3a.5.5 0 0 0 0 1h2a.5.5 0 0 0 0-1h-2z"/> | ||||||
|  |                                 </svg> | ||||||
|  |                             </a> | ||||||
|  |                             <a | ||||||
|  |                                 href="#" | ||||||
|  |                                 class="btn btn-danger" | ||||||
|  |                                 title="Delete Exam" | ||||||
|  |                             > | ||||||
|  |                                 <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" fill="currentColor" class="bi bi-person-x-fill" viewBox="0 0 16 16"> | ||||||
|  |                                     <path fill-rule="evenodd" d="M1 14s-1 0-1-1 1-4 6-4 6 3 6 4-1 1-1 1H1zm5-6a3 3 0 1 0 0-6 3 3 0 0 0 0 6zm6.146-2.854a.5.5 0 0 1 .708 0L14 6.293l1.146-1.147a.5.5 0 0 1 .708.708L14.707 7l1.147 1.146a.5.5 0 0 1-.708.708L14 7.707l-1.146 1.147a.5.5 0 0 1-.708-.708L13.293 7l-1.147-1.146a.5.5 0 0 1 0-.708z"></path> | ||||||
|  |                                 </svg> | ||||||
|  |                             </button> | ||||||
|  |                         </td> | ||||||
|  |                     </tr> | ||||||
|  |                 {% endfor %} | ||||||
|  |             </tbody> | ||||||
|  |         </table> | ||||||
|  |         {% block custom_data_script %} | ||||||
|  |             <script> | ||||||
|  |                 $(document).ready(function() { | ||||||
|  |                     $('#test-table').DataTable({ | ||||||
|  |                         'columnDefs': [ | ||||||
|  |                             {'sortable': false, 'targets': [1,3,5]}, | ||||||
|  |                         {'searchable': false, 'targets': [3,5]} | ||||||
|  |                         ], | ||||||
|  |                         'order': [[0, 'desc'], [2, 'asc']], | ||||||
|  |                         'buttons': [ | ||||||
|  |                             'copy', 'excel', 'pdf' | ||||||
|  |                         ], | ||||||
|  |                         'responsive': 'true', | ||||||
|  |                         'colReorder': 'true', | ||||||
|  |                         'fixedHeader': 'true' | ||||||
|  |                     }); | ||||||
|  |                 } ); | ||||||
|  |                 $('#test-table').show(); | ||||||
|  |                 $(window).trigger('resize'); | ||||||
|  |             </script> | ||||||
|  |         {% endblock %} | ||||||
|  |     {% else %} | ||||||
|  |             <div class="alert alert-primary alert-db-empty"> | ||||||
|  |                 <i class="bi bi-info-circle-fill" aria-title="Alert" title="Alert"></i> | ||||||
|  |                 No exams have been created. Use the form below to create an exam. | ||||||
|  |             </div> | ||||||
|  |     {% endif %} | ||||||
|  |  | ||||||
|  |     <div class="form-container"> | ||||||
|  |         <form name="form-create-test" class="form-signin"> | ||||||
|  |             <h2 class="form-signin-heading">Create Exam</h2> | ||||||
|  |             {{ form.hidden_tag() }} | ||||||
|  |             <div class="form-date-input"> | ||||||
|  |                 {{ form.expiry(placeholder="Enter Expiry Date", class_ = "datepicker") }} | ||||||
|  |                 {{ form.expiry.label }} | ||||||
|  |             </div> | ||||||
|  |             <div class="form-select-input"> | ||||||
|  |                 {{ form.time_limit(placeholder="Select Time Limit") }} | ||||||
|  |                 {{ form.time_limit.label }} | ||||||
|  |             </div> | ||||||
|  |             {% include "admin/components/client-alerts.html" %} | ||||||
|  |             <div class="container form-submission-button"> | ||||||
|  |                 <div class="row"> | ||||||
|  |                     <div class="col text-center"> | ||||||
|  |                         <button title="Create Exam" class="btn btn-md btn-success btn-block" type="submit"> | ||||||
|  |                             <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" fill="currentColor" class="bi bi-person-plus-fill" viewBox="0 0 20 20"> | ||||||
|  |                                 <path d="M1 14s-1 0-1-1 1-4 6-4 6 3 6 4-1 1-1 1H1zm5-6a3 3 0 1 0 0-6 3 3 0 0 0 0 6z"/> | ||||||
|  |                                 <path fill-rule="evenodd" d="M13.5 5a.5.5 0 0 1 .5.5V7h1.5a.5.5 0 0 1 0 1H14v1.5a.5.5 0 0 1-1 0V8h-1.5a.5.5 0 0 1 0-1H13V5.5a.5.5 0 0 1 .5-.5z"/> | ||||||
|  |                             </svg> | ||||||
|  |                             Create Exam | ||||||
|  |                         </button> | ||||||
|  |                     </div> | ||||||
|  |                 </div> | ||||||
|  |             </div> | ||||||
|  |         </form> | ||||||
|  |     </div> | ||||||
|  | {% endblock %} | ||||||
| @@ -10,25 +10,8 @@ from main import db | |||||||
| from uuid import uuid4 | from uuid import uuid4 | ||||||
| import secrets | import secrets | ||||||
| from main import mail | from main import mail | ||||||
| from datetime import datetime, timedelta | from datetime import datetime, date, timedelta | ||||||
|  | from .models import Test | ||||||
| from .forms import CreateUserForm |  | ||||||
|  |  | ||||||
| cookie_consent = Blueprint( |  | ||||||
|     'cookie_consent', |  | ||||||
|     __name__ |  | ||||||
| ) |  | ||||||
| @cookie_consent.route('/') |  | ||||||
| def _cookies(): |  | ||||||
|     resp = redirect('/') |  | ||||||
|     resp.set_cookie( |  | ||||||
|         key = 'cookie_consent', |  | ||||||
|         value = 'True', |  | ||||||
|         max_age = timedelta(days=14) if request.cookies.get('remember') == 'True' else 'Session', |  | ||||||
|         path = '/', |  | ||||||
|         expires = datetime.utcnow() + timedelta(days=14) if request.cookies.get('remember') else 'Session' |  | ||||||
|     ) |  | ||||||
|     return resp |  | ||||||
|  |  | ||||||
| views = Blueprint( | views = Blueprint( | ||||||
|     'admin_views', |     'admin_views', | ||||||
| @@ -99,6 +82,7 @@ def settings(): | |||||||
| @admin_account_required | @admin_account_required | ||||||
| @login_required | @login_required | ||||||
| def users(): | def users(): | ||||||
|  |     from .forms import CreateUserForm | ||||||
|     form = CreateUserForm() |     form = CreateUserForm() | ||||||
|     if request.method == 'GET': |     if request.method == 'GET': | ||||||
|         users_list = decrypt_find(db.users, {}) |         users_list = decrypt_find(db.users, {}) | ||||||
| @@ -259,8 +243,32 @@ def questions(): | |||||||
| def upload_questions(): | def upload_questions(): | ||||||
|     return render_template('/admin/settings/upload-questions.html') |     return render_template('/admin/settings/upload-questions.html') | ||||||
|  |  | ||||||
| @views.route('/tests/') | @views.route('/tests/', methods=['GET','POST']) | ||||||
| @admin_account_required | @admin_account_required | ||||||
| @login_required | @login_required | ||||||
| def tests(): | def tests(): | ||||||
|     return render_template('/admin/tests.html') |     from .forms import CreateTest | ||||||
|  |     form = CreateTest() | ||||||
|  |     form.time_limit.default='none' | ||||||
|  |     form.process() | ||||||
|  |     if request.method == 'GET': | ||||||
|  |         tests = decrypt_find(db.tests, {}) | ||||||
|  |         return render_template('/admin/tests.html', tests = tests, form = form) | ||||||
|  |     if request.method == 'POST': | ||||||
|  |         if form.validate_on_submit(): | ||||||
|  |             expiry = request.form.get('expiry') | ||||||
|  |             if datetime.strptime(expiry, '%Y-%m-%d').date() < date.today(): | ||||||
|  |                 return jsonify({'error': 'The expiry date cannot be in the past.'}), 400 | ||||||
|  |             creator_id = get_id_from_cookie() | ||||||
|  |             creator = decrypt_find_one(db.users, { '_id': creator_id } )['username'] | ||||||
|  |             test = Test( | ||||||
|  |                 _id = uuid4().hex, | ||||||
|  |                 expiry = expiry, | ||||||
|  |                 time_limit = request.form.get('time_limit'), | ||||||
|  |                 creator = creator | ||||||
|  |             ) | ||||||
|  |             test.create() | ||||||
|  |             return jsonify({'success': 'New exam created.'}), 200 | ||||||
|  |         else: | ||||||
|  |             errors = [*form.expiry.errors, *form.time_limit.errors] | ||||||
|  |             return jsonify({ 'error': errors}), 400 | ||||||
							
								
								
									
										18
									
								
								ref-test/common/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								ref-test/common/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | from datetime import datetime, timedelta | ||||||
|  | from flask import Blueprint, redirect, request | ||||||
|  |  | ||||||
|  | cookie_consent = Blueprint( | ||||||
|  |     'cookie_consent', | ||||||
|  |     __name__ | ||||||
|  | ) | ||||||
|  | @cookie_consent.route('/') | ||||||
|  | def _cookies(): | ||||||
|  |     resp = redirect('/') | ||||||
|  |     resp.set_cookie( | ||||||
|  |         key = 'cookie_consent', | ||||||
|  |         value = 'True', | ||||||
|  |         max_age = timedelta(days=14) if request.cookies.get('remember') == 'True' else 'Session', | ||||||
|  |         path = '/', | ||||||
|  |         expires = datetime.utcnow() + timedelta(days=14) if request.cookies.get('remember') else 'Session' | ||||||
|  |     ) | ||||||
|  |     return resp | ||||||
| @@ -37,7 +37,9 @@ if __name__ == '__main__': | |||||||
|     if not check_keyfile_exists(): |     if not check_keyfile_exists(): | ||||||
|         generate_keyfile() |         generate_keyfile() | ||||||
|  |  | ||||||
|     from admin.views import views as admin_views, cookie_consent |     from common import cookie_consent | ||||||
|  |  | ||||||
|  |     from admin.views import views as admin_views | ||||||
|     from admin.auth import auth as admin_auth |     from admin.auth import auth as admin_auth | ||||||
|     from admin.results import results |     from admin.results import results | ||||||
|     from quiz.views import views as quiz_views |     from quiz.views import views as quiz_views | ||||||
|   | |||||||
| @@ -3,6 +3,4 @@ from flask import Blueprint | |||||||
| auth = Blueprint( | auth = Blueprint( | ||||||
|     'quiz_auth', |     'quiz_auth', | ||||||
|     __name__, |     __name__, | ||||||
|     template_folder='templates', |  | ||||||
|     static_folder='static' |  | ||||||
| ) | ) | ||||||
| @@ -1,9 +1,11 @@ | |||||||
| from flask_wtf import FlaskForm | from flask_wtf import FlaskForm | ||||||
| from wtforms import StringField, PasswordField, BooleanField | from wtforms import StringField, PasswordField, BooleanField | ||||||
| from wtforms.validators import InputRequired, Email, Length | from wtforms.validators import InputRequired, Email, Length, Optional | ||||||
|  |  | ||||||
| class StartQuiz(FlaskForm): | class StartQuiz(FlaskForm): | ||||||
|     given_name = StringField('Given Name', validators=[InputRequired(), Length(max=15)]) |     first_name = StringField('First Name(s)', validators=[InputRequired(), Length(max=30)]) | ||||||
|     surname = StringField('Surname', validators=[InputRequired(), Length(max=15)]) |     surname = StringField('Surname', validators=[InputRequired(), Length(max=30)]) | ||||||
|     email = StringField('Email Address', validators=[InputRequired(), Email(message='You must enter a valid email address.'), Length(max=50)]) |     email = StringField('Email Address', validators=[InputRequired(), Email(message='You must enter a valid email address.'), Length(max=50)]) | ||||||
|     club = StringField('Affiliated Club', validators=[InputRequired(), Length(max=50)]) |     club = StringField('Affiliated Club (Optional)', validators=[Optional(), Length(max=50)]) | ||||||
|  |     auth_code = StringField('Exam Code', validators=[InputRequired(), Length(min=14, max=14)]) | ||||||
|  |     user_code = StringField('User Code (Optional)', validators=[Optional(), Length(min=6, max=6)]) | ||||||
| @@ -0,0 +1,190 @@ | |||||||
|  | .bg-light { | ||||||
|  |     background-color: #EBE3E1!important; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | body { | ||||||
|  |     padding: 80px 0; | ||||||
|  |     line-height: 1.5; | ||||||
|  |     font-size: 14pt; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .site-footer { | ||||||
|  |     background-color: lightgray; | ||||||
|  |     font-size: small; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .site-footer p { | ||||||
|  |     margin: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .quiz-container { | ||||||
|  |     max-width: 720px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-container { | ||||||
|  |     display: -ms-flexbox; | ||||||
|  |     display: flex; | ||||||
|  |     -ms-flex-align: center; | ||||||
|  |     align-items: center; | ||||||
|  |     padding-top: 40px; | ||||||
|  |     padding-bottom: 40px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-quiz-start { | ||||||
|  |     width: 100%; | ||||||
|  |     max-width: 420px; | ||||||
|  |     padding: 15px; | ||||||
|  |     margin: auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-heading { | ||||||
|  |     margin-bottom: 2rem; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-label-group { | ||||||
|  |     position: relative; | ||||||
|  |     margin-bottom: 2rem; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-label-group input, | ||||||
|  | .form-label-group label { | ||||||
|  |     padding: var(--input-padding-y) var(--input-padding-x); | ||||||
|  |     font-size: 16pt; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-label-group label { | ||||||
|  |     position: absolute; | ||||||
|  |     top: 0; | ||||||
|  |     left: 0; | ||||||
|  |     display: block; | ||||||
|  |     width: 100%; | ||||||
|  |     margin-bottom: 0; /* Override default `<label>` margin */ | ||||||
|  |     line-height: 1.5; | ||||||
|  |     color: #495057; | ||||||
|  |     cursor: text; /* Match the input under the label */ | ||||||
|  |     border: 1px solid transparent; | ||||||
|  |     border-radius: .25rem; | ||||||
|  |     transition: all .1s ease-in-out; | ||||||
|  |     z-index: -1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-label-group input { | ||||||
|  |     background-color: transparent; | ||||||
|  |     border: none; | ||||||
|  |     border-radius: 0%; | ||||||
|  |     border-bottom: 2px solid #585858; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-label-group input:active, .form-label-group input:focus { | ||||||
|  |     background-color: transparent; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-label-group input::-webkit-input-placeholder { | ||||||
|  |     color: transparent; | ||||||
|  | } | ||||||
|  |    | ||||||
|  | .form-label-group input:-ms-input-placeholder { | ||||||
|  |     color: transparent; | ||||||
|  | } | ||||||
|  |    | ||||||
|  | .form-label-group input::-ms-input-placeholder { | ||||||
|  |     color: transparent; | ||||||
|  | } | ||||||
|  |    | ||||||
|  | .form-label-group input::-moz-placeholder { | ||||||
|  |     color: transparent; | ||||||
|  | } | ||||||
|  |    | ||||||
|  | .form-label-group input::placeholder { | ||||||
|  |     color: transparent; | ||||||
|  | } | ||||||
|  |    | ||||||
|  | .form-label-group input:not(:placeholder-shown) { | ||||||
|  |     padding-top: calc(var(--input-padding-y) + var(--input-padding-y) * (2 / 3)); | ||||||
|  |     padding-bottom: calc(var(--input-padding-y) / 3); | ||||||
|  | } | ||||||
|  |    | ||||||
|  | .form-label-group input:not(:placeholder-shown) ~ label { | ||||||
|  |     padding-top: calc(var(--input-padding-y) / 3); | ||||||
|  |     padding-bottom: calc(var(--input-padding-y) / 3); | ||||||
|  |     font-size: 12px; | ||||||
|  |     color: #777; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-check { | ||||||
|  |     margin-bottom: 2rem; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .checkbox input { | ||||||
|  |     transform: scale(1.5); | ||||||
|  |     margin-right: 1rem; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .signin-forgot-password { | ||||||
|  |     font-size: 14pt; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-submission-button { | ||||||
|  |     margin-bottom: 2rem; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-submission-button button, .form-submission-button a { | ||||||
|  |     margin: 1rem; | ||||||
|  |     vertical-align: middle; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-submission-button button span, .form-submission-button button svg, .form-submission-button a span, .form-submission-button a svg { | ||||||
|  |     margin: 0 2px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /*! Generated by Font Squirrel (https://www.fontsquirrel.com) on November 23, 2021 */ | ||||||
|  |  | ||||||
|  | @font-face { | ||||||
|  |     font-family: 'opendyslexic3bold'; | ||||||
|  |     src: url('../fonts/opendyslexic3-bold-webfont.woff2') format('woff2'), | ||||||
|  |          url('../fonts/opendyslexic3-bold-webfont.woff') format('woff'); | ||||||
|  |     font-weight: normal; | ||||||
|  |     font-style: normal; | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @font-face { | ||||||
|  |     font-family: 'opendyslexic3regular'; | ||||||
|  |     src: url('../fonts/opendyslexic3-regular-webfont.woff2') format('woff2'), | ||||||
|  |          url('../fonts/opendyslexic3-regular-webfont.woff') format('woff'); | ||||||
|  |     font-weight: normal; | ||||||
|  |     font-style: normal; | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @font-face { | ||||||
|  |     font-family: 'opendyslexicmonoregular'; | ||||||
|  |     src: url('../fonts/opendyslexicmono-regular-webfont.woff2') format('woff2'), | ||||||
|  |          url('../fonts/opendyslexicmono-regular-webfont.woff') format('woff'); | ||||||
|  |     font-weight: normal; | ||||||
|  |     font-style: normal; | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Fallback for Edge | ||||||
|  | -------------------------------------------------- */ | ||||||
|  | @supports (-ms-ime-align: auto) { | ||||||
|  |     .form-label-group label { | ||||||
|  |       display: none; | ||||||
|  |     } | ||||||
|  |     .form-label-group input::-ms-input-placeholder { | ||||||
|  |       color: #777; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |    | ||||||
|  |   /* Fallback for IE | ||||||
|  |   -------------------------------------------------- */ | ||||||
|  | @media all and (-ms-high-contrast: none), (-ms-high-contrast: active) { | ||||||
|  |     .form-label-group label { | ||||||
|  |         display: none; | ||||||
|  |     } | ||||||
|  |     .form-label-group input:-ms-input-placeholder { | ||||||
|  |         color: #777; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |    | ||||||
| @@ -0,0 +1,13 @@ | |||||||
|  | $(document).ready(function() { | ||||||
|  |     $("#od-font-test").click(function(){ | ||||||
|  |         $("body").css("font-family", "opendyslexic3regular") | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     $('.auth-code-input').keyup(function() { | ||||||
|  |         var input = $(this).val().split("-").join("").split("—").join(""); // remove hyphens and mdashes | ||||||
|  |         if (input.length > 0) { | ||||||
|  |           input = input.match(new RegExp('.{1,4}', 'g')).join("—"); | ||||||
|  |         } | ||||||
|  |         $(this).val(input); | ||||||
|  |     }); | ||||||
|  | }); | ||||||
| @@ -1,43 +0,0 @@ | |||||||
| <!DOCTYPE html> |  | ||||||
| <html> |  | ||||||
|     <head> |  | ||||||
|         <meta charset="utf8" /> |  | ||||||
|         <meta name="viewport" content="width=device-width, initial-scale=1" /> |  | ||||||
|         <link |  | ||||||
|             href="https://cdn.jsdelivr.net/npm/bootstrap@5.0.2/dist/css/bootstrap.min.css" |  | ||||||
|             rel="stylesheet" |  | ||||||
|             integrity="sha384-EVSTQN3/azprG1Anm3QDgpJLIm9Nao0Yz1ztcQTwFspd3yD65VohhpuuCOmLASjC" |  | ||||||
|             crossorigin="anonymous"> |  | ||||||
|         <link |  | ||||||
|             rel="stylesheet" |  | ||||||
|             href="https://cdn.jsdelivr.net/npm/@fortawesome/fontawesome-free@5.15.4/css/fontawesome.min.css" |  | ||||||
|             crossorigin="anonymous" |  | ||||||
|         /> |  | ||||||
|         <title>{% block title %} Site Title {% endblock %}</title> |  | ||||||
|     </head> |  | ||||||
|     <body> |  | ||||||
|  |  | ||||||
|         <nav class="navbar fixed-top navbar-expand-md navbar-dark bg-dark"> |  | ||||||
|             <div class="container"> |  | ||||||
|                 <a href="/" class="navbar-brand mb-0 h1">SKA Refereeing Test </a> |  | ||||||
|             </div> |  | ||||||
|         </nav> |  | ||||||
|  |  | ||||||
|         <!-- JQuery, Popper, and Bootstrap js dependencies --> |  | ||||||
|         <script |  | ||||||
|             src="https://code.jquery.com/jquery-3.6.0.slim.min.js" |  | ||||||
|             integrity="sha256-u7e5khyithlIdTpu22PHhENmPcRdFiHRjhAuHcs05RI=" |  | ||||||
|             crossorigin="anonymous"> |  | ||||||
|         </script> |  | ||||||
|         <script |  | ||||||
|             src="https://cdn.jsdelivr.net/npm/bootstrap@5.1.3/dist/js/bootstrap.bundle.min.js" |  | ||||||
|             integrity="sha384-ka7Sk0Gln4gmtz2MlQnikT1wXgYsOg+OMhuP+IlRH9sENBO0LRn5q+8nbTov4+1p" |  | ||||||
|             crossorigin="anonymous"> |  | ||||||
|         </script> |  | ||||||
|         <!-- Custom js --> |  | ||||||
|         <script |  | ||||||
|             type="text/javascript" |  | ||||||
|             src="{{ url_for('.static', filename='js/script.js') }}" |  | ||||||
|         ></script> |  | ||||||
|     </body> |  | ||||||
| </html> |  | ||||||
							
								
								
									
										59
									
								
								ref-test/quiz/templates/quiz/components/base.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								ref-test/quiz/templates/quiz/components/base.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,59 @@ | |||||||
|  | <!DOCTYPE html> | ||||||
|  | <html> | ||||||
|  |     <head> | ||||||
|  |         <meta charset="utf8" /> | ||||||
|  |         <meta name="viewport" content="width=device-width, initial-scale=1" /> | ||||||
|  |         <link | ||||||
|  |             href="https://cdn.jsdelivr.net/npm/bootstrap@5.0.2/dist/css/bootstrap.min.css" | ||||||
|  |             rel="stylesheet" | ||||||
|  |             integrity="sha384-EVSTQN3/azprG1Anm3QDgpJLIm9Nao0Yz1ztcQTwFspd3yD65VohhpuuCOmLASjC" | ||||||
|  |             crossorigin="anonymous"> | ||||||
|  |         <link | ||||||
|  |             rel="stylesheet" | ||||||
|  |             href="https://cdn.jsdelivr.net/npm/bootstrap-icons@1.7.1/font/bootstrap-icons.css"> | ||||||
|  |         <link  | ||||||
|  |             rel="stylesheet" | ||||||
|  |             href="{{ url_for('.static', filename='css/style.css') }}" | ||||||
|  |         /> | ||||||
|  |         <title>{% block title %} SKA Referee Test {% endblock %}</title> | ||||||
|  |     </head> | ||||||
|  |     <body class="bg-light"> | ||||||
|  |  | ||||||
|  |         {% block navbar %} | ||||||
|  |             {% include "quiz/components/navbar.html" %} | ||||||
|  |         {% endblock %} | ||||||
|  |          | ||||||
|  |         <div class="container quiz-container"> | ||||||
|  |             {% block top_alerts %} | ||||||
|  |                 {% include "quiz/components/server-alerts.html" %} | ||||||
|  |             {% endblock %} | ||||||
|  |             {% block content %}{% endblock %} | ||||||
|  |         </div> | ||||||
|  |  | ||||||
|  |         <footer class="container site-footer"> | ||||||
|  |             {% include "quiz/components/footer.html" %} | ||||||
|  |         </footer> | ||||||
|  |  | ||||||
|  |         <!-- JQuery, Popper, and Bootstrap js dependencies --> | ||||||
|  |         <script | ||||||
|  |             src="https://code.jquery.com/jquery-3.6.0.min.js" | ||||||
|  |             integrity="sha256-/xUj+3OJU5yExlq6GSYGSHk7tPXikynS7ogEvDej/m4=" | ||||||
|  |             crossorigin="anonymous"> | ||||||
|  |         </script> | ||||||
|  |         <script | ||||||
|  |             src="https://cdn.jsdelivr.net/npm/@popperjs/core@2.10.2/dist/umd/popper.min.js" | ||||||
|  |             integrity="sha384-7+zCNj/IqJ95wo16oMtfsKbZ9ccEh31eOz1HGyDuCQ6wgnyJNSYdrPa03rtR1zdB" | ||||||
|  |             crossorigin="anonymous"> | ||||||
|  |         </script> | ||||||
|  |         <script | ||||||
|  |             src="https://cdn.jsdelivr.net/npm/bootstrap@5.1.3/dist/js/bootstrap.min.js" | ||||||
|  |             integrity="sha384-QJHtvGhmr9XOIpI6YVutG+2QOK9T+ZnN4kzFN1RtK3zEFEIsxhlmWl5/YESvpZ13" | ||||||
|  |             crossorigin="anonymous" | ||||||
|  |         ></script> | ||||||
|  |         <!-- Custom js --> | ||||||
|  |         <script | ||||||
|  |             type="text/javascript" | ||||||
|  |             src="{{ url_for('.static', filename='js/script.js') }}" | ||||||
|  |         ></script> | ||||||
|  |     </body> | ||||||
|  | </html> | ||||||
							
								
								
									
										0
									
								
								ref-test/quiz/templates/quiz/components/footer.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								ref-test/quiz/templates/quiz/components/footer.html
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										5
									
								
								ref-test/quiz/templates/quiz/components/navbar.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								ref-test/quiz/templates/quiz/components/navbar.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | <nav class="navbar fixed-top navbar-expand-md navbar-dark bg-dark"> | ||||||
|  |     <div class="container"> | ||||||
|  |         <a href="/" class="navbar-brand mb-0 h1">SKA Refereeing Test </a> | ||||||
|  |     </div> | ||||||
|  | </nav> | ||||||
							
								
								
									
										24
									
								
								ref-test/quiz/templates/quiz/index.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								ref-test/quiz/templates/quiz/index.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | |||||||
|  | {% extends "quiz/components/base.html" %} | ||||||
|  |  | ||||||
|  | {% block content %} | ||||||
|  |     <h1>SKA Refereeing Theory Exam</h1> | ||||||
|  |  | ||||||
|  |     <p> | ||||||
|  |         This app will allow you to take the Exam on-line. This app should also allow you to adjust the way the quiz is rendered to suit your access needs. This could include using a screen reader, changing the display font size or typeface, or navigating questions and answers via the keyboard. | ||||||
|  |     </p> | ||||||
|  |  | ||||||
|  |     <p> | ||||||
|  |         Instructions | ||||||
|  |     </p> | ||||||
|  |  | ||||||
|  |     <p> | ||||||
|  |         Other Info | ||||||
|  |     </p> | ||||||
|  |  | ||||||
|  |     <p> | ||||||
|  |         When you are ready to begin the quiz, click the following button. | ||||||
|  |     </p> | ||||||
|  |  | ||||||
|  |     <a href="{{ url_for('quiz_views.start') }}" class="btn btn-success">Take the Quiz</a> | ||||||
|  |  | ||||||
|  | {% endblock %} | ||||||
							
								
								
									
										0
									
								
								ref-test/quiz/templates/quiz/privacy.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								ref-test/quiz/templates/quiz/privacy.html
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										43
									
								
								ref-test/quiz/templates/quiz/start-quiz.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								ref-test/quiz/templates/quiz/start-quiz.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | |||||||
|  | {% extends "quiz/components/base.html" %} | ||||||
|  | {% import "bootstrap/wtf.html" as wtf %} | ||||||
|  |  | ||||||
|  | {% block content %} | ||||||
|  |     <div class="form-container"> | ||||||
|  |         <form name="form-quiz-start" class="form-quiz-start"> | ||||||
|  |             <h2 class="form-heading">Take the Exam</h2> | ||||||
|  |             {{ form.hidden_tag() }} | ||||||
|  |             <div class="form-label-group"> | ||||||
|  |                 {{ form.first_name(class_="form-control", autofocus=true, placeholder="Enter First Name(s)") }} | ||||||
|  |                 {{ form.first_name.label }} | ||||||
|  |             </div> | ||||||
|  |             <div class="form-label-group"> | ||||||
|  |                 {{ form.surname(class_="form-control", placeholder="Enter Surname") }} | ||||||
|  |                 {{ form.surname.label }} | ||||||
|  |             </div> | ||||||
|  |             <div class="form-label-group"> | ||||||
|  |                 {{ form.email(class_="form-control", placeholder="Enter Email Address") }} | ||||||
|  |                 {{ form.email.label }} | ||||||
|  |             </div> | ||||||
|  |             <div class="form-label-group"> | ||||||
|  |                 {{ form.club(class_="form-control", placeholder="Enter Affiliated Club") }} | ||||||
|  |                 {{ form.club.label }} | ||||||
|  |             </div> | ||||||
|  |             <div class="form-label-group"> | ||||||
|  |                 {{ form.auth_code(class_="form-control auth-code-input", placeholder="Enter Exam Code") }} | ||||||
|  |                 {{ form.auth_code.label }} | ||||||
|  |             </div> | ||||||
|  |             <div class="form-label-group"> | ||||||
|  |                 {{ form.user_code(class_="form-control", placeholder="Enter User Code") }} | ||||||
|  |                 {{ form.user_code.label }} | ||||||
|  |             </div> | ||||||
|  |             {% include "admin/components/client-alerts.html" %} | ||||||
|  |             <div class="container form-submission-button"> | ||||||
|  |                 <div class="row"> | ||||||
|  |                     <div class="col text-center"> | ||||||
|  |                         <button class="btn btn-md btn-success btn-block" type="submit">Start Quiz</button> | ||||||
|  |                     </div> | ||||||
|  |                 </div> | ||||||
|  |             </div> | ||||||
|  |         </form> | ||||||
|  |     </div> | ||||||
|  | {% endblock %} | ||||||
| @@ -1,8 +1,9 @@ | |||||||
| from flask import Blueprint | from flask import Blueprint, render_template | ||||||
|  |  | ||||||
| views = Blueprint( | views = Blueprint( | ||||||
|     'quiz_views', |     'quiz_views', | ||||||
|     __name__, |     __name__, | ||||||
|  |     static_url_path='', | ||||||
|     template_folder='templates', |     template_folder='templates', | ||||||
|     static_folder='static' |     static_folder='static' | ||||||
| ) | ) | ||||||
| @@ -10,7 +11,13 @@ views = Blueprint( | |||||||
| @views.route('/') | @views.route('/') | ||||||
| @views.route('/home/') | @views.route('/home/') | ||||||
| def home(): | def home(): | ||||||
|     return f'<h1>Ref Test Home Page</h1>' |     return render_template('/quiz/index.html') | ||||||
|  |  | ||||||
|  | @views.route('/start/', methods = ['GET', 'POST']) | ||||||
|  | def start(): | ||||||
|  |     from .forms import StartQuiz | ||||||
|  |     form = StartQuiz() | ||||||
|  |     return render_template('/quiz/start-quiz.html', form=form) | ||||||
|  |  | ||||||
| @views.route('/privacy/') | @views.route('/privacy/') | ||||||
| def privacy(): | def privacy(): | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| from pymongo import collection | from pymongo import collection | ||||||
| from . import encrypt, decrypt | from . import encrypt, decrypt | ||||||
| encrypted_parameters = ['username', 'email', 'name', 'club'] | encrypted_parameters = ['username', 'email', 'name', 'club', 'creator'] | ||||||
|  |  | ||||||
| def decrypt_find(collection:collection, query:dict): | def decrypt_find(collection:collection, query:dict): | ||||||
|     cursor = collection.find({}) |     cursor = collection.find({}) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user