import * as React from 'react'; import { HTMLProps } from 'react'; import styles from '@patternfly/react-styles/css/components/FormControl/form-control'; import { css } from '@patternfly/react-styles'; import { capitalize, ValidatedOptions, canUseDOM } from '../../helpers'; import { FormControlIcon } from '../FormControl/FormControlIcon'; export enum TextAreResizeOrientation { horizontal = 'horizontal', vertical = 'vertical', both = 'both' } export enum TextAreaReadOnlyVariant { default = 'default', plain = 'plain' } export interface TextAreaProps extends Omit, 'onChange' | 'ref'> { /** Additional classes added to the text area. */ className?: string; /** Flag to show if the text area is required. */ isRequired?: boolean; /** Flag to show if the text area is disabled. */ isDisabled?: boolean; /** Read only variant. */ readOnlyVariant?: 'default' | 'plain'; /** Flag to modify height based on contents. */ autoResize?: boolean; /** Value to indicate if the text area is modified to show that validation state. * If set to success, text area will be modified to indicate valid state. * If set to error, text area will be modified to indicate error state. */ validated?: 'success' | 'warning' | 'error' | 'default'; /** Value of the text area. */ value?: string | number; /** A callback for when the text area value changes. */ onChange?: (event: React.ChangeEvent, value: string) => void; /** Sets the orientation to limit the resize to */ resizeOrientation?: 'horizontal' | 'vertical' | 'both'; /** Custom flag to show that the text area requires an associated id or aria-label. */ 'aria-label'?: string; /** @hide A reference object to attach to the text area. */ innerRef?: React.RefObject; } class TextAreaBase extends React.Component { static displayName = 'TextArea'; static defaultProps: TextAreaProps = { innerRef: React.createRef(), className: '', isRequired: false, isDisabled: false, validated: 'default', resizeOrientation: 'both', 'aria-label': null as string }; inputRef = React.createRef(); private setAutoHeight = (field: HTMLTextAreaElement) => { const parent = field.parentElement; parent.style.setProperty('height', 'inherit'); const computed = window.getComputedStyle(field); // Calculate the height const height = parseInt(computed.getPropertyValue('border-top-width')) + parseInt(computed.getPropertyValue('padding-top')) + field.scrollHeight + parseInt(computed.getPropertyValue('padding-bottom')) + parseInt(computed.getPropertyValue('border-bottom-width')); parent.style.setProperty('height', `${height}px`); }; constructor(props: TextAreaProps) { super(props); if (!props.id && !props['aria-label']) { // eslint-disable-next-line no-console console.error('TextArea: TextArea requires either an id or aria-label to be specified'); } } componentDidMount(): void { const inputRef = this.props.innerRef || this.inputRef; if (this.props.autoResize && canUseDOM) { const field = inputRef.current; this.setAutoHeight(field); } } private handleChange = (event: React.ChangeEvent) => { // https://gomakethings.com/automatically-expand-a-textarea-as-the-user-types-using-vanilla-javascript/ const field = event.currentTarget; if (this.props.autoResize && canUseDOM) { this.setAutoHeight(field); } if (this.props.onChange) { this.props.onChange(event, field.value); } }; render() { const { className, value, validated, isRequired, isDisabled, readOnlyVariant, resizeOrientation, innerRef, disabled, /* eslint-disable @typescript-eslint/no-unused-vars */ autoResize, onChange, /* eslint-enable @typescript-eslint/no-unused-vars */ onBlur, onFocus, ...props } = this.props; const orientation = `resize${capitalize(resizeOrientation)}` as | 'resizeVertical' | 'resizeHorizontal' | 'resizeBoth'; const hasStatusIcon = ['success', 'error', 'warning'].includes(validated); return (