mirror of
https://gitee.com/ant-design-vue/ant-design-vue.git
synced 2024-12-05 13:39:40 +08:00
205 lines
6.8 KiB
Vue
205 lines
6.8 KiB
Vue
import { onBeforeUpdate, computed, onBeforeUnmount, onMounted, ref, defineComponent } from 'vue';
|
|
import classNames from '../_util/classNames';
|
|
import {
|
|
FormItemInputContext,
|
|
NoFormStatus,
|
|
useInjectFormItemContext,
|
|
} from '../form/FormItemContext';
|
|
import useConfigInject from '../config-provider/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';
|
|
import { NoCompactStyle, useCompactItemContext } from '../space/Compact';
|
|
|
|
// CSSINJS
|
|
import useStyle from './style';
|
|
import { useInjectDisabled } from '../config-provider/DisabledContext';
|
|
|
|
export default defineComponent({
|
|
compatConfig: { MODE: 3 },
|
|
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);
|
|
|
|
// ===================== Compact Item =====================
|
|
const { compactSize, compactItemClassnames } = useCompactItemContext(prefixCls, direction);
|
|
const mergedSize = computed(() => {
|
|
return compactSize.value || size.value;
|
|
});
|
|
// Style
|
|
const [wrapSSR, hashId] = useStyle(prefixCls);
|
|
|
|
const disabled = useInjectDisabled();
|
|
|
|
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<any[]>([]);
|
|
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);
|
|
formItemContext.onFieldBlur();
|
|
};
|
|
|
|
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 || (() => <CloseCircleFilled />);
|
|
return wrapSSR(
|
|
<VcInput
|
|
{...attrs}
|
|
{...omit(rest, ['onUpdate:value', 'onChange', 'onInput'])}
|
|
onChange={triggerChange}
|
|
id={id}
|
|
disabled={props.disabled ?? disabled.value}
|
|
ref={inputRef}
|
|
prefixCls={prefixClsValue}
|
|
autocomplete={autocomplete.value}
|
|
onBlur={handleBlur}
|
|
onFocus={handleFocus}
|
|
suffix={suffixNode}
|
|
allowClear={allowClear}
|
|
addonAfter={
|
|
addonAfter && (
|
|
<NoCompactStyle>
|
|
<NoFormStatus>{addonAfter}</NoFormStatus>
|
|
</NoCompactStyle>
|
|
)
|
|
}
|
|
addonBefore={
|
|
addonBefore && (
|
|
<NoCompactStyle>
|
|
<NoFormStatus>{addonBefore}</NoFormStatus>
|
|
</NoCompactStyle>
|
|
)
|
|
}
|
|
class={[attrs.class, compactItemClassnames.value]}
|
|
inputClassName={classNames(
|
|
{
|
|
[`${prefixClsValue}-sm`]: mergedSize.value === 'small',
|
|
[`${prefixClsValue}-lg`]: mergedSize.value === 'large',
|
|
[`${prefixClsValue}-rtl`]: direction.value === 'rtl',
|
|
[`${prefixClsValue}-borderless`]: !bordered,
|
|
},
|
|
!inputHasPrefixSuffix && getStatusClassNames(prefixClsValue, mergedStatus.value),
|
|
hashId.value,
|
|
)}
|
|
affixWrapperClassName={classNames(
|
|
{
|
|
[`${prefixClsValue}-affix-wrapper-sm`]: mergedSize.value === 'small',
|
|
[`${prefixClsValue}-affix-wrapper-lg`]: mergedSize.value === 'large',
|
|
[`${prefixClsValue}-affix-wrapper-rtl`]: direction.value === 'rtl',
|
|
[`${prefixClsValue}-affix-wrapper-borderless`]: !bordered,
|
|
},
|
|
getStatusClassNames(`${prefixClsValue}-affix-wrapper`, mergedStatus.value, hasFeedback),
|
|
hashId.value,
|
|
)}
|
|
wrapperClassName={classNames(
|
|
{
|
|
[`${prefixClsValue}-group-rtl`]: direction.value === 'rtl',
|
|
},
|
|
hashId.value,
|
|
)}
|
|
groupClassName={classNames(
|
|
{
|
|
[`${prefixClsValue}-group-wrapper-sm`]: mergedSize.value === 'small',
|
|
[`${prefixClsValue}-group-wrapper-lg`]: mergedSize.value === 'large',
|
|
[`${prefixClsValue}-group-wrapper-rtl`]: direction.value === 'rtl',
|
|
},
|
|
getStatusClassNames(`${prefixClsValue}-group-wrapper`, mergedStatus.value, hasFeedback),
|
|
hashId.value,
|
|
)}
|
|
v-slots={{ ...slots, clearIcon }}
|
|
></VcInput>,
|
|
);
|
|
};
|
|
},
|
|
});
|