UNPKG

@ui-tool/core

Version:
263 lines 41.2 kB
import { ChangeDetectionStrategy, Component, ContentChildren, Inject, Input } from '@angular/core'; import { AbstractControl, NgControl } from '@angular/forms'; import { Subscription } from 'rxjs'; import { MULTIPLE_VALIDATION_SUMMARIZER_OPTIONS, MULTIPLE_VALIDATION_SUMMARIZER_OPTIONS_PROVIDER, MULTIPLE_VALIDATION_SUMMARIZER_SERVICE } from '../../../constants/injectors/multiple-validation-summarizer-injectors'; import { VALIDATION_SUMMARIZER_OPTIONS_PROVIDER, VALIDATION_SUMMARIZER_SERVICE } from '../../../constants/injectors/validation-summarizer-injectors'; import { v4 as uuid } from 'uuid'; import { MultipleValidationSummarizerItemContextDirective } from './directives/multiple-validation-summarizer-item-context.directive'; import { VALIDATION_SUMMARIZER_OPTIONS } from '../../../constants/injectors/internal-injectors'; import * as i0 from "@angular/core"; import * as i1 from "../validation-summarizer/validation-summarizer.component"; import * as i2 from "@angular/common"; const basicValidationHandler = (ngControl) => { if (!ngControl) { return false; } return (ngControl.invalid && (ngControl.dirty || ngControl.touched)) || false; }; export class MultipleValidationSummarizerComponent { //#endregion //#region Constructor constructor(validationSummarizerService, optionsProvider, changeDetectorRef) { this.validationSummarizerService = validationSummarizerService; this.optionsProvider = optionsProvider; this.changeDetectorRef = changeDetectorRef; // Id to control status change subscription. this._idToControlStatusChangeSubscription = {}; // Subscription about validation context changes. this._validationContextChangesSubscription = null; // Context directive watch list. this.itemContexts = null; const options = this.optionsProvider.getOption(); this._groupId = options?.groupId || uuid(); this._idToLabel = {}; this._idToTemplate = {}; this._idToInstance = {}; this._idToValidationError = {}; this._visibilityHandler = options.visibilityHandler || basicValidationHandler; this._hasInvalidField = false; this._subscription = new Subscription(); } //#endregion //#region Accessors // Id to label mapping. get idToLabel() { return this._idToLabel; } // Id to template mapping. get idToTemplate() { return this._idToTemplate; } get idToInstance() { return this._idToInstance; } get groupId() { return this._groupId; } // Id of group the multiple validation summarizer belongs to. set groupId(value) { this._groupId = value; } // Whether there is any invalid field or not. get hasInvalidField() { return this._hasInvalidField; } // tslint:disable-next-line:no-input-rename set visibilityHandler(value) { if (!value) { this._visibilityHandler = basicValidationHandler; return; } this._visibilityHandler = value; } get visibilityHandler() { return this._visibilityHandler; } //#endregion //#region Life cycle hooks ngOnInit() { } // Called when content has been initialized. ngAfterContentInit() { // Handle control initial changes. this.handleControlChanges(); this.changeDetectorRef.markForCheck(); // Hook control changes. const hookControlChangesSubscription = this.itemContexts ?.changes .subscribe(() => { this.handleControlChanges(); this.changeDetectorRef.markForCheck(); }); this._subscription.add(hookControlChangesSubscription); } // Called when component is destroyed. ngOnDestroy() { this.deleteControlStatusChangeSubscriptions(); this._validationContextChangesSubscription?.unsubscribe(); this._subscription?.unsubscribe(); } //#endregion //#region Methods // Whether control has validation error or not. shouldControlHasValidationError(id) { if (!id || !this._idToValidationError) { return false; } const validationError = this._idToValidationError[id]; return validationError !== null && validationError !== undefined; } //#endregion //#region Internal methods handleControlChanges() { // Unsubscribe the previous control status. this._validationContextChangesSubscription?.unsubscribe(); // Clear the mapping. this._idToLabel = {}; this._idToTemplate = {}; this._hasInvalidField = false; this._idToValidationError = {}; // Invalid item collection. if (!this.itemContexts || !this.itemContexts.length) { return; } // Handle control changes. for (const itemContext of this.itemContexts) { this.handleContextChangedEvent(itemContext, { name: 'label', value: itemContext.label }); this.handleContextChangedEvent(itemContext, { name: 'template', value: itemContext.template }); this.handleContextChangedEvent(itemContext, { name: 'instance', value: itemContext.instance }); const validationContextChangeSubscription = itemContext.updateEvent .subscribe((context) => { this.handleContextChangedEvent(itemContext, context, true); }); this._validationContextChangesSubscription?.add(validationContextChangeSubscription); } // Check whether there is any invalid field or not. this._hasInvalidField = this.shouldAnyFieldInvalid(); this.changeDetectorRef.markForCheck(); } // Handle context change event. handleContextChangedEvent(itemContext, context, markAsChanged) { switch (context.name) { case 'label': this._idToLabel[itemContext.id] = context.value; break; case 'template': this._idToTemplate[itemContext.id] = context.value; break; case 'instance': // Unsubscribe this instance subscription. let controlChangesSubscription = this._idToControlStatusChangeSubscription[itemContext.id]; controlChangesSubscription?.unsubscribe(); delete this._idToInstance[itemContext.id]; let valueChangesObservable = null; if (itemContext.instance instanceof AbstractControl) { valueChangesObservable = itemContext.instance.statusChanges; } else if (itemContext.instance instanceof NgControl) { valueChangesObservable = itemContext.instance.statusChanges; } if (valueChangesObservable) { this._idToInstance[itemContext.id] = context.value; controlChangesSubscription = valueChangesObservable .subscribe(() => { const instance = this._idToInstance[itemContext.id]; if (!instance) { return; } this._hasInvalidField = this.shouldAnyFieldInvalid(); this.changeDetectorRef.markForCheck(); }); this._subscription.add(controlChangesSubscription); } break; } if (markAsChanged) { this.changeDetectorRef.markForCheck(); } } // Remove control status change subscription. deleteControlStatusChangeSubscriptions() { const ids = Object.keys(this._idToControlStatusChangeSubscription); for (const id of ids) { const controlStatusChangeSubscription = this._idToControlStatusChangeSubscription[id]; controlStatusChangeSubscription?.unsubscribe(); } this._idToControlStatusChangeSubscription = {}; } // Whether there is any invalid field or not. shouldAnyFieldInvalid() { const ids = Object.keys(this._idToInstance); if (!ids || !ids.length) { return false; } let hasValidationError = false; for (const id of ids) { const instance = this.idToInstance[id]; if (!instance) { continue; } const validationErrors = this.validationSummarizerService.loadControlValidationErrors(instance); if (!validationErrors) { continue; } this._idToValidationError[id] = validationErrors; const shouldValidationErrorVisible = this.visibilityHandler ? this._visibilityHandler(instance) : false; if (!shouldValidationErrorVisible) { continue; } hasValidationError = true; } return hasValidationError; } } MultipleValidationSummarizerComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.11", ngImport: i0, type: MultipleValidationSummarizerComponent, deps: [{ token: MULTIPLE_VALIDATION_SUMMARIZER_SERVICE }, { token: MULTIPLE_VALIDATION_SUMMARIZER_OPTIONS_PROVIDER }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); MultipleValidationSummarizerComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.11", type: MultipleValidationSummarizerComponent, selector: "cms-multiple-validation-summarizer", inputs: { groupId: ["group-id", "groupId"], visibilityHandler: ["visibility-handler", "visibilityHandler"] }, providers: [ { provide: VALIDATION_SUMMARIZER_SERVICE, useExisting: MULTIPLE_VALIDATION_SUMMARIZER_SERVICE }, { provide: VALIDATION_SUMMARIZER_OPTIONS, useExisting: MULTIPLE_VALIDATION_SUMMARIZER_OPTIONS }, { provide: VALIDATION_SUMMARIZER_OPTIONS_PROVIDER, useExisting: MULTIPLE_VALIDATION_SUMMARIZER_OPTIONS_PROVIDER, multi: false } ], queries: [{ propertyName: "itemContexts", predicate: MultipleValidationSummarizerItemContextDirective }], ngImport: i0, template: "<ul *ngIf=\"hasInvalidField\">\n <ng-container *ngFor=\"let itemContext of itemContexts\">\n <cms-validation-summarizer *ngIf=\"idToInstance[itemContext.id] && shouldControlHasValidationError(itemContext.id)\"\n [instance]=\"idToInstance[itemContext.id]\"\n [label]=\"idToLabel[itemContext.id]\"\n [validation-template]=\"idToTemplate[itemContext.id] || validationSummarizerTemplate\">\n </cms-validation-summarizer>\n </ng-container>\n</ul>\n\n\n<!--Default validation summary template-->\n<ng-template\n #validationSummarizerTemplate\n let-ngControl=\"ngControl\"\n let-controlLabel=\"controlLabel\"\n let-validationMessages=\"validationMessages\"\n>\n <li *ngFor=\"let validationMessage of validationMessages;\">\n {{controlLabel}} {{validationMessage | json}}\n </li>\n</ng-template>\n", components: [{ type: i1.ValidationSummarizerComponent, selector: "cms-validation-summarizer", inputs: ["group-id", "instance", "label", "validation-template", "maximum-messages", "visibility-handler"] }], directives: [{ type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], pipes: { "json": i2.JsonPipe }, changeDetection: i0.ChangeDetectionStrategy.OnPush }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.11", ngImport: i0, type: MultipleValidationSummarizerComponent, decorators: [{ type: Component, args: [{ selector: 'cms-multiple-validation-summarizer', changeDetection: ChangeDetectionStrategy.OnPush, providers: [ { provide: VALIDATION_SUMMARIZER_SERVICE, useExisting: MULTIPLE_VALIDATION_SUMMARIZER_SERVICE }, { provide: VALIDATION_SUMMARIZER_OPTIONS, useExisting: MULTIPLE_VALIDATION_SUMMARIZER_OPTIONS }, { provide: VALIDATION_SUMMARIZER_OPTIONS_PROVIDER, useExisting: MULTIPLE_VALIDATION_SUMMARIZER_OPTIONS_PROVIDER, multi: false } ], template: "<ul *ngIf=\"hasInvalidField\">\n <ng-container *ngFor=\"let itemContext of itemContexts\">\n <cms-validation-summarizer *ngIf=\"idToInstance[itemContext.id] && shouldControlHasValidationError(itemContext.id)\"\n [instance]=\"idToInstance[itemContext.id]\"\n [label]=\"idToLabel[itemContext.id]\"\n [validation-template]=\"idToTemplate[itemContext.id] || validationSummarizerTemplate\">\n </cms-validation-summarizer>\n </ng-container>\n</ul>\n\n\n<!--Default validation summary template-->\n<ng-template\n #validationSummarizerTemplate\n let-ngControl=\"ngControl\"\n let-controlLabel=\"controlLabel\"\n let-validationMessages=\"validationMessages\"\n>\n <li *ngFor=\"let validationMessage of validationMessages;\">\n {{controlLabel}} {{validationMessage | json}}\n </li>\n</ng-template>\n" }] }], ctorParameters: function () { return [{ type: undefined, decorators: [{ type: Inject, args: [MULTIPLE_VALIDATION_SUMMARIZER_SERVICE] }] }, { type: undefined, decorators: [{ type: Inject, args: [MULTIPLE_VALIDATION_SUMMARIZER_OPTIONS_PROVIDER] }] }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { itemContexts: [{ type: ContentChildren, args: [MultipleValidationSummarizerItemContextDirective] }], groupId: [{ type: Input, args: ['group-id'] }], visibilityHandler: [{ type: Input, args: ['visibility-handler'] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"multiple-validation-summarizer.component.js","sourceRoot":"","sources":["../../../../../../../../libs/core/src/modules/validator/multiple-validation-summarizer/multiple-validation-summarizer.component.ts","../../../../../../../../libs/core/src/modules/validator/multiple-validation-summarizer/multiple-validation-summarizer.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterContentInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ContentChildren,\n  Inject,\n  Input,\n  OnDestroy,\n  OnInit,\n  QueryList,\n  TemplateRef\n} from '@angular/core';\nimport {AbstractControl, FormControlDirective, FormGroup, NgControl, ValidationErrors} from '@angular/forms';\nimport {Observable, Subscription} from 'rxjs';\nimport {\n  MULTIPLE_VALIDATION_SUMMARIZER_OPTIONS,\n  MULTIPLE_VALIDATION_SUMMARIZER_OPTIONS_PROVIDER,\n  MULTIPLE_VALIDATION_SUMMARIZER_SERVICE\n} from '../../../constants/injectors/multiple-validation-summarizer-injectors';\nimport {VALIDATION_SUMMARIZER_OPTIONS_PROVIDER,\n  VALIDATION_SUMMARIZER_SERVICE} from '../../../constants/injectors/validation-summarizer-injectors';\nimport {MULTIPLE_VALIDATION_SUMMARIZER_CONTEXT_CHANGED_EVENT} from '../../../constants/multiple-validation-summarizer-constants';\nimport {IValidationSummarizerService} from '../../../services/interfaces/validation-summarizers/validation-summarizer-service.interface';\nimport {v4 as uuid} from 'uuid';\nimport {MultipleValidationSummarizerItemContextDirective} from './directives/multiple-validation-summarizer-item-context.directive';\nimport {MultipleValidationSummarizerItemContext} from '../../../models/interfaces/multiple-validation-summarizers/multiple-validation-summarizer-item-context';\nimport {IValidationSummarizerOptionProvider} from '../../../providers/interfaces/validation-summarizer-options-provider.interface';\nimport {VALIDATION_SUMMARIZER_OPTIONS} from '../../../constants/injectors/internal-injectors';\n\nconst basicValidationHandler = (ngControl: AbstractControl | NgControl): boolean => {\n  if (!ngControl) {\n    return false;\n  }\n\n  return (ngControl.invalid && (ngControl.dirty || ngControl.touched)) || false;\n};\n\n@Component({\n  selector: 'cms-multiple-validation-summarizer',\n  templateUrl: 'multiple-validation-summarizer.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [\n    {\n      provide: VALIDATION_SUMMARIZER_SERVICE,\n      useExisting: MULTIPLE_VALIDATION_SUMMARIZER_SERVICE\n    },\n    {\n      provide: VALIDATION_SUMMARIZER_OPTIONS,\n      useExisting: MULTIPLE_VALIDATION_SUMMARIZER_OPTIONS\n    },\n    {\n      provide: VALIDATION_SUMMARIZER_OPTIONS_PROVIDER,\n      useExisting: MULTIPLE_VALIDATION_SUMMARIZER_OPTIONS_PROVIDER,\n      multi: false\n    }\n  ]\n})\nexport class MultipleValidationSummarizerComponent implements OnInit, AfterContentInit, OnDestroy {\n\n  //#region Properties\n\n  // Group id.\n  private _groupId: string;\n\n  // Id to label.\n  private _idToLabel: { [id: string]: string };\n\n  // Id to template.\n  private _idToTemplate: { [id: string]: TemplateRef<MultipleValidationSummarizerItemContext> | null };\n\n  // Id to instance.\n  private _idToInstance: { [id: string]: AbstractControl | FormGroup | FormControlDirective | null };\n\n  // Id to validation errors.\n  private _idToValidationError: { [id: string]: ValidationErrors };\n\n  // Handler for handling summarizer visibility.\n  // tslint:disable-next-line:variable-name\n  private _visibilityHandler: ((ngControl: AbstractControl | NgControl) => boolean);\n\n  // Whether there is any invalid field or not.\n  private _hasInvalidField: boolean;\n\n  // Id to control status change subscription.\n  private _idToControlStatusChangeSubscription: { [id: string]: Subscription | null } = {};\n\n  // Subscription about validation context changes.\n  private _validationContextChangesSubscription: Subscription | null = null;\n\n  // Subscription watch list.\n  private _subscription: Subscription;\n\n  // Context directive watch list.\n  @ContentChildren(MultipleValidationSummarizerItemContextDirective)\n  public itemContexts: QueryList<MultipleValidationSummarizerItemContextDirective> | null = null;\n\n  //#endregion\n\n  //#region Accessors\n\n  // Id to label mapping.\n  public get idToLabel(): { [id: string]: string } {\n    return this._idToLabel;\n  }\n\n  // Id to template mapping.\n  public get idToTemplate(): { [id: string]: TemplateRef<MultipleValidationSummarizerItemContext> | null } {\n    return this._idToTemplate;\n  }\n\n  public get idToInstance(): { [id: string]: AbstractControl | FormGroup | FormControlDirective | null } {\n    return this._idToInstance;\n  }\n\n  public get groupId(): string {\n    return this._groupId;\n  }\n\n  // Id of group the multiple validation summarizer belongs to.\n  @Input('group-id')\n  public set groupId(value: string) {\n    this._groupId = value;\n  }\n\n  // Whether there is any invalid field or not.\n  public get hasInvalidField(): boolean {\n    return this._hasInvalidField;\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    if (!value) {\n      this._visibilityHandler = basicValidationHandler;\n      return;\n    }\n    this._visibilityHandler = value;\n  }\n\n  public get visibilityHandler(): (((ngControl: AbstractControl | NgControl) => boolean) | null) {\n    return this._visibilityHandler;\n  }\n\n  //#endregion\n\n  //#region Constructor\n\n  public constructor(@Inject(MULTIPLE_VALIDATION_SUMMARIZER_SERVICE)\n                     protected readonly validationSummarizerService: IValidationSummarizerService,\n                     @Inject(MULTIPLE_VALIDATION_SUMMARIZER_OPTIONS_PROVIDER)\n                     protected readonly optionsProvider: IValidationSummarizerOptionProvider,\n                     protected readonly changeDetectorRef: ChangeDetectorRef) {\n\n    const options = this.optionsProvider.getOption();\n    this._groupId = options?.groupId || uuid();\n    this._idToLabel = {};\n    this._idToTemplate = {};\n    this._idToInstance = {};\n    this._idToValidationError = {};\n    this._visibilityHandler = options.visibilityHandler || basicValidationHandler;\n    this._hasInvalidField = false;\n    this._subscription = new Subscription();\n  }\n\n  //#endregion\n\n  //#region Life cycle hooks\n\n  public ngOnInit(): void {\n  }\n\n  // Called when content has been initialized.\n  public ngAfterContentInit(): void {\n    // Handle control initial changes.\n    this.handleControlChanges();\n    this.changeDetectorRef.markForCheck();\n\n    // Hook control changes.\n    const hookControlChangesSubscription = this.itemContexts\n      ?.changes\n      .subscribe(() => {\n        this.handleControlChanges();\n        this.changeDetectorRef.markForCheck();\n      });\n    this._subscription.add(hookControlChangesSubscription);\n  }\n\n  // Called when component is destroyed.\n  public ngOnDestroy(): void {\n    this.deleteControlStatusChangeSubscriptions();\n    this._validationContextChangesSubscription?.unsubscribe();\n    this._subscription?.unsubscribe();\n  }\n\n  //#endregion\n\n  //#region Methods\n\n  // Whether control has validation error or not.\n  public shouldControlHasValidationError(id: string): boolean {\n    if (!id || !this._idToValidationError) {\n      return false;\n    }\n\n    const validationError = this._idToValidationError[id];\n    return validationError !== null && validationError !== undefined;\n  }\n\n  //#endregion\n\n  //#region Internal methods\n\n  private handleControlChanges(): void {\n\n    // Unsubscribe the previous control status.\n    this._validationContextChangesSubscription?.unsubscribe();\n\n    // Clear the mapping.\n    this._idToLabel = {};\n    this._idToTemplate = {};\n    this._hasInvalidField = false;\n    this._idToValidationError = {};\n\n    // Invalid item collection.\n    if (!this.itemContexts || !this.itemContexts.length) {\n      return;\n    }\n\n    // Handle control changes.\n    for (const itemContext of this.itemContexts) {\n      this.handleContextChangedEvent(itemContext, {name: 'label', value: itemContext.label});\n      this.handleContextChangedEvent(itemContext, {name: 'template', value: itemContext.template});\n      this.handleContextChangedEvent(itemContext, {name: 'instance', value: itemContext.instance});\n\n      const validationContextChangeSubscription = itemContext.updateEvent\n        .subscribe(\n          (context: MULTIPLE_VALIDATION_SUMMARIZER_CONTEXT_CHANGED_EVENT) => {\n            this.handleContextChangedEvent(itemContext, context, true);\n          });\n      this._validationContextChangesSubscription?.add(validationContextChangeSubscription);\n    }\n\n    // Check whether there is any invalid field or not.\n    this._hasInvalidField = this.shouldAnyFieldInvalid();\n    this.changeDetectorRef.markForCheck();\n  }\n\n  // Handle context change event.\n  private handleContextChangedEvent(itemContext: MultipleValidationSummarizerItemContextDirective,\n                                    context: MULTIPLE_VALIDATION_SUMMARIZER_CONTEXT_CHANGED_EVENT,\n                                    markAsChanged?: boolean): void {\n    switch (context.name) {\n      case 'label':\n        this._idToLabel[itemContext.id] = context.value;\n        break;\n\n      case 'template':\n        this._idToTemplate[itemContext.id] = context.value;\n        break;\n\n      case 'instance':\n\n        // Unsubscribe this instance subscription.\n        let controlChangesSubscription = this._idToControlStatusChangeSubscription[itemContext.id];\n        controlChangesSubscription?.unsubscribe();\n        delete this._idToInstance[itemContext.id];\n\n        let valueChangesObservable: Observable<any> | null = null;\n        if (itemContext.instance instanceof AbstractControl) {\n          valueChangesObservable = (itemContext.instance as AbstractControl).statusChanges;\n        } else if (itemContext.instance instanceof NgControl) {\n          valueChangesObservable = (itemContext.instance as NgControl).statusChanges;\n        }\n\n        if (valueChangesObservable) {\n          this._idToInstance[itemContext.id] = context.value;\n          controlChangesSubscription = valueChangesObservable\n            .subscribe(() => {\n              const instance = this._idToInstance[itemContext.id];\n              if (!instance) {\n                return;\n              }\n\n              this._hasInvalidField = this.shouldAnyFieldInvalid();\n              this.changeDetectorRef.markForCheck();\n            });\n\n          this._subscription.add(controlChangesSubscription);\n        }\n\n        break;\n    }\n\n    if (markAsChanged) {\n      this.changeDetectorRef.markForCheck();\n    }\n  }\n\n  // Remove control status change subscription.\n  private deleteControlStatusChangeSubscriptions(): void {\n    const ids = Object.keys(this._idToControlStatusChangeSubscription);\n    for (const id of ids) {\n      const controlStatusChangeSubscription = this._idToControlStatusChangeSubscription[id];\n      controlStatusChangeSubscription?.unsubscribe();\n    }\n\n    this._idToControlStatusChangeSubscription = {};\n  }\n\n  // Whether there is any invalid field or not.\n  private shouldAnyFieldInvalid(): boolean {\n    const ids = Object.keys(this._idToInstance);\n    if (!ids || !ids.length) {\n      return false;\n    }\n\n    let hasValidationError = false;\n\n    for (const id of ids) {\n      const instance = this.idToInstance[id];\n      if (!instance) {\n        continue;\n      }\n\n      const validationErrors = this.validationSummarizerService.loadControlValidationErrors(instance as any);\n      if (!validationErrors) {\n        continue;\n      }\n\n      this._idToValidationError[id] = validationErrors;\n      const shouldValidationErrorVisible = this.visibilityHandler ? this._visibilityHandler(instance) : false;\n      if (!shouldValidationErrorVisible) {\n        continue;\n      }\n\n      hasValidationError = true;\n    }\n\n    return hasValidationError;\n  }\n\n  //#endregion\n}\n","<ul *ngIf=\"hasInvalidField\">\n  <ng-container *ngFor=\"let itemContext of itemContexts\">\n    <cms-validation-summarizer *ngIf=\"idToInstance[itemContext.id] && shouldControlHasValidationError(itemContext.id)\"\n                               [instance]=\"idToInstance[itemContext.id]\"\n                               [label]=\"idToLabel[itemContext.id]\"\n                               [validation-template]=\"idToTemplate[itemContext.id] || validationSummarizerTemplate\">\n    </cms-validation-summarizer>\n  </ng-container>\n</ul>\n\n\n<!--Default validation summary template-->\n<ng-template\n  #validationSummarizerTemplate\n  let-ngControl=\"ngControl\"\n  let-controlLabel=\"controlLabel\"\n  let-validationMessages=\"validationMessages\"\n>\n  <li *ngFor=\"let validationMessage of validationMessages;\">\n    {{controlLabel}} {{validationMessage | json}}\n  </li>\n</ng-template>\n"]}