21
21
use Symfony \Component \Console \Output \OutputInterface ;
22
22
use Symfony \Component \Console \Style \SymfonyStyle ;
23
23
use Symfony \Component \Finder \Finder ;
24
- use Symfony \UX \TwigComponent \Attribute \ExposeInTemplate ;
25
24
use Symfony \UX \TwigComponent \ComponentFactory ;
26
25
use Symfony \UX \TwigComponent \ComponentMetadata ;
27
- use Symfony \UX \TwigComponent \Twig \ PropsNode ;
26
+ use Symfony \UX \TwigComponent \ComponentPropertiesExtractor ;
28
27
use Twig \Environment ;
29
28
use Twig \Loader \FilesystemLoader ;
30
29
@@ -37,6 +36,7 @@ public function __construct(
37
36
private string $ twigTemplatesPath ,
38
37
private ComponentFactory $ componentFactory ,
39
38
private Environment $ twig ,
39
+ private readonly ComponentPropertiesExtractor $ componentPropertiesExtractor ,
40
40
private readonly array $ componentClassMap ,
41
41
?string $ anonymousDirectory = null ,
42
42
) {
@@ -212,12 +212,18 @@ private function displayComponentDetails(SymfonyStyle $io, string $name): void
212
212
['Template ' , $ metadata ->getTemplate ()],
213
213
]);
214
214
215
+ $ properties = $ this ->componentPropertiesExtractor ->getComponentProperties ($ metadata );
216
+ $ propertiesAsArrayOfStrings = array_filter (array_map (
217
+ fn (array $ property ) => $ property ['display ' ],
218
+ $ properties ,
219
+ ));
220
+
215
221
// Anonymous Component
216
222
if ($ metadata ->isAnonymous ()) {
217
223
$ table ->addRows ([
218
224
['Type ' , '<comment>Anonymous</comment> ' ],
219
225
new TableSeparator (),
220
- ['Properties ' , implode ("\n" , $ this -> getAnonymousComponentProperties ( $ metadata ) )],
226
+ ['Properties ' , implode ("\n" , $ propertiesAsArrayOfStrings )],
221
227
]);
222
228
$ table ->render ();
223
229
@@ -229,7 +235,7 @@ private function displayComponentDetails(SymfonyStyle $io, string $name): void
229
235
new TableSeparator (),
230
236
// ['Attributes Var', $metadata->get('attributes_var')],
231
237
['Public Props ' , $ metadata ->isPublicPropsExposed () ? 'Yes ' : 'No ' ],
232
- ['Properties ' , implode ("\n" , $ this -> getComponentProperties ( $ metadata ) )],
238
+ ['Properties ' , implode ("\n" , $ propertiesAsArrayOfStrings )],
233
239
]);
234
240
235
241
$ logMethod = function (\ReflectionMethod $ m ) {
@@ -280,80 +286,4 @@ private function displayComponentsTable(SymfonyStyle $io, array $components): vo
280
286
}
281
287
$ table ->render ();
282
288
}
283
-
284
- /**
285
- * @return array<string, string>
286
- */
287
- private function getComponentProperties (ComponentMetadata $ metadata ): array
288
- {
289
- $ properties = [];
290
- $ reflectionClass = new \ReflectionClass ($ metadata ->getClass ());
291
- foreach ($ reflectionClass ->getProperties () as $ property ) {
292
- $ propertyName = $ property ->getName ();
293
-
294
- if ($ metadata ->isPublicPropsExposed () && $ property ->isPublic ()) {
295
- $ type = $ property ->getType ();
296
- if ($ type instanceof \ReflectionNamedType) {
297
- $ typeName = $ type ->getName ();
298
- } else {
299
- $ typeName = (string ) $ type ;
300
- }
301
- $ value = $ property ->getDefaultValue ();
302
- $ propertyDisplay = $ typeName .' $ ' .$ propertyName .(null !== $ value ? ' = ' .json_encode ($ value ) : '' );
303
- $ properties [$ property ->name ] = $ propertyDisplay ;
304
- }
305
-
306
- foreach ($ property ->getAttributes (ExposeInTemplate::class) as $ exposeAttribute ) {
307
- /** @var ExposeInTemplate $attribute */
308
- $ attribute = $ exposeAttribute ->newInstance ();
309
- $ properties [$ property ->name ] = $ attribute ->name ?? $ property ->name ;
310
- }
311
- }
312
-
313
- return $ properties ;
314
- }
315
-
316
- /**
317
- * Extract properties from {% props %} tag in anonymous template.
318
- *
319
- * @return array<string, string>
320
- */
321
- private function getAnonymousComponentProperties (ComponentMetadata $ metadata ): array
322
- {
323
- $ source = $ this ->twig ->load ($ metadata ->getTemplate ())->getSourceContext ();
324
- $ tokenStream = $ this ->twig ->tokenize ($ source );
325
- $ moduleNode = $ this ->twig ->parse ($ tokenStream );
326
-
327
- $ propsNode = null ;
328
- foreach ($ moduleNode ->getNode ('body ' ) as $ bodyNode ) {
329
- foreach ($ bodyNode as $ node ) {
330
- if (PropsNode::class === $ node ::class) {
331
- $ propsNode = $ node ;
332
- break 2 ;
333
- }
334
- }
335
- }
336
- if (!$ propsNode instanceof PropsNode) {
337
- return [];
338
- }
339
-
340
- $ propertyNames = $ propsNode ->getAttribute ('names ' );
341
- $ properties = array_combine ($ propertyNames , $ propertyNames );
342
- foreach ($ propertyNames as $ propName ) {
343
- if ($ propsNode ->hasNode ($ propName )
344
- && ($ valueNode = $ propsNode ->getNode ($ propName ))
345
- && $ valueNode ->hasAttribute ('value ' )
346
- ) {
347
- $ value = $ valueNode ->getAttribute ('value ' );
348
- if (\is_bool ($ value )) {
349
- $ value = $ value ? 'true ' : 'false ' ;
350
- } else {
351
- $ value = json_encode ($ value );
352
- }
353
- $ properties [$ propName ] = $ propName .' = ' .$ value ;
354
- }
355
- }
356
-
357
- return $ properties ;
358
- }
359
289
}
0 commit comments