UNPKG

@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).

117 lines 16.5 kB
import { DateUtils } from '../../shared/utils/date-utils'; export const DATE_ERROR = { INVALID_FORMAT: 'err_invalid_format', INVALID_DAY: 'err_invalid_day', INVALID_MONTH: 'err_invalid_month', INVALID_DATE: 'err_invalid_date', REQUIRED: 'err_required', }; /** * Cette classe représente les valeurs saisies par un utilisateur sur 3 champs distincts, jour, mois année. * 🔥 Bien que ces propriétés soient en théorie des nombres, à l'exécution, on reçoit des strings. * - Chaque valeur est initialisée à `undefined` mais peut aussi avoir la valeur "" (chaine vide). * 👆 Par défaut la date est une date UTC. * L'objectif de la classe est de : * - Gérer des dates saisies par l'utilisateur, potentiellement avec des valeurs `undefined` ou "" * - Manipuler les mois de 1 à 12 (et non de 0 à 11) * - D'encapsuler l'api [Date](https://developer.mozilla.org/fr/docs/Web/JavaScript/Reference/Global_Objects/Date) et plus tard l'API Temporal */ export class DateModel { constructor(fullYear, monthNum, dayNum) { this.isDateUtc = true; this.fullYear = fullYear; this.monthNum = monthNum; this.dayNum = dayNum; // Les valeurs de 0 à 99 correspondent aux années 1900 à 1999 [MDN](https://developer.mozilla.org/fr/docs/Web/JavaScript/Reference/Global_Objects/Date/Date) if (this.fullYear && 0 <= this.fullYear && this.fullYear < 100) this.fullYear += 1900; } get month() { return this.monthNum ? this.monthNum - 1 : -1; } /** * @param value : 3 formes basiques pour utiliser la méthode * - string : Une chaîne de caractères qui représente une date, selon le format reconnu par la méthode `Date.parse()` * - number : Une valeur entière qui représente le nombre de millisecondes depuis le premier janvier 1970 */ static of(value, loggerService) { const date = DateUtils.dateUtcOf(value); if (!date && typeof value === 'string') loggerService.warn(`La date '${value}' n'est pas valide`); return !date ? new DateModel() : new DateModel(date.getUTCFullYear(), date.getUTCMonth() + 1, date.getUTCDate()); } isValid() { return this.toDate() !== undefined; } toDate() { let date = undefined; let fullYear = Number(this.fullYear); let monthNum = Number(this.monthNum); let dayNum = Number(this.dayNum); // on vérifie quand même qu'on a bien affaire à des nombres if (fullYear && monthNum && dayNum && !isNaN(fullYear) && !isNaN(monthNum) && !isNaN(dayNum) && 1 <= monthNum && monthNum <= 12 && 1 <= dayNum && dayNum <= 31) { date = this.isDateUtc ? new Date(Date.UTC(fullYear, this.month, dayNum)) : new Date(fullYear, this.month, dayNum); // Date inexistante if (date.getFullYear() !== fullYear || date.getMonth() !== this.month || date.getDate() !== dayNum) date = undefined; } return date; } /** * Valide le model. * @param required indique si la date est requise ou non * @return une liste de codes d'erreur ou tableau vide */ validate(required = false) { const errors = []; const dayNum = this.dayNum; const monthNum = this.monthNum; const fullYear = this.fullYear; // Format if (!this.isNumber(dayNum) || !this.isNumber(monthNum) || !this.isNumber(fullYear)) { errors.push(DATE_ERROR.INVALID_FORMAT); } // Erreur sur le jour, le mois else { if (dayNum && (dayNum < 1 || dayNum > 31)) errors.push(DATE_ERROR.INVALID_DAY); if (monthNum && (monthNum < 1 || monthNum > 12)) errors.push(DATE_ERROR.INVALID_MONTH); } // All touched : erreur sur la date elle-même if (errors.length === 0 && this.allTouched()) { // Tous les champs sont remplis, on vérifie la date if (this.allFilled()) { const value = this.toDate(); if (!value) errors.push(DATE_ERROR.INVALID_DATE); } // Au moins 1 champ n'est pas rempli else { if (required) errors.push(DATE_ERROR.REQUIRED); } } return errors; } allTouched() { return this.fullYear !== undefined && this.monthNum !== undefined && this.dayNum !== undefined; } allFilled() { return !!this.dayNum && !!this.monthNum && !!this.fullYear; } /** @return true si undefined ou number */ isNumber(value) { return value === undefined || !isNaN(value); } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"date.model.js","sourceRoot":"","sources":["../../../../../../projects/ngx-dsfr-components/src/lib/patterns/date/date.model.ts"],"names":[],"mappings":"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","sourcesContent":["import { LoggerService } from '../../shared/services/logger.service';\nimport { DateUtils } from '../../shared/utils/date-utils';\n\nexport const DATE_ERROR = {\n  INVALID_FORMAT: 'err_invalid_format',\n  INVALID_DAY: 'err_invalid_day',\n  INVALID_MONTH: 'err_invalid_month',\n  INVALID_DATE: 'err_invalid_date',\n  REQUIRED: 'err_required',\n};\n\n/**\n * Cette classe représente les valeurs saisies par un utilisateur sur 3 champs distincts, jour, mois année.\n * 🔥 Bien que ces propriétés soient en théorie des nombres, à l'exécution, on reçoit des strings.\n * - Chaque valeur est initialisée à `undefined` mais peut aussi avoir la valeur \"\" (chaine vide).\n * 👆 Par défaut la date est une date UTC.\n * L'objectif de la classe est de :\n * - Gérer des dates saisies par l'utilisateur, potentiellement avec des valeurs `undefined` ou \"\"\n * - Manipuler les mois de 1 à 12 (et non de 0 à 11)\n * - D'encapsuler l'api [Date](https://developer.mozilla.org/fr/docs/Web/JavaScript/Reference/Global_Objects/Date) et plus tard l'API Temporal\n */\nexport class DateModel {\n  // Une date est le nombre de millisecondes écoulées depuis le premier janvier 1970 sur l'échelle UTC (idem epoch UNIX)\n  // On distingue les dates relatives au temps universal coordonné (UTC) du temps de la machine de l'utilisateur.\n  fullYear: number | undefined;\n  monthNum: number | undefined;\n  dayNum: number | undefined;\n  private isDateUtc = true;\n\n  constructor(fullYear?: number, monthNum?: number, dayNum?: number) {\n    this.fullYear = fullYear;\n    this.monthNum = monthNum;\n    this.dayNum = dayNum;\n\n    // Les valeurs de 0 à 99 correspondent aux années 1900 à 1999 [MDN](https://developer.mozilla.org/fr/docs/Web/JavaScript/Reference/Global_Objects/Date/Date)\n    if (this.fullYear && 0 <= this.fullYear && this.fullYear < 100) this.fullYear += 1900;\n  }\n\n  private get month(): number {\n    return this.monthNum ? this.monthNum - 1 : -1;\n  }\n\n  /**\n   * @param value : 3 formes basiques pour utiliser la méthode\n   * - string : Une chaîne de caractères qui représente une date, selon le format reconnu par la méthode `Date.parse()`\n   * - number : Une valeur entière qui représente le nombre de millisecondes depuis le premier janvier 1970\n   */\n  static of(value: string | number | Date | undefined | null, loggerService: LoggerService): DateModel {\n    const date = DateUtils.dateUtcOf(value);\n    if (!date && typeof value === 'string') loggerService.warn(`La date '${value}' n'est pas valide`);\n    return !date ? new DateModel() : new DateModel(date.getUTCFullYear(), date.getUTCMonth() + 1, date.getUTCDate());\n  }\n\n  isValid(): boolean {\n    return this.toDate() !== undefined;\n  }\n\n  toDate(): Date | undefined {\n    let date = undefined;\n    let fullYear = Number(this.fullYear);\n    let monthNum = Number(this.monthNum);\n    let dayNum = Number(this.dayNum);\n    // on vérifie quand même qu'on a bien affaire à des nombres\n    if (\n      fullYear &&\n      monthNum &&\n      dayNum &&\n      !isNaN(fullYear) &&\n      !isNaN(monthNum) &&\n      !isNaN(dayNum) &&\n      1 <= monthNum &&\n      monthNum <= 12 &&\n      1 <= dayNum &&\n      dayNum <= 31\n    ) {\n      date = this.isDateUtc ? new Date(Date.UTC(fullYear, this.month, dayNum)) : new Date(fullYear, this.month, dayNum);\n\n      // Date inexistante\n      if (date.getFullYear() !== fullYear || date.getMonth() !== this.month || date.getDate() !== dayNum)\n        date = undefined;\n    }\n    return date;\n  }\n\n  /**\n   * Valide le model.\n   * @param required indique si la date est requise ou non\n   * @return une liste de codes d'erreur ou tableau vide\n   */\n  validate(required = false): string[] {\n    const errors = [];\n    const dayNum = this.dayNum;\n    const monthNum = this.monthNum;\n    const fullYear = this.fullYear;\n\n    // Format\n    if (!this.isNumber(dayNum) || !this.isNumber(monthNum) || !this.isNumber(fullYear)) {\n      errors.push(DATE_ERROR.INVALID_FORMAT);\n    }\n\n    // Erreur sur le jour, le mois\n    else {\n      if (dayNum && (dayNum < 1 || dayNum > 31)) errors.push(DATE_ERROR.INVALID_DAY);\n      if (monthNum && (monthNum < 1 || monthNum > 12)) errors.push(DATE_ERROR.INVALID_MONTH);\n    }\n\n    // All touched : erreur sur la date elle-même\n    if (errors.length === 0 && this.allTouched()) {\n      // Tous les champs sont remplis, on vérifie la date\n      if (this.allFilled()) {\n        const value = this.toDate();\n        if (!value) errors.push(DATE_ERROR.INVALID_DATE);\n      }\n      // Au moins 1 champ n'est pas rempli\n      else {\n        if (required) errors.push(DATE_ERROR.REQUIRED);\n      }\n    }\n\n    return errors;\n  }\n\n  private allTouched(): boolean {\n    return this.fullYear !== undefined && this.monthNum !== undefined && this.dayNum !== undefined;\n  }\n\n  private allFilled(): boolean {\n    return !!this.dayNum && !!this.monthNum && !!this.fullYear;\n  }\n\n  /** @return true si undefined ou number */\n  private isNumber(value: any) {\n    return value === undefined || !isNaN(value);\n  }\n}\n"]}