@villedemontreal/angular-ui 14.0.1 → 14.2.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.
- package/esm2020/lib/alert/alert.component.mjs +2 -2
- package/esm2020/lib/alert/index.mjs +3 -3
- package/esm2020/lib/alert/module.mjs +2 -2
- package/esm2020/lib/avatar/avatar.component.mjs +2 -2
- package/esm2020/lib/avatar/index.mjs +2 -2
- package/esm2020/lib/avatar/module.mjs +2 -2
- package/esm2020/lib/badge/badge.component.mjs +2 -2
- package/esm2020/lib/badge/index.mjs +2 -2
- package/esm2020/lib/badge/module.mjs +2 -2
- package/esm2020/lib/bao.module.mjs +23 -11
- package/esm2020/lib/breadcrumb/breadcrumb.component.mjs +2 -2
- package/esm2020/lib/breadcrumb/index.mjs +2 -2
- package/esm2020/lib/breadcrumb/module.mjs +2 -2
- package/esm2020/lib/button/button.component.mjs +3 -3
- package/esm2020/lib/button/index.mjs +2 -2
- package/esm2020/lib/button/module.mjs +2 -2
- package/esm2020/lib/card/card.component.mjs +2 -2
- package/esm2020/lib/card/index.mjs +2 -2
- package/esm2020/lib/card/module.mjs +2 -2
- package/esm2020/lib/checkbox/checkbox-group.component.mjs +2 -2
- package/esm2020/lib/checkbox/checkbox.component.mjs +1 -1
- package/esm2020/lib/checkbox/index.mjs +2 -2
- package/esm2020/lib/checkbox/module.mjs +2 -2
- package/esm2020/lib/common-components/error-text/errorText.component.mjs +2 -2
- package/esm2020/lib/common-components/guiding-text/guidingText.component.mjs +2 -2
- package/esm2020/lib/common-components/index.mjs +2 -2
- package/esm2020/lib/common-components/label-text/labelText.component.mjs +4 -4
- package/esm2020/lib/common-components/module.mjs +6 -3
- package/esm2020/lib/common-components/title-text/titleText.component.mjs +2 -2
- package/esm2020/lib/core/breakpoints.mjs +13 -0
- package/esm2020/lib/core/colors.mjs +1 -1
- package/esm2020/lib/core/index.mjs +8 -0
- package/esm2020/lib/dropdown-menu/dropdown-menu.component.mjs +38 -8
- package/esm2020/lib/dropdown-menu/index.mjs +2 -2
- package/esm2020/lib/dropdown-menu/module.mjs +2 -2
- package/esm2020/lib/file/file-input.component.mjs +319 -0
- package/esm2020/lib/file/file-intl.mjs +65 -0
- package/esm2020/lib/file/file-preview.component.mjs +86 -0
- package/esm2020/lib/file/index.mjs +10 -0
- package/esm2020/lib/file/module.mjs +56 -0
- package/esm2020/lib/header-info/header-info.component.mjs +2 -2
- package/esm2020/lib/header-info/index.mjs +2 -2
- package/esm2020/lib/header-info/module.mjs +2 -2
- package/esm2020/lib/hyperlink/hyperlink.component.mjs +2 -2
- package/esm2020/lib/hyperlink/index.mjs +2 -2
- package/esm2020/lib/hyperlink/module.mjs +2 -2
- package/esm2020/lib/icon/bao-icon-registry.mjs +2 -2
- package/esm2020/lib/icon/icon.component.mjs +2 -2
- package/esm2020/lib/icon/icons-dictionary.mjs +3 -2
- package/esm2020/lib/icon/index.mjs +2 -2
- package/esm2020/lib/icon/module.mjs +2 -2
- package/esm2020/lib/list/index.mjs +2 -2
- package/esm2020/lib/list/list.component.mjs +2 -2
- package/esm2020/lib/list/module.mjs +2 -2
- package/esm2020/lib/modal/index.mjs +2 -2
- package/esm2020/lib/modal/modal-config.mjs +2 -2
- package/esm2020/lib/modal/modal-container.mjs +18 -32
- package/esm2020/lib/modal/modal-directives.mjs +2 -2
- package/esm2020/lib/modal/modal-ref.mjs +4 -1
- package/esm2020/lib/modal/modal.mjs +11 -32
- package/esm2020/lib/modal/module.mjs +5 -20
- package/esm2020/lib/radio/index.mjs +2 -2
- package/esm2020/lib/radio/module.mjs +2 -2
- package/esm2020/lib/radio/radio-group.component.mjs +2 -2
- package/esm2020/lib/radio/radio.component.mjs +1 -1
- package/esm2020/lib/shared/enum/display-mode.mjs +1 -1
- package/esm2020/lib/shared/index.mjs +2 -2
- package/esm2020/lib/snack-bar/index.mjs +12 -0
- package/esm2020/lib/snack-bar/module.mjs +51 -0
- package/esm2020/lib/snack-bar/simple-snack-bar.component.mjs +92 -0
- package/esm2020/lib/snack-bar/snack-bar-animations.mjs +28 -0
- package/esm2020/lib/snack-bar/snack-bar-config.mjs +44 -0
- package/esm2020/lib/snack-bar/snack-bar-container.mjs +273 -0
- package/esm2020/lib/snack-bar/snack-bar-ref.mjs +75 -0
- package/esm2020/lib/snack-bar/snack-bar.mjs +251 -0
- package/esm2020/lib/summary/index.mjs +2 -2
- package/esm2020/lib/summary/list-summary.component.mjs +2 -2
- package/esm2020/lib/summary/module.mjs +2 -2
- package/esm2020/lib/summary/summary.component.mjs +2 -2
- package/esm2020/lib/system-header/index.mjs +8 -0
- package/esm2020/lib/system-header/module.mjs +33 -0
- package/esm2020/lib/system-header/system-header.component.mjs +128 -0
- package/esm2020/lib/tabs/index.mjs +2 -2
- package/esm2020/lib/tabs/module.mjs +2 -2
- package/esm2020/lib/tabs/tabs.component.mjs +2 -2
- package/esm2020/lib/tag/index.mjs +2 -2
- package/esm2020/lib/tag/module.mjs +2 -2
- package/esm2020/lib/tag/tag.component.mjs +2 -2
- package/esm2020/public-api.mjs +5 -2
- package/fesm2015/villedemontreal-angular-ui.mjs +1745 -314
- package/fesm2015/villedemontreal-angular-ui.mjs.map +1 -1
- package/fesm2020/villedemontreal-angular-ui.mjs +1743 -314
- package/fesm2020/villedemontreal-angular-ui.mjs.map +1 -1
- package/lib/alert/index.d.ts +1 -1
- package/lib/bao.module.d.ts +14 -11
- package/lib/button/button.component.d.ts +3 -3
- package/lib/core/breakpoints.d.ts +7 -0
- package/lib/core/index.d.ts +2 -0
- package/lib/dropdown-menu/dropdown-menu.component.d.ts +10 -4
- package/lib/file/file-input.component.d.ts +123 -0
- package/lib/file/file-intl.d.ts +44 -0
- package/lib/file/file-preview.component.d.ts +29 -0
- package/lib/file/index.d.ts +4 -0
- package/lib/file/module.d.ts +13 -0
- package/lib/modal/modal-container.d.ts +4 -5
- package/lib/modal/modal-ref.d.ts +1 -0
- package/lib/modal/modal.d.ts +3 -4
- package/lib/modal/module.d.ts +1 -2
- package/lib/snack-bar/index.d.ts +6 -0
- package/lib/snack-bar/module.d.ts +13 -0
- package/lib/snack-bar/simple-snack-bar.component.d.ts +51 -0
- package/lib/snack-bar/snack-bar-animations.d.ts +8 -0
- package/lib/snack-bar/snack-bar-config.d.ts +55 -0
- package/lib/snack-bar/snack-bar-container.d.ts +111 -0
- package/lib/snack-bar/snack-bar-ref.d.ts +51 -0
- package/lib/snack-bar/snack-bar.d.ts +89 -0
- package/lib/system-header/index.d.ts +2 -0
- package/lib/system-header/module.d.ts +9 -0
- package/lib/system-header/system-header.component.d.ts +41 -0
- package/package.json +1 -1
- package/public-api.d.ts +3 -0
- package/esm2020/lib/modal/modal-animations.mjs +0 -29
- package/lib/modal/modal-animations.d.ts +0 -8
|
@@ -0,0 +1,319 @@
|
|
|
1
|
+
/*
|
|
2
|
+
* Copyright (c) 2024 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 "@angular/common";
|
|
13
|
+
import * as i3 from "../common-components/error-text/errorText.component";
|
|
14
|
+
import * as i4 from "../common-components/label-text/labelText.component";
|
|
15
|
+
import * as i5 from "../button/button.component";
|
|
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');
|
|
154
|
+
if (helperText) {
|
|
155
|
+
this._helperTextId = `bao-guiding-text-${fileTextUniqueId++}`;
|
|
156
|
+
this.renderer.setAttribute(helperText.firstElementChild, '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: "14.2.1", 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: "14.0.0", version: "14.2.1", 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-file-preview:first-child{margin-top:1rem}bao-file-input .bao-error{margin-top:.5rem}\n"], dependencies: [{ kind: "directive", type: i0.forwardRef(function () { return i2.NgIf; }), selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i0.forwardRef(function () { return i3.BaoErrorTextComponent; }), selector: "bao-error, [bao-error]" }, { kind: "component", type: i0.forwardRef(function () { return i4.BaoLabelTextComponent; }), selector: "bao-label, [bao-label]", inputs: ["required"] }, { kind: "component", type: i0.forwardRef(function () { return i5.BaoButtonComponent; }), selector: "button[bao-button]", inputs: ["displayType", "level", "size", "loading", "reversed", "loadingSpinnerAriaLabel", "fullWidth"] }, { kind: "directive", type: i0.forwardRef(function () { return BaoFileDropzoneIntructions; }), selector: "bao-file-dropzone-instructions, [bao-file-dropzone-instructions]" }, { kind: "directive", type: i0.forwardRef(function () { return BaoFileDropDirective; }), selector: "[baoFileDrop]", outputs: ["fileDrop"] }], encapsulation: i0.ViewEncapsulation.None });
|
|
189
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.1", 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-file-preview:first-child{margin-top:1rem}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: "14.2.1", ngImport: i0, type: BaoFileDropDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
|
|
277
|
+
BaoFileDropDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "14.2.1", 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: "14.2.1", 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: "14.2.1", ngImport: i0, type: BaoFileDropzoneIntructions, deps: [{ token: i0.Renderer2 }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Directive });
|
|
309
|
+
BaoFileDropzoneIntructions.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "14.2.1", 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: "14.2.1", 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) 2024 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    );\n    if (helperText) {\n      this._helperTextId = `bao-guiding-text-${fileTextUniqueId++}`;\n      this.renderer.setAttribute(\n        helperText.firstElementChild,\n        'id',\n        this._helperTextId\n      );\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) 2024 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: "14.2.1", ngImport: i0, type: BaoFileIntl, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
31
|
+
BaoFileIntl.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "14.2.1", ngImport: i0, type: BaoFileIntl, providedIn: 'root' });
|
|
32
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.1", 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: "14.2.1", ngImport: i0, type: BaoFileIntlEnglish, deps: null, target: i0.ɵɵFactoryTarget.Injectable });
|
|
50
|
+
BaoFileIntlEnglish.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "14.2.1", ngImport: i0, type: BaoFileIntlEnglish });
|
|
51
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.1", 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,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmlsZS1pbnRsLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvYW5ndWxhci11aS9zcmMvbGliL2ZpbGUvZmlsZS1pbnRsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7O0dBSUc7QUFFSCxPQUFPLEVBQUUsVUFBVSxFQUFFLFFBQVEsRUFBRSxRQUFRLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDL0QsT0FBTyxFQUFFLE9BQU8sRUFBRSxNQUFNLE1BQU0sQ0FBQzs7QUFFL0I7OztHQUdHO0FBRUgsTUFBTSxPQUFPLFdBQVc7SUFEeEI7UUFFRTs7O1dBR0c7UUFDTSxZQUFPLEdBQWtCLElBQUksT0FBTyxFQUFRLENBQUM7UUFFdEQsdUNBQXVDO1FBQ3ZDLHdCQUFtQixHQUFHLFdBQVcsQ0FBQztRQUVsQyx3Q0FBd0M7UUFDeEMsZ0NBQTJCLEdBQUcsOEJBQThCLENBQUM7UUFFN0QsOERBQThEO1FBQzlELDJCQUFzQixHQUFHLHlDQUF5QyxDQUFDO1FBRW5FLHFFQUFxRTtRQUNyRSxnQ0FBMkIsR0FBRyx5Q0FBeUMsQ0FBQztLQUN6RTs7d0dBbEJZLFdBQVc7NEdBQVgsV0FBVyxjQURFLE1BQU07MkZBQ25CLFdBQVc7a0JBRHZCLFVBQVU7bUJBQUMsRUFBRSxVQUFVLEVBQUUsTUFBTSxFQUFFOztBQXNCbEMsTUFBTSxPQUFPLGtCQUFtQixTQUFRLFdBQVc7SUFEbkQ7O1FBRUUsdUNBQXVDO1FBQ3ZDLHdCQUFtQixHQUFHLFFBQVEsQ0FBQztRQUUvQix3Q0FBd0M7UUFDeEMsZ0NBQTJCLEdBQUcsd0JBQXdCLENBQUM7UUFFdkQsOERBQThEO1FBQzlELDJCQUFzQixHQUFHLG9DQUFvQyxDQUFDO1FBRTlELHFFQUFxRTtRQUNyRSxnQ0FBMkIsR0FBRyx5Q0FBeUMsQ0FBQztLQUN6RTs7K0dBWlksa0JBQWtCO21IQUFsQixrQkFBa0I7MkZBQWxCLGtCQUFrQjtrQkFEOUIsVUFBVTs7QUFlWCxvQkFBb0I7QUFDcEIsTUFBTSxVQUFVLDhCQUE4QixDQUFDLFVBQXVCO0lBQ3BFLE9BQU8sVUFBVSxJQUFJLElBQUksV0FBVyxFQUFFLENBQUM7QUFDekMsQ0FBQztBQUVELG9CQUFvQjtBQUNwQixNQUFNLENBQUMsTUFBTSxzQkFBc0IsR0FBRztJQUNwQyx3RkFBd0Y7SUFDeEYsT0FBTyxFQUFFLFdBQVc7SUFDcEIsSUFBSSxFQUFFLENBQUMsQ0FBQyxJQUFJLFFBQVEsRUFBRSxFQUFFLElBQUksUUFBUSxFQUFFLEVBQUUsV0FBVyxDQUFDLENBQUM7SUFDckQsVUFBVSxFQUFFLDhCQUE4QjtDQUMzQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiAqIENvcHlyaWdodCAoYykgMjAyNCBWaWxsZSBkZSBNb250cmVhbC4gQWxsIHJpZ2h0cyByZXNlcnZlZC5cbiAqIExpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgbGljZW5zZS5cbiAqIFNlZSBMSUNFTlNFIGZpbGUgaW4gdGhlIHByb2plY3Qgcm9vdCBmb3IgZnVsbCBsaWNlbnNlIGluZm9ybWF0aW9uLlxuICovXG5cbmltcG9ydCB7IEluamVjdGFibGUsIE9wdGlvbmFsLCBTa2lwU2VsZiB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgU3ViamVjdCB9IGZyb20gJ3J4anMnO1xuXG4vKipcbiAqIFRvIG1vZGlmeSB0aGUgbGFiZWxzIGFuZCB0ZXh0IGRpc3BsYXllZCwgY3JlYXRlIGEgbmV3IGluc3RhbmNlIG9mIEJhb0ZpbGVJbnRsIGFuZFxuICogaW5jbHVkZSBpdCBpbiBhIGN1c3RvbSBwcm92aWRlclxuICovXG5ASW5qZWN0YWJsZSh7IHByb3ZpZGVkSW46ICdyb290JyB9KVxuZXhwb3J0IGNsYXNzIEJhb0ZpbGVJbnRsIHtcbiAgLyoqXG4gICAqIFN0cmVhbSB0byBlbWl0IGZyb20gd2hlbiBsYWJlbHMgYXJlIGNoYW5nZWQuIFVzZSB0aGlzIHRvIG5vdGlmeSBjb21wb25lbnRzIHdoZW4gdGhlIGxhYmVscyBoYXZlXG4gICAqIGNoYW5nZWQgYWZ0ZXIgaW5pdGlhbGl6YXRpb24uXG4gICAqL1xuICByZWFkb25seSBjaGFuZ2VzOiBTdWJqZWN0PHZvaWQ+ID0gbmV3IFN1YmplY3Q8dm9pZD4oKTtcblxuICAvKiogVGhlIGxhYmVsIGZvciBidXR0b24gaW4gZHJvcHpvbmUgKi9cbiAgZHJvcHpvbmVCdXR0b25MYWJlbCA9ICdQYXJjb3VyaXInO1xuXG4gIC8qKiBUaGUgZGVmYXVsdCBkcm9wem9uZSBpbnN0cnVjdGlvbnMgKi9cbiAgZGVmYXVsdERyb3B6b25lSW5zdHJ1Y3Rpb25zID0gJ291IGTDqXBvc2VyIHZvdHJlIGZpY2hpZXIgaWNpJztcblxuICAvKiogRXJyb3IgbWVzc2FnZSBkaXNwbGF5ZWQgd2hlbiB1cGxvYWRlZCBmaWxlIGlzIHRvbyBsYXJnZSAqL1xuICBmaWxlVG9vQmlnRXJyb3JNZXNzYWdlID0gJ0xhIHRhaWxsZSBkZSBjZSBmaWNoaWVyIGVzdCB0cm9wIGdyYW5kZSc7XG5cbiAgLyoqIEVycm9yIG1lc3NhZ2UgZGlzcGxheWVkIHdoZW4gdXBsb2FkZWQgZmlsZSBoYXMgYW4gaW52YWxpZCB0eXBlICovXG4gIGludmFsaWRGaWxlVHlwZUVycm9yTWVzc2FnZSA9IFwiQ2UgZm9ybWF0IGRlIGZpY2hpZXIgbidlc3QgcGFzIGF1dG9yaXPDqVwiO1xufVxuXG5ASW5qZWN0YWJsZSgpXG5leHBvcnQgY2xhc3MgQmFvRmlsZUludGxFbmdsaXNoIGV4dGVuZHMgQmFvRmlsZUludGwge1xuICAvKiogVGhlIGxhYmVsIGZvciBidXR0b24gaW4gZHJvcHpvbmUgKi9cbiAgZHJvcHpvbmVCdXR0b25MYWJlbCA9ICdCcm93c2UnO1xuXG4gIC8qKiBUaGUgZGVmYXVsdCBkcm9wem9uZSBpbnN0cnVjdGlvbnMgKi9cbiAgZGVmYXVsdERyb3B6b25lSW5zdHJ1Y3Rpb25zID0gJ29yIGRyb3AgeW91ciBmaWxlIGhlcmUnO1xuXG4gIC8qKiBFcnJvciBtZXNzYWdlIGRpc3BsYXllZCB3aGVuIHVwbG9hZGVkIGZpbGUgaXMgdG9vIGxhcmdlICovXG4gIGZpbGVUb29CaWdFcnJvck1lc3NhZ2UgPSAnVGhlIHNpemUgb2YgdGhpcyBmaWxlIGlzIHRvbyBsYXJnZSc7XG5cbiAgLyoqIEVycm9yIG1lc3NhZ2UgZGlzcGxheWVkIHdoZW4gdXBsb2FkZWQgZmlsZSBoYXMgYW4gaW52YWxpZCB0eXBlICovXG4gIGludmFsaWRGaWxlVHlwZUVycm9yTWVzc2FnZSA9ICdUaGUgZm9ybWF0IG9mIHRoaXMgZmlsZSBpcyB1bmF1dGhvcml6ZWQnO1xufVxuXG4vKiogQGRvY3MtcHJpdmF0ZSAqL1xuZXhwb3J0IGZ1bmN0aW9uIEJBT19GSUxFX0lOVExfUFJPVklERVJfRkFDVE9SWShwYXJlbnRJbnRsOiBCYW9GaWxlSW50bCkge1xuICByZXR1cm4gcGFyZW50SW50bCB8fCBuZXcgQmFvRmlsZUludGwoKTtcbn1cblxuLyoqIEBkb2NzLXByaXZhdGUgKi9cbmV4cG9ydCBjb25zdCBCQU9fRklMRV9JTlRMX1BST1ZJREVSID0ge1xuICAvLyBJZiB0aGVyZSBpcyBhbHJlYWR5IGFuIEJhb0ZpbGVJbnRsIGF2YWlsYWJsZSwgdXNlIHRoYXQuIE90aGVyd2lzZSwgcHJvdmlkZSBhIG5ldyBvbmUuXG4gIHByb3ZpZGU6IEJhb0ZpbGVJbnRsLFxuICBkZXBzOiBbW25ldyBPcHRpb25hbCgpLCBuZXcgU2tpcFNlbGYoKSwgQmFvRmlsZUludGxdXSxcbiAgdXNlRmFjdG9yeTogQkFPX0ZJTEVfSU5UTF9QUk9WSURFUl9GQUNUT1JZXG59O1xuIl19
|
|
@@ -0,0 +1,86 @@
|
|
|
1
|
+
/*
|
|
2
|
+
* Copyright (c) 2024 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 "@angular/common";
|
|
9
|
+
import * as i2 from "../icon/icon.component";
|
|
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: "14.2.1", ngImport: i0, type: BaoFilePreviewComponent, deps: [{ token: i0.ElementRef }, { token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Component });
|
|
75
|
+
BaoFilePreviewComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.2.1", 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>.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"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.BaoIconComponent, selector: "bao-icon", inputs: ["color", "size", "svgIcon", "title"], exportAs: ["baoIcon"] }], encapsulation: i0.ViewEncapsulation.None });
|
|
76
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.1", 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>.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) 2024 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) 2024 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,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9hbmd1bGFyLXVpL3NyYy9saWIvZmlsZS9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7OztHQUlHO0FBQ0gsY0FBYyxVQUFVLENBQUM7QUFDekIsY0FBYyx3QkFBd0IsQ0FBQztBQUN2QyxjQUFjLDBCQUEwQixDQUFDO0FBQ3pDLGNBQWMsYUFBYSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiAqIENvcHlyaWdodCAoYykgMjAyNCBWaWxsZSBkZSBNb250cmVhbC4gQWxsIHJpZ2h0cyByZXNlcnZlZC5cbiAqIExpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgbGljZW5zZS5cbiAqIFNlZSBMSUNFTlNFIGZpbGUgaW4gdGhlIHByb2plY3Qgcm9vdCBmb3IgZnVsbCBsaWNlbnNlIGluZm9ybWF0aW9uLlxuICovXG5leHBvcnQgKiBmcm9tICcuL21vZHVsZSc7XG5leHBvcnQgKiBmcm9tICcuL2ZpbGUtaW5wdXQuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vZmlsZS1wcmV2aWV3LmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2ZpbGUtaW50bCc7XG4iXX0=
|