Selecting Chunks—The Fast Way

A fast, simple Snippet to select different Chunks to add your web pages.

By Bob Ray  |  May 31, 2023  |  2 min read
Selecting Chunks—The Fast Way

It can make your life easier if you use the same Template on many pages, but you might not want them all to look exactly the same. Say, for example, that you want to vary the header depending on which parent the page is under. You could do something like this:

[[*parent:is=`1`:then=`[[$headerChunk1]]`]]
[[*parent:is=`5`:then=`[[$headerChunk2]]`]]
[[*parent:is=`7`:then=`[[$headerChunk3]]`]]
[[*parent:is=`9`:or:is=`11`:then=`$headerChunk4]]`]]

The performance of this code will be painfully slow because not only will each line have to be parsed and evaluated, the nested Chunk tags will all execute and the all four Chunks will be retrieved even though only one (or none) will be used.

You could improve the performance significantly by using Jason Coward’s famous “Mosquitoes” technique:

[[[[*parent:is=`1`:then=`$headerChunk1`]]]]
[[[[*parent:is=`5`:then=`$headerChunk2`]]]]
[[[[*parent:is=`7`:then=`$headerChunk3`]]]]
[[[[*parent:is=`9`:or:is=`11`:then=`$headerChunk4`]]]]

This will help a lot since at most only one Chunk will be processed, but it will still be fairly slow compared to a dedicated Snippet. Each of the conditional tags will have to be parsed and then evaluated even though most of them won’t result in any output. What if we could use a single tag that didn’t contain a conditional statement?

The trick is to put the logic in a very simple Snippet that uses the PHP switch statement to make the choice. You should be able to use this method even if you’re not that familiar with writing PHP code. If you care at all about page-load times (and you should), learning a little PHP is essential.

The Code

Here’s the tag:

[[SelectChunk? &parentId=`[[*parent]]`]]

Here’s the Snippet code:

<?php
/* SelectChunk snippet */

/* get the parent property from the $scriptProperties array */
$parent = $modx->getOption('parentId', $scriptProperties, null);

/* Set a default chunk to use */
$chunkName = 'MyDefaultChunk';

/* Set $chunkName based on the parent */
switch($parent) {
    case 1:
        $chunkName = 'headerChunk1';
        break;
    case 5:
        $chunkName = 'headerChunk2';
        break;
    case 7:
        $chunkName = 'headerChunk3';
        break;
    case 9:
    case 11:
        $chunkName = 'headerChunk4';
        break;
}

/* Return the appropriate chunk name*/
return $modx->getChunk($chunkName);

The first line of PHP code in our Snippet calls $modx->getOption() to get the &parentId property we sent in the Snippet tag. The [[*parent]] tag inside the SelectChunk tag will be replaced with the content of the parent field of the Resource, which as you probably know, holds the Resource ID of the Resource’s parent (or 0 if the Resource is at the root of the tree).

We discussed the $modx->getOption() method in a previous article. To review, it takes four arguments, separated by commas. In this case, the first one is the name of the property we’re looking for (parentId). The second argument is the array to look in, which in the case of Snippet properties is always the $scriptProperties variable. The third (optional) argument is the default value to use if the property is not sent in the tag (in this case, null, because we want to make sure it’s not a number). As long as you include the &parentId property in the tag, the third argument will never be used. The fourth argument (optional) is a true/false option. If true, you’ll get the default value (from the third argument) if the property in the $ScriptProperties array is set but empty.

To sum up that first line: after it executes the $parent variable will be set to the value sent in the &parent property. Since we’re using a tag to send the parent field. The variable will be set to the ID of the Resource’s parent.

The Switch Statement

The PHP switch statement is very simple. The first line tells PHP what value to use to select which bit of code to execute. In this case, we’re “switching” on the value of the $parent variable. That’s followed by an opening curly brace ({), then the options, followed by a closing curly brace (}).

Between those curly braces. each option starts with the word case followed by a value, then a colon. Indented below that there will be one or more lines of PHP code to execute followed by the break; statement, which tells PHP “break out of the switch statement”, in other words, to jump to the line after the closing curly brace.

When PHP encounters the switch statement, it checks the value in parentheses at the beginning, then looks for a case that matches it. If it finds one, it executes the code for that option, if not it does nothing and jumps past the end of the switch statement. It does this with blinding speed, using a jump table to jump to the appropriate case.

One benchmark test I found ran a switch statement with 15 cases 3.5 million times. The total run took about a third of a second. That means seven passes through the fairly long case statement took about 0.00000066 seconds.

Notice the final two cases in our switch statement. There’s no break; statement between them. The code below them will execute if the parent is either 9 or 11. You can have as many cases as you like here, and you could have similar sets of multiple cases anywhere in the switch statement if you want to execute the same code for multiple cases.

Remember, though, that a given section of PHP code needs a break; statement at the end. If it’s not there, the execution will “fall through” to the next case and its code will execute too. Leaving out the break; statement at the end of a case is the most common mistake made by people learning to use a switch statement. Multiple cases execute, and they have trouble figuring out why.

In our example Snippet, the code in each case of the switch statement simply sets the value of the $chunkName variable to the name of the appropriate Chunk.

At the end of our code, we simply return the content of the selected chunk with return $modx->getChunk($chunkName);. The contents of that Chunk will replace the SelectChunk Snippet tag.

Other Uses

With fairly minor modifications, the code above could be used to select Chunks based on almost any characteristic of the current Resource. The value at the top can be a string, so for example, you could send the pagetitle or the value of a TV in the Snippet tag and adjust the code accordingly. You can also return a string directly rather than getting a Chunk. Here’s an example that assumes TV1 holds the name of a color:

[[!SelectChunk? &tvValue=`[[*Tv1]]` ]]

$tvValue = $modx->getOption('tvValue', $scriptProperties, 'someValue');
$output = 'someValue';
switch($tvValue) {
   case 'red':
        $output = "The color is red"
        break;
   case 'blue':
        $output = "The color is blue"
        break;
   case 'green':
        $output = "The color is green"
        break;
}
return $output;

Because we’re evaluating a string here, rather than a number, the value in each case has to be in quotes.

The only restriction for using a switch statement is that there has to be a single value in the parentheses at the top of the switch statement and the case has to match it. You can’t do any of these:

    case $parent > 5:
    case $parent != 6:
    case empty($parent):

In those cases, you would have to construct an elseif tree like this:

if ($parent > 5) {
     /* some code */
} elseif ($parent != 6) {
    /* some code */
} elseif (empty($parent)) {
    /* some code */
} else {
    /* default case here */
}

Using the elseif method would still be way faster than creating the same options with conditional output modifiers.


Bob Ray is the author of the MODX: The Official Guide and dozens of MODX Extras including QuickEmail, NewsPublisher, SiteCheck, GoRevo, Personalize, EZfaq, MyComponent and many more. His website is Bob’s Guides. It not only includes a plethora of MODX tutorials but there are some really great bread recipes there, as well.