Implemented MegaMenu
parent
c909888e56
commit
ba96148172
|
@ -8,6 +8,29 @@
|
||||||
<span class="p-menuitem-text">{{category.label}}</span>
|
<span class="p-menuitem-text">{{category.label}}</span>
|
||||||
<span v-if="category.items" :class="getCategorySubMenuIcon(category)"></span>
|
<span v-if="category.items" :class="getCategorySubMenuIcon(category)"></span>
|
||||||
</a>
|
</a>
|
||||||
|
<div class="p-megamenu-panel" v-if="category.items">
|
||||||
|
<div class="p-grid">
|
||||||
|
<div v-for="(column,columnIndex) of category.items" :key="category.label + '_column_' + columnIndex" :class="getColumnClassName(category)">
|
||||||
|
<ul v-for="(submenu,submenuIndex) of column" class="p-megamenu-submenu" :key="submenu.label + '_submenu_' + submenuIndex">
|
||||||
|
<li :class="getSubmenuHeaderClass(submenu)" :style="submenu.style">{{submenu.label}}</li>
|
||||||
|
<template v-for="(item, i) of submenu.items">
|
||||||
|
<li role="menuitem" :class="getSubmenuItemClass(item)" :style="item.style" v-if="item.visible !== false && !item.separator" :key="item.label + i">
|
||||||
|
<router-link v-if="item.to" :to="item.to" class="p-menuitem-link" @click.native="onLeafClick($event, item)">
|
||||||
|
<span :class="['p-menuitem-icon', item.icon]"></span>
|
||||||
|
<span class="p-menuitem-text">{{item.label}}</span>
|
||||||
|
</router-link>
|
||||||
|
<a v-else :href="item.url||'#'" class="p-menuitem-link" :target="item.target" @click="onLeafClick($event, item)">
|
||||||
|
<span :class="['p-menuitem-icon', item.icon]"></span>
|
||||||
|
<span class="p-menuitem-text">{{item.label}}</span>
|
||||||
|
<span :class="getSubmenuIcon()" v-if="item.items"></span>
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
<li class="p-menu-separator" :style="item.style" v-if="item.visible !== false && item.separator" :key="'separator' + i"></li>
|
||||||
|
</template>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</li>
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
<div class="p-menubar-custom" v-if="$slots.default">
|
<div class="p-menubar-custom" v-if="$slots.default">
|
||||||
|
@ -17,6 +40,8 @@
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
|
import DomHandler from '../utils/DomHandler';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
props: {
|
props: {
|
||||||
model: {
|
model: {
|
||||||
|
@ -28,23 +53,156 @@ export default {
|
||||||
default: 'horizontal'
|
default: 'horizontal'
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
documentClickListener: null,
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
activeItem: null
|
activeItem: null
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
beforeDestroy() {
|
||||||
|
this.unbindDocumentClickListener();
|
||||||
|
},
|
||||||
methods: {
|
methods: {
|
||||||
itemClick(event) {
|
onLeafClick(event, item) {
|
||||||
|
if (item.disabled) {
|
||||||
|
event.preventDefault();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!item.url) {
|
||||||
|
event.preventDefault();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (item.command) {
|
||||||
|
item.command({
|
||||||
|
originalEvent: event,
|
||||||
|
item: item
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
this.activeItem = null;
|
||||||
},
|
},
|
||||||
onCategoryMouseEnter(event, category) {
|
onCategoryMouseEnter(event, category) {
|
||||||
|
if (category.disabled) {
|
||||||
|
event.preventDefault();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.activeItem) {
|
||||||
|
this.activeItem = category;
|
||||||
|
}
|
||||||
},
|
},
|
||||||
onCategoryClick(event, category) {
|
onCategoryClick(event, category) {
|
||||||
|
if (category.disabled) {
|
||||||
|
event.preventDefault();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!category.url) {
|
||||||
|
event.preventDefault();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (category.command) {
|
||||||
|
category.command({
|
||||||
|
originalEvent: event,
|
||||||
|
item: category
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (category.items) {
|
||||||
|
if (this.activeItem && this.activeItem === category) {
|
||||||
|
this.activeItem = null;
|
||||||
|
this.unbindDocumentClickListener();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
this.activeItem = category;
|
||||||
|
this.bindDocumentClickListener();
|
||||||
|
}
|
||||||
|
}
|
||||||
},
|
},
|
||||||
onCategoryKeydown(event, category) {
|
onCategoryKeydown(event, category) {
|
||||||
|
let listItem = event.currentTarget.parentElement;
|
||||||
|
|
||||||
|
switch(event.which) {
|
||||||
|
//down
|
||||||
|
case 40:
|
||||||
|
if (this.horizontal)
|
||||||
|
this.expandMenu(category);
|
||||||
|
else
|
||||||
|
this.navigateToNextItem(listItem);
|
||||||
|
|
||||||
|
event.preventDefault();
|
||||||
|
break;
|
||||||
|
|
||||||
|
//up
|
||||||
|
case 38:
|
||||||
|
if (this.vertical)
|
||||||
|
this.navigateToPrevItem(listItem);
|
||||||
|
else if (category.items && category === this.activeItem)
|
||||||
|
this.collapseMenu();
|
||||||
|
|
||||||
|
event.preventDefault();
|
||||||
|
break;
|
||||||
|
|
||||||
|
//right
|
||||||
|
case 39:
|
||||||
|
if (this.horizontal)
|
||||||
|
this.navigateToNextItem(listItem);
|
||||||
|
else
|
||||||
|
this.expandMenu(category);
|
||||||
|
|
||||||
|
event.preventDefault()
|
||||||
|
break;
|
||||||
|
|
||||||
|
//left
|
||||||
|
case 37:
|
||||||
|
if (this.horizontal)
|
||||||
|
this.navigateToPrevItem(listItem);
|
||||||
|
else if (category.items && category === this.activeItem)
|
||||||
|
this.collapseMenu();
|
||||||
|
|
||||||
|
event.preventDefault();
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
expandMenu(item) {
|
||||||
|
if (item.items) {
|
||||||
|
this.activeItem = item;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
collapseMenu(item) {
|
||||||
|
this.activeItem = null;
|
||||||
|
},
|
||||||
|
findNextItem(item) {
|
||||||
|
let nextItem = item.nextElementSibling;
|
||||||
|
|
||||||
|
if (nextItem)
|
||||||
|
return DomHandler.hasClass(nextItem, 'p-disabled') || !DomHandler.hasClass(nextItem, 'p-menuitem') ? this.findNextItem(nextItem) : nextItem;
|
||||||
|
else
|
||||||
|
return null;
|
||||||
|
},
|
||||||
|
findPrevItem(item) {
|
||||||
|
let prevItem = item.previousElementSibling;
|
||||||
|
|
||||||
|
if (prevItem)
|
||||||
|
return DomHandler.hasClass(prevItem, 'p-disabled') || !DomHandler.hasClass(prevItem, 'p-menuitem') ? this.findPrevItem(prevItem) : prevItem;
|
||||||
|
else
|
||||||
|
return null;
|
||||||
|
},
|
||||||
|
navigateToNextItem(listItem) {
|
||||||
|
var nextItem = this.findNextItem(listItem);
|
||||||
|
if (nextItem) {
|
||||||
|
nextItem.children[0].focus();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
navigateToPrevItem(listItem) {
|
||||||
|
var prevItem = this.findPrevItem(listItem);
|
||||||
|
if (prevItem) {
|
||||||
|
prevItem.children[0].focus();
|
||||||
|
}
|
||||||
},
|
},
|
||||||
getCategoryClass(category) {
|
getCategoryClass(category) {
|
||||||
return ['p-menuitem', {
|
return ['p-menuitem', {
|
||||||
|
@ -60,7 +218,60 @@ export default {
|
||||||
},
|
},
|
||||||
getCategoryIcon(category) {
|
getCategoryIcon(category) {
|
||||||
return ['p-menuitem-icon', category.icon];
|
return ['p-menuitem-icon', category.icon];
|
||||||
}
|
},
|
||||||
|
getColumnClassName(category) {
|
||||||
|
let length = category.items ? category.items.length: 0;
|
||||||
|
let columnClass;
|
||||||
|
|
||||||
|
switch(length) {
|
||||||
|
case 2:
|
||||||
|
columnClass= 'p-col-6';
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 3:
|
||||||
|
columnClass= 'p-col-4';
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 4:
|
||||||
|
columnClass= 'p-col-3';
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 6:
|
||||||
|
columnClass= 'p-col-2';
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
columnClass= 'p-col-12';
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return columnClass;
|
||||||
|
},
|
||||||
|
getSubmenuHeaderClass(submenu) {
|
||||||
|
return ['p-megamenu-submenu-header', submenu.class, {'p-disabled': submenu.disabled}];
|
||||||
|
},
|
||||||
|
getSubmenuItemClass(item) {
|
||||||
|
return ['p-menuitem', item.class, {'p-disabled': item.disabled}];
|
||||||
|
},
|
||||||
|
bindDocumentClickListener() {
|
||||||
|
if (!this.documentClickListener) {
|
||||||
|
this.documentClickListener = (event) => {
|
||||||
|
console.log(event);
|
||||||
|
if (this.$el && !this.$el.contains(event.target)) {
|
||||||
|
this.activeItem = null;
|
||||||
|
this.unbindDocumentClickListener();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
document.addEventListener('click', this.documentClickListener);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
unbindDocumentClickListener() {
|
||||||
|
if (this.documentClickListener) {
|
||||||
|
document.removeEventListener('click', this.documentClickListener);
|
||||||
|
this.documentClickListener = null;
|
||||||
|
}
|
||||||
|
},
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
containerClass() {
|
containerClass() {
|
||||||
|
|
|
@ -8,7 +8,11 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="content-section implementation">
|
<div class="content-section implementation">
|
||||||
|
<h3>Horizontal</h3>
|
||||||
<MegaMenu :model="items" />
|
<MegaMenu :model="items" />
|
||||||
|
|
||||||
|
<h3>Vertical</h3>
|
||||||
|
<MegaMenu :model="items" orientation="vertical"/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<MegaMenuDoc />
|
<MegaMenuDoc />
|
||||||
|
@ -126,7 +130,7 @@ export default {
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
{
|
{
|
||||||
label: 'Technology 4',
|
label: 'Setting 4',
|
||||||
items: [{label: 'Setting 4.1'}, {label: 'Setting 4.2'}]
|
items: [{label: 'Setting 4.1'}, {label: 'Setting 4.2'}]
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
|
Loading…
Reference in New Issue