PonyConf/planning/utils.py

157 lines
6.1 KiB
Python
Raw Normal View History

2016-10-16 21:36:03 +00:00
from django.db.models import Q
2016-10-16 21:50:18 +00:00
from django.utils.safestring import mark_safe
from django.utils.html import escape
2016-10-16 21:36:03 +00:00
from django.utils.timezone import localtime
from datetime import datetime, timedelta
from copy import deepcopy
from collections import OrderedDict, namedtuple
from itertools import islice
from .models import Room
from proposals.models import Talk
Event = namedtuple('Event', ['talk', 'row', 'rowcount'])
class Program:
2016-10-18 20:54:36 +00:00
def __init__(self, site, empty_rooms=False, talk_filter=None):
2016-10-16 21:36:03 +00:00
self.talks = Talk.objects.\
2016-10-17 12:22:34 +00:00
filter(site=site, room__isnull=False, start_date__isnull=False).\
2016-10-16 21:36:03 +00:00
filter(Q(duration__gt=0) | Q(event__duration__gt=0)).\
exclude(accepted=False).\
order_by('start_date')
2016-10-18 20:54:36 +00:00
if talk_filter:
self.talks = self.talks.filter(talk_filter)
2016-10-17 12:22:34 +00:00
if empty_rooms:
self.rooms = Room.objects.filter(site=site)
else:
self.rooms = Room.objects.filter(talk__in=self.talks.all()).order_by('name').distinct()
2016-10-18 21:52:33 +00:00
self.days = {}
2016-10-16 21:36:03 +00:00
for talk in self.talks.all():
duration = talk.estimated_duration()
assert(duration)
2016-10-18 21:52:33 +00:00
dt1 = talk.start_date
d1 = localtime(dt1).date()
if d1 not in self.days.keys():
self.days[d1] = {'timeslots': []}
dt2 = dt1 + timedelta(minutes=duration)
d2 = localtime(dt2).date()
if d2 not in self.days.keys():
self.days[d2] = {'timeslots': []}
if dt1 not in self.days[d1]['timeslots']:
self.days[d1]['timeslots'].append(dt1)
if dt2 not in self.days[d2]['timeslots']:
self.days[d2]['timeslots'].append(dt2)
2016-10-16 21:36:03 +00:00
self.cols = OrderedDict([(room, 1) for room in self.rooms])
2016-10-18 21:52:33 +00:00
for day in self.days.keys():
self.days[day]['timeslots'] = sorted(self.days[day]['timeslots'])
self.days[day]['rows'] = OrderedDict([(timeslot, OrderedDict([(room, []) for room in self.rooms])) for timeslot in self.days[day]['timeslots'][:-1]])
2016-10-16 21:36:03 +00:00
2016-10-16 21:49:18 +00:00
for talk in self.talks.all():
2016-10-16 21:36:03 +00:00
self._add_talk(talk)
def _add_talk(self, talk):
room = talk.room
2016-10-18 21:52:33 +00:00
dt1 = talk.start_date
d1 = localtime(dt1).date()
dt2 = talk.start_date + timedelta(minutes=talk.estimated_duration())
d2 = localtime(dt2).date()
assert(d1 == d2) # this is a current limitation
dt1 = self.days[d1]['timeslots'].index(dt1)
dt2 = self.days[d1]['timeslots'].index(dt2)
2016-10-16 21:36:03 +00:00
col = None
2016-10-18 21:52:33 +00:00
for row, timeslot in enumerate(islice(self.days[d1]['timeslots'], dt1, dt2)):
2016-10-16 21:36:03 +00:00
if col is None:
col = 0
2016-10-18 21:52:33 +00:00
while col < len(self.days[d1]['rows'][timeslot][room]) and self.days[d1]['rows'][timeslot][room][col]:
2016-10-16 21:36:03 +00:00
col += 1
self.cols[room] = max(self.cols[room], col+1)
2016-10-18 21:52:33 +00:00
event = Event(talk=talk, row=row, rowcount=dt2-dt1)
while len(self.days[d1]['rows'][timeslot][room]) <= col:
self.days[d1]['rows'][timeslot][room].append(None)
self.days[d1]['rows'][timeslot][room][col] = event
2016-10-16 21:36:03 +00:00
def _header(self):
output = '<td>Room</td>'
room_cell = '<td%(options)s>%(name)s<br><b>%(label)s</b></td>'
for room, colspan in self.cols.items():
2016-10-16 21:55:27 +00:00
options = ' style="min-width: 100px;" colspan="%d"' % colspan
2016-10-16 21:49:18 +00:00
output += room_cell % {'name': escape(room.name), 'label': escape(room.label), 'options': options}
2016-10-16 21:36:03 +00:00
return '<tr>%s</tr>' % output
def _body(self):
2016-10-18 21:52:33 +00:00
output = ''
for day in self.days.keys():
output += self._day_header(day)
output += self._day(day)
return output
def _day_header(self, day):
row = '<tr><td colspan="%(colcount)s"><h3>%(day)s</h3></td></tr>'
colcount = 1
for room, col in self.cols.items():
colcount += col
return row % {
'colcount': colcount,
'day': datetime.strftime(day, '%A %d %B'),
}
def _day(self, day):
2016-10-16 21:36:03 +00:00
output = []
2016-10-18 21:52:33 +00:00
rows = self.days[day]['rows']
for ts, rooms in rows.items():
output.append(self._row(day, ts, rooms))
2016-10-16 21:36:03 +00:00
return '\n'.join(output)
2016-10-18 21:52:33 +00:00
def _row(self, day, ts, rooms):
row = '<tr style="%(style)s">%(timeslot)s%(content)s</tr>'
cell = '<td%(options)s>%(content)s</td>'
content = ''
for room, events in rooms.items():
colspan = 1
for i in range(self.cols[room]):
options = ' colspan="%d"' % colspan
cellcontent = ''
if i < len(events) and events[i]:
event = events[i]
if event.row != 0:
continue
options = ' rowspan="%d" bgcolor="%s"' % (event.rowcount, event.talk.event.color)
cellcontent = str(event.talk) + '' + event.talk.get_speakers_str()
elif (i+1 > len(events) or not events[i+1]) and i+1 < self.cols[room]:
colspan += 1
continue
colspan = 1
content += cell % {'options': options, 'content': escape(cellcontent)}
style, timeslot = self._timeslot(day, ts)
return row % {
'style': style,
'timeslot': timeslot,
'content': content,
}
def _timeslot(self, day, ts):
2016-10-16 21:36:03 +00:00
template = '<td>%(content)s</td>'
start = ts
2016-10-18 21:52:33 +00:00
end = self.days[day]['timeslots'][self.days[day]['timeslots'].index(ts)+1]
2016-10-16 21:36:03 +00:00
duration = (end - start).seconds / 60
print(start, end, duration)
date_to_string = lambda date: datetime.strftime(localtime(date), '%H:%M')
2016-10-16 21:49:18 +00:00
style = 'height: %dpx;' % int(duration * 1.2)
2016-10-16 21:36:03 +00:00
timeslot = '<td>%s %s</td>' % tuple(map(date_to_string, [start, end]))
return style, timeslot
def __str__(self):
template = """<table class="table table-bordered text-center">\n%(header)s\n%(body)s\n</table>"""
return mark_safe(template % {
'header': self._header(),
'body': self._body(),
})