symposion_app/vendor/symposion/schedule/timetable.py

60 lines
1.9 KiB
Python
Raw Normal View History

2012-08-30 06:51:07 +00:00
import itertools
2012-09-20 02:48:54 +00:00
from django.db.models import Count, Min
from symposion.schedule.models import Room, Slot, SlotRoom
2012-08-30 06:52:50 +00:00
2012-08-30 06:51:07 +00:00
2012-08-30 06:53:31 +00:00
class TimeTable(object):
2014-07-30 18:19:26 +00:00
2012-08-30 06:51:07 +00:00
def __init__(self, day):
self.day = day
self.slots_qs = Slot.objects.filter(day=day)\
.select_related('kind', 'content_ptr__speaker__user')\
.prefetch_related('content_ptr__additional_speakers')
self._times = sorted(
set(itertools.chain(*self.slots_qs.values_list("start", "end")))
)
def slots(self):
if not hasattr(self, '_slots'):
filters = {
"room_count": Count("slotroom"),
"order": Min("slotroom__room__order")
}
self._slots = self.slots_qs.annotate(**filters).order_by("order")
return self._slots
2014-07-30 18:19:26 +00:00
2012-08-30 06:51:07 +00:00
def rooms(self):
if not hasattr(self, '_rooms'):
self._rooms = Room.objects\
.filter(slotroom__slot__in=self.slots_qs)\
.distinct()\
.order_by("order")
return self._rooms
2014-07-30 18:19:26 +00:00
2012-08-30 06:51:07 +00:00
def __iter__(self):
slots = self.slots()
2012-08-30 06:51:07 +00:00
row = []
total_room_count = self.rooms().count()
for time, next_time in pairwise(self._times):
row = {"time": time, "end": next_time, "slots": []}
row_slots = [ slot for slot in slots if slot.start == time]
for slot in row_slots:
slot.rowspan = TimeTable.rowspan(self._times, slot.start, slot.end)
slot.colspan = slot.room_count if not slot.exclusive else total_room_count
row["slots"].append(slot)
2012-08-30 06:51:07 +00:00
if row["slots"] or next_time is None:
yield row
2014-07-30 18:19:26 +00:00
2012-08-30 06:51:07 +00:00
@staticmethod
def rowspan(times, start, end):
return times.index(end) - times.index(start)
2012-08-30 06:51:07 +00:00
def pairwise(iterable):
a, b = itertools.tee(iterable)
next(b)
return itertools.zip_longest(a, b)