-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathUserInputPrompts.php
174 lines (147 loc) Β· 4.88 KB
/
UserInputPrompts.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
<?php
declare(strict_types=1);
namespace Syntatis\Codex\Companion\Console\ProjectInitCommand\Howdy;
use Symfony\Component\Console\Style\StyleInterface;
use Syntatis\Codex\Companion\Contracts\Executable;
use Syntatis\Codex\Companion\Exceptions\MissingRequiredInfo;
use Syntatis\Codex\Companion\Helpers\Strings\PHPNamespace;
use Syntatis\Codex\Companion\Helpers\Strings\PHPVendorPrefix;
use Syntatis\Codex\Companion\Helpers\Strings\WPPluginDescription;
use Syntatis\Codex\Companion\Helpers\Strings\WPPluginName;
use Syntatis\Codex\Companion\Helpers\Strings\WPPluginSlug;
use Syntatis\Codex\Companion\Projects\Howdy\InitializeFiles;
use Syntatis\Utils\Str;
use Syntatis\Utils\Val;
use function array_filter;
use function array_keys;
use function count;
use function in_array;
use function is_string;
use const ARRAY_FILTER_USE_BOTH;
/** @phpstan-import-type ValidatedItems from InitializeFiles */
class UserInputPrompts implements Executable
{
private StyleInterface $style;
/** @phpstan-var ValidatedItems */
private array $inputs;
/** @phpstan-var ValidatedItems */
private array $props;
/**
* @param array<string,string|null> $props The list of properties from the "howdy" project.
* @param StyleInterface $style Output for the console to interact with the user.
*/
public function __construct(array $props, StyleInterface $style)
{
$missing = [];
if (! $this->evelProjectProps($props, $missing)) {
throw new MissingRequiredInfo(...$missing);
}
$this->props = $props;
$this->style = $style;
}
/**
* @param array<string,string|null> $props
* @param array<string,string|null> $missing
*
* @phpstan-assert-if-true ValidatedItems $props
*/
private function evelProjectProps(array $props, array &$missing): bool
{
$required = [
'php_vendor_prefix',
'php_namespace',
'wp_plugin_name',
'wp_plugin_slug',
];
$missing = array_keys(
array_filter(
[
'php_vendor_prefix' => $props['php_vendor_prefix'] ?? null,
'php_namespace' => $props['php_namespace'] ?? null,
'wp_plugin_name' => $props['wp_plugin_name'] ?? null,
'wp_plugin_slug' => $props['wp_plugin_slug'] ?? null,
],
static fn ($val, $key) => in_array($key, $required, true) && (! is_string($val) || Val::isBlank($val)),
ARRAY_FILTER_USE_BOTH,
),
);
return count($missing) <= 0;
}
public function execute(): int
{
$this->style->text([
'To get started with your new WordPress plugin project, please provide the',
'Plugin slug. The Plugin slug should be in all-lowercase and use hyphens',
'to separate words e.g. <comment>acme-awesome-plugin</comment>.',
]);
/** @var WPPluginSlug $wpPluginSlug */
$wpPluginSlug = $this->style->ask(
'Plugin slug',
null,
static fn ($value) => new WPPluginSlug(is_string($value) ? $value : ''),
);
/** @var WPPluginName $wpPluginName */
$wpPluginName = $this->style->ask(
'Plugin name',
Str::toTitleCase((string) $wpPluginSlug),
static fn ($value) => new WPPluginName(is_string($value) ? $value : ''),
);
/** @var PHPNamespace $phpNamespace */
$phpNamespace = $this->style->ask(
'PHP namespace',
Str::toPascalCase((string) $wpPluginSlug),
static fn ($value) => new PHPNamespace(is_string($value) ? $value : ''),
);
/** @var PHPVendorPrefix $vendorPrefix */
$vendorPrefix = $this->style->ask(
'PHP vendor prefix',
Str::toPascalCase((string) $wpPluginSlug) . '\\Vendor',
static fn ($value) => new PHPVendorPrefix(is_string($value) ? $value : ''),
);
/**
* Each of the values below are already validated when the users input their values.
* Users are not allowed to provide empty values. Otherwise, the class will throw
* them an error exception, and ask them to provide a valid value.
*/
$inputs = [
'php_vendor_prefix' => (string) $vendorPrefix,
'php_namespace' => (string) $phpNamespace,
'wp_plugin_name' => (string) $wpPluginName,
'wp_plugin_slug' => (string) $wpPluginSlug,
];
$this->inputs = $this->promptOptionals($inputs);
return 0;
}
/** @phpstan-return ValidatedItems */
public function getInputs(): array
{
return $this->inputs;
}
/** @phpstan-return ValidatedItems */
public function getProjectProps(): array
{
return $this->props;
}
/**
* @phpstan-param ValidatedItems $inputs
*
* @phpstan-return ValidatedItems
*/
private function promptOptionals(array $inputs): array
{
// Description is an optional prop.
if (isset($this->props['wp_plugin_description']) && ! Val::isBlank($this->props['wp_plugin_description'])) {
/** @var WPPluginDescription $wpPluginDesc */
$wpPluginDesc = $this->style->ask(
'Plugin description',
$this->props['wp_plugin_description'],
static fn ($value) => new WPPluginDescription(is_string($value) ? $value : ''),
);
$wpPluginDesc = (string) $wpPluginDesc;
if (! Val::isBlank($wpPluginDesc)) {
$inputs['wp_plugin_description'] = $wpPluginDesc;
}
}
return $inputs;
}
}