remove feincms3 dependency
This commit is contained in:
		@ -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
 | 
				
			||||||
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user