2

我正在使用 Django Wagtail 构建一个站点,并且无法弄清楚如何添加将列出父索引页面的所有子页面的侧边栏菜单。例如,我有一个standard_index_page.html在 Admin 中创建的父页面,然后我使用standard_page.html模板添加了该页面的子页面。

在我的 standard_index_page.html 模板中,我有以下代码

{% standard_index_listing calling_page=self %}

它显示所有带有链接的子页面,但我还想在子页面上显示所有子链接的列表。

我希望这是有道理的,有人可以伸出援助之手。谢谢你。

4

1 回答 1

9

本质上,您遍历了由Django-Treebeard提供给 Wagtail 的页面层次结构的树结构。

许多前端框架不允许多级菜单,因为有些人认为它超出了最佳实践。但是,使用诸如SmartMenus之类的库,您可以稍微费点力气地显示此结构。

对于我的需要,没有简单的解决方案。所以,虽然我想分享一个例子来说明我是如何做到这一点的,但它可能缺少解释。如果您有任何问题,我很乐意为您解答。

我为此苦苦挣扎了一段时间,虽然可能有更简单的方法来遍历树,但随着需求的扩大,我构建了以下方法。它允许我们遍历站点中的所有活动页面,检查当前页面何时在菜单中呈现,并允许对呈现进行细粒度控制。

这是我们要做的:

  • 创建一些模板标签,这些标签将获取当前站点的站点根目录,循环通过站点根目录的直接子项,并循环通过任何较低级别的子项,同时在每个级别发现时循环通过当前菜单项的子项

在您的基本模板中,这意味着我们需要:

  • {% load demo_tags %}导入我们的自定义模板标签
  • 调用{% top_menu calling_page=self %}获取和呈现站点根目录的所有直接子项。这些项目将显示在标准菜单栏上。
  • {% top_menu_children parent=menuitem %}在渲染的模板中调用{% top_menu %}获取并渲染所有二级和更低级别的子页面。这包括将鼠标悬停在父菜单项上时要显示的所有菜单项。

这是我创建的用于遍历页面层次结构的所有级别的自定义demo_tags.py文件。这样做的好处是它不需要提供任何自定义上下文数据;它与 Wagtail 一起开箱即用!

@register.assignment_tag(takes_context=True)
def get_site_root(context):
    ''' 
    Returns a core.Page, not the implementation-specific model used
    so object-comparison to self will return false as objects would differ  
    '''
    return context['request'].site.root_page


def has_menu_children(page):
    '''
    Returns boolean of whether children pages exist to the page supplied
    '''   
    return page.get_children().live().in_menu().exists()    


@register.inclusion_tag('info_site/tags/top_menu.html', takes_context=True)
def top_menu(context, parent, calling_page=None):
    '''
    Retrieves the top menu items - the immediate children of the parent page
    The has_menu_children method is necessary in many cases. For example, a bootstrap menu requires
    a dropdown class to be applied to a parent
    '''
    root = get_site_root(context)
    try:
        is_root_page = (root.id == calling_page.id)
    except:
        is_root_page = False

    menuitems = parent.get_children().filter(
        live=True,
        show_in_menus=True
    ).order_by('title')

    for menuitem in menuitems:
        menuitem.show_dropdown = has_menu_children(menuitem)

    return {
        'calling_page': calling_page,
        'menuitems': menuitems,
        'is_root_page':is_root_page,
        # required by the pageurl tag that we want to use within this template
        'request': context['request'],
    }


@register.inclusion_tag('my_site/tags/top_menu_children.html', takes_context=True)
def top_menu_children(context, parent, sub=False, level=0):
    ''' Retrieves the children of the top menu items for the drop downs '''
    menuitems_children = parent.get_children().order_by('title')
    menuitems_children = menuitems_children.live().in_menu()

    for menuitem in menuitems_children:
        menuitem.show_dropdown = has_menu_children(menuitem)

    levelstr= "".join('a' for i in range(level)) # for indentation
    level += 1

    return {
        'parent': parent,
        'menuitems_children': menuitems_children,
        'sub': sub,
        'level':level,
        'levelstr':levelstr,
        # required by the pageurl tag that we want to use within this template
        'request': context['request'],
    }

本质上,页面呈现三个级别:

  • 站点根目录由{% get_site_root %}
  • 第一级的孩子被称为{% top_menu %}
  • 二级和更低级别的子级由 调用{% top_menu_children %},每当菜单中显示的页面在呈现此标记时具有子级时都会调用该子级。

为此,我们需要创建要由我们的top_menutop_menu_children模板标签呈现的模板。

请注意 - 这些都是为 Bootstrap 3 的导航栏类构建的,并根据我的需要进行了定制。只需根据您的需要定制这些。整个菜单构建过程由 调用{% top_menu_children %},因此将此标记放置在您希望呈现菜单的基本模板中。更改top_menu.html以反映菜单的整体结构以及如何呈现每个menuitem。更改children_items.html以反映您希望如何呈现所有顶级菜单项的子项,在任何深度。

my_site/tags/top_menu.html

{% load demo_tags wagtailcore_tags static %}
{% get_site_root as site_root %}

{# FOR TOP-LEVEL CHILDREN OF SITE ROOT; In a nav or sidebar, these are the menu items we'd generally show before hovering. #}

<div class="container">
    <div class="collapse navbar-collapse" id="navbar-collapse-3">
        <ul class="nav navbar-nav navbar-left">
            {% for menuitem in menuitems %}
                <li class="{% if menuitem.active %}active{% endif %}">
                    {% if menuitem.show_dropdown %}
                        <a href="{{ menuitem.url }}">{{ menuitem.title }}
                            <span class="hidden-lg hidden-md hidden-sm visible-xs-inline">
                                <span class="glyphicon glyphicon-chevron-right"></span>
                            </span>
                        </a>
                        {% top_menu_children parent=menuitem %}
                    {% else %}
                        <a href="{% pageurl menuitem %}">{{ menuitem.title }}</a>
                    {% endif %}
                </li>
            {% endfor %}
        </ul>
    </div>
</div>

my_site/tags/children_items.html

{% load demo_tags wagtailcore_tags %}

{# For second- and lower-level decendents of site root; These are items not shown prior to hovering on their parent menuitem, hence the separate templates (and template tags) #}

<ul class="dropdown-menu">
    {% for child in menuitems_children %}
        {% if child.show_dropdown %}
            <li>
                <a href="{% pageurl child %}">
                    {% for i in levelstr %}&nbsp&nbsp{% endfor %}
                    {{ child.title }}
                    <span class="glyphicon glyphicon-chevron-right"></span>
                </a>
                {# On the next line, we're calling the same template tag we're rendering. We only do this when there are child pages of the menu item being rendered. #}
                {% top_menu_children parent=child sub=True level=level %}
                {# ^^^^ SmartMenus is made to render menus with as many levels as we like. Bootstrap considers this outside of best practices and, with version 3, has deprecated the ability to do so. Best practices are made to be broken, right :] #}
            </li>
        {% else %}
            <li>
                <a href="{% pageurl child %}">
                    <!-- Allows for indentation based on depth of page in the site structure -->
                    {% for i in levelstr %}&nbsp&nbsp{% endfor %}
                    {{ child.title }}
                </a>
            </li>
        {% endif %}
    {% endfor %}
</ul>

现在,在您的基本级别模板中(假设您正在使用一个;如果没有,请使用它:))您可以遍历菜单,同时保持对您inclusion_tag的 s 使用的模板的混乱清除。

my_site/base.html

<ul class="nav navbar-nav navbar-left">
    {% for menuitem in menuitems %}
        <li class="{% if menuitem.active %}active{% endif %}">
            {% if menuitem.show_dropdown %}
                <a href="{{ menuitem.url }}">{{ menuitem.title }}
                    <span class="hidden-lg hidden-md hidden-sm visible-xs-inline">
                        <span class="glyphicon glyphicon-chevron-right"></span>
                    </span>
                </a>
                {% top_menu_children parent=menuitem %}
            {% else %}
                <a href="{% pageurl menuitem %}">{{ menuitem.title }}</a>
            {% endif %}
        </li>
    {% endfor %}
</ul>

我写了一篇关于此的博客文章 -查看更多详细信息。或者,前往Thermaline.com看看它的实际效果,尽管我认为目前还没有多层次的深度。如果有,它们会自动呈现:)

现在,这个例子是一个导航栏,但它可以很容易地适应一个侧边栏。

您需要做的就是:

  • 包含demo_tags在您的基本模板中
  • 调用{% top_menu %}您希望呈现菜单的位置。
  • 自定义top_menu.htmlchildren_items.html呈现第一个页面,然后呈现所有后续级别的页面。

向 Tivix 大喊他们在两级菜单上的帖子,这对我来说是一个很好的起点!

于 2016-06-29T04:35:57.527 回答