@js-sugar/angular-bootstrap
Version:
JS-Sugar Bootstrap components for Angular
104 lines • 18.4 kB
JavaScript
import { HostListener } from '@angular/core';
import { Component, Input, ContentChild, Optional, HostBinding, Inject, ElementRef } from '@angular/core';
import { FormControl, NgControl } from '@angular/forms';
import { Locale, LOCALE } from '@js-sugar/angular';
import * as i0 from "@angular/core";
import * as i1 from "@angular/forms";
import * as i2 from "@angular/common";
export class FormFieldComponent {
constructor(_renderer, _form, _localSpecifier) {
this._renderer = _renderer;
this._form = _form;
this._localSpecifier = _localSpecifier;
this.showErrors = true;
this.showRequiredMarker = true;
this.showSubscripts = true;
this._isRequired = false;
}
get firstLocalizedError() { return this._firstLocalizedError; }
get localizedErrors() { return this._localizedErrors; }
get touched() { return this._ngControl.touched; }
get dirty() { return this._ngControl.dirty; }
get isRequired() { return this._isRequired; }
_onFocusOut() { this._validate(); }
ngOnInit() {
if (!this._ngControl) {
throw Error('FormFieldComponent requires a NgControl');
}
if (this._localSpecifier) {
if (this._localSpecifier instanceof Locale) {
this._locale = this._localSpecifier;
}
else {
this._locale = this._localSpecifier.get();
this._localeChangeSubscription = this._localSpecifier.change.subscribe(e => this._locale = e.newLocale);
}
}
}
ngAfterContentInit() {
this._isRequired = this._isRequiredField();
this._statusChangeSubscription = this._ngControl.statusChanges.subscribe(change => {
if (this.dirty || this.touched) {
this._validate();
}
});
}
_validate() {
const invalid = this._ngControl.status === 'INVALID';
if (invalid && this._locale) {
this._localizedErrors = this._locale.translateErrors(this._ngControl.errors);
const firstKey = Object.keys(this._localizedErrors)[0];
this._firstLocalizedError = this._localizedErrors[firstKey];
}
else {
this._localizedErrors = this._firstLocalizedError = undefined;
}
this._renderer.addClass(this._ngControlElement.nativeElement, invalid ? 'is-invalid' : 'is-valid');
this._renderer.removeClass(this._ngControlElement.nativeElement, invalid ? 'is-valid' : 'is-invalid');
}
_isRequiredField() {
const validator = this._ngControl?.validator;
const errors = validator && validator(new FormControl(null));
return errors != null && errors['required'] === true;
}
ngOnDestroy() {
this._localeChangeSubscription?.unsubscribe();
this._statusChangeSubscription?.unsubscribe();
}
}
FormFieldComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.1.1", ngImport: i0, type: FormFieldComponent, deps: [{ token: i0.Renderer2 }, { token: i1.NgForm, optional: true }, { token: LOCALE, optional: true }], target: i0.ɵɵFactoryTarget.Component });
FormFieldComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.1.1", type: FormFieldComponent, selector: "jss-form-field, jss-field, jssb-form-field, jssb-field", inputs: { label: "label", hint: "hint", showErrors: "showErrors", showRequiredMarker: "showRequiredMarker", showSubscripts: "showSubscripts" }, host: { listeners: { "focusout": "_onFocusOut()" }, properties: { "class.jss-required-form-field": "this.isRequired" } }, queries: [{ propertyName: "_ngControl", first: true, predicate: NgControl, descendants: true, static: true }, { propertyName: "_ngControlElement", first: true, predicate: NgControl, descendants: true, read: ElementRef, static: true }], exportAs: ["jssFormField"], ngImport: i0, template: "<label *ngIf=\"label\" class=\"jss-form-field-label\">\r\n {{label}}\r\n <span *ngIf=\"isRequired && showRequiredMarker\" class=\"jss-form-field-required-marker\">*</span>\r\n</label>\r\n\r\n<div class=\"input-group\">\r\n <ng-content></ng-content>\r\n</div>\r\n\r\n<small *ngIf=\"showSubscripts\" class=\"jss-form-field-subscript\">\r\n <span *ngIf=\"showErrors && firstLocalizedError && (touched || dirty); else hintTemplate\" class=\"jss-form-field-error text-danger\">\r\n {{firstLocalizedError.text}}\r\n </span>\r\n <ng-template #hintTemplate>\r\n <span *ngIf=\"hint\" class=\"jss-form-field-hint\">{{hint}}</span>\r\n </ng-template>\r\n</small>\r\n", styles: [":host{display:block}.jss-form-field-subscript{display:block;height:1.5em;line-height:1.5em}\n"], dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.1.1", ngImport: i0, type: FormFieldComponent, decorators: [{
type: Component,
args: [{ selector: 'jss-form-field, jss-field, jssb-form-field, jssb-field', exportAs: 'jssFormField', template: "<label *ngIf=\"label\" class=\"jss-form-field-label\">\r\n {{label}}\r\n <span *ngIf=\"isRequired && showRequiredMarker\" class=\"jss-form-field-required-marker\">*</span>\r\n</label>\r\n\r\n<div class=\"input-group\">\r\n <ng-content></ng-content>\r\n</div>\r\n\r\n<small *ngIf=\"showSubscripts\" class=\"jss-form-field-subscript\">\r\n <span *ngIf=\"showErrors && firstLocalizedError && (touched || dirty); else hintTemplate\" class=\"jss-form-field-error text-danger\">\r\n {{firstLocalizedError.text}}\r\n </span>\r\n <ng-template #hintTemplate>\r\n <span *ngIf=\"hint\" class=\"jss-form-field-hint\">{{hint}}</span>\r\n </ng-template>\r\n</small>\r\n", styles: [":host{display:block}.jss-form-field-subscript{display:block;height:1.5em;line-height:1.5em}\n"] }]
}], ctorParameters: function () { return [{ type: i0.Renderer2 }, { type: i1.NgForm, decorators: [{
type: Optional
}] }, { type: undefined, decorators: [{
type: Optional
}, {
type: Inject,
args: [LOCALE]
}] }]; }, propDecorators: { label: [{
type: Input
}], hint: [{
type: Input
}], showErrors: [{
type: Input
}], showRequiredMarker: [{
type: Input
}], showSubscripts: [{
type: Input
}], _ngControl: [{
type: ContentChild,
args: [NgControl, { static: true }]
}], _ngControlElement: [{
type: ContentChild,
args: [NgControl, { static: true, read: ElementRef }]
}], isRequired: [{
type: HostBinding,
args: ['class.jss-required-form-field']
}], _onFocusOut: [{
type: HostListener,
args: ['focusout']
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-field.component.js","sourceRoot":"","sources":["../../../../../../projects/angular-bootstrap/src/lib/components/form-field/form-field.component.ts","../../../../../../projects/angular-bootstrap/src/lib/components/form-field/form-field.component.html"],"names":[],"mappings":"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","sourcesContent":["import { HostListener } from '@angular/core';\r\nimport { Component, Input, ContentChild, AfterContentInit, Optional, HostBinding, Inject, OnInit, OnDestroy, ElementRef, Renderer2 } from '@angular/core';\r\nimport { FormControl, NgControl, NgForm } from '@angular/forms';\r\nimport { Locale, LocaleProvider, LOCALE, LocalizedValidationError } from '@js-sugar/angular';\r\nimport { Subscription } from 'rxjs';\r\n\r\n@Component({\r\n  selector: 'jss-form-field, jss-field, jssb-form-field, jssb-field',\r\n  exportAs: 'jssFormField',\r\n  templateUrl: './form-field.component.html',\r\n  styleUrls: ['./form-field.component.scss'],\r\n})\r\nexport class FormFieldComponent implements OnInit, AfterContentInit, OnDestroy {\r\n  @Input() label?: string;\r\n  @Input() hint?: string;\r\n  @Input() showErrors = true;\r\n  @Input() showRequiredMarker = true;\r\n  @Input() showSubscripts = true;\r\n  @ContentChild(NgControl, { static: true }) private _ngControl!: NgControl;\r\n  @ContentChild(NgControl, { static: true, read: ElementRef }) private _ngControlElement!: ElementRef;\r\n  private _isRequired = false;\r\n  private _locale?: Locale;\r\n  private _localizedErrors?: { [key: string]: LocalizedValidationError };\r\n  private _firstLocalizedError?: LocalizedValidationError;\r\n  private _statusChangeSubscription?: Subscription;\r\n  private _localeChangeSubscription?: Subscription;\r\n\r\n  /**\r\n   * Returns the first (localized) error of the control \r\n   */\r\n  get firstLocalizedError() { return this._firstLocalizedError; }\r\n\r\n  /**\r\n   * Returns localized validation errors of the control \r\n   * If no locale is provided in the project, returns undefined.\r\n   */\r\n  get localizedErrors() { return this._localizedErrors; }\r\n\r\n  /** Reports whether the control is touched. */\r\n  get touched(): boolean { return this._ngControl.touched!; }\r\n\r\n  /** Reports whether the control is dirty. */\r\n  get dirty(): boolean { return this._ngControl.dirty!; }\r\n\r\n  /** Returns true if this form-field is required, otherwise returns false. */\r\n  @HostBinding('class.jss-required-form-field') get isRequired(): boolean { return this._isRequired; }\r\n\r\n  @HostListener('focusout') private _onFocusOut() { this._validate(); }\r\n\r\n  constructor(\r\n    private _renderer: Renderer2,\r\n    @Optional() private _form: NgForm,\r\n    @Optional() @Inject(LOCALE) private _localSpecifier?: Locale | LocaleProvider,\r\n  ) {\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    if (!this._ngControl) {\r\n      throw Error('FormFieldComponent requires a NgControl');\r\n    }\r\n\r\n    if (this._localSpecifier) {\r\n      if (this._localSpecifier instanceof Locale) {\r\n        this._locale = this._localSpecifier;\r\n      } else {\r\n        this._locale = this._localSpecifier.get();\r\n        this._localeChangeSubscription = this._localSpecifier.change.subscribe(e => this._locale = e.newLocale);\r\n      }\r\n    }\r\n  }\r\n\r\n  ngAfterContentInit(): void {\r\n    this._isRequired = this._isRequiredField();\r\n    this._statusChangeSubscription = this._ngControl.statusChanges!.subscribe(change => {\r\n      if (this.dirty || this.touched) {\r\n        this._validate();\r\n      }\r\n    });\r\n  }\r\n\r\n  private _validate() {\r\n    const invalid = this._ngControl.status === 'INVALID';\r\n    if (invalid && this._locale) {\r\n      this._localizedErrors = this._locale.translateErrors(this._ngControl.errors);\r\n      const firstKey = Object.keys(this._localizedErrors)[0];\r\n      this._firstLocalizedError = this._localizedErrors[firstKey];\r\n    } else {\r\n      this._localizedErrors = this._firstLocalizedError = undefined;\r\n    }\r\n\r\n    this._renderer.addClass(this._ngControlElement.nativeElement, invalid ? 'is-invalid' : 'is-valid');\r\n    this._renderer.removeClass(this._ngControlElement.nativeElement, invalid ? 'is-valid' : 'is-invalid');\r\n  }\r\n\r\n  private _isRequiredField(): boolean {\r\n    const validator = this._ngControl?.validator;\r\n    const errors = validator && validator(new FormControl(null));\r\n    return errors != null && errors['required'] === true;\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this._localeChangeSubscription?.unsubscribe();\r\n    this._statusChangeSubscription?.unsubscribe();\r\n  }\r\n}\r\n","<label *ngIf=\"label\" class=\"jss-form-field-label\">\r\n  {{label}}\r\n  <span *ngIf=\"isRequired && showRequiredMarker\" class=\"jss-form-field-required-marker\">*</span>\r\n</label>\r\n\r\n<div class=\"input-group\">\r\n  <ng-content></ng-content>\r\n</div>\r\n\r\n<small *ngIf=\"showSubscripts\" class=\"jss-form-field-subscript\">\r\n  <span *ngIf=\"showErrors && firstLocalizedError && (touched || dirty); else hintTemplate\" class=\"jss-form-field-error text-danger\">\r\n    {{firstLocalizedError.text}}\r\n  </span>\r\n  <ng-template #hintTemplate>\r\n    <span *ngIf=\"hint\" class=\"jss-form-field-hint\">{{hint}}</span>\r\n  </ng-template>\r\n</small>\r\n"]}