remove feincms3 dependency
This commit is contained in:
parent
4b57cd6643
commit
4caca505c4
|
@ -3,7 +3,7 @@ import copy
|
||||||
from django.contrib import admin
|
from django.contrib import admin
|
||||||
from django.utils.translation import ugettext_lazy as _
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
from content_editor.admin import ContentEditor
|
from content_editor.admin import ContentEditor, ContentEditorInline
|
||||||
from feincms3 import plugins
|
from feincms3 import plugins
|
||||||
from feincms3.admin import TreeAdmin
|
from feincms3.admin import TreeAdmin
|
||||||
|
|
||||||
|
@ -16,8 +16,9 @@ from aircox.admin.mixins import UnrelatedInlineMixin
|
||||||
@admin.register(models.Site)
|
@admin.register(models.Site)
|
||||||
class SiteAdmin(ContentEditor):
|
class SiteAdmin(ContentEditor):
|
||||||
inlines = [
|
inlines = [
|
||||||
plugins.richtext.RichTextInline.create(models.SiteRichText),
|
ContentEditorInline.create(models.SiteRichText),
|
||||||
plugins.image.ImageInline.create(models.SiteImage),
|
ContentEditorInline.create(models.SiteImage),
|
||||||
|
ContentEditorInline.create(models.SiteLink),
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
|
@ -36,36 +37,58 @@ class PageDiffusionPlaylist(UnrelatedInlineMixin, TracksInline):
|
||||||
|
|
||||||
|
|
||||||
@admin.register(models.Page)
|
@admin.register(models.Page)
|
||||||
class PageAdmin(ContentEditor, TreeAdmin):
|
class PageAdmin(ContentEditor):
|
||||||
list_display = ["indented_title", "move_column", "is_active"]
|
list_display = ["title", "parent", "status"]
|
||||||
prepopulated_fields = {"slug": ("title",)}
|
prepopulated_fields = {"slug": ("title",)}
|
||||||
# readonly_fields = ('diffusion',)
|
# readonly_fields = ('diffusion',)
|
||||||
|
|
||||||
fieldsets = (
|
fieldsets = (
|
||||||
(_('Main'), {
|
(_('Main'), {
|
||||||
'fields': ['title', 'slug', 'by_program', 'summary'],
|
'fields': ['title', 'slug', 'as_program', 'headline'],
|
||||||
'classes': ('tabbed', 'uncollapse')
|
'classes': ('tabbed', 'uncollapse')
|
||||||
}),
|
}),
|
||||||
(_('Settings'), {
|
(_('Settings'), {
|
||||||
'fields': ['show_author', 'featured', 'allow_comments',
|
'fields': ['featured', 'allow_comments',
|
||||||
'status', 'static_path', 'path'],
|
'status', 'static_path', 'path'],
|
||||||
'classes': ('tabbed',)
|
'classes': ('tabbed',)
|
||||||
}),
|
}),
|
||||||
(_('Infos'), {
|
#(_('Infos'), {
|
||||||
'fields': ['diffusion'],
|
# 'fields': ['diffusion'],
|
||||||
'classes': ('tabbed',)
|
# 'classes': ('tabbed',)
|
||||||
}),
|
#}),
|
||||||
)
|
)
|
||||||
|
|
||||||
inlines = [
|
inlines = [
|
||||||
plugins.richtext.RichTextInline.create(models.PageRichText),
|
ContentEditorInline.create(models.PageRichText),
|
||||||
plugins.image.ImageInline.create(models.PageImage),
|
ContentEditorInline.create(models.PageImage),
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
@admin.register(models.DiffusionPage)
|
||||||
|
class DiffusionPageAdmin(PageAdmin):
|
||||||
|
fieldsets = copy.deepcopy(PageAdmin.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
|
||||||
|
|
||||||
|
|
||||||
|
@admin.register(models.ProgramPage)
|
||||||
|
class DiffusionPageAdmin(PageAdmin):
|
||||||
|
fieldsets = copy.deepcopy(PageAdmin.fieldsets)
|
||||||
|
fieldsets[1][1]['fields'].insert(0, 'program')
|
||||||
|
|
||||||
|
inlines = PageAdmin.inlines + [
|
||||||
|
PageDiffusionPlaylist
|
||||||
]
|
]
|
||||||
|
|
||||||
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
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -1,2 +1,3 @@
|
||||||
import './js';
|
import './js';
|
||||||
|
import './styles.scss';
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
import Vue from 'vue';
|
import Vue from 'vue';
|
||||||
import Buefy from 'buefy';
|
import Buefy from 'buefy';
|
||||||
import 'buefy/dist/buefy.css';
|
|
||||||
|
|
||||||
Vue.use(Buefy);
|
Vue.use(Buefy);
|
||||||
|
|
||||||
|
|
|
@ -1,16 +1,18 @@
|
||||||
|
from django.core.validators import RegexValidator
|
||||||
from django.db import models
|
from django.db import models
|
||||||
|
from django.db.models import F
|
||||||
|
from django.db.models.functions import Concat, Substr
|
||||||
from django.utils.translation import ugettext_lazy as _
|
from django.utils.translation import ugettext_lazy as _
|
||||||
from django.contrib.auth import models as auth
|
|
||||||
|
|
||||||
from content_editor.models import Region, create_plugin_base
|
from content_editor.models import Region, create_plugin_base
|
||||||
from feincms3 import plugins
|
|
||||||
from feincms3.pages import AbstractPage
|
|
||||||
|
|
||||||
from model_utils.models import TimeStampedModel, StatusModel
|
from model_utils.models import TimeStampedModel, StatusModel
|
||||||
|
from model_utils.managers import InheritanceManager
|
||||||
from model_utils import Choices
|
from model_utils import Choices
|
||||||
from filer.fields.image import FilerImageField
|
from filer.fields.image import FilerImageField
|
||||||
|
|
||||||
from aircox import models as aircox
|
from aircox import models as aircox
|
||||||
|
from . import plugins
|
||||||
|
|
||||||
|
|
||||||
class Site(models.Model):
|
class Site(models.Model):
|
||||||
|
@ -34,6 +36,11 @@ class Site(models.Model):
|
||||||
related_name='+',
|
related_name='+',
|
||||||
)
|
)
|
||||||
|
|
||||||
|
default = models.BooleanField(_('default site'),
|
||||||
|
default=False,
|
||||||
|
help_text=_('Use as default site'),
|
||||||
|
)
|
||||||
|
|
||||||
# meta descriptors
|
# meta descriptors
|
||||||
description = models.CharField(
|
description = models.CharField(
|
||||||
_('Description'), max_length=128,
|
_('Description'), max_length=128,
|
||||||
|
@ -52,38 +59,119 @@ class Site(models.Model):
|
||||||
|
|
||||||
SitePlugin = create_plugin_base(Site)
|
SitePlugin = create_plugin_base(Site)
|
||||||
|
|
||||||
class SiteRichText(plugins.richtext.RichText, SitePlugin):
|
class SiteRichText(plugins.RichText, SitePlugin):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
class SiteImage(plugins.Image, SitePlugin):
|
||||||
|
pass
|
||||||
|
|
||||||
class SiteImage(plugins.image.Image, SitePlugin):
|
class SiteLink(plugins.Link, SitePlugin):
|
||||||
caption = models.CharField(_("caption"), max_length=200, blank=True)
|
css_class="navbar-item"
|
||||||
|
|
||||||
|
|
||||||
|
#-----------------------------------------------------------------------
|
||||||
|
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 Page(AbstractPage, TimeStampedModel, StatusModel):
|
Inspired by Feincms3.
|
||||||
STATUS = Choices('draft', 'published')
|
"""
|
||||||
|
STATUS = Choices('draft', 'announced', 'published')
|
||||||
|
|
||||||
|
parent = models.ForeignKey(
|
||||||
|
'self', models.CASCADE,
|
||||||
|
verbose_name=_('parent page'),
|
||||||
|
blank=True, null=True,
|
||||||
|
)
|
||||||
|
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=r"^/(|.+/)$",
|
||||||
|
message=_("Path must start and end with a slash (/)."),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
)
|
||||||
|
static_path = models.BooleanField(
|
||||||
|
_('static path'), default=False,
|
||||||
|
help_text=_('Update path using parent\'s page path and page title')
|
||||||
|
)
|
||||||
|
|
||||||
|
objects = InheritanceManager()
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
abstract = True
|
||||||
|
|
||||||
|
def __init__(self, *args, **kwargs):
|
||||||
|
super().__init__(*args, **kwargs)
|
||||||
|
self._initial_path = self.path
|
||||||
|
self._initial_parent = self.parent
|
||||||
|
self._initial_slug = self.slug
|
||||||
|
|
||||||
|
def view(self, request, *args, **kwargs):
|
||||||
|
""" Page view function """
|
||||||
|
from django.http import HttpResponse
|
||||||
|
return HttpResponse('Not implemented')
|
||||||
|
|
||||||
|
def update_descendants(self):
|
||||||
|
""" Update descendants pages' path if required. """
|
||||||
|
if self.path == self._initial_path:
|
||||||
|
return
|
||||||
|
|
||||||
|
# FIXME: draft -> draft children?
|
||||||
|
expr = Concat(self.path, Substr(F('path'), len(self._initial_path)))
|
||||||
|
BasePage.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[-1] != '/':
|
||||||
|
self.path += '/'
|
||||||
|
if self.path[0] != '/':
|
||||||
|
self.path = '/' + 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)
|
||||||
|
|
||||||
|
|
||||||
|
class Page(BasePage, TimeStampedModel):
|
||||||
|
""" User's pages """
|
||||||
regions = [
|
regions = [
|
||||||
Region(key="main", title=_("Content")),
|
Region(key="main", title=_("Content")),
|
||||||
]
|
]
|
||||||
|
|
||||||
# metadata
|
# metadata
|
||||||
by = models.ForeignKey(
|
as_program = models.ForeignKey(
|
||||||
auth.User, models.SET_NULL, blank=True, null=True,
|
|
||||||
verbose_name=_('Author'),
|
|
||||||
)
|
|
||||||
by_program = models.ForeignKey(
|
|
||||||
aircox.Program, models.SET_NULL, blank=True, null=True,
|
aircox.Program, models.SET_NULL, blank=True, null=True,
|
||||||
related_name='authored_pages',
|
related_name='published_pages',
|
||||||
limit_choices_to={'schedule__isnull': False},
|
limit_choices_to={'schedule__isnull': False},
|
||||||
verbose_name=_('Show program as author'),
|
verbose_name=_('Show program as author'),
|
||||||
help_text=_("If nothing is selected, display user's name"),
|
help_text=_("Show program as author"),
|
||||||
)
|
)
|
||||||
|
|
||||||
# options
|
# options
|
||||||
show_author = models.BooleanField(
|
|
||||||
_('Show author'), default=True,
|
|
||||||
)
|
|
||||||
featured = models.BooleanField(
|
featured = models.BooleanField(
|
||||||
_('featured'), default=False,
|
_('featured'), default=False,
|
||||||
)
|
)
|
||||||
|
@ -92,36 +180,46 @@ class Page(AbstractPage, TimeStampedModel, StatusModel):
|
||||||
)
|
)
|
||||||
|
|
||||||
# content
|
# content
|
||||||
title = models.CharField(
|
headline = models.TextField(
|
||||||
_('title'), max_length=64,
|
_('headline'), max_length=128, blank=True, null=True,
|
||||||
)
|
|
||||||
summary = models.TextField(
|
|
||||||
_('Summary'),
|
|
||||||
max_length=128, blank=True, null=True,
|
|
||||||
)
|
)
|
||||||
cover = FilerImageField(
|
cover = FilerImageField(
|
||||||
on_delete=models.SET_NULL, null=True, blank=True,
|
on_delete=models.SET_NULL, null=True, blank=True,
|
||||||
verbose_name=_('Cover'),
|
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)
|
||||||
|
|
||||||
|
|
||||||
|
class DiffusionPage(Page):
|
||||||
diffusion = models.OneToOneField(
|
diffusion = models.OneToOneField(
|
||||||
aircox.Diffusion, models.CASCADE,
|
aircox.Diffusion, models.CASCADE,
|
||||||
blank=True, null=True,
|
blank=True, null=True,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class ProgramPage(Page):
|
||||||
program = models.OneToOneField(
|
program = models.OneToOneField(
|
||||||
aircox.Program, models.CASCADE,
|
aircox.Program, models.CASCADE,
|
||||||
blank=True, null=True,
|
blank=True, null=True,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
#-----------------------------------------------------------------------
|
||||||
PagePlugin = create_plugin_base(Page)
|
PagePlugin = create_plugin_base(Page)
|
||||||
|
|
||||||
class PageRichText(plugins.richtext.RichText, PagePlugin):
|
class PageRichText(plugins.RichText, PagePlugin):
|
||||||
|
pass
|
||||||
|
|
||||||
|
class PageImage(plugins.Image, PagePlugin):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
class PageImage(plugins.image.Image, PagePlugin):
|
|
||||||
caption = models.CharField(_("caption"), max_length=200, blank=True)
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -7,19 +7,21 @@
|
||||||
"license": "AGPL",
|
"license": "AGPL",
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@fortawesome/fontawesome-free": "^5.8.2",
|
"@fortawesome/fontawesome-free": "^5.8.2",
|
||||||
"mini-css-extract-plugin": "^0.5.0",
|
"bulma": "^0.7.5",
|
||||||
"css-loader": "^2.1.1",
|
"css-loader": "^2.1.1",
|
||||||
|
"extract-text-webpack-plugin": "^4.0.0-beta.0",
|
||||||
"file-loader": "^3.0.1",
|
"file-loader": "^3.0.1",
|
||||||
|
"mini-css-extract-plugin": "^0.5.0",
|
||||||
|
"node-sass": "^4.12.0",
|
||||||
|
"sass-loader": "^7.1.0",
|
||||||
|
"style-loader": "^0.23.1",
|
||||||
"ttf-loader": "^1.0.2",
|
"ttf-loader": "^1.0.2",
|
||||||
"vue-loader": "^15.7.0",
|
"vue-loader": "^15.7.0",
|
||||||
"vue-style-loader": "^4.1.2",
|
"vue-style-loader": "^4.1.2",
|
||||||
"webpack": "^4.32.2",
|
"webpack": "^4.32.2",
|
||||||
"webpack-bundle-analyzer": "^3.3.2",
|
|
||||||
"webpack-bundle-tracker": "^0.4.2-beta",
|
|
||||||
"webpack-cli": "^3.3.2"
|
"webpack-cli": "^3.3.2"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"bootstrap": "^4.3.1",
|
|
||||||
"buefy": "^0.7.8",
|
"buefy": "^0.7.8",
|
||||||
"vue": "^2.6.10"
|
"vue": "^2.6.10"
|
||||||
}
|
}
|
||||||
|
|
12
aircox_web/plugins/richtext.py
Normal file
12
aircox_web/plugins/richtext.py
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
from django.db import models
|
||||||
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
|
from ckeditor.fields import RichTextField
|
||||||
|
|
||||||
|
|
||||||
|
class RichText(models.Model):
|
||||||
|
text = RichTextField(_('text'))
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
abstract = True
|
||||||
|
|
|
@ -1,35 +1,18 @@
|
||||||
from django.utils.html import format_html, mark_safe
|
from django.utils.html import format_html, mark_safe
|
||||||
from feincms3.renderer import TemplatePluginRenderer
|
from content_editor.renderer import PluginRenderer
|
||||||
|
|
||||||
from .models import *
|
from .models import *
|
||||||
|
|
||||||
|
|
||||||
site_renderer = TemplatePluginRenderer()
|
site_renderer = PluginRenderer()
|
||||||
site_renderer.register_string_renderer(
|
site_renderer._renderers.clear()
|
||||||
SiteRichText,
|
site_renderer.register(SiteRichText, lambda plugin: mark_safe(plugin.text))
|
||||||
lambda plugin: mark_safe(plugin.text),
|
site_renderer.register(SiteImage, lambda plugin: plugin.render())
|
||||||
)
|
site_renderer.register(SiteLink, lambda plugin: plugin.render())
|
||||||
site_renderer.register_string_renderer(
|
|
||||||
SiteImage,
|
|
||||||
lambda plugin: format_html(
|
|
||||||
'<figure><img src="{}" alt=""/><figcaption>{}</figcaption></figure>',
|
|
||||||
plugin.image.url,
|
|
||||||
plugin.caption,
|
|
||||||
),
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
page_renderer = TemplatePluginRenderer()
|
page_renderer = PluginRenderer()
|
||||||
page_renderer.register_string_renderer(
|
page_renderer._renderers.clear()
|
||||||
PageRichText,
|
page_renderer.register(PageRichText, lambda plugin: mark_safe(plugin.text))
|
||||||
lambda plugin: mark_safe(plugin.text),
|
page_renderer.register(PageImage, lambda plugin: plugin.render())
|
||||||
)
|
|
||||||
page_renderer.register_string_renderer(
|
|
||||||
PageImage,
|
|
||||||
lambda plugin: format_html(
|
|
||||||
'<figure><img src="{}" alt=""/><figcaption>{}</figcaption></figure>',
|
|
||||||
plugin.image.url,
|
|
||||||
plugin.caption,
|
|
||||||
),
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
{% load static thumbnail feincms3 %}
|
{% load static i18n thumbnail %}
|
||||||
<html>
|
<html>
|
||||||
<head>
|
<head>
|
||||||
<meta charset="utf-8">
|
<meta charset="utf-8">
|
||||||
|
@ -9,7 +9,7 @@
|
||||||
|
|
||||||
{% block assets %}
|
{% 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/main.css" %}"/>
|
||||||
<link rel="stylesheet" type="text/css" href="{% static "aircox_web/assets/vendor.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/main.js" %}"></script>
|
||||||
<script src="{% static "aircox_web/assets/vendor.js" %}"></script>
|
<script src="{% static "aircox_web/assets/vendor.js" %}"></script>
|
||||||
|
@ -20,18 +20,27 @@
|
||||||
{% block extra_head %}{% endblock %}
|
{% block extra_head %}{% endblock %}
|
||||||
</head>
|
</head>
|
||||||
<body id="app">
|
<body id="app">
|
||||||
<nav class="navbar" role="navigation" aria-label="main navigation">
|
<nav class="navbar has-shadow" role="navigation" aria-label="main navigation">
|
||||||
{% render_region regions "topnav" %}
|
<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>
|
</nav>
|
||||||
|
|
||||||
<div class="columns">
|
<div class="columns">
|
||||||
<aside class="column">
|
<aside class="column">
|
||||||
{% render_region regions "sidenav" %}
|
{{ site_regions.sidenav }}
|
||||||
</aside>
|
</aside>
|
||||||
<main class="column is-three-quarters">
|
<main class="column is-three-quarters">
|
||||||
{% block main %}
|
{% block main %}
|
||||||
<h1>{{ page.title }}</h1>
|
<h1>{{ page.title }}</h1>
|
||||||
{% render_region page_regions "main" %}
|
{{ regions.main }}
|
||||||
{% endblock main %}
|
{% endblock main %}
|
||||||
</main>
|
</main>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -3,7 +3,7 @@ from django.conf.urls import url
|
||||||
from . import views
|
from . import views
|
||||||
|
|
||||||
urlpatterns = [
|
urlpatterns = [
|
||||||
url(r"^(?P<path>[-\w/]+)/$", views.page_detail, name="page"),
|
url(r"^(?P<path>[-\w/]+)/$", views.route_page, name="page"),
|
||||||
url(r"^$", views.page_detail, name="root"),
|
url(r"^$", views.route_page, name="root"),
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
|
@ -1,22 +1,48 @@
|
||||||
|
from django.db.models import Q
|
||||||
from django.shortcuts import get_object_or_404, render
|
from django.shortcuts import get_object_or_404, render
|
||||||
|
from django.views.generic.base import TemplateView
|
||||||
|
|
||||||
from feincms3.regions import Regions
|
from content_editor.contents import contents_for_item
|
||||||
|
|
||||||
from .models import Site, Page
|
from .models import Site, Page
|
||||||
from .renderer import site_renderer, page_renderer
|
from .renderer import site_renderer, page_renderer
|
||||||
|
|
||||||
|
|
||||||
def page_detail(request, path=None):
|
def route_page(request, path=None, *args, site=None, **kwargs):
|
||||||
|
# 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
|
||||||
page = get_object_or_404(
|
page = get_object_or_404(
|
||||||
# TODO: published
|
# TODO: published
|
||||||
Page.objects.all(),
|
Page.objects.select_subclasses()
|
||||||
|
.filter(Q(status=Page.STATUS.published) |
|
||||||
|
Q(status=Page.STATUS.announced)),
|
||||||
path="/{}/".format(path) if path else "/",
|
path="/{}/".format(path) if path else "/",
|
||||||
)
|
)
|
||||||
site = Site.objects.all().first()
|
kwargs['page'] = page
|
||||||
return render(request, "aircox_web/page.html", {
|
return page.view(request, *args, site=site, **kwargs)
|
||||||
'site': site,
|
|
||||||
"regions": Regions.from_item(site, renderer=site_renderer, timeout=60),
|
|
||||||
"page": page,
|
class PageView(TemplateView):
|
||||||
"page_regions": Regions.from_item(page, renderer=page_renderer, timeout=60),
|
""" Base view class for pages. """
|
||||||
})
|
template_name = 'aircox_web/page.html'
|
||||||
|
|
||||||
|
site = None
|
||||||
|
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'))
|
||||||
|
|
||||||
|
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)
|
||||||
|
return super().get_context_data(**kwargs)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -48,9 +48,10 @@ module.exports = (env, argv) => Object({
|
||||||
sideEffects: false
|
sideEffects: false
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
test: /\.css$/,
|
test: /\.scss$/,
|
||||||
use: [ { loader: MiniCssExtractPlugin.loader },
|
use: [ { loader: MiniCssExtractPlugin.loader },
|
||||||
'css-loader' ]
|
{ loader: 'css-loader' },
|
||||||
|
{ loader: 'sass-loader' , options: { sourceMap: true }} ],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
// TODO: remove ttf eot svg
|
// TODO: remove ttf eot svg
|
||||||
|
@ -70,8 +71,6 @@ module.exports = (env, argv) => Object({
|
||||||
resolve: {
|
resolve: {
|
||||||
alias: {
|
alias: {
|
||||||
js: path.resolve(__dirname, 'assets/js'),
|
js: path.resolve(__dirname, 'assets/js'),
|
||||||
vue: path.resolve(__dirname, 'assets/vue'),
|
|
||||||
css: path.resolve(__dirname, 'assets/css'),
|
|
||||||
vue: 'vue/dist/vue.esm.browser.js',
|
vue: 'vue/dist/vue.esm.browser.js',
|
||||||
// buefy: 'buefy/dist/buefy.js',
|
// buefy: 'buefy/dist/buefy.js',
|
||||||
},
|
},
|
||||||
|
|
|
@ -9,11 +9,10 @@ mutagen>=1.37
|
||||||
pyyaml>=3.12
|
pyyaml>=3.12
|
||||||
|
|
||||||
django-filer>=1.5.0
|
django-filer>=1.5.0
|
||||||
|
django-ckeditor>=5.7.1
|
||||||
django-admin-sortable2>=0.7.2
|
django-admin-sortable2>=0.7.2
|
||||||
django-content-editor>=1.4.2
|
django-content-editor>=1.4.2
|
||||||
feincms3[all]>=0.31.0
|
|
||||||
|
|
||||||
bleach>=1.4.3
|
|
||||||
django-honeypot>=0.5.0
|
django-honeypot>=0.5.0
|
||||||
|
|
||||||
gunicorn>=19.6.0
|
gunicorn>=19.6.0
|
||||||
|
|
Loading…
Reference in New Issue
Block a user