UNPKG

@doku-dev/doku-fragment

Version:

A new Angular UI library that moving away from Bootstrap and built from scratch.

258 lines 41.6 kB
import { CommonModule, DOCUMENT, DatePipe } from '@angular/common'; import { ChangeDetectionStrategy, Component, Host, HostBinding, Inject, Injector, Input, LOCALE_ID, Optional, Self, ViewEncapsulation, createComponent, } from '@angular/core'; import { Subject, delay, distinctUntilChanged, filter, fromEvent, map, skip, startWith, takeUntil, } from 'rxjs'; import { getClickType } from '../../../utils/get-click-type'; import { updateFloatingPosition } from '../../../utils/update-floating-position'; import { DOKU_FORM_FIELD_ACCESSOR, } from '../../form-field'; import { DokuDatePickerBase } from '../base/date-picker-base.component'; import { DokuDatePickerBasicProps } from '../common/date-picker-basic-props.component'; import * as i0 from "@angular/core"; import * as i1 from "@angular/common"; import * as i2 from "@angular/forms"; import * as i3 from "../../form-field"; export class DokuDatePicker extends DokuDatePickerBasicProps { constructor(renderer, envInjector, injector, appRef, ngZone, datePipe, cdr, document, localeId, ngControl, formField) { super(); this.renderer = renderer; this.envInjector = envInjector; this.injector = injector; this.appRef = appRef; this.ngZone = ngZone; this.datePipe = datePipe; this.cdr = cdr; this.document = document; this.localeId = localeId; this.ngControl = ngControl; this.formField = formField; this.class = 'd-date-picker'; /** * Whether to close date picker dropdown after selecting the date. * * @default true */ this.closeOnDateClick = true; /** * Date format that will be used for formatting displayed value in the input field. It follows Angular DatePipe's format options. * * @default 'dd/MM/yyyy' */ this.dateFormat = 'dd/MM/yyyy'; /** * Placeholder of the date picker input. * * @default 'dd/mm/yyyy' */ this.placeholder = 'dd/mm/yyyy'; this.isOpen = false; this.destroy$ = new Subject(); if (this.ngControl) { this.ngControl.valueAccessor = this; } this.onClickHandler(); } get formattedValue() { return this.value ? this.datePipe.transform(this.value, this.dateFormat, undefined, this.localeId) : null; } get inputWrapperElement() { return this.formField?.['inputWrapperElement']; } ngOnInit() { this.notifyChange$ .pipe(filter((change) => change === 'value' || change === 'minDate' || change === 'maxDate'), startWith(this.value?.toISOString() || null), map(() => this.value?.toISOString() || null), distinctUntilChanged(), skip(1), takeUntil(this.destroy$)) .subscribe((value) => { this.valueChange.emit(value); }); this.valueChange.pipe(distinctUntilChanged(), takeUntil(this.destroy$)).subscribe((value) => { this.onChange?.(value); this.onTouched?.(); this.cdr.detectChanges(); }); } ngAfterViewInit() { this.ngControl?.statusChanges ?.pipe(startWith(this.ngControl.status), distinctUntilChanged(), delay(0), takeUntil(this.destroy$)) .subscribe(() => { this.disabled = !!this.ngControl?.disabled; }); this.ngControl?.statusChanges ?.pipe(distinctUntilChanged(), delay(0), takeUntil(this.destroy$)) .subscribe((status) => { if (status === 'VALID') { this.onValidate?.('valid'); } else if (status === 'INVALID') { this.onValidate?.('invalid'); } else { this.onValidate?.(); } }); this.notifyChange$ .pipe(filter((change) => change === 'disabled' || change === 'readonly'), startWith({ disabled: this.disabled, readonly: this.readonly }), map(() => ({ disabled: this.disabled, readonly: this.readonly })), distinctUntilChanged((prev, current) => JSON.stringify(prev) === JSON.stringify(current)), delay(0), takeUntil(this.destroy$)) .subscribe(({ disabled, readonly }) => { this.onDisable?.(disabled); this.onReadonly?.(readonly); this.setInputWrapperCursorState(this.inputWrapperElement, { disabled, readonly }); }); } ngOnDestroy() { this.close(); this.destroy$.next(true); this.destroy$.complete(); this.cleanup?.(); } registerOnDisable(fn) { this.onDisable = fn; } registerOnValidate(fn) { this.onValidate = fn; } registerOnReadonly(fn) { this.onReadonly = fn; } writeValue(value) { this.value = value; } registerOnChange(fn) { this.onChange = fn; } registerOnTouched(fn) { this.onTouched = fn; } setDisabledState(isDisabled) { this.onDisable?.(isDisabled); } /** * Open the date picker dropdown. */ open() { if (this.isOpen || this.disabled || this.readonly) return; this.isOpen = true; this.portalElement = this.createPortalElement(); this.datePickerBaseRef = this.createDatePickerComponent(); this.renderer.appendChild(this.portalElement, this.datePickerBaseRef.location.nativeElement); this.renderer.appendChild(this.document.body, this.portalElement); this.doAutoUpdateDropdownPosition(); } /** * Close the date picker dropdown. */ close() { if (!this.isOpen) return; this.isOpen = false; if (this.portalElement) this.renderer.removeChild(this.document.body, this.portalElement); this.datePickerBaseRef?.destroy(); } /** * Toggle the date picker dropdown. */ toggle() { this.isOpen ? this.close() : this.open(); } createPortalElement() { const el = this.renderer.createElement('div'); this.renderer.addClass(el, 'd-date-picker-portal'); return el; } createComponentRef(props) { const ref = createComponent(DokuDatePickerBase, { environmentInjector: this.envInjector, elementInjector: props.elementInjector, }); ref.setInput('value', this.value); ref.setInput('minDate', this.minDate); ref.setInput('maxDate', this.maxDate); return ref; } createDatePickerComponent() { const elementInjector = Injector.create({ providers: [], parent: this.injector }); const ref = this.createComponentRef({ elementInjector }); this.appRef.attachView(ref.hostView); const valueChangeListener = ref.instance.valueChange.subscribe((value) => { this.value = value?.start; if (this.closeOnDateClick) this.close(); }); ref.onDestroy(() => { valueChangeListener.unsubscribe(); }); ref.changeDetectorRef.detectChanges(); return ref; } onClickHandler() { fromEvent(this.document, 'click') .pipe(takeUntil(this.destroy$)) .subscribe((event) => { const { clickOutside, clickTrigger } = getClickType(event, [this.inputWrapperElement], [this.portalElement]); if (clickTrigger) return this.toggle(); // Prevent dropdown from being closed when choosing month or year. const nodeName = event.target.nodeName.toLowerCase(); if (nodeName === 'doku-select-option') return; if (clickOutside) return this.close(); }); } doAutoUpdateDropdownPosition() { this.ngZone.runOutsideAngular(() => { if (!this.inputWrapperElement || !this.portalElement) return; this.cleanup = updateFloatingPosition({ triggerElement: this.inputWrapperElement, floatingElement: this.portalElement, placement: 'bottom-start', autoUpdate: true, middleware: { flip: true, shift: true, }, }); }); } setInputWrapperCursorState(inputWrapperElement, props) { if (!inputWrapperElement) return; let cursor = 'pointer'; if (props?.readonly) cursor = 'text'; if (props?.disabled) cursor = 'not-allowed'; inputWrapperElement.style.cursor = cursor; } } DokuDatePicker.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuDatePicker, deps: [{ token: i0.Renderer2 }, { token: i0.EnvironmentInjector }, { token: i0.Injector }, { token: i0.ApplicationRef }, { token: i0.NgZone }, { token: i1.DatePipe }, { token: i0.ChangeDetectorRef }, { token: DOCUMENT }, { token: LOCALE_ID }, { token: i2.NgControl, optional: true, self: true }, { token: i3.DokuFormField, host: true, optional: true }], target: i0.ɵɵFactoryTarget.Component }); DokuDatePicker.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: DokuDatePicker, isStandalone: true, selector: "doku-date-picker", inputs: { closeOnDateClick: "closeOnDateClick", dateFormat: "dateFormat", placeholder: "placeholder" }, host: { properties: { "class": "this.class" } }, providers: [DatePipe, { provide: DOKU_FORM_FIELD_ACCESSOR, useExisting: DokuDatePicker }], exportAs: ["dokuDatePicker"], usesInheritance: true, ngImport: i0, template: "<span *ngIf=\"!formattedValue\" class=\"d-date-picker-placeholder\">\n {{ placeholder }}\n</span>\n\n<span *ngIf=\"formattedValue\" class=\"d-date-picker-value\">\n {{ formattedValue }}\n</span>\n\n<span class=\"icon-calendar\">\n <ng-container *ngTemplateOutlet=\"iconCalendar\"></ng-container>\n</span>\n\n<ng-template #iconCalendar>\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\">\n <path\n d=\"M19 5H5C4.44772 5 4 5.44772 4 6V17C4 17.5523 4.44772 18 5 18H19C19.5523 18 20 17.5523 20 17V6C20 5.44772 19.5523 5 19 5Z\"\n stroke=\"currentColor\"\n />\n <path d=\"M4 7.94336H20\" stroke=\"currentColor\" />\n <ellipse cx=\"7.69213\" cy=\"11.3774\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n <ellipse cx=\"7.69213\" cy=\"14.3207\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n <ellipse cx=\"10.6462\" cy=\"11.3774\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n <ellipse cx=\"10.6462\" cy=\"14.3207\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n <ellipse cx=\"13.5998\" cy=\"11.3774\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n <ellipse cx=\"13.5998\" cy=\"14.3207\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n <ellipse cx=\"16.554\" cy=\"11.3774\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n <ellipse cx=\"16.554\" cy=\"14.3207\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n </svg>\n</ng-template>\n", dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuDatePicker, decorators: [{ type: Component, args: [{ selector: 'doku-date-picker', exportAs: 'dokuDatePicker', standalone: true, imports: [CommonModule, DokuDatePickerBase], encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, providers: [DatePipe, { provide: DOKU_FORM_FIELD_ACCESSOR, useExisting: DokuDatePicker }], template: "<span *ngIf=\"!formattedValue\" class=\"d-date-picker-placeholder\">\n {{ placeholder }}\n</span>\n\n<span *ngIf=\"formattedValue\" class=\"d-date-picker-value\">\n {{ formattedValue }}\n</span>\n\n<span class=\"icon-calendar\">\n <ng-container *ngTemplateOutlet=\"iconCalendar\"></ng-container>\n</span>\n\n<ng-template #iconCalendar>\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\">\n <path\n d=\"M19 5H5C4.44772 5 4 5.44772 4 6V17C4 17.5523 4.44772 18 5 18H19C19.5523 18 20 17.5523 20 17V6C20 5.44772 19.5523 5 19 5Z\"\n stroke=\"currentColor\"\n />\n <path d=\"M4 7.94336H20\" stroke=\"currentColor\" />\n <ellipse cx=\"7.69213\" cy=\"11.3774\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n <ellipse cx=\"7.69213\" cy=\"14.3207\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n <ellipse cx=\"10.6462\" cy=\"11.3774\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n <ellipse cx=\"10.6462\" cy=\"14.3207\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n <ellipse cx=\"13.5998\" cy=\"11.3774\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n <ellipse cx=\"13.5998\" cy=\"14.3207\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n <ellipse cx=\"16.554\" cy=\"11.3774\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n <ellipse cx=\"16.554\" cy=\"14.3207\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n </svg>\n</ng-template>\n" }] }], ctorParameters: function () { return [{ type: i0.Renderer2 }, { type: i0.EnvironmentInjector }, { type: i0.Injector }, { type: i0.ApplicationRef }, { type: i0.NgZone }, { type: i1.DatePipe }, { type: i0.ChangeDetectorRef }, { type: Document, decorators: [{ type: Inject, args: [DOCUMENT] }] }, { type: undefined, decorators: [{ type: Inject, args: [LOCALE_ID] }] }, { type: i2.NgControl, decorators: [{ type: Optional }, { type: Self }] }, { type: i3.DokuFormField, decorators: [{ type: Optional }, { type: Host }] }]; }, propDecorators: { class: [{ type: HostBinding, args: ['class'] }], closeOnDateClick: [{ type: Input }], dateFormat: [{ type: Input }], placeholder: [{ type: Input }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"date-picker.component.js","sourceRoot":"","sources":["../../../../../../../projects/doku-fragment/src/lib/date-picker/date-picker-float/date-picker.component.ts","../../../../../../../projects/doku-fragment/src/lib/date-picker/date-picker-float/date-picker.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule, DOCUMENT, DatePipe, NgClass } from '@angular/common';\nimport {\n  AfterViewInit,\n  ApplicationRef,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ComponentRef,\n  EnvironmentInjector,\n  Host,\n  HostBinding,\n  Inject,\n  Injector,\n  Input,\n  LOCALE_ID,\n  NgZone,\n  OnDestroy,\n  OnInit,\n  Optional,\n  Renderer2,\n  Self,\n  ViewEncapsulation,\n  createComponent,\n} from '@angular/core';\nimport { ControlValueAccessor, NgControl } from '@angular/forms';\nimport {\n  Subject,\n  delay,\n  distinctUntilChanged,\n  filter,\n  fromEvent,\n  map,\n  skip,\n  startWith,\n  takeUntil,\n} from 'rxjs';\nimport { getClickType } from '../../../utils/get-click-type';\nimport { updateFloatingPosition } from '../../../utils/update-floating-position';\nimport {\n  DOKU_FORM_FIELD_ACCESSOR,\n  DokuFormField,\n  DokuFormFieldAccessor,\n  DokuFormFieldAccessorValidateValue,\n} from '../../form-field';\nimport { DokuDatePickerBase } from '../base/date-picker-base.component';\nimport { DokuDatePickerBasicProps } from '../common/date-picker-basic-props.component';\n\n@Component({\n  selector: 'doku-date-picker',\n  exportAs: 'dokuDatePicker',\n  standalone: true,\n  imports: [CommonModule, DokuDatePickerBase],\n  templateUrl: './date-picker.component.html',\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [DatePipe, { provide: DOKU_FORM_FIELD_ACCESSOR, useExisting: DokuDatePicker }],\n})\nexport class DokuDatePicker\n  extends DokuDatePickerBasicProps\n  implements OnDestroy, OnInit, ControlValueAccessor, DokuFormFieldAccessor, AfterViewInit\n{\n  @HostBinding('class')\n  protected readonly class: NgClass['ngClass'] = 'd-date-picker';\n\n  /**\n   * Whether to close date picker dropdown after selecting the date.\n   *\n   * @default true\n   */\n  @Input() closeOnDateClick = true;\n\n  /**\n   * Date format that will be used for formatting displayed value in the input field. It follows Angular DatePipe's format options.\n   *\n   * @default 'dd/MM/yyyy'\n   */\n  @Input() dateFormat = 'dd/MM/yyyy';\n\n  /**\n   * Placeholder of the date picker input.\n   *\n   * @default 'dd/mm/yyyy'\n   */\n  @Input() placeholder = 'dd/mm/yyyy';\n\n  private isOpen = false;\n\n  private portalElement?: HTMLDivElement;\n  private datePickerBaseRef?: ComponentRef<DokuDatePickerBase>;\n\n  private destroy$ = new Subject();\n  private cleanup?: () => void;\n\n  constructor(\n    private renderer: Renderer2,\n    protected envInjector: EnvironmentInjector,\n    private injector: Injector,\n    protected appRef: ApplicationRef,\n    private ngZone: NgZone,\n    private datePipe: DatePipe,\n    private cdr: ChangeDetectorRef,\n    @Inject(DOCUMENT) private document: Document,\n    @Inject(LOCALE_ID) private localeId: string,\n    @Optional() @Self() private ngControl?: NgControl,\n    @Optional() @Host() private formField?: DokuFormField\n  ) {\n    super();\n\n    if (this.ngControl) {\n      this.ngControl.valueAccessor = this;\n    }\n\n    this.onClickHandler();\n  }\n\n  protected get formattedValue(): string | null {\n    return this.value\n      ? this.datePipe.transform(this.value, this.dateFormat, undefined, this.localeId)\n      : null;\n  }\n\n  private get inputWrapperElement(): HTMLElement | undefined {\n    return this.formField?.['inputWrapperElement'];\n  }\n\n  ngOnInit(): void {\n    this.notifyChange$\n      .pipe(\n        filter((change) => change === 'value' || change === 'minDate' || change === 'maxDate'),\n        startWith(this.value?.toISOString() || null),\n        map(() => this.value?.toISOString() || null),\n        distinctUntilChanged(),\n        skip(1),\n        takeUntil(this.destroy$)\n      )\n      .subscribe((value) => {\n        this.valueChange.emit(value);\n      });\n\n    this.valueChange.pipe(distinctUntilChanged(), takeUntil(this.destroy$)).subscribe((value) => {\n      this.onChange?.(value);\n      this.onTouched?.();\n      this.cdr.detectChanges();\n    });\n  }\n\n  ngAfterViewInit(): void {\n    this.ngControl?.statusChanges\n      ?.pipe(\n        startWith(this.ngControl.status),\n        distinctUntilChanged(),\n        delay(0),\n        takeUntil(this.destroy$)\n      )\n      .subscribe(() => {\n        this.disabled = !!this.ngControl?.disabled;\n      });\n\n    this.ngControl?.statusChanges\n      ?.pipe(distinctUntilChanged(), delay(0), takeUntil(this.destroy$))\n      .subscribe((status) => {\n        if (status === 'VALID') {\n          this.onValidate?.('valid');\n        } else if (status === 'INVALID') {\n          this.onValidate?.('invalid');\n        } else {\n          this.onValidate?.();\n        }\n      });\n\n    this.notifyChange$\n      .pipe(\n        filter((change) => change === 'disabled' || change === 'readonly'),\n        startWith({ disabled: this.disabled, readonly: this.readonly }),\n        map(() => ({ disabled: this.disabled, readonly: this.readonly })),\n        distinctUntilChanged((prev, current) => JSON.stringify(prev) === JSON.stringify(current)),\n        delay(0),\n        takeUntil(this.destroy$)\n      )\n      .subscribe(({ disabled, readonly }) => {\n        this.onDisable?.(disabled);\n        this.onReadonly?.(readonly);\n        this.setInputWrapperCursorState(this.inputWrapperElement, { disabled, readonly });\n      });\n  }\n\n  ngOnDestroy(): void {\n    this.close();\n    this.destroy$.next(true);\n    this.destroy$.complete();\n    this.cleanup?.();\n  }\n\n  onDisable?: ((value: boolean) => void) | undefined;\n  onReadonly?: ((value: boolean) => void) | undefined;\n  onValidate?: ((value?: DokuFormFieldAccessorValidateValue | undefined) => void) | undefined;\n  registerOnDisable(fn: (value: boolean) => void): void {\n    this.onDisable = fn;\n  }\n  registerOnValidate?(fn: (value?: DokuFormFieldAccessorValidateValue | undefined) => void): void {\n    this.onValidate = fn;\n  }\n  registerOnReadonly(fn: (value: boolean) => void): void {\n    this.onReadonly = fn;\n  }\n\n  onChange?: (value: unknown) => void;\n  onTouched?: () => void;\n  writeValue(value?: Date | string | null): void {\n    this.value = value;\n  }\n  registerOnChange(fn: (value: unknown) => void): void {\n    this.onChange = fn;\n  }\n  registerOnTouched(fn: () => void): void {\n    this.onTouched = fn;\n  }\n  setDisabledState?(isDisabled: boolean): void {\n    this.onDisable?.(isDisabled);\n  }\n\n  /**\n   * Open the date picker dropdown.\n   */\n  open() {\n    if (this.isOpen || this.disabled || this.readonly) return;\n    this.isOpen = true;\n\n    this.portalElement = this.createPortalElement();\n    this.datePickerBaseRef = this.createDatePickerComponent();\n\n    this.renderer.appendChild(this.portalElement, this.datePickerBaseRef.location.nativeElement);\n    this.renderer.appendChild(this.document.body, this.portalElement);\n\n    this.doAutoUpdateDropdownPosition();\n  }\n\n  /**\n   * Close the date picker dropdown.\n   */\n  close() {\n    if (!this.isOpen) return;\n    this.isOpen = false;\n\n    if (this.portalElement) this.renderer.removeChild(this.document.body, this.portalElement);\n    this.datePickerBaseRef?.destroy();\n  }\n\n  /**\n   * Toggle the date picker dropdown.\n   */\n  toggle() {\n    this.isOpen ? this.close() : this.open();\n  }\n\n  private createPortalElement() {\n    const el = this.renderer.createElement('div') as HTMLDivElement;\n    this.renderer.addClass(el, 'd-date-picker-portal');\n    return el;\n  }\n\n  protected createComponentRef(props: { elementInjector: Injector }) {\n    const ref = createComponent(DokuDatePickerBase, {\n      environmentInjector: this.envInjector,\n      elementInjector: props.elementInjector,\n    });\n    ref.setInput('value', this.value);\n    ref.setInput('minDate', this.minDate);\n    ref.setInput('maxDate', this.maxDate);\n    return ref;\n  }\n\n  private createDatePickerComponent() {\n    const elementInjector = Injector.create({ providers: [], parent: this.injector });\n\n    const ref = this.createComponentRef({ elementInjector });\n\n    this.appRef.attachView(ref.hostView);\n\n    const valueChangeListener = ref.instance.valueChange.subscribe((value) => {\n      this.value = value?.start;\n      if (this.closeOnDateClick) this.close();\n    });\n\n    ref.onDestroy(() => {\n      valueChangeListener.unsubscribe();\n    });\n\n    ref.changeDetectorRef.detectChanges();\n    return ref;\n  }\n\n  private onClickHandler() {\n    fromEvent(this.document, 'click')\n      .pipe(takeUntil(this.destroy$))\n      .subscribe((event) => {\n        const { clickOutside, clickTrigger } = getClickType(\n          event,\n          [this.inputWrapperElement],\n          [this.portalElement]\n        );\n\n        if (clickTrigger) return this.toggle();\n\n        // Prevent dropdown from being closed when choosing month or year.\n        const nodeName = (event.target as HTMLElement).nodeName.toLowerCase();\n        if (nodeName === 'doku-select-option') return;\n\n        if (clickOutside) return this.close();\n      });\n  }\n\n  private doAutoUpdateDropdownPosition() {\n    this.ngZone.runOutsideAngular(() => {\n      if (!this.inputWrapperElement || !this.portalElement) return;\n      this.cleanup = updateFloatingPosition({\n        triggerElement: this.inputWrapperElement,\n        floatingElement: this.portalElement,\n        placement: 'bottom-start',\n        autoUpdate: true,\n        middleware: {\n          flip: true,\n          shift: true,\n        },\n      });\n    });\n  }\n\n  private setInputWrapperCursorState(\n    inputWrapperElement?: HTMLElement,\n    props?: { disabled?: boolean; readonly?: boolean }\n  ): void {\n    if (!inputWrapperElement) return;\n\n    let cursor = 'pointer';\n    if (props?.readonly) cursor = 'text';\n    if (props?.disabled) cursor = 'not-allowed';\n    inputWrapperElement.style.cursor = cursor;\n  }\n}\n","<span *ngIf=\"!formattedValue\" class=\"d-date-picker-placeholder\">\n  {{ placeholder }}\n</span>\n\n<span *ngIf=\"formattedValue\" class=\"d-date-picker-value\">\n  {{ formattedValue }}\n</span>\n\n<span class=\"icon-calendar\">\n  <ng-container *ngTemplateOutlet=\"iconCalendar\"></ng-container>\n</span>\n\n<ng-template #iconCalendar>\n  <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\">\n    <path\n      d=\"M19 5H5C4.44772 5 4 5.44772 4 6V17C4 17.5523 4.44772 18 5 18H19C19.5523 18 20 17.5523 20 17V6C20 5.44772 19.5523 5 19 5Z\"\n      stroke=\"currentColor\"\n    />\n    <path d=\"M4 7.94336H20\" stroke=\"currentColor\" />\n    <ellipse cx=\"7.69213\" cy=\"11.3774\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n    <ellipse cx=\"7.69213\" cy=\"14.3207\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n    <ellipse cx=\"10.6462\" cy=\"11.3774\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n    <ellipse cx=\"10.6462\" cy=\"14.3207\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n    <ellipse cx=\"13.5998\" cy=\"11.3774\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n    <ellipse cx=\"13.5998\" cy=\"14.3207\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n    <ellipse cx=\"16.554\" cy=\"11.3774\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n    <ellipse cx=\"16.554\" cy=\"14.3207\" rx=\"0.984615\" ry=\"0.981132\" fill=\"currentColor\" />\n  </svg>\n</ng-template>\n"]}