from django.core.exceptions import ObjectDoesNotExist from django.contrib import admin from django.utils.translation import ugettext_lazy as _ from django.shortcuts import render from django.conf import settings from .models import Language, Type, MusicFocus, Category, Topic, RTRCategory, Host, Note, RRule, Schedule, Show, TimeSlot from .forms import MusicFocusForm, CollisionForm from datetime import date, datetime, time, timedelta class ActivityFilter(admin.SimpleListFilter): title = _("Activity") def lookups(self, request, model_admin): return ( ('yes', _("active")), ('no', _("inactive")) ) def queryset(self, request, queryset): if self.parameter_name == 'has_timeslots': # active/inactive Schedules if self.value() == 'yes': return queryset.filter(until__gt=datetime.now()).distinct() if self.value() == 'no': return queryset.filter(until__lt=datetime.now()).distinct() if self.parameter_name == 'has_schedules_timeslots': # active/inactive Shows if self.value() == 'yes': return queryset.filter(schedules__until__gt=datetime.now()).distinct() if self.value() == 'no': return queryset.filter(schedules__until__lt=datetime.now()).distinct() if self.parameter_name == 'has_shows_schedules_timeslots': # active/inactive Hosts if self.value() == 'yes': return queryset.filter(shows__schedules__until__gt=datetime.now()).distinct() if self.value() == 'no': return queryset.filter(shows__schedules__until__lt=datetime.now()).distinct() class ActiveSchedulesFilter(ActivityFilter): parameter_name = 'has_timeslots' class ActiveShowsFilter(ActivityFilter): parameter_name = 'has_schedules_timeslots' class ActiveHostsFilter(ActivityFilter): parameter_name = 'has_shows_schedules_timeslots' class TypeAdmin(admin.ModelAdmin): list_display = ('type', 'admin_color', 'enabled') prepopulated_fields = {'slug': ('type',)} class MusicFocusAdmin(admin.ModelAdmin): form = MusicFocusForm list_display = ('focus', 'abbrev', 'admin_buttons') prepopulated_fields = {'slug': ('focus',)} class CategoryAdmin(admin.ModelAdmin): list_display = ('category', 'abbrev', 'admin_buttons') prepopulated_fields = {'slug': ('category',)} class LanguageAdmin(admin.ModelAdmin): list_display = ('name',) class TopicAdmin(admin.ModelAdmin): list_display = ('topic', 'abbrev', 'admin_buttons') prepopulated_fields = {'slug': ('topic',)} class RTRCategoryAdmin(admin.ModelAdmin): list_display = ('rtrcategory', 'abbrev', ) prepopulated_fields = {'slug': ('rtrcategory',)} class HostAdmin(admin.ModelAdmin): list_display = ('name',) list_filter = (ActiveHostsFilter, 'is_always_visible',) class NoteAdmin(admin.ModelAdmin): date_hierarchy = 'start' list_display = ('title', 'show', 'start', 'status') fields = (( 'show', 'timeslot'), 'title', 'slug', 'summary', 'content', 'image', 'status', 'cba_id') prepopulated_fields = {'slug': ('title',)} list_filter = ('status',) ordering = ('timeslot',) save_as = True class Media: js = [ settings.MEDIA_URL + 'js/calendar/lib/moment.min.js', settings.MEDIA_URL + 'js/note_change.js', ] def get_queryset(self, request): if request.user.is_superuser: # Superusers see notes of all shows shows = Show.objects.all() else: # Users only see notes of shows they own shows = request.user.shows.all() return super(NoteAdmin, self).get_queryset(request).filter(show__in=shows) def formfield_for_foreignkey(self, db_field, request=None, **kwargs): four_weeks_ago = datetime.now() - timedelta(weeks=4) in_twelve_weeks = datetime.now() + timedelta(weeks=12) if db_field.name == 'timeslot': # Adding/Editing a note: load timeslots of the user's shows into the dropdown # TODO: Don't show any timeslot in the select by default. # User should first choose show, then timeslots are loaded into the select via ajax. # # How to do this while not constraining the queryset? # Saving won't be possible otherwise, if queryset doesn't contain the selectable elements beforehand #kwargs['queryset'] = TimeSlot.objects.filter(show=-1) # Superusers see every timeslot for every show if request.user.is_superuser: kwargs['queryset'] = TimeSlot.objects.filter(start__gt=four_weeks_ago, start__lt=in_twelve_weeks) # note__isnull=True # Users see timeslots of shows they own else: kwargs['queryset'] = TimeSlot.objects.filter(show__in=request.user.shows.all(), start__gt=four_weeks_ago, start__lt=in_twelve_weeks) # note__isnull=True if db_field.name == 'show': # Adding/Editing a note: load user's shows into the dropdown # Superusers see all shows if not request.user.is_superuser: kwargs['queryset'] = Show.objects.filter(pk__in=request.user.shows.all()) return super(NoteAdmin, self).formfield_for_foreignkey(db_field, request, **kwargs) def save_model(self, request, obj, form, change): # Save the creator when adding a note if not change: obj.user = request.user obj.save() class TimeSlotInline(admin.TabularInline): model = TimeSlot ordering = ('-end',) class TimeSlotAdmin(admin.ModelAdmin): model = TimeSlot class ScheduleAdmin(admin.ModelAdmin): actions = ('renew',) inlines = (TimeSlotInline,) fields = (('rrule', 'byweekday'), ('dstart', 'tstart', 'tend'), 'until', 'is_repetition', 'automation_id', 'fallback_playlist_id') list_display = ('get_show_name', 'byweekday', 'rrule', 'tstart', 'tend', 'until') list_filter = (ActiveSchedulesFilter, 'byweekday', 'rrule', 'is_repetition') ordering = ('byweekday', 'dstart') save_on_top = True search_fields = ('show__name',) def renew(self, request, queryset): next_year = date.today().year + 1 until = date(next_year, 12, 31) renewed = queryset.update(until=until) if renewed == 1: message = _("1 schedule was renewed until %s") % until else: message = _("%s schedule were renewed until %s") % (renewed, until) self.message_user(request, message) renew.short_description = _("Renew selected schedules") def get_show_name(self, obj): return obj.show.name get_show_name.admin_order_field = 'show' get_show_name.short_description = "Show" class ScheduleInline(admin.TabularInline): model = Schedule ordering = ('pk', '-until', 'byweekday') class ShowAdmin(admin.ModelAdmin): filter_horizontal = ('hosts', 'owners', 'musicfocus', 'category', 'topic', 'language') inlines = (ScheduleInline,) list_display = ('name', 'short_description') list_filter = (ActiveShowsFilter, 'type', 'category', 'topic', 'musicfocus', 'rtrcategory', 'language') ordering = ('slug',) prepopulated_fields = {'slug': ('name',)} search_fields = ('name', 'short_description', 'description') fields = ( 'predecessor', 'type', 'name', 'slug', 'image', 'logo', 'short_description', 'description', 'email', 'website', 'hosts', 'owners', 'language', 'category', 'rtrcategory', 'topic', 'musicfocus', 'fallback_pool', 'cba_series_id', ) def get_queryset(self, request): if request.user.is_superuser: # Superusers see all shows shows = Show.objects.all() else: # Users only see shows they own shows = request.user.shows.all() return super(ShowAdmin, self).get_queryset(request).filter(pk__in=shows) class Media: js = [ settings.MEDIA_URL + 'js/calendar/lib/moment.min.js', settings.MEDIA_URL + 'js/show_change.js', ] css = { 'all': ('/program/styles.css',) } def formfield_for_foreignkey(self, db_field, request=None, **kwargs): try: show_id = int(request.get_full_path().split('/')[-2]) except ValueError: show_id = None if db_field.name == 'predecessor' and show_id: kwargs['queryset'] = Show.objects.exclude(pk=show_id) return super(ShowAdmin, self).formfield_for_foreignkey(db_field, request, **kwargs) def save_formset(self, request, form, formset, change): """ Is called after the "save show"-form or collision-form were submitted Saves the show after first submit If any changes in schedules happened * added/changed schedules are used to generate new timeslots and matched against existing ones, which will be displayed in the collision form If a collision form was submitted * save the current schedule * delete/create timeslots and relink notes after confirmation Each step passes on to response_add or response_change which will * either display the collision form for the next step * or redirect to the original show-form if the resolving process has been finished (= if either max_steps was surpassed or end_reached was True) """ self.end_reached = False schedule_instances = formset.save(commit=False) # If there are no schedules to save, do nothing if schedule_instances: show_id = schedule_instances[0].show.id else: self.end_reached = True schedule = [] timeslots = [] max_steps = int(len(schedule_instances)) if len(schedule_instances) > 0 else 1 step = 1 if request.POST.get('step') == None: # First save-show submit # Generate thumbnails if form.instance.image.name and settings.THUMBNAIL_SIZES: for size in settings.THUMBNAIL_SIZES: thumbnail = form.instance.image.crop[size].name # Save show data only form.save(); # Delete schedules (as well as related timeslots and notes) if flagged as such for obj in formset.deleted_objects: obj.delete() # If nothing else changed, do nothing and redirect to show-form if not formset.changed_objects and not formset.new_objects: self.end_reached = True else: # If a collision form was submitted step = int(request.POST.get('step')) if request.POST.get('num_inputs') != None and int(request.POST.get('num_inputs')) > 0: print("Resolving conflicts...") '''Declare and retrieve variables''' # Either datetimes as string (e.g. '2017-01-01 00:00:00 - 2017-01-01 01:00:00') to create # or ints of colliding timeslots to keep otherwise resolved_timeslots = [] # IDs of colliding timeslots found in the db. If there's no corresponding collision to the # same index in create_timeslot, value will be None collisions = [] # Datetimes as string (e.g. '2017-01-01 00:00:00 - 2017-01-01 01:00:00') for timeslots to create create_timeslots = [] # IDs of timeslots to delete delete_timeslots = set() # Number of timeslots to be generated num_inputs = int(request.POST.get('num_inputs')) # Numbers of notes to relink for existing timeslots and newly created ones # each of them relating to one of these POST vars: # POST.ntids[idx][id] and POST.ntids[idx][note_id] contain ids of existing timeslots and note_ids to link, while # POST.ntind[idx][id] and POST.ntind[idx][note_id] contain indices of corresponding elements in create_timeslots # and note_ids which will be linked after they're created and thus split into two lists beforehand num_ntids = int(request.POST.get('num_ntids')) num_ntind = int(request.POST.get('num_ntind')) # Retrieve POST vars of current schedule schedule_id = int(request.POST.get('ps_save_id')) if request.POST.get('ps_save_id') != 'None' else None rrule = RRule.objects.get(pk=int(request.POST.get('ps_save_rrule_id'))) show = Show.objects.get(pk=show_id) byweekday = int(request.POST.get('ps_save_byweekday')) tstart = datetime.strptime(request.POST.get('ps_save_tstart'), '%H:%M').time() tend = datetime.strptime(request.POST.get('ps_save_tend'), '%H:%M').time() dstart = datetime.strptime(request.POST.get('ps_save_dstart'), '%Y-%m-%d').date() if dstart < datetime.today().date(): # Create or delete upcoming timeslots only dstart = datetime.today().date() until = datetime.strptime(request.POST.get('ps_save_until'), '%Y-%m-%d').date() is_repetition = request.POST.get('ps_save_is_repetition') automation_id = int(request.POST.get('ps_save_automation_id')) if request.POST.get('ps_save_automation_id') != 'None' else 0 fallback_playlist_id = int(request.POST.get('fallback_playlist_id')) if request.POST.get('ps_save_fallback_playlist_id') != 'None' else 0 # Put timeslot POST vars into lists with same indices for i in range(num_inputs): resolved_ts = request.POST.get('resolved_timeslots[' + str(i) + ']') if resolved_ts != None: resolved_timeslots.append( resolved_ts ) create_timeslots.append( request.POST.get('create_timeslots[' + str(i) + ']') ) # May contain None collisions.append( request.POST.get('collisions[' + str(i) + ']') ) # May contain None else: num_inputs -= 1 '''Prepare resolved timeslots''' # Separate timeslots to delete from those to create keep_collisions = [] for x in range(num_inputs): if resolved_timeslots[x] == None or resolved_timeslots[x].isdigit(): # If it's a digit, keep the existing timeslot by preventing the new one from being created create_timeslots[x] = None keep_collisions.append(int(collisions[x])) else: # Otherwise collect the timeslot ids to be deleted later if len(collisions[x]) > 0: delete_timeslots.add(int(collisions[x])) # Collect IDs of upcoming timeslots of the same schedule to delete except those in keep_collision if schedule_id != None: for ts in TimeSlot.objects.filter(start__gte=dstart,end__lte=until,schedule_id=schedule_id).exclude(pk__in=keep_collisions).values_list('id', flat=True): delete_timeslots.add(ts) '''Save schedule''' new_schedule = Schedule(pk=schedule_id, rrule=rrule, byweekday=byweekday, show=show, dstart=dstart, tstart=tstart, tend=tend, until=until, is_repetition=is_repetition, automation_id=automation_id, fallback_playlist_id=fallback_playlist_id) # Only save schedule if any timeslots changed if len(resolved_timeslots) > 0: new_schedule.save() '''Relink notes to existing timeslots and prepare those to be linked''' # Relink notes with existing timeslot ids for i in range(num_ntids): try: note = Note.objects.get(pk=int(request.POST.get('ntids[' + str(i) + '][note_id]'))) note.timeslot_id = int(request.POST.get('ntids[' + str(i) + '][id]')) note.save(update_fields=["timeslot_id"]) print("Rewrote note " + str(note.id) + "...to timeslot_id " + str(note.timeslot_id)) except ObjectDoesNotExist: pass # Put list indices of yet to be created timeslots and note_ids in corresponding lists to relink them during creation note_indices = [] note_ids = [] for i in range(num_ntind): note_indices.append( int(request.POST.get('ntind[' + str(i) + '][id]')) ) note_ids.append( int(request.POST.get('ntind[' + str(i) + '][note_id]')) ) '''Database changes for resolved timeslots and relinked notes for newly created''' for idx, ts in enumerate(create_timeslots): if ts != None: start_end = ts.split(' - ') # Only create upcoming timeslots if datetime.strptime(start_end[0], "%Y-%m-%d %H:%M:%S") > datetime.today(): timeslot_created = TimeSlot.objects.create(schedule=new_schedule, is_repetition=new_schedule.is_repetition, start=start_end[0], end=start_end[1]) # Link a note to the new timeslot if idx in note_indices: note_idx = note_indices.index( idx ) # Get the note_id's index... note_id = note_ids[note_idx] # ...which contains the note_id to relate to try: note = Note.objects.get(pk=note_id) note.timeslot_id = timeslot_created.id note.save(update_fields=["timeslot_id"]) print("Timeslot " + str(timeslot_created.id) + " linked to note " + str(note_id)) except ObjectDoesNotExist: pass # Finally delete discarded timeslots for timeslot_id in delete_timeslots: TimeSlot.objects.filter(pk=timeslot_id).delete() if step > max_steps: self.end_reached = True ''' Everything below here is called when a new collision is loaded before being handed over to the client ''' # Generate timeslots from current schedule k = 1 for instance in schedule_instances: if isinstance(instance, Schedule): if k == step: timeslots = Schedule.generate_timeslots(instance) schedule = instance break k += 1 # Get collisions for timeslots collisions = Schedule.get_collisions(timeslots) # Get notes of colliding timeslots notes = [] for id in collisions: try: notes.append( Note.objects.get(timeslot_id=id) ) except ObjectDoesNotExist: pass self.schedule = schedule self.timeslots = timeslots self.collisions = collisions self.num_collisions = len([ s for s in self.collisions if s != 'None']) # Number of real collisions displayed to the user self.notes = notes self.showform = form self.schedulesform = formset self.step = step + 1 # Becomes upcoming step self.max_steps = max_steps # Pass it on to response_add() or response_change() return self def response_add(self, request, obj): return ShowAdmin.respond(self, request, obj) def response_change(self, request, obj): return ShowAdmin.respond(self, request, obj) def respond(self, request, obj): """ Redirects to the show-change-form if no schedules changed or resolving has been finished (or any other form validation error occured) Displays the collision form for the current schedule otherwise """ if self.end_reached: return super(ShowAdmin, self).response_change(request, obj) timeslots_to_collisions = list(zip(self.timeslots, self.collisions)) # myform = CollisionForm(self.timeslots, self.collisions) return render(request, 'collisions.html', {'self' : self, 'obj': obj, 'request': request, 'timeslots': self.timeslots, 'collisions': self.collisions, 'schedule': self.schedule, 'timeslots_to_collisions': timeslots_to_collisions, 'schedulesform': self.schedulesform, 'showform': self.showform, 'num_inputs': len(self.timeslots), 'step': self.step, 'max_steps': self.max_steps, 'now': datetime.now(), 'num_collisions': self.num_collisions}) admin.site.register(Language, LanguageAdmin) admin.site.register(Type, TypeAdmin) admin.site.register(MusicFocus, MusicFocusAdmin) admin.site.register(Category, CategoryAdmin) admin.site.register(Topic, TopicAdmin) admin.site.register(RTRCategory, RTRCategoryAdmin) admin.site.register(Host, HostAdmin) admin.site.register(Note, NoteAdmin) #admin.site.register(Schedule, ScheduleAdmin) admin.site.register(TimeSlot, TimeSlotAdmin) admin.site.register(Show, ShowAdmin)