import type { CSSProperties } from 'vue'; import { computed, defineComponent, getCurrentInstance, nextTick, onMounted, ref, watch, watchEffect, } from 'vue'; import ClearableLabeledInput from './ClearableLabeledInput'; import ResizableTextArea from './ResizableTextArea'; import { textAreaProps } from './inputProps'; import type { InputFocusOptions } from './Input'; import { fixControlledValue, resolveOnChange, triggerFocus } from './Input'; import classNames from '../_util/classNames'; import { useInjectFormItemContext } from '../form/FormItemContext'; import type { FocusEventHandler } from '../_util/EventInterface'; import useConfigInject from '../_util/hooks/useConfigInject'; import omit from '../_util/omit'; import type { VueNode } from '../_util/type'; function fixEmojiLength(value: string, maxLength: number) { return [...(value || '')].slice(0, maxLength).join(''); } function setTriggerValue( isCursorInEnd: boolean, preValue: string, triggerValue: string, maxLength: number, ) { let newTriggerValue = triggerValue; if (isCursorInEnd) { // 光标在尾部,直接截断 newTriggerValue = fixEmojiLength(triggerValue, maxLength!); } else if ( [...(preValue || '')].length < triggerValue.length && [...(triggerValue || '')].length > maxLength! ) { // 光标在中间,如果最后的值超过最大值,则采用原先的值 newTriggerValue = preValue; } return newTriggerValue; } export default defineComponent({ name: 'ATextarea', inheritAttrs: false, props: textAreaProps(), setup(props, { attrs, expose, emit }) { const formItemContext = useInjectFormItemContext(); const stateValue = ref(props.value === undefined ? props.defaultValue : props.value); const resizableTextArea = ref(); const mergedValue = ref(''); const { prefixCls, size, direction } = useConfigInject('input', props); const showCount = computed(() => { return (props.showCount as any) === '' || props.showCount || false; }); // Max length value const hasMaxLength = computed(() => Number(props.maxlength) > 0); const compositing = ref(false); const oldCompositionValueRef = ref(); const oldSelectionStartRef = ref(0); const onInternalCompositionStart = (e: CompositionEvent) => { compositing.value = true; // 拼音输入前保存一份旧值 oldCompositionValueRef.value = mergedValue.value as string; // 保存旧的光标位置 oldSelectionStartRef.value = (e.currentTarget as any).selectionStart; emit('compositionstart', e); }; const onInternalCompositionEnd = (e: CompositionEvent) => { compositing.value = false; let triggerValue = (e.currentTarget as any).value; if (hasMaxLength.value) { const isCursorInEnd = oldSelectionStartRef.value >= props.maxlength + 1 || oldSelectionStartRef.value === oldCompositionValueRef.value?.length; triggerValue = setTriggerValue( isCursorInEnd, oldCompositionValueRef.value as string, triggerValue, props.maxlength, ); } // Patch composition onChange when value changed if (triggerValue !== mergedValue.value) { setValue(triggerValue); resolveOnChange(e.currentTarget as any, e, triggerChange, triggerValue); } emit('compositionend', e); }; const instance = getCurrentInstance(); watch( () => props.value, () => { if ('value' in instance.vnode.props || {}) { stateValue.value = props.value ?? ''; } }, ); const focus = (option?: InputFocusOptions) => { triggerFocus(resizableTextArea.value?.textArea, option); }; const blur = () => { resizableTextArea.value?.textArea?.blur(); }; const setValue = (value: string | number, callback?: Function) => { if (stateValue.value === value) { return; } if (props.value === undefined) { stateValue.value = value; } else { nextTick(() => { if (resizableTextArea.value.textArea.value !== mergedValue.value) { resizableTextArea.value?.instance.update?.(); } }); } nextTick(() => { callback && callback(); }); }; const handleKeyDown = (e: KeyboardEvent) => { if (e.keyCode === 13) { emit('pressEnter', e); } emit('keydown', e); }; const onBlur: FocusEventHandler = e => { const { onBlur } = props; onBlur?.(e); formItemContext.onFieldBlur(); }; const triggerChange = (e: Event) => { emit('update:value', (e.target as HTMLInputElement).value); emit('change', e); emit('input', e); formItemContext.onFieldChange(); }; const handleReset = (e: MouseEvent) => { resolveOnChange(resizableTextArea.value.textArea, e, triggerChange); setValue('', () => { focus(); }); }; const handleChange = (e: Event) => { const { composing } = e.target as any; let triggerValue = (e.target as any).value; compositing.value = !!((e as any).isComposing || composing); if ((compositing.value && props.lazy) || stateValue.value === triggerValue) return; if (hasMaxLength.value) { // 1. 复制粘贴超过maxlength的情况 2.未超过maxlength的情况 const target = e.target as any; const isCursorInEnd = target.selectionStart >= props.maxlength! + 1 || target.selectionStart === triggerValue.length || !target.selectionStart; triggerValue = setTriggerValue( isCursorInEnd, mergedValue.value as string, triggerValue, props.maxlength!, ); } resolveOnChange(e.currentTarget as any, e, triggerChange, triggerValue); setValue(triggerValue); }; const renderTextArea = () => { const { style, class: customClass } = attrs; const { bordered = true } = props; const resizeProps = { ...omit(props, ['allowClear']), ...attrs, style: showCount.value ? {} : style, class: { [`${prefixCls.value}-borderless`]: !bordered, [`${customClass}`]: customClass && !showCount.value, [`${prefixCls.value}-sm`]: size.value === 'small', [`${prefixCls.value}-lg`]: size.value === 'large', }, showCount: null, prefixCls: prefixCls.value, onInput: handleChange, onChange: handleChange, onBlur, onKeydown: handleKeyDown, onCompositionstart: onInternalCompositionStart, onCompositionend: onInternalCompositionEnd, }; if (props.valueModifiers?.lazy) { delete resizeProps.onInput; } return ( ); }; onMounted(() => { if (process.env.NODE_ENV === 'test') { if (props.autofocus) { focus(); } } }); expose({ focus, blur, resizableTextArea, }); watchEffect(() => { let val = fixControlledValue(stateValue.value) as string; if ( !compositing.value && hasMaxLength.value && (props.value === null || props.value === undefined) ) { // fix #27612 将value转为数组进行截取,解决 '😂'.length === 2 等emoji表情导致的截取乱码的问题 val = fixEmojiLength(val, props.maxlength); } mergedValue.value = val; }); return () => { const { maxlength, bordered = true, hidden } = props; const { style, class: customClass } = attrs; const inputProps: any = { ...props, ...attrs, prefixCls: prefixCls.value, inputType: 'text', handleReset, direction: direction.value, bordered, style: showCount.value ? undefined : style, }; let textareaNode = ( ); if (showCount.value) { const valueLength = [...mergedValue.value].length; let dataCount: VueNode = ''; if (typeof showCount.value === 'object') { dataCount = showCount.value.formatter({ count: valueLength, maxlength }); } else { dataCount = `${valueLength}${hasMaxLength.value ? ` / ${maxlength}` : ''}`; } textareaNode = ( ); } return textareaNode; }; }, });