* $val = $obj->label; // equivalent to $val = $obj->getLabel(); * $obj->label = 'Nette'; // equivalent to $obj->setLabel('Nette'); * * Property names are case-sensitive, and they are written in the camelCaps * or PascalCaps. * * Event functionality is provided by declaration of property named 'on{Something}' * Multiple handlers are allowed. * * public $onClick; // declaration in class * $this->onClick[] = 'callback'; // attaching event handler * if (!empty($this->onClick)) ... // are there any handlers? * $this->onClick($sender, $arg); // raises the event with arguments * * * Adding method to class (i.e. to all instances) works similar to JavaScript * prototype property. The syntax for adding a new method is: * * function MyClass_prototype_newMethod(MyClass $obj, $arg, ...) { ... } * $obj = new MyClass; * $obj->newMethod($x); // equivalent to MyClass_prototype_newMethod($obj, $x); * * * @author David Grudl * @copyright Copyright (c) 2004, 2008 David Grudl * @package Nette */ abstract class Object { /** * Returns the name of the class of this object. * * @return string */ final public function getClass() { return get_class($this); } /** * Access to reflection. * * @return ReflectionObject */ final public function getReflection() { return new ReflectionObject($this); } /** * Call to undefined method. * * @param string method name * @param array arguments * @return mixed * @throws ::MemberAccessException */ public function __call($name, $args) { $class = get_class($this); if ($name === '') { throw new /*::*/MemberAccessException("Call to class '$class' method without name."); } // event functionality if (self::hasEvent($class, $name)) { $list = $this->$name; if (is_array($list) || $list instanceof Traversable) { foreach ($list as $handler) { call_user_func_array($handler, $args); } } return; } // object prototypes support Class__method() // (or use class Class__method { static function ... } with autoloading?) $cl = $class; do { if (function_exists($nm = $cl . '_prototype_' . $name)) { array_unshift($args, $this); return call_user_func_array($nm, $args); } } while ($cl = get_parent_class($cl)); throw new /*::*/MemberAccessException("Call to undefined method $class::$name()."); } /** * Call to undefined static method. * * @param string method name (in lower case!) * @param array arguments * @return mixed * @throws ::MemberAccessException */ public static function __callStatic($name, $args) { $class = get_called_class(); throw new /*::*/MemberAccessException("Call to undefined static method $class::$name()."); } /** * Returns property value. Do not call directly. * * @param string property name * @return mixed property value * @throws ::MemberAccessException if the property is not defined. */ public function &__get($name) { $class = get_class($this); if ($name === '') { throw new /*::*/MemberAccessException("Cannot read an class '$class' property without name."); } // property getter support $m = 'get' . $name; if (self::hasAccessor($class, $m)) { // ampersands: // - uses &__get() because declaration should be forward compatible (e.g. with Nette::Web::Html) // - doesn't call &$this->$m because user could bypass property setter by: $x = & $obj->property; $x = 'new value'; $val = $this->$m(); return $val; } else { throw new /*::*/MemberAccessException("Cannot read an undeclared property $class::\$$name."); } } /** * Sets value of a property. Do not call directly. * * @param string property name * @param mixed property value * @return void * @throws ::MemberAccessException if the property is not defined or is read-only */ public function __set($name, $value) { $class = get_class($this); if ($name === '') { throw new /*::*/MemberAccessException("Cannot assign to an class '$class' property without name."); } // property setter support if (self::hasAccessor($class, 'get' . $name)) { $m = 'set' . $name; if (self::hasAccessor($class, $m)) { $this->$m($value); } else { throw new /*::*/MemberAccessException("Cannot assign to a read-only property $class::\$$name."); } } else { throw new /*::*/MemberAccessException("Cannot assign to an undeclared property $class::\$$name."); } } /** * Is property defined? * * @param string property name * @return bool */ public function __isset($name) { return $name !== '' && self::hasAccessor(get_class($this), 'get' . $name); } /** * Access to undeclared property. * * @param string property name * @return void * @throws ::MemberAccessException */ public function __unset($name) { $class = get_class($this); throw new /*::*/MemberAccessException("Cannot unset an property $class::\$$name."); } /** * Has property an accessor? * * @param string class name * @param string method name * @return bool */ private static function hasAccessor($c, $m) { static $cache; if (!isset($cache[$c])) { // get_class_methods returns private, protected and public methods of Object (doesn't matter) // and ONLY PUBLIC methods of descendants (perfect!) // but returns static methods too (nothing doing...) // and is much faster than reflection // (works good since 5.0.4) $cache[$c] = array_flip(get_class_methods($c)); } // case-sensitive checking, capitalize the fourth character $m[3] = $m[3] & "\xDF"; return isset($cache[$c][$m]); } /** * Is property an event? * * @param string class name * @param string method name * @return bool */ private static function hasEvent($c, $m) { return preg_match('#^on[A-Z]#', $m) && property_exists($c, $m); } }