import { Component, ElementRef, EventEmitter, HostListener, Input, OnChanges, OnDestroy, OnInit, Output, SimpleChanges, ViewChild } from "@angular/core"; import {AbstractControl, FormArray, FormControl} from "@angular/forms"; import {HelperFunctions} from "../../utils/HelperFunctions.class"; import {Observable, of, Subscription} from "rxjs"; import {MatSelect} from "@angular/material/select"; import {MatAutocompleteSelectedEvent} from "@angular/material/autocomplete"; import {map, startWith} from "rxjs/operators"; import {MatChipInputEvent} from "@angular/material/chips"; export interface Option { icon?: string, iconClass?: string, value: any, label: string } @Component({ selector: '[dashboard-input]', template: `
{{label + (required ? ' *' : '')}}
{{hint}}
{{placeholder}} {{option.label}}
{{getLabel(chip.value)}}
{{placeholder}}
{{option.label}}
{{formControl.errors.error}} {{warning}} {{note}}
{{label}} `, styleUrls: ['input.component.css'] }) export class InputComponent implements OnInit, OnDestroy, OnChanges { /** Basic information */ @Input('formInput') formControl: AbstractControl; @Input('type') type: 'text' | 'textarea' | 'select' | 'checkbox' | 'chips' = 'text'; @Input('label') label: string; @Input('rows') rows: number = 3; /** Select | chips available options */ @Input('options') options: Option[]; @Input('hint') hint = null; @Input('placeholder') placeholder = ''; @Input() inputClass: string = 'input-box'; /** Extra element Right or Left of the input */ @Input() extraLeft: boolean = true; @Input() gridSmall: boolean = false; @Input() hideControl: boolean = false; @Input() flex: 'middle' | 'top' | 'bottom' = 'middle'; /** Icon Right or Left on the input */ @Input() icon: string = null; @Input() iconLeft: boolean = false; /** Custom messages */ @Input() warning: string = null; @Input() note: string = null; /** Chip options */ @Input() removable: boolean = true; @Input() addExtraChips: boolean = false; @Input() smallChip: boolean = false; @Input() panelWidth: number = 300; @Input() panelClass: string = null; @Input() showOptionsOnEmpty: boolean = true; @Output() focusEmitter: EventEmitter = new EventEmitter(); /** Internal basic information */ public required: boolean = false; private initValue: any; /** Chips */ public filteredOptions: Observable; public searchControl: FormControl; private subscriptions: any[] = []; @ViewChild('select') select: MatSelect; @ViewChild('searchInput') searchInput: ElementRef; focused: boolean = false; constructor(private elementRef: ElementRef) { } @HostListener('document:click', ['$event']) clickOut(event) { this.focused = !!this.elementRef.nativeElement.contains(event.target); this.focusEmitter.emit(this.focused); } ngOnInit(): void { this.reset(); } ngOnChanges(changes: SimpleChanges) { if (changes.formControl) { this.reset(); } } get formAsArray(): FormArray { return (this.formControl); } reset() { this.unsubscribe(); this.initValue = HelperFunctions.copy(this.formControl.value); if (this.options && this.type === 'chips') { this.filteredOptions = of(this.options); this.searchControl = new FormControl(''); this.filteredOptions = this.searchControl.valueChanges.pipe(startWith(''), map(option => this.filter(option))); } if (this.formControl && this.formControl.validator) { let validator = this.formControl.validator({} as AbstractControl); this.required = (validator && validator.required); } this.subscriptions.push(this.formControl.valueChanges.subscribe(value => { value = (value === '') ? null : value; if (this.initValue === value || (this.initValue === '' && value === null)) { this.formControl.markAsPristine(); } })); if (!this.formControl.value) { this.formControl.setValue(''); } } unsubscribe() { this.subscriptions.forEach(subscription => { if (subscription instanceof Subscription) { subscription.unsubscribe(); } }); } openSelect() { if (this.select) { this.select.open(); } } ngOnDestroy(): void { this.unsubscribe(); } stopPropagation() { event.stopPropagation(); } removed(index: number) { this.formAsArray.removeAt(index); this.formAsArray.markAsDirty(); this.searchControl.setValue(''); this.searchInput.nativeElement.focus(); this.searchInput.nativeElement.value = ''; this.stopPropagation(); } selected(event: MatAutocompleteSelectedEvent): void { this.formAsArray.push(new FormControl(event.option.value)); this.formAsArray.markAsDirty(); this.searchControl.setValue(''); this.searchInput.nativeElement.focus(); this.searchInput.nativeElement.value = ''; this.stopPropagation(); } private filter(value: string): Option[] { let options = this.options.filter(option => !this.formAsArray.value.find(value => option.value === value)); if ((!value || value.length == 0) && !this.showOptionsOnEmpty) { return []; } const filterValue = value.toString().toLowerCase(); return options.filter(option => option.label.toLowerCase().indexOf(filterValue) != -1); } add(event: MatChipInputEvent) { if (this.addExtraChips && event.value) { this.stopPropagation(); this.formAsArray.push(new FormControl(event.value)); this.formAsArray.markAsDirty(); this.searchControl.setValue(''); this.searchInput.nativeElement.value = ''; } } getLabel(value: any) { let option = this.options.find(option => option.value === value); return (option) ? option.label : value; } }