UNPKG

@doku-dev/doku-fragment

Version:

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

185 lines 30 kB
import { CommonModule } from '@angular/common'; import { ChangeDetectionStrategy, Component, EventEmitter, HostBinding, Input, Optional, Output, Self, ViewEncapsulation, } from '@angular/core'; import { FormsModule } from '@angular/forms'; import { ReplaySubject, Subject, delay, distinctUntilChanged, filter, map, pairwise, startWith, takeUntil, tap, } from 'rxjs'; import { DOKU_FORM_FIELD_ACCESSOR, } from '../form-field'; import { DokuInputTime } from './input-time.directive'; import * as i0 from "@angular/core"; import * as i1 from "@angular/forms"; import * as i2 from "@angular/common"; export class DokuTimePicker { constructor(cdr, ngControl) { this.cdr = cdr; this.ngControl = ngControl; this.classes = 'd-time-picker'; /** * Whether to disable the time picker. * @default false */ this.disabled = false; /** * Whether to show the seconds input. * @default false */ this.showSeconds = false; /** * Value of the time picker in ISO String format. * @default '' */ this.value = ''; /** * Minimum time that can be selected. Value in ISO String format. * @default '' */ this.min = ''; /** * Maximum time that can be selected. Value in ISO String format. * @default '' */ this.max = ''; /** * An event emitted when the time changes on blur and condition met. * * Conditions: * - All fields are filled. * - All fields were filled previously, but there is an empty field on the current change. * This case, the value will be an empty string. * - Previous and current values are different. */ this.timeChange = new EventEmitter(); this.hourValue = ''; this.minuteValue = ''; this.secondValue = ''; this.notifyOnBlur$ = new Subject(); this.destroy$ = new ReplaySubject(); this.fieldOptions = { withoutInputStyle: true }; if (this.ngControl) { this.ngControl.valueAccessor = this; } } registerOnDisable(fn) { this.onDisable = fn; } registerOnValidate(fn) { this.onValidate = fn; } writeValue(value) { this.setValueFromISOString(value); this.cdr.detectChanges(); } registerOnChange(fn) { this.onChange = fn; } registerOnTouched(fn) { this.onTouched = fn; } setDisabledState(isDisabled) { this.disabled = isDisabled; this.cdr.detectChanges(); } ngOnInit() { this.notifyOnBlur$ .pipe(tap(() => { this.onTouched?.(); this.setMinMaxTime(this.constructFinalValue()); }), map(() => this.constructFinalValue()), startWith(this.constructFinalValue()), pairwise(), filter(([prev, current]) => (!prev && !!current) || (!!prev && !current) || (!!prev && !!current)), distinctUntilChanged(([, prev], [, current]) => JSON.stringify(prev) === JSON.stringify(current)), takeUntil(this.destroy$)) .subscribe(([, currentValue]) => { this.timeChange.emit(currentValue); this.onChange?.(currentValue); this.onValidate?.(currentValue ? 'valid' : 'invalid'); }); } ngAfterViewInit() { 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?.(); } }); } ngOnChanges(changes) { if (changes['disabled']?.previousValue !== changes['disabled']?.currentValue) { this.onDisable?.(!!this.disabled); } if (changes['value']?.previousValue !== changes['value']?.currentValue) { this.setValueFromISOString(changes['value']?.currentValue); } } ngOnDestroy() { this.destroy$.next(true); this.destroy$.complete(); } constructFinalValue() { if (!this.hourValue || !this.minuteValue) return ''; if (this.showSeconds && !this.secondValue) return ''; const time = this.value ? new Date(this.value) : new Date(); time.setHours(parseInt(this.hourValue)); time.setMinutes(parseInt(this.minuteValue)); time.setSeconds(parseInt(this.showSeconds ? this.secondValue : '0')); time.setMilliseconds(0); return time.toISOString(); } setMinMaxTime(currentTime) { if (this.min && new Date(currentTime) < new Date(this.min)) { this.setValueFromISOString(this.min); } if (this.max && new Date(currentTime) > new Date(this.max)) { this.setValueFromISOString(this.max); } } getTimeFromISODateString(value) { const time = new Date(value || ''); const isValid = time instanceof Date && !isNaN(time.getTime()); if (!isValid) return { hour: '', minute: '', second: '' }; return { hour: time.getHours().toString(), minute: time.getMinutes().toString(), second: time.getSeconds().toString(), }; } setValueFromISOString(value) { const { hour, minute, second } = this.getTimeFromISODateString(value); this.hourValue = this.addPadToValueItem(hour); this.minuteValue = this.addPadToValueItem(minute); this.secondValue = this.addPadToValueItem(second); } addPadToValueItem(value) { return value ? value.padStart(2, '0') : value; } } DokuTimePicker.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuTimePicker, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.NgControl, optional: true, self: true }], target: i0.ɵɵFactoryTarget.Component }); DokuTimePicker.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: DokuTimePicker, isStandalone: true, selector: "doku-time-picker", inputs: { disabled: "disabled", showSeconds: "showSeconds", value: "value", min: "min", max: "max" }, outputs: { timeChange: "timeChange" }, host: { properties: { "class": "this.classes" } }, providers: [{ provide: DOKU_FORM_FIELD_ACCESSOR, useExisting: DokuTimePicker }], exportAs: ["dokuTimePicker"], usesOnChanges: true, ngImport: i0, template: "<div class=\"input-wrapper\">\n <input\n #inputHour\n doku-input-time\n name=\"hour\"\n type=\"text\"\n placeholder=\"HH\"\n maxlength=\"2\"\n [disabled]=\"disabled\"\n [(ngModel)]=\"hourValue\"\n (blur)=\"notifyOnBlur$.next(true)\"\n autocomplete=\"off\"\n />\n</div>\n<div class=\"input-wrapper\">\n <input\n #inputMinute\n doku-input-time\n name=\"minute\"\n type=\"text\"\n placeholder=\"MM\"\n maxlength=\"2\"\n [disabled]=\"disabled\"\n [(ngModel)]=\"minuteValue\"\n (blur)=\"notifyOnBlur$.next(true)\"\n autocomplete=\"off\"\n />\n</div>\n<div class=\"input-wrapper\" *ngIf=\"showSeconds\">\n <input\n #inputSecond\n doku-input-time\n name=\"second\"\n type=\"text\"\n placeholder=\"SS\"\n maxlength=\"2\"\n [disabled]=\"disabled\"\n [(ngModel)]=\"secondValue\"\n (blur)=\"notifyOnBlur$.next(true)\"\n autocomplete=\"off\"\n />\n</div>\n", dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: DokuInputTime, selector: "[doku-input-time]", exportAs: ["dokuInputTime"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.MaxLengthValidator, selector: "[maxlength][formControlName],[maxlength][formControl],[maxlength][ngModel]", inputs: ["maxlength"] }, { kind: "directive", type: i1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuTimePicker, decorators: [{ type: Component, args: [{ selector: 'doku-time-picker', exportAs: 'dokuTimePicker', standalone: true, imports: [CommonModule, DokuInputTime, FormsModule], encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, providers: [{ provide: DOKU_FORM_FIELD_ACCESSOR, useExisting: DokuTimePicker }], template: "<div class=\"input-wrapper\">\n <input\n #inputHour\n doku-input-time\n name=\"hour\"\n type=\"text\"\n placeholder=\"HH\"\n maxlength=\"2\"\n [disabled]=\"disabled\"\n [(ngModel)]=\"hourValue\"\n (blur)=\"notifyOnBlur$.next(true)\"\n autocomplete=\"off\"\n />\n</div>\n<div class=\"input-wrapper\">\n <input\n #inputMinute\n doku-input-time\n name=\"minute\"\n type=\"text\"\n placeholder=\"MM\"\n maxlength=\"2\"\n [disabled]=\"disabled\"\n [(ngModel)]=\"minuteValue\"\n (blur)=\"notifyOnBlur$.next(true)\"\n autocomplete=\"off\"\n />\n</div>\n<div class=\"input-wrapper\" *ngIf=\"showSeconds\">\n <input\n #inputSecond\n doku-input-time\n name=\"second\"\n type=\"text\"\n placeholder=\"SS\"\n maxlength=\"2\"\n [disabled]=\"disabled\"\n [(ngModel)]=\"secondValue\"\n (blur)=\"notifyOnBlur$.next(true)\"\n autocomplete=\"off\"\n />\n</div>\n" }] }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }, { type: i1.NgControl, decorators: [{ type: Optional }, { type: Self }] }]; }, propDecorators: { classes: [{ type: HostBinding, args: ['class'] }], disabled: [{ type: Input }], showSeconds: [{ type: Input }], value: [{ type: Input }], min: [{ type: Input }], max: [{ type: Input }], timeChange: [{ type: Output }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"time-picker.component.js","sourceRoot":"","sources":["../../../../../../projects/doku-fragment/src/lib/time-picker/time-picker.component.ts","../../../../../../projects/doku-fragment/src/lib/time-picker/time-picker.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule, NgClass } from '@angular/common';\nimport {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  HostBinding,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Optional,\n  Output,\n  Self,\n  SimpleChanges,\n  ViewEncapsulation,\n} from '@angular/core';\nimport { ControlValueAccessor, FormsModule, NgControl } from '@angular/forms';\nimport {\n  ReplaySubject,\n  Subject,\n  delay,\n  distinctUntilChanged,\n  filter,\n  map,\n  pairwise,\n  startWith,\n  takeUntil,\n  tap,\n} from 'rxjs';\nimport {\n  DOKU_FORM_FIELD_ACCESSOR,\n  DokuFormFieldAccessor,\n  DokuFormFieldAccessorValidateValue,\n  DokuFormFieldOptions,\n} from '../form-field';\nimport { DokuInputTime } from './input-time.directive';\n\nexport type DokuTimeType = 'hour' | 'minute' | 'second';\n\nexport interface DokuTimePartialOptions {\n  hour?: boolean;\n  minute?: boolean;\n  second?: boolean;\n}\n\nexport type DokuTimePickerValue = string;\n\n@Component({\n  selector: 'doku-time-picker',\n  exportAs: 'dokuTimePicker',\n  standalone: true,\n  imports: [CommonModule, DokuInputTime, FormsModule],\n  templateUrl: './time-picker.component.html',\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [{ provide: DOKU_FORM_FIELD_ACCESSOR, useExisting: DokuTimePicker }],\n})\nexport class DokuTimePicker\n  implements\n    DokuFormFieldAccessor,\n    OnChanges,\n    OnDestroy,\n    OnInit,\n    ControlValueAccessor,\n    AfterViewInit\n{\n  @HostBinding('class')\n  protected readonly classes: NgClass['ngClass'] = 'd-time-picker';\n\n  /**\n   * Whether to disable the time picker.\n   * @default false\n   */\n  @Input() disabled = false;\n\n  /**\n   * Whether to show the seconds input.\n   * @default false\n   */\n  @Input() showSeconds = false;\n\n  /**\n   * Value of the time picker in ISO String format.\n   * @default ''\n   */\n  @Input() value = '';\n\n  /**\n   * Minimum time that can be selected. Value in ISO String format.\n   * @default ''\n   */\n  @Input() min = '';\n\n  /**\n   * Maximum time that can be selected. Value in ISO String format.\n   * @default ''\n   */\n  @Input() max = '';\n\n  /**\n   * An event emitted when the time changes on blur and condition met.\n   *\n   * Conditions:\n   * - All fields are filled.\n   * - All fields were filled previously, but there is an empty field on the current change.\n   * This case, the value will be an empty string.\n   * - Previous and current values are different.\n   */\n  @Output() timeChange = new EventEmitter<DokuTimePickerValue>();\n\n  protected hourValue = '';\n  protected minuteValue = '';\n  protected secondValue = '';\n\n  protected notifyOnBlur$ = new Subject();\n  private destroy$ = new ReplaySubject();\n\n  constructor(private cdr: ChangeDetectorRef, @Optional() @Self() private ngControl: NgControl) {\n    if (this.ngControl) {\n      this.ngControl.valueAccessor = this;\n    }\n  }\n\n  fieldOptions?: DokuFormFieldOptions | undefined = { withoutInputStyle: true };\n  onDisable?: ((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\n  onChange?: (value: string) => void;\n  onTouched?: () => void;\n  writeValue(value: string): void {\n    this.setValueFromISOString(value);\n    this.cdr.detectChanges();\n  }\n  registerOnChange(fn: (value: string) => void): void {\n    this.onChange = fn;\n  }\n  registerOnTouched(fn: () => void): void {\n    this.onTouched = fn;\n  }\n  setDisabledState?(isDisabled: boolean): void {\n    this.disabled = isDisabled;\n    this.cdr.detectChanges();\n  }\n\n  ngOnInit(): void {\n    this.notifyOnBlur$\n      .pipe(\n        tap(() => {\n          this.onTouched?.();\n          this.setMinMaxTime(this.constructFinalValue());\n        }),\n        map(() => this.constructFinalValue()),\n        startWith(this.constructFinalValue()),\n        pairwise(),\n        filter(\n          ([prev, current]) => (!prev && !!current) || (!!prev && !current) || (!!prev && !!current)\n        ),\n        distinctUntilChanged(\n          ([, prev], [, current]) => JSON.stringify(prev) === JSON.stringify(current)\n        ),\n        takeUntil(this.destroy$)\n      )\n      .subscribe(([, currentValue]) => {\n        this.timeChange.emit(currentValue);\n        this.onChange?.(currentValue);\n        this.onValidate?.(currentValue ? 'valid' : 'invalid');\n      });\n  }\n\n  ngAfterViewInit(): void {\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\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['disabled']?.previousValue !== changes['disabled']?.currentValue) {\n      this.onDisable?.(!!this.disabled);\n    }\n\n    if (changes['value']?.previousValue !== changes['value']?.currentValue) {\n      this.setValueFromISOString(changes['value']?.currentValue);\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next(true);\n    this.destroy$.complete();\n  }\n\n  private constructFinalValue(): DokuTimePickerValue {\n    if (!this.hourValue || !this.minuteValue) return '';\n    if (this.showSeconds && !this.secondValue) return '';\n\n    const time = this.value ? new Date(this.value) : new Date();\n    time.setHours(parseInt(this.hourValue));\n    time.setMinutes(parseInt(this.minuteValue));\n    time.setSeconds(parseInt(this.showSeconds ? this.secondValue : '0'));\n    time.setMilliseconds(0);\n    return time.toISOString();\n  }\n\n  private setMinMaxTime(currentTime: string): void {\n    if (this.min && new Date(currentTime) < new Date(this.min)) {\n      this.setValueFromISOString(this.min);\n    }\n\n    if (this.max && new Date(currentTime) > new Date(this.max)) {\n      this.setValueFromISOString(this.max);\n    }\n  }\n\n  private getTimeFromISODateString(value: string | null): {\n    hour: string;\n    minute: string;\n    second: string;\n  } {\n    const time = new Date(value || '');\n    const isValid = time instanceof Date && !isNaN(time.getTime());\n    if (!isValid) return { hour: '', minute: '', second: '' };\n    return {\n      hour: time.getHours().toString(),\n      minute: time.getMinutes().toString(),\n      second: time.getSeconds().toString(),\n    };\n  }\n\n  private setValueFromISOString(value: string | null) {\n    const { hour, minute, second } = this.getTimeFromISODateString(value);\n    this.hourValue = this.addPadToValueItem(hour);\n    this.minuteValue = this.addPadToValueItem(minute);\n    this.secondValue = this.addPadToValueItem(second);\n  }\n\n  private addPadToValueItem(value: string) {\n    return value ? value.padStart(2, '0') : value;\n  }\n}\n","<div class=\"input-wrapper\">\n  <input\n    #inputHour\n    doku-input-time\n    name=\"hour\"\n    type=\"text\"\n    placeholder=\"HH\"\n    maxlength=\"2\"\n    [disabled]=\"disabled\"\n    [(ngModel)]=\"hourValue\"\n    (blur)=\"notifyOnBlur$.next(true)\"\n    autocomplete=\"off\"\n  />\n</div>\n<div class=\"input-wrapper\">\n  <input\n    #inputMinute\n    doku-input-time\n    name=\"minute\"\n    type=\"text\"\n    placeholder=\"MM\"\n    maxlength=\"2\"\n    [disabled]=\"disabled\"\n    [(ngModel)]=\"minuteValue\"\n    (blur)=\"notifyOnBlur$.next(true)\"\n    autocomplete=\"off\"\n  />\n</div>\n<div class=\"input-wrapper\" *ngIf=\"showSeconds\">\n  <input\n    #inputSecond\n    doku-input-time\n    name=\"second\"\n    type=\"text\"\n    placeholder=\"SS\"\n    maxlength=\"2\"\n    [disabled]=\"disabled\"\n    [(ngModel)]=\"secondValue\"\n    (blur)=\"notifyOnBlur$.next(true)\"\n    autocomplete=\"off\"\n  />\n</div>\n"]}