Behaviors & Dynamic Class Extension

Introduction

Dynamic class extension provides the following benefits:

  1. Add new properties and methods dynamically.
  2. Bind to local events fired in models, widgets, and other locations, even those in the core modules or other plugins.
  3. Add additional behaviours (private traits, see below) to classes.
  4. Must extend the Winter\Storm\Extension\Extendable class or implement the Winter\Storm\Extension\ExtendableTrait.
  5. Already-defined methods and properties cannot be overridden.
  6. Extendable classes are protected against having undefined properties set on first use and must use $object->addDynamicProperty(); instead.
  7. Dynamically added methods cannot override methods defined in code.

See below for a sample of what is possible with dynamic class extension:

// Dynamically extend a model that belongs to a third party plugin
Post::extend(function ($model) {
    // Bind to an event that's only fired locally
    $model->bindEvent('model.afterSave', function () use ($model) {
        if (!$model->isValid()) {
            throw new \Exception("Invalid Model!");
        }
    });

    // Add a new property to the class
    $model->addDynamicProperty('tagsCache', null);

    // Add a new method to the class
    $model->addDynamicMethod('getTagsAttribute', function() use ($model) {
        if ($model->tagsCache) {
            return $model->tagsCache;
        } else {
            return $model->tagsCache = $model->tags()->lists('name');
        }
    });
});

Dynamic Class Extension also adds the ability for classes to have private traits, also known as Behaviors. These are similar to native PHP Traits except they have some distinct benefits:

  1. Behaviors have their own constructor.
  2. Behaviors can have private or protected methods.
  3. Methods and property names can conflict safely.
  4. Provide a safe mechanism for shared functionality across controllers whilst still maintaining their own state.
  5. Classes can be extended with behaviors dynamically.

The best of example of the power of behaviors would be the backend form, list, and relation ControllerBehaviors that implement the majority of CRUD requirements in Winter CMS for any controllers that implement them.

Comparison to Traits

Where you might use a PHP trait like this:

class MyClass
{
    use \Winter\Storm\UtilityFunctions;
    use \Winter\Storm\DeferredBinding;
}

A behavior is used in a similar fashion:

class MyClass extends \Winter\Storm\Extension\Extendable
{
    public $implement = [
        'Winter.Storm.UtilityFunctions',
        'Winter.Storm.DeferredBinding',
    ];
}

Where you might define a trait like this:

trait UtilityFunctions
{
    public function sayHello()
    {
        echo "Hello from " . get_class($this);
    }
}

A behavior is defined like this:

class UtilityFunctions extends \Winter\Storm\Extension\ExtensionBase
{
    protected $parent;

    public function __construct($parent)
    {
        $this->parent = $parent;
    }

    public function sayHello()
    {
        echo "Hello from " . get_class($this->parent);
    }
}

The extended object is always passed as the first parameter to the Behavior's constructor.

To summarize:

  • Extend \Winter\Storm\Extension\ExtensionBase to declare your class as a Behaviour
  • The class wanting to -implement- the Behaviour needs to extend \Winter\Storm\Extension\Extendable

NOTE: See Using traits instead of base classes

Extending Constructors

Any class that uses the Extendable or ExtendableTrait can have its constructor extended with the static extend method. The argument should pass a closure that will be called as part of the class constructor.

MyNamespace\Controller::extend(function ($controller) {
    //
});

By default, extension closures used with the extend method will run in the scope of the current class. For example, if you are extending another class in your Plugin initialization class (Plugin.php), the closure's scope will be of that Plugin initialization class. This allows you to call methods and properties from your class.

However, if you would like to scope the closure to the class that you are extending, you may pass true as the second parameter to change the scope of the closure. This changes the $this magic variable to be of the class you are extending, not of the current class.

namespace Acme\Plugin;

class Plugin extends PluginBase
{
    public function boot()
    {
        MyNamespace\Controller::extend(function ($controller) {
            // Scoped to \Acme\Plugin\Plugin
        });

        MyNamespace\Controller::extend(function () {
            // Scoped to \MyNamespace\Controller
        }, true);
    }
}

This has the benefit of allowing you to reference private or protected methods and properties in the class that you are extending.

Note that when this local scope is enabled, by default, no class is passed to the closure as the first parameter. You can, optionally, provide an "outer" scope to pass through to the first parameter of the closure by adding an object as the third parameter. This allows you to, for example, still use methods and properties from your Plugin class inside the locally-scoped closure.

namespace Acme\Plugin;

class Plugin extends PluginBase
{
    public function boot()
    {
        \MyNamespace\Controller::extend(function ($outer) {
            // Scoped to \MyNamespace\Controller
            // $outer = \Acme\Plugin\Plugin
        }, true, $this);
    }
}

Dynamically declaring properties

Properties can be declared on an extendable object by calling addDynamicProperty and passing a property name and value.

Post::extend(function($model) {
    $model->addDynamicProperty('tagsCache', null);
});

NOTE: Attempting to set undeclared properties through normal means ($this->foo = 'bar';) on an object that implements the Winter\Storm\Extension\ExtendableTrait will not work. It won't throw an exception, but it will not autodeclare the property either. addDynamicProperty must be called in order to set previously undeclared properties on extendable objects.

Retrieving Dynamic Properties

Properties created dynamically can be retrieved with the getDynamicProperties function inherited from the ExtendableTrait.

So retrieving all dynamic properties would look like this:

$model->getDynamicProperties();

This will return an associative array [key => value], with the key being the dynamic property name and the value being the property value.

If we know what property we want we can simply append the key (property name) to the function:

$model->getDynamicProperties()[$key];

Dynamically Creating Methods

Methods can be created to an extendable object by calling addDynamicMethod and passing a method name and callable object, like a Closure.

Post::extend(function($model) {
    $model->addDynamicProperty('tagsCache', null);

    $model->addDynamicMethod('getTagsAttribute', function() use ($model) {
        if ($model->tagsCache) {
            return $model->tagsCache;
        } else {
            return $model->tagsCache = $model->tags()->lists('name');
        }
    });
});

Checking the Existence of a Method

You can check for the existence of a method in an Extendable class by using the methodExists method - similar to the PHP method_exists() function. This will detect both standard methods and dynamic methods that have been added through a addDynamicMethod call. methodExists accepts one parameter: a string of the method name to check the existence of.

Post::extend(function($model) {
    $model->addDynamicMethod('getTagsAttribute', function () use ($model) {
        return $model->tagsCache;
    });
});

$post = new Post;

$post->methodExists('getTagsAttribute'); // true
$post->methodExists('missingMethod'); // false

List all Available Methods

To retrieve a list of all available methods in an Extendable class, you can use the getClassMethods method. This method operates similar to the PHP get_class_methods() function in that it returns an array of available methods in a class, but in addition to defined methods in the class, it will also list any methods provided by an extension or through an addDynamicMethod call.

Post::extend(function($model) {
    $model->addDynamicMethod('getTagsAttribute', function () use ($model) {
        return $model->tagsCache;
    });
});

$post = new Post;

$methods = $post->getClassMethods();

/**
 * $methods = [
 *   0 => '__construct',
 *   1 => 'extend',
 *   2 => 'getTagsAttribute',
 *   ...
 * ];
 */

Dynamically Implementing Behaviors

This unique ability to extend constructors allows behaviors to be implemented dynamically, for example:

/**
 * Extend the Winter.Users Users controller to include the RelationController behavior too
 */
Winter\Users\Controllers\Users::extend(function($controller) {
    // Implement the list controller behavior dynamically
    $controller->implement[] = \Backend\Behaviors\RelationController::class;

    // Declare the relationConfig property dynamically for the RelationController behavior to use
    $controller->addDynamicProperty('relationConfig', '$/myvendor/myplugin/controllers/users/config_relation.yaml');
});

Local Extensions

Extendable classes can be provided local extensions by calling the extend() method directly on an instance of an extendable class. Compared to the statically-called ::extend() method, this has the effect of extending just a single instance of a given model after constructions. This means that behaviors will be loaded and available.

For example, you may choose to add a callback method to a model only in certain circumstances, such as during a search.

public function searchModel($query, $model)
{
    $model->extend(function () use ($query) {
        $this->addDynamicMethod('afterFetch', function () use ($query) {
            $this->searchQuery = $query;
        });
    });

    return $model->where('name', 'like', '%' . $query . '%')->get();
}

By design, all local extensions are scoped to the class that is being extended, which means that $this inside the callback method will call methods and properties within the class being extended. You may provide an "outer" scope as the second parameter of the extend() method to pass through to the first parameter of the closure.

public function searchModel($query, $model)
{
    $model->extend(function ($outerScope) use ($query) {
        // "$this" will be scoped to the $model class.
        // "$outerScope" will be scoped to the outside class (ie. the one that defined the "searchModel" method)

        $this->addDynamicMethod('afterFetch', function () use ($query, $outerScope) {
            $this->searchQuery = $query;
            $this->searchClass = $outerScope;
        }, true);
    }, $this);

    return $model->where('name', 'like', '%' . $query . '%')->get();
}

Usage Examples

Behavior / Extension class

<?php namespace MyNamespace\Behaviors;

class FormController extends \Winter\Storm\Extension\ExtensionBase
{
    /**
        * @var Reference to the extended object.
        */
    protected $controller;

    /**
        * Constructor
        */
    public function __construct($controller)
    {
        $this->controller = $controller;
    }

    public function someMethod()
    {
        return "I come from the FormController Behavior!";
    }

    public function otherMethod()
    {
        return "You might not see me...";
    }
}

Extending a class

This Controller class will implement the FormController behavior and then the methods will become available (mixed in) to the class. We will override the otherMethod method.

<?php namespace MyNamespace;

class Controller extends \Winter\Storm\Extension\Extendable
{
    /**
     * Implement the FormController behavior
     */
    public $implement = [
        'MyNamespace.Behaviors.FormController'
    ];

    public function otherMethod()
    {
        return "I come from the main Controller!";
    }
}

Using the Extension

$controller = new MyNamespace\Controller;

// Prints: I come from the FormController Behavior!
echo $controller->someMethod();

// Prints: I come from the main Controller!
echo $controller->otherMethod();

// Prints: You might not see me...
echo $controller->asExtension('FormController')->otherMethod();

Detecting utilized extensions

To check if an object has been extended with a behavior, you may use the isClassExtendedWith method on the object.

$controller->isClassExtendedWith(\Backend\Behaviors\RelationController::class);

Below is an example of dynamically extending a UsersController of a third-party plugin utilizing this method to avoid preventing other plugins from also extending the afore-mentioned third-party plugin.

UsersController::extend(function($controller) {
    // Implement behavior if not already implemented
    if (!$controller->isClassExtendedWith(\Backend\Behaviors\RelationController::class)) {
        $controller->implement[] = \Backend\Behaviors\RelationController::class;
    }

    // Define property if not already defined
    if (!isset($controller->relationConfig)) {
        $controller->addDynamicProperty('relationConfig');
    }

    // Splice in configuration safely
    $myConfigPath = '$/myvendor/myplugin/controllers/users/config_relation.yaml';

    $controller->relationConfig = $controller->mergeConfig(
        $controller->relationConfig,
        $myConfigPath
    );
}

Soft Definitions

If a behavior class does not exist, like a trait, a Class not found error will be thrown. In some cases you may wish to suppress this error, for conditional implementation if a behavior is present in the system. You can do this by placing an @ symbol at the beginning of the class name.

class User extends \Winter\Storm\Extension\Extendable
{
    public $implement = ['@Winter.Translate.Behaviors.TranslatableModel'];
}

If the class name Winter\Translate\Behaviors\TranslatableModel does not exist, no error will be thrown. This is the equivalent of the following code:

class User extends \Winter\Storm\Extension\Extendable
{
    public $implement = [];

    public function __construct()
    {
        if (class_exists('Winter\Translate\Behaviors\TranslatableModel')) {
            $this->implement[] = 'Winter.Translate.Behaviors.TranslatableModel';
        }

        parent::__construct();
    }
}

Using Traits instead of Base Classes

For those cases where you may not wish to extend the ExtensionBase or Extendable classes, you can use the traits instead. Your classes will have to be implemented as follows:

First let's create the class that will act as a Behaviour, ie. can be -implemented- by other classes.

<?php namespace MyNamespace\Behaviours;

class WaveBehaviour
{
    use \Winter\Storm\Extension\ExtensionTrait;

    /**
        * When using the Extensiontrait, your behaviour also has to implement this method
        * @see \Winter\Storm\Extension\ExtensionBase
        */
    public static function extend(callable $callback)
    {
        self::extensionExtendCallback($callback);
    }

    public function wave()
    {
        echo "*waves*<br>";
    }
}

Now let's create the class that is able to -implement- behaviours using the ExtendableTrait.

class AI
{
    use \Winter\Storm\Extension\ExtendableTrait;

    /**
        * @var array Extensions implemented by this class.
        */
    public $implement;
    /**
        * Constructor
        */
    public function __construct()
    {
        $this->extendableConstruct();
    }
    public function __get($name)
    {
        return $this->extendableGet($name);
    }
    public function __set($name, $value)
    {
        $this->extendableSet($name, $value);
    }
    public function __call($name, $params)
    {
        return $this->extendableCall($name, $params);
    }
    public static function __callStatic($name, $params)
    {
        return self::extendableCallStatic($name, $params);
    }
    public static function extend(callable $callback)
    {
        self::extendableExtendCallback($callback);
    }

    public function youGotBrains()
    {
        echo "I've got an AI!<br>";
    }
}

The AI class is now able to use behaviours. Let's extend it and have this class implement the WaveBehaviour.

<?php namespace MyNamespace\Classes;

class Robot extends AI
{
    public $implement = [
        'MyNamespace.Behaviours.WaveBehaviour'
    ];

    public function identify()
    {
        echo "I'm a Robot<br>";
        echo $this->youGotBrains();
        echo $this->wave();
    }
}

You can now utilize the Robot as follows:

$robot = new Robot();
$robot->identify();

Which will output:

I'm a Robot
I've got an AI!
*waves*

Remember:

  • When using the ExtensionTrait the methods from ExtensionBase should be applied to the class.
  • When using the ExtendableTrait the methods from Extendable should be applied to the class.
Copyright © 2024 Winter CMS