select.component.mjs 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102
  1. import { ChangeDetectionStrategy, Component, EventEmitter, forwardRef, Input, Optional, Output, ViewChild, } from '@angular/core';
  2. import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
  3. import { ButtonComponent } from '../../components/atoms/button/button.component';
  4. import { IconComponent } from '../../components/atoms/icon/icon.component';
  5. import { OptionComponent } from '../option/option.component';
  6. import { Subject, debounceTime } from 'rxjs';
  7. import { LabelComponent } from '../label/label.component';
  8. import { AlertComponent } from '../../components/atoms/alert/alert.component';
  9. import { provideControlContainer, provideValueAccessor } from '../utils/form.util';
  10. import { FormGenericComponent } from '../../services/form-generic.abstract';
  11. import { FormErrorComponent } from '../error/error.component';
  12. import * as i0 from "@angular/core";
  13. import * as i1 from "@angular/forms";
  14. export class SelectComponent extends FormGenericComponent {
  15. constructor(formGroupDirective, destroyRef) {
  16. super(formGroupDirective);
  17. this.formGroupDirective = formGroupDirective;
  18. this.destroyRef = destroyRef;
  19. this.focusOut$ = new Subject();
  20. this.isOpen = false;
  21. this.option = '';
  22. this.key = '';
  23. this.options = [];
  24. this.label = '';
  25. this.helper = '';
  26. this.placeholder = '';
  27. this.value = '';
  28. this.disposition = 'vertical';
  29. this.fullSize = true;
  30. this.showError = false;
  31. this.selected = new EventEmitter();
  32. }
  33. ngOnInit() {
  34. this.selectOption(this.value, false);
  35. this.focusOut$.pipe(takeUntilDestroyed(this.destroyRef), debounceTime(200)).subscribe(() => {
  36. this.triggerMarkCheck();
  37. this.close();
  38. });
  39. }
  40. toggle() {
  41. this.isOpen = !this.isOpen;
  42. if (!this.isOpen) {
  43. this.triggerTouched();
  44. }
  45. }
  46. close() {
  47. this.isOpen = false;
  48. this.triggerTouched();
  49. }
  50. selectOption(key, propagate = true) {
  51. const selectedOption = this.options.find(option => option.key === key);
  52. this.option = key === null ? this.placeholder : selectedOption?.label;
  53. if (propagate) {
  54. this.selected.emit(key);
  55. this.triggerChange(selectedOption?.key ?? null);
  56. this.close();
  57. }
  58. }
  59. writeValue(value) {
  60. this.triggerMarkCheck();
  61. this.selectOption(value, false);
  62. }
  63. static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.5", ngImport: i0, type: SelectComponent, deps: [{ token: i1.FormGroupDirective, optional: true }, { token: i0.DestroyRef }], target: i0.ɵɵFactoryTarget.Component }); }
  64. static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.5", type: SelectComponent, isStandalone: true, selector: "ct-select", inputs: { formControlName: "formControlName", key: "key", options: "options", label: "label", helper: "helper", placeholder: "placeholder", value: "value", disposition: "disposition", fullSize: "fullSize", showError: "showError" }, 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: "<div class=\"flex gap-1 font-primary\" [class.flex-col]=\"disposition === 'vertical'\">\n @if (label) {\n <ct-form-label [text]=\"label\" [hasError]=\"hasErrors(formControl)\" [showError]=\"showError\" />\n }\n <div class=\"flex flex-col\" [class.w-full]=\"fullSize\" [class.w-fit]=\"!fullSize\">\n <div\n tabindex=\"0\"\n #triggerElement\n class=\"flex cursor-pointer flex-row items-center justify-between rounded-md border bg-surface px-4 py-2 text-sm text-on-surface dark:bg-surface-dark dark:text-on-surface-dark\"\n [class.border-error]=\"hasErrors(formControl) && showError\"\n [class.border-neutral-300]=\"!hasErrors(formControl) || !showError\"\n [class.text-on-surface]=\"option !== placeholder\"\n (focusout)=\"focusOut$.next($event)\"\n (click)=\"toggle()\">\n <span [class.text-neutral-500]=\"option === placeholder\">{{ option }}</span>\n <ct-icon class=\"flex\" [size]=\"'sm'\" [icon]=\"isOpen ? 'chevron-up' : 'chevron-down'\" />\n </div>\n @if (isOpen) {\n <div class=\"relative flex flex-col\">\n <div\n #optionsPanel\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 <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 </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: 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", "showError"] }, { kind: "component", type: FormErrorComponent, selector: "ct-form-alert", inputs: ["errors", "size", "hasErrors", "fullSize", "helper", "errorMessages"] }], viewProviders: [provideControlContainer()], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
  65. }
  66. i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.5", ngImport: i0, type: SelectComponent, decorators: [{
  67. type: Component,
  68. args: [{ selector: 'ct-select', standalone: true, imports: [OptionComponent, ButtonComponent, IconComponent, LabelComponent, AlertComponent, FormErrorComponent], providers: [provideValueAccessor(forwardRef(() => SelectComponent))], 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 [text]=\"label\" [hasError]=\"hasErrors(formControl)\" [showError]=\"showError\" />\n }\n <div class=\"flex flex-col\" [class.w-full]=\"fullSize\" [class.w-fit]=\"!fullSize\">\n <div\n tabindex=\"0\"\n #triggerElement\n class=\"flex cursor-pointer flex-row items-center justify-between rounded-md border bg-surface px-4 py-2 text-sm text-on-surface dark:bg-surface-dark dark:text-on-surface-dark\"\n [class.border-error]=\"hasErrors(formControl) && showError\"\n [class.border-neutral-300]=\"!hasErrors(formControl) || !showError\"\n [class.text-on-surface]=\"option !== placeholder\"\n (focusout)=\"focusOut$.next($event)\"\n (click)=\"toggle()\">\n <span [class.text-neutral-500]=\"option === placeholder\">{{ option }}</span>\n <ct-icon class=\"flex\" [size]=\"'sm'\" [icon]=\"isOpen ? 'chevron-up' : 'chevron-down'\" />\n </div>\n @if (isOpen) {\n <div class=\"relative flex flex-col\">\n <div\n #optionsPanel\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 <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 </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" }]
  69. }], ctorParameters: () => [{ type: i1.FormGroupDirective, decorators: [{
  70. type: Optional
  71. }] }, { type: i0.DestroyRef }], propDecorators: { formControlName: [{
  72. type: Input
  73. }], key: [{
  74. type: Input,
  75. args: [{ required: true }]
  76. }], options: [{
  77. type: Input,
  78. args: [{ required: true }]
  79. }], label: [{
  80. type: Input
  81. }], helper: [{
  82. type: Input
  83. }], placeholder: [{
  84. type: Input
  85. }], value: [{
  86. type: Input
  87. }], disposition: [{
  88. type: Input
  89. }], fullSize: [{
  90. type: Input
  91. }], showError: [{
  92. type: Input
  93. }], selected: [{
  94. type: Output
  95. }], trigger: [{
  96. type: ViewChild,
  97. args: ['triggerElement']
  98. }], panel: [{
  99. type: ViewChild,
  100. args: ['optionsPanel']
  101. }] } });
  102. //# 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  Optional,\n  Output,\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 { provideControlContainer, provideValueAccessor } from '../utils/form.util';\nimport { FormGenericComponent } from '../../services/form-generic.abstract';\nimport { FormErrorComponent } from '../error/error.component';\nimport { ComponentDisposition } from '../../model/components/component-disposition.enum';\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\n  @Input() formControlName!: string;\n  @Input({ required: true }) key = '';\n  @Input({ required: true }) options: FormOptionBase[] = [];\n  @Input() label = '';\n  @Input() helper = '';\n  @Input() placeholder = '';\n  @Input() value: any = '';\n  @Input() disposition: ComponentDisposition | string = 'vertical';\n  @Input() fullSize: boolean = true;\n  @Input() showError: boolean = false;\n\n  @Output() selected = new EventEmitter<any>();\n\n  @ViewChild('triggerElement') trigger!: ElementRef<HTMLButtonElement>;\n  @ViewChild('optionsPanel') panel!: ElementRef<HTMLDivElement>;\n\n  constructor(\n    @Optional() protected override 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.triggerMarkCheck();\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.triggerMarkCheck();\n    this.selectOption(value, false);\n  }\n}\n","<div class=\"flex gap-1 font-primary\" [class.flex-col]=\"disposition === 'vertical'\">\n  @if (label) {\n    <ct-form-label [text]=\"label\" [hasError]=\"hasErrors(formControl)\" [showError]=\"showError\" />\n  }\n  <div class=\"flex flex-col\" [class.w-full]=\"fullSize\" [class.w-fit]=\"!fullSize\">\n    <div\n      tabindex=\"0\"\n      #triggerElement\n      class=\"flex cursor-pointer flex-row items-center justify-between rounded-md border bg-surface px-4 py-2 text-sm text-on-surface dark:bg-surface-dark dark:text-on-surface-dark\"\n      [class.border-error]=\"hasErrors(formControl) && showError\"\n      [class.border-neutral-300]=\"!hasErrors(formControl) || !showError\"\n      [class.text-on-surface]=\"option !== placeholder\"\n      (focusout)=\"focusOut$.next($event)\"\n      (click)=\"toggle()\">\n      <span [class.text-neutral-500]=\"option === placeholder\">{{ option }}</span>\n      <ct-icon class=\"flex\" [size]=\"'sm'\" [icon]=\"isOpen ? 'chevron-up' : 'chevron-down'\" />\n    </div>\n    @if (isOpen) {\n      <div class=\"relative flex flex-col\">\n        <div\n          #optionsPanel\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          <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  </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"]}