Speed up Page Loads With a Plugin

A clever strategy for speeding up page loads: a Plugin that pre-processes TV values and saves them for display.

By Bob Ray  |  April 18, 2023  |  6 min read
Speed up Page Loads With a Plugin

In a previous article, we looked at some Plugins that execute when a Resource is saved in the Manager. In this one, we’ll see some Plugin code that can dramatically speed up page loads by storing processed information in the Resource’s introtext field every time the Resource is saved in the Manager.

For a variety of reasons, processing TVs every time a page is loaded can really slow things down. Especially if the TV tags are in a Tpl Chunk used in an aggregating Snippet like getResources. What if we could store the processed results of those tags in the Resource’s introtext field just once, when the Resource is saved. Think of all the time that would be saved later on. This example can serve as an alternative to getResources, but it can also be used to dramatically speed up getResources by placing all the TV data in a single Resource field.

This assumes that you aren’t using the introtext field for anything. You could also use the description field for this.

The following example Plugin should be connected to the OnDocFormSave event. To use it, paste the code into a Plugin and, on the “System Events” tab, check the box next to OnDocFormSave. Then, save the Plugin. In the code of a Plugin attached to that event, the Resource being saved is always available as the $resource variable.

More Fun With Dogs and Cats

In an earlier article, we used an example that set TV values with the characteristics of dogs and cats. In this one, we’ll assume that we have the same TVs, but we’ll set the introtext (summary) field of each Resource using a Tpl Chunk containing placeholders for TV values that hold information about the animal.

As we did in the previous article, each cat or dog will have its own Resource with TVs for the breed, weight, birthdate, and description. All the dogs are stored under the “Dogs” folder, which has an ID of 12. All cats will be stored under the “Cats” folder, which has an ID of 22.

The method shown here can be used in a lot of different situations to essentially “pre-process” tags and save formatted output so the tags don’t have to be parsed on every page load. If you don’t want to use the introtext field, you could always store the formatted output in another field or in a dedicated TV. You could also use the MIGX Extra for this.

In this Plugin, we’ll leave out the code from the previous article that limits execution to new Resources. We want the Plugin to execute whether it’s a new Rsource or not. That way, when you change any of the relevant TV values, the introtext field will be updated when you save the Resource.

Our Tpl Chunk might look like this (let’s call it “IntrotextTpl”):

### [[+name]]

<b>Species: [[+species]]</b>

    * Breed: [[+breed]]
    * Weight: [[+weight]
    * Birth date: [[+dob]]
    * Description: [[+desc]]

Our Plugin will use that Tpl Chunk to process the tags and write the complete description to the introtext field of the Resource.

The Code

Here’s the code for our Plugin:

$dogFolderId = 12; // You'd change these two values
$catFolderId = 22; // to the IDs of actual pages

/* We'll use this array to hold the values
   to place in the chunk */
$fields = array(
'weight' => $resource->getTVValue('weight'),
);

/* Handle dogs */
if ($resource->get('parent') == $dogFolderId) {
$fields['species'] = 'Canine';
}

/* Handle cats */
if ($resource->get('parent') == $catFolderId) {
$fields['species'] = 'Feline';
}

/* Handle the other fields */
$fields['name'] = $resource->getTVValue('name');
$fields['breed'] = $resource->getTVValue('breed');
$fields['weight'] = $resource->getTVValue('weight');
$fields['desc'] = $resource->getTVValue('desc');
$fields['dob'] = $resource->getTVValue('birthdate');

$tpl = $modx->getChunk('IntrotextTpl', $fields);
$resource->set('introtext', $tpl);
$resource->save();
return '';

The MODX getChunk() method will retrieve a Chunk and replace its placeholder tags with the values sent in the array in the second argument (in this case, $fields).

Now, you can display a fully-formatted description for any animal just by putting this tag on the page:

[[*introtext]]

This will give you a lot faster page loads than creating a regular Tpl Chunk for the page since all the parsing is done once when the Resource is saved, instead of doing it whenever the Resource is viewed.

The example above is a little contrived, because in reality, you might have the animal’s name in the pagetitle field and its description in the content field. In that case you might add this code in the Plugin to set those placeholders:

$fields['name'] = $resource->get('pagetitle');
$fields['desc'] = $resource->get('content');

It has probably occurred to you that you could create a similar display with getResources, though the page-load times with our example would be much faster. And with getResources, it would be trickier to display cats and dogs in the same list unless you created a species TV, which would slow things down even more. If you wanted to display cats and dogs separately, it would be easy to use our example to generate a very fast display for each group using getResources, but without having getResources retrieve or process any TVs, which would speed things up immensely. You’d just put this tag in the AnimalDescriptionTpl Chunk:

[[+introtext]]

The code on the page to display descriptions for all the animals might look something like this:

### Dogs

[[!getResources? &parents=`12` &tpl=`AnimalDescriptionTpl`]]

### Cats

[[!getResources? &parents=`22` &tpl=`AnimalDescriptionTpl`]]

Now, getResources doesn’t have to process any TVs at all. Notice that we don’t have to use the Tpl we created above that formats the animal’s description at all here. The fully-formatted description is already in the introtext field, which is why this method is so much faster.

A Final Note

One of the great things about this method is that you can actually see the results in the introtext (summary) field when you edit the Resource. Don’t be tempted to edit that field when editing a Resource, though. Your changes will be overwritten as soon as you save the Resource. If you want to change what’s there, modify the Tpl Chunk used in your Plugin. Remember that the changes won’t show up for a particular Resource until that specific Resource is saved in the Manager. Clearing the Site Cache will not update the introtext field.

If you are using this technique and have other admin users, you might want to hide the introtext field from them with Form Customization, so they won’t be tempted to edit it.


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.