import { ChangeDetectionStrategy, Component, EventEmitter, forwardRef, Input, Output, signal, ViewChild, } from '@angular/core';
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
import { ButtonComponent } from '../../components/atoms/button/button.component';
import { IconComponent } from '../../components/atoms/icon/icon.component';
import { OptionComponent } from '../option/option.component';
import { Subject, debounceTime } from 'rxjs';
import { LabelComponent } from '../label/label.component';
import { AlertComponent } from '../../components/atoms/alert/alert.component';
import { TypeEnum } from '../../model/components/type.enum';
import { provideControlContainer, provideValueAccessor } from '../../utils/form.util';
import { FormGenericComponent } from '../../services/form-generic.abstract';
import { FormErrorComponent } from '../error/error.component';
import * as i0 from "@angular/core";
import * as i1 from "@angular/forms";
export class SelectComponent extends FormGenericComponent {
constructor(formGroupDirective, destroyRef) {
super(formGroupDirective);
this.destroyRef = destroyRef;
this.focusOut$ = new Subject();
this.isOpen = false;
this.option = '';
this.option$ = signal('');
this.placeholder = '';
this.label = '';
this.value = '';
this.type = TypeEnum.PRIMARY;
this.fullSize = true;
this.shouldShowError = true;
this.key = '';
this.options = [];
this.selected = new EventEmitter();
}
ngOnInit() {
this.selectOption(this.value, false);
this.focusOut$.pipe(takeUntilDestroyed(this.destroyRef), debounceTime(200)).subscribe(() => {
this.triggerChangeDetection();
this.close();
});
}
toggle() {
this.isOpen = !this.isOpen;
if (!this.isOpen) {
this.triggerTouched();
}
}
close() {
this.isOpen = false;
this.triggerTouched();
}
selectOption(key, propagate = true) {
const selectedOption = this.options.find(option => option.key === key);
this.option = key === null ? this.placeholder : selectedOption?.label;
if (propagate) {
this.selected.emit(key);
this.triggerChange(selectedOption?.key ?? null);
this.close();
}
}
writeValue(value) {
this.triggerChangeDetection();
this.selectOption(value, false);
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.2", ngImport: i0, type: SelectComponent, deps: [{ token: i1.FormGroupDirective }, { token: i0.DestroyRef }], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.2", type: SelectComponent, isStandalone: true, selector: "ct-select", inputs: { formControlName: "formControlName", placeholder: "placeholder", label: "label", value: "value", type: "type", fullSize: "fullSize", shouldShowError: "shouldShowError", key: "key", options: "options" }, outputs: { selected: "selected" }, providers: [provideValueAccessor(forwardRef(() => SelectComponent))], viewQueries: [{ propertyName: "trigger", first: true, predicate: ["triggerElement"], descendants: true }, { propertyName: "panel", first: true, predicate: ["optionsPanel"], descendants: true }], usesInheritance: true, ngImport: i0, template: "\n
\n {{ option }}\n \n
\n@if (isOpen) {\n \n
\n \n @for (option of options; track option) {\n \n }\n
\n
\n}\n@if (hasErrors(formControl)) {\n \n}\n", dependencies: [{ kind: "component", type: OptionComponent, selector: "ct-option", inputs: ["label", "icon"] }, { kind: "component", type: IconComponent, selector: "ct-icon", inputs: ["icon", "fill", "strokeWidth", "strokeColor", "size", "variant"] }, { kind: "component", type: LabelComponent, selector: "ct-form-label", inputs: ["text", "type", "formControlName", "hasError"] }, { kind: "component", type: FormErrorComponent, selector: "ct-form-error", inputs: ["errors"] }], viewProviders: [provideControlContainer()], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.2", ngImport: i0, type: SelectComponent, decorators: [{
type: Component,
args: [{ selector: 'ct-select', standalone: true, imports: [OptionComponent, ButtonComponent, IconComponent, LabelComponent, AlertComponent, FormErrorComponent], providers: [provideValueAccessor(forwardRef(() => SelectComponent))], viewProviders: [provideControlContainer()], changeDetection: ChangeDetectionStrategy.OnPush, template: "\n\n {{ option }}\n \n
\n@if (isOpen) {\n \n
\n \n @for (option of options; track option) {\n \n }\n
\n
\n}\n@if (hasErrors(formControl)) {\n \n}\n" }]
}], ctorParameters: () => [{ type: i1.FormGroupDirective }, { type: i0.DestroyRef }], propDecorators: { formControlName: [{
type: Input
}], placeholder: [{
type: Input
}], label: [{
type: Input
}], value: [{
type: Input
}], type: [{
type: Input
}], fullSize: [{
type: Input
}], shouldShowError: [{
type: Input
}], key: [{
type: Input,
args: [{ required: true }]
}], options: [{
type: Input,
args: [{ required: true }]
}], selected: [{
type: Output
}], trigger: [{
type: ViewChild,
args: ['triggerElement']
}], panel: [{
type: ViewChild,
args: ['optionsPanel']
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select.component.js","sourceRoot":"","sources":["../../../../../../projects/circletone/src/lib/forms/select/select.component.ts","../../../../../../projects/circletone/src/lib/forms/select/select.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  DestroyRef,\n  ElementRef,\n  EventEmitter,\n  forwardRef,\n  Input,\n  OnInit,\n  Output,\n  signal,\n  ViewChild,\n} from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { ButtonComponent } from '../../components/atoms/button/button.component';\nimport { IconComponent } from '../../components/atoms/icon/icon.component';\nimport { FormOptionBase } from '../../model/forms/form-base.model';\nimport { OptionComponent } from '../option/option.component';\nimport { Subject, debounceTime } from 'rxjs';\nimport { FormGroupDirective } from '@angular/forms';\nimport { LabelComponent } from '../label/label.component';\nimport { AlertComponent } from '../../components/atoms/alert/alert.component';\nimport { TypeEnum } from '../../model/components/type.enum';\nimport { provideControlContainer, provideValueAccessor } from '../../utils/form.util';\nimport { FormGenericComponent } from '../../services/form-generic.abstract';\nimport { FormErrorComponent } from '../error/error.component';\n\n@Component({\n  selector: 'ct-select',\n  standalone: true,\n  imports: [OptionComponent, ButtonComponent, IconComponent, LabelComponent, AlertComponent, FormErrorComponent],\n  templateUrl: './select.component.html',\n  providers: [provideValueAccessor(forwardRef(() => SelectComponent))],\n  viewProviders: [provideControlContainer()],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class SelectComponent extends FormGenericComponent implements OnInit {\n  focusOut$ = new Subject<any>();\n  isOpen = false;\n  option = '';\n  option$ = signal('');\n\n  @Input() formControlName!: string;\n  @Input() placeholder = '';\n  @Input() label = '';\n  @Input() value: any = '';\n  @Input() type: TypeEnum | string = TypeEnum.PRIMARY;\n  @Input() fullSize: boolean = true;\n  @Input() shouldShowError: boolean = true;\n  @Input({ required: true }) key = '';\n  @Input({ required: true }) options: FormOptionBase[] = [];\n  @Output() selected = new EventEmitter<any>();\n\n  @ViewChild('triggerElement') trigger!: ElementRef<HTMLButtonElement>;\n  @ViewChild('optionsPanel') panel!: ElementRef<HTMLDivElement>;\n\n  constructor(\n    formGroupDirective: FormGroupDirective,\n    private readonly destroyRef: DestroyRef\n  ) {\n    super(formGroupDirective);\n  }\n\n  ngOnInit(): void {\n    this.selectOption(this.value, false);\n    this.focusOut$.pipe(takeUntilDestroyed(this.destroyRef), debounceTime(200)).subscribe(() => {\n      this.triggerChangeDetection();\n      this.close();\n    });\n  }\n\n  toggle(): void {\n    this.isOpen = !this.isOpen;\n    if (!this.isOpen) {\n      this.triggerTouched();\n    }\n  }\n\n  close(): void {\n    this.isOpen = false;\n    this.triggerTouched();\n  }\n\n  selectOption(key: any, propagate = true) {\n    const selectedOption = this.options.find(option => option.key === key)!;\n    this.option = key === null ? this.placeholder : selectedOption?.label;\n    if (propagate) {\n      this.selected.emit(key);\n      this.triggerChange(selectedOption?.key ?? null);\n      this.close();\n    }\n  }\n\n  writeValue(value: any) {\n    this.triggerChangeDetection();\n    this.selectOption(value, false);\n  }\n}\n","<ct-form-label [text]=\"label\" [hasError]=\"hasErrors(formControl)\" />\n<div\n  class=\"flex flex-row items-center justify-between gap-1.5 rounded-md border bg-surface-variant px-4 py-2 text-sm text-on-surface-variant dark:bg-surface-variant-dark dark:text-on-surface-variant-dark\"\n  [class.border-error]=\"hasErrors(formControl)\"\n  [class.border-neutral-300]=\"!hasErrors(formControl)\"\n  [class.text-neutral-400]=\"option === placeholder\"\n  [class.text-on-surface-variant]=\"option !== placeholder\"\n  #triggerElement\n  tabindex=\"0\"\n  (focusout)=\"focusOut$.next($event)\"\n  (click)=\"toggle()\">\n  {{ option }}\n  <ct-icon class=\"flex\" [size]=\"'sm'\" [icon]=\"isOpen ? 'chevron-up' : 'chevron-down'\" />\n</div>\n@if (isOpen) {\n  <div class=\"relative\">\n    <div\n      #optionsPanel\n      [class.w-full]=\"fullSize\"\n      [class.w-fit]=\"!fullSize\"\n      class=\"absolute top-1 z-10 flex flex-col rounded-md border border-neutral-400 bg-surface-variant text-on-surface-variant dark:bg-surface-variant-dark dark:text-on-surface-variant-dark\">\n      <ct-option (click)=\"selectOption(null)\" [label]=\"placeholder\" />\n      @for (option of options; track option) {\n        <ct-option (click)=\"selectOption(option.key)\" [label]=\"option.label\" [icon]=\"option.icon!\" />\n      }\n    </div>\n  </div>\n}\n@if (hasErrors(formControl)) {\n  <ct-form-error [errors]=\"formControl.errors ?? {}\" />\n}\n"]}