joeni

Refinement of roster module in administration

The functionality of the roster has been moved to its own module, to keep it nicely seperated from what is the view code.
Several updates in the models have been made, so the migrations have been included.
Added a couple models to the admin registry so elements can be added in the admin interface.
Also added some new style classes for the roster to indicate important things like a change, new event, or notification.

Author
Maarten 'Vngngdn' Vangeneugden
Date
Feb. 4, 2018, 7:21 p.m.
Hash
838a4c30e3ac78883df4d4a301ef8929e5be6ffe
Parent
db2baced09f3dd36acf8ca5ebb352e88c7b30ed1
Modified files
administration/admin.py
administration/migrations/0012_auto_20180204_1349.py
administration/migrations/0013_auto_20180204_1444.py
administration/models.py
administration/roster.py
administration/templates/administration/roster.djhtml
administration/views.py
courses/admin.py
courses/migrations/0005_auto_20180204_1349.py
courses/migrations/0006_auto_20180204_1349.py
courses/models.py
static/css/base.css

administration/admin.py

5 additions and 0 deletions.

View changes Hide changes
1
1
from .models import *
2
2
from django.contrib.auth.admin import UserAdmin
3
3
4
4
admin.site.register(User, UserAdmin)
5
5
6
6
admin.site.register(Curriculum)
7
7
admin.site.register(UserData)
8
8
admin.site.register(CourseResult)
9
9
admin.site.register(PreRegistration)
10
10
admin.site.register(Room)
11
11
admin.site.register(RoomReservation)
12
12
admin.site.register(Degree)
13
13
14
14
admin.site.register(CourseEvent)
+
15
admin.site.register(CourseEvent)
15
16
admin.site.register(UniversityEvent)
16
17
admin.site.register(StudyEvent)
17
18
+
19
admin.site.register(ExamCommissionDecision)
+
20
admin.site.register(EducationDepartmentMessages)
+
21
+
22

administration/migrations/0012_auto_20180204_1349.py

57 additions and 0 deletions.

View changes Hide changes
+
1
+
2
import administration.models
+
3
from django.conf import settings
+
4
from django.db import migrations, models
+
5
import django.db.models.deletion
+
6
+
7
+
8
class Migration(migrations.Migration):
+
9
+
10
    dependencies = [
+
11
        ('administration', '0011_auto_20180128_1935'),
+
12
    ]
+
13
+
14
    operations = [
+
15
        migrations.CreateModel(
+
16
            name='EducationDepartmentMessages',
+
17
            fields=[
+
18
                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+
19
                ('date', models.DateField(auto_now_add=True)),
+
20
                ('title', models.CharField(help_text='A short, well-describing title for this message.', max_length=64)),
+
21
                ('text', models.TextField(help_text='The message text. Org syntax available.')),
+
22
            ],
+
23
            options={
+
24
                'verbose_name': 'Decision of the exam commission',
+
25
                'verbose_name_plural': 'Decisions of the exam commission',
+
26
            },
+
27
        ),
+
28
        migrations.CreateModel(
+
29
            name='ExamCommissionDecision',
+
30
            fields=[
+
31
                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+
32
                ('date', models.DateField(auto_now_add=True)),
+
33
                ('text', models.TextField(help_text='The text describing the decision. Org syntax available.')),
+
34
                ('user', models.ForeignKey(help_text='The recipient of this decision.', on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL)),
+
35
            ],
+
36
            options={
+
37
                'verbose_name': 'Decision of the exam commission',
+
38
                'verbose_name_plural': 'Decisions of the exam commission',
+
39
            },
+
40
        ),
+
41
        migrations.AddField(
+
42
            model_name='courseresult',
+
43
            name='year',
+
44
            field=models.PositiveIntegerField(default=2018, help_text="The academic year this course took place in. If 2018 is entered, then that means academic year '2018-2019'."),
+
45
        ),
+
46
        migrations.AlterField(
+
47
            model_name='event',
+
48
            name='begin_time',
+
49
            field=models.DateTimeField(help_text="The begin date and time that this event takes place. This value must be a quarter of an hour (0, 15, 30, 45), and take place <em>before</em> this event's end time.", validators=[administration.models.validate_event_time], verbose_name='begin time'),
+
50
        ),
+
51
        migrations.AlterField(
+
52
            model_name='event',
+
53
            name='end_time',
+
54
            field=models.DateTimeField(help_text="The end date and time that this event takes place. This value must be a quarter of an hour (0, 15, 30, 45), and take place <em>after</em> this event's begin time.", validators=[administration.models.validate_event_time], verbose_name='end time'),
+
55
        ),
+
56
    ]
+
57

administration/migrations/0013_auto_20180204_1444.py

29 additions and 0 deletions.

View changes Hide changes
+
1
+
2
from django.conf import settings
+
3
from django.db import migrations, models
+
4
import django.db.models.deletion
+
5
+
6
+
7
class Migration(migrations.Migration):
+
8
+
9
    dependencies = [
+
10
        ('administration', '0012_auto_20180204_1349'),
+
11
    ]
+
12
+
13
    operations = [
+
14
        migrations.AlterModelOptions(
+
15
            name='educationdepartmentmessages',
+
16
            options={'verbose_name': 'Message of the education department', 'verbose_name_plural': 'Messages of the education department'},
+
17
        ),
+
18
        migrations.AlterField(
+
19
            model_name='courseevent',
+
20
            name='group',
+
21
            field=models.ForeignKey(blank=True, help_text="Some courses have multiple groups. If that's the case, and this event is only for a specific group, then that group must be referenced here.", null=True, on_delete=django.db.models.deletion.CASCADE, to='courses.CourseGroup'),
+
22
        ),
+
23
        migrations.AlterField(
+
24
            model_name='curriculum',
+
25
            name='student',
+
26
            field=models.ForeignKey(limit_choices_to={'groups': 1}, on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL, unique_for_year='year'),
+
27
        ),
+
28
    ]
+
29

administration/models.py

6 additions and 4 deletions.

View changes Hide changes
1
1
from django.core.exceptions import ValidationError
2
2
from django.core.validators import MaxValueValidator
3
3
from django.utils.translation import ugettext_lazy as _
4
4
from django.contrib.auth.models import AbstractUser
5
5
import datetime
6
6
import os
7
7
import uuid
8
8
9
9
def validate_IBAN(value):
10
10
    """ Validates if the given value qualifies as a valid IBAN number.
11
11
    This validator checks if the structure is valid, and calculates the control
12
12
    number if the structure is correct. If the control number fails, or the
13
13
    structure is invalid, a ValidationError will be raised. In that case,
14
14
    the Error will specify whether the structure is incorrect, or the control
15
15
    number is not valid.
16
16
    """
17
17
    # FIXME: This function is not complete. When there's time, implement
18
18
    # as specified at https://nl.wikipedia.org/wiki/International_Bank_Account_Number#Structuur
19
19
    if False:
20
20
        raise ValidationError(
21
21
            _('%(value)s is not a valid IBAN number.'),
22
22
            params={'value': value},)
23
23
def validate_BIC(value):
24
24
    """ Same functionality as validate_IBAN, but for BIC-codes. """
25
25
    # FIXME: This function is not complete. When there's time, implement
26
26
    # as specified at https://nl.wikipedia.org/wiki/Business_Identifier_Code
27
27
    pass
28
28
29
29
class User(AbstractUser):
30
30
    """ Replacement for the standard Django User model. """
31
31
    number = models.AutoField(
32
32
        primary_key=True,
33
33
        help_text=_("The number assigned to this user."),
34
34
        )
35
35
    created = models.DateField(auto_now_add=True)
36
36
37
37
class UserData(models.Model):
38
38
    user = models.OneToOneField(User, on_delete=models.CASCADE)
39
39
    first_name = models.CharField(max_length=64, blank=False)
40
40
    last_name = models.CharField(max_length=64, blank=False)
41
41
    title = models.CharField(
42
42
        max_length=64,
43
43
        blank=True,
44
44
        help_text=_("The academic title of this user, if applicable."),
45
45
        )
46
46
    DOB = models.DateField(
47
47
        blank=False,
48
48
        #editable=False,
49
49
        help_text=_("The date of birth of this user."),
50
50
        )
51
51
    POB = models.CharField(
52
52
        max_length=64,
53
53
        blank=False,
54
54
        #editable=False,
55
55
        help_text=_("The place of birth of this user."),
56
56
        )
57
57
    nationality = models.CharField(
58
58
        max_length=64,
59
59
        blank=False,
60
60
        help_text=_("The current nationality of this user."),
61
61
        default="Belg",
62
62
        )
63
63
    # XXX: What if this starts with zeros?
64
64
    national_registry_number = models.BigIntegerField(
65
65
        blank=True,  # Only possible if Belgian
66
66
        # TODO Validator!
67
67
        #editable=False,
68
68
        help_text=_("The assigned national registry number of this user."),
69
69
        )
70
70
    civil_status = models.CharField(
71
71
        max_length=32,
72
72
        choices = (
73
73
            ("Single", _("Single")),
74
74
            ("Married", _("Married")),
75
75
            ("Divorced", _("Divorced")),
76
76
            ("Widowed", _("Widowed")),
77
77
            ("Partnership", _("Partnership")),
78
78
            ),
79
79
        blank=False,
80
80
        # There may be more; consult http://www.aantrekkingskracht.com/trefwoord/burgerlijke-staat
81
81
        # for more information.
82
82
        help_text=_("The civil/marital status of the user."),
83
83
        )
84
84
85
85
    is_staff = models.BooleanField(
86
86
        default=False,
87
87
        help_text=_("Determines if this user is part of the university's staff."),
88
88
        )
89
89
    is_student = models.BooleanField(
90
90
        default=True,
91
91
        help_text=_("Indicates if this user is a student at the university."),
92
92
        )
93
93
94
94
    # Home address
95
95
    home_street = models.CharField(max_length=64, blank=False)
96
96
    home_number = models.PositiveSmallIntegerField(blank=False)
97
97
    home_bus = models.CharField(max_length=10, null=True, blank=True)
98
98
    home_postal_code = models.PositiveIntegerField(blank=False)
99
99
    home_city = models.CharField(max_length=64, blank=False)
100
100
    home_country = models.CharField(max_length=64, blank=False, default="België")
101
101
    home_telephone = models.CharField(
102
102
        max_length=64,
103
103
        help_text=_("The telephone number for the house address. Prefix 0 can be presented with the national call code in the system (\"32\" for Belgium)."),
104
104
        )
105
105
    # Study address
106
106
    study_street = models.CharField(max_length=64, blank=True)
107
107
    study_number = models.PositiveSmallIntegerField(blank=True)
108
108
    study_bus = models.CharField(max_length=10, null=True, blank=True)
109
109
    study_postal_code = models.PositiveSmallIntegerField(blank=True)
110
110
    study_country = models.CharField(max_length=64, blank=True)
111
111
    study_telephone = models.CharField(
112
112
        blank=True,
113
113
        max_length=64,
114
114
        help_text=_("The telephone number for the study address. Prefix 0 can be presented with the national call code in the system."),
115
115
        )
116
116
    study_cellphone = models.CharField(
117
117
        max_length=64,
118
118
        help_text=_("The cellphone number of the person. Prefix 0 can be presented with then national call code in the system."),
119
119
        )
120
120
    # Titularis address
121
121
    # XXX: These fields are only required if this differs from the user itself.
122
122
    titularis_street = models.CharField(max_length=64, null=True, blank=True)
123
123
    titularis_number = models.PositiveSmallIntegerField(null=True)
124
124
    titularis_bus = models.CharField(max_length=10, null=True, blank=True)
125
125
    titularis_postal_code = models.PositiveSmallIntegerField(null=True)
126
126
    titularis_country = models.CharField(max_length=64, null=True, blank=True)
127
127
    titularis_telephone = models.CharField(
128
128
        max_length=64,
129
129
        help_text=_("The telephone number of the titularis. Prefix 0 can be presented with the national call code in the system."),
130
130
        null=True,
131
131
        )
132
132
133
133
    # Financial details
134
134
    bank_account_number = models.CharField(
135
135
        max_length=34,  # Max length of all IBAN account numbers
136
136
        validators=[validate_IBAN],
137
137
        help_text=_("The IBAN of this user. No spaces!"),
138
138
        )
139
139
    BIC = models.CharField(
140
140
        max_length=11,
141
141
        validators=[validate_BIC],
142
142
        help_text=_("The BIC of this user's bank."),
143
143
        )
144
144
145
145
""" NOTE: What about all the other features that should be in the administration?
146
146
While there are a lot of things to cover, as of now, I have no way to know which
147
147
ones are still valid, which are deprecated, and so on...
148
148
Additionally, every feature may have a different set of requirements, data,
149
149
and it's very likely making an abstract class won't do any good. Thus I have
150
150
decided to postpone making additional tables and forms for these features until
151
151
I have clearance about certain aspects. """
152
152
153
153
class Curriculum(models.Model):
154
154
    """ The curriculum of a particular student.
155
155
    Every academic year, a student has to hand in a curriculum (s)he wishes to
156
156
    follow. This is then reviewed by a committee. A curriculum exists of all the
157
157
    courses one wants to partake in in a certain year. """
158
158
    student = models.ForeignKey(
159
159
        "User",
160
160
        on_delete=models.CASCADE,
161
161
        limit_choices_to={'is_student': True},
162
-
        null=False,
+
162
        null=False,
163
163
        #editable=False,
164
164
        unique_for_year="year",  # Only 1 curriculum per year
165
165
        )
166
166
    year = models.DateField(
167
167
        auto_now_add=True,
168
168
        db_index=True,
169
169
        help_text=_("The academic year for which this curriculum is. "
170
170
                    "If this field is equal to 2008, then that means "
171
171
                    "this curriculum is for the academic year "
172
172
                    "2008-2009."),
173
173
        )
174
174
    last_modified = models.DateTimeField(
175
175
        auto_now=True,
176
176
        help_text=_("The last timestamp that this was updated."),
177
177
        )
178
178
    course_programmes = models.ManyToManyField(
179
179
        "courses.CourseProgramme",
180
180
        null=False,
181
181
        help_text=_("All the course programmes included in this curriculum."),
182
182
        )
183
183
    approved = models.NullBooleanField(
184
184
        default=None,
185
185
        help_text=_("Indicates if this curriculum has been approved. If true, "
186
186
                    "that means the responsible committee has reviewed and "
187
187
                    "approved the student for this curriculum. False otherwise. "
188
188
                    "If review is still pending, the value is NULL. Modifying "
189
189
                    "the curriculum implies this setting is set to NULL again."),
190
190
        )
191
191
    note = models.TextField(
192
192
        blank=True,
193
193
        help_text=_("Additional notes regarding this curriculum. This has "
194
194
                    "multiple uses. For the student, it is used to clarify "
195
195
                    "any questions, or to motivate why (s)he wants to take a "
196
196
                    "course for which the requirements were not met. "
197
197
                    "The reviewing committee can use this field to argument "
198
198
                    "their decision, especially for when the curriculum is "
199
199
                    "denied."),
200
200
        )
201
201
202
202
    def courses(self):
203
203
        """ Returns a set of all the courses that are in this curriculum.
204
204
        This is not the same as CourseProgrammes, as these can differ depending
205
205
        on which study one follows. """
206
206
        course_set = set()
207
207
        for course_programme in self.course_programmes:
208
208
            course_set.add(course_programme.course)
209
209
        return course_set
210
210
211
211
    def curriculum_type(self):
212
212
        """ Returns the type of this curriculum. At the moment, this is
213
213
        either a standard programme, or an individualized programme. """
214
214
        # Currently: A standard programme means: All courses are from the
215
215
        # same study, ánd from the same year. Additionally, all courses
216
216
        # from that year must've been taken.
217
217
        # FIXME: Need a way to determine what is the standard programme.
218
218
        # If not possible, make this a charfield with options or something
219
219
        pass
220
220
221
221
    def __str__(self):
222
222
        year = self.year.year
223
223
        if self.year.month < 7:
224
224
            return str(self.student) +" | "+ str(year-1) +"-"+ str(year)
225
225
        else:
226
226
            return str(self.student) +" | "+ str(year) +"-"+ str(year+1)
227
227
228
228
229
229
class CourseResult(models.Model):
230
230
    """ A student has to obtain a certain course result. These are stored here,
231
231
    together with all the appropriate information. """
232
232
    # TODO: Validate that a course programme for a student can only be made once per year for each course, if possible.
233
233
    CRED = _("Credit acquired")
234
234
    FAIL = _("Credit not acquired")
235
235
    TLRD = _("Tolerated")
236
236
    ITLD = _("Tolerance used")
237
237
    BDRG = _("Fraud committed")
238
238
    VRST = _("Exemption")
239
239
    STOP = _("Course cancelled")
240
240
    # Possible to add more in the future
241
241
242
242
    student = models.ForeignKey(
243
243
        "User",
244
244
        on_delete=models.CASCADE,
245
245
        limit_choices_to={'is_student': True},
246
246
        null=False,
247
247
        )
248
248
    course_programme = models.ForeignKey(
249
249
        "courses.CourseProgramme",
250
250
        on_delete=models.PROTECT,
251
251
        null=False,
252
252
        )
253
253
    year = models.PositiveIntegerField(
254
254
        null=False,
255
255
        default=datetime.date.today().year,
256
256
        help_text=_("The academic year this course took place in. If 2018 is entered, "
257
257
                    "then that means academic year '2018-2019'."),
258
258
        )
259
259
    released = models.DateField(
260
260
        auto_now=True,
261
261
        help_text=_("The date that this result was last updated."),
262
262
        )
263
263
    first_score = models.PositiveSmallIntegerField(
264
264
        null=True,  # It's possible a score does not exist.
265
265
        validators=[MaxValueValidator(
266
266
            20,
267
267
            _("The score mustn't be higher than 20."),
268
268
            )],
269
269
        )
270
270
    second_score = models.PositiveSmallIntegerField(
271
271
        null=True,
272
272
        validators=[MaxValueValidator(
273
273
            20,
274
274
            _("The score mustn't be higher than 20."),
275
275
            )],
276
276
        )
277
277
    result = models.CharField(
278
278
        max_length=10,
279
279
        choices = (
280
280
            ("CRED", CRED),
281
281
            ("FAIL", FAIL),
282
282
            ("TLRD", TLRD),
283
283
            ("ITLD", ITLD),
284
284
            ),
285
285
        blank=False,
286
286
        help_text=_("The final result this record constitutes."),
287
287
        )
288
288
289
289
    def __str__(self):
290
290
        stdnum = str(self.student.number)
291
291
        result = self.result
292
292
        if result == "CRED":
293
293
            if self.first_score < 10:
294
294
                result = "C" + self.first_score + "1"
295
295
            else:
296
296
                result = "C" + self.second_score + "2"
297
297
        course = str(self.course_programme.course)
298
298
        return stdnum +" ("+ result +") | "+ course
299
299
300
300
class PreRegistration(models.Model):
301
301
    """ At the beginning of the new academic year, students can register
302
302
    themselves at the university. Online, they can do a preregistration already.
303
303
    These records are stored here and can later be retrieved for the actual
304
304
    registration process.
305
305
    Note: The current system in use at Hasselt University provides a password system.
306
306
    That will be eliminated here. Just make sure that the entered details are correct.
307
307
    Should there be an error, and the same email address is used to update something,
308
308
    a mail will be sent to that address to verify this was a genuine update."""
309
309
    created = models.DateField(auto_now_add=True)
310
310
    first_name = models.CharField(max_length=64, blank=False, help_text=_("Your first name."))
311
311
    last_name = models.CharField(max_length=64, blank=False, help_text=_("Your last name."))
312
312
    additional_names = models.CharField(max_length=64, blank=True, help_text=_("Any additional names."))
313
313
    title = models.CharField(
314
314
        max_length=64,
315
315
        blank=True,
316
316
        help_text=_("Any additional titles, prefixes, ..."),
317
317
        )
318
318
    DOB = models.DateField(
319
319
        blank=False,
320
320
        #editable=False,
321
321
        help_text=_("Your date of birth."),
322
322
        )
323
323
    POB = models.CharField(
324
324
        max_length=64,
325
325
        blank=False,
326
326
        #editable=False,
327
327
        help_text=_("The place you were born."),
328
328
        )
329
329
    nationality = models.CharField(
330
330
        max_length=64,
331
331
        blank=False,
332
332
        help_text=_("Your current nationality."),
333
333
        )
334
334
    national_registry_number = models.BigIntegerField(
335
335
        null=True,
336
336
        help_text=_("If you have one, your national registry number."),
337
337
        )
338
338
    civil_status = models.CharField(
339
339
        max_length=32,
340
340
        choices = (
341
341
            ("Single", _("Single")),
342
342
            ("Married", _("Married")),
343
343
            ("Divorced", _("Divorced")),
344
344
            ("Widowed", _("Widowed")),
345
345
            ("Partnership", _("Partnership")),
346
346
            ),
347
347
        blank=False,
348
348
        # There may be more; consult http://www.aantrekkingskracht.com/trefwoord/burgerlijke-staat
349
349
        # for more information.
350
350
        help_text=_("Your civil/marital status."),
351
351
        )
352
352
    email = models.EmailField(
353
353
        blank=False,
354
354
        unique=True,
355
355
        help_text=_("The e-mail address we will use to communicate until your actual registration."),
356
356
        )
357
357
    study = models.ForeignKey(
358
358
        "courses.Study",
359
359
        on_delete=models.PROTECT,
360
360
        null=False,
361
361
        help_text=_("The study you wish to follow. Be sure to provide all legal"
362
362
                    "documents that are required for this study with this "
363
363
                    "application, or bring them with you to the final registration."),
364
364
        )
365
365
    study_type = models.CharField(
366
366
        max_length=32,
367
367
        choices = (
368
368
            ("Diplom contract", _("Diplom contract")),
369
369
            ("Exam contract", _("Exam contract")),
370
370
            ("Credit contract", _("Credit contract")),
371
371
            ),
372
372
        blank=False,
373
373
        help_text=_("The type of study contract you wish to follow."),
374
374
        )
375
375
    document = models.FileField(
376
376
        upload_to="pre-enrollment/%Y",
377
377
        help_text=_("Any legal documents regarding your enrollment."),
378
378
        )
379
379
    # XXX: If the database in production is PostgreSQL, comment document, and
380
380
    # uncomment the next column.
381
381
    """documents = models.ArrayField(
382
382
        models.FileField(upload_to="pre-enrollment/%Y"),
383
383
        help_text=_("Any legal documents regarding your enrollment."),
384
384
        )"""
385
385
386
386
    def __str__(self):
387
387
        name = self.last_name +" "+ self.first_name
388
388
        dob = self.DOB.strftime("%d/%m/%Y")
389
389
        return name +" | "+ dob
390
390
391
391
392
392
# Planning and organization related tables
393
393
class Room(models.Model):
394
394
    """ Represents a room in the university.
395
395
    Rooms can have a number of properties, which are stored in the database.
396
396
    """
397
397
    # Types of rooms
398
398
    LABORATORY = _("Laboratory")  # Chemistry/Physics equipped rooms
399
399
    CLASS_ROOM = _("Class room")  # Simple class rooms
400
400
    AUDITORIUM = _("Auditorium")  # Large rooms with ample seating and equipment for lectures
401
401
    PC_ROOM    = _("PC room"   )  # Rooms equipped for executing PC related tasks
402
402
    PUBLIC_ROOM= _("Public room") # Restaurants, restrooms, ... general public spaces
403
403
    OFFICE     = _("Office"    )  # Private offices for staff
404
404
    PRIVATE_ROOM = _("Private room")  # Rooms accessible for a limited public; cleaning cupboards, kitchens, ...
405
405
    WORKSHOP   = _("Workshop"  )  # Rooms with hardware equipment to build and work on materials
406
406
    OTHER      = _("Other"     )  # Rooms that do not fit in any other category
407
407
408
408
409
409
    name = models.CharField(
410
410
        max_length=20,
411
411
        primary_key=True,
412
412
        blank=False,
413
413
        help_text=_("The name of this room. If more appropriate, this can be the colloquial name."),
414
414
        )
415
415
    seats = models.PositiveSmallIntegerField(
416
416
        help_text=_("The amount of available seats in this room. This can be handy for exams for example."),
417
417
        )
418
418
    wheelchair_accessible = models.BooleanField(default=True)
419
419
    exams_equipped = models.BooleanField(
420
420
        default=True,
421
421
        help_text=_("Indicates if exams can reasonably be held in this room."),
422
422
        )
423
423
    computers_available = models.PositiveSmallIntegerField(
424
424
        default=False,
425
425
        help_text=_("Indicates how many computers are available in this room."),
426
426
        )
427
427
    projector_available = models.BooleanField(
428
428
        default=False,
429
429
        help_text=_("Indicates if a projector is available at this room."),
430
430
        )
431
431
    blackboards_available = models.PositiveSmallIntegerField(
432
432
        help_text=_("The amount of blackboards available in this room."),
433
433
        )
434
434
    whiteboards_available = models.PositiveSmallIntegerField(
435
435
        help_text=_("The amount of whiteboards available in this room."),
436
436
        )
437
437
    category = models.CharField(
438
438
        max_length=16,
439
439
        blank=False,
440
440
        choices = (
441
441
            ("LABORATORY", LABORATORY),
442
442
            ("CLASS_ROOM", CLASS_ROOM),
443
443
            ("AUDITORIUM", AUDITORIUM),
444
444
            ("PC_ROOM", PC_ROOM),
445
445
            ("PUBLIC_ROOM", PUBLIC_ROOM),
446
446
            ("OFFICE", OFFICE),
447
447
            ("PRIVATE_ROOM", PRIVATE_ROOM),
448
448
            ("WORKSHOP", WORKSHOP),
449
449
            ("OTHER", OTHER),
450
450
            ),
451
451
        help_text=_("The category that best suits the character of this room."),
452
452
        )
453
453
    reservable = models.BooleanField(
454
454
        default=True,
455
455
        help_text=_("Indicates if this room can be reserved for something."),
456
456
        )
457
457
    note = models.TextField(
458
458
        blank=True,
459
459
        help_text=_("If some additional info is required for this room, like a "
460
460
                    "characteristic property (e.g. 'Usually occupied by 2BACH "
461
461
                    "informatics'), state it here."),
462
462
        )
463
463
    # TODO: Add a campus/building field or not?
464
464
465
465
    def reservation_possible(self, begin, end, seats=None):
466
466
        """ Returns a boolean indicating if reservating during the given time
467
467
        is possible. If the begin overlaps with a reservation's end or vice versa,
468
468
        this is regarded as possible.
469
469
        Takes seats as optional argument. If not specified, it is assumed the entire
470
470
        room has to be reserved. """
471
471
        if self.reservable is False:
472
472
            return False
473
473
        if seats is not None and seats < 0: raise ValueError(_("seats ∈ ℕ"))
474
474
475
475
        reservations = RoomReservation.objects.filter(room=self)
476
476
        for reservation in reservations:
477
477
            if reservation.end <= begin or reservation.begin >= end:
478
478
                continue  # Can be trivially skipped, no overlap here
479
479
            elif seats is None or reservation.seats is None:
480
480
                return False  # The whole room cannot be reserved -> False
481
481
            elif seats + reservation.seats > self.seats:
482
482
                    return False  # Total amount of seats exceeds the available amount -> False
483
483
        return True  # No overlappings found -> True
484
484
485
485
    def __str__(self):
486
486
        return self.name
487
487
488
488
class RoomReservation(models.Model):
489
489
    """ Rooms are to be reserved from time to time. They can be reserved
490
490
    by externals, for something else, and whatnot. That is stored in this table.
491
491
    """
492
492
    room = models.ForeignKey(
493
493
        "Room",
494
494
        on_delete=models.CASCADE,
495
495
        null=False,
496
496
        #editable=False,
497
497
        db_index=True,
498
498
        limit_choices_to={"reservable": True},
499
499
        help_text=_("The room that is being reserved at this point."),
500
500
        )
501
501
    reservator = models.ForeignKey(
502
502
        "User",
503
503
        on_delete=models.CASCADE,
504
504
        null=False,
505
505
        #editable=False,
506
506
        help_text=_("The person that made the reservation (and thus responsible)."),
507
507
        )
508
508
    timestamp = models.DateTimeField(auto_now_add=True)
509
509
    start_time = models.DateTimeField(
510
510
        null=False,
511
511
        help_text=_("The time that this reservation starts."),
512
512
        )
513
513
    end_time = models.DateTimeField(
514
514
        null=False,
515
515
        help_text=_("The time that this reservation ends."),
516
516
        )
517
517
    seats = models.PositiveSmallIntegerField(
518
518
        null=True,
519
519
        help_text=_("Indicates how many seats are required. If this is left null, "
520
520
                    "it is assumed the entire room has to be reserved."),
521
521
        )
522
522
    reason = models.CharField(
523
523
        max_length=64,
524
524
        blank=True,
525
525
        help_text=_("The reason for this reservation, if useful."),
526
526
        )
527
527
    note = models.TextField(
528
528
        blank=True,
529
529
        help_text=_("If some additional info is required for this reservation, "
530
530
                    "state it here."),
531
531
        )
532
532
533
533
    def __str__(self):
534
534
        start = self.start_time.strftime("%H:%M")
535
535
        end = self.end_time.strftime("%H:%M")
536
536
        return str(self.room) +" | "+ start +"-"+ end
537
537
538
538
class Degree(models.Model):
539
539
    """ Contains all degrees that were achieved at this university.
540
540
    There are no foreign keys in this field. This allows system
541
541
    administrators to safely remove accounts from alumni, without
542
542
    the risk of breaking referential integrity or accidentally removing
543
543
    degrees.
544
544
    While keeping some fields editable that look like they shouldn't be
545
545
    (e.g. first_name), this makes it possible for alumni to have a name change
546
546
    later in their life, and still being able to get a copy of their degree. """
547
547
    """ Reason for an ID field for every degree:
548
548
    This system allows for employers to verify that a certain applicant has indeed,
549
549
    achieved the degrees (s)he proclaims to have. Because of privacy concerns,
550
550
    a university cannot disclose information about alumni.
551
551
    That's where the degree ID comes in. This ID can be printed on all future
552
552
    degrees. The employer can then visit the university's website, and simply
553
553
    enter the ID. The website will then simply print what study is attached to
554
554
    this degree, but not disclose names or anything identifiable. This strikes
555
555
    thé perfect balance between (easy and digital) degree verification for employers, and maintaining
556
556
    alumni privacy to the highest extent possible. """
557
557
    id = models.UUIDField(primary_key=True, default=uuid.uuid4, editable=False)
558
558
    first_name = models.CharField(
559
559
        max_length=64,
560
560
        blank=False,
561
561
        )
562
562
    last_name = models.CharField(
563
563
        max_length=64,
564
564
        blank=False,
565
565
        )
566
566
    additional_names = models.CharField(
567
567
        max_length=64,
568
568
        blank=True,
569
569
        )
570
570
    DOB = models.DateField(null=False)#editable=False, null=False)  # This can't be changed, of course
571
571
    POB = models.CharField(
572
572
        max_length=64,
573
573
        blank=False,
574
574
        #editable=False,
575
575
        )
576
576
    # The study also has to be a charfield, because if a study is removed,
577
577
    # The information will be lost.
578
578
    study = models.CharField(
579
579
        max_length=64,
580
580
        blank=False,
581
581
        #editable=False,
582
582
        )
583
583
    achieved = models.DateField(null=False)#editable=False, null=False)
584
584
    user = models.ForeignKey(
585
585
        "User",
586
586
        on_delete=models.SET_NULL,
587
587
        null=True,
588
588
        help_text=_("The person that achieved this degree, if (s)he still has "
589
589
                    "an account at this university. If the account is deleted "
590
590
                    "at a later date, this field will be set to NULL, but the "
591
591
                    "other fields will be retained."),
592
592
        )
593
593
594
594
    def __str__(self):
595
595
        return self.first_name +" "+ self.last_name +" | "+ self.study
596
596
597
597
598
598
# Classes regarding roster items
599
599
600
600
def validate_event_time(time):
601
601
    """Checks if the time is a quarter of an hour (0, 15, 30, or 45)."""
602
602
    if time.minute not in [0, 15, 30, 45]:
603
603
        raise ValidationError(
604
604
            _('%(time)s is not in the quarter of an hour.'),
605
605
            params={'time': time},
606
606
        )
607
607
608
608
class Event(models.Model):
609
609
    """An event that will show up in the roster of accounts that need to be
610
610
    aware of this event. This can be a multitude of things, like colleges
611
611
    for certain courses, meetings like blood donations, and so on. There are
612
612
    specialized classes for certain types of events that take place."""
613
613
    begin_time = models.DateTimeField(
614
614
        null=False,
615
615
        help_text=_("The begin date and time that this event takes place. "
616
616
                    "This value must be a quarter of an hour (0, 15, 30, 45), "
617
617
                    "and take place <em>before</em> this event's end time."),
618
618
        verbose_name=_("begin time"),
619
619
        validators=[validate_event_time],
620
620
        )
621
621
    end_time = models.DateTimeField(
622
622
        null=False,
623
623
        help_text=_("The end date and time that this event takes place. "
624
624
                    "This value must be a quarter of an hour (0, 15, 30, 45), "
625
625
                    "and take place <em>after</em> this event's begin time."),
626
626
        verbose_name=_("end time"),
627
627
        validators=[validate_event_time],
628
628
        )
629
629
    note = models.TextField(
630
630
        blank=True,
631
631
        help_text=_("Optional. If necessary, this field allows for additional "
632
632
                    "information that can be shown to the people for whom this "
633
633
                    "event is."),
634
634
        )
635
635
    created = models.DateTimeField(
636
636
        auto_now_add=True,
637
637
        )
638
638
    last_update = models.DateTimeField(
639
639
        auto_now=True,
640
640
        )
641
641
642
642
class CourseEvent(Event):
643
643
    """An event related to a particular course. This includes a location,
644
644
    a group (if applicable), and other data."""
645
645
    course = models.ForeignKey(
646
646
        "courses.CourseProgramme",
647
647
        on_delete=models.CASCADE,
648
648
        null=False,
649
649
        )
650
650
    docent = models.ForeignKey(
651
651
        "User",
652
652
        on_delete=models.PROTECT,
653
653
        null=False,
654
654
        limit_choices_to={'is_staff': True},
655
655
        help_text=_("The person who will be the main overseer of this event."),
656
656
        )
657
657
    room = models.ForeignKey(
658
658
        "Room",
659
659
        on_delete=models.PROTECT,
660
660
        null=False,
661
661
        help_text=_("The room in which this event will be held."),
662
662
        )
+
663
        )
663
664
    subject = models.CharField(
664
665
        max_length=32,
665
666
        blank=False,
666
667
        help_text=_("The subject of this event. Examples are 'Hoorcollege', "
667
668
                    "'Zelfstudie', ..."),
668
669
        )
669
670
    group = models.ForeignKey(
670
671
        "courses.Group",
671
-
        on_delete = models.CASCADE,
+
672
        on_delete = models.CASCADE,
672
673
        null=True,
673
674
        help_text=_("Some courses have multiple groups. If that's the case, "
+
675
        help_text=_("Some courses have multiple groups. If that's the case, "
674
676
                    "and this event is only for a specific group, then that "
675
677
                    "group must be referenced here."),
676
678
        )
677
679
678
680
class UniversityEvent(Event):
679
681
    """University wide events. These include events like blood donations for the
680
682
    Red Cross, for example."""
681
683
    pass
682
684
683
685
class StudyEvent(Event):
684
686
    """An event that is linked to a particular study, like lectures from guest
685
687
    speakers about a certain subject."""
686
688
    pass
687
689
688
690
class ExamCommissionDecision(models.Model):
689
691
    """The Exam commission can make certain decisions regarding individual
690
692
    students. Every decision on its own is stored in this table, and is linked
691
693
    to the recipient's account."""
692
694
    user = models.ForeignKey(
693
695
        User,
694
696
        on_delete=models.CASCADE,
695
697
        null=False,
696
698
        help_text=_("The recipient of this decision."),
697
699
        )
698
700
    date = models.DateField(auto_now_add=True)
699
701
    text = models.TextField(
700
702
        blank=False,
701
703
        help_text=_("The text describing the decision. Org syntax available.")
702
704
        )
703
705
    def __str__(self):
704
706
        return str(self.user) + " | " + str(self.date)
705
707
706
708
    class Meta:
707
709
        verbose_name = _("Decision of the exam commission")
708
710
        verbose_name_plural = _("Decisions of the exam commission")
709
711
710
712
class EducationDepartmentMessages(models.Model):
711
713
    """The department of education can issue messages that are to be shown to
712
714
    all students. Their contents are stored here."""
713
715
    date = models.DateField(auto_now_add=True)
714
716
    title = models.CharField(
715
717
        max_length=64,
716
718
        blank=False,
717
719
        help_text=_("A short, well-describing title for this message."),
718
720
        )
719
721
    text = models.TextField(
720
722
        blank=False,
721
723
        help_text=_("The message text. Org syntax available.")
722
724
        )
723
725
    def __str__(self):
724
726
        return str(self.date) + " | " + str(self.title)
725
727
726
728
    class Meta:
727
729
        verbose_name = _("Decision of the exam commission")
728
-
        verbose_name_plural = _("Decisions of the exam commission")
729
-
+
730
        verbose_name_plural = _("Messages of the education department")
+
731

administration/roster.py

106 additions and 0 deletions.

View changes Hide changes
+
1
building of the roster. """
+
2
from django.shortcuts import render
+
3
from collections import OrderedDict
+
4
import datetime
+
5
from django.urls import reverse # Why?
+
6
from django.utils.translation import gettext as _
+
7
from .models import *
+
8
import administration
+
9
+
10
def same_day(datetime_a, datetime_b):
+
11
    """True if both a and b are on the same day, false otherwise."""
+
12
    return (
+
13
        datetime_a.day == datetime_b.day and
+
14
        datetime_a.month == datetime_b.month and
+
15
        datetime_a.year == datetime_b.year)
+
16
+
17
+
18
def make_quarter_buckets(events):
+
19
    """Returns a dict with all days that the given events take place.
+
20
    Every quarter between the first and last event will get a bucket in the dict,
+
21
    with the keys in format "dd-mm-yyyy". Also quarters without events will be
+
22
    included, but will simply have empty buckets."""
+
23
    quarters = OrderedDict()  # This is the first time I ever use an OrderedDict and I intend it to be my last one as well.
+
24
    current_quarter = datetime.datetime.now().replace(hour=8, minute=0)
+
25
    while current_quarter.hour != 20 or current_quarter.minute != 00:
+
26
        quarters[current_quarter.strftime("%H:%M")] = list()
+
27
        current_quarter += datetime.timedelta(minutes=15)
+
28
    for event in events:
+
29
        quarters[event.begin_time.strftime("%H:%M")].append(event)
+
30
+
31
    return quarters  # Yay! ^.^
+
32
+
33
+
34
def create_roster_rows(events):
+
35
    """Creates the rows for use in the roster table.
+
36
    None of the times in the given events may overlap, and all must start and
+
37
    end at a quarter of the hour (so :00, :15, :30, or :45). If you think you're above this,
+
38
    I'll raise you a ValueError, kind sir.
+
39
    Events must be of administration.models.Event type."""
+
40
    for event in events:
+
41
        for other_event in events:
+
42
            if other_event is not event and (
+
43
                    (event.begin_time >= other_event.begin_time and event.begin_time < other_event.end_time)
+
44
                    or (event.end_time > other_event.begin_time and event.end_time <= other_event.end_time)
+
45
                    ):
+
46
                raise ValueError("One of the events overlaps with another event.")
+
47
        if event.begin_time.minute not in [0, 15, 30, 45] or event.end_time.minute not in [0, 15, 30, 45]:
+
48
            raise ValueError("One of the events did not begin or end on a quarter.")
+
49
+
50
    # XXX: I assume here the given queryset is sorted:
+
51
    first_day = events[0].begin_time
+
52
    last_day = events.last().begin_time
+
53
    # All events validated
+
54
    quarter_buckets = make_quarter_buckets(events)
+
55
    skip_row = dict()
+
56
    table_code = list()
+
57
+
58
    for quarter_bucket in quarter_buckets.keys():
+
59
        # Preparing time column
+
60
        quarter_line = "<tr><td style='font-size: xx-small;'>"
+
61
        if quarter_bucket[3:] == "00":
+
62
            quarter_line += quarter_bucket
+
63
        quarter_line += "</td>"
+
64
+
65
        current_day = first_day
+
66
        while not same_day(current_day, last_day+datetime.timedelta(days=1)):
+
67
            for event in quarter_buckets[quarter_bucket]:
+
68
                if same_day(current_day, event.begin_time):
+
69
                    quarters = (event.end_time - event.begin_time).seconds // 60 // 15
+
70
                    skip_row[quarter_bucket] = quarters
+
71
                    event_line = "<td "
+
72
                    if isinstance(event, CourseEvent) and (datetime.datetime.now(datetime.timezone.utc) - event.last_update).days > 5:
+
73
                        event_line += "style='backgrond-color: #"+event.course.color+"; color: white;' "
+
74
                    elif (datetime.datetime.now(datetime.timezone.utc) - event.last_update).days <= 5:
+
75
                        event_line += "style='backgrond-color: yellow; color: red;' "
+
76
                    if event.note:
+
77
                        event_line +="title='" +event.note+ "' "
+
78
+
79
                    # FIXME: From here, I just assume the events are all CourseEvents, because
+
80
                    # that is the most important one to implement for the prototype.
+
81
                    if quarters > 1:
+
82
                        event_line += "rowspan='"+str(quarters)+"' "
+
83
+
84
                    event_line +=">"
+
85
                    event_line += str(
+
86
                        #"<a href=" + reverse("courses-course-index", args="")+"> "  # FIXME so that this links to the course's index page
+
87
                        "<a href=""> "
+
88
                        + str(event.course)
+
89
                        + "<br />"
+
90
                        + str(event.docent)
+
91
                        + "<br />"
+
92
                        + str(event.room) + " (" + str(event.subject) + ")</a></td>")
+
93
                    quarter_line += event_line
+
94
                else:
+
95
                    if quarter_bucket in skip_row:
+
96
                        skip_row[quarter_bucket] -= 1
+
97
                        if skip_row[quarter_bucket] == 0:
+
98
                            del skip_row[quarter_bucket]
+
99
                            quarter_line += "<td></td>"
+
100
                    else:
+
101
                        quarter_line += "<td></td>"
+
102
            current_day += datetime.timedelta(days=1)
+
103
        quarter_line += "</tr>"
+
104
        table_code.append(quarter_line)
+
105
    return table_code
+
106

administration/templates/administration/roster.djhtml

43 additions and 43 deletions.

View changes Hide changes
1
1
{% cycle "hour" "first quarter" "half" "last quarter" as hour silent %}
2
2
{# "silent" blocks the cycle operator from printing the cycler, and in subsequent calls #}
3
3
{% load i18n %}
4
4
5
5
{% block title %}
6
6
    {% trans "Roster | ◀ Joeni /▶" %}
7
-
{% endblock %}
+
7
{% endblock %}
8
8
9
9
{% block main %}
10
10
    {% include "administration/nav.djhtml" %}
11
11
    <h1>{% trans "Personal timetable" %}</h1>
12
12
    <h2>{% trans "Explanation" %}</h2>
13
13
    <p>
14
14
        {% trans "Personal roster from" %} {{ begin|date }} {% trans "to" %} {{ end|date }}
15
15
    </p>
16
16
    <p>
17
17
        {% blocktrans %}
18
18
            Some fields may have additional information that might be of interest
19
19
            to you. This information is shown in different ways with colour codes.
20
20
        {% endblocktrans %}
21
21
    </p>
22
22
    <p>
23
-
        {% blocktrans %}
24
-
            Most fields have a single colour, determined by the course. This makes
25
-
            it easier to differentiate on a glance what hours are for what courses.
26
-
        {% endblocktrans %}
27
-
    </p>
28
-
    <p>
29
-
        {% blocktrans %}
30
-
            Some fields have a
31
-
            <span style="background-color:yellow; color: red; border: medium dotted red;">
32
-
                bright yellow background, with red text and a red dotted border.</span>
33
-
            This indicates this event had one or more of its properties changed
34
-
            in the last five days. This can be the room, the hours, the subject, ...
35
-
            You're encouraged to take note of that.
36
-
        {% endblocktrans %}
37
-
    </p>
38
-
    <p>
39
-
        {% blocktrans %}
40
-
            Some fields are <span style="border: medium dashed black; background-color: white; color: black;">
41
-
            white with a dashed black border.</span> This indicates this event
42
-
            is new, and was added in the last five days.
43
-
        {% endblocktrans %}
44
-
    </p>
45
-
    <p>
46
-
        {% blocktrans %}
47
-
            Fields that flash <span style="color:orange; border: medium solid orange;">
48
-
            orange with the same coloured border</span> have a note attached to
49
-
            them by the docent/speaker. Hover over the event to display the note.
50
-
        {% endblocktrans %}
51
-
    </p>
52
-
53
23
    <h2>{% trans "Main hour roster" %}
54
-
    <table>
+
24
        <dt><span class="event-update">
+
25
            {% trans "Recent event update" %}
+
26
        </span></dt>
+
27
        <dd>
+
28
            {% blocktrans %}
+
29
                This event had one or more of its properties changed
+
30
                in the last five days. This can be the room, the hours, the subject, ...
+
31
                You're encouraged to take note of that.
+
32
            {% endblocktrans %}
+
33
        </dd>
+
34
        <dt><span class="event-new">
+
35
            {% trans "New event" %}
+
36
        </span></dt>
+
37
        <dd>
+
38
            {% blocktrans %}
+
39
                This is a new event, added in the last five days.
+
40
            {% endblocktrans %}
+
41
        </dd>
+
42
        <dt><span class="event-note">
+
43
            {% trans "Notification available" %}
+
44
        </span></dt>
+
45
        <dd>
+
46
            {% blocktrans %}
+
47
                This event has a note attached to it by the docent. Hover over
+
48
                the event to display the note.
+
49
            {% endblocktrans %}
+
50
        </dd>
+
51
    </dl>
+
52
+
53
    <h2>{% trans "Main hour roster" %}</h2>
+
54
    <style>
+
55
        table tr td {
+
56
        border: medium solid red;
+
57
        }
+
58
    </style>
+
59
    <table>
55
60
        <th>
56
61
            <td></td> {# Empty row for hours #}
57
-
            {% for day in days %}
+
62
            {% for day in days %}
58
63
                <td>{{ day|date:"l (d/m)" }}</td>
59
64
            {% endfor %}
60
65
        </th>
61
66
        {% for time, events in time_blocks %}
62
-
            <tr>
63
-
                {% if hour == "hour" %}
+
67
            {{ element|safe }}
+
68
            <!--<tr>
+
69
                {% if hour == "hour" %}
64
70
                    <td>{{ time }}</td>
65
71
                {% else %}
66
72
                    <td></td>
67
73
                {% endif %}
68
74
                {% cycle hour %}
69
75
                <td>{{ time }}</td>
70
76
                <!--<td rowspan="5">AI</td>
71
-
                <td>Dinsdag</td>
72
-
                <td>Dinsdag</td>
73
-
                <td>Woensdag</td>
74
-
                <td>Dondeddag</td>
75
-
                <td>Vdijdag</td>
76
-
                <td>Zaterdag</td>-->
77
-
            </tr>
78
-
        {% endfor %}
+
77
            </tr>-->
+
78
        {% endfor %}
79
79
    </table>
80
80
{% endblock main %}
81
81

administration/views.py

14 additions and 103 deletions.

View changes Hide changes
1
1
from collections import OrderedDict
2
2
from django.http import HttpResponseRedirect
3
3
import datetime
4
4
from django.urls import reverse # Why?
5
5
from django.utils.translation import gettext as _
6
6
from .models import *
7
7
from .forms import UserDataForm
8
8
import administration
+
9
import administration
9
10
from django.contrib.auth.decorators import login_required
10
11
from django.contrib.auth import authenticate
11
12
12
13
13
-
def make_day_buckets(events):
14
-
    """Returns a dict with all days that the given events take place.
15
-
    Every day between the first and last event will get a bucket in the dict,
16
-
    with the keys in format "dd-mm-yyyy". Also days without events will be
17
-
    included, but will simply have empty buckets."""
18
-
    first_event = events[0]
19
-
    last_event = events[0]
20
-
    days = OrderedDict()  # This is the first time I ever use an OrderedDict and I intend it to be my last one as well.
21
-
    for event in events:
22
-
        if event < first_event:
23
-
            first_event = event
24
-
        if event > last_event:
25
-
            last_event = event
26
-
    days_count = (last_event - first_event).days
27
-
    event_counter = first_event
28
-
    for i in range(days_count):
29
-
        days[event_counter.strftime("%d-%m-%Y")] = list()
30
-
        event_counter += datetime.timedelta(days=1)
31
-
    for event in events:
32
-
        days[event.strftime("%d-%m-%Y")].append(event)
33
-
34
-
    return days  # Yay! ^.^
35
-
36
-
37
-
def create_roster_rows(events):
38
-
    """Creates the rows for use in the roster table.
39
-
    None of the times in the given events may overlap, and all must start and
40
-
    end at a quarter of the hour (so :00, :15, :30, or :45). If you think you're above this,
41
-
    I'll raise you a ValueError, kind sir.
42
-
    Events must be of administration.models.Event type."""
43
-
    for event in events:
44
-
        for other_event in events:
45
-
            if (
46
-
                    (event.begin_time > other_event.begin_time and event.begin_time > other_event.end_time)
47
-
                    or (event.end_time > other_event.begin_time and event.end_time > other_event.end_time)
48
-
                    ):
49
-
                raise ValueError("One of the events overlaps with another event.")
50
-
        if event.begin_time.minute not in [0, 15, 30, 45] or event.end_time.minute not in [0, 15, 30, 45]:
51
-
            raise ValueError("One of the events did not begin or end on a quarter.")
52
-
53
-
    # All events validated
54
-
    days = make_day_buckets(events)
55
-
56
-
    table_code = list()
57
-
    for hour in range(8, 20):
58
-
        for quarter in [0, 15, 30, 45]:
59
-
            quarter_line = "<tr><td>"
60
-
            if hour < 10:
61
-
                quarter_line += "0"
62
-
            quarter_line += str(hour) + ":"
63
-
            if quarter == 0:
64
-
                quarter_line += "0"
65
-
            quarter_line += str(quarter) + "</td>"
66
-
67
-
            for day in days:
68
-
                for event in day:
69
-
                    if event.begin_time.hour == hour and event.begin_time.minute == quarter:
70
-
                        quarters = (event.end_time - event.begin_time).minutes // 15
71
-
                        event_line = "<td "
72
-
                        if isinstance(event, administration.CourseEvent) and (datetime.datetime.today() - event.last_update).days > 5:
73
-
                            event_line += "style='backgrond-color: #"+event.course.color+"; color: white;' "
74
-
                        elif (datetime.datetime.today() - event.last_update).days <= 5:
75
-
                            event_line += "style='backgrond-color: yellow; color: red;' "
76
-
                        if event.note:
77
-
                            event_line +="title='" +event.note+ "' "
78
-
79
-
                        # FIXME: From here, I just assume the events are all CourseEvents, because
80
-
                        # that is the most important one to implement for the prototype.
81
-
                        if quarters > 1:
82
-
                            event_line += "rowspan='"+str(quarters)+"' "
83
-
84
-
                        event_line +=">"
85
-
                        event_line += str(
86
-
                            + "<a href=" + reverse("courses-course-index", args="")+"> "  # FIXME so that this links to the course's index page
87
-
                            + str(event.course)
88
-
                            + "<br />"
89
-
                            + str(event.docent)
90
-
                            + "<br />"
91
-
                            + str(event.room) + " (" + str(event.subject) + ")</a></td>")
92
-
                        quarter_line += event_line
93
-
                    else:
94
-
                        quarter_line += "<td></td>"
95
-
96
-
            quarter_line += "</tr>"
97
-
            code += quarter_line
98
-
    return code
99
-
100
-
101
-
102
-
103
-
104
-
105
-
106
-
@login_required
107
14
def roster(request, begin=None, end=None):
108
15
    """Collects and renders the data that has to be displayed in the roster.
109
16
110
17
    The begin and end date can be specified. Only roster points in that range
111
18
    will be included in the response. If no begin and end are specified, it will
112
19
    take the current week as begin and end point. If it's
113
20
    weekend, it will take next week."""
114
21
115
22
    # TODO Handle given begin and end
116
23
    context = dict()
117
24
    template = "administration/roster.djhtml"
118
25
119
26
    if begin is None or end is None:
120
27
        today = datetime.date.today()
121
28
        if today.isoweekday() in {6,7}:  # Weekend
122
29
            begin = today + datetime.timedelta(days=8-today.isoweekday())
123
30
            end = today + datetime.timedelta(days=13-today.isoweekday())
124
31
        else:  # Same week
125
32
            begin = today - datetime.timedelta(days=today.weekday())
126
33
            end = today + datetime.timedelta(days=5-today.isoweekday())
127
34
    context['begin'] = begin
128
35
    context['end'] = end
129
36
130
37
    days = [begin]
131
38
    while (end-days[-1]).days != 0:
132
39
        # Human translation: Keep adding days until the last day in the array of
133
40
        # days is the same day as the last day the user wants to see the roster for.
134
41
        days.append(days[-1] + datetime.timedelta(days=1))
135
42
    context['days'] = days
136
43
137
44
    # Collecting events
138
45
    course_events = CourseEvent.objects.filter(begin_time__gte=begin).filter(end_time__lte=end)
139
-
    #university_events = UniversityEvent.objects.filter(begin_time__gte=begin).filter(end_time__lte=end)
+
46
    #university_events = UniversityEvent.objects.filter(begin_time__gte=begin).filter(end_time__lte=end)
140
47
    #study_events = StudyEvent.objects.filter(begin_time__gte=begin).filter(end_time__lte=end)
141
48
    #events = Event.objects.filter(begin_time__gte=begin).filter(end_time__lte=end)
142
49
+
50
143
51
    # Producing time blocks for display in the table
144
52
    time_blocks = []
145
-
    for i in range(8, 20):
+
53
    for i in range(8, 20):
146
54
        time_block = str(i)
147
-
        for j in range(0, 60, 15):
+
55
        for j in range(0, 60, 15):
148
56
            if j == 0:
149
57
                time_blocks.append([time_block + ":00",""])
150
-
                continue
151
-
            time_blocks.append([time_block + ":" + str(j), ""])
152
-
    context['time_blocks'] = time_blocks
153
-
    #context['
154
-
155
-
+
58
            else:
+
59
                times.append(time + ":" + str(j))
+
60
156
61
+
62
    for i in range(len(times)):
+
63
        time_blocks.append([times[i],table_code[i]])
+
64
157
65
    return render(request, template, context)
+
66
    context['time_blocks'] = table_code
+
67
    #print(time_blocks)
+
68
    return render(request, template, context)
158
69
    # TODO Finish!
159
70
160
71
def index(request):
161
72
    template = "administration/index.djhtml"
162
73
    context = {}
163
74
    return render(request, template, context)
164
75
165
76
    pass
166
77
167
78
def pre_registration(request):
168
79
    user_data_form = UserDataForm()
169
80
    template = "administration/pre_registration.djhtml"
170
81
    context = dict()
171
82
172
83
    if request.method == 'POST':
173
84
        user_data_form = UserDataForm(request.POST)
174
85
        context['user_data_form'] = user_data_form
175
86
        if user_data_form.is_valid():
176
87
            user_data_form.save()
177
88
            context['messsage'] = _("Your registration has been completed. You will receive an e-mail shortly.")
178
89
        else:
179
90
            context['messsage'] = _("The data you supplied had errors. Please review your submission.")
180
91
    else:
181
92
        context['user_data_form'] = UserDataForm(instance = user_data)
182
93
183
94
    return render(request, template, context)
184
95
    pass
185
96
186
97
@login_required
187
98
def settings(request):
188
99
    user_data = UserData.objects.get(user=request.user)
189
100
    user_data_form = UserDataForm(instance = user_data)
190
101
    template = "administration/settings.djhtml"
191
102
    context = dict()
192
103
193
104
    if request.method == 'POST':
194
105
        user_data_form = UserDataForm(request.POST, instance = user_data)
195
106
        context['user_data_form'] = user_data_form
196
107
        if user_data_form.is_valid():
197
108
            user_data_form.save()
198
109
            context['messsage'] = _("Your settings were successfully updated.")
199
110
        else:
200
111
            context['messsage'] = _("The data you supplied had errors. Please review your submission.")
201
112
    else:
202
113
        context['user_data_form'] = UserDataForm(instance = user_data)
203
114
204
115
    return render(request, template, context)
205
116
206
117
@login_required
207
118
def bulletin_board(request):
208
119
    context = dict()
209
120
    context['exam_commission_decisions'] = ExamCommissionDecision.objects.filter(user=request.user)
210
121
    context['education_department_messages'] = ExamCommissionDecision.objects.filter(user=request.user)
211
122
    template = "administration/bulletin_board.djhtml"
212
123
    return render(request, template, context)
213
124
214
125
def jobs(request):
215
126
    context = dict()
216
127
    template = "administration/jobs.djhtml"
217
128
    #@context['decisions'] = ExamCommissionDecision.objects.filter(user=request.user)
218
129
    return render(request, template, context)
219
130
220
131
221
132
def curriculum(request):
222
133
    return render(request, template, context)
223
134
224
135
def result(request):
225
136
    return render(request, template, context)
226
137
227
138
@login_required
228
139
def results(request):
229
140
    results = CourseResult.objects.filter(student=request.user)
230
141
    template = "administration/results.djhtml"
231
142
    # TODO
232
143
    return render(request, template, context)
233
144
234
145
def forms(request):
235
146
    return render(request, template, context)
236
147
237
148
def rooms(request):
238
149
    template = "administration/rooms.djhtml"
239
150
    return render(request, template, context)
240
151
241
152
def room_reservate(request):
242
153
    return render(request, template, context)
243
154
244
155
def login(request):
245
156
    context = dict()
246
157
    if request.method == "POST":
247
158
        name = request.POST['name']
248
159
        passphrase = request.POST['pass']
249
160
        user = authenticate(username=name, password=passphrase)
250
161
        if user is not None: # The user was successfully authenticated
251
162
            print("YA")
252
163
            return HttpResponseRedirect(request.POST['next'])
253
164
        else: # User credentials were wrong
254
165
            context['next'] = request.POST['next']
255
166
            context['message'] = _("The given credentials were not correct.")
256
167
    else:
257
168
        context['next'] = request.GET.get('next', None)
258
169
        if context['next'] is None:
259
170
            context['next'] = reverse('administration-index')
260
171
261
172
    template = 'administration/login.djhtml'
262
173
263
174
    return render(request, template, context)
264
175

courses/admin.py

1 addition and 0 deletions.

View changes Hide changes
1
1
from .models import *
2
2
3
3
admin.site.register(Course)
4
4
admin.site.register(Prerequisite)
5
5
admin.site.register(CourseProgramme)
6
6
admin.site.register(Study)
7
7
admin.site.register(StudyProgramme)
8
8
admin.site.register(Assignment)
9
9
admin.site.register(Announcement)
10
10
admin.site.register(Upload)
11
11
admin.site.register(StudyGroup)
12
12
+
13

courses/migrations/0005_auto_20180204_1349.py

51 additions and 0 deletions.

View changes Hide changes
+
1
+
2
from django.conf import settings
+
3
from django.db import migrations, models
+
4
import django.db.models.deletion
+
5
+
6
+
7
class Migration(migrations.Migration):
+
8
+
9
    dependencies = [
+
10
        migrations.swappable_dependency(settings.AUTH_USER_MODEL),
+
11
        ('courses', '0004_auto_20180124_0049'),
+
12
    ]
+
13
+
14
    operations = [
+
15
        migrations.AddField(
+
16
            model_name='course',
+
17
            name='co_owners',
+
18
            field=models.ManyToManyField(blank=True, help_text='If applicable: The co-owners of this course.', limit_choices_to={'is_staff': True}, related_name='co_owners', to=settings.AUTH_USER_MODEL),
+
19
        ),
+
20
        migrations.AlterField(
+
21
            model_name='course',
+
22
            name='color',
+
23
            field=models.CharField(default='E73B2B', help_text="The color for this course. Must be an hexadecimal code. Some standard colors if you don't know what to pick: <ul><li>0076BE: Faculty of Sciences / Blue</li><li>C0D633: Faculty of Transportation Sciences / Green</li><li>F4802D: Faculty of Architecture and Arts / Orange</li><li>00ACEE: Faculty of Business Economics / Turquoise</li><li>9C3591: Faculty of Medicine and Life Sciences / Purple</li><li>5BC4BA: Faculty of Engineering Technology / Light blue</li><li>E41F3A: Faculty of Law / Red</li></ul>", max_length=6),
+
24
        ),
+
25
        migrations.AlterField(
+
26
            model_name='course',
+
27
            name='educating_team',
+
28
            field=models.ManyToManyField(blank=True, help_text='The remaining team members of this course.', limit_choices_to={'is_staff': True}, related_name='educating_team', to=settings.AUTH_USER_MODEL),
+
29
        ),
+
30
        migrations.AlterField(
+
31
            model_name='prerequisite',
+
32
            name='ECTS_for_required_study',
+
33
            field=models.PositiveSmallIntegerField(blank=True, help_text='The amount of obtained ECTS points for the required course, if any.', null=True),
+
34
        ),
+
35
        migrations.AlterField(
+
36
            model_name='prerequisite',
+
37
            name='in_curriculum',
+
38
            field=models.ManyToManyField(blank=True, help_text='All courses that have to be in the curriculum to follow this. If a credit was achieved, that course can be omitted.', related_name='in_curriculum', to='courses.Course'),
+
39
        ),
+
40
        migrations.AlterField(
+
41
            model_name='prerequisite',
+
42
            name='required_study',
+
43
            field=models.ForeignKey(blank=True, help_text='If one must have a certain amount of obtained ECTS points for a particular course, state that course here.', null=True, on_delete=django.db.models.deletion.CASCADE, to='courses.Study'),
+
44
        ),
+
45
        migrations.AlterField(
+
46
            model_name='prerequisite',
+
47
            name='sequentialities',
+
48
            field=models.ManyToManyField(blank=True, help_text="All courses for which a credit must've been received in order to follow the course.", related_name='sequentialities', to='courses.Course'),
+
49
        ),
+
50
    ]
+
51

courses/migrations/0006_auto_20180204_1349.py

18 additions and 0 deletions.

View changes Hide changes
+
1
+
2
from django.db import migrations
+
3
+
4
+
5
class Migration(migrations.Migration):
+
6
+
7
    dependencies = [
+
8
        ('administration', '0012_auto_20180204_1349'),
+
9
        ('courses', '0005_auto_20180204_1349'),
+
10
    ]
+
11
+
12
    operations = [
+
13
        migrations.RenameModel(
+
14
            old_name='Group',
+
15
            new_name='CourseGroup',
+
16
        ),
+
17
    ]
+
18

courses/models.py

1 addition and 1 deletion.

View changes Hide changes
1
1
from joeni import constants
2
2
from django.utils.translation import ugettext_lazy as _
3
3
4
4
def validate_hex_color(value):
5
5
    pass  # TODO
6
6
7
7
class Course(models.Model):
8
8
    """ Represents a course that is taught at the university. """
9
9
    number = models.PositiveSmallIntegerField(
10
10
        primary_key=True,
11
11
        blank=False,
12
12
        help_text=_("The number associated with this course. A leading '0' will be added if the number is smaller than 1000."),
13
13
        )
14
14
    name = models.CharField(
15
15
        max_length=64,
16
16
        blank=False,
17
17
        help_text=_("The name of this course, in the language that it is taught. Translations are for the appropriate template."),
18
18
        )
19
19
    color = models.CharField(
20
20
        max_length=6,
21
21
        blank=False,
22
22
        default=constants.COLORS['UHasselt default'],
23
23
        help_text=_("The color for this course. Must be an hexadecimal code. "
24
24
                    "Some standard colors if you don't know what to pick: "
25
25
                    "<ul><li>0076BE: Faculty of Sciences / Blue</li>"
26
26
                    "<li>C0D633: Faculty of Transportation Sciences / Green</li>"
27
27
                    "<li>F4802D: Faculty of Architecture and Arts / Orange</li>"
28
28
                    "<li>00ACEE: Faculty of Business Economics / Turquoise</li>"
29
29
                    "<li>9C3591: Faculty of Medicine and Life Sciences / Purple</li>"
30
30
                    "<li>5BC4BA: Faculty of Engineering Technology / Light blue</li>"
31
31
                    "<li>E41F3A: Faculty of Law / Red</li></ul>"),
32
32
        #validators=['validate_hex_color'], # TODO
33
33
        )
34
34
    slug_name = models.SlugField(
35
35
        blank=False,
36
36
        allow_unicode=True,
37
37
        unique=True,
38
38
        help_text=_("A so-called 'slug name' for this course."),
39
39
        )
40
40
    # TODO: Add a potential thingy magicky to auto fill the slug name on the course name
41
41
    contact_person = models.ForeignKey(
42
42
        "administration.User",
43
43
        on_delete=models.PROTECT,  # A course must have a contact person
44
44
        limit_choices_to={'is_staff': True},
45
45
        null=False,
46
46
        help_text=_("The person to contact regarding this course."),
47
47
        related_name="contact_person",
48
48
        )
49
49
    coordinator = models.ForeignKey(
50
50
        "administration.User",
51
51
        on_delete=models.PROTECT,  # A course must have a coordinator
52
52
        limit_choices_to={'is_staff': True},
53
53
        null=False,
54
54
        help_text=_("The person whom's the coordinator of this course."),
55
55
        related_name="coordinator",
56
56
        )
57
57
    co_owners = models.ManyToManyField(
58
58
        "administration.User",
59
59
        limit_choices_to={'is_staff': True},
60
60
        blank=True,  # Allows empty in form validation, and M->M implies null=True
61
61
        help_text=_("If applicable: The co-owners of this course."),
62
62
        related_name="co_owners",
63
63
        )
64
64
65
65
    educating_team = models.ManyToManyField(
66
66
        "administration.User",
67
67
        # No on_delete, since M->M cannot be required at database level
68
68
        limit_choices_to={'is_staff': True},
69
69
        blank=True,
70
70
        help_text=_("The remaining team members of this course."),
71
71
        related_name="educating_team",
72
72
        )
73
73
    language = models.CharField(
74
74
        max_length=64,
75
75
        choices = (
76
76
            ('NL', _("Dutch")),
77
77
            ('EN', _("English")),
78
78
            ('FR', _("French")),
79
79
            ),
80
80
        null=False,
81
81
        help_text=_("The language in which this course is given."),
82
82
        )
83
83
84
84
    def course_team(self):
85
85
        """ Returns a set of all Users that are part of the team of this course. """
86
86
        return set(
87
87
            self.contact_person,
88
88
            self.coordinator,
89
89
            self.educating_team,
90
90
            )
91
91
92
92
    def __str__(self):
93
93
        number = str(self.number)
94
94
        for i in [10,100,1000]:
95
95
            if self.number < i:
96
96
                number = "0" + number
97
97
        return "(" + number + ") " + self.name
98
98
99
99
100
100
class Prerequisite(models.Model):
101
101
    """ Represents a collection of prerequisites a student must have obtained
102
102
    before being allowed to partake in this course.
103
103
    It's possible that, if a student has obtained credits in a certain set of
104
104
    courses, a certain part of the prerequisites do not have to be obtained.
105
105
    Because of this, make a different record for each different set. In other
106
106
    words: If one set of prerequisites is obtained, and another one isn't, BUT
107
107
    they point to the same course, the student is allowed to partake. """
108
108
    course = models.ForeignKey(
109
109
        "Course",
110
110
        on_delete=models.CASCADE,
111
111
        null=False,
112
112
        help_text=_("The course that these prerequisites are for."),
113
113
        related_name="prerequisite_course",
114
114
        )
115
115
    name = models.CharField(
116
116
        max_length=64,
117
117
        blank=True,
118
118
        help_text=_("To specify a name for this set, if necessary."),
119
119
        )
120
120
    sequentialities = models.ManyToManyField(
121
121
        "Course",
122
122
        help_text=_("All courses for which a credit must've been received in order to follow the course."),
123
123
        blank=True,
124
124
        related_name="sequentialities",
125
125
        )
126
126
    in_curriculum = models.ManyToManyField(
127
127
        "Course",
128
128
        help_text=_("All courses that have to be in the curriculum to follow this. If a credit was achieved, that course can be omitted."),
129
129
        blank=True,
130
130
        related_name="in_curriculum",
131
131
        )
132
132
    required_study = models.ForeignKey(
133
133
        "Study",
134
134
        on_delete=models.CASCADE,
135
135
        blank=True,
136
136
        null=True,
137
137
        help_text=_("If one must have a certain amount of obtained ECTS points for a particular course, state that course here."),
138
138
        )
139
139
    ECTS_for_required_study = models.PositiveSmallIntegerField(
140
140
        blank=True,
141
141
        null=True,
142
142
        help_text=_("The amount of obtained ECTS points for the required course, if any."),
143
143
        )
144
144
145
145
    def __str__(self):
146
146
        if self.name == "":
147
147
            return _("Prerequisites for %(course)s") % {'course': str(self.course)}
148
148
        else:
149
149
            return self.name + " | " + str(self.course)
150
150
151
151
152
152
class CourseProgramme(models.Model):
153
153
    """ It's possible that a course is taught in multiple degree programmes; For
154
154
    example: Calculus can easily be taught to physics and mathematics students
155
155
    alike. In this table, these relations are set up, and the related properties
156
156
    are defined as well. """
157
157
    study = models.ForeignKey(
158
158
        "Study",
159
159
        on_delete=models.CASCADE,
160
160
        null=False,
161
161
        help_text=_("The study in which the course is taught."),
162
162
        )
163
163
    course = models.ForeignKey(
164
164
        "Course",
165
165
        on_delete=models.CASCADE,
166
166
        null=False,
167
167
        help_text=_("The course that this programme is for."),
168
168
        )
169
169
    study_programme = models.ForeignKey(
170
170
        "StudyProgramme",
171
171
        on_delete=models.CASCADE,
172
172
        null=False,
173
173
        help_text=_("The study programme that this course belongs to."),
174
174
        )
175
175
    programme_type = models.CharField(
176
176
        max_length=1,
177
177
        blank=False,
178
178
        choices = (
179
179
            ('C', _("Compulsory")),
180
180
            ('O', _("Optional")),
181
181
            ),
182
182
        help_text=_("Type of this course for this study."),
183
183
        )
184
184
    study_hours = models.PositiveSmallIntegerField(
185
185
        blank=False,
186
186
        help_text=_("The required amount of hours to study this course."),
187
187
        )
188
188
    ECTS = models.PositiveSmallIntegerField(
189
189
        blank=False,
190
190
        help_text=_("The amount of ECTS points attached to this course."),
191
191
        )
192
192
    semester = models.PositiveSmallIntegerField(
193
193
        blank=False,
194
194
        choices = (
195
195
            (1, _("First semester")),
196
196
            (2, _("Second semester")),
197
197
            (3, _("Full year course")),
198
198
            (4, _("Taught in first quarter")),
199
199
            (5, _("Taught in second quarter")),
200
200
            (6, _("Taught in third quarter")),
201
201
            (7, _("Taught in fourth quarter")),
202
202
            ),
203
203
        help_text=_("The period in which this course is being taught in this study."),
204
204
        )
205
205
    year = models.PositiveSmallIntegerField(
206
206
        blank=False,
207
207
        help_text=_("The year in which this course is taught for this study."),
208
208
        )
209
209
    second_chance = models.BooleanField(
210
210
        default=True,
211
211
        help_text=_("Defines if a second chance exam is planned for this course."),
212
212
        )
213
213
    tolerable = models.BooleanField(
214
214
        default=True,
215
215
        help_text=_("Defines if a failed result can be tolerated."),
216
216
        )
217
217
    scoring = models.CharField(
218
218
        max_length=2,
219
219
        choices = (
220
220
            ('N', _("Numerical")),
221
221
            ('FP', _("Fail/Pass")),
222
222
            ),
223
223
        default='N',
224
224
        blank=False,
225
225
        help_text=_("How the obtained score for this course is given."),
226
226
        )
227
227
228
228
    def __str__(self):
229
229
        return str(self.study) + " - " + str(self.course)
230
230
231
231
class Study(models.Model):
232
232
    """ Defines a certain study that can be followed at the university.
233
233
    This also includes abridged study programmes, like transition programmes.
234
234
    Other information, such as descriptions, are kept in the template file
235
235
    of this study, which can be manually edited. Joeni searches for a file
236
236
    with the exact name as the study + ".html". So if the study is called
237
237
    "Bachelor of Informatics", it will search for "Bachelor of Informatics.html".
238
238
    """
239
239
    # Degree types
240
240
    BSc = _("Bachelor of Science")
241
241
    MSc = _("Master of Science")
242
242
    LLB = _("Bachelor of Laws")
243
243
    LLM = _("Master of Laws")
244
244
    BA  = _("Bachelor of Arts")
245
245
    MA  = _("Master of Arts")
246
246
    ir  = _("Engineer")
247
247
    ing = _("Technological Engineer")
248
248
    # Faculties
249
249
    FoMaLS = _("Faculty of Medicine and Life Sciences")
250
250
    FoS    = _("Faculty of Sciences")
251
251
    FoTS   = _("Faculty of Transportation Sciences")
252
252
    FoAaA  = _("Faculty of Architecture and Arts")