2020-10-15 01:48:43 +00:00
|
|
|
import { App, defineComponent, inject, PropType, VNodeTypes } from 'vue';
|
2020-08-31 08:53:19 +00:00
|
|
|
import classNames from '../_util/classNames';
|
2020-03-07 11:45:13 +00:00
|
|
|
import omit from 'omit.js';
|
|
|
|
import PropTypes from '../_util/vue-types';
|
|
|
|
import VcMentions from '../vc-mentions';
|
|
|
|
import { mentionsProps } from '../vc-mentions/src/mentionsProps';
|
|
|
|
import Spin from '../spin';
|
|
|
|
import BaseMixin from '../_util/BaseMixin';
|
2020-09-30 02:47:18 +00:00
|
|
|
import { defaultConfigProvider } from '../config-provider';
|
2020-08-01 08:37:06 +00:00
|
|
|
import { getOptionProps, getComponent, getSlot } from '../_util/props-util';
|
2020-10-15 01:48:43 +00:00
|
|
|
import { RenderEmptyHandler } from '../config-provider/renderEmpty';
|
2020-03-07 11:45:13 +00:00
|
|
|
|
|
|
|
const { Option } = VcMentions;
|
|
|
|
|
2020-10-15 01:48:43 +00:00
|
|
|
interface MentionsConfig {
|
|
|
|
prefix?: string | string[];
|
|
|
|
split?: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface OptionProps {
|
|
|
|
value: string;
|
|
|
|
disabled: boolean;
|
|
|
|
children: VNodeTypes;
|
|
|
|
[key: string]: any;
|
|
|
|
}
|
|
|
|
|
2020-03-07 11:45:13 +00:00
|
|
|
function loadingFilterOption() {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-10-19 08:43:10 +00:00
|
|
|
function getMentions(value = '', config: MentionsConfig) {
|
2020-03-07 11:45:13 +00:00
|
|
|
const { prefix = '@', split = ' ' } = config || {};
|
|
|
|
const prefixList = Array.isArray(prefix) ? prefix : [prefix];
|
|
|
|
|
|
|
|
return value
|
|
|
|
.split(split)
|
|
|
|
.map((str = '') => {
|
|
|
|
let hitPrefix = null;
|
|
|
|
|
|
|
|
prefixList.some(prefixStr => {
|
|
|
|
const startStr = str.slice(0, prefixStr.length);
|
|
|
|
if (startStr === prefixStr) {
|
|
|
|
hitPrefix = prefixStr;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
|
|
|
|
if (hitPrefix !== null) {
|
|
|
|
return {
|
|
|
|
prefix: hitPrefix,
|
|
|
|
value: str.slice(hitPrefix.length),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
})
|
|
|
|
.filter(entity => !!entity && !!entity.value);
|
|
|
|
}
|
|
|
|
|
2020-10-15 01:48:43 +00:00
|
|
|
const Mentions = defineComponent({
|
2020-03-07 11:45:13 +00:00
|
|
|
name: 'AMentions',
|
|
|
|
mixins: [BaseMixin],
|
|
|
|
inheritAttrs: false,
|
|
|
|
Option: { ...Option, name: 'AMentionsOption' },
|
|
|
|
getMentions,
|
|
|
|
props: {
|
|
|
|
...mentionsProps,
|
2020-10-10 10:16:28 +00:00
|
|
|
loading: PropTypes.looseBool,
|
2020-10-15 01:48:43 +00:00
|
|
|
onFocus: {
|
|
|
|
type: Function as PropType<(e: FocusEvent) => void>,
|
|
|
|
},
|
|
|
|
onBlur: {
|
|
|
|
type: Function as PropType<(e: FocusEvent) => void>,
|
|
|
|
},
|
|
|
|
onSelect: {
|
|
|
|
type: Function as PropType<(option: OptionProps, prefix: string) => void>,
|
|
|
|
},
|
|
|
|
onChange: {
|
|
|
|
type: Function as PropType<(text: string) => void>,
|
|
|
|
},
|
2020-03-07 11:45:13 +00:00
|
|
|
},
|
2020-10-15 01:48:43 +00:00
|
|
|
emits: ['update:value', 'change', 'focus', 'blur', 'select'],
|
2020-06-20 02:08:45 +00:00
|
|
|
setup() {
|
|
|
|
return {
|
2020-09-30 02:47:18 +00:00
|
|
|
configProvider: inject('configProvider', defaultConfigProvider),
|
2020-06-20 02:08:45 +00:00
|
|
|
};
|
2020-03-07 11:45:13 +00:00
|
|
|
},
|
|
|
|
data() {
|
|
|
|
return {
|
|
|
|
focused: false,
|
|
|
|
};
|
|
|
|
},
|
|
|
|
mounted() {
|
2020-06-22 09:02:11 +00:00
|
|
|
this.$nextTick(() => {
|
2020-08-01 08:37:06 +00:00
|
|
|
if (process.env.NODE_ENV === 'test') {
|
|
|
|
if (this.autofocus) {
|
|
|
|
this.focus();
|
|
|
|
}
|
2020-03-07 11:45:13 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
methods: {
|
2020-10-15 01:48:43 +00:00
|
|
|
handleFocus(e: FocusEvent) {
|
|
|
|
this.$emit('focus', e);
|
2020-03-07 11:45:13 +00:00
|
|
|
this.setState({
|
|
|
|
focused: true,
|
|
|
|
});
|
|
|
|
},
|
2020-10-15 01:48:43 +00:00
|
|
|
handleBlur(e: FocusEvent) {
|
|
|
|
this.$emit('blur', e);
|
2020-03-07 11:45:13 +00:00
|
|
|
this.setState({
|
|
|
|
focused: false,
|
|
|
|
});
|
|
|
|
},
|
2020-10-15 01:48:43 +00:00
|
|
|
handleSelect(...args: [OptionProps, string]) {
|
2020-03-07 11:45:13 +00:00
|
|
|
this.$emit('select', ...args);
|
|
|
|
this.setState({
|
|
|
|
focused: true,
|
|
|
|
});
|
|
|
|
},
|
2020-10-15 01:48:43 +00:00
|
|
|
handleChange(val: string) {
|
2020-06-20 02:08:45 +00:00
|
|
|
this.$emit('update:value', val);
|
2020-08-14 10:11:30 +00:00
|
|
|
this.$emit('change', val);
|
2020-03-07 11:45:13 +00:00
|
|
|
},
|
2020-10-15 01:48:43 +00:00
|
|
|
getNotFoundContent(renderEmpty: RenderEmptyHandler) {
|
2020-06-20 02:08:45 +00:00
|
|
|
const notFoundContent = getComponent(this, 'notFoundContent');
|
2020-03-07 11:45:13 +00:00
|
|
|
if (notFoundContent !== undefined) {
|
|
|
|
return notFoundContent;
|
|
|
|
}
|
|
|
|
|
2020-06-20 02:08:45 +00:00
|
|
|
return renderEmpty('Select');
|
2020-03-07 11:45:13 +00:00
|
|
|
},
|
|
|
|
getOptions() {
|
|
|
|
const { loading } = this.$props;
|
2020-08-01 08:37:06 +00:00
|
|
|
const children = getSlot(this);
|
2020-03-07 11:45:13 +00:00
|
|
|
|
|
|
|
if (loading) {
|
|
|
|
return (
|
|
|
|
<Option value="ANTD_SEARCHING" disabled>
|
|
|
|
<Spin size="small" />
|
|
|
|
</Option>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return children;
|
|
|
|
},
|
|
|
|
getFilterOption() {
|
|
|
|
const { filterOption, loading } = this.$props;
|
|
|
|
if (loading) {
|
|
|
|
return loadingFilterOption;
|
|
|
|
}
|
|
|
|
return filterOption;
|
|
|
|
},
|
|
|
|
focus() {
|
2020-10-15 01:48:43 +00:00
|
|
|
(this.$refs.vcMentions as any).focus();
|
2020-03-07 11:45:13 +00:00
|
|
|
},
|
|
|
|
blur() {
|
2020-10-15 01:48:43 +00:00
|
|
|
(this.$refs.vcMentions as any).blur();
|
2020-03-07 11:45:13 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
render() {
|
|
|
|
const { focused } = this.$data;
|
|
|
|
const { getPrefixCls, renderEmpty } = this.configProvider;
|
|
|
|
const {
|
|
|
|
prefixCls: customizePrefixCls,
|
|
|
|
disabled,
|
|
|
|
getPopupContainer,
|
|
|
|
...restProps
|
2020-10-15 01:48:43 +00:00
|
|
|
} = getOptionProps(this) as any;
|
2020-06-22 09:02:11 +00:00
|
|
|
const { class: className, ...otherAttrs } = this.$attrs;
|
2020-03-07 11:45:13 +00:00
|
|
|
const prefixCls = getPrefixCls('mentions', customizePrefixCls);
|
2020-08-01 08:37:06 +00:00
|
|
|
const otherProps = omit(restProps, ['loading', 'onUpdate:value']);
|
2020-03-07 11:45:13 +00:00
|
|
|
|
2020-06-22 09:02:11 +00:00
|
|
|
const mergedClassName = classNames(className, {
|
2020-03-07 11:45:13 +00:00
|
|
|
[`${prefixCls}-disabled`]: disabled,
|
|
|
|
[`${prefixCls}-focused`]: focused,
|
|
|
|
});
|
|
|
|
|
|
|
|
const mentionsProps = {
|
2020-06-20 02:08:45 +00:00
|
|
|
prefixCls,
|
|
|
|
notFoundContent: this.getNotFoundContent(renderEmpty),
|
|
|
|
...otherProps,
|
|
|
|
disabled,
|
|
|
|
filterOption: this.getFilterOption(),
|
|
|
|
getPopupContainer,
|
|
|
|
children: this.getOptions(),
|
2020-03-07 11:45:13 +00:00
|
|
|
class: mergedClassName,
|
2020-06-20 02:08:45 +00:00
|
|
|
rows: 1,
|
2020-06-22 09:02:11 +00:00
|
|
|
...otherAttrs,
|
2020-08-01 08:37:06 +00:00
|
|
|
onChange: this.handleChange,
|
|
|
|
onSelect: this.handleSelect,
|
|
|
|
onFocus: this.handleFocus,
|
|
|
|
onBlur: this.handleBlur,
|
2020-03-07 11:45:13 +00:00
|
|
|
ref: 'vcMentions',
|
|
|
|
};
|
|
|
|
|
|
|
|
return <VcMentions {...mentionsProps} />;
|
|
|
|
},
|
2020-10-15 01:48:43 +00:00
|
|
|
});
|
2020-03-07 11:45:13 +00:00
|
|
|
|
|
|
|
/* istanbul ignore next */
|
2020-10-15 01:48:43 +00:00
|
|
|
Mentions.install = function(app: App) {
|
2020-06-20 02:08:45 +00:00
|
|
|
app.component(Mentions.name, Mentions);
|
|
|
|
app.component(Mentions.Option.name, Mentions.Option);
|
2020-10-13 11:14:56 +00:00
|
|
|
return app;
|
2020-03-07 11:45:13 +00:00
|
|
|
};
|
|
|
|
|
2020-10-24 14:09:54 +00:00
|
|
|
export default Mentions as typeof Mentions & {
|
|
|
|
readonly Option: typeof Option;
|
|
|
|
};
|