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

125 lines 17.6 kB
import { Component, EventEmitter, forwardRef, Input, Output, ViewChild, ViewEncapsulation, } from '@angular/core'; import { NG_VALUE_ACCESSOR } from '@angular/forms'; import { DefaultControlComponent, newUniqueId } from '../../shared'; import * as i0 from "@angular/core"; import * as i1 from "../../shared"; import * as i2 from "@angular/common"; export class DsfrUploadComponent extends DefaultControlComponent { constructor(_renderer, _elementRef, i18n) { super(); this._renderer = _renderer; this._elementRef = _elementRef; this.i18n = i18n; /** Permet l'ajout de plusieurs fichiers à la fois. */ this.multiple = false; /** Evénement émis à la sélection d'un ou plusieurs fichiers. */ this.fileSelect = new EventEmitter(); /** Evénement émis lorsque l'utilisateur annule la sélection. */ this.fileCancel = new EventEmitter(); } /** @deprecated (@since 1.4.0) utiliser hint à la place. */ get description() { return this.hint; } /** * Description pour l'upload (précise les contraintes au niveau du ou des fichiers attendus : format, poids attendus, nombre de fichiers possible…). * * @deprecated (@since 1.4.0) utiliser hint à la place. * * @internal */ set description(value) { this.hint = value; } /** * Surcharge permettant de gérer la mise à jour du DOM. * * @internal */ writeValue(value) { // afin de se prémunir des initialisations avec une chaîne de caractères (vide ou pas) il faut s'assurer du type // d'argument transmis en entrée et forcer la chaîne vide le cas échéant const isTypeFileList = Object.prototype.toString.call(value).includes(FileList.name); if (!isTypeFileList) { value = ''; } super.writeValue(value); if (this.inputFile) { if (isTypeFileList && value?.length) { this._renderer.setProperty(this.inputFile.nativeElement, 'files', value); } else { this._renderer.setProperty(this.inputFile.nativeElement, 'value', ''); this._renderer.setProperty(this.inputFile.nativeElement, 'files', null); } } } /** @internal */ ngOnInit() { super.ngOnInit(); this.name ??= 'file-upload'; // 'file-upload' est la valeur statique jusqu'en 1.3 this.messagesGroupId = newUniqueId(); } /** @internal */ ngOnChanges(changes) { if ([changes['label']]) this.label ||= this.i18n.t('upload.label'); // default if ([changes['hint']]) this.hint ||= this.i18n.t('upload.hint'); // default } /** * Permet de programmatiquement réinitialiser le champ après une sélection de fichier(s). */ reset() { this.inputFile.nativeElement.value = ''; } /** @internal */ onCancel() { this.fileCancel.emit(); } /** @internal */ onFileChange(event) { const target = event.target; this.value = target.files ?? undefined; if (this.value) this.fileSelect.emit(this.value); } /** @internal */ hasMessages() { return !!this.error; } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: DsfrUploadComponent, deps: [{ token: i0.Renderer2 }, { token: i0.ElementRef }, { token: i1.I18nService }], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: DsfrUploadComponent, selector: "dsfr-form-upload, dsfr-upload", inputs: { error: "error", multiple: "multiple", accept: "accept", description: "description" }, outputs: { fileSelect: "fileSelect", fileCancel: "fileCancel" }, providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => DsfrUploadComponent), multi: true, }, ], viewQueries: [{ propertyName: "inputFile", first: true, predicate: ["inputFile"], descendants: true, static: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<div class=\"fr-upload-group\" [ngClass]=\"{ 'fr-input-group--error': error, 'fr-input-group--disabled': disabled }\">\n <label class=\"fr-label\" [for]=\"inputId\"\n >{{ label }}\n <span class=\"fr-hint-text\">{{ hint }}</span>\n </label>\n <input\n #inputFile\n class=\"fr-upload\"\n type=\"file\"\n [id]=\"inputId || null\"\n [name]=\"name || null\"\n [multiple]=\"multiple\"\n [attr.accept]=\"accept || null\"\n [disabled]=\"disabled\"\n [attr.aria-describedby]=\"hasMessages() ? messagesGroupId : null\"\n (change)=\"onFileChange($event)\"\n (cancel)=\"onCancel()\" />\n <p *ngIf=\"hasMessages()\" [id]=\"messagesGroupId\" class=\"fr-error-text\">\n {{ error }}\n </p>\n</div>\n", dependencies: [{ kind: "directive", type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], encapsulation: i0.ViewEncapsulation.None }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: DsfrUploadComponent, decorators: [{ type: Component, args: [{ selector: 'dsfr-form-upload, dsfr-upload', encapsulation: ViewEncapsulation.None, providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => DsfrUploadComponent), multi: true, }, ], template: "<div class=\"fr-upload-group\" [ngClass]=\"{ 'fr-input-group--error': error, 'fr-input-group--disabled': disabled }\">\n <label class=\"fr-label\" [for]=\"inputId\"\n >{{ label }}\n <span class=\"fr-hint-text\">{{ hint }}</span>\n </label>\n <input\n #inputFile\n class=\"fr-upload\"\n type=\"file\"\n [id]=\"inputId || null\"\n [name]=\"name || null\"\n [multiple]=\"multiple\"\n [attr.accept]=\"accept || null\"\n [disabled]=\"disabled\"\n [attr.aria-describedby]=\"hasMessages() ? messagesGroupId : null\"\n (change)=\"onFileChange($event)\"\n (cancel)=\"onCancel()\" />\n <p *ngIf=\"hasMessages()\" [id]=\"messagesGroupId\" class=\"fr-error-text\">\n {{ error }}\n </p>\n</div>\n" }] }], ctorParameters: function () { return [{ type: i0.Renderer2 }, { type: i0.ElementRef }, { type: i1.I18nService }]; }, propDecorators: { inputFile: [{ type: ViewChild, args: ['inputFile', { static: true }] }], error: [{ type: Input }], multiple: [{ type: Input }], accept: [{ type: Input }], fileSelect: [{ type: Output }], fileCancel: [{ type: Output }], description: [{ type: Input }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"upload.component.js","sourceRoot":"","sources":["../../../../../../projects/ngx-dsfr-components/src/lib/forms/form-upload/upload.component.ts","../../../../../../projects/ngx-dsfr-components/src/lib/forms/form-upload/upload.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ElementRef,\n  EventEmitter,\n  forwardRef,\n  Input,\n  OnChanges,\n  OnInit,\n  Output,\n  Renderer2,\n  SimpleChanges,\n  ViewChild,\n  ViewEncapsulation,\n} from '@angular/core';\nimport { NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { DefaultControlComponent, I18nService, newUniqueId } from '../../shared';\n\n@Component({\n  selector: 'dsfr-form-upload, dsfr-upload',\n  templateUrl: './upload.component.html',\n  encapsulation: ViewEncapsulation.None,\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => DsfrUploadComponent),\n      multi: true,\n    },\n  ],\n})\nexport class DsfrUploadComponent extends DefaultControlComponent<any> implements OnInit, OnChanges {\n  /** @internal */\n  @ViewChild('inputFile', { static: true })\n  inputFile: ElementRef;\n\n  /** Texte d'erreur. */\n  @Input() error: string;\n\n  /** Permet l'ajout de plusieurs fichiers à la fois. */\n  @Input() multiple = false;\n\n  /** Spécifie un filtre pour les types de fichiers que l'utilisateur peut sélectionner (`@since 1.9.0`). */\n  @Input() accept: string;\n\n  /** Evénement émis à la sélection d'un ou plusieurs fichiers. */\n  @Output()\n  fileSelect = new EventEmitter<FileList>();\n\n  /** Evénement émis lorsque l'utilisateur annule la sélection. */\n  @Output()\n  fileCancel = new EventEmitter<FileList>();\n\n  /** @internal */\n  messagesGroupId: string;\n\n  constructor(\n    private readonly _renderer: Renderer2,\n    private readonly _elementRef: ElementRef,\n    private readonly i18n: I18nService,\n  ) {\n    super();\n  }\n\n  /** @deprecated (@since 1.4.0) utiliser hint à la place. */\n  get description(): string | undefined {\n    return this.hint;\n  }\n\n  /**\n   * Description pour l'upload (précise les contraintes au niveau du ou des fichiers attendus : format, poids attendus, nombre de fichiers possible…).\n   *\n   * @deprecated (@since 1.4.0) utiliser hint à la place.\n   *\n   * @internal\n   */\n  @Input() set description(value: string | undefined) {\n    this.hint = value;\n  }\n\n  /**\n   * Surcharge permettant de gérer la mise à jour du DOM.\n   *\n   * @internal\n   */\n  override writeValue(value: any | undefined): void {\n    // afin de se prémunir des initialisations avec une chaîne de caractères (vide ou pas) il faut s'assurer du type\n    // d'argument transmis en entrée et forcer la chaîne vide le cas échéant\n    const isTypeFileList = Object.prototype.toString.call(value).includes(FileList.name);\n    if (!isTypeFileList) {\n      value = '';\n    }\n    super.writeValue(value);\n    if (this.inputFile) {\n      if (isTypeFileList && value?.length) {\n        this._renderer.setProperty(this.inputFile.nativeElement, 'files', value);\n      } else {\n        this._renderer.setProperty(this.inputFile.nativeElement, 'value', '');\n        this._renderer.setProperty(this.inputFile.nativeElement, 'files', null);\n      }\n    }\n  }\n\n  /** @internal */\n  ngOnInit() {\n    super.ngOnInit();\n    this.name ??= 'file-upload'; // 'file-upload' est la valeur statique jusqu'en 1.3\n    this.messagesGroupId = newUniqueId();\n  }\n\n  /** @internal */\n  ngOnChanges(changes: SimpleChanges) {\n    if ([changes['label']]) this.label ||= this.i18n.t('upload.label'); // default\n    if ([changes['hint']]) this.hint ||= this.i18n.t('upload.hint'); // default\n  }\n\n  /**\n   * Permet de programmatiquement réinitialiser le champ après une sélection de fichier(s).\n   */\n  reset() {\n    this.inputFile.nativeElement.value = '';\n  }\n\n  /** @internal */\n  onCancel() {\n    this.fileCancel.emit();\n  }\n\n  /** @internal */\n  onFileChange(event: Event) {\n    const target = <HTMLInputElement>event.target;\n    this.value = target.files ?? undefined;\n    if (this.value) this.fileSelect.emit(this.value);\n  }\n\n  /** @internal */\n  hasMessages(): boolean {\n    return !!this.error;\n  }\n}\n","<div class=\"fr-upload-group\" [ngClass]=\"{ 'fr-input-group--error': error, 'fr-input-group--disabled': disabled }\">\n  <label class=\"fr-label\" [for]=\"inputId\"\n    >{{ label }}\n    <span class=\"fr-hint-text\">{{ hint }}</span>\n  </label>\n  <input\n    #inputFile\n    class=\"fr-upload\"\n    type=\"file\"\n    [id]=\"inputId || null\"\n    [name]=\"name || null\"\n    [multiple]=\"multiple\"\n    [attr.accept]=\"accept || null\"\n    [disabled]=\"disabled\"\n    [attr.aria-describedby]=\"hasMessages() ? messagesGroupId : null\"\n    (change)=\"onFileChange($event)\"\n    (cancel)=\"onCancel()\" />\n  <p *ngIf=\"hasMessages()\" [id]=\"messagesGroupId\" class=\"fr-error-text\">\n    {{ error }}\n  </p>\n</div>\n"]}