You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
ant-design-vue/components/vc-menu/Menu.jsx

232 lines
6.3 KiB

import PropTypes from '../_util/vue-types'
import { Provider, create } from '../_util/store'
import { default as SubPopupMenu, getActiveKey } from './SubPopupMenu'
import BaseMixin from '../_util/BaseMixin'
import hasProp, { getOptionProps } from '../_util/props-util'
import commonPropsType from './commonPropsType'
const Menu = {
name: 'Menu',
props: {
...commonPropsType,
selectable: PropTypes.bool.def(true),
},
mixins: [BaseMixin],
data () {
const props = getOptionProps(this)
let selectedKeys = props.defaultSelectedKeys
let openKeys = props.defaultOpenKeys
if ('selectedKeys' in props) {
selectedKeys = props.selectedKeys || []
}
if ('openKeys' in props) {
openKeys = props.openKeys || []
}
this.store = create({
selectedKeys,
openKeys,
activeKey: { '0-menu-': getActiveKey({ ...props, children: this.$slots.default || [] }, props.activeKey) },
})
// this.isRootMenu = true // 声明在props上
return {}
},
watch: {
selectedKeys (val) {
this.store.setState({
selectedKeys: val || [],
})
},
openKeys (val) {
this.store.setState({
openKeys: val || [],
})
},
// '$props': {
// handler: function (nextProps) {
// if (hasProp(this, 'selectedKeys')) {
// this.setState({
// sSelectedKeys: nextProps.selectedKeys || [],
// })
// }
// if (hasProp(this, 'openKeys')) {
// this.setState({
// sOpenKeys: nextProps.openKeys || [],
// })
// }
// },
// deep: true,
// },
},
methods: {
// onDestroy (key) {
// const state = this.$data
// const sSelectedKeys = state.sSelectedKeys
// const sOpenKeys = state.sOpenKeys
// let index = sSelectedKeys.indexOf(key)
// if (!hasProp(this, 'selectedKeys') && index !== -1) {
// sSelectedKeys.splice(index, 1)
// }
// index = sOpenKeys.indexOf(key)
// if (!hasProp(this, 'openKeys') && index !== -1) {
// sOpenKeys.splice(index, 1)
// }
// },
onSelect (selectInfo) {
const props = this.$props
if (props.selectable) {
// root menu
let selectedKeys = this.store.getState().selectedKeys
const selectedKey = selectInfo.key
if (props.multiple) {
selectedKeys = selectedKeys.concat([selectedKey])
} else {
selectedKeys = [selectedKey]
}
if (!hasProp(this, 'selectedKeys')) {
this.store.setState({
selectedKeys,
})
}
this.__emit('select', {
...selectInfo,
selectedKeys: selectedKeys,
})
}
},
onClick (e) {
this.__emit('click', e)
},
// onKeyDown needs to be exposed as a instance method
// e.g., in rc-select, we need to navigate menu item while
// current active item is rc-select input box rather than the menu itself
onKeyDown (e, callback) {
this.$refs.innerMenu.getWrappedInstance().onKeyDown(e, callback)
},
onOpenChange (event) {
const openKeys = this.store.getState().openKeys.concat()
let changed = false
const processSingle = (e) => {
let oneChanged = false
if (e.open) {
oneChanged = openKeys.indexOf(e.key) === -1
if (oneChanged) {
openKeys.push(e.key)
}
} else {
const index = openKeys.indexOf(e.key)
oneChanged = index !== -1
if (oneChanged) {
openKeys.splice(index, 1)
}
}
changed = changed || oneChanged
}
if (Array.isArray(event)) {
// batch change call
event.forEach(processSingle)
} else {
processSingle(event)
}
if (changed) {
if (!hasProp(this, 'openKeys')) {
this.store.setState({ openKeys })
}
this.__emit('openChange', openKeys)
}
},
onDeselect (selectInfo) {
const props = this.$props
if (props.selectable) {
const selectedKeys = this.store.getState().selectedKeys.concat()
const selectedKey = selectInfo.key
const index = selectedKeys.indexOf(selectedKey)
if (index !== -1) {
selectedKeys.splice(index, 1)
}
if (!hasProp(this, 'selectedKeys')) {
this.store.setState({
selectedKeys,
})
}
this.__emit('deselect', {
...selectInfo,
selectedKeys: selectedKeys,
})
}
},
getOpenTransitionName () {
const props = this.$props
let transitionName = props.openTransitionName
const animationName = props.openAnimation
if (!transitionName && typeof animationName === 'string') {
transitionName = `${props.prefixCls}-open-${animationName}`
}
return transitionName
},
// isInlineMode () {
// return this.$props.mode === 'inline'
// },
// lastOpenSubMenu () {
// let lastOpen = []
// const { sOpenKeys } = this.$data
// if (sOpenKeys.length) {
// lastOpen = this.getFlatInstanceArray().filter((c) => {
// return c && sOpenKeys.indexOf(c.eventKey) !== -1
// })
// }
// return lastOpen[0]
// },
// renderMenuItem (c, i, subIndex) {
// if (!c) {
// return null
// }
// const state = this.$data
// const extraProps = {
// openKeys: state.sOpenKeys,
// selectedKeys: state.sSelectedKeys,
// triggerSubMenuAction: this.$props.triggerSubMenuAction,
// isRootMenu: this.isRootMenu,
// }
// return this.renderCommonMenuItem(c, i, subIndex, extraProps)
// },
},
render () {
const props = getOptionProps(this)
const subPopupMenuProps = {
props: {
...props,
openTransitionName: this.getOpenTransitionName(),
parentMenu: this,
children: this.$slots.default || [],
__propsSymbol__: Symbol(),
},
class: `${props.prefixCls}-root`,
on: {
click: this.onClick,
openChange: this.onOpenChange,
deselect: this.onDeselect,
select: this.onSelect,
},
ref: 'innerMenu',
}
return (
<Provider store={this.store}>
<SubPopupMenu {...subPopupMenuProps} />
</Provider>
)
},
}
export default Menu