diff --git a/src/input/__test__/index.test.jsx b/src/input/__test__/index.test.jsx index 722ca4de0..370b36a8a 100644 --- a/src/input/__test__/index.test.jsx +++ b/src/input/__test__/index.test.jsx @@ -76,7 +76,7 @@ describe('Input.vue', async () => { const wrapper = mount(); const closeIcon = wrapper.findComponent(CloseCircleFilledIcon); expect(closeIcon.exists()).toBeTruthy(); - await closeIcon.trigger('click'); + await closeIcon.trigger('touchend'); expect(value.value).toBe(''); expect(handleClear).toBeCalled(); }); @@ -128,6 +128,7 @@ describe('Input.vue', async () => { it(': type=password', async () => { const wrapper = mount(); + expect(wrapper.find('.t-icon-browse-off').exists()).toBeTruthy(); wrapper.find('.t-icon-browse-off').trigger('click'); await wrapper.vm.$nextTick(); expect(wrapper.find('.t-icon-browse').exists()).toBeTruthy(); @@ -140,32 +141,119 @@ describe('Input.vue', async () => { expect(attrDom1.attributes('type')).toBe('password'); }); - it(': onBlur', async () => { - const onBlur = vi.fn(); - const wrapper = mount(); - await nextTick(); - const input = wrapper.find('.t-input__wrap input'); - await input.trigger('blur'); - expect(onBlur).toBeCalled(); + it(': type=password and disabled', async () => { + const wrapper = mount(); + expect(wrapper.find('.t-icon-browse-off').exists()).toBeTruthy(); + wrapper.find('.t-icon-browse-off').trigger('click'); + await wrapper.vm.$nextTick(); + expect(wrapper.find('.t-icon-browse-off').exists()).toBeTruthy(); + await wrapper.setProps({ + disabled: false, + }); + wrapper.find('.t-icon-browse-off').trigger('click'); + await wrapper.vm.$nextTick(); + const attrDom = wrapper.find('input'); + expect(attrDom.attributes('type')).toBe('text'); + wrapper.find('.t-icon-browse').trigger('click'); + await wrapper.vm.$nextTick(); + expect(wrapper.find('.t-icon-browse-off').exists()).toBeTruthy(); + const attrDom1 = wrapper.find('input'); + expect(attrDom1.attributes('type')).toBe('password'); }); - it(': onFocus', async () => { - const onFocus = vi.fn(); - const wrapper = mount(); - const input = wrapper.find('.t-input__wrap input'); - await input.trigger('focus'); - expect(onFocus).toBeCalled(); + it(': autofocus', async () => { + const value = ref('123'); + const wrapper = mount(); + await wrapper.vm.$nextTick(); + expect(wrapper.find('.t-icon-close-circle-filled').exists()).toBeTruthy(); + wrapper.vm.blur(); + await wrapper.vm.$nextTick(); + expect(wrapper.find('.t-icon-close-circle-filled').exists()).toBeFalsy(); + wrapper.vm.focus(); + await wrapper.vm.$nextTick(); + expect(wrapper.find('.t-icon-close-circle-filled').exists()).toBeTruthy(); + }); - it(': onChange', async () => { - const value = ref(''); - const onChange = vi.fn(); - const wrapper = mount(); - const el = wrapper.find('.t-input__wrap input').element; - await simulateEvent(el, '文本', 'input'); - expect(onChange).toBeCalledTimes(1); - expect(onChange).toHaveBeenCalledWith('文本'); + it(': clearTrigger=always', async () => { + const value = ref('123'); + const handleClear = vi.fn(); + const wrapper = mount(); + expect(wrapper.find('.t-icon-close-circle-filled').exists()).toBeTruthy(); + await wrapper.find('.t-icon-close-circle-filled').trigger('touchend'); + expect(value.value).toBe(''); + expect(handleClear).toBeCalled(); }); + + it(': clearTrigger=always and disabled', async () => { + const value = ref('123'); + const handleClear = vi.fn(); + const wrapper = mount(); + expect(wrapper.find('.t-icon-close-circle-filled').exists()).toBeFalsy(); + await wrapper.setProps({ + disabled: false, + }); + expect(wrapper.find('.t-icon-close-circle-filled').exists()).toBeTruthy(); + }); + + it(': clearTrigger=always and readonly', async () => { + const value = ref('123'); + const handleClear = vi.fn(); + const wrapper = mount(); + expect(wrapper.find('.t-icon-close-circle-filled').exists()).toBeFalsy(); + await wrapper.setProps({ + readonly: false, + }); + expect(wrapper.find('.t-icon-close-circle-filled').exists()).toBeTruthy(); + }); + + it(': clearTrigger=focus', async () => { + const value = ref('123'); + const handleClear = vi.fn(); + const wrapper = mount(); + expect(wrapper.find('.t-icon-close-circle-filled').exists()).toBeFalsy(); + wrapper.vm.focus(); + await wrapper.vm.$nextTick(); + expect(wrapper.find('.t-icon-close-circle-filled').exists()).toBeTruthy(); + + await wrapper.find('.t-icon-close-circle-filled').trigger('touchend'); + expect(value.value).toBe(''); + expect(handleClear).toBeCalled(); + }); + + it(': clearTrigger=focus and disabled', async () => { + const value = ref('123'); + const handleClear = vi.fn(); + const wrapper = mount(); + expect(wrapper.find('.t-icon-close-circle-filled').exists()).toBeFalsy(); + wrapper.vm.focus(); + await wrapper.vm.$nextTick(); + expect(wrapper.find('.t-icon-close-circle-filled').exists()).toBeFalsy(); + await wrapper.setProps({ + disabled: false, + }); + wrapper.vm.focus(); + await wrapper.vm.$nextTick(); + expect(wrapper.find('.t-icon-close-circle-filled').exists()).toBeTruthy(); + }); + + it(': clearTrigger=focus and readonly', async () => { + const value = ref('123'); + const handleClear = vi.fn(); + const wrapper = mount(); + expect(wrapper.find('.t-icon-close-circle-filled').exists()).toBeFalsy(); + wrapper.vm.focus(); + await wrapper.vm.$nextTick(); + expect(wrapper.find('.t-icon-close-circle-filled').exists()).toBeFalsy(); + await wrapper.setProps({ + readonly: false, + }); + wrapper.vm.focus(); + await wrapper.vm.$nextTick(); + expect(wrapper.find('.t-icon-close-circle-filled').exists()).toBeTruthy(); + }); + + }); describe('event', async () => { it(': focus && blur', async () => { @@ -200,6 +288,33 @@ describe('Input.vue', async () => { await $input.trigger('compositionend'); expect(onCompositionend).toBeCalled(); }); + + it(': onBlur', async () => { + const onBlur = vi.fn(); + const wrapper = mount(); + await nextTick(); + const input = wrapper.find('.t-input__wrap input'); + await input.trigger('blur'); + expect(onBlur).toBeCalled(); + }); + + it(': onFocus', async () => { + const onFocus = vi.fn(); + const wrapper = mount(); + const input = wrapper.find('.t-input__wrap input'); + await input.trigger('focus'); + expect(onFocus).toBeCalled(); + }); + + it(': onChange', async () => { + const value = ref(''); + const onChange = vi.fn(); + const wrapper = mount(); + const el = wrapper.find('.t-input__wrap input').element; + await simulateEvent(el, '文本', 'input'); + expect(onChange).toBeCalledTimes(1); + expect(onChange).toHaveBeenCalledWith('文本'); + }); }); describe('slots', async () => { diff --git a/src/input/input.en-US.md b/src/input/input.en-US.md index 6a50570ca..c42abd7b6 100644 --- a/src/input/input.en-US.md +++ b/src/input/input.en-US.md @@ -12,6 +12,7 @@ allowInputOverMax | Boolean | false | allow to continue input on value length is autocomplete | String | undefined | attribute of input element, [see here](https://developer.mozilla.org/en-US/docs/Web/HTML/Attributes/autocomplete) | N autofocus | Boolean | false | autofocus on first rendered | N borderless | Boolean | false | input without border | N +clearTrigger | String | always | show clear icon, clicked to clear input value。options: always / focus | N clearable | Boolean | false | show clear icon, clicked to clear input value | N disabled | Boolean | undefined | make input to be disabled | N format | Function | - | input value formatter, `type=number` does not work. if you need to format number, `InputNumber` Component might be better。Typescript:`InputFormatType` `type InputFormatType = (value: InputValue) => string`。[see more ts definition](https://github.com/Tencent/tdesign-mobile-vue/tree/develop/src/input/type.ts) | N @@ -22,7 +23,7 @@ maxlength | String / Number | - | \- | N name | String | - | \- | N placeholder | String | undefined | \- | N prefixIcon | Slot / Function | - | Typescript:`TNode`。[see more ts definition](https://github.com/Tencent/tdesign-mobile-vue/blob/develop/src/common.ts) | N -readonly | Boolean | false | \- | N +readonly | Boolean | undefined | \- | N size | String | small | `deprecated`。options: small/medium。Typescript:`'medium' \| 'small'` | N spellCheck | Boolean | false | attribute of input element, [see here](https://developer.mozilla.org/en-US/docs/Web/HTML/Attributes/spellcheck) | N status | String | undefined | options: default/success/warning/error | N @@ -34,7 +35,7 @@ value | String / Number | - | input value。`v-model` and `v-model:value` is sup defaultValue | String / Number | - | input value。uncontrolled property。Typescript:`InputValue` `type InputValue = string \| number`。[see more ts definition](https://github.com/Tencent/tdesign-mobile-vue/tree/develop/src/input/type.ts) | N onBlur | Function | | Typescript:`(value: InputValue, context: { e: FocusEvent }) => void`
| N onChange | Function | | Typescript:`(value: InputValue, context?: { e?: InputEvent \| MouseEvent \| CompositionEvent; trigger: 'input' \| 'initial' \| 'clear' }) => void`
trigger on input value changed | N -onClear | Function | | Typescript:`(context: { e: MouseEvent }) => void`
| N +onClear | Function | | Typescript:`(context: { e: TouchEvent }) => void`
| N onFocus | Function | | Typescript:`(value: InputValue, context: { e: FocusEvent }) => void`
| N onValidate | Function | | Typescript:`(context: { error?: 'exceed-maximum' \| 'below-minimum' }) => void`
trigger on text length being over max length or max character | N @@ -44,7 +45,7 @@ name | params | description -- | -- | -- blur | `(value: InputValue, context: { e: FocusEvent })` | \- change | `(value: InputValue, context?: { e?: InputEvent \| MouseEvent \| CompositionEvent; trigger: 'input' \| 'initial' \| 'clear' })` | trigger on input value changed -clear | `(context: { e: MouseEvent })` | \- +clear | `(context: { e: TouchEvent })` | \- focus | `(value: InputValue, context: { e: FocusEvent })` | \- validate | `(context: { error?: 'exceed-maximum' \| 'below-minimum' })` | trigger on text length being over max length or max character diff --git a/src/input/input.md b/src/input/input.md index 985a700c1..1a54b6a60 100644 --- a/src/input/input.md +++ b/src/input/input.md @@ -11,6 +11,7 @@ allowInputOverMax | Boolean | false | 超出 `maxlength` 或 `maxcharacter` 之 autocomplete | String | undefined | 是否开启自动填充功能,HTML5 原生属性,[点击查看详情](https://developer.mozilla.org/en-US/docs/Web/HTML/Attributes/autocomplete) | N autofocus | Boolean | false | 自动聚焦 | N borderless | Boolean | false | 是否开启无边框模式 | N +clearTrigger | String | always | 清空图标触发方式,仅在输入框有值时有效。可选项:always / focus | N clearable | Boolean | false | 是否可清空 | N disabled | Boolean | undefined | 是否禁用输入框 | N format | Function | - | 【开发中】指定输入框展示值的格式。TS 类型:`InputFormatType` `type InputFormatType = (value: InputValue) => string`。[详细类型定义](https://github.com/Tencent/tdesign-mobile-vue/tree/develop/src/input/type.ts) | N @@ -21,7 +22,7 @@ maxlength | String / Number | - | 用户最多可以输入的文本长度,一 name | String | - | 名称 | N placeholder | String | undefined | 占位符 | N prefixIcon | Slot / Function | - | 组件前置图标。TS 类型:`TNode`。[通用类型定义](https://github.com/Tencent/tdesign-mobile-vue/blob/develop/src/common.ts) | N -readonly | Boolean | false | 只读状态 | N +readonly | Boolean | undefined | 只读状态 | N size | String | small | 已废弃。输入框尺寸。可选项:small/medium。TS 类型:`'medium' \| 'small'` | N spellCheck | Boolean | false | 是否开启拼写检查,HTML5 原生属性,[点击查看详情](https://developer.mozilla.org/en-US/docs/Web/HTML/Attributes/spellcheck) | N status | String | undefined | 输入框状态。默认情况会由组件内部根据实际情况呈现,如果文本过长引起的状态变化。可选项:default/success/warning/error | N @@ -33,7 +34,7 @@ value | String / Number | - | 输入框的值。支持语法糖 `v-model` 或 `v defaultValue | String / Number | - | 输入框的值。非受控属性。TS 类型:`InputValue` `type InputValue = string \| number`。[详细类型定义](https://github.com/Tencent/tdesign-mobile-vue/tree/develop/src/input/type.ts) | N onBlur | Function | | TS 类型:`(value: InputValue, context: { e: FocusEvent }) => void`
失去焦点时触发 | N onChange | Function | | TS 类型:`(value: InputValue, context?: { e?: InputEvent \| MouseEvent \| CompositionEvent; trigger: 'input' \| 'initial' \| 'clear' }) => void`
输入框值发生变化时触发。`trigger=initial` 表示传入的数据不符合预期,组件自动处理后触发 change 告知父组件。如:初始值长度超过 `maxlength` 限制 | N -onClear | Function | | TS 类型:`(context: { e: MouseEvent }) => void`
清空按钮点击时触发 | N +onClear | Function | | TS 类型:`(context: { e: TouchEvent }) => void`
清空按钮点击时触发 | N onFocus | Function | | TS 类型:`(value: InputValue, context: { e: FocusEvent }) => void`
获得焦点时触发 | N onValidate | Function | | TS 类型:`(context: { error?: 'exceed-maximum' \| 'below-minimum' }) => void`
【暂不支持】字数超出限制时触发 | N @@ -43,7 +44,7 @@ onValidate | Function | | TS 类型:`(context: { error?: 'exceed-maximum' \| -- | -- | -- blur | `(value: InputValue, context: { e: FocusEvent })` | 失去焦点时触发 change | `(value: InputValue, context?: { e?: InputEvent \| MouseEvent \| CompositionEvent; trigger: 'input' \| 'initial' \| 'clear' })` | 输入框值发生变化时触发。`trigger=initial` 表示传入的数据不符合预期,组件自动处理后触发 change 告知父组件。如:初始值长度超过 `maxlength` 限制 -clear | `(context: { e: MouseEvent })` | 清空按钮点击时触发 +clear | `(context: { e: TouchEvent })` | 清空按钮点击时触发 focus | `(value: InputValue, context: { e: FocusEvent })` | 获得焦点时触发 validate | `(context: { error?: 'exceed-maximum' \| 'below-minimum' })` | 【暂不支持】字数超出限制时触发 diff --git a/src/input/input.tsx b/src/input/input.tsx index 019f249fd..6a6693bad 100644 --- a/src/input/input.tsx +++ b/src/input/input.tsx @@ -1,10 +1,9 @@ -import { PropType, ref, computed, defineComponent, toRefs, nextTick, watch } from 'vue'; +import { PropType, ref, computed, defineComponent, nextTick, watch } from 'vue'; import { BrowseIcon as TBrowseIcon, BrowseOffIcon as TBrowseOffIcon, CloseCircleFilledIcon as TCloseCircleFilledIcon, } from 'tdesign-icons-vue-next'; -import { useFocus } from '@vueuse/core'; import config from '../config'; import InputProps from './props'; import { InputValue, TdInputProps } from './type'; @@ -35,17 +34,16 @@ export default defineComponent({ }, }, setup(props, context) { - const readerTNodeJSX = useTNodeJSX(); + const renderTNodeJSX = useTNodeJSX(); const inputClass = usePrefixClass('input'); const isDisabled = useFormDisabled(); const inputRef = ref(); - const { autofocus } = toRefs(props); const [innerValue] = useDefault(props, context.emit, 'value', 'change'); const status = props.status || 'default'; const renderType = ref(props.type); - const { focused } = useFocus(inputRef, { initialValue: props.autofocus }); + const focused = ref(false); const inputClasses = computed(() => [ `${inputClass.value}__control`, @@ -64,6 +62,14 @@ export default defineComponent({ [`${inputClass.value}--border`]: !props.borderless, }, ]); + const showClear = computed(() => { + if (isDisabled.value || props.readonly === true) return false; + + if (props.clearable && innerValue.value && innerValue.value.length > 0) { + return props.clearTrigger === 'always' || (props.clearTrigger === 'focus' && focused.value); + } + return false; + }); const setInputValue = (v: InputValue = '') => { const input = inputRef.value as HTMLInputElement; @@ -102,24 +108,30 @@ export default defineComponent({ const focus = () => { focused.value = true; + inputRef.value?.focus(); }; const blur = () => { focused.value = false; - // inputRef.value?.blur(); + inputRef.value?.blur(); }; extendAPI({ focus, blur }); - const handleClear = (e: MouseEvent) => { + const handleClear = (e: TouchEvent) => { + e.preventDefault(); innerValue.value = ''; - focused.value = true; + focus(); props.onClear?.({ e }); }; + const handleFocus = (e: FocusEvent) => { + focused.value = true; props.onFocus?.(innerValue.value, { e }); }; + const handleBlur = (e: FocusEvent) => { + focused.value = false; props.onBlur?.(innerValue.value, { e }); }; @@ -128,16 +140,22 @@ export default defineComponent({ }; const handlePwdIconClick = () => { + if (isDisabled.value) return; + renderType.value = renderType.value === 'password' ? 'text' : 'password'; }; - watch(autofocus, (autofocus, prevAutofocus) => { - if (autofocus === true) { - nextTick(() => { - focused.value = true; - }); - } - }); + watch( + () => props.autofocus, + (v) => { + if (v === true) { + nextTick(() => { + focus(); + }); + } + }, + { immediate: true }, + ); watch( () => props.type, @@ -148,9 +166,9 @@ export default defineComponent({ ); return () => { - const readerPrefix = () => { - const prefixIcon = readerTNodeJSX('prefixIcon'); - const label = readerTNodeJSX('label'); + const renderPrefix = () => { + const prefixIcon = renderTNodeJSX('prefixIcon'); + const label = renderTNodeJSX('label'); return (
@@ -159,26 +177,27 @@ export default defineComponent({
); }; - const readerClearable = () => { - if (props.clearable && innerValue.value && innerValue.value.length > 0) { + const renderClearable = () => { + if (showClear.value) { return ( -
+
); } + return null; }; - const readerSuffix = () => { - const suffix = readerTNodeJSX('suffix'); + const renderSuffix = () => { + const suffix = renderTNodeJSX('suffix'); if (!suffix) { return null; } return
{suffix}
; }; - const readerSuffixIcon = () => { - let suffixIcon = readerTNodeJSX('suffixIcon'); + const renderSuffixIcon = () => { + let suffixIcon = renderTNodeJSX('suffixIcon'); if (props.type === 'password') { if (renderType.value === 'password') { suffixIcon = ; @@ -193,8 +212,8 @@ export default defineComponent({ return
{suffixIcon}
; }; - const readerTips = () => { - const tips = readerTNodeJSX('tips'); + const renderTips = () => { + const tips = renderTNodeJSX('tips'); if (!tips) { return null; } @@ -203,7 +222,7 @@ export default defineComponent({ return (
- {readerPrefix()} + {renderPrefix()}
- {readerClearable()} - {readerSuffix()} - {readerSuffixIcon()} + {renderClearable()} + {renderSuffix()} + {renderSuffixIcon()}
- {readerTips()} + {renderTips()}
); diff --git a/src/input/props.ts b/src/input/props.ts index 86d3be5ba..d63a0bf1a 100644 --- a/src/input/props.ts +++ b/src/input/props.ts @@ -28,6 +28,15 @@ export default { autofocus: Boolean, /** 是否开启无边框模式 */ borderless: Boolean, + /** 清空图标触发方式,仅在输入框有值时有效 */ + clearTrigger: { + type: String as PropType, + default: 'always' as TdInputProps['clearTrigger'], + validator(val: TdInputProps['clearTrigger']): boolean { + if (!val) return true; + return ['always', 'focus'].includes(val); + }, + }, /** 是否可清空 */ clearable: Boolean, /** 是否禁用输入框 */ @@ -75,7 +84,10 @@ export default { type: Function as PropType, }, /** 只读状态 */ - readonly: Boolean, + readonly: { + type: Boolean, + default: undefined, + }, /** 是否开启拼写检查,HTML5 原生属性,[点击查看详情](https://developer.mozilla.org/en-US/docs/Web/HTML/Attributes/spellcheck) */ spellCheck: Boolean, /** 输入框状态。默认情况会由组件内部根据实际情况呈现,如果文本过长引起的状态变化 */ diff --git a/src/input/type.ts b/src/input/type.ts index 20fb58c27..b90968778 100644 --- a/src/input/type.ts +++ b/src/input/type.ts @@ -31,6 +31,11 @@ export interface TdInputProps { * @default false */ borderless?: boolean; + /** + * 清空图标触发方式,仅在输入框有值时有效 + * @default always + */ + clearTrigger?: 'always' | 'focus'; /** * 是否可清空 * @default false @@ -130,7 +135,7 @@ export interface TdInputProps { /** * 清空按钮点击时触发 */ - onClear?: (context: { e: MouseEvent }) => void; + onClear?: (context: { e: TouchEvent }) => void; /** * 获得焦点时触发 */