ant-design-vue/components/menu/src/Menu.vue

200 lines
5.5 KiB
Vue

<script>
import PropTypes from '../../_util/vue-types'
import MenuMixin from './MenuMixin'
import StateMixin from '../../_util/StateMixin'
import hasProp from '../../_util/hasProp'
const Menu = {
name: 'Menu',
props: {
defaultSelectedKeys: PropTypes.arrayOf(PropTypes.string).def([]),
selectedKeys: PropTypes.arrayOf(PropTypes.string),
defaultOpenKeys: PropTypes.arrayOf(PropTypes.string).def([]),
openKeys: PropTypes.arrayOf(PropTypes.string),
mode: PropTypes.oneOf(['horizontal', 'vertical', 'vertical-left', 'vertical-right', 'inline']).def('vertical'),
getPopupContainer: PropTypes.func,
openTransitionName: PropTypes.string,
openAnimation: PropTypes.oneOfType([PropTypes.string, PropTypes.object]),
subMenuOpenDelay: PropTypes.number.def(0),
subMenuCloseDelay: PropTypes.number.def(0.1),
forceSubMenuRender: PropTypes.bool,
triggerSubMenuAction: PropTypes.string.def('click'),
level: PropTypes.number.def(1),
selectable: PropTypes.bool.def(true),
multiple: PropTypes.bool,
children: PropTypes.any,
},
mixins: [StateMixin, MenuMixin],
data () {
const props = this.$props
let sSelectedKeys = props.defaultSelectedKeys
let sOpenKeys = props.defaultOpenKeys
sSelectedKeys = props.selectedKeys || []
sOpenKeys = props.openKeys || []
this.isRootMenu = true
return {
sSelectedKeys,
sOpenKeys,
}
},
watch: {
'$props': {
handler: function (nextProps) {
const props = {}
if (nextProps.selectedKeys === undefined) {
props.sSelectedKeys = nextProps.selectedKeys || []
}
if (nextProps.openKeys === undefined) {
props.sOpenKeys = nextProps.openKeys || []
}
this.setState(props)
},
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 sSelectedKeys = this.$data.sSelectedKeys
const selectedKey = selectInfo.key
if (props.multiple) {
sSelectedKeys = sSelectedKeys.concat([selectedKey])
} else {
sSelectedKeys = [selectedKey]
}
if (!hasProp(this, 'selectedKeys')) {
this.setState({
sSelectedKeys,
})
}
this.$emit('select', {
...selectInfo,
sSelectedKeys,
})
}
},
onClick (e) {
this.$emit('click', e)
},
onOpenChange (e_) {
const sOpenKeys = this.$data.sOpenKeys.concat()
let changed = false
const processSingle = (e) => {
let oneChanged = false
if (e.open) {
oneChanged = sOpenKeys.indexOf(e.key) === -1
if (oneChanged) {
sOpenKeys.push(e.key)
}
} else {
const index = sOpenKeys.indexOf(e.key)
oneChanged = index !== -1
if (oneChanged) {
sOpenKeys.splice(index, 1)
}
}
changed = changed || oneChanged
}
if (Array.isArray(e_)) {
// batch change call
e_.forEach(processSingle)
} else {
processSingle(e_)
}
if (changed) {
if (!hasProp(this, 'openKeys')) {
this.setState({ sOpenKeys })
}
this.$emit('openChange', sOpenKeys)
}
},
onDeselect (selectInfo) {
const props = this.$props
if (props.selectable) {
const sSelectedKeys = this.$data.sSelectedKeys.concat()
const selectedKey = selectInfo.key
const index = sSelectedKeys.indexOf(selectedKey)
if (index !== -1) {
sSelectedKeys.splice(index, 1)
}
if (!hasProp(this, 'selectedKeys')) {
this.setState({
sSelectedKeys,
})
}
this.$emit('deselect', {
...selectInfo,
sSelectedKeys,
})
}
},
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.props.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: true,
}
return this.renderCommonMenuItem(c, i, subIndex, extraProps)
},
},
render () {
const props = { ...this.$props }
props.class = ` ${props.prefixCls}-root`
return this.renderRoot(props, this.$slots.default)
},
}
export default Menu
</script>