fundamental-ngx
Version:
SAP Fundamentals, implemented in Angular
407 lines • 29.7 kB
JavaScript
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,constantProperty,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';
/**
* Preset options for the popover body width.
* * `at-least` will apply a minimum width to the body equivalent to the width of the control.
* * `equal` will apply a width to the body equivalent to the width of the control.
* * Leave blank for no effect.
*/
this.fillControlMode = 'at-least';
/**
* 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) {
if (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);
this.popoverRef.updatePopover();
}
/**
* @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 [fillControlMode]=\"fillControlMode\"\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 *ngFor=\"let value of displayedValues\">\n <label 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 }],
fillControlMode: [{ 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;
/**
* Preset options for the popover body width.
* * `at-least` will apply a minimum width to the body equivalent to the width of the control.
* * `equal` will apply a width to the body equivalent to the width of the control.
* * Leave blank for no effect.
* @type {?}
*/
MultiInputComponent.prototype.fillControlMode;
/**
* 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';\nimport { PopoverFillMode } from '../popover/popover-directive/popover.directive';\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    /**\n     * Preset options for the popover body width.\n     * * `at-least` will apply a minimum width to the body equivalent to the width of the control.\n     * * `equal` will apply a width to the body equivalent to the width of the control.\n     * * Leave blank for no effect.\n     */\n    @Input()\n    fillControlMode: PopoverFillMode = 'at-least';\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        if (selected) {\n            this.selected = selected;\n        }\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        this.popoverRef.updatePopover();\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"]}