UNPKG

fundamental-ngx

Version:

SAP Fiori Fundamentals, implemented in Angular

388 lines 28 kB
/** * @fileoverview added by tsickle * @suppress {checkTypes,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc */ import { Component, ElementRef, EventEmitter, forwardRef, HostBinding, HostListener, Input, Output, ViewChild, ViewEncapsulation } from '@angular/core'; import { NG_VALUE_ACCESSOR } from '@angular/forms'; import { PopoverComponent } from '../popover/popover.component'; /** * Input field with multiple selection enabled. Should be used when a user can select between a * limited number of pre-defined options with a filter-enabled context. * * Supports Angular Forms. */ export class MultiInputComponent { /** * @hidden * @param {?} elRef */ constructor(elRef) { this.elRef = elRef; /** * @hidden */ this.multiInputClass = true; /** * Placeholder for the input field. */ this.placeholder = ''; /** * Whether the input is disabled. */ this.disabled = false; /** * Whether the input is in compact mode. */ this.compact = false; /** * Max height of the popover. Any overflowing elements will be accessible through scrolling. */ this.maxHeight = '300px'; /** * Icon of the button on the right of the input field. */ this.glyph = 'navigation-down-arrow'; /** * Values to be displayed in the unfiltered dropdown. */ this.dropdownValues = []; /** * Whether the search term should be highlighted in results. */ this.highlight = true; /** * Selected dropdown items. */ this.selected = []; /** * Filter function. Accepts an array and a string as arguments, and outputs an array. * An arrow function can be used to access the *this* keyword in the calling component. * See multi input examples for details. */ this.filterFn = this.defaultFilter; /** * Display function. Accepts an object of the same type as the * items passed to dropdownValues as argument, and outputs a string. * An arrow function can be used to access the *this* keyword in the calling component. * See multi input examples for details. */ this.displayFn = this.defaultDisplay; /** * Aria label for the multi input body. */ this.multiInputBodyLabel = 'Multi input body'; /** * Event emitted when the search term changes. Use *$event* to access the new term. */ this.searchTermChange = new EventEmitter(); /** * Event emitted when the selected items change. Use *$event* to access the new selected array. */ this.selectedChange = new EventEmitter(); /** * @hidden */ this.displayedValues = []; /** * @hidden */ this.isOpen = false; /** * @hidden */ this.onChange = (/** * @return {?} */ () => { }); /** * @hidden */ this.onTouched = (/** * @return {?} */ () => { }); } /** * @hidden * @return {?} */ ngOnInit() { if (this.dropdownValues) { this.displayedValues = this.dropdownValues; } } /** * @hidden * @param {?} changes * @return {?} */ ngOnChanges(changes) { if (this.dropdownValues && (changes.dropdownValues || changes.searchTerm)) { if (this.searchTerm) { this.displayedValues = this.filterFn(this.dropdownValues, this.searchTerm); } else { this.displayedValues = this.dropdownValues; } } } /** * @hidden * @param {?} fn * @return {?} */ registerOnChange(fn) { this.onChange = fn; } /** * @hidden * @param {?} fn * @return {?} */ registerOnTouched(fn) { this.onTouched = fn; } /** * @hidden * @param {?} isDisabled * @return {?} */ setDisabledState(isDisabled) { this.disabled = isDisabled; } /** * @hidden * @param {?} selected * @return {?} */ writeValue(selected) { this.selected = selected; } /** * @hidden * @param {?} checked * @param {?} value * @return {?} */ handleSelect(checked, value) { /** @type {?} */ const previousLength = this.selected.length; if (checked) { this.selected.push(value); } else { this.selected.splice(this.selected.indexOf(value), 1); } // Handle popover placement update if ((previousLength === 0 && this.selected.length === 1) || (previousLength === 1 && this.selected.length === 0)) { this.popoverRef.updatePopover(); } this.onChange(this.selected); this.selectedChange.emit(this.selected); } /** * @hidden * @return {?} */ handleSearchTermChange() { this.searchTermChange.emit(this.searchTerm); this.displayedValues = this.filterFn(this.dropdownValues, this.searchTerm); } /** * @private * @param {?} contentArray * @param {?} searchTerm * @return {?} */ defaultFilter(contentArray, searchTerm) { /** @type {?} */ const searchLower = searchTerm.toLocaleLowerCase(); return contentArray.filter((/** * @param {?} item * @return {?} */ item => { if (item) { return this.displayFn(item).toLocaleLowerCase().includes(searchLower); } })); } /** * @private * @param {?} str * @return {?} */ defaultDisplay(str) { return str; } /** * @hidden * @param {?} event * @return {?} */ clickHandler(event) { event.stopPropagation(); if (!this.elRef.nativeElement.contains(event.target)) { this.isOpen = false; } } } MultiInputComponent.decorators = [ { type: Component, args: [{ selector: 'fd-multi-input', template: "<div class=\"fd-multi-input-field\">\n <fd-popover [(isOpen)]=\"isOpen\"\n [triggers]=\"[]\"\n [disabled]=\"disabled\"\n [fillControl]=\"true\"\n class=\"fd-multi-input-popover-custom\">\n <fd-popover-control>\n <div class=\"fd-combobox-control\"\n [attr.aria-label]=\"multiInputBodyLabel\"\n [attr.aria-expanded]=\"isOpen\">\n <div class=\"fd-input-group fd-input-group--after\" [ngClass]=\"{'fd-input-group--compact': compact}\">\n <input type=\"text\" class=\"fd-input\"\n [ngClass]=\"{'fd-input--compact': compact}\"\n [placeholder]=\"placeholder\"\n [disabled]=\"disabled\"\n [(ngModel)]=\"searchTerm\"\n (ngModelChange)=\"handleSearchTermChange()\"\n (keypress)=\"isOpen = true\"\n (click)=\"isOpen = !isOpen\">\n <span class=\"fd-input-group__addon fd-input-group__addon--after\n fd-input-group__addon--button\">\n <button class=\"fd-button--light\" type=\"button\"\n [ngClass]=\"('sap-icon--' + glyph)\"\n [disabled]=\"disabled\"\n (click)=\"isOpen = !isOpen\"></button>\n </span>\n </div>\n </div>\n </fd-popover-control>\n <fd-popover-body [attr.aria-hidden]=\"!isOpen\">\n <fd-menu class=\"fd-multi-input-menu-overflow\"\n *ngIf=\"displayedValues && displayedValues.length\"\n [style.maxHeight]=\"maxHeight\">\n <ul fd-menu-list>\n <li fd-menu-item *ngFor=\"let value of displayedValues\">\n <label class=\"fd-menu__item\">\n <input type=\"checkbox\" class=\"fd-checkbox\"\n [ngModel]=\"selected ? selected.indexOf(value) !== -1 : false\"\n (ngModelChange)=\"handleSelect($event, value)\">\n <span [innerHtml]=\"value | displayFnPipe:displayFn | highlight:searchTerm:highlight\"></span>\n </label>\n </li>\n </ul>\n </fd-menu>\n <ng-content></ng-content>\n </fd-popover-body>\n </fd-popover>\n</div>\n<div class=\"fd-multi-input-tags\">\n <fd-token *ngFor=\"let token of selected\"\n (onCloseClick)=\"handleSelect(false, token)\"\n class=\"fd-multi-input-token-spacing\">\n {{token | displayFnPipe:displayFn}}\n </fd-token>\n</div>\n\n", host: { '(blur)': 'onTouched()', '[class.fd-multi-input-custom]': 'true' }, providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef((/** * @return {?} */ () => MultiInputComponent)), multi: true } ], encapsulation: ViewEncapsulation.None, styles: [".fd-multi-input-custom{display:block}.fd-multi-input-custom .fd-multi-input-popover-size{overflow:auto;display:block}.fd-multi-input-custom .fd-multi-input-popover-custom{display:block}.fd-multi-input-custom .fd-multi-input-menu-overflow{overflow:auto}.fd-multi-input-custom .fd-multi-input-token-spacing{margin:0 4px 4px 0}.fd-multi-input-custom .fd-multi-input-token-spacing:last-child{margin-right:0}"] }] } ]; /** @nocollapse */ MultiInputComponent.ctorParameters = () => [ { type: ElementRef } ]; MultiInputComponent.propDecorators = { popoverRef: [{ type: ViewChild, args: [PopoverComponent,] }], multiInputClass: [{ type: HostBinding, args: ['class.fd-multi-input',] }], placeholder: [{ type: Input }], disabled: [{ type: Input }], compact: [{ type: Input }], maxHeight: [{ type: Input }], glyph: [{ type: Input }], dropdownValues: [{ type: Input }], searchTerm: [{ type: Input }], highlight: [{ type: Input }], selected: [{ type: Input }], filterFn: [{ type: Input }], displayFn: [{ type: Input }], multiInputBodyLabel: [{ type: Input }], searchTermChange: [{ type: Output }], selectedChange: [{ type: Output }], clickHandler: [{ type: HostListener, args: ['document:click', ['$event'],] }] }; if (false) { /** * @hidden * @type {?} */ MultiInputComponent.prototype.popoverRef; /** * @hidden * @type {?} */ MultiInputComponent.prototype.multiInputClass; /** * Placeholder for the input field. * @type {?} */ MultiInputComponent.prototype.placeholder; /** * Whether the input is disabled. * @type {?} */ MultiInputComponent.prototype.disabled; /** * Whether the input is in compact mode. * @type {?} */ MultiInputComponent.prototype.compact; /** * Max height of the popover. Any overflowing elements will be accessible through scrolling. * @type {?} */ MultiInputComponent.prototype.maxHeight; /** * Icon of the button on the right of the input field. * @type {?} */ MultiInputComponent.prototype.glyph; /** * Values to be displayed in the unfiltered dropdown. * @type {?} */ MultiInputComponent.prototype.dropdownValues; /** * Search term, or more specifically the value of the inner input field. * @type {?} */ MultiInputComponent.prototype.searchTerm; /** * Whether the search term should be highlighted in results. * @type {?} */ MultiInputComponent.prototype.highlight; /** * Selected dropdown items. * @type {?} */ MultiInputComponent.prototype.selected; /** * Filter function. Accepts an array and a string as arguments, and outputs an array. * An arrow function can be used to access the *this* keyword in the calling component. * See multi input examples for details. * @type {?} */ MultiInputComponent.prototype.filterFn; /** * Display function. Accepts an object of the same type as the * items passed to dropdownValues as argument, and outputs a string. * An arrow function can be used to access the *this* keyword in the calling component. * See multi input examples for details. * @type {?} */ MultiInputComponent.prototype.displayFn; /** * Aria label for the multi input body. * @type {?} */ MultiInputComponent.prototype.multiInputBodyLabel; /** * Event emitted when the search term changes. Use *$event* to access the new term. * @type {?} */ MultiInputComponent.prototype.searchTermChange; /** * Event emitted when the selected items change. Use *$event* to access the new selected array. * @type {?} */ MultiInputComponent.prototype.selectedChange; /** * @hidden * @type {?} */ MultiInputComponent.prototype.displayedValues; /** * @hidden * @type {?} */ MultiInputComponent.prototype.isOpen; /** * @hidden * @type {?} */ MultiInputComponent.prototype.onChange; /** * @hidden * @type {?} */ MultiInputComponent.prototype.onTouched; /** * @type {?} * @private */ MultiInputComponent.prototype.elRef; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"multi-input.component.js","sourceRoot":"ng://fundamental-ngx/","sources":["lib/multi-input/multi-input.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    Component,\n    ElementRef,\n    EventEmitter,\n    forwardRef,\n    HostBinding,\n    HostListener,\n    Input,\n    OnChanges,\n    OnInit,\n    Output,\n    SimpleChanges,\n    ViewChild,\n    ViewEncapsulation\n} from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { PopoverComponent } from '../popover/popover.component';\n\n/**\n * Input field with multiple selection enabled. Should be used when a user can select between a\n * limited number of pre-defined options with a filter-enabled context.\n *\n * Supports Angular Forms.\n */\n@Component({\n    selector: 'fd-multi-input',\n    templateUrl: './multi-input.component.html',\n    styleUrls: ['./multi-input.component.scss'],\n    host: {\n        '(blur)': 'onTouched()',\n        '[class.fd-multi-input-custom]': 'true'\n    },\n    providers: [\n        {\n            provide: NG_VALUE_ACCESSOR,\n            useExisting: forwardRef(() => MultiInputComponent),\n            multi: true\n        }\n    ],\n    encapsulation: ViewEncapsulation.None\n})\nexport class MultiInputComponent implements OnInit, ControlValueAccessor, OnChanges {\n\n    /** @hidden */\n    @ViewChild(PopoverComponent)\n    popoverRef: PopoverComponent;\n\n    /** @hidden */\n    @HostBinding('class.fd-multi-input')\n    multiInputClass = true;\n\n    /** Placeholder for the input field. */\n    @Input()\n    placeholder: string = '';\n\n    /** Whether the input is disabled. */\n    @Input()\n    disabled: boolean = false;\n\n    /** Whether the input is in compact mode. */\n    @Input()\n    compact: boolean = false;\n\n    /** Max height of the popover. Any overflowing elements will be accessible through scrolling. */\n    @Input()\n    maxHeight: string = '300px';\n\n    /** Icon of the button on the right of the input field. */\n    @Input()\n    glyph: string = 'navigation-down-arrow';\n\n    /** Values to be displayed in the unfiltered dropdown. */\n    @Input()\n    dropdownValues: any[] = [];\n\n    /** Search term, or more specifically the value of the inner input field. */\n    @Input()\n    searchTerm: string;\n\n    /** Whether the search term should be highlighted in results. */\n    @Input()\n    highlight: boolean = true;\n\n    /** Selected dropdown items. */\n    @Input()\n    selected: any[] = [];\n\n    /** Filter function. Accepts an array and a string as arguments, and outputs an array.\n     * An arrow function can be used to access the *this* keyword in the calling component.\n     * See multi input examples for details. */\n    @Input()\n    filterFn: Function = this.defaultFilter;\n\n    /** Display function. Accepts an object of the same type as the\n     * items passed to dropdownValues as argument, and outputs a string.\n     * An arrow function can be used to access the *this* keyword in the calling component.\n     * See multi input examples for details. */\n    @Input()\n    displayFn: Function = this.defaultDisplay;\n\n    /** Aria label for the multi input body. */\n    @Input()\n    multiInputBodyLabel: string = 'Multi input body';\n\n    /** Event emitted when the search term changes. Use *$event* to access the new term. */\n    @Output()\n    readonly searchTermChange: EventEmitter<string> = new EventEmitter<string>();\n\n    /** Event emitted when the selected items change. Use *$event* to access the new selected array. */\n    @Output()\n    readonly selectedChange: EventEmitter<any[]> = new EventEmitter<any[]>();\n\n    /** @hidden */\n    displayedValues: any[] = [];\n\n    /** @hidden */\n    isOpen = false;\n\n    /** @hidden */\n    onChange: Function = () => {};\n\n    /** @hidden */\n    onTouched: Function = () => {};\n\n    /** @hidden */\n    constructor(private elRef: ElementRef) {}\n\n    /** @hidden */\n    ngOnInit() {\n        if (this.dropdownValues) {\n            this.displayedValues = this.dropdownValues;\n        }\n    }\n\n    /** @hidden */\n    ngOnChanges(changes: SimpleChanges) {\n        if (this.dropdownValues && (changes.dropdownValues || changes.searchTerm)) {\n            if (this.searchTerm) {\n                this.displayedValues = this.filterFn(this.dropdownValues, this.searchTerm);\n            } else {\n                this.displayedValues =  this.dropdownValues;\n            }\n        }\n    }\n\n    /** @hidden */\n    registerOnChange(fn: any): void {\n        this.onChange = fn;\n    }\n\n    /** @hidden */\n    registerOnTouched(fn: any): void {\n        this.onTouched = fn;\n    }\n\n    /** @hidden */\n    setDisabledState(isDisabled: boolean): void {\n        this.disabled = isDisabled;\n    }\n\n    /** @hidden */\n    writeValue(selected: any[]): void {\n        this.selected = selected;\n    }\n\n    /** @hidden */\n    handleSelect(checked: any, value: any): void {\n        const previousLength = this.selected.length;\n        if (checked) {\n            this.selected.push(value);\n        } else {\n            this.selected.splice(this.selected.indexOf(value), 1);\n        }\n\n        // Handle popover placement update\n        if ((previousLength === 0 && this.selected.length === 1) ||\n            (previousLength === 1 && this.selected.length === 0)) {\n            this.popoverRef.updatePopover();\n        }\n\n        this.onChange(this.selected);\n        this.selectedChange.emit(this.selected);\n    }\n\n    /** @hidden */\n    handleSearchTermChange(): void {\n        this.searchTermChange.emit(this.searchTerm);\n        this.displayedValues = this.filterFn(this.dropdownValues, this.searchTerm);\n    }\n\n    private defaultFilter(contentArray: any[], searchTerm: string): any[] {\n        const searchLower = searchTerm.toLocaleLowerCase();\n        return contentArray.filter(item => {\n            if (item) {\n                return this.displayFn(item).toLocaleLowerCase().includes(searchLower);\n            }\n        });\n    }\n\n    private defaultDisplay(str: string): string {\n        return str;\n    }\n\n    /** @hidden */\n    @HostListener('document:click', ['$event'])\n    clickHandler(event) {\n        event.stopPropagation();\n        if (!this.elRef.nativeElement.contains(event.target)) {\n            this.isOpen = false;\n        }\n    }\n\n}\n"]}