dynamic-mat-table
Version:
dynamic-mat-table is an Angular component for presenting large and complex data with a lightning fast performance (at least 10x faster) and excellent level of control over the presentation.
146 lines • 23 kB
JavaScript
import { ChangeDetectionStrategy, Component, HostBinding, Output, ViewChild, Input, EventEmitter, ChangeDetectorRef, QueryList, ContentChildren } from '@angular/core';
import { TableService } from '../../dynamic-mat-table.service';
import { TextFilter } from './compare/text-filter';
import { NumberFilter } from './compare/number-filter';
import { transition, trigger, query, style, stagger, animate } from '@angular/animations';
import { TableIntl } from '../../../international/table-Intl';
import { MatMenuTrigger } from '@angular/material/menu';
import { isNullorUndefined } from '../../../cores/type';
const listAnimation = trigger('listAnimation', [
transition('* <=> *', [
query(':enter', [style({ opacity: 0 }), stagger('10ms', animate('400ms ease-out', style({ opacity: 1 })))], { optional: true }),
])
]);
export class HeaderFilterComponent {
constructor(languagePack, service, cdr) {
this.languagePack = languagePack;
this.service = service;
this.cdr = cdr;
this.filterChanged = new EventEmitter();
this.filterList = [];
}
get filters() {
if (isNullorUndefined(this.filterList) === true || this.filterList.length === 0) {
this.filterList = [];
this.addNewFilter(this.field.type || 'text');
}
return this.filterList;
}
set filters(values) {
this.filterList = values;
}
get hasValue() {
return this.filterList && this.filterList.filter(f => f.hasValue() === true).length > 0;
}
get showTrigger() {
if (this.menu === undefined) {
return false;
}
else {
return this.menu.menuOpen || this.hasValue;
}
}
ngOnDestroy() {
if (this.eventsSubscription) {
this.eventsSubscription.unsubscribe();
}
}
ngOnInit() {
if (isNullorUndefined(this.filters)) {
this.filters = [];
this.addNewFilter(this.field.type);
}
}
addNewFilter(type = 'text') {
switch (type || 'text') {
case 'text': {
this.filterList.push(new TextFilter(this.languagePack));
break;
}
case 'number': {
this.filterList.push(new NumberFilter(this.languagePack));
break;
}
case 'date': {
// this.compare = new DateCompare(service);
break;
}
case 'boolean': {
// this.compare = new BooleanCompare(service);
break;
}
default: this.filterList.push(new TextFilter(this.languagePack));
}
this.filters[this.filters.length - 1].selectedIndex = 0;
return this.filters[this.filters.length - 1];
}
ngAfterViewInit() {
if (this.menu) {
this.eventsSubscription = this.menu.menuOpened.subscribe(() => this.focusToLastInput());
}
}
focusToLastInput() {
setTimeout(() => {
if (this.filterInputList.length > 0) {
this.filterInputList.last.focus();
}
});
}
filterAction_OnClick(index, action) {
if (action === 0 || action === 1) { // and or
this.filters[index].type = action === 0 ? 'and' : 'or';
if (this.filters.length === index + 1) {
this.addNewFilter(this.field.type);
this.focusToLastInput();
}
}
else if (action === 2 && this.filters.length > 1) { // delete
setTimeout(() => {
this.filters.splice(index, 1);
this.cdr.detectChanges();
this.focusToLastInput();
}); // bug for delete filter item(unwanted reaction close menu)
}
}
clearColumn_OnClick() {
this.filterList = [];
this.filterChanged.emit(this.filterList);
}
applyFilter_OnClick() {
this.filterChanged.emit(this.filterList);
}
}
/** @type {!Array<{type: !Function, args: (undefined|!Array<?>)}>} */
HeaderFilterComponent.decorators = [
{ type: Component, args: [{
// tslint:disable-next-line:component-selector
selector: 'header-filter',
template: "<ng-content></ng-content>\r\n\r\n<mat-menu filter-event #filterMenu=\"matMenu\" class=\"menu\">\r\n <ng-template matMenuContent>\r\n\r\n <div filter-event class=\"menu-title\">\r\n {{field?.header}}\r\n </div>\r\n <div [@listAnimation]=\"filters.length\" filter-event *ngFor=\"let filter of filters; let index = index\"\r\n class=\"filter-panel\">\r\n\r\n <mat-form-field>\r\n <mat-select [value]=\"filter.selectedIndex\" [panelClass]=\"'mat-elevation-z10'\"\r\n (selectionChange)=\"filter.selectedIndex = $event.value;\" placeholder='Conditions'\r\n (keyup.enter)=\"applyFilter_OnClick()\">\r\n <mat-option *ngFor=\"let op of filter.getOperations(); let selectedIndex=index\" [value]=\"selectedIndex\">\r\n {{ op.text }}\r\n </mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n\r\n <div *ngFor=\"let ctrl of filter?.parameters\">\r\n <mat-form-field class=\"input-field\">\r\n <mat-label>{{ctrl.text}}</mat-label>\r\n <input matInput #filterInput=\"matInput\" [(ngModel)]=\"ctrl.value\" [placeholder]=\"\"\r\n (keyup.enter)=\"applyFilter_OnClick()\" autocomplete=\"off\" />\r\n </mat-form-field>\r\n </div>\r\n\r\n <div class=\"or-and\">\r\n <span *ngIf=\"filters?.length !== index+1\" class=\"selected-filter-type\">{{ filter?.type === 'and' ?\r\n languagePack.filterLabels.And : languagePack.filterLabels.Or}}</span>\r\n <span class=\"svg\">\r\n <mat-icon (click)=\"filterAction_OnClick(index,0)\">add</mat-icon>\r\n </span>\r\n <span class=\"svg\">\r\n <mat-icon (click)=\"filterAction_OnClick(index,1)\" style=\"transform: rotate(90deg);\">drag_handle</mat-icon>\r\n </span>\r\n <span class=\"svg\">\r\n <mat-icon (click)=\"filterAction_OnClick(index,2)\">clear</mat-icon>\r\n </span>\r\n </div>\r\n\r\n </div>\r\n\r\n <div filter-event class=\"menu-action\">\r\n <button mat-raised-button type=\"button\" (click)=\"clearColumn_OnClick()\">{{ languagePack.filterLabels.Clear\r\n }}</button>\r\n <button mat-raised-button type=\"button\" color=\"primary\" (click)=\"applyFilter_OnClick()\">{{\r\n languagePack.filterLabels.Search}}</button>\r\n </div>\r\n </ng-template>\r\n</mat-menu>\r\n\r\n<span class=\"trigger\" [matMenuTriggerFor]=\"filterMenu\" *ngIf=\"field.filter !== 'none'\">\r\n <mat-icon>filter_list</mat-icon>\r\n</span>\r\n",
changeDetection: ChangeDetectionStrategy.OnPush,
animations: [listAnimation],
styles: ["@media print{.print-preview{background-color:#fff;position:fixed;width:100%;height:auto;z-index:99999999;margin:0;padding:0;top:0;left:0;overflow:visible;display:block}}.disable-backdrop-click .cdk-overlay-backdrop.cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing{pointer-events:none}:host{display:flex;align-items:center;width:100%;align-self:stretch}.trigger{color:#0000004d;display:flex;opacity:0;transform:translateY(-5px);cursor:pointer;transition-duration:.4s;transition-property:opacity,transform;position:sticky;right:0px;z-index:1;padding-left:0 8px}:host.has-value .trigger{opacity:1;color:#0000008a}:host:hover .trigger,:host.show-trigger .trigger{opacity:1;transform:translateY(-1px)}::ng-deep .mat-menu-content:not(:empty){padding:0!important}.mat-menu-item-highlighted:not([disabled]),.mat-menu-item.cdk-keyboard-focused:not([disabled]),.mat-menu-item.cdk-program-focused:not([disabled]),.mat-menu-item:hover:not([disabled]){background-color:inherit}.input-field{margin-top:-15px}.menu-title{font-weight:bolder;top:-8px;position:sticky;background-color:#fff;z-index:1}.menu-action{position:sticky;bottom:-8px;padding-top:10px;padding-bottom:0;background-color:#fff}.menu-action button{width:calc(50% - 10px);margin:5px;border-radius:10px}.filter-panel{border-radius:5px;background-color:#fdfbfb;border:solid 1px #efefef;transition:all .5s;padding:5px;overflow:hidden;font-size:14px;margin-top:10px;display:flex;flex-direction:column}.filter-panel:nth-child(2){margin-top:0!important}.filter-panel:hover{border:solid 1px #d1d1d1}.filter-panel:hover .svg{opacity:1;transform:translateY(-1px)}.or-and{display:inherit!important;text-align:right;margin:-12px 0;height:35px;cursor:inherit;font-size:12px}.svg{color:#0000004d;display:flex;opacity:0;transform:translateY(-5px);transition-duration:.4s;transition-property:opacity,transform;margin-left:5px;padding:2px;border-radius:5px;color:#4c4c4c;cursor:pointer;display:inline-block!important;height:24px}.svg mat-icon{margin:0;vertical-align:top;border-radius:5px}.svg mat-icon:hover{color:#fff;background-color:#89898a}.svg:hover{background-color:#f8f8f8}.selected-filter-type{float:left;color:#fff;background-color:#89898a;border-radius:5px;padding:0 4px;line-height:24px}::ng-deep .menu{padding:8px;-webkit-user-select:none;-moz-user-select:none;user-select:none}\n"]
},] }
];
/**
* @type {function(): !Array<(null|{
* type: ?,
* decorators: (undefined|!Array<{type: !Function, args: (undefined|!Array<?>)}>),
* })>}
* @nocollapse
*/
HeaderFilterComponent.ctorParameters = () => [
{ type: TableIntl },
{ type: TableService },
{ type: ChangeDetectorRef }
];
/** @type {!Object<string, !Array<{type: !Function, args: (undefined|!Array<?>)}>>} */
HeaderFilterComponent.propDecorators = {
field: [{ type: Input }],
filterChanged: [{ type: Output }],
filterInputList: [{ type: ContentChildren, args: ['filterInput',] }],
menu: [{ type: ViewChild, args: [MatMenuTrigger, { static: true },] }],
filters: [{ type: Input }],
hasValue: [{ type: HostBinding, args: ['class.has-value',] }],
showTrigger: [{ type: HostBinding, args: ['class.show-trigger',] }]
};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"header-filter.component.js","sourceRoot":"","sources":["../../../../../../../projects/dynamic-mat-table/src/lib/dynamic-mat-table/extensions/filter/header-filter.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAiB,uBAAuB,EAAE,SAAS,EAAE,WAAW,EAAE,MAAM,EAAE,SAAS,EACjF,KAAK,EAAE,YAAY,EAAU,iBAAiB,EAAE,SAAS,EAA2B,eAAe,EAAE,MAAM,eAAe,CAAC;AAEpI,OAAO,EAAE,YAAY,EAAE,MAAM,iCAAiC,CAAC;AAC/D,OAAO,EAAE,UAAU,EAAE,MAAM,uBAAuB,CAAC;AACnD,OAAO,EAAE,YAAY,EAAE,MAAM,yBAAyB,CAAC;AAEvD,OAAO,EAAE,UAAU,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,qBAAqB,CAAC;AAC1F,OAAO,EAAE,SAAS,EAAE,MAAM,mCAAmC,CAAC;AAE9D,OAAO,EAAE,cAAc,EAAE,MAAM,wBAAwB,CAAC;AACxD,OAAO,EAAE,iBAAiB,EAAE,MAAM,qBAAqB,CAAC;AAGxD,MAAM,aAAa,GAAG,OAAO,CAAC,eAAe,EAAE;IAC7C,UAAU,CAAC,SAAS,EAAE;QACpB,KAAK,CAAC,QAAQ,EACZ,CAAC,KAAK,CAAC,EAAE,OAAO,EAAE,CAAC,EAAE,CAAC,EAAE,OAAO,CAAC,MAAM,EAAE,OAAO,CAAC,gBAAgB,EAAE,KAAK,CAAC,EAAE,OAAO,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,EAC1F,EAAE,QAAQ,EAAE,IAAI,EAAE,CACnB;KACF,CAAC;CACH,CAAC,CAAC;AAUH,MAAM,OAAO,qBAAqB;IAmChC,YAAmB,YAAuB,EAAS,OAAqB,EAAU,GAAsB;QAArF,iBAAY,GAAZ,YAAY,CAAW;QAAS,YAAO,GAAP,OAAO,CAAc;QAAU,QAAG,GAAH,GAAG,CAAmB;QAjC9F,kBAAa,GAAmC,IAAI,YAAY,EAAoB,CAAC;QAKvF,eAAU,GAAqB,EAAE,CAAC;IA6B1C,CAAC;IA3BD,IACI,OAAO;QACT,IAAK,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,KAAK,IAAI,IAAI,IAAI,CAAC,UAAU,CAAC,MAAM,KAAK,CAAC,EAAE;YAChF,IAAI,CAAC,UAAU,GAAG,EAAE,CAAC;YACrB,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,IAAI,MAAM,CAAC,CAAC;SAC9C;QACD,OAAO,IAAI,CAAC,UAAU,CAAC;IACzB,CAAC;IACD,IAAI,OAAO,CAAC,MAAwB;QAClC,IAAI,CAAC,UAAU,GAAG,MAAM,CAAC;IAC3B,CAAC;IAED,IACI,QAAQ;QACV,OAAO,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,UAAU,CAAC,MAAM,CAAE,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,EAAE,KAAK,IAAI,CAAC,CAAC,MAAM,GAAG,CAAC,CAAC;IAC3F,CAAC;IAED,IACI,WAAW;QACb,IAAG,IAAI,CAAC,IAAI,KAAK,SAAS,EAAE;YAC1B,OAAO,KAAK,CAAC;SACd;aAAM;YACL,OAAO,IAAI,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,QAAQ,CAAC;SAC5C;IACH,CAAC;IAKD,WAAW;QACT,IAAI,IAAI,CAAC,kBAAkB,EAAE;YAC3B,IAAI,CAAC,kBAAkB,CAAC,WAAW,EAAE,CAAC;SACvC;IACH,CAAC;IAED,QAAQ;QACN,IAAI,iBAAiB,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE;YACnC,IAAI,CAAC,OAAO,GAAG,EAAE,CAAC;YAClB,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;SACpC;IACH,CAAC;IAED,YAAY,CAAC,OAAe,MAAM;QAChC,QAAQ,IAAI,IAAI,MAAM,EAAE;YACtB,KAAK,MAAM,CAAC,CAAC;gBACX,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,UAAU,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC;gBACxD,MAAM;aACP;YACD,KAAK,QAAQ,CAAC,CAAC;gBACb,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,YAAY,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC;gBAC1D,MAAM;aACP;YACD,KAAK,MAAM,CAAC,CAAC;gBACX,2CAA2C;gBAC3C,MAAM;aACP;YACD,KAAK,SAAS,CAAC,CAAC;gBACd,8CAA8C;gBAC9C,MAAM;aACP;YACD,OAAO,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,UAAU,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC;SAClE;QACD,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,aAAa,GAAG,CAAC,CAAC;QACxD,OAAO,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;IAC/C,CAAC;IAED,eAAe;QACb,IAAI,IAAI,CAAC,IAAI,EAAE;YACb,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC,CAAC;SACzF;IACH,CAAC;IAED,gBAAgB;QACd,UAAU,CAAC,GAAG,EAAE;YACd,IAAI,IAAI,CAAC,eAAe,CAAC,MAAM,GAAG,CAAC,EAAE;gBACnC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC;aACnC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED,oBAAoB,CAAC,KAAK,EAAE,MAAM;QAChC,IAAI,MAAM,KAAK,CAAC,IAAI,MAAM,KAAK,CAAC,EAAE,EAAE,SAAS;YAC3C,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,IAAI,GAAG,MAAM,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC;YACvD,IAAI,IAAI,CAAC,OAAO,CAAC,MAAM,KAAK,KAAK,GAAG,CAAC,EAAE;gBACrC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;gBACnC,IAAI,CAAC,gBAAgB,EAAE,CAAC;aACzB;SACF;aAAM,IAAI,MAAM,KAAK,CAAC,IAAI,IAAI,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,EAAE,SAAS;YAC7D,UAAU,CAAC,GAAG,EAAE;gBACd,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;gBAC9B,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;gBACzB,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC1B,CAAC,CAAC,CAAC,CAAC,2DAA2D;SAChE;IACH,CAAC;IAED,mBAAmB;QACjB,IAAI,CAAC,UAAU,GAAG,EAAE,CAAC;QACrB,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAC3C,CAAC;IAED,mBAAmB;QACjB,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAC3C,CAAC;;;;YAxHF,SAAS,SAAC;gBACT,8CAA8C;gBAC9C,QAAQ,EAAE,eAAe;gBACzB,q+EAA6C;gBAE7C,eAAe,EAAE,uBAAuB,CAAC,MAAM;gBAC/C,UAAU,EAAE,CAAC,aAAa,CAAC;;aAC5B;;;;;;;;;;YAtBQ,SAAS;YALT,YAAY;YAFiB,iBAAiB;;;;oBA+BpD,KAAK;4BACL,MAAM;8BAEN,eAAe,SAAC,aAAa;mBAC7B,SAAS,SAAC,cAAc,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE;sBAI1C,KAAK;uBAYL,WAAW,SAAC,iBAAiB;0BAK7B,WAAW,SAAC,oBAAoB","sourcesContent":["import { AfterViewInit, ChangeDetectionStrategy, Component, HostBinding, Output, ViewChild,\r\n         Input, EventEmitter, OnInit, ChangeDetectorRef, QueryList, ViewChildren, OnDestroy, ContentChildren } from '@angular/core';\r\nimport { TableField } from './../../../models/table-field.model';\r\nimport { TableService } from '../../dynamic-mat-table.service';\r\nimport { TextFilter } from './compare/text-filter';\r\nimport { NumberFilter } from './compare/number-filter';\r\nimport { AbstractFilter } from './compare/abstract-filter';\r\nimport { transition, trigger, query, style, stagger, animate } from '@angular/animations';\r\nimport { TableIntl } from '../../../international/table-Intl';\r\nimport { MatInput } from '@angular/material/input';\r\nimport { MatMenuTrigger } from '@angular/material/menu';\r\nimport { isNullorUndefined } from '../../../cores/type';\r\nimport { Subscription } from 'rxjs';\r\n\r\nconst listAnimation = trigger('listAnimation', [\r\n  transition('* <=> *', [\r\n    query(':enter',\r\n      [style({ opacity: 0 }), stagger('10ms', animate('400ms ease-out', style({ opacity: 1 })))],\r\n      { optional: true }\r\n    ),\r\n  ])\r\n]);\r\n\r\n@Component({\r\n  // tslint:disable-next-line:component-selector\r\n  selector: 'header-filter',\r\n  templateUrl: './header-filter.component.html',\r\n  styleUrls: ['./header-filter.component.scss'],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  animations: [listAnimation]\r\n})\r\nexport class HeaderFilterComponent implements OnInit, AfterViewInit, OnDestroy {\r\n  @Input() field?: TableField<any>;\r\n  @Output() filterChanged: EventEmitter<AbstractFilter[]> = new EventEmitter<AbstractFilter[]>();\r\n\r\n  @ContentChildren('filterInput') filterInputList !: QueryList<MatInput>;\r\n  @ViewChild(MatMenuTrigger, { static: true }) menu !: MatMenuTrigger;\r\n\r\n  private filterList: AbstractFilter[] = [];\r\n  private eventsSubscription: Subscription;\r\n  @Input()\r\n  get filters(): AbstractFilter[] {\r\n    if ( isNullorUndefined(this.filterList) === true || this.filterList.length === 0) {\r\n      this.filterList = [];\r\n      this.addNewFilter(this.field.type || 'text');\r\n    }\r\n    return this.filterList;\r\n  }\r\n  set filters(values: AbstractFilter[]) {\r\n    this.filterList = values;\r\n  }\r\n\r\n  @HostBinding('class.has-value')\r\n  get hasValue(): boolean {\r\n    return this.filterList && this.filterList.filter( f => f.hasValue() === true).length > 0;\r\n  }\r\n\r\n  @HostBinding('class.show-trigger')\r\n  get showTrigger(): boolean {\r\n    if(this.menu === undefined) {\r\n      return false;\r\n    } else {\r\n      return this.menu.menuOpen || this.hasValue;\r\n    }\r\n  }\r\n\r\n  constructor(public languagePack: TableIntl, public service: TableService, private cdr: ChangeDetectorRef) {\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    if (this.eventsSubscription) {\r\n      this.eventsSubscription.unsubscribe();\r\n    }\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    if (isNullorUndefined(this.filters)) {\r\n      this.filters = [];\r\n      this.addNewFilter(this.field.type);\r\n    }\r\n  }\r\n\r\n  addNewFilter(type: string = 'text') {\r\n    switch (type || 'text') {\r\n      case 'text': {\r\n        this.filterList.push(new TextFilter(this.languagePack));\r\n        break;\r\n      }\r\n      case 'number': {\r\n        this.filterList.push(new NumberFilter(this.languagePack));\r\n        break;\r\n      }\r\n      case 'date': {\r\n        // this.compare = new DateCompare(service);\r\n        break;\r\n      }\r\n      case 'boolean': {\r\n        // this.compare = new BooleanCompare(service);\r\n        break;\r\n      }\r\n      default: this.filterList.push(new TextFilter(this.languagePack));\r\n    }\r\n    this.filters[this.filters.length - 1].selectedIndex = 0;\r\n    return this.filters[this.filters.length - 1];\r\n  }\r\n\r\n  ngAfterViewInit() {\r\n    if (this.menu) {\r\n      this.eventsSubscription = this.menu.menuOpened.subscribe(() => this.focusToLastInput());\r\n    }\r\n  }\r\n\r\n  focusToLastInput() {\r\n    setTimeout(() => {\r\n      if (this.filterInputList.length > 0) {\r\n        this.filterInputList.last.focus();\r\n      }\r\n    });\r\n  }\r\n\r\n  filterAction_OnClick(index, action) {\r\n    if (action === 0 || action === 1) { // and or\r\n      this.filters[index].type = action === 0 ? 'and' : 'or';\r\n      if (this.filters.length === index + 1) {\r\n        this.addNewFilter(this.field.type);\r\n        this.focusToLastInput();\r\n      }\r\n    } else if (action === 2 && this.filters.length > 1) { // delete\r\n      setTimeout(() => {\r\n        this.filters.splice(index, 1);\r\n        this.cdr.detectChanges();\r\n        this.focusToLastInput();\r\n      }); // bug for delete filter item(unwanted reaction close menu)\r\n    }\r\n  }\r\n\r\n  clearColumn_OnClick() {\r\n    this.filterList = [];\r\n    this.filterChanged.emit(this.filterList);\r\n  }\r\n\r\n  applyFilter_OnClick() {\r\n    this.filterChanged.emit(this.filterList);\r\n  }\r\n\r\n}\r\n\r\n"]}