UNPKG

design-angular-kit

Version:

Un toolkit Angular conforme alle linee guida di design per i servizi web della PA

112 lines 23.7 kB
import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output } from '@angular/core'; import { debounceTime, distinctUntilChanged, map, Observable, of, switchMap } from 'rxjs'; import { AsyncPipe, NgTemplateOutlet } from '@angular/common'; import { ReactiveFormsModule } from '@angular/forms'; import { ItIconComponent } from '../../utils/icon/icon.component'; import { ItMarkMatchingTextPipe } from '../../../pipes/mark-matching-text.pipe'; import { ItAbstractFormComponent } from '../../../abstracts/abstract-form.component'; import { inputToBoolean } from '../../../utils/coercion'; import * as i0 from "@angular/core"; import * as i1 from "@angular/forms"; export class ItSearchComponent extends ItAbstractFormComponent { constructor() { super(...arguments); /** * Time span [ms] has passed without another source emission, to delay data filtering. * Useful when the user is typing multiple letters * @default 300 [ms] */ this.debounceTime = 300; /** * The input placeholder */ this.placeholder = ''; /** * The input label even get labelWaria icon */ this.labelWaria = undefined; /** * Show the label */ this.forceShowLabel = true; /** * Fired when the Search Item has been selected */ this.searchSelectedEvent = new EventEmitter(); this.showAutocompletion = false; /** Observable da cui vengono emessi i risultati dell'auto completamento */ this.searchResults$ = new Observable(); } ngOnInit() { super.ngOnInit(); this.searchResults$ = this.getSearchResults$(); } /** * Create the search list */ getSearchResults$() { return this.control.valueChanges.pipe(debounceTime(this.debounceTime), // Delay filter data after time span has passed without another source emission, useful when the user is typing multiple letters distinctUntilChanged(), // Only if searchValue is distinct in comparison to the last value switchMap(searchedValue => { if (!this.searchData) { return of({ searchedValue, relatedEntries: [], }); } const autoCompleteData$ = Array.isArray(this.searchData) ? of(this.searchData) : this.searchData(searchedValue); return autoCompleteData$.pipe(map(searchData => { if (!searchedValue || typeof searchedValue === 'number') { return { searchedValue, relatedEntries: [] }; } const lowercaseValue = searchedValue.toLowerCase(); const relatedEntries = searchData.filter(item => item.value?.toLowerCase().includes(lowercaseValue)); return { searchedValue, relatedEntries }; })); })); } onEntryClick(entry, event) { // Se non è stato definito un link associato all'elemento dell'search, probabilmente il desiderata // non è effettuare la navigazione al default '#', pertanto in tal caso meglio annullare la navigazione. if (!entry.link) { event.preventDefault(); } this.searchSelectedEvent.next(entry); this.control.setValue(entry.value); this.showAutocompletion = false; } searchItemTrackByValueFn(index, item) { return item.value; } onKeyDown() { this.showAutocompletion = true; } get isActiveLabel() { const value = this.control.value; return this.forceShowLabel && (!!value || !!this.placeholder); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.6", ngImport: i0, type: ItSearchComponent, deps: null, target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.6", type: ItSearchComponent, isStandalone: true, selector: "it-search", inputs: { searchData: "searchData", big: ["big", "big", inputToBoolean], debounceTime: "debounceTime", placeholder: "placeholder", labelWaria: "labelWaria", forceShowLabel: ["forceShowLabel", "forceShowLabel", inputToBoolean] }, outputs: { searchSelectedEvent: "searchSelectedEvent" }, usesInheritance: true, ngImport: i0, template: "<div class=\"form-group\" [class.autocomplete-wrapper-big]=\"big\">\n @if (label) {\n <label [for]=\"id\" [class.visually-hidden]=\"!isActiveLabel\" [class.active]=\"isActiveLabel\">\n {{ label }}\n </label>\n }\n\n <input\n [id]=\"id\"\n type=\"search\"\n class=\"autocomplete form-control\"\n [placeholder]=\"placeholder\"\n [formControl]=\"control\"\n [class.is-invalid]=\"isInvalid\"\n [class.is-valid]=\"isValid\"\n (blur)=\"markAsTouched()\"\n (keydown)=\"onKeyDown()\" />\n\n <span class=\"autocomplete-icon\" aria-hidden=\"true\">\n <it-icon [labelWaria]=\"labelWaria\" name=\"search\" size=\"sm\"></it-icon>\n </span>\n\n @if (searchResults$ | async; as autocomplete) {\n <ul class=\"autocomplete-list\" [class.autocomplete-list-show]=\"autocomplete.relatedEntries?.length && showAutocompletion\">\n @for (entry of autocomplete.relatedEntries; track searchItemTrackByValueFn($index, entry)) {\n <li>\n <a [href]=\"entry.link\" (click)=\"onEntryClick(entry, $event)\">\n @if (entry.avatarSrcPath) {\n <div class=\"avatar size-sm\">\n <img [src]=\"entry.avatarSrcPath\" [alt]=\"entry.avatarAltText\" />\n </div>\n }\n @if (entry.icon) {\n <it-icon [name]=\"entry.icon\" size=\"sm\"></it-icon>\n }\n <span class=\"autocomplete-list-text\">\n <span [innerHTML]=\"entry.value | itMarkMatchingText: autocomplete.searchedValue\"></span>\n @if (entry.label) {\n <em>{{ entry.label }}</em>\n }\n </span>\n </a>\n </li>\n }\n </ul>\n }\n\n @if (isInvalid) {\n <div class=\"form-feedback just-validate-error-label\" [id]=\"id + '-error'\">\n <div #customError>\n <ng-content select=\"[error]\"></ng-content>\n </div>\n @if (!customError.hasChildNodes()) {\n {{ invalidMessage | async }}\n }\n </div>\n }\n</div>\n", dependencies: [{ kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "component", type: ItIconComponent, selector: "it-icon", inputs: ["name", "size", "color", "padded", "svgClass", "title", "labelWaria"] }, { kind: "pipe", type: ItMarkMatchingTextPipe, name: "itMarkMatchingText" }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.6", ngImport: i0, type: ItSearchComponent, decorators: [{ type: Component, args: [{ standalone: true, selector: 'it-search', imports: [AsyncPipe, ItIconComponent, ItMarkMatchingTextPipe, NgTemplateOutlet, ReactiveFormsModule], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"form-group\" [class.autocomplete-wrapper-big]=\"big\">\n @if (label) {\n <label [for]=\"id\" [class.visually-hidden]=\"!isActiveLabel\" [class.active]=\"isActiveLabel\">\n {{ label }}\n </label>\n }\n\n <input\n [id]=\"id\"\n type=\"search\"\n class=\"autocomplete form-control\"\n [placeholder]=\"placeholder\"\n [formControl]=\"control\"\n [class.is-invalid]=\"isInvalid\"\n [class.is-valid]=\"isValid\"\n (blur)=\"markAsTouched()\"\n (keydown)=\"onKeyDown()\" />\n\n <span class=\"autocomplete-icon\" aria-hidden=\"true\">\n <it-icon [labelWaria]=\"labelWaria\" name=\"search\" size=\"sm\"></it-icon>\n </span>\n\n @if (searchResults$ | async; as autocomplete) {\n <ul class=\"autocomplete-list\" [class.autocomplete-list-show]=\"autocomplete.relatedEntries?.length && showAutocompletion\">\n @for (entry of autocomplete.relatedEntries; track searchItemTrackByValueFn($index, entry)) {\n <li>\n <a [href]=\"entry.link\" (click)=\"onEntryClick(entry, $event)\">\n @if (entry.avatarSrcPath) {\n <div class=\"avatar size-sm\">\n <img [src]=\"entry.avatarSrcPath\" [alt]=\"entry.avatarAltText\" />\n </div>\n }\n @if (entry.icon) {\n <it-icon [name]=\"entry.icon\" size=\"sm\"></it-icon>\n }\n <span class=\"autocomplete-list-text\">\n <span [innerHTML]=\"entry.value | itMarkMatchingText: autocomplete.searchedValue\"></span>\n @if (entry.label) {\n <em>{{ entry.label }}</em>\n }\n </span>\n </a>\n </li>\n }\n </ul>\n }\n\n @if (isInvalid) {\n <div class=\"form-feedback just-validate-error-label\" [id]=\"id + '-error'\">\n <div #customError>\n <ng-content select=\"[error]\"></ng-content>\n </div>\n @if (!customError.hasChildNodes()) {\n {{ invalidMessage | async }}\n }\n </div>\n }\n</div>\n" }] }], propDecorators: { searchData: [{ type: Input, args: [{ required: true }] }], big: [{ type: Input, args: [{ transform: inputToBoolean }] }], debounceTime: [{ type: Input }], placeholder: [{ type: Input }], labelWaria: [{ type: Input }], forceShowLabel: [{ type: Input, args: [{ transform: inputToBoolean }] }], searchSelectedEvent: [{ type: Output }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"search.component.js","sourceRoot":"","sources":["../../../../../../../projects/design-angular-kit/src/lib/components/form/search/search.component.ts","../../../../../../../projects/design-angular-kit/src/lib/components/form/search/search.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, EventEmitter, Input, OnInit, Output } from '@angular/core';\nimport { debounceTime, distinctUntilChanged, map, Observable, of, switchMap } from 'rxjs';\nimport { AsyncPipe, NgTemplateOutlet } from '@angular/common';\nimport { ReactiveFormsModule } from '@angular/forms';\nimport { ItIconComponent } from '../../utils/icon/icon.component';\nimport { ItMarkMatchingTextPipe } from '../../../pipes/mark-matching-text.pipe';\nimport { ItAbstractFormComponent } from '../../../abstracts/abstract-form.component';\nimport { SearchItem } from '../../../interfaces/form';\nimport { inputToBoolean } from '../../../utils/coercion';\n\n@Component({\n  standalone: true,\n  selector: 'it-search',\n  templateUrl: './search.component.html',\n  imports: [AsyncPipe, ItIconComponent, ItMarkMatchingTextPipe, NgTemplateOutlet, ReactiveFormsModule],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ItSearchComponent extends ItAbstractFormComponent<string | null | undefined> implements OnInit {\n  /**\n   * Indicates the list of searchable elements on which to base the input search system\n   * If you need to retrieve items via API, can pass a function of Observable\n   * @default undefined\n   */\n  @Input({ required: true }) searchData!: Array<SearchItem> | ((search?: string | null) => Observable<Array<SearchItem>>);\n\n  /**\n   * To get a large version of Search\n   */\n  @Input({ transform: inputToBoolean }) big?: boolean;\n\n  /**\n   * Time span [ms] has passed without another source emission, to delay data filtering.\n   * Useful when the user is typing multiple letters\n   * @default 300 [ms]\n   */\n  @Input() debounceTime = 300;\n\n  /**\n   * The input placeholder\n   */\n  @Input() placeholder = '';\n\n  /**\n   * The input label even get labelWaria icon\n   */\n  @Input() labelWaria: string | undefined = undefined;\n\n  /**\n   * Show the label\n   */\n  @Input({ transform: inputToBoolean }) forceShowLabel: boolean = true;\n\n  /**\n   * Fired when the Search Item has been selected\n   */\n  @Output() searchSelectedEvent: EventEmitter<SearchItem> = new EventEmitter();\n\n  protected showAutocompletion = false;\n\n  /** Observable da cui vengono emessi i risultati dell'auto completamento */\n  protected searchResults$: Observable<{\n    searchedValue: string | undefined | null;\n    relatedEntries: Array<SearchItem>;\n  }> = new Observable();\n\n  override ngOnInit() {\n    super.ngOnInit();\n    this.searchResults$ = this.getSearchResults$();\n  }\n\n  /**\n   * Create the search list\n   */\n  private getSearchResults$(): Observable<{\n    searchedValue: string | null | undefined;\n    relatedEntries: Array<SearchItem>;\n  }> {\n    return this.control.valueChanges.pipe(\n      debounceTime(this.debounceTime), // Delay filter data after time span has passed without another source emission, useful when the user is typing multiple letters\n      distinctUntilChanged(), // Only if searchValue is distinct in comparison to the last value\n      switchMap(searchedValue => {\n        if (!this.searchData) {\n          return of({\n            searchedValue,\n            relatedEntries: <Array<SearchItem>>[],\n          });\n        }\n\n        const autoCompleteData$ = Array.isArray(this.searchData) ? of(this.searchData) : this.searchData(searchedValue);\n        return autoCompleteData$.pipe(\n          map(searchData => {\n            if (!searchedValue || typeof searchedValue === 'number') {\n              return { searchedValue, relatedEntries: [] };\n            }\n\n            const lowercaseValue = searchedValue.toLowerCase();\n            const relatedEntries = searchData.filter(item => item.value?.toLowerCase().includes(lowercaseValue));\n\n            return { searchedValue, relatedEntries };\n          })\n        );\n      })\n    );\n  }\n\n  protected onEntryClick(entry: SearchItem, event: Event) {\n    // Se non è stato definito un link associato all'elemento dell'search, probabilmente il desiderata\n    // non è effettuare la navigazione al default '#', pertanto in tal caso meglio annullare la navigazione.\n    if (!entry.link) {\n      event.preventDefault();\n    }\n\n    this.searchSelectedEvent.next(entry);\n    this.control.setValue(entry.value);\n    this.showAutocompletion = false;\n  }\n\n  protected searchItemTrackByValueFn(index: number, item: SearchItem) {\n    return item.value;\n  }\n\n  protected onKeyDown() {\n    this.showAutocompletion = true;\n  }\n\n  protected get isActiveLabel(): boolean {\n    const value = this.control.value;\n    return this.forceShowLabel && (!!value || !!this.placeholder);\n  }\n}\n","<div class=\"form-group\" [class.autocomplete-wrapper-big]=\"big\">\n  @if (label) {\n    <label [for]=\"id\" [class.visually-hidden]=\"!isActiveLabel\" [class.active]=\"isActiveLabel\">\n      {{ label }}\n    </label>\n  }\n\n  <input\n    [id]=\"id\"\n    type=\"search\"\n    class=\"autocomplete form-control\"\n    [placeholder]=\"placeholder\"\n    [formControl]=\"control\"\n    [class.is-invalid]=\"isInvalid\"\n    [class.is-valid]=\"isValid\"\n    (blur)=\"markAsTouched()\"\n    (keydown)=\"onKeyDown()\" />\n\n  <span class=\"autocomplete-icon\" aria-hidden=\"true\">\n    <it-icon [labelWaria]=\"labelWaria\" name=\"search\" size=\"sm\"></it-icon>\n  </span>\n\n  @if (searchResults$ | async; as autocomplete) {\n    <ul class=\"autocomplete-list\" [class.autocomplete-list-show]=\"autocomplete.relatedEntries?.length && showAutocompletion\">\n      @for (entry of autocomplete.relatedEntries; track searchItemTrackByValueFn($index, entry)) {\n        <li>\n          <a [href]=\"entry.link\" (click)=\"onEntryClick(entry, $event)\">\n            @if (entry.avatarSrcPath) {\n              <div class=\"avatar size-sm\">\n                <img [src]=\"entry.avatarSrcPath\" [alt]=\"entry.avatarAltText\" />\n              </div>\n            }\n            @if (entry.icon) {\n              <it-icon [name]=\"entry.icon\" size=\"sm\"></it-icon>\n            }\n            <span class=\"autocomplete-list-text\">\n              <span [innerHTML]=\"entry.value | itMarkMatchingText: autocomplete.searchedValue\"></span>\n              @if (entry.label) {\n                <em>{{ entry.label }}</em>\n              }\n            </span>\n          </a>\n        </li>\n      }\n    </ul>\n  }\n\n  @if (isInvalid) {\n    <div class=\"form-feedback just-validate-error-label\" [id]=\"id + '-error'\">\n      <div #customError>\n        <ng-content select=\"[error]\"></ng-content>\n      </div>\n      @if (!customError.hasChildNodes()) {\n        {{ invalidMessage | async }}\n      }\n    </div>\n  }\n</div>\n"]}