show required extensions
[edumed.git] / stage2 / forms.py
1 # -*- coding: utf-8 -*-
2 from decimal import Decimal
3
4 from django import forms
5 from django.core import validators
6 from django.template.defaultfilters import filesizeformat
7 from django.utils.safestring import mark_safe
8
9 from stage2.models import Attachment, Mark, FieldOptionSet, FieldOption
10
11
12 class AttachmentForm(forms.ModelForm):
13     assignment_id = forms.CharField(widget=forms.HiddenInput)
14
15     class Meta:
16         model = Attachment
17         fields = ['file']
18
19     def __init__(self, assignment, file_no, label, options, *args, **kwargs):
20         prefix = 'att%s-%s' % (assignment.id, file_no)
21         super(AttachmentForm, self).__init__(*args, prefix=prefix, **kwargs)
22         self.fields['assignment_id'].initial = assignment.id
23         extensions = options.get('ext')
24         max_mb = options.get('max', 20)
25         self.max_upload_size = max_mb * 1024 * 1024
26         self.fields['file'].widget.attrs['data-max'] = max_mb
27         label_extra = ['maks. %s MB' % max_mb]
28         if extensions:
29             label_extra.append('format: %s' % ', '.join(extensions))
30         self.fields['file'].label = label + u' (%s)' % '; '.join(label_extra)
31         if extensions:
32             self.fields['file'].widget.attrs['data-ext'] = '|'.join(extensions)
33         self.extensions = extensions
34
35     def clean_file(self):
36         file = self.cleaned_data['file']
37         if file.size > self.max_upload_size:
38             raise forms.ValidationError(
39                 u'Prosimy o wysłanie pliku o rozmiarze najwyżej %s. Aktualny rozmiar pliku: %s' % (
40                     filesizeformat(self.max_upload_size), filesizeformat(file.size)))
41         if self.extensions and ('.' not in file.name or file.name.rsplit('.', 1)[1].lower() not in self.extensions):
42             raise forms.ValidationError(u'Niepoprawne rozszerzenie, powinno być jedno z: %s' % ', '.join(self.extensions))
43         return file
44
45
46 class AssignmentFieldForm(forms.Form):
47     value = forms.CharField(required=False)
48     assignment_id = forms.CharField(widget=forms.HiddenInput)
49
50     def __init__(self, label, field_no, options, answer, *args, **kwargs):
51         prefix = 'field%s-%s' % (answer.id, field_no)
52         super(AssignmentFieldForm, self).__init__(prefix=prefix, *args, **kwargs)
53         self.answer = answer
54         self.label = label
55         self.fields['value'].label = label
56         self.type = options['type']
57         self.fields['assignment_id'].initial = answer.assignment.id
58         max_length = options.get('max_length')
59         if max_length:
60             self.fields['value'].validators.append(validators.MaxLengthValidator(int(max_length)))
61             self.fields['value'].label += u' (maks. %s znaków)' % max_length
62         if options.get('widget') == 'area':
63             self.fields['value'].widget = forms.Textarea(attrs={'cols': 80, 'rows': 25})
64         if self.type == 'options':
65             option_set = FieldOptionSet.objects.get(name=options['option_set'])
66             self.fields['value'].widget = forms.Select(choices=option_set.choices(answer))
67             value_options = answer.fieldoption_set.all()
68             if value_options:
69                 self.fields['value'].initial = value_options.get().id
70         else:
71             value = answer.field_values.get(label)
72             self.fields['value'].initial = value or ''
73
74     def clean_value(self):
75         if self.type == 'options':
76             value = self.cleaned_data['value']
77             if value:
78                 try:
79                     option = FieldOption.objects.get(id=int(value))
80                 except (FieldOption.DoesNotExist, ValueError):
81                     raise forms.ValidationError(u'Nieprawidłowa wartość.')
82                 if option.answer != self.answer and option.answer is not None:
83                     raise forms.ValidationError(u'Ta opcja została już wybrana przez kogoś innego.')
84                 return option
85         return self.cleaned_data['value']
86
87     def save(self):
88         value = self.cleaned_data['value']
89         if self.type == 'options':
90             option = value
91             if option:
92                 if option.answer != self.answer:
93                     # not thread-safe :/
94                     assert not option.answer
95                     for opt in self.answer.fieldoption_set.all():
96                         opt.answer = None
97                         opt.save()
98                     option.answer = self.answer
99                     option.save()
100             else:
101                 for opt in self.answer.fieldoption_set.all():
102                     opt.answer = None
103                     opt.save()
104         else:
105             self.answer.field_values[self.label] = value
106             self.answer.save()
107
108
109 class MarkForm(forms.ModelForm):
110     answer_id = forms.CharField(widget=forms.HiddenInput)
111
112     class Meta:
113         model = Mark
114         fields = ['points']
115         widgets = {
116             'points': forms.TextInput(attrs={'type': 'number', 'min': 0, 'step': '0.5'})
117         }
118
119     def __init__(self, answer, criterion, *args, **kwargs):
120         super(MarkForm, self).__init__(*args, **kwargs)
121         self.fields['answer_id'].initial = answer.id
122         points_field = self.fields['points']
123         points_field.label = mark_safe(criterion.form_label())
124         points_field.help_text = '(max %s)' % criterion.max_points
125         points_field.min_value = Decimal(0)
126         points_field.max_value = Decimal(criterion.max_points)
127         points_field.widget.attrs['max'] = criterion.max_points