104 lines
3.3 KiB
Vue
104 lines
3.3 KiB
Vue
import classNames from '../_util/classNames';
|
|
import { isValidElement } from '../_util/props-util';
|
|
import { cloneElement } from '../_util/vnode';
|
|
import Input from './Input';
|
|
import EyeOutlined from '@ant-design/icons-vue/EyeOutlined';
|
|
import EyeInvisibleOutlined from '@ant-design/icons-vue/EyeInvisibleOutlined';
|
|
import type { InputProps } from './inputProps';
|
|
import inputProps from './inputProps';
|
|
import PropTypes from '../_util/vue-types';
|
|
import BaseMixin from '../_util/BaseMixin';
|
|
import { computed, defineComponent, ref } from 'vue';
|
|
import useConfigInject from '../_util/hooks/useConfigInject';
|
|
import omit from '../_util/omit';
|
|
|
|
const ActionMap = {
|
|
click: 'onClick',
|
|
hover: 'onMouseover',
|
|
};
|
|
const defaultIconRender = (visible: boolean) =>
|
|
visible ? <EyeOutlined /> : <EyeInvisibleOutlined />;
|
|
export default defineComponent({
|
|
name: 'AInputPassword',
|
|
mixins: [BaseMixin],
|
|
inheritAttrs: false,
|
|
props: {
|
|
...inputProps,
|
|
prefixCls: PropTypes.string,
|
|
inputPrefixCls: PropTypes.string,
|
|
action: PropTypes.string.def('click'),
|
|
visibilityToggle: PropTypes.looseBool.def(true),
|
|
iconRender: PropTypes.func,
|
|
},
|
|
setup(props, { slots, attrs, expose }) {
|
|
const visible = ref(false);
|
|
const onVisibleChange = () => {
|
|
const { disabled } = props;
|
|
if (disabled) {
|
|
return;
|
|
}
|
|
visible.value = !visible.value;
|
|
};
|
|
const inputRef = ref();
|
|
const focus = () => {
|
|
inputRef.value?.focus();
|
|
};
|
|
const blur = () => {
|
|
inputRef.value?.blur();
|
|
};
|
|
expose({
|
|
focus,
|
|
blur,
|
|
});
|
|
const getIcon = (prefixCls: string) => {
|
|
const { action, iconRender = slots.iconRender || defaultIconRender } = props;
|
|
const iconTrigger = ActionMap[action!] || '';
|
|
const icon = iconRender(visible.value);
|
|
const iconProps = {
|
|
[iconTrigger]: onVisibleChange,
|
|
class: `${prefixCls}-icon`,
|
|
key: 'passwordIcon',
|
|
onMousedown: (e: MouseEvent) => {
|
|
// Prevent focused state lost
|
|
// https://github.com/ant-design/ant-design/issues/15173
|
|
e.preventDefault();
|
|
},
|
|
onMouseup: (e: MouseEvent) => {
|
|
// Prevent caret position change
|
|
// https://github.com/ant-design/ant-design/issues/23524
|
|
e.preventDefault();
|
|
},
|
|
};
|
|
return cloneElement(isValidElement(icon) ? icon : <span>{icon}</span>, iconProps);
|
|
};
|
|
const { prefixCls, getPrefixCls } = useConfigInject('input-password', props);
|
|
const inputPrefixCls = computed(() => getPrefixCls('input', props.inputPrefixCls));
|
|
const renderPassword = () => {
|
|
const { size, visibilityToggle, ...restProps } = props;
|
|
|
|
const suffixIcon = visibilityToggle && getIcon(prefixCls.value);
|
|
const inputClassName = classNames(prefixCls.value, attrs.class, {
|
|
[`${prefixCls.value}-${size}`]: !!size,
|
|
});
|
|
|
|
const omittedProps = {
|
|
...omit(restProps, ['suffix', 'iconRender', 'action']),
|
|
...attrs,
|
|
type: visible.value ? 'text' : 'password',
|
|
class: inputClassName,
|
|
prefixCls: inputPrefixCls.value,
|
|
suffix: suffixIcon,
|
|
} as InputProps;
|
|
|
|
if (size) {
|
|
omittedProps.size = size;
|
|
}
|
|
|
|
return <Input ref={inputRef} {...omittedProps} v-slots={slots} />;
|
|
};
|
|
return () => {
|
|
return renderPassword();
|
|
};
|
|
},
|
|
});
|