287f82b87c325a26780592e422b92c6623fda39a
[prawokultury.git] / prawokultury / contact_forms.py
1 # -*- coding: utf-8 -*-
2 from __future__ import unicode_literals
3
4 from django.conf import settings
5 from django import forms
6 from django.core.mail import send_mail
7
8 from contact.forms import ContactForm
9 from contact.models import Contact
10 from contact.fields import HeaderField
11 from django.utils.functional import lazy
12 from django.utils.translation import ugettext_lazy as _
13 from django.utils.safestring import mark_safe
14 from migdal.models import Entry
15
16 from prawokultury.countries import COUNTRIES, TRAVEL_GRANT_COUNTRIES
17
18 mark_safe_lazy = lazy(mark_safe, unicode)
19
20
21 class RegistrationForm(ContactForm):
22     form_tag = 'register'
23
24     save_as_tag = '2018'
25     conference_name = u'CopyCamp 2018'
26     notify_on_register = False
27     
28     form_title = _('Registration')
29     admin_list = ['first_name', 'last_name', 'organization']
30
31     mailing_field = 'agree_mailing'
32
33     travel_grant_countries = TRAVEL_GRANT_COUNTRIES
34
35     first_name = forms.CharField(label=_('First name'), max_length=128)
36     last_name = forms.CharField(label=_('Last name'), max_length=128)
37     contact = forms.EmailField(label=_('E-mail'), max_length=128)
38     organization = forms.CharField(label=_('Organization'), max_length=256, required=False)
39     country = forms.ChoiceField(
40         label=_('Country of residence'), choices=[('', '--------')] + zip(COUNTRIES, COUNTRIES), required=False)
41     travel_grant = forms.BooleanField(
42         label=_('I require financial assistance to attend CopyCamp 2018.'), required=False)
43     travel_grant_motivation = forms.CharField(
44         label=_('Please write us about yourself and why you want to come to CopyCamp. '
45                 'This information will help us evaluate your travel grant application:'),
46         help_text=_('Financial assistance for German audience is possible '
47                     'thanks to the funds of the German Federal Foreign Office transferred by '
48                     'the Foundation for Polish-German Cooperation.'),
49         widget=forms.Textarea, max_length=600, required=False)
50
51     days = forms.ChoiceField(
52        label=_("I'm planning to show up on"),
53        choices=[
54            ('both', _('Both days of the conference')),
55            ('only-28th', _('October 5th only')),
56            ('only-29th', _('October 6th only')),
57        ], widget=forms.RadioSelect())
58
59     # ankieta
60     times_attended = forms.ChoiceField(
61         required=False,
62         label=_("1. How many times have you attended CopyCamp?"),
63         choices=[
64             ('0', _('not yet')),
65             ('1', _('once')),
66             ('2', _('twice')),
67             ('3', _('three times')),
68             ('4', _('four times')),
69             ('5', _('five times')),
70         ], widget=forms.RadioSelect())
71     age = forms.ChoiceField(
72         required=False,
73         label=_("2. Please indicate your age bracket:"),
74         choices=[
75             ('0-19', _('19 or below')),
76             ('20-25', _('20-25')),
77             ('26-35', _('26-35')),
78             ('36-45', _('36-45')),
79             ('46-55', _('46-55')),
80             ('56-65', _('56-65')),
81             ('66+', _('66 or above')),
82         ], widget=forms.RadioSelect())
83     areas = forms.MultipleChoiceField(
84         required=False,
85         label=_("3. Please indicate up to 3 areas you feel most affiliated with"),
86         choices=[
87             ('sztuki plastyczne', _('visual art')),
88             ('literatura', _('literature')),
89             ('muzyka', _('music')),
90             ('teatr', _('theatre')),
91             ('film', _('film production')),
92             ('wydawanie', _('publishing')),
93             ('prawo', _('law')),
94             ('ekonomia', _('economy')),
95             ('socjologia', _('sociology')),
96             ('technika', _('technology')),
97             ('edukacja', _('education')),
98             ('studia', _('higher education')),
99             ('nauka', _('academic research')),
100             ('biblioteki', _('library science')),
101             ('administracja', _('public administration')),
102             ('ngo', _('nonprofit organisations')),
103             ('other', _('other (please specify below)')),
104         ], widget=forms.CheckboxSelectMultiple())
105     areas_other = forms.CharField(required=False, label=_('Fill if you selected “other” above'))
106     source = forms.ChoiceField(
107         required=False,
108         label=_("4. Please indicate how you received information about the conference:"),
109         choices=[
110             ('znajomi', _('through friends sharing on the web')),
111             ('znajomi2', _('through friends by other means')),
112             ('prasa', _('through press')),
113             ('fnp', _('directly through the Foundation\'s facebook or website')),
114             ('www', _('through other websites (please specify below)')),
115             ('other', _('other (please specify below)')),
116         ], widget=forms.RadioSelect())
117     source_other = forms.CharField(required=False, label=_('Fill if you selected “other” or “other website” above'))
118     motivation = forms.ChoiceField(
119         required=False,
120         label=_("6. Please indicate the most important factor for your willingness to participate:"),
121         choices=[
122             ('speaker', _('listening to particular speaker(s)')),
123             ('networking', _('good networking occasion')),
124             ('partnering', _('partnering with organisations present at the event')),
125             ('other', _('other (please specify below)')),
126         ], widget=forms.RadioSelect())
127     motivation_other = forms.CharField(required=False, label=_('Fill if you selected “other” above'))
128
129     agree_mailing = forms.BooleanField(
130         label=_('I want to receive e-mails about future CopyCamps '
131                 'and similar activities of the Modern Poland Foundation'),
132         required=False
133     )
134     agree_license = forms.BooleanField(
135         label=_('Permission for publication'),
136         help_text=mark_safe_lazy(_(
137             u'I agree to having materials, recorded during the conference, released under the terms of '
138             u'<a href="http://creativecommons.org/licenses/by-sa/3.0/deed">CC\u00a0BY-SA</a> license and '
139             u'to publishing my image.')),
140         required=False
141     )
142
143     def __init__(self, *args, **kwargs):
144         super(RegistrationForm, self).__init__(*args, **kwargs)
145         self.started = getattr(settings, 'REGISTRATION_STARTED', False)
146         self.limit_reached = Contact.objects.filter(form_tag=self.save_as_tag).count() >= settings.REGISTRATION_LIMIT
147         try:
148             url = Entry.objects.get(slug_pl='regulamin').get_absolute_url()
149             self.fields['agree_toc'] = forms.BooleanField(
150                 required=True,
151                 label=mark_safe(_('I accept <a href="%s">Terms and Conditions of CopyCamp</a>') % url)
152             )
153         except Entry.DoesNotExist:
154             pass
155
156     def clean_areas(self):
157         data = self.cleaned_data['areas']
158         if len(data) > 3:
159             raise forms.ValidationError(_('Select at most 3 areas'))
160         return data
161
162     def clean(self):
163         cleaned_data = self.cleaned_data
164         if 'travel_grant' in cleaned_data:
165             country = cleaned_data['country']
166             travel_grant = cleaned_data['travel_grant']
167             motivation = cleaned_data['travel_grant_motivation']
168             if country not in self.travel_grant_countries and travel_grant:
169                 raise forms.ValidationError(_('Travel grant is not provided for the selected country'))
170             if travel_grant and not motivation:
171                 self._errors['travel_grant_motivation'] = _('Please provide this information')
172                 raise forms.ValidationError(_('To apply for a travel grant you must provide additional information.'))
173             if not travel_grant and motivation:
174                 cleaned_data['motivation'] = ''
175         return cleaned_data
176
177     def main_fields(self):
178         return [self[name] for name in (
179             'first_name', 'last_name', 'contact', 'organization', 'country',
180             'travel_grant', 'travel_grant_motivation', 'days')]
181
182     def survey_fields(self):
183         return [self[name] for name in (
184             'times_attended', 'age',
185             'areas', 'areas_other', 'source', 'source_other', 'motivation', 'motivation_other')]
186
187     def agreement_fields(self):
188         return [self[name] for name in ('agree_mailing', 'agree_license', 'agree_toc')]
189
190
191 class RegisterSpeaker(RegistrationForm):
192     form_tag = 'register-speaker'
193     save_as_tag = '2019-speaker'
194     form_title = _('Open call for presentations')
195     notify_on_register = False
196     mailing_field = 'agree_mailing'
197
198     bio = forms.CharField(label=mark_safe_lazy(
199         _('Short biographical note in Polish (max. 500 characters)')),
200                           widget=forms.Textarea, max_length=500, required=True)
201     bio_en = forms.CharField(label=_('Short biographical note in English (max. 500 characters, not required)'), widget=forms.Textarea,
202                              max_length=500, required=False)
203     photo = forms.FileField(label=_('Photo'), required=False)
204     phone = forms.CharField(label=_('Phone number'), max_length=64,
205                             required=False,
206                             help_text=_('(used only for organizational purposes)'))
207
208     presentation_title = forms.CharField(
209         label=mark_safe_lazy(_('Presentation title in Polish')),
210         max_length=256)
211     presentation_title_en = forms.CharField(
212         label=_('Presentation title in English (not required)'), max_length=256, required=False)
213     presentation_summary = forms.CharField(label=_('Presentation summary (max. 1800 characters)'),
214                                            widget=forms.Textarea, max_length=1800)
215
216     # presentation_post_conference_publication = forms.BooleanField(
217     #     label=_('I am interested in including my paper in the post-conference publication'),
218     #     required=False
219     # )
220
221     agree_data = None
222
223     agree_terms = forms.BooleanField(
224         label=mark_safe_lazy(
225             _(u'I accept <a href="/en/info/terms-and-conditions/">CopyCamp Terms and Conditions</a>.'))
226     )
227
228     def __init__(self, *args, **kw):
229         super(RegisterSpeaker, self).__init__(*args, **kw)
230         self.started = getattr(settings, 'REGISTRATION_SPEAKER_STARTED', False)
231         self.closed = getattr(settings, 'REGISTRATION_SPEAKER_CLOSED', False)
232         self.fields.keyOrder = [
233             'first_name',
234             'last_name',
235             'contact',
236             'phone',
237             'organization',
238             'bio',
239             'bio_en',
240             'photo',
241             'presentation_title',
242             'presentation_title_en',
243             'presentation_summary',
244             # 'presentation_post_conference_publication',
245
246             'agree_mailing',
247             # 'agree_data',
248             'agree_license',
249             'agree_terms',
250         ]
251
252
253 class RemindForm(ContactForm):
254     form_tag = 'remind-me'
255     save_as_tag = 'remind-me-2019'
256     form_title = u'CopyCamp 2019'
257     notify_on_register = False
258     notify_user = False
259
260
261 class NextForm(ContactForm):
262     form_tag = '/next'
263     form_title = _('Next CopyCamp')
264
265     name = forms.CharField(label=_('Name'), max_length=128)
266     contact = forms.EmailField(label=_('E-mail'), max_length=128)
267     organization = forms.CharField(label=_('Organization'),
268                                    max_length=256, required=False)
269
270
271 def workshop_field(label, help=None):
272     return forms.BooleanField(label=_(label), required=False, help_text=help)
273
274
275 class WorkshopForm(ContactForm):
276     form_tag = 'workshops'
277     save_as_tag = 'workshops-2018'
278     conference_name = u'CopyCamp 2018'
279     form_title = _('Workshop')
280     notify_on_register = False
281     mailing_field = 'agree_mailing'
282
283     first_name = forms.CharField(label=_('First name'), max_length=128)
284     last_name = forms.CharField(label=_('Last name'), max_length=128)
285     contact = forms.EmailField(label=_('E-mail'), max_length=128)
286     organization = forms.CharField(label=_('Organization'), max_length=256, required=False)
287     country = forms.ChoiceField(
288         label=_('Country of residence'), choices=[('', '--------')] + zip(COUNTRIES, COUNTRIES), required=False)
289
290     _header = HeaderField(
291         label=mark_safe_lazy(_("<h3>I'll take a part in workshops</h3>")),
292         help_text=_('Only workshops with any spots left are visible here.'))
293
294     _h1 = HeaderField(label=mark_safe_lazy(_("<strong>Friday, October 5th, 11 a.m.–1 p.m.</strong>")))
295
296     w_dobosz = workshop_field(
297         u'Elżbieta Dobosz, Urząd Patentowy RP: Ochrona wzornictwa, co można chronić, co warto chronić i w jaki sposób',
298         u'Uczestnicy mogą przedstawić na warsztatach swoje wzory – '
299         u'rozwiązania wizualne ze wszystkich kategorii produktów.')
300     w_kozak = workshop_field(
301         u'Łukasz Kozak i Krzysztof Siewicz: Projekt : Upiór – wprowadzenie i warsztaty dla twórców gier')
302     w_secker = workshop_field(
303         u'Jane Secker and Chris Morrison: Embedding Copyright literacy using games-based learning',
304         _(u'The workshop will be conducted in English.'))
305
306     _h2 = HeaderField(label=mark_safe_lazy(_("<strong>Saturday, October 6th, 11 a.m.–1 p.m.</strong>")))
307
308     w_kakareko = workshop_field(
309         u'Ksenia Kakareko: Regulacje prawne dotyczące wykorzystania materiałów zdigitalizowanych')
310     w_kakareko_question = forms.CharField(
311         label=u'Możesz opisać sprawy, z którymi najczęściej spotykasz się jako pracownik instytucji posiadającej '
312               u'zdigitalizowane zbiory lub jako użytkownik tych zbiorów '
313               u'(max 800 znaków)',
314         max_length=800, widget=forms.Textarea, required=False)
315     w_sikorska = workshop_field(
316         u'Krzysztof Siewicz: Autor: projektant / prawo autorskie dla projektantów')
317     w_sikorska_question = forms.CharField(
318         label=u'Jeżeli chcesz, możesz przesłać prowadzącemu swoje pytanie dotyczące prawa autorskiego, '
319               u'co pomoże mu lepiej przygotować warsztaty '
320               u'(max 800 znaków)',
321         max_length=800, widget=forms.Textarea, required=False)
322     w_sztoldman = workshop_field(
323         u'dr Agnieszka Sztoldman, Aleksandra Burda, SMM Legal: Spory o pieniądze w branżach IP-driven')
324
325     _header_1 = HeaderField(label='')
326     _header_2 = HeaderField(label='')
327
328     start_workshops = ('dobosz', 'kozak', 'secker', 'kakareko', 'sikorska', 'sztoldman')
329
330     slots = (
331         ('_h1', 'dobosz', 'kozak', 'secker'),
332         ('_h2', 'kakareko', 'sikorska', 'sztoldman'),
333     )
334
335     limits = {
336         'dobosz': 30,
337         'kozak': 30,
338         'secker': 30,
339         'kakareko': 30,
340         'sikorska': 30,
341         'sztoldman': 30,
342     }
343
344     agree_mailing = forms.BooleanField(
345         label=_('I am interested in receiving information about the Modern Poland Foundation\'s activities by e-mail'),
346         required=False)
347     agree_license = forms.BooleanField(
348         label=_('Permission for publication'),
349         help_text=mark_safe_lazy(_(
350             u'I agree to having materials, recorded during the conference, released under the terms of '
351             u'<a href="http://creativecommons.org/licenses/by-sa/3.0/deed">CC\u00a0BY-SA</a> '
352             u'license and to publishing my image.')),
353         required=False)
354
355     def __init__(self, *args, **kwargs):
356         super(WorkshopForm, self).__init__(*args, **kwargs)
357         try:
358             url = Entry.objects.get(slug_pl='regulamin').get_absolute_url()
359             self.fields['agree_toc'] = forms.BooleanField(
360                 required=True,
361                 label=mark_safe(_('I accept <a href="%s">Terms and Conditions of CopyCamp</a>') % url)
362             )
363         except Entry.DoesNotExist:
364             pass
365         counts = {k: 0 for k in self.start_workshops}
366         for contact in Contact.objects.filter(form_tag=self.save_as_tag):
367             for workshop in self.start_workshops:
368                 if contact.body.get('w_%s' % workshop, False):
369                     counts[workshop] += 1
370                     # if workshop == 'youtube' and counts[workshop] == 30:
371                     #     send_mail(u'Warsztaty YouTube', u'Przekroczono limit 30 osób na warsztaty YouTube',
372                     #               'no-reply@copycamp.pl',
373                     #               ['krzysztof.siewicz@nowoczesnapolska.org.pl'],
374                     #               fail_silently=True)
375
376         some_full = False
377         for k, v in counts.items():
378             if v >= self.limits[k]:
379                 some_full = True
380                 if 'w_%s' % k in self.fields:
381                     del self.fields['w_%s' % k]
382         if not some_full:
383             self.fields['_header'].help_text = None
384
385     def clean(self):
386         for slot in self.slots:
387             if sum(1 for w in slot if self.cleaned_data.get('w_%s' % w)) > 1:
388                 self._errors[slot[0]] = [_("You can't choose more than one workshop during the same period")]
389         if not any(self.cleaned_data.get('w_%s' % w) for w in self.start_workshops):
390             self._errors['_header'] = [_("Please choose at least one workshop.")]
391         return self.cleaned_data