我正在尝试使用Twig 1.34的WordPress Timber插件实现从WordPress插件高级自定义字段(ACF)中对转发器字段的输出进行排序。从下面的ACF排序的基本PHP示例来自https://www.advancedcustomfields.com/resources/how-to-sorting-a-repeater-field/,在ACF论坛中我的问题没有可用的答案。
所以我试图将这个函数从示例转换为Timber / Twig:
// get repeater field data
$repeater = get_field('repeater');
// vars
$order = array();
// populate order
foreach( $repeater as $i => $row ) {
$order[ $i ] = $row['id'];
}
// multisort
array_multisort( $order, SORT_DESC, $repeater );
// loop through repeater
if( $repeater ): ?>
<ul>
<?php foreach( $repeater as $i => $row ): ?>
<li><?php echo $row['id']; ?>. <?php echo $row['name']; ?></li>
<?php endforeach; ?>
</ul>
<?php endif; ?>
我在view.twig
文件中实现的内容如下所示。我的ACF转发器字段称为时间轴,有两个子字段,日期和描述。
{% set repeater = timeline %}
{% for i, row in repeater %}
{{ row.date}}
{{ row.description}}
{% endfor %}
输出ACF转发器timeline
的现有字段(数组) - 两个字段的日期和描述 - 按照最早的第一个的日期顺序,这是默认值,如下所示:
2010年1月
Lorum Ipsum blah blah blah blah
2011年7月
Lorum Ipsum blah blah blah Lorum Ipsum
但我想先按最新日期排序。
根据这个答案Twig_Error_Syntax for "Unknown filter" with a Twig filter in Timber我需要创建一个使用Twig_SimpleFilter
的过滤函数,并在转发器字段数组上进行排序。
我已对此进行了测试,该答案中的示例rot13
过滤器适用于{{ 'test text'|rot13 }}
。
但是当尝试使用相同的Twig_SimpleFilter
结构和代码按日期对数组进行排序时,即在我的主题的functions.php
文件中使用它:
add_filter('timber/twig', function($twig) {
$twig->addExtension(new Twig_Extension_StringLoader());
$twig->addFilter(
new Twig_SimpleFilter(
'sort_timeline',
function($string) {
$repeater = get_field('timeline');
$order = array();
foreach( $repeater as $i => $row ) {
$order[ $i ] = $row['date'];
}
array_multisort( $order, SORT_DESC, $repeater );
}
)
);
return $twig;
});
并在{% for i, row in repeater|sort_timeline %}
view.twig
过滤器
{% set repeater = timeline %}
{% for i, row in repeater|sort_timeline %}
{{ row.date}}
{{ row.description}}
{% endfor %}
我得到的只是一个白屏,在php日志中没有错误。
FWIW,使用像rot13
这样的{% for i, row in repeater|rot13 %}
过滤器也会显示白屏,因此整体Twig_SimpleFilter
出现问题。
(顺便说一下,我也意识到我可能需要在$row['date']
中转换php日期格式才能正确排序,可能使用strtotime
,因为它现在只是月份和年份。)
在Twig中使用过滤器是否正确尝试此操作?或者是否可以直接在array_multisort( $order, SORT_DESC, $repeater );
模板中调整和使用.twig
函数?
编辑4/17/18
@ num8er的代码适用于php5 +。 php7运算符似乎是一个问题。
并且,下面的其他功能将对转发器的后端行进行排序;更改值并添加到主题的functions.php
文件中。见https://support.advancedcustomfields.com/forums/topic/sort-repeater-in-back-end-where-data-is-entered/
要获取要排序的行的MySQL field_123456789
名称,请转到ACF导出页面并导出字段组。 https://support.advancedcustomfields.com/forums/topic/how-to-retrieve-a-group-field-key/
add_filter('acf/load_value/name=timeline', 'my_acf_load_value', 10, 3); //repeater name is timeline
function my_acf_load_value( $rows)
{
foreach( $rows as $key => $row ) {
$column_id[ $key ] = $row['field_5967e7639a09b'];
}
array_multisort( $column_id, SORT_DESC, $rows );
return $rows;
}
答案 0 :(得分:2)
直接解决您的问题。
试试这段代码:
add_filter('timber/twig', function($twig) {
$twig->addExtension(new Twig_Extension_StringLoader());
$twig->addFilter(
new Twig_SimpleFilter(
'timeline_latest_first',
function($timeline) {
usort($timeline, function($a, $b) {
if(strtotime($a['date']) === strtotime($b['date'])) return 0;
return strtotime($a['date']) > strtotime($b['date']) ? -1 : 1;
// or simply (if php 7.x):
// return -1*(strtotime($a['date']) <=> strtotime($b['date']));
});
return $timeline;
}
)
);
return $twig;
});
{% set timeline_sorted = timeline|timeline_latest_first %}
{% for i, row in timeline_sorted %}
{{ row.date}}
{{ row.description}}
{% endfor %}
答案 1 :(得分:0)
在Twig中,您可以创建自定义过滤器并将其添加到正在运行的Twig环境中。这样,您就可以创建自己的multisort
过滤器:
$multisort_filter = new Twig_Filter('multisort', function ($repeaters) {
$order = array();
foreach( $repeaters as $i => $row ) {
$order[ $i ] = $row['id'];
}
// multisort
array_multisort( $order, SORT_DESC, $repeaters );
return $repeaters;
});
然后将其添加到树枝:
$twig = new Twig_Environment($loader);
$twig->addFilter($filter);
现在,您可以从模板中调用multisort
过滤器。
{% for i, row in repeater|multisort %}
{{ row.date}}
{{ row.description}}
{% endfor %}
有关扩展Twig的更多信息,请参阅他们的文档:https://twig.symfony.com/doc/2.x/advanced.html