forked from rc/aircox
		
	bkp before branching
This commit is contained in:
		@ -18,13 +18,6 @@ class StreamInline(admin.TabularInline):
 | 
			
		||||
    model = Stream
 | 
			
		||||
    extra = 1
 | 
			
		||||
 | 
			
		||||
class NameableAdmin(admin.ModelAdmin):
 | 
			
		||||
    fields = [ 'name' ]
 | 
			
		||||
 | 
			
		||||
    list_display = ['id', 'name']
 | 
			
		||||
    list_filter = []
 | 
			
		||||
    search_fields = ['name',]
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@admin.register(Stream)
 | 
			
		||||
class StreamAdmin(admin.ModelAdmin):
 | 
			
		||||
@ -32,15 +25,19 @@ class StreamAdmin(admin.ModelAdmin):
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@admin.register(Program)
 | 
			
		||||
class ProgramAdmin(NameableAdmin):
 | 
			
		||||
class ProgramAdmin(admin.ModelAdmin):
 | 
			
		||||
    def schedule(self, obj):
 | 
			
		||||
        return Schedule.objects.filter(program = obj).count() > 0
 | 
			
		||||
        return Schedule.objects.filter(program=obj).count() > 0
 | 
			
		||||
 | 
			
		||||
    schedule.boolean = True
 | 
			
		||||
    schedule.short_description = _("Schedule")
 | 
			
		||||
 | 
			
		||||
    list_display = ('id', 'name', 'active', 'schedule', 'sync', 'station')
 | 
			
		||||
    fields = NameableAdmin.fields + [ 'active', 'station','sync' ]
 | 
			
		||||
    inlines = [ ScheduleInline, StreamInline ]
 | 
			
		||||
    list_display = ('name', 'id', 'active', 'schedule', 'sync', 'station')
 | 
			
		||||
    fields = ['name', 'slug', 'active', 'station', 'sync']
 | 
			
		||||
    prepopulated_fields = {'slug': ('name',)}
 | 
			
		||||
    search_fields = ['name']
 | 
			
		||||
 | 
			
		||||
    inlines = [ScheduleInline, StreamInline]
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@admin.register(Schedule)
 | 
			
		||||
@ -64,7 +61,6 @@ class ScheduleAdmin(admin.ModelAdmin):
 | 
			
		||||
                    'time', 'duration', 'timezone', 'rerun']
 | 
			
		||||
    list_editable = ['time', 'timezone', 'duration']
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    def get_readonly_fields(self, request, obj=None):
 | 
			
		||||
        if obj:
 | 
			
		||||
            return ['program', 'date', 'frequency']
 | 
			
		||||
@ -79,6 +75,7 @@ class PortInline(admin.StackedInline):
 | 
			
		||||
 | 
			
		||||
@admin.register(Station)
 | 
			
		||||
class StationAdmin(admin.ModelAdmin):
 | 
			
		||||
    prepopulated_fields = {'slug': ('name',)}
 | 
			
		||||
    inlines = [PortInline]
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -2,19 +2,17 @@ from django.contrib import admin
 | 
			
		||||
from django.utils.translation import ugettext as _, ugettext_lazy
 | 
			
		||||
 | 
			
		||||
from aircox.models import Sound
 | 
			
		||||
from .base import NameableAdmin
 | 
			
		||||
from .playlist import TracksInline
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@admin.register(Sound)
 | 
			
		||||
class SoundAdmin(NameableAdmin):
 | 
			
		||||
class SoundAdmin(admin.ModelAdmin):
 | 
			
		||||
    fields = None
 | 
			
		||||
    list_display = ['id', 'name', 'program', 'type', 'duration', 'mtime',
 | 
			
		||||
                    'public', 'good_quality', 'path']
 | 
			
		||||
    list_filter = ('program', 'type', 'good_quality', 'public')
 | 
			
		||||
    fieldsets = [
 | 
			
		||||
        (None, {'fields': NameableAdmin.fields +
 | 
			
		||||
                          ['path', 'type', 'program', 'diffusion']}),
 | 
			
		||||
        (None, {'fields': ['name', 'path', 'type', 'program', 'diffusion']}),
 | 
			
		||||
        (None, {'fields': ['embed', 'duration', 'public', 'mtime']}),
 | 
			
		||||
        (None, {'fields': ['good_quality']})
 | 
			
		||||
    ]
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										135
									
								
								aircox/models.py
									
									
									
									
									
								
							
							
						
						
									
										135
									
								
								aircox/models.py
									
									
									
									
									
								
							@ -11,9 +11,9 @@ from django.contrib.contenttypes.fields import (GenericForeignKey,
 | 
			
		||||
                                                GenericRelation)
 | 
			
		||||
from django.contrib.contenttypes.models import ContentType
 | 
			
		||||
from django.db import models
 | 
			
		||||
from django.db.models import Q
 | 
			
		||||
from django.db.models import F, Q
 | 
			
		||||
from django.db.models.functions import Concat, Substr
 | 
			
		||||
from django.db.transaction import atomic
 | 
			
		||||
from django.template.defaultfilters import slugify
 | 
			
		||||
from django.utils import timezone as tz
 | 
			
		||||
from django.utils.functional import cached_property
 | 
			
		||||
from django.utils.html import strip_tags
 | 
			
		||||
@ -26,28 +26,6 @@ from taggit.managers import TaggableManager
 | 
			
		||||
logger = logging.getLogger('aircox.core')
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class Nameable(models.Model):
 | 
			
		||||
    name = models.CharField(
 | 
			
		||||
        _('name'),
 | 
			
		||||
        max_length=128,
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    class Meta:
 | 
			
		||||
        abstract = True
 | 
			
		||||
 | 
			
		||||
    @property
 | 
			
		||||
    def slug(self):
 | 
			
		||||
        """
 | 
			
		||||
        Slug based on the name. We replace '-' by '_'
 | 
			
		||||
        """
 | 
			
		||||
        return slugify(self.name).replace('-', '_')
 | 
			
		||||
 | 
			
		||||
    def __str__(self):
 | 
			
		||||
        # if self.pk:
 | 
			
		||||
        #    return '#{} {}'.format(self.pk, self.name)
 | 
			
		||||
        return '{}'.format(self.name)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#
 | 
			
		||||
# Station related classes
 | 
			
		||||
#
 | 
			
		||||
@ -67,7 +45,7 @@ def default_station():
 | 
			
		||||
    return Station.objects.default()
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class Station(Nameable):
 | 
			
		||||
class Station(models.Model):
 | 
			
		||||
    """
 | 
			
		||||
    Represents a radio station, to which multiple programs are attached
 | 
			
		||||
    and that is used as the top object for everything.
 | 
			
		||||
@ -76,6 +54,8 @@ class Station(Nameable):
 | 
			
		||||
    Theses are set up when needed (at the first access to these elements)
 | 
			
		||||
    then cached.
 | 
			
		||||
    """
 | 
			
		||||
    name = models.CharField(_('name'), max_length=64)
 | 
			
		||||
    slug = models.SlugField(_('slug'), max_length=64, unique=True)
 | 
			
		||||
    path = models.CharField(
 | 
			
		||||
        _('path'),
 | 
			
		||||
        help_text=_('path to the working directory'),
 | 
			
		||||
@ -199,6 +179,9 @@ class Station(Nameable):
 | 
			
		||||
            logs = logs[:count]
 | 
			
		||||
        return logs
 | 
			
		||||
 | 
			
		||||
    def __str__(self):
 | 
			
		||||
        return self.name
 | 
			
		||||
 | 
			
		||||
    def save(self, make_sources=True, *args, **kwargs):
 | 
			
		||||
        if not self.path:
 | 
			
		||||
            self.path = os.path.join(
 | 
			
		||||
@ -223,7 +206,7 @@ class ProgramManager(models.Manager):
 | 
			
		||||
        return qs.filter(station=station, **kwargs)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class Program(Nameable):
 | 
			
		||||
class Program(models.Model):
 | 
			
		||||
    """
 | 
			
		||||
    A Program can either be a Streamed or a Scheduled program.
 | 
			
		||||
 | 
			
		||||
@ -241,6 +224,8 @@ class Program(Nameable):
 | 
			
		||||
        verbose_name=_('station'),
 | 
			
		||||
        on_delete=models.CASCADE,
 | 
			
		||||
    )
 | 
			
		||||
    name = models.CharField(_('name'), max_length=64)
 | 
			
		||||
    slug = models.SlugField(_('slug'), max_length=64, unique=True)
 | 
			
		||||
    active = models.BooleanField(
 | 
			
		||||
        _('active'),
 | 
			
		||||
        default=True,
 | 
			
		||||
@ -254,14 +239,10 @@ class Program(Nameable):
 | 
			
		||||
 | 
			
		||||
    objects = ProgramManager()
 | 
			
		||||
 | 
			
		||||
    # TODO: use unique slug
 | 
			
		||||
    @property
 | 
			
		||||
    def path(self):
 | 
			
		||||
        """
 | 
			
		||||
        Return the path to the programs directory
 | 
			
		||||
        """
 | 
			
		||||
        return os.path.join(settings.AIRCOX_PROGRAMS_DIR,
 | 
			
		||||
                            self.slug + '_' + str(self.id))
 | 
			
		||||
        """ Return program's directory path """
 | 
			
		||||
        return os.path.join(settings.AIRCOX_PROGRAMS_DIR, self.slug)
 | 
			
		||||
 | 
			
		||||
    def ensure_dir(self, subdir=None):
 | 
			
		||||
        """
 | 
			
		||||
@ -299,26 +280,8 @@ class Program(Nameable):
 | 
			
		||||
    def __init__(self, *kargs, **kwargs):
 | 
			
		||||
        super().__init__(*kargs, **kwargs)
 | 
			
		||||
 | 
			
		||||
        if self.name:
 | 
			
		||||
            self.__original_path = self.path
 | 
			
		||||
 | 
			
		||||
    def save(self, *kargs, **kwargs):
 | 
			
		||||
        super().save(*kargs, **kwargs)
 | 
			
		||||
 | 
			
		||||
        if hasattr(self, '__original_path') and \
 | 
			
		||||
                self.__original_path != self.path and \
 | 
			
		||||
                os.path.exists(self.__original_path) and \
 | 
			
		||||
                not os.path.exists(self.path):
 | 
			
		||||
            logger.info('program #%s\'s name changed to %s. Change dir name',
 | 
			
		||||
                        self.id, self.name)
 | 
			
		||||
            shutil.move(self.__original_path, self.path)
 | 
			
		||||
 | 
			
		||||
            sounds = Sound.objects.filter(
 | 
			
		||||
                path__startswith=self.__original_path)
 | 
			
		||||
 | 
			
		||||
            for sound in sounds:
 | 
			
		||||
                sound.path.replace(self.__original_path, self.path)
 | 
			
		||||
                sound.save()
 | 
			
		||||
        if self.slug:
 | 
			
		||||
            self.__initial_path = self.path
 | 
			
		||||
 | 
			
		||||
    @classmethod
 | 
			
		||||
    def get_from_path(cl, path):
 | 
			
		||||
@ -346,6 +309,23 @@ class Program(Nameable):
 | 
			
		||||
    def is_show(self):
 | 
			
		||||
        return self.schedule_set.count() != 0
 | 
			
		||||
 | 
			
		||||
    def __str__(self):
 | 
			
		||||
        return self.name
 | 
			
		||||
 | 
			
		||||
    def save(self, *kargs, **kwargs):
 | 
			
		||||
        super().save(*kargs, **kwargs)
 | 
			
		||||
 | 
			
		||||
        path_ = getattr(self, '__initial_path', None)
 | 
			
		||||
        if path_ is not None and path_ != self.path and \
 | 
			
		||||
                os.path.exists(path_) and not os.path.exists(self.path):
 | 
			
		||||
            logger.info('program #%s\'s dir changed to %s - update it.',
 | 
			
		||||
                        self.id, self.name)
 | 
			
		||||
 | 
			
		||||
            shutil.move(path_, self.path)
 | 
			
		||||
            Sound.objects.filter(path__startswith=path_) \
 | 
			
		||||
                 .update(path=Concat('path', Substr(F('path'), len(path_))))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class Stream(models.Model):
 | 
			
		||||
    """
 | 
			
		||||
@ -427,15 +407,15 @@ class Schedule(models.Model):
 | 
			
		||||
        _('frequency'),
 | 
			
		||||
        choices=[(int(y), {
 | 
			
		||||
            'ponctual': _('ponctual'),
 | 
			
		||||
            'first': _('first week of the month'),
 | 
			
		||||
            'second': _('second week of the month'),
 | 
			
		||||
            'third': _('third week of the month'),
 | 
			
		||||
            'fourth': _('fourth week of the month'),
 | 
			
		||||
            'last': _('last week of the month'),
 | 
			
		||||
            'first_and_third': _('first and third weeks of the month'),
 | 
			
		||||
            'second_and_fourth': _('second and fourth weeks of the month'),
 | 
			
		||||
            'every': _('every week'),
 | 
			
		||||
            'one_on_two': _('one week on two'),
 | 
			
		||||
            'first': _('1st {day} of the month'),
 | 
			
		||||
            'second': _('2nd {day} of the month'),
 | 
			
		||||
            'third': _('3rd {day} of the month'),
 | 
			
		||||
            'fourth': _('4th {day} of the month'),
 | 
			
		||||
            'last': _('last {day} of the month'),
 | 
			
		||||
            'first_and_third': _('1st and 3rd {day}s of the month'),
 | 
			
		||||
            'second_and_fourth': _('2nd and 4th {day}s of the month'),
 | 
			
		||||
            'every': _('{day}'),
 | 
			
		||||
            'one_on_two': _('one {day} on two'),
 | 
			
		||||
        }[x]) for x, y in Frequency.__members__.items()],
 | 
			
		||||
    )
 | 
			
		||||
    initial = models.ForeignKey(
 | 
			
		||||
@ -454,11 +434,22 @@ class Schedule(models.Model):
 | 
			
		||||
 | 
			
		||||
        return pytz.timezone(self.timezone)
 | 
			
		||||
 | 
			
		||||
    @property
 | 
			
		||||
    def datetime(self):
 | 
			
		||||
        """ Datetime for this schedule (timezone unaware) """
 | 
			
		||||
        import datetime
 | 
			
		||||
        return datetime.datetime.combine(self.date, self.time)
 | 
			
		||||
    @cached_property
 | 
			
		||||
    def start(self):
 | 
			
		||||
        """ Datetime of the start (timezone unaware) """
 | 
			
		||||
        return tz.datetime.combine(self.date, self.time)
 | 
			
		||||
 | 
			
		||||
    @cached_property
 | 
			
		||||
    def end(self):
 | 
			
		||||
        """ Datetime of the end """
 | 
			
		||||
        return self.start + utils.to_timedelta(self.duration)
 | 
			
		||||
 | 
			
		||||
    def get_frequency_verbose(self):
 | 
			
		||||
        """ Return frequency formated for display """
 | 
			
		||||
        from django.template.defaultfilters import date
 | 
			
		||||
        return self.get_frequency_display().format(
 | 
			
		||||
            day=date(self.date, 'l')
 | 
			
		||||
        )
 | 
			
		||||
 | 
			
		||||
    # initial cached data
 | 
			
		||||
    __initial = None
 | 
			
		||||
@ -630,7 +621,7 @@ class Schedule(models.Model):
 | 
			
		||||
 | 
			
		||||
        delta = None
 | 
			
		||||
        if self.initial:
 | 
			
		||||
            delta = self.datetime - self.initial.datetime
 | 
			
		||||
            delta = self.start - self.initial.start
 | 
			
		||||
 | 
			
		||||
        # FIXME: daylight saving bug: delta misses an hour when diffusion and
 | 
			
		||||
        #        rerun are not on the same daylight-saving timezone
 | 
			
		||||
@ -916,9 +907,12 @@ class Diffusion(models.Model):
 | 
			
		||||
                self.check_conflicts()
 | 
			
		||||
 | 
			
		||||
    def __str__(self):
 | 
			
		||||
        return '{self.program.name} {date} #{self.pk}'.format(
 | 
			
		||||
            self=self, date=self.local_start.strftime('%Y/%m/%d %H:%M%z')
 | 
			
		||||
        str_ = '{self.program.name} {date}'.format(
 | 
			
		||||
            self=self, date=self.local_start.strftime('%Y/%m/%d %H:%M%z'),
 | 
			
		||||
        )
 | 
			
		||||
        if self.initial:
 | 
			
		||||
            str_ += ' ({})'.format(_('rerun'))
 | 
			
		||||
        return str_
 | 
			
		||||
 | 
			
		||||
    class Meta:
 | 
			
		||||
        verbose_name = _('Diffusion')
 | 
			
		||||
@ -928,7 +922,7 @@ class Diffusion(models.Model):
 | 
			
		||||
        )
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class Sound(Nameable):
 | 
			
		||||
class Sound(models.Model):
 | 
			
		||||
    """
 | 
			
		||||
    A Sound is the representation of a sound file that can be either an excerpt
 | 
			
		||||
    or a complete archive of the related diffusion.
 | 
			
		||||
@ -939,6 +933,7 @@ class Sound(Nameable):
 | 
			
		||||
        excerpt = 0x02,
 | 
			
		||||
        removed = 0x03,
 | 
			
		||||
 | 
			
		||||
    name = models.CharField(_('name'), max_length=64)
 | 
			
		||||
    program = models.ForeignKey(
 | 
			
		||||
        Program,
 | 
			
		||||
        verbose_name=_('program'),
 | 
			
		||||
 | 
			
		||||
@ -4,8 +4,6 @@ from django.contrib import admin
 | 
			
		||||
from django.utils.translation import ugettext_lazy as _
 | 
			
		||||
 | 
			
		||||
from content_editor.admin import ContentEditor, ContentEditorInline
 | 
			
		||||
from feincms3 import plugins
 | 
			
		||||
from feincms3.admin import TreeAdmin
 | 
			
		||||
 | 
			
		||||
from aircox import models as aircox
 | 
			
		||||
from . import models
 | 
			
		||||
@ -38,39 +36,21 @@ class PageDiffusionPlaylist(UnrelatedInlineMixin, TracksInline):
 | 
			
		||||
        view_obj.save()
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@admin.register(models.Page)
 | 
			
		||||
class PageAdmin(admin.ModelAdmin):
 | 
			
		||||
    list_display = ["title", "parent", "status"]
 | 
			
		||||
    list_editable = ['status']
 | 
			
		||||
    prepopulated_fields = {"slug": ("title",)}
 | 
			
		||||
 | 
			
		||||
    fieldsets = (
 | 
			
		||||
        (_('Main'), {
 | 
			
		||||
            'fields': ['title', 'slug']
 | 
			
		||||
        }),
 | 
			
		||||
        (_('Settings'), {
 | 
			
		||||
            'fields': ['status', 'static_path', 'path'],
 | 
			
		||||
        }),
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@admin.register(models.Article)
 | 
			
		||||
class ArticleAdmin(ContentEditor, PageAdmin):
 | 
			
		||||
class ArticleAdmin(ContentEditor):
 | 
			
		||||
    fieldsets = (
 | 
			
		||||
        (_('Main'), {
 | 
			
		||||
            'fields': ['title', 'slug', 'as_program', 'cover', 'headline'],
 | 
			
		||||
            'fields': ['title', 'slug', 'cover', 'headline'],
 | 
			
		||||
            'classes': ('tabbed', 'uncollapse')
 | 
			
		||||
        }),
 | 
			
		||||
        (_('Settings'), {
 | 
			
		||||
            'fields': ['featured', 'allow_comments',
 | 
			
		||||
                       'status', 'static_path', 'path'],
 | 
			
		||||
            'fields': ['featured', 'as_program', 'allow_comments', 'status'],
 | 
			
		||||
            'classes': ('tabbed',)
 | 
			
		||||
        }),
 | 
			
		||||
        #(_('Infos'), {
 | 
			
		||||
        #    'fields': ['diffusion'],
 | 
			
		||||
        #    'classes': ('tabbed',)
 | 
			
		||||
        #}),
 | 
			
		||||
    )
 | 
			
		||||
    list_display = ["title", "parent", "status"]
 | 
			
		||||
    list_editable = ['status']
 | 
			
		||||
    prepopulated_fields = {"slug": ("title",)}
 | 
			
		||||
 | 
			
		||||
    inlines = [
 | 
			
		||||
        ContentEditorInline.create(models.ArticleRichText),
 | 
			
		||||
 | 
			
		||||
@ -27,18 +27,26 @@ $body-background-color: $light;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/** page **/
 | 
			
		||||
img.cover {
 | 
			
		||||
    border: 0.2em black solid;
 | 
			
		||||
.page {
 | 
			
		||||
    .header {
 | 
			
		||||
        margin-bottom: 1.5em;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .headline {
 | 
			
		||||
        font-size: 1.4em;
 | 
			
		||||
        padding: 0.2em 0em;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .cover {
 | 
			
		||||
        float: right;
 | 
			
		||||
        max-width: 40%;
 | 
			
		||||
        margin: 1em;
 | 
			
		||||
        border: 0.2em black solid;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    p {
 | 
			
		||||
        padding: 0.4em 0em;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
.headline {
 | 
			
		||||
    font-size: 1.2em;
 | 
			
		||||
    padding: 0.2em 0em;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
img.cover {
 | 
			
		||||
    float: right;
 | 
			
		||||
    max-width: 40%;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -23,27 +23,26 @@ class PagePathConverter(StringConverter):
 | 
			
		||||
        return mark_safe(value)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#class WeekConverter:
 | 
			
		||||
#    """ Converter for date as YYYYY/WW """
 | 
			
		||||
#    regex = r'[0-9]{4}/[0-9]{2}/?'
 | 
			
		||||
#
 | 
			
		||||
#    def to_python(self, value):
 | 
			
		||||
#        value = value.split('/')
 | 
			
		||||
#        return datetime.date(int(value[0]), int(value[1]), int(value[2]))
 | 
			
		||||
#
 | 
			
		||||
#    def to_url(self, value):
 | 
			
		||||
#        return '{:04d}/{:02d}/'.format(*value.isocalendar())
 | 
			
		||||
class WeekConverter:
 | 
			
		||||
    """ Converter for date as YYYYY/WW """
 | 
			
		||||
    regex = r'[0-9]{4}/[0-9]{2}'
 | 
			
		||||
 | 
			
		||||
    def to_python(self, value):
 | 
			
		||||
        return datetime.datetime.strptime(value + '/1', '%G/%V/%u').date()
 | 
			
		||||
 | 
			
		||||
    def to_url(self, value):
 | 
			
		||||
        return '{:04d}/{:02d}'.format(*value.isocalendar())
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class DateConverter:
 | 
			
		||||
    """ Converter for date as YYYY/MM/DD """
 | 
			
		||||
    regex = r'[0-9]{4}/[0-9]{2}/[0-9]{2}/?'
 | 
			
		||||
    regex = r'[0-9]{4}/[0-9]{2}/[0-9]{2}'
 | 
			
		||||
 | 
			
		||||
    def to_python(self, value):
 | 
			
		||||
        value = value.split('/')
 | 
			
		||||
        return datetime.date(int(value[0]), int(value[1]), int(value[2]))
 | 
			
		||||
 | 
			
		||||
    def to_url(self, value):
 | 
			
		||||
        return '{:04d}/{:02d}/{:02d}/'.format(value.year, value.month,
 | 
			
		||||
                                              value.day)
 | 
			
		||||
        return '{:04d}/{:02d}/{:02d}'.format(value.year, value.month,
 | 
			
		||||
                                             value.day)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -1,7 +1,6 @@
 | 
			
		||||
from django.core.validators import RegexValidator
 | 
			
		||||
from django.db import models
 | 
			
		||||
from django.db.models import F, Q
 | 
			
		||||
from django.db.models.functions import Concat, Substr
 | 
			
		||||
from django.urls import reverse
 | 
			
		||||
from django.utils.translation import ugettext_lazy as _
 | 
			
		||||
 | 
			
		||||
from content_editor.models import Region, create_plugin_base
 | 
			
		||||
@ -13,42 +12,44 @@ from filer.fields.image import FilerImageField
 | 
			
		||||
 | 
			
		||||
from aircox import models as aircox
 | 
			
		||||
from . import plugins
 | 
			
		||||
from .converters import PagePathConverter
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class Site(models.Model):
 | 
			
		||||
    station = models.ForeignKey(
 | 
			
		||||
        aircox.Station, on_delete=models.SET_NULL, null=True,
 | 
			
		||||
    )
 | 
			
		||||
    #hosts = models.TextField(
 | 
			
		||||
    #    _('hosts'),
 | 
			
		||||
    #    help_text=_('website addresses (one per line)'),
 | 
			
		||||
    #)
 | 
			
		||||
 | 
			
		||||
    # main settings
 | 
			
		||||
    title = models.CharField(
 | 
			
		||||
        _('Title'), max_length=32,
 | 
			
		||||
        help_text=_('Website title used at various places'),
 | 
			
		||||
        _('title'), max_length=32,
 | 
			
		||||
        help_text=_('website title displayed to users'),
 | 
			
		||||
    )
 | 
			
		||||
    logo = FilerImageField(
 | 
			
		||||
        on_delete=models.SET_NULL, null=True, blank=True,
 | 
			
		||||
        verbose_name=_('Logo'),
 | 
			
		||||
        verbose_name=_('logo'),
 | 
			
		||||
        related_name='+',
 | 
			
		||||
    )
 | 
			
		||||
    favicon = FilerImageField(
 | 
			
		||||
        on_delete=models.SET_NULL, null=True, blank=True,
 | 
			
		||||
        verbose_name=_('Favicon'),
 | 
			
		||||
        verbose_name=_('favicon'),
 | 
			
		||||
        related_name='+',
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    default = models.BooleanField(_('default site'),
 | 
			
		||||
    default = models.BooleanField(_('is default'),
 | 
			
		||||
        default=False,
 | 
			
		||||
        help_text=_('Use as default site'),
 | 
			
		||||
        help_text=_('use this website by default'),
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    # meta descriptors
 | 
			
		||||
    description = models.CharField(
 | 
			
		||||
        _('Description'), max_length=128,
 | 
			
		||||
        _('description'), max_length=128,
 | 
			
		||||
        blank=True, null=True,
 | 
			
		||||
    )
 | 
			
		||||
    tags = models.CharField(
 | 
			
		||||
        _('Tags'), max_length=128,
 | 
			
		||||
        _('tags'), max_length=128,
 | 
			
		||||
        blank=True, null=True,
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
@ -72,9 +73,8 @@ class SiteLink(plugins.Link, SitePlugin):
 | 
			
		||||
 | 
			
		||||
#-----------------------------------------------------------------------
 | 
			
		||||
class PageQueryset(InheritanceQuerySet):
 | 
			
		||||
    def active(self):
 | 
			
		||||
        return self.filter(Q(status=Page.STATUS.announced) |
 | 
			
		||||
                           Q(status=Page.STATUS.published))
 | 
			
		||||
    def live(self):
 | 
			
		||||
        return self.filter(status=Page.STATUS.published)
 | 
			
		||||
 | 
			
		||||
    def descendants(self, page, direct=True, inclusive=True):
 | 
			
		||||
        qs = self.filter(parent=page) if direct else \
 | 
			
		||||
@ -97,7 +97,7 @@ class Page(StatusModel):
 | 
			
		||||
    Base class for views whose url path can be defined by users.
 | 
			
		||||
    Page parenting is based on foreignkey to parent and page path.
 | 
			
		||||
    """
 | 
			
		||||
    STATUS = Choices('draft', 'announced', 'published')
 | 
			
		||||
    STATUS = Choices('draft', 'published', 'trash')
 | 
			
		||||
 | 
			
		||||
    parent = models.ForeignKey(
 | 
			
		||||
        'self', models.CASCADE,
 | 
			
		||||
@ -106,31 +106,15 @@ class Page(StatusModel):
 | 
			
		||||
    )
 | 
			
		||||
    title = models.CharField(max_length=128)
 | 
			
		||||
    slug = models.SlugField(_('slug'))
 | 
			
		||||
    path = models.CharField(
 | 
			
		||||
        _("path"), max_length=1000,
 | 
			
		||||
        blank=True, db_index=True, unique=True,
 | 
			
		||||
        validators=[RegexValidator(
 | 
			
		||||
            regex=PagePathConverter.regex,
 | 
			
		||||
            message=_('Path accepts alphanumeric and "_-" characters '
 | 
			
		||||
                      'and must be surrounded by "/"')
 | 
			
		||||
        )],
 | 
			
		||||
    )
 | 
			
		||||
    static_path = models.BooleanField(
 | 
			
		||||
        _('static path'), default=False,
 | 
			
		||||
        # FIXME: help
 | 
			
		||||
        help_text=_('Update path using parent\'s page path and page title')
 | 
			
		||||
    )
 | 
			
		||||
    headline = models.TextField(
 | 
			
		||||
        _('headline'), max_length=128, blank=True, null=True,
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    objects = PageQueryset.as_manager()
 | 
			
		||||
 | 
			
		||||
    def __init__(self, *args, **kwargs):
 | 
			
		||||
        super().__init__(*args, **kwargs)
 | 
			
		||||
        self._initial_path = self.path
 | 
			
		||||
        self._initial_parent = self.parent
 | 
			
		||||
        self._initial_slug = self.slug
 | 
			
		||||
    @property
 | 
			
		||||
    def path(self):
 | 
			
		||||
        return reverse('page', kwargs={'slug': self.slug})
 | 
			
		||||
 | 
			
		||||
    def get_view_class(self):
 | 
			
		||||
        """ Page view class"""
 | 
			
		||||
@ -141,52 +125,12 @@ class Page(StatusModel):
 | 
			
		||||
        view = self.get_view_class().as_view(site=site, page=self)
 | 
			
		||||
        return view(request, *args, **kwargs)
 | 
			
		||||
 | 
			
		||||
    def update_descendants(self):
 | 
			
		||||
        """ Update descendants pages' path if required. """
 | 
			
		||||
        if self.path == self._initial_path:
 | 
			
		||||
            return
 | 
			
		||||
 | 
			
		||||
        # FIXME: draft -> draft children?
 | 
			
		||||
        # FIXME: Page.objects (can't use Page since its an abstract model)
 | 
			
		||||
        if len(self._initial_path):
 | 
			
		||||
            expr = Concat('path', Substr(F('path'), len(self._initial_path)))
 | 
			
		||||
            Page.objects.filter(path__startswith=self._initial_path) \
 | 
			
		||||
                        .update(path=expr)
 | 
			
		||||
 | 
			
		||||
    def sync_generations(self, update_descendants=True):
 | 
			
		||||
        """
 | 
			
		||||
        Update fields (path, ...) based on parent. Update childrens if
 | 
			
		||||
        ``update_descendants`` is True.
 | 
			
		||||
        """
 | 
			
		||||
        # TODO: set parent based on path (when static path)
 | 
			
		||||
        # TODO: ensure unique path fallback
 | 
			
		||||
        if self.path == self._initial_path and \
 | 
			
		||||
                self.slug == self._initial_slug and \
 | 
			
		||||
                self.parent == self._initial_parent:
 | 
			
		||||
            return
 | 
			
		||||
 | 
			
		||||
        if not self.title or not self.path or self.static_path and \
 | 
			
		||||
                self.slug != self._initial_slug:
 | 
			
		||||
            self.path = self.parent.path + self.slug \
 | 
			
		||||
                if self.parent is not None else '/' + self.slug
 | 
			
		||||
 | 
			
		||||
        if self.path[0] != '/':
 | 
			
		||||
            self.path = '/' + self.path
 | 
			
		||||
        if self.path[-1] != '/':
 | 
			
		||||
            self.path += '/'
 | 
			
		||||
        if update_descendants:
 | 
			
		||||
            self.update_descendants()
 | 
			
		||||
 | 
			
		||||
    def save(self, *args, update_descendants=True, **kwargs):
 | 
			
		||||
        self.sync_generations(update_descendants)
 | 
			
		||||
        super().save(*args, **kwargs)
 | 
			
		||||
 | 
			
		||||
    def __str__(self):
 | 
			
		||||
        return '{}: {}'.format(self._meta.verbose_name,
 | 
			
		||||
                               self.title or self.pk)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class Article(Page, TimeStampedModel):
 | 
			
		||||
class Article(Page):
 | 
			
		||||
    """ User's pages """
 | 
			
		||||
    regions = [
 | 
			
		||||
        Region(key="content", title=_("Content")),
 | 
			
		||||
@ -195,7 +139,7 @@ class Article(Page, TimeStampedModel):
 | 
			
		||||
    # metadata
 | 
			
		||||
    as_program = models.ForeignKey(
 | 
			
		||||
        aircox.Program, models.SET_NULL, blank=True, null=True,
 | 
			
		||||
        related_name='published_pages',
 | 
			
		||||
        related_name='+',
 | 
			
		||||
        # SO#51948640
 | 
			
		||||
        # limit_choices_to={'schedule__isnull': False},
 | 
			
		||||
        verbose_name=_('Show program as author'),
 | 
			
		||||
@ -216,27 +160,31 @@ class Article(Page, TimeStampedModel):
 | 
			
		||||
        verbose_name=_('Cover'),
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    def get_view_class(self):
 | 
			
		||||
        from .views import ArticleView
 | 
			
		||||
        return ArticleView
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class DiffusionPage(Article):
 | 
			
		||||
    diffusion = models.OneToOneField(
 | 
			
		||||
        aircox.Diffusion, models.CASCADE,
 | 
			
		||||
        related_name='page',
 | 
			
		||||
        limit_choices_to={'initial__isnull': True}
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    @property
 | 
			
		||||
    def path(self):
 | 
			
		||||
        return reverse('diffusion-page', kwargs={'slug': self.slug})
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class ProgramPage(Article):
 | 
			
		||||
    detail_url_name = 'program-page'
 | 
			
		||||
 | 
			
		||||
    program = models.OneToOneField(
 | 
			
		||||
        aircox.Program, models.CASCADE,
 | 
			
		||||
        related_name='page',
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    def get_view_class(self):
 | 
			
		||||
        from .views import ProgramView
 | 
			
		||||
        return ProgramView
 | 
			
		||||
    @property
 | 
			
		||||
    def path(self):
 | 
			
		||||
        return reverse('program-page', kwargs={'slug': self.slug})
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#-----------------------------------------------------------------------
 | 
			
		||||
 | 
			
		||||
@ -1,17 +0,0 @@
 | 
			
		||||
{% extends "aircox_web/page.html" %}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
{% block main %}
 | 
			
		||||
<section class="is-inline-block">
 | 
			
		||||
    <img class="cover" src="{{ page.cover.url }}"/>
 | 
			
		||||
    {% block headline %}
 | 
			
		||||
    {{ page.headline }}
 | 
			
		||||
    {% endblock %}
 | 
			
		||||
 | 
			
		||||
    {% block content %}
 | 
			
		||||
    {{ regions.main }}
 | 
			
		||||
    {% endblock %}
 | 
			
		||||
</section>
 | 
			
		||||
 | 
			
		||||
{% endblock %}
 | 
			
		||||
 | 
			
		||||
@ -3,6 +3,7 @@
 | 
			
		||||
Context variables:
 | 
			
		||||
- object: the actual diffusion
 | 
			
		||||
- page: current parent page in which item is rendered
 | 
			
		||||
- hide_schedule: if True, do not display start time
 | 
			
		||||
{% endcomment %}
 | 
			
		||||
 | 
			
		||||
{% with page as context_page %}
 | 
			
		||||
@ -10,29 +11,41 @@ Context variables:
 | 
			
		||||
{% diffusion_page object as page %}
 | 
			
		||||
<article class="media">
 | 
			
		||||
    <div class="media-left">
 | 
			
		||||
      <figure class="image is-64x64">
 | 
			
		||||
          <img src="{% thumbnail page.cover|default:site.logo 128x128 crop=scale %}">
 | 
			
		||||
      </figure>
 | 
			
		||||
        <img src="{% thumbnail page.cover|default:site.logo 128x128 crop=scale %}">
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="media-content">
 | 
			
		||||
      <div class="content">
 | 
			
		||||
        <p>
 | 
			
		||||
          {% if page and context_page != page %}
 | 
			
		||||
          <strong><a href="{{ page.path }}">{{ page.title }}</a></strong>
 | 
			
		||||
          {% else %}
 | 
			
		||||
          <strong>{{ page.title|default:program.name }}</strong>
 | 
			
		||||
          {% endif %}
 | 
			
		||||
          {% if object.page is page %}
 | 
			
		||||
          — <a href="{{ program.page.path }}">{{ program.name }}</a></small>
 | 
			
		||||
          {% endif %}
 | 
			
		||||
          {% if object.initial %}
 | 
			
		||||
          {% with object.initial.date as date %}
 | 
			
		||||
          <span class="tag is-info" title="{% blocktrans %}Rerun of {{ date }}{% endblocktrans %}">
 | 
			
		||||
              {% trans "rerun" %}
 | 
			
		||||
          </span>
 | 
			
		||||
          {% endwith %}
 | 
			
		||||
          {% endif %}
 | 
			
		||||
          <br>
 | 
			
		||||
        <div>
 | 
			
		||||
            <h4 class="subtitle is-size-4 is-inline-block">
 | 
			
		||||
            {% if page and context_page != page %}
 | 
			
		||||
            <a href="{{ page.path }}">{{ page.title }}</a>
 | 
			
		||||
            {% else %}
 | 
			
		||||
            {{ page.title|default:program.name }}
 | 
			
		||||
            {% endif %}
 | 
			
		||||
            </h4>
 | 
			
		||||
 | 
			
		||||
            <span class="has-text-weight-normal">
 | 
			
		||||
            {% if object.page is page and context_page != program.page %}
 | 
			
		||||
            — <a href="{% url "program-page" slug=program.page.slug %}">{{ program.page.title }}</a>
 | 
			
		||||
            {% endif %}
 | 
			
		||||
 | 
			
		||||
            {% if not hide_schedule %}
 | 
			
		||||
            <time datetime="{{ object.start|date:"c" }}" title="{{ object.start }}"
 | 
			
		||||
                  class="has-text-weight-light is-size-6">
 | 
			
		||||
                  — {{ object.start|date:"d M, H:i" }}
 | 
			
		||||
            </time>
 | 
			
		||||
            {% endif %}
 | 
			
		||||
 | 
			
		||||
            {% if object.initial %}
 | 
			
		||||
            {% with object.initial.date as date %}
 | 
			
		||||
            <span class="tag is-info" title="{% blocktrans %}Rerun of {{ date }}{% endblocktrans %}">
 | 
			
		||||
                {% trans "rerun" %}
 | 
			
		||||
            </span>
 | 
			
		||||
            {% endwith %}
 | 
			
		||||
            {% endif %}
 | 
			
		||||
            </span>
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
        <div class="content">
 | 
			
		||||
          {{ page.headline|default:program.page.headline }}
 | 
			
		||||
        </p>
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
@ -1,35 +1,53 @@
 | 
			
		||||
{% extends "aircox_web/page.html" %}
 | 
			
		||||
{% load i18n aircox_web %}
 | 
			
		||||
 | 
			
		||||
{% block main %}
 | 
			
		||||
{{ block.super }}
 | 
			
		||||
{% block title %}
 | 
			
		||||
{% if program %}
 | 
			
		||||
    {% with program.name as program %}
 | 
			
		||||
    {% blocktrans %}Diffusions of {{ program }}{% endblocktrans %}
 | 
			
		||||
    {% endwith %}
 | 
			
		||||
{% else %}
 | 
			
		||||
    {% trans "All diffusions" %}
 | 
			
		||||
    {% endif %}
 | 
			
		||||
{% endblock %}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
{% block header %}
 | 
			
		||||
{{ block.super }}
 | 
			
		||||
{% if program %}
 | 
			
		||||
<h4 class="subtitle is-size-3">
 | 
			
		||||
    <a href="{% url "page" path=program.page.path %}">❬ {{ program.name }}</a></li>
 | 
			
		||||
</h4>
 | 
			
		||||
{% include "aircox_web/program_header.html" %}
 | 
			
		||||
{% endif %}
 | 
			
		||||
{% endblock %}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
{% block content %}
 | 
			
		||||
<section class="section">
 | 
			
		||||
    {% for object in object_list %}
 | 
			
		||||
    <div class="columns">
 | 
			
		||||
        <div class="column is-one-fifth has-text-right">
 | 
			
		||||
            <time datetime="{{ object.start|date:"c" }}" title="{{ object.start }}">
 | 
			
		||||
                {{ object.start|date:"d M, H:i" }}
 | 
			
		||||
            </time>
 | 
			
		||||
        </div>
 | 
			
		||||
        <div class="column">
 | 
			
		||||
            {% include "aircox_web/diffusion_item.html" %}
 | 
			
		||||
        </div>
 | 
			
		||||
    </div>
 | 
			
		||||
    {% with object.diffusion as object %}
 | 
			
		||||
    {% include "aircox_web/diffusion_item.html" %}
 | 
			
		||||
    {% endwith %}
 | 
			
		||||
    {% endfor %}
 | 
			
		||||
</section>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
{% if is_paginated %}
 | 
			
		||||
<nav class="pagination is-centered" role="pagination" aria-label="{% trans "pagination" %}">
 | 
			
		||||
    {% if page_obj.has_previous %}
 | 
			
		||||
    <a href="?page={{ page_obj.previous_page_number }}" class="pagination-previous">
 | 
			
		||||
        {% trans "Previous" %}</a>
 | 
			
		||||
    {% else %}
 | 
			
		||||
    <a class="pagination-previous" disabled>
 | 
			
		||||
    {% endif %}
 | 
			
		||||
        {% trans "Previous" %}</a>
 | 
			
		||||
 | 
			
		||||
    {% if page_obj.has_next %}
 | 
			
		||||
    <a href="?page={{ page_obj.next_page_number }}" class="pagination-next">
 | 
			
		||||
        {% trans "Next" %}</a>
 | 
			
		||||
    {% else %}
 | 
			
		||||
    <a class="pagination-next" disabled>
 | 
			
		||||
    {% endif %}
 | 
			
		||||
        {% trans "Next" %}</a>
 | 
			
		||||
 | 
			
		||||
    <ul class="pagination-list">
 | 
			
		||||
    {% for i in paginator.page_range %}
 | 
			
		||||
 | 
			
		||||
@ -25,6 +25,7 @@
 | 
			
		||||
    {% endif %}
 | 
			
		||||
 | 
			
		||||
    {# <h4 class="subtitle size-4">{{ date }}</h4> #}
 | 
			
		||||
    {% with True as hide_schedule %}
 | 
			
		||||
    <table class="table is-striped is-hoverable is-fullwidth">
 | 
			
		||||
        {% for object in object_list reversed %}
 | 
			
		||||
        <tr>
 | 
			
		||||
@ -46,6 +47,7 @@
 | 
			
		||||
        </tr>
 | 
			
		||||
        {% endfor %}
 | 
			
		||||
    </table>
 | 
			
		||||
    {% endwith %}
 | 
			
		||||
</section>
 | 
			
		||||
{% endblock %}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -1,59 +1,36 @@
 | 
			
		||||
{% extends "aircox_web/base.html" %}
 | 
			
		||||
{% load static i18n thumbnail %}
 | 
			
		||||
<html>
 | 
			
		||||
    <head>
 | 
			
		||||
        <meta charset="utf-8">
 | 
			
		||||
        <meta name="application-name" content="aircox">
 | 
			
		||||
        <meta name="description" content="{{ site.description }}">
 | 
			
		||||
        <meta name="keywords" content="{{ site.tags }}">
 | 
			
		||||
        <link rel="icon" href="{% thumbnail site.favicon 32x32 crop %}" />
 | 
			
		||||
{% comment %}
 | 
			
		||||
Context:
 | 
			
		||||
- cover: cover image
 | 
			
		||||
- title: title
 | 
			
		||||
- page: page
 | 
			
		||||
{% endcomment %}
 | 
			
		||||
 | 
			
		||||
        {% block assets %}
 | 
			
		||||
        <link rel="stylesheet" type="text/css" href="{% static "aircox_web/assets/main.css" %}"/>
 | 
			
		||||
        <script src="{% static "aircox_web/assets/main.js" %}"></script>
 | 
			
		||||
        <script src="{% static "aircox_web/assets/vendor.js" %}"></script>
 | 
			
		||||
        {% endblock %}
 | 
			
		||||
 | 
			
		||||
        <title>{% block title %}{% if title %}{{ title }} -- {% endif %}{{ site.title }}{% endblock %}</title>
 | 
			
		||||
 | 
			
		||||
        {% block extra_head %}{% endblock %}
 | 
			
		||||
    </head>
 | 
			
		||||
    <body>
 | 
			
		||||
        <div id="app">
 | 
			
		||||
            <nav class="navbar has-shadow" role="navigation" aria-label="main navigation">
 | 
			
		||||
                <div class="container">
 | 
			
		||||
                    <div class="navbar-brand">
 | 
			
		||||
                        <a href="/" title="{% trans "Home" %}" class="navbar-item">
 | 
			
		||||
                            <img src="{{ site.logo.url }}" class="logo"/>
 | 
			
		||||
                        </a>
 | 
			
		||||
                    </div>
 | 
			
		||||
                    <div class="navbar-menu">
 | 
			
		||||
                        <div class="navbar-start">
 | 
			
		||||
                            {{ site_regions.topnav }}
 | 
			
		||||
                        </div>
 | 
			
		||||
                    </div>
 | 
			
		||||
                </div>
 | 
			
		||||
            </nav>
 | 
			
		||||
 | 
			
		||||
            <div class="container">
 | 
			
		||||
                <div class="columns">
 | 
			
		||||
                    <aside class="column is-one-quarter">
 | 
			
		||||
                        {% block left-sidebar %}
 | 
			
		||||
                        {{ site_regions.sidenav }}
 | 
			
		||||
                        {% endblock %}
 | 
			
		||||
                    </aside>
 | 
			
		||||
                    <main class="column page">
 | 
			
		||||
                        <header class="header">
 | 
			
		||||
                            {% block header %}
 | 
			
		||||
                            <h1 class="title is-1">{{ title }}</h1>
 | 
			
		||||
                            {% endblock %}
 | 
			
		||||
                        </header>
 | 
			
		||||
 | 
			
		||||
                        {% block main %}{% endblock main %}
 | 
			
		||||
                    </main>
 | 
			
		||||
                </div>
 | 
			
		||||
            </div>
 | 
			
		||||
        </div>
 | 
			
		||||
    </body>
 | 
			
		||||
</html>
 | 
			
		||||
{% block head_title %}
 | 
			
		||||
    {% block title %}{{ title }}{% endblock %}
 | 
			
		||||
    {% if title %} — {% endif %}
 | 
			
		||||
    {{ site.title }}
 | 
			
		||||
{% endblock %}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
{% block main %}
 | 
			
		||||
{% block headline %}
 | 
			
		||||
{% if page and page.headline %}
 | 
			
		||||
<p class="headline">{{ page.headline }}</p>
 | 
			
		||||
{% endif %}
 | 
			
		||||
{% endblock %}
 | 
			
		||||
 | 
			
		||||
{% block content %}
 | 
			
		||||
{{ regions.content }}
 | 
			
		||||
{% endblock %}
 | 
			
		||||
{% endblock %}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
{% block side_nav %}
 | 
			
		||||
{% if cover is not None %}
 | 
			
		||||
<img class="cover" src="{{ cover.url }}" class="cover"/>
 | 
			
		||||
{% endif %}
 | 
			
		||||
{% endblock %}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -1,27 +1,8 @@
 | 
			
		||||
{% extends "aircox_web/article.html" %}
 | 
			
		||||
{% load i18n %}
 | 
			
		||||
 | 
			
		||||
{% block headline %}
 | 
			
		||||
<section class="is-size-5">
 | 
			
		||||
    {% for schedule in program.schedule_set.all %}
 | 
			
		||||
    <p>
 | 
			
		||||
        <strong>{{ schedule.datetime|date:"l H:i" }}</strong>
 | 
			
		||||
        <small>
 | 
			
		||||
            {{ schedule.get_frequency_display }}
 | 
			
		||||
            {% if schedule.initial %}
 | 
			
		||||
            {% with schedule.initial.date as date %}
 | 
			
		||||
            <span title="{% blocktrans %}Rerun of {{ date }}{% endblocktrans %}">
 | 
			
		||||
                / {% trans "rerun" %}
 | 
			
		||||
            </span>
 | 
			
		||||
            {% endwith %}
 | 
			
		||||
            {% endif %}
 | 
			
		||||
        </small>
 | 
			
		||||
    </p>
 | 
			
		||||
    {% endfor %}
 | 
			
		||||
</section>
 | 
			
		||||
{% extends "aircox_web/page.html" %}
 | 
			
		||||
 | 
			
		||||
{% block header %}
 | 
			
		||||
{{ block.super }}
 | 
			
		||||
 | 
			
		||||
{% include "aircox_web/program_header.html" %}
 | 
			
		||||
{% endblock %}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -1,6 +1,10 @@
 | 
			
		||||
{% extends "aircox_web/page.html" %}
 | 
			
		||||
{% load i18n aircox_web %}
 | 
			
		||||
 | 
			
		||||
{% block title %}
 | 
			
		||||
{% trans "Timetable" %}
 | 
			
		||||
{% endblock %}
 | 
			
		||||
 | 
			
		||||
{% block main %}
 | 
			
		||||
{{ block.super }}
 | 
			
		||||
 | 
			
		||||
@ -27,6 +31,7 @@
 | 
			
		||||
            </li>
 | 
			
		||||
        </template>
 | 
			
		||||
 | 
			
		||||
        {% with True as hide_schedule %}
 | 
			
		||||
        <template v-slot:default="{value}">
 | 
			
		||||
            {% for day, diffusions in by_date.items %}
 | 
			
		||||
            <noscript><h4 class="subtitle is-4">{{ day|date:"l d F Y" }}</h4></noscript>
 | 
			
		||||
@ -46,6 +51,7 @@
 | 
			
		||||
            </div>
 | 
			
		||||
            {% endfor %}
 | 
			
		||||
        </template>
 | 
			
		||||
        {% endwith %}
 | 
			
		||||
    </a-tabs>
 | 
			
		||||
</section>
 | 
			
		||||
{% endblock %}
 | 
			
		||||
 | 
			
		||||
@ -8,19 +8,22 @@ from aircox_web.models import Page
 | 
			
		||||
random.seed()
 | 
			
		||||
register = template.Library()
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@register.simple_tag(name='diffusion_page')
 | 
			
		||||
def do_diffusion_page(diffusion):
 | 
			
		||||
    """ Return page for diffusion. """
 | 
			
		||||
    for obj in (diffusion, diffusion.program):
 | 
			
		||||
        page = getattr(obj, 'page', None)
 | 
			
		||||
        if page is not None and page.status is not Page.STATUS.draft:
 | 
			
		||||
        if page is not None and page.status == Page.STATUS.published:
 | 
			
		||||
            return page
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@register.simple_tag(name='unique_id')
 | 
			
		||||
def do_unique_id(prefix=''):
 | 
			
		||||
    value = str(random.random()).replace('.', '')
 | 
			
		||||
    return prefix + '_' + value if prefix else value
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@register.filter(name='is_diffusion')
 | 
			
		||||
def do_is_diffusion(obj):
 | 
			
		||||
    return isinstance(obj, aircox.Diffusion)
 | 
			
		||||
 | 
			
		||||
@ -2,22 +2,29 @@ from django.conf.urls import url
 | 
			
		||||
from django.urls import path, register_converter
 | 
			
		||||
 | 
			
		||||
from . import views, models
 | 
			
		||||
from .converters import PagePathConverter, DateConverter
 | 
			
		||||
from .converters import PagePathConverter, DateConverter, WeekConverter
 | 
			
		||||
 | 
			
		||||
register_converter(PagePathConverter, 'page_path')
 | 
			
		||||
register_converter(DateConverter, 'date')
 | 
			
		||||
register_converter(WeekConverter, 'week')
 | 
			
		||||
 | 
			
		||||
urlpatterns = [
 | 
			
		||||
    path('programs/<slug:slug>/',
 | 
			
		||||
        views.ProgramPageView.as_view(), name='program-page'),
 | 
			
		||||
    path('programs/<slug:program_slug>/diffusions/',
 | 
			
		||||
         views.DiffusionsView.as_view(), name='diffusion-list'),
 | 
			
		||||
 | 
			
		||||
    path('diffusion/<slug:slug>/',
 | 
			
		||||
        views.ProgramPageView.as_view(), name='diffusion-page'),
 | 
			
		||||
 | 
			
		||||
    path('diffusions/',
 | 
			
		||||
         views.TimetableView.as_view(), name='timetable'),
 | 
			
		||||
    path('diffusions/<date:date>',
 | 
			
		||||
    path('diffusions/<week:date>/',
 | 
			
		||||
         views.TimetableView.as_view(), name='timetable'),
 | 
			
		||||
    path('diffusions/all',
 | 
			
		||||
         views.DiffusionsView.as_view(), name='diffusion-list'),
 | 
			
		||||
    path('diffusions/<slug:program>',
 | 
			
		||||
         views.DiffusionsView.as_view(), name='diffusion-list'),
 | 
			
		||||
    path('logs/', views.LogsView.as_view(), name='logs'),
 | 
			
		||||
    path('logs/<date:date>', views.LogsView.as_view(), name='logs'),
 | 
			
		||||
    path('logs/<date:date>/', views.LogsView.as_view(), name='logs'),
 | 
			
		||||
    path('<page_path:path>', views.route_page, name='page'),
 | 
			
		||||
]
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -1,16 +1,17 @@
 | 
			
		||||
from collections import OrderedDict, deque
 | 
			
		||||
import datetime
 | 
			
		||||
 | 
			
		||||
from django.db.models import Q
 | 
			
		||||
from django.core.paginator import Paginator
 | 
			
		||||
from django.shortcuts import get_object_or_404
 | 
			
		||||
from django.utils.translation import ugettext_lazy as _
 | 
			
		||||
from django.views.generic import TemplateView, ListView
 | 
			
		||||
from django.views.generic import TemplateView, DetailView, ListView
 | 
			
		||||
from django.views.generic.base import TemplateResponseMixin, ContextMixin
 | 
			
		||||
 | 
			
		||||
from content_editor.contents import contents_for_item
 | 
			
		||||
 | 
			
		||||
from aircox import models as aircox
 | 
			
		||||
from .models import Site, Page
 | 
			
		||||
from .models import Site, Page, DiffusionPage, ProgramPage
 | 
			
		||||
from .renderer import site_renderer, page_renderer
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -35,6 +36,7 @@ def route_page(request, path=None, *args, model=None, site=None, **kwargs):
 | 
			
		||||
 | 
			
		||||
class BaseView(TemplateResponseMixin, ContextMixin):
 | 
			
		||||
    title = None
 | 
			
		||||
    cover = None
 | 
			
		||||
    site = None
 | 
			
		||||
 | 
			
		||||
    def dispatch(self, request, *args, site=None, **kwargs):
 | 
			
		||||
@ -48,64 +50,85 @@ class BaseView(TemplateResponseMixin, ContextMixin):
 | 
			
		||||
            kwargs['site_regions'] = contents.render_regions(site_renderer)
 | 
			
		||||
 | 
			
		||||
        kwargs.setdefault('site', self.site)
 | 
			
		||||
        if self.title is not None:
 | 
			
		||||
            kwargs.setdefault('title', self.title)
 | 
			
		||||
        kwargs.setdefault('cover', self.cover)
 | 
			
		||||
        return super().get_context_data(**kwargs)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class ArticleView(BaseView, TemplateView):
 | 
			
		||||
class PageView(BaseView):
 | 
			
		||||
    """ Base view class for pages. """
 | 
			
		||||
    template_name = 'aircox_web/article.html'
 | 
			
		||||
    template_name = 'aircox_web/page.html'
 | 
			
		||||
    context_object_name = 'page'
 | 
			
		||||
    page = None
 | 
			
		||||
 | 
			
		||||
    def get_context_data(self, **kwargs):
 | 
			
		||||
        # article content
 | 
			
		||||
        page = kwargs.setdefault('page', self.page or self.kwargs.get('page'))
 | 
			
		||||
        if kwargs.get('regions') is None:
 | 
			
		||||
            contents = contents_for_item(page, page_renderer._renderers.keys())
 | 
			
		||||
            kwargs['regions'] = contents.render_regions(page_renderer)
 | 
			
		||||
    def get_queryset(self):
 | 
			
		||||
        return super().get_queryset().live()
 | 
			
		||||
 | 
			
		||||
        kwargs.setdefault('title', page.title)
 | 
			
		||||
    def get_context_data(self, **kwargs):
 | 
			
		||||
        page = getattr(self, 'object', None)
 | 
			
		||||
        if page is not None:
 | 
			
		||||
            if kwargs.get('regions') is None:
 | 
			
		||||
                contents = contents_for_item(
 | 
			
		||||
                    page, page_renderer._renderers.keys())
 | 
			
		||||
                kwargs['regions'] = contents.render_regions(page_renderer)
 | 
			
		||||
 | 
			
		||||
            kwargs.setdefault('title', page.title)
 | 
			
		||||
            kwargs.setdefault('cover', page.cover)
 | 
			
		||||
            kwargs.setdefault('page', page)
 | 
			
		||||
        return super().get_context_data(**kwargs)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class ProgramView(ArticleView):
 | 
			
		||||
class ProgramPageView(PageView, DetailView):
 | 
			
		||||
    """ Base view class for pages. """
 | 
			
		||||
    template_name = 'aircox_web/program.html'
 | 
			
		||||
    next_diffs_count = 5
 | 
			
		||||
    model = ProgramPage
 | 
			
		||||
 | 
			
		||||
    def get_queryset(self):
 | 
			
		||||
        return super().get_queryset().select_related('program')
 | 
			
		||||
 | 
			
		||||
    def get_context_data(self, program=None, **kwargs):
 | 
			
		||||
        # TODO: pagination
 | 
			
		||||
        program = program or self.page.program
 | 
			
		||||
        #next_diffs = program.diffusion_set.on_air().after().order_by('start')
 | 
			
		||||
        return super().get_context_data(
 | 
			
		||||
            program=program,
 | 
			
		||||
            # next_diffs=next_diffs[:self.next_diffs_count],
 | 
			
		||||
            **kwargs,
 | 
			
		||||
        kwargs.setdefault('program', self.object.program)
 | 
			
		||||
        kwargs['diffusions'] = DiffusionPage.objects.filter(
 | 
			
		||||
            diffusion__program=kwargs['program']
 | 
			
		||||
        )
 | 
			
		||||
        return super().get_context_data(**kwargs)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class DiffusionView(ArticleView):
 | 
			
		||||
class DiffusionView(PageView):
 | 
			
		||||
    template_name = 'aircox_web/diffusion.html'
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
# TODO: pagination: in template, only a limited number of pages displayed
 | 
			
		||||
 | 
			
		||||
# DiffusionsView use diffusion instead of diffusion page for different reasons:
 | 
			
		||||
# more straightforward, it handles reruns
 | 
			
		||||
class DiffusionsView(BaseView, ListView):
 | 
			
		||||
    template_name = 'aircox_web/diffusions.html'
 | 
			
		||||
    model = aircox.Diffusion
 | 
			
		||||
    paginate_by = 10
 | 
			
		||||
    title = _('Diffusions')
 | 
			
		||||
    model = DiffusionPage
 | 
			
		||||
    paginate_by = 30
 | 
			
		||||
    program = None
 | 
			
		||||
 | 
			
		||||
    # TODO: get program object + display program title when filtered by program
 | 
			
		||||
    # TODO: pagination: in template, only a limited number of pages displayed
 | 
			
		||||
    def get(self, request, *args, **kwargs):
 | 
			
		||||
        program_slug = kwargs.get('program_slug')
 | 
			
		||||
        if program_slug:
 | 
			
		||||
            self.program = get_object_or_404(
 | 
			
		||||
                aircox.Program, slug=kwargs.get('program_slug'))
 | 
			
		||||
        return super().get(request, *args, **kwargs)
 | 
			
		||||
 | 
			
		||||
    def get_queryset(self):
 | 
			
		||||
        qs = super().get_queryset().station(self.site.station).on_air() \
 | 
			
		||||
                    .filter(initial__isnull=True) #TODO, page__isnull=False)
 | 
			
		||||
        program = self.kwargs.get('program')
 | 
			
		||||
        if program:
 | 
			
		||||
            qs = qs.filter(program__page__slug=program)
 | 
			
		||||
        return qs.order_by('-start')
 | 
			
		||||
        qs = super().get_queryset().live() \
 | 
			
		||||
                    .select_related('diffusion')
 | 
			
		||||
        if self.program:
 | 
			
		||||
            qs = qs.filter(diffusion__program=self.program)
 | 
			
		||||
        else:
 | 
			
		||||
            qs = qs.select_related('diffusion__program')
 | 
			
		||||
        return qs.order_by('-diffusion__start')
 | 
			
		||||
 | 
			
		||||
    def get_context_data(self, **kwargs):
 | 
			
		||||
        program = kwargs.setdefault('program', self.program)
 | 
			
		||||
        if program is not None and hasattr(program, 'page'):
 | 
			
		||||
            kwargs.setdefault('cover', program.page.cover)
 | 
			
		||||
            kwargs.setdefault('page', program.page)
 | 
			
		||||
        return super().get_context_data(**kwargs)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class TimetableView(BaseView, ListView):
 | 
			
		||||
@ -120,9 +143,9 @@ class TimetableView(BaseView, ListView):
 | 
			
		||||
    end = None
 | 
			
		||||
 | 
			
		||||
    def get_queryset(self):
 | 
			
		||||
        self.date = self.kwargs.get('date', datetime.date.today())
 | 
			
		||||
        self.date = self.kwargs.get('date') or datetime.date.today()
 | 
			
		||||
        self.start = self.date - datetime.timedelta(days=self.date.weekday())
 | 
			
		||||
        self.end = self.date + datetime.timedelta(days=7-self.date.weekday())
 | 
			
		||||
        self.end = self.start + datetime.timedelta(days=7)
 | 
			
		||||
        return super().get_queryset().station(self.site.station) \
 | 
			
		||||
                                     .range(self.start, self.end) \
 | 
			
		||||
                                     .order_by('start')
 | 
			
		||||
 | 
			
		||||
		Reference in New Issue
	
	Block a user