import { onBeforeUpdate, computed, onBeforeUnmount, onMounted, ref, defineComponent } from 'vue'; import classNames from '../_util/classNames'; import { FormItemInputContext, NoFormStatus, useInjectFormItemContext, } from '../form/FormItemContext'; import useConfigInject from '../_util/hooks/useConfigInject'; import { getMergedStatus, getStatusClassNames } from '../_util/statusUtils'; import type { InputFocusOptions } from '../vc-input/utils/commonUtils'; import { hasPrefixSuffix } from '../vc-input/utils/commonUtils'; import VcInput from '../vc-input/Input'; import inputProps from './inputProps'; import omit from '../_util/omit'; import CloseCircleFilled from '@ant-design/icons-vue/CloseCircleFilled'; export default defineComponent({ name: 'AInput', inheritAttrs: false, props: inputProps(), setup(props, { slots, attrs, expose, emit }) { const inputRef = ref(); const formItemContext = useInjectFormItemContext(); const formItemInputContext = FormItemInputContext.useInject(); const mergedStatus = computed(() => getMergedStatus(formItemInputContext.status, props.status)); const { direction, prefixCls, size, autocomplete } = useConfigInject('input', props); const focus = (option?: InputFocusOptions) => { inputRef.value?.focus(option); }; const blur = () => { inputRef.value?.blur(); }; const setSelectionRange = ( start: number, end: number, direction?: 'forward' | 'backward' | 'none', ) => { inputRef.value?.setSelectionRange(start, end, direction); }; const select = () => { inputRef.value?.select(); }; expose({ focus, blur, input: inputRef, setSelectionRange, select, }); // ===================== Remove Password value ===================== const removePasswordTimeoutRef = ref([]); const removePasswordTimeout = () => { removePasswordTimeoutRef.value.push( setTimeout(() => { if ( inputRef.value?.input && inputRef.value?.input.getAttribute('type') === 'password' && inputRef.value?.input.hasAttribute('value') ) { inputRef.value?.input.removeAttribute('value'); } }), ); }; onMounted(() => { removePasswordTimeout(); }); onBeforeUpdate(() => { removePasswordTimeoutRef.value.forEach(item => clearTimeout(item)); }); onBeforeUnmount(() => { removePasswordTimeoutRef.value.forEach(item => clearTimeout(item)); }); const handleBlur = (e: FocusEvent) => { removePasswordTimeout(); emit('blur', e); }; const handleFocus = (e: FocusEvent) => { removePasswordTimeout(); emit('focus', e); }; const triggerChange = (e: Event) => { emit('update:value', (e.target as HTMLInputElement).value); emit('change', e); emit('input', e); formItemContext.onFieldChange(); }; return () => { const { hasFeedback, feedbackIcon } = formItemInputContext; const { allowClear, bordered = true, prefix = slots.prefix?.(), suffix = slots.suffix?.(), addonAfter = slots.addonAfter?.(), addonBefore = slots.addonBefore?.(), id = formItemContext.id?.value, ...rest } = props; const suffixNode = (hasFeedback || suffix) && ( <> {suffix} {hasFeedback && feedbackIcon} ); const prefixClsValue = prefixCls.value; const inputHasPrefixSuffix = hasPrefixSuffix({ prefix, suffix }) || !!hasFeedback; const clearIcon = slots.clearIcon || (() => ); return ( {addonAfter}} addonBefore={addonBefore && {addonBefore}} inputClassName={classNames( { [`${prefixClsValue}-sm`]: size.value === 'small', [`${prefixClsValue}-lg`]: size.value === 'large', [`${prefixClsValue}-rtl`]: direction.value === 'rtl', [`${prefixClsValue}-borderless`]: !bordered, }, !inputHasPrefixSuffix && getStatusClassNames(prefixClsValue, mergedStatus.value), )} affixWrapperClassName={classNames( { [`${prefixClsValue}-affix-wrapper-sm`]: size.value === 'small', [`${prefixClsValue}-affix-wrapper-lg`]: size.value === 'large', [`${prefixClsValue}-affix-wrapper-rtl`]: direction.value === 'rtl', [`${prefixClsValue}-affix-wrapper-borderless`]: !bordered, }, getStatusClassNames(`${prefixClsValue}-affix-wrapper`, mergedStatus.value, hasFeedback), )} wrapperClassName={classNames({ [`${prefixClsValue}-group-rtl`]: direction.value === 'rtl', })} groupClassName={classNames( { [`${prefixClsValue}-group-wrapper-sm`]: size.value === 'small', [`${prefixClsValue}-group-wrapper-lg`]: size.value === 'large', [`${prefixClsValue}-group-wrapper-rtl`]: direction.value === 'rtl', }, getStatusClassNames(`${prefixClsValue}-group-wrapper`, mergedStatus.value, hasFeedback), )} v-slots={{ ...slots, clearIcon }} > ); }; }, });