ProgramPageListView + article list view
This commit is contained in:
parent
abaccf9ded
commit
595af5a69d
Binary file not shown.
|
@ -7220,7 +7220,8 @@ aside .media .content {
|
||||||
font-size: 1.5rem !important;
|
font-size: 1.5rem !important;
|
||||||
height: 2.5em !important; }
|
height: 2.5em !important; }
|
||||||
.player .media-content {
|
.player .media-content {
|
||||||
padding: 0.2em; }
|
padding-top: 0.4em;
|
||||||
|
padding-left: 0.4em; }
|
||||||
.player .button {
|
.player .button {
|
||||||
font-size: 1.5rem !important;
|
font-size: 1.5rem !important;
|
||||||
height: 2.5em;
|
height: 2.5em;
|
||||||
|
|
|
@ -393,7 +393,7 @@ eval("//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n\n/* harmony default export */ __w
|
||||||
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
||||||
|
|
||||||
"use strict";
|
"use strict";
|
||||||
eval("/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"a\", function() { return render; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"b\", function() { return staticRenderFns; });\nvar render = function() {\n var _vm = this\n var _h = _vm.$createElement\n var _c = _vm._self._c || _h\n return _c(\"div\", { staticClass: \"media\" }, [\n _c(\"div\", { staticClass: \"media-left\" }, [\n _c(\n \"div\",\n {\n staticClass: \"button\",\n attrs: { title: _vm.buttonTitle, \"aria-label\": _vm.buttonTitle },\n on: {\n click: function($event) {\n return _vm.toggle()\n }\n }\n },\n [\n _vm.playing\n ? _c(\"span\", { staticClass: \"fas fa-pause\" })\n : _c(\"span\", { staticClass: \"fas fa-play\" })\n ]\n ),\n _vm._v(\" \"),\n _c(\n \"audio\",\n {\n ref: \"audio\",\n on: {\n playing: _vm.onChange,\n ended: _vm.onChange,\n pause: _vm.onChange\n }\n },\n [_vm._t(\"sources\")],\n 2\n )\n ]),\n _vm._v(\" \"),\n _vm.onAir && _vm.onAir.cover\n ? _c(\"div\", { staticClass: \"media-left media-cover\" }, [\n _c(\"img\", { staticClass: \"cover\", attrs: { src: _vm.onAir.cover } })\n ])\n : _vm._e(),\n _vm._v(\" \"),\n _vm.onAir && _vm.onAir.type == \"track\"\n ? _c(\n \"div\",\n { staticClass: \"media-content\" },\n [_vm._t(\"track\", null, { onAir: _vm.onAir, liveInfo: _vm.liveInfo })],\n 2\n )\n : _vm.onAir && _vm.onAir.type == \"diffusion\"\n ? _c(\n \"div\",\n { staticClass: \"media-content\" },\n [\n _vm._t(\"diffusion\", null, {\n onAir: _vm.onAir,\n liveInfo: _vm.liveInfo\n })\n ],\n 2\n )\n : _c(\"div\", [_vm._t(\"empty\")], 2)\n ])\n}\nvar staticRenderFns = []\nrender._withStripped = true\n\n\n\n//# sourceURL=webpack:///./assets/vue/player.vue?./node_modules/vue-loader/lib/loaders/templateLoader.js??vue-loader-options!./node_modules/vue-loader/lib??vue-loader-options");
|
eval("/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"a\", function() { return render; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"b\", function() { return staticRenderFns; });\nvar render = function() {\n var _vm = this\n var _h = _vm.$createElement\n var _c = _vm._self._c || _h\n return _c(\"div\", { staticClass: \"media\" }, [\n _c(\"div\", { staticClass: \"media-left\" }, [\n _c(\n \"div\",\n {\n staticClass: \"button\",\n attrs: { title: _vm.buttonTitle, \"aria-label\": _vm.buttonTitle },\n on: {\n click: function($event) {\n return _vm.toggle()\n }\n }\n },\n [\n _vm.playing\n ? _c(\"span\", { staticClass: \"fas fa-pause\" })\n : _c(\"span\", { staticClass: \"fas fa-play\" })\n ]\n ),\n _vm._v(\" \"),\n _c(\n \"audio\",\n {\n ref: \"audio\",\n on: {\n playing: _vm.onChange,\n ended: _vm.onChange,\n pause: _vm.onChange\n }\n },\n [_vm._t(\"sources\")],\n 2\n )\n ]),\n _vm._v(\" \"),\n _vm.onAir && _vm.onAir.cover\n ? _c(\"div\", { staticClass: \"media-left media-cover\" }, [\n _c(\"img\", { staticClass: \"cover\", attrs: { src: _vm.onAir.cover } })\n ])\n : _vm._e(),\n _vm._v(\" \"),\n _vm.onAir && _vm.onAir.type == \"track\"\n ? _c(\n \"div\",\n { staticClass: \"media-content\" },\n [_vm._t(\"track\", null, { onAir: _vm.onAir, liveInfo: _vm.liveInfo })],\n 2\n )\n : _vm.onAir && _vm.onAir.type == \"diffusion\"\n ? _c(\n \"div\",\n { staticClass: \"media-content\" },\n [\n _vm._t(\"diffusion\", null, {\n onAir: _vm.onAir,\n liveInfo: _vm.liveInfo\n })\n ],\n 2\n )\n : _c(\"div\", { staticClass: \"media-content\" }, [_vm._t(\"empty\")], 2)\n ])\n}\nvar staticRenderFns = []\nrender._withStripped = true\n\n\n\n//# sourceURL=webpack:///./assets/vue/player.vue?./node_modules/vue-loader/lib/loaders/templateLoader.js??vue-loader-options!./node_modules/vue-loader/lib??vue-loader-options");
|
||||||
|
|
||||||
/***/ }),
|
/***/ }),
|
||||||
|
|
||||||
|
|
|
@ -42,6 +42,10 @@
|
||||||
</h4>
|
</h4>
|
||||||
<div class="">[[ onAir.info ]]</div>
|
<div class="">[[ onAir.info ]]</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
<template v-slot:empty>
|
||||||
|
<h4 class="title is-4">{{ station.name }}</h4>
|
||||||
|
</template>
|
||||||
</a-player>
|
</a-player>
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
|
@ -39,6 +39,8 @@ urls = [
|
||||||
views.ProgramDetailView.as_view(), name='program-detail'),
|
views.ProgramDetailView.as_view(), name='program-detail'),
|
||||||
path(_('programs/<slug:program_slug>/episodes/'),
|
path(_('programs/<slug:program_slug>/episodes/'),
|
||||||
views.EpisodeListView.as_view(), name='diffusion-list'),
|
views.EpisodeListView.as_view(), name='diffusion-list'),
|
||||||
|
path(_('programs/<slug:program_slug>/articles/'),
|
||||||
|
views.ArticleListView.as_view(), name='article-list'),
|
||||||
|
|
||||||
path(_('episodes/'),
|
path(_('episodes/'),
|
||||||
views.EpisodeListView.as_view(), name='diffusion-list'),
|
views.EpisodeListView.as_view(), name='diffusion-list'),
|
||||||
|
|
|
@ -25,7 +25,7 @@ class LiveAPIView(BaseLogListView, BaseAPIView, ListAPIView):
|
||||||
queryset = Log.objects.all()
|
queryset = Log.objects.all()
|
||||||
|
|
||||||
def get(self, *args, **kwargs):
|
def get(self, *args, **kwargs):
|
||||||
self.min_date = tz.now() - tz.timedelta(minutes=5)
|
self.min_date = tz.now() - tz.timedelta(minutes=20)
|
||||||
return super().get(*args, **kwargs)
|
return super().get(*args, **kwargs)
|
||||||
|
|
||||||
def get_serializer(self, queryset, *args, **kwargs):
|
def get_serializer(self, queryset, *args, **kwargs):
|
||||||
|
|
|
@ -1,12 +1,14 @@
|
||||||
from ..models import Article
|
from ..models import Article
|
||||||
from .page import PageListView
|
from .program import ProgramPageListView
|
||||||
|
|
||||||
|
|
||||||
__all__ = ['ArticleListView']
|
__all__ = ['ArticleListView']
|
||||||
|
|
||||||
|
|
||||||
class ArticleListView(PageListView):
|
class ArticleListView(ProgramPageListView):
|
||||||
model = Article
|
model = Article
|
||||||
|
template_name = 'aircox/article_list.html'
|
||||||
|
show_headline = True
|
||||||
is_static = False
|
is_static = False
|
||||||
|
|
||||||
def get_queryset(self):
|
def get_queryset(self):
|
||||||
|
|
|
@ -7,8 +7,7 @@ from django.views.generic import ListView
|
||||||
|
|
||||||
from ..models import Diffusion, Episode, Page, Program, Sound
|
from ..models import Diffusion, Episode, Page, Program, Sound
|
||||||
from .base import BaseView
|
from .base import BaseView
|
||||||
from .page import PageListView
|
from .program import ProgramPageDetailView, ProgramPageListView
|
||||||
from .program import ProgramPageDetailView
|
|
||||||
|
|
||||||
|
|
||||||
__all__ = ['EpisodeDetailView', 'DiffusionListView', 'TimetableView']
|
__all__ = ['EpisodeDetailView', 'DiffusionListView', 'TimetableView']
|
||||||
|
@ -30,32 +29,11 @@ class EpisodeDetailView(ProgramPageDetailView):
|
||||||
return super().get_context_data(**kwargs)
|
return super().get_context_data(**kwargs)
|
||||||
|
|
||||||
|
|
||||||
# TODO: pagination: in template, only a limited number of pages displayed
|
class EpisodeListView(ProgramPageListView):
|
||||||
class EpisodeListView(PageListView):
|
|
||||||
model = Episode
|
model = Episode
|
||||||
|
template_name = 'aircox/diffusion_list.html'
|
||||||
item_template_name = 'aircox/episode_item.html'
|
item_template_name = 'aircox/episode_item.html'
|
||||||
show_headline = True
|
show_headline = True
|
||||||
template_name = 'aircox/diffusion_list.html'
|
|
||||||
program = None
|
|
||||||
|
|
||||||
def get(self, request, *args, **kwargs):
|
|
||||||
program_slug = kwargs.get('program_slug')
|
|
||||||
if program_slug:
|
|
||||||
self.program = get_object_or_404(Program, slug=program_slug)
|
|
||||||
return super().get(request, *args, **kwargs)
|
|
||||||
|
|
||||||
def get_queryset(self):
|
|
||||||
qs = super().get_queryset()
|
|
||||||
if self.program:
|
|
||||||
qs = qs.filter(program=self.program)
|
|
||||||
return qs
|
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
|
||||||
program = kwargs.setdefault('program', self.program)
|
|
||||||
if program is not None:
|
|
||||||
kwargs.setdefault('cover', program.cover)
|
|
||||||
kwargs.setdefault('parent', program)
|
|
||||||
return super().get_context_data(**kwargs)
|
|
||||||
|
|
||||||
|
|
||||||
class TimetableView(BaseView, ListView):
|
class TimetableView(BaseView, ListView):
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
|
|
||||||
|
from django.core.exceptions import FieldDoesNotExist
|
||||||
from django.http import Http404
|
from django.http import Http404
|
||||||
from django.views.generic import DetailView, ListView
|
from django.views.generic import DetailView, ListView
|
||||||
|
|
||||||
|
@ -45,10 +46,11 @@ class PageDetailView(BaseView, DetailView):
|
||||||
return super().get_context_data(**kwargs)
|
return super().get_context_data(**kwargs)
|
||||||
|
|
||||||
|
|
||||||
|
# TODO: pagination: in template, only a limited number of pages displayed
|
||||||
class PageListView(BaseView, ListView):
|
class PageListView(BaseView, ListView):
|
||||||
template_name = 'aircox/page_list.html'
|
template_name = 'aircox/page_list.html'
|
||||||
item_template_name = 'aircox/page_item.html'
|
item_template_name = 'aircox/page_item.html'
|
||||||
paginate_by = 10
|
paginate_by = 20
|
||||||
show_headline = True
|
show_headline = True
|
||||||
show_side_nav = True
|
show_side_nav = True
|
||||||
categories = None
|
categories = None
|
||||||
|
@ -82,3 +84,4 @@ class PageListView(BaseView, ListView):
|
||||||
return super().get_context_data(**kwargs)
|
return super().get_context_data(**kwargs)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -1,15 +1,19 @@
|
||||||
|
from django.core.exceptions import ObjectDoesNotExist
|
||||||
|
from django.shortcuts import get_object_or_404
|
||||||
|
|
||||||
from aircox.models import Episode, Program
|
from aircox.models import Episode, Program
|
||||||
from .page import PageDetailView
|
from .page import PageDetailView, PageListView
|
||||||
|
|
||||||
|
|
||||||
__all__ = ['ProgramPageDetailView', 'ProgramDetailView']
|
__all__ = ['ProgramPageDetailView', 'ProgramDetailView']
|
||||||
|
|
||||||
|
|
||||||
class ProgramPageDetailView(PageDetailView):
|
class ProgramPageDetailView(PageDetailView):
|
||||||
""" Base view class for rendering content of a specific programs. """
|
"""
|
||||||
|
Base view class for a page that is displayed as a program's child page.
|
||||||
|
"""
|
||||||
show_side_nav = True
|
show_side_nav = True
|
||||||
list_count=5
|
list_count = 5
|
||||||
|
|
||||||
def get_episodes_queryset(self, program):
|
def get_episodes_queryset(self, program):
|
||||||
return program.episode_set.published().order_by('-date')
|
return program.episode_set.published().order_by('-date')
|
||||||
|
@ -21,9 +25,40 @@ class ProgramPageDetailView(PageDetailView):
|
||||||
program=program, episodes=episodes[:self.list_count], **kwargs)
|
program=program, episodes=episodes[:self.list_count], **kwargs)
|
||||||
|
|
||||||
|
|
||||||
|
class ProgramPageListView(PageListView):
|
||||||
|
"""
|
||||||
|
Base list view class rendering pages as a program's child page.
|
||||||
|
Retrieved program from it slug provided by `kwargs['program_slug']`.
|
||||||
|
|
||||||
|
This view class can be used with or without providing a program.
|
||||||
|
"""
|
||||||
|
program = None
|
||||||
|
|
||||||
|
def get(self, request, *args, **kwargs):
|
||||||
|
slug = kwargs.get('program_slug', None)
|
||||||
|
if slug is not None:
|
||||||
|
self.program = get_object_or_404(
|
||||||
|
Program.objects.select_related('cover'), slug=slug)
|
||||||
|
return super().get(request, *args, **kwargs)
|
||||||
|
|
||||||
|
def get_queryset(self):
|
||||||
|
return super().get_queryset().filter(program=self.program) \
|
||||||
|
if self.program else super().get_queryset()
|
||||||
|
|
||||||
|
def get_context_data(self, **kwargs):
|
||||||
|
program = kwargs.setdefault('program', self.program)
|
||||||
|
if program is not None:
|
||||||
|
kwargs.setdefault('cover', program.cover)
|
||||||
|
kwargs.setdefault('parent', program)
|
||||||
|
return super().get_context_data(**kwargs)
|
||||||
|
|
||||||
|
|
||||||
class ProgramDetailView(ProgramPageDetailView):
|
class ProgramDetailView(ProgramPageDetailView):
|
||||||
model = Program
|
model = Program
|
||||||
|
|
||||||
|
def get_articles_queryset(self, program):
|
||||||
|
return program.article_set.published().order_by('-date')
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
def get_context_data(self, **kwargs):
|
||||||
kwargs.setdefault('program', self.object)
|
kwargs.setdefault('program', self.object)
|
||||||
return super().get_context_data(**kwargs)
|
return super().get_context_data(**kwargs)
|
||||||
|
|
|
@ -105,7 +105,8 @@ aside {
|
||||||
}
|
}
|
||||||
|
|
||||||
.media-content {
|
.media-content {
|
||||||
padding: 0.2em;
|
padding-top: 0.4em;
|
||||||
|
padding-left: 0.4em;
|
||||||
}
|
}
|
||||||
|
|
||||||
.button {
|
.button {
|
||||||
|
|
|
@ -19,7 +19,7 @@
|
||||||
<div class="media-content" v-else-if="onAir && onAir.type == 'diffusion'">
|
<div class="media-content" v-else-if="onAir && onAir.type == 'diffusion'">
|
||||||
<slot name="diffusion" :onAir="onAir" :liveInfo="liveInfo"></slot>
|
<slot name="diffusion" :onAir="onAir" :liveInfo="liveInfo"></slot>
|
||||||
</div>
|
</div>
|
||||||
<div v-else><slot name="empty"></slot></div>
|
<div class="media-content" v-else><slot name="empty"></slot></div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
Loading…
Reference in New Issue
Block a user