Skip to content

Commit b774cee

Browse files
committed
feat: Added Infuse decorator
1 parent 058b161 commit b774cee

File tree

4 files changed

+104
-27
lines changed

4 files changed

+104
-27
lines changed

src/Decorators/Filter.php

+26-6
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,13 @@
2525
*/
2626
#[\Attribute( \Attribute::IS_REPEATABLE | \Attribute::TARGET_METHOD )]
2727
class Filter extends Hook implements Can_Invoke {
28+
/**
29+
* Array of lazy loaded handlers.
30+
*
31+
* @var array<string,bool>
32+
*/
33+
public static array $lazy = array();
34+
2835
/**
2936
* The handler.
3037
*
@@ -143,7 +150,10 @@ protected function init_handler( string $strategy ): bool {
143150
return $this->can_load();
144151
}
145152

146-
\do_action( "{$this->handler->id}_{$strategy}_init" );
153+
if ( ! isset( static::$lazy[ $this->handler->id ] ) ) {
154+
static::$lazy[ $this->handler->id ] = true;
155+
\do_action( "{$this->handler->id}_{$strategy}_init", $this->handler );
156+
}
147157

148158
return $this->handler->loaded;
149159
}
@@ -236,17 +246,27 @@ protected function fire_hook( mixed ...$args ): mixed {
236246
protected function get_cb_args( array $args ): array {
237247
if ( $this->params ) {
238248
foreach ( $this->params as $param ) {
239-
$args[] = ! \str_starts_with( $param, '!value:' )
240-
? $this->container->get( $param )
241-
: \str_replace( '!value:', '', $param );
249+
$args[] = $this->get_cb_arg( $param );
242250
}
243-
244-
$args[] = $this;
245251
}
246252

247253
return $args;
248254
}
249255

256+
protected function get_cb_arg( string $param ): mixed {
257+
if ( '!self.hook' === $param ) {
258+
return $this;
259+
}
260+
261+
if ( '!self.handler' === $param ) {
262+
return $this->handler;
263+
}
264+
265+
return ! \str_starts_with( $param, '!value:' )
266+
? $this->container->get( $param )
267+
: \str_replace( '!value:', '', $param );
268+
}
269+
250270
/**
251271
* Handle an exception.
252272
*

src/Decorators/Handler.php

+24-20
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,6 @@
99
namespace XWP\DI\Decorators;
1010

1111
use Closure;
12-
use DI\Attribute\Inject;
1312
use DI\Container;
1413
use ReflectionClass;
1514
use XWP\DI\Interfaces\Can_Handle;
@@ -28,6 +27,13 @@
2827
*/
2928
#[\Attribute( \Attribute::TARGET_CLASS )]
3029
class Handler extends Hook implements Can_Handle {
30+
/**
31+
* Array of handlers that have been initialized.
32+
*
33+
* @var array<string,bool>
34+
*/
35+
protected static array $did_init = array();
36+
3137
/**
3238
* Handler classname.
3339
*
@@ -146,8 +152,9 @@ public function load(): bool {
146152
}
147153

148154
$this->instance ??= $this->initialize();
155+
$this->loaded = $this->instance::class === $this->classname;
149156

150-
return $this->on_initialize();
157+
return $this->loaded && $this->on_initialize();
151158
}
152159

153160
/**
@@ -165,35 +172,32 @@ protected function initialize(): object {
165172
* @return bool
166173
*/
167174
protected function on_initialize(): bool {
168-
$this->loaded = true;
169-
170-
$method = 'on_initialize';
171-
172-
if ( \method_exists( $this->classname, $method ) ) {
175+
if ( ! \method_exists( $this->instance, 'on_initialize' ) ) {
176+
return static::$did_init[ $this->id ] ??= true;
177+
}
173178

174-
$this->container->call(
175-
array( $this->classname, $method ),
176-
$this->resolve_params( $method ),
177-
);
179+
if ( isset( static::$did_init[ $this->id ] ) ) {
180+
return true;
178181
}
179182

180-
return $this->loaded;
183+
$this->container->call(
184+
array( $this->instance, 'on_initialize' ),
185+
$this->resolve_params( 'on_initialize' ),
186+
);
187+
188+
return static::$did_init[ $this->id ] ??= true;
181189
}
182190

183191
/**
184192
* Resolve the parameters for a method.
185193
*
186194
* @param string $method Method name.
187-
* @return array<string,mixed>
195+
* @return array<mixed>
188196
*/
189197
protected function resolve_params( string $method ): array {
190-
return \array_map(
191-
'\DI\get',
192-
Reflection::get_decorator(
193-
$this->reflector->getMethod( $method ),
194-
Inject::class,
195-
)?->getParameters() ?? array(),
196-
);
198+
$injector = Reflection::get_decorator( $this->reflector->getMethod( $method ), Infuse::class );
199+
200+
return $injector ? $injector->resolve( $this ) : array();
197201
}
198202

199203
/**

src/Decorators/Infuse.php

+53
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,53 @@
1+
<?php
2+
/**
3+
* Infuse decorator class file.
4+
*
5+
* @package eXtended WordPress
6+
* @subpackage Dependency Injection
7+
*/
8+
9+
namespace XWP\DI\Decorators;
10+
11+
use XWP\DI\Interfaces\Can_Handle;
12+
13+
/**
14+
* Infuse decorator.
15+
*/
16+
#[\Attribute( \Attribute::TARGET_METHOD )]
17+
class Infuse {
18+
/**
19+
* The parameters to inject.
20+
*
21+
* @var array<string>
22+
*/
23+
protected array $params;
24+
25+
/**
26+
* Constructor.
27+
*
28+
* @param string|array<string> ...$params The parameters to inject.
29+
*/
30+
public function __construct( string|array ...$params, ) {
31+
$this->params = \is_array( $params[0] ) ? $params[0] : $params;
32+
}
33+
34+
/**
35+
* Resolve the parameters.
36+
*
37+
* @template T of object
38+
* @param Can_Handle<T> $h The handler.
39+
* @return array<mixed>
40+
*/
41+
public function resolve( Can_Handle $h ) {
42+
$params = \array_diff( $this->params, array( '!self.handler' ) );
43+
$hook_it = $params !== $this->params;
44+
45+
$params = \array_map( '\DI\get', $params );
46+
47+
if ( $hook_it ) {
48+
$params[] = \DI\value( $h );
49+
}
50+
51+
return $params;
52+
}
53+
}

src/Invoker.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -280,7 +280,7 @@ protected function queue_methods( Can_Handle $handler ): static {
280280
function () use ( $handler ) {
281281
$this->init_handler( $handler );
282282
},
283-
0,
283+
-1,
284284
0,
285285
);
286286
}

0 commit comments

Comments
 (0)