123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311 |
- <?php
- /**
- * @file Extendable Object Faces API. Provided by the faces module.
- */
- if (!interface_exists('FacesExtenderInterface', FALSE)) {
- /**
- * Interface for extenders.
- */
- interface FacesExtenderInterface {
- /**
- * Constructs an instance of the extender.
- */
- function __construct(FacesExtendable $object);
- /**
- * Returns the extended object.
- */
- public function getExtendable();
- }
- /**
- * The Exception thrown by the FacesExtendable.
- */
- class FacesExtendableException extends ErrorException {}
- }
- if (!class_exists('FacesExtender', FALSE)) {
- /**
- * A common base class for FacesExtenders. Extenders may access protected
- * methods and properties of the extendable using the property() and call()
- * methods.
- */
- abstract class FacesExtender implements FacesExtenderInterface {
- /**
- * @var FacesExtendable
- */
- protected $object;
- function __construct(FacesExtendable $object) {
- $this->object = $object;
- }
- /**
- * Returns the extended object.
- */
- public function getExtendable() {
- return $this->object;
- }
- /**
- * Makes protected properties of the extendable accessible.
- */
- protected function &property($name) {
- $var =& $this->object->property($name);
- return $var;
- }
- /**
- * Invokes any method on the extended object. May be used to invoke
- * protected methods.
- *
- * @param $name
- * The method name.
- * @param $arguments
- * An array of arguments to pass to the method.
- */
- protected function call($name, array $args = array()) {
- return $this->object->call($name, $args);
- }
- }
- }
- if (!class_exists('FacesExtendable', FALSE)) {
- /**
- * An extendable base class.
- */
- abstract class FacesExtendable {
- protected $facesMethods = array();
- protected $faces = array();
- protected $facesIncludes = array();
- protected $facesClassInstances = array();
- static protected $facesIncluded = array();
- /**
- * Wraps calls to module_load_include() to prevent multiple inclusions.
- *
- * @see module_load_include()
- */
- protected static function load_include($args) {
- $args += array('type' => 'inc', 'module' => '', 'name' => NULL);
- $key = implode(':', $args);
- if (!isset(self::$facesIncluded[$key])) {
- self::$facesIncluded[$key] = TRUE;
- module_load_include($args['type'], $args['module'], $args['name']);
- }
- }
- /**
- * Magic method: Invoke the dynamically implemented methods.
- */
- function __call($name, $arguments = array()) {
- if (isset($this->facesMethods[$name])) {
- $method = $this->facesMethods[$name];
- // Include code, if necessary.
- if (isset($this->facesIncludes[$name])) {
- self::load_include($this->facesIncludes[$name]);
- $this->facesIncludes[$name] = NULL;
- }
- if (isset($method[0])) {
- // We always pass the object reference and the name of the method.
- $arguments[] = $this;
- $arguments[] = $name;
- return call_user_func_array($method[0], $arguments);
- }
- // Call the method on the extender object, but don't use extender()
- // for performance reasons.
- if (!isset($this->facesClassInstances[$method[1]])) {
- $this->facesClassInstances[$method[1]] = new $method[1]($this);
- }
- return call_user_func_array(array($this->facesClassInstances[$method[1]], $name), $arguments);
- }
- $class = check_plain(get_class($this));
- throw new FacesExtendableException("There is no method $name for this instance of the class $class.");
- }
- /**
- * Returns the extender object for the given class. May be used to
- * explicitly invoke a specific extender, e.g. a function overriding a
- * method may use that to explicitly invoke the original extender.
- */
- public function extender($class) {
- if (!isset($this->facesClassInstances[$class])) {
- $this->facesClassInstances[$class] = new $class($this);
- }
- return $this->facesClassInstances[$class];
- }
- /**
- * Returns whether the object can face as the given interface, thus it
- * returns TRUE if this oject has been extended by an appropriate
- * implementation.
- *
- * @param $interface
- * Optional. A interface to test for. If it's omitted, all interfaces that
- * the object can be faced as are returned.
- * @return
- * Whether the object can face as the interface or an array of interface
- * names.
- */
- public function facesAs($interface = NULL) {
- if (!isset($interface)) {
- return array_values($this->faces);
- }
- return in_array($interface, $this->faces) || $this instanceof $interface;
- }
- /**
- * Extend the object by a class to implement the given interfaces.
- *
- * @param $interface
- * The interface name or an array of interface names.
- * @param $class
- * The extender class, which has to implement the FacesExtenderInterface.
- * @param $include
- * An optional array describing the file to include before invoking the
- * class. The array entries known are 'type', 'module', and 'name'
- * matching the parameters of module_load_include(). Only 'module' is
- * required as 'type' defaults to 'inc' and 'name' to NULL.
- */
- public function extendByClass($interface, $className, array $includes = array()) {
- $parents = class_implements($className);
- if (!in_array('FacesExtenderInterface', $parents)) {
- throw new FacesExtendableException("The class " . check_plain($className) . " doesn't implement the FacesExtenderInterface.");
- }
- $interfaces = is_array($interface) ? $interface : array($interface);
- foreach ($interfaces as $interface) {
- if (!in_array($interface, $parents)) {
- throw new FacesExtendableException("The class " . check_plain($className) . " doesn't implement the interface " . check_plain($interface) . ".");
- }
- $this->faces[$interface] = $interface;
- $this->faces += class_implements($interface);
- $face_methods = get_class_methods($interface);
- $this->addIncludes($face_methods, $includes);
- foreach ($face_methods as $method) {
- $this->facesMethods[$method] = array(1 => $className);
- }
- }
- }
- /**
- * Extend the object by the given functions to implement the given
- * interface. There has to be an implementation function for each method of
- * the interface.
- *
- * @param $interface
- * The interface name or FALSE to extend the object without a given
- * interface.
- * @param $methods
- * An array, where the keys are methods of the given interface and the
- * values the callback functions to use.
- * @param $includes
- * An optional array to describe files to include before invoking the
- * callbacks. You may pass a single array describing one include for all
- * callbacks or an array of arrays, keyed by the method names. Look at the
- * extendByClass() $include parameter for more details about how to
- * describe a single file.
- */
- public function extend($interface, array $callbacks = array(), array $includes = array()) {
- $face_methods = $interface ? get_class_methods($interface) : array_keys($callbacks);
- if ($interface) {
- if (array_diff($face_methods, array_keys($callbacks))) {
- throw new FacesExtendableException("Missing methods for implementing the interface " . check_plain($interface) . ".");
- }
- $this->faces[$interface] = $interface;
- $this->faces += class_implements($interface);
- }
- $this->addIncludes($face_methods, $includes);
- foreach ($face_methods as $method) {
- $this->facesMethods[$method] = array(0 => $callbacks[$method]);
- }
- }
- /**
- * Override the implementation of an extended method.
- *
- * @param $methods
- * An array of methods of the interface, that should be overriden, where
- * the keys are methods to override and the values the callback functions
- * to use.
- * @param $includes
- * An optional array to describe files to include before invoking the
- * callbacks. You may pass a single array describing one include for all
- * callbacks or an array of arrays, keyed by the method names. Look at the
- * extendByClass() $include parameter for more details about how to
- * describe a single file.
- */
- public function override(array $callbacks = array(), array $includes = array()) {
- if (array_diff_key($callbacks, $this->facesMethods)) {
- throw new FacesExtendableException("A not implemented method is to be overridden.");
- }
- $this->addIncludes(array_keys($callbacks), $includes);
- foreach ($callbacks as $method => $callback) {
- $this->facesMethods[$method] = array(0 => $callback);
- }
- }
- /**
- * Adds in include files for the given methods while removing any old files.
- * If a single include file is described, it's added for all methods.
- */
- protected function addIncludes($methods, $includes) {
- $includes = isset($includes['module']) && is_string($includes['module']) ? array_fill_keys($methods, $includes) : $includes;
- $this->facesIncludes = $includes + array_diff_key($this->facesIncludes, array_flip($methods));
- }
- /**
- * Only serialize what is really necessary.
- */
- public function __sleep() {
- return array('facesMethods', 'faces', 'facesIncludes');
- }
- /**
- * Destroys all references to created instances so that PHP's garbage
- * collection can do its work. This is needed as PHP's gc has troubles with
- * circular references until PHP < 5.3.
- */
- public function destroy() {
- // Avoid circular references.
- $this->facesClassInstances = array();
- }
- /**
- * Makes protected properties accessible.
- */
- public function &property($name) {
- if (property_exists($this, $name)) {
- return $this->$name;
- }
- }
- /**
- * Invokes any method.
- *
- * This also allows to pass arguments by reference, so it may be used to
- * pass arguments by reference to dynamically extended methods.
- *
- * @param $name
- * The method name.
- * @param $arguments
- * An array of arguments to pass to the method.
- */
- public function call($name, array $args = array()) {
- if (method_exists($this, $name)) {
- return call_user_func_array(array($this, $name), $args);
- }
- return $this->__call($name, $args);
- }
- }
- }
|