2018-03-26 14:32:43 -07:00
|
|
|
// @flow
|
2022-01-24 11:07:09 -05:00
|
|
|
import 'easymde/dist/easymde.min.css';
|
|
|
|
import { FF_MAX_CHARS_DEFAULT } from 'constants/form-field';
|
2019-05-07 21:42:56 -04:00
|
|
|
import React from 'react';
|
2022-01-24 11:07:09 -05:00
|
|
|
import type { ElementRef, Node } from 'react';
|
2019-04-03 00:56:58 -05:00
|
|
|
|
2018-03-26 14:32:43 -07:00
|
|
|
type Props = {
|
2018-06-14 16:10:50 -04:00
|
|
|
affixClass?: string, // class applied to prefix/postfix label
|
2018-09-21 19:20:58 -06:00
|
|
|
autoFocus?: boolean,
|
2019-02-18 12:24:56 -05:00
|
|
|
blockWrap: boolean,
|
2019-09-30 23:16:46 +02:00
|
|
|
charCount?: number,
|
2022-01-24 11:07:09 -05:00
|
|
|
children?: React$Node,
|
|
|
|
defaultValue?: string | number,
|
|
|
|
disabled?: boolean,
|
|
|
|
error?: string | boolean,
|
|
|
|
helper?: string | React$Node,
|
|
|
|
inputButton?: React$Node,
|
|
|
|
label?: string | Node,
|
|
|
|
labelOnLeft: boolean,
|
2020-03-24 23:15:05 -04:00
|
|
|
max?: number,
|
2022-01-24 11:07:09 -05:00
|
|
|
min?: number,
|
|
|
|
name: string,
|
|
|
|
placeholder?: string | number,
|
|
|
|
postfix?: string,
|
|
|
|
prefix?: string,
|
|
|
|
range?: number,
|
|
|
|
readOnly?: boolean,
|
|
|
|
stretch?: boolean,
|
|
|
|
textAreaMaxLength?: number,
|
|
|
|
type?: string,
|
2021-04-23 15:59:48 -04:00
|
|
|
value?: string | number,
|
2022-01-24 11:07:09 -05:00
|
|
|
onChange?: (any) => any,
|
|
|
|
render?: () => React$Node,
|
2018-03-26 14:32:43 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
export class FormField extends React.PureComponent<Props> {
|
2022-01-24 11:07:09 -05:00
|
|
|
static defaultProps = { labelOnLeft: false, blockWrap: true };
|
2019-02-13 12:27:20 -04:00
|
|
|
|
2019-04-24 10:02:08 -04:00
|
|
|
input: { current: ElementRef<any> };
|
2019-02-20 00:20:29 -05:00
|
|
|
|
2019-02-18 12:24:56 -05:00
|
|
|
constructor(props: Props) {
|
2018-09-21 19:20:58 -06:00
|
|
|
super(props);
|
|
|
|
this.input = React.createRef();
|
|
|
|
}
|
|
|
|
|
|
|
|
componentDidMount() {
|
|
|
|
const { autoFocus } = this.props;
|
|
|
|
const input = this.input.current;
|
|
|
|
|
2022-01-24 11:07:09 -05:00
|
|
|
if (input && autoFocus) input.focus();
|
2018-09-21 19:20:58 -06:00
|
|
|
}
|
|
|
|
|
2018-03-26 14:32:43 -07:00
|
|
|
render() {
|
|
|
|
const {
|
2018-06-14 16:10:50 -04:00
|
|
|
affixClass,
|
2018-09-21 19:20:58 -06:00
|
|
|
autoFocus,
|
2019-02-18 12:24:56 -05:00
|
|
|
blockWrap,
|
2019-09-30 23:16:46 +02:00
|
|
|
charCount,
|
2022-01-24 11:07:09 -05:00
|
|
|
children,
|
|
|
|
error,
|
|
|
|
helper,
|
|
|
|
inputButton,
|
|
|
|
label,
|
|
|
|
labelOnLeft,
|
|
|
|
name,
|
|
|
|
postfix,
|
|
|
|
prefix,
|
|
|
|
stretch,
|
|
|
|
textAreaMaxLength,
|
|
|
|
type,
|
|
|
|
render,
|
2018-03-26 14:32:43 -07:00
|
|
|
...inputProps
|
|
|
|
} = this.props;
|
2022-01-24 11:07:09 -05:00
|
|
|
|
2018-04-05 17:26:20 -04:00
|
|
|
const errorMessage = typeof error === 'object' ? error.message : error;
|
2022-01-24 11:07:09 -05:00
|
|
|
|
|
|
|
// Ideally, the character count should (and can) be appended to the
|
|
|
|
// SimpleMDE's "options::status" bar. However, I couldn't figure out how
|
|
|
|
// to pass the current value to it's callback, nor query the current
|
|
|
|
// text length from the callback. So, we'll use our own widget.
|
|
|
|
const hasCharCount = charCount !== undefined && charCount >= 0;
|
|
|
|
const countInfo = hasCharCount && textAreaMaxLength !== undefined && (
|
|
|
|
<span className="comment__char-count-mde">{`${charCount || '0'}/${textAreaMaxLength}`}</span>
|
|
|
|
);
|
2019-02-18 12:24:56 -05:00
|
|
|
const Wrapper = blockWrap
|
2019-11-22 16:13:00 -05:00
|
|
|
? ({ children: innerChildren }) => <fieldset-section class="radio">{innerChildren}</fieldset-section>
|
|
|
|
: ({ children: innerChildren }) => <span className="radio">{innerChildren}</span>;
|
2019-02-18 12:24:56 -05:00
|
|
|
|
2022-01-24 11:07:09 -05:00
|
|
|
const inputSimple = (type: string) => (
|
|
|
|
<>
|
|
|
|
<input id={name} type={type} {...inputProps} />
|
|
|
|
<label htmlFor={name}>{label}</label>
|
|
|
|
</>
|
|
|
|
);
|
2018-07-28 17:48:54 -06:00
|
|
|
|
2022-01-24 11:07:09 -05:00
|
|
|
const inputSelect = (selectClass: string) => (
|
|
|
|
<fieldset-section class={selectClass}>
|
|
|
|
{(label || errorMessage) && (
|
|
|
|
<label htmlFor={name}>{errorMessage ? <span className="error__text">{errorMessage}</span> : label}</label>
|
|
|
|
)}
|
|
|
|
<select id={name} {...inputProps}>
|
|
|
|
{children}
|
|
|
|
</select>
|
|
|
|
</fieldset-section>
|
|
|
|
);
|
2021-02-26 15:38:33 +08:00
|
|
|
|
2022-01-24 11:07:09 -05:00
|
|
|
const input = () => {
|
|
|
|
switch (type) {
|
|
|
|
case 'radio':
|
|
|
|
return <Wrapper>{inputSimple('radio')}</Wrapper>;
|
|
|
|
case 'checkbox':
|
|
|
|
return <div className="checkbox">{inputSimple('checkbox')}</div>;
|
|
|
|
case 'range':
|
|
|
|
return <div>{inputSimple('range')}</div>;
|
|
|
|
case 'select':
|
|
|
|
return inputSelect('');
|
|
|
|
case 'select-tiny':
|
|
|
|
return inputSelect('select--slim');
|
|
|
|
case 'textarea':
|
|
|
|
return (
|
2019-02-13 12:27:20 -04:00
|
|
|
<fieldset-section>
|
2022-11-04 08:42:36 -04:00
|
|
|
{label && (
|
2022-01-24 11:07:09 -05:00
|
|
|
<div className="form-field__two-column">
|
2020-05-21 15:25:37 +08:00
|
|
|
<label htmlFor={name}>{label}</label>
|
|
|
|
</div>
|
2022-01-24 11:07:09 -05:00
|
|
|
)}
|
2022-11-04 08:42:36 -04:00
|
|
|
<textarea
|
|
|
|
type={type}
|
|
|
|
id={name}
|
|
|
|
maxLength={textAreaMaxLength || FF_MAX_CHARS_DEFAULT}
|
|
|
|
ref={this.input}
|
|
|
|
{...inputProps}
|
|
|
|
/>
|
|
|
|
<div className="form-field__textarea-info">{countInfo}</div>
|
2019-02-13 12:27:20 -04:00
|
|
|
</fieldset-section>
|
2022-01-24 11:07:09 -05:00
|
|
|
);
|
|
|
|
default:
|
|
|
|
const inputElement = <input type={type} id={name} {...inputProps} ref={this.input} />;
|
|
|
|
const inner = inputButton ? (
|
|
|
|
<input-submit>
|
|
|
|
{inputElement}
|
|
|
|
{inputButton}
|
|
|
|
</input-submit>
|
|
|
|
) : (
|
|
|
|
inputElement
|
|
|
|
);
|
2019-02-13 12:27:20 -04:00
|
|
|
|
2022-01-24 11:07:09 -05:00
|
|
|
return (
|
2019-02-13 12:27:20 -04:00
|
|
|
<fieldset-section>
|
2020-02-06 13:49:05 -05:00
|
|
|
{(label || errorMessage) && (
|
2019-11-22 16:13:00 -05:00
|
|
|
<label htmlFor={name}>
|
2020-04-13 15:16:07 -04:00
|
|
|
{errorMessage ? <span className="error__text">{errorMessage}</span> : label}
|
2019-11-22 16:13:00 -05:00
|
|
|
</label>
|
|
|
|
)}
|
2019-09-26 12:07:11 -04:00
|
|
|
{prefix && <label htmlFor={name}>{prefix}</label>}
|
2019-02-13 12:27:20 -04:00
|
|
|
{inner}
|
|
|
|
</fieldset-section>
|
2022-01-24 11:07:09 -05:00
|
|
|
);
|
2018-03-26 14:32:43 -07:00
|
|
|
}
|
2022-01-24 11:07:09 -05:00
|
|
|
};
|
2018-03-26 14:32:43 -07:00
|
|
|
|
|
|
|
return (
|
2022-01-24 11:07:09 -05:00
|
|
|
<>
|
|
|
|
{type && input()}
|
2019-03-21 11:22:23 -04:00
|
|
|
{helper && <div className="form-field__help">{helper}</div>}
|
2022-01-24 11:07:09 -05:00
|
|
|
</>
|
2018-03-26 14:32:43 -07:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export default FormField;
|