Get a Field Value—FAST!!!

A quick and efficient way to get the value of a single field for a MODX object like a Resource, User, or User Profile.

By Bob Ray  |  August 8, 2023  |  6 min read
Get a Field Value—FAST!!!

Suppose you want the value of a single field from some MODX object (most often, a Resource, user, or user profile). Let’s use the example of a Resource. You know something about the Resource (e.g., its pagetitle, alias, or id). The usual way of getting a field value in code is to get the Resource object with $doc = $modx->getObject() and then get the field with $doc->get('fieldName');. You also have to test the return value of getObject() because if it fails, the call to get() will throw a PHP error.

Here’s an example of the traditional method that will work in both Revo 2 and Revo 3. It assumes that you know the Resource’s pagetitle, and you want its introtext field:

/* Make it run in either MODX 2 or MODX 3 */

$prefix = $modx->getVersionData()['version'] >= 3
  ? 'MODX\Revolution\\'
  : '';

$pTitle = 'Products';
$doc = $modx->getObject($prefix . 'modResource', array('pagetitle' => $pTitle));
if ($doc) {
    $pTitle = $doc->get('pagetitle');
}

This method is extremely convenient and the code is short. It’s somewhat slow, though, and also wasteful of memory. You only want the value of that one field, but you’re pulling an entire Resource object into memory. If you want to see just how wasteful that is, put a tag for this Snippet on a page:

return print_r($modx->resource, true);

The Resource object is monstrous compared to the size of the field you want. What if there were an easy way to get just the field? With a little xPDO magic, there is.

A Concrete Example

This, much faster and more efficient method does the same thing we did above: it gets the introtext field of the “Products” Resource.

$prefix = $modx->getVersionData()['version'] >= 3
  ? 'MODX\Revolution\\'
  : '';
$query = $modx->newQuery($prefix . 'modResource', array(
    'pagetitle' => 'Products',
));
$query->select('introtext');
$intro = $modx->getValue($query->prepare());

Breaking It Down

The method above will work for any MODX object with very little modification. The first argument to $modx->newQuery() is the class of the object you want to get a field value from. It might be a modUser, modResource, modUserProfile, object or any other MODX object. For Revo 3, these would have the prefix MODX\Revolution\.

The array member will always contain what you already know about the object, with the name of the field you know on the left, and its value on the right. In this case, it’s 'pagetitle' => 'Products'. It could just a well be 'id' => '23' if you knew the ID of the Resource.

Finally, the value in the $query->select() line is the name of the field you want the value of, in this case, introtext.

If anything goes wrong (the object isn’t found or the field isn’t set), $modx->getValue() will return false.

It’s fast, it’s efficient, and it doesn’t care about the fields you don’t want.

Let’s look at a couple examples of using this method to get user information. We’ll assume that you want the email address of the user. The email address is in the user profile, not the modUser object. In each modUserProfile object, the user’s ID is in the internalKey field, so we need to use that field in the array argument to $modx->newQuery().

Case 1—You Know the User’s ID

If you know the user’s ID, you can query the modUserProfile object directly:

$prefix = $modx->getVersionData()['version'] >= 3
  ? 'MODX\Revolution\\'
  : '';

$userId = 12;
$query = $modx->newQuery($prefix . 'modUserProfile', array(
    'internalKey' => $userId,
));
$query->select('email');
$email = $modx->getValue($query->prepare());

This was really easy because we knew the user’s ID, but what if we only had the username?

Case 2—You Know the User’s username

The username is not stored in the user profile, so we need an extra step here. Luckily, we can use our method for both steps. Neither the user object, nor the user profile object need to be retrieved, just the two fields we want:

$prefix = $modx->getVersionData()['version'] >= 3
  ? 'MODX\Revolution\\'
  : '';

/* Get the user ID */
$userName = 'BobRay';
$query1 = $modx->newQuery($prefix . 'modUser', array(
    'username' => $userName,
));
$query1->select('id');
$userId = $modx->getValue($query1->prepare());

/* Make sure we got it */
if ($userId === false) {
    /* Error! */
} else {
    /* Get the email */
    $query2 = $modx->newQuery($prefix . 'modUserProfile', array(
        'internalKey' => $userId,
    ));
    $query2->select('email');

    $email = $modx->getValue($query2->prepare());
}

The test of the $userId variable is important. If no user is found with that username, the $userId will be false. Using that value in the second query may actually return a user, but not the one you’re looking for—definitely not what you want to happen.

More Fun With Users

The user-related fields in the Resource object (createdby, editedby, publishedby, and deletedby) are stored as IDs. Let’s use our method to get the username of the user who created the current Resource:

$prefix = $modx->getVersionData()['version'] >= 3
  ? 'MODX\Revolution\\'
  : '';

$userId = $modx->resource->get('createdby');
$query = $modx->newQuery($prefix . 'modUser', array(
    'id' => $userId,
));

$query->select('username');
$userName = $modx->getValue($query->prepare());

What if we want the full name? It’s in the user profile, so we’d do this:

$prefix = $modx->getVersionData()['version'] >= 3
  ? 'MODX\Revolution\\'
  : '';
$userId = $modx->resource->get('createby');

$query = $modx->newQuery($prefix . 'modUserProfile', array(
    'internalKey' => $userId,
));

$query->select('fullname');
$fullName = $modx->getValue($query->prepare());

Some Caveats

This method is fast and efficient, but there are a few catches you need to be aware of.

First, getValue() will always return a string containing the raw value of the field. With date fields, for example, it will return the raw timestamp (as opposed to get(), which will return a formatted time string). For boolean (true/false) fields like published and isfolder it will return '0' or '1'.

If a date field is empty (say, pub_date when there isn’t one set), getValue() will return '0'. If you try to display that empty date, you’ll get a result something like this:

Wednesday, December 31 1969

For empty text fields like introtext and description, it will return an empty string ('').

If a field might be empty, you should also test for that before tyring to display the field:

if (!empty($field)) {
    /* Go ahead and display it */
}

If you use it to get the content of an object such as a Resource, Chunk, Template, Snippet, or Plugin, it will not process any of the tags in that field, though if you return the value in a Snippet, MODX will process the tags before the page is displayed. You also have to know the actual field name of the content field. It’s content for Resources, but not for the other objects.

You can see the fields of each object here. The fields are the same for Revo 2.

As written above, the method won’t get a TV value for a Resource, or even the default value of the TV.

In the next few articles, we’ll look at some trickier uses for this method and see how to use it with boolean fields, date fields, TVs, and some other objects.


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.