import { getComponent, getSlot } from '../_util/props-util'; import PropTypes from '../_util/vue-types'; import Trigger from '../vc-trigger'; import Menus from './Menus'; import KeyCode from '../_util/KeyCode'; import arrayTreeFilter from 'array-tree-filter'; import shallowEqualArrays from 'shallow-equal/arrays'; import { hasProp, getEvents } from '../_util/props-util'; import BaseMixin from '../_util/BaseMixin'; import { cloneElement } from '../_util/vnode'; import syncWatch from '../_util/syncWatch'; const BUILT_IN_PLACEMENTS = { bottomLeft: { points: ['tl', 'bl'], offset: [0, 4], overflow: { adjustX: 1, adjustY: 1, }, }, topLeft: { points: ['bl', 'tl'], offset: [0, -4], overflow: { adjustX: 1, adjustY: 1, }, }, bottomRight: { points: ['tr', 'br'], offset: [0, 4], overflow: { adjustX: 1, adjustY: 1, }, }, topRight: { points: ['br', 'tr'], offset: [0, -4], overflow: { adjustX: 1, adjustY: 1, }, }, }; export default { name: 'Cascader', mixins: [BaseMixin], inheritAttrs: false, // model: { // prop: 'value', // event: 'change', // }, props: { value: PropTypes.array, defaultValue: PropTypes.array, options: PropTypes.array, // onChange: PropTypes.func, // onPopupVisibleChange: PropTypes.func, popupVisible: PropTypes.bool, disabled: PropTypes.bool.def(false), transitionName: PropTypes.string.def(''), popupClassName: PropTypes.string.def(''), popupStyle: PropTypes.object.def(() => ({})), popupPlacement: PropTypes.string.def('bottomLeft'), prefixCls: PropTypes.string.def('rc-cascader'), dropdownMenuColumnStyle: PropTypes.object, builtinPlacements: PropTypes.object.def(BUILT_IN_PLACEMENTS), loadData: PropTypes.func, changeOnSelect: PropTypes.bool, // onKeyDown: PropTypes.func, expandTrigger: PropTypes.string.def('click'), fieldNames: PropTypes.object.def(() => ({ label: 'label', value: 'value', children: 'children', })), expandIcon: PropTypes.any, loadingIcon: PropTypes.any, getPopupContainer: PropTypes.func, }, data() { let initialValue = []; const { value, defaultValue, popupVisible } = this; if (hasProp(this, 'value')) { initialValue = value || []; } else if (hasProp(this, 'defaultValue')) { initialValue = defaultValue || []; } this.children = undefined; // warning(!('filedNames' in props), // '`filedNames` of Cascader is a typo usage and deprecated, please use `fieldNames` instead.'); this.defaultFieldNames = { label: 'label', value: 'value', children: 'children' }; return { sPopupVisible: popupVisible, sActiveValue: initialValue, sValue: initialValue, }; }, watch: { value: syncWatch(function(val, oldValue) { if (!shallowEqualArrays(val, oldValue)) { const newValues = { sValue: val || [], }; // allow activeValue diff from value // https://github.com/ant-design/ant-design/issues/2767 if (!hasProp(this, 'loadData')) { newValues.sActiveValue = val || []; } this.setState(newValues); } }), popupVisible: syncWatch(function(val) { this.setState({ sPopupVisible: val, }); }), }, methods: { getPopupDOMNode() { return this.trigger.getPopupDomNode(); }, getFieldName(name) { const { defaultFieldNames, fieldNames } = this; return fieldNames[name] || defaultFieldNames[name]; }, getFieldNames() { return this.fieldNames; }, getCurrentLevelOptions() { const { options = [], sActiveValue = [] } = this; const result = arrayTreeFilter( options, (o, level) => o[this.getFieldName('value')] === sActiveValue[level], { childrenKeyName: this.getFieldName('children') }, ); if (result[result.length - 2]) { return result[result.length - 2][this.getFieldName('children')]; } return [...options].filter(o => !o.disabled); }, getActiveOptions(activeValue) { return arrayTreeFilter( this.options || [], (o, level) => o[this.getFieldName('value')] === activeValue[level], { childrenKeyName: this.getFieldName('children') }, ); }, setPopupVisible(popupVisible) { if (!hasProp(this, 'popupVisible')) { this.setState({ sPopupVisible: popupVisible }); } // sync activeValue with value when panel open if (popupVisible && !this.sPopupVisible) { this.setState({ sActiveValue: this.sValue, }); } this.__emit('popupVisibleChange', popupVisible); }, handleChange(options, setProps, e) { if (e.type !== 'keydown' || e.keyCode === KeyCode.ENTER) { const value = options.map(o => o[this.getFieldName('value')]); this.__emit('update:value', value); this.__emit('change', value, options); this.setPopupVisible(setProps.visible); } }, handlePopupVisibleChange(popupVisible) { this.setPopupVisible(popupVisible); }, handleMenuSelect(targetOption, menuIndex, e) { // Keep focused state for keyboard support const triggerNode = this.trigger.getRootDomNode(); if (triggerNode && triggerNode.focus) { triggerNode.focus(); } const { changeOnSelect, loadData, expandTrigger } = this; if (!targetOption || targetOption.disabled) { return; } let { sActiveValue } = this; sActiveValue = sActiveValue.slice(0, menuIndex + 1); sActiveValue[menuIndex] = targetOption[this.getFieldName('value')]; const activeOptions = this.getActiveOptions(sActiveValue); if ( targetOption.isLeaf === false && !targetOption[this.getFieldName('children')] && loadData ) { if (changeOnSelect) { this.handleChange(activeOptions, { visible: true }, e); } this.setState({ sActiveValue }); loadData(activeOptions); return; } const newState = {}; if ( !targetOption[this.getFieldName('children')] || !targetOption[this.getFieldName('children')].length ) { this.handleChange(activeOptions, { visible: false }, e); // set value to activeValue when select leaf option newState.sValue = sActiveValue; // add e.type judgement to prevent `onChange` being triggered by mouseEnter } else if (changeOnSelect && (e.type === 'click' || e.type === 'keydown')) { if (expandTrigger === 'hover') { this.handleChange(activeOptions, { visible: false }, e); } else { this.handleChange(activeOptions, { visible: true }, e); } // set value to activeValue on every select newState.sValue = sActiveValue; } newState.sActiveValue = sActiveValue; // not change the value by keyboard if (hasProp(this, 'value') || (e.type === 'keydown' && e.keyCode !== KeyCode.ENTER)) { delete newState.sValue; } this.setState(newState); }, handleItemDoubleClick() { const { changeOnSelect } = this.$props; if (changeOnSelect) { this.setPopupVisible(false); } }, handleKeyDown(e) { const children = this.children; // https://github.com/ant-design/ant-design/issues/6717 // Don't bind keyboard support when children specify the onKeyDown if (children) { const keydown = getEvents(children).onKeydown; if (keydown) { keydown(e); return; } } const activeValue = [...this.sActiveValue]; const currentLevel = activeValue.length - 1 < 0 ? 0 : activeValue.length - 1; const currentOptions = this.getCurrentLevelOptions(); const currentIndex = currentOptions .map(o => o[this.getFieldName('value')]) .indexOf(activeValue[currentLevel]); if ( e.keyCode !== KeyCode.DOWN && e.keyCode !== KeyCode.UP && e.keyCode !== KeyCode.LEFT && e.keyCode !== KeyCode.RIGHT && e.keyCode !== KeyCode.ENTER && e.keyCode !== KeyCode.SPACE && e.keyCode !== KeyCode.BACKSPACE && e.keyCode !== KeyCode.ESC && e.keyCode !== KeyCode.TAB ) { return; } // Press any keys above to reopen menu if ( !this.sPopupVisible && e.keyCode !== KeyCode.BACKSPACE && e.keyCode !== KeyCode.LEFT && e.keyCode !== KeyCode.RIGHT && e.keyCode !== KeyCode.ESC && e.keyCode !== KeyCode.TAB ) { this.setPopupVisible(true); return; } if (e.keyCode === KeyCode.DOWN || e.keyCode === KeyCode.UP) { e.preventDefault(); let nextIndex = currentIndex; if (nextIndex !== -1) { if (e.keyCode === KeyCode.DOWN) { nextIndex += 1; nextIndex = nextIndex >= currentOptions.length ? 0 : nextIndex; } else { nextIndex -= 1; nextIndex = nextIndex < 0 ? currentOptions.length - 1 : nextIndex; } } else { nextIndex = 0; } activeValue[currentLevel] = currentOptions[nextIndex][this.getFieldName('value')]; } else if (e.keyCode === KeyCode.LEFT || e.keyCode === KeyCode.BACKSPACE) { e.preventDefault(); activeValue.splice(activeValue.length - 1, 1); } else if (e.keyCode === KeyCode.RIGHT) { e.preventDefault(); if ( currentOptions[currentIndex] && currentOptions[currentIndex][this.getFieldName('children')] ) { activeValue.push( currentOptions[currentIndex][this.getFieldName('children')][0][ this.getFieldName('value') ], ); } } else if (e.keyCode === KeyCode.ESC || e.keyCode === KeyCode.TAB) { this.setPopupVisible(false); return; } if (!activeValue || activeValue.length === 0) { this.setPopupVisible(false); } const activeOptions = this.getActiveOptions(activeValue); const targetOption = activeOptions[activeOptions.length - 1]; this.handleMenuSelect(targetOption, activeOptions.length - 1, e); this.__emit('keydown', e); }, saveTrigger(node) { this.trigger = node; }, }, render() { const { $props, sActiveValue, handleMenuSelect, sPopupVisible, handlePopupVisibleChange, handleKeyDown, } = this; const { prefixCls, transitionName, popupClassName, options = [], disabled, builtinPlacements, popupPlacement, ...restProps } = $props; // Did not show popup when there is no options let menus =
; let emptyMenuClassName = ''; if (options && options.length > 0) { const loadingIcon = getComponent(this, 'loadingIcon'); const expandIcon = getComponent(this, 'expandIcon') || '>'; const menusProps = { ...$props, ...this.$attrs, fieldNames: this.getFieldNames(), defaultFieldNames: this.defaultFieldNames, activeValue: sActiveValue, visible: sPopupVisible, loadingIcon, expandIcon, onSelect: handleMenuSelect, onItemDoubleClick: this.handleItemDoubleClick, }; menus = ; } else { emptyMenuClassName = ` ${prefixCls}-menus-empty`; } const triggerProps = { ...restProps, ...this.$attrs, disabled, popupPlacement, builtinPlacements, popupTransitionName: transitionName, action: disabled ? [] : ['click'], popupVisible: disabled ? false : sPopupVisible, prefixCls: `${prefixCls}-menus`, popupClassName: popupClassName + emptyMenuClassName, popup: menus, onPopupVisibleChange: handlePopupVisibleChange, ref: this.saveTrigger, }; const children = getSlot(this); this.children = children; return ( {children && cloneElement(children[0], { onKeydown: handleKeyDown, tabindex: disabled ? undefined : 0, })} ); }, };