0

我用以下内容创建了一个新的 page-test.php。

$query = array('category_name' => 'blog, portfolio');
$context['posts'] = Timber::get_posts($query);

这仅显示来自这些类别的帖子,这很棒,但我想将它们分组到页面上的特定 div 中。目前我无法获取我的自定义页面或戏弄树枝文件(我不知道我是否需要两者或只需要一个)来覆盖默认树枝页面。我所做的更改要么破坏了页面,要么似乎什么都不做。

我确信这是完全错误的。

{% extends "base.twig" %}

{% block content %}
    {% for post in posts %}
        <div class="blog">
            {% include ['tease-'~post.post_type~'.twig', 'tease.twig'] %}
        </div>
    {% endfor %}

    {% for post in posts %}
        <div class="portfolio">
            {% include ['tease-'~post.post_type~'.twig', 'tease.twig'] %}
        </div>
    {% endfor %}
{% endblock %}

我不知道如何让特定的树枝文件只显示我想要的内容。

提前感谢您的帮助。

请注意,我以前使用过 Expression Engine、Statamic 和 Craft,所以我对这种类型的东西非常熟悉(虽然不是 php),但由于某种原因,这让我感到失望。

4

2 回答 2

5

为什么您的第一种方法不起作用

当您获得两个类别的帖子时,您会获得按日期排序的混合帖子,这是您第一次尝试时得到的。你遍历了所有的帖子,一次是在一个带有类blog的 div中,一次是在一个带有类投资组合的 div 中。您没有添加检查帖子是否实际上在类别博客投资组合中。

据我了解,您想post.post_type在包含中使用来区分类别。但是,post_type不是类别,您的帖子将是“帖子”,您的页面将是“页面”(或使用自定义帖子类型时的任何其他名称)。您的 include 试图找到tease-post.twig,它可能不存在。

所以这之间的唯一区别是:

{% for post in posts %}
    <div class="blog">
        {% include ['tease-'~post.post_type~'.twig', 'tease.twig'] %}
    </div>
{% endfor %}

和这个:

{% for post in posts %}
    <div class="portfolio">
        {% include ['tease-'~post.post_type~'.twig', 'tease.twig'] %}
    </div>
{% endfor %}

到底是只有类名上的div;)。

现在要获取帖子的类别,您可以通过post.categorytwig 文件访问它。

{% for post in posts %}
    <div class="blog">
        {% include ['tease-' ~ post.category.slug ~ '.twig', 'tease.twig'] %}
    </div>
{% endfor %}

但是,您的博客 div 中仍然会有来自投资组合类别的帖子。

你怎么能做到

Timber 的美妙之处在于,您可以在将帖子处理到模板之前对其进行处理。您还可以获取您从中获得的帖子Timber::get_posts()并将它们拆分、过滤、扩展它们,无论您想要什么。

现在我们可以在将它们交给模板之前按类别对它们进行排序。

一种方法是在查询中传递一个附加参数。但是为了学习 Timber,我们将采取另一种方法。

我们将为每个类别创建一个包含子数组的新数组。我们称它为新数组$sorted_posts。然后,我们遍历所有帖子,获取该帖子的类别并填充$sorted_posts,以便子数组的键是帖子第一个类别的 slug。像这样:

$query = array(
    'category_name' => 'blog,portfolio',
);

$posts = Timber::get_posts( $query );

$sorted_posts = array();

foreach ( $posts as $post ) {
    // Get first category of post
    $category = $post->category();

    // Fill post back to sorted_posts
    $sorted_posts[ $category->slug ][] = $post;
}

// Add sorted posts to context
$context['posts'] = $sorted_posts;

A var_dump()on$sorted_posts可能如下所示,共有 5 篇文章(3 篇作品集文章和 2 篇博客文章)。

$sorted_posts = array (size=2)
    'portfolio' => array (size=3)
        0 => object(Timber\Post)[379]
        1 => object(Timber\Post)[430]
        2 => object(Timber\Post)[375]
    'blog' => array (size=2)
        0 => object(Timber\Post)[378]
        1 => object(Timber\Post)[429]

在你的 twig 文件中,你可以遍历这些子数组:

{% extends 'base.twig' %}

{% block content %}

<div class="blog">
    <h3>Blog</h3>
    {% for post in posts.blog %}
        {% include ['tease-blog.twig', 'tease.twig'] %}
    {% endfor %}
</div>

<div class="portfolio">
    <h3>Portfolio</h3>
    {% for post in posts.portfolio %}
        {% include ['tease-portfolio.twig', 'tease.twig'] %}
    {% endfor %}
</div>

{% endblock %}

这种方法无法处理的是当您为帖子分配两个或更多类别时。

为什么您的第二个解决方案有效

您在第二种方法中所做的是通过仅获取一个类别的帖子来绕过帖子排序。这样,您$context['posts']就知道里面只有带有 categorystory的帖子,并且$context['posts2']里面只有带有 category 的帖子news。您不会得到必须首先排序的混合帖子数组。

现在的好处是:您可能找到了最适合您的案例的解决方案,因为它使用的数据库查询比单独获取每个帖子的类别要少。

当我这样做时(如“你怎么能做到”)……</p>

foreach ( $posts as $post ) {
    $category = $post->category();
}

… 然后 WordPress 将在额外的数据库查询中获取每个帖子的类别。老实说,我不知道是否有可能获得包含该类别的帖子。如果您没有必须在首页上显示的许多帖子,那么您或您的网站访问者在性能方面的感知差异可能可以忽略不计。

TL;DR:干得​​好,坚持你找到的解决方案!

于 2016-05-28T15:14:14.910 回答
1

所以我得到了这个工作,但我确信这是一个非常糟糕的解决方案。

我在 page-test.php 中添加了第二个查询,所以帖子是故事,帖子 2 是新闻。

$query = array('category_name' => 'Story');
$context['posts'] = Timber::get_posts($query);
$query2 = array('category_name' => 'News');
$context['posts2'] = Timber::get_posts($query2);

然后我添加了第二个 div,它引用了 posts2 查询并且有效。

{% extends "base.twig" %}

{% block content %}
<div class="blog">
        <h3>Story</h3>
    {% for post in posts %}
        {% include ['tease-story.twig', 'tease.twig'] %}        
    {% endfor %}
    </div>

    <div class="news">
        <h3>News</h3>
    {% for post in posts2 %}
        {% include ['tease-news.twig', 'tease.twig'] %}
    {% endfor %}
    </div>
{% endblock %}

我还创建了一个专门的“tease-story.twig”和“tease-news.twig”作为测试,但实际上并不需要这些。

老实说,我不确定我在这里做了什么,但它确实有效。

如果有人可以帮助澄清,将不胜感激。

于 2016-03-30T15:41:42.670 回答