Extending DataObjects
You can add properties and methods to existing DataObject
subclasses like Member
without hacking core code or subclassing by using an Extension
. See the Extending SilverStripe guide for more information.
The following documentation outlines some common hooks that the Extension API provides specifically for managing data records. Note that this is not an exhaustive list - we encourage you to look at the source code to see what other extension hooks are available.
onBeforeWrite
You can customise saving behavior for each DataObject
, e.g. for adding workflow or data customization. The function is
triggered when calling write()
to save the object to the database. This includes saving a page in the CMS or altering
a record via code.
Example: Make sure the player has a valid and unique player number for their team when being assigned a new team.
use SilverStripe\Control\HTTPResponse_Exception;
use SilverStripe\Security\Security;
use SilverStripe\ORM\DataObject;
class Player extends DataObject
{
private static $db = [
// ...
'Number' => 'Int',
];
private static $has_one = [
'Team' => Team::class,
];
public function onBeforeWrite()
{
// Use $this->isInDb() to check if the record is being written to the database for the first time
if (!$this->isInDb() && $this->Team()->exists()) {
$this->Number = $this->Team()->getAvailablePlayerNumber();
}
// If the player changed teams
if ($this->isChanged('TeamID') && $this->Team()->exists()) {
// If the player's number is already used by someone else on this team
if (in_array($this->Number, $this->Team()->Players()->exclude('ID', $this->ID)->column('Number'))) {
// Assign a new player number
$this->Number = $this->Team()->getAvailablePlayerNumber();
}
}
// CAUTION: You are required to call parent::onBeforeWrite(), otherwise
// SilverStripe will not execute the request.
parent::onBeforeWrite();
}
}
onBeforeDelete
Triggered before executing delete()
on an existing object. It can be useful if you need to make sure you clean up some other data/files/etc which aren't directly associated with the actual DataObject
record.
use SilverStripe\ORM\DataObject;
class Player extends DataObject
{
// ...
public function onBeforeDelete()
{
/* Do some cleanup here relevant to your project before deleting the actual database record */
// CAUTION: You are required to call parent::onBeforeDelete(), otherwise
// SilverStripe will not execute the request.
parent::onBeforeDelete();
}
}
$this->isInDb()
to toggle
these two modes, as shown in the example above.