Solución:
Yo tuve el mismo problema. Intenté todo y finalmente pude resolver usando este método:
Agregué este oyente en el componente personalizado. También puede hacerlo en un evento de ‘desenfoque’.
@HostListener('focusout', ['$event.target'])
onFocusout() {
this.onTouched();
}
Y también llamando a onTouched al establecer cualquier valor.
writeValue(value: any) {
this.onTouched();
this.Value = value ? value : '';
}
Usé un poco su respuesta y el enlace que proporcionó para llegar a esta solución:
@Component({
selector: 'app-custom-input',
templateUrl: './custom-input.component.html',
styleUrls: ['./custom-input.component.css'],
providers: [{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => CustomInputComponent),
multi: true
}]
})
export class CustomInputComponent implements OnInit, ControlValueAccessor {
...
_control: NgControl;
disabled: boolean;
constructor(@Inject(INJECTOR) private injector: Injector) {
}
ngOnInit() {
this._control = this.injector.get(NgControl);
}
...
custom-input.component.html
<div class="input-wrap">
<mat-form-field appearance="outline">
<mat-label>{{Label}}</mat-label>
<input matInput
[formControl]="_control.control" // <== this what makes it work
[attr.maxlength]="MaxLength"
[placeholder]="PlaceHolder ? PlaceHolder : ''"
[type]="type ? type: 'text'"
>
</mat-form-field>
</div>
Nota: No estoy vinculado a las salidas de MatInput (sí). la directiva formControl que pasa el control a MatInput lo maneja por nosotros.
hizo un ejemplo para ti
Esto creará una validación de errores de Angular Material
Mecanografiado:
import { Component, OnInit, Input, EventEmitter, Output, forwardRef, Injector } from '@angular/core';
import { ControlValueAccessor, NG_VALUE_ACCESSOR, FormControl, NgForm, FormGroupDirective, NgControl } from '@angular/forms';
import { ErrorStateMatcher } from '@angular/material';
export class CustomFieldErrorMatcher implements ErrorStateMatcher {
constructor(private customControl: FormControl,private errors:any) { }
isErrorState(control: FormControl | null, form: FormGroupDirective | NgForm | null): boolean {
return this.customControl && this.customControl.touched &&(this.customControl.invalid || this.errors);
}
}
@Component({
selector: 'app-input-textbox',
templateUrl: './input-textbox.component.html',
styleUrls: ['./input-textbox.component.scss'],
providers: [
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => InputTextboxComponent),
multi: true
}
]
})
export class InputTextboxComponent implements OnInit, ControlValueAccessor {
@Input() MaxLength: string;
@Input() FocusIn: boolean;
@Input() Width: string;
@Input() Value: string;
@Input() type: string;
@Input() Label: string;
@Input() Hint: string;
@Input() PlaceHolder: string;
@Output() saveValue = new EventEmitter();
@Output() onStateChange = new EventEmitter();
@Input() errors: any = null;
disabled: boolean;
control: FormControl;
constructor(public injector: Injector) {}
ngOnInit(){}
ngAfterViewInit(): void {
const ngControl: NgControl = this.injector.get(NgControl, null);
if (ngControl) {
setTimeout(() => {
this.control = ngControl.control as FormControl;
})
}
}
saveValueAction(e) {
this.saveValue.emit(e.target.value);
}
//control value accessor init
writeValue(value: any) {
this.Value = value ? value : '';
}
onChange(e) {
this.Value = e;
}
onTouched() {
this.onStateChange.emit();
}
registerOnChange(fn: any) { this.onChange = fn; }
registerOnTouched(fn: any) { this.onTouched = fn; }
setDisabledState(isDisabled) { this.disabled = isDisabled; }
errorMatcher() {
return new CustomFieldErrorMatcher(this.control,this.errors)
}
readonly errorStateMatcher: ErrorStateMatcher = {
isErrorState: (ctrl: FormControl) => (ctrl && ctrl.invalid)
};
}
HTML
<div>
<mat-form-field>
<mat-label>{{Label}}</mat-label>
<input
matInput
[attr.maxlength] = "MaxLength"
[value]="Value ? Value : ''"
[placeholder]="PlaceHolder ? PlaceHolder : ''"
[type]="type ? type: 'text'"
[ngModel]="Value"
[errorStateMatcher]="errorMatcher()"
(input)="onChange($event.target.value)"
(blur)="onTouched()"
(change)="saveValueAction($event)"
(ngModelChange)="Value=$event;onChange($event)"
>
<mat-hint>{{Hint}}</mat-hint>
</mat-form-field>
</div>
¡Haz clic para puntuar esta entrada!
(Votos: 0 Promedio: 0)