|
| 1 | +from django import forms |
| 2 | +from django.conf import settings |
| 3 | +from django.core.exceptions import ValidationError |
| 4 | +from django.core.files.storage import FileSystemStorage |
| 5 | +from django.utils import timezone |
| 6 | +from django.utils.safestring import mark_safe |
| 7 | +from django.utils.translation import gettext_lazy as _ |
| 8 | + |
| 9 | +from app.mixins import BootstrapFormMixin |
| 10 | +from application.models import Application |
| 11 | + |
| 12 | + |
| 13 | +YEARS = [(year, str(year)) for year in range(timezone.now().year - 1, timezone.now().year + 6)] |
| 14 | +DEFAULT_YEAR = timezone.now().year + 1 |
| 15 | +EXTENSIONS = getattr(settings, 'SUPPORTED_RESUME_EXTENSIONS', None) |
| 16 | + |
| 17 | +HACK_NAME = getattr(settings, 'HACKATHON_NAME') |
| 18 | +EXTRA_NAME = [' 2016 Fall', ' 2016 Winter', ' 2017 Fall', ' 2017 Winter', ' 2018', ' 2019', ' 2021', ' 2022'] |
| 19 | +PREVIOUS_HACKS = [(i, HACK_NAME + EXTRA_NAME[i]) for i in range(0, len(EXTRA_NAME))] |
| 20 | +HACK_DAYS = [(x, x) for x in ['Friday', 'Saturday', 'Sunday']] |
| 21 | +ENGLISH_LEVELS = [(x, x) for x in ['1', '2', '3', '4', '5']] |
| 22 | + |
| 23 | + |
| 24 | +class ApplicationForm(BootstrapFormMixin, forms.ModelForm): |
| 25 | + |
| 26 | + diet_notice = forms.BooleanField( |
| 27 | + label=_('I authorize %s to use my food allergies and intolerances information to ' |
| 28 | + 'manage the catering service only.') % getattr(settings, 'HACKATHON_ORG') |
| 29 | + ) |
| 30 | + |
| 31 | + terms_and_conditions = forms.BooleanField( |
| 32 | + label=mark_safe(_('I\'ve read, understand and accept <a href="/terms_and_conditions" target="_blank">%s ' |
| 33 | + 'Terms & Conditions</a> and <a href="/privacy_and_cookies" target="_blank">%s ' |
| 34 | + 'Privacy and Cookies Policy</a>.' % ( |
| 35 | + getattr(settings, 'HACKATHON_NAME', ''), getattr(settings, 'HACKATHON_NAME', '') |
| 36 | + ))) |
| 37 | + ) |
| 38 | + |
| 39 | + exclude_save = ['terms_and_conditions', 'diet_notice'] |
| 40 | + |
| 41 | + def save(self, commit=True): |
| 42 | + model_fields = [field.name for field in self.Meta.model._meta.fields] |
| 43 | + extra_fields = [field for field in self.declared_fields if field not in model_fields and |
| 44 | + field not in self.exclude_save] |
| 45 | + files_fields = getattr(self, 'files', {}) |
| 46 | + extra_data = {field: data for field, data in self.cleaned_data.items() |
| 47 | + if field in extra_fields and field not in files_fields.keys()} |
| 48 | + self.instance.form_data = extra_data |
| 49 | + instance = super().save(commit) |
| 50 | + if commit: |
| 51 | + self.save_files(instance=instance) |
| 52 | + return instance |
| 53 | + |
| 54 | + def save_files(self, instance): |
| 55 | + files_fields = getattr(self, 'files', {}) |
| 56 | + fs = FileSystemStorage() |
| 57 | + for field_name, file in files_fields.items(): |
| 58 | + file_path = '%s/%s/%s/%s_%s.%s' % (instance.edition.name, instance.type.name, field_name, |
| 59 | + instance.get_full_name().replace(' ', '-'), instance.get_uuid, |
| 60 | + file.name.split('.')[-1]) |
| 61 | + if fs.exists(file_path): |
| 62 | + fs.delete(file_path) |
| 63 | + fs.save(name=file_path, content=file) |
| 64 | + form_data = instance.form_data |
| 65 | + form_data[field_name] = {'type': 'file', 'path': file_path} |
| 66 | + instance.form_data = form_data |
| 67 | + if len(files_fields) > 0: |
| 68 | + instance.save() |
| 69 | + return files_fields.keys() |
| 70 | + |
| 71 | + def get_hidden_edit_fields(self): |
| 72 | + return self.exclude_save.copy() |
| 73 | + |
| 74 | + def __init__(self, *args, **kwargs): |
| 75 | + super().__init__(*args, **kwargs) |
| 76 | + self.initial.update(self.instance.form_data) |
| 77 | + instance = kwargs.get('instance', None) |
| 78 | + hidden_fields = self.get_hidden_edit_fields() |
| 79 | + if instance is not None and instance._state.db is not None: # instance in DB |
| 80 | + for hidden_field in hidden_fields: |
| 81 | + self.fields.get(hidden_field).required = False |
| 82 | + |
| 83 | + def get_bootstrap_field_info(self): |
| 84 | + fields = super().get_bootstrap_field_info() |
| 85 | + instance = getattr(self, 'instance', None) |
| 86 | + if instance is None or instance._state.db is None: # instance not in DB |
| 87 | + policy_fields = self.get_policy_fields() |
| 88 | + fields.update({ |
| 89 | + _('HackUPC Polices'): { |
| 90 | + 'fields': policy_fields, |
| 91 | + 'description': '<p style="color: margin-top: 1em;display: block;' |
| 92 | + 'margin-bottom: 1em;line-height: 1.25em;">We, Hackers at UPC, ' |
| 93 | + 'process your information to organize an awesome hackathon. It ' |
| 94 | + 'will also include images and videos of yourself during the event. ' |
| 95 | + 'Your data will be used for admissions mainly.' |
| 96 | + 'For more information on the processing of your ' |
| 97 | + 'personal data and on how to exercise your rights of access, ' |
| 98 | + 'rectification, suppression, limitation, portability and opposition ' |
| 99 | + 'please visit our Privacy and Cookies Policy.</p>' |
| 100 | + }}) |
| 101 | + fields[next(iter(fields))]['fields'].append({'name': 'promotional_code'}) |
| 102 | + return fields |
| 103 | + |
| 104 | + def get_policy_fields(self): |
| 105 | + return [{'name': 'terms_and_conditions', 'space': 12}, {'name': 'diet_notice', 'space': 12}] |
| 106 | + |
| 107 | + def clean_promotional_code(self): |
| 108 | + promotional_code = self.cleaned_data.get('promotional_code', None) |
| 109 | + if promotional_code is not None: |
| 110 | + if promotional_code.usages != -1 and promotional_code.application_set.count() >= promotional_code.usages: |
| 111 | + raise ValidationError('This code is out of usages or not for this type') |
| 112 | + return promotional_code |
| 113 | + |
| 114 | + class Meta: |
| 115 | + model = Application |
| 116 | + description = '' |
| 117 | + exclude = ['user', 'uuid', 'data', 'submission_date', 'status_update_date', 'status', 'contacted_by', 'type', |
| 118 | + 'last_modified', 'edition'] |
| 119 | + help_texts = { |
| 120 | + 'gender': _('This is for demographic purposes. You can skip this question if you want.'), |
| 121 | + 'other_diet': _('Please fill here in your dietary requirements. ' |
| 122 | + 'We want to make sure we have food for you!'), |
| 123 | + 'origin': "Please select one of the dropdown options or write 'Others'. If the dropdown doesn't show up," |
| 124 | + " type following this schema: <strong>city, nation, country</strong>" |
| 125 | + } |
| 126 | + labels = { |
| 127 | + 'gender': _('What gender do you identify as?'), |
| 128 | + 'other_gender': _('Self-describe'), |
| 129 | + 'tshirt_size': _('What\'s your t-shirt size?'), |
| 130 | + 'diet': _('Dietary requirements'), |
| 131 | + } |
| 132 | + widgets = { |
| 133 | + 'promotional_code': forms.HiddenInput |
| 134 | + } |
0 commit comments