import { Input, Component, ContentChildren, EventEmitter, Output, QueryList, SimpleChanges, HostBinding, AfterContentInit } from "@angular/core"; import { ValidatorComponent } from './validators/base.validator.component'; import { RegexValidatorComponent } from './validators/regex.validator.component'; import { RequiredValidatorComponent } from './validators/required.validator.component'; import { ValidatableComponent } from './validatable.component'; import { CustomValidatorComponent } from './validators/custom.validator.component'; import { template } from "./validation.component.html"; @Component({ selector: 'sdc-validation', template }) export class ValidationComponent implements AfterContentInit { @Input() public validateElement: ValidatableComponent; @Input() public disabled: boolean; @Output() public validityChanged: EventEmitter = new EventEmitter(); @HostBinding('class') classes; // @ContentChildren does not recieve type any or IValidator or ValidatorComponent, so need to create @ContentChildren for each validator type. @ContentChildren(RegexValidatorComponent) public regexValidator: QueryList; @ContentChildren(RequiredValidatorComponent) public requireValidator: QueryList; @ContentChildren(CustomValidatorComponent) public customValidator: QueryList; private supportedValidator: Array>; constructor() { this.disabled = false; this.classes = 'sdc-validation'; } ngAfterContentInit(): void { this.supportedValidator = [ this.regexValidator, this.requireValidator, this.customValidator ]; this.validateElement.notifier.subscribe( (value) => { const validationResult = this.validateOnChange(value); this.validateElement.valid = validationResult; }, (error) => console.log('Validation subscribe error') ); // init validateElement.valid. const value = this.validateElement.getValue(); this.validateElement.notifier.next(value); } public validate = (): boolean => { const value = this.validateElement.getValue(); return this.validateOnChange(value); } private validateOnChange(value: any): boolean { if (this.disabled) { return true; } /** * Iterate over all validators types (required, regex, etc...), and inside each iterate over * all validators with same type, and return boolean result true in case all validations passed. */ const validationResult: boolean = this.supportedValidator.reduce((sum, validatorName) => { const response: boolean = validatorName.reduce((_sum, validator) => { return _sum && validator.validate(value); }, true); return sum && response; }, true); if (this.validateElement.valid !== validationResult) { this.validityChanged.emit(validationResult); } return validationResult; } }