17
17
use phpDocumentor \Reflection \DocBlock \Tags \Author ;
18
18
use phpDocumentor \Reflection \DocBlock \Tags \Covers ;
19
19
use phpDocumentor \Reflection \DocBlock \Tags \Deprecated ;
20
+ use phpDocumentor \Reflection \DocBlock \Tags \Factory \Factory ;
20
21
use phpDocumentor \Reflection \DocBlock \Tags \Generic ;
21
22
use phpDocumentor \Reflection \DocBlock \Tags \InvalidTag ;
22
23
use phpDocumentor \Reflection \DocBlock \Tags \Link as LinkTag ;
40
41
use ReflectionParameter ;
41
42
use Webmozart \Assert \Assert ;
42
43
44
+ use function array_key_exists ;
43
45
use function array_merge ;
44
46
use function array_slice ;
45
47
use function call_user_func_array ;
46
48
use function count ;
47
49
use function get_class ;
50
+ use function is_object ;
48
51
use function preg_match ;
52
+ use function sprintf ;
49
53
use function strpos ;
50
54
use function trim ;
51
55
@@ -72,7 +76,7 @@ final class StandardTagFactory implements TagFactory
72
76
public const REGEX_TAGNAME = '[\w\-\_ \\\\:]+ ' ;
73
77
74
78
/**
75
- * @var array<class-string<Tag>> An array with a tag as a key, and an
79
+ * @var array<class-string<Tag>|Factory > An array with a tag as a key, and an
76
80
* FQCN to a class that handles it as an array value.
77
81
*/
78
82
private $ tagHandlerMappings = [
@@ -162,18 +166,25 @@ public function addService(object $service, ?string $alias = null): void
162
166
$ this ->serviceLocator [$ alias ?: get_class ($ service )] = $ service ;
163
167
}
164
168
165
- public function registerTagHandler (string $ tagName , string $ handler ): void
169
+ /** {@inheritDoc} */
170
+ public function registerTagHandler (string $ tagName , $ handler ): void
166
171
{
167
172
Assert::stringNotEmpty ($ tagName );
168
- Assert::classExists ($ handler );
169
- Assert::implementsInterface ($ handler , Tag::class);
170
-
171
173
if (strpos ($ tagName , '\\' ) && $ tagName [0 ] !== '\\' ) {
172
174
throw new InvalidArgumentException (
173
175
'A namespaced tag must have a leading backslash as it must be fully qualified '
174
176
);
175
177
}
176
178
179
+ if (is_object ($ handler )) {
180
+ Assert::isInstanceOf ($ handler , Factory::class);
181
+ $ this ->tagHandlerMappings [$ tagName ] = $ handler ;
182
+
183
+ return ;
184
+ }
185
+
186
+ Assert::classExists ($ handler );
187
+ Assert::implementsInterface ($ handler , Tag::class);
177
188
$ this ->tagHandlerMappings [$ tagName ] = $ handler ;
178
189
}
179
190
@@ -210,6 +221,10 @@ private function createTag(string $body, string $name, TypeContext $context): Ta
210
221
$ this ->getServiceLocatorWithDynamicParameters ($ context , $ name , $ body )
211
222
);
212
223
224
+ if (array_key_exists ('tagLine ' , $ arguments )) {
225
+ $ arguments ['tagLine ' ] = sprintf ('@%s %s ' , $ name , $ body );
226
+ }
227
+
213
228
try {
214
229
$ callable = [$ handlerClassName , 'create ' ];
215
230
Assert::isCallable ($ callable );
@@ -225,9 +240,9 @@ private function createTag(string $body, string $name, TypeContext $context): Ta
225
240
/**
226
241
* Determines the Fully Qualified Class Name of the Factory or Tag (containing a Factory Method `create`).
227
242
*
228
- * @return class-string<Tag>
243
+ * @return class-string<Tag>|Factory
229
244
*/
230
- private function findHandlerClassName (string $ tagName , TypeContext $ context ): string
245
+ private function findHandlerClassName (string $ tagName , TypeContext $ context )
231
246
{
232
247
$ handlerClassName = Generic::class;
233
248
if (isset ($ this ->tagHandlerMappings [$ tagName ])) {
@@ -268,18 +283,18 @@ private function getArgumentsForParametersFromWiring(array $parameters, array $l
268
283
}
269
284
}
270
285
286
+ $ parameterName = $ parameter ->getName ();
271
287
if (isset ($ locator [$ typeHint ])) {
272
- $ arguments [] = $ locator [$ typeHint ];
288
+ $ arguments [$ parameterName ] = $ locator [$ typeHint ];
273
289
continue ;
274
290
}
275
291
276
- $ parameterName = $ parameter ->getName ();
277
292
if (isset ($ locator [$ parameterName ])) {
278
- $ arguments [] = $ locator [$ parameterName ];
293
+ $ arguments [$ parameterName ] = $ locator [$ parameterName ];
279
294
continue ;
280
295
}
281
296
282
- $ arguments [] = null ;
297
+ $ arguments [$ parameterName ] = null ;
283
298
}
284
299
285
300
return $ arguments ;
@@ -289,12 +304,14 @@ private function getArgumentsForParametersFromWiring(array $parameters, array $l
289
304
* Retrieves a series of ReflectionParameter objects for the static 'create' method of the given
290
305
* tag handler class name.
291
306
*
292
- * @param class-string $handlerClassName
307
+ * @param class-string|Factory $handler
293
308
*
294
309
* @return ReflectionParameter[]
295
310
*/
296
- private function fetchParametersForHandlerFactoryMethod (string $ handlerClassName ): array
311
+ private function fetchParametersForHandlerFactoryMethod ($ handler ): array
297
312
{
313
+ $ handlerClassName = is_object ($ handler ) ? get_class ($ handler ) : $ handler ;
314
+
298
315
if (!isset ($ this ->tagHandlerParameterCache [$ handlerClassName ])) {
299
316
$ methodReflection = new ReflectionMethod ($ handlerClassName , 'create ' );
300
317
$ this ->tagHandlerParameterCache [$ handlerClassName ] = $ methodReflection ->getParameters ();
0 commit comments