Chris@0
|
1 <?php
|
Chris@0
|
2
|
Chris@0
|
3 namespace DrupalCodeGenerator\Helper;
|
Chris@0
|
4
|
Chris@0
|
5 use DrupalCodeGenerator\Utils;
|
Chris@0
|
6 use Symfony\Component\Console\Exception\InvalidOptionException;
|
Chris@0
|
7 use Symfony\Component\Console\Helper\Helper;
|
Chris@0
|
8 use Symfony\Component\Console\Input\InputInterface;
|
Chris@0
|
9 use Symfony\Component\Console\Output\OutputInterface;
|
Chris@0
|
10 use Symfony\Component\Console\Question\ChoiceQuestion;
|
Chris@0
|
11 use Symfony\Component\Console\Question\ConfirmationQuestion;
|
Chris@0
|
12 use Symfony\Component\Console\Question\Question;
|
Chris@0
|
13
|
Chris@0
|
14 /**
|
Chris@0
|
15 * Generator input handler.
|
Chris@0
|
16 */
|
Chris@0
|
17 class InputHandler extends Helper {
|
Chris@0
|
18
|
Chris@0
|
19 use QuestionSettersTrait;
|
Chris@0
|
20
|
Chris@0
|
21 /**
|
Chris@0
|
22 * {@inheritdoc}
|
Chris@0
|
23 */
|
Chris@0
|
24 public function getName() {
|
Chris@0
|
25 return 'dcg_input_handler';
|
Chris@0
|
26 }
|
Chris@0
|
27
|
Chris@0
|
28 /**
|
Chris@0
|
29 * Interacts with the user and returns variables for templates.
|
Chris@0
|
30 *
|
Chris@0
|
31 * @param \Symfony\Component\Console\Input\InputInterface $input
|
Chris@0
|
32 * Input instance.
|
Chris@0
|
33 * @param \Symfony\Component\Console\Output\OutputInterface $output
|
Chris@0
|
34 * Output instance.
|
Chris@0
|
35 * @param \Symfony\Component\Console\Question\Question[] $questions
|
Chris@0
|
36 * List of questions that the user should answer.
|
Chris@0
|
37 * @param array $vars
|
Chris@0
|
38 * Array of predefined template variables.
|
Chris@0
|
39 *
|
Chris@0
|
40 * @return array
|
Chris@0
|
41 * Template variables.
|
Chris@0
|
42 */
|
Chris@0
|
43 public function collectVars(InputInterface $input, OutputInterface $output, array $questions, array $vars = []) {
|
Chris@0
|
44
|
Chris@0
|
45 // A user can pass answers through the command line option.
|
Chris@0
|
46 $answers = NULL;
|
Chris@0
|
47 if ($answers_raw = $input->getOption('answers')) {
|
Chris@0
|
48 $answers = json_decode($answers_raw, TRUE);
|
Chris@0
|
49 if (!is_array($answers)) {
|
Chris@0
|
50 throw new InvalidOptionException('Answers should be encoded in JSON format.');
|
Chris@0
|
51 }
|
Chris@0
|
52 }
|
Chris@0
|
53
|
Chris@0
|
54 /** @var \Symfony\Component\Console\Helper\QuestionHelper $question_helper */
|
Chris@0
|
55 $question_helper = $this->getHelperSet()->get('question');
|
Chris@0
|
56
|
Chris@0
|
57 /** @var \DrupalCodeGenerator\Command\GeneratorInterface $command */
|
Chris@0
|
58 $command = $this->getHelperSet()->getCommand();
|
Chris@0
|
59 $directory = $command->getDirectory();
|
Chris@0
|
60
|
Chris@0
|
61 foreach ($questions as $name => $question) {
|
Chris@0
|
62 /** @var \Symfony\Component\Console\Question\Question $question */
|
Chris@0
|
63 $default_value = $question->getDefault();
|
Chris@0
|
64
|
Chris@0
|
65 // Make some assumptions based on question name.
|
Chris@0
|
66 if ($default_value === NULL) {
|
Chris@0
|
67 switch ($name) {
|
Chris@0
|
68 case 'name':
|
Chris@0
|
69 $root_directory = basename(Utils::getExtensionRoot($directory) ?: $directory);
|
Chris@0
|
70 $default_value = Utils::machine2human($root_directory);
|
Chris@0
|
71 break;
|
Chris@0
|
72
|
Chris@0
|
73 case 'machine_name':
|
Chris@0
|
74 $default_value = function (array $vars) use ($directory) {
|
Chris@0
|
75 return Utils::human2machine(isset($vars['name']) ? $vars['name'] : basename($directory));
|
Chris@0
|
76 };
|
Chris@0
|
77 break;
|
Chris@0
|
78 }
|
Chris@0
|
79 }
|
Chris@0
|
80
|
Chris@0
|
81 // Turn the callback into a value acceptable for Symfony question helper.
|
Chris@0
|
82 if (is_callable($default_value)) {
|
Chris@0
|
83 // Do not treat simple strings as callable because they may match PHP
|
Chris@0
|
84 // builtin functions.
|
Chris@0
|
85 if (!is_string($default_value) || strpos('::', $default_value) !== FALSE) {
|
Chris@0
|
86 $default_value = call_user_func($default_value, $vars);
|
Chris@0
|
87 }
|
Chris@0
|
88 }
|
Chris@0
|
89 // Default value may have tokens.
|
Chris@0
|
90 $default_value = Utils::tokenReplace($default_value, $vars);
|
Chris@0
|
91 $this->setQuestionDefault($question, $default_value);
|
Chris@0
|
92
|
Chris@0
|
93 if ($answers) {
|
Chris@0
|
94 if (array_key_exists($name, $answers)) {
|
Chris@0
|
95 $answer = $answers[$name];
|
Chris@4
|
96 // Validate provided answer.
|
Chris@4
|
97 if ($validator = $question->getValidator()) {
|
Chris@4
|
98 $validator($answer);
|
Chris@4
|
99 }
|
Chris@0
|
100 // Turn 'yes/no' string into boolean.
|
Chris@0
|
101 if ($question instanceof ConfirmationQuestion && !is_bool($answer)) {
|
Chris@0
|
102 $answer = strcasecmp($answer, 'yes') == 0;
|
Chris@0
|
103 }
|
Chris@0
|
104 }
|
Chris@0
|
105 else {
|
Chris@0
|
106 $answer = $default_value;
|
Chris@0
|
107 }
|
Chris@0
|
108 }
|
Chris@0
|
109 else {
|
Chris@0
|
110 $this->formatQuestionText($question);
|
Chris@0
|
111 $answer = $question_helper->ask($input, $output, $question);
|
Chris@0
|
112 }
|
Chris@0
|
113
|
Chris@0
|
114 $vars[$name] = $answer;
|
Chris@0
|
115 }
|
Chris@0
|
116
|
Chris@0
|
117 return $vars;
|
Chris@0
|
118 }
|
Chris@0
|
119
|
Chris@0
|
120 /**
|
Chris@0
|
121 * Formats question text.
|
Chris@0
|
122 *
|
Chris@0
|
123 * @param \Symfony\Component\Console\Question\Question $question
|
Chris@0
|
124 * The question.
|
Chris@0
|
125 */
|
Chris@0
|
126 protected function formatQuestionText(Question $question) {
|
Chris@0
|
127 $question_text = $question->getQuestion();
|
Chris@0
|
128 $default_value = $question->getDefault();
|
Chris@0
|
129
|
Chris@0
|
130 $question_text = "\n <info>$question_text</info>";
|
Chris@0
|
131 if (is_bool($default_value)) {
|
Chris@0
|
132 $default_value = $default_value ? 'Yes' : 'No';
|
Chris@0
|
133 }
|
Chris@0
|
134 if ($default_value) {
|
Chris@0
|
135 $question_text .= " [<comment>$default_value</comment>]";
|
Chris@0
|
136 }
|
Chris@0
|
137 $question_text .= ":";
|
Chris@0
|
138 if ($question instanceof ChoiceQuestion) {
|
Chris@0
|
139 $question->setPrompt(' ➤➤➤ ');
|
Chris@0
|
140 }
|
Chris@0
|
141 else {
|
Chris@0
|
142 $question_text .= "\n ➤ ";
|
Chris@0
|
143 }
|
Chris@0
|
144
|
Chris@0
|
145 $this->setQuestionText($question, $question_text);
|
Chris@0
|
146 }
|
Chris@0
|
147
|
Chris@0
|
148 /**
|
Chris@0
|
149 * Normalizes questions.
|
Chris@0
|
150 *
|
Chris@0
|
151 * @param \Symfony\Component\Console\Question\Question[] $questions
|
Chris@0
|
152 * Questions to normalize.
|
Chris@0
|
153 *
|
Chris@0
|
154 * @return \Symfony\Component\Console\Question\Question[]
|
Chris@0
|
155 * Normalized questions
|
Chris@0
|
156 *
|
Chris@0
|
157 * @deprecated
|
Chris@0
|
158 * Use Symfony\Component\Console\Question\Question to define questions.
|
Chris@0
|
159 *
|
Chris@0
|
160 * @codeCoverageIgnore
|
Chris@0
|
161 */
|
Chris@0
|
162 protected function normalizeQuestions(array $questions) {
|
Chris@0
|
163 return array_map(function ($question) {
|
Chris@0
|
164 // Support array syntax.
|
Chris@0
|
165 if (is_array($question)) {
|
Chris@0
|
166 if (count($question) > 2) {
|
Chris@0
|
167 throw new \OutOfBoundsException('The question array is too long.');
|
Chris@0
|
168 }
|
Chris@0
|
169 list($question_text, $default_value) = array_pad($question, 2, NULL);
|
Chris@0
|
170 $question = new Question($question_text, $default_value);
|
Chris@0
|
171 }
|
Chris@0
|
172 return $question;
|
Chris@0
|
173 }, $questions);
|
Chris@0
|
174 }
|
Chris@0
|
175
|
Chris@0
|
176 }
|