访问模板文件 Wagtail 中的子页面模型属性

Accessing Subpage Model properties in template file Wagtail

这是我的 model.py 文件

...

class LinkFields(models.Model):
    link_external = models.URLField(
        "External link",
        blank=True,
        null=True,
        help_text='Set an external link if you want to describe the event from an other web site',
    )
    link_page = models.ForeignKey(
        'wagtailcore.Page',
        null=True,
        on_delete=models.SET_NULL,
        blank=True,
        related_name='+',
        help_text='Choose an existing page (event must have already been created)',
    )

    @property
    def link(self):
        if self.link_page:
            return self.link_page.url
        else:
            return self.link_external

    panels = [
        FieldPanel('link_external'),
        PageChooserPanel('link_page'),
    ]

    class Meta:
        abstract = True

class RelatedLink(LinkFields):
    title = models.CharField(max_length=255, help_text="Link title")

    panels = [
        FieldPanel('title'),
        MultiFieldPanel(LinkFields.panels, "Link"),
    ]

    class Meta:
        abstract = True


class HomePage(Page):
    body = RichTextField(blank=True)

    content_panels = Page.content_panels + [
        FieldPanel('body', classname="full"),
        InlinePanel('related_links', label="Related events"),
    ]


class EventsPage(Page):

    # Database fields

    date = models.DateField("Event date")
    topicTag = models.CharField(max_length = 25)
    name = models.CharField(max_length = 50)
    city = models.CharField(max_length = 20, default="Where it is")
    place = models.CharField(max_length = 20, blank=True)
    body = RichTextField(blank=True)

    # Search index configuration

    search_fields = Page.search_fields + (
        index.SearchField('topicTag'),
        index.SearchField('place'),
        index.FilterField('date'),
    )

    # Editor panels configuration

    content_panels = Page.content_panels + [
        FieldPanel('date'),
        FieldPanel('topicTag'),
        FieldPanel('name', classname="title"),
        FieldPanel('city'),
        FieldPanel('place'),
        FieldPanel('body'),
    ]


class EventsRelatedLink(Orderable, RelatedLink):
    page = ParentalKey('HomePage', related_name='related_links')

EventsPages 是 HomePage 的子页面,它本身显示 links 后续子页面。我的问题是每个子页面的 link 我在主页上显示我想访问所谓的子页面的属性以显示一种预览。

这是我的 home_page.html 模板:

{% extends "base.html" %}

{% load wagtailcore_tags %}

{% block body_class %}template-homepage{% endblock %}

{% block content %}
   <h1>THE <span class="pink">OPEN SOURCE</span> INNOVATION SPRING</h1>

   <div class="intro">{{ page.body|richtext }}</div>

   {% if page.related_links.all %}
      <ul>
    {% for item in page.related_links.all %}
       <li><a href="{{ item.link }}">{{ item.title }}</a></li>
       <br><div class="description">{{ item.date }}
         <span class="pink">[{{ item.topicTag }}]</span>
          {{ item.name }} <span class="pink">[{{ item.city }}]</span>
       </div>
    {% endfor %}
      </ul>
   {% endif %}
{% endblock %}

当然,所有带有 * 作为 EventsPage 模型属性的 "item.*" 都不起作用。我很确定我走错了路,但我需要帮助才能做到这一点。

首先,{{ item.date }} 将不起作用,因为 item 是一个 RelatedLink 对象,而不是页面对象。要访问页面对象,您需要 {{ item.link_page.date }}(当然,当编辑器提供 link_external 而不是 link_page 时,还需要一些替代行为)。但是,仅此更改本身是不够的 - 它适用于 item.link_page.title 等核心页面属性,但不适用于 date.

等 event-specific 属性

为什么会发生这种情况的解释:每当您通过 EventPage.objects.all()page.get_children() 或(在本例中)page.related_links 等操作检索一组页面时,通常所有结果都将是同一类型。如果这是一个可能涵盖多种不同类型页面的操作(例如 - EventPage.objects.all() 只能 return EventPage 对象,但 page.get_children()page.related_links 可以 return 任何类型的页面),然后为了保持它们相同,它必须 return 它们作为 "lowest common denominator" - 所有页面类型都源自的基本页面模型。这提供了title等所有页面通用的字段,但special-purpose字段中的none。

(这是数据库查找工作方式的限制 - 如果我们想以最具体的形式获取每个页面对象,数据库查询将不得不一次检查所有 page-related 表,这是非常低效的。)

考虑到这一点,有几个选项可供您选择:

首先,如果您知道您网站上的编辑只会 select EventPage 作为相关链接 - 根据您的 [,这似乎是一个安全的假设=60=] 并且您正在编写诸如 {{ item.date }}{{ item.city }} 之类的东西,这对于其他类型的页面可能没有意义 - 然后您可以通过使外键指向 EventPage:

link_page = models.ForeignKey(
    'myapp.EventPage',  # replace 'myapp' with your application name
    null=True,
    on_delete=models.SET_NULL,
    blank=True,
    related_name='+',
    help_text='Choose an existing page (event must have already been created)',
)

进行此更改后,Wagtail 管理中的页面选择器将只允许您为此字段 select EventPages,并且 item.link_page 将为您提供完整的 EventPage 个对象,允许您编写 {{ item.link_page.date }}.

其次:如果您需要在相关链接中支持其他页面类型 - 例如,也许您有一个 NewsPage 模型也有一个 date 字段,并且希望能够select 这些也是相关链接 - 然后您可以通过调用 .specific 将基本 Page 对象转换为更具体的页面类型。例如:{{ item.link_page.specific.date }}。缺点是 .specific 会在您每次使用它时执行额外的数据库查找,这可能会损害性能。作为一个简单的优化,您可以使用 {% with %} 标记将其分配给变量:

{% for item in page.related_links.all %}
   {% with item.link_page.specific as related_page %}
      <li><a href="{{ item.link }}">{{ item.title }}</a></li>
      <br><div class="description">{{ related_page.date }}
        <span class="pink">[{{ related_page.topicTag }}]</span>
         {{ related_page.name }} <span class="pink">[{{ related_page.city }}]</span>
      </div>
   {% endwith %}
{% endfor %}