123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117 |
- import { ChangeDetectionStrategy, Component, DestroyRef, EventEmitter, Input, Output, forwardRef, inject, } from '@angular/core';
- import { FormGenericComponent } from '../../services/form-generic.abstract';
- import { FormErrorComponent } from '../error/error.component';
- import { LabelComponent } from '../label/label.component';
- import { AsyncPipe, NgClass } from '@angular/common';
- import { IconComponent } from '../../components/atoms/icon/icon.component';
- import { OptionComponent } from '../option/option.component';
- import { provideControlContainer, provideValueAccessor } from '../utils/form.util';
- import { InputComponent } from '../input/input.component';
- import { BehaviorSubject, Subject, debounceTime } from 'rxjs';
- import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
- import * as i0 from "@angular/core";
- import * as i1 from "@angular/forms";
- export class AutocompleteComponent extends FormGenericComponent {
- constructor(formGroupDirective) {
- super(formGroupDirective);
- this.destroyRef = inject(DestroyRef);
- this.key = '';
- this.options = [];
- this.label = '';
- this.icon = '';
- this.helper = '';
- this.placeholder = '';
- this.value = '';
- this.inputValue = '';
- this.iconPosition = 'left';
- this.disposition = 'vertical';
- this.fullSize = true;
- this.multiple = false;
- this.showError = false;
- this.typed = new EventEmitter();
- this.selected = new EventEmitter();
- this.focusOut$ = new Subject();
- this.inputValue$ = new BehaviorSubject('');
- this.focusOut$.pipe(takeUntilDestroyed(this.destroyRef), debounceTime(150)).subscribe(() => {
- this.options = [];
- this.triggerTouched();
- this.triggerMarkCheck();
- });
- }
- input(value) {
- if (!value)
- this.options = [];
- if (this.value && !this.multiple) {
- this.setValue(null);
- this.inputValue$.next('');
- return;
- }
- if (this.multiple) {
- this.inputValue$.next(value);
- }
- this.typed.emit(value);
- }
- setValue(value, propagate = true) {
- this.value = value;
- if (propagate) {
- this.triggerChange(value);
- this.selected.emit(value);
- }
- if (value) {
- const selectedOption = this.options.find(option => option.key === value);
- if (!this.multiple)
- this.inputValue$.next(selectedOption ? selectedOption.label : this.placeholder);
- else {
- const currentWords = this.inputValue$.getValue().split(' ');
- // replacing the last word with the current 'label' value
- currentWords[currentWords.length - 1] = selectedOption ? selectedOption.label : this.placeholder;
- this.inputValue$.next(currentWords.join(' '));
- }
- }
- }
- writeValue(value) {
- this.triggerMarkCheck();
- this.setValue(value, false);
- }
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.5", ngImport: i0, type: AutocompleteComponent, deps: [{ token: i1.FormGroupDirective }], target: i0.ɵɵFactoryTarget.Component }); }
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.5", type: AutocompleteComponent, isStandalone: true, selector: "ct-autocomplete", inputs: { formControlName: "formControlName", key: "key", options: "options", label: "label", icon: "icon", helper: "helper", placeholder: "placeholder", value: "value", inputValue: "inputValue", iconPosition: "iconPosition", disposition: "disposition", fullSize: "fullSize", multiple: "multiple", showError: "showError" }, outputs: { typed: "typed", selected: "selected" }, providers: [provideValueAccessor(forwardRef(() => AutocompleteComponent))], usesInheritance: true, ngImport: i0, template: "<div class=\"flex gap-1 font-primary\" [class.flex-col]=\"disposition === 'vertical'\">\n @if (label) {\n <ct-form-label\n [text]=\"label\"\n [hasError]=\"hasErrors(formControl)\"\n [showError]=\"showError\"\n [formControlName]=\"formControlName\" />\n }\n <div class=\"flex flex-col\" [class.w-full]=\"fullSize\" [class.w-fit]=\"!fullSize\">\n <ct-input\n key=\"autocomplete\"\n [canDelete]=\"true\"\n [helper]=\"helper\"\n [value]=\"inputValue$ | async\"\n [placeholder]=\"placeholder\"\n (typed)=\"input($event)\"\n (focusout)=\"focusOut$.next($event)\" />\n @if (options && options.length > 0) {\n <div class=\"relative flex flex-col\">\n <div\n [class.w-full]=\"fullSize\"\n [class.w-fit]=\"!fullSize\"\n class=\"absolute top-0.5 z-10 flex max-h-72 flex-col overflow-y-auto rounded-md border border-neutral-400 bg-surface text-on-surface shadow-lg dark:bg-surface-dark dark:text-on-surface-dark\">\n @for (option of options; track option) {\n <ct-option (click)=\"setValue(option.key)\" [label]=\"option.label\" [icon]=\"option.icon!\" />\n }\n </div>\n </div>\n }\n </div>\n @if (formControl) {\n <ct-form-alert\n [hasErrors]=\"hasErrors(formControl)\"\n [helper]=\"helper\"\n [errors]=\"formControl.errors ?? {}\"\n size=\"xs\" />\n }\n</div>\n", dependencies: [{ kind: "component", type: FormErrorComponent, selector: "ct-form-alert", inputs: ["errors", "size", "hasErrors", "fullSize", "helper", "errorMessages"] }, { kind: "component", type: LabelComponent, selector: "ct-form-label", inputs: ["text", "type", "formControlName", "hasError", "showError"] }, { kind: "component", type: OptionComponent, selector: "ct-option", inputs: ["label", "icon"] }, { kind: "component", type: InputComponent, selector: "ct-input", inputs: ["formControlName", "key", "type", "value", "label", "icon", "helper", "placeholder", "symbol", "canDelete", "fullSize", "showError", "min", "max", "iconPosition", "disposition"], outputs: ["typed"] }, { kind: "pipe", type: AsyncPipe, name: "async" }], viewProviders: [provideControlContainer()], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
- }
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.5", ngImport: i0, type: AutocompleteComponent, decorators: [{
- type: Component,
- args: [{ selector: 'ct-autocomplete', standalone: true, imports: [IconComponent, FormErrorComponent, LabelComponent, OptionComponent, InputComponent, NgClass, AsyncPipe], providers: [provideValueAccessor(forwardRef(() => AutocompleteComponent))], viewProviders: [provideControlContainer()], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"flex gap-1 font-primary\" [class.flex-col]=\"disposition === 'vertical'\">\n @if (label) {\n <ct-form-label\n [text]=\"label\"\n [hasError]=\"hasErrors(formControl)\"\n [showError]=\"showError\"\n [formControlName]=\"formControlName\" />\n }\n <div class=\"flex flex-col\" [class.w-full]=\"fullSize\" [class.w-fit]=\"!fullSize\">\n <ct-input\n key=\"autocomplete\"\n [canDelete]=\"true\"\n [helper]=\"helper\"\n [value]=\"inputValue$ | async\"\n [placeholder]=\"placeholder\"\n (typed)=\"input($event)\"\n (focusout)=\"focusOut$.next($event)\" />\n @if (options && options.length > 0) {\n <div class=\"relative flex flex-col\">\n <div\n [class.w-full]=\"fullSize\"\n [class.w-fit]=\"!fullSize\"\n class=\"absolute top-0.5 z-10 flex max-h-72 flex-col overflow-y-auto rounded-md border border-neutral-400 bg-surface text-on-surface shadow-lg dark:bg-surface-dark dark:text-on-surface-dark\">\n @for (option of options; track option) {\n <ct-option (click)=\"setValue(option.key)\" [label]=\"option.label\" [icon]=\"option.icon!\" />\n }\n </div>\n </div>\n }\n </div>\n @if (formControl) {\n <ct-form-alert\n [hasErrors]=\"hasErrors(formControl)\"\n [helper]=\"helper\"\n [errors]=\"formControl.errors ?? {}\"\n size=\"xs\" />\n }\n</div>\n" }]
- }], ctorParameters: () => [{ type: i1.FormGroupDirective }], propDecorators: { formControlName: [{
- type: Input
- }], key: [{
- type: Input,
- args: [{ required: true }]
- }], options: [{
- type: Input,
- args: [{ required: true }]
- }], label: [{
- type: Input
- }], icon: [{
- type: Input
- }], helper: [{
- type: Input
- }], placeholder: [{
- type: Input
- }], value: [{
- type: Input
- }], inputValue: [{
- type: Input
- }], iconPosition: [{
- type: Input
- }], disposition: [{
- type: Input
- }], fullSize: [{
- type: Input
- }], multiple: [{
- type: Input
- }], showError: [{
- type: Input
- }], typed: [{
- type: Output
- }], selected: [{
- type: Output
- }] } });
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"autocomplete.component.js","sourceRoot":"","sources":["../../../../../../projects/circletone/src/lib/forms/autocomplete/autocomplete.component.ts","../../../../../../projects/circletone/src/lib/forms/autocomplete/autocomplete.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  DestroyRef,\n  EventEmitter,\n  Input,\n  Output,\n  forwardRef,\n  inject,\n} from '@angular/core';\nimport { FormGenericComponent } from '../../services/form-generic.abstract';\nimport { ComponentPosition } from '../../model/components/component-position.type';\nimport { FormGroupDirective } from '@angular/forms';\nimport { FormErrorComponent } from '../error/error.component';\nimport { LabelComponent } from '../label/label.component';\nimport { AsyncPipe, NgClass } from '@angular/common';\nimport { IconComponent } from '../../components/atoms/icon/icon.component';\nimport { FormOptionBase } from '../../model/forms/form-base.model';\nimport { OptionComponent } from '../option/option.component';\nimport { provideControlContainer, provideValueAccessor } from '../utils/form.util';\nimport { InputComponent } from '../input/input.component';\nimport { BehaviorSubject, Subject, debounceTime } from 'rxjs';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { ComponentDisposition } from '../../model/components/component-disposition.enum';\n\n@Component({\n  selector: 'ct-autocomplete',\n  standalone: true,\n  imports: [IconComponent, FormErrorComponent, LabelComponent, OptionComponent, InputComponent, NgClass, AsyncPipe],\n  templateUrl: './autocomplete.component.html',\n  providers: [provideValueAccessor(forwardRef(() => AutocompleteComponent))],\n  viewProviders: [provideControlContainer()],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class AutocompleteComponent extends FormGenericComponent {\n  private readonly destroyRef = inject(DestroyRef);\n  @Input() formControlName!: string;\n  @Input({ required: true }) key = '';\n  @Input({ required: true }) options: FormOptionBase[] = [];\n  @Input() label = '';\n  @Input() icon = '';\n  @Input() helper = '';\n  @Input() placeholder = '';\n  @Input() value: any = '';\n  @Input() inputValue = '';\n  @Input() iconPosition: ComponentPosition = 'left';\n  @Input() disposition: ComponentDisposition = 'vertical';\n  @Input() fullSize = true;\n  @Input() multiple = false;\n  @Input() showError = false;\n\n  @Output() typed = new EventEmitter<string>();\n  @Output() selected = new EventEmitter<any>();\n\n  focusOut$ = new Subject<any>();\n  inputValue$ = new BehaviorSubject<string>('');\n\n  constructor(formGroupDirective: FormGroupDirective) {\n    super(formGroupDirective);\n\n    this.focusOut$.pipe(takeUntilDestroyed(this.destroyRef), debounceTime(150)).subscribe(() => {\n      this.options = [];\n      this.triggerTouched();\n      this.triggerMarkCheck();\n    });\n  }\n\n  input(value: string) {\n    if (!value) this.options = [];\n    if (this.value && !this.multiple) {\n      this.setValue(null);\n      this.inputValue$.next('');\n      return;\n    }\n    if (this.multiple) {\n      this.inputValue$.next(value);\n    }\n    this.typed.emit(value);\n  }\n\n  setValue(value: any, propagate = true) {\n    this.value = value;\n\n    if (propagate) {\n      this.triggerChange(value);\n      this.selected.emit(value);\n    }\n\n    if (value) {\n      const selectedOption = this.options.find(option => option.key === value)!;\n      if (!this.multiple) this.inputValue$.next(selectedOption ? selectedOption.label : this.placeholder);\n      else {\n        const currentWords = this.inputValue$.getValue().split(' ');\n        // replacing the last word with the current 'label' value\n        currentWords[currentWords.length - 1] = selectedOption ? selectedOption.label : this.placeholder;\n        this.inputValue$.next(currentWords.join(' '));\n      }\n    }\n  }\n\n  writeValue(value: any) {\n    this.triggerMarkCheck();\n    this.setValue(value, false);\n  }\n}\n","<div class=\"flex gap-1 font-primary\" [class.flex-col]=\"disposition === 'vertical'\">\n  @if (label) {\n    <ct-form-label\n      [text]=\"label\"\n      [hasError]=\"hasErrors(formControl)\"\n      [showError]=\"showError\"\n      [formControlName]=\"formControlName\" />\n  }\n  <div class=\"flex flex-col\" [class.w-full]=\"fullSize\" [class.w-fit]=\"!fullSize\">\n    <ct-input\n      key=\"autocomplete\"\n      [canDelete]=\"true\"\n      [helper]=\"helper\"\n      [value]=\"inputValue$ | async\"\n      [placeholder]=\"placeholder\"\n      (typed)=\"input($event)\"\n      (focusout)=\"focusOut$.next($event)\" />\n    @if (options && options.length > 0) {\n      <div class=\"relative flex flex-col\">\n        <div\n          [class.w-full]=\"fullSize\"\n          [class.w-fit]=\"!fullSize\"\n          class=\"absolute top-0.5 z-10 flex max-h-72 flex-col overflow-y-auto rounded-md border border-neutral-400 bg-surface text-on-surface shadow-lg dark:bg-surface-dark dark:text-on-surface-dark\">\n          @for (option of options; track option) {\n            <ct-option (click)=\"setValue(option.key)\" [label]=\"option.label\" [icon]=\"option.icon!\" />\n          }\n        </div>\n      </div>\n    }\n  </div>\n  @if (formControl) {\n    <ct-form-alert\n      [hasErrors]=\"hasErrors(formControl)\"\n      [helper]=\"helper\"\n      [errors]=\"formControl.errors ?? {}\"\n      size=\"xs\" />\n  }\n</div>\n"]}
|