UNPKG

@stratio/egeo

Version:
298 lines 31.1 kB
/** * @fileoverview added by tsickle * Generated from: lib/st-menu/st-menu-options/st-menu-options.component.ts * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc */ /* * © 2017 Stratio Big Data Inc., Sucursal en España. * * This software is licensed under the Apache License, Version 2.0. * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. * See the terms of the License for more details. * * SPDX-License-Identifier: Apache-2.0. */ import { ChangeDetectionStrategy, ChangeDetectorRef, Component, ElementRef, EventEmitter, Input, Output, ViewChild } from '@angular/core'; import { FormControl } from '@angular/forms'; import { Subject } from 'rxjs'; import { debounceTime } from 'rxjs/operators'; import { StMenuPosition } from './../st-menu.model'; /** * @template T */ export class StMenuOptionsComponent { // px per millisecond /** * @param {?} _cd */ constructor(_cd) { this._cd = _cd; this.position = StMenuPosition.right; this.level = 0; this.animate = true; this.searchBox = new EventEmitter(); this.selectedOption = new EventEmitter(); this.initialized = false; this.scrollTop = 0; this._scrollSubject = new Subject(); this._SCROLL_SPEED = 120; // px per second // px per second this._SCROLL_SPEED_MILLIS = this._SCROLL_SPEED / 1000; // px per millisecond } /** * @return {?} */ ngAfterViewInit() { if (this.dynamicHeight) { this._checkScrollButtons(); } this.initialized = true; this._cd.detectChanges(); this._scrollSubscription = this._scrollSubject .pipe(debounceTime(50)) .subscribe((/** * @return {?} */ () => this._checkScrollButtons())); } /** * @return {?} */ ngOnDestroy() { cancelAnimationFrame(this._scrollAnimationHandler); if (this._scrollSubscription) { this._scrollSubscription.unsubscribe(); } } /** * Set the submenu position when an option is hovered * * @param {?} event MouseEnter event when an option is hovered * @param {?} item Hovered item of the list * @return {?} */ showMenu(event, item) { if (this.openedItem !== item && (!this.activeOption || this.activeOption.value !== item)) { this.openedItem = item; if (item.subMenus) { /** @type {?} */ const target = (/** @type {?} */ (event.target)); this.menuPosition = target.offsetTop; if (this.dynamicHeight) { this.maxHeightChild = window.innerHeight - target.getBoundingClientRect().top - 30; } } } } /** * Saves current scroll position and check the scroll buttons state * @return {?} */ onScroll() { this.scrollTop = this.childListElement.nativeElement.scrollTop; if (this.dynamicHeight) { this._scrollSubject.next(); } } /** * Generates a ngFor trackBy id from StMenuModel name and icon values * @param {?} index * @param {?} item * @return {?} */ trackByFn(index, item) { return item.name + (item.icon || ''); } /** * Starts a scroll animation frame * @param {?} reverse * @return {?} */ scrollTo(reverse) { /** @type {?} */ const startScrollAnimationTime = new Date().getTime(); // initial animation time in milliseconds /** @type {?} */ const startScrollAnimationPosition = this.childListElement.nativeElement.scrollTop; this.openedItem = null; this._scrollAnimationHandler = requestAnimationFrame((/** * @return {?} */ () => this._scrollAnimation(reverse, startScrollAnimationTime, startScrollAnimationPosition))); } /** * Stop current scroll animation frame * @return {?} */ stopScroll() { cancelAnimationFrame(this._scrollAnimationHandler); } /** * Clear the search input value * @return {?} */ removeSearch() { this.searchFormControl.setValue(''); } /** * Check the selected option and emits if it's valid * @param {?} event * @param {?} item * @return {?} */ onSelectOption(event, item) { if ((!item.subMenus || !item.subMenus.length) && (!this.activeOption || this.activeOption !== item.value)) { this.selectedOption.emit(item); } else { // Prevents close menu event.stopPropagation(); } } /** * Generates menu item id * @param {?} item * @return {?} */ getItemId(item) { return (item.icon || '') + item.name.replace(/\W/g, '_') + '-option'; } /** * Recursive scroll animation frame method * @private * @param {?} reverse sense of the scroll animation * @param {?} startScrollAnimationTime * @param {?} startScrollAnimationPosition * @return {?} */ _scrollAnimation(reverse, startScrollAnimationTime, startScrollAnimationPosition) { /** @type {?} */ const current = new Date().getTime(); /** @type {?} */ const runtime = current - startScrollAnimationTime; // initialPosition in px + (reverse: negative or positive sense * millis from animation start * px per millis) /** @type {?} */ const position = startScrollAnimationPosition + ((reverse ? 1 : -1) * runtime * this._SCROLL_SPEED_MILLIS); this.childListElement.nativeElement.scrollTo(0, position); this._scrollAnimationHandler = requestAnimationFrame((/** * @return {?} */ () => this._scrollAnimation(reverse, startScrollAnimationTime, startScrollAnimationPosition))); } /** * Enable or disable the top and bottom scroll buttons depending of the scrollList position * @private * @return {?} */ _checkScrollButtons() { /** @type {?} */ const scrollList = this.childListElement.nativeElement; this.scrollTopEnabled = scrollList.scrollTop > 0; this.scrollBottomEnabled = scrollList.getBoundingClientRect().height + scrollList.scrollTop < scrollList.scrollHeight - 1; this._cd.markForCheck(); } } StMenuOptionsComponent.decorators = [ { type: Component, args: [{ selector: 'st-menu-options', template: "<div class=\"st-menu-options\"\n [style.marginTop.px]=\"marginTop\"\n [ngClass]=\"{\n 'st-menu-options--left': position === 'left',\n 'st-menu-options--initilized': initialized\n }\">\n\n <div class=\"scroll-button scroll-button--top\"\n *ngIf=\"dynamicHeight\"\n [style.opacity]=\"scrollTopEnabled ? 1 : 0\"\n (mouseover)=\"scrollTo(false)\"\n (mouseout)=\"stopScroll()\">\n <span class=\"icon-arrow2_up\"></span>\n </div>\n\n <div class=\"st-menu-options__container\"\n [ngClass]=\"{\n 'st-menu-options__container--animation': animate && level === 0\n }\">\n <ul class=\"st-menu-options__list st-custom-scrollbar\"\n #childList\n [style.maxHeight.px]=\"maxHeight\"\n (scroll)=\"onScroll()\">\n\n <li class=\"st-menu-options__item st-menu-options__search\" *ngIf=\"searchFormControl\">\n <input class=\"st-menu-options__input\"\n [attr.id]=\"qaTag + '-search'\"\n type=\"text\"\n autocomplete=\"off\"\n [formControl]=\"searchFormControl\"\n [placeholder]=\"searchLabel\" />\n <span class=\"st-menu-options__delete icon-cross\"\n (click)=\"removeSearch()\"\n *ngIf=\"searchFormControl.value.length\"></span>\n </li>\n\n <li class=\"st-menu-options__item\"\n *ngFor=\"let item of menuOptions; let i = index; trackBy: trackByFn\"\n [ngClass]=\"{\n 'st-menu-options__item--active' : openedItem === item,\n 'disabled' : (activeOption && activeOption === item.value) || item.disabled,\n 'st-menu-options__item--success' : item.status === 'success',\n 'st-menu-options__item--warning' : item.status === 'warning',\n 'st-menu-options__item--critical' : item.status === 'critical',\n 'separator': item.separator\n }\"\n [attr.id]=\"getItemId(item)\"\n (click)=\"!item.disabled && onSelectOption($event, item)\"\n (mouseenter)=\"showMenu($event, item)\">\n\n <div class=\"item-content\">\n <i *ngIf=\"item.icon\" [ngClass]=\"item.icon\" class=\"item-icon\"></i>\n <span class=\"label\">{{item.name}}</span>\n <i class=\"icon icon-arrow2_right has-submenu\" *ngIf=\"item.subMenus && item.subMenus.length\"></i>\n </div>\n\n <st-menu-options\n *ngIf=\"item.subMenus && item.subMenus.length && openedItem === item\"\n [style.top.px]=\"menuPosition - scrollTop\"\n [menuOptions]=\"item.subMenus\"\n [position]=\"position\"\n [maxHeight]=\"maxHeightChild\"\n [activeOption]=\"activeOption\"\n [dynamicHeight]=\"dynamicHeight\"\n [level]=\"level + 1\"\n (selectedOption)=\"selectedOption.emit($event)\"\n class=\"st-menu-options__submenu\">\n </st-menu-options>\n </li>\n </ul>\n </div>\n\n <div class=\"scroll-button scroll-button--bottom\"\n *ngIf=\"dynamicHeight\"\n [style.opacity]=\"scrollBottomEnabled ? 1 : 0\"\n (mouseover)=\"scrollTo(true)\"\n (mouseout)=\"stopScroll()\">\n <span class=\"icon-arrow2_down\"></span>\n </div>\n</div>\n", changeDetection: ChangeDetectionStrategy.OnPush, styles: ["@charset \"UTF-8\";.st-menu-options{background-color:#fff;font-size:.938rem;left:100%;position:absolute;margin-bottom:2.5rem;visibility:hidden;top:100%;-webkit-transform:translate3d(0,0,0)}.st-menu-options--left{right:100%;left:auto}.st-menu-options--initilized{visibility:visible}.st-menu-options__container{overflow:hidden;width:100%;box-shadow:0 2px 4px 0 rgba(0,0,0,.5);border-radius:5px}.st-menu-options__container--animation{-webkit-animation:.25s ease-out fadeInFromNone;animation:.25s ease-out fadeInFromNone}.st-menu-options__search{border-bottom:1px solid #cfcfcf;margin:0;padding:0;position:relative}.st-menu-options__input{outline:0;width:100%;min-width:12.5rem}.st-menu-options__delete{position:absolute;right:.625rem;cursor:pointer;top:.6875rem}.st-menu-options__submenu{position:absolute;width:100%;left:0;top:0;margin-left:0}.st-menu-options__submenu.active{visibility:visible}.st-menu-options__list{list-style:none;top:0;left:0;width:100%;background:#fff;min-width:100%;overflow-y:auto;overflow-x:hidden}.st-menu-options__item{color:#585858;cursor:pointer;font-size:.875rem;justify-content:space-between;padding:.75rem .5rem .6875rem 1.25rem;-webkit-tap-highlight-color:transparent}.st-menu-options__item>div{display:flex;flex-direction:row}.st-menu-options__item>div .label{padding-right:.75rem;white-space:nowrap;font-size:.9375rem;line-height:1.25rem}.st-menu-options__item .has-submenu{font-size:.6875rem;margin-top:5px;margin-left:auto}.st-menu-options__item .item-icon{font-size:.8125rem;margin-right:.375rem;margin-top:.1875rem;color:#a0a0a0}.st-menu-options__item>span{white-space:nowrap}.st-menu-options__item--active{background-color:#e6f1fc;visibility:visible}.st-menu-options__item--active>.item-content .icon.has-submenu,.st-menu-options__item--active>.item-content .item-icon{color:var(--egeo-theme-brand,#37b5e4)}.st-menu-options__item--success,.st-menu-options__item--success>.item-content .item-icon{color:#0776df}.st-menu-options__item--warning,.st-menu-options__item--warning>.item-content .item-icon{color:#fa932f}.st-menu-options__item--critical,.st-menu-options__item--critical>.item-content .item-icon{color:#df2935}.st-menu-options__item.disabled{color:#b8b8b8!important}.st-menu-options__item.separator{border-top:1px solid #cdd6df}.scroll-button{background:#fff;color:#6c7b8b;border:1px solid #cfcfcf;font-size:.625rem;margin-left:-1px;padding:1px 0 0;position:absolute;text-align:center;width:calc(100% + 2px);z-index:1}.scroll-button--top{top:-.875rem}.scroll-button--bottom{bottom:-.9375rem}@-webkit-keyframes fadeInFromNone{0%{max-height:0}100%{display:block;max-height:18.75rem}}@keyframes fadeInFromNone{0%{max-height:0}100%{display:block;max-height:18.75rem}}"] }] } ]; /** @nocollapse */ StMenuOptionsComponent.ctorParameters = () => [ { type: ChangeDetectorRef } ]; StMenuOptionsComponent.propDecorators = { maxHeight: [{ type: Input }], dynamicHeight: [{ type: Input }], activeOption: [{ type: Input }], menuOptions: [{ type: Input }], position: [{ type: Input }], searchFormControl: [{ type: Input }], searchLabel: [{ type: Input }], level: [{ type: Input }], marginTop: [{ type: Input }], animate: [{ type: Input }], qaTag: [{ type: Input }], searchBox: [{ type: Output }], selectedOption: [{ type: Output }], childListElement: [{ type: ViewChild, args: ['childList', { static: false },] }] }; if (false) { /** @type {?} */ StMenuOptionsComponent.prototype.maxHeight; /** @type {?} */ StMenuOptionsComponent.prototype.dynamicHeight; /** @type {?} */ StMenuOptionsComponent.prototype.activeOption; /** @type {?} */ StMenuOptionsComponent.prototype.menuOptions; /** @type {?} */ StMenuOptionsComponent.prototype.position; /** @type {?} */ StMenuOptionsComponent.prototype.searchFormControl; /** @type {?} */ StMenuOptionsComponent.prototype.searchLabel; /** @type {?} */ StMenuOptionsComponent.prototype.level; /** @type {?} */ StMenuOptionsComponent.prototype.marginTop; /** @type {?} */ StMenuOptionsComponent.prototype.animate; /** @type {?} */ StMenuOptionsComponent.prototype.qaTag; /** @type {?} */ StMenuOptionsComponent.prototype.searchBox; /** @type {?} */ StMenuOptionsComponent.prototype.selectedOption; /** @type {?} */ StMenuOptionsComponent.prototype.childListElement; /** @type {?} */ StMenuOptionsComponent.prototype.openedItem; /** @type {?} */ StMenuOptionsComponent.prototype.menuPosition; /** @type {?} */ StMenuOptionsComponent.prototype.maxHeightChild; /** @type {?} */ StMenuOptionsComponent.prototype.scrollTopEnabled; /** @type {?} */ StMenuOptionsComponent.prototype.scrollBottomEnabled; /** @type {?} */ StMenuOptionsComponent.prototype.initialized; /** @type {?} */ StMenuOptionsComponent.prototype.scrollTop; /** * @type {?} * @private */ StMenuOptionsComponent.prototype._scrollAnimationHandler; /** * @type {?} * @private */ StMenuOptionsComponent.prototype._scrollSubject; /** * @type {?} * @private */ StMenuOptionsComponent.prototype._scrollSubscription; /** * @type {?} * @private */ StMenuOptionsComponent.prototype._SCROLL_SPEED; /** * @type {?} * @private */ StMenuOptionsComponent.prototype._SCROLL_SPEED_MILLIS; /** * @type {?} * @private */ StMenuOptionsComponent.prototype._cd; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"st-menu-options.component.js","sourceRoot":"ng://@stratio/egeo/","sources":["lib/st-menu/st-menu-options/st-menu-options.component.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * © 2017 Stratio Big Data Inc., Sucursal en España.\n *\n * This software is licensed under the Apache License, Version 2.0.\n * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;\n * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.\n * See the terms of the License for more details.\n *\n * SPDX-License-Identifier: Apache-2.0.\n */\n\nimport {\n   AfterViewInit,\n   ChangeDetectionStrategy,\n   ChangeDetectorRef,\n   Component,\n   ElementRef,\n   EventEmitter,\n   Input,\n   Output,\n   ViewChild,\n   OnDestroy\n} from '@angular/core';\nimport { FormControl } from '@angular/forms';\nimport { Subject, Subscription } from 'rxjs';\nimport { debounceTime } from 'rxjs/operators';\nimport { StMenuModel, StMenuPosition } from './../st-menu.model';\n\n@Component({\n   selector: 'st-menu-options',\n   templateUrl: './st-menu-options.component.html',\n   styleUrls: ['./st-menu-options.component.scss'],\n   changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class StMenuOptionsComponent<T> implements AfterViewInit, OnDestroy {\n\n   @Input() maxHeight: number;\n   @Input() dynamicHeight: boolean;\n   @Input() activeOption: any;\n   @Input() menuOptions: Array<StMenuModel<T>>;\n   @Input() position: StMenuPosition = StMenuPosition.right;\n   @Input() searchFormControl: FormControl;\n   @Input() searchLabel: string;\n   @Input() level: number = 0;\n   @Input() marginTop: number;\n   @Input() animate: boolean = true;\n   @Input() qaTag: string;\n\n   @Output() searchBox: EventEmitter<string> = new EventEmitter<string>();\n   @Output() selectedOption: EventEmitter<StMenuModel<T>> = new EventEmitter<StMenuModel<T>>();\n\n   @ViewChild('childList', {static: false}) childListElement: ElementRef<HTMLElement>;\n\n   public openedItem: StMenuModel<T>;\n   public menuPosition: number;\n   public maxHeightChild: number;\n   public scrollTopEnabled: boolean;\n   public scrollBottomEnabled: boolean;\n   public initialized: boolean = false;\n   public scrollTop: number = 0;\n\n   private _scrollAnimationHandler: number;\n\n   private _scrollSubject: Subject<void> = new Subject();\n   private _scrollSubscription: Subscription;\n\n   private readonly _SCROLL_SPEED: number = 120;                                // px per second\n   private readonly _SCROLL_SPEED_MILLIS: number = this._SCROLL_SPEED / 1000;   // px per millisecond\n\n   constructor(private _cd: ChangeDetectorRef) { }\n\n   ngAfterViewInit(): void {\n      if (this.dynamicHeight) {\n         this._checkScrollButtons();\n      }\n      this.initialized = true;\n      this._cd.detectChanges();\n      this._scrollSubscription = this._scrollSubject\n         .pipe(debounceTime(50))\n         .subscribe(() => this._checkScrollButtons());\n   }\n\n   ngOnDestroy(): void {\n      cancelAnimationFrame(this._scrollAnimationHandler);\n      if (this._scrollSubscription) {\n         this._scrollSubscription.unsubscribe();\n      }\n   }\n\n   /**\n    * Set the submenu position when an option is hovered\n    *\n    * @param event MouseEnter event when an option is hovered\n    * @param item Hovered item of the list\n    */\n   public showMenu(event: MouseEvent, item: StMenuModel<T>): void {\n      if (this.openedItem !== item && (!this.activeOption || this.activeOption.value !== item)) {\n         this.openedItem = item;\n         if (item.subMenus) {\n            const target: HTMLElement = event.target as HTMLElement;\n            this.menuPosition = target.offsetTop;\n            if (this.dynamicHeight) {\n               this.maxHeightChild = window.innerHeight - target.getBoundingClientRect().top - 30;\n            }\n         }\n      }\n   }\n\n   /**\n    * Saves current scroll position and check the scroll buttons state\n    */\n   public onScroll(): void {\n      this.scrollTop = this.childListElement.nativeElement.scrollTop;\n      if (this.dynamicHeight) {\n         this._scrollSubject.next();\n      }\n   }\n\n   /**\n    * Generates a ngFor trackBy id from StMenuModel name and icon values\n    * @param index\n    * @param item\n    */\n   public trackByFn(index: number, item: StMenuModel<T>): string {\n      return item.name + (item.icon || '');\n   }\n\n   /**\n    * Starts a scroll animation frame\n    * @param reverse\n    */\n   public scrollTo(reverse: boolean): void {\n      const startScrollAnimationTime = new Date().getTime(); // initial animation time in milliseconds\n      const startScrollAnimationPosition = this.childListElement.nativeElement.scrollTop; // initial animation position\n      this.openedItem = null;\n      this._scrollAnimationHandler = requestAnimationFrame(() => this._scrollAnimation(reverse, startScrollAnimationTime,  startScrollAnimationPosition));\n   }\n\n   /**\n    * Stop current scroll animation frame\n    */\n   public stopScroll(): void {\n     cancelAnimationFrame(this._scrollAnimationHandler);\n   }\n\n   /**\n    * Clear the search input value\n    */\n   public removeSearch(): void {\n      this.searchFormControl.setValue('');\n   }\n\n   /**\n    * Check the selected option and emits if it's valid\n    * @param event\n    * @param item\n    */\n   public onSelectOption(event: Event, item: StMenuModel<T>): void {\n      if ((!item.subMenus || !item.subMenus.length) && (!this.activeOption || this.activeOption !== item.value)) {\n         this.selectedOption.emit(item);\n      } else {\n         // Prevents close menu\n         event.stopPropagation();\n      }\n   }\n\n   /**\n    * Generates menu item id\n    * @param item\n    */\n   public getItemId(item: StMenuModel<T>): string {\n      return (item.icon || '') + item.name.replace(/\\W/g, '_') + '-option';\n   }\n\n   /**\n    * Recursive scroll animation frame method\n    * @param reverse sense of the scroll animation\n    */\n   private _scrollAnimation(reverse: boolean,  startScrollAnimationTime: number, startScrollAnimationPosition: number): void {\n      const current: number = new Date().getTime();\n      const runtime: number = current - startScrollAnimationTime;\n      // initialPosition in px + (reverse: negative or positive sense * millis from animation start * px per millis)\n      const position: number = startScrollAnimationPosition + ((reverse ? 1 : -1) * runtime * this._SCROLL_SPEED_MILLIS);\n      this.childListElement.nativeElement.scrollTo(0, position);\n      this._scrollAnimationHandler = requestAnimationFrame(() => this._scrollAnimation(reverse, startScrollAnimationTime, startScrollAnimationPosition));\n   }\n\n   /**\n    * Enable or disable the top and bottom scroll buttons depending of the scrollList position\n    */\n   private _checkScrollButtons(): void {\n      const scrollList: HTMLElement = this.childListElement.nativeElement;\n      this.scrollTopEnabled = scrollList.scrollTop > 0;\n      this.scrollBottomEnabled = scrollList.getBoundingClientRect().height + scrollList.scrollTop < scrollList.scrollHeight - 1;\n      this._cd.markForCheck();\n   }\n}\n"]}