forked from rc/aircox
work on admin interface
This commit is contained in:
parent
21f3e89101
commit
83e425349b
51
cms/admin.py
51
cms/admin.py
|
@ -4,7 +4,6 @@ from django.utils.translation import ugettext as _, ugettext_lazy
|
||||||
import aircox.cms.models as models
|
import aircox.cms.models as models
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
class PostAdmin(admin.ModelAdmin):
|
class PostAdmin(admin.ModelAdmin):
|
||||||
list_display = [ 'title', 'date', 'author', 'published', 'post_tags']
|
list_display = [ 'title', 'date', 'author', 'published', 'post_tags']
|
||||||
list_editable = [ 'published' ]
|
list_editable = [ 'published' ]
|
||||||
|
@ -42,6 +41,56 @@ class CommentAdmin(admin.ModelAdmin):
|
||||||
return post.content[:256]
|
return post.content[:256]
|
||||||
content_slice.short_description = _('content')
|
content_slice.short_description = _('content')
|
||||||
|
|
||||||
|
|
||||||
|
class PostInline(admin.StackedInline):
|
||||||
|
extra = 1
|
||||||
|
max_num = 1
|
||||||
|
verbose_name = _('Post')
|
||||||
|
|
||||||
|
fieldsets = [
|
||||||
|
(None, {
|
||||||
|
'fields': ['title', 'content', 'image', 'tags']
|
||||||
|
}),
|
||||||
|
(None, {
|
||||||
|
'fields': ['date', 'published', 'author']
|
||||||
|
})
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
def inject_related_inline(post_model, prepend = False, inline = None):
|
||||||
|
"""
|
||||||
|
Create an inline class and inject it into the related model admin class.
|
||||||
|
Clean-up bound attributes.
|
||||||
|
"""
|
||||||
|
class InlineModel(PostInline):
|
||||||
|
model = post_model
|
||||||
|
verbose_name = _('Related post')
|
||||||
|
|
||||||
|
inline = inline or InlineModel
|
||||||
|
|
||||||
|
# remove bound attributes
|
||||||
|
for none, dic in inline.fieldsets:
|
||||||
|
if not dic.get('fields'):
|
||||||
|
continue
|
||||||
|
dic['fields'] = [ v for v in dic['fields']
|
||||||
|
if v not in post_model._relation.bindings.keys() ]
|
||||||
|
|
||||||
|
inject_inline(post_model._meta.get_field('related').rel.to,
|
||||||
|
inline, prepend)
|
||||||
|
|
||||||
|
def inject_inline(model, inline, prepend = False):
|
||||||
|
registry = admin.site._registry
|
||||||
|
if not model in registry:
|
||||||
|
return TypeError('{} not in admin registry'.format(model.__name__))
|
||||||
|
|
||||||
|
inlines = list(registry[model].inlines) or []
|
||||||
|
if prepend:
|
||||||
|
inlines.insert(0, inline)
|
||||||
|
else:
|
||||||
|
inlines.append(inline)
|
||||||
|
registry[model].inlines = inlines
|
||||||
|
|
||||||
|
|
||||||
admin.site.register(models.Article, PostAdmin)
|
admin.site.register(models.Article, PostAdmin)
|
||||||
admin.site.register(models.Comment, CommentAdmin)
|
admin.site.register(models.Comment, CommentAdmin)
|
||||||
|
|
||||||
|
|
|
@ -288,7 +288,8 @@ class RelatedPostBase (models.base.ModelBase):
|
||||||
post = model.objects.filter(related = instance)
|
post = model.objects.filter(related = instance)
|
||||||
if post.count():
|
if post.count():
|
||||||
post = post[0]
|
post = post[0]
|
||||||
elif rel.auto_create:
|
elif rel.auto_create(instance) if callable(rel.auto_create) else \
|
||||||
|
rel.auto_create:
|
||||||
post = model(related = instance)
|
post = model(related = instance)
|
||||||
else:
|
else:
|
||||||
return
|
return
|
||||||
|
@ -379,8 +380,8 @@ class RelatedPost (Post, metaclass = RelatedPostBase):
|
||||||
* field_args: dict of arguments to pass to the ForeignKey constructor,
|
* field_args: dict of arguments to pass to the ForeignKey constructor,
|
||||||
such as: ForeignKey(related_model, **field_args)
|
such as: ForeignKey(related_model, **field_args)
|
||||||
* auto_create: automatically create a RelatedPost for each new item of
|
* auto_create: automatically create a RelatedPost for each new item of
|
||||||
the related object and init it with bounded values. Use signals
|
the related object and init it with bounded values. Use 'post_save'
|
||||||
'', ''.
|
signal. If auto_create is callable, use `auto_create(related_object)`.
|
||||||
|
|
||||||
Be careful with post_to_rel!
|
Be careful with post_to_rel!
|
||||||
* There is no check of permissions when related object is synchronised
|
* There is no check of permissions when related object is synchronised
|
||||||
|
|
|
@ -11,22 +11,22 @@ from aircox.programs.models import *
|
||||||
#
|
#
|
||||||
# Inlines
|
# Inlines
|
||||||
#
|
#
|
||||||
class SoundInline (admin.TabularInline):
|
class SoundInline(admin.TabularInline):
|
||||||
model = Sound
|
model = Sound
|
||||||
|
|
||||||
|
|
||||||
class ScheduleInline (admin.TabularInline):
|
class ScheduleInline(admin.TabularInline):
|
||||||
model = Schedule
|
model = Schedule
|
||||||
extra = 1
|
extra = 1
|
||||||
|
|
||||||
class StreamInline (admin.TabularInline):
|
class StreamInline(admin.TabularInline):
|
||||||
fields = ['delay', 'begin', 'end']
|
fields = ['delay', 'begin', 'end']
|
||||||
model = Stream
|
model = Stream
|
||||||
extra = 1
|
extra = 1
|
||||||
|
|
||||||
|
|
||||||
# from suit.admin import SortableTabularInline, SortableModelAdmin
|
# from suit.admin import SortableTabularInline, SortableModelAdmin
|
||||||
#class TrackInline (SortableTabularInline):
|
#class TrackInline(SortableTabularInline):
|
||||||
# fields = ['artist', 'name', 'tags', 'position']
|
# fields = ['artist', 'name', 'tags', 'position']
|
||||||
# form = TrackForm
|
# form = TrackForm
|
||||||
# model = Track
|
# model = Track
|
||||||
|
@ -34,7 +34,7 @@ class StreamInline (admin.TabularInline):
|
||||||
# extra = 10
|
# extra = 10
|
||||||
|
|
||||||
|
|
||||||
class NameableAdmin (admin.ModelAdmin):
|
class NameableAdmin(admin.ModelAdmin):
|
||||||
fields = [ 'name' ]
|
fields = [ 'name' ]
|
||||||
|
|
||||||
list_display = ['id', 'name']
|
list_display = ['id', 'name']
|
||||||
|
@ -43,7 +43,7 @@ class NameableAdmin (admin.ModelAdmin):
|
||||||
|
|
||||||
|
|
||||||
@admin.register(Sound)
|
@admin.register(Sound)
|
||||||
class SoundAdmin (NameableAdmin):
|
class SoundAdmin(NameableAdmin):
|
||||||
fields = None
|
fields = None
|
||||||
list_display = ['id', 'name', 'duration', 'type', 'mtime', 'good_quality', 'removed', 'public']
|
list_display = ['id', 'name', 'duration', 'type', 'mtime', 'good_quality', 'removed', 'public']
|
||||||
fieldsets = [
|
fieldsets = [
|
||||||
|
@ -55,17 +55,17 @@ class SoundAdmin (NameableAdmin):
|
||||||
|
|
||||||
|
|
||||||
@admin.register(Stream)
|
@admin.register(Stream)
|
||||||
class StreamAdmin (admin.ModelAdmin):
|
class StreamAdmin(admin.ModelAdmin):
|
||||||
list_display = ('id', 'program', 'delay', 'begin', 'end')
|
list_display = ('id', 'program', 'delay', 'begin', 'end')
|
||||||
|
|
||||||
|
|
||||||
@admin.register(Station)
|
@admin.register(Station)
|
||||||
class StationAdmin (NameableAdmin):
|
class StationAdmin(NameableAdmin):
|
||||||
fields = NameableAdmin.fields + [ 'active', 'public', 'fallback' ]
|
fields = NameableAdmin.fields + [ 'active', 'public', 'fallback' ]
|
||||||
|
|
||||||
@admin.register(Program)
|
@admin.register(Program)
|
||||||
class ProgramAdmin (NameableAdmin):
|
class ProgramAdmin(NameableAdmin):
|
||||||
def schedule (self, obj):
|
def schedule(self, obj):
|
||||||
return Schedule.objects.filter(program = obj).count() > 0
|
return Schedule.objects.filter(program = obj).count() > 0
|
||||||
schedule.boolean = True
|
schedule.boolean = True
|
||||||
schedule.short_description = _("Schedule")
|
schedule.short_description = _("Schedule")
|
||||||
|
@ -76,7 +76,7 @@ class ProgramAdmin (NameableAdmin):
|
||||||
inlines = [ ScheduleInline, StreamInline ]
|
inlines = [ ScheduleInline, StreamInline ]
|
||||||
|
|
||||||
# SO#8074161
|
# SO#8074161
|
||||||
#def get_form (self, request, obj=None, **kwargs):
|
#def get_form(self, request, obj=None, **kwargs):
|
||||||
#if obj:
|
#if obj:
|
||||||
# if Schedule.objects.filter(program = obj).count():
|
# if Schedule.objects.filter(program = obj).count():
|
||||||
# self.inlines.remove(StreamInline)
|
# self.inlines.remove(StreamInline)
|
||||||
|
@ -84,22 +84,38 @@ class ProgramAdmin (NameableAdmin):
|
||||||
# self.inlines.remove(ScheduleInline)
|
# self.inlines.remove(ScheduleInline)
|
||||||
#return super().get_form(request, obj, **kwargs)
|
#return super().get_form(request, obj, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
|
class DiffusionInline(admin.StackedInline):
|
||||||
|
model = Diffusion
|
||||||
|
extra = 0
|
||||||
|
fields = ['type', 'start', 'end']
|
||||||
|
|
||||||
@admin.register(Diffusion)
|
@admin.register(Diffusion)
|
||||||
class DiffusionAdmin (admin.ModelAdmin):
|
class DiffusionAdmin(admin.ModelAdmin):
|
||||||
def archives (self, obj):
|
def archives(self, obj):
|
||||||
sounds = [ str(s) for s in obj.get_archives()]
|
sounds = [ str(s) for s in obj.get_archives()]
|
||||||
return ', '.join(sounds) if sounds else ''
|
return ', '.join(sounds) if sounds else ''
|
||||||
|
|
||||||
def conflicts (self, obj):
|
def conflicts(self, obj):
|
||||||
if obj.type == Diffusion.Type.unconfirmed:
|
if obj.type == Diffusion.Type.unconfirmed:
|
||||||
return ', '.join([ str(d) for d in obj.get_conflicts()])
|
return ', '.join([ str(d) for d in obj.get_conflicts()])
|
||||||
return ''
|
return ''
|
||||||
|
|
||||||
list_display = ('id', 'type', 'start', 'end', 'program', 'initial', 'archives', 'conflicts')
|
def end_time(self, obj):
|
||||||
|
return obj.end.strftime('%H:%M')
|
||||||
|
end_time.short_description = _('end')
|
||||||
|
|
||||||
|
def first(self, obj):
|
||||||
|
return obj.initial.start if obj.initial else ''
|
||||||
|
|
||||||
|
list_display = ('id', 'program', 'start', 'end_time', 'type', 'first', 'archives', 'conflicts')
|
||||||
list_filter = ('type', 'start', 'program')
|
list_filter = ('type', 'start', 'program')
|
||||||
list_editable = ('type', 'start', 'end')
|
list_editable = ('type',)
|
||||||
|
ordering = ('-start', 'id')
|
||||||
|
|
||||||
fields = ['type', 'start', 'end', 'initial', 'program', 'sounds']
|
fields = ['type', 'start', 'end', 'initial', 'program', 'sounds']
|
||||||
|
inlines = [ DiffusionInline ]
|
||||||
|
|
||||||
|
|
||||||
def get_form(self, request, obj=None, **kwargs):
|
def get_form(self, request, obj=None, **kwargs):
|
||||||
if request.user.has_perm('aircox_program.programming'):
|
if request.user.has_perm('aircox_program.programming'):
|
||||||
|
@ -111,32 +127,40 @@ class DiffusionAdmin (admin.ModelAdmin):
|
||||||
self.readonly_fields += ['program', 'sounds']
|
self.readonly_fields += ['program', 'sounds']
|
||||||
return super().get_form(request, obj, **kwargs)
|
return super().get_form(request, obj, **kwargs)
|
||||||
|
|
||||||
|
def get_object(self, *args, **kwargs):
|
||||||
|
"""
|
||||||
|
We want rerun to redirect to the given object.
|
||||||
|
"""
|
||||||
|
obj = super().get_object(*args, **kwargs)
|
||||||
|
if obj.initial:
|
||||||
|
obj = obj.initial
|
||||||
|
return obj
|
||||||
|
|
||||||
def get_queryset(self, request):
|
def get_queryset(self, request):
|
||||||
qs = super(DiffusionAdmin, self).get_queryset(request)
|
qs = super().get_queryset(request)
|
||||||
if ('_changelist_filters' in request.GET or \
|
if request.GET and '_changelist_filters' in request.GET and \
|
||||||
'type__exact' in request.GET) and \
|
request.GET.get('type__exact') == Diffusion.Type.unconfirmed:
|
||||||
str(Diffusion.Type.unconfirmed) in request.GET['type__exact']:
|
|
||||||
return qs
|
return qs
|
||||||
return qs.exclude(type = Diffusion.Type.unconfirmed)
|
return qs.exclude(type = Diffusion.Type.unconfirmed)
|
||||||
|
|
||||||
|
|
||||||
@admin.register(Log)
|
@admin.register(Log)
|
||||||
class LogAdmin (admin.ModelAdmin):
|
class LogAdmin(admin.ModelAdmin):
|
||||||
list_display = ['id', 'date', 'source', 'comment', 'related_object']
|
list_display = ['id', 'date', 'source', 'comment', 'related_object']
|
||||||
list_filter = ['date', 'source', 'related_type']
|
list_filter = ['date', 'source', 'related_type']
|
||||||
|
|
||||||
|
|
||||||
@admin.register(Schedule)
|
@admin.register(Schedule)
|
||||||
class ScheduleAdmin (admin.ModelAdmin):
|
class ScheduleAdmin(admin.ModelAdmin):
|
||||||
def program_name (self, obj):
|
def program_name(self, obj):
|
||||||
return obj.program.name
|
return obj.program.name
|
||||||
program_name.short_description = _('Program')
|
program_name.short_description = _('Program')
|
||||||
|
|
||||||
def day (self, obj):
|
def day(self, obj):
|
||||||
return obj.date.strftime('%A')
|
return obj.date.strftime('%A')
|
||||||
day.short_description = _('Day')
|
day.short_description = _('Day')
|
||||||
|
|
||||||
def rerun (self, obj):
|
def rerun(self, obj):
|
||||||
return obj.initial != None
|
return obj.initial != None
|
||||||
rerun.short_description = _('Rerun')
|
rerun.short_description = _('Rerun')
|
||||||
rerun.boolean = True
|
rerun.boolean = True
|
||||||
|
|
|
@ -7,19 +7,17 @@ import aircox.website.models as models
|
||||||
import aircox.website.forms as forms
|
import aircox.website.forms as forms
|
||||||
|
|
||||||
|
|
||||||
class TrackInline (SortableTabularInline):
|
class TrackInline(SortableTabularInline):
|
||||||
fields = ['artist', 'name', 'tags', 'position']
|
fields = ['artist', 'name', 'tags', 'position']
|
||||||
form = forms.TrackForm
|
form = forms.TrackForm
|
||||||
model = programs.Track
|
model = programs.Track
|
||||||
sortable = 'position'
|
sortable = 'position'
|
||||||
extra = 10
|
extra = 10
|
||||||
|
|
||||||
|
|
||||||
class DiffusionPostAdmin(cms.RelatedPostAdmin):
|
|
||||||
inlines = [TrackInline]
|
|
||||||
|
|
||||||
|
|
||||||
admin.site.register(models.Program, cms.RelatedPostAdmin)
|
admin.site.register(models.Program, cms.RelatedPostAdmin)
|
||||||
admin.site.register(models.Diffusion, DiffusionPostAdmin)
|
admin.site.register(models.Diffusion, cms.RelatedPostAdmin)
|
||||||
|
|
||||||
|
cms.inject_inline(programs.Diffusion, TrackInline, True)
|
||||||
|
cms.inject_related_inline(models.Diffusion, True)
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
from django import forms
|
from django import forms
|
||||||
|
|
||||||
import autocomplete_light.shortcuts as al
|
#import autocomplete_light.shortcuts as al
|
||||||
from autocomplete_light.contrib.taggit_field import TaggitWidget
|
#from autocomplete_light.contrib.taggit_field import TaggitWidget
|
||||||
|
|
||||||
import aircox.programs.models as programs
|
import aircox.programs.models as programs
|
||||||
|
|
||||||
|
@ -11,9 +11,9 @@ class TrackForm (forms.ModelForm):
|
||||||
model = programs.Track
|
model = programs.Track
|
||||||
fields = ['artist', 'name', 'tags', 'position']
|
fields = ['artist', 'name', 'tags', 'position']
|
||||||
widgets = {
|
widgets = {
|
||||||
'artist': al.TextWidget('TrackArtistAutocomplete'),
|
# 'artist': al.TextWidget('TrackArtistAutocomplete'),
|
||||||
'name': al.TextWidget('TrackNameAutocomplete'),
|
# 'name': al.TextWidget('TrackNameAutocomplete'),
|
||||||
'tags': TaggitWidget('TagAutocomplete'),
|
# 'tags': TaggitWidget('TagAutocomplete'),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -33,7 +33,9 @@ class Diffusion (RelatedPost):
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
rel_to_post = True
|
rel_to_post = True
|
||||||
auto_create = True
|
|
||||||
|
def auto_create(object):
|
||||||
|
return not object.initial
|
||||||
|
|
||||||
def __init__(self, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
super().__init__(*args, **kwargs)
|
super().__init__(*args, **kwargs)
|
||||||
|
|
Loading…
Reference in New Issue
Block a user