@villedemontreal/angular-ui 13.0.0 → 13.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (89) hide show
  1. package/esm2020/lib/alert/alert.component.mjs +2 -2
  2. package/esm2020/lib/alert/index.mjs +2 -2
  3. package/esm2020/lib/alert/module.mjs +2 -2
  4. package/esm2020/lib/avatar/avatar.component.mjs +2 -2
  5. package/esm2020/lib/avatar/index.mjs +2 -2
  6. package/esm2020/lib/avatar/module.mjs +2 -2
  7. package/esm2020/lib/badge/badge.component.mjs +2 -2
  8. package/esm2020/lib/badge/index.mjs +2 -2
  9. package/esm2020/lib/badge/module.mjs +2 -2
  10. package/esm2020/lib/bao.module.mjs +9 -5
  11. package/esm2020/lib/breadcrumb/breadcrumb.component.mjs +2 -2
  12. package/esm2020/lib/breadcrumb/index.mjs +2 -2
  13. package/esm2020/lib/breadcrumb/module.mjs +2 -2
  14. package/esm2020/lib/button/button.component.mjs +2 -2
  15. package/esm2020/lib/button/index.mjs +2 -2
  16. package/esm2020/lib/button/module.mjs +2 -2
  17. package/esm2020/lib/card/card.component.mjs +2 -2
  18. package/esm2020/lib/card/index.mjs +2 -2
  19. package/esm2020/lib/card/module.mjs +2 -2
  20. package/esm2020/lib/checkbox/checkbox-group.component.mjs +2 -2
  21. package/esm2020/lib/checkbox/checkbox.component.mjs +1 -1
  22. package/esm2020/lib/checkbox/index.mjs +2 -2
  23. package/esm2020/lib/checkbox/module.mjs +2 -2
  24. package/esm2020/lib/common-components/error-text/errorText.component.mjs +2 -2
  25. package/esm2020/lib/common-components/guiding-text/guidingText.component.mjs +2 -2
  26. package/esm2020/lib/common-components/index.mjs +2 -2
  27. package/esm2020/lib/common-components/label-text/labelText.component.mjs +4 -4
  28. package/esm2020/lib/common-components/module.mjs +6 -3
  29. package/esm2020/lib/common-components/title-text/titleText.component.mjs +2 -2
  30. package/esm2020/lib/core/colors.mjs +1 -1
  31. package/esm2020/lib/dropdown-menu/dropdown-menu.component.mjs +9 -2
  32. package/esm2020/lib/dropdown-menu/index.mjs +2 -2
  33. package/esm2020/lib/dropdown-menu/module.mjs +2 -2
  34. package/esm2020/lib/file/file-input.component.mjs +319 -0
  35. package/esm2020/lib/file/file-intl.mjs +65 -0
  36. package/esm2020/lib/file/file-preview.component.mjs +86 -0
  37. package/esm2020/lib/file/index.mjs +10 -0
  38. package/esm2020/lib/file/module.mjs +58 -0
  39. package/esm2020/lib/header-info/header-info.component.mjs +2 -2
  40. package/esm2020/lib/header-info/index.mjs +2 -2
  41. package/esm2020/lib/header-info/module.mjs +2 -2
  42. package/esm2020/lib/hyperlink/hyperlink.component.mjs +2 -2
  43. package/esm2020/lib/hyperlink/index.mjs +2 -2
  44. package/esm2020/lib/hyperlink/module.mjs +2 -2
  45. package/esm2020/lib/icon/bao-icon-registry.mjs +2 -2
  46. package/esm2020/lib/icon/icon.component.mjs +2 -2
  47. package/esm2020/lib/icon/icons-dictionary.mjs +3 -2
  48. package/esm2020/lib/icon/index.mjs +2 -2
  49. package/esm2020/lib/icon/module.mjs +2 -2
  50. package/esm2020/lib/list/index.mjs +2 -2
  51. package/esm2020/lib/list/list.component.mjs +2 -2
  52. package/esm2020/lib/list/module.mjs +2 -2
  53. package/esm2020/lib/modal/index.mjs +2 -2
  54. package/esm2020/lib/modal/modal-animations.mjs +2 -2
  55. package/esm2020/lib/modal/modal-config.mjs +2 -2
  56. package/esm2020/lib/modal/modal-container.mjs +1 -1
  57. package/esm2020/lib/modal/modal-directives.mjs +2 -2
  58. package/esm2020/lib/modal/modal-ref.mjs +1 -1
  59. package/esm2020/lib/modal/modal.mjs +2 -2
  60. package/esm2020/lib/modal/module.mjs +2 -2
  61. package/esm2020/lib/radio/index.mjs +2 -2
  62. package/esm2020/lib/radio/module.mjs +2 -2
  63. package/esm2020/lib/radio/radio-group.component.mjs +2 -2
  64. package/esm2020/lib/radio/radio.component.mjs +1 -1
  65. package/esm2020/lib/shared/enum/display-mode.mjs +1 -1
  66. package/esm2020/lib/shared/index.mjs +2 -2
  67. package/esm2020/lib/summary/index.mjs +2 -2
  68. package/esm2020/lib/summary/list-summary.component.mjs +2 -2
  69. package/esm2020/lib/summary/module.mjs +2 -2
  70. package/esm2020/lib/summary/summary.component.mjs +2 -2
  71. package/esm2020/lib/tabs/index.mjs +2 -2
  72. package/esm2020/lib/tabs/module.mjs +2 -2
  73. package/esm2020/lib/tabs/tabs.component.mjs +2 -2
  74. package/esm2020/lib/tag/index.mjs +2 -2
  75. package/esm2020/lib/tag/module.mjs +2 -2
  76. package/esm2020/lib/tag/tag.component.mjs +2 -2
  77. package/esm2020/public-api.mjs +3 -2
  78. package/fesm2015/villedemontreal-angular-ui.mjs +596 -78
  79. package/fesm2015/villedemontreal-angular-ui.mjs.map +1 -1
  80. package/fesm2020/villedemontreal-angular-ui.mjs +596 -78
  81. package/fesm2020/villedemontreal-angular-ui.mjs.map +1 -1
  82. package/lib/bao.module.d.ts +2 -1
  83. package/lib/file/file-input.component.d.ts +123 -0
  84. package/lib/file/file-intl.d.ts +44 -0
  85. package/lib/file/file-preview.component.d.ts +29 -0
  86. package/lib/file/index.d.ts +4 -0
  87. package/lib/file/module.d.ts +13 -0
  88. package/package.json +1 -1
  89. package/public-api.d.ts +1 -0
@@ -0,0 +1,319 @@
1
+ /*
2
+ * Copyright (c) 2023 Ville de Montreal. All rights reserved.
3
+ * Licensed under the MIT license.
4
+ * See LICENSE file in the project root for full license information.
5
+ */
6
+ import { Component, ContentChildren, Directive, EventEmitter, forwardRef, HostListener, Input, Output, ViewChild, ViewChildren, ViewEncapsulation } from '@angular/core';
7
+ import { NG_VALUE_ACCESSOR } from '@angular/forms';
8
+ import { BaoErrorTextComponent } from '../common-components';
9
+ import { BaoFilePreviewComponent } from './file-preview.component';
10
+ import * as i0 from "@angular/core";
11
+ import * as i1 from "./file-intl";
12
+ import * as i2 from "../common-components/label-text/labelText.component";
13
+ import * as i3 from "../button/button.component";
14
+ import * as i4 from "../common-components/error-text/errorText.component";
15
+ import * as i5 from "@angular/common";
16
+ /**
17
+ * Unique number to generate a unique ID
18
+ */
19
+ let fileInputUniqueId = 0;
20
+ let fileTextUniqueId = 0;
21
+ export class BaoFileInputComponent {
22
+ constructor(intl, elementRef, renderer, cdr) {
23
+ this.intl = intl;
24
+ this.elementRef = elementRef;
25
+ this.renderer = renderer;
26
+ this.cdr = cdr;
27
+ /**
28
+ * Size of the file input label
29
+ */
30
+ this.size = 'medium';
31
+ /**
32
+ * Maximum size accepted for uploaded files
33
+ */
34
+ this.maximalFileSize = -1;
35
+ /**
36
+ * Accepted types of files
37
+ */
38
+ this.acceptedMIMETypes = [];
39
+ /**
40
+ * Is field required
41
+ */
42
+ this.required = false;
43
+ /**
44
+ * Is field disabled
45
+ */
46
+ this.disabled = false;
47
+ /**
48
+ * File selected to be uploaded
49
+ */
50
+ this.uploadedFile = new EventEmitter();
51
+ this.insertDefaultInstructions = false;
52
+ this.isFileTooBig = false;
53
+ this.isFileTypeInvalid = false;
54
+ /**
55
+ * Saves the registerOnChange function so the component can call it whenever it wants.
56
+ */
57
+ // eslint-disable-next-line @typescript-eslint/no-empty-function, @typescript-eslint/no-unused-vars
58
+ this.propagateChange = (_) => { };
59
+ /**
60
+ * Saves the registerOnTouched function so the component can call it whenever it wants.
61
+ */
62
+ // eslint-disable-next-line @typescript-eslint/no-empty-function
63
+ this.propagateTouched = () => { };
64
+ this._intlChanges = intl.changes.subscribe(() => this.cdr.markForCheck());
65
+ }
66
+ get nativeElement() {
67
+ return this.elementRef.nativeElement;
68
+ }
69
+ enterKeyEvent() {
70
+ if (document.activeElement.id === this.inputId) {
71
+ document.getElementById(this.inputId).click();
72
+ }
73
+ }
74
+ tabKeyEvent() {
75
+ if (document.activeElement.id === this.inputId) {
76
+ this.renderer.addClass(this.dropzoneElement.nativeElement, 'dropzone-focus');
77
+ }
78
+ }
79
+ shiftTabKeyEvent() {
80
+ if (document.activeElement.id === this.inputId) {
81
+ this.renderer.addClass(this.dropzoneElement.nativeElement, 'dropzone-focus');
82
+ }
83
+ }
84
+ ngAfterContentInit() {
85
+ this._errorForm.changes.subscribe(() => this.setErrorTextsAttribute());
86
+ if (!this.inputId) {
87
+ this.inputId = `file-input-${fileInputUniqueId++}`;
88
+ }
89
+ // If no content was added for dropzone instructions, add default text.
90
+ const dropzoneElement = Array.from(this.nativeElement.children).find((el) => el.className === 'file-drop-zone');
91
+ if (!Array.from(dropzoneElement.children).find(el => el.localName === 'bao-file-dropzone-instructions')) {
92
+ this.insertDefaultInstructions = true;
93
+ }
94
+ this.setDescribedByAttribute();
95
+ this._files.changes.subscribe((files) => {
96
+ const filesList = files.map((el) => el.file);
97
+ this.setValue(filesList);
98
+ });
99
+ }
100
+ ngAfterViewInit() {
101
+ this._errorTexts.changes.subscribe(() => this.setErrorTextsAttribute());
102
+ }
103
+ ngOnDestroy() {
104
+ this._intlChanges.unsubscribe();
105
+ }
106
+ /**
107
+ * Implements ControlValueAccessor interface
108
+ */
109
+ writeValue(obj) {
110
+ this._value = obj;
111
+ }
112
+ /**
113
+ * Implements ControlValueAccessor interface
114
+ */
115
+ registerOnChange(fn) {
116
+ this.propagateChange = fn;
117
+ }
118
+ /**
119
+ * Implements ControlValueAccessor interface
120
+ */
121
+ registerOnTouched(fn) {
122
+ this.propagateTouched = fn;
123
+ }
124
+ /**
125
+ * Implements ControlValueAccessor interface
126
+ */
127
+ setDisabledState(isDisabled) {
128
+ this.disabled = isDisabled;
129
+ }
130
+ uploadFile(file) {
131
+ if (!this.disabled) {
132
+ this.isFileTypeInvalid = false;
133
+ this.isFileTooBig = false;
134
+ this.uploader.nativeElement.value = '';
135
+ if (this.maximalFileSize > 0 && file.size > this.maximalFileSize) {
136
+ this.isFileTooBig = true;
137
+ }
138
+ if (this.acceptedMIMETypes.length > 0 &&
139
+ this.acceptedMIMETypes.indexOf(file.type) < 0) {
140
+ this.isFileTypeInvalid = true;
141
+ }
142
+ if (!this.isFileTooBig && !this.isFileTypeInvalid) {
143
+ this.uploadedFile.emit(file);
144
+ }
145
+ }
146
+ }
147
+ setValue(value) {
148
+ this._value = value;
149
+ this.propagateChange(this._value);
150
+ this.propagateTouched();
151
+ }
152
+ setDescribedByAttribute() {
153
+ const helperText = Array.from(this.nativeElement.children).find((el) => el.localName === 'bao-guiding-text').firstElementChild;
154
+ if (helperText) {
155
+ this._helperTextId = `bao-guiding-text-${fileTextUniqueId++}`;
156
+ this.renderer.setAttribute(helperText, 'id', this._helperTextId);
157
+ const inputElement = Array.from(this.nativeElement.children)
158
+ .find((el) => el.className == 'file-drop-zone')
159
+ .children.item(1);
160
+ this.renderer.setAttribute(inputElement, 'aria-describedby', this._helperTextId);
161
+ }
162
+ }
163
+ setErrorTextsAttribute() {
164
+ const textsIds = [];
165
+ const errors = Array.from(this.nativeElement.children).filter((el) => el.localName == 'bao-error');
166
+ errors.forEach((errorText) => {
167
+ const errorTextId = `bao-error-${fileTextUniqueId++}`;
168
+ this.renderer.setAttribute(errorText.firstElementChild, 'id', errorTextId);
169
+ textsIds.push(errorTextId);
170
+ });
171
+ const inputElement = Array.from(this.nativeElement.children)
172
+ .find((el) => el.classList.contains('file-drop-zone'))
173
+ .children.item(1);
174
+ if (this._helperTextId) {
175
+ textsIds.unshift(this._helperTextId);
176
+ }
177
+ this.renderer.setAttribute(inputElement, 'aria-describedby', textsIds.join(' '));
178
+ }
179
+ }
180
+ BaoFileInputComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: BaoFileInputComponent, deps: [{ token: i1.BaoFileIntl }, { token: i0.ElementRef }, { token: i0.Renderer2 }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
181
+ BaoFileInputComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.12", type: BaoFileInputComponent, selector: "bao-file-input, [bao-file-input]", inputs: { inputId: ["id", "inputId"], label: "label", size: "size", maximalFileSize: "maximalFileSize", acceptedMIMETypes: "acceptedMIMETypes", required: "required", disabled: "disabled" }, outputs: { uploadedFile: "uploadedFile" }, host: { listeners: { "window:keyup.enter": "enterKeyEvent()", "window:keyup.tab": "tabKeyEvent()", "window:keyup.shift.tab": "shiftTabKeyEvent()" }, properties: { "class.bao-file-label-small": "size === \"small\"", "class.bao-file-label-medium": "size === \"medium\"", "class.bao-file-input-disabled": "disabled" }, classAttribute: "bao-file-input" }, providers: [
182
+ {
183
+ provide: NG_VALUE_ACCESSOR,
184
+ // tslint:disable-next-line:no-forward-ref
185
+ useExisting: forwardRef(() => BaoFileInputComponent),
186
+ multi: true
187
+ }
188
+ ], queries: [{ propertyName: "_files", predicate: BaoFilePreviewComponent, descendants: true }, { propertyName: "_errorForm", predicate: BaoErrorTextComponent, descendants: true }], viewQueries: [{ propertyName: "uploader", first: true, predicate: ["uploader"], descendants: true }, { propertyName: "dropzoneElement", first: true, predicate: ["dropzone"], descendants: true }, { propertyName: "_errorTexts", predicate: BaoErrorTextComponent, descendants: true }], ngImport: i0, template: "<label bao-label [required]=\"required\" [for]=\"inputId\">{{ label }}</label>\n<ng-content select=\"bao-guiding-text\"></ng-content>\n<div\n baoFileDrop\n class=\"file-drop-zone\"\n (fileDrop)=\"uploadFile($event)\"\n #dropzone\n>\n <button\n bao-button\n type=\"button\"\n displayType=\"utility\"\n level=\"secondary\"\n [disabled]=\"disabled\"\n (click)=\"uploader.click()\"\n aria-hidden=\"true\"\n tabIndex=\"-1\"\n >\n {{ intl.dropzoneButtonLabel }}\n </button>\n <input\n [id]=\"inputId\"\n type=\"file\"\n class=\"sr-only\"\n [disabled]=\"disabled\"\n (change)=\"uploadFile($event.target.files[0])\"\n #uploader\n />\n <ng-container\n ><div #ref>\n <ng-content select=\"bao-file-dropzone-instructions\"></ng-content></div\n ></ng-container>\n <ng-container *ngIf=\"ref.childNodes.length === 0\"\n ><bao-file-dropzone-instructions>{{\n intl.defaultDropzoneInstructions\n }}</bao-file-dropzone-instructions></ng-container\n >\n</div>\n<bao-error *ngIf=\"isFileTooBig\">\n {{ intl.fileTooBigErrorMessage }}\n</bao-error>\n<bao-error *ngIf=\"isFileTypeInvalid\">\n {{ intl.invalidFileTypeErrorMessage }}\n</bao-error>\n<ng-content select=\"bao-error\"></ng-content>\n<ng-content></ng-content>\n", styles: ["bao-file-input{width:100%;display:inline-flex;flex-direction:column}bao-file-input>ul{padding:0;margin:0}bao-file-input .bao-label>span{font-size:inherit;font-weight:inherit}bao-file-input.bao-file-label-small label{font-weight:700;font-size:.875rem;line-height:1.25rem}bao-file-input.bao-file-label-medium label{font-weight:700;font-size:1rem;line-height:1.5rem}bao-file-input .bao-guiding-text{margin-bottom:.5rem}bao-file-input .file-drop-zone{padding:.5rem;background-color:#fff;border-radius:.25rem;border-style:dashed;border-color:#ced4da;border-width:1px;display:inline-flex;align-items:center}bao-file-input .file-drop-zone:focus-within.dropzone-focus{box-shadow:0 0 0 .1875rem #98bcde;background-color:#eefaf8}bao-file-input .file-drop-zone>.bao-button{margin-right:.5rem}bao-file-input .file-drop-zone.drag-over{background-color:#eefaf8;border-color:#097d6c;cursor:drag}bao-file-input .file-drop-zone.drag-over>.bao-button{background-color:#eefaf8}bao-file-input.bao-file-input-disabled .file-drop-zone{background-color:#f8f9fa;border-color:#ced4da}bao-file-input.bao-file-input-disabled .file-drop-zone .bao-button{background-color:#f8f9fa}bao-file-input.bao-file-input-disabled .file-drop-zone .bao-button:hover{background-color:#f8f9fa}bao-file-input .bao-error{margin-top:.5rem}\n"], components: [{ type: i0.forwardRef(function () { return i2.BaoLabelTextComponent; }), selector: "bao-label, [bao-label]", inputs: ["required"] }, { type: i0.forwardRef(function () { return i3.BaoButtonComponent; }), selector: "button[bao-button]", inputs: ["displayType", "level", "size", "loading", "reversed", "loadingSpinnerAriaLabel", "fullWidth"] }, { type: i0.forwardRef(function () { return i4.BaoErrorTextComponent; }), selector: "bao-error, [bao-error]" }], directives: [{ type: i0.forwardRef(function () { return BaoFileDropDirective; }), selector: "[baoFileDrop]", outputs: ["fileDrop"] }, { type: i0.forwardRef(function () { return i5.NgIf; }), selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i0.forwardRef(function () { return BaoFileDropzoneIntructions; }), selector: "bao-file-dropzone-instructions, [bao-file-dropzone-instructions]" }], encapsulation: i0.ViewEncapsulation.None });
189
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: BaoFileInputComponent, decorators: [{
190
+ type: Component,
191
+ args: [{ selector: 'bao-file-input, [bao-file-input]', providers: [
192
+ {
193
+ provide: NG_VALUE_ACCESSOR,
194
+ // tslint:disable-next-line:no-forward-ref
195
+ useExisting: forwardRef(() => BaoFileInputComponent),
196
+ multi: true
197
+ }
198
+ ], encapsulation: ViewEncapsulation.None, host: {
199
+ class: 'bao-file-input',
200
+ '[class.bao-file-label-small]': 'size === "small"',
201
+ '[class.bao-file-label-medium]': 'size === "medium"',
202
+ '[class.bao-file-input-disabled]': 'disabled'
203
+ }, template: "<label bao-label [required]=\"required\" [for]=\"inputId\">{{ label }}</label>\n<ng-content select=\"bao-guiding-text\"></ng-content>\n<div\n baoFileDrop\n class=\"file-drop-zone\"\n (fileDrop)=\"uploadFile($event)\"\n #dropzone\n>\n <button\n bao-button\n type=\"button\"\n displayType=\"utility\"\n level=\"secondary\"\n [disabled]=\"disabled\"\n (click)=\"uploader.click()\"\n aria-hidden=\"true\"\n tabIndex=\"-1\"\n >\n {{ intl.dropzoneButtonLabel }}\n </button>\n <input\n [id]=\"inputId\"\n type=\"file\"\n class=\"sr-only\"\n [disabled]=\"disabled\"\n (change)=\"uploadFile($event.target.files[0])\"\n #uploader\n />\n <ng-container\n ><div #ref>\n <ng-content select=\"bao-file-dropzone-instructions\"></ng-content></div\n ></ng-container>\n <ng-container *ngIf=\"ref.childNodes.length === 0\"\n ><bao-file-dropzone-instructions>{{\n intl.defaultDropzoneInstructions\n }}</bao-file-dropzone-instructions></ng-container\n >\n</div>\n<bao-error *ngIf=\"isFileTooBig\">\n {{ intl.fileTooBigErrorMessage }}\n</bao-error>\n<bao-error *ngIf=\"isFileTypeInvalid\">\n {{ intl.invalidFileTypeErrorMessage }}\n</bao-error>\n<ng-content select=\"bao-error\"></ng-content>\n<ng-content></ng-content>\n", styles: ["bao-file-input{width:100%;display:inline-flex;flex-direction:column}bao-file-input>ul{padding:0;margin:0}bao-file-input .bao-label>span{font-size:inherit;font-weight:inherit}bao-file-input.bao-file-label-small label{font-weight:700;font-size:.875rem;line-height:1.25rem}bao-file-input.bao-file-label-medium label{font-weight:700;font-size:1rem;line-height:1.5rem}bao-file-input .bao-guiding-text{margin-bottom:.5rem}bao-file-input .file-drop-zone{padding:.5rem;background-color:#fff;border-radius:.25rem;border-style:dashed;border-color:#ced4da;border-width:1px;display:inline-flex;align-items:center}bao-file-input .file-drop-zone:focus-within.dropzone-focus{box-shadow:0 0 0 .1875rem #98bcde;background-color:#eefaf8}bao-file-input .file-drop-zone>.bao-button{margin-right:.5rem}bao-file-input .file-drop-zone.drag-over{background-color:#eefaf8;border-color:#097d6c;cursor:drag}bao-file-input .file-drop-zone.drag-over>.bao-button{background-color:#eefaf8}bao-file-input.bao-file-input-disabled .file-drop-zone{background-color:#f8f9fa;border-color:#ced4da}bao-file-input.bao-file-input-disabled .file-drop-zone .bao-button{background-color:#f8f9fa}bao-file-input.bao-file-input-disabled .file-drop-zone .bao-button:hover{background-color:#f8f9fa}bao-file-input .bao-error{margin-top:.5rem}\n"] }]
204
+ }], ctorParameters: function () { return [{ type: i1.BaoFileIntl }, { type: i0.ElementRef }, { type: i0.Renderer2 }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { inputId: [{
205
+ type: Input,
206
+ args: ['id']
207
+ }], label: [{
208
+ type: Input
209
+ }], size: [{
210
+ type: Input
211
+ }], maximalFileSize: [{
212
+ type: Input
213
+ }], acceptedMIMETypes: [{
214
+ type: Input
215
+ }], required: [{
216
+ type: Input
217
+ }], disabled: [{
218
+ type: Input
219
+ }], uploadedFile: [{
220
+ type: Output
221
+ }], _files: [{
222
+ type: ContentChildren,
223
+ args: [BaoFilePreviewComponent, { descendants: true }]
224
+ }], _errorForm: [{
225
+ type: ContentChildren,
226
+ args: [BaoErrorTextComponent, { descendants: true }]
227
+ }], _errorTexts: [{
228
+ type: ViewChildren,
229
+ args: [BaoErrorTextComponent]
230
+ }], uploader: [{
231
+ type: ViewChild,
232
+ args: ['uploader', { static: false }]
233
+ }], dropzoneElement: [{
234
+ type: ViewChild,
235
+ args: ['dropzone', { static: false }]
236
+ }], enterKeyEvent: [{
237
+ type: HostListener,
238
+ args: ['window:keyup.enter']
239
+ }], tabKeyEvent: [{
240
+ type: HostListener,
241
+ args: ['window:keyup.tab']
242
+ }], shiftTabKeyEvent: [{
243
+ type: HostListener,
244
+ args: ['window:keyup.shift.tab']
245
+ }] } });
246
+ export class BaoFileDropDirective {
247
+ constructor() {
248
+ this.fileDrop = new EventEmitter();
249
+ this._isDragOver = false;
250
+ }
251
+ onDragOver(event) {
252
+ this.preventAndStop(event);
253
+ this._isDragOver = true;
254
+ }
255
+ onDragLeave(event) {
256
+ this.preventAndStop(event);
257
+ this._isDragOver = false;
258
+ }
259
+ onDrop(event) {
260
+ this.preventAndStop(event);
261
+ this._isDragOver = false;
262
+ const transfer = this.getDataTransfer(event);
263
+ this.fileDrop.emit(transfer.files[0]);
264
+ }
265
+ preventAndStop(event) {
266
+ event.preventDefault();
267
+ event.stopPropagation();
268
+ }
269
+ getDataTransfer(event) {
270
+ // eslint-disable-next-line @typescript-eslint/no-unsafe-return
271
+ return event.dataTransfer
272
+ ? event.dataTransfer
273
+ : event.originalEvent.dataTransfer;
274
+ }
275
+ }
276
+ BaoFileDropDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: BaoFileDropDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
277
+ BaoFileDropDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "13.3.12", type: BaoFileDropDirective, selector: "[baoFileDrop]", outputs: { fileDrop: "fileDrop" }, host: { listeners: { "dragover": "onDragOver($event)", "dragleave": "onDragLeave($event)", "drop": "onDrop($event)" }, properties: { "class.drag-over": "_isDragOver == true" } }, ngImport: i0 });
278
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: BaoFileDropDirective, decorators: [{
279
+ type: Directive,
280
+ args: [{
281
+ selector: '[baoFileDrop]',
282
+ host: { '[class.drag-over]': '_isDragOver == true' }
283
+ }]
284
+ }], propDecorators: { fileDrop: [{
285
+ type: Output
286
+ }], onDragOver: [{
287
+ type: HostListener,
288
+ args: ['dragover', ['$event']]
289
+ }], onDragLeave: [{
290
+ type: HostListener,
291
+ args: ['dragleave', ['$event']]
292
+ }], onDrop: [{
293
+ type: HostListener,
294
+ args: ['drop', ['$event']]
295
+ }] } });
296
+ export class BaoFileDropzoneIntructions {
297
+ constructor(renderer, elementRef) {
298
+ this.renderer = renderer;
299
+ this.elementRef = elementRef;
300
+ }
301
+ get nativeElement() {
302
+ return this.elementRef.nativeElement;
303
+ }
304
+ ngAfterContentInit() {
305
+ this.renderer.setAttribute(this.nativeElement, 'aria-hidden', 'true');
306
+ }
307
+ }
308
+ BaoFileDropzoneIntructions.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: BaoFileDropzoneIntructions, deps: [{ token: i0.Renderer2 }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Directive });
309
+ BaoFileDropzoneIntructions.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "13.3.12", type: BaoFileDropzoneIntructions, selector: "bao-file-dropzone-instructions, [bao-file-dropzone-instructions]", host: { classAttribute: "bao-file-dropzone-instructions" }, ngImport: i0 });
310
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: BaoFileDropzoneIntructions, decorators: [{
311
+ type: Directive,
312
+ args: [{
313
+ selector: 'bao-file-dropzone-instructions, [bao-file-dropzone-instructions]',
314
+ host: {
315
+ class: 'bao-file-dropzone-instructions'
316
+ }
317
+ }]
318
+ }], ctorParameters: function () { return [{ type: i0.Renderer2 }, { type: i0.ElementRef }]; } });
319
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-input.component.js","sourceRoot":"","sources":["../../../../../projects/angular-ui/src/lib/file/file-input.component.ts","../../../../../projects/angular-ui/src/lib/file/file-input.component.html"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2023 Ville de Montreal. All rights reserved.\n * Licensed under the MIT license.\n * See LICENSE file in the project root for full license information.\n */\n\nimport {\n  AfterContentInit,\n  AfterViewInit,\n  ChangeDetectorRef,\n  Component,\n  ContentChildren,\n  Directive,\n  ElementRef,\n  EventEmitter,\n  forwardRef,\n  HostListener,\n  Input,\n  OnDestroy,\n  Output,\n  QueryList,\n  Renderer2,\n  ViewChild,\n  ViewChildren,\n  ViewEncapsulation\n} from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { BaoErrorTextComponent } from '../common-components';\nimport { Subscription } from 'rxjs';\nimport { BaoFileIntl } from './file-intl';\nimport { BaoFilePreviewComponent } from './file-preview.component';\n\n/**\n * Unique number to generate a unique ID\n */\nlet fileInputUniqueId = 0;\nlet fileTextUniqueId = 0;\n\n@Component({\n  selector: 'bao-file-input, [bao-file-input]',\n  templateUrl: './file-input.component.html',\n  styleUrls: ['./file-input.component.scss'],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      // tslint:disable-next-line:no-forward-ref\n      useExisting: forwardRef(() => BaoFileInputComponent),\n      multi: true\n    }\n  ],\n  encapsulation: ViewEncapsulation.None,\n  host: {\n    class: 'bao-file-input',\n    '[class.bao-file-label-small]': 'size === \"small\"',\n    '[class.bao-file-label-medium]': 'size === \"medium\"',\n    '[class.bao-file-input-disabled]': 'disabled'\n  }\n})\nexport class BaoFileInputComponent\n  implements AfterContentInit, AfterViewInit, OnDestroy, ControlValueAccessor\n{\n  /**\n   * Id of the file input field\n   */\n  @Input('id') public inputId?: string;\n\n  /**\n   * Label of field to be displayed above\n   */\n  @Input() public label: string;\n\n  /**\n   * Size of the file input label\n   */\n  @Input() public size: 'small' | 'medium' = 'medium';\n\n  /**\n   * Maximum size accepted for uploaded files\n   */\n  @Input() public maximalFileSize = -1;\n\n  /**\n   * Accepted types of files\n   */\n  @Input() public acceptedMIMETypes: string[] = [];\n\n  /**\n   * Is field required\n   */\n  @Input() public required? = false;\n\n  /**\n   * Is field disabled\n   */\n  @Input() public disabled? = false;\n\n  /**\n   * File selected to be uploaded\n   */\n  @Output() public uploadedFile: EventEmitter<File> = new EventEmitter<File>();\n\n  /**\n   * List of files that have been uploaded so far\n   */\n  @ContentChildren(BaoFilePreviewComponent, { descendants: true })\n  private _files: QueryList<BaoFilePreviewComponent>;\n\n  /**\n   * Form errors when component is used in a form\n   */\n  @ContentChildren(BaoErrorTextComponent, { descendants: true })\n  private _errorForm: QueryList<BaoErrorTextComponent>;\n\n  /**\n   * Error texts\n   */\n  @ViewChildren(BaoErrorTextComponent)\n  private _errorTexts: QueryList<BaoErrorTextComponent>;\n\n  /**\n   * File input that triggers uploading when clicked\n   */\n  @ViewChild('uploader', { static: false }) private uploader: ElementRef;\n\n  @ViewChild('dropzone', { static: false })\n  private dropzoneElement: ElementRef<HTMLElement>;\n\n  public insertDefaultInstructions = false;\n  public isFileTooBig = false;\n  public isFileTypeInvalid = false;\n  private _value: File[];\n  private _intlChanges: Subscription;\n  private _helperTextId: string;\n\n  constructor(\n    public intl: BaoFileIntl,\n    private elementRef: ElementRef<HTMLElement>,\n    private renderer: Renderer2,\n    private cdr: ChangeDetectorRef\n  ) {\n    this._intlChanges = intl.changes.subscribe(() => this.cdr.markForCheck());\n  }\n\n  get nativeElement(): HTMLElement {\n    return this.elementRef.nativeElement;\n  }\n\n  @HostListener('window:keyup.enter')\n  enterKeyEvent() {\n    if (document.activeElement.id === this.inputId) {\n      document.getElementById(this.inputId).click();\n    }\n  }\n\n  @HostListener('window:keyup.tab')\n  tabKeyEvent() {\n    if (document.activeElement.id === this.inputId) {\n      this.renderer.addClass(\n        this.dropzoneElement.nativeElement,\n        'dropzone-focus'\n      );\n    }\n  }\n\n  @HostListener('window:keyup.shift.tab')\n  shiftTabKeyEvent() {\n    if (document.activeElement.id === this.inputId) {\n      this.renderer.addClass(\n        this.dropzoneElement.nativeElement,\n        'dropzone-focus'\n      );\n    }\n  }\n\n  public ngAfterContentInit(): void {\n    this._errorForm.changes.subscribe(() => this.setErrorTextsAttribute());\n    if (!this.inputId) {\n      this.inputId = `file-input-${fileInputUniqueId++}`;\n    }\n    // If no content was added for dropzone instructions, add default text.\n    const dropzoneElement = Array.from(this.nativeElement.children).find(\n      (el: HTMLElement) => el.className === 'file-drop-zone'\n    );\n    if (\n      !Array.from(dropzoneElement.children).find(\n        el => el.localName === 'bao-file-dropzone-instructions'\n      )\n    ) {\n      this.insertDefaultInstructions = true;\n    }\n    this.setDescribedByAttribute();\n    this._files.changes.subscribe(\n      (files: QueryList<BaoFilePreviewComponent>) => {\n        const filesList: File[] = files.map(\n          (el: BaoFilePreviewComponent) => el.file\n        );\n        this.setValue(filesList);\n      }\n    );\n  }\n\n  public ngAfterViewInit(): void {\n    this._errorTexts.changes.subscribe(() => this.setErrorTextsAttribute());\n  }\n\n  public ngOnDestroy(): void {\n    this._intlChanges.unsubscribe();\n  }\n\n  /**\n   * Implements ControlValueAccessor interface\n   */\n  public writeValue(obj: any): void {\n    this._value = obj;\n  }\n  /**\n   * Implements ControlValueAccessor interface\n   */\n  public registerOnChange(fn: (value: any) => void): void {\n    this.propagateChange = fn;\n  }\n  /**\n   * Implements ControlValueAccessor interface\n   */\n  public registerOnTouched(fn: any): void {\n    this.propagateTouched = fn;\n  }\n  /**\n   * Implements ControlValueAccessor interface\n   */\n  public setDisabledState(isDisabled: boolean) {\n    this.disabled = isDisabled;\n  }\n\n  public uploadFile(file: File) {\n    if (!this.disabled) {\n      this.isFileTypeInvalid = false;\n      this.isFileTooBig = false;\n      this.uploader.nativeElement.value = '';\n      if (this.maximalFileSize > 0 && file.size > this.maximalFileSize) {\n        this.isFileTooBig = true;\n      }\n      if (\n        this.acceptedMIMETypes.length > 0 &&\n        this.acceptedMIMETypes.indexOf(file.type) < 0\n      ) {\n        this.isFileTypeInvalid = true;\n      }\n      if (!this.isFileTooBig && !this.isFileTypeInvalid) {\n        this.uploadedFile.emit(file);\n      }\n    }\n  }\n\n  /**\n   * Saves the registerOnChange function so the component can call it whenever it wants.\n   */\n  // eslint-disable-next-line @typescript-eslint/no-empty-function, @typescript-eslint/no-unused-vars\n  public propagateChange = (_: any) => {};\n\n  /**\n   * Saves the registerOnTouched function so the component can call it whenever it wants.\n   */\n  // eslint-disable-next-line @typescript-eslint/no-empty-function\n  public propagateTouched = () => {};\n\n  private setValue(value: File[]) {\n    this._value = value;\n    this.propagateChange(this._value);\n    this.propagateTouched();\n  }\n\n  private setDescribedByAttribute(): void {\n    const helperText = Array.from(this.nativeElement.children).find(\n      (el: HTMLElement) => el.localName === 'bao-guiding-text'\n    ).firstElementChild;\n    if (helperText) {\n      this._helperTextId = `bao-guiding-text-${fileTextUniqueId++}`;\n      this.renderer.setAttribute(helperText, 'id', this._helperTextId);\n      const inputElement = Array.from(this.nativeElement.children)\n        .find((el: HTMLElement) => el.className == 'file-drop-zone')\n        .children.item(1);\n      this.renderer.setAttribute(\n        inputElement,\n        'aria-describedby',\n        this._helperTextId\n      );\n    }\n  }\n\n  private setErrorTextsAttribute(): void {\n    const textsIds = [];\n    const errors = Array.from(this.nativeElement.children).filter(\n      (el: HTMLElement) => el.localName == 'bao-error'\n    );\n    errors.forEach((errorText: HTMLElement) => {\n      const errorTextId = `bao-error-${fileTextUniqueId++}`;\n      this.renderer.setAttribute(\n        errorText.firstElementChild,\n        'id',\n        errorTextId\n      );\n      textsIds.push(errorTextId);\n    });\n    const inputElement = Array.from(this.nativeElement.children)\n      .find((el: HTMLElement) => el.classList.contains('file-drop-zone'))\n      .children.item(1);\n    if (this._helperTextId) {\n      textsIds.unshift(this._helperTextId);\n    }\n    this.renderer.setAttribute(\n      inputElement,\n      'aria-describedby',\n      textsIds.join(' ')\n    );\n  }\n}\n\n@Directive({\n  selector: '[baoFileDrop]',\n  host: { '[class.drag-over]': '_isDragOver == true' }\n})\nexport class BaoFileDropDirective {\n  @Output() public fileDrop: EventEmitter<File> = new EventEmitter<File>();\n  private _isDragOver = false;\n\n  @HostListener('dragover', ['$event'])\n  public onDragOver(event: DragEvent): void {\n    this.preventAndStop(event);\n    this._isDragOver = true;\n  }\n\n  @HostListener('dragleave', ['$event'])\n  public onDragLeave(event: DragEvent): void {\n    this.preventAndStop(event);\n    this._isDragOver = false;\n  }\n\n  @HostListener('drop', ['$event'])\n  public onDrop(event: DragEvent): void {\n    this.preventAndStop(event);\n    this._isDragOver = false;\n    const transfer = this.getDataTransfer(event);\n    this.fileDrop.emit(transfer.files[0]);\n  }\n\n  private preventAndStop(event: DragEvent): void {\n    event.preventDefault();\n    event.stopPropagation();\n  }\n\n  private getDataTransfer(event: DragEvent | any): DataTransfer {\n    // eslint-disable-next-line @typescript-eslint/no-unsafe-return\n    return event.dataTransfer\n      ? event.dataTransfer\n      : event.originalEvent.dataTransfer;\n  }\n}\n\n@Directive({\n  selector: 'bao-file-dropzone-instructions, [bao-file-dropzone-instructions]',\n  host: {\n    class: 'bao-file-dropzone-instructions'\n  }\n})\nexport class BaoFileDropzoneIntructions implements AfterContentInit {\n  constructor(\n    private renderer: Renderer2,\n    private elementRef: ElementRef<HTMLElement>\n  ) {}\n\n  get nativeElement(): HTMLElement {\n    return this.elementRef.nativeElement;\n  }\n\n  ngAfterContentInit(): void {\n    this.renderer.setAttribute(this.nativeElement, 'aria-hidden', 'true');\n  }\n}\n","<label bao-label [required]=\"required\" [for]=\"inputId\">{{ label }}</label>\n<ng-content select=\"bao-guiding-text\"></ng-content>\n<div\n  baoFileDrop\n  class=\"file-drop-zone\"\n  (fileDrop)=\"uploadFile($event)\"\n  #dropzone\n>\n  <button\n    bao-button\n    type=\"button\"\n    displayType=\"utility\"\n    level=\"secondary\"\n    [disabled]=\"disabled\"\n    (click)=\"uploader.click()\"\n    aria-hidden=\"true\"\n    tabIndex=\"-1\"\n  >\n    {{ intl.dropzoneButtonLabel }}\n  </button>\n  <input\n    [id]=\"inputId\"\n    type=\"file\"\n    class=\"sr-only\"\n    [disabled]=\"disabled\"\n    (change)=\"uploadFile($event.target.files[0])\"\n    #uploader\n  />\n  <ng-container\n    ><div #ref>\n      <ng-content select=\"bao-file-dropzone-instructions\"></ng-content></div\n  ></ng-container>\n  <ng-container *ngIf=\"ref.childNodes.length === 0\"\n    ><bao-file-dropzone-instructions>{{\n      intl.defaultDropzoneInstructions\n    }}</bao-file-dropzone-instructions></ng-container\n  >\n</div>\n<bao-error *ngIf=\"isFileTooBig\">\n  {{ intl.fileTooBigErrorMessage }}\n</bao-error>\n<bao-error *ngIf=\"isFileTypeInvalid\">\n  {{ intl.invalidFileTypeErrorMessage }}\n</bao-error>\n<ng-content select=\"bao-error\"></ng-content>\n<ng-content></ng-content>\n"]}
@@ -0,0 +1,65 @@
1
+ /*
2
+ * Copyright (c) 2023 Ville de Montreal. All rights reserved.
3
+ * Licensed under the MIT license.
4
+ * See LICENSE file in the project root for full license information.
5
+ */
6
+ import { Injectable, Optional, SkipSelf } from '@angular/core';
7
+ import { Subject } from 'rxjs';
8
+ import * as i0 from "@angular/core";
9
+ /**
10
+ * To modify the labels and text displayed, create a new instance of BaoFileIntl and
11
+ * include it in a custom provider
12
+ */
13
+ export class BaoFileIntl {
14
+ constructor() {
15
+ /**
16
+ * Stream to emit from when labels are changed. Use this to notify components when the labels have
17
+ * changed after initialization.
18
+ */
19
+ this.changes = new Subject();
20
+ /** The label for button in dropzone */
21
+ this.dropzoneButtonLabel = 'Parcourir';
22
+ /** The default dropzone instructions */
23
+ this.defaultDropzoneInstructions = 'ou déposer votre fichier ici';
24
+ /** Error message displayed when uploaded file is too large */
25
+ this.fileTooBigErrorMessage = 'La taille de ce fichier est trop grande';
26
+ /** Error message displayed when uploaded file has an invalid type */
27
+ this.invalidFileTypeErrorMessage = "Ce format de fichier n'est pas autorisé";
28
+ }
29
+ }
30
+ BaoFileIntl.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: BaoFileIntl, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
31
+ BaoFileIntl.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: BaoFileIntl, providedIn: 'root' });
32
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: BaoFileIntl, decorators: [{
33
+ type: Injectable,
34
+ args: [{ providedIn: 'root' }]
35
+ }] });
36
+ export class BaoFileIntlEnglish extends BaoFileIntl {
37
+ constructor() {
38
+ super(...arguments);
39
+ /** The label for button in dropzone */
40
+ this.dropzoneButtonLabel = 'Browse';
41
+ /** The default dropzone instructions */
42
+ this.defaultDropzoneInstructions = 'or drop your file here';
43
+ /** Error message displayed when uploaded file is too large */
44
+ this.fileTooBigErrorMessage = 'The size of this file is too large';
45
+ /** Error message displayed when uploaded file has an invalid type */
46
+ this.invalidFileTypeErrorMessage = 'The format of this file is unauthorized';
47
+ }
48
+ }
49
+ BaoFileIntlEnglish.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: BaoFileIntlEnglish, deps: null, target: i0.ɵɵFactoryTarget.Injectable });
50
+ BaoFileIntlEnglish.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: BaoFileIntlEnglish });
51
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: BaoFileIntlEnglish, decorators: [{
52
+ type: Injectable
53
+ }] });
54
+ /** @docs-private */
55
+ export function BAO_FILE_INTL_PROVIDER_FACTORY(parentIntl) {
56
+ return parentIntl || new BaoFileIntl();
57
+ }
58
+ /** @docs-private */
59
+ export const BAO_FILE_INTL_PROVIDER = {
60
+ // If there is already an BaoFileIntl available, use that. Otherwise, provide a new one.
61
+ provide: BaoFileIntl,
62
+ deps: [[new Optional(), new SkipSelf(), BaoFileIntl]],
63
+ useFactory: BAO_FILE_INTL_PROVIDER_FACTORY
64
+ };
65
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmlsZS1pbnRsLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvYW5ndWxhci11aS9zcmMvbGliL2ZpbGUvZmlsZS1pbnRsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7O0dBSUc7QUFFSCxPQUFPLEVBQUUsVUFBVSxFQUFFLFFBQVEsRUFBRSxRQUFRLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDL0QsT0FBTyxFQUFFLE9BQU8sRUFBRSxNQUFNLE1BQU0sQ0FBQzs7QUFFL0I7OztHQUdHO0FBRUgsTUFBTSxPQUFPLFdBQVc7SUFEeEI7UUFFRTs7O1dBR0c7UUFDTSxZQUFPLEdBQWtCLElBQUksT0FBTyxFQUFRLENBQUM7UUFFdEQsdUNBQXVDO1FBQ3ZDLHdCQUFtQixHQUFHLFdBQVcsQ0FBQztRQUVsQyx3Q0FBd0M7UUFDeEMsZ0NBQTJCLEdBQUcsOEJBQThCLENBQUM7UUFFN0QsOERBQThEO1FBQzlELDJCQUFzQixHQUFHLHlDQUF5QyxDQUFDO1FBRW5FLHFFQUFxRTtRQUNyRSxnQ0FBMkIsR0FBRyx5Q0FBeUMsQ0FBQztLQUN6RTs7eUdBbEJZLFdBQVc7NkdBQVgsV0FBVyxjQURFLE1BQU07NEZBQ25CLFdBQVc7a0JBRHZCLFVBQVU7bUJBQUMsRUFBRSxVQUFVLEVBQUUsTUFBTSxFQUFFOztBQXNCbEMsTUFBTSxPQUFPLGtCQUFtQixTQUFRLFdBQVc7SUFEbkQ7O1FBRUUsdUNBQXVDO1FBQ3ZDLHdCQUFtQixHQUFHLFFBQVEsQ0FBQztRQUUvQix3Q0FBd0M7UUFDeEMsZ0NBQTJCLEdBQUcsd0JBQXdCLENBQUM7UUFFdkQsOERBQThEO1FBQzlELDJCQUFzQixHQUFHLG9DQUFvQyxDQUFDO1FBRTlELHFFQUFxRTtRQUNyRSxnQ0FBMkIsR0FBRyx5Q0FBeUMsQ0FBQztLQUN6RTs7Z0hBWlksa0JBQWtCO29IQUFsQixrQkFBa0I7NEZBQWxCLGtCQUFrQjtrQkFEOUIsVUFBVTs7QUFlWCxvQkFBb0I7QUFDcEIsTUFBTSxVQUFVLDhCQUE4QixDQUFDLFVBQXVCO0lBQ3BFLE9BQU8sVUFBVSxJQUFJLElBQUksV0FBVyxFQUFFLENBQUM7QUFDekMsQ0FBQztBQUVELG9CQUFvQjtBQUNwQixNQUFNLENBQUMsTUFBTSxzQkFBc0IsR0FBRztJQUNwQyx3RkFBd0Y7SUFDeEYsT0FBTyxFQUFFLFdBQVc7SUFDcEIsSUFBSSxFQUFFLENBQUMsQ0FBQyxJQUFJLFFBQVEsRUFBRSxFQUFFLElBQUksUUFBUSxFQUFFLEVBQUUsV0FBVyxDQUFDLENBQUM7SUFDckQsVUFBVSxFQUFFLDhCQUE4QjtDQUMzQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiAqIENvcHlyaWdodCAoYykgMjAyMyBWaWxsZSBkZSBNb250cmVhbC4gQWxsIHJpZ2h0cyByZXNlcnZlZC5cbiAqIExpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgbGljZW5zZS5cbiAqIFNlZSBMSUNFTlNFIGZpbGUgaW4gdGhlIHByb2plY3Qgcm9vdCBmb3IgZnVsbCBsaWNlbnNlIGluZm9ybWF0aW9uLlxuICovXG5cbmltcG9ydCB7IEluamVjdGFibGUsIE9wdGlvbmFsLCBTa2lwU2VsZiB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgU3ViamVjdCB9IGZyb20gJ3J4anMnO1xuXG4vKipcbiAqIFRvIG1vZGlmeSB0aGUgbGFiZWxzIGFuZCB0ZXh0IGRpc3BsYXllZCwgY3JlYXRlIGEgbmV3IGluc3RhbmNlIG9mIEJhb0ZpbGVJbnRsIGFuZFxuICogaW5jbHVkZSBpdCBpbiBhIGN1c3RvbSBwcm92aWRlclxuICovXG5ASW5qZWN0YWJsZSh7IHByb3ZpZGVkSW46ICdyb290JyB9KVxuZXhwb3J0IGNsYXNzIEJhb0ZpbGVJbnRsIHtcbiAgLyoqXG4gICAqIFN0cmVhbSB0byBlbWl0IGZyb20gd2hlbiBsYWJlbHMgYXJlIGNoYW5nZWQuIFVzZSB0aGlzIHRvIG5vdGlmeSBjb21wb25lbnRzIHdoZW4gdGhlIGxhYmVscyBoYXZlXG4gICAqIGNoYW5nZWQgYWZ0ZXIgaW5pdGlhbGl6YXRpb24uXG4gICAqL1xuICByZWFkb25seSBjaGFuZ2VzOiBTdWJqZWN0PHZvaWQ+ID0gbmV3IFN1YmplY3Q8dm9pZD4oKTtcblxuICAvKiogVGhlIGxhYmVsIGZvciBidXR0b24gaW4gZHJvcHpvbmUgKi9cbiAgZHJvcHpvbmVCdXR0b25MYWJlbCA9ICdQYXJjb3VyaXInO1xuXG4gIC8qKiBUaGUgZGVmYXVsdCBkcm9wem9uZSBpbnN0cnVjdGlvbnMgKi9cbiAgZGVmYXVsdERyb3B6b25lSW5zdHJ1Y3Rpb25zID0gJ291IGTDqXBvc2VyIHZvdHJlIGZpY2hpZXIgaWNpJztcblxuICAvKiogRXJyb3IgbWVzc2FnZSBkaXNwbGF5ZWQgd2hlbiB1cGxvYWRlZCBmaWxlIGlzIHRvbyBsYXJnZSAqL1xuICBmaWxlVG9vQmlnRXJyb3JNZXNzYWdlID0gJ0xhIHRhaWxsZSBkZSBjZSBmaWNoaWVyIGVzdCB0cm9wIGdyYW5kZSc7XG5cbiAgLyoqIEVycm9yIG1lc3NhZ2UgZGlzcGxheWVkIHdoZW4gdXBsb2FkZWQgZmlsZSBoYXMgYW4gaW52YWxpZCB0eXBlICovXG4gIGludmFsaWRGaWxlVHlwZUVycm9yTWVzc2FnZSA9IFwiQ2UgZm9ybWF0IGRlIGZpY2hpZXIgbidlc3QgcGFzIGF1dG9yaXPDqVwiO1xufVxuXG5ASW5qZWN0YWJsZSgpXG5leHBvcnQgY2xhc3MgQmFvRmlsZUludGxFbmdsaXNoIGV4dGVuZHMgQmFvRmlsZUludGwge1xuICAvKiogVGhlIGxhYmVsIGZvciBidXR0b24gaW4gZHJvcHpvbmUgKi9cbiAgZHJvcHpvbmVCdXR0b25MYWJlbCA9ICdCcm93c2UnO1xuXG4gIC8qKiBUaGUgZGVmYXVsdCBkcm9wem9uZSBpbnN0cnVjdGlvbnMgKi9cbiAgZGVmYXVsdERyb3B6b25lSW5zdHJ1Y3Rpb25zID0gJ29yIGRyb3AgeW91ciBmaWxlIGhlcmUnO1xuXG4gIC8qKiBFcnJvciBtZXNzYWdlIGRpc3BsYXllZCB3aGVuIHVwbG9hZGVkIGZpbGUgaXMgdG9vIGxhcmdlICovXG4gIGZpbGVUb29CaWdFcnJvck1lc3NhZ2UgPSAnVGhlIHNpemUgb2YgdGhpcyBmaWxlIGlzIHRvbyBsYXJnZSc7XG5cbiAgLyoqIEVycm9yIG1lc3NhZ2UgZGlzcGxheWVkIHdoZW4gdXBsb2FkZWQgZmlsZSBoYXMgYW4gaW52YWxpZCB0eXBlICovXG4gIGludmFsaWRGaWxlVHlwZUVycm9yTWVzc2FnZSA9ICdUaGUgZm9ybWF0IG9mIHRoaXMgZmlsZSBpcyB1bmF1dGhvcml6ZWQnO1xufVxuXG4vKiogQGRvY3MtcHJpdmF0ZSAqL1xuZXhwb3J0IGZ1bmN0aW9uIEJBT19GSUxFX0lOVExfUFJPVklERVJfRkFDVE9SWShwYXJlbnRJbnRsOiBCYW9GaWxlSW50bCkge1xuICByZXR1cm4gcGFyZW50SW50bCB8fCBuZXcgQmFvRmlsZUludGwoKTtcbn1cblxuLyoqIEBkb2NzLXByaXZhdGUgKi9cbmV4cG9ydCBjb25zdCBCQU9fRklMRV9JTlRMX1BST1ZJREVSID0ge1xuICAvLyBJZiB0aGVyZSBpcyBhbHJlYWR5IGFuIEJhb0ZpbGVJbnRsIGF2YWlsYWJsZSwgdXNlIHRoYXQuIE90aGVyd2lzZSwgcHJvdmlkZSBhIG5ldyBvbmUuXG4gIHByb3ZpZGU6IEJhb0ZpbGVJbnRsLFxuICBkZXBzOiBbW25ldyBPcHRpb25hbCgpLCBuZXcgU2tpcFNlbGYoKSwgQmFvRmlsZUludGxdXSxcbiAgdXNlRmFjdG9yeTogQkFPX0ZJTEVfSU5UTF9QUk9WSURFUl9GQUNUT1JZXG59O1xuIl19
@@ -0,0 +1,86 @@
1
+ /*
2
+ * Copyright (c) 2023 Ville de Montreal. All rights reserved.
3
+ * Licensed under the MIT license.
4
+ * See LICENSE file in the project root for full license information.
5
+ */
6
+ import { Component, Input, ViewEncapsulation } from '@angular/core';
7
+ import * as i0 from "@angular/core";
8
+ import * as i1 from "../icon/icon.component";
9
+ import * as i2 from "@angular/common";
10
+ const KILO_THRESHOLD = 1000;
11
+ const MEGA_THRESHOLD = 1000000;
12
+ export class BaoFilePreviewComponent {
13
+ constructor(elementRef, renderer) {
14
+ this.elementRef = elementRef;
15
+ this.renderer = renderer;
16
+ /**
17
+ * Is file loading
18
+ */
19
+ this.isLoading = false;
20
+ /**
21
+ * True if projected content has no icon and file does not have a thumbnail.
22
+ */
23
+ this.insertGenericIcon = false;
24
+ this.thumbnailURL = '';
25
+ }
26
+ get nativeElement() {
27
+ return this.elementRef.nativeElement;
28
+ }
29
+ get fileSize() {
30
+ return this.formatSize(this.file.size);
31
+ }
32
+ ngAfterContentInit() {
33
+ this.getThumbnail();
34
+ this.setIcon();
35
+ }
36
+ setIcon() {
37
+ // If no icon is in the projected content, generic icon is added
38
+ const contentIcon = Array.from(this.nativeElement.children.item(0).children).find((el) => el.localName === 'bao-icon');
39
+ if (!contentIcon) {
40
+ this.insertGenericIcon = true;
41
+ }
42
+ else {
43
+ this.renderer.addClass(contentIcon, 'bao-file-media');
44
+ }
45
+ }
46
+ getThumbnail() {
47
+ if (this.file &&
48
+ (this.file.type === 'image/png' || this.file.type === 'image/jpeg')) {
49
+ const reader = new FileReader();
50
+ reader.onload = (event) => {
51
+ this.thumbnailURL = event.target.result;
52
+ };
53
+ reader.onerror = () => {
54
+ this.thumbnailURL = '';
55
+ };
56
+ reader.readAsDataURL(this.file);
57
+ }
58
+ }
59
+ formatSize(size) {
60
+ if (size >= KILO_THRESHOLD && size / KILO_THRESHOLD < KILO_THRESHOLD) {
61
+ return this.getSizeAndUnit(size, KILO_THRESHOLD, 'Ko');
62
+ }
63
+ const sizeDividedByKoMultiplicator = size / KILO_THRESHOLD;
64
+ if (sizeDividedByKoMultiplicator >= KILO_THRESHOLD) {
65
+ const toFixed = sizeDividedByKoMultiplicator > 10 ? 0 : 1;
66
+ return this.getSizeAndUnit(size, MEGA_THRESHOLD, 'Mo', toFixed);
67
+ }
68
+ return `${size} octets`;
69
+ }
70
+ getSizeAndUnit(size, multiplicator, unit, toFixed = 0) {
71
+ return `${(size / multiplicator).toFixed(toFixed)} ${unit}`;
72
+ }
73
+ }
74
+ BaoFilePreviewComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: BaoFilePreviewComponent, deps: [{ token: i0.ElementRef }, { token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Component });
75
+ BaoFilePreviewComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.12", type: BaoFilePreviewComponent, selector: "bao-file-preview, [bao-file-preview]", inputs: { file: "file", isLoading: "isLoading" }, host: { classAttribute: "bao-file-preview" }, ngImport: i0, template: "<div class=\"bao-file-info\">\n <ng-content select=\"bao-icon\"></ng-content>\n <bao-icon\n *ngIf=\"insertGenericIcon && !thumbnailURL\"\n class=\"bao-file-media\"\n svgIcon=\"icon-file\"\n >\n </bao-icon>\n <ng-container *ngIf=\"thumbnailURL && !isLoading\">\n <img\n class=\"bao-file-media\"\n [src]=\"thumbnailURL\"\n width=\"40px\"\n height=\"40px\"\n />\n </ng-container>\n <div class=\"bao-file-text\">\n <div class=\"bao-file-name\">{{ file.name }}</div>\n <div class=\"bao-file-size\">{{ fileSize }}</div>\n </div>\n</div>\n<ng-container *ngIf=\"!isLoading\">\n <ng-content select=\"button[bao-button]\"></ng-content>\n <ng-content select=\"baoDropdownTriggerFor\"></ng-content>\n</ng-container>\n<bao-icon\n *ngIf=\"isLoading\"\n class=\"loading-spinner\"\n svgIcon=\"icon-spinner\"\n title=\"chargement\"\n></bao-icon>\n", styles: [".bao-file-preview{display:flex;align-items:center;justify-content:space-between;padding-top:.625rem;padding-bottom:.625rem;list-style-type:none}.bao-file-preview:first-child{margin-top:1rem}.bao-file-preview>.bao-file-info{display:flex;align-items:center}.bao-file-preview>.bao-file-info>.bao-icon{color:#adb2bd;flex-shrink:0}.bao-file-preview>.bao-file-info>.bao-file-media{margin-right:1rem}.bao-file-preview>.bao-file-info>.bao-file-text{display:inline-flex;flex-direction:column;margin-right:1rem}.bao-file-preview>.bao-file-info>.bao-file-text>.bao-file-name{overflow:hidden;font-weight:700;font-size:.875rem;line-height:1.25rem;color:#212529}.bao-file-preview>.bao-file-info>.bao-file-text>.bao-file-size{font-weight:400;font-size:.75rem;line-height:1rem;color:#637381}.bao-file-preview .loading-spinner{color:#097d6c}.bao-file-preview .bao-dropdown-menu-container{position:absolute;margin-left:auto}\n"], components: [{ type: i1.BaoIconComponent, selector: "bao-icon", inputs: ["color", "size", "svgIcon", "title"], exportAs: ["baoIcon"] }], directives: [{ type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], encapsulation: i0.ViewEncapsulation.None });
76
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: BaoFilePreviewComponent, decorators: [{
77
+ type: Component,
78
+ args: [{ selector: 'bao-file-preview, [bao-file-preview]', encapsulation: ViewEncapsulation.None, host: {
79
+ class: 'bao-file-preview'
80
+ }, template: "<div class=\"bao-file-info\">\n <ng-content select=\"bao-icon\"></ng-content>\n <bao-icon\n *ngIf=\"insertGenericIcon && !thumbnailURL\"\n class=\"bao-file-media\"\n svgIcon=\"icon-file\"\n >\n </bao-icon>\n <ng-container *ngIf=\"thumbnailURL && !isLoading\">\n <img\n class=\"bao-file-media\"\n [src]=\"thumbnailURL\"\n width=\"40px\"\n height=\"40px\"\n />\n </ng-container>\n <div class=\"bao-file-text\">\n <div class=\"bao-file-name\">{{ file.name }}</div>\n <div class=\"bao-file-size\">{{ fileSize }}</div>\n </div>\n</div>\n<ng-container *ngIf=\"!isLoading\">\n <ng-content select=\"button[bao-button]\"></ng-content>\n <ng-content select=\"baoDropdownTriggerFor\"></ng-content>\n</ng-container>\n<bao-icon\n *ngIf=\"isLoading\"\n class=\"loading-spinner\"\n svgIcon=\"icon-spinner\"\n title=\"chargement\"\n></bao-icon>\n", styles: [".bao-file-preview{display:flex;align-items:center;justify-content:space-between;padding-top:.625rem;padding-bottom:.625rem;list-style-type:none}.bao-file-preview:first-child{margin-top:1rem}.bao-file-preview>.bao-file-info{display:flex;align-items:center}.bao-file-preview>.bao-file-info>.bao-icon{color:#adb2bd;flex-shrink:0}.bao-file-preview>.bao-file-info>.bao-file-media{margin-right:1rem}.bao-file-preview>.bao-file-info>.bao-file-text{display:inline-flex;flex-direction:column;margin-right:1rem}.bao-file-preview>.bao-file-info>.bao-file-text>.bao-file-name{overflow:hidden;font-weight:700;font-size:.875rem;line-height:1.25rem;color:#212529}.bao-file-preview>.bao-file-info>.bao-file-text>.bao-file-size{font-weight:400;font-size:.75rem;line-height:1rem;color:#637381}.bao-file-preview .loading-spinner{color:#097d6c}.bao-file-preview .bao-dropdown-menu-container{position:absolute;margin-left:auto}\n"] }]
81
+ }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.Renderer2 }]; }, propDecorators: { file: [{
82
+ type: Input
83
+ }], isLoading: [{
84
+ type: Input
85
+ }] } });
86
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-preview.component.js","sourceRoot":"","sources":["../../../../../projects/angular-ui/src/lib/file/file-preview.component.ts","../../../../../projects/angular-ui/src/lib/file/file-preview.component.html"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2023 Ville de Montreal. All rights reserved.\n * Licensed under the MIT license.\n * See LICENSE file in the project root for full license information.\n */\n\nimport {\n  AfterContentInit,\n  Component,\n  ElementRef,\n  Input,\n  Renderer2,\n  ViewEncapsulation\n} from '@angular/core';\n\nconst KILO_THRESHOLD = 1000;\nconst MEGA_THRESHOLD = 1000000;\n\n@Component({\n  selector: 'bao-file-preview, [bao-file-preview]',\n  templateUrl: './file-preview.component.html',\n  styleUrls: ['./file-preview.component.scss'],\n  encapsulation: ViewEncapsulation.None,\n  host: {\n    class: 'bao-file-preview'\n  }\n})\nexport class BaoFilePreviewComponent implements AfterContentInit {\n  /**\n   * Uploaded file to display in list.\n   */\n  @Input() public file: File;\n\n  /**\n   * Is file loading\n   */\n  @Input() public isLoading = false;\n\n  /**\n   * True if projected content has no icon and file does not have a thumbnail.\n   */\n  public insertGenericIcon = false;\n\n  public thumbnailURL = '';\n\n  constructor(\n    private elementRef: ElementRef<HTMLElement>,\n    private renderer: Renderer2\n  ) {}\n\n  get nativeElement(): HTMLElement {\n    return this.elementRef.nativeElement;\n  }\n\n  get fileSize(): string {\n    return this.formatSize(this.file.size);\n  }\n\n  ngAfterContentInit(): void {\n    this.getThumbnail();\n    this.setIcon();\n  }\n\n  private setIcon() {\n    // If no icon is in the projected content, generic icon is added\n    const contentIcon = Array.from(\n      this.nativeElement.children.item(0).children\n    ).find((el: HTMLElement) => el.localName === 'bao-icon');\n    if (!contentIcon) {\n      this.insertGenericIcon = true;\n    } else {\n      this.renderer.addClass(contentIcon, 'bao-file-media');\n    }\n  }\n\n  private getThumbnail() {\n    if (\n      this.file &&\n      (this.file.type === 'image/png' || this.file.type === 'image/jpeg')\n    ) {\n      const reader = new FileReader();\n      reader.onload = (event: any) => {\n        this.thumbnailURL = event.target.result;\n      };\n\n      reader.onerror = () => {\n        this.thumbnailURL = '';\n      };\n      reader.readAsDataURL(this.file);\n    }\n  }\n\n  private formatSize(size: number) {\n    if (size >= KILO_THRESHOLD && size / KILO_THRESHOLD < KILO_THRESHOLD) {\n      return this.getSizeAndUnit(size, KILO_THRESHOLD, 'Ko');\n    }\n    const sizeDividedByKoMultiplicator = size / KILO_THRESHOLD;\n    if (sizeDividedByKoMultiplicator >= KILO_THRESHOLD) {\n      const toFixed = sizeDividedByKoMultiplicator > 10 ? 0 : 1;\n      return this.getSizeAndUnit(size, MEGA_THRESHOLD, 'Mo', toFixed);\n    }\n    return `${size} octets`;\n  }\n\n  private getSizeAndUnit(\n    size: number,\n    multiplicator: number,\n    unit: string,\n    toFixed = 0\n  ): string {\n    return `${(size / multiplicator).toFixed(toFixed)} ${unit}`;\n  }\n}\n","<div class=\"bao-file-info\">\n  <ng-content select=\"bao-icon\"></ng-content>\n  <bao-icon\n    *ngIf=\"insertGenericIcon && !thumbnailURL\"\n    class=\"bao-file-media\"\n    svgIcon=\"icon-file\"\n  >\n  </bao-icon>\n  <ng-container *ngIf=\"thumbnailURL && !isLoading\">\n    <img\n      class=\"bao-file-media\"\n      [src]=\"thumbnailURL\"\n      width=\"40px\"\n      height=\"40px\"\n    />\n  </ng-container>\n  <div class=\"bao-file-text\">\n    <div class=\"bao-file-name\">{{ file.name }}</div>\n    <div class=\"bao-file-size\">{{ fileSize }}</div>\n  </div>\n</div>\n<ng-container *ngIf=\"!isLoading\">\n  <ng-content select=\"button[bao-button]\"></ng-content>\n  <ng-content select=\"baoDropdownTriggerFor\"></ng-content>\n</ng-container>\n<bao-icon\n  *ngIf=\"isLoading\"\n  class=\"loading-spinner\"\n  svgIcon=\"icon-spinner\"\n  title=\"chargement\"\n></bao-icon>\n"]}
@@ -0,0 +1,10 @@
1
+ /*
2
+ * Copyright (c) 2023 Ville de Montreal. All rights reserved.
3
+ * Licensed under the MIT license.
4
+ * See LICENSE file in the project root for full license information.
5
+ */
6
+ export * from './module';
7
+ export * from './file-input.component';
8
+ export * from './file-preview.component';
9
+ export * from './file-intl';
10
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9hbmd1bGFyLXVpL3NyYy9saWIvZmlsZS9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7OztHQUlHO0FBQ0gsY0FBYyxVQUFVLENBQUM7QUFDekIsY0FBYyx3QkFBd0IsQ0FBQztBQUN2QyxjQUFjLDBCQUEwQixDQUFDO0FBQ3pDLGNBQWMsYUFBYSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiAqIENvcHlyaWdodCAoYykgMjAyMyBWaWxsZSBkZSBNb250cmVhbC4gQWxsIHJpZ2h0cyByZXNlcnZlZC5cbiAqIExpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgbGljZW5zZS5cbiAqIFNlZSBMSUNFTlNFIGZpbGUgaW4gdGhlIHByb2plY3Qgcm9vdCBmb3IgZnVsbCBsaWNlbnNlIGluZm9ybWF0aW9uLlxuICovXG5leHBvcnQgKiBmcm9tICcuL21vZHVsZSc7XG5leHBvcnQgKiBmcm9tICcuL2ZpbGUtaW5wdXQuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vZmlsZS1wcmV2aWV3LmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2ZpbGUtaW50bCc7XG4iXX0=