remove feincms3 dependency

This commit is contained in:
bkfox 2019-07-01 05:02:13 +02:00
parent 4b57cd6643
commit 4caca505c4
12 changed files with 254 additions and 103 deletions

View File

@ -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

View File

@ -1,2 +1,3 @@
import './js'; import './js';
import './styles.scss';

View File

@ -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);

View File

@ -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)

View File

@ -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"
} }

View 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

View File

@ -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,
),
)

View File

@ -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>

View File

@ -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"),
] ]

View File

@ -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)

View File

@ -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',
}, },

View File

@ -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