300 lines
		
	
	
		
			8.4 KiB
		
	
	
	
		
			Vue
		
	
	
			
		
		
	
	
			300 lines
		
	
	
		
			8.4 KiB
		
	
	
	
		
			Vue
		
	
	
| /**
 | |
|  * Cursor rule:
 | |
|  * 1. Only `showSearch` enabled
 | |
|  * 2. Only `open` is `true`
 | |
|  * 3. When typing, set `open` to `true` which hit rule of 2
 | |
|  *
 | |
|  * Accessibility:
 | |
|  * - https://www.w3.org/TR/wai-aria-practices/examples/combobox/aria1.1pattern/listbox-combo.html
 | |
|  */
 | |
| 
 | |
| import KeyCode from '../../_util/KeyCode';
 | |
| import MultipleSelector from './MultipleSelector';
 | |
| import SingleSelector from './SingleSelector';
 | |
| import { LabelValueType, RawValueType, CustomTagProps } from '../interface/generator';
 | |
| import { RenderNode, Mode } from '../interface';
 | |
| import useLock from '../hooks/useLock';
 | |
| import { defineComponent, VNodeChild } from 'vue';
 | |
| import createRef, { RefObject } from '../../_util/createRef';
 | |
| import PropTypes from '../../_util/vue-types';
 | |
| import { VueNode } from '../../_util/type';
 | |
| 
 | |
| export interface InnerSelectorProps {
 | |
|   prefixCls: string;
 | |
|   id: string;
 | |
|   mode: Mode;
 | |
|   inputRef: RefObject;
 | |
|   placeholder?: VNodeChild;
 | |
|   disabled?: boolean;
 | |
|   autofocus?: boolean;
 | |
|   autocomplete?: string;
 | |
|   values: LabelValueType[];
 | |
|   showSearch?: boolean;
 | |
|   searchValue: string;
 | |
|   accessibilityIndex: number;
 | |
|   open: boolean;
 | |
|   tabindex?: number | string;
 | |
|   onInputKeyDown: EventHandlerNonNull;
 | |
|   onInputMouseDown: EventHandlerNonNull;
 | |
|   onInputChange: EventHandlerNonNull;
 | |
|   onInputPaste: EventHandlerNonNull;
 | |
|   onInputCompositionStart: EventHandlerNonNull;
 | |
|   onInputCompositionEnd: EventHandlerNonNull;
 | |
| }
 | |
| 
 | |
| export interface SelectorProps {
 | |
|   id: string;
 | |
|   prefixCls: string;
 | |
|   showSearch?: boolean;
 | |
|   open: boolean;
 | |
|   /** Display in the Selector value, it's not same as `value` prop */
 | |
|   values: LabelValueType[];
 | |
|   multiple: boolean;
 | |
|   mode: Mode;
 | |
|   searchValue: string;
 | |
|   activeValue: string;
 | |
|   inputElement: VueNode;
 | |
| 
 | |
|   autofocus?: boolean;
 | |
|   accessibilityIndex: number;
 | |
|   tabindex?: number | string;
 | |
|   disabled?: boolean;
 | |
|   placeholder?: VNodeChild;
 | |
|   removeIcon?: RenderNode;
 | |
| 
 | |
|   // Tags
 | |
|   maxTagCount?: number;
 | |
|   maxTagTextLength?: number;
 | |
|   maxTagPlaceholder?: VNodeChild;
 | |
|   tagRender?: (props: CustomTagProps) => VNodeChild;
 | |
| 
 | |
|   /** Check if `tokenSeparators` contains `\n` or `\r\n` */
 | |
|   tokenWithEnter?: boolean;
 | |
| 
 | |
|   // Motion
 | |
|   choiceTransitionName?: string;
 | |
| 
 | |
|   onToggleOpen: (open?: boolean) => void;
 | |
|   /** `onSearch` returns go next step boolean to check if need do toggle open */
 | |
|   onSearch: (searchText: string, fromTyping: boolean, isCompositing: boolean) => boolean;
 | |
|   onSearchSubmit: (searchText: string) => void;
 | |
|   onSelect: (value: RawValueType, option: { selected: boolean }) => void;
 | |
|   onInputKeyDown?: EventHandlerNonNull;
 | |
| 
 | |
|   /**
 | |
|    * @private get real dom for trigger align.
 | |
|    * This may be removed after React provides replacement of `findDOMNode`
 | |
|    */
 | |
|   domRef: () => HTMLDivElement;
 | |
| }
 | |
| 
 | |
| const Selector = defineComponent<SelectorProps>({
 | |
|   name: 'Selector',
 | |
|   setup(props) {
 | |
|     const inputRef = createRef();
 | |
|     let compositionStatus = false;
 | |
| 
 | |
|     // ====================== Input ======================
 | |
|     const [getInputMouseDown, setInputMouseDown] = useLock(0);
 | |
| 
 | |
|     const onInternalInputKeyDown = (event: KeyboardEvent) => {
 | |
|       const { which } = event;
 | |
| 
 | |
|       if (which === KeyCode.UP || which === KeyCode.DOWN) {
 | |
|         event.preventDefault();
 | |
|       }
 | |
| 
 | |
|       if (props.onInputKeyDown) {
 | |
|         props.onInputKeyDown(event);
 | |
|       }
 | |
| 
 | |
|       if (which === KeyCode.ENTER && props.mode === 'tags' && !compositionStatus && !props.open) {
 | |
|         // When menu isn't open, OptionList won't trigger a value change
 | |
|         // So when enter is pressed, the tag's input value should be emitted here to let selector know
 | |
|         props.onSearchSubmit((event.target as HTMLInputElement).value);
 | |
|       }
 | |
| 
 | |
|       if (![KeyCode.SHIFT, KeyCode.TAB, KeyCode.BACKSPACE, KeyCode.ESC].includes(which)) {
 | |
|         props.onToggleOpen(true);
 | |
|       }
 | |
|     };
 | |
| 
 | |
|     /**
 | |
|      * We can not use `findDOMNode` sine it will get warning,
 | |
|      * have to use timer to check if is input element.
 | |
|      */
 | |
|     const onInternalInputMouseDown = () => {
 | |
|       setInputMouseDown(true);
 | |
|     };
 | |
| 
 | |
|     // When paste come, ignore next onChange
 | |
|     let pastedText = null;
 | |
| 
 | |
|     const triggerOnSearch = (value: string) => {
 | |
|       if (props.onSearch(value, true, compositionStatus) !== false) {
 | |
|         props.onToggleOpen(true);
 | |
|       }
 | |
|     };
 | |
| 
 | |
|     const onInputCompositionStart = () => {
 | |
|       compositionStatus = true;
 | |
|     };
 | |
| 
 | |
|     const onInputCompositionEnd = () => {
 | |
|       compositionStatus = false;
 | |
|     };
 | |
| 
 | |
|     const onInputChange = (event: { target: { value: any } }) => {
 | |
|       let {
 | |
|         target: { value },
 | |
|       } = event;
 | |
| 
 | |
|       // Pasted text should replace back to origin content
 | |
|       if (props.tokenWithEnter && pastedText && /[\r\n]/.test(pastedText)) {
 | |
|         // CRLF will be treated as a single space for input element
 | |
|         const replacedText = pastedText.replace(/\r\n/g, ' ').replace(/[\r\n]/g, ' ');
 | |
|         value = value.replace(replacedText, pastedText);
 | |
|       }
 | |
| 
 | |
|       pastedText = null;
 | |
| 
 | |
|       triggerOnSearch(value);
 | |
|     };
 | |
| 
 | |
|     const onInputPaste = (e: ClipboardEvent) => {
 | |
|       const { clipboardData } = e;
 | |
|       const value = clipboardData.getData('text');
 | |
| 
 | |
|       pastedText = value;
 | |
|     };
 | |
| 
 | |
|     const onClick = ({ target }) => {
 | |
|       if (target !== inputRef.current) {
 | |
|         // Should focus input if click the selector
 | |
|         const isIE = (document.body.style as any).msTouchAction !== undefined;
 | |
|         if (isIE) {
 | |
|           setTimeout(() => {
 | |
|             inputRef.current.focus();
 | |
|           });
 | |
|         } else {
 | |
|           inputRef.current.focus();
 | |
|         }
 | |
|       }
 | |
|     };
 | |
| 
 | |
|     const onMousedown = (event: MouseEvent) => {
 | |
|       const inputMouseDown = getInputMouseDown();
 | |
|       if (event.target !== inputRef.current && !inputMouseDown) {
 | |
|         event.preventDefault();
 | |
|       }
 | |
| 
 | |
|       if ((props.mode !== 'combobox' && (!props.showSearch || !inputMouseDown)) || !props.open) {
 | |
|         if (props.open) {
 | |
|           props.onSearch('', true, false);
 | |
|         }
 | |
|         props.onToggleOpen();
 | |
|       }
 | |
|     };
 | |
| 
 | |
|     return {
 | |
|       focus: () => {
 | |
|         inputRef.current.focus();
 | |
|       },
 | |
|       blur: () => {
 | |
|         inputRef.current.blur();
 | |
|       },
 | |
|       onMousedown,
 | |
|       onClick,
 | |
|       onInputPaste,
 | |
|       inputRef,
 | |
|       onInternalInputKeyDown,
 | |
|       onInternalInputMouseDown,
 | |
|       onInputChange,
 | |
|       onInputCompositionEnd,
 | |
|       onInputCompositionStart,
 | |
|     };
 | |
|   },
 | |
|   render() {
 | |
|     const { prefixCls, domRef, multiple } = this.$props as SelectorProps;
 | |
|     const {
 | |
|       onMousedown,
 | |
|       onClick,
 | |
|       inputRef,
 | |
|       onInputPaste,
 | |
|       onInternalInputKeyDown,
 | |
|       onInternalInputMouseDown,
 | |
|       onInputChange,
 | |
|       onInputCompositionStart,
 | |
|       onInputCompositionEnd,
 | |
|     } = this as any;
 | |
|     const sharedProps = {
 | |
|       inputRef,
 | |
|       onInputKeyDown: onInternalInputKeyDown,
 | |
|       onInputMouseDown: onInternalInputMouseDown,
 | |
|       onInputChange,
 | |
|       onInputPaste,
 | |
|       onInputCompositionStart,
 | |
|       onInputCompositionEnd,
 | |
|     };
 | |
|     const selectNode = multiple ? (
 | |
|       <MultipleSelector {...this.$props} {...sharedProps} />
 | |
|     ) : (
 | |
|       <SingleSelector {...this.$props} {...sharedProps} />
 | |
|     );
 | |
|     return (
 | |
|       <div ref={domRef} class={`${prefixCls}-selector`} onClick={onClick} onMousedown={onMousedown}>
 | |
|         {selectNode}
 | |
|       </div>
 | |
|     );
 | |
|   },
 | |
| });
 | |
| 
 | |
| Selector.inheritAttrs = false;
 | |
| Selector.props = {
 | |
|   id: PropTypes.string,
 | |
|   prefixCls: PropTypes.string,
 | |
|   showSearch: PropTypes.looseBool,
 | |
|   open: PropTypes.looseBool,
 | |
|   /** Display in the Selector value, it's not same as `value` prop */
 | |
|   values: PropTypes.array,
 | |
|   multiple: PropTypes.looseBool,
 | |
|   mode: PropTypes.string,
 | |
|   searchValue: PropTypes.string,
 | |
|   activeValue: PropTypes.string,
 | |
|   inputElement: PropTypes.any,
 | |
| 
 | |
|   autofocus: PropTypes.looseBool,
 | |
|   accessibilityIndex: PropTypes.number,
 | |
|   tabindex: PropTypes.oneOfType([PropTypes.number, PropTypes.string]),
 | |
|   disabled: PropTypes.looseBool,
 | |
|   placeholder: PropTypes.any,
 | |
|   removeIcon: PropTypes.any,
 | |
| 
 | |
|   // Tags
 | |
|   maxTagCount: PropTypes.number,
 | |
|   maxTagTextLength: PropTypes.number,
 | |
|   maxTagPlaceholder: PropTypes.any,
 | |
|   tagRender: PropTypes.func,
 | |
| 
 | |
|   /** Check if `tokenSeparators` contains `\n` or `\r\n` */
 | |
|   tokenWithEnter: PropTypes.looseBool,
 | |
| 
 | |
|   // Motion
 | |
|   choiceTransitionName: PropTypes.string,
 | |
| 
 | |
|   onToggleOpen: PropTypes.func,
 | |
|   /** `onSearch` returns go next step boolean to check if need do toggle open */
 | |
|   onSearch: PropTypes.func,
 | |
|   onSearchSubmit: PropTypes.func,
 | |
|   onSelect: PropTypes.func,
 | |
|   onInputKeyDown: PropTypes.func,
 | |
| 
 | |
|   /**
 | |
|    * @private get real dom for trigger align.
 | |
|    * This may be removed after React provides replacement of `findDOMNode`
 | |
|    */
 | |
|   domRef: PropTypes.func,
 | |
| };
 | |
| 
 | |
| export default Selector;
 |