296 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			Vue
		
	
	
			
		
		
	
	
			296 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			Vue
		
	
	
| 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<string>();
 | ||
|     const oldSelectionStartRef = ref<number>(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 (
 | ||
|         <ResizableTextArea
 | ||
|           {...resizeProps}
 | ||
|           id={resizeProps.id ?? formItemContext.id.value}
 | ||
|           ref={resizableTextArea}
 | ||
|           maxlength={props.maxlength}
 | ||
|         />
 | ||
|       );
 | ||
|     };
 | ||
| 
 | ||
|     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 = (
 | ||
|         <ClearableLabeledInput
 | ||
|           {...inputProps}
 | ||
|           value={mergedValue.value}
 | ||
|           v-slots={{ element: renderTextArea }}
 | ||
|         />
 | ||
|       );
 | ||
| 
 | ||
|       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 = (
 | ||
|           <div
 | ||
|             hidden={hidden}
 | ||
|             class={classNames(
 | ||
|               `${prefixCls.value}-textarea`,
 | ||
|               {
 | ||
|                 [`${prefixCls.value}-textarea-rtl`]: direction.value === 'rtl',
 | ||
|               },
 | ||
|               `${prefixCls.value}-textarea-show-count`,
 | ||
|               customClass,
 | ||
|             )}
 | ||
|             style={style as CSSProperties}
 | ||
|             data-count={typeof dataCount !== 'object' ? dataCount : undefined}
 | ||
|           >
 | ||
|             {textareaNode}
 | ||
|           </div>
 | ||
|         );
 | ||
|       }
 | ||
|       return textareaNode;
 | ||
|     };
 | ||
|   },
 | ||
| });
 |