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

118 lines 25.6 kB
import { Component, EventEmitter, forwardRef, Input, Output, ViewEncapsulation, } from '@angular/core'; import { NG_VALUE_ACCESSOR } from '@angular/forms'; import { DefaultControlComponent } from '../../shared'; import * as i0 from "@angular/core"; import * as i1 from "../../shared"; import * as i2 from "@angular/common"; import * as i3 from "@angular/forms"; export class DsfrFormSelectComponent extends DefaultControlComponent { /** @internal */ constructor(i18n) { super(); this.i18n = i18n; /** * Indique que le champ est obligatoire, nécessaire du point de vue de l'accessibilité. */ this.required = false; /** * Output équivalent de (ngModelChange) si celui n'est pas disponible. */ this.selectChange = new EventEmitter(); /** * @internal * https://github.com/angular/angular/issues/50233 : issue performance with ng-container/ng-template inside select for Firefox * Dans le cas ou il n'y a pas de groupe, supprimer le ng-container avec ngFor côté template * fixme: vérifier/améliorer le fix lors du passage à angular 17. */ this.noGroup = true; /** @internal fonction de comparaison */ this.compareFn = Object.is; this.trackByIndex = (index) => { return index; }; } /** * @deprecated since 1.2, utiliser `placeholder` à la place (sans H majuscule). */ get placeHolder() { return this.placeholder; } /** * @deprecated since 1.2, utiliser `placeholder` à la place (sans H majuscule). */ set placeHolder(value) { this.placeholder = value; } /** Personnalisation de la comparaison. CompareWith est une fonction qui a deux arguments: valeur option1 et valeur option2. Si compareWith est fournie, sélection de l'option en fonction du retour de la fonction. */ set compareWith(fn) { if (typeof fn !== 'function') { throw Error('`compareWith` must be a function.'); } this.compareFn = fn; } /** @internal */ ngOnInit() { if (this.options && this.options.length && this.options.find((o) => o.options)) { this.noGroup = false; } super.ngOnInit(); this.messagesGroupId = `${this.inputId}-messages`; this.placeholder ??= this.i18n.t('select.placeholder'); } ngOnChanges({ value }) { if (value && this.value && this.options?.length) { if (!this.findOption(this.value, this.options)) { this.value = undefined; } } } /** @internal */ onChange() { // Comme on a le double binding, la valeur est déjà à jour // this.value = (<HTMLSelectElement>event?.target)?.value; this.selectChange.emit(this.value); } findOption(value, options) { return options?.find((option) => this.compareFn(value, option.value) || option.value === value || (option.options && this.findOption(value, option.options))); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: DsfrFormSelectComponent, deps: [{ token: i1.I18nService }], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: DsfrFormSelectComponent, selector: "dsfr-form-select", inputs: { placeholder: "placeholder", options: "options", required: "required", ariaLabel: "ariaLabel", error: "error", valid: "valid", placeHolder: "placeHolder", compareWith: "compareWith" }, outputs: { selectChange: "selectChange" }, providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => DsfrFormSelectComponent), multi: true, }, ], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<div\n [ngClass]=\"{\n 'fr-select-group': true,\n 'fr-select-group--error': error,\n 'fr-select-group--valid': valid,\n 'fr-select-group--disabled': disabled\n }\">\n <label class=\"fr-label\" [for]=\"inputId\">\n <ng-container *ngIf=\"label\">{{ label }}</ng-container>\n <ng-content *ngIf=\"!label\" select=\"[label]\"></ng-content>\n <span *ngIf=\"hint\" class=\"fr-hint-text\">{{ hint }}</span>\n </label>\n\n <!-- Select sans groupes. Duplication du select pour pb perf. affichage d'options sur firefox si utilisation d'un ng-container -->\n <select\n *ngIf=\"noGroup\"\n class=\"fr-select\"\n [id]=\"inputId\"\n [attr.name]=\"name || null\"\n [(ngModel)]=\"value\"\n [attr.aria-describedby]=\"messagesGroupId\"\n [attr.aria-label]=\"ariaLabel\"\n [disabled]=\"disabled\"\n [attr.required]=\"required || null\"\n [compareWith]=\"compareFn\"\n (change)=\"onChange()\">\n <option *ngIf=\"!value\" [value]=\"value\" selected hidden disabled>{{ placeholder }}</option>\n <option\n *ngFor=\"let option of options; trackBy: trackByIndex\"\n [ngValue]=\"option.value\"\n [attr.aria-selected]=\"compareFn(option.value, value)\"\n [disabled]=\"option.disabled || null\">\n {{ option.label }}\n </option>\n </select>\n\n <!-- Select avec groupes. Ne sera pas optimis\u00E9 pour FF, voir pour un fix avec angular 17 -->\n <select\n *ngIf=\"!noGroup\"\n class=\"fr-select\"\n [id]=\"inputId\"\n [attr.name]=\"name || null\"\n [(ngModel)]=\"value\"\n [attr.aria-describedby]=\"messagesGroupId\"\n [attr.aria-label]=\"ariaLabel\"\n [disabled]=\"disabled\"\n [attr.required]=\"required || null\"\n [compareWith]=\"compareFn\"\n (change)=\"onChange()\">\n <option *ngIf=\"!value\" [value]=\"value\" selected hidden disabled>{{ placeholder }}</option>\n <ng-container *ngFor=\"let option of options; trackBy: trackByIndex\">\n <option\n *ngIf=\"!option.options\"\n [ngValue]=\"option.value\"\n [attr.aria-selected]=\"compareFn(option.value, value)\"\n [disabled]=\"option.disabled || null\">\n {{ option.label }}\n </option>\n </ng-container>\n <ng-container *ngFor=\"let option of options; trackBy: trackByIndex\">\n <optgroup *ngIf=\"option.options\" [label]=\"option.label\">\n <option\n *ngFor=\"let option of option.options; trackBy: trackByIndex\"\n [ngValue]=\"option.value\"\n [attr.aria-selected]=\"compareFn(option.value, value)\"\n [disabled]=\"option.disabled || null\">\n {{ option.label }}\n </option>\n </optgroup>\n </ng-container>\n </select>\n\n <div [id]=\"messagesGroupId\" class=\"fr-messages-group\" aria-live=\"polite\">\n <p *ngIf=\"error\" class=\"fr-message fr-message--error\">{{ error }}</p>\n <p *ngIf=\"valid\" class=\"fr-message fr-message--valid\">{{ valid }}</p>\n </div>\n</div>\n", dependencies: [{ kind: "directive", type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i3.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i3.SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: ["compareWith"] }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }], encapsulation: i0.ViewEncapsulation.None }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: DsfrFormSelectComponent, decorators: [{ type: Component, args: [{ selector: 'dsfr-form-select', encapsulation: ViewEncapsulation.None, providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => DsfrFormSelectComponent), multi: true, }, ], template: "<div\n [ngClass]=\"{\n 'fr-select-group': true,\n 'fr-select-group--error': error,\n 'fr-select-group--valid': valid,\n 'fr-select-group--disabled': disabled\n }\">\n <label class=\"fr-label\" [for]=\"inputId\">\n <ng-container *ngIf=\"label\">{{ label }}</ng-container>\n <ng-content *ngIf=\"!label\" select=\"[label]\"></ng-content>\n <span *ngIf=\"hint\" class=\"fr-hint-text\">{{ hint }}</span>\n </label>\n\n <!-- Select sans groupes. Duplication du select pour pb perf. affichage d'options sur firefox si utilisation d'un ng-container -->\n <select\n *ngIf=\"noGroup\"\n class=\"fr-select\"\n [id]=\"inputId\"\n [attr.name]=\"name || null\"\n [(ngModel)]=\"value\"\n [attr.aria-describedby]=\"messagesGroupId\"\n [attr.aria-label]=\"ariaLabel\"\n [disabled]=\"disabled\"\n [attr.required]=\"required || null\"\n [compareWith]=\"compareFn\"\n (change)=\"onChange()\">\n <option *ngIf=\"!value\" [value]=\"value\" selected hidden disabled>{{ placeholder }}</option>\n <option\n *ngFor=\"let option of options; trackBy: trackByIndex\"\n [ngValue]=\"option.value\"\n [attr.aria-selected]=\"compareFn(option.value, value)\"\n [disabled]=\"option.disabled || null\">\n {{ option.label }}\n </option>\n </select>\n\n <!-- Select avec groupes. Ne sera pas optimis\u00E9 pour FF, voir pour un fix avec angular 17 -->\n <select\n *ngIf=\"!noGroup\"\n class=\"fr-select\"\n [id]=\"inputId\"\n [attr.name]=\"name || null\"\n [(ngModel)]=\"value\"\n [attr.aria-describedby]=\"messagesGroupId\"\n [attr.aria-label]=\"ariaLabel\"\n [disabled]=\"disabled\"\n [attr.required]=\"required || null\"\n [compareWith]=\"compareFn\"\n (change)=\"onChange()\">\n <option *ngIf=\"!value\" [value]=\"value\" selected hidden disabled>{{ placeholder }}</option>\n <ng-container *ngFor=\"let option of options; trackBy: trackByIndex\">\n <option\n *ngIf=\"!option.options\"\n [ngValue]=\"option.value\"\n [attr.aria-selected]=\"compareFn(option.value, value)\"\n [disabled]=\"option.disabled || null\">\n {{ option.label }}\n </option>\n </ng-container>\n <ng-container *ngFor=\"let option of options; trackBy: trackByIndex\">\n <optgroup *ngIf=\"option.options\" [label]=\"option.label\">\n <option\n *ngFor=\"let option of option.options; trackBy: trackByIndex\"\n [ngValue]=\"option.value\"\n [attr.aria-selected]=\"compareFn(option.value, value)\"\n [disabled]=\"option.disabled || null\">\n {{ option.label }}\n </option>\n </optgroup>\n </ng-container>\n </select>\n\n <div [id]=\"messagesGroupId\" class=\"fr-messages-group\" aria-live=\"polite\">\n <p *ngIf=\"error\" class=\"fr-message fr-message--error\">{{ error }}</p>\n <p *ngIf=\"valid\" class=\"fr-message fr-message--valid\">{{ valid }}</p>\n </div>\n</div>\n" }] }], ctorParameters: function () { return [{ type: i1.I18nService }]; }, propDecorators: { placeholder: [{ type: Input }], options: [{ type: Input }], required: [{ type: Input }], ariaLabel: [{ type: Input }], error: [{ type: Input }], valid: [{ type: Input }], selectChange: [{ type: Output }], placeHolder: [{ type: Input }], compareWith: [{ type: Input }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-select.component.js","sourceRoot":"","sources":["../../../../../../projects/ngx-dsfr-components/src/lib/forms/form-select/form-select.component.ts","../../../../../../projects/ngx-dsfr-components/src/lib/forms/form-select/form-select.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  forwardRef,\n  Input,\n  OnChanges,\n  OnInit,\n  Output,\n  SimpleChanges,\n  ViewEncapsulation,\n} from '@angular/core';\nimport { NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { DefaultControlComponent, I18nService } from '../../shared';\nimport { DsfrSelectOption } from './form-select.model';\n\n@Component({\n  selector: 'dsfr-form-select',\n  templateUrl: './form-select.component.html',\n  encapsulation: ViewEncapsulation.None,\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => DsfrFormSelectComponent),\n      multi: true,\n    },\n  ],\n})\nexport class DsfrFormSelectComponent extends DefaultControlComponent<any> implements OnInit, OnChanges {\n  /**\n   * Permet de personnaliser la première option (non sélectionnable). Cette propriété est positionnée à une valeur par\n   * défaut (ex : Sélectionnez une option) internationalisée.\n   */\n  @Input() placeholder: string;\n\n  /**\n   * Le modèle de présentation permettant de transmettre la liste des options.\n   */\n  @Input() options: DsfrSelectOption[];\n\n  /**\n   * Indique que le champ est obligatoire, nécessaire du point de vue de l'accessibilité.\n   */\n  @Input() required = false;\n\n  /**\n   * Permet d'ajouter un attribut 'aria-label' sur le champ de formulaire en cas de besoin.\n   */\n  @Input() ariaLabel: string;\n\n  /**\n   * Message d'erreur, quand il est présent les couleurs du contrôle changent.\n   */\n  @Input() error: string;\n\n  /**\n   * Message de validation, quand il est présent les couleurs du contrôle changent.\n   */\n  @Input() valid: string;\n\n  /**\n   * Output équivalent de (ngModelChange) si celui n'est pas disponible.\n   */\n  @Output() selectChange = new EventEmitter<any>();\n\n  /** @internal */ messagesGroupId: string;\n\n  /**\n   * @internal\n   * https://github.com/angular/angular/issues/50233 : issue performance with ng-container/ng-template inside select for Firefox\n   * Dans le cas ou il n'y a pas de groupe, supprimer le ng-container avec ngFor côté template\n   * fixme: vérifier/améliorer le fix lors du passage à angular 17.\n   */\n  noGroup: boolean = true;\n\n  /** @internal fonction de comparaison */\n  compareFn: (o1: any, o2: any) => boolean = Object.is;\n\n  /** @internal */\n  constructor(private i18n: I18nService) {\n    super();\n  }\n\n  /**\n   * @deprecated since 1.2, utiliser `placeholder` à la place (sans H majuscule).\n   */\n  get placeHolder(): string {\n    return this.placeholder;\n  }\n\n  /**\n   * @deprecated since 1.2, utiliser `placeholder` à la place (sans H majuscule).\n   */\n  @Input() set placeHolder(value: string) {\n    this.placeholder = value;\n  }\n\n  /** Personnalisation de la comparaison. CompareWith est une fonction qui a deux arguments: valeur option1 et valeur option2.\n  Si compareWith est fournie, sélection de l'option en fonction du retour de la fonction. */\n  @Input()\n  set compareWith(fn: (o1: any, o2: any) => boolean) {\n    if (typeof fn !== 'function') {\n      throw Error('`compareWith` must be a function.');\n    }\n\n    this.compareFn = fn;\n  }\n\n  /** @internal */\n  ngOnInit() {\n    if (this.options && this.options.length && this.options.find((o) => o.options)) {\n      this.noGroup = false;\n    }\n    super.ngOnInit();\n    this.messagesGroupId = `${this.inputId}-messages`;\n    this.placeholder ??= this.i18n.t('select.placeholder');\n  }\n\n  ngOnChanges({ value }: SimpleChanges): void {\n    if (value && this.value && this.options?.length) {\n      if (!this.findOption(this.value, this.options)) {\n        this.value = undefined;\n      }\n    }\n  }\n\n  /** @internal */\n  onChange() {\n    // Comme on a le double binding, la valeur est déjà à jour\n    // this.value = (<HTMLSelectElement>event?.target)?.value;\n    this.selectChange.emit(this.value);\n  }\n\n  trackByIndex = (index: number): number => {\n    return index;\n  };\n\n  private findOption(value: any, options: DsfrSelectOption[]): any {\n    return options?.find(\n      (option) =>\n        this.compareFn(value, option.value) ||\n        option.value === value ||\n        (option.options && this.findOption(value, option.options)),\n    );\n  }\n}\n","<div\n  [ngClass]=\"{\n    'fr-select-group': true,\n    'fr-select-group--error': error,\n    'fr-select-group--valid': valid,\n    'fr-select-group--disabled': disabled\n  }\">\n  <label class=\"fr-label\" [for]=\"inputId\">\n    <ng-container *ngIf=\"label\">{{ label }}</ng-container>\n    <ng-content *ngIf=\"!label\" select=\"[label]\"></ng-content>\n    <span *ngIf=\"hint\" class=\"fr-hint-text\">{{ hint }}</span>\n  </label>\n\n  <!-- Select sans groupes. Duplication du select pour pb perf. affichage d'options sur firefox si utilisation d'un ng-container -->\n  <select\n    *ngIf=\"noGroup\"\n    class=\"fr-select\"\n    [id]=\"inputId\"\n    [attr.name]=\"name || null\"\n    [(ngModel)]=\"value\"\n    [attr.aria-describedby]=\"messagesGroupId\"\n    [attr.aria-label]=\"ariaLabel\"\n    [disabled]=\"disabled\"\n    [attr.required]=\"required || null\"\n    [compareWith]=\"compareFn\"\n    (change)=\"onChange()\">\n    <option *ngIf=\"!value\" [value]=\"value\" selected hidden disabled>{{ placeholder }}</option>\n    <option\n      *ngFor=\"let option of options; trackBy: trackByIndex\"\n      [ngValue]=\"option.value\"\n      [attr.aria-selected]=\"compareFn(option.value, value)\"\n      [disabled]=\"option.disabled || null\">\n      {{ option.label }}\n    </option>\n  </select>\n\n  <!-- Select avec groupes. Ne sera pas optimisé pour FF, voir pour un fix avec angular 17 -->\n  <select\n    *ngIf=\"!noGroup\"\n    class=\"fr-select\"\n    [id]=\"inputId\"\n    [attr.name]=\"name || null\"\n    [(ngModel)]=\"value\"\n    [attr.aria-describedby]=\"messagesGroupId\"\n    [attr.aria-label]=\"ariaLabel\"\n    [disabled]=\"disabled\"\n    [attr.required]=\"required || null\"\n    [compareWith]=\"compareFn\"\n    (change)=\"onChange()\">\n    <option *ngIf=\"!value\" [value]=\"value\" selected hidden disabled>{{ placeholder }}</option>\n    <ng-container *ngFor=\"let option of options; trackBy: trackByIndex\">\n      <option\n        *ngIf=\"!option.options\"\n        [ngValue]=\"option.value\"\n        [attr.aria-selected]=\"compareFn(option.value, value)\"\n        [disabled]=\"option.disabled || null\">\n        {{ option.label }}\n      </option>\n    </ng-container>\n    <ng-container *ngFor=\"let option of options; trackBy: trackByIndex\">\n      <optgroup *ngIf=\"option.options\" [label]=\"option.label\">\n        <option\n          *ngFor=\"let option of option.options; trackBy: trackByIndex\"\n          [ngValue]=\"option.value\"\n          [attr.aria-selected]=\"compareFn(option.value, value)\"\n          [disabled]=\"option.disabled || null\">\n          {{ option.label }}\n        </option>\n      </optgroup>\n    </ng-container>\n  </select>\n\n  <div [id]=\"messagesGroupId\" class=\"fr-messages-group\" aria-live=\"polite\">\n    <p *ngIf=\"error\" class=\"fr-message fr-message--error\">{{ error }}</p>\n    <p *ngIf=\"valid\" class=\"fr-message fr-message--valid\">{{ valid }}</p>\n  </div>\n</div>\n"]}