Manipulating Craft's ElementCriteriaModel objects with Twig
Using Twig to manipulate Craft's ElementCriteriaModel objects makes for leaner and meaner templates. Relatively complex functionalities can also be built pretty easily.
Introduction
While working on the templates of my latest Craft CMS projects, I found myself manipulating ElementCriteriaModel objects with Twig rather frequently. It is a very simple technique but one that is quite flexible and can be used in a wide variety of contexts to solve common problems.
Here are the basic steps:
- Create an empty array.
- Use
craft.entries.ids()
several times to get the IDs of the entries you need to display. - Use Twig's array filters merge and slice to assemble those arrays of ids into a single one using the array we created in step one. Craft's own without and intersect filters should also be part of your arsenal.
- Use
craft.entries.id(myCustomArray).fixedOrder(true).find()
to output exactly the entries you want using your newly created array of entry ids.
Let's go through a few use cases for that pattern.
Complement user-selected entries with recent entries
This is a very common use case. On the homepage of a website, we let administrators choose six projects to highlight using an entries field. In the event they do not pick six of them, we want to complement that list with the most recent projects. We don't want any duplicates in that list.
Using the simple method described above, that's how we can implement this:
{#
1. Create array containing Ids of chosen projects
2. If not up to 6 items, get the Ids of the 6 most recent projects and remove chosen projects Ids from that list
3. Get your final ElementCriteria model using your list of Ids
#}
{% set projectsIds = entry.homeProjects.ids() %}
{% if projectsIds | length < 6 %}
{% set recentProjectsIds = craft.entries.section('projects').limit(6).ids() | without(projectsIds) %}
{% set projectsIds = projectsIds | merge(recentProjectsIds) | slice(0,6) %}
{% endif %}
{% set projects = craft.entries.section('projects').id(projectsIds).fixedOrder(true).find() %}
{# use 'projects' in a for loop for display #}
Display related items on an entry page
Here is another albeit similar scenario. On an entry page, we want to display 5 related entries. Related entries are specified manually by administrators using an entries field.
When no related entries have been found or when all slots are not used, we want to complement our list using entries sharing at least one category with the currently viewed entry. If that's still not enough entries to complete the list, we will just use the most recent articles.
Again, no duplicates allowed and the currently viewed entry should never appear in that list.
{#
1. Create array containing Ids of related articles
2. If not up to 5 items, get 6 entries sharing categories w/ the current entry
3. If not up to 5 items, get the 6 most recent entries
4. No duplicates allowed. Current entry should never appear in that list
#}
{% set articlesIds = entry.articleRelated.ids() %}
{% if articlesIds | length < 5 %}
{% set currentCategories = entry.articleCategories %}
{% if currentCategories | length %}
{% set sameCategoryIds = craft.entries.section('articles').relatedTo(currentCategories).limit(6).ids() | without(articlesIds) | without(entry.id) %}
{% set articlesIds = articlesIds | merge(sameCategoryIds) | slice(0,5) %}
{% endif %}
{% endif %}
{% if articlesIds | length < 5 %}
{% set recentIds = craft.entries.section('articles').limit(6).ids() | without(articlesIds) | without(entry.id) %}
{% set articlesIds = articlesIds | merge(recentIds) | slice(0,5) %}
{% endif %}
{% set relatedArticles = craft.entries.section('articles').id(articlesIds).fixedOrder(true).find() %}
{# use 'relatedArticles' in a for loop for display #}
Order entries by categories
Another use case would be to display entries sorted by categories and reflect the order these have in our category group. An example that came up in a recent project: sponsors and levels of sponsorship.
We want to list the sponsor levels (categories) and preserve the order they have in the control panel. Then, we want to list all sponsors under each category. The client will be able to easily add a sponsor level and assign new sponsors to it. The client will also be able to reorder his sponsorship levels easily if needed.
{#
- Create an empty array to hold our entries ids
- Get the list of categories in the order they are in in the CP
- For each category, get a list of entry ids related to that category
- Merge those arrays together
#}
{% set sponsorsIds = [] %}
{% set sponsorsCategories = craft.categories.group('sponsorLevels').find() %}
{% for category in sponsorsCategories %}
{% set categorySponsorsIds = craft.entries.section('sponsors').relatedTo(category).ids() %}
{% set sponsorsIds = sponsorsIds | merge(categorySponsorsIds) %}
{% endfor %}
{% set sponsors = craft.entries.section('sponsors').id(sponsorsIds).fixedOrder(true).find() %}
{# use 'sponsors' in a for loop for display #}
There you go, three use cases for a simple but powerful technique. I hope you'll find this little trick useful when working on your next Craft project.