Пользовательский компонент ng-bootstrap datepicker с проверкой реактивных форм
Я пытаюсь инкапсулировать указатель даты ng-bootstrap (Angular 4 и Bootstrap 4) внутри пользовательского компонента, добавляя проверку реактивных форм тоже.
Я хотел бы знать, если я использовал правильный подход, поэтому я покажу вам свой код.
Это пользовательский компонент:
@Component({
selector: 'ngb-date-time-picker',
template: `
<div class="form-inline" [hidden]="timeOnly">
<div class="form-group">
<div class="input-group">
<input
readonly
class="form-control form-control-danger"
[placeholder]="placeholder"
name="date"
[(ngModel)]="dateStruct"
(ngModelChange)="updateDate()"
ngbDatepicker
#datePicker="ngbDatepicker">
<div class="input-group-addon btn-toggle-date-picker" (click)="datePicker.toggle()" >
<i class="material-icons md-18">date_range</i>
</div>
</div>
</div>
</div>
<ngb-timepicker [(ngModel)]="timeStruct" (ngModelChange)="updateTime()" [meridian]="true" [hidden]="dateOnly"></ngb-timepicker>
`,
styles: [`
.form-group {
width: 100%;
}
.btn-toggle-date-picker {
cursor: pointer;
}
.btn-toggle-date-picker:hover {
background-color: #ccc;
}
`],
providers: [
NgbDatepickerConfig,
I18n,
{ provide: NgbDatepickerI18n, useClass: NgbMultiLanguageDatepickerI18n },
{ provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => NgbDateTimePickerComponent), multi: true },
{ provide: NG_VALIDATORS, useExisting: forwardRef(() => NgbDateTimePickerComponent), multi: true }
]
})
export class NgbDateTimePickerComponent implements OnInit, OnChanges, ControlValueAccessor {
@Input() placeholder: string;
@Input() isRequired: boolean = true;
protected onChange: any = () => { };
protected onTouched: any = () => { };
private _date: Date = null;
//get accessor
@Input()
get date(): Date {
return this._date;
};
//set accessor
set date(val: Date) {
this._date = val;
this.onChange(val);
}
@Output() dateChange: EventEmitter<Date> = new EventEmitter<Date>();
@Input() minDate: Date = null;
@Input() dateOnly: boolean = false;
@Input() timeOnly: boolean = false;
@Input() startOfDay: boolean = false;
@Input() endOfDay: boolean = false;
dateStruct: NgbDateStruct;
timeStruct: NgbTimeStruct;
constructor(private config: NgbDatepickerConfig) { }
ngOnInit(): void {
if (this.minDate != null) {
this.config.minDate = { year: getYear(this.minDate), month: getMonth(this.minDate) + 1, day: getDate(this.minDate) };
}
}
ngOnChanges(changes: SimpleChanges): void {
if (changes.date && changes.date.currentValue !== changes.date.previousValue && changes.date.currentValue != null) {
this.dateStruct = {
day: getDate(this.date),
month: getMonth(this.date) + 1,
year: getYear(this.date)
};
this.timeStruct = {
second: getSeconds(this.date),
minute: getMinutes(this.date),
hour: getHours(this.date),
};
} else if (changes.date && changes.date.currentValue == null) {
this.dateStruct = null;
this.timeStruct = null;
}
}
updateDate(): void {
if (this._date == null) {
this._date = new Date();
}
const newDate: Date = setYear(setMonth(setDate(this._date, this.dateStruct.day), this.dateStruct.month - 1), this.dateStruct.year);
if (this.startOfDay) {
setHours(setMinutes(setSeconds(setMilliseconds(newDate, 0), 0), 0), 0);
newDate.setHours(0, 0, 0, 0);
} else if (this.endOfDay) {
newDate.setHours(23, 59, 59, 999);
} else {
let now: Date = new Date();
newDate.setHours(getHours(now), getMinutes(now), getSeconds(now), getMilliseconds(now));
}
this.dateChange.next(newDate);
}
updateTime(): void {
const newDate: Date = setHours(setMinutes(setSeconds(this._date, this.timeStruct.second), this.timeStruct.minute), this.timeStruct.hour);
this.dateChange.next(newDate);
}
public writeValue(value: any): void {
if (value) {
if (value === "") value = null;
this.date = value;
}
}
public registerOnChange(fn: (_: any) => {}): void { this.onChange = fn; }
public registerOnTouched(fn: () => {}): void { this.onTouched = fn; }
validate(c: FormControl) {
if (this.isRequired) {
let res = Validators.required(c)
return res;
}
return null;
}
}
Это компонент, из которого я использую пользовательский компонент:
@Component({
moduleId: module.id.toString(),
template: require('./user-add-edit.component.html'),
styles: [require('./user-add-edit.component.scss')],
// make slide in/out animation available to this component
animations: [slideInOutAnimation],
// attach the slide in/out animation to the host (root) element of this component
host: { '[@slideInOutAnimation]': '' }
})
export class UserAddEditComponent implements OnInit, CanComponentDeactivate {
title: string = 'Aggiungi Abbonato';
isBusy: Subscription;
isEditMode: boolean = false;
isCancelling: boolean = false;
user: IUser = <IUser>{};
model: IPutUser = <IPutUser>{ };
addEditForm: FormGroup;
uploader: FileUploader = new FileUploader({ autoUpload: true });
uploading: boolean = false;
minDate: Date = new Date();
isCertificateChanged: boolean = false;
// Workaround for https://github.com/valor-software/ng2-file-upload/issues/220
@ViewChild('fileInput') fileInput: ElementRef
form: FormGroup;
counterValue = 3;
minValue = 0;
maxValue = 12;
constructor(private route: ActivatedRoute, private router: Router, private appService: AppService, private pubSubService: PubSubService,
private modalService: ModalService, private formBuilder: FormBuilder, @Inject('ORIGIN_URL') private originUrl: string,
public authService: AuthService, private toastyService: ToastyService)
{
// Uploader configuration
this.uploader.options.url = this.originUrl + '/api/certificates';
this.uploader.authToken = this.authService.getAuthToken();
this.uploader.onBeforeUploadItem = (item: FileItem) => this.uploading = true;
this.uploader.onCompleteItem = (item: FileItem, response: string, status: number, headers: ParsedResponseHeaders) => {
this.uploading = false;
var resp = JSON.parse(response);
if (resp.error) {
console.error(resp.error);
this.toastyService.error(resp.error);
}
this.model.medicalCertificateUrl = resp.url;
this.toastyService.success("Caricamento completato.");
};
this.uploader.onAfterAddingFile = (item: FileItem) => {
// Workaround for https://github.com/valor-software/ng2-file-upload/issues/220
this.fileInput.nativeElement.value = '';
//this.setMedicalCertificateExpiryRequired(true);
this.addEditForm.addControl("medicalCertificateExpiry", new FormControl(""));
};
}
public ngOnInit(): void {
let userId: string = this.route.snapshot.params['id'];
if (userId) {
this.title = 'Modifica Abbonato';
this.isEditMode = true;
this.isBusy = this.appService.getUser(userId)
.subscribe(user => {
this.user = user;
this.model = <IPutUser>{
id: this.user.id,
firstName: this.user.firstName,
lastName: this.user.lastName,
taxCode: this.user.taxCode,
userName: this.user.userName,
phoneNumber: this.user.phoneNumber,
email: this.user.email,
medicalCertificateUrl: this.user.medicalCertificateUrl,
medicalCertificateExpiry: this.user.medicalCertificateExpiry
};
if (user.medicalCertificateUrl != null) {
this.addEditForm.addControl("medicalCertificateExpiry", new FormControl(""));
this.addEditForm.controls.medicalCertificateExpiry.patchValue(user.medicalCertificateExpiry);
}
this.onValueChanged();
});
}
this.initForm();
this.form = this.formBuilder.group({
counter: this.counterValue
});
}
dateChanged(event: any) {
console.log("dateChanged", event);
}
save(): void {
let apiCall: Observable<any>;
if (this.isEditMode) {
apiCall = this.appService.putUser(this.model);
} else {
apiCall = this.appService.postUser(this.model);
}
this.isBusy = apiCall.subscribe(() => {
this.router.navigate(['registry']);
this.pubSubService.publish('registry-updated');
this.pubSubService.publish('check-alerts');
});
}
removeCertificate(showToast: boolean = true): void {
this.appService.deleteCertificate(this.model.medicalCertificateUrl).subscribe(() => {
this.model.medicalCertificateUrl = null;
this.model.medicalCertificateExpiry = null;
this.addEditForm.removeControl("medicalCertificateExpiry");
if (showToast) {
this.toastyService.success("Certificato rimosso con successo.");
}
});
}
canDeactivate(): Promise<boolean> | boolean {
if (this.isCancelling) {
let title: string = 'Conferma';
let message: string = 'Sicuro di voler annullare le modifiche?';
let isModelChanged: boolean = false;
if (this.isEditMode) {
this.isCertificateChanged = this.user.medicalCertificateUrl !== this.model.medicalCertificateUrl;
isModelChanged = this.user.firstName !== this.model.firstName
|| this.user.lastName !== this.model.lastName
|| this.user.taxCode !== this.model.taxCode
|| this.user.userName !== this.model.userName
|| this.user.phoneNumber !== this.model.phoneNumber
|| this.user.email !== this.model.email
|| this.isCertificateChanged
|| ((this.user.medicalCertificateExpiry != null && this.model.medicalCertificateExpiry != null) ? (this.user.medicalCertificateExpiry.getTime() !== this.model.medicalCertificateExpiry.getTime()) : false)
|| (this.user.medicalCertificateExpiry != null && this.model.medicalCertificateExpiry == null)
|| (this.user.medicalCertificateExpiry == null && this.model.medicalCertificateExpiry != null);
} else {
if (this.model.medicalCertificateUrl) {
this.isCertificateChanged = true;
}
if (this.model.firstName || this.model.lastName || this.model.taxCode || this.model.userName || this.model.phoneNumber || this.model.email
|| this.model.medicalCertificateUrl || this.model.medicalCertificateExpiry) {
isModelChanged = true;
}
}
if (isModelChanged) {
return this.modalService.confirm(title, message, { cancelButtonLabel: 'Annulla', showClose: true })
.then(ok => {
if (ok && this.isCertificateChanged && this.model.medicalCertificateUrl) {
this.removeCertificate(false);
}
this.isCancelling = false;
return ok;
});
}
}
return true;
}
private initForm() {
this.addEditForm = this.formBuilder.group({
firstName: ['', Validators.required],
lastName: ['', Validators.required],
taxCode: ['', [Validators.required, Validators.pattern(/^[a-zA-Z]{6}[0-9]{2}[a-zA-Z][0-9]{2}[a-zA-Z][0-9]{3}[a-zA-Z]$/)]],
userName: ['', [Validators.required, Validators.minLength(6)]],
phoneNumber: ['', [Validators.required, Validators.pattern(/^\d+$/)]],
email: ['', [Validators.required, Validators.pattern(/^[a-zA-Z0-9\+\.\_\%\-\+]{1,256}\@[a-zA-Z0-9][a-zA-Z0-9\-]{0,64}(\.[a-zA-Z0-9][a-zA-Z0-9\-]{0,25})$/)]],
});
this.addEditForm.valueChanges
.subscribe(data => this.onValueChanged(data));
this.onValueChanged(); // (re)set validation messages now
}
onValueChanged(data?: any): void {
if (!this.addEditForm) { return; }
const form = this.addEditForm;
for (const field in this.formErrors) {
// clear previous error message (if any)
this.formErrors[field] = '';
const control = form.get(field);
if (control && control.dirty && !control.valid) {
const messages = this.validationMessages[field];
for (const key in control.errors) {
this.formErrors[field] += messages[key] + ' ';
}
}
}
}
formErrors: any = {
'firstName': '',
'lastName': '',
'taxCode': '',
'userName': '',
'phoneNumber': '',
'email': '',
'medicalCertificateExpiry': ''
};
validationMessages: any = {
'firstName': {
'required': 'Nome é obbligatorio.'
},
'lastName': {
'required': 'Cognome é obbligatorio.'
},
'taxCode': {
'required': 'Codice Fiscale é obbligatorio.',
'pattern': 'Codice Fiscale non valido.'
},
'userName': {
'required': 'Username é obbligatorio.',
'minlength': 'Username deve avere almeno 6 caratteri.'
},
'phoneNumber': {
'required': 'Telefono é obbligatorio.',
'pattern': 'Telefono non valido.'
},
'email': {
'required': 'Email é obbligatorio.',
'pattern': 'Email non valida.'
},
'medicalCertificateExpiry': {
'required': 'Scadenza Certificato Medico é obbligatorio.'
}
};
}
И это мнение:
<div class="side-form">
<div [ngBusy]="isBusy"></div>
<h1 style="margin-bottom:1em;">{{title}}</h1>
<div class="form-container">
<form name="form" (ngSubmit)="f.form.valid && save()" #f="ngForm" [formGroup]="addEditForm" novalidate>
<div class="form-group" [ngClass]="{ 'has-danger': formErrors.firstName }">
<label class="form-control-label" for="firstName">Nome</label>
<input type="text" id="firstName" name="firstName" placeholder="Nome" [(ngModel)]="model.firstName" formControlName="firstName" class="form-control form-control-danger" required />
<div *ngIf="formErrors.firstName" class="form-control-feedback">
{{ formErrors.firstName }}
</div>
</div>
<div class="form-group" [ngClass]="{ 'has-danger': formErrors.lastName }">
<label class="form-control-label" for="lastName">Cognome</label>
<input type="text" id="lastName" name="lastName" placeholder="Cognome" [(ngModel)]="model.lastName" formControlName="lastName" class="form-control form-control-danger" required />
<div *ngIf="formErrors.lastName" class="form-control-feedback">
{{ formErrors.lastName }}
</div>
</div>
<div class="form-group" [ngClass]="{ 'has-danger': formErrors.taxCode }">
<label class="form-control-label" for="taxCode">Codice Fiscale</label>
<input type="text" id="taxCode" name="taxCode" placeholder="Codice Fiscale" [(ngModel)]="model.taxCode" formControlName="taxCode" class="form-control form-control-danger" required style="text-transform:uppercase"/>
<div *ngIf="formErrors.taxCode" class="form-control-feedback">
{{ formErrors.taxCode }}
</div>
</div>
<div class="form-group" [ngClass]="{ 'has-danger': formErrors.userName }">
<label class="form-control-label" for="userName">Username</label>
<input type="text" id="userName" name="userName" placeholder="Username" [(ngModel)]="model.userName" formControlName="userName" class="form-control form-control-danger" required min="6" />
<div *ngIf="formErrors.userName" class="form-control-feedback">
{{ formErrors.userName }}
</div>
</div>
<div class="form-group" [ngClass]="{ 'has-danger': formErrors.phoneNumber }">
<label class="form-control-label" for="phoneNumber">Telefono</label>
<input type="text" id="phoneNumber" name="phoneNumber" placeholder="Telefono" [(ngModel)]="model.phoneNumber" formControlName="phoneNumber" class="form-control form-control-danger" required />
<div *ngIf="formErrors.phoneNumber" class="form-control-feedback">
{{ formErrors.phoneNumber }}
</div>
</div>
<div class="form-group" [ngClass]="{ 'has-danger': formErrors.email }">
<label class="form-control-label" for="email">Email</label>
<input type="email" id="email" name="email" placeholder="Email" [(ngModel)]="model.email" formControlName="email" class="form-control form-control-danger" required /> <!-- email -->
<div *ngIf="formErrors.email" class="form-control-feedback">
{{ formErrors.email }}
</div>
</div>
<div class="form-group" [ngClass]="{ 'has-danger': formErrors.medicalCertificateUrl }">
<label class="form-control-label" for="medicalCertificateUrl">Certificato Medico</label>
<div>
<input #fileInput type="file" ng2FileSelect [uploader]="uploader" style="display:none;" />
<button id="browse" name="browse" type="button" class="btn btn-default" (click)="fileInput.click()" style="display:inline-block;" [disabled]="model.medicalCertificateUrl">Scegli file...</button>
<img *ngIf="uploading" src="" />
<span style="margin-left:.5em" *ngIf="!model.medicalCertificateUrl">Nessun file selezionato</span>
<div *ngIf="model.medicalCertificateUrl" style="display:inline-block">
<span style="margin-left:.5em">{{ model.medicalCertificateUrl }}</span>
<button type="button" class="btn btn-danger btn-clear" (click)="removeCertificate()" title="Rimuovi certificato">
<i class="material-icons md-18">clear</i>
</button>
</div>
</div>
</div>
<div class="form-group" [ngClass]="{ 'has-danger': formErrors.medicalCertificateExpiry }" *ngIf="model.medicalCertificateUrl">
<label class="form-control-label" for="medicalCertificateExpiry">Scadenza Certificato Medico</label>
<ngb-date-time-picker id="medicalCertificateExpiry" name="medicalCertificateExpiry"
formControlName="medicalCertificateExpiry"
[(date)]="model.medicalCertificateExpiry"
(dateChange)="dateChanged($event)"
[dateOnly]="true"
[endOfDay]="true"
[minDate]="minDate"
placeholder="Scadenza...">
</ngb-date-time-picker>
<div *ngIf="formErrors.medicalCertificateExpiry" class="form-control-feedback">
{{ formErrors.medicalCertificateExpiry }}
</div>
</div>
<div class="form-group">
<button type="button" class="btn btn-secondary" routerLink="/registry" (click)="isCancelling = true">Annulla</button>
<button type="submit" class="btn btn-primary" [disabled]="!addEditForm.valid">Salva</button>
</div>
</form>
</div>
</div>
Что вы думаете об этом? Есть ли что-то, что можно улучшить?
Спасибо всем заранее:)