gpt4 book ai didi

django - 使用索引页子链接创建 Django Wagtail 侧边栏

转载 作者:行者123 更新时间:2023-12-02 06:30:43 24 4
gpt4 key购买 nike

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

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

{% standard_index_listing calling_page=self %}

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

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

最佳答案

本质上,您遍历了由 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_menu 渲染的模板。和 top_menu_children模板标签。

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

    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>

    我写了一篇关于这个的博客文章 - check it out更多细节。或者,前往 Thermaline.com看看它的实际效果,尽管我认为现在没有多层次的深度。如果有,它们会自动呈现:)

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

    您需要做的就是:
  • 包含 demo_tags在您的基本模板中
  • 调用 {% top_menu %}您希望渲染菜单的位置。
  • 定制 top_menu.htmlchildren_items.html渲染
    首先是所有后续级别的页面。

  • 为他们的 post on two-level menus 向 Tivix 大喊大叫这对我来说是一个很好的起点!

    关于django - 使用索引页子链接创建 Django Wagtail 侧边栏,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38088589/

    24 4 0
    Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
    广告合作:1813099741@qq.com 6ren.com