@edugouvfr/ngx-dsfr
Version:
NgxDsfr est un portage Angular des éléments d'interface du Système de Design de l'État Français (DSFR).
132 lines • 21.8 kB
JavaScript
import { Component, EventEmitter, forwardRef, Input, Output, ViewEncapsulation } from '@angular/core';
import { NG_VALIDATORS, NG_VALUE_ACCESSOR } from '@angular/forms';
import { DefaultValueAccessorComponent } from '../../shared';
import { DateModel } from './date.model';
import * as i0 from "@angular/core";
import * as i1 from "../../shared";
import * as i2 from "../../shared/services/logger.service";
import * as i3 from "@angular/forms";
import * as i4 from "../../forms/fieldset/form-fieldset.component";
import * as i5 from "../../forms/fieldset/form-fieldset-element.directive";
import * as i6 from "../../forms/form-input/form-input.component";
const PATH_ERR_I18N = {
err_invalid_format: 'date.error.invalid.format',
err_invalid_day: 'date.error.invalid.day',
err_invalid_month: 'date.error.invalid.month',
err_invalid_date: 'date.error.invalid.date',
err_required: 'date.error.required',
};
export class DsfrDateComponent extends DefaultValueAccessorComponent {
/** @internal */
constructor(i18n, logger) {
super();
this.i18n = i18n;
this.logger = logger;
/**
* Indique si les champs de saisie doivent être auto-complétés à partir de la date de naissance de l'utilisateur.
*/
this.autocomplete = false;
/**
* Indique si la date est obligatoire.
*/
this.required = false;
/**
* Signale le changement de date.
*/
// Certainement inutile vs ngModelChange, à voir !
this.dateChange = new EventEmitter();
/** @internal */ this.dateModel = new DateModel(); // View-Model
/** @internal */ this.validationErrors = null;
/** Permet de stocker la fonction pouvant être appelée pour forcer une revalidation. */
this.fnOnValidatorChange = () => { };
}
get errors() {
const internalError = this.validationErrors === null ? [] : this.validationErrors.map((err) => Object.values(err)[0]);
return internalError.length > 0 ? internalError : this.error;
}
// Sur changement de la date, on réécrit jour mois année
/** @internal */
writeValue(value) {
// Bien que la value soit de type date, le développeur peut, à notre insu, nous envoyer une chaine ou un nombre (comme Storybook), ex : Date.now()
this.dateModel = DateModel.of(value, this.logger);
super.writeValue(this.dateModel.toDate());
this.dateChange.emit(this.value);
}
// Sur changement du modèle, c.-à-d. jour, mois ou année, on récrit la date
/** @internal */
onFocusOut() {
this.validationErrors = this.internalValidate();
this.value = this.dateModel.toDate();
this.dateChange.emit(this.value);
}
/** @internal */
registerOnValidatorChange(fn) {
this.fnOnValidatorChange = fn;
}
/** @internal */
validate(control) {
return this.internalValidate();
}
/**
* Valide le composant et retourne ValidationErrors
*/
internalValidate() {
const errors = this.dateModel.validate(this.required);
// Pas d'erreur, on retourne null
if (errors.length === 0)
return null;
// On transforme la liste des codes d'erreur en ValidationErrors
return errors.map((codeErr) => {
// @ts-ignore
const pathErr = PATH_ERR_I18N[codeErr];
const labelErr = this.i18n.t(pathErr);
const validationErr = {};
// @ts-ignore
validationErr[codeErr] = labelErr;
return validationErr;
});
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: DsfrDateComponent, deps: [{ token: i1.I18nService }, { token: i2.LoggerService }], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: DsfrDateComponent, selector: "dsfr-date", inputs: { autocomplete: "autocomplete", error: "error", hint: "hint", legend: "legend", required: "required", valid: "valid" }, outputs: { dateChange: "dateChange" }, providers: [
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => DsfrDateComponent),
multi: true,
},
{
provide: NG_VALIDATORS,
useExisting: forwardRef(() => DsfrDateComponent),
multi: true,
},
], usesInheritance: true, ngImport: i0, template: "<dsfr-fieldset [legend]=\"legend\" [hint]=\"hint\" [error]=\"errors\" [valid]=\"valid\">\n <!-- Day -->\n <dsfr-form-input\n *fieldsetElement=\"'fr-fieldset__element--inline fr-fieldset__element--number'\"\n [label]=\"i18n.t('date.day.label')\"\n [maxLength]=\"2\"\n [hint]=\"i18n.t('date.day.hint')\"\n [autocomplete]=\"autocomplete ? 'bday-day' : undefined\"\n [required]=\"required\"\n [disabled]=\"disabled\"\n [(ngModel)]=\"dateModel.dayNum\"\n (focusout)=\"onFocusOut()\"></dsfr-form-input>\n <!-- Month -->\n <dsfr-form-input\n *fieldsetElement=\"'fr-fieldset__element--inline fr-fieldset__element--number'\"\n [label]=\"i18n.t('date.month.label')\"\n [maxLength]=\"2\"\n [hint]=\"i18n.t('date.month.hint')\"\n [autocomplete]=\"autocomplete ? 'bday-month' : undefined\"\n [required]=\"required\"\n [disabled]=\"disabled\"\n [(ngModel)]=\"dateModel.monthNum\"\n (focusout)=\"onFocusOut()\"></dsfr-form-input>\n <!-- Year -->\n <dsfr-form-input\n *fieldsetElement=\"'fr-fieldset__element--inline fr-fieldset__element--inline-grow fr-fieldset__element--year'\"\n [label]=\"i18n.t('date.year.label')\"\n [maxLength]=\"4\"\n [hint]=\"i18n.t('date.year.hint')\"\n [autocomplete]=\"autocomplete ? 'bday-year' : undefined\"\n [required]=\"required\"\n [disabled]=\"disabled\"\n [(ngModel)]=\"dateModel.fullYear\"\n (focusout)=\"onFocusOut()\"></dsfr-form-input>\n</dsfr-fieldset>\n", dependencies: [{ kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { kind: "directive", type: i3.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i4.DsfrFormFieldsetComponent, selector: "dsfr-fieldset, dsfr-form-fieldset", inputs: ["inline"] }, { kind: "directive", type: i5.DsfrFormFieldsetElementDirective, selector: "[fieldsetElement]", inputs: ["fieldsetElement"] }, { kind: "component", type: i6.DsfrFormInputComponent, selector: "dsfr-form-input", inputs: ["rows", "width", "textarea"] }], encapsulation: i0.ViewEncapsulation.None }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: DsfrDateComponent, decorators: [{
type: Component,
args: [{ selector: 'dsfr-date', encapsulation: ViewEncapsulation.None, providers: [
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => DsfrDateComponent),
multi: true,
},
{
provide: NG_VALIDATORS,
useExisting: forwardRef(() => DsfrDateComponent),
multi: true,
},
], template: "<dsfr-fieldset [legend]=\"legend\" [hint]=\"hint\" [error]=\"errors\" [valid]=\"valid\">\n <!-- Day -->\n <dsfr-form-input\n *fieldsetElement=\"'fr-fieldset__element--inline fr-fieldset__element--number'\"\n [label]=\"i18n.t('date.day.label')\"\n [maxLength]=\"2\"\n [hint]=\"i18n.t('date.day.hint')\"\n [autocomplete]=\"autocomplete ? 'bday-day' : undefined\"\n [required]=\"required\"\n [disabled]=\"disabled\"\n [(ngModel)]=\"dateModel.dayNum\"\n (focusout)=\"onFocusOut()\"></dsfr-form-input>\n <!-- Month -->\n <dsfr-form-input\n *fieldsetElement=\"'fr-fieldset__element--inline fr-fieldset__element--number'\"\n [label]=\"i18n.t('date.month.label')\"\n [maxLength]=\"2\"\n [hint]=\"i18n.t('date.month.hint')\"\n [autocomplete]=\"autocomplete ? 'bday-month' : undefined\"\n [required]=\"required\"\n [disabled]=\"disabled\"\n [(ngModel)]=\"dateModel.monthNum\"\n (focusout)=\"onFocusOut()\"></dsfr-form-input>\n <!-- Year -->\n <dsfr-form-input\n *fieldsetElement=\"'fr-fieldset__element--inline fr-fieldset__element--inline-grow fr-fieldset__element--year'\"\n [label]=\"i18n.t('date.year.label')\"\n [maxLength]=\"4\"\n [hint]=\"i18n.t('date.year.hint')\"\n [autocomplete]=\"autocomplete ? 'bday-year' : undefined\"\n [required]=\"required\"\n [disabled]=\"disabled\"\n [(ngModel)]=\"dateModel.fullYear\"\n (focusout)=\"onFocusOut()\"></dsfr-form-input>\n</dsfr-fieldset>\n" }]
}], ctorParameters: function () { return [{ type: i1.I18nService }, { type: i2.LoggerService }]; }, propDecorators: { autocomplete: [{
type: Input
}], error: [{
type: Input
}], hint: [{
type: Input
}], legend: [{
type: Input
}], required: [{
type: Input
}], valid: [{
type: Input
}], dateChange: [{
type: Output
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"date.component.js","sourceRoot":"","sources":["../../../../../../projects/ngx-dsfr-components/src/lib/patterns/date/date.component.ts","../../../../../../projects/ngx-dsfr-components/src/lib/patterns/date/date.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, forwardRef, Input, Output, ViewEncapsulation } from '@angular/core';\nimport { AbstractControl, NG_VALIDATORS, NG_VALUE_ACCESSOR, ValidationErrors, Validator } from '@angular/forms';\nimport { DefaultValueAccessorComponent, I18nService } from '../../shared';\nimport { LoggerService } from '../../shared/services/logger.service';\nimport { DateModel } from './date.model';\n\nconst PATH_ERR_I18N = {\n  err_invalid_format: 'date.error.invalid.format',\n  err_invalid_day: 'date.error.invalid.day',\n  err_invalid_month: 'date.error.invalid.month',\n  err_invalid_date: 'date.error.invalid.date',\n  err_required: 'date.error.required',\n};\n\n@Component({\n  selector: 'dsfr-date',\n  templateUrl: './date.component.html',\n  encapsulation: ViewEncapsulation.None,\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => DsfrDateComponent),\n      multi: true,\n    },\n    {\n      provide: NG_VALIDATORS,\n      useExisting: forwardRef(() => DsfrDateComponent),\n      multi: true,\n    },\n  ],\n})\nexport class DsfrDateComponent extends DefaultValueAccessorComponent<Date> implements Validator {\n  /**\n   * Indique si les champs de saisie doivent être auto-complétés à partir de la date de naissance de l'utilisateur.\n   */\n  @Input() autocomplete = false;\n\n  /**\n   * Modèle de présentation de données internes.\n   */\n  @Input() error: string | string[];\n\n  /**\n   * Texte de description additionnel du fieldset.\n   */\n  @Input() hint: string;\n\n  /**\n   * Légende du fieldset.\n   */\n  @Input() legend: string;\n\n  /**\n   * Indique si la date est obligatoire.\n   */\n  @Input() required = false;\n\n  /**\n   * Texte de succès.\n   */\n  @Input() valid: string;\n\n  /**\n   * Signale le changement de date.\n   */\n  // Certainement inutile vs ngModelChange, à voir !\n  @Output() dateChange: EventEmitter<Date> = new EventEmitter();\n\n  /** @internal */ dateModel = new DateModel(); // View-Model\n  /** @internal */ validationErrors: ValidationErrors | null = null;\n\n  /** @internal */\n  constructor(\n    public i18n: I18nService,\n    private logger: LoggerService,\n  ) {\n    super();\n  }\n\n  get errors(): string | string[] {\n    const internalError =\n      this.validationErrors === null ? [] : this.validationErrors.map((err: {}) => Object.values(err)[0]);\n    return internalError.length > 0 ? internalError : this.error;\n  }\n\n  // Sur changement de la date, on réécrit jour mois année\n  /** @internal */\n  writeValue(value: Date | undefined) {\n    // Bien que la value soit de type date, le développeur peut, à notre insu, nous envoyer une chaine ou un nombre (comme Storybook), ex : Date.now()\n    this.dateModel = DateModel.of(value, this.logger);\n    super.writeValue(this.dateModel.toDate());\n    this.dateChange.emit(this.value);\n  }\n\n  // Sur changement du modèle, c.-à-d. jour, mois ou année, on récrit la date\n  /** @internal */\n  onFocusOut() {\n    this.validationErrors = this.internalValidate();\n    this.value = this.dateModel.toDate();\n    this.dateChange.emit(this.value);\n  }\n\n  /** @internal */\n  registerOnValidatorChange(fn: () => void): void {\n    this.fnOnValidatorChange = fn;\n  }\n\n  /** @internal */\n  validate(control: AbstractControl): ValidationErrors | null {\n    return this.internalValidate();\n  }\n\n  /** Permet de stocker la fonction pouvant être appelée pour forcer une revalidation. */\n  private fnOnValidatorChange = () => {};\n\n  /**\n   * Valide le composant et retourne ValidationErrors\n   */\n  private internalValidate(): ValidationErrors | null {\n    const errors = this.dateModel.validate(this.required);\n\n    // Pas d'erreur, on retourne null\n    if (errors.length === 0) return null;\n\n    // On transforme la liste des codes d'erreur en ValidationErrors\n    return errors.map((codeErr) => {\n      // @ts-ignore\n      const pathErr = PATH_ERR_I18N[codeErr];\n      const labelErr = this.i18n.t(pathErr);\n      const validationErr = {};\n      // @ts-ignore\n      validationErr[codeErr] = labelErr;\n      return validationErr;\n    });\n  }\n}\n","<dsfr-fieldset [legend]=\"legend\" [hint]=\"hint\" [error]=\"errors\" [valid]=\"valid\">\n  <!-- Day -->\n  <dsfr-form-input\n    *fieldsetElement=\"'fr-fieldset__element--inline fr-fieldset__element--number'\"\n    [label]=\"i18n.t('date.day.label')\"\n    [maxLength]=\"2\"\n    [hint]=\"i18n.t('date.day.hint')\"\n    [autocomplete]=\"autocomplete ? 'bday-day' : undefined\"\n    [required]=\"required\"\n    [disabled]=\"disabled\"\n    [(ngModel)]=\"dateModel.dayNum\"\n    (focusout)=\"onFocusOut()\"></dsfr-form-input>\n  <!-- Month -->\n  <dsfr-form-input\n    *fieldsetElement=\"'fr-fieldset__element--inline fr-fieldset__element--number'\"\n    [label]=\"i18n.t('date.month.label')\"\n    [maxLength]=\"2\"\n    [hint]=\"i18n.t('date.month.hint')\"\n    [autocomplete]=\"autocomplete ? 'bday-month' : undefined\"\n    [required]=\"required\"\n    [disabled]=\"disabled\"\n    [(ngModel)]=\"dateModel.monthNum\"\n    (focusout)=\"onFocusOut()\"></dsfr-form-input>\n  <!-- Year -->\n  <dsfr-form-input\n    *fieldsetElement=\"'fr-fieldset__element--inline fr-fieldset__element--inline-grow fr-fieldset__element--year'\"\n    [label]=\"i18n.t('date.year.label')\"\n    [maxLength]=\"4\"\n    [hint]=\"i18n.t('date.year.hint')\"\n    [autocomplete]=\"autocomplete ? 'bday-year' : undefined\"\n    [required]=\"required\"\n    [disabled]=\"disabled\"\n    [(ngModel)]=\"dateModel.fullYear\"\n    (focusout)=\"onFocusOut()\"></dsfr-form-input>\n</dsfr-fieldset>\n"]}