UNPKG

@clr/angular

Version:

Angular components for Clarity

110 lines 14.7 kB
/* * Copyright (c) 2016-2025 Broadcom. All Rights Reserved. * The term "Broadcom" refers to Broadcom Inc. and/or its subsidiaries. * This software is released under MIT license. * The full license information can be found in LICENSE in the root directory of this project. */ import { NgForOf } from '@angular/common'; import { Directive, Input, } from '@angular/core'; import * as i0 from "@angular/core"; import * as i1 from "./providers/option-selection.service"; import * as i2 from "../../utils/popover/providers/popover-position.service"; export class ClrOptionItems { constructor(template, differs, optionService, positionService, vcr) { this.template = template; this.differs = differs; this.optionService = optionService; this.positionService = positionService; this.subscriptions = []; this.filter = ''; this.differ = null; this.iterableProxy = new NgForOf(vcr, template, differs); this.subscriptions.push(optionService.inputChanged.subscribe(filter => { this.filter = filter; this.updateItems(); })); } set rawItems(items) { this._rawItems = items ? items : []; this.updateItems(); } set trackBy(value) { this.iterableProxy.ngForTrackBy = value; } set field(field) { this._filterField = field; this.optionService.displayField = field; } get hasResults() { // explicity return `undefined` instead of `false` if the answer is not known return this.filteredItems ? this.filteredItems.length : undefined; } ngDoCheck() { if (!this.differ) { this.differ = this.differs.find(this.filteredItems).create(this.iterableProxy.ngForTrackBy); } if (this.differ) { const changes = this.differ.diff(this.filteredItems); if (changes) { this.iterableProxy.ngDoCheck(); this.positionService.realign(); } } } ngOnDestroy() { this.subscriptions.forEach(sub => sub.unsubscribe()); } updateItems() { if (!this._rawItems || this.filter === undefined || this.filter === null) { return; } const normalizedFilterValue = normalizeValue(this.filter); if (this.optionService.showAllOptions) { this.filteredItems = this._rawItems; } else if (this._filterField) { this.filteredItems = this._rawItems.filter(item => { const objValue = item[this._filterField]; return objValue ? normalizeValue(objValue).includes(normalizedFilterValue) : false; }); } else { // Filter by all item object values this.filteredItems = this._rawItems.filter(item => { if (typeof item !== 'object') { return normalizeValue(item).includes(normalizedFilterValue); } const objValues = Object.values(item).filter(value => { return value !== null && value !== undefined ? normalizeValue(value).includes(normalizedFilterValue) : false; }); return objValues.length > 0; }); } this.iterableProxy.ngForOf = this.filteredItems; } } ClrOptionItems.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: ClrOptionItems, deps: [{ token: i0.TemplateRef }, { token: i0.IterableDiffers }, { token: i1.OptionSelectionService }, { token: i2.ClrPopoverPositionService }, { token: i0.ViewContainerRef }], target: i0.ɵɵFactoryTarget.Directive }); ClrOptionItems.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "15.2.2", type: ClrOptionItems, selector: "[clrOptionItems][clrOptionItemsOf]", inputs: { rawItems: ["clrOptionItemsOf", "rawItems"], trackBy: ["clrOptionItemsTrackBy", "trackBy"], field: ["clrOptionItemsField", "field"] }, ngImport: i0 }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: ClrOptionItems, decorators: [{ type: Directive, args: [{ selector: '[clrOptionItems][clrOptionItemsOf]', }] }], ctorParameters: function () { return [{ type: i0.TemplateRef }, { type: i0.IterableDiffers }, { type: i1.OptionSelectionService }, { type: i2.ClrPopoverPositionService }, { type: i0.ViewContainerRef }]; }, propDecorators: { rawItems: [{ type: Input, args: ['clrOptionItemsOf'] }], trackBy: [{ type: Input, args: ['clrOptionItemsTrackBy'] }], field: [{ type: Input, args: ['clrOptionItemsField'] }] } }); function normalizeValue(value) { return value .toString() .normalize('NFD') .replace(/\p{Diacritic}/gu, '') .toLowerCase(); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"option-items.directive.js","sourceRoot":"","sources":["../../../../../projects/angular/src/forms/combobox/option-items.directive.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2016-2025 Broadcom. All Rights Reserved.\n * The term \"Broadcom\" refers to Broadcom Inc. and/or its subsidiaries.\n * This software is released under MIT license.\n * The full license information can be found in LICENSE in the root directory of this project.\n */\n\nimport { NgForOf, NgForOfContext } from '@angular/common';\nimport {\n  Directive,\n  DoCheck,\n  Input,\n  IterableDiffer,\n  IterableDiffers,\n  OnDestroy,\n  TemplateRef,\n  TrackByFunction,\n  ViewContainerRef,\n} from '@angular/core';\nimport { Subscription } from 'rxjs';\n\nimport { ClrPopoverPositionService } from '../../utils/popover/providers/popover-position.service';\nimport { OptionSelectionService } from './providers/option-selection.service';\n\n@Directive({\n  selector: '[clrOptionItems][clrOptionItemsOf]',\n})\nexport class ClrOptionItems<T> implements DoCheck, OnDestroy {\n  private iterableProxy: NgForOf<T>;\n  private _rawItems: T[];\n  private filteredItems: T[];\n  private subscriptions: Subscription[] = [];\n  private filter = '';\n  private _filterField: string;\n  private differ: IterableDiffer<T> | null = null;\n\n  constructor(\n    public template: TemplateRef<NgForOfContext<T>>,\n    private differs: IterableDiffers,\n    private optionService: OptionSelectionService<T>,\n    private positionService: ClrPopoverPositionService,\n    vcr: ViewContainerRef\n  ) {\n    this.iterableProxy = new NgForOf<T>(vcr, template, differs);\n    this.subscriptions.push(\n      optionService.inputChanged.subscribe(filter => {\n        this.filter = filter;\n        this.updateItems();\n      })\n    );\n  }\n\n  @Input('clrOptionItemsOf')\n  set rawItems(items: T[]) {\n    this._rawItems = items ? items : [];\n    this.updateItems();\n  }\n\n  @Input('clrOptionItemsTrackBy')\n  set trackBy(value: TrackByFunction<T>) {\n    this.iterableProxy.ngForTrackBy = value;\n  }\n\n  @Input('clrOptionItemsField')\n  set field(field: string) {\n    this._filterField = field;\n    this.optionService.displayField = field;\n  }\n\n  get hasResults() {\n    // explicity return `undefined` instead of `false` if the answer is not known\n    return this.filteredItems ? this.filteredItems.length : undefined;\n  }\n\n  ngDoCheck() {\n    if (!this.differ) {\n      this.differ = this.differs.find(this.filteredItems).create(this.iterableProxy.ngForTrackBy);\n    }\n    if (this.differ) {\n      const changes = this.differ.diff(this.filteredItems);\n      if (changes) {\n        this.iterableProxy.ngDoCheck();\n        this.positionService.realign();\n      }\n    }\n  }\n\n  ngOnDestroy() {\n    this.subscriptions.forEach(sub => sub.unsubscribe());\n  }\n  private updateItems() {\n    if (!this._rawItems || this.filter === undefined || this.filter === null) {\n      return;\n    }\n\n    const normalizedFilterValue = normalizeValue(this.filter);\n\n    if (this.optionService.showAllOptions) {\n      this.filteredItems = this._rawItems;\n    } else if (this._filterField) {\n      this.filteredItems = this._rawItems.filter(item => {\n        const objValue = (item as any)[this._filterField];\n        return objValue ? normalizeValue(objValue).includes(normalizedFilterValue) : false;\n      });\n    } else {\n      // Filter by all item object values\n      this.filteredItems = this._rawItems.filter(item => {\n        if (typeof item !== 'object') {\n          return normalizeValue(item).includes(normalizedFilterValue);\n        }\n        const objValues = Object.values(item).filter(value => {\n          return value !== null && value !== undefined ? normalizeValue(value).includes(normalizedFilterValue) : false;\n        });\n        return objValues.length > 0;\n      });\n    }\n    this.iterableProxy.ngForOf = this.filteredItems;\n  }\n}\n\nfunction normalizeValue(value: any) {\n  return value\n    .toString()\n    .normalize('NFD')\n    .replace(/\\p{Diacritic}/gu, '')\n    .toLowerCase();\n}\n"]}