fix: mentions keydown keyup not work

pull/2682/head
tangjinzhou 2020-08-01 16:37:06 +08:00
parent 1eb76e2300
commit 94d355aba7
5 changed files with 55 additions and 63 deletions

@ -1 +1 @@
Subproject commit 4b4c51535a790c0b06818d5d17f973148d8e65db Subproject commit d572203fb82bed45597abcab8ba7b78796962580

View File

@ -1,7 +1,7 @@
import { mount } from '@vue/test-utils'; import { mount } from '@vue/test-utils';
import * as Vue from 'vue';
import Mentions from '..'; import Mentions from '..';
import focusTest from '../../../tests/shared/focusTest'; import focusTest from '../../../tests/shared/focusTest';
import { sleep } from '../../../tests/utils';
const { getMentions } = Mentions; const { getMentions } = Mentions;
@ -18,12 +18,8 @@ function triggerInput(wrapper, text = '') {
} }
describe('Mentions', () => { describe('Mentions', () => {
beforeAll(() => { beforeEach(() => {
jest.useFakeTimers(); document.body.innerHTML = '';
});
afterAll(() => {
jest.useRealTimers();
}); });
it('getMentions', () => { it('getMentions', () => {
@ -40,50 +36,43 @@ describe('Mentions', () => {
]); ]);
}); });
it('focus', () => { fit('focus', async () => {
const onFocus = jest.fn(); const onFocus = jest.fn();
const onBlur = jest.fn(); const onBlur = jest.fn();
const wrapper = mount({ const wrapper = mount(
render() { {
return <Mentions onFocus={onFocus} onBlur={onBlur} />; render() {
return <Mentions onFocus={onFocus} onBlur={onBlur} />;
},
}, },
}); { sync: false, attachTo: 'body' },
);
await sleep();
wrapper.find('textarea').trigger('focus'); wrapper.find('textarea').trigger('focus');
await sleep();
expect(wrapper.find('.ant-mentions').classes('ant-mentions-focused')).toBeTruthy(); expect(wrapper.find('.ant-mentions').classes('ant-mentions-focused')).toBeTruthy();
expect(onFocus).toHaveBeenCalled(); expect(onFocus).toHaveBeenCalled();
wrapper.find('textarea').trigger('blur'); wrapper.find('textarea').trigger('blur');
jest.runAllTimers(); await sleep(500);
expect(wrapper.classes()).not.toContain('ant-mentions-focused'); expect(wrapper.classes()).not.toContain('ant-mentions-focused');
expect(onBlur).toHaveBeenCalled(); expect(onBlur).toHaveBeenCalled();
}); });
it('loading', done => { it('loading', async () => {
const wrapper = mount( const wrapper = mount(
{ {
render() { render() {
return <Mentions loading />; return <Mentions loading />;
}, },
}, },
{ sync: false }, { sync: false, attachTo: 'body' },
); );
await sleep(500);
triggerInput(wrapper, '@'); triggerInput(wrapper, '@');
Vue.nextTick(() => { await sleep(500);
mount( expect($$('.ant-mentions-dropdown-menu-item').length).toBeTruthy();
{ expect($$('.ant-spin')).toBeTruthy();
render() {
return wrapper.find({ name: 'Trigger' }).vm.getComponent();
},
},
{ sync: false },
);
Vue.nextTick(() => {
expect($$('.ant-mentions-dropdown-menu-item').length).toBeTruthy();
expect($$('.ant-spin')).toBeTruthy();
done();
});
});
}); });
focusTest(Mentions); focusTest(Mentions);

View File

@ -7,7 +7,7 @@ import { mentionsProps } from '../vc-mentions/src/mentionsProps';
import Spin from '../spin'; import Spin from '../spin';
import BaseMixin from '../_util/BaseMixin'; import BaseMixin from '../_util/BaseMixin';
import { ConfigConsumerProps } from '../config-provider'; import { ConfigConsumerProps } from '../config-provider';
import { getOptionProps, getComponent, filterEmpty, getSlot } from '../_util/props-util'; import { getOptionProps, getComponent, getSlot } from '../_util/props-util';
const { Option } = VcMentions; const { Option } = VcMentions;
@ -53,6 +53,11 @@ const Mentions = {
props: { props: {
...mentionsProps, ...mentionsProps,
loading: PropTypes.bool, loading: PropTypes.bool,
onFocus: PropTypes.func,
onBlur: PropTypes.func,
onSelect: PropTypes.func,
onChange: PropTypes.func,
'onUpdate:value': PropTypes.func,
}, },
setup() { setup() {
return { return {
@ -66,31 +71,33 @@ const Mentions = {
}, },
mounted() { mounted() {
this.$nextTick(() => { this.$nextTick(() => {
if (this.autofocus) { if (process.env.NODE_ENV === 'test') {
this.focus(); if (this.autofocus) {
this.focus();
}
} }
}); });
}, },
methods: { methods: {
onFocus(...args) { handleFocus(...args) {
this.$emit('focus', ...args); this.$emit('focus', ...args);
this.setState({ this.setState({
focused: true, focused: true,
}); });
}, },
onBlur(...args) { handleBlur(...args) {
this.$emit('blur', ...args); this.$emit('blur', ...args);
this.setState({ this.setState({
focused: false, focused: false,
}); });
}, },
onSelect(...args) { handleSelect(...args) {
this.$emit('select', ...args); this.$emit('select', ...args);
this.setState({ this.setState({
focused: true, focused: true,
}); });
}, },
onChange(val) { handleChange(val) {
this.$emit('change', val); this.$emit('change', val);
this.$emit('update:value', val); this.$emit('update:value', val);
}, },
@ -104,7 +111,7 @@ const Mentions = {
}, },
getOptions() { getOptions() {
const { loading } = this.$props; const { loading } = this.$props;
const children = filterEmpty(getSlot(this) || []); const children = getSlot(this);
if (loading) { if (loading) {
return ( return (
@ -140,7 +147,7 @@ const Mentions = {
} = getOptionProps(this); } = getOptionProps(this);
const { class: className, ...otherAttrs } = this.$attrs; const { class: className, ...otherAttrs } = this.$attrs;
const prefixCls = getPrefixCls('mentions', customizePrefixCls); const prefixCls = getPrefixCls('mentions', customizePrefixCls);
const otherProps = omit(restProps, ['loading']); const otherProps = omit(restProps, ['loading', 'onUpdate:value']);
const mergedClassName = classNames(className, { const mergedClassName = classNames(className, {
[`${prefixCls}-disabled`]: disabled, [`${prefixCls}-disabled`]: disabled,
@ -158,10 +165,10 @@ const Mentions = {
class: mergedClassName, class: mergedClassName,
rows: 1, rows: 1,
...otherAttrs, ...otherAttrs,
onChange: this.onChange, onChange: this.handleChange,
onSelect: this.onSelect, onSelect: this.handleSelect,
onFocus: this.onFocus, onFocus: this.handleFocus,
onBlur: this.onBlur, onBlur: this.handleBlur,
ref: 'vcMentions', ref: 'vcMentions',
}; };

View File

@ -249,7 +249,6 @@ const Mentions = {
prefixCls, prefixCls,
placement, placement,
transitionName, transitionName,
autofocus,
notFoundContent, notFoundContent,
getPopupContainer, getPopupContainer,
...restProps ...restProps
@ -268,24 +267,21 @@ const Mentions = {
]); ]);
const options = measuring ? this.getOptions() : []; const options = measuring ? this.getOptions() : [];
const textareaProps = {
...inputProps,
...otherAttrs,
onChange: noop,
onSelect: noop,
value,
onInput: this.onChange,
onBlur: this.onInputBlur,
onKeydown: this.onKeyDown,
onKeyup: this.onKeyUp,
onFocus: this.onInputFocus,
};
return ( return (
<div class={classNames(prefixCls, className)} style={style}> <div class={classNames(prefixCls, className)} style={style}>
<textarea <textarea ref="textarea" {...textareaProps} />
ref="textarea"
{...{
...inputProps,
...otherAttrs,
onChange: noop,
onSelect: noop,
}}
value={value}
onInput={this.onChange}
onBlur={this.onInputBlur}
onKeyDown={this.onKeyDown}
onKeyUp={this.onKeyUp}
onFocus={this.onInputFocus}
/>
{measuring && ( {measuring && (
<div ref="measure" class={`${prefixCls}-measure`}> <div ref="measure" class={`${prefixCls}-measure`}>
{value.slice(0, measureLocation)} {value.slice(0, measureLocation)}

View File

@ -4,7 +4,7 @@
</div> </div>
</template> </template>
<script> <script>
import demo from '../antdv-demo/docs/modal/demo/index'; import demo from '../antdv-demo/docs/mentions/demo/index';
export default { export default {
components: { components: {