import { nextTick, PropType, VNode } from 'vue'; import ResizeObserver from '../vc-resize-observer'; import omit from 'omit.js'; import classNames from '../_util/classNames'; import calculateNodeHeight from './calculateNodeHeight'; import raf from '../_util/raf'; import warning from '../_util/warning'; import BaseMixin from '../_util/BaseMixin'; import inputProps from './inputProps'; import PropTypes from '../_util/vue-types'; import { getOptionProps } from '../_util/props-util'; import { defineComponent, withDirectives } from 'vue'; import antInput from '../_util/antInputDirective'; const RESIZE_STATUS_NONE = 0; const RESIZE_STATUS_RESIZING = 1; const RESIZE_STATUS_RESIZED = 2; export interface AutoSizeType { minRows?: number; maxRows?: number; } const TextAreaProps = { ...inputProps, autosize: { type: [Boolean, Object] as PropType, default: undefined }, autoSize: { type: [Boolean, Object] as PropType, default: undefined }, onResize: PropTypes.func, }; const ResizableTextArea = defineComponent({ name: 'ResizableTextArea', mixins: [BaseMixin], inheritAttrs: false, props: TextAreaProps, setup() { return { nextFrameActionId: undefined, textArea: null, resizeFrameId: undefined, }; }, data() { return { textareaStyles: {}, resizeStatus: RESIZE_STATUS_NONE, }; }, watch: { value() { nextTick(() => { this.resizeTextarea(); }); }, }, mounted() { this.resizeTextarea(); }, beforeUnmount() { raf.cancel(this.nextFrameActionId); raf.cancel(this.resizeFrameId); }, methods: { saveTextArea(textArea: HTMLTextAreaElement) { this.textArea = textArea; }, handleResize(size: { width: number; height: number }) { const { resizeStatus } = this.$data; if (resizeStatus !== RESIZE_STATUS_NONE) { return; } this.$emit('resize', size); }, resizeOnNextFrame() { raf.cancel(this.nextFrameActionId); this.nextFrameActionId = raf(this.resizeTextarea); }, resizeTextarea() { const autoSize = this.$props.autoSize || this.$props.autosize; if (!autoSize || !this.textArea) { return; } const { minRows, maxRows } = autoSize; const textareaStyles = calculateNodeHeight(this.textArea, false, minRows, maxRows); this.setState({ textareaStyles, resizeStatus: RESIZE_STATUS_RESIZING }, () => { raf.cancel(this.resizeFrameId); this.resizeFrameId = raf(() => { this.setState({ resizeStatus: RESIZE_STATUS_RESIZED }, () => { this.resizeFrameId = raf(() => { this.setState({ resizeStatus: RESIZE_STATUS_NONE }); this.fixFirefoxAutoScroll(); }); }); }); }); }, // https://github.com/ant-design/ant-design/issues/21870 fixFirefoxAutoScroll() { try { if (document.activeElement === this.textArea) { const currentStart = this.textArea.selectionStart; const currentEnd = this.textArea.selectionEnd; this.textArea.setSelectionRange(currentStart, currentEnd); } } catch (e) { // Fix error in Chrome: // Failed to read the 'selectionStart' property from 'HTMLInputElement' // http://stackoverflow.com/q/21177489/3040605 } }, renderTextArea() { const props: any = { ...getOptionProps(this), ...this.$attrs }; const { prefixCls, autoSize, autosize, disabled, class: className } = props; const { textareaStyles, resizeStatus } = this.$data; warning( autosize === undefined, 'Input.TextArea', 'autosize is deprecated, please use autoSize instead.', ); const otherProps = omit(props, [ 'prefixCls', 'onPressEnter', 'autoSize', 'autosize', 'defaultValue', 'allowClear', 'type', 'lazy', ]); const cls = classNames(prefixCls, className, { [`${prefixCls}-disabled`]: disabled, }); // Fix https://github.com/ant-design/ant-design/issues/6776 // Make sure it could be reset when using form.getFieldDecorator if ('value' in otherProps) { otherProps.value = otherProps.value || ''; } const style = { ...props.style, ...textareaStyles, ...(resizeStatus === RESIZE_STATUS_RESIZING ? { overflowX: 'hidden', overflowY: 'hidden' } : null), }; const textareaProps: any = { ...otherProps, style, class: cls, }; if (!textareaProps.autofocus) { delete textareaProps.autofocus; } return ( {withDirectives((