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:
All the code below will run fine in Revo 3 without the $prefix
variable, but it will put a “deprecated” message in the Deprecation log, which will slow down the Snippet.
/* 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.
Credit for the method used in this article goes entirely to Jason Coward (OpenGeek). I would never have come up with it on my own.
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.