2018-01-20 20:09:57 +01:00
|
|
|
from django.utils.safestring import mark_safe
|
2018-01-27 04:38:11 +01:00
|
|
|
from django import forms
|
2018-01-21 16:51:29 +01:00
|
|
|
from django.forms.widgets import TextInput, Media
|
|
|
|
from django.utils.translation import ugettext_lazy as _
|
2018-04-02 04:22:30 +02:00
|
|
|
from django.utils import timezone
|
2018-01-21 16:51:29 +01:00
|
|
|
|
2018-02-02 05:26:11 +01:00
|
|
|
from django.contrib.auth.models import User
|
2018-03-25 20:40:29 +02:00
|
|
|
from .models.locale import Country, SPR, City
|
2018-04-25 16:36:01 +02:00
|
|
|
from .models.profiles import Team, UserProfile
|
2018-04-24 16:22:16 +02:00
|
|
|
from .models.events import (
|
|
|
|
Event,
|
|
|
|
EventComment,
|
|
|
|
CommonEvent,
|
|
|
|
EventSeries,
|
|
|
|
Place,
|
|
|
|
EventPhoto,
|
2018-04-25 16:36:01 +02:00
|
|
|
)
|
|
|
|
from .models.speakers import (
|
|
|
|
Speaker,
|
|
|
|
Talk,
|
2018-04-24 16:22:16 +02:00
|
|
|
Presentation,
|
|
|
|
SpeakerRequest,
|
|
|
|
)
|
2018-04-15 17:31:12 +02:00
|
|
|
import recurrence
|
2017-12-30 04:27:05 +01:00
|
|
|
|
2018-04-02 04:22:30 +02:00
|
|
|
import pytz
|
2018-01-21 16:51:29 +01:00
|
|
|
from datetime import time
|
2018-01-21 17:18:58 +01:00
|
|
|
from time import strptime, strftime
|
2018-01-21 16:51:29 +01:00
|
|
|
|
2018-01-20 22:50:11 +01:00
|
|
|
class Lookup(TextInput):
|
|
|
|
input_type = 'text'
|
2018-01-20 20:09:57 +01:00
|
|
|
template_name = 'forms/widgets/lookup.html'
|
|
|
|
add_id_index = False
|
|
|
|
checked_attribute = {'selected': True}
|
|
|
|
option_inherits_attrs = False
|
|
|
|
|
2018-03-25 20:40:29 +02:00
|
|
|
def __init__(self, source, key="id", label='__str__', attrs=None):
|
2018-01-20 20:09:57 +01:00
|
|
|
super().__init__(attrs)
|
|
|
|
self.source = source
|
|
|
|
self.key = key
|
|
|
|
self.label = label
|
|
|
|
|
|
|
|
def get_context(self, name, value, attrs):
|
|
|
|
context = super().get_context(name, value, attrs)
|
|
|
|
return context
|
|
|
|
|
2018-01-21 18:09:18 +01:00
|
|
|
def format_value(self, value):
|
|
|
|
if value is not None:
|
2018-03-25 20:40:29 +02:00
|
|
|
lookup_query = {self.key: value}
|
|
|
|
lookup_object = self.source.objects.get(**lookup_query)
|
|
|
|
lookup_field = getattr(lookup_object, self.label)
|
|
|
|
if callable(lookup_field):
|
|
|
|
lookup_value = lookup_field()
|
|
|
|
else:
|
|
|
|
lookup_value = lookup_field
|
|
|
|
return mark_safe('<option value="%s">%s</option>' % (value, lookup_value))
|
2018-01-21 18:09:18 +01:00
|
|
|
else:
|
|
|
|
return mark_safe('<option value="">--------</option>')
|
|
|
|
|
2018-01-27 04:38:11 +01:00
|
|
|
class DateWidget(forms.DateInput):
|
2018-01-21 18:09:18 +01:00
|
|
|
"""A more-friendly date widget with a p% if widget.value != None %} value="{{ widget.value|stringformat:'s' }}"{% endif %op-up calendar.
|
2018-01-21 16:51:29 +01:00
|
|
|
"""
|
|
|
|
template_name = 'forms/widgets/date.html'
|
|
|
|
def __init__(self, attrs=None):
|
|
|
|
self.date_class = 'datepicker'
|
|
|
|
if not attrs:
|
|
|
|
attrs = {}
|
|
|
|
if 'date_class' in attrs:
|
|
|
|
self.date_class = attrs.pop('date_class')
|
|
|
|
if 'class' not in attrs:
|
|
|
|
attrs['class'] = 'date'
|
|
|
|
|
|
|
|
super(DateWidget, self).__init__(attrs=attrs)
|
|
|
|
|
|
|
|
|
2018-01-27 04:38:11 +01:00
|
|
|
class TimeWidget(forms.MultiWidget):
|
2018-01-21 16:51:29 +01:00
|
|
|
"""A more-friendly time widget.
|
|
|
|
"""
|
|
|
|
def __init__(self, attrs=None):
|
|
|
|
self.time_class = 'timepicker'
|
|
|
|
if not attrs:
|
|
|
|
attrs = {}
|
|
|
|
if 'time_class' in attrs:
|
|
|
|
self.time_class = attrs.pop('time_class')
|
|
|
|
if 'class' not in attrs:
|
|
|
|
attrs['class'] = 'time'
|
|
|
|
|
|
|
|
widgets = (
|
2018-01-27 04:38:11 +01:00
|
|
|
forms.Select(attrs=attrs, choices=[(i + 1, "%02d" % (i + 1)) for i in range(0, 12)]),
|
|
|
|
forms.Select(attrs=attrs, choices=[(i, "%02d" % i) for i in range(00, 60, 15)]),
|
|
|
|
forms.Select(attrs=attrs, choices=[('AM', _('AM')), ('PM', _('PM'))])
|
2018-01-21 16:51:29 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
super(TimeWidget, self).__init__(widgets, attrs)
|
|
|
|
|
|
|
|
def decompress(self, value):
|
|
|
|
if isinstance(value, str):
|
|
|
|
try:
|
|
|
|
value = strptime(value, '%I:%M %p')
|
|
|
|
except:
|
|
|
|
value = strptime(value, '%H:%M:%S')
|
|
|
|
hour = int(value.tm_hour)
|
|
|
|
minute = int(value.tm_min)
|
|
|
|
if hour < 12:
|
|
|
|
meridian = 'AM'
|
|
|
|
else:
|
|
|
|
meridian = 'PM'
|
|
|
|
hour -= 12
|
|
|
|
return (hour, minute, meridian)
|
|
|
|
elif isinstance(value, time):
|
|
|
|
hour = int(value.strftime("%I"))
|
|
|
|
minute = int(value.strftime("%M"))
|
|
|
|
meridian = value.strftime("%p")
|
|
|
|
return (hour, minute, meridian)
|
|
|
|
return (None, None, None)
|
|
|
|
|
|
|
|
def value_from_datadict(self, data, files, name):
|
|
|
|
value = super(TimeWidget, self).value_from_datadict(data, files, name)
|
|
|
|
t = strptime("%02d:%02d %s" % (int(value[0]), int(value[1]), value[2]), "%I:%M %p")
|
|
|
|
return strftime("%H:%M:%S", t)
|
|
|
|
|
|
|
|
def format_output(self, rendered_widgets):
|
|
|
|
return '<span class="%s">%s%s%s</span>' % (
|
|
|
|
self.time_class,
|
|
|
|
rendered_widgets[0], rendered_widgets[1], rendered_widgets[2]
|
|
|
|
)
|
|
|
|
|
2018-01-27 04:38:11 +01:00
|
|
|
class DateTimeWidget(forms.SplitDateTimeWidget):
|
2018-01-21 16:51:29 +01:00
|
|
|
"""
|
|
|
|
A more-friendly date/time widget.
|
|
|
|
"""
|
|
|
|
def __init__(self, attrs=None, date_format=None, time_format=None):
|
|
|
|
super(DateTimeWidget, self).__init__(attrs, date_format, time_format)
|
|
|
|
self.widgets = (
|
|
|
|
DateWidget(attrs=attrs),
|
|
|
|
TimeWidget(attrs=attrs),
|
|
|
|
)
|
|
|
|
|
|
|
|
def decompress(self, value):
|
|
|
|
if value:
|
|
|
|
d = strftime("%Y-%m-%d", value.timetuple())
|
|
|
|
t = strftime("%I:%M %p", value.timetuple())
|
|
|
|
return (d, t)
|
|
|
|
else:
|
|
|
|
return (None, None)
|
|
|
|
|
|
|
|
def format_output(self, rendered_widgets):
|
|
|
|
return '%s %s' % (rendered_widgets[0], rendered_widgets[1])
|
|
|
|
|
2018-01-21 17:18:58 +01:00
|
|
|
def value_from_datadict(self, data, files, name):
|
|
|
|
values = super(DateTimeWidget, self).value_from_datadict(data, files, name)
|
|
|
|
return ' '.join(values)
|
|
|
|
|
2018-01-27 04:38:11 +01:00
|
|
|
class TeamForm(forms.ModelForm):
|
2017-12-30 04:27:05 +01:00
|
|
|
class Meta:
|
|
|
|
model = Team
|
2018-02-27 05:03:56 +01:00
|
|
|
fields = ['name', 'description', 'category', 'city', 'web_url', 'tz']
|
2018-01-21 18:09:18 +01:00
|
|
|
widgets = {
|
2018-03-25 20:40:29 +02:00
|
|
|
'city': Lookup(source=City),
|
2018-01-21 18:09:18 +01:00
|
|
|
}
|
2018-01-29 16:06:46 +01:00
|
|
|
raw_id_fields = ('city')
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
self.fields['city'].required = True
|
2017-12-30 04:27:05 +01:00
|
|
|
|
2018-01-27 04:38:11 +01:00
|
|
|
class NewTeamForm(forms.ModelForm):
|
2017-12-30 04:27:05 +01:00
|
|
|
class Meta:
|
|
|
|
model = Team
|
2018-04-17 03:52:24 +02:00
|
|
|
fields = ['name', 'city', 'tz']
|
2018-01-20 20:09:57 +01:00
|
|
|
widgets = {
|
2018-03-25 20:40:29 +02:00
|
|
|
'city': Lookup(source=City),
|
2018-01-20 20:09:57 +01:00
|
|
|
}
|
2018-01-25 22:50:19 +01:00
|
|
|
raw_id_fields = ('city')
|
2018-01-29 16:06:46 +01:00
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
self.fields['city'].required = True
|
2017-12-31 22:54:35 +01:00
|
|
|
|
2018-04-17 03:52:24 +02:00
|
|
|
class TeamDefinitionForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = Team
|
|
|
|
fields = ['category', 'web_url', 'description']
|
|
|
|
|
2018-01-27 04:38:11 +01:00
|
|
|
class DeleteTeamForm(forms.Form):
|
|
|
|
confirm = forms.BooleanField(label="Yes, delete team", required=True)
|
|
|
|
|
|
|
|
class TeamEventForm(forms.ModelForm):
|
2017-12-31 22:54:35 +01:00
|
|
|
class Meta:
|
|
|
|
model = Event
|
2018-04-14 17:14:40 +02:00
|
|
|
fields = ['name', 'start_time', 'end_time', 'summary', 'web_url', 'announce_url', 'tags']
|
2018-01-20 20:09:57 +01:00
|
|
|
widgets = {
|
2018-03-25 20:40:29 +02:00
|
|
|
'place': Lookup(source=Place),
|
2018-01-21 18:09:18 +01:00
|
|
|
'start_time': DateTimeWidget,
|
|
|
|
'end_time': DateTimeWidget
|
2018-01-20 20:09:57 +01:00
|
|
|
}
|
2018-04-02 04:22:30 +02:00
|
|
|
def __init__(self, *args, **kargs):
|
|
|
|
super().__init__(*args, **kargs)
|
|
|
|
event_tz = pytz.timezone(self.instance.tz)
|
|
|
|
if self.instance.local_start_time: self.initial['start_time'] = self.instance.local_start_time
|
|
|
|
if self.instance.local_end_time: self.initial['end_time'] = self.instance.local_end_time
|
|
|
|
print("Initial: %s" % self.initial)
|
|
|
|
|
|
|
|
def clean(self):
|
|
|
|
cleaned_data = super().clean()
|
|
|
|
event_tz = pytz.timezone(self.instance.tz)
|
|
|
|
print("Clean: %s" % cleaned_data)
|
|
|
|
cleaned_data['start_time'] = pytz.utc.localize(timezone.make_naive(event_tz.localize(timezone.make_naive(cleaned_data['start_time']))))
|
|
|
|
cleaned_data['end_time'] = pytz.utc.localize(timezone.make_naive(event_tz.localize(timezone.make_naive(cleaned_data['end_time']))))
|
|
|
|
return cleaned_data
|
2017-12-31 22:54:35 +01:00
|
|
|
|
2018-01-27 04:38:11 +01:00
|
|
|
class NewTeamEventForm(forms.ModelForm):
|
2018-04-15 17:31:12 +02:00
|
|
|
recurrences = recurrence.forms.RecurrenceField(label="Repeat", required=False)
|
2017-12-31 22:54:35 +01:00
|
|
|
class Meta:
|
|
|
|
model = Event
|
2018-04-15 17:31:12 +02:00
|
|
|
fields = ['name', 'start_time', 'end_time', 'recurrences', 'summary']
|
2018-01-20 20:09:57 +01:00
|
|
|
widgets = {
|
2018-01-21 16:51:29 +01:00
|
|
|
'start_time': DateTimeWidget,
|
|
|
|
'end_time': DateTimeWidget
|
2018-01-20 20:09:57 +01:00
|
|
|
}
|
2018-04-02 04:22:30 +02:00
|
|
|
def __init__(self, *args, **kargs):
|
|
|
|
super().__init__(*args, **kargs)
|
|
|
|
event_tz = pytz.timezone(self.instance.tz)
|
|
|
|
if self.instance.local_start_time: self.initial['start_time'] = self.instance.local_start_time
|
|
|
|
if self.instance.local_end_time: self.initial['end_time'] = self.instance.local_end_time
|
|
|
|
print("Initial: %s" % self.initial)
|
|
|
|
|
|
|
|
def clean(self):
|
|
|
|
cleaned_data = super().clean()
|
|
|
|
event_tz = pytz.timezone(self.instance.tz)
|
|
|
|
print("Clean: %s" % cleaned_data)
|
|
|
|
cleaned_data['start_time'] = pytz.utc.localize(timezone.make_naive(event_tz.localize(timezone.make_naive(cleaned_data['start_time']))))
|
|
|
|
cleaned_data['end_time'] = pytz.utc.localize(timezone.make_naive(event_tz.localize(timezone.make_naive(cleaned_data['end_time']))))
|
|
|
|
return cleaned_data
|
2018-01-20 20:09:57 +01:00
|
|
|
|
2018-01-27 04:38:11 +01:00
|
|
|
class DeleteEventForm(forms.Form):
|
|
|
|
confirm = forms.BooleanField(label="Yes, delete event", required=True)
|
|
|
|
|
2018-04-14 17:14:40 +02:00
|
|
|
class EventSeriesForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = EventSeries
|
|
|
|
fields = ['name', 'start_time', 'end_time', 'recurrences', 'summary']
|
|
|
|
widgets = {
|
|
|
|
'start_time': TimeWidget,
|
|
|
|
'end_time': TimeWidget
|
|
|
|
}
|
|
|
|
|
|
|
|
class DeleteEventSeriesForm(forms.Form):
|
|
|
|
confirm = forms.BooleanField(label="Yes, delete series", required=True)
|
|
|
|
|
2018-03-17 22:37:56 +01:00
|
|
|
class UploadEventPhotoForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = EventPhoto
|
|
|
|
fields = ['src', 'title', 'caption']
|
|
|
|
|
2018-03-24 05:00:38 +01:00
|
|
|
class EventCommentForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = EventComment
|
|
|
|
fields = ['body']
|
|
|
|
|
2018-01-27 04:38:11 +01:00
|
|
|
class NewPlaceForm(forms.ModelForm):
|
2018-01-21 05:18:02 +01:00
|
|
|
class Meta:
|
|
|
|
model = Place
|
|
|
|
fields = ['name', 'address', 'city', 'longitude', 'latitude', 'place_url', 'tz']
|
|
|
|
widgets = {
|
2018-03-25 20:40:29 +02:00
|
|
|
'city': Lookup(source=City),
|
2018-01-21 05:18:02 +01:00
|
|
|
}
|
2018-01-29 18:48:14 +01:00
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
self.fields['city'].required = True
|
2017-12-31 22:54:35 +01:00
|
|
|
|
2018-02-02 05:26:11 +01:00
|
|
|
class UserForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = User
|
|
|
|
fields = ['email']
|
|
|
|
|
|
|
|
class UserProfileForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = UserProfile
|
2018-04-21 16:28:04 +02:00
|
|
|
fields = ['avatar', 'realname', 'city', 'tz', 'send_notifications']
|
2018-02-26 17:53:50 +01:00
|
|
|
labels = {
|
|
|
|
'send_notifications': _('Send me notification emails'),
|
|
|
|
}
|
2018-04-21 16:28:04 +02:00
|
|
|
widgets = {
|
|
|
|
'city': Lookup(source=City),
|
|
|
|
}
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
self.fields['city'].required = True
|
2018-02-26 17:53:50 +01:00
|
|
|
|
2018-03-04 20:10:37 +01:00
|
|
|
class ConfirmProfileForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = UserProfile
|
2018-04-21 16:28:04 +02:00
|
|
|
fields = ['avatar', 'realname', 'city']
|
|
|
|
widgets = {
|
|
|
|
'city': Lookup(source=City),
|
|
|
|
}
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
self.fields['city'].required = True
|
2018-03-04 20:10:37 +01:00
|
|
|
|
2018-02-26 17:53:50 +01:00
|
|
|
class SendNotificationsForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = UserProfile
|
|
|
|
fields = ['send_notifications']
|
|
|
|
labels = {
|
|
|
|
'send_notifications': _('Send me notification emails'),
|
|
|
|
}
|
2018-02-02 05:26:11 +01:00
|
|
|
|
2018-02-15 02:01:43 +01:00
|
|
|
class SearchForm(forms.Form):
|
2018-03-25 20:40:29 +02:00
|
|
|
city = forms.IntegerField(required=False, widget=Lookup(source=City, label='name'))
|
2018-02-15 02:01:43 +01:00
|
|
|
distance = forms.IntegerField(label=_("Distance(km)"), required=True)
|
|
|
|
class Meta:
|
|
|
|
widgets ={
|
2018-03-25 20:40:29 +02:00
|
|
|
'city': Lookup(source=City, label='name'),
|
2018-02-15 02:01:43 +01:00
|
|
|
}
|
2018-03-22 05:15:09 +01:00
|
|
|
|
|
|
|
class NewCommonEventForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = CommonEvent
|
|
|
|
fields = [
|
|
|
|
'name',
|
|
|
|
'start_time',
|
|
|
|
'end_time',
|
|
|
|
'summary',
|
|
|
|
|
|
|
|
'country',
|
|
|
|
'spr',
|
|
|
|
'city',
|
|
|
|
'place',
|
|
|
|
|
|
|
|
'web_url',
|
|
|
|
'announce_url',
|
|
|
|
|
|
|
|
'category',
|
|
|
|
'tags',
|
|
|
|
]
|
|
|
|
widgets ={
|
2018-03-25 20:40:29 +02:00
|
|
|
'country': Lookup(source=Country, label='name'),
|
|
|
|
'spr': Lookup(source=SPR, label='name'),
|
|
|
|
'city': Lookup(source=City, label='name'),
|
|
|
|
'place': Lookup(source=Place, label='name'),
|
2018-03-22 05:15:09 +01:00
|
|
|
'start_time': DateTimeWidget,
|
|
|
|
'end_time': DateTimeWidget
|
|
|
|
}
|
|
|
|
|
2018-04-24 16:22:16 +02:00
|
|
|
class SpeakerBioForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = Speaker
|
|
|
|
fields = ['avatar', 'title', 'bio', 'categories']
|
|
|
|
|
2018-04-25 16:36:01 +02:00
|
|
|
class DeleteSpeakerForm(forms.Form):
|
|
|
|
confirm = forms.BooleanField(label="Yes, delete series", required=True)
|
|
|
|
|
2018-04-24 16:22:16 +02:00
|
|
|
class UserTalkForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = Talk
|
|
|
|
fields = ['speaker', 'title', 'abstract', 'talk_type', 'web_url', 'category']
|
|
|
|
|
|
|
|
class DeleteTalkForm(forms.Form):
|
|
|
|
confirm = forms.BooleanField(label="Yes, delete series", required=True)
|
|
|
|
|
|
|
|
class SchedulePresentationForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = Presentation
|
|
|
|
fields = ['start_time']
|
|
|
|
|