ngx-ui-hero
Version:
Simple, fast and reliable utilities for Angular.
194 lines • 43.3 kB
JavaScript
import { Component, EventEmitter, Inject, Input, IterableDiffers, Optional, Output, ViewChild } from '@angular/core';
import { NG_ASYNC_VALIDATORS, NG_VALIDATORS, NG_VALUE_ACCESSOR, NgModel } from '@angular/forms';
import { ElementBase } from '../../base/element-base';
import { INPUT_FORMS_CONFIG } from '../../input-forms-config.constants';
import * as i0 from "@angular/core";
import * as i1 from "@angular/common";
import * as i2 from "@angular/forms";
import * as i3 from "../../../ui/directives/click-outside.directive";
import * as i4 from "../../../ui/directives/auto-focus.directive";
import * as i5 from "ngx-bootstrap/tooltip";
import * as i6 from "../../../ui/pipes/search-by.pipe";
let identifier = 0;
export class InputMultiselectComponent extends ElementBase {
constructor(validators, asyncValidators, config, iterableDiffers) {
super(validators, asyncValidators, config);
this.config = config;
this.iterableDiffers = iterableDiffers;
this.placeholder = 'Select...';
this.searchPlaceholder = 'Search...';
this.displayTextProperty = 'text';
this.valueProperty = 'value';
this.emptyMessage = 'No results found.';
this.selectAllButtonLabel = 'Select all';
this.clearSelectionButtonLabel = 'Clear selection';
this.maxCountOfLabelsToShow = 3;
this.onChange = new EventEmitter();
this.identifier = `input-multiselect-${identifier++}`;
if (config.multiSelect) {
Object.assign(this, config.multiSelect);
}
this._differModel = this.iterableDiffers.find([]).create(null);
this._differOptions = this.iterableDiffers.find([]).create(null);
}
get options() {
return this._options;
}
set options(value) {
this._options = value;
this.Init();
}
ngOnInit() {
this.Init();
}
ngDoCheck() {
let changesInModel = this._differModel.diff(this.value);
let changesInOptions = this._differOptions.diff(this._options);
if (changesInModel || (changesInOptions && !this.modelInitialized)) {
this.Init();
}
}
Init() {
if (!this.value || !this.options || this.value.length == 0 || this.options.length == 0) {
if ((!this.value || this.value.length == 0) && this.options) {
this.ToggleAllItemsSelection(false, false);
}
return;
}
for (let i = 0; i < this.options.length; i++) {
this.options[i].selected = this.value.filter(x => this.options[i][this.valueProperty] == x[this.valueProperty]).length > 0;
}
this.modelInitialized = true;
}
ToggleDropDown(event, value) {
if ((value == false && !this.showOptions) || (value == undefined && this.disabled))
return;
if (value == undefined) {
if (this.showOptions) {
this.setComboTouched();
}
this.showOptions = !this.showOptions;
}
else {
if (!value && this.showOptions) {
this.setComboTouched();
}
this.showOptions = value;
}
this.clearSearch();
}
ToggleItemSelected(item) {
if (this.disabled) {
return;
}
item.selected = !item.selected;
this.updateModel();
this.onChange.emit();
}
ToggleAllItemsSelection(value, emitChanges) {
if (this.disabled) {
return;
}
if (!this.options || this.options.length == 0)
return;
for (let i = 0; i < this.options.length; i++) {
this.options[i].selected = value;
}
this.updateModel();
this.clearSearch();
if (emitChanges)
this.onChange.emit();
}
ItemSelectedCheckChanged() {
if (this.disabled) {
return;
}
this.updateModel();
this.onChange.emit();
}
RemoveItem(item, index, event) {
if (this.disabled || !this.options || this.options.length == 0) {
return;
}
let itemToRemove = this.options.find(x => x[this.valueProperty] == item[this.valueProperty]);
if (itemToRemove) {
this.ToggleItemSelected(itemToRemove);
this.clearSearch();
}
else {
this.value.splice(index, 1);
}
event.stopPropagation();
}
OnComboPressed(event) {
if (event.keyCode == 13) {
this.ToggleDropDown(null);
event.preventDefault();
}
}
clearSearch() {
this.search = '';
}
updateModel() {
if (!this.options || this.options.length == 0) {
this.value = [];
return;
}
this.value = this.options.filter(x => x.selected);
}
setComboTouched() {
this.comboTouched = true;
}
}
InputMultiselectComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: InputMultiselectComponent, deps: [{ token: NG_VALIDATORS, optional: true }, { token: NG_ASYNC_VALIDATORS, optional: true }, { token: INPUT_FORMS_CONFIG }, { token: i0.IterableDiffers }], target: i0.ɵɵFactoryTarget.Component });
InputMultiselectComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: InputMultiselectComponent, selector: "input-multiselect", inputs: { placeholder: "placeholder", searchPlaceholder: "searchPlaceholder", displayTextProperty: "displayTextProperty", valueProperty: "valueProperty", emptyMessage: "emptyMessage", selectAllButtonLabel: "selectAllButtonLabel", clearSelectionButtonLabel: "clearSelectionButtonLabel", maxCountOfLabelsToShow: "maxCountOfLabelsToShow", options: "options" }, outputs: { onChange: "onChange" }, providers: [{
provide: NG_VALUE_ACCESSOR,
useExisting: InputMultiselectComponent,
multi: true
}], viewQueries: [{ propertyName: "model", first: true, predicate: NgModel, descendants: true, static: true }], usesInheritance: true, ngImport: i0, template: "<div [id]=\"identifier\" class=\"ui-multiselect\">\n <label *ngIf=\"label\">\n {{label}}\n <span class=\"required-symbol\" *ngIf=\"isRequired\"> *</span>\n <i class=\"fa fa-question-circle ml-2\" tooltip=\"{{help}}\" container=\"body\" *ngIf=\"help\"></i>\n </label>\n\n <input type=\"hidden\" [(ngModel)]=\"value\" />\n <div class=\"combobox d-flex align-items-center justify-content-between\" \n [ngClass]=\"{\n 'has-validations': showValidations,\n 'open': showOptions, \n 'disabled': disabled,\n 'is-valid': !showOptions && !disabled && comboTouched && (!isRequired || value?.length > 0),\n 'is-invalid': !showOptions && !disabled && comboTouched && isRequired && (!value || value.length == 0)\n }\"\n (click)=\"ToggleDropDown($event)\" (keyup)=\"OnComboPressed($event)\" tabindex=\"0\">\n\n <span *ngIf=\"!value || value.length == 0\">{{placeholder}}</span>\n\n <div *ngIf=\"value?.length > 0\" class=\"selected-items d-flex align-items-center\">\n <span class=\"badge\" *ngFor=\"let item of value | slice:0:maxCountOfLabelsToShow; let i = index;\" (click)=\"RemoveItem(item, i, $event)\">\n <span class=\"text-truncate\">{{item[displayTextProperty]}}</span>\n <i class=\"fa fa-times ml-2\" *ngIf=\"!disabled\"></i>\n </span>\n <span *ngIf=\"value.length > maxCountOfLabelsToShow\">...</span>\n </div>\n\n <i class=\"fa\" [ngClass]=\"{'fa-caret-down': !showOptions, 'fa-caret-up': showOptions}\"></i>\n </div>\n\n <div class=\"dropdown-options shadow fadeInDown\" *ngIf=\"showOptions\" (clickOutside)=\"ToggleDropDown($event, false)\">\n <div class=\"options-container\" *ngIf=\"options?.length > 0\">\n <div class=\"toolbar d-flex align-items-center\">\n <input type=\"text\" class=\"form-control flex-grow-1\" name=\"{{identifier}}-search\" [(ngModel)]=\"search\" placeholder=\"{{searchPlaceholder}}\" auto-focus />\n <button type=\"button\" class=\"btn btn-light ml-1 flex-shrink-0\" (click)=\"ToggleAllItemsSelection(true, true)\" [disabled]=\"disabled\"><i class=\"fa fa-check-square\"></i> {{selectAllButtonLabel}}</button>\n <button type=\"button\" class=\"btn btn-light ml-1 flex-shrink-0\" (click)=\"ToggleAllItemsSelection(false, true)\" [disabled]=\"disabled\"><i class=\"fa fa-eraser\"></i> {{clearSelectionButtonLabel}}</button>\n </div>\n\n <ul>\n <li class=\"d-flex\" *ngFor=\"let item of options | searchBy : displayTextProperty : search\" [ngClass]=\"{'selected': item.selected}\">\n <div class=\"checkbox mr-3 flex-shrink-0\">\n <input type=\"checkbox\" name=\"{{identifier}}-option\" [(ngModel)]=\"item.selected\" (change)=\"ItemSelectedCheckChanged()\" [disabled]=\"disabled\" />\n </div>\n <span class=\"link\" (click)=\"ToggleItemSelected(item)\">{{item[displayTextProperty]}}</span>\n </li>\n </ul>\n\n <small class=\"empty-results\" *ngIf=\"(options | searchBy : displayTextProperty : search).length == 0\">{{emptyMessage}}</small>\n </div>\n\n <small class=\"empty-results\" *ngIf=\"!options || options.length == 0\">{{emptyMessage}}</small>\n </div>\n</div>\n\n<small class=\"text-muted\" *ngIf=\"description\">{{description}}</small>\n", styles: ["@keyframes fadeInDown{0%{opacity:0;transform:translate3d(0,-5%,0)}to{opacity:1;transform:translateZ(0)}}.fadeInDown{animation-name:fadeInDown;animation-duration:.35s;animation-fill-mode:both}.ui-multiselect{position:relative}.ui-multiselect *{-webkit-user-select:none;user-select:none}.ui-multiselect>.combobox{outline:none;background-color:#fff;padding:.375rem .4rem .375rem .75rem;border:1px solid #ced4da;border-radius:.25rem}.ui-multiselect>.combobox>span{color:#6c757d;font-size:1rem;line-height:1.5}.ui-multiselect>.combobox>i{color:#495057;font-size:14px;margin-left:10px}.ui-multiselect>.combobox .selected-items{margin-left:-5px;overflow:hidden}.ui-multiselect>.combobox .selected-items .badge{margin-right:3px;line-height:1;color:#888;background-color:#e0e0e0;border-bottom:2px solid #ccc;border-left:2px solid #ccc;padding:5px 7px 4px}.ui-multiselect>.combobox .selected-items .badge>span{display:inline-block;max-width:150px}.ui-multiselect>.combobox .selected-items .badge:hover{cursor:pointer;color:#fff;background-color:#007bff;border-bottom:2px solid #00559a;border-left:2px solid #00559a}.ui-multiselect>.combobox.open{border-radius:.25rem .25rem 0 0}.ui-multiselect>.combobox.disabled{background-color:#e9ecef;border-color:#ced4da}.ui-multiselect>.combobox.has-validations.is-invalid{border-color:#dc3545}.ui-multiselect>.combobox.has-validations.is-invalid.open{box-shadow:0 0 0 .2rem #dc354540}.ui-multiselect>.combobox.has-validations.is-valid{border-color:#28a745}.ui-multiselect>.combobox.has-validations.is-valid.open{box-shadow:0 0 0 .2rem #28a74540}.ui-multiselect>.dropdown-options{background-color:#fff;border:1px solid #ced4da;border-radius:0 0 .25rem .25rem;position:absolute;top:100%;left:0;width:100%;z-index:20;margin-top:-1px;min-width:300px}.ui-multiselect>.dropdown-options>.options-container{display:inherit;position:initial;font-family:inherit;background-color:inherit;box-shadow:none;height:auto;width:auto;color:inherit;font-size:inherit;letter-spacing:inherit;line-height:inherit;padding:inherit;border-radius:inherit}.ui-multiselect>.dropdown-options .toolbar{padding:3px;border-bottom:1px solid #ced4da;background-color:#f8f9fa}.ui-multiselect>.dropdown-options .toolbar input[type=text]{background-color:transparent;border:0;padding:.375rem .55rem;box-shadow:none!important;min-width:0}.ui-multiselect>.dropdown-options ul{max-height:300px;overflow-y:scroll;overflow-x:hidden;margin:0;padding:0}.ui-multiselect>.dropdown-options ul li{list-style:none;border-bottom:1px solid #ced4da}.ui-multiselect>.dropdown-options ul li>.checkbox{background-color:#f0f0f0;border-right:1px solid #ced4da;padding:12px 15px;display:flex;align-items:center}.ui-multiselect>.dropdown-options ul li>.checkbox input[type=checkbox]{width:17px;height:17px}.ui-multiselect>.dropdown-options ul li>span{cursor:pointer;padding:8px 0}.ui-multiselect>.dropdown-options ul li.selected{color:#007bff}.ui-multiselect>.dropdown-options ul li:hover{background-color:#f9f9f9}.ui-multiselect>.dropdown-options ul li:last-child{border-bottom:0}.ui-multiselect>.dropdown-options .empty-results{font-style:italic;padding:12px;display:block;color:#6b6b6b}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.CheckboxControlValueAccessor, selector: "input[type=checkbox][formControlName],input[type=checkbox][formControl],input[type=checkbox][ngModel]" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: i3.ClickOutsideDirective, selector: "[clickOutside]", inputs: ["clickOutsideEnabled", "attachOutsideOnClick", "delayClickOutsideInit", "emitOnBlur", "exclude", "excludeBeforeClick", "clickOutsideEvents"], outputs: ["clickOutside"] }, { kind: "directive", type: i4.AutoFocusDirective, selector: "[auto-focus]" }, { kind: "directive", type: i5.TooltipDirective, selector: "[tooltip], [tooltipHtml]", inputs: ["containerClass", "tooltipAnimation", "tooltipFadeDuration", "isOpen", "tooltipHtml", "tooltip", "tooltipPlacement", "placement", "tooltipIsOpen", "tooltipEnable", "isDisabled", "tooltipAppendToBody", "container", "tooltipClass", "tooltipContext", "tooltipPopupDelay", "delay", "tooltipTrigger", "triggers", "adaptivePosition"], outputs: ["tooltipChange", "tooltipStateChanged", "onShown", "onHidden"], exportAs: ["bs-tooltip"] }, { kind: "pipe", type: i1.SlicePipe, name: "slice" }, { kind: "pipe", type: i6.SearchByPipe, name: "searchBy" }] });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: InputMultiselectComponent, decorators: [{
type: Component,
args: [{ selector: 'input-multiselect', providers: [{
provide: NG_VALUE_ACCESSOR,
useExisting: InputMultiselectComponent,
multi: true
}], template: "<div [id]=\"identifier\" class=\"ui-multiselect\">\n <label *ngIf=\"label\">\n {{label}}\n <span class=\"required-symbol\" *ngIf=\"isRequired\"> *</span>\n <i class=\"fa fa-question-circle ml-2\" tooltip=\"{{help}}\" container=\"body\" *ngIf=\"help\"></i>\n </label>\n\n <input type=\"hidden\" [(ngModel)]=\"value\" />\n <div class=\"combobox d-flex align-items-center justify-content-between\" \n [ngClass]=\"{\n 'has-validations': showValidations,\n 'open': showOptions, \n 'disabled': disabled,\n 'is-valid': !showOptions && !disabled && comboTouched && (!isRequired || value?.length > 0),\n 'is-invalid': !showOptions && !disabled && comboTouched && isRequired && (!value || value.length == 0)\n }\"\n (click)=\"ToggleDropDown($event)\" (keyup)=\"OnComboPressed($event)\" tabindex=\"0\">\n\n <span *ngIf=\"!value || value.length == 0\">{{placeholder}}</span>\n\n <div *ngIf=\"value?.length > 0\" class=\"selected-items d-flex align-items-center\">\n <span class=\"badge\" *ngFor=\"let item of value | slice:0:maxCountOfLabelsToShow; let i = index;\" (click)=\"RemoveItem(item, i, $event)\">\n <span class=\"text-truncate\">{{item[displayTextProperty]}}</span>\n <i class=\"fa fa-times ml-2\" *ngIf=\"!disabled\"></i>\n </span>\n <span *ngIf=\"value.length > maxCountOfLabelsToShow\">...</span>\n </div>\n\n <i class=\"fa\" [ngClass]=\"{'fa-caret-down': !showOptions, 'fa-caret-up': showOptions}\"></i>\n </div>\n\n <div class=\"dropdown-options shadow fadeInDown\" *ngIf=\"showOptions\" (clickOutside)=\"ToggleDropDown($event, false)\">\n <div class=\"options-container\" *ngIf=\"options?.length > 0\">\n <div class=\"toolbar d-flex align-items-center\">\n <input type=\"text\" class=\"form-control flex-grow-1\" name=\"{{identifier}}-search\" [(ngModel)]=\"search\" placeholder=\"{{searchPlaceholder}}\" auto-focus />\n <button type=\"button\" class=\"btn btn-light ml-1 flex-shrink-0\" (click)=\"ToggleAllItemsSelection(true, true)\" [disabled]=\"disabled\"><i class=\"fa fa-check-square\"></i> {{selectAllButtonLabel}}</button>\n <button type=\"button\" class=\"btn btn-light ml-1 flex-shrink-0\" (click)=\"ToggleAllItemsSelection(false, true)\" [disabled]=\"disabled\"><i class=\"fa fa-eraser\"></i> {{clearSelectionButtonLabel}}</button>\n </div>\n\n <ul>\n <li class=\"d-flex\" *ngFor=\"let item of options | searchBy : displayTextProperty : search\" [ngClass]=\"{'selected': item.selected}\">\n <div class=\"checkbox mr-3 flex-shrink-0\">\n <input type=\"checkbox\" name=\"{{identifier}}-option\" [(ngModel)]=\"item.selected\" (change)=\"ItemSelectedCheckChanged()\" [disabled]=\"disabled\" />\n </div>\n <span class=\"link\" (click)=\"ToggleItemSelected(item)\">{{item[displayTextProperty]}}</span>\n </li>\n </ul>\n\n <small class=\"empty-results\" *ngIf=\"(options | searchBy : displayTextProperty : search).length == 0\">{{emptyMessage}}</small>\n </div>\n\n <small class=\"empty-results\" *ngIf=\"!options || options.length == 0\">{{emptyMessage}}</small>\n </div>\n</div>\n\n<small class=\"text-muted\" *ngIf=\"description\">{{description}}</small>\n", styles: ["@keyframes fadeInDown{0%{opacity:0;transform:translate3d(0,-5%,0)}to{opacity:1;transform:translateZ(0)}}.fadeInDown{animation-name:fadeInDown;animation-duration:.35s;animation-fill-mode:both}.ui-multiselect{position:relative}.ui-multiselect *{-webkit-user-select:none;user-select:none}.ui-multiselect>.combobox{outline:none;background-color:#fff;padding:.375rem .4rem .375rem .75rem;border:1px solid #ced4da;border-radius:.25rem}.ui-multiselect>.combobox>span{color:#6c757d;font-size:1rem;line-height:1.5}.ui-multiselect>.combobox>i{color:#495057;font-size:14px;margin-left:10px}.ui-multiselect>.combobox .selected-items{margin-left:-5px;overflow:hidden}.ui-multiselect>.combobox .selected-items .badge{margin-right:3px;line-height:1;color:#888;background-color:#e0e0e0;border-bottom:2px solid #ccc;border-left:2px solid #ccc;padding:5px 7px 4px}.ui-multiselect>.combobox .selected-items .badge>span{display:inline-block;max-width:150px}.ui-multiselect>.combobox .selected-items .badge:hover{cursor:pointer;color:#fff;background-color:#007bff;border-bottom:2px solid #00559a;border-left:2px solid #00559a}.ui-multiselect>.combobox.open{border-radius:.25rem .25rem 0 0}.ui-multiselect>.combobox.disabled{background-color:#e9ecef;border-color:#ced4da}.ui-multiselect>.combobox.has-validations.is-invalid{border-color:#dc3545}.ui-multiselect>.combobox.has-validations.is-invalid.open{box-shadow:0 0 0 .2rem #dc354540}.ui-multiselect>.combobox.has-validations.is-valid{border-color:#28a745}.ui-multiselect>.combobox.has-validations.is-valid.open{box-shadow:0 0 0 .2rem #28a74540}.ui-multiselect>.dropdown-options{background-color:#fff;border:1px solid #ced4da;border-radius:0 0 .25rem .25rem;position:absolute;top:100%;left:0;width:100%;z-index:20;margin-top:-1px;min-width:300px}.ui-multiselect>.dropdown-options>.options-container{display:inherit;position:initial;font-family:inherit;background-color:inherit;box-shadow:none;height:auto;width:auto;color:inherit;font-size:inherit;letter-spacing:inherit;line-height:inherit;padding:inherit;border-radius:inherit}.ui-multiselect>.dropdown-options .toolbar{padding:3px;border-bottom:1px solid #ced4da;background-color:#f8f9fa}.ui-multiselect>.dropdown-options .toolbar input[type=text]{background-color:transparent;border:0;padding:.375rem .55rem;box-shadow:none!important;min-width:0}.ui-multiselect>.dropdown-options ul{max-height:300px;overflow-y:scroll;overflow-x:hidden;margin:0;padding:0}.ui-multiselect>.dropdown-options ul li{list-style:none;border-bottom:1px solid #ced4da}.ui-multiselect>.dropdown-options ul li>.checkbox{background-color:#f0f0f0;border-right:1px solid #ced4da;padding:12px 15px;display:flex;align-items:center}.ui-multiselect>.dropdown-options ul li>.checkbox input[type=checkbox]{width:17px;height:17px}.ui-multiselect>.dropdown-options ul li>span{cursor:pointer;padding:8px 0}.ui-multiselect>.dropdown-options ul li.selected{color:#007bff}.ui-multiselect>.dropdown-options ul li:hover{background-color:#f9f9f9}.ui-multiselect>.dropdown-options ul li:last-child{border-bottom:0}.ui-multiselect>.dropdown-options .empty-results{font-style:italic;padding:12px;display:block;color:#6b6b6b}\n"] }]
}], ctorParameters: function () { return [{ type: undefined, decorators: [{
type: Optional
}, {
type: Inject,
args: [NG_VALIDATORS]
}] }, { type: undefined, decorators: [{
type: Optional
}, {
type: Inject,
args: [NG_ASYNC_VALIDATORS]
}] }, { type: undefined, decorators: [{
type: Inject,
args: [INPUT_FORMS_CONFIG]
}] }, { type: i0.IterableDiffers }]; }, propDecorators: { placeholder: [{
type: Input
}], searchPlaceholder: [{
type: Input
}], displayTextProperty: [{
type: Input
}], valueProperty: [{
type: Input
}], emptyMessage: [{
type: Input
}], selectAllButtonLabel: [{
type: Input
}], clearSelectionButtonLabel: [{
type: Input
}], maxCountOfLabelsToShow: [{
type: Input
}], onChange: [{
type: Output
}], model: [{
type: ViewChild,
args: [NgModel, { static: true }]
}], options: [{
type: Input,
args: ['options']
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"input-multiselect.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-ui-hero/src/lib/input-forms/components/input-multiselect/input-multiselect.component.ts","../../../../../../../projects/ngx-ui-hero/src/lib/input-forms/components/input-multiselect/input-multiselect.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, DoCheck, EventEmitter, Inject, Input, IterableDiffers, OnInit, Optional, Output, ViewChild } from '@angular/core';\nimport { NG_ASYNC_VALIDATORS, NG_VALIDATORS, NG_VALUE_ACCESSOR, NgModel } from '@angular/forms';\n\nimport { ElementBase } from '../../base/element-base';\nimport { AsyncValidatorArray, ValidatorArray } from '../../base/validate';\nimport { InputFormsConfig } from '../../input-forms-config';\nimport { INPUT_FORMS_CONFIG } from '../../input-forms-config.constants';\n\nlet identifier = 0;\n\n@Component({\n  selector: 'input-multiselect',\n  templateUrl: './input-multiselect.component.html',\n  styleUrls: ['./input-multiselect.component.scss'],\n  providers: [{\n    provide: NG_VALUE_ACCESSOR,\n    useExisting: InputMultiselectComponent,\n    multi: true\n  }]\n})\nexport class InputMultiselectComponent extends ElementBase<any> implements OnInit, DoCheck {\n  private _differModel: any;  \n  private _differOptions: any;  \n  private _options: Array<any>;\n  showOptions: boolean;\n  comboTouched: boolean;\n  modelInitialized: boolean;\n  search: string;\n\n  @Input() public placeholder = 'Select...';\n  @Input() public searchPlaceholder = 'Search...';\n  @Input() public displayTextProperty: string = 'text';\n  @Input() public valueProperty: string = 'value';\n  @Input() public emptyMessage: string = 'No results found.';\n  @Input() public selectAllButtonLabel: string = 'Select all';\n  @Input() public clearSelectionButtonLabel: string = 'Clear selection';\n  @Input() public maxCountOfLabelsToShow: number = 3;\n  @Output() public onChange = new EventEmitter<any>();\n  @ViewChild(NgModel, {static: true}) model: NgModel;\n\n  get options(): Array<any> {\n    return this._options;\n  }    \n  @Input('options')\n  set options(value: Array<any>) {\n    this._options = value;\n    this.Init();\n  }\n  \n  public identifier = `input-multiselect-${identifier++}`;\n\n  constructor(\n    @Optional() @Inject(NG_VALIDATORS) validators: ValidatorArray,\n    @Optional() @Inject(NG_ASYNC_VALIDATORS) asyncValidators: AsyncValidatorArray,\n    @Inject( INPUT_FORMS_CONFIG ) public config: InputFormsConfig,\n    private iterableDiffers: IterableDiffers,\n  ) {\n    super(validators, asyncValidators, config);\n\n    if (config.multiSelect) {\n      Object.assign(this, config.multiSelect);\n    }\n\n    this._differModel = this.iterableDiffers.find([]).create(null);\n    this._differOptions = this.iterableDiffers.find([]).create(null);\n  }\n\n  ngOnInit(): void {\n    this.Init();\n  }\n  ngDoCheck(): void {\n    let changesInModel = this._differModel.diff(this.value);\n    let changesInOptions = this._differOptions.diff(this._options);\n\n    if (changesInModel || (changesInOptions && !this.modelInitialized)) {\n      this.Init();\n    }\n  }\n\n  Init(): void {\n    if (!this.value || !this.options || this.value.length == 0 || this.options.length == 0) {      \n      if ((!this.value || this.value.length == 0) && this.options) {\n        this.ToggleAllItemsSelection(false, false);\n      }\n\n      return;\n    }\n\n    for (let i = 0; i < this.options.length; i++) {\n      this.options[i].selected = this.value.filter(x => this.options[i][this.valueProperty] == x[this.valueProperty]).length > 0;\n    }\n\n    this.modelInitialized = true;\n  }\n\n  ToggleDropDown(event: MouseEvent, value?: boolean): void {\n    if ((value == false && !this.showOptions) || (value == undefined && this.disabled)) return;\n\n    if (value == undefined) { \n      if (this.showOptions) {\n        this.setComboTouched();\n      }\n\n      this.showOptions = !this.showOptions;      \n    } else {\n      if (!value && this.showOptions) {\n        this.setComboTouched();\n      }\n\n      this.showOptions = value;\n    }\n    \n    this.clearSearch();\n  }\n  ToggleItemSelected(item: any): void {\n    if (this.disabled) {\n      return;\n    }\n    \n    item.selected = !item.selected;\n    this.updateModel();\n    this.onChange.emit();\n  }\n  ToggleAllItemsSelection(value: boolean, emitChanges: boolean): void {\n    if (this.disabled) {\n      return;\n    }\n    \n    if (!this.options || this.options.length == 0) return;\n\n    for (let i = 0; i < this.options.length; i++) {\n      this.options[i].selected = value;\n    }\n\n    this.updateModel();\n    this.clearSearch();\n\n    if (emitChanges) this.onChange.emit();\n  }\n  ItemSelectedCheckChanged(): void {\n    if (this.disabled) {\n      return;\n    }\n    \n    this.updateModel();\n    this.onChange.emit();\n  }  \n\n  RemoveItem(item: any, index: number, event: any) {\n    if (this.disabled || !this.options || this.options.length == 0) {\n      return;\n    }\n    \n    let itemToRemove = this.options.find(x => x[this.valueProperty] == item[this.valueProperty]);\n\n    if (itemToRemove) {\n      this.ToggleItemSelected(itemToRemove);\n      this.clearSearch();\n    } else {\n      this.value.splice(index, 1);\n    }\n\n    event.stopPropagation();\n  }\n  \n  OnComboPressed(event: KeyboardEvent): void {\n    if (event.keyCode == 13) {\n      this.ToggleDropDown(null);\n      event.preventDefault();\n    }\n  }\n\n  private clearSearch(): void {\n    this.search = '';\n  }\n  private updateModel(): void {\n    if (!this.options || this.options.length == 0) {\n      this.value = [];\n      return;\n    }\n    \n    this.value = this.options.filter(x => x.selected);\n  }\n  private setComboTouched(): void {\n    this.comboTouched = true;\n  }\n\n}\n","<div [id]=\"identifier\" class=\"ui-multiselect\">\n    <label *ngIf=\"label\">\n        {{label}}\n        <span class=\"required-symbol\" *ngIf=\"isRequired\"> *</span>\n        <i class=\"fa fa-question-circle ml-2\" tooltip=\"{{help}}\" container=\"body\" *ngIf=\"help\"></i>\n    </label>\n\n    <input type=\"hidden\" [(ngModel)]=\"value\" />\n    <div class=\"combobox d-flex align-items-center justify-content-between\" \n        [ngClass]=\"{\n            'has-validations': showValidations,\n            'open': showOptions, \n            'disabled': disabled,\n            'is-valid': !showOptions && !disabled && comboTouched && (!isRequired || value?.length > 0),\n            'is-invalid': !showOptions && !disabled && comboTouched && isRequired && (!value || value.length == 0)\n        }\"\n        (click)=\"ToggleDropDown($event)\" (keyup)=\"OnComboPressed($event)\" tabindex=\"0\">\n\n        <span *ngIf=\"!value || value.length == 0\">{{placeholder}}</span>\n\n        <div *ngIf=\"value?.length > 0\" class=\"selected-items d-flex align-items-center\">\n            <span class=\"badge\" *ngFor=\"let item of value | slice:0:maxCountOfLabelsToShow; let i = index;\" (click)=\"RemoveItem(item, i, $event)\">\n                <span class=\"text-truncate\">{{item[displayTextProperty]}}</span>\n                <i class=\"fa fa-times ml-2\" *ngIf=\"!disabled\"></i>\n            </span>\n            <span *ngIf=\"value.length > maxCountOfLabelsToShow\">...</span>\n        </div>\n\n        <i class=\"fa\" [ngClass]=\"{'fa-caret-down': !showOptions, 'fa-caret-up': showOptions}\"></i>\n    </div>\n\n    <div class=\"dropdown-options shadow fadeInDown\" *ngIf=\"showOptions\" (clickOutside)=\"ToggleDropDown($event, false)\">\n        <div class=\"options-container\" *ngIf=\"options?.length > 0\">\n            <div class=\"toolbar d-flex align-items-center\">\n                <input type=\"text\" class=\"form-control flex-grow-1\" name=\"{{identifier}}-search\" [(ngModel)]=\"search\" placeholder=\"{{searchPlaceholder}}\" auto-focus />\n                <button type=\"button\" class=\"btn btn-light ml-1 flex-shrink-0\" (click)=\"ToggleAllItemsSelection(true, true)\" [disabled]=\"disabled\"><i class=\"fa fa-check-square\"></i> {{selectAllButtonLabel}}</button>\n                <button type=\"button\" class=\"btn btn-light ml-1 flex-shrink-0\" (click)=\"ToggleAllItemsSelection(false, true)\" [disabled]=\"disabled\"><i class=\"fa fa-eraser\"></i> {{clearSelectionButtonLabel}}</button>\n            </div>\n\n            <ul>\n                <li class=\"d-flex\" *ngFor=\"let item of options | searchBy : displayTextProperty : search\" [ngClass]=\"{'selected': item.selected}\">\n                    <div class=\"checkbox mr-3 flex-shrink-0\">\n                        <input type=\"checkbox\" name=\"{{identifier}}-option\" [(ngModel)]=\"item.selected\" (change)=\"ItemSelectedCheckChanged()\" [disabled]=\"disabled\" />\n                    </div>\n                    <span class=\"link\" (click)=\"ToggleItemSelected(item)\">{{item[displayTextProperty]}}</span>\n                </li>\n            </ul>\n\n            <small class=\"empty-results\" *ngIf=\"(options | searchBy : displayTextProperty : search).length == 0\">{{emptyMessage}}</small>\n        </div>\n\n        <small class=\"empty-results\" *ngIf=\"!options || options.length == 0\">{{emptyMessage}}</small>\n    </div>\n</div>\n\n<small class=\"text-muted\" *ngIf=\"description\">{{description}}</small>\n"]}