Skip to content

Commit c024f5d

Browse files
committed
Review page added
1 parent 220dd71 commit c024f5d

19 files changed

+454
-48
lines changed

app/settings.py

+3
Original file line numberDiff line numberDiff line change
@@ -47,12 +47,14 @@
4747
'django.contrib.sessions',
4848
'django.contrib.messages',
4949
'django.contrib.staticfiles',
50+
'django_tables2',
5051
'django_jwt',
5152
'django_jwt.server',
5253
'django_bootstrap5',
5354
'corsheaders',
5455
'user',
5556
'application',
57+
'review',
5658
]
5759

5860
MIDDLEWARE = [
@@ -79,6 +81,7 @@
7981
'django.template.context_processors.request',
8082
'django.contrib.auth.context_processors.auth',
8183
'django.contrib.messages.context_processors.messages',
84+
'django.template.context_processors.request',
8285
'app.template.app_variables',
8386
],
8487
'libraries': {

app/templates/components/bootstrap5_form.html

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@
1616
{% for title, zone in form.get_fields.items %}
1717
<fieldset class="bootstrap-fieldset">
1818
{% if not form_text_disabled %}
19-
<h3 class="heading-primary">{{ title }}</h3>
19+
{% if title %}<h3 class="heading-primary">{{ title }}</h3>{% endif %}
2020
{% if zone.description %}
2121
<p>{{ zone.description|safe }}</p>
2222
{% endif %}

app/templatetags/util.py

+5
Original file line numberDiff line numberDiff line change
@@ -6,3 +6,8 @@
66
@register.filter
77
def get_type(value):
88
return type(value).__name__
9+
10+
11+
@register.filter
12+
def get_item(dict_item, value):
13+
return dict_item.get(value, None)

app/urls.py

+1
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@
2424
path('', BaseView.as_view(), name='home'),
2525
path('auth/', include('user.urls')),
2626
path('application/', include('application.urls')),
27+
path('review/', include('review.urls')),
2728
]
2829

2930
# JWT fake login on DEBUG for development purposes

application/admin.py

+1
Original file line numberDiff line numberDiff line change
@@ -4,3 +4,4 @@
44

55
admin.site.register(models.Application)
66
admin.site.register(models.ApplicationTypeConfig)
7+
admin.site.register(models.ApplicationLog)

application/forms.py

+7-27
Original file line numberDiff line numberDiff line change
@@ -132,15 +132,6 @@ class HackerForm(ApplicationForm):
132132
'fields': [{'name': 'country', 'space': 6}, {'name': 'origin', 'space': 6}], }
133133
}
134134

135-
under_age = forms.TypedChoiceField(
136-
required=True,
137-
label=_('How old are you?'),
138-
initial=False,
139-
coerce=lambda x: x == 'True',
140-
choices=((False, _('18 or over')), (True, _('Between 14 (included) and 18'))),
141-
widget=forms.RadioSelect
142-
)
143-
144135
phone_number = forms.CharField(validators=[RegexValidator(regex=r'^\+?1?\d{9,15}$')], required=False,
145136
help_text=_("Phone number must be entered in the format: +#########'. "
146137
"Up to 15 digits allowed."),
@@ -211,6 +202,13 @@ class Meta(ApplicationForm.Meta):
211202
'university': {'url': static('data/universities.json')},
212203
'degree': {'url': static('data/degrees.json')},
213204
}
205+
icon_link = {
206+
'resume': 'bi bi-file-pdf-fill',
207+
'github': 'bi bi-github',
208+
'devpost': 'bi bi-collection-fill',
209+
'linkedin': 'bi bi-linkedin',
210+
'site': 'bi bi-globe',
211+
}
214212

215213

216214
class VolunteerForm(ApplicationForm):
@@ -261,15 +259,6 @@ class VolunteerForm(ApplicationForm):
261259
widget=forms.RadioSelect
262260
)
263261

264-
under_age = forms.TypedChoiceField(
265-
required=True,
266-
label=_('How old are you?'),
267-
initial=False,
268-
coerce=lambda x: x == 'True',
269-
choices=((False, _('18 or over')), (True, _('Between 14 (included) and 18'))),
270-
widget=forms.RadioSelect
271-
)
272-
273262
origin = forms.CharField(max_length=300, label=_('From which city?'))
274263

275264
country = forms.CharField(max_length=300, label=_('From which country are you joining us?'))
@@ -368,15 +357,6 @@ class MentorForm(ApplicationForm):
368357
'description': _('Tell us a bit about your experience and preferences in this type of event.')},
369358
}
370359

371-
under_age = forms.TypedChoiceField(
372-
required=True,
373-
label=_('How old are you?'),
374-
initial=False,
375-
coerce=lambda x: x == 'True',
376-
choices=((False, _('18 or over')), (True, _('Between 14 (included) and 18'))),
377-
widget=forms.RadioSelect
378-
)
379-
380360
university = forms.CharField(max_length=300, label=_('What university do you study at?'),
381361
help_text=_('Current or most recent school you attended.'))
382362

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
# Generated by Django 4.0.4 on 2022-08-10 10:24
2+
3+
from django.db import migrations, models
4+
import django.db.models.deletion
5+
import django.utils.timezone
6+
7+
8+
class Migration(migrations.Migration):
9+
10+
dependencies = [
11+
('application', '0003_alter_application_user'),
12+
]
13+
14+
operations = [
15+
migrations.AddField(
16+
model_name='applicationlog',
17+
name='date',
18+
field=models.DateTimeField(default=django.utils.timezone.now),
19+
),
20+
migrations.AlterField(
21+
model_name='applicationlog',
22+
name='application',
23+
field=models.ForeignKey(db_index=False, on_delete=django.db.models.deletion.CASCADE, related_name='logs', to='application.application'),
24+
),
25+
]

application/models.py

+3-4
Original file line numberDiff line numberDiff line change
@@ -11,8 +11,6 @@
1111
from django.utils.http import urlsafe_base64_encode, urlsafe_base64_decode
1212
from django.utils.translation import gettext_lazy as _
1313

14-
from user.models import User
15-
1614

1715
class FileField(dict):
1816
def __init__(self, data, url) -> None:
@@ -222,11 +220,12 @@ class Meta:
222220

223221
class ApplicationLog(models.Model):
224222
id = models.BigAutoField(primary_key=True)
225-
application = models.ForeignKey(Application, on_delete=models.CASCADE, db_index=False)
226-
user = models.ForeignKey(User, on_delete=models.RESTRICT, db_index=False)
223+
application = models.ForeignKey(Application, on_delete=models.CASCADE, db_index=False, related_name='logs')
224+
user = models.ForeignKey(settings.AUTH_USER_MODEL, on_delete=models.RESTRICT, db_index=False)
227225
name = models.CharField(max_length=20)
228226
comment = models.TextField(blank=True)
229227
data = models.TextField(blank=True)
228+
date = models.DateTimeField(default=timezone.now)
230229

231230
class NotFound:
232231
pass

application/templates/application_form.html

+1-1
Original file line numberDiff line numberDiff line change
@@ -30,7 +30,7 @@ <h3 class="mb-1">{% translate 'Status' %} {% if request.user.is_organizer %}{{ a
3030

3131
<div class="d-grid gap-4 d-flex justify-content-around">
3232
{% if edit %}
33-
<a href="{% url 'change_status_application' application_form.instance.get_uuid application_form.instance.STATUS_CANCELLED %}" onclick="return confirm('{% trans 'Are you sure you want to cancel this application?' %}')" class="btn btn-danger col-4">{% trans 'Cancel application' %}</a>
33+
<a href="{% url 'change_status_application' application_form.instance.get_uuid application_form.instance.STATUS_CANCELLED %}" onclick="return confirm('{% trans "Did you changed your mind? Are you sure you want to cancel this application?" %}')" class="btn btn-danger col-4">{% trans 'Cancel application process' %}</a>
3434
{% endif %}
3535
{% if not application_form.is_read_only %}
3636
<button type="submit" class="btn btn-primary col-4">{% if edit %}{% trans 'Save' %}{% else %}{% trans 'Apply' %}{% endif %}</button>

application/views.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -135,7 +135,7 @@ def get_application(self):
135135
return application
136136

137137
def application_can_edit(self, application, application_type):
138-
return self.request.user.is_organizer or application.can_edit() and application_type.active()
138+
return self.request.user.is_organizer or (application.can_edit() and application_type.active())
139139

140140
def get_context_data(self, **kwargs):
141141
context = super().get_context_data(**kwargs)
@@ -144,7 +144,7 @@ def get_context_data(self, **kwargs):
144144
ApplicationForm = self.get_form(application_type)
145145
application_form = ApplicationForm(instance=application)
146146
user_form = UserProfileForm(instance=application.user)
147-
if self.application_can_edit(application, application_type):
147+
if not self.application_can_edit(application, application_type):
148148
application_form.set_read_only()
149149
context.update({'edit': True, 'application_form': application_form, 'full_name': application.get_full_name(),
150150
'application_type': application_type, 'user_form': user_form})

review/forms.py

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
from django import forms
2+
from django.utils.translation import gettext_lazy as _
3+
4+
from app.mixins import BootstrapFormMixin
5+
from application.models import ApplicationLog
6+
7+
8+
class CommentForm(BootstrapFormMixin, forms.ModelForm):
9+
bootstrap_field_info = {'': {'fields': [{'name': 'name'}, {'name': 'application'},
10+
{'name': 'comment', 'space': 12}]}}
11+
12+
name = forms.CharField(initial='Comment', widget=forms.HiddenInput())
13+
14+
def __init__(self, *args, **kwargs):
15+
super().__init__(*args, **kwargs)
16+
if kwargs.get('instance', None) is not None:
17+
self.fields.get('comment').label = ''
18+
19+
class Meta:
20+
model = ApplicationLog
21+
fields = ('comment', 'application', 'name')
22+
labels = {
23+
'comment': _('Add new comment')
24+
}
25+
widgets = {
26+
'comment': forms.Textarea(attrs={'rows': 2}),
27+
'application': forms.HiddenInput()
28+
}

review/migrations/0001_initial.py

+32
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
# Generated by Django 4.0.4 on 2022-08-10 07:03
2+
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+
initial = True
11+
12+
dependencies = [
13+
('application', '0003_alter_application_user'),
14+
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
15+
]
16+
17+
operations = [
18+
migrations.CreateModel(
19+
name='Vote',
20+
fields=[
21+
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
22+
('tech', models.IntegerField(choices=[(1, '1'), (2, '2'), (3, '3'), (4, '4'), (5, '5')], null=True)),
23+
('personal', models.IntegerField(choices=[(1, '1'), (2, '2'), (3, '3'), (4, '4'), (5, '5')], null=True)),
24+
('calculated_vote', models.FloatField(null=True)),
25+
('application', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='application.application')),
26+
('user', models.ForeignKey(null=True, on_delete=django.db.models.deletion.SET_NULL, to=settings.AUTH_USER_MODEL)),
27+
],
28+
options={
29+
'unique_together': {('application', 'user')},
30+
},
31+
),
32+
]

review/models.py

+70-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,72 @@
1+
from django.conf import settings
2+
from django.contrib.auth.models import User
13
from django.db import models
4+
from django.db.models import Avg, F
25

3-
# Create your models here.
6+
7+
MAX_VOTES = getattr(settings, 'MAX_VOTES', 5)
8+
TECH_WEIGHT = 0.2
9+
PERSONAL_WEIGHT = 0.8
10+
11+
VOTES = [(i, str(i)) for i in range(1, MAX_VOTES + 1)]
12+
13+
14+
class Vote(models.Model):
15+
application = models.ForeignKey('application.Application', on_delete=models.CASCADE)
16+
user = models.ForeignKey(settings.AUTH_USER_MODEL, null=True, on_delete=models.SET_NULL)
17+
tech = models.IntegerField(choices=VOTES, null=True)
18+
personal = models.IntegerField(choices=VOTES, null=True)
19+
calculated_vote = models.FloatField(null=True)
20+
21+
def save(self, force_insert=False, force_update=False, using=None, update_fields=None):
22+
"""
23+
We are overriding this in order to standarize each review vote with the
24+
new vote.
25+
Also we store a calculated vote for each vote so that we don't need to
26+
do it later.
27+
28+
Thanks to Django awesomeness we do all the calculations with only 3
29+
queries to the database. 2 selects and 1 update. The performance is way
30+
faster than I thought.
31+
If improvements need to be done using a better DB than SQLite should
32+
increase performance. As long as the database can handle aggregations
33+
efficiently this will be good.
34+
35+
By casassg
36+
"""
37+
super().save(force_insert, force_update, using, update_fields)
38+
39+
# only recalculate when values are different than None
40+
if not self.personal or not self.tech:
41+
return
42+
43+
# Retrieve averages
44+
avgs = User.objects.filter(id=self.user_id).aggregate(
45+
tech=Avg('vote__tech'),
46+
pers=Avg('vote__personal'))
47+
p_avg = round(avgs['pers'], 2)
48+
t_avg = round(avgs['tech'], 2)
49+
50+
# Calculate standard deviation for each scores
51+
sds = User.objects.filter(id=self.user_id).aggregate(
52+
tech=Avg((F('vote__tech') - t_avg) * (F('vote__tech') - t_avg)),
53+
pers=Avg((F('vote__personal') - p_avg) *
54+
(F('vote__personal') - p_avg)))
55+
56+
# Alternatively, if standard deviation is 0.0, set it as 1.0 to avoid
57+
# division by 0.0 in the update statement
58+
p_sd = round(sds['pers'], 2) or 1.0
59+
t_sd = round(sds['tech'], 2) or 1.0
60+
61+
# Apply standarization. Standarization formula:
62+
# x(new) = (x - u)/o
63+
# where u is the mean and o is the standard deviation
64+
#
65+
# See this: http://www.dataminingblog.com/standardization-vs-
66+
# normalization/
67+
personal = PERSONAL_WEIGHT * (F('personal') - p_avg) / p_sd
68+
tech = TECH_WEIGHT * (F('tech') - t_avg) / t_sd
69+
Vote.objects.filter(user=self.user).update(calculated_vote=(personal + tech) * MAX_VOTES / 10)
70+
71+
class Meta:
72+
unique_together = ('application', 'user')

0 commit comments

Comments
 (0)