2017-08-18 22:57:27 +02:00
|
|
|
<?php declare(strict_types=1);
|
2011-04-18 19:02:30 +02:00
|
|
|
|
2014-02-06 14:44:16 +01:00
|
|
|
namespace PhpParser\Node\Stmt;
|
|
|
|
|
|
|
|
use PhpParser\Error;
|
2016-11-23 22:58:18 +01:00
|
|
|
use PhpParser\Node;
|
2014-02-06 14:44:16 +01:00
|
|
|
|
2015-01-31 22:59:38 +01:00
|
|
|
class Class_ extends ClassLike
|
2011-04-18 19:02:30 +02:00
|
|
|
{
|
|
|
|
const MODIFIER_PUBLIC = 1;
|
|
|
|
const MODIFIER_PROTECTED = 2;
|
|
|
|
const MODIFIER_PRIVATE = 4;
|
|
|
|
const MODIFIER_STATIC = 8;
|
|
|
|
const MODIFIER_ABSTRACT = 16;
|
|
|
|
const MODIFIER_FINAL = 32;
|
2021-07-21 12:43:29 +02:00
|
|
|
const MODIFIER_READONLY = 64;
|
2011-04-18 19:02:30 +02:00
|
|
|
|
2017-04-19 11:20:05 +02:00
|
|
|
const VISIBILITY_MODIFIER_MASK = 7; // 1 | 2 | 4
|
2014-11-13 20:18:49 +01:00
|
|
|
|
2015-02-28 18:44:28 +01:00
|
|
|
/** @var int Type */
|
2016-07-25 13:33:19 +02:00
|
|
|
public $flags;
|
2015-02-28 18:44:28 +01:00
|
|
|
/** @var null|Node\Name Name of extended class */
|
|
|
|
public $extends;
|
|
|
|
/** @var Node\Name[] Names of implemented interfaces */
|
|
|
|
public $implements;
|
|
|
|
|
2011-10-28 19:06:24 +02:00
|
|
|
/**
|
|
|
|
* Constructs a class node.
|
|
|
|
*
|
2017-04-28 19:09:39 +02:00
|
|
|
* @param string|Node\Identifier|null $name Name
|
2011-10-28 19:06:24 +02:00
|
|
|
* @param array $subNodes Array of the following optional subnodes:
|
2020-09-13 21:01:17 +02:00
|
|
|
* 'flags' => 0 : Flags
|
|
|
|
* 'extends' => null : Name of extended class
|
|
|
|
* 'implements' => array(): Names of implemented interfaces
|
|
|
|
* 'stmts' => array(): Statements
|
2021-04-25 21:11:36 +02:00
|
|
|
* 'attrGroups' => array(): PHP attribute groups
|
2012-04-29 23:32:09 +02:00
|
|
|
* @param array $attributes Additional attributes
|
2011-10-28 19:06:24 +02:00
|
|
|
*/
|
2017-08-13 14:06:08 +02:00
|
|
|
public function __construct($name, array $subNodes = [], array $attributes = []) {
|
2019-05-12 14:55:21 +02:00
|
|
|
$this->attributes = $attributes;
|
2017-04-24 22:32:40 +02:00
|
|
|
$this->flags = $subNodes['flags'] ?? $subNodes['type'] ?? 0;
|
2017-04-28 19:09:39 +02:00
|
|
|
$this->name = \is_string($name) ? new Node\Identifier($name) : $name;
|
2017-04-24 22:32:40 +02:00
|
|
|
$this->extends = $subNodes['extends'] ?? null;
|
2017-08-13 14:06:08 +02:00
|
|
|
$this->implements = $subNodes['implements'] ?? [];
|
|
|
|
$this->stmts = $subNodes['stmts'] ?? [];
|
2020-09-13 21:01:17 +02:00
|
|
|
$this->attrGroups = $subNodes['attrGroups'] ?? [];
|
2011-08-04 12:03:34 +02:00
|
|
|
}
|
|
|
|
|
2017-04-28 21:40:59 +02:00
|
|
|
public function getSubNodeNames() : array {
|
2020-09-13 21:01:17 +02:00
|
|
|
return ['attrGroups', 'flags', 'name', 'extends', 'implements', 'stmts'];
|
2015-02-28 18:44:28 +01:00
|
|
|
}
|
|
|
|
|
2017-01-24 08:38:55 +01:00
|
|
|
/**
|
2017-01-26 00:16:54 +01:00
|
|
|
* Whether the class is explicitly abstract.
|
|
|
|
*
|
2017-01-24 08:38:55 +01:00
|
|
|
* @return bool
|
|
|
|
*/
|
2017-04-28 21:40:59 +02:00
|
|
|
public function isAbstract() : bool {
|
2016-07-25 13:33:19 +02:00
|
|
|
return (bool) ($this->flags & self::MODIFIER_ABSTRACT);
|
2012-07-07 16:08:37 +02:00
|
|
|
}
|
|
|
|
|
2017-01-24 08:38:55 +01:00
|
|
|
/**
|
2017-01-26 00:16:54 +01:00
|
|
|
* Whether the class is final.
|
|
|
|
*
|
2017-01-24 08:38:55 +01:00
|
|
|
* @return bool
|
|
|
|
*/
|
2017-04-28 21:40:59 +02:00
|
|
|
public function isFinal() : bool {
|
2016-07-25 13:33:19 +02:00
|
|
|
return (bool) ($this->flags & self::MODIFIER_FINAL);
|
2012-07-07 16:08:37 +02:00
|
|
|
}
|
|
|
|
|
2017-01-24 08:38:55 +01:00
|
|
|
/**
|
2017-01-26 00:16:54 +01:00
|
|
|
* Whether the class is anonymous.
|
|
|
|
*
|
2017-01-24 08:38:55 +01:00
|
|
|
* @return bool
|
|
|
|
*/
|
2017-04-28 21:40:59 +02:00
|
|
|
public function isAnonymous() : bool {
|
2015-04-26 23:04:31 +02:00
|
|
|
return null === $this->name;
|
|
|
|
}
|
|
|
|
|
2014-09-30 20:23:25 +02:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2011-04-18 19:02:30 +02:00
|
|
|
public static function verifyModifier($a, $b) {
|
2017-04-19 11:20:05 +02:00
|
|
|
if ($a & self::VISIBILITY_MODIFIER_MASK && $b & self::VISIBILITY_MODIFIER_MASK) {
|
2014-02-06 14:44:16 +01:00
|
|
|
throw new Error('Multiple access type modifiers are not allowed');
|
2011-05-29 12:20:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if ($a & self::MODIFIER_ABSTRACT && $b & self::MODIFIER_ABSTRACT) {
|
2014-02-06 14:44:16 +01:00
|
|
|
throw new Error('Multiple abstract modifiers are not allowed');
|
2011-05-29 12:20:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if ($a & self::MODIFIER_STATIC && $b & self::MODIFIER_STATIC) {
|
2014-02-06 14:44:16 +01:00
|
|
|
throw new Error('Multiple static modifiers are not allowed');
|
2011-05-29 12:20:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if ($a & self::MODIFIER_FINAL && $b & self::MODIFIER_FINAL) {
|
2014-02-06 14:44:16 +01:00
|
|
|
throw new Error('Multiple final modifiers are not allowed');
|
2011-05-29 12:20:47 +02:00
|
|
|
}
|
|
|
|
|
2021-07-21 12:43:29 +02:00
|
|
|
if ($a & self::MODIFIER_READONLY && $b & self::MODIFIER_READONLY) {
|
|
|
|
throw new Error('Multiple readonly modifiers are not allowed');
|
|
|
|
}
|
|
|
|
|
2011-06-03 17:44:23 +02:00
|
|
|
if ($a & 48 && $b & 48) {
|
2014-02-12 20:23:12 +01:00
|
|
|
throw new Error('Cannot use the final modifier on an abstract class member');
|
2011-05-29 12:20:47 +02:00
|
|
|
}
|
2011-04-18 19:02:30 +02:00
|
|
|
}
|
2019-08-30 20:37:35 +02:00
|
|
|
|
2018-01-10 18:57:48 +01:00
|
|
|
public function getType() : string {
|
2017-11-12 21:25:57 +01:00
|
|
|
return 'Stmt_Class';
|
|
|
|
}
|
2014-01-23 13:33:02 +01:00
|
|
|
}
|