UNPKG

primevue

Version:

[![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](https://opensource.org/licenses/MIT) [![npm version](https://badge.fury.io/js/primevue.svg)](https://badge.fury.io/js/primevue) [![Discord Chat](https://img.shields.io/discord/55794023

506 lines (469 loc) 20.1 kB
this.primevue = this.primevue || {}; this.primevue.contextmenu = (function (utils, Ripple, vue) { 'use strict'; function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; } var Ripple__default = /*#__PURE__*/_interopDefaultLegacy(Ripple); var script = { name: 'ContextMenuSub', emits: ['leaf-click'], props: { model: { type: Array, default: null }, root: { type: Boolean, default: false }, parentActive: { type: Boolean, default: false }, template: { type: Object, default: null } }, watch: { parentActive(newValue) { if (!newValue) { this.activeItem = null; } } }, data() { return { activeItem: null } }, methods: { onItemMouseEnter(event, item) { if (item.disabled) { event.preventDefault(); return; } this.activeItem = item; }, onItemClick(event, item, navigate) { if (item.disabled) { event.preventDefault(); return; } if (item.command) { item.command({ originalEvent: event, item: item }); } if (item.items) { if (this.activeItem && item === this.activeItem) this.activeItem = null; else this.activeItem = item; } if (!item.items) { this.onLeafClick(); } if (item.to && navigate) { navigate(event); } }, onLeafClick() { this.activeItem = null; this.$emit('leaf-click'); }, onEnter() { this.position(); }, position() { const parentItem = this.$refs.container.parentElement; const containerOffset = utils.DomHandler.getOffset(this.$refs.container.parentElement); const viewport = utils.DomHandler.getViewport(); const sublistWidth = this.$refs.container.offsetParent ? this.$refs.container.offsetWidth : utils.DomHandler.getHiddenElementOuterWidth(this.$refs.container); const itemOuterWidth = utils.DomHandler.getOuterWidth(parentItem.children[0]); this.$refs.container.style.top = '0px'; if ((parseInt(containerOffset.left, 10) + itemOuterWidth + sublistWidth) > (viewport.width - utils.DomHandler.calculateScrollbarWidth())) { this.$refs.container.style.left = -1 * sublistWidth + 'px'; } else { this.$refs.container.style.left = itemOuterWidth + 'px'; } }, getItemClass(item) { return [ 'p-menuitem', item.class, { 'p-menuitem-active': this.activeItem === item } ] }, getLinkClass(item) { return ['p-menuitem-link', {'p-disabled': item.disabled}]; }, visible(item) { return (typeof item.visible === 'function' ? item.visible() : item.visible !== false); } }, computed: { containerClass() { return {'p-submenu-list': !this.root}; } }, directives: { 'ripple': Ripple__default['default'] } }; const _hoisted_1 = { class: "p-menuitem-text" }; const _hoisted_2 = { class: "p-menuitem-text" }; const _hoisted_3 = { key: 0, class: "p-submenu-icon pi pi-angle-right" }; function render(_ctx, _cache, $props, $setup, $data, $options) { const _component_router_link = vue.resolveComponent("router-link"); const _component_ContextMenuSub = vue.resolveComponent("ContextMenuSub", true); const _directive_ripple = vue.resolveDirective("ripple"); return (vue.openBlock(), vue.createBlock(vue.Transition, { name: "p-contextmenusub", onEnter: $options.onEnter }, { default: vue.withCtx(() => [ ($props.root ? true : $props.parentActive) ? (vue.openBlock(), vue.createBlock("ul", { key: 0, ref: "container", class: $options.containerClass, role: "menu" }, [ (vue.openBlock(true), vue.createBlock(vue.Fragment, null, vue.renderList($props.model, (item, i) => { return (vue.openBlock(), vue.createBlock(vue.Fragment, { key: item.label + i.toString() }, [ ($options.visible(item) && !item.separator) ? (vue.openBlock(), vue.createBlock("li", { key: 0, role: "none", class: $options.getItemClass(item), style: item.style, onMouseenter: $event => ($options.onItemMouseEnter($event, item)) }, [ (!$props.template) ? (vue.openBlock(), vue.createBlock(vue.Fragment, { key: 0 }, [ (item.to && !item.disabled) ? (vue.openBlock(), vue.createBlock(_component_router_link, { key: 0, to: item.to, custom: "" }, { default: vue.withCtx(({navigate, href}) => [ vue.withDirectives(vue.createVNode("a", { href: href, onClick: $event => ($options.onItemClick($event, item, navigate)), class: $options.getLinkClass(item), role: "menuitem" }, [ vue.createVNode("span", { class: ['p-menuitem-icon', item.icon] }, null, 2), vue.createVNode("span", _hoisted_1, vue.toDisplayString(item.label), 1) ], 10, ["href", "onClick"]), [ [_directive_ripple] ]) ]), _: 2 }, 1032, ["to"])) : vue.withDirectives((vue.openBlock(), vue.createBlock("a", { key: 1, href: item.url, class: $options.getLinkClass(item), target: item.target, onClick: $event => ($options.onItemClick($event, item)), "aria-haspopup": item.items != null, "aria-expanded": item === $data.activeItem, role: "menuitem", tabindex: item.disabled ? null : '0' }, [ vue.createVNode("span", { class: ['p-menuitem-icon', item.icon] }, null, 2), vue.createVNode("span", _hoisted_2, vue.toDisplayString(item.label), 1), (item.items) ? (vue.openBlock(), vue.createBlock("span", _hoisted_3)) : vue.createCommentVNode("", true) ], 10, ["href", "target", "onClick", "aria-haspopup", "aria-expanded", "tabindex"])), [ [_directive_ripple] ]) ], 64)) : (vue.openBlock(), vue.createBlock(vue.resolveDynamicComponent($props.template), { key: 1, item: item }, null, 8, ["item"])), ($options.visible(item) && item.items) ? (vue.openBlock(), vue.createBlock(_component_ContextMenuSub, { model: item.items, key: item.label + '_sub_', template: $props.template, onLeafClick: $options.onLeafClick, parentActive: item === $data.activeItem }, null, 8, ["model", "template", "onLeafClick", "parentActive"])) : vue.createCommentVNode("", true) ], 46, ["onMouseenter"])) : vue.createCommentVNode("", true), ($options.visible(item) && item.separator) ? (vue.openBlock(), vue.createBlock("li", { class: ['p-menu-separator', item.class], style: item.style, key: 'separator' + i.toString(), role: "separator" }, null, 6)) : vue.createCommentVNode("", true) ], 64)) }), 128)) ], 2)) : vue.createCommentVNode("", true) ]), _: 1 }, 8, ["onEnter"])) } script.render = render; var script$1 = { name: 'ContextMenu', inheritAttrs: false, props: { model: { type: Array, default: null }, appendTo: { type: String, default: 'body' }, autoZIndex: { type: Boolean, default: true }, baseZIndex: { type: Number, default: 0 }, global: { type: Boolean, default: false } }, target: null, outsideClickListener: null, resizeListener: null, documentContextMenuListener: null, pageX: null, pageY: null, container: null, data() { return { visible: false }; }, beforeUnmount() { this.unbindResizeListener(); this.unbindOutsideClickListener(); this.unbindDocumentContextMenuListener(); if (this.container && this.autoZIndex) { utils.ZIndexUtils.clear(this.container); } this.container = null; }, mounted() { if (this.global) { this.bindDocumentContextMenuListener(); } }, methods: { itemClick(event) { const item = event.item; if (item.command) { item.command(event); event.originalEvent.preventDefault(); } this.hide(); }, toggle(event) { if (this.visible) this.hide(); else this.show(event); }, onLeafClick() { this.hide(); }, show(event) { this.pageX = event.pageX; this.pageY = event.pageY; if (this.visible) this.position(); else this.visible = true; event.stopPropagation(); event.preventDefault(); }, hide() { this.visible = false; }, onEnter(el) { this.position(); this.bindOutsideClickListener(); this.bindResizeListener(); if (this.autoZIndex) { utils.ZIndexUtils.set('menu', el, this.baseZIndex + this.$primevue.config.zIndex.menu); } }, onLeave() { this.unbindOutsideClickListener(); this.unbindResizeListener(); }, onAfterLeave(el) { if (this.autoZIndex) { utils.ZIndexUtils.clear(el); } }, position() { let left = this.pageX + 1; let top = this.pageY + 1; let width = this.container.offsetParent ? this.container.offsetWidth : utils.DomHandler.getHiddenElementOuterWidth(this.container); let height = this.container.offsetParent ? this.container.offsetHeight : utils.DomHandler.getHiddenElementOuterHeight(this.container); let viewport = utils.DomHandler.getViewport(); //flip if (left + width - document.body.scrollLeft > viewport.width) { left -= width; } //flip if (top + height - document.body.scrollTop > viewport.height) { top -= height; } //fit if (left < document.body.scrollLeft) { left = document.body.scrollLeft; } //fit if (top < document.body.scrollTop) { top = document.body.scrollTop; } this.container.style.left = left + 'px'; this.container.style.top = top + 'px'; }, bindOutsideClickListener() { if (!this.outsideClickListener) { this.outsideClickListener = (event) => { if (this.visible && this.container && !this.container.contains(event.target) && !event.ctrlKey) { this.hide(); } }; document.addEventListener('click', this.outsideClickListener); } }, unbindOutsideClickListener() { if (this.outsideClickListener) { document.removeEventListener('click', this.outsideClickListener); this.outsideClickListener = null; } }, bindResizeListener() { if (!this.resizeListener) { this.resizeListener = () => { if (this.visible) { this.hide(); } }; window.addEventListener('resize', this.resizeListener); } }, unbindResizeListener() { if (this.resizeListener) { window.removeEventListener('resize', this.resizeListener); this.resizeListener = null; } }, bindDocumentContextMenuListener() { if (!this.documentContextMenuListener) { this.documentContextMenuListener = (event) => { this.show(event); }; document.addEventListener('contextmenu', this.documentContextMenuListener); } }, unbindDocumentContextMenuListener() { if(this.documentContextMenuListener) { document.removeEventListener('contextmenu', this.documentContextMenuListener); this.documentContextMenuListener = null; } }, containerRef(el) { this.container = el; } }, computed: { containerClass() { return ['p-contextmenu p-component', { 'p-input-filled': this.$primevue.config.inputStyle === 'filled', 'p-ripple-disabled': this.$primevue.config.ripple === false }] } }, components: { 'ContextMenuSub': script } }; function render$1(_ctx, _cache, $props, $setup, $data, $options) { const _component_ContextMenuSub = vue.resolveComponent("ContextMenuSub"); return (vue.openBlock(), vue.createBlock(vue.Teleport, { to: $props.appendTo }, [ vue.createVNode(vue.Transition, { name: "p-contextmenu", onEnter: $options.onEnter, onLeave: $options.onLeave, onAfterLeave: $options.onAfterLeave }, { default: vue.withCtx(() => [ ($data.visible) ? (vue.openBlock(), vue.createBlock("div", vue.mergeProps({ key: 0, ref: $options.containerRef, class: $options.containerClass }, _ctx.$attrs), [ vue.createVNode(_component_ContextMenuSub, { model: $props.model, root: true, onLeafClick: $options.onLeafClick, template: _ctx.$slots.item }, null, 8, ["model", "onLeafClick", "template"]) ], 16)) : vue.createCommentVNode("", true) ]), _: 1 }, 8, ["onEnter", "onLeave", "onAfterLeave"]) ], 8, ["to"])) } function styleInject(css, ref) { if ( ref === void 0 ) ref = {}; var insertAt = ref.insertAt; if (!css || typeof document === 'undefined') { return; } var head = document.head || document.getElementsByTagName('head')[0]; var style = document.createElement('style'); style.type = 'text/css'; if (insertAt === 'top') { if (head.firstChild) { head.insertBefore(style, head.firstChild); } else { head.appendChild(style); } } else { head.appendChild(style); } if (style.styleSheet) { style.styleSheet.cssText = css; } else { style.appendChild(document.createTextNode(css)); } } var css_248z = "\n.p-contextmenu {\n position: absolute;\n}\n.p-contextmenu ul {\n margin: 0;\n padding: 0;\n list-style: none;\n}\n.p-contextmenu .p-submenu-list {\n position: absolute;\n min-width: 100%;\n z-index: 1;\n}\n.p-contextmenu .p-menuitem-link {\n cursor: pointer;\n display: -webkit-box;\n display: -ms-flexbox;\n display: flex;\n -webkit-box-align: center;\n -ms-flex-align: center;\n align-items: center;\n text-decoration: none;\n overflow: hidden;\n position: relative;\n}\n.p-contextmenu .p-menuitem-text {\n line-height: 1;\n}\n.p-contextmenu .p-menuitem {\n position: relative;\n}\n.p-contextmenu .p-menuitem-link .p-submenu-icon {\n margin-left: auto;\n}\n.p-contextmenu-enter-from {\n opacity: 0;\n}\n.p-contextmenu-enter-active {\n -webkit-transition: opacity 250ms;\n transition: opacity 250ms;\n}\n"; styleInject(css_248z); script$1.render = render$1; return script$1; }(primevue.utils, primevue.ripple, Vue));