-
Notifications
You must be signed in to change notification settings - Fork 0
Contains filter for pages, users and files fields #566
Comments
PS: This is not to be confused with the |
Maybe |
How would the filter know what is stored in the fields? It could be pages, files or a list of completely different elements. So to be honest, I think this is too specific to be included in the core. It would be great as a plugin however. |
The filter doesn‘t have to know what is stored in the field from my understanding: If the filter accepted string values only as in my example, it would just have to convert the field value with the If the filter also accepted Kirby objects – a page, file or user to filter by – the filter function can deduce the context from this object so the info is there in plain sight. |
Why is this too specific if there are three core fields missing a simple filtering process? Simple in the sense that the given examples in the docs are complex PHP when I look at it compared to the other filters. |
In your example above you used But that leaves us with another issue: There can be different separators for the fields (a YAML list, a comma-separated list or a list separated with a completely custom character). Also the method on the objects could actually return something else than a string (for example an array with strings). Ideally the
I was referring to the original example with |
But there are filters for these cases already, aren‘t there? Like
|
Yes, I forgot about the |
I just noticed something interesting: if I use $page->children()->filterBy('related', 'my-id', '-'); // returns all pages containing 'my-id' in the related field It seems like Kirby is internally trimming the white space when splitting which makes it possible to parse the YAML of the pages, users and files fields. So actually the |
I'm not sure if this is just happening by accident, but if not the docs should be updated accordingly because this simplifies content filtering a lot. |
As an additional note: it seems like Kirby does always store content for the pages, users and files fields using the list syntax – one item per row, prefixed with a dash – no matter if the field allows for single or multiple items. So it's Users:
- asd9ofus for single items and Users:
- asd9ofus
- dfjh898d
- 24kj4ksg for multiple. So splitting by |
It is. Your solution is a clever hack, but not an "official" feature I'd add to the docs. |
I think I don't agree that it's a hack because it's not misusing a function in a way it wasn't intended to be used. So I actually like the simplicity of the approach conceptually. On a more general note, when I started using Kirby, I found the concept of defining a split character on filters or using the $collection->filterBy('category', '=', 'article');
$collection->filterBy('date', '>', 'today'); I'm fine with the shortcut to skip the comparison argument for the most common equals filter. The number of arguments is reduced from three to two so I understand this is some kind of shortcut. To filter lists, I expected a comparison filter as well. Something like this: // finding pages that have news selected in their tags field
$collection->filterBy('tags', 'has', 'news');
$collection->filterBy('tags', 'lists', 'news'); A custom split character could be part of that comparison argument, too. Or it could be a fourth argument: $collection->filterBy('tags', 'lists ,' 'news');
$collection->filterBy('tags', 'lists', 'news', ','); This would keep the logical order field, compares, term which is very understandable in the context of the other filters. Or it would extend the logical order by an additional argument field, compares, term, by which would make it clear that this filter is doing more than other filters (three vs. four arguments). Instead, for filtering lists, we have field, term, split which also has three arguments but works completely different than the other filters with three arguments. It's like a shortcut within a shortcut. And this is what I'd call a "clever hack" 😉 I'd always avoid that for clarity reasons. This cleverness made it very, very hard for me to understand and memorize filtering in Kirby from the beginning. It's a big flaw in my eyes and the reason why I have to read the filtering compendium in the docs every single time I have to set up a filter. So – independent of the implementation – I think it would help filtering in Kirby a lot to introduce a |
PS: My guess is that filtering is the way it is for historical reasons, too. So while it made sense at the time and while you are used to it please think about it again with the eyes of a newbie. Thanks :) |
I created a small test plugin and this seems to work fine for pages, users and files fields as well as for character-separated content: Kirby::plugin('hananils/has-filter', [
'collectionFilters' => [
'has' => function ($collection, $field, $test, $split = false) {
foreach ($collection->data as $key => $item) {
$reference = $collection->getAttribute($item, $field);
if ($split !== false) {
$values = Str::split($reference->value(), $split);
} else {
$values = $reference->yaml();
}
if ($reference && in_array($test, $values)) {
continue;
}
unset($collection->$key);
}
return $collection;
}
]
]); Filtering syntax: /**
* Pages:
*
* - page-1
* - page-2
*/
$collection->filterBy('pages', 'has', 'page-1');
/**
* Tags: this, is, good, news
*/
$collection->filterBy('tags', 'has', 'news', ',')); It could be enhanced to also allow for (unnested) structure fields: /**
* Structure:
*
* -
* key: value
*/
$collection->filterBy('structure', 'has', ['key', 'value']); Would be great to see something alike adapted in the core. |
@lukasbestle @bastianallgeier Is this something that should be moved to Nolt (I don't know how to best do this without loosing too much context) or is this something that can be converted to a normal issue? |
I'd say we can either leave this here for now or you can create a Nolt idea with a summarized description and a link to this issue. We will archive this repo at some point but the existing issues should still be accessible by URL. |
It's a common use case to create content relations with the pages, users or files fields. While this works pretty straightforward in the interface, on the front-end side filtering gets complex. Citing the docs:
This is very cumbersome when you have to do a lot of relation filtering.
It would be very helpful if Kirby provided a
has
filter that would replicate above functionality:There would also be the need for a reversed filter:
These filters could also have shortcuts using
?
and!
:The text was updated successfully, but these errors were encountered: