mirror of
https://github.com/danog/PHP-Parser.git
synced 2024-11-27 04:14:44 +01:00
a2d7e8977a
Instead of storing subnodes in a subNodes dictionary, they are now stored as simple properties. This requires declarating the properties, assigning them in the constructor, overriding the getSubNodeNames() method and passing NULL to the first argument of the NodeAbstract constructor. [Deprecated: It's still possible to use the old mode of operation for custom nodes by passing an array of subnodes to the constructor.] The only behavior difference this should cause is that getSubNodeNames() will always return the original subnode names and skip any additional properties that were dynamically added. E.g. this means that the "namespacedName" node added by the NameResolver visitor is not treated as a subnode, but as a dynamic property instead. This change improves performance and memory usage.
103 lines
3.5 KiB
PHP
103 lines
3.5 KiB
PHP
<?php
|
|
|
|
namespace PhpParser\Node\Stmt;
|
|
|
|
use PhpParser\Node;
|
|
use PhpParser\Error;
|
|
|
|
class Class_ extends ClassLike
|
|
{
|
|
const MODIFIER_PUBLIC = 1;
|
|
const MODIFIER_PROTECTED = 2;
|
|
const MODIFIER_PRIVATE = 4;
|
|
const MODIFIER_STATIC = 8;
|
|
const MODIFIER_ABSTRACT = 16;
|
|
const MODIFIER_FINAL = 32;
|
|
|
|
const VISIBILITY_MODIFER_MASK = 7; // 1 | 2 | 4
|
|
|
|
/** @var int Type */
|
|
public $type;
|
|
/** @var null|Node\Name Name of extended class */
|
|
public $extends;
|
|
/** @var Node\Name[] Names of implemented interfaces */
|
|
public $implements;
|
|
|
|
protected static $specialNames = array(
|
|
'self' => true,
|
|
'parent' => true,
|
|
'static' => true,
|
|
);
|
|
|
|
/**
|
|
* Constructs a class node.
|
|
*
|
|
* @param string $name Name
|
|
* @param array $subNodes Array of the following optional subnodes:
|
|
* 'type' => 0 : Type
|
|
* 'extends' => null : Name of extended class
|
|
* 'implements' => array(): Names of implemented interfaces
|
|
* 'stmts' => array(): Statements
|
|
* @param array $attributes Additional attributes
|
|
*/
|
|
public function __construct($name, array $subNodes = array(), array $attributes = array()) {
|
|
parent::__construct(null, $attributes);
|
|
$this->type = isset($subNodes['type']) ? $subNodes['type'] : 0;
|
|
$this->name = $name;
|
|
$this->extends = isset($subNodes['extends']) ? $subNodes['extends'] : null;
|
|
$this->implements = isset($subNodes['implements']) ? $subNodes['implements'] : array();
|
|
$this->stmts = isset($subNodes['stmts']) ? $subNodes['stmts'] : array();
|
|
|
|
if (isset(self::$specialNames[(string) $this->name])) {
|
|
throw new Error(sprintf('Cannot use \'%s\' as class name as it is reserved', $this->name));
|
|
}
|
|
|
|
if (isset(self::$specialNames[(string) $this->extends])) {
|
|
throw new Error(sprintf('Cannot use \'%s\' as class name as it is reserved', $this->extends));
|
|
}
|
|
|
|
foreach ($this->implements as $interface) {
|
|
if (isset(self::$specialNames[(string) $interface])) {
|
|
throw new Error(sprintf('Cannot use \'%s\' as interface name as it is reserved', $interface));
|
|
}
|
|
}
|
|
}
|
|
|
|
public function getSubNodeNames() {
|
|
return array('type', 'name', 'extends', 'implements', 'stmts');
|
|
}
|
|
|
|
public function isAbstract() {
|
|
return (bool) ($this->type & self::MODIFIER_ABSTRACT);
|
|
}
|
|
|
|
public function isFinal() {
|
|
return (bool) ($this->type & self::MODIFIER_FINAL);
|
|
}
|
|
|
|
/**
|
|
* @internal
|
|
*/
|
|
public static function verifyModifier($a, $b) {
|
|
if ($a & self::VISIBILITY_MODIFER_MASK && $b & self::VISIBILITY_MODIFER_MASK) {
|
|
throw new Error('Multiple access type modifiers are not allowed');
|
|
}
|
|
|
|
if ($a & self::MODIFIER_ABSTRACT && $b & self::MODIFIER_ABSTRACT) {
|
|
throw new Error('Multiple abstract modifiers are not allowed');
|
|
}
|
|
|
|
if ($a & self::MODIFIER_STATIC && $b & self::MODIFIER_STATIC) {
|
|
throw new Error('Multiple static modifiers are not allowed');
|
|
}
|
|
|
|
if ($a & self::MODIFIER_FINAL && $b & self::MODIFIER_FINAL) {
|
|
throw new Error('Multiple final modifiers are not allowed');
|
|
}
|
|
|
|
if ($a & 48 && $b & 48) {
|
|
throw new Error('Cannot use the final modifier on an abstract class member');
|
|
}
|
|
}
|
|
}
|