@doku-dev/doku-fragment
Version:
A new Angular UI library that moving away from Bootstrap and built from scratch.
129 lines • 16.9 kB
JavaScript
import { Directive, HostBinding, Input, Optional, Self, } from '@angular/core';
import { ReplaySubject, delay, distinctUntilChanged, map, startWith, takeUntil } from 'rxjs';
import { DOKU_FORM_FIELD_ACCESSOR, } from '../form-field';
import * as i0 from "@angular/core";
import * as i1 from "@angular/forms";
export class DokuInput {
get disabled() {
return this._disabled;
}
set disabled(value) {
this._disabled = value !== false;
this.notifyState$.next('disabled');
}
get readonly() {
return this._readonly;
}
set readonly(value) {
this._readonly = value !== false;
this.notifyState$.next('readonly');
}
constructor(elementRef, ngControl) {
this.elementRef = elementRef;
this.ngControl = ngControl;
this.classes = 'd-field-input';
this._disabled = false;
this._readonly = false;
this.notifyState$ = new ReplaySubject();
this.destroy$ = new ReplaySubject();
}
ngAfterViewInit() {
this.elementRef.nativeElement.onfocus = () => {
this.onFocus?.();
};
this.elementRef.nativeElement.onblur = () => {
this.onBlur?.();
};
this.ngControl?.statusChanges
?.pipe(startWith(this.ngControl.status), distinctUntilChanged(), takeUntil(this.destroy$))
.subscribe(() => {
this.disabled = !!this.ngControl?.disabled;
});
this.ngControl?.statusChanges
?.pipe(map((status) => ({
status: status,
state: {
pristine: this.ngControl?.control?.pristine,
untouched: this.ngControl?.control?.untouched,
},
})), distinctUntilChanged((prev, curr) => JSON.stringify(prev) === JSON.stringify(curr)), delay(0), takeUntil(this.destroy$))
.subscribe(({ status, state }) => {
if (status === 'VALID') {
this.onValidate?.('valid', state);
}
else if (status === 'INVALID') {
this.onValidate?.('invalid', state);
}
else {
this.onValidate?.(undefined, state);
}
});
this.notifyState$
.pipe(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);
// Handle element native attributes
const el = this.elementRef.nativeElement;
disabled ? el.setAttribute('disabled', '') : el.removeAttribute('disabled');
readonly ? el.setAttribute('readonly', '') : el.removeAttribute('readonly');
});
}
ngOnDestroy() {
this.destroy$.next(1);
this.destroy$.complete();
}
registerOnFocus(fn) {
this.onFocus = fn;
}
registerOnBlur(fn) {
this.onBlur = fn;
}
registerOnDisable(fn) {
this.onDisable = fn;
}
registerOnReadonly(fn) {
this.onReadonly = fn;
}
registerOnValidate(fn) {
this.onValidate = fn;
}
onClickWrapperElement() {
this.elementRef.nativeElement.focus();
}
}
DokuInput.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuInput, deps: [{ token: i0.ElementRef }, { token: i1.NgControl, optional: true, self: true }], target: i0.ɵɵFactoryTarget.Directive });
DokuInput.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "15.2.9", type: DokuInput, isStandalone: true, selector: "input[doku-input], textarea[doku-input]", inputs: { disabled: "disabled", readonly: "readonly" }, host: { properties: { "class": "this.classes" } }, providers: [
{
provide: DOKU_FORM_FIELD_ACCESSOR,
useExisting: DokuInput,
multi: true,
},
], exportAs: ["dokuInput"], ngImport: i0 });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuInput, decorators: [{
type: Directive,
args: [{
selector: 'input[doku-input], textarea[doku-input]',
exportAs: 'dokuInput',
standalone: true,
providers: [
{
provide: DOKU_FORM_FIELD_ACCESSOR,
useExisting: DokuInput,
multi: true,
},
],
}]
}], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i1.NgControl, decorators: [{
type: Optional
}, {
type: Self
}] }]; }, propDecorators: { classes: [{
type: HostBinding,
args: ['class']
}], disabled: [{
type: Input
}], readonly: [{
type: Input
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"input.directive.js","sourceRoot":"","sources":["../../../../../../projects/doku-fragment/src/lib/input/input.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { NgClass } from '@angular/common';\nimport {\n  AfterViewInit,\n  Directive,\n  ElementRef,\n  HostBinding,\n  Input,\n  OnDestroy,\n  Optional,\n  Self,\n} from '@angular/core';\nimport { NgControl } from '@angular/forms';\nimport { ReplaySubject, delay, distinctUntilChanged, map, startWith, takeUntil } from 'rxjs';\nimport {\n  DOKU_FORM_FIELD_ACCESSOR,\n  DokuFormFieldAccessor,\n  DokuFormFieldAccessorValidateState,\n  DokuFormFieldAccessorValidateValue,\n} from '../form-field';\n\n@Directive({\n  selector: 'input[doku-input], textarea[doku-input]',\n  exportAs: 'dokuInput',\n  standalone: true,\n  providers: [\n    {\n      provide: DOKU_FORM_FIELD_ACCESSOR,\n      useExisting: DokuInput,\n      multi: true,\n    },\n  ],\n})\nexport class DokuInput implements DokuFormFieldAccessor, AfterViewInit, OnDestroy {\n  @HostBinding('class')\n  protected readonly classes: NgClass['ngClass'] = 'd-field-input';\n\n  @Input()\n  get disabled(): boolean {\n    return this._disabled;\n  }\n  set disabled(value: boolean | string) {\n    this._disabled = value !== false;\n    this.notifyState$.next('disabled');\n  }\n  private _disabled = false;\n\n  @Input()\n  get readonly(): boolean {\n    return this._readonly;\n  }\n  set readonly(value: boolean | string) {\n    this._readonly = value !== false;\n    this.notifyState$.next('readonly');\n  }\n  private _readonly = false;\n\n  private notifyState$ = new ReplaySubject<'disabled' | 'readonly'>();\n  private destroy$ = new ReplaySubject();\n\n  constructor(private elementRef: ElementRef, @Optional() @Self() private ngControl?: NgControl) {}\n\n  ngAfterViewInit(): void {\n    this.elementRef.nativeElement.onfocus = () => {\n      this.onFocus?.();\n    };\n\n    this.elementRef.nativeElement.onblur = () => {\n      this.onBlur?.();\n    };\n\n    this.ngControl?.statusChanges\n      ?.pipe(startWith(this.ngControl.status), distinctUntilChanged(), takeUntil(this.destroy$))\n      .subscribe(() => {\n        this.disabled = !!this.ngControl?.disabled;\n      });\n\n    this.ngControl?.statusChanges\n      ?.pipe(\n        map((status) => ({\n          status: status,\n          state: {\n            pristine: this.ngControl?.control?.pristine,\n            untouched: this.ngControl?.control?.untouched,\n          } as DokuFormFieldAccessorValidateState,\n        })),\n        distinctUntilChanged((prev, curr) => JSON.stringify(prev) === JSON.stringify(curr)),\n        delay(0),\n        takeUntil(this.destroy$)\n      )\n      .subscribe(({ status, state }) => {\n        if (status === 'VALID') {\n          this.onValidate?.('valid', state);\n        } else if (status === 'INVALID') {\n          this.onValidate?.('invalid', state);\n        } else {\n          this.onValidate?.(undefined, state);\n        }\n      });\n\n    this.notifyState$\n      .pipe(\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        // Handle element native attributes\n        const el = this.elementRef.nativeElement as HTMLInputElement;\n        disabled ? el.setAttribute('disabled', '') : el.removeAttribute('disabled');\n        readonly ? el.setAttribute('readonly', '') : el.removeAttribute('readonly');\n      });\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next(1);\n    this.destroy$.complete();\n  }\n\n  onBlur?: (() => void) | undefined;\n  onFocus?: (() => void) | undefined;\n  onDisable?: ((value: boolean) => void) | undefined;\n  onReadonly?: ((value: boolean) => void) | undefined;\n  onValidate?:\n    | ((\n        value?: DokuFormFieldAccessorValidateValue | undefined,\n        state?: DokuFormFieldAccessorValidateState\n      ) => void)\n    | undefined;\n\n  registerOnFocus(fn: () => void): void {\n    this.onFocus = fn;\n  }\n\n  registerOnBlur(fn: () => void): void {\n    this.onBlur = fn;\n  }\n\n  registerOnDisable(fn: (value: boolean) => void): void {\n    this.onDisable = fn;\n  }\n\n  registerOnReadonly(fn: (value: boolean) => void): void {\n    this.onReadonly = fn;\n  }\n\n  registerOnValidate(fn: (value?: DokuFormFieldAccessorValidateValue | undefined) => void): void {\n    this.onValidate = fn;\n  }\n\n  onClickWrapperElement(): void {\n    this.elementRef.nativeElement.focus();\n  }\n}\n"]}