123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354 |
- Custom Annotation Classes
- =========================
- If you want to define your own annotations, you just have to group them
- in a namespace and register this namespace in the ``AnnotationRegistry``.
- Annotation classes have to contain a class-level docblock with the text
- ``@Annotation``:
- .. code-block:: php
- namespace MyCompany\Annotations;
- /** @Annotation */
- class Bar
- {
- // some code
- }
- Inject annotation values
- ------------------------
- The annotation parser checks if the annotation constructor has arguments,
- if so then it will pass the value array, otherwise it will try to inject
- values into public properties directly:
- .. code-block:: php
- namespace MyCompany\Annotations;
- /**
- * @Annotation
- *
- * Some Annotation using a constructor
- */
- class Bar
- {
- private $foo;
- public function __construct(array $values)
- {
- $this->foo = $values['foo'];
- }
- }
- /**
- * @Annotation
- *
- * Some Annotation without a constructor
- */
- class Foo
- {
- public $bar;
- }
- Annotation Target
- -----------------
- ``@Target`` indicates the kinds of class elements to which an annotation
- type is applicable. Then you could define one or more targets:
- - ``CLASS`` Allowed in class docblocks
- - ``PROPERTY`` Allowed in property docblocks
- - ``METHOD`` Allowed in the method docblocks
- - ``ALL`` Allowed in class, property and method docblocks
- - ``ANNOTATION`` Allowed inside other annotations
- If the annotations is not allowed in the current context, an
- ``AnnotationException`` is thrown.
- .. code-block:: php
- namespace MyCompany\Annotations;
- /**
- * @Annotation
- * @Target({"METHOD","PROPERTY"})
- */
- class Bar
- {
- // some code
- }
- /**
- * @Annotation
- * @Target("CLASS")
- */
- class Foo
- {
- // some code
- }
- Attribute types
- ---------------
- The annotation parser checks the given parameters using the phpdoc
- annotation ``@var``, The data type could be validated using the ``@var``
- annotation on the annotation properties or using the ``@Attributes`` and
- ``@Attribute`` annotations.
- If the data type does not match you get an ``AnnotationException``
- .. code-block:: php
- namespace MyCompany\Annotations;
- /**
- * @Annotation
- * @Target({"METHOD","PROPERTY"})
- */
- class Bar
- {
- /** @var mixed */
- public $mixed;
- /** @var boolean */
- public $boolean;
- /** @var bool */
- public $bool;
- /** @var float */
- public $float;
- /** @var string */
- public $string;
- /** @var integer */
- public $integer;
- /** @var array */
- public $array;
- /** @var SomeAnnotationClass */
- public $annotation;
- /** @var array<integer> */
- public $arrayOfIntegers;
- /** @var array<SomeAnnotationClass> */
- public $arrayOfAnnotations;
- }
- /**
- * @Annotation
- * @Target({"METHOD","PROPERTY"})
- * @Attributes({
- * @Attribute("stringProperty", type = "string"),
- * @Attribute("annotProperty", type = "SomeAnnotationClass"),
- * })
- */
- class Foo
- {
- public function __construct(array $values)
- {
- $this->stringProperty = $values['stringProperty'];
- $this->annotProperty = $values['annotProperty'];
- }
- // some code
- }
- Annotation Required
- -------------------
- ``@Required`` indicates that the field must be specified when the
- annotation is used. If it is not used you get an ``AnnotationException``
- stating that this value can not be null.
- Declaring a required field:
- .. code-block:: php
- /**
- * @Annotation
- * @Target("ALL")
- */
- class Foo
- {
- /** @Required */
- public $requiredField;
- }
- Usage:
- .. code-block:: php
- /** @Foo(requiredField="value") */
- public $direction; // Valid
- /** @Foo */
- public $direction; // Required field missing, throws an AnnotationException
- Enumerated values
- -----------------
- - An annotation property marked with ``@Enum`` is a field that accepts a
- fixed set of scalar values.
- - You should use ``@Enum`` fields any time you need to represent fixed
- values.
- - The annotation parser checks the given value and throws an
- ``AnnotationException`` if the value does not match.
- Declaring an enumerated property:
- .. code-block:: php
- /**
- * @Annotation
- * @Target("ALL")
- */
- class Direction
- {
- /**
- * @Enum({"NORTH", "SOUTH", "EAST", "WEST"})
- */
- public $value;
- }
- Annotation usage:
- .. code-block:: php
- /** @Direction("NORTH") */
- public $direction; // Valid value
- /** @Direction("NORTHEAST") */
- public $direction; // Invalid value, throws an AnnotationException
- Constants
- ---------
- The use of constants and class constants is available on the annotations
- parser.
- The following usages are allowed:
- .. code-block:: php
- namespace MyCompany\Entity;
- use MyCompany\Annotations\Foo;
- use MyCompany\Annotations\Bar;
- use MyCompany\Entity\SomeClass;
- /**
- * @Foo(PHP_EOL)
- * @Bar(Bar::FOO)
- * @Foo({SomeClass::FOO, SomeClass::BAR})
- * @Bar({SomeClass::FOO_KEY = SomeClass::BAR_VALUE})
- */
- class User
- {
- }
- Be careful with constants and the cache !
- .. note::
- The cached reader will not re-evaluate each time an annotation is
- loaded from cache. When a constant is changed the cache must be
- cleaned.
- Usage
- -----
- Using the library API is simple. Using the annotations described in the
- previous section, you can now annotate other classes with your
- annotations:
- .. code-block:: php
- namespace MyCompany\Entity;
- use MyCompany\Annotations\Foo;
- use MyCompany\Annotations\Bar;
- /**
- * @Foo(bar="foo")
- * @Bar(foo="bar")
- */
- class User
- {
- }
- Now we can write a script to get the annotations above:
- .. code-block:: php
- $reflClass = new ReflectionClass('MyCompany\Entity\User');
- $classAnnotations = $reader->getClassAnnotations($reflClass);
- foreach ($classAnnotations AS $annot) {
- if ($annot instanceof \MyCompany\Annotations\Foo) {
- echo $annot->bar; // prints "foo";
- } else if ($annot instanceof \MyCompany\Annotations\Bar) {
- echo $annot->foo; // prints "bar";
- }
- }
- You have a complete API for retrieving annotation class instances from a
- class, property or method docblock:
- Reader API
- ~~~~~~~~~~
- Access all annotations of a class
- ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
- .. code-block:: php
- public function getClassAnnotations(\ReflectionClass $class);
- Access one annotation of a class
- ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
- .. code-block:: php
- public function getClassAnnotation(\ReflectionClass $class, $annotationName);
- Access all annotations of a method
- ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
- .. code-block:: php
- public function getMethodAnnotations(\ReflectionMethod $method);
- Access one annotation of a method
- ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
- .. code-block:: php
- public function getMethodAnnotation(\ReflectionMethod $method, $annotationName);
- Access all annotations of a property
- ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
- .. code-block:: php
- public function getPropertyAnnotations(\ReflectionProperty $property);
- Access one annotation of a property
- ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
- .. code-block:: php
- public function getPropertyAnnotation(\ReflectionProperty $property, $annotationName);
|