Skip to content

Commit 7143e8c

Browse files
committed
feat: Added dynamic actions and filters
1 parent 772fbf5 commit 7143e8c

File tree

5 files changed

+182
-15
lines changed

5 files changed

+182
-15
lines changed

src/Decorators/Action.php

-11
Original file line numberDiff line numberDiff line change
@@ -21,21 +21,10 @@
2121
*/
2222
#[\Attribute( \Attribute::IS_REPEATABLE | \Attribute::TARGET_METHOD )]
2323
class Action extends Filter {
24-
/**
25-
* Get the hook type.
26-
*
27-
* @return string
28-
*/
2924
protected function get_type(): string {
3025
return 'action';
3126
}
3227

33-
/**
34-
* Indirect call to the hook callback.
35-
*
36-
* @param mixed ...$args Arguments passed to the hook callback.
37-
* @return mixed Always null.
38-
*/
3928
public function invoke( mixed ...$args ): mixed {
4029
parent::invoke( ...$args );
4130

src/Decorators/Dynamic_Action.php

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
<?php //phpcs:disable Squiz.Commenting.FunctionComment.Missing
2+
/**
3+
* Dynamic_Action class file.
4+
*
5+
* @package eXtended WordPress
6+
* @subpackage Dependency Injection
7+
*/
8+
9+
namespace XWP\DI\Decorators;
10+
11+
/**
12+
* Dynamic action decorator
13+
*
14+
* @template T of object
15+
* @template H of Ajax_Handler<T>
16+
* @extends Dynamic_Filter<T,H>
17+
*/
18+
#[\Attribute( \Attribute::TARGET_METHOD )]
19+
class Dynamic_Action extends Dynamic_Filter {
20+
protected function get_type(): string {
21+
return 'action';
22+
}
23+
24+
public function invoke( mixed ...$args ): mixed {
25+
parent::invoke( ...$args );
26+
27+
return null;
28+
}
29+
}

src/Decorators/Dynamic_Filter.php

+137
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,137 @@
1+
<?php //phpcs:disable Squiz.Commenting.FunctionComment.Missing
2+
/**
3+
* Dynamic_Filter class file.
4+
*
5+
* @package eXtended WordPress
6+
* @subpackage Dependency Injection
7+
*/
8+
9+
namespace XWP\DI\Decorators;
10+
11+
use Closure;
12+
use Reflector;
13+
14+
/**
15+
* Dynamic filter decorator
16+
*
17+
* @template T of object
18+
* @template H of Ajax_Handler<T>
19+
* @extends Filter<T,H>
20+
*/
21+
#[\Attribute( \Attribute::TARGET_METHOD )]
22+
class Dynamic_Filter extends Filter {
23+
/**
24+
* Variables to fetch.
25+
*
26+
* @var array<string,string>
27+
*/
28+
protected array $vars;
29+
30+
/**
31+
* Raw variables for substitution.
32+
*
33+
* @var string|array<string>|Closure():array<string>
34+
*/
35+
protected Closure|string|array $raw_vars;
36+
37+
/**
38+
* Extra variables to pass to the callback.
39+
*
40+
* @var array<string,string>
41+
*/
42+
protected array $extra;
43+
44+
/**
45+
* Constructor.
46+
*
47+
* @param string $tag Hook tag.
48+
* @param string|array<string>|callable():array<string> $vars Variables to mix into the tag.
49+
* @param int $context Hook context.
50+
* @param Closure|string|int|array{class-string,string} $priority Hook priority.
51+
* @param int|null $args The number of arguments to pass to the callback.
52+
* @param array<int,string> $params The parameters to pass to the callback.
53+
*/
54+
public function __construct(
55+
string $tag,
56+
callable|array|string $vars,
57+
int $context = self::CTX_GLOBAL,
58+
Closure|array|int|string $priority = 10,
59+
?int $args = null,
60+
array $params = array(),
61+
) {
62+
$this->raw_vars = $vars;
63+
64+
parent::__construct(
65+
tag: $tag,
66+
priority: $priority,
67+
context: $context,
68+
invoke: self::INV_PROXIED,
69+
args: $args,
70+
params: $params,
71+
);
72+
}
73+
74+
/**
75+
* Process variables.
76+
*
77+
* @param string|array<string>|callable():array<string> $vars Variables to mix into the tag.
78+
* @return array<int,string>|array<string,string>
79+
*/
80+
private function process_vars( string|callable|array $vars ): array {
81+
if ( \is_callable( $vars ) ) {
82+
return $vars();
83+
}
84+
85+
if ( \is_string( $vars ) && $this->container->has( $vars ) ) {
86+
return $this->container->get( $vars );
87+
}
88+
89+
return $vars;
90+
}
91+
92+
/**
93+
* Parse variables.
94+
*
95+
* @param string|array<string>|callable():array<string> $vars Variables to mix into the tag.
96+
* @return array<string,string>
97+
*/
98+
protected function parse_vars( string|callable|array $vars ): array {
99+
$parsed = array();
100+
101+
foreach ( $this->process_vars( $vars ) as $key => $val ) {
102+
$key = \is_int( $key ) ? $val : $key;
103+
104+
$parsed[ $key ] = $val;
105+
}
106+
107+
return $parsed;
108+
}
109+
110+
public function with_reflector( Reflector $r ): static {
111+
$this->args ??= $r->getNumberOfParameters() - 1;
112+
113+
return parent::with_reflector( $r );
114+
}
115+
116+
public function load_hook( ?string $tag = null ): bool {
117+
$res = true;
118+
119+
foreach ( $this->parse_vars( $this->raw_vars ) as $var => $param ) {
120+
$tag = $this->define_tag( $this->tag, array( $var ) );
121+
122+
$this->extra[ $tag ] = $param;
123+
124+
$res = $res && parent::load_hook( $tag );
125+
}
126+
127+
return $res;
128+
}
129+
130+
protected function get_cb_args( array $args ): array {
131+
$args = parent::get_cb_args( $args );
132+
133+
$args[] = $this->extra[ $this->current() ];
134+
135+
return $args;
136+
}
137+
}

src/Decorators/Filter.php

+11
Original file line numberDiff line numberDiff line change
@@ -91,6 +91,17 @@ protected function get_type(): string {
9191
return 'filter';
9292
}
9393

94+
/**
95+
* Get the current hook.
96+
*
97+
* @return string
98+
*/
99+
protected function current(): string {
100+
$cb = "current_{$this->get_type()}";
101+
102+
return $cb();
103+
}
104+
94105
public function with_reflector( Reflector $r ): static {
95106
$this->args ??= $r->getNumberOfParameters();
96107

src/Interfaces/Can_Invoke.php

+5-4
Original file line numberDiff line numberDiff line change
@@ -15,11 +15,12 @@
1515
* @template THndl of Can_Handle<TInst>
1616
* @extends Can_Hook<TInst,\ReflectionMethod>
1717
*
18-
* @property-read bool $firing Is the hook firing?
19-
* @property-read int $fired Number of times the hook has fired.
20-
18+
* @property-read bool $firing Is the hook firing?
19+
* @property-read int $fired Number of times the hook has fired.
20+
* @property-read string $method The method name.
21+
*
2122
* @property-read array{TInst,string} $target The target method.
22-
* @property-read Thndl $handler The handler instance.
23+
* @property-read THndl $handler The handler instance.
2324
*/
2425
interface Can_Invoke extends Can_Hook {
2526
/**

0 commit comments

Comments
 (0)