UNPKG

@doku-dev/doku-fragment

Version:

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

252 lines 35.1 kB
import { CommonModule } from '@angular/common'; import { ChangeDetectionStrategy, Component, ContentChild, ContentChildren, HostBinding, Input, ViewChild, ViewEncapsulation, } from '@angular/core'; import { ReplaySubject, filter, fromEvent, takeUntil } from 'rxjs'; import { DokuFieldError } from './field-helpers/field-error.component'; import { DokuFieldHint } from './field-helpers/field-hint.component'; import { DokuFieldSuccess } from './field-helpers/field-success.component'; import { DOKU_FORM_FIELD_ACCESSOR } from './token'; import * as i0 from "@angular/core"; import * as i1 from "@angular/common"; export class DokuFormField { get shouldShowFieldHelper() { if (this.hasFieldSuccess && this.showSuccessBehavior && this.isSuccess) return this.isSuccess; if (this.hasFieldError && this.isError) return this.isError; if (this.hasFieldHint) return true; return false; } get isError() { return this._isError; } set isError(value) { this._isError = value; this.cdRef.markForCheck(); } get isSuccess() { return this._isSuccess; } set isSuccess(value) { this._isSuccess = value; this.cdRef.markForCheck(); } set isFocus(value) { this._isFocus = this.isDisabled || this.isReadonly ? false : value; } get isFocus() { return this._isFocus; } constructor(cdRef) { this.cdRef = cdRef; /** * Whether to show the success behavior when the validation is valid. * @default false */ this.showSuccessBehavior = false; this.classes = 'd-form-field'; this.hasFieldHint = false; this.hasFieldError = false; this.hasFieldSuccess = false; this._isError = false; this._isSuccess = false; this._isFocus = false; this.isDisabled = false; this.isReadonly = false; this.isTextarea = false; this.destroy$ = new ReplaySubject(); } get inputWrapperElement() { return this.inputWrapperElementRef?.nativeElement; } ngOnChanges(changes) { if (changes['isErrorState']) { this.useCustomErrorStateMatcher(changes['isErrorState'].currentValue); } if (changes['isSuccessState']) { this.useCustomSuccessStateMatcher(changes['isSuccessState'].currentValue); } } ngAfterContentInit() { this.hasFieldHint = !!this.fieldHint?.length; this.hasFieldError = !!this.fieldError?.length; this.hasFieldSuccess = !!this.fieldSuccess?.length; this.handleFormFieldOptions(); this.handleRegisterOnFocus(); this.handleRegisterOnBlur(); this.handleRegisterOnDisable(); this.handleRegisterOnReadonly(); this.handleRegisterOnValidate(); this.handleTextarea(); } ngAfterViewInit() { this.handleOnClickWrapperElement(); } ngOnDestroy() { this.destroy$.next(1); this.destroy$.complete(); } handleFormFieldOptions() { if (!this.formFieldAccessor?.fieldOptions) return; this.fieldOptions = this.formFieldAccessor.fieldOptions; } handleRegisterOnFocus() { if (!this.formFieldAccessor) return; const focusHandler = () => { this.isFocus = true; this.cdRef.markForCheck(); }; this.formFieldAccessor.registerOnFocus?.(focusHandler); } handleRegisterOnBlur() { if (!this.formFieldAccessor) return; const blurHandler = () => { this.isFocus = false; this.cdRef.markForCheck(); }; this.formFieldAccessor.registerOnBlur?.(blurHandler); } handleRegisterOnDisable() { if (!this.formFieldAccessor) return; const disableHandler = (value) => { this.isDisabled = value; this.cdRef.markForCheck(); }; this.formFieldAccessor.registerOnDisable?.(disableHandler); } handleRegisterOnReadonly() { if (!this.formFieldAccessor) return; const readonlyHandler = (value) => { this.isReadonly = value; this.cdRef.markForCheck(); }; this.formFieldAccessor.registerOnReadonly?.(readonlyHandler); } handleRegisterOnValidate() { if (!this.formFieldAccessor) return; const validationHandler = (value, state) => { if (typeof this.isErrorState === 'boolean') return; if (typeof this.isSuccessState === 'boolean') return; const isControlReset = state?.pristine && state.untouched; if (isControlReset) { this.isError = false; this.isSuccess = false; } else if (value === 'valid') { this.isError = false; this.isSuccess = true; } else if (value === 'invalid') { this.isError = true; this.isSuccess = false; } else { this.isError = false; this.isSuccess = false; } }; this.formFieldAccessor.registerOnValidate?.(validationHandler); } handleOnClickWrapperElement() { if (!this.formFieldAccessor) return; fromEvent(this.inputWrapperElement, 'click') .pipe(filter((ev) => !this.isElementPrefixOrSuffixClicked(ev.target)), takeUntil(this.destroy$)) .subscribe((e) => { this.formFieldAccessor?.onClickWrapperElement?.(e); }); } handleTextarea() { if (!this.formFieldAccessor) return; const isTextarea = this.formFieldAccessor.elementRef?.nativeElement?.nodeName?.toLowerCase() === 'textarea'; if (!isTextarea) return; this.isTextarea = true; } isElementPrefixOrSuffixClicked(srcElement) { const prefixElement = this.inputWrapperElement.querySelector('[doku-field-prefix]'); const suffixElement = this.inputWrapperElement.querySelector('[doku-field-suffix]'); let clicked = false; if (prefixElement) clicked = this.checkElementChildren(srcElement, prefixElement); if (clicked) return clicked; if (suffixElement) clicked = this.checkElementChildren(srcElement, suffixElement); return clicked; } checkElementChildren(clickedElement, checkElement) { if (!clickedElement || !checkElement) return false; if (clickedElement === checkElement) return true; if (!checkElement.hasChildNodes()) return false; return Array.from(checkElement.children).some((child) => this.checkElementChildren(clickedElement, child)); } useCustomErrorStateMatcher(value) { this.isError = value; } useCustomSuccessStateMatcher(value) { this.isSuccess = value; } } DokuFormField.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuFormField, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); DokuFormField.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: DokuFormField, isStandalone: true, selector: "doku-form-field", inputs: { showSuccessBehavior: "showSuccessBehavior", isErrorState: "isErrorState", isSuccessState: "isSuccessState" }, host: { properties: { "class.d-field-show-success": "this.showSuccessBehavior", "class": "this.classes", "class.d-form-field-error": "this.isError", "class.d-form-field-success": "this.isSuccess", "class.d-field-disabled": "this.isDisabled", "class.d-field-readonly": "this.isReadonly", "class.d-field-textarea": "this.isTextarea" } }, queries: [{ propertyName: "formFieldAccessor", first: true, predicate: DOKU_FORM_FIELD_ACCESSOR, descendants: true }, { propertyName: "fieldHint", predicate: DokuFieldHint }, { propertyName: "fieldError", predicate: DokuFieldError }, { propertyName: "fieldSuccess", predicate: DokuFieldSuccess }], viewQueries: [{ propertyName: "inputWrapperElementRef", first: true, predicate: ["inputWrapper"], descendants: true }], exportAs: ["dokuFormField"], usesOnChanges: true, ngImport: i0, template: "<ng-content select=\"doku-field-label\"></ng-content>\n\n<div\n #inputWrapper\n class=\"d-field-input-wrapper\"\n [class.d-field-focus]=\"isFocus\"\n [class.no-style]=\"fieldOptions?.withoutInputStyle\"\n>\n <ng-content select=\"[doku-field-prefix]\"></ng-content>\n <ng-content></ng-content>\n <ng-content select=\"[doku-field-suffix]\"></ng-content>\n</div>\n\n<div *ngIf=\"shouldShowFieldHelper\" class=\"d-field-helper-text d-text-body-s\">\n <ng-container *ngIf=\"hasFieldHint && !isError && (!isSuccess || !showSuccessBehavior)\">\n <ng-content select=\"doku-field-hint\"></ng-content>\n </ng-container>\n <ng-container *ngIf=\"hasFieldError && isError\">\n <ng-content select=\"doku-field-error\"></ng-content>\n </ng-container>\n <ng-container *ngIf=\"hasFieldSuccess && isSuccess && showSuccessBehavior\">\n <ng-content select=\"doku-field-success\"></ng-content>\n </ng-container>\n</div>\n", dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuFormField, decorators: [{ type: Component, args: [{ selector: 'doku-form-field', exportAs: 'dokuFormField', standalone: true, imports: [CommonModule], encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-content select=\"doku-field-label\"></ng-content>\n\n<div\n #inputWrapper\n class=\"d-field-input-wrapper\"\n [class.d-field-focus]=\"isFocus\"\n [class.no-style]=\"fieldOptions?.withoutInputStyle\"\n>\n <ng-content select=\"[doku-field-prefix]\"></ng-content>\n <ng-content></ng-content>\n <ng-content select=\"[doku-field-suffix]\"></ng-content>\n</div>\n\n<div *ngIf=\"shouldShowFieldHelper\" class=\"d-field-helper-text d-text-body-s\">\n <ng-container *ngIf=\"hasFieldHint && !isError && (!isSuccess || !showSuccessBehavior)\">\n <ng-content select=\"doku-field-hint\"></ng-content>\n </ng-container>\n <ng-container *ngIf=\"hasFieldError && isError\">\n <ng-content select=\"doku-field-error\"></ng-content>\n </ng-container>\n <ng-container *ngIf=\"hasFieldSuccess && isSuccess && showSuccessBehavior\">\n <ng-content select=\"doku-field-success\"></ng-content>\n </ng-container>\n</div>\n" }] }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { showSuccessBehavior: [{ type: HostBinding, args: ['class.d-field-show-success'] }, { type: Input }], isErrorState: [{ type: Input }], isSuccessState: [{ type: Input }], classes: [{ type: HostBinding, args: ['class'] }], isError: [{ type: HostBinding, args: ['class.d-form-field-error'] }], isSuccess: [{ type: HostBinding, args: ['class.d-form-field-success'] }], isDisabled: [{ type: HostBinding, args: ['class.d-field-disabled'] }], isReadonly: [{ type: HostBinding, args: ['class.d-field-readonly'] }], isTextarea: [{ type: HostBinding, args: ['class.d-field-textarea'] }], formFieldAccessor: [{ type: ContentChild, args: [DOKU_FORM_FIELD_ACCESSOR] }], fieldHint: [{ type: ContentChildren, args: [DokuFieldHint, { descendants: false }] }], fieldError: [{ type: ContentChildren, args: [DokuFieldError, { descendants: false }] }], fieldSuccess: [{ type: ContentChildren, args: [DokuFieldSuccess, { descendants: false }] }], inputWrapperElementRef: [{ type: ViewChild, args: ['inputWrapper'] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-field.component.js","sourceRoot":"","sources":["../../../../../../projects/doku-fragment/src/lib/form-field/form-field.component.ts","../../../../../../projects/doku-fragment/src/lib/form-field/form-field.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  AfterContentInit,\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ContentChild,\n  ContentChildren,\n  ElementRef,\n  HostBinding,\n  Input,\n  OnChanges,\n  OnDestroy,\n  QueryList,\n  SimpleChanges,\n  ViewChild,\n  ViewEncapsulation,\n} from '@angular/core';\nimport { ReplaySubject, filter, fromEvent, takeUntil } from 'rxjs';\nimport { DokuFieldError } from './field-helpers/field-error.component';\nimport { DokuFieldHint } from './field-helpers/field-hint.component';\nimport { DokuFieldSuccess } from './field-helpers/field-success.component';\nimport {\n  DokuFormFieldAccessor,\n  DokuFormFieldAccessorValidateState,\n  DokuFormFieldAccessorValidateValue,\n  DokuFormFieldOptions,\n} from './form-field-accessor';\nimport { DOKU_FORM_FIELD_ACCESSOR } from './token';\n\n@Component({\n  selector: 'doku-form-field',\n  exportAs: 'dokuFormField',\n  standalone: true,\n  imports: [CommonModule],\n  templateUrl: './form-field.component.html',\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class DokuFormField implements AfterContentInit, AfterViewInit, OnDestroy, OnChanges {\n  /**\n   * Whether to show the success behavior when the validation is valid.\n   * @default false\n   */\n  @HostBinding('class.d-field-show-success')\n  @Input()\n  showSuccessBehavior = false;\n\n  /**\n   * Override built-in error state handler.\n   * @default undefined\n   */\n  @Input() isErrorState?: boolean;\n\n  /**\n   * Override built-in success state handler.\n   */\n  @Input() isSuccessState?: boolean;\n\n  @HostBinding('class')\n  protected readonly classes = 'd-form-field';\n\n  protected fieldOptions?: DokuFormFieldOptions;\n\n  protected hasFieldHint = false;\n  protected hasFieldError = false;\n  protected hasFieldSuccess = false;\n  protected get shouldShowFieldHelper(): boolean {\n    if (this.hasFieldSuccess && this.showSuccessBehavior && this.isSuccess) return this.isSuccess;\n    if (this.hasFieldError && this.isError) return this.isError;\n    if (this.hasFieldHint) return true;\n    return false;\n  }\n\n  @HostBinding('class.d-form-field-error')\n  protected get isError(): boolean {\n    return this._isError;\n  }\n  protected set isError(value: boolean) {\n    this._isError = value;\n    this.cdRef.markForCheck();\n  }\n  private _isError = false;\n\n  @HostBinding('class.d-form-field-success')\n  protected get isSuccess(): boolean {\n    return this._isSuccess;\n  }\n  protected set isSuccess(value: boolean) {\n    this._isSuccess = value;\n    this.cdRef.markForCheck();\n  }\n  private _isSuccess = false;\n\n  private _isFocus = false;\n  protected set isFocus(value: boolean) {\n    this._isFocus = this.isDisabled || this.isReadonly ? false : value;\n  }\n  protected get isFocus(): boolean {\n    return this._isFocus;\n  }\n\n  @HostBinding('class.d-field-disabled')\n  protected isDisabled = false;\n\n  @HostBinding('class.d-field-readonly')\n  protected isReadonly = false;\n\n  @HostBinding('class.d-field-textarea')\n  protected isTextarea = false;\n\n  @ContentChild(DOKU_FORM_FIELD_ACCESSOR) private formFieldAccessor?: DokuFormFieldAccessor;\n\n  @ContentChildren(DokuFieldHint, { descendants: false })\n  private fieldHint?: QueryList<DokuFieldHint>;\n  @ContentChildren(DokuFieldError, { descendants: false })\n  private fieldError?: QueryList<DokuFieldError>;\n  @ContentChildren(DokuFieldSuccess, { descendants: false })\n  private fieldSuccess?: QueryList<DokuFieldSuccess>;\n\n  @ViewChild('inputWrapper') private inputWrapperElementRef?: ElementRef;\n\n  private destroy$ = new ReplaySubject();\n\n  constructor(private cdRef: ChangeDetectorRef) {}\n\n  protected get inputWrapperElement(): HTMLElement {\n    return this.inputWrapperElementRef?.nativeElement;\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['isErrorState']) {\n      this.useCustomErrorStateMatcher(changes['isErrorState'].currentValue);\n    }\n    if (changes['isSuccessState']) {\n      this.useCustomSuccessStateMatcher(changes['isSuccessState'].currentValue);\n    }\n  }\n\n  ngAfterContentInit(): void {\n    this.hasFieldHint = !!this.fieldHint?.length;\n    this.hasFieldError = !!this.fieldError?.length;\n    this.hasFieldSuccess = !!this.fieldSuccess?.length;\n\n    this.handleFormFieldOptions();\n    this.handleRegisterOnFocus();\n    this.handleRegisterOnBlur();\n    this.handleRegisterOnDisable();\n    this.handleRegisterOnReadonly();\n    this.handleRegisterOnValidate();\n    this.handleTextarea();\n  }\n\n  ngAfterViewInit(): void {\n    this.handleOnClickWrapperElement();\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next(1);\n    this.destroy$.complete();\n  }\n\n  private handleFormFieldOptions() {\n    if (!this.formFieldAccessor?.fieldOptions) return;\n    this.fieldOptions = this.formFieldAccessor.fieldOptions;\n  }\n\n  private handleRegisterOnFocus() {\n    if (!this.formFieldAccessor) return;\n    const focusHandler = () => {\n      this.isFocus = true;\n      this.cdRef.markForCheck();\n    };\n    this.formFieldAccessor.registerOnFocus?.(focusHandler);\n  }\n\n  private handleRegisterOnBlur() {\n    if (!this.formFieldAccessor) return;\n    const blurHandler = () => {\n      this.isFocus = false;\n      this.cdRef.markForCheck();\n    };\n    this.formFieldAccessor.registerOnBlur?.(blurHandler);\n  }\n\n  private handleRegisterOnDisable() {\n    if (!this.formFieldAccessor) return;\n    const disableHandler = (value: boolean) => {\n      this.isDisabled = value;\n      this.cdRef.markForCheck();\n    };\n    this.formFieldAccessor.registerOnDisable?.(disableHandler);\n  }\n\n  private handleRegisterOnReadonly() {\n    if (!this.formFieldAccessor) return;\n    const readonlyHandler = (value: boolean) => {\n      this.isReadonly = value;\n      this.cdRef.markForCheck();\n    };\n    this.formFieldAccessor.registerOnReadonly?.(readonlyHandler);\n  }\n\n  private handleRegisterOnValidate() {\n    if (!this.formFieldAccessor) return;\n    const validationHandler = (\n      value?: DokuFormFieldAccessorValidateValue,\n      state?: DokuFormFieldAccessorValidateState\n    ) => {\n      if (typeof this.isErrorState === 'boolean') return;\n      if (typeof this.isSuccessState === 'boolean') return;\n\n      const isControlReset = state?.pristine && state.untouched;\n\n      if (isControlReset) {\n        this.isError = false;\n        this.isSuccess = false;\n      } else if (value === 'valid') {\n        this.isError = false;\n        this.isSuccess = true;\n      } else if (value === 'invalid') {\n        this.isError = true;\n        this.isSuccess = false;\n      } else {\n        this.isError = false;\n        this.isSuccess = false;\n      }\n    };\n    this.formFieldAccessor.registerOnValidate?.(validationHandler);\n  }\n\n  private handleOnClickWrapperElement() {\n    if (!this.formFieldAccessor) return;\n    fromEvent(this.inputWrapperElement, 'click')\n      .pipe(\n        filter((ev) => !this.isElementPrefixOrSuffixClicked(ev.target as HTMLElement)),\n        takeUntil(this.destroy$)\n      )\n      .subscribe((e) => {\n        this.formFieldAccessor?.onClickWrapperElement?.(e);\n      });\n  }\n\n  private handleTextarea() {\n    if (!this.formFieldAccessor) return;\n    const isTextarea =\n      (this.formFieldAccessor as any).elementRef?.nativeElement?.nodeName?.toLowerCase() ===\n      'textarea';\n    if (!isTextarea) return;\n    this.isTextarea = true;\n  }\n\n  private isElementPrefixOrSuffixClicked(srcElement: HTMLElement): boolean {\n    const prefixElement: HTMLElement | null =\n      this.inputWrapperElement.querySelector('[doku-field-prefix]');\n    const suffixElement: HTMLElement | null =\n      this.inputWrapperElement.querySelector('[doku-field-suffix]');\n\n    let clicked = false;\n\n    if (prefixElement) clicked = this.checkElementChildren(srcElement, prefixElement);\n    if (clicked) return clicked;\n    if (suffixElement) clicked = this.checkElementChildren(srcElement, suffixElement);\n    return clicked;\n  }\n\n  private checkElementChildren(clickedElement: HTMLElement, checkElement?: HTMLElement): boolean {\n    if (!clickedElement || !checkElement) return false;\n    if (clickedElement === checkElement) return true;\n    if (!checkElement.hasChildNodes()) return false;\n    return Array.from(checkElement.children).some((child) =>\n      this.checkElementChildren(clickedElement, child as HTMLElement)\n    );\n  }\n\n  private useCustomErrorStateMatcher(value: boolean): void {\n    this.isError = value;\n  }\n\n  private useCustomSuccessStateMatcher(value: boolean): void {\n    this.isSuccess = value;\n  }\n}\n","<ng-content select=\"doku-field-label\"></ng-content>\n\n<div\n  #inputWrapper\n  class=\"d-field-input-wrapper\"\n  [class.d-field-focus]=\"isFocus\"\n  [class.no-style]=\"fieldOptions?.withoutInputStyle\"\n>\n  <ng-content select=\"[doku-field-prefix]\"></ng-content>\n  <ng-content></ng-content>\n  <ng-content select=\"[doku-field-suffix]\"></ng-content>\n</div>\n\n<div *ngIf=\"shouldShowFieldHelper\" class=\"d-field-helper-text d-text-body-s\">\n  <ng-container *ngIf=\"hasFieldHint && !isError && (!isSuccess || !showSuccessBehavior)\">\n    <ng-content select=\"doku-field-hint\"></ng-content>\n  </ng-container>\n  <ng-container *ngIf=\"hasFieldError && isError\">\n    <ng-content select=\"doku-field-error\"></ng-content>\n  </ng-container>\n  <ng-container *ngIf=\"hasFieldSuccess && isSuccess && showSuccessBehavior\">\n    <ng-content select=\"doku-field-success\"></ng-content>\n  </ng-container>\n</div>\n"]}