Преглед на файлове

Initial commit for base app with streamfield blocks, about page, gallery page and people snippet

Edd Baldry преди 8 години
родител
ревизия
3407b16f1b

BIN
bakerydemo/.DS_Store


+ 42 - 0
bakerydemo/base/forms.py

@@ -0,0 +1,42 @@
+from django.db import models
+from modelcluster.fields import ParentalKey
+from wagtail.wagtailadmin.edit_handlers import (
+    FieldPanel,
+    StreamFieldPanel,
+    FieldRowPanel,
+    InlinePanel,
+    MultiFieldPanel)
+from wagtail.wagtailimages.edit_handlers import ImageChooserPanel
+from wagtail.wagtailcore.fields import StreamField, RichTextField
+from wagtail.wagtailforms.models import AbstractEmailForm, AbstractFormField
+from .blocks import FormPageBlock
+from core.models import SiteSettingsTemplateMixin
+
+
+class FormField(AbstractFormField):
+    page = ParentalKey('FormPage', related_name='form_fields')
+
+
+class FormPage(SiteSettingsTemplateMixin, AbstractEmailForm):
+    header_image = models.ForeignKey(
+        'core.OvercastImage',
+        null=True,
+        blank=True,
+        on_delete=models.SET_NULL,
+        related_name='+'
+    )
+    body = StreamField(FormPageBlock())
+    thank_you_text = RichTextField(blank=True)
+    content_panels = AbstractEmailForm.content_panels + [
+        ImageChooserPanel('header_image'),
+        StreamFieldPanel('body'),
+        InlinePanel('form_fields', label="Form fields"),
+        FieldPanel('thank_you_text', classname="full"),
+        MultiFieldPanel([
+            FieldRowPanel([
+                FieldPanel('from_address', classname="col6"),
+                FieldPanel('to_address', classname="col6"),
+            ]),
+            FieldPanel('subject'),
+        ], "Email"),
+    ]

+ 58 - 0
bakerydemo/base/migrations/0001_initial.py

@@ -0,0 +1,58 @@
+# -*- coding: utf-8 -*-
+# Generated by Django 1.9.9 on 2017-02-09 14:32
+from __future__ import unicode_literals
+
+from django.db import migrations, models
+import django.db.models.deletion
+import wagtail.wagtailcore.blocks
+import wagtail.wagtailcore.fields
+import wagtail.wagtailembeds.blocks
+import wagtail.wagtailimages.blocks
+
+
+class Migration(migrations.Migration):
+
+    initial = True
+
+    dependencies = [
+        ('wagtailimages', '0013_make_rendition_upload_callable'),
+        ('wagtailcore', '0028_merge'),
+    ]
+
+    operations = [
+        migrations.CreateModel(
+            name='AboutPage',
+            fields=[
+                ('page_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='wagtailcore.Page')),
+                ('body', wagtail.wagtailcore.fields.StreamField((('heading_block', wagtail.wagtailcore.blocks.StructBlock((('heading_text', wagtail.wagtailcore.blocks.CharBlock(classname='title', required=True)), ('size', wagtail.wagtailcore.blocks.ChoiceBlock(blank=True, choices=[('', 'Select a header size'), ('h2', 'H2'), ('h3', 'H3'), ('h4', 'H4')], required=False))))), ('paragraph_block', wagtail.wagtailcore.blocks.RichTextBlock(icon='fa-paragraph', template='blocks/paragraph_block.html')), ('image_block', wagtail.wagtailcore.blocks.StructBlock((('images', wagtail.wagtailcore.blocks.ListBlock(wagtail.wagtailcore.blocks.StructBlock((('image', wagtail.wagtailimages.blocks.ImageChooserBlock(required=True)), ('caption', wagtail.wagtailcore.blocks.CharBlock(required=False)), ('attribution', wagtail.wagtailcore.blocks.CharBlock(required=False)))))),))), ('block_quote', wagtail.wagtailcore.blocks.StructBlock((('attribute_name', wagtail.wagtailcore.blocks.CharBlock(blank=True, label='e.g. Guy Picciotto', required=False)),))), ('embed_block', wagtail.wagtailembeds.blocks.EmbedBlock(help_text='Insert an embed URL e.g https://www.youtube.com/embed/SGJFWirQ3ks', icon='fa-s15', template='blocks/embed_block.html'))), blank=True, verbose_name='About page detail')),
+                ('image', models.ForeignKey(blank=True, help_text='Location image', null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='+', to='wagtailimages.Image')),
+            ],
+            options={
+                'abstract': False,
+            },
+            bases=('wagtailcore.page',),
+        ),
+        migrations.CreateModel(
+            name='GalleryPage',
+            fields=[
+                ('page_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='wagtailcore.Page')),
+                ('choices', models.CharField(help_text='Text to describe the index page', max_length=255)),
+                ('introduction', models.TextField(blank=True, help_text='Text to describe the index page')),
+                ('image', models.ForeignKey(blank=True, help_text='Location listing image', null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='+', to='wagtailimages.Image')),
+            ],
+            options={
+                'abstract': False,
+            },
+            bases=('wagtailcore.page',),
+        ),
+        migrations.CreateModel(
+            name='People',
+            fields=[
+                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('first_name', models.CharField(max_length=254, verbose_name='First name')),
+                ('last_name', models.CharField(max_length=254, verbose_name='Last name')),
+                ('job_title', models.CharField(max_length=254, verbose_name='Job title')),
+                ('image', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='+', to='wagtailimages.Image')),
+            ],
+        ),
+    ]

+ 27 - 0
bakerydemo/base/migrations/0002_auto_20170209_1454.py

@@ -0,0 +1,27 @@
+# -*- coding: utf-8 -*-
+# Generated by Django 1.9.9 on 2017-02-09 14:54
+from __future__ import unicode_literals
+
+from django.db import migrations
+import wagtail.wagtailcore.blocks
+import wagtail.wagtailcore.fields
+import wagtail.wagtailembeds.blocks
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        ('base', '0001_initial'),
+    ]
+
+    operations = [
+        migrations.AlterModelOptions(
+            name='people',
+            options={'verbose_name': 'Person', 'verbose_name_plural': 'People'},
+        ),
+        migrations.AlterField(
+            model_name='aboutpage',
+            name='body',
+            field=wagtail.wagtailcore.fields.StreamField((('heading_block', wagtail.wagtailcore.blocks.StructBlock((('heading_text', wagtail.wagtailcore.blocks.CharBlock(classname='title', required=True)), ('size', wagtail.wagtailcore.blocks.ChoiceBlock(blank=True, choices=[('', 'Select a header size'), ('h2', 'H2'), ('h3', 'H3'), ('h4', 'H4')], required=False))))), ('paragraph_block', wagtail.wagtailcore.blocks.RichTextBlock(icon='fa-paragraph', template='blocks/paragraph_block.html')), ('image_block', wagtail.wagtailcore.blocks.StructBlock(())), ('block_quote', wagtail.wagtailcore.blocks.StructBlock((('attribute_name', wagtail.wagtailcore.blocks.CharBlock(blank=True, label='e.g. Guy Picciotto', required=False)),))), ('embed_block', wagtail.wagtailembeds.blocks.EmbedBlock(help_text='Insert an embed URL e.g https://www.youtube.com/embed/SGJFWirQ3ks', icon='fa-s15', template='blocks/embed_block.html'))), blank=True, verbose_name='About page detail'),
+        ),
+    ]

+ 155 - 0
bakerydemo/base/models.py

@@ -0,0 +1,155 @@
+from __future__ import unicode_literals
+
+from django.db import models
+
+from wagtail.wagtailcore.models import Page, Orderable, Collection
+from wagtail.wagtailsearch import index
+from wagtail.wagtailimages.edit_handlers import ImageChooserPanel
+from wagtail.wagtailcore.fields import StreamField
+from wagtail.wagtailadmin.edit_handlers import (
+        FieldPanel, InlinePanel, FieldRowPanel, StreamFieldPanel)
+from wagtail.wagtailsnippets.models import register_snippet
+from wagtail.wagtailsnippets.edit_handlers import SnippetChooserPanel
+from blocks import BaseStreamBlock
+
+
+@register_snippet
+class People(models.Model):
+    first_name = models.CharField("First name", max_length=254)
+    last_name = models.CharField("Last name", max_length=254)
+    job_title = models.CharField("Job title", max_length=254)
+
+    image = models.ForeignKey(
+        'wagtailimages.Image',
+        null=True,
+        blank=True,
+        on_delete=models.SET_NULL,
+        related_name='+'
+    )
+
+    panels = [
+        FieldPanel('first_name', classname="col6"),
+        FieldPanel('last_name', classname="col6"),
+        FieldPanel('job_title'),
+        ImageChooserPanel('image')
+    ]
+
+    def __str__(self):
+        return self.first_name + " " + self.last_name
+
+    class Meta:
+        verbose_name = 'Person'
+        verbose_name_plural = 'People'
+
+
+# class AboutLocationRelationship(Orderable, models.Model):
+#     """
+#     This defines the relationship between the `LocationPage` within the `locations`
+#     app and the About page below allowing us to add locations to the about
+#     section.
+#     """
+#     about = ParentalKey(
+#         'About', related_name='location_about_relationship'
+#     )
+#     locations = models.ForeignKey(
+#         'locations.LocationPage', related_name='about_location_relationship'
+#     )
+#     panels = [
+#         PageChooserPanel('locations')
+#     ]
+
+
+class AboutPage(Page):
+    """
+    The About Page
+    """
+    image = models.ForeignKey(
+        'wagtailimages.Image',
+        null=True,
+        blank=True,
+        on_delete=models.SET_NULL,
+        related_name='+',
+        help_text='Location image'
+    )
+
+    body = StreamField(
+        BaseStreamBlock(), verbose_name="About page detail", blank=True
+        )
+    # We've defined the StreamBlock() within blocks.py that we've imported on
+    # line 12. Defining it in a different file gives us consistency across the
+    # site, though StreamFields _can_ be created on a per model basis if you
+    # have a use case for it
+
+    content_panels = Page.content_panels + [
+        ImageChooserPanel('image'),
+        StreamFieldPanel('body'),
+        #    InlinePanel(
+        #        'about_location_relationship',
+        #        label='Locations',
+        #        min_num=None
+        #        ),
+    ]
+
+    # parent_page_types = [
+    #    'home.HomePage'
+    # ]
+
+    # Defining what content type can sit under the parent
+    # The empty array means that no children can be placed under the
+    # LocationPage page model
+    subpage_types = []
+
+    # api_fields = ['image', 'body']
+
+
+def getImageCollections():
+    # We return all collections to a list that don't have the name root.
+    return [(
+        collection.id, collection.name
+        ) for collection in Collection.objects.all().exclude(
+        name='Root'
+        )]
+
+
+class GalleryPage(Page):
+    """
+    This is a page to list all the locations on the site
+    """
+
+    CHOICES_LIST = getImageCollections()
+    # To return our collection choices for the editor to access we need to
+    # make the choices list a variable rather than a function
+
+    choices = models.CharField(
+        max_length=255, choices=CHOICES_LIST
+        )
+
+    image = models.ForeignKey(
+        'wagtailimages.Image',
+        null=True,
+        blank=True,
+        on_delete=models.SET_NULL,
+        related_name='+',
+        help_text='Location listing image'
+    )
+
+    introduction = models.TextField(
+        help_text='Text to describe the index page',
+        blank=True)
+
+    content_panels = Page.content_panels + [
+        FieldPanel('choices'),
+        ImageChooserPanel('image'),
+        FieldPanel('introduction')
+    ]
+
+    # parent_page_types = [
+    #     'home.HomePage'
+    # ]
+
+    # Defining what content type can sit under the parent. Since it's a blank
+    # array no subpage can be added
+    subpage_types = [
+    ]
+
+    # api_fields = ['introduction']

BIN
bakerydemo/settings/__pycache__/base.cpython-34.pyc


+ 1 - 0
bakerydemo/settings/base.py

@@ -46,6 +46,7 @@ INSTALLED_APPS = [
     'modelcluster',
     'compressor',
     'taggit',
+    'wagtailfontawesome',
 
     'django.contrib.admin',
     'django.contrib.auth',

+ 13 - 0
bakerydemo/templates/about_page.html

@@ -0,0 +1,13 @@
+{% extends "base.html" %}
+{% load wagtailimages_tags %}
+
+{% block content %}
+	{{ page.title }}
+
+    <div class="image">
+        {% image page.image width-500 as photo %}
+              <img src="{{ photo.url }}" width="{{ photo.width }}" height="{{ photo.height }}" alt="{{ photo.alt }}" />
+	</div>
+
+	{{ page.body }}
+{% endblock content %}

+ 13 - 0
bakerydemo/templates/base/about_page.html

@@ -0,0 +1,13 @@
+{% extends "base.html" %}
+{% load wagtailimages_tags %}
+
+{% block content %}
+    {{ page.title }}
+
+    <div class="image">
+        {% image page.image width-500 as photo %}
+              <img src="{{ photo.url }}" width="{{ photo.width }}" height="{{ photo.height }}" alt="{{ photo.alt }}" />
+    </div>
+
+    {{ page.body }}
+{% endblock content %}

+ 9 - 0
bakerydemo/templates/blocks/blockquote.html

@@ -0,0 +1,9 @@
+{% load wagtailimages_tags %}
+
+<div class="block text blockquote">
+  <div class="wrap">
+      <p class="text">{{ self.text }}</p>
+      <p class="attribute_name">{{ self.attribute_name}}</p>
+      <p class="attribute_group">{{ self.attribute_group}}</p>
+  </div>
+</div>

+ 3 - 0
bakerydemo/templates/blocks/embed_block.html

@@ -0,0 +1,3 @@
+{% load wagtailimages_tags %}
+
+{{ self }}

+ 15 - 0
bakerydemo/templates/blocks/heading_block.html

@@ -0,0 +1,15 @@
+{% if self.size == 'h2' %}
+	    <h2>{{ self.heading_text }}</h2>
+
+	{% elif self.size == 'h3' %}
+	    <h3>{{ self.heading_text }}</h3>
+
+	{% elif self.size == 'h4' %}
+	    <h4>{{ self.heading_text }}</h4>
+
+{% endif %}
+
+{% comment %}
+	Content is coming from the StandardBlock StreamField
+	class within `blocks.py`
+{% endcomment %}

+ 8 - 0
bakerydemo/templates/blocks/image_block.html

@@ -0,0 +1,8 @@
+{% load wagtailimages_tags %}
+
+
+
+<figure>
+  {% image self.image fill-600x600 %}
+      <figcaption>{{ self.caption }} - {{ self.attribution }}</figcaption>
+</figure>

+ 1 - 0
bakerydemo/templates/blocks/paragraph_block.html

@@ -0,0 +1 @@
+        {{ self }}

+ 61 - 0
blocks.py

@@ -0,0 +1,61 @@
+from wagtail.wagtailimages.blocks import ImageChooserBlock
+from wagtail.wagtailembeds.blocks import EmbedBlock
+from wagtail.wagtailcore.blocks import (
+    StructBlock,
+    TextBlock,
+    StreamBlock,
+    RichTextBlock,
+    CharBlock,
+    ListBlock,
+    ChoiceBlock,
+    PageChooserBlock
+)
+
+
+class ImageBlock(StructBlock):
+    image = ImageChooserBlock(required=True)
+    caption = CharBlock(required=False)
+    attribution = CharBlock(required=False)
+
+    class Meta:
+        icon = 'image'
+        template = "blocks/image_block.html"
+
+
+class HeadingBlock(StructBlock):
+    heading_text = CharBlock(classname="title", required=True)
+    size = ChoiceBlock(choices=[
+            ('', 'Select a header size'),
+            ('h2', 'H2'),
+            ('h3', 'H3'),
+            ('h4', 'H4')
+        ], blank=True, required=False)
+
+    class Meta:
+        icon = "title"
+        template = "blocks/heading_block.html"
+
+
+class BlockQuote(StructBlock):
+    text = TextBlock(),
+    attribute_name = CharBlock(
+            blank=True, required=False, label='e.g. Guy Picciotto')
+
+    class Meta:
+        icon = "fa-quote-left"
+        template = "blocks/blockquote.html"
+
+
+# StreamBlocks
+class BaseStreamBlock(StreamBlock):
+    heading_block = HeadingBlock()
+    paragraph_block = RichTextBlock(
+        icon="fa-paragraph",
+        template="blocks/paragraph_block.html"
+    )
+    image_block = ImageBlock()
+    block_quote = BlockQuote()
+    embed_block = EmbedBlock(
+        help_text='Insert an embed URL e.g https://www.youtube.com/embed/SGJFWirQ3ks',
+        icon="fa-s15",
+        template="blocks/embed_block.html")