primeng
Version:
PrimeNG is an open source UI library for Angular featuring a rich set of 80+ components, a theme designer, various theme alternatives such as Material, Bootstrap, Tailwind, premium templates and professional support. In addition, it integrates with PrimeB
865 lines (857 loc) • 85.4 kB
JavaScript
export * from 'primeng/types/tieredmenu';
import * as i0 from '@angular/core';
import { Injectable, InjectionToken, input, EventEmitter, signal, inject, forwardRef, numberAttribute, booleanAttribute, ViewChild, Output, Input, Inject, ViewEncapsulation, Component, computed, effect, ContentChildren, ContentChild, ChangeDetectionStrategy, NgModule } from '@angular/core';
import { style } from '@primeuix/styles/tieredmenu';
import { BaseStyle } from 'primeng/base';
import * as i2 from '@angular/common';
import { isPlatformBrowser, CommonModule } from '@angular/common';
import * as i3 from '@angular/router';
import { RouterModule } from '@angular/router';
import { nestedPosition, resolve, isNotEmpty, uuid, isEmpty, focus, isTouchDevice, isPrintableCharacter, findSingle, addStyle, relativePosition, absolutePosition, getOuterWidth, appendChild, findLastIndex } from '@primeuix/utils';
import * as i6 from 'primeng/api';
import { SharedModule, PrimeTemplate } from 'primeng/api';
import { BaseComponent, PARENT_INSTANCE } from 'primeng/basecomponent';
import * as i1 from 'primeng/bind';
import { Bind, BindModule } from 'primeng/bind';
import { ConnectedOverlayScrollHandler } from 'primeng/dom';
import { AngleRightIcon } from 'primeng/icons';
import * as i5 from 'primeng/motion';
import { MotionModule } from 'primeng/motion';
import { Ripple } from 'primeng/ripple';
import * as i4 from 'primeng/tooltip';
import { TooltipModule } from 'primeng/tooltip';
import { ZIndexUtils } from 'primeng/utils';
const inlineStyles = {
submenu: ({ instance, processedItem }) => ({ display: instance.isItemActive(processedItem) ? 'flex' : 'none' })
};
const classes = {
root: ({ instance }) => [
'p-tieredmenu p-component',
{
'p-tieredmenu-overlay': instance.popup,
'p-tieredmenu-mobile': instance.queryMatches()
}
],
start: 'p-tieredmenu-start',
rootList: 'p-tieredmenu-root-list',
item: ({ instance, processedItem }) => [
'p-tieredmenu-item',
{
'p-tieredmenu-item-active': instance.isItemActive(processedItem),
'p-focus': instance.isItemFocused(processedItem),
'p-disabled': instance.isItemDisabled(processedItem)
}
],
itemContent: 'p-tieredmenu-item-content',
itemLink: 'p-tieredmenu-item-link',
itemIcon: 'p-tieredmenu-item-icon',
itemLabel: 'p-tieredmenu-item-label',
itemBadge: 'p-menuitem-badge',
submenuIcon: 'p-tieredmenu-submenu-icon',
submenu: 'p-tieredmenu-submenu',
separator: 'p-tieredmenu-separator',
end: 'p-tieredmenu-end'
};
class TieredMenuStyle extends BaseStyle {
name = 'tieredmenu';
style = style;
classes = classes;
inlineStyles = inlineStyles;
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "21.0.5", ngImport: i0, type: TieredMenuStyle, deps: null, target: i0.ɵɵFactoryTarget.Injectable });
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "21.0.5", ngImport: i0, type: TieredMenuStyle });
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "21.0.5", ngImport: i0, type: TieredMenuStyle, decorators: [{
type: Injectable
}] });
/**
*
* TieredMenu displays submenus in nested overlays.
*
* [Live Demo](https://www.primeng.org/menu/)
*
* @module tieredmenustyle
*
*/
var TieredMenuClasses;
(function (TieredMenuClasses) {
/**
* Class name of the root element
*/
TieredMenuClasses["root"] = "p-tieredmenu";
/**
* Class name of the start element
*/
TieredMenuClasses["start"] = "p-tieredmenu-start";
/**
* Class name of the root list element
*/
TieredMenuClasses["rootList"] = "p-tieredmenu-root-list";
/**
* Class name of the item element
*/
TieredMenuClasses["item"] = "p-tieredmenu-item";
/**
* Class name of the item content element
*/
TieredMenuClasses["itemContent"] = "p-tieredmenu-item-content";
/**
* Class name of the item link element
*/
TieredMenuClasses["itemLink"] = "p-tieredmenu-item-link";
/**
* Class name of the item icon element
*/
TieredMenuClasses["itemIcon"] = "p-tieredmenu-item-icon";
/**
* Class name of the item label element
*/
TieredMenuClasses["itemLabel"] = "p-tieredmenu-item-label";
/**
* Class name of the submenu icon element
*/
TieredMenuClasses["submenuIcon"] = "p-tieredmenu-submenu-icon";
/**
* Class name of the submenu element
*/
TieredMenuClasses["submenu"] = "p-tieredmenu-submenu";
/**
* Class name of the separator element
*/
TieredMenuClasses["separator"] = "p-tieredmenu-separator";
/**
* Class name of the end element
*/
TieredMenuClasses["end"] = "p-tieredmenu-end";
})(TieredMenuClasses || (TieredMenuClasses = {}));
const TIEREDMENU_INSTANCE = new InjectionToken('TIEREDMENU_INSTANCE');
const TIEREDMENUSUB_INSTANCE = new InjectionToken('TIEREDMENUSUB_INSTANCE');
class TieredMenuSub extends BaseComponent {
el;
renderer;
tieredMenu;
get visible() {
return this._visible;
}
set visible(value) {
this._visible = value;
if (this._visible || this.root) {
this.render.set(true);
}
}
items;
itemTemplate;
root = false;
autoDisplay;
autoZIndex = true;
baseZIndex = 0;
popup;
menuId;
ariaLabel;
ariaLabelledBy;
level = 0;
focusedItemId;
activeItemPath = input([], ...(ngDevMode ? [{ debugName: "activeItemPath" }] : []));
motionOptions;
tabindex = 0;
inlineStyles;
itemClick = new EventEmitter();
itemMouseEnter = new EventEmitter();
menuFocus = new EventEmitter();
menuBlur = new EventEmitter();
menuKeydown = new EventEmitter();
sublistViewChild;
render = signal(false, ...(ngDevMode ? [{ debugName: "render" }] : []));
_componentStyle = inject(TieredMenuStyle);
bindDirectiveInstance = inject(Bind, { self: true });
$pcTieredMenu = inject(TIEREDMENU_INSTANCE, { optional: true, skipSelf: true }) ?? undefined;
$pcTieredMenuSub = inject(TIEREDMENUSUB_INSTANCE, { optional: true, skipSelf: true }) ?? undefined;
_visible = false;
onAfterViewChecked() {
this.bindDirectiveInstance.setAttrs(this.ptms(['host', 'root']));
}
constructor(el, renderer, tieredMenu) {
super();
this.el = el;
this.renderer = renderer;
this.tieredMenu = tieredMenu;
}
positionSubmenu(sublist) {
if (isPlatformBrowser(this.tieredMenu.platformId)) {
if (sublist) {
nestedPosition(sublist, this.level);
}
}
}
getItemProp(processedItem, name, params = null) {
return processedItem && processedItem.item ? resolve(processedItem.item[name], params) : undefined;
}
getItemId(processedItem) {
return processedItem.item?.id ?? `${this.menuId}_${processedItem.key}`;
}
getItemKey(processedItem) {
return this.getItemId(processedItem);
}
getItemLabel(processedItem) {
return this.getItemProp(processedItem, 'label');
}
getAriaSetSize() {
return this.items.filter((processedItem) => this.isItemVisible(processedItem) && !this.getItemProp(processedItem, 'separator')).length;
}
getAriaPosInset(index) {
return (index -
this.items.slice(0, index).filter((processedItem) => {
const isItemVisible = this.isItemVisible(processedItem);
const isVisibleSeparator = isItemVisible && this.getItemProp(processedItem, 'separator');
return !isItemVisible || isVisibleSeparator;
}).length +
1);
}
isItemVisible(processedItem) {
return this.getItemProp(processedItem, 'visible') !== false;
}
isItemActive(processedItem) {
if (this.activeItemPath()) {
return this.activeItemPath().some((path) => path.key === processedItem.key);
}
return false;
}
isItemDisabled(processedItem) {
return this.getItemProp(processedItem, 'disabled');
}
isItemFocused(processedItem) {
return this.focusedItemId === this.getItemId(processedItem);
}
isItemGroup(processedItem) {
return isNotEmpty(processedItem.items);
}
// TODO: will be removed later. Helper method to get PT from parent ContextMenu if available, otherwise use own PT
_ptm(section, options) {
return this.$pcTieredMenu ? this.$pcTieredMenu.ptm(section, options) : this.ptm(section, options);
}
getPTOptions(processedItem, index, key) {
return this._ptm(key, {
context: {
item: processedItem.item,
index,
active: this.isItemActive(processedItem),
focused: this.isItemFocused(processedItem),
disabled: this.isItemDisabled(processedItem)
}
});
}
onItemMouseEnter(param) {
if (this.autoDisplay) {
const { event, processedItem } = param;
this.itemMouseEnter.emit({ originalEvent: event, processedItem });
}
}
onItemClick(event, processedItem) {
this.getItemProp(processedItem, 'command', { originalEvent: event, item: processedItem.item });
this.itemClick.emit({ originalEvent: event, processedItem, isFocus: true });
}
onBeforeEnter(event) {
this.positionSubmenu(event.element);
}
onAfterLeave() {
this.render.set(false);
}
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "21.0.5", ngImport: i0, type: TieredMenuSub, deps: [{ token: i0.ElementRef }, { token: i0.Renderer2 }, { token: forwardRef(() => TieredMenu) }], target: i0.ɵɵFactoryTarget.Component });
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "21.0.5", type: TieredMenuSub, isStandalone: true, selector: "p-tieredMenuSub, p-tieredmenusub", inputs: { visible: { classPropertyName: "visible", publicName: "visible", isSignal: false, isRequired: false, transformFunction: null }, items: { classPropertyName: "items", publicName: "items", isSignal: false, isRequired: false, transformFunction: null }, itemTemplate: { classPropertyName: "itemTemplate", publicName: "itemTemplate", isSignal: false, isRequired: false, transformFunction: null }, root: { classPropertyName: "root", publicName: "root", isSignal: false, isRequired: false, transformFunction: booleanAttribute }, autoDisplay: { classPropertyName: "autoDisplay", publicName: "autoDisplay", isSignal: false, isRequired: false, transformFunction: booleanAttribute }, autoZIndex: { classPropertyName: "autoZIndex", publicName: "autoZIndex", isSignal: false, isRequired: false, transformFunction: booleanAttribute }, baseZIndex: { classPropertyName: "baseZIndex", publicName: "baseZIndex", isSignal: false, isRequired: false, transformFunction: numberAttribute }, popup: { classPropertyName: "popup", publicName: "popup", isSignal: false, isRequired: false, transformFunction: booleanAttribute }, menuId: { classPropertyName: "menuId", publicName: "menuId", isSignal: false, isRequired: false, transformFunction: null }, ariaLabel: { classPropertyName: "ariaLabel", publicName: "ariaLabel", isSignal: false, isRequired: false, transformFunction: null }, ariaLabelledBy: { classPropertyName: "ariaLabelledBy", publicName: "ariaLabelledBy", isSignal: false, isRequired: false, transformFunction: null }, level: { classPropertyName: "level", publicName: "level", isSignal: false, isRequired: false, transformFunction: numberAttribute }, focusedItemId: { classPropertyName: "focusedItemId", publicName: "focusedItemId", isSignal: false, isRequired: false, transformFunction: null }, activeItemPath: { classPropertyName: "activeItemPath", publicName: "activeItemPath", isSignal: true, isRequired: false, transformFunction: null }, motionOptions: { classPropertyName: "motionOptions", publicName: "motionOptions", isSignal: false, isRequired: false, transformFunction: null }, tabindex: { classPropertyName: "tabindex", publicName: "tabindex", isSignal: false, isRequired: false, transformFunction: numberAttribute }, inlineStyles: { classPropertyName: "inlineStyles", publicName: "inlineStyles", isSignal: false, isRequired: false, transformFunction: null } }, outputs: { itemClick: "itemClick", itemMouseEnter: "itemMouseEnter", menuFocus: "menuFocus", menuBlur: "menuBlur", menuKeydown: "menuKeydown" }, providers: [
{ provide: TIEREDMENUSUB_INSTANCE, useExisting: forwardRef(() => TieredMenuSub) },
{ provide: PARENT_INSTANCE, useExisting: forwardRef(() => TieredMenuSub) }
], viewQueries: [{ propertyName: "sublistViewChild", first: true, predicate: ["sublist"], descendants: true }], usesInheritance: true, hostDirectives: [{ directive: i1.Bind }], ngImport: i0, template: `
@if (render()) {
<ul
#sublist
role="menu"
[class]="root ? cx('rootList') : cx('submenu')"
[id]="menuId + '_list'"
[tabindex]="tabindex"
[attr.aria-label]="ariaLabel"
[attr.aria-labelledBy]="ariaLabelledBy"
[attr.aria-activedescendant]="focusedItemId"
[attr.aria-orientation]="'vertical'"
[pBind]="_ptm(root ? 'rootList' : 'submenu')"
(keydown)="menuKeydown.emit($event)"
(focus)="menuFocus.emit($event)"
(blur)="menuBlur.emit($event)"
[style]="inlineStyles"
[pMotion]="root ? true : visible"
[pMotionDisabled]="root"
[pMotionAppear]="true"
[pMotionName]="'p-anchored-overlay'"
[pMotionOptions]="motionOptions"
(pMotionOnBeforeEnter)="onBeforeEnter($event)"
(pMotionOnAfterLeave)="onAfterLeave()"
>
<ng-template ngFor let-processedItem [ngForOf]="items" let-index="index">
<li
*ngIf="isItemVisible(processedItem) && getItemProp(processedItem, 'separator')"
[attr.id]="getItemId(processedItem)"
[style]="getItemProp(processedItem, 'style')"
[class]="cn(cx('separator'), getItemProp(processedItem, 'class'), getItemProp(processedItem, 'styleClass'))"
role="separator"
[pBind]="_ptm('separator')"
></li>
<li
#listItem
*ngIf="isItemVisible(processedItem) && !getItemProp(processedItem, 'separator')"
role="menuitem"
[attr.id]="getItemId(processedItem)"
[attr.data-p-highlight]="isItemActive(processedItem)"
[attr.data-p-focused]="isItemFocused(processedItem)"
[attr.data-p-disabled]="isItemDisabled(processedItem)"
[attr.aria-label]="getItemLabel(processedItem)"
[attr.aria-disabled]="isItemDisabled(processedItem) || undefined"
[attr.aria-haspopup]="isItemGroup(processedItem) && !getItemProp(processedItem, 'to') ? 'menu' : undefined"
[attr.aria-expanded]="isItemGroup(processedItem) ? isItemActive(processedItem) : undefined"
[attr.aria-setsize]="getAriaSetSize()"
[attr.aria-posinset]="getAriaPosInset(index)"
[ngStyle]="getItemProp(processedItem, 'style')"
[class]="cn(cx('item', { processedItem }), getItemProp(processedItem, 'styleClass'))"
[pBind]="getPTOptions(processedItem, index, 'item')"
pTooltip
[tooltipOptions]="getItemProp(processedItem, 'tooltipOptions')"
[pTooltipUnstyled]="unstyled()"
>
<div [class]="cx('itemContent')" [pBind]="getPTOptions(processedItem, index, 'itemContent')" (click)="onItemClick($event, processedItem)" (mouseenter)="onItemMouseEnter({ $event, processedItem })">
<ng-container *ngIf="!itemTemplate">
<a
*ngIf="!getItemProp(processedItem, 'routerLink')"
[attr.href]="getItemProp(processedItem, 'url')"
[attr.data-automationid]="getItemProp(processedItem, 'automationId')"
[attr.title]="getItemProp(processedItem, 'title')"
[target]="getItemProp(processedItem, 'target')"
[class]="cn(cx('itemLink'), getItemProp(processedItem, 'linkClass'))"
[ngStyle]="getItemProp(processedItem, 'linkStyle')"
[attr.tabindex]="-1"
[pBind]="getPTOptions(processedItem, index, 'itemLink')"
pRipple
>
<span
*ngIf="getItemProp(processedItem, 'icon')"
[class]="cn(cx('itemIcon'), getItemProp(processedItem, 'icon'), getItemProp(processedItem, 'iconClass'))"
[ngStyle]="getItemProp(processedItem, 'iconStyle')"
[pBind]="getPTOptions(processedItem, index, 'itemIcon')"
[attr.tabindex]="-1"
>
</span>
<span
*ngIf="getItemProp(processedItem, 'escape'); else htmlLabel"
[class]="cn(cx('itemLabel'), getItemProp(processedItem, 'labelClass'))"
[ngStyle]="getItemProp(processedItem, 'labelStyle')"
[pBind]="getPTOptions(processedItem, index, 'itemLabel')"
>
{{ getItemLabel(processedItem) }}
</span>
<ng-template #htmlLabel>
<span
[class]="cn(cx('itemLabel'), getItemProp(processedItem, 'labelClass'))"
[ngStyle]="getItemProp(processedItem, 'labelStyle')"
[innerHTML]="getItemLabel(processedItem)"
[pBind]="getPTOptions(processedItem, index, 'itemLabel')"
></span>
</ng-template>
<span *ngIf="getItemProp(processedItem, 'badge')" [class]="cn(cx('itemBadge'), getItemProp(processedItem, 'badgeStyleClass'))">{{ getItemProp(processedItem, 'badge') }}</span>
<ng-container *ngIf="isItemGroup(processedItem)">
<svg
data-p-icon="angle-right"
*ngIf="!tieredMenu.submenuIconTemplate && !tieredMenu._submenuIconTemplate"
[class]="cx('submenuIcon')"
[pBind]="getPTOptions(processedItem, index, 'submenuIcon')"
[attr.aria-hidden]="true"
/>
<ng-template *ngTemplateOutlet="tieredMenu.submenuIconTemplate || tieredMenu._submenuIconTemplate" [attr.aria-hidden]="true"></ng-template>
</ng-container>
</a>
<a
*ngIf="getItemProp(processedItem, 'routerLink')"
[routerLink]="getItemProp(processedItem, 'routerLink')"
[attr.data-automationid]="getItemProp(processedItem, 'automationId')"
[attr.title]="getItemProp(processedItem, 'title')"
[attr.tabindex]="-1"
[queryParams]="getItemProp(processedItem, 'queryParams')"
[routerLinkActive]="'p-tieredmenu-item-link-active'"
[routerLinkActiveOptions]="getItemProp(processedItem, 'routerLinkActiveOptions') || { exact: false }"
[target]="getItemProp(processedItem, 'target')"
[class]="cn(cx('itemLink'), getItemProp(processedItem, 'linkClass'))"
[ngStyle]="getItemProp(processedItem, 'linkStyle')"
[fragment]="getItemProp(processedItem, 'fragment')"
[queryParamsHandling]="getItemProp(processedItem, 'queryParamsHandling')"
[preserveFragment]="getItemProp(processedItem, 'preserveFragment')"
[skipLocationChange]="getItemProp(processedItem, 'skipLocationChange')"
[replaceUrl]="getItemProp(processedItem, 'replaceUrl')"
[state]="getItemProp(processedItem, 'state')"
[pBind]="getPTOptions(processedItem, index, 'itemLink')"
pRipple
>
<span
*ngIf="getItemProp(processedItem, 'icon')"
[class]="cn(cx('itemIcon'), getItemProp(processedItem, 'icon'), getItemProp(processedItem, 'iconClass'))"
[ngStyle]="getItemProp(processedItem, 'iconStyle')"
[pBind]="getPTOptions(processedItem, index, 'itemIcon')"
[attr.aria-hidden]="true"
[attr.tabindex]="-1"
>
</span>
<span
*ngIf="getItemProp(processedItem, 'escape'); else htmlLabel"
[class]="cn(cx('itemLabel'), getItemProp(processedItem, 'labelClass'))"
[ngStyle]="getItemProp(processedItem, 'labelStyle')"
[pBind]="getPTOptions(processedItem, index, 'itemLabel')"
>
{{ getItemLabel(processedItem) }}
</span>
<ng-template #htmlLabel>
<span
[class]="cn(cx('itemLabel'), getItemProp(processedItem, 'labelClass'))"
[ngStyle]="getItemProp(processedItem, 'labelStyle')"
[innerHTML]="getItemLabel(processedItem)"
[pBind]="getPTOptions(processedItem, index, 'itemLabel')"
></span>
</ng-template>
<span *ngIf="getItemProp(processedItem, 'badge')" [class]="cn(cx('itemBadge'), getItemProp(processedItem, 'badgeStyleClass'))">{{ getItemProp(processedItem, 'badge') }}</span>
<ng-container *ngIf="isItemGroup(processedItem)">
<svg
data-p-icon="angle-right"
*ngIf="!tieredMenu.submenuIconTemplate && !tieredMenu._submenuIconTemplate"
[class]="cx('submenuIcon')"
[pBind]="getPTOptions(processedItem, index, 'submenuIcon')"
[attr.aria-hidden]="true"
/>
<ng-template *ngTemplateOutlet="tieredMenu.submenuIconTemplate || tieredMenu._submenuIconTemplate" [attr.aria-hidden]="true"></ng-template>
</ng-container>
</a>
</ng-container>
<ng-container *ngIf="itemTemplate">
<ng-template *ngTemplateOutlet="itemTemplate; context: { $implicit: processedItem.item, hasSubmenu: getItemProp(processedItem, 'items') }"></ng-template>
</ng-container>
</div>
<p-tieredmenusub
*ngIf="isItemVisible(processedItem) && isItemGroup(processedItem)"
[items]="processedItem.items"
[itemTemplate]="itemTemplate"
[autoDisplay]="autoDisplay"
[menuId]="menuId"
[visible]="isItemActive(processedItem) && isItemGroup(processedItem)"
[activeItemPath]="activeItemPath()"
[focusedItemId]="focusedItemId"
[ariaLabelledBy]="getItemId(processedItem)"
[level]="level + 1"
(itemClick)="itemClick.emit($event)"
(itemMouseEnter)="onItemMouseEnter($event)"
[pt]="pt()"
[motionOptions]="motionOptions"
[unstyled]="unstyled()"
></p-tieredmenusub>
</li>
</ng-template>
</ul>
}
`, isInline: true, dependencies: [{ kind: "component", type: TieredMenuSub, selector: "p-tieredMenuSub, p-tieredmenusub", inputs: ["visible", "items", "itemTemplate", "root", "autoDisplay", "autoZIndex", "baseZIndex", "popup", "menuId", "ariaLabel", "ariaLabelledBy", "level", "focusedItemId", "activeItemPath", "motionOptions", "tabindex", "inlineStyles"], outputs: ["itemClick", "itemMouseEnter", "menuFocus", "menuBlur", "menuKeydown"] }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i2.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "ngmodule", type: RouterModule }, { kind: "directive", type: i3.RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "directive", type: i3.RouterLinkActive, selector: "[routerLinkActive]", inputs: ["routerLinkActiveOptions", "ariaCurrentWhenActive", "routerLinkActive"], outputs: ["isActiveChange"], exportAs: ["routerLinkActive"] }, { kind: "directive", type: Ripple, selector: "[pRipple]" }, { kind: "ngmodule", type: TooltipModule }, { kind: "directive", type: i4.Tooltip, selector: "[pTooltip]", inputs: ["tooltipPosition", "tooltipEvent", "positionStyle", "tooltipStyleClass", "tooltipZIndex", "escape", "showDelay", "hideDelay", "life", "positionTop", "positionLeft", "autoHide", "fitContent", "hideOnEscape", "pTooltip", "tooltipDisabled", "tooltipOptions", "appendTo", "ptTooltip", "pTooltipPT", "pTooltipUnstyled"] }, { kind: "directive", type: i1.Bind, selector: "[pBind]", inputs: ["pBind"] }, { kind: "component", type: AngleRightIcon, selector: "[data-p-icon=\"angle-right\"]" }, { kind: "ngmodule", type: SharedModule }, { kind: "ngmodule", type: BindModule }, { kind: "ngmodule", type: MotionModule }, { kind: "directive", type: i5.MotionDirective, selector: "[pMotion]", inputs: ["pMotion", "pMotionName", "pMotionType", "pMotionSafe", "pMotionDisabled", "pMotionAppear", "pMotionEnter", "pMotionLeave", "pMotionDuration", "pMotionHideStrategy", "pMotionEnterFromClass", "pMotionEnterToClass", "pMotionEnterActiveClass", "pMotionLeaveFromClass", "pMotionLeaveToClass", "pMotionLeaveActiveClass", "pMotionOptions"], outputs: ["pMotionOnBeforeEnter", "pMotionOnEnter", "pMotionOnAfterEnter", "pMotionOnEnterCancelled", "pMotionOnBeforeLeave", "pMotionOnLeave", "pMotionOnAfterLeave", "pMotionOnLeaveCancelled"] }], encapsulation: i0.ViewEncapsulation.None });
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "21.0.5", ngImport: i0, type: TieredMenuSub, decorators: [{
type: Component,
args: [{
selector: 'p-tieredMenuSub, p-tieredmenusub',
standalone: true,
imports: [CommonModule, RouterModule, Ripple, TooltipModule, AngleRightIcon, SharedModule, BindModule, MotionModule],
template: `
@if (render()) {
<ul
#sublist
role="menu"
[class]="root ? cx('rootList') : cx('submenu')"
[id]="menuId + '_list'"
[tabindex]="tabindex"
[attr.aria-label]="ariaLabel"
[attr.aria-labelledBy]="ariaLabelledBy"
[attr.aria-activedescendant]="focusedItemId"
[attr.aria-orientation]="'vertical'"
[pBind]="_ptm(root ? 'rootList' : 'submenu')"
(keydown)="menuKeydown.emit($event)"
(focus)="menuFocus.emit($event)"
(blur)="menuBlur.emit($event)"
[style]="inlineStyles"
[pMotion]="root ? true : visible"
[pMotionDisabled]="root"
[pMotionAppear]="true"
[pMotionName]="'p-anchored-overlay'"
[pMotionOptions]="motionOptions"
(pMotionOnBeforeEnter)="onBeforeEnter($event)"
(pMotionOnAfterLeave)="onAfterLeave()"
>
<ng-template ngFor let-processedItem [ngForOf]="items" let-index="index">
<li
*ngIf="isItemVisible(processedItem) && getItemProp(processedItem, 'separator')"
[attr.id]="getItemId(processedItem)"
[style]="getItemProp(processedItem, 'style')"
[class]="cn(cx('separator'), getItemProp(processedItem, 'class'), getItemProp(processedItem, 'styleClass'))"
role="separator"
[pBind]="_ptm('separator')"
></li>
<li
#listItem
*ngIf="isItemVisible(processedItem) && !getItemProp(processedItem, 'separator')"
role="menuitem"
[attr.id]="getItemId(processedItem)"
[attr.data-p-highlight]="isItemActive(processedItem)"
[attr.data-p-focused]="isItemFocused(processedItem)"
[attr.data-p-disabled]="isItemDisabled(processedItem)"
[attr.aria-label]="getItemLabel(processedItem)"
[attr.aria-disabled]="isItemDisabled(processedItem) || undefined"
[attr.aria-haspopup]="isItemGroup(processedItem) && !getItemProp(processedItem, 'to') ? 'menu' : undefined"
[attr.aria-expanded]="isItemGroup(processedItem) ? isItemActive(processedItem) : undefined"
[attr.aria-setsize]="getAriaSetSize()"
[attr.aria-posinset]="getAriaPosInset(index)"
[ngStyle]="getItemProp(processedItem, 'style')"
[class]="cn(cx('item', { processedItem }), getItemProp(processedItem, 'styleClass'))"
[pBind]="getPTOptions(processedItem, index, 'item')"
pTooltip
[tooltipOptions]="getItemProp(processedItem, 'tooltipOptions')"
[pTooltipUnstyled]="unstyled()"
>
<div [class]="cx('itemContent')" [pBind]="getPTOptions(processedItem, index, 'itemContent')" (click)="onItemClick($event, processedItem)" (mouseenter)="onItemMouseEnter({ $event, processedItem })">
<ng-container *ngIf="!itemTemplate">
<a
*ngIf="!getItemProp(processedItem, 'routerLink')"
[attr.href]="getItemProp(processedItem, 'url')"
[attr.data-automationid]="getItemProp(processedItem, 'automationId')"
[attr.title]="getItemProp(processedItem, 'title')"
[target]="getItemProp(processedItem, 'target')"
[class]="cn(cx('itemLink'), getItemProp(processedItem, 'linkClass'))"
[ngStyle]="getItemProp(processedItem, 'linkStyle')"
[attr.tabindex]="-1"
[pBind]="getPTOptions(processedItem, index, 'itemLink')"
pRipple
>
<span
*ngIf="getItemProp(processedItem, 'icon')"
[class]="cn(cx('itemIcon'), getItemProp(processedItem, 'icon'), getItemProp(processedItem, 'iconClass'))"
[ngStyle]="getItemProp(processedItem, 'iconStyle')"
[pBind]="getPTOptions(processedItem, index, 'itemIcon')"
[attr.tabindex]="-1"
>
</span>
<span
*ngIf="getItemProp(processedItem, 'escape'); else htmlLabel"
[class]="cn(cx('itemLabel'), getItemProp(processedItem, 'labelClass'))"
[ngStyle]="getItemProp(processedItem, 'labelStyle')"
[pBind]="getPTOptions(processedItem, index, 'itemLabel')"
>
{{ getItemLabel(processedItem) }}
</span>
<ng-template #htmlLabel>
<span
[class]="cn(cx('itemLabel'), getItemProp(processedItem, 'labelClass'))"
[ngStyle]="getItemProp(processedItem, 'labelStyle')"
[innerHTML]="getItemLabel(processedItem)"
[pBind]="getPTOptions(processedItem, index, 'itemLabel')"
></span>
</ng-template>
<span *ngIf="getItemProp(processedItem, 'badge')" [class]="cn(cx('itemBadge'), getItemProp(processedItem, 'badgeStyleClass'))">{{ getItemProp(processedItem, 'badge') }}</span>
<ng-container *ngIf="isItemGroup(processedItem)">
<svg
data-p-icon="angle-right"
*ngIf="!tieredMenu.submenuIconTemplate && !tieredMenu._submenuIconTemplate"
[class]="cx('submenuIcon')"
[pBind]="getPTOptions(processedItem, index, 'submenuIcon')"
[attr.aria-hidden]="true"
/>
<ng-template *ngTemplateOutlet="tieredMenu.submenuIconTemplate || tieredMenu._submenuIconTemplate" [attr.aria-hidden]="true"></ng-template>
</ng-container>
</a>
<a
*ngIf="getItemProp(processedItem, 'routerLink')"
[routerLink]="getItemProp(processedItem, 'routerLink')"
[attr.data-automationid]="getItemProp(processedItem, 'automationId')"
[attr.title]="getItemProp(processedItem, 'title')"
[attr.tabindex]="-1"
[queryParams]="getItemProp(processedItem, 'queryParams')"
[routerLinkActive]="'p-tieredmenu-item-link-active'"
[routerLinkActiveOptions]="getItemProp(processedItem, 'routerLinkActiveOptions') || { exact: false }"
[target]="getItemProp(processedItem, 'target')"
[class]="cn(cx('itemLink'), getItemProp(processedItem, 'linkClass'))"
[ngStyle]="getItemProp(processedItem, 'linkStyle')"
[fragment]="getItemProp(processedItem, 'fragment')"
[queryParamsHandling]="getItemProp(processedItem, 'queryParamsHandling')"
[preserveFragment]="getItemProp(processedItem, 'preserveFragment')"
[skipLocationChange]="getItemProp(processedItem, 'skipLocationChange')"
[replaceUrl]="getItemProp(processedItem, 'replaceUrl')"
[state]="getItemProp(processedItem, 'state')"
[pBind]="getPTOptions(processedItem, index, 'itemLink')"
pRipple
>
<span
*ngIf="getItemProp(processedItem, 'icon')"
[class]="cn(cx('itemIcon'), getItemProp(processedItem, 'icon'), getItemProp(processedItem, 'iconClass'))"
[ngStyle]="getItemProp(processedItem, 'iconStyle')"
[pBind]="getPTOptions(processedItem, index, 'itemIcon')"
[attr.aria-hidden]="true"
[attr.tabindex]="-1"
>
</span>
<span
*ngIf="getItemProp(processedItem, 'escape'); else htmlLabel"
[class]="cn(cx('itemLabel'), getItemProp(processedItem, 'labelClass'))"
[ngStyle]="getItemProp(processedItem, 'labelStyle')"
[pBind]="getPTOptions(processedItem, index, 'itemLabel')"
>
{{ getItemLabel(processedItem) }}
</span>
<ng-template #htmlLabel>
<span
[class]="cn(cx('itemLabel'), getItemProp(processedItem, 'labelClass'))"
[ngStyle]="getItemProp(processedItem, 'labelStyle')"
[innerHTML]="getItemLabel(processedItem)"
[pBind]="getPTOptions(processedItem, index, 'itemLabel')"
></span>
</ng-template>
<span *ngIf="getItemProp(processedItem, 'badge')" [class]="cn(cx('itemBadge'), getItemProp(processedItem, 'badgeStyleClass'))">{{ getItemProp(processedItem, 'badge') }}</span>
<ng-container *ngIf="isItemGroup(processedItem)">
<svg
data-p-icon="angle-right"
*ngIf="!tieredMenu.submenuIconTemplate && !tieredMenu._submenuIconTemplate"
[class]="cx('submenuIcon')"
[pBind]="getPTOptions(processedItem, index, 'submenuIcon')"
[attr.aria-hidden]="true"
/>
<ng-template *ngTemplateOutlet="tieredMenu.submenuIconTemplate || tieredMenu._submenuIconTemplate" [attr.aria-hidden]="true"></ng-template>
</ng-container>
</a>
</ng-container>
<ng-container *ngIf="itemTemplate">
<ng-template *ngTemplateOutlet="itemTemplate; context: { $implicit: processedItem.item, hasSubmenu: getItemProp(processedItem, 'items') }"></ng-template>
</ng-container>
</div>
<p-tieredmenusub
*ngIf="isItemVisible(processedItem) && isItemGroup(processedItem)"
[items]="processedItem.items"
[itemTemplate]="itemTemplate"
[autoDisplay]="autoDisplay"
[menuId]="menuId"
[visible]="isItemActive(processedItem) && isItemGroup(processedItem)"
[activeItemPath]="activeItemPath()"
[focusedItemId]="focusedItemId"
[ariaLabelledBy]="getItemId(processedItem)"
[level]="level + 1"
(itemClick)="itemClick.emit($event)"
(itemMouseEnter)="onItemMouseEnter($event)"
[pt]="pt()"
[motionOptions]="motionOptions"
[unstyled]="unstyled()"
></p-tieredmenusub>
</li>
</ng-template>
</ul>
}
`,
encapsulation: ViewEncapsulation.None,
providers: [
{ provide: TIEREDMENUSUB_INSTANCE, useExisting: forwardRef(() => TieredMenuSub) },
{ provide: PARENT_INSTANCE, useExisting: forwardRef(() => TieredMenuSub) }
],
hostDirectives: [Bind]
}]
}], ctorParameters: () => [{ type: i0.ElementRef }, { type: i0.Renderer2 }, { type: TieredMenu, decorators: [{
type: Inject,
args: [forwardRef(() => TieredMenu)]
}] }], propDecorators: { visible: [{
type: Input
}], items: [{
type: Input
}], itemTemplate: [{
type: Input
}], root: [{
type: Input,
args: [{ transform: booleanAttribute }]
}], autoDisplay: [{
type: Input,
args: [{ transform: booleanAttribute }]
}], autoZIndex: [{
type: Input,
args: [{ transform: booleanAttribute }]
}], baseZIndex: [{
type: Input,
args: [{ transform: numberAttribute }]
}], popup: [{
type: Input,
args: [{ transform: booleanAttribute }]
}], menuId: [{
type: Input
}], ariaLabel: [{
type: Input
}], ariaLabelledBy: [{
type: Input
}], level: [{
type: Input,
args: [{ transform: numberAttribute }]
}], focusedItemId: [{
type: Input
}], activeItemPath: [{ type: i0.Input, args: [{ isSignal: true, alias: "activeItemPath", required: false }] }], motionOptions: [{
type: Input
}], tabindex: [{
type: Input,
args: [{ transform: numberAttribute }]
}], inlineStyles: [{
type: Input
}], itemClick: [{
type: Output
}], itemMouseEnter: [{
type: Output
}], menuFocus: [{
type: Output
}], menuBlur: [{
type: Output
}], menuKeydown: [{
type: Output
}], sublistViewChild: [{
type: ViewChild,
args: ['sublist']
}] } });
/**
* TieredMenu displays submenus in nested overlays.
* @group Components
*/
class TieredMenu extends BaseComponent {
overlayService;
/**
* An array of menuitems.
* @group Props
*/
set model(value) {
this._model = value;
this._processedItems = this.createProcessedItems(this._model || []);
}
get model() {
return this._model;
}
/**
* Defines if menu would displayed as a popup.
* @group Props
*/
popup;
/**
* Inline style of the component.
* @group Props
*/
style;
/**
* Style class of the component.
* @group Props
*/
styleClass;
/**
* The breakpoint to define the maximum width boundary.
* @group Props
*/
breakpoint = '960px';
/**
* Whether to automatically manage layering.
* @group Props
*/
autoZIndex = true;
/**
* Base zIndex value to use in layering.
* @group Props
*/
baseZIndex = 0;
/**
* Whether to show a root submenu on mouse over.
* @defaultValue true
* @group Props
*/
autoDisplay = true;
/**
* Transition options of the show animation.
* @group Props
* @deprecated since v21.0.0, use `motionOptions` instead.
*/
showTransitionOptions = '.12s cubic-bezier(0, 0, 0.2, 1)';
/**
* Transition options of the hide animation.
* @group Props
* @deprecated since v21.0.0, use `motionOptions` instead.
*/
hideTransitionOptions = '.1s linear';
/**
* Current id state as a string.
* @group Props
*/
id;
/**
* Defines a string value that labels an interactive element.
* @group Props
*/
ariaLabel;
/**
* Identifier of the underlying input element.
* @group Props
*/
ariaLabelledBy;
/**
* When present, it specifies that the component should be disabled.
* @group Props
*/
disabled = false;
/**
* Index of the element in tabbing order.
* @group Props
*/
tabindex = 0;
/**
* Target element to attach the overlay, valid values are "body" or a local ng-template variable of another element (note: use binding with brackets for template variables, e.g. [appendTo]="mydiv" for a div element having #mydiv as variable name).
* @defaultValue 'self'
* @group Props
*/
appendTo = input(undefined, ...(ngDevMode ? [{ debugName: "appendTo" }] : []));
/**
* The motion options.
* @group Props
*/
motionOptions = input(undefined, ...(ngDevMode ? [{ debugName: "motionOptions" }] : []));
computedMotionOptions = computed(() => {
return {
...this.ptm('motion'),
...this.motionOptions()
};
}, ...(ngDevMode ? [{ debugName: "computedMotionOptions" }] : []));
/**
* Callback to invoke when overlay menu is shown.
* @group Emits
*/
onShow = new EventEmitter();
/**
* Callback to invoke when overlay menu is hidden.
* @group Emits
*/
onHide = new EventEmitter();
rootmenu;
containerViewChild;
/**
* Custom submenu icon template.
* @group Templates
*/
submenuIconTemplate;
/**
* Custom item template.
* @param {TieredMenuItemTemplateContext} context - item context.
* @see {@link TieredMenuItemTemplateContext}
* @group Templates
*/
itemTemplate;
templates;
$appendTo = computed(() => this.appendTo() || this.config.overlayAppendTo(), ...(ngDevMode ? [{ debugName: "$appendTo" }] : []));
render = signal(false, ...(ngDevMode ? [{ debugName: "render" }] : []));
container;
outsideClickListener;
resizeListener;
scrollHandler;
target;
relatedTarget;
visible;
dirty = false;
focused = false;
activeItemPath = signal([], ...(ngDevMode ? [{ debugName: "activeItemPath" }] : []));
number = signal(0, ...(ngDevMode ? [{ debugName: "number" }] : []));
focusedItemInfo = signal({ index: -1, level: 0, parentKey: '', item: null }, ...(ngDevMode ? [{ debugName: "focusedItemInfo"