1
0
mirror of https://github.com/dg/dibi.git synced 2025-10-25 11:46:35 +02:00
Files
php-dibi/dibi/libs/NObject.php
2007-11-18 02:29:11 +00:00

243 lines
6.8 KiB
PHP

<?php
/**
* dibi - tiny'n'smart database abstraction layer
* ----------------------------------------------
*
* Copyright (c) 2005, 2007 David Grudl aka -dgx- (http://www.dgx.cz)
*
* This source file is subject to the "dibi license" that is bundled
* with this package in the file license.txt.
*
* For more information please see http://php7.org/dibi/
*
* @copyright Copyright (c) 2004, 2007 David Grudl
* @license http://php7.org/nette/license Nette license
* @link http://php7.org/nette/
* @package Nette
*/
/**
* NObject is the ultimate ancestor of all instantiable classes.
*
* It defines some handful methods and enhances object core of PHP:
* - access to undeclared members throws exceptions
* - ability to add new methods to class (extension methods)
* - support for conventional properties with getters and setters
*
* Properties is a syntactic sugar which allows access public getter and setter
* methods as normal object variables. A property is defined by a getter method
* and optional setter method (no setter method means read-only property).
* <code>
* $val = $obj->Label; // equivalent to $val = $obj->getLabel();
* $obj->Label = 'Nette'; // equivalent to $obj->setLabel('Nette');
* </code>
* Property names are case-sensitive, and they are written in the PascalCaps,
* in contrast to camelCaps used by normal members.
*
* Adding method to class (i.e. to all instances) works similar to JavaScript
* prototype property. The syntax for adding a new method is:
* <code>
* function MyClass_prototype_newMethod(MyClass $obj, $arg, ...) { ... }
* $obj = new MyClass;
* $obj->newMethod($x); // equivalent to MyClass_prototype_newMethod($obj, $x);
* </code>
*
* @author David Grudl
* @copyright Copyright (c) 2004, 2007 David Grudl
* @license http://php7.org/nette/license Nette license
* @link http://php7.org/nette/
* @package Nette
*/
abstract class NObject
{
/**
* 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 BadMethodCallException
*/
protected function __call($name, $args)
{
if ($name === '') {
throw new BadMethodCallException("Call to method without name");
}
// object prototypes support Class__method()
// (or use class Class__method { static function ... } with autoloading?)
$cl = $class = get_class($this);
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 BadMethodCallException("Call to undefined method $class::$name()");
}
/**
* Returns property value. Do not call directly.
*
* @param string property name
* @return mixed property value or the event handler list
* @throws LogicException if the property is not defined.
*/
protected function &__get($name)
{
if ($name === '') {
throw new LogicException("Cannot read an property without name");
}
// property getter support
$class = get_class($this);
$m = 'get' . $name;
if (self::isCallable($class, $m)) {
// ampersands:
// - using &__get() because declaration should be forward compatible (e.g. with NHtml)
// - not using &$this->$m because user could bypass property setter by: $x = & $obj->property; $x = 'new value';
$val = $this->$m();
return $val;
} else {
throw new LogicException("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 LogicException if the property is not defined or is read-only
*/
protected function __set($name, $value)
{
if ($name === '') {
throw new LogicException('Cannot assign to an property without name');
}
// property setter support
$class = get_class($this);
if (self::isCallable($class, 'get' . $name)) {
$m = 'set' . $name;
if (self::isCallable($class, $m)) {
$this->$m($value);
} else {
throw new LogicException("Cannot assign to a read-only property $class::\$$name");
}
} else {
throw new LogicException("Cannot assign to an undeclared property $class::\$$name");
}
}
/**
* Is property defined?
*
* @param string property name
* @return bool
*/
protected function __isset($name)
{
return $name !== '' && self::isCallable(get_class($this), 'get' . $name);
}
/**
* Access to undeclared property
*
* @param string property name
* @return void
* @throws LogicException
*/
protected function __unset($name)
{
$class = get_class($this);
throw new LogicException("Cannot unset an property $class::\$$name");
}
/**
* Does public method exist? (case sensitive)
*
* @param string class name
* @param string method name
* @return bool
*/
private static function isCallable($c, $m)
{
static $cache;
if (!isset($cache[$c])) {
// get_class_methods returns private, protected and public methods of NObject (doesn't matter)
// only 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));
}
return isset($cache[$c][$m]);
}
}
/**
* NClass is the ultimate ancestor of all uninstantiable classes.
*
* @author David Grudl
* @copyright Copyright (c) 2004, 2007 David Grudl
* @license http://php7.org/nette/license Nette license
* @link http://php7.org/nette/
* @package Nette
*/
abstract class NClass
{
final public function __construct()
{
throw new LogicException("Cannot instantiate static class " . get_class($this));
}
}