Browse Source

Basic templatetags and templates for main menu and breadcrumbs

Edd Baldry 8 years ago
parent
commit
9b952f5fbb

+ 0 - 0
bakerydemo/base/templatetags/__init__.py


+ 89 - 0
bakerydemo/base/templatetags/navigation_tags.py

@@ -0,0 +1,89 @@
+from django import template
+from django.template import Template
+from django.utils.http import urlencode
+from wagtail.wagtailcore.models import Page
+
+register = template.Library()
+# https://docs.djangoproject.com/en/1.9/howto/custom-template-tags/
+
+
+@register.assignment_tag(takes_context=True)
+def get_site_root(context):
+    # This returns a core.Page. The main menu needs to have the site.root_page
+    # defined else will return an object attribute error ('str' object has no
+    # attribute 'get_children')
+    return context['request'].site.root_page
+
+
+def has_menu_children(page):
+    # This is used by the top_menu property
+    # get_children is a Treebeard API thing
+    # https://tabo.pe/projects/django-treebeard/docs/4.0.1/api.html
+    return page.get_children().live().in_menu().exists()
+
+
+def has_children(page):
+    # Generically allow index pages to list their children
+    return page.get_children().live().exists()
+
+
+def is_active(page, current_page):
+    # To give us active state on main navigation
+    return (current_page.url.startswith(page.url) if current_page else False)
+
+
+# Retrieves the top menu items - the immediate children of the parent page
+# The has_menu_children method is necessary because the Foundation menu requires
+# a dropdown class to be applied to a parent
+@register.inclusion_tag('tags/top_menu.html', takes_context=True)
+def top_menu(context, parent, calling_page=None):
+    menuitems = parent.get_children().live().in_menu()
+    for menuitem in menuitems:
+        menuitem.show_dropdown = has_menu_children(menuitem)
+        # We don't directly check if calling_page is None since the template
+        # engine can pass an empty string to calling_page
+        # if the variable passed as calling_page does not exist.
+        menuitem.active = (calling_page.url.startswith(menuitem.url)
+                           if calling_page else False)
+    return {
+        'calling_page': calling_page,
+        'menuitems': menuitems,
+        # required by the pageurl tag that we want to use within this template
+        'request': context['request'],
+    }
+
+
+# Retrieves the children of the top menu items for the drop downs
+@register.inclusion_tag('tags/top_menu_children.html', takes_context=True)
+def top_menu_children(context, parent, calling_page=None):
+    menuitems_children = parent.get_children()
+    menuitems_children = menuitems_children.live().in_menu()
+    for menuitem in menuitems_children:
+        menuitem.has_dropdown = has_menu_children(menuitem)
+        # We don't directly check if calling_page is None since the template
+        # engine can pass an empty string to calling_page
+        # if the variable passed as calling_page does not exist.
+        menuitem.active = (calling_page.url.startswith(menuitem.url)
+                           if calling_page else False)
+        menuitem.children = menuitem.get_children().live().in_menu()
+    return {
+        'parent': parent,
+        'menuitems_children': menuitems_children,
+        # required by the pageurl tag that we want to use within this template
+        'request': context['request'],
+    }
+
+
+@register.inclusion_tag('tags/breadcrumbs.html', takes_context=True)
+def breadcrumbs(context):
+    self = context.get('self')
+    if self is None or self.depth <= 2:
+        # When on the home page, displaying breadcrumbs is irrelevant.
+        ancestors = ()
+    else:
+        ancestors = Page.objects.ancestor_of(
+            self, inclusive=True).filter(depth__gt=1)
+    return {
+        'ancestors': ancestors,
+        'request': context['request'],
+    }

+ 26 - 3
bakerydemo/templates/base.html

@@ -1,4 +1,4 @@
-{% load static wagtailuserbar %}
+{% load navigation_tags static wagtailuserbar %}
 
 <!DOCTYPE html>
 <!--[if lt IE 7]>      <html class="no-js lt-ie9 lt-ie8 lt-ie7"> <![endif]-->
@@ -8,7 +8,18 @@
     <head>
         <meta charset="utf-8">
         <meta http-equiv="X-UA-Compatible" content="IE=edge">
-        <title>{% block title %}{% if self.seo_title %}{{ self.seo_title }}{% else %}{{ self.title }}{% endif %}{% endblock %}{% block title_suffix %}{% endblock %}</title>
+        <title>
+            {% block title %}
+                {% block title_prefix %}
+                    Wagtail demo bakery
+                {% endblock %}
+                {% if self.seo_title %}
+                    {{ self.seo_title }}
+                {% else %}
+                    {{ self.title }}
+                {% endif %}
+            {% endblock %}
+        </title>
         <meta name="description" content="">
         <meta name="viewport" content="width=device-width, initial-scale=1">
     </head>
@@ -16,7 +27,19 @@
     <body class="{% block body_class %}{% endblock %}">
         {% wagtailuserbar %}
 
-        {% block content %}{% endblock %}
+        {% block main_navigation %}
+            {% get_site_root as site_root %}
+            {% top_menu parent=site_root calling_page=self %}
+            {# main_menu is defined in base/templatetags/navigation_tags.py #}
+        {% endblock %}
+
+        {% block breadcrumbs %}
+            {% breadcrumbs %}
+            {# breadcrumbs is defined in base/templatetags/navigation_tags.py #}
+        {% endblock %}
+
+        {% block content %}
+        {% endblock %}
 
     </body>
 </html>

+ 17 - 0
bakerydemo/templates/tags/breadcrumbs.html

@@ -0,0 +1,17 @@
+{% load wagtailcore_tags %}
+
+{% if ancestors %}
+  <nav class="breadcrumbs" aria-label="You are here:" role="navigation">
+    <ul>
+      {% for ancestor in ancestors %}
+        {% if forloop.last %}
+          <li>
+            <span class="show-for-sr">Current: </span> {{ ancestor }}
+          </li>
+        {% else %}
+          <li><a href="{% pageurl ancestor %}">{{ ancestor }}</a></li>
+        {% endif %}
+      {% endfor %}
+    </ul>
+  </nav>
+{% endif %}

+ 36 - 0
bakerydemo/templates/tags/sidebar_menu.html

@@ -0,0 +1,36 @@
+{% load navigation_tags wagtailcore_tags %}
+{% get_site_root as site_root %}
+
+{% if calling_page|has_protocol_parent and calling_page.content_type.model == 'standardpage' %}
+
+  <div class="off-canvas position-left reveal-for-large bla" id="offCanvasLeft" data-off-canvas>
+    {% protocol_menu calling_page=calling_page %}
+  </div>
+
+{% elif ancestor.has_children %}
+
+  <div class="off-canvas position-left reveal-for-large" id="offCanvasLeft" data-off-canvas>
+    <nav class="sidebar-nav">
+      <h3>In this section</h3>
+      <ul class="vertical menu">
+        {% for menuitem in ancestor.children %}
+          <li class="{% if menuitem.is_active %}is-active{% endif %}">
+            <a href="{% pageurl menuitem %}">{{ menuitem.title }}</a>
+
+            {% if menuitem.is_active and menuitem.has_children %}
+              <ul class="nested vertical menu is-active">
+                {% for child in menuitem.children %}
+                  <li>
+                    <a href="{% pageurl child %}">{{ child.title }}</a>
+                  </li>
+
+                {% endfor %}
+              </ul>
+            {% endif %}
+          </li>
+        {% endfor %}
+      </ul>
+    </nav>
+  </div>
+
+{% endif %}

+ 35 - 0
bakerydemo/templates/tags/table_of_contents_menu.html

@@ -0,0 +1,35 @@
+{% if section_pages %}
+  <nav class="sidebar-nav section-nav">
+    <h3>In this section</h3>
+    <ul class="vertical menu">
+      {% for section_page in section_pages %}
+        <li class="{% if section_page.is_active %}is-active{% endif %}">
+          <a href="{{section_page.url}}">{{ section_page.title }}</a>
+        </li>
+      {% endfor %}
+    </ul>
+  </nav>
+{% endif %}
+{% if article_headings %}
+  <div data-sticky-container>
+    <div class="sticky" data-sticky data-sticky-on="large" data-margin-top="3" data-top-anchor="on-this-page:top" data-btm-anchor="main-end:bottom">
+      <nav class="sidebar-nav article-nav">
+        <h3 id="on-this-page">On this page</h3>
+        <ul class="vertical menu" data-magellan data-bar-offset="60">
+          {% for heading in article_headings %}
+            <li>
+              <a href="#{{heading.value|slugify}}">{{heading.value}}</a>
+              {% if heading.children %}
+                <ul>
+                  {% for subheading in heading.children %}
+                    <li><a href="#{{subheading|slugify}}">{{subheading}}</a>
+                  {% endfor %}
+                </ul>
+              {% endif %}
+            </li>
+          {% endfor %}
+        </ul>
+      </nav>
+    </div>
+  </div>
+{% endif %}

+ 15 - 0
bakerydemo/templates/tags/top_menu.html

@@ -0,0 +1,15 @@
+{% load navigation_tags wagtailcore_tags %}
+{% get_site_root as site_root %}
+
+
+<div class="navigation-bar">
+    <ul id="main-menu">
+      {% for menuitem in menuitems %}
+        <li class="{% if menuitem.show_dropdown %}has-submenu{% endif %} {% if menuitem.active %}current{% endif %}">
+        <a href="{% pageurl menuitem %}">{{ menuitem.title }}</a>
+        {% top_menu_children parent=menuitem %}
+        </li>
+      {% endfor %}
+    </ul>
+  </div>
+</div>

+ 6 - 0
bakerydemo/templates/tags/top_menu_children.html

@@ -0,0 +1,6 @@
+{% load navigation_tags wagtailcore_tags %}
+<ul class="submenu">
+  {% for child in menuitems_children %}
+    <li><a href="{% pageurl child %}">{{ child.title }}</a></li>
+  {% endfor %}
+</ul>