Merge "SpecialBlock: Scroll to the error's fieldset instead of field"
[mediawiki.git] / includes / htmlform / fields / HTMLSelectAndOtherField.php
blob55ff21da0b14cae20a190f7ebee442feab5d9d50
1 <?php
3 namespace MediaWiki\HTMLForm\Field;
5 use InvalidArgumentException;
6 use MediaWiki\Html\Html;
7 use MediaWiki\Request\WebRequest;
8 use MediaWiki\Widget\SelectWithInputWidget;
10 /**
11 * Double field with a dropdown list constructed from a system message in the format
12 * * Optgroup header
13 * ** <option value>
14 * * New Optgroup header
15 * Plus a text field underneath for an additional reason. The 'value' of the field is
16 * "<select>: <extra reason>", or "<extra reason>" if nothing has been selected in the
17 * select dropdown.
19 * @stable to extend
20 * @todo FIXME: If made 'required', only the text field should be compulsory.
22 class HTMLSelectAndOtherField extends HTMLSelectField {
23 private const FIELD_CLASS = 'mw-htmlform-select-and-other-field';
24 /** @var string[] */
25 private $mFlatOptions;
27 /**
28 * @stable to call
29 * @inheritDoc
31 public function __construct( $params ) {
32 if ( array_key_exists( 'other', $params ) ) {
33 // Do nothing
34 } elseif ( array_key_exists( 'other-message', $params ) ) {
35 $params['other'] = $this->getMessage( $params['other-message'] )->plain();
36 } else {
37 $params['other'] = $this->msg( 'htmlform-selectorother-other' )->plain();
40 parent::__construct( $params );
42 if ( $this->getOptions() === null ) {
43 throw new InvalidArgumentException( 'HTMLSelectAndOtherField called without any options' );
45 if ( !in_array( 'other', $this->mOptions, true ) ) {
46 // Have 'other' always as first element
47 $this->mOptions = [ $params['other'] => 'other' ] + $this->mOptions;
49 $this->mFlatOptions = self::flattenOptions( $this->getOptions() );
52 public function getInputHTML( $value ) {
53 $select = parent::getInputHTML( $value[1] );
55 $textAttribs = [
56 'size' => $this->getSize(),
59 if ( isset( $this->mParams['maxlength-unit'] ) ) {
60 $textAttribs['data-mw-maxlength-unit'] = $this->mParams['maxlength-unit'];
63 $allowedParams = [
64 'required',
65 'autofocus',
66 'multiple',
67 'disabled',
68 'tabindex',
69 'maxlength', // gets dynamic with javascript, see mediawiki.htmlform.js
70 'maxlength-unit', // 'bytes' or 'codepoints', see mediawiki.htmlform.js
73 $textAttribs += $this->getAttributes( $allowedParams );
75 $textbox = Html::input( $this->mName . '-other', $value[2], 'text', $textAttribs );
77 $wrapperAttribs = [
78 'id' => $this->mID,
79 'class' => self::FIELD_CLASS
81 if ( $this->mClass !== '' ) {
82 $wrapperAttribs['class'] .= ' ' . $this->mClass;
84 return Html::rawElement(
85 'div',
86 $wrapperAttribs,
87 "$select<br />\n$textbox"
91 protected function getOOUIModules() {
92 return [ 'mediawiki.widgets.SelectWithInputWidget' ];
95 public function getInputOOUI( $value ) {
96 $this->mParent->getOutput()->addModuleStyles( 'mediawiki.widgets.SelectWithInputWidget.styles' );
98 # TextInput
99 $textAttribs = [
100 'name' => $this->mName . '-other',
101 'value' => $value[2],
104 $allowedParams = [
105 'required',
106 'autofocus',
107 'multiple',
108 'disabled',
109 'tabindex',
110 'maxlength',
113 $textAttribs += \OOUI\Element::configFromHtmlAttributes(
114 $this->getAttributes( $allowedParams )
117 # DropdownInput
118 $dropdownInputAttribs = [
119 'name' => $this->mName,
120 'options' => $this->getOptionsOOUI(),
121 'value' => $value[1],
124 $allowedParams = [
125 'tabindex',
126 'disabled',
129 $dropdownInputAttribs += \OOUI\Element::configFromHtmlAttributes(
130 $this->getAttributes( $allowedParams )
133 $disabled = false;
134 if ( isset( $this->mParams[ 'disabled' ] ) && $this->mParams[ 'disabled' ] ) {
135 $disabled = true;
138 $inputClasses = [ self::FIELD_CLASS ];
139 if ( $this->mClass !== '' ) {
140 $inputClasses = array_merge( $inputClasses, explode( ' ', $this->mClass ) );
142 return $this->getInputWidget( [
143 'id' => $this->mID,
144 'disabled' => $disabled,
145 'textinput' => $textAttribs,
146 'dropdowninput' => $dropdownInputAttribs,
147 'or' => false,
148 'required' => $this->mParams[ 'required' ] ?? false,
149 'classes' => $inputClasses,
150 'data' => [
151 'maxlengthUnit' => $this->mParams['maxlength-unit'] ?? 'bytes'
153 ] );
157 * @stable to override
158 * @param array $params
159 * @return \MediaWiki\Widget\SelectWithInputWidget
161 public function getInputWidget( $params ) {
162 return new SelectWithInputWidget( $params );
165 public function getInputCodex( $value, $hasErrors ) {
166 $select = parent::getInputCodex( $value[1], $hasErrors );
168 // Set up attributes for the text input.
169 $textInputAttribs = [
170 'size' => $this->getSize(),
171 'name' => $this->mName . '-other'
174 if ( isset( $this->mParams['maxlength-unit'] ) ) {
175 $textInputAttribs['data-mw-maxlength-unit'] = $this->mParams['maxlength-unit'];
178 $allowedParams = [
179 'required',
180 'autofocus',
181 'multiple',
182 'disabled',
183 'tabindex',
184 'maxlength', // gets dynamic with javascript, see mediawiki.htmlform.js
185 'maxlength-unit', // 'bytes' or 'codepoints', see mediawiki.htmlform.js
188 $textInputAttribs += $this->getAttributes( $allowedParams );
190 // Get text input HTML.
191 $textInput = HTMLTextField::buildCodexComponent(
192 $value[2],
193 $hasErrors,
194 'text',
195 $this->mName . '-other',
196 $textInputAttribs
199 // Set up the wrapper element and return the entire component.
200 $wrapperAttribs = [
201 'id' => $this->mID,
202 'class' => [ self::FIELD_CLASS ]
204 if ( $this->mClass !== '' ) {
205 $wrapperAttribs['class'][] = $this->mClass;
207 return Html::rawElement(
208 'div',
209 $wrapperAttribs,
210 "$select<br />\n$textInput"
215 * @inheritDoc
217 public function getDefault() {
218 $default = parent::getDefault();
220 // Default values of empty form
221 $final = '';
222 $list = 'other';
223 $text = '';
225 if ( $default !== null ) {
226 $final = $default;
227 // Assume the default is a text value, with the 'other' option selected.
228 // Then check if that assumption is correct, and update $list and $text if not.
229 $text = $final;
230 foreach ( $this->mFlatOptions as $option ) {
231 $match = $option . $this->msg( 'colon-separator' )->inContentLanguage()->text();
232 if ( str_starts_with( $final, $match ) ) {
233 $list = $option;
234 $text = substr( $final, strlen( $match ) );
235 break;
240 return [ $final, $list, $text ];
244 * @param WebRequest $request
246 * @return array ["<overall message>","<select value>","<text field value>"]
248 public function loadDataFromRequest( $request ) {
249 if ( $request->getCheck( $this->mName ) ) {
250 $list = $request->getText( $this->mName );
251 $text = $request->getText( $this->mName . '-other' );
253 // Should be built the same as in mediawiki.htmlform.js
254 if ( $list == 'other' ) {
255 $final = $text;
256 } elseif ( !in_array( $list, $this->mFlatOptions, true ) ) {
257 # User has spoofed the select form to give an option which wasn't
258 # in the original offer. Sulk...
259 $final = $text;
260 } elseif ( $text == '' ) {
261 $final = $list;
262 } else {
263 $final = $list . $this->msg( 'colon-separator' )->inContentLanguage()->text() . $text;
265 return [ $final, $list, $text ];
267 return $this->getDefault();
270 public function getSize() {
271 return $this->mParams['size'] ?? 45;
274 public function validate( $value, $alldata ) {
275 # HTMLSelectField forces $value to be one of the options in the select
276 # field, which is not useful here. But we do want the validation further up
277 # the chain
278 $p = parent::validate( $value[1], $alldata );
280 if ( $p !== true ) {
281 return $p;
284 if ( isset( $this->mParams['required'] )
285 && $this->mParams['required'] !== false
286 && $value[0] === ''
288 return $this->msg( 'htmlform-required' );
291 return true;
295 /** @deprecated class alias since 1.42 */
296 class_alias( HTMLSelectAndOtherField::class, 'HTMLSelectAndOtherField' );