@clr/angular
Version:
Angular components for Clarity
156 lines (152 loc) • 18.2 kB
JavaScript
/*
* 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 { DOCUMENT } from '@angular/common';
import { Component, ContentChildren, Inject, Input, Optional, } from '@angular/core';
import { fromEvent } from 'rxjs';
import { POPOVER_HOST_ANCHOR } from '../../popover/common/popover-host-anchor.token';
import { IF_ACTIVE_ID } from '../../utils/conditional/if-active.service';
import { ClrLoadingState } from '../../utils/loading/loading';
import { LoadingListener } from '../../utils/loading/loading-listener';
import { ClrOption } from './option';
import * as i0 from "@angular/core";
import * as i1 from "./providers/option-selection.service";
import * as i2 from "../../utils/i18n/common-strings.service";
import * as i3 from "./providers/combobox-focus-handler.service";
import * as i4 from "../../utils/popover/providers/popover-toggle.service";
import * as i5 from "@angular/common";
import * as i6 from "../../progress/spinner/spinner";
let nbOptionsComponents = 0;
export class ClrOptions {
constructor(optionSelectionService, id, el, commonStrings, focusHandler, toggleService, parentHost, document) {
this.optionSelectionService = optionSelectionService;
this.id = id;
this.el = el;
this.commonStrings = commonStrings;
this.focusHandler = focusHandler;
this.toggleService = toggleService;
this.document = document;
this.loading = false;
this.subscriptions = [];
if (!parentHost) {
throw new Error('clr-options should only be used inside of a clr-combobox');
}
if (!this.optionsId) {
this.optionsId = 'clr-options-' + nbOptionsComponents++;
}
}
get items() {
return this._items;
}
set items(items) {
this._items = items;
this.focusHandler.addOptionValues(this._items.map(option => option.optionProxy));
}
/**
* Tests if the list of options is empty, meaning it doesn't contain any items
*/
get emptyOptions() {
return !this.optionSelectionService.loading && this.items.length === 0;
}
get noResultsElementId() {
return `${this.optionsId}-no-results`;
}
ngAfterViewInit() {
this.focusHandler.listbox = this.el.nativeElement;
this.subscriptions.push(fromEvent(this.document, 'scroll', { capture: true }).subscribe(event => {
if (this.toggleService.open &&
event.target !== this.el.nativeElement &&
event.target !== this.focusHandler.textInput) {
this.toggleService.open = false;
}
}), this.items.changes.subscribe(items => {
if (items.length) {
setTimeout(() => {
this.focusHandler.focusFirstActive();
});
}
else {
this.focusHandler.pseudoFocus.pop();
}
}));
}
ngOnDestroy() {
this.subscriptions.forEach(sub => sub.unsubscribe());
}
searchText(input) {
return this.commonStrings.parse(this.commonStrings.keys.comboboxSearching, { INPUT: input });
}
loadingStateChange(state) {
this.loading = state === ClrLoadingState.LOADING;
}
}
ClrOptions.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: ClrOptions, deps: [{ token: i1.OptionSelectionService }, { token: IF_ACTIVE_ID }, { token: i0.ElementRef }, { token: i2.ClrCommonStringsService }, { token: i3.ComboboxFocusHandler }, { token: i4.ClrPopoverToggleService }, { token: POPOVER_HOST_ANCHOR, optional: true }, { token: DOCUMENT }], target: i0.ɵɵFactoryTarget.Component });
ClrOptions.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.2", type: ClrOptions, selector: "clr-options", inputs: { optionsId: ["id", "optionsId"] }, host: { properties: { "class.clr-combobox-options": "true", "attr.role": "\"listbox\"", "id": "optionsId" } }, providers: [{ provide: LoadingListener, useExisting: ClrOptions }], queries: [{ propertyName: "items", predicate: ClrOption, descendants: true }], ngImport: i0, template: `
<div *ngIf="optionSelectionService.loading" class="clr-combobox-options-loading">
<clr-spinner clrInline>
{{ commonStrings.keys.loading }}
</clr-spinner>
<span class="clr-combobox-options-text">
{{ searchText(optionSelectionService.currentInput) }}
</span>
</div>
<!-- Rendered if data set is empty -->
<div *ngIf="emptyOptions" [id]="noResultsElementId" role="option">
<span class="clr-combobox-options-empty-text">
{{ commonStrings.keys.comboboxNoResults }}
</span>
</div>
<!--Option Groups and Options will be projected here-->
<ng-content></ng-content>
`, isInline: true, dependencies: [{ kind: "directive", type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i6.ClrSpinner, selector: "clr-spinner", inputs: ["clrInline", "clrInverse", "clrSmall", "clrMedium"] }] });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: ClrOptions, decorators: [{
type: Component,
args: [{
selector: 'clr-options',
template: `
<div *ngIf="optionSelectionService.loading" class="clr-combobox-options-loading">
<clr-spinner clrInline>
{{ commonStrings.keys.loading }}
</clr-spinner>
<span class="clr-combobox-options-text">
{{ searchText(optionSelectionService.currentInput) }}
</span>
</div>
<!-- Rendered if data set is empty -->
<div *ngIf="emptyOptions" [id]="noResultsElementId" role="option">
<span class="clr-combobox-options-empty-text">
{{ commonStrings.keys.comboboxNoResults }}
</span>
</div>
<!--Option Groups and Options will be projected here-->
<ng-content></ng-content>
`,
providers: [{ provide: LoadingListener, useExisting: ClrOptions }],
host: {
'[class.clr-combobox-options]': 'true',
'[attr.role]': '"listbox"',
'[id]': 'optionsId',
},
}]
}], ctorParameters: function () { return [{ type: i1.OptionSelectionService }, { type: undefined, decorators: [{
type: Inject,
args: [IF_ACTIVE_ID]
}] }, { type: i0.ElementRef }, { type: i2.ClrCommonStringsService }, { type: i3.ComboboxFocusHandler }, { type: i4.ClrPopoverToggleService }, { type: i0.ElementRef, decorators: [{
type: Optional
}, {
type: Inject,
args: [POPOVER_HOST_ANCHOR]
}] }, { type: undefined, decorators: [{
type: Inject,
args: [DOCUMENT]
}] }]; }, propDecorators: { optionsId: [{
type: Input,
args: ['id']
}], items: [{
type: ContentChildren,
args: [ClrOption, { descendants: true }]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"options.js","sourceRoot":"","sources":["../../../../../projects/angular/src/forms/combobox/options.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 { DOCUMENT } from '@angular/common';\nimport {\n  AfterViewInit,\n  Component,\n  ContentChildren,\n  ElementRef,\n  Inject,\n  Input,\n  OnDestroy,\n  Optional,\n  QueryList,\n} from '@angular/core';\nimport { fromEvent, Subscription } from 'rxjs';\n\nimport { POPOVER_HOST_ANCHOR } from '../../popover/common/popover-host-anchor.token';\nimport { IF_ACTIVE_ID } from '../../utils/conditional/if-active.service';\nimport { ClrCommonStringsService } from '../../utils/i18n/common-strings.service';\nimport { ClrLoadingState } from '../../utils/loading/loading';\nimport { LoadingListener } from '../../utils/loading/loading-listener';\nimport { ClrPopoverToggleService } from '../../utils/popover/providers/popover-toggle.service';\nimport { ClrOption } from './option';\nimport { ComboboxFocusHandler } from './providers/combobox-focus-handler.service';\nimport { OptionSelectionService } from './providers/option-selection.service';\n\nlet nbOptionsComponents = 0;\n\n@Component({\n  selector: 'clr-options',\n  template: `\n    <div *ngIf=\"optionSelectionService.loading\" class=\"clr-combobox-options-loading\">\n      <clr-spinner clrInline>\n        {{ commonStrings.keys.loading }}\n      </clr-spinner>\n      <span class=\"clr-combobox-options-text\">\n        {{ searchText(optionSelectionService.currentInput) }}\n      </span>\n    </div>\n\n    <!-- Rendered if data set is empty -->\n    <div *ngIf=\"emptyOptions\" [id]=\"noResultsElementId\" role=\"option\">\n      <span class=\"clr-combobox-options-empty-text\">\n        {{ commonStrings.keys.comboboxNoResults }}\n      </span>\n    </div>\n\n    <!--Option Groups and Options will be projected here-->\n    <ng-content></ng-content>\n  `,\n  providers: [{ provide: LoadingListener, useExisting: ClrOptions }],\n  host: {\n    '[class.clr-combobox-options]': 'true',\n    '[attr.role]': '\"listbox\"',\n    '[id]': 'optionsId',\n  },\n})\nexport class ClrOptions<T> implements AfterViewInit, LoadingListener, OnDestroy {\n  @Input('id') optionsId: string;\n\n  loading = false;\n  _items: QueryList<ClrOption<T>>;\n\n  private subscriptions: Subscription[] = [];\n\n  constructor(\n    public optionSelectionService: OptionSelectionService<T>,\n    @Inject(IF_ACTIVE_ID) public id: number,\n    public el: ElementRef<HTMLElement>,\n    public commonStrings: ClrCommonStringsService,\n    private focusHandler: ComboboxFocusHandler<T>,\n    private toggleService: ClrPopoverToggleService,\n    @Optional()\n    @Inject(POPOVER_HOST_ANCHOR)\n    parentHost: ElementRef<HTMLElement>,\n    @Inject(DOCUMENT) private document: any\n  ) {\n    if (!parentHost) {\n      throw new Error('clr-options should only be used inside of a clr-combobox');\n    }\n\n    if (!this.optionsId) {\n      this.optionsId = 'clr-options-' + nbOptionsComponents++;\n    }\n  }\n\n  @ContentChildren(ClrOption, { descendants: true })\n  get items(): QueryList<ClrOption<T>> {\n    return this._items;\n  }\n  set items(items: QueryList<ClrOption<T>>) {\n    this._items = items;\n    this.focusHandler.addOptionValues(this._items.map(option => option.optionProxy));\n  }\n\n  /**\n   * Tests if the list of options is empty, meaning it doesn't contain any items\n   */\n  get emptyOptions() {\n    return !this.optionSelectionService.loading && this.items.length === 0;\n  }\n\n  get noResultsElementId() {\n    return `${this.optionsId}-no-results`;\n  }\n\n  ngAfterViewInit() {\n    this.focusHandler.listbox = this.el.nativeElement;\n\n    this.subscriptions.push(\n      fromEvent(this.document, 'scroll', { capture: true }).subscribe(event => {\n        if (\n          this.toggleService.open &&\n          (event as Event).target !== this.el.nativeElement &&\n          (event as Event).target !== this.focusHandler.textInput\n        ) {\n          this.toggleService.open = false;\n        }\n      }),\n      this.items.changes.subscribe(items => {\n        if (items.length) {\n          setTimeout(() => {\n            this.focusHandler.focusFirstActive();\n          });\n        } else {\n          this.focusHandler.pseudoFocus.pop();\n        }\n      })\n    );\n  }\n\n  ngOnDestroy() {\n    this.subscriptions.forEach(sub => sub.unsubscribe());\n  }\n\n  searchText(input: string) {\n    return this.commonStrings.parse(this.commonStrings.keys.comboboxSearching, { INPUT: input });\n  }\n\n  loadingStateChange(state: ClrLoadingState): void {\n    this.loading = state === ClrLoadingState.LOADING;\n  }\n}\n"]}