@ui-tool/core
Version:
172 lines • 30.1 kB
JavaScript
import { ChangeDetectionStrategy, ChangeDetectorRef, Component, InjectFlags, Input } from '@angular/core';
import { AbstractControl, NgControl } from '@angular/forms';
import { v4 as uuid } from 'uuid';
import { Subscription } from 'rxjs';
import { VALIDATION_SUMMARIZER_OPTIONS_PROVIDER, VALIDATION_SUMMARIZER_SERVICE } from '../../../constants/injectors/validation-summarizer-injectors';
import * as i0 from "@angular/core";
import * as i1 from "@angular/common";
import * as i2 from "./validation-summarizer-item/validation-summarizer-item.directive";
import * as i3 from "../../../pipes/sanitizers/to-trusted-html/to-trusted-html.pipe";
export class ValidationSummarizerComponent {
//#endregion
//#region Constructor
constructor(injector) {
this.injector = injector;
// tslint:disable-next-line:variable-name
this._maxValidationMessages = 0;
// Service resolve.
this.validationSummarizerService = injector.get(VALIDATION_SUMMARIZER_SERVICE, null, InjectFlags.Optional);
const validationSummarizerOptions = injector.get(VALIDATION_SUMMARIZER_OPTIONS_PROVIDER);
this._changeDetectorRef = injector.get(ChangeDetectorRef);
this._options = validationSummarizerOptions.getOption();
this._groupId = this._options.groupId || uuid();
this._maxValidationMessages = this._options.maximumMessages || 0;
this._visibilityHandler = this._options.visibilityHandler || null;
this.controlLabel = '';
this._control = null;
this.alternativeTemplate = null;
this._subscription = new Subscription();
}
//#endregion
//#region Accessors
// Id of group the validation summarizer belongs to.
// This can be used for identifying whether to apply validation summarizer item template builder or not.
get groupId() {
return this._groupId;
}
// Set item group id.
// tslint:disable-next-line:no-input-rename
set groupId(value) {
if (!value || !value.length) {
return;
}
this._groupId = value;
}
// Instance of the control that needs to be validated.
set ngControl(control) {
this._control = control;
// Unsubscribe subscription.
this._hookStatusChangesSubscription?.unsubscribe();
let statusChangesObservable = null;
if (this._control instanceof AbstractControl) {
statusChangesObservable = this._control.statusChanges;
}
else if (this._control instanceof NgControl) {
statusChangesObservable = this._control.statusChanges;
}
this._hookStatusChangesSubscription = statusChangesObservable?.subscribe(() => {
this._templateContext = this.getTemplateContext();
this._changeDetectorRef.markForCheck();
});
}
// Get the instance of control that needs to be validated.
get ngControl() {
return this._control;
}
// Get template context.
get templateContext() {
return this._templateContext;
}
// Maximum number of validation messages.
get maximumValidationMessages() {
return this._maxValidationMessages;
}
// Maximum number of validation messages.
set maximumValidationMessages(value) {
if (isNaN(value)) {
this._maxValidationMessages = 0;
return;
}
this._maxValidationMessages = value;
}
// tslint:disable-next-line:no-input-rename
set visibilityHandler(value) {
this._visibilityHandler = value;
}
get visibilityHandler() {
return this._visibilityHandler;
}
// Validation summarizer options.
get options() {
return this._options;
}
//#endregion
//#region Life cycle hooks
ngOnInit() {
this._templateContext = this.getTemplateContext();
this._changeDetectorRef.markForCheck();
}
ngOnDestroy() {
this._hookStatusChangesSubscription?.unsubscribe();
this._subscription?.unsubscribe();
}
//#endregion
//#region Methods
ableToDisplayValidationMessages(ngControl) {
if (!ngControl) {
return false;
}
// Visibility handler is defined.
if (this.visibilityHandler) {
return this.visibilityHandler(ngControl);
}
if (!ngControl) {
return false;
}
const ableToDisplay = ngControl.invalid && (ngControl.dirty || ngControl.touched) === true;
return true === ableToDisplay;
}
//#endregion
//#region Internal methods
loadValidationMessages(maximumValidationMessages) {
if (!this.validationSummarizerService || !this.ngControl) {
return [];
}
let messages = this.validationSummarizerService
.loadControlValidationMessages(this.controlLabel, this.ngControl);
if (!messages) {
return [];
}
if (!maximumValidationMessages || isNaN(maximumValidationMessages)) {
return messages;
}
if (maximumValidationMessages < 1) {
return messages;
}
messages = messages.slice(0, maximumValidationMessages);
return messages;
}
// Get validation template context.
getTemplateContext() {
return {
ngControl: this.ngControl,
controlLabel: this.controlLabel,
validationMessages: this.loadValidationMessages(this.maximumValidationMessages)
};
}
}
ValidationSummarizerComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.11", ngImport: i0, type: ValidationSummarizerComponent, deps: [{ token: i0.Injector }], target: i0.ɵɵFactoryTarget.Component });
ValidationSummarizerComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.11", type: ValidationSummarizerComponent, selector: "cms-validation-summarizer", inputs: { groupId: ["group-id", "groupId"], ngControl: ["instance", "ngControl"], controlLabel: ["label", "controlLabel"], alternativeTemplate: ["validation-template", "alternativeTemplate"], maximumValidationMessages: ["maximum-messages", "maximumValidationMessages"], visibilityHandler: ["visibility-handler", "visibilityHandler"] }, ngImport: i0, template: "<!--Only displayed when control is available-->\n<ng-container *ngIf=\"ableToDisplayValidationMessages(ngControl)\"\n [ngSwitch]=\"options.useValidationItemBuilder\">\n\n <!--Use item template builder-->\n <ng-container *ngSwitchCase=\"true\">\n <ng-container *ngIf=\"templateContext.validationMessages && templateContext.validationMessages.length > 0\">\n <ng-template validationSummarizerItem\n [containerId]=\"groupId\"\n [validationMessages]=\"templateContext.validationMessages\"\n [ngControl]=\"ngControl\"\n [controlLabel]=\"controlLabel\"></ng-template>\n </ng-container>\n </ng-container>\n\n\n <!--Using item template-->\n <ng-container *ngSwitchDefault>\n <ng-template\n [ngTemplateOutlet]=\"alternativeTemplate || defaultTemplate\"\n [ngTemplateOutletContext]=\"templateContext\"\n >\n </ng-template>\n </ng-container>\n\n</ng-container>\n\n<!--Default validation summary template-->\n<ng-template\n #defaultTemplate\n let-ngControl=\"ngControl\"\n let-controlLabel=\"controlLabel\"\n let-validationMessages=\"validationMessages\"\n>\n <ul class=\"validation-summary\"\n *ngIf=\"validationMessages && validationMessages.length > 0\">\n <li *ngFor=\"let validationMessage of validationMessages;\"\n [innerHTML]=\"validationMessage.content | toTrustedHtml\">\n </li>\n </ul>\n</ng-template>\n", styles: ["ul.validation-summary{margin-top:5px}ul.validation-summary li{color:red}\n"], directives: [{ type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i1.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { type: i1.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { type: i2.ValidationSummarizerItemDirective, selector: "[validationSummarizerItem]", inputs: ["containerId", "validationMessages", "ngControl", "controlLabel"] }, { type: i1.NgSwitchDefault, selector: "[ngSwitchDefault]" }, { type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], pipes: { "toTrustedHtml": i3.ToTrustedHtmlPipe }, changeDetection: i0.ChangeDetectionStrategy.OnPush });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.11", ngImport: i0, type: ValidationSummarizerComponent, decorators: [{
type: Component,
args: [{ selector: 'cms-validation-summarizer', changeDetection: ChangeDetectionStrategy.OnPush, template: "<!--Only displayed when control is available-->\n<ng-container *ngIf=\"ableToDisplayValidationMessages(ngControl)\"\n [ngSwitch]=\"options.useValidationItemBuilder\">\n\n <!--Use item template builder-->\n <ng-container *ngSwitchCase=\"true\">\n <ng-container *ngIf=\"templateContext.validationMessages && templateContext.validationMessages.length > 0\">\n <ng-template validationSummarizerItem\n [containerId]=\"groupId\"\n [validationMessages]=\"templateContext.validationMessages\"\n [ngControl]=\"ngControl\"\n [controlLabel]=\"controlLabel\"></ng-template>\n </ng-container>\n </ng-container>\n\n\n <!--Using item template-->\n <ng-container *ngSwitchDefault>\n <ng-template\n [ngTemplateOutlet]=\"alternativeTemplate || defaultTemplate\"\n [ngTemplateOutletContext]=\"templateContext\"\n >\n </ng-template>\n </ng-container>\n\n</ng-container>\n\n<!--Default validation summary template-->\n<ng-template\n #defaultTemplate\n let-ngControl=\"ngControl\"\n let-controlLabel=\"controlLabel\"\n let-validationMessages=\"validationMessages\"\n>\n <ul class=\"validation-summary\"\n *ngIf=\"validationMessages && validationMessages.length > 0\">\n <li *ngFor=\"let validationMessage of validationMessages;\"\n [innerHTML]=\"validationMessage.content | toTrustedHtml\">\n </li>\n </ul>\n</ng-template>\n", styles: ["ul.validation-summary{margin-top:5px}ul.validation-summary li{color:red}\n"] }]
}], ctorParameters: function () { return [{ type: i0.Injector }]; }, propDecorators: { groupId: [{
type: Input,
args: ['group-id']
}], ngControl: [{
type: Input,
args: ['instance']
}], controlLabel: [{
type: Input,
args: ['label']
}], alternativeTemplate: [{
type: Input,
args: ['validation-template']
}], maximumValidationMessages: [{
type: Input,
args: ['maximum-messages']
}], visibilityHandler: [{
type: Input,
args: ['visibility-handler']
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"validation-summarizer.component.js","sourceRoot":"","sources":["../../../../../../../../libs/core/src/modules/validator/validation-summarizer/validation-summarizer.component.ts","../../../../../../../../libs/core/src/modules/validator/validation-summarizer/validation-summarizer.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  InjectFlags,\n  Injector,\n  Input,\n  OnDestroy,\n  OnInit,\n  TemplateRef\n} from '@angular/core';\nimport {AbstractControl, NgControl} from '@angular/forms';\nimport {IValidationSummarizerService} from '../../../services/interfaces/validation-summarizers/validation-summarizer-service.interface';\nimport {ValidationMessage} from '../../../models/implementations/validation-summarizers/validation-message';\nimport {IValidationSummarizerOptions} from '../../../models/interfaces/validation-summarizers/validation-summarizer-options.interface';\nimport {v4 as uuid} from 'uuid';\nimport {IValidationSummarizerModuleOptions} from '../../../models/interfaces/validation-summarizers/validation-summarizer-module-options.interface';\nimport {Observable, Subscription} from 'rxjs';\nimport {VALIDATION_SUMMARIZER_OPTIONS_PROVIDER, VALIDATION_SUMMARIZER_SERVICE} from '../../../constants/injectors/validation-summarizer-injectors';\n\n\n@Component({\n  // tslint:disable-next-line:component-selector\n  selector: 'cms-validation-summarizer',\n  templateUrl: 'validation-summarizer.component.html',\n  styleUrls: ['validation-summarizer.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class ValidationSummarizerComponent implements OnInit, OnDestroy {\n\n  //#region Properties\n\n  // Context of template.\n  // tslint:disable-next-line:variable-name\n  private _templateContext: any;\n\n  // Component id.\n  // tslint:disable-next-line:variable-name\n  protected _groupId: string;\n\n  // tslint:disable-next-line:variable-name\n  protected _control: AbstractControl | NgControl | null | undefined;\n\n  // tslint:disable-next-line:variable-name\n  protected _maxValidationMessages = 0;\n\n  // Validation summarizer options.\n  // tslint:disable-next-line:variable-name\n  protected _options: IValidationSummarizerModuleOptions;\n\n  // Service for validating controls.\n  protected validationSummarizerService: IValidationSummarizerService | null;\n\n  // Handler for handling summarizer visibility.\n  // tslint:disable-next-line:variable-name\n  protected _visibilityHandler: ((ngControl: AbstractControl | NgControl) => boolean) | null;\n\n  // Hook status changes subscription\n  // tslint:disable-next-line:variable-name\n  private _hookStatusChangesSubscription: Subscription | undefined;\n\n  // For marking component as changed.\n  // tslint:disable-next-line:variable-name\n  private _changeDetectorRef: ChangeDetectorRef;\n\n  // Subscription watch list.\n  private readonly _subscription: Subscription;\n\n  //#endregion\n\n  //#region Accessors\n\n  // Id of group the validation summarizer belongs to.\n  // This can be used for identifying whether to apply validation summarizer item template builder or not.\n  public get groupId(): string {\n    return this._groupId;\n  }\n\n  // Set item group id.\n  // tslint:disable-next-line:no-input-rename\n  @Input('group-id')\n  public set groupId(value: string) {\n\n    if (!value || !value.length) {\n      return;\n    }\n\n    this._groupId = value;\n  }\n\n  // Instance of the control that needs to be validated.\n  @Input('instance')\n  public set ngControl(control: AbstractControl | NgControl | null | undefined) {\n    this._control = control;\n\n    // Unsubscribe subscription.\n    this._hookStatusChangesSubscription?.unsubscribe();\n\n    let statusChangesObservable: Observable<any> | null = null;\n    if (this._control instanceof AbstractControl) {\n      statusChangesObservable = (this._control as AbstractControl).statusChanges;\n    } else if (this._control instanceof NgControl) {\n      statusChangesObservable = (this._control as NgControl).statusChanges;\n    }\n\n    this._hookStatusChangesSubscription = statusChangesObservable?.subscribe(() => {\n      this._templateContext = this.getTemplateContext();\n      this._changeDetectorRef.markForCheck();\n    });\n  }\n\n  // Get the instance of control that needs to be validated.\n  public get ngControl(): AbstractControl | NgControl | null | undefined {\n    return this._control;\n  }\n\n  // Label of control.\n  // tslint:disable-next-line:no-input-rename\n  @Input('label')\n  public controlLabel: string;\n\n  // Alternative template for validation summary.\n  // tslint:disable-next-line:no-input-rename\n  @Input('validation-template')\n  public alternativeTemplate: TemplateRef<any> | null;\n\n  // Get template context.\n  public get templateContext(): any {\n    return this._templateContext;\n  }\n\n  // Maximum number of validation messages.\n  public get maximumValidationMessages(): number {\n    return this._maxValidationMessages;\n  }\n\n  // Maximum number of validation messages.\n  @Input('maximum-messages')\n  public set maximumValidationMessages(value: number) {\n    if (isNaN(value)) {\n      this._maxValidationMessages = 0;\n      return;\n    }\n\n    this._maxValidationMessages = value;\n  }\n\n  // tslint:disable-next-line:no-input-rename\n  @Input('visibility-handler')\n  public set visibilityHandler(value: ((ngControl: AbstractControl | NgControl) => boolean) | null) {\n    this._visibilityHandler = value;\n  }\n\n  public get visibilityHandler(): ((ngControl: AbstractControl | NgControl) => boolean) | null {\n    return this._visibilityHandler;\n  }\n\n  // Validation summarizer options.\n  public get options(): IValidationSummarizerOptions {\n    return this._options;\n  }\n\n  //#endregion\n\n  //#region Constructor\n\n  public constructor(protected injector: Injector) {\n\n    // Service resolve.\n    this.validationSummarizerService = injector.get(VALIDATION_SUMMARIZER_SERVICE,\n      null, InjectFlags.Optional);\n\n    const validationSummarizerOptions = injector.get(VALIDATION_SUMMARIZER_OPTIONS_PROVIDER);\n    this._changeDetectorRef = injector.get(ChangeDetectorRef);\n\n    this._options = validationSummarizerOptions.getOption();\n\n    this._groupId = this._options.groupId || uuid();\n    this._maxValidationMessages = this._options.maximumMessages || 0;\n    this._visibilityHandler = this._options.visibilityHandler || null;\n\n    this.controlLabel = '';\n    this._control = null;\n    this.alternativeTemplate = null;\n    this._subscription = new Subscription();\n  }\n\n  //#endregion\n\n  //#region Life cycle hooks\n\n  public ngOnInit(): void {\n    this._templateContext = this.getTemplateContext();\n    this._changeDetectorRef.markForCheck();\n  }\n\n  public ngOnDestroy(): void {\n    this._hookStatusChangesSubscription?.unsubscribe();\n    this._subscription?.unsubscribe();\n  }\n\n  //#endregion\n\n\n  //#region Methods\n\n  public ableToDisplayValidationMessages(ngControl: AbstractControl | NgControl | null | undefined): boolean {\n\n    if (!ngControl) {\n      return false;\n    }\n\n    // Visibility handler is defined.\n    if (this.visibilityHandler) {\n      return this.visibilityHandler(ngControl);\n    }\n\n    if (!ngControl) {\n      return false;\n    }\n\n    const ableToDisplay = ngControl.invalid && (ngControl.dirty || ngControl.touched) === true;\n    return true === ableToDisplay;\n  }\n\n  //#endregion\n\n  //#region Internal methods\n\n  protected loadValidationMessages(maximumValidationMessages: number | null): ValidationMessage[] {\n\n    if (!this.validationSummarizerService || !this.ngControl) {\n      return [];\n    }\n\n    let messages = this.validationSummarizerService\n      .loadControlValidationMessages(this.controlLabel, this.ngControl);\n\n    if (!messages) {\n      return [];\n    }\n\n    if (!maximumValidationMessages || isNaN(maximumValidationMessages)) {\n      return messages;\n    }\n\n    if (maximumValidationMessages < 1) {\n      return messages;\n    }\n\n    messages = messages.slice(0, maximumValidationMessages);\n    return messages;\n  }\n\n  // Get validation template context.\n  private getTemplateContext(): any {\n    return {\n      ngControl: this.ngControl,\n      controlLabel: this.controlLabel,\n      validationMessages: this.loadValidationMessages(this.maximumValidationMessages)\n    };\n  }\n\n  //#endregion\n}\n","<!--Only displayed when control is available-->\n<ng-container *ngIf=\"ableToDisplayValidationMessages(ngControl)\"\n              [ngSwitch]=\"options.useValidationItemBuilder\">\n\n  <!--Use item template builder-->\n  <ng-container *ngSwitchCase=\"true\">\n    <ng-container *ngIf=\"templateContext.validationMessages && templateContext.validationMessages.length > 0\">\n      <ng-template validationSummarizerItem\n                   [containerId]=\"groupId\"\n                   [validationMessages]=\"templateContext.validationMessages\"\n                   [ngControl]=\"ngControl\"\n                   [controlLabel]=\"controlLabel\"></ng-template>\n    </ng-container>\n  </ng-container>\n\n\n  <!--Using item template-->\n  <ng-container *ngSwitchDefault>\n    <ng-template\n      [ngTemplateOutlet]=\"alternativeTemplate || defaultTemplate\"\n      [ngTemplateOutletContext]=\"templateContext\"\n    >\n    </ng-template>\n  </ng-container>\n\n</ng-container>\n\n<!--Default validation summary template-->\n<ng-template\n  #defaultTemplate\n  let-ngControl=\"ngControl\"\n  let-controlLabel=\"controlLabel\"\n  let-validationMessages=\"validationMessages\"\n>\n  <ul class=\"validation-summary\"\n      *ngIf=\"validationMessages && validationMessages.length > 0\">\n    <li *ngFor=\"let validationMessage of validationMessages;\"\n        [innerHTML]=\"validationMessage.content | toTrustedHtml\">\n    </li>\n  </ul>\n</ng-template>\n"]}