forked from rc/aircox
		
	work on website; fix stuffs on aircox too
This commit is contained in:
		@ -15,6 +15,8 @@ from aircox.admin.mixins import UnrelatedInlineMixin
 | 
			
		||||
 | 
			
		||||
@admin.register(models.Site)
 | 
			
		||||
class SiteAdmin(ContentEditor):
 | 
			
		||||
    list_display = ['title', 'station']
 | 
			
		||||
 | 
			
		||||
    inlines = [
 | 
			
		||||
        ContentEditorInline.create(models.SiteRichText),
 | 
			
		||||
        ContentEditorInline.create(models.SiteImage),
 | 
			
		||||
@ -37,14 +39,26 @@ class PageDiffusionPlaylist(UnrelatedInlineMixin, TracksInline):
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@admin.register(models.Page)
 | 
			
		||||
class PageAdmin(ContentEditor):
 | 
			
		||||
class PageAdmin(admin.ModelAdmin):
 | 
			
		||||
    list_display = ["title", "parent", "status"]
 | 
			
		||||
    list_editable = ['status']
 | 
			
		||||
    prepopulated_fields = {"slug": ("title",)}
 | 
			
		||||
    # readonly_fields = ('diffusion',)
 | 
			
		||||
 | 
			
		||||
    fieldsets = (
 | 
			
		||||
        (_('Main'), {
 | 
			
		||||
            'fields': ['title', 'slug', 'as_program', 'headline'],
 | 
			
		||||
            'fields': ['title', 'slug']
 | 
			
		||||
        }),
 | 
			
		||||
        (_('Settings'), {
 | 
			
		||||
            'fields': ['status', 'static_path', 'path'],
 | 
			
		||||
        }),
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@admin.register(models.Article)
 | 
			
		||||
class ArticleAdmin(ContentEditor, PageAdmin):
 | 
			
		||||
    fieldsets = (
 | 
			
		||||
        (_('Main'), {
 | 
			
		||||
            'fields': ['title', 'slug', 'as_program', 'cover', 'headline'],
 | 
			
		||||
            'classes': ('tabbed', 'uncollapse')
 | 
			
		||||
        }),
 | 
			
		||||
        (_('Settings'), {
 | 
			
		||||
@ -59,36 +73,27 @@ class PageAdmin(ContentEditor):
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    inlines = [
 | 
			
		||||
        ContentEditorInline.create(models.PageRichText),
 | 
			
		||||
        ContentEditorInline.create(models.PageImage),
 | 
			
		||||
        ContentEditorInline.create(models.ArticleRichText),
 | 
			
		||||
        ContentEditorInline.create(models.ArticleImage),
 | 
			
		||||
    ]
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@admin.register(models.DiffusionPage)
 | 
			
		||||
class DiffusionPageAdmin(PageAdmin):
 | 
			
		||||
    fieldsets = copy.deepcopy(PageAdmin.fieldsets)
 | 
			
		||||
class DiffusionPageAdmin(ArticleAdmin):
 | 
			
		||||
    fieldsets = copy.deepcopy(ArticleAdmin.fieldsets)
 | 
			
		||||
    fieldsets[1][1]['fields'].insert(0, 'diffusion')
 | 
			
		||||
 | 
			
		||||
    inlines = PageAdmin.inlines + [
 | 
			
		||||
        PageDiffusionPlaylist
 | 
			
		||||
    ]
 | 
			
		||||
 | 
			
		||||
    # TODO: permissions
 | 
			
		||||
    #def get_inline_instances(self, request, obj=None):
 | 
			
		||||
    #    inlines = super().get_inline_instances(request, obj)
 | 
			
		||||
    #    if obj and obj.diffusion:
 | 
			
		||||
    #        inlines.insert(0, PageDiffusionPlaylist(self.model, self.admin_site))
 | 
			
		||||
    #    return inlines
 | 
			
		||||
    def get_inline_instances(self, request, obj=None):
 | 
			
		||||
        inlines = super().get_inline_instances(request, obj)
 | 
			
		||||
        if obj and obj.diffusion:
 | 
			
		||||
            inlines.insert(0, PageDiffusionPlaylist(self.model, self.admin_site))
 | 
			
		||||
        return inlines
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@admin.register(models.ProgramPage)
 | 
			
		||||
class DiffusionPageAdmin(PageAdmin):
 | 
			
		||||
    fieldsets = copy.deepcopy(PageAdmin.fieldsets)
 | 
			
		||||
class ProgramPageAdmin(ArticleAdmin):
 | 
			
		||||
    fieldsets = copy.deepcopy(ArticleAdmin.fieldsets)
 | 
			
		||||
    fieldsets[1][1]['fields'].insert(0, 'program')
 | 
			
		||||
 | 
			
		||||
    inlines = PageAdmin.inlines + [
 | 
			
		||||
        PageDiffusionPlaylist
 | 
			
		||||
    ]
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -1,3 +1,5 @@
 | 
			
		||||
import './js';
 | 
			
		||||
import './styles.scss';
 | 
			
		||||
import './noscript.scss';
 | 
			
		||||
import './vue';
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -3,9 +3,12 @@ import Buefy from 'buefy';
 | 
			
		||||
 | 
			
		||||
Vue.use(Buefy);
 | 
			
		||||
 | 
			
		||||
var app = new Vue({
 | 
			
		||||
  el: '#app',
 | 
			
		||||
})
 | 
			
		||||
window.addEventListener('load', () => {
 | 
			
		||||
    var app = new Vue({
 | 
			
		||||
      el: '#app',
 | 
			
		||||
      delimiters: [ '[[', ']]' ],
 | 
			
		||||
    })
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -10,9 +10,12 @@ $body-background-color: $light;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.navbar.has-shadow {
 | 
			
		||||
    box-shadow: 0em 0.1em 0.5em rgba(0,0,0,0.1);
 | 
			
		||||
    box-shadow: 0em 0.05em 0.5em rgba(0,0,0,0.1);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
.navbar-brand img {
 | 
			
		||||
    min-height: 6em;
 | 
			
		||||
}
 | 
			
		||||
@ -20,5 +23,22 @@ $body-background-color: $light;
 | 
			
		||||
.navbar-menu .navbar-item:not(:last-child) {
 | 
			
		||||
    border-right: 1px $grey solid;
 | 
			
		||||
}
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/** page **/
 | 
			
		||||
img.cover {
 | 
			
		||||
    border: 0.2em black solid;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
.headline {
 | 
			
		||||
    font-size: 1.2em;
 | 
			
		||||
    padding: 0.2em 0em;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
img.cover {
 | 
			
		||||
    float: right;
 | 
			
		||||
    max-width: 40%;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										11
									
								
								aircox_web/assets/vue/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								aircox_web/assets/vue/index.js
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,11 @@
 | 
			
		||||
import Vue from 'vue';
 | 
			
		||||
 | 
			
		||||
import Tab from './tab.vue';
 | 
			
		||||
import Tabs from './tabs.vue';
 | 
			
		||||
 | 
			
		||||
Vue.component('a-tab', Tab);
 | 
			
		||||
Vue.component('a-tabs', Tabs);
 | 
			
		||||
 | 
			
		||||
export {Tab, Tabs};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										31
									
								
								aircox_web/assets/vue/tab.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								aircox_web/assets/vue/tab.vue
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,31 @@
 | 
			
		||||
<template>
 | 
			
		||||
    <li @click.prevent="onclick"
 | 
			
		||||
        :class="{'is-active': $parent.value == value}">
 | 
			
		||||
        <slot></slot>
 | 
			
		||||
    </li>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
<script>
 | 
			
		||||
export default {
 | 
			
		||||
    props: {
 | 
			
		||||
        value: { default: undefined },
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    methods: {
 | 
			
		||||
        select() {
 | 
			
		||||
            this.$parent.selectTab(this);
 | 
			
		||||
        },
 | 
			
		||||
 | 
			
		||||
        onclick(event) {
 | 
			
		||||
            this.select();
 | 
			
		||||
            /*if(event.target.href != document.location)
 | 
			
		||||
                window.history.pushState(
 | 
			
		||||
                    { url: event.target.href },
 | 
			
		||||
                    event.target.innerText + ' - ' + document.title,
 | 
			
		||||
                    event.target.href
 | 
			
		||||
                ) */
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										45
									
								
								aircox_web/assets/vue/tabs.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								aircox_web/assets/vue/tabs.vue
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,45 @@
 | 
			
		||||
<template>
 | 
			
		||||
    <div>
 | 
			
		||||
        <div class="tabs is-centered">
 | 
			
		||||
            <ul><slot name="tabs" :value="value" /></ul>
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
        <slot :value="value"/>
 | 
			
		||||
    </div>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
<script>
 | 
			
		||||
export default {
 | 
			
		||||
    props: {
 | 
			
		||||
        default: { default: null },
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    data() {
 | 
			
		||||
        return {
 | 
			
		||||
            value: this.default,
 | 
			
		||||
        }
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    computed: {
 | 
			
		||||
        tab() {
 | 
			
		||||
            const vnode = this.$slots.default && this.$slots.default.find(
 | 
			
		||||
                elm => elm.child && elm.child.value == this.value
 | 
			
		||||
            );
 | 
			
		||||
            return vnode && vnode.child;
 | 
			
		||||
        }
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    methods: {
 | 
			
		||||
        selectTab(tab) {
 | 
			
		||||
            const value = tab.value;
 | 
			
		||||
            if(this.value === value)
 | 
			
		||||
                return;
 | 
			
		||||
 | 
			
		||||
            this.value = value;
 | 
			
		||||
            this.$emit('select', {target: this, value: value, tab: tab});
 | 
			
		||||
        },
 | 
			
		||||
    },
 | 
			
		||||
}
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										49
									
								
								aircox_web/converters.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								aircox_web/converters.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,49 @@
 | 
			
		||||
import datetime
 | 
			
		||||
 | 
			
		||||
from django.utils.safestring import mark_safe
 | 
			
		||||
from django.urls.converters import StringConverter
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class PagePathConverter(StringConverter):
 | 
			
		||||
    """ Match path for pages, including surrounding slashes. """
 | 
			
		||||
    regex = r'/?|([-_a-zA-Z0-9]+/)*?'
 | 
			
		||||
 | 
			
		||||
    def to_python(self, value):
 | 
			
		||||
        if not value or value[0] != '/':
 | 
			
		||||
            value = '/' + value
 | 
			
		||||
        if len(value) > 1 and value[-1] != '/':
 | 
			
		||||
            value = value + '/'
 | 
			
		||||
        return value
 | 
			
		||||
 | 
			
		||||
    def to_url(self, value):
 | 
			
		||||
        if value[0] == '/':
 | 
			
		||||
            value = value[1:]
 | 
			
		||||
        if value[-1] != '/':
 | 
			
		||||
            value = value + '/'
 | 
			
		||||
        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 DateConverter:
 | 
			
		||||
    """ Converter for date as YYYY/MM/DD """
 | 
			
		||||
    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)
 | 
			
		||||
 | 
			
		||||
@ -1,18 +1,19 @@
 | 
			
		||||
from django.core.validators import RegexValidator
 | 
			
		||||
from django.db import models
 | 
			
		||||
from django.db.models import F
 | 
			
		||||
from django.db.models import F, Q
 | 
			
		||||
from django.db.models.functions import Concat, Substr
 | 
			
		||||
from django.utils.translation import ugettext_lazy as _
 | 
			
		||||
 | 
			
		||||
from content_editor.models import Region, create_plugin_base
 | 
			
		||||
 | 
			
		||||
from model_utils.models import TimeStampedModel, StatusModel
 | 
			
		||||
from model_utils.managers import InheritanceManager
 | 
			
		||||
from model_utils.managers import InheritanceQuerySet
 | 
			
		||||
from model_utils import Choices
 | 
			
		||||
from filer.fields.image import FilerImageField
 | 
			
		||||
 | 
			
		||||
from aircox import models as aircox
 | 
			
		||||
from . import plugins
 | 
			
		||||
from .converters import PagePathConverter
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class Site(models.Model):
 | 
			
		||||
@ -70,12 +71,31 @@ class SiteLink(plugins.Link, SitePlugin):
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#-----------------------------------------------------------------------
 | 
			
		||||
class BasePage(StatusModel):
 | 
			
		||||
    """
 | 
			
		||||
    Base abstract class for views whose url path is defined by users.
 | 
			
		||||
    Page parenting is based on foreignkey to parent and page path.
 | 
			
		||||
class PageQueryset(InheritanceQuerySet):
 | 
			
		||||
    def active(self):
 | 
			
		||||
        return self.filter(Q(status=Page.STATUS.announced) |
 | 
			
		||||
                           Q(status=Page.STATUS.published))
 | 
			
		||||
 | 
			
		||||
    Inspired by Feincms3.
 | 
			
		||||
    def descendants(self, page, direct=True, inclusive=True):
 | 
			
		||||
        qs = self.filter(parent=page) if direct else \
 | 
			
		||||
             self.filter(path__startswith=page.path)
 | 
			
		||||
        if not inclusive:
 | 
			
		||||
            qs = qs.exclude(pk=page.pk)
 | 
			
		||||
        return qs
 | 
			
		||||
 | 
			
		||||
    def ancestors(self, page, inclusive=True):
 | 
			
		||||
        path, paths = page.path, []
 | 
			
		||||
        index = path.find('/')
 | 
			
		||||
        while index != -1 and index+1 < len(path):
 | 
			
		||||
            paths.append(path[0:index+1])
 | 
			
		||||
            index = path.find('/', index+1)
 | 
			
		||||
        return self.filter(path__in=paths)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
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')
 | 
			
		||||
 | 
			
		||||
@ -89,22 +109,22 @@ class BasePage(StatusModel):
 | 
			
		||||
    path = models.CharField(
 | 
			
		||||
        _("path"), max_length=1000,
 | 
			
		||||
        blank=True, db_index=True, unique=True,
 | 
			
		||||
        validators=[
 | 
			
		||||
            RegexValidator(
 | 
			
		||||
                regex=r"^/(|.+/)$",
 | 
			
		||||
                message=_("Path must start and end with a slash (/)."),
 | 
			
		||||
            )
 | 
			
		||||
        ],
 | 
			
		||||
        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 = InheritanceManager()
 | 
			
		||||
 | 
			
		||||
    class Meta:
 | 
			
		||||
        abstract = True
 | 
			
		||||
    objects = PageQueryset.as_manager()
 | 
			
		||||
 | 
			
		||||
    def __init__(self, *args, **kwargs):
 | 
			
		||||
        super().__init__(*args, **kwargs)
 | 
			
		||||
@ -112,10 +132,14 @@ class BasePage(StatusModel):
 | 
			
		||||
        self._initial_parent = self.parent
 | 
			
		||||
        self._initial_slug = self.slug
 | 
			
		||||
 | 
			
		||||
    def view(self, request, *args, **kwargs):
 | 
			
		||||
    def get_view_class(self):
 | 
			
		||||
        """ Page view class"""
 | 
			
		||||
        raise NotImplementedError('not implemented')
 | 
			
		||||
 | 
			
		||||
    def view(self, request, *args, site=None, **kwargs):
 | 
			
		||||
        """ Page view function """
 | 
			
		||||
        from django.http import HttpResponse
 | 
			
		||||
        return HttpResponse('Not implemented')
 | 
			
		||||
        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. """
 | 
			
		||||
@ -123,8 +147,10 @@ class BasePage(StatusModel):
 | 
			
		||||
            return
 | 
			
		||||
 | 
			
		||||
        # FIXME: draft -> draft children?
 | 
			
		||||
        expr = Concat(self.path, Substr(F('path'), len(self._initial_path)))
 | 
			
		||||
        BasePage.objects.filter(path__startswith=self._initial_path) \
 | 
			
		||||
        # 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):
 | 
			
		||||
@ -141,13 +167,13 @@ class BasePage(StatusModel):
 | 
			
		||||
 | 
			
		||||
        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 \
 | 
			
		||||
            self.path = self.parent.path + self.slug \
 | 
			
		||||
                if self.parent is not None else '/' + self.slug
 | 
			
		||||
 | 
			
		||||
        if self.path[-1] != '/':
 | 
			
		||||
            self.path += '/'
 | 
			
		||||
        if self.path[0] != '/':
 | 
			
		||||
            self.path = '/' + self.path
 | 
			
		||||
        if self.path[-1] != '/':
 | 
			
		||||
            self.path += '/'
 | 
			
		||||
        if update_descendants:
 | 
			
		||||
            self.update_descendants()
 | 
			
		||||
 | 
			
		||||
@ -155,18 +181,23 @@ class BasePage(StatusModel):
 | 
			
		||||
        self.sync_generations(update_descendants)
 | 
			
		||||
        super().save(*args, **kwargs)
 | 
			
		||||
 | 
			
		||||
    def __str__(self):
 | 
			
		||||
        return '{}: {}'.format(self._meta.verbose_name,
 | 
			
		||||
                               self.title or self.pk)
 | 
			
		||||
 | 
			
		||||
class Page(BasePage, TimeStampedModel):
 | 
			
		||||
 | 
			
		||||
class Article(Page, TimeStampedModel):
 | 
			
		||||
    """ User's pages """
 | 
			
		||||
    regions = [
 | 
			
		||||
        Region(key="main", title=_("Content")),
 | 
			
		||||
        Region(key="content", title=_("Content")),
 | 
			
		||||
    ]
 | 
			
		||||
 | 
			
		||||
    # metadata
 | 
			
		||||
    as_program = models.ForeignKey(
 | 
			
		||||
        aircox.Program, models.SET_NULL, blank=True, null=True,
 | 
			
		||||
        related_name='published_pages',
 | 
			
		||||
        limit_choices_to={'schedule__isnull': False},
 | 
			
		||||
        # SO#51948640
 | 
			
		||||
        # limit_choices_to={'schedule__isnull': False},
 | 
			
		||||
        verbose_name=_('Show program as author'),
 | 
			
		||||
        help_text=_("Show program as author"),
 | 
			
		||||
    )
 | 
			
		||||
@ -180,45 +211,41 @@ class Page(BasePage, TimeStampedModel):
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    # content
 | 
			
		||||
    headline = models.TextField(
 | 
			
		||||
        _('headline'), max_length=128, blank=True, null=True,
 | 
			
		||||
    )
 | 
			
		||||
    cover = FilerImageField(
 | 
			
		||||
        on_delete=models.SET_NULL, null=True, blank=True,
 | 
			
		||||
        verbose_name=_('Cover'),
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    def get_view_class(self):
 | 
			
		||||
        from .views import PageView
 | 
			
		||||
        return PageView
 | 
			
		||||
 | 
			
		||||
    def view(self, request, *args, **kwargs):
 | 
			
		||||
        """ Page view function """
 | 
			
		||||
        view = self.get_view_class().as_view()
 | 
			
		||||
        return view(request, *args, **kwargs)
 | 
			
		||||
        from .views import ArticleView
 | 
			
		||||
        return ArticleView
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class DiffusionPage(Page):
 | 
			
		||||
class DiffusionPage(Article):
 | 
			
		||||
    diffusion = models.OneToOneField(
 | 
			
		||||
        aircox.Diffusion, models.CASCADE,
 | 
			
		||||
        blank=True, null=True,
 | 
			
		||||
        related_name='page',
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class ProgramPage(Page):
 | 
			
		||||
class ProgramPage(Article):
 | 
			
		||||
    program = models.OneToOneField(
 | 
			
		||||
        aircox.Program, models.CASCADE,
 | 
			
		||||
        blank=True, null=True,
 | 
			
		||||
        related_name='page',
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    def get_view_class(self):
 | 
			
		||||
        from .views import ProgramView
 | 
			
		||||
        return ProgramView
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#-----------------------------------------------------------------------
 | 
			
		||||
PagePlugin = create_plugin_base(Page)
 | 
			
		||||
ArticlePlugin = create_plugin_base(Article)
 | 
			
		||||
 | 
			
		||||
class PageRichText(plugins.RichText, PagePlugin):
 | 
			
		||||
class ArticleRichText(plugins.RichText, ArticlePlugin):
 | 
			
		||||
    pass
 | 
			
		||||
 | 
			
		||||
class PageImage(plugins.Image, PagePlugin):
 | 
			
		||||
class ArticleImage(plugins.Image, ArticlePlugin):
 | 
			
		||||
    pass
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -18,6 +18,7 @@
 | 
			
		||||
    "ttf-loader": "^1.0.2",
 | 
			
		||||
    "vue-loader": "^15.7.0",
 | 
			
		||||
    "vue-style-loader": "^4.1.2",
 | 
			
		||||
    "vue-template-compiler": "^2.6.10",
 | 
			
		||||
    "webpack": "^4.32.2",
 | 
			
		||||
    "webpack-cli": "^3.3.2"
 | 
			
		||||
  },
 | 
			
		||||
 | 
			
		||||
@ -13,6 +13,6 @@ site_renderer.register(SiteLink, lambda plugin: plugin.render())
 | 
			
		||||
 | 
			
		||||
page_renderer = PluginRenderer()
 | 
			
		||||
page_renderer._renderers.clear()
 | 
			
		||||
page_renderer.register(PageRichText, lambda plugin: mark_safe(plugin.text))
 | 
			
		||||
page_renderer.register(PageImage, lambda plugin: plugin.render())
 | 
			
		||||
page_renderer.register(ArticleRichText, lambda plugin: mark_safe(plugin.text))
 | 
			
		||||
page_renderer.register(ArticleImage, lambda plugin: plugin.render())
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										17
									
								
								aircox_web/templates/aircox_web/article.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								aircox_web/templates/aircox_web/article.html
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,17 @@
 | 
			
		||||
{% 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 %}
 | 
			
		||||
 | 
			
		||||
@ -1,50 +0,0 @@
 | 
			
		||||
{% 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 %}" />
 | 
			
		||||
 | 
			
		||||
        {% block assets %}
 | 
			
		||||
        <link rel="stylesheet" type="text/css" href="{% static "aircox_web/assets/main.css" %}"/>
 | 
			
		||||
        <!-- <link rel="stylesheet" type="text/css" href="{% static "aircox_web/assets/vendor.css" %}"/> -->
 | 
			
		||||
 | 
			
		||||
        <script src="{% static "aircox_web/assets/main.js" %}"></script>
 | 
			
		||||
        <script src="{% static "aircox_web/assets/vendor.js" %}"></script>
 | 
			
		||||
        {% endblock %}
 | 
			
		||||
 | 
			
		||||
        <title>{% block title %}{{ site.title }}{% endblock %}</title>
 | 
			
		||||
 | 
			
		||||
        {% block extra_head %}{% endblock %}
 | 
			
		||||
    </head>
 | 
			
		||||
    <body id="app">
 | 
			
		||||
        <nav class="navbar has-shadow" role="navigation" aria-label="main navigation">
 | 
			
		||||
            <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>
 | 
			
		||||
        </nav>
 | 
			
		||||
 | 
			
		||||
        <div class="columns">
 | 
			
		||||
            <aside class="column">
 | 
			
		||||
                {{ site_regions.sidenav }}
 | 
			
		||||
            </aside>
 | 
			
		||||
            <main class="column is-three-quarters">
 | 
			
		||||
                {% block main %}
 | 
			
		||||
                <h1>{{ page.title }}</h1>
 | 
			
		||||
                {{ regions.main }}
 | 
			
		||||
                {% endblock main %}
 | 
			
		||||
            </main>
 | 
			
		||||
        </div>
 | 
			
		||||
    </body>
 | 
			
		||||
</html>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										43
									
								
								aircox_web/templates/aircox_web/diffusion_item.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								aircox_web/templates/aircox_web/diffusion_item.html
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,43 @@
 | 
			
		||||
{% load i18n easy_thumbnails_tags aircox_web %}
 | 
			
		||||
{% comment %}
 | 
			
		||||
Context variables:
 | 
			
		||||
- object: the actual diffusion
 | 
			
		||||
- page: current parent page in which item is rendered
 | 
			
		||||
{% endcomment %}
 | 
			
		||||
 | 
			
		||||
{% with page as context_page %}
 | 
			
		||||
{% with object.program as program %}
 | 
			
		||||
{% 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>
 | 
			
		||||
    </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>
 | 
			
		||||
          {{ page.headline|default:program.page.headline }}
 | 
			
		||||
        </p>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
</article>
 | 
			
		||||
{% endwith %}
 | 
			
		||||
{% endwith %}
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										46
									
								
								aircox_web/templates/aircox_web/diffusions.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								aircox_web/templates/aircox_web/diffusions.html
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,46 @@
 | 
			
		||||
{% extends "aircox_web/page.html" %}
 | 
			
		||||
{% load i18n aircox_web %}
 | 
			
		||||
 | 
			
		||||
{% block main %}
 | 
			
		||||
{{ block.super }}
 | 
			
		||||
 | 
			
		||||
<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>
 | 
			
		||||
    {% endfor %}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
{% 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>
 | 
			
		||||
    {% endif %}
 | 
			
		||||
 | 
			
		||||
    {% if page_obj.has_next %}
 | 
			
		||||
    <a href="?page={{ page_obj.next_page_number }}" class="pagination-next">
 | 
			
		||||
        {% trans "Next" %}</a>
 | 
			
		||||
    {% endif %}
 | 
			
		||||
 | 
			
		||||
    <ul class="pagination-list">
 | 
			
		||||
    {% for i in paginator.page_range %}
 | 
			
		||||
        <li>
 | 
			
		||||
            <a class="pagination-link {% if page_obj.number == i %}is-current{% endif %}"
 | 
			
		||||
               href="?page={{ i }}">{{ i }}</a>
 | 
			
		||||
        </li>
 | 
			
		||||
    {% endfor %}
 | 
			
		||||
    </ul>
 | 
			
		||||
</nav>
 | 
			
		||||
{% endif %}
 | 
			
		||||
</section>
 | 
			
		||||
{% endblock %}
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										19
									
								
								aircox_web/templates/aircox_web/log_item.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								aircox_web/templates/aircox_web/log_item.html
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,19 @@
 | 
			
		||||
{% load i18n %}
 | 
			
		||||
 | 
			
		||||
{% with object.track as track %}
 | 
			
		||||
<span class="has-text-info is-size-5">♬</span>
 | 
			
		||||
<span>{{ track.title }}</span>
 | 
			
		||||
{% with track.artist as artist %}
 | 
			
		||||
{% with track.info as info %}
 | 
			
		||||
<span class="has-text-grey-dark has-text-weight-light">
 | 
			
		||||
    {% blocktrans %}
 | 
			
		||||
    by {{ artist }}
 | 
			
		||||
    {% endblocktrans %}
 | 
			
		||||
    {% if info %}
 | 
			
		||||
    ({% blocktrans %}<i>{{ info }}</i>{% endblocktrans %})
 | 
			
		||||
    {% endif %}
 | 
			
		||||
</span>
 | 
			
		||||
{% endwith %}
 | 
			
		||||
{% endwith %}
 | 
			
		||||
{% endwith %}
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										51
									
								
								aircox_web/templates/aircox_web/logs.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								aircox_web/templates/aircox_web/logs.html
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,51 @@
 | 
			
		||||
{% extends "aircox_web/page.html" %}
 | 
			
		||||
{% load i18n aircox_web %}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
{% block main %}
 | 
			
		||||
{{ block.super }}
 | 
			
		||||
 | 
			
		||||
<section class="section">
 | 
			
		||||
    {% if dates %}
 | 
			
		||||
    <nav class="tabs is-centered" aria-label="{% trans "Other days' logs" %}">
 | 
			
		||||
        <ul>
 | 
			
		||||
        {% for day in dates %}
 | 
			
		||||
            <li {% if day == date %}class="is-active"{% endif %}>
 | 
			
		||||
                <a href="{% url "logs" date=day %}">
 | 
			
		||||
                    {{ day|date:"d b" }}
 | 
			
		||||
                </a>
 | 
			
		||||
            </li>
 | 
			
		||||
 | 
			
		||||
        {% if forloop.last and day > min_date %}
 | 
			
		||||
            <li>...</li>
 | 
			
		||||
        {% endif %}
 | 
			
		||||
        {% endfor %}
 | 
			
		||||
        </ul>
 | 
			
		||||
    </nav>
 | 
			
		||||
    {% endif %}
 | 
			
		||||
 | 
			
		||||
    {# <h4 class="subtitle size-4">{{ date }}</h4> #}
 | 
			
		||||
    <table class="table is-striped is-hoverable is-fullwidth">
 | 
			
		||||
        {% for object in object_list reversed %}
 | 
			
		||||
        <tr>
 | 
			
		||||
        {% if object|is_diffusion %}
 | 
			
		||||
            <td>
 | 
			
		||||
                <time datetime="{{ object.start }}" title="{{ object.start }}">
 | 
			
		||||
                    {{ object.start|date:"H:i" }} - {{ object.end|date:"H:i" }}
 | 
			
		||||
                </time>
 | 
			
		||||
            </td>
 | 
			
		||||
            <td>{% include "aircox_web/diffusion_item.html" %}</td>
 | 
			
		||||
        {% else %}
 | 
			
		||||
            <td>
 | 
			
		||||
                <time datetime="{{ object.date }}" title="{{ object.date }}">
 | 
			
		||||
                    {{ object.date|date:"H:i" }}
 | 
			
		||||
                </time>
 | 
			
		||||
            </td>
 | 
			
		||||
            <td>{% include "aircox_web/log_item.html" %}</td>
 | 
			
		||||
        {% endif %}
 | 
			
		||||
        </tr>
 | 
			
		||||
        {% endfor %}
 | 
			
		||||
    </table>
 | 
			
		||||
</section>
 | 
			
		||||
{% endblock %}
 | 
			
		||||
 | 
			
		||||
@ -1,8 +1,59 @@
 | 
			
		||||
{% 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 %}" />
 | 
			
		||||
 | 
			
		||||
{% block title %}{{ page.title }} -- {{ block.super }}{% endblock %}
 | 
			
		||||
        {% 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 main %}
 | 
			
		||||
<h1 class="title">{{ page.title }}</h1>
 | 
			
		||||
{% endblock %}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										27
									
								
								aircox_web/templates/aircox_web/program.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								aircox_web/templates/aircox_web/program.html
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,27 @@
 | 
			
		||||
{% 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>
 | 
			
		||||
 | 
			
		||||
{{ block.super }}
 | 
			
		||||
 | 
			
		||||
{% endblock %}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										52
									
								
								aircox_web/templates/aircox_web/timetable.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								aircox_web/templates/aircox_web/timetable.html
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,52 @@
 | 
			
		||||
{% extends "aircox_web/page.html" %}
 | 
			
		||||
{% load i18n aircox_web %}
 | 
			
		||||
 | 
			
		||||
{% block main %}
 | 
			
		||||
{{ block.super }}
 | 
			
		||||
 | 
			
		||||
<section class="section">
 | 
			
		||||
    <h3 class="subtitle size-3">
 | 
			
		||||
        {% blocktrans %}From <b>{{ start }}</b> to <b>{{ end }}</b>{% endblocktrans %}
 | 
			
		||||
    </h3>
 | 
			
		||||
 | 
			
		||||
    {% unique_id "timetable" as timetable_id %}
 | 
			
		||||
    <a-tabs default="{{ date }}">
 | 
			
		||||
        <template v-slot:tabs="scope" noscript="hidden">
 | 
			
		||||
            <li><a href="{% url "timetable" date=prev_date %}"><</a></li>
 | 
			
		||||
 | 
			
		||||
            {% for day in by_date.keys %}
 | 
			
		||||
            <a-tab value="{{ day }}">
 | 
			
		||||
                <a href="{% url "timetable" date=day %}">
 | 
			
		||||
                    {{ day|date:"D. d" }}
 | 
			
		||||
                </a>
 | 
			
		||||
            </a-tab>
 | 
			
		||||
            {% endfor %}
 | 
			
		||||
 | 
			
		||||
            <li>
 | 
			
		||||
                <a href="{% url "timetable" date=next_date %}">></a>
 | 
			
		||||
            </li>
 | 
			
		||||
        </template>
 | 
			
		||||
 | 
			
		||||
        <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>
 | 
			
		||||
            <div id="{{timetable_id}}-{{ day|date:"Y-m-d" }}" v-if="value == '{{ day }}'">
 | 
			
		||||
                {% for object in diffusions %}
 | 
			
		||||
                <div class="columns">
 | 
			
		||||
                    <div class="column is-one-fifth has-text-right">
 | 
			
		||||
                        <time datetime="{{ object.start|date:"c" }}">
 | 
			
		||||
                            {{ object.start|date:"H:i" }} - {{ object.end|date:"H:i" }}
 | 
			
		||||
                        </time>
 | 
			
		||||
                    </div>
 | 
			
		||||
                    <div class="column">
 | 
			
		||||
                        {% include "aircox_web/diffusion_item.html" %}
 | 
			
		||||
                    </div>
 | 
			
		||||
                </div>
 | 
			
		||||
                {% endfor %}
 | 
			
		||||
            </div>
 | 
			
		||||
            {% endfor %}
 | 
			
		||||
        </template>
 | 
			
		||||
    </a-tabs>
 | 
			
		||||
</section>
 | 
			
		||||
{% endblock %}
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										28
									
								
								aircox_web/templatetags/aircox_web.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								aircox_web/templatetags/aircox_web.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,28 @@
 | 
			
		||||
import random
 | 
			
		||||
 | 
			
		||||
from django import template
 | 
			
		||||
 | 
			
		||||
from aircox import models as aircox
 | 
			
		||||
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:
 | 
			
		||||
            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)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -1,9 +1,23 @@
 | 
			
		||||
from django.conf.urls import url
 | 
			
		||||
from django.urls import path, register_converter
 | 
			
		||||
 | 
			
		||||
from . import views
 | 
			
		||||
from . import views, models
 | 
			
		||||
from .converters import PagePathConverter, DateConverter
 | 
			
		||||
 | 
			
		||||
register_converter(PagePathConverter, 'page_path')
 | 
			
		||||
register_converter(DateConverter, 'date')
 | 
			
		||||
 | 
			
		||||
urlpatterns = [
 | 
			
		||||
    url(r"^(?P<path>[-\w/]+)/$", views.route_page, name="page"),
 | 
			
		||||
    url(r"^$", views.route_page, name="root"),
 | 
			
		||||
    path('diffusions/',
 | 
			
		||||
         views.TimetableView.as_view(), name='timetable'),
 | 
			
		||||
    path('diffusions/<date: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('<page_path:path>', views.route_page, name='page'),
 | 
			
		||||
]
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -1,48 +1,241 @@
 | 
			
		||||
from django.db.models import Q
 | 
			
		||||
from django.shortcuts import get_object_or_404, render
 | 
			
		||||
from django.views.generic.base import TemplateView
 | 
			
		||||
from collections import OrderedDict, deque
 | 
			
		||||
import datetime
 | 
			
		||||
 | 
			
		||||
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.base import TemplateResponseMixin, ContextMixin
 | 
			
		||||
 | 
			
		||||
from content_editor.contents import contents_for_item
 | 
			
		||||
 | 
			
		||||
from aircox import models as aircox
 | 
			
		||||
from .models import Site, Page
 | 
			
		||||
from .renderer import site_renderer, page_renderer
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def route_page(request, path=None, *args, site=None, **kwargs):
 | 
			
		||||
def route_page(request, path=None, *args, model=None, site=None, **kwargs):
 | 
			
		||||
    """
 | 
			
		||||
    Route request to page of the provided path. If model is provided, uses
 | 
			
		||||
    it.
 | 
			
		||||
    """
 | 
			
		||||
    # TODO/FIXME: django site framework | site from request host
 | 
			
		||||
    # TODO: extra page kwargs (as in pepr)
 | 
			
		||||
    site = Site.objects.all().order_by('-default').first() \
 | 
			
		||||
           if site is None else site
 | 
			
		||||
 | 
			
		||||
    model = model if model is not None else Page
 | 
			
		||||
    page = get_object_or_404(
 | 
			
		||||
        # TODO: published
 | 
			
		||||
        Page.objects.select_subclasses()
 | 
			
		||||
                    .filter(Q(status=Page.STATUS.published) |
 | 
			
		||||
                            Q(status=Page.STATUS.announced)),
 | 
			
		||||
        path="/{}/".format(path) if path else "/",
 | 
			
		||||
        model.objects.select_subclasses().active(),
 | 
			
		||||
        path=path
 | 
			
		||||
    )
 | 
			
		||||
    kwargs['page'] = page
 | 
			
		||||
    return page.view(request, *args, site=site, **kwargs)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class PageView(TemplateView):
 | 
			
		||||
    """ Base view class for pages. """
 | 
			
		||||
    template_name = 'aircox_web/page.html'
 | 
			
		||||
 | 
			
		||||
class BaseView(TemplateResponseMixin, ContextMixin):
 | 
			
		||||
    title = None
 | 
			
		||||
    site = None
 | 
			
		||||
 | 
			
		||||
    def dispatch(self, request, *args, site=None, **kwargs):
 | 
			
		||||
        self.site = site if site is not None else \
 | 
			
		||||
            Site.objects.all().order_by('-default').first()
 | 
			
		||||
        return super().dispatch(request, *args, **kwargs)
 | 
			
		||||
 | 
			
		||||
    def get_context_data(self, **kwargs):
 | 
			
		||||
        if kwargs.get('site_regions') is None:
 | 
			
		||||
            contents = contents_for_item(self.site, site_renderer._renderers.keys())
 | 
			
		||||
            kwargs['site_regions'] = contents.render_regions(site_renderer)
 | 
			
		||||
 | 
			
		||||
        kwargs.setdefault('site', self.site)
 | 
			
		||||
        if self.title is not None:
 | 
			
		||||
            kwargs.setdefault('title', self.title)
 | 
			
		||||
        return super().get_context_data(**kwargs)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class ArticleView(BaseView, TemplateView):
 | 
			
		||||
    """ Base view class for pages. """
 | 
			
		||||
    template_name = 'aircox_web/article.html'
 | 
			
		||||
    page = None
 | 
			
		||||
 | 
			
		||||
    def get_context_data(self, **kwargs):
 | 
			
		||||
        page = kwargs.setdefault('page', self.page or self.kwargs.get('site'))
 | 
			
		||||
        site = kwargs.setdefault('site', self.site or self.kwargs.get('site'))
 | 
			
		||||
 | 
			
		||||
        # 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)
 | 
			
		||||
 | 
			
		||||
        if kwargs.get('site_regions') is None:
 | 
			
		||||
            contents = contents_for_item(site, site_renderer._renderers.keys())
 | 
			
		||||
            kwargs['site_regions'] = contents.render_regions(site_renderer)
 | 
			
		||||
        kwargs.setdefault('title', page.title)
 | 
			
		||||
        return super().get_context_data(**kwargs)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class ProgramView(ArticleView):
 | 
			
		||||
    """ Base view class for pages. """
 | 
			
		||||
    template_name = 'aircox_web/program.html'
 | 
			
		||||
    next_diffs_count = 5
 | 
			
		||||
 | 
			
		||||
    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,
 | 
			
		||||
        )
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class DiffusionView(ArticleView):
 | 
			
		||||
    template_name = 'aircox_web/diffusion.html'
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class DiffusionsView(BaseView, ListView):
 | 
			
		||||
    template_name = 'aircox_web/diffusions.html'
 | 
			
		||||
    model = aircox.Diffusion
 | 
			
		||||
    paginate_by = 10
 | 
			
		||||
    title = _('Diffusions')
 | 
			
		||||
    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_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')
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class TimetableView(BaseView, ListView):
 | 
			
		||||
    """ View for timetables """
 | 
			
		||||
    template_name = 'aircox_web/timetable.html'
 | 
			
		||||
    model = aircox.Diffusion
 | 
			
		||||
 | 
			
		||||
    title = _('Timetable')
 | 
			
		||||
 | 
			
		||||
    date = None
 | 
			
		||||
    start = None
 | 
			
		||||
    end = None
 | 
			
		||||
 | 
			
		||||
    def get_queryset(self):
 | 
			
		||||
        self.date = self.kwargs.get('date', datetime.date.today())
 | 
			
		||||
        self.start = self.date - datetime.timedelta(days=self.date.weekday())
 | 
			
		||||
        self.end = self.date + datetime.timedelta(days=7-self.date.weekday())
 | 
			
		||||
        return super().get_queryset().station(self.site.station) \
 | 
			
		||||
                                     .range(self.start, self.end) \
 | 
			
		||||
                                     .order_by('start')
 | 
			
		||||
 | 
			
		||||
    def get_context_data(self, **kwargs):
 | 
			
		||||
        # regoup by dates
 | 
			
		||||
        by_date = OrderedDict()
 | 
			
		||||
        date = self.start
 | 
			
		||||
        while date < self.end:
 | 
			
		||||
            by_date[date] = []
 | 
			
		||||
            date += datetime.timedelta(days=1)
 | 
			
		||||
 | 
			
		||||
        for diffusion in self.object_list:
 | 
			
		||||
            if not diffusion.date in by_date:
 | 
			
		||||
                continue
 | 
			
		||||
            by_date[diffusion.date].append(diffusion)
 | 
			
		||||
 | 
			
		||||
        return super().get_context_data(
 | 
			
		||||
            by_date=by_date,
 | 
			
		||||
            date=self.date,
 | 
			
		||||
            start=self.start,
 | 
			
		||||
            end=self.end - datetime.timedelta(days=1),
 | 
			
		||||
            prev_date=self.start - datetime.timedelta(days=1),
 | 
			
		||||
            next_date=self.end + datetime.timedelta(days=1),
 | 
			
		||||
            **kwargs
 | 
			
		||||
        )
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class LogViewBase(ListView):
 | 
			
		||||
    station = None
 | 
			
		||||
    date = None
 | 
			
		||||
    delta = None
 | 
			
		||||
 | 
			
		||||
    def get_queryset(self):
 | 
			
		||||
        # only get logs for tracks: log for diffusion will be retrieved
 | 
			
		||||
        # by the diffusions' queryset.
 | 
			
		||||
        return super().get_queryset().station(self.station).on_air() \
 | 
			
		||||
                      .at(self.date).filter(track__isnull=False)
 | 
			
		||||
 | 
			
		||||
    def get_diffusions_queryset(self):
 | 
			
		||||
        return aircox.Diffusion.objects.station(self.station).on_air() \
 | 
			
		||||
                               .at(self.date)
 | 
			
		||||
 | 
			
		||||
    def get_object_list(self, queryset):
 | 
			
		||||
        diffs = deque(self.get_diffusions_queryset().order_by('start'))
 | 
			
		||||
        logs = list(queryset.order_by('date'))
 | 
			
		||||
        if not len(diffs):
 | 
			
		||||
            return logs
 | 
			
		||||
 | 
			
		||||
        object_list = []
 | 
			
		||||
        diff = diffs.popleft()
 | 
			
		||||
        last_collision = None
 | 
			
		||||
 | 
			
		||||
        # diff.start < log on first diff
 | 
			
		||||
        # diff.end > log on last diff
 | 
			
		||||
 | 
			
		||||
        for index, log in enumerate(logs):
 | 
			
		||||
            # get next diff
 | 
			
		||||
            if diff.end < log.date:
 | 
			
		||||
                diff = diffs.popleft() if len(diffs) else None
 | 
			
		||||
 | 
			
		||||
            # no more diff that can collide: return list
 | 
			
		||||
            if diff is None:
 | 
			
		||||
                return object_list + logs[index:]
 | 
			
		||||
 | 
			
		||||
            # diff colliding with log
 | 
			
		||||
            if diff.start <= log.date <= diff.end:
 | 
			
		||||
                if object_list[-1] is not diff:
 | 
			
		||||
                    object_list.append(diff)
 | 
			
		||||
                last_collision = log
 | 
			
		||||
            else:
 | 
			
		||||
                # add last colliding log: track
 | 
			
		||||
                if last_collision is not None:
 | 
			
		||||
                    object_list.append(last_collision)
 | 
			
		||||
 | 
			
		||||
                object_list.append(log)
 | 
			
		||||
                last_collision = None
 | 
			
		||||
        return object_list
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class LogsView(BaseView, LogViewBase):
 | 
			
		||||
    """ View for timetables """
 | 
			
		||||
    template_name = 'aircox_web/logs.html'
 | 
			
		||||
    model = aircox.Log
 | 
			
		||||
    title = _('Logs')
 | 
			
		||||
 | 
			
		||||
    date = None
 | 
			
		||||
    max_age = 10
 | 
			
		||||
 | 
			
		||||
    min_date = None
 | 
			
		||||
 | 
			
		||||
    def get(self, request, *args, **kwargs):
 | 
			
		||||
        self.station = self.site.station
 | 
			
		||||
 | 
			
		||||
        today = datetime.date.today()
 | 
			
		||||
        self.min_date = today - datetime.timedelta(days=self.max_age)
 | 
			
		||||
        self.date = min(max(self.min_date, self.kwargs['date']), today) \
 | 
			
		||||
            if 'date' in self.kwargs else today
 | 
			
		||||
        return super().get(request, *args, **kwargs)
 | 
			
		||||
 | 
			
		||||
    def get_context_data(self, **kwargs):
 | 
			
		||||
        today = datetime.date.today()
 | 
			
		||||
        max_date = min(max(self.date + datetime.timedelta(days=3),
 | 
			
		||||
                           self.min_date + datetime.timedelta(days=6)), today)
 | 
			
		||||
 | 
			
		||||
        return super().get_context_data(
 | 
			
		||||
            date=self.date,
 | 
			
		||||
            min_date=self.min_date,
 | 
			
		||||
            dates=(date for date in (
 | 
			
		||||
                max_date - datetime.timedelta(days=i)
 | 
			
		||||
                for i in range(0, 7)) if date >= self.min_date
 | 
			
		||||
            ),
 | 
			
		||||
            object_list=self.get_object_list(self.object_list),
 | 
			
		||||
            **kwargs
 | 
			
		||||
        )
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -3,7 +3,7 @@ const webpack = require('webpack');
 | 
			
		||||
 | 
			
		||||
const MiniCssExtractPlugin = require("mini-css-extract-plugin");
 | 
			
		||||
// const { createLodashAliases } = require('lodash-loader');
 | 
			
		||||
const { VueLoaderPlugin } = require('vue-loader');
 | 
			
		||||
const VueLoaderPlugin = require('vue-loader/lib/plugin');
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
module.exports = (env, argv) => Object({
 | 
			
		||||
@ -29,6 +29,13 @@ module.exports = (env, argv) => Object({
 | 
			
		||||
 | 
			
		||||
                    test: /[\\/]node_modules[\\/]/,
 | 
			
		||||
                },
 | 
			
		||||
 | 
			
		||||
                /*noscript: {
 | 
			
		||||
                    name: 'noscript',
 | 
			
		||||
                    chunks: 'initial',
 | 
			
		||||
                    enforce: true,
 | 
			
		||||
                    test: /noscript/,
 | 
			
		||||
                }*/
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    },
 | 
			
		||||
@ -43,6 +50,7 @@ module.exports = (env, argv) => Object({
 | 
			
		||||
 | 
			
		||||
    module: {
 | 
			
		||||
        rules: [
 | 
			
		||||
            { test: /\.vue$/, loader: 'vue-loader' },
 | 
			
		||||
            {
 | 
			
		||||
                test: /\/node_modules\//,
 | 
			
		||||
                sideEffects: false
 | 
			
		||||
@ -64,7 +72,6 @@ module.exports = (env, argv) => Object({
 | 
			
		||||
                    }
 | 
			
		||||
                }],
 | 
			
		||||
            },
 | 
			
		||||
            { test: /\.vue$/, use: 'vue-loader' },
 | 
			
		||||
        ],
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
		Reference in New Issue
	
	Block a user