ng-material-multilevel-menu-2
Version:
697 lines (685 loc) • 73.4 kB
JavaScript
(function (global, factory) {
typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports, require('@angular/core'), require('@angular/material'), require('@angular/router'), require('@angular/animations'), require('@angular/common')) :
typeof define === 'function' && define.amd ? define('ng-material-multilevel-menu-2', ['exports', '@angular/core', '@angular/material', '@angular/router', '@angular/animations', '@angular/common'], factory) :
(factory((global['ng-material-multilevel-menu-2'] = {}),global.ng.core,global.ng.material,global.ng.router,global.ng.animations,global.ng.common));
}(this, (function (exports,i0,material,router,animations,common) { 'use strict';
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,missingReturn,uselessCode} checked by tsc
*/
var MaterialsModule = /** @class */ (function () {
function MaterialsModule() {
}
MaterialsModule.decorators = [
{ type: i0.NgModule, args: [{
imports: [
material.MatIconModule,
material.MatListModule,
material.MatRippleModule,
],
declarations: [],
exports: [
material.MatIconModule,
material.MatListModule,
material.MatRippleModule,
]
},] },
];
return MaterialsModule;
}());
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,missingReturn,uselessCode} checked by tsc
*/
var MultilevelMenuService = /** @class */ (function () {
function MultilevelMenuService() {
}
/**
* @return {?}
*/
MultilevelMenuService.prototype.generateId = /**
* @return {?}
*/
function () {
/** @type {?} */
var text = '';
/** @type {?} */
var possible = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';
for (var i = 0; i < 20; i++) {
text += possible.charAt(Math.floor(Math.random() * possible.length));
}
return text;
};
/**
* @param {?} nodes
* @return {?}
*/
MultilevelMenuService.prototype.addRandomId = /**
* @param {?} nodes
* @return {?}
*/
function (nodes) {
var _this = this;
nodes.forEach(function (node) {
node.id = _this.generateId();
if (node.items !== undefined) {
_this.addRandomId(node.items);
}
});
};
/**
* @param {?} node
* @param {?} nodeId
* @return {?}
*/
MultilevelMenuService.prototype.recursiveCheckId = /**
* @param {?} node
* @param {?} nodeId
* @return {?}
*/
function (node, nodeId) {
var _this = this;
if (node.id === nodeId) {
return true;
}
else {
if (node.items !== undefined) {
return node.items.some(function (nestedNode) {
return _this.recursiveCheckId(nestedNode, nodeId);
});
}
}
};
/**
* @param {?} nodes
* @param {?} link
* @return {?}
*/
MultilevelMenuService.prototype.recursiveCheckLink = /**
* @param {?} nodes
* @param {?} link
* @return {?}
*/
function (nodes, link) {
for (var nodeIndex = 0; nodeIndex < nodes.length; nodeIndex++) {
/** @type {?} */
var node = nodes[nodeIndex];
for (var key in node) {
if (node.hasOwnProperty(key)) {
if (encodeURI(node.link) === link) {
this.foundLinkObject = node;
}
else {
if (node.items !== undefined) {
this.recursiveCheckLink(node.items, link);
}
}
}
}
}
};
/**
* @param {?} node
* @param {?} link
* @return {?}
*/
MultilevelMenuService.prototype.getMatchedObjectByUrl = /**
* @param {?} node
* @param {?} link
* @return {?}
*/
function (node, link) {
this.recursiveCheckLink(node, link);
return this.foundLinkObject;
};
MultilevelMenuService.decorators = [
{ type: i0.Injectable, args: [{
providedIn: 'root'
},] },
];
/** @nocollapse */ MultilevelMenuService.ngInjectableDef = i0.defineInjectable({ factory: function MultilevelMenuService_Factory() { return new MultilevelMenuService(); }, token: MultilevelMenuService, providedIn: "root" });
return MultilevelMenuService;
}());
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,missingReturn,uselessCode} checked by tsc
*/
/** @type {?} */
var CONSTANT = {
PADDING_AT_START: true,
DEFAULT_CLASS_NAME: "amml-container",
DEFAULT_LIST_CLASS_NAME: "amml-item",
SELECTED_LIST_CLASS_NAME: "selected-amml-item",
DEFAULT_SELECTED_FONT_COLOR: "#1976d2",
DEFAULT_LIST_BACKGROUND_COLOR: "transparent",
DEFAULT_LIST_FONT_COLOR: "rgba(0,0,0,.87)",
ERROR_MESSAGE: "Invalid data for material Multilevel List Component"
};
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,missingReturn,uselessCode} checked by tsc
*/
var NgMaterialMultilevelMenuComponent = /** @class */ (function () {
function NgMaterialMultilevelMenuComponent(router$$1, multilevelMenuService) {
this.router = router$$1;
this.multilevelMenuService = multilevelMenuService;
this.configuration = null;
this.selectedItem = new i0.EventEmitter();
this.selectedLabel = new i0.EventEmitter();
this.hasDivider = true;
this.nodeConfig = {
paddingAtStart: true,
listBackgroundColor: null,
fontColor: null,
selectedListFontColor: null,
interfaceWithRoute: null,
collapseOnSelect: null,
highlightOnSelect: false,
rtlLayout: false
};
this.isInvalidConfig = true;
}
/**
* @return {?}
*/
NgMaterialMultilevelMenuComponent.prototype.ngOnChanges = /**
* @return {?}
*/
function () {
this.checkValiddata();
this.detectInvalidConfig();
};
/**
* @return {?}
*/
NgMaterialMultilevelMenuComponent.prototype.ngOnInit = /**
* @return {?}
*/
function () {
var _this = this;
if (this.configuration !== null &&
this.configuration !== undefined &&
this.configuration !== '' &&
this.configuration.interfaceWithRoute !== null &&
this.configuration.interfaceWithRoute) {
this.router.events.subscribe(function (event) {
if (event instanceof router.NavigationEnd) {
_this.updateNodeByURL(event.url);
}
});
this.updateNodeByURL(this.router.url);
}
};
/**
* @param {?} url
* @return {?}
*/
NgMaterialMultilevelMenuComponent.prototype.updateNodeByURL = /**
* @param {?} url
* @return {?}
*/
function (url) {
/** @type {?} */
var foundNode = this.multilevelMenuService.getMatchedObjectByUrl(this.items, url);
if (foundNode !== undefined &&
foundNode.link !== undefined &&
foundNode.link !== null &&
foundNode.link !== '') {
this.currentNode = foundNode;
this.selectedListItem(foundNode);
}
};
/**
* @return {?}
*/
NgMaterialMultilevelMenuComponent.prototype.checkValiddata = /**
* @return {?}
*/
function () {
if (this.items.length === 0) {
console.warn(CONSTANT.ERROR_MESSAGE);
}
else {
this.items = this.items.filter(function (n) { return !n.hidden; });
this.multilevelMenuService.addRandomId(this.items);
}
};
/**
* @return {?}
*/
NgMaterialMultilevelMenuComponent.prototype.detectInvalidConfig = /**
* @return {?}
*/
function () {
if (this.configuration === null ||
this.configuration === undefined ||
this.configuration === '') {
this.isInvalidConfig = true;
}
else {
this.isInvalidConfig = false;
/** @type {?} */
var config = this.configuration;
if (config.paddingAtStart !== undefined &&
config.paddingAtStart !== null &&
typeof config.paddingAtStart === 'boolean') {
this.nodeConfig.paddingAtStart = config.paddingAtStart;
}
if (config.listBackgroundColor !== '' &&
config.listBackgroundColor !== null &&
config.listBackgroundColor !== undefined) {
this.nodeConfig.listBackgroundColor = config.listBackgroundColor;
}
if (config.fontColor !== '' &&
config.fontColor !== null &&
config.fontColor !== undefined) {
this.nodeConfig.fontColor = config.fontColor;
}
if (config.selectedListFontColor !== '' &&
config.selectedListFontColor !== null &&
config.selectedListFontColor !== undefined) {
this.nodeConfig.selectedListFontColor = config.selectedListFontColor;
}
if (config.interfaceWithRoute !== null &&
config.interfaceWithRoute !== undefined &&
typeof config.interfaceWithRoute === 'boolean') {
this.nodeConfig.interfaceWithRoute = config.interfaceWithRoute;
}
if (config.collapseOnSelect !== null &&
config.collapseOnSelect !== undefined &&
typeof config.collapseOnSelect === 'boolean') {
this.nodeConfig.collapseOnSelect = config.collapseOnSelect;
}
if (config.highlightOnSelect !== null &&
config.highlightOnSelect !== undefined &&
typeof config.highlightOnSelect === 'boolean') {
this.nodeConfig.highlightOnSelect = config.highlightOnSelect;
}
if (config.rtlLayout !== null &&
config.rtlLayout !== undefined &&
typeof config.rtlLayout === 'boolean') {
this.nodeConfig.rtlLayout = config.rtlLayout;
}
}
};
/**
* @return {?}
*/
NgMaterialMultilevelMenuComponent.prototype.getClassName = /**
* @return {?}
*/
function () {
if (this.isInvalidConfig) {
return CONSTANT.DEFAULT_CLASS_NAME;
}
else {
if (this.configuration.classname !== '' &&
this.configuration.classname !== null &&
this.configuration.classname !== undefined) {
return CONSTANT.DEFAULT_CLASS_NAME + " " + this.configuration.classname;
}
else {
return CONSTANT.DEFAULT_CLASS_NAME;
}
}
};
/**
* @return {?}
*/
NgMaterialMultilevelMenuComponent.prototype.getGlobalStyle = /**
* @return {?}
*/
function () {
if (!this.isInvalidConfig) {
/** @type {?} */
var styles = {
background: null
};
if (this.configuration.backgroundColor !== '' &&
this.configuration.backgroundColor !== null &&
this.configuration.backgroundColor !== undefined) {
styles.background = this.configuration.backgroundColor;
}
return styles;
}
};
/**
* @return {?}
*/
NgMaterialMultilevelMenuComponent.prototype.isRtlLayout = /**
* @return {?}
*/
function () {
return this.nodeConfig.rtlLayout;
};
/**
* @param {?} event
* @return {?}
*/
NgMaterialMultilevelMenuComponent.prototype.selectedListItem = /**
* @param {?} event
* @return {?}
*/
function (event) {
this.currentNode = event;
if (event.items === undefined &&
(!event.onSelected || typeof event.onSelected !== 'function')) {
this.selectedItem.emit(event);
}
else {
this.selectedLabel.emit(event);
}
};
NgMaterialMultilevelMenuComponent.decorators = [
{ type: i0.Component, args: [{
selector: 'ng-material-multilevel-menu',
template: "<div [ngClass]=\"getClassName()\" [ngStyle]=\"getGlobalStyle()\" *ngIf='items.length !== 0' [dir]=\"isRtlLayout() ? 'rtl' : 'ltr'\">\n <mat-list>\n <ng-list-item *ngFor=\"let node of items\" [nodeConfiguration]='nodeConfig' [node]='node' [selectedNode]='currentNode' [hasDivider]=\"hasDivider\" (selectedItem)=\"selectedListItem($event)\n \">\n </ng-list-item>\n </mat-list>\n</div>",
styles: [".amml-item{line-height:48px;display:flex;justify-content:space-between;position:relative}.anml-data{width:100%;text-transform:capitalize;display:flex;justify-content:start}.amml-icon-fa{font-size:20px}.amml-icon{line-height:48px}.active{color:#1976d2}div[dir=ltr] .amml-icon{margin-right:15px}div[dir=ltr] .amml-submenu{margin-left:16px}div[dir=rtl] .amml-icon{margin-left:15px}div[dir=rtl] .amml-submenu{margin-right:16px}"]
},] },
];
NgMaterialMultilevelMenuComponent.ctorParameters = function () {
return [
{ type: router.Router },
{ type: MultilevelMenuService }
];
};
NgMaterialMultilevelMenuComponent.propDecorators = {
items: [{ type: i0.Input }],
configuration: [{ type: i0.Input }],
selectedItem: [{ type: i0.Output }],
selectedLabel: [{ type: i0.Output }],
hasDivider: [{ type: i0.Input }]
};
return NgMaterialMultilevelMenuComponent;
}());
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,missingReturn,uselessCode} checked by tsc
*/
var ListItemComponent = /** @class */ (function () {
function ListItemComponent(router$$1, multilevelMenuService) {
var _a;
this.router = router$$1;
this.multilevelMenuService = multilevelMenuService;
this.level = 1;
this.nodeConfiguration = null;
this.selectedItem = new i0.EventEmitter();
this.hasDivider = true;
this.isSelected = false;
this.expanded = false;
this.firstInitializer = false;
this.selectedListClasses = (_a = {},
_a[CONSTANT.DEFAULT_LIST_CLASS_NAME] = true,
_a[CONSTANT.SELECTED_LIST_CLASS_NAME] = false,
_a);
}
/**
* @return {?}
*/
ListItemComponent.prototype.ngOnChanges = /**
* @return {?}
*/
function () {
this.nodeChildren =
this.node && this.node.items
? this.node.items.filter(function (n) { return !n.hidden; })
: [];
if (this.selectedNode !== undefined && this.selectedNode !== null) {
this.setSelectedClass(this.multilevelMenuService.recursiveCheckId(this.node, this.selectedNode.id));
}
};
/**
* @param {?} isFound
* @return {?}
*/
ListItemComponent.prototype.setSelectedClass = /**
* @param {?} isFound
* @return {?}
*/
function (isFound) {
var _a;
if (isFound) {
if (!this.firstInitializer) {
this.expanded = true;
}
this.isSelected =
this.nodeConfiguration.highlightOnSelect ||
this.selectedNode.items === undefined
? true
: false;
}
else {
this.isSelected = false;
if (this.nodeConfiguration.collapseOnSelect) {
this.expanded = false;
}
}
this.selectedListClasses = (_a = {},
_a[CONSTANT.DEFAULT_LIST_CLASS_NAME] = true,
_a[CONSTANT.SELECTED_LIST_CLASS_NAME] = this.isSelected,
_a);
this.setClasses();
};
/**
* @return {?}
*/
ListItemComponent.prototype.getPaddingAtStart = /**
* @return {?}
*/
function () {
return this.nodeConfiguration.paddingAtStart ? true : false;
};
/**
* @return {?}
*/
ListItemComponent.prototype.getListStyle = /**
* @return {?}
*/
function () {
/** @type {?} */
var styles = {
background: CONSTANT.DEFAULT_LIST_BACKGROUND_COLOR,
color: CONSTANT.DEFAULT_LIST_FONT_COLOR
};
if (this.nodeConfiguration.listBackgroundColor !== null) {
styles.background = this.nodeConfiguration.listBackgroundColor;
}
if (this.isSelected) {
this.nodeConfiguration.selectedListFontColor !== null
? (styles.color = this.nodeConfiguration.selectedListFontColor)
: (styles.color = CONSTANT.DEFAULT_SELECTED_FONT_COLOR);
}
else if (this.nodeConfiguration.fontColor !== null) {
styles.color = this.nodeConfiguration.fontColor;
}
return styles;
};
/**
* @param {?} node
* @return {?}
*/
ListItemComponent.prototype.getListIcon = /**
* @param {?} node
* @return {?}
*/
function (node) {
if (node.icon !== null && node.icon !== undefined && node.icon !== '') {
return "icon";
}
else if (node.faIcon !== null &&
node.faIcon !== undefined &&
node.faIcon !== '') {
return "faicon";
}
else if (node.imageIcon !== null &&
node.imageIcon !== undefined &&
node.imageIcon !== '') {
return "imageicon";
}
else {
return "";
}
};
/**
* @return {?}
*/
ListItemComponent.prototype.hasItems = /**
* @return {?}
*/
function () {
return this.nodeChildren.length > 0 ? true : false;
};
/**
* @return {?}
*/
ListItemComponent.prototype.isRtlLayout = /**
* @return {?}
*/
function () {
return this.nodeConfiguration.rtlLayout;
};
/**
* @return {?}
*/
ListItemComponent.prototype.setClasses = /**
* @return {?}
*/
function () {
var _a;
this.classes = (_a = {},
_a['level-' + this.level] = true,
_a['amml-submenu'] = this.hasItems() && this.getPaddingAtStart(),
_a);
};
/**
* @param {?} node
* @return {?}
*/
ListItemComponent.prototype.expand = /**
* @param {?} node
* @return {?}
*/
function (node) {
this.expanded = !this.expanded;
this.firstInitializer = true;
this.setClasses();
if (this.nodeConfiguration.interfaceWithRoute !== null &&
this.nodeConfiguration.interfaceWithRoute &&
node.link !== undefined &&
node.link) {
if (node.externalRedirect !== undefined && node.externalRedirect) {
window.location.href = node.link;
}
else {
this.router.navigate([node.link]);
}
}
else if (node.onSelected && typeof node.onSelected === 'function') {
node.onSelected(node);
this.selectedListItem(node);
}
else if (node.items === undefined ||
this.nodeConfiguration.collapseOnSelect) {
this.selectedListItem(node);
}
};
/**
* @param {?} node
* @return {?}
*/
ListItemComponent.prototype.selectedListItem = /**
* @param {?} node
* @return {?}
*/
function (node) {
this.selectedItem.emit(node);
};
ListItemComponent.decorators = [
{ type: i0.Component, args: [{
selector: 'ng-list-item',
template: "<mat-list-item matRipple [ngClass]=\"selectedListClasses\" *ngIf=\"!node.hidden\" (click)=\"expand(node)\" title=\"{{node.label}}\" [ngStyle]=\"getListStyle()\">\n <div class=\"anml-data\" [dir]=\"isRtlLayout() ? 'rtl' : 'ltr'\">\n <div class=\"icon-container\" [ngSwitch]=\"getListIcon(node)\">\n <span *ngSwitchCase=\"'faicon'\" class=\"amml-icon amml-icon-fa\">\n <i [ngClass]=\"node.faIcon\"></i>\n </span>\n <mat-icon *ngSwitchCase=\"'icon'\" class=\"amml-icon\">\n {{node.icon}}\n </mat-icon>\n <img matListAvatar *ngSwitchCase=\"'imageicon'\" class=\"amml-icon\" src=\"{{node.imageIcon}}\" alt=\"{{node.label}}\" />\n </div>\n <span class=\"label\">{{node.label}}</span>\n </div>\n <ng-container *ngIf='hasItems()'>\n <mat-icon *ngIf='!isRtlLayout()' [@isExpandedLTR]=\"hasItems() && expanded ? 'yes' : 'no'\">\n keyboard_arrow_down\n </mat-icon>\n <mat-icon *ngIf='isRtlLayout()' [@isExpandedRTL]=\"hasItems() && expanded ? 'yes' : 'no'\">\n keyboard_arrow_down\n </mat-icon>\n </ng-container>\n</mat-list-item>\n\n<mat-divider *ngIf=\"hasDivider\"></mat-divider>\n\n<div *ngIf=\"hasItems() && expanded\" [@slideInOut] [dir]=\"isRtlLayout() ? 'rtl' : 'ltr'\" [ngClass]=\"classes\">\n <ng-list-item *ngFor=\"let singleNode of nodeChildren\" [hasDivider]=\"hasDivider\" [nodeConfiguration]='nodeConfiguration' [node]='singleNode' [level]=\"level + 1\" [selectedNode]='selectedNode' (selectedItem)=\"selectedListItem($event)\">\n </ng-list-item>\n</div>",
styles: [".amml-item{line-height:48px;position:relative;cursor:pointer}.anml-data{width:100%;text-transform:capitalize;display:flex;justify-content:start;height:48px}.amml-icon-fa{font-size:20px}.amml-icon,.label{line-height:48px}div[dir=ltr] .amml-icon{margin-right:16px}div[dir=ltr].amml-submenu,div[dir=rtl] .amml-icon{margin-left:16px}div[dir=rtl].amml-submenu{margin-right:16px}"],
animations: [
animations.trigger('slideInOut', [
animations.state('in', animations.style({ height: '*', opacity: 0 })),
animations.transition(':leave', [
animations.style({ height: '*', opacity: 0.2 }),
animations.group([
animations.animate(200, animations.style({ height: 0 })),
animations.animate('200ms ease-out', animations.style({ opacity: 0 }))
])
]),
animations.transition(':enter', [
animations.style({ height: '0', opacity: 0 }),
animations.group([
animations.animate(200, animations.style({ height: '*' })),
animations.animate('400ms ease-out', animations.style({ opacity: 1 }))
])
])
]),
animations.trigger('isExpandedLTR', [
animations.state('no', animations.style({ transform: 'rotate(-90deg)' })),
animations.state('yes', animations.style({ transform: 'rotate(0deg)' })),
animations.transition('no => yes', animations.animate(200)),
animations.transition('yes => no', animations.animate(200))
]),
animations.trigger('isExpandedRTL', [
animations.state('no', animations.style({ transform: 'rotate(90deg)' })),
animations.state('yes', animations.style({ transform: 'rotate(0deg)' })),
animations.transition('no => yes', animations.animate(200)),
animations.transition('yes => no', animations.animate(200))
])
]
},] },
];
ListItemComponent.ctorParameters = function () {
return [
{ type: router.Router },
{ type: MultilevelMenuService }
];
};
ListItemComponent.propDecorators = {
node: [{ type: i0.Input }],
level: [{ type: i0.Input }],
selectedNode: [{ type: i0.Input }],
nodeConfiguration: [{ type: i0.Input }],
selectedItem: [{ type: i0.Output }],
hasDivider: [{ type: i0.Input }]
};
return ListItemComponent;
}());
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,missingReturn,uselessCode} checked by tsc
*/
var NgMaterialMultilevelMenuModule = /** @class */ (function () {
function NgMaterialMultilevelMenuModule() {
}
NgMaterialMultilevelMenuModule.decorators = [
{ type: i0.NgModule, args: [{
imports: [
common.CommonModule,
MaterialsModule
],
declarations: [NgMaterialMultilevelMenuComponent, ListItemComponent],
exports: [NgMaterialMultilevelMenuComponent]
},] },
];
return NgMaterialMultilevelMenuModule;
}());
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,missingReturn,uselessCode} checked by tsc
*/
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,missingReturn,uselessCode} checked by tsc
*/
exports.NgMaterialMultilevelMenuModule = NgMaterialMultilevelMenuModule;
exports.ɵd = ListItemComponent;
exports.ɵa = MaterialsModule;
exports.ɵc = MultilevelMenuService;
exports.ɵb = NgMaterialMultilevelMenuComponent;
Object.defineProperty(exports, '__esModule', { value: true });
})));
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"ng-material-multilevel-menu-2.umd.js.map","sources":["ng://ng-material-multilevel-menu-2/lib/materials.module.ts","ng://ng-material-multilevel-menu-2/lib/multilevel-menu.service.ts","ng://ng-material-multilevel-menu-2/lib/constants.ts","ng://ng-material-multilevel-menu-2/lib/ng-material-multilevel-menu.component.ts","ng://ng-material-multilevel-menu-2/lib/list-item/list-item.component.ts","ng://ng-material-multilevel-menu-2/lib/ng-material-multilevel-menu.module.ts"],"sourcesContent":["import { NgModule } from '@angular/core';\r\n\r\nimport {\r\n  MatIconModule,\r\n  MatListModule,\r\n  MatRippleModule,\r\n} from '@angular/material';\r\n\r\n@NgModule({\r\n  imports: [\r\n    MatIconModule,\r\n    MatListModule,\r\n    MatRippleModule,\r\n  ],\r\n  declarations: [],\r\n  exports: [\r\n    MatIconModule,\r\n    MatListModule,\r\n    MatRippleModule,\r\n  ]\r\n})\r\nexport class MaterialsModule { }\r\n","import { Injectable } from '@angular/core';\r\nimport { MultilevelNodes } from './app.model';\r\n\r\n@Injectable({\r\n  providedIn: 'root'\r\n})\r\nexport class MultilevelMenuService {\r\n  foundLinkObject: MultilevelNodes;\r\n  generateId(): string {\r\n    let text = '';\r\n    const possible = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';\r\n    for (let i = 0; i < 20; i++) {\r\n      text += possible.charAt(Math.floor(Math.random() * possible.length));\r\n    }\r\n    return text;\r\n  }\r\n  addRandomId(nodes: MultilevelNodes[]): void {\r\n    nodes.forEach((node: MultilevelNodes) => {\r\n      node.id = this.generateId();\r\n      if (node.items !== undefined) {\r\n        this.addRandomId(node.items);\r\n      }\r\n    });\r\n  }\r\n  recursiveCheckId(node: MultilevelNodes, nodeId: string): boolean {\r\n    if (node.id === nodeId) {\r\n      return true;\r\n    } else {\r\n      if (node.items !== undefined) {\r\n        return node.items.some((nestedNode: MultilevelNodes) => {\r\n          return this.recursiveCheckId(nestedNode, nodeId);\r\n        });\r\n      }\r\n    }\r\n  }\r\n  recursiveCheckLink(nodes: MultilevelNodes[], link: string): void {\r\n    for (let nodeIndex = 0; nodeIndex < nodes.length; nodeIndex++) {\r\n      const node = nodes[nodeIndex];\r\n      for (const key in node) {\r\n        if (node.hasOwnProperty(key)) {\r\n          if (encodeURI(node.link) === link) {\r\n            this.foundLinkObject = node;\r\n          } else {\r\n            if (node.items !== undefined) {\r\n              this.recursiveCheckLink(node.items, link);\r\n            }\r\n          }\r\n        }\r\n      }\r\n    }\r\n  }\r\n  getMatchedObjectByUrl(node: MultilevelNodes[], link: string): MultilevelNodes {\r\n    this.recursiveCheckLink(node, link);\r\n    return this.foundLinkObject;\r\n  }\r\n}\r\n","export const CONSTANT = {\r\n    PADDING_AT_START: true,\r\n    DEFAULT_CLASS_NAME: `amml-container`,\r\n    DEFAULT_LIST_CLASS_NAME: `amml-item`,\r\n    SELECTED_LIST_CLASS_NAME: `selected-amml-item`,\r\n    DEFAULT_SELECTED_FONT_COLOR: `#1976d2`,\r\n    DEFAULT_LIST_BACKGROUND_COLOR: `transparent`,\r\n    DEFAULT_LIST_FONT_COLOR: `rgba(0,0,0,.87)`,\r\n    ERROR_MESSAGE: `Invalid data for material Multilevel List Component`\r\n};\r\n","import {\r\n  Component,\r\n  OnInit,\r\n  OnChanges,\r\n  Input,\r\n  Output,\r\n  EventEmitter\r\n} from '@angular/core';\r\nimport { Router, NavigationEnd } from '@angular/router';\r\n\r\nimport { MultilevelMenuService } from './multilevel-menu.service';\r\n\r\nimport { Configuration, MultilevelNodes, BackgroundStyle } from './app.model';\r\nimport { CONSTANT } from './constants';\r\n\r\n@Component({\r\n  selector: 'ng-material-multilevel-menu',\r\n  template: `<div [ngClass]=\"getClassName()\" [ngStyle]=\"getGlobalStyle()\" *ngIf='items.length !== 0' [dir]=\"isRtlLayout() ? 'rtl' : 'ltr'\">\r\n    <mat-list>\r\n        <ng-list-item *ngFor=\"let node of items\" [nodeConfiguration]='nodeConfig' [node]='node' [selectedNode]='currentNode' [hasDivider]=\"hasDivider\" (selectedItem)=\"selectedListItem($event)\r\n    \">\r\n        </ng-list-item>\r\n    </mat-list>\r\n</div>`,\r\n  styles: [`.amml-item{line-height:48px;display:flex;justify-content:space-between;position:relative}.anml-data{width:100%;text-transform:capitalize;display:flex;justify-content:start}.amml-icon-fa{font-size:20px}.amml-icon{line-height:48px}.active{color:#1976d2}div[dir=ltr] .amml-icon{margin-right:15px}div[dir=ltr] .amml-submenu{margin-left:16px}div[dir=rtl] .amml-icon{margin-left:15px}div[dir=rtl] .amml-submenu{margin-right:16px}`]\r\n})\r\nexport class NgMaterialMultilevelMenuComponent implements OnInit, OnChanges {\r\n  @Input() items: MultilevelNodes[];\r\n  @Input() configuration: Configuration = null;\r\n  @Output() selectedItem = new EventEmitter<MultilevelNodes>();\r\n  @Output() selectedLabel = new EventEmitter<MultilevelNodes>();\r\n  @Input() hasDivider = true;\r\n  currentNode: MultilevelNodes;\r\n  nodeConfig: Configuration = {\r\n    paddingAtStart: true,\r\n    listBackgroundColor: null,\r\n    fontColor: null,\r\n    selectedListFontColor: null,\r\n    interfaceWithRoute: null,\r\n    collapseOnSelect: null,\r\n    highlightOnSelect: false,\r\n    rtlLayout: false\r\n  };\r\n  isInvalidConfig = true;\r\n  constructor(\r\n    private router: Router,\r\n    private multilevelMenuService: MultilevelMenuService\r\n  ) {}\r\n  ngOnChanges() {\r\n    this.checkValiddata();\r\n    this.detectInvalidConfig();\r\n  }\r\n  ngOnInit() {\r\n    if (\r\n      this.configuration !== null &&\r\n      this.configuration !== undefined &&\r\n      this.configuration !== '' &&\r\n      this.configuration.interfaceWithRoute !== null &&\r\n      this.configuration.interfaceWithRoute\r\n    ) {\r\n      this.router.events.subscribe(event => {\r\n        if (event instanceof NavigationEnd) {\r\n          this.updateNodeByURL(event.url);\r\n        }\r\n      });\r\n      this.updateNodeByURL(this.router.url);\r\n    }\r\n  }\r\n  updateNodeByURL(url: string): void {\r\n    const foundNode = this.multilevelMenuService.getMatchedObjectByUrl(\r\n      this.items,\r\n      url\r\n    );\r\n    if (\r\n      foundNode !== undefined &&\r\n      foundNode.link !== undefined &&\r\n      foundNode.link !== null &&\r\n      foundNode.link !== ''\r\n    ) {\r\n      this.currentNode = foundNode;\r\n      this.selectedListItem(foundNode);\r\n    }\r\n  }\r\n  checkValiddata(): void {\r\n    if (this.items.length === 0) {\r\n      console.warn(CONSTANT.ERROR_MESSAGE);\r\n    } else {\r\n      this.items = this.items.filter(n => !n.hidden);\r\n      this.multilevelMenuService.addRandomId(this.items);\r\n    }\r\n  }\r\n  detectInvalidConfig(): void {\r\n    if (\r\n      this.configuration === null ||\r\n      this.configuration === undefined ||\r\n      this.configuration === ''\r\n    ) {\r\n      this.isInvalidConfig = true;\r\n    } else {\r\n      this.isInvalidConfig = false;\r\n      const config = this.configuration;\r\n      if (\r\n        config.paddingAtStart !== undefined &&\r\n        config.paddingAtStart !== null &&\r\n        typeof config.paddingAtStart === 'boolean'\r\n      ) {\r\n        this.nodeConfig.paddingAtStart = config.paddingAtStart;\r\n      }\r\n      if (\r\n        config.listBackgroundColor !== '' &&\r\n        config.listBackgroundColor !== null &&\r\n        config.listBackgroundColor !== undefined\r\n      ) {\r\n        this.nodeConfig.listBackgroundColor = config.listBackgroundColor;\r\n      }\r\n      if (\r\n        config.fontColor !== '' &&\r\n        config.fontColor !== null &&\r\n        config.fontColor !== undefined\r\n      ) {\r\n        this.nodeConfig.fontColor = config.fontColor;\r\n      }\r\n      if (\r\n        config.selectedListFontColor !== '' &&\r\n        config.selectedListFontColor !== null &&\r\n        config.selectedListFontColor !== undefined\r\n      ) {\r\n        this.nodeConfig.selectedListFontColor = config.selectedListFontColor;\r\n      }\r\n      if (\r\n        config.interfaceWithRoute !== null &&\r\n        config.interfaceWithRoute !== undefined &&\r\n        typeof config.interfaceWithRoute === 'boolean'\r\n      ) {\r\n        this.nodeConfig.interfaceWithRoute = config.interfaceWithRoute;\r\n      }\r\n      if (\r\n        config.collapseOnSelect !== null &&\r\n        config.collapseOnSelect !== undefined &&\r\n        typeof config.collapseOnSelect === 'boolean'\r\n      ) {\r\n        this.nodeConfig.collapseOnSelect = config.collapseOnSelect;\r\n      }\r\n      if (\r\n        config.highlightOnSelect !== null &&\r\n        config.highlightOnSelect !== undefined &&\r\n        typeof config.highlightOnSelect === 'boolean'\r\n      ) {\r\n        this.nodeConfig.highlightOnSelect = config.highlightOnSelect;\r\n      }\r\n      if (\r\n        config.rtlLayout !== null &&\r\n        config.rtlLayout !== undefined &&\r\n        typeof config.rtlLayout === 'boolean'\r\n      ) {\r\n        this.nodeConfig.rtlLayout = config.rtlLayout;\r\n      }\r\n    }\r\n  }\r\n  getClassName(): string {\r\n    if (this.isInvalidConfig) {\r\n      return CONSTANT.DEFAULT_CLASS_NAME;\r\n    } else {\r\n      if (\r\n        this.configuration.classname !== '' &&\r\n        this.configuration.classname !== null &&\r\n        this.configuration.classname !== undefined\r\n      ) {\r\n        return `${CONSTANT.DEFAULT_CLASS_NAME} ${this.configuration.classname}`;\r\n      } else {\r\n        return CONSTANT.DEFAULT_CLASS_NAME;\r\n      }\r\n    }\r\n  }\r\n  getGlobalStyle(): BackgroundStyle {\r\n    if (!this.isInvalidConfig) {\r\n      const styles = {\r\n        background: null\r\n      };\r\n      if (\r\n        this.configuration.backgroundColor !== '' &&\r\n        this.configuration.backgroundColor !== null &&\r\n        this.configuration.backgroundColor !== undefined\r\n      ) {\r\n        styles.background = this.configuration.backgroundColor;\r\n      }\r\n      return styles;\r\n    }\r\n  }\r\n  isRtlLayout(): boolean {\r\n    return this.nodeConfig.rtlLayout;\r\n  }\r\n  selectedListItem(event: MultilevelNodes): void {\r\n    this.currentNode = event;\r\n    if (\r\n      event.items === undefined &&\r\n      (!event.onSelected || typeof event.onSelected !== 'function')\r\n    ) {\r\n      this.selectedItem.emit(event);\r\n    } else {\r\n      this.selectedLabel.emit(event);\r\n    }\r\n  }\r\n}\r\n","import {\r\n  Component,\r\n  OnChanges,\r\n  Input,\r\n  Output,\r\n  EventEmitter\r\n} from '@angular/core';\r\nimport { Router } from '@angular/router';\r\nimport {\r\n  trigger,\r\n  style,\r\n  transition,\r\n  animate,\r\n  state,\r\n  group\r\n} from '@angular/animations';\r\n\r\nimport { MultilevelMenuService } from './../multilevel-menu.service';\r\n\r\nimport { Configuration, MultilevelNodes, ListStyle } from './../app.model';\r\nimport { CONSTANT } from './../constants';\r\n\r\n@Component({\r\n  selector: 'ng-list-item',\r\n  template: `<mat-list-item matRipple [ngClass]=\"selectedListClasses\" *ngIf=\"!node.hidden\" (click)=\"expand(node)\" title=\"{{node.label}}\" [ngStyle]=\"getListStyle()\">\r\n    <div class=\"anml-data\" [dir]=\"isRtlLayout() ? 'rtl' : 'ltr'\">\r\n        <div class=\"icon-container\" [ngSwitch]=\"getListIcon(node)\">\r\n            <span *ngSwitchCase=\"'faicon'\" class=\"amml-icon amml-icon-fa\">\r\n        <i [ngClass]=\"node.faIcon\"></i>\r\n      </span>\r\n            <mat-icon *ngSwitchCase=\"'icon'\" class=\"amml-icon\">\r\n                {{node.icon}}\r\n            </mat-icon>\r\n            <img matListAvatar *ngSwitchCase=\"'imageicon'\" class=\"amml-icon\" src=\"{{node.imageIcon}}\" alt=\"{{node.label}}\" />\r\n        </div>\r\n        <span class=\"label\">{{node.label}}</span>\r\n    </div>\r\n    <ng-container *ngIf='hasItems()'>\r\n        <mat-icon *ngIf='!isRtlLayout()' [@isExpandedLTR]=\"hasItems() && expanded ? 'yes' : 'no'\">\r\n            keyboard_arrow_down\r\n        </mat-icon>\r\n        <mat-icon *ngIf='isRtlLayout()' [@isExpandedRTL]=\"hasItems() && expanded ? 'yes' : 'no'\">\r\n            keyboard_arrow_down\r\n        </mat-icon>\r\n    </ng-container>\r\n</mat-list-item>\r\n\r\n<mat-divider *ngIf=\"hasDivider\"></mat-divider>\r\n\r\n<div *ngIf=\"hasItems() && expanded\" [@slideInOut] [dir]=\"isRtlLayout() ? 'rtl' : 'ltr'\" [ngClass]=\"classes\">\r\n    <ng-list-item *ngFor=\"let singleNode of nodeChildren\" [hasDivider]=\"hasDivider\" [nodeConfiguration]='nodeConfiguration' [node]='singleNode' [level]=\"level + 1\" [selectedNode]='selectedNode' (selectedItem)=\"selectedListItem($event)\">\r\n    </ng-list-item>\r\n</div>`,\r\n  styles: [`.amml-item{line-height:48px;position:relative;cursor:pointer}.anml-data{width:100%;text-transform:capitalize;display:flex;justify-content:start;height:48px}.amml-icon-fa{font-size:20px}.amml-icon,.label{line-height:48px}div[dir=ltr] .amml-icon{margin-right:16px}div[dir=ltr].amml-submenu,div[dir=rtl] .amml-icon{margin-left:16px}div[dir=rtl].amml-submenu{margin-right:16px}`],\r\n  animations: [\r\n    trigger('slideInOut', [\r\n      state('in', style({ height: '*', opacity: 0 })),\r\n      transition(':leave', [\r\n        style({ height: '*', opacity: 0.2 }),\r\n        group([\r\n          animate(200, style({ height: 0 })),\r\n          animate('200ms ease-out', style({ opacity: 0 }))\r\n        ])\r\n      ]),\r\n      transition(':enter', [\r\n        style({ height: '0', opacity: 0 }),\r\n        group([\r\n          animate(200, style({ height: '*' })),\r\n          animate('400ms ease-out', style({ opacity: 1 }))\r\n        ])\r\n      ])\r\n    ]),\r\n    trigger('isExpandedLTR', [\r\n      state('no', style({ transform: 'rotate(-90deg)' })),\r\n      state('yes', style({ transform: 'rotate(0deg)' })),\r\n\r\n      transition('no => yes', animate(200)),\r\n      transition('yes => no', animate(200))\r\n    ]),\r\n    trigger('isExpandedRTL', [\r\n      state('no', style({ transform: 'rotate(90deg)' })),\r\n      state('yes', style({ transform: 'rotate(0deg)' })),\r\n\r\n      transition('no => yes', animate(200)),\r\n      transition('yes => no', animate(200))\r\n    ])\r\n  ]\r\n})\r\nexport class ListItemComponent implements OnChanges {\r\n  @Input() node: MultilevelNodes;\r\n  @Input() level = 1;\r\n  @Input() selectedNode: MultilevelNodes;\r\n  @Input() node