@ecodev/natural 62.1.2 → 63.0.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/fesm2022/ecodev-natural-vanilla.mjs +1193 -0
- package/fesm2022/ecodev-natural-vanilla.mjs.map +1 -0
- package/fesm2022/ecodev-natural.mjs +436 -425
- package/fesm2022/ecodev-natural.mjs.map +1 -1
- package/lib/classes/network-activity.service.d.ts +54 -0
- package/lib/classes/validators.d.ts +1 -1
- package/lib/modules/columns-picker/columns-picker.component.d.ts +2 -2
- package/lib/modules/common/services/seo.provider.d.ts +2 -2
- package/lib/modules/dropdown-components/type-select/type-select.component.d.ts +1 -1
- package/lib/modules/file/abstract-file.d.ts +6 -3
- package/lib/modules/file/component/file.component.d.ts +2 -2
- package/lib/modules/file/file-drop.directive.d.ts +2 -3
- package/lib/modules/fixed-button-detail/fixed-button-detail.component.d.ts +2 -3
- package/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.component.d.ts +3 -3
- package/lib/modules/icon/icon.module.d.ts +2 -2
- package/lib/modules/panels/panels.service.d.ts +1 -2
- package/lib/modules/relations/relations.component.d.ts +3 -3
- package/lib/modules/search/dropdown-container/dropdown-container.component.d.ts +2 -3
- package/lib/modules/search/group/group.component.d.ts +2 -3
- package/lib/modules/search/input/input.component.d.ts +5 -5
- package/lib/modules/search/search/search.component.d.ts +2 -2
- package/lib/modules/select/abstract-select.component.d.ts +3 -3
- package/lib/modules/select/select/select.component.d.ts +1 -1
- package/lib/modules/sidenav/sidenav-container/sidenav-container.component.d.ts +1 -1
- package/lib/modules/table-button/table-button.component.d.ts +4 -2
- package/package.json +16 -14
- package/public-api.d.ts +1 -0
- package/src/lib/_natural.theme.scss +1 -2
- package/vanilla/index.d.ts +5 -0
- package/vanilla/package.json +3 -0
- package/vanilla/public-api.d.ts +11 -0
- package/vanilla/src/lib/classes/crypto.d.ts +8 -0
- package/vanilla/src/lib/classes/data-source.d.ts +32 -0
- package/vanilla/src/lib/classes/query-variable-manager-utils.d.ts +2 -0
- package/vanilla/src/lib/classes/query-variable-manager.d.ts +91 -0
- package/vanilla/src/lib/classes/signing.d.ts +7 -0
- package/vanilla/src/lib/classes/utility.d.ts +77 -0
- package/vanilla/src/lib/modules/search/classes/graphql-doctrine.types.d.ts +83 -0
- package/vanilla/src/lib/modules/search/classes/utils.d.ts +17 -0
- package/vanilla/src/lib/modules/search/types/dropdown-component.d.ts +20 -0
- package/vanilla/src/lib/modules/search/types/facet.d.ts +75 -0
- package/vanilla/src/lib/modules/search/types/values.d.ts +32 -0
- package/vanilla/src/lib/services/abstract-model.service.d.ts +244 -0
- package/vanilla/src/lib/services/debounce.service.d.ts +52 -0
- package/vanilla/src/lib/types/types.d.ts +100 -0
- package/esm2022/ecodev-natural.mjs +0 -5
- package/esm2022/lib/classes/abstract-detail.mjs +0 -229
- package/esm2022/lib/classes/abstract-editable-list.mjs +0 -99
- package/esm2022/lib/classes/abstract-list.mjs +0 -461
- package/esm2022/lib/classes/abstract-navigable-list.mjs +0 -133
- package/esm2022/lib/classes/apollo-utils.mjs +0 -59
- package/esm2022/lib/classes/crypto.mjs +0 -23
- package/esm2022/lib/classes/cumulative-changes.mjs +0 -50
- package/esm2022/lib/classes/data-source.mjs +0 -71
- package/esm2022/lib/classes/providers.mjs +0 -13
- package/esm2022/lib/classes/query-variable-manager-utils.mjs +0 -14
- package/esm2022/lib/classes/query-variable-manager.mjs +0 -172
- package/esm2022/lib/classes/rxjs.mjs +0 -54
- package/esm2022/lib/classes/signing.mjs +0 -38
- package/esm2022/lib/classes/tld.mjs +0 -1476
- package/esm2022/lib/classes/utility.mjs +0 -234
- package/esm2022/lib/classes/validators.mjs +0 -179
- package/esm2022/lib/directives/http-prefix.directive.mjs +0 -47
- package/esm2022/lib/modules/alert/alert.service.mjs +0 -53
- package/esm2022/lib/modules/alert/confirm.component.mjs +0 -16
- package/esm2022/lib/modules/alert/public-api.mjs +0 -6
- package/esm2022/lib/modules/avatar/component/avatar.component.mjs +0 -203
- package/esm2022/lib/modules/avatar/public-api.mjs +0 -6
- package/esm2022/lib/modules/avatar/service/avatar.service.mjs +0 -63
- package/esm2022/lib/modules/avatar/sources/gravatar.mjs +0 -29
- package/esm2022/lib/modules/avatar/sources/image.mjs +0 -13
- package/esm2022/lib/modules/avatar/sources/initials.mjs +0 -39
- package/esm2022/lib/modules/avatar/sources/source.mjs +0 -16
- package/esm2022/lib/modules/columns-picker/columns-picker.component.mjs +0 -145
- package/esm2022/lib/modules/columns-picker/public-api.mjs +0 -5
- package/esm2022/lib/modules/columns-picker/types.mjs +0 -2
- package/esm2022/lib/modules/common/directives/background-density.directive.mjs +0 -63
- package/esm2022/lib/modules/common/directives/linkable-tab.directive.mjs +0 -93
- package/esm2022/lib/modules/common/directives/src-density.directive.mjs +0 -72
- package/esm2022/lib/modules/common/pipes/capitalize.pipe.mjs +0 -24
- package/esm2022/lib/modules/common/pipes/ellipsis.pipe.mjs +0 -17
- package/esm2022/lib/modules/common/pipes/enum.pipe.mjs +0 -24
- package/esm2022/lib/modules/common/pipes/time-ago.pipe.mjs +0 -140
- package/esm2022/lib/modules/common/public-api.mjs +0 -14
- package/esm2022/lib/modules/common/services/memory-storage.mjs +0 -110
- package/esm2022/lib/modules/common/services/seo.provider.mjs +0 -23
- package/esm2022/lib/modules/common/services/seo.service.mjs +0 -235
- package/esm2022/lib/modules/detail-header/detail-header.component.mjs +0 -84
- package/esm2022/lib/modules/detail-header/public-api.mjs +0 -5
- package/esm2022/lib/modules/dialog-trigger/dialog-trigger.component.mjs +0 -72
- package/esm2022/lib/modules/dialog-trigger/public-api.mjs +0 -5
- package/esm2022/lib/modules/dropdown-components/abstract-association-select-component.directive.mjs +0 -100
- package/esm2022/lib/modules/dropdown-components/public-api.mjs +0 -14
- package/esm2022/lib/modules/dropdown-components/type-boolean/type-boolean.component.mjs +0 -39
- package/esm2022/lib/modules/dropdown-components/type-date/type-date.component.mjs +0 -173
- package/esm2022/lib/modules/dropdown-components/type-date-range/type-date-range.component.mjs +0 -134
- package/esm2022/lib/modules/dropdown-components/type-hierarchic-selector/type-hierarchic-selector.component.mjs +0 -80
- package/esm2022/lib/modules/dropdown-components/type-natural-select/type-natural-select.component.mjs +0 -48
- package/esm2022/lib/modules/dropdown-components/type-number/type-number.component.mjs +0 -110
- package/esm2022/lib/modules/dropdown-components/type-options/type-options.component.mjs +0 -64
- package/esm2022/lib/modules/dropdown-components/type-select/type-select.component.mjs +0 -175
- package/esm2022/lib/modules/dropdown-components/type-text/type-text.component.mjs +0 -62
- package/esm2022/lib/modules/dropdown-components/types.mjs +0 -41
- package/esm2022/lib/modules/dropdown-components/utils.mjs +0 -35
- package/esm2022/lib/modules/file/abstract-file.mjs +0 -230
- package/esm2022/lib/modules/file/component/file.component.mjs +0 -172
- package/esm2022/lib/modules/file/file-drop.directive.mjs +0 -111
- package/esm2022/lib/modules/file/file-select.directive.mjs +0 -26
- package/esm2022/lib/modules/file/file.service.mjs +0 -43
- package/esm2022/lib/modules/file/public-api.mjs +0 -9
- package/esm2022/lib/modules/file/types.mjs +0 -2
- package/esm2022/lib/modules/file/utils.mjs +0 -129
- package/esm2022/lib/modules/fixed-button/fixed-button.component.mjs +0 -30
- package/esm2022/lib/modules/fixed-button/public-api.mjs +0 -5
- package/esm2022/lib/modules/fixed-button-detail/fixed-button-detail.component.mjs +0 -56
- package/esm2022/lib/modules/fixed-button-detail/public-api.mjs +0 -5
- package/esm2022/lib/modules/hierarchic-selector/classes/flat-node.mjs +0 -18
- package/esm2022/lib/modules/hierarchic-selector/classes/hierarchic-configuration.mjs +0 -2
- package/esm2022/lib/modules/hierarchic-selector/classes/hierarchic-filters-configuration.mjs +0 -2
- package/esm2022/lib/modules/hierarchic-selector/classes/model-node.mjs +0 -14
- package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.component.mjs +0 -398
- package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.service.mjs +0 -243
- package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector-dialog/hierarchic-selector-dialog.component.mjs +0 -38
- package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector-dialog/hierarchic-selector-dialog.service.mjs +0 -22
- package/esm2022/lib/modules/hierarchic-selector/public-api.mjs +0 -10
- package/esm2022/lib/modules/icon/icon.directive.mjs +0 -96
- package/esm2022/lib/modules/icon/icon.module.mjs +0 -33
- package/esm2022/lib/modules/icon/public-api.mjs +0 -6
- package/esm2022/lib/modules/logger/error-handler.mjs +0 -87
- package/esm2022/lib/modules/logger/error.module.mjs +0 -22
- package/esm2022/lib/modules/logger/public-api.mjs +0 -6
- package/esm2022/lib/modules/matomo/matomo.service.mjs +0 -96
- package/esm2022/lib/modules/matomo/public-api.mjs +0 -5
- package/esm2022/lib/modules/panels/abstract-panel.mjs +0 -76
- package/esm2022/lib/modules/panels/fallback-if-no-opened-panels.urlmatcher.mjs +0 -12
- package/esm2022/lib/modules/panels/panels.component.mjs +0 -27
- package/esm2022/lib/modules/panels/panels.module.mjs +0 -10
- package/esm2022/lib/modules/panels/panels.service.mjs +0 -329
- package/esm2022/lib/modules/panels/panels.urlmatcher.mjs +0 -75
- package/esm2022/lib/modules/panels/public-api.mjs +0 -11
- package/esm2022/lib/modules/panels/types.mjs +0 -3
- package/esm2022/lib/modules/relations/public-api.mjs +0 -5
- package/esm2022/lib/modules/relations/relations.component.mjs +0 -254
- package/esm2022/lib/modules/search/classes/graphql-doctrine.mjs +0 -111
- package/esm2022/lib/modules/search/classes/graphql-doctrine.types.mjs +0 -14
- package/esm2022/lib/modules/search/classes/transformers.mjs +0 -142
- package/esm2022/lib/modules/search/classes/url.mjs +0 -53
- package/esm2022/lib/modules/search/classes/utils.mjs +0 -25
- package/esm2022/lib/modules/search/dropdown-container/dropdown-container-animations.mjs +0 -44
- package/esm2022/lib/modules/search/dropdown-container/dropdown-container.component.mjs +0 -87
- package/esm2022/lib/modules/search/dropdown-container/dropdown-ref.mjs +0 -24
- package/esm2022/lib/modules/search/dropdown-container/dropdown.service.mjs +0 -90
- package/esm2022/lib/modules/search/facet-selector/facet-selector.component.mjs +0 -45
- package/esm2022/lib/modules/search/group/group.component.mjs +0 -53
- package/esm2022/lib/modules/search/input/input.component.mjs +0 -365
- package/esm2022/lib/modules/search/public-api.mjs +0 -7
- package/esm2022/lib/modules/search/search/search.component.mjs +0 -102
- package/esm2022/lib/modules/search/types/dropdown-component.mjs +0 -2
- package/esm2022/lib/modules/search/types/facet.mjs +0 -2
- package/esm2022/lib/modules/search/types/values.mjs +0 -2
- package/esm2022/lib/modules/select/abstract-select.component.mjs +0 -232
- package/esm2022/lib/modules/select/public-api.mjs +0 -7
- package/esm2022/lib/modules/select/select/select.component.mjs +0 -310
- package/esm2022/lib/modules/select/select-enum/select-enum.component.mjs +0 -57
- package/esm2022/lib/modules/select/select-hierarchic/select-hierarchic.component.mjs +0 -155
- package/esm2022/lib/modules/sidenav/public-api.mjs +0 -9
- package/esm2022/lib/modules/sidenav/sidenav/sidenav.component.mjs +0 -15
- package/esm2022/lib/modules/sidenav/sidenav-container/sidenav-container.component.mjs +0 -90
- package/esm2022/lib/modules/sidenav/sidenav-content/sidenav-content.component.mjs +0 -11
- package/esm2022/lib/modules/sidenav/sidenav-stack.service.mjs +0 -50
- package/esm2022/lib/modules/sidenav/sidenav.service.mjs +0 -196
- package/esm2022/lib/modules/stamp/public-api.mjs +0 -5
- package/esm2022/lib/modules/stamp/stamp.component.mjs +0 -23
- package/esm2022/lib/modules/table-button/public-api.mjs +0 -5
- package/esm2022/lib/modules/table-button/table-button.component.mjs +0 -78
- package/esm2022/lib/services/abstract-model.service.mjs +0 -526
- package/esm2022/lib/services/debounce.service.mjs +0 -149
- package/esm2022/lib/services/enum.service.mjs +0 -64
- package/esm2022/lib/services/link-mutation.service.mjs +0 -154
- package/esm2022/lib/services/persistence.service.mjs +0 -115
- package/esm2022/lib/services/swiss-parsing-date-adapter.service.mjs +0 -63
- package/esm2022/lib/types/types.mjs +0 -2
- package/esm2022/public-api.mjs +0 -46
|
@@ -1,230 +0,0 @@
|
|
|
1
|
-
import { Directive, EventEmitter, HostListener, Inject, Input, Output, } from '@angular/core';
|
|
2
|
-
import { acceptType, createInvisibleFileInputWrap, detectSwipe, eventToFiles, fileListToArray, isDirectory, isFileInput, stopEvent, } from './utils';
|
|
3
|
-
import { DOCUMENT } from '@angular/common';
|
|
4
|
-
import { forkJoin, map, of, tap } from 'rxjs';
|
|
5
|
-
import * as i0 from "@angular/core";
|
|
6
|
-
import * as i1 from "./file.service";
|
|
7
|
-
/**
|
|
8
|
-
* A master base set of logic intended to support file select/drag/drop operations
|
|
9
|
-
*
|
|
10
|
-
* In most cases you probably want click-to-select and drag-to-select, so you should use:
|
|
11
|
-
*
|
|
12
|
-
* <div naturalFileDrop [selectable]="true"></div>
|
|
13
|
-
*
|
|
14
|
-
* @dynamic
|
|
15
|
-
*/
|
|
16
|
-
export class NaturalAbstractFile {
|
|
17
|
-
element;
|
|
18
|
-
naturalFileService;
|
|
19
|
-
document;
|
|
20
|
-
fileElement;
|
|
21
|
-
/**
|
|
22
|
-
* Whether we should accept a single file or multiple files
|
|
23
|
-
*/
|
|
24
|
-
multiple = false;
|
|
25
|
-
/**
|
|
26
|
-
* Comma-separated list of unique file type specifiers. Like the native element
|
|
27
|
-
* it can be a mix of mime-type and file extensions.
|
|
28
|
-
*
|
|
29
|
-
* See https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/file#accept
|
|
30
|
-
*/
|
|
31
|
-
accept = '';
|
|
32
|
-
/**
|
|
33
|
-
* Maximum file size in bytes. 0 means no validation at all.
|
|
34
|
-
*/
|
|
35
|
-
maxSize = 0;
|
|
36
|
-
/**
|
|
37
|
-
* Disable the file selection entirely
|
|
38
|
-
*/
|
|
39
|
-
fileSelectionDisabled = false;
|
|
40
|
-
/**
|
|
41
|
-
* Whether the user can click on the element to select something
|
|
42
|
-
*
|
|
43
|
-
* This has only effect during initialization. Subsequent changes will have
|
|
44
|
-
* no effect.
|
|
45
|
-
*/
|
|
46
|
-
selectable = false;
|
|
47
|
-
/**
|
|
48
|
-
* If true, the file selection will be broadcast through `NaturalFileService.filesChanged`.
|
|
49
|
-
*
|
|
50
|
-
* It is useful to set this to false if there is two upload on a page with different purpose
|
|
51
|
-
* and the second upload should not be confused with the first one.
|
|
52
|
-
*/
|
|
53
|
-
broadcast = true;
|
|
54
|
-
/**
|
|
55
|
-
* The single valid file that has been selected.
|
|
56
|
-
*
|
|
57
|
-
* It is for convenience of use, and will only emit if there is at least one
|
|
58
|
-
* valid file. See `filesChange` for a more complete output.
|
|
59
|
-
*/
|
|
60
|
-
fileChange = new EventEmitter();
|
|
61
|
-
/**
|
|
62
|
-
* The list of files that have been selected.
|
|
63
|
-
*/
|
|
64
|
-
filesChange = new EventEmitter();
|
|
65
|
-
constructor(element, naturalFileService, document) {
|
|
66
|
-
this.element = element;
|
|
67
|
-
this.naturalFileService = naturalFileService;
|
|
68
|
-
this.document = document;
|
|
69
|
-
}
|
|
70
|
-
ngOnDestroy() {
|
|
71
|
-
delete this.fileElement; // faster memory release of dom element
|
|
72
|
-
}
|
|
73
|
-
ngOnInit() {
|
|
74
|
-
if (this.selectable) {
|
|
75
|
-
this.enableSelecting();
|
|
76
|
-
}
|
|
77
|
-
this.getFileElement().multiple = this.multiple;
|
|
78
|
-
}
|
|
79
|
-
ngOnChanges(changes) {
|
|
80
|
-
if (changes.accept) {
|
|
81
|
-
this.getFileElement().setAttribute('accept', changes.accept.currentValue || '*');
|
|
82
|
-
}
|
|
83
|
-
}
|
|
84
|
-
getFileElement() {
|
|
85
|
-
if (this.fileElement) {
|
|
86
|
-
return this.fileElement;
|
|
87
|
-
}
|
|
88
|
-
// elm is a file input
|
|
89
|
-
if (isFileInput(this.element.nativeElement)) {
|
|
90
|
-
this.fileElement = this.element.nativeElement;
|
|
91
|
-
return this.fileElement;
|
|
92
|
-
}
|
|
93
|
-
// create foo file input
|
|
94
|
-
const label = createInvisibleFileInputWrap(this.document);
|
|
95
|
-
this.fileElement = label.getElementsByTagName('input')[0];
|
|
96
|
-
this.fileElement.addEventListener('change', this.changeFn.bind(this));
|
|
97
|
-
this.element.nativeElement.appendChild(label);
|
|
98
|
-
return this.fileElement;
|
|
99
|
-
}
|
|
100
|
-
enableSelecting() {
|
|
101
|
-
const elm = this.element.nativeElement;
|
|
102
|
-
if (isFileInput(elm)) {
|
|
103
|
-
const bindedHandler = () => this.beforeSelect();
|
|
104
|
-
elm.addEventListener('click', bindedHandler);
|
|
105
|
-
elm.addEventListener('touchstart', bindedHandler);
|
|
106
|
-
return;
|
|
107
|
-
}
|
|
108
|
-
else {
|
|
109
|
-
const bindedHandler = (event) => this.clickHandler(event);
|
|
110
|
-
elm.addEventListener('click', bindedHandler);
|
|
111
|
-
elm.addEventListener('touchstart', bindedHandler);
|
|
112
|
-
elm.addEventListener('touchend', bindedHandler);
|
|
113
|
-
}
|
|
114
|
-
}
|
|
115
|
-
handleFiles(files) {
|
|
116
|
-
const selection = {
|
|
117
|
-
valid: [],
|
|
118
|
-
invalid: [],
|
|
119
|
-
};
|
|
120
|
-
forkJoin(files.map(file => this.validate(file).pipe(tap(error => {
|
|
121
|
-
if (error) {
|
|
122
|
-
selection.invalid.push({
|
|
123
|
-
file: file,
|
|
124
|
-
error: error,
|
|
125
|
-
});
|
|
126
|
-
}
|
|
127
|
-
else {
|
|
128
|
-
selection.valid.push(file);
|
|
129
|
-
}
|
|
130
|
-
})))).subscribe(() => {
|
|
131
|
-
if (selection.valid.length) {
|
|
132
|
-
this.fileChange.emit(selection.valid[0]);
|
|
133
|
-
}
|
|
134
|
-
if (selection.valid.length || selection.invalid.length) {
|
|
135
|
-
this.filesChange.emit(selection);
|
|
136
|
-
if (this.broadcast) {
|
|
137
|
-
this.naturalFileService.filesChanged.next(selection);
|
|
138
|
-
}
|
|
139
|
-
}
|
|
140
|
-
this.getFileElement().value = '';
|
|
141
|
-
});
|
|
142
|
-
}
|
|
143
|
-
/**
|
|
144
|
-
* Called when input has files
|
|
145
|
-
*/
|
|
146
|
-
changeFn(event) {
|
|
147
|
-
if (!(event.target instanceof HTMLInputElement)) {
|
|
148
|
-
return;
|
|
149
|
-
}
|
|
150
|
-
const fileList = event.target.files;
|
|
151
|
-
if (!fileList) {
|
|
152
|
-
return;
|
|
153
|
-
}
|
|
154
|
-
stopEvent(event);
|
|
155
|
-
this.handleFiles(fileListToArray(fileList));
|
|
156
|
-
}
|
|
157
|
-
clickHandler(event) {
|
|
158
|
-
const elm = this.element.nativeElement;
|
|
159
|
-
if (elm.getAttribute('disabled') || this.fileSelectionDisabled) {
|
|
160
|
-
return false;
|
|
161
|
-
}
|
|
162
|
-
// prevent the click if it is a swipe
|
|
163
|
-
if (detectSwipe(event)) {
|
|
164
|
-
return true;
|
|
165
|
-
}
|
|
166
|
-
const fileElm = this.getFileElement();
|
|
167
|
-
fileElm.click();
|
|
168
|
-
this.beforeSelect();
|
|
169
|
-
return false;
|
|
170
|
-
}
|
|
171
|
-
beforeSelect() {
|
|
172
|
-
if (!this.fileElement) {
|
|
173
|
-
return;
|
|
174
|
-
}
|
|
175
|
-
// if no files in array, be sure browser doesnt prevent reselect of same file (see github issue 27)
|
|
176
|
-
this.fileElement.value = '';
|
|
177
|
-
}
|
|
178
|
-
onChange(event) {
|
|
179
|
-
const fileList = this.getFileElement().files;
|
|
180
|
-
const files = fileList ? fileListToArray(fileList) : eventToFiles(event);
|
|
181
|
-
if (!files.length) {
|
|
182
|
-
return;
|
|
183
|
-
}
|
|
184
|
-
stopEvent(event);
|
|
185
|
-
this.handleFiles(files);
|
|
186
|
-
}
|
|
187
|
-
validate(file) {
|
|
188
|
-
return forkJoin({
|
|
189
|
-
accept: of(acceptType(this.accept, file.type, file.name)),
|
|
190
|
-
fileSize: of(!(this.maxSize && file.size > this.maxSize)),
|
|
191
|
-
directory: isDirectory(file),
|
|
192
|
-
}).pipe(map(result => {
|
|
193
|
-
for (const [key, value] of Object.entries(result)) {
|
|
194
|
-
if (!value) {
|
|
195
|
-
return key;
|
|
196
|
-
}
|
|
197
|
-
}
|
|
198
|
-
return null;
|
|
199
|
-
}));
|
|
200
|
-
}
|
|
201
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalAbstractFile, deps: [{ token: i0.ElementRef }, { token: i1.NaturalFileService }, { token: DOCUMENT }], target: i0.ɵɵFactoryTarget.Directive });
|
|
202
|
-
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.3", type: NaturalAbstractFile, isStandalone: true, inputs: { multiple: "multiple", accept: "accept", maxSize: "maxSize", fileSelectionDisabled: "fileSelectionDisabled", selectable: "selectable", broadcast: "broadcast" }, outputs: { fileChange: "fileChange", filesChange: "filesChange" }, host: { listeners: { "change": "onChange($event)" } }, usesOnChanges: true, ngImport: i0 });
|
|
203
|
-
}
|
|
204
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalAbstractFile, decorators: [{
|
|
205
|
-
type: Directive,
|
|
206
|
-
args: [{ standalone: true }]
|
|
207
|
-
}], ctorParameters: () => [{ type: i0.ElementRef }, { type: i1.NaturalFileService }, { type: Document, decorators: [{
|
|
208
|
-
type: Inject,
|
|
209
|
-
args: [DOCUMENT]
|
|
210
|
-
}] }], propDecorators: { multiple: [{
|
|
211
|
-
type: Input
|
|
212
|
-
}], accept: [{
|
|
213
|
-
type: Input
|
|
214
|
-
}], maxSize: [{
|
|
215
|
-
type: Input
|
|
216
|
-
}], fileSelectionDisabled: [{
|
|
217
|
-
type: Input
|
|
218
|
-
}], selectable: [{
|
|
219
|
-
type: Input
|
|
220
|
-
}], broadcast: [{
|
|
221
|
-
type: Input
|
|
222
|
-
}], fileChange: [{
|
|
223
|
-
type: Output
|
|
224
|
-
}], filesChange: [{
|
|
225
|
-
type: Output
|
|
226
|
-
}], onChange: [{
|
|
227
|
-
type: HostListener,
|
|
228
|
-
args: ['change', ['$event']]
|
|
229
|
-
}] } });
|
|
230
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstract-file.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/file/abstract-file.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    Directive,\n    ElementRef,\n    EventEmitter,\n    HostListener,\n    Inject,\n    Input,\n    OnChanges,\n    OnDestroy,\n    OnInit,\n    Output,\n    SimpleChanges,\n} from '@angular/core';\nimport {\n    acceptType,\n    createInvisibleFileInputWrap,\n    detectSwipe,\n    eventToFiles,\n    fileListToArray,\n    isDirectory,\n    isFileInput,\n    stopEvent,\n} from './utils';\nimport {NaturalFileService} from './file.service';\nimport {DOCUMENT} from '@angular/common';\nimport {forkJoin, map, Observable, ObservableInput, of, tap} from 'rxjs';\n\nexport type InvalidFile = {\n    file: File;\n    error: string;\n};\n\nexport type FileSelection = {\n    /**\n     * The list of files that have been selected.\n     */\n    valid: File[];\n\n    /**\n     * The list of files that have been selected but are invalid according to validators.\n     */\n    invalid: InvalidFile[];\n};\n\n/**\n * A master base set of logic intended to support file select/drag/drop operations\n *\n * In most cases you probably want click-to-select and drag-to-select, so you should use:\n *\n *     <div naturalFileDrop [selectable]=\"true\"></div>\n *\n * @dynamic\n */\n@Directive({standalone: true})\nexport abstract class NaturalAbstractFile implements OnInit, OnDestroy, OnChanges {\n    private fileElement?: HTMLInputElement;\n\n    /**\n     * Whether we should accept a single file or multiple files\n     */\n    @Input() public multiple = false;\n\n    /**\n     * Comma-separated list of unique file type specifiers. Like the native element\n     * it can be a mix of mime-type and file extensions.\n     *\n     * See https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/file#accept\n     */\n    @Input() public accept = '';\n\n    /**\n     * Maximum file size in bytes. 0 means no validation at all.\n     */\n    @Input() public maxSize = 0;\n\n    /**\n     * Disable the file selection entirely\n     */\n    @Input() public fileSelectionDisabled = false;\n\n    /**\n     * Whether the user can click on the element to select something\n     *\n     * This has only effect during initialization. Subsequent changes will have\n     * no effect.\n     */\n    @Input() public selectable = false;\n\n    /**\n     * If true, the file selection will be broadcast through `NaturalFileService.filesChanged`.\n     *\n     * It is useful to set this to false if there is two upload on a page with different purpose\n     * and the second upload should not be confused with the first one.\n     */\n    @Input() public broadcast = true;\n\n    /**\n     * The single valid file that has been selected.\n     *\n     * It is for convenience of use, and will only emit if there is at least one\n     * valid file. See `filesChange` for a more complete output.\n     */\n    @Output() public readonly fileChange = new EventEmitter<File>();\n\n    /**\n     * The list of files that have been selected.\n     */\n    @Output() public readonly filesChange = new EventEmitter<FileSelection>();\n\n    public constructor(\n        private readonly element: ElementRef<HTMLElement>,\n        protected readonly naturalFileService: NaturalFileService,\n        @Inject(DOCUMENT) private readonly document: Document,\n    ) {}\n\n    public ngOnDestroy(): void {\n        delete this.fileElement; // faster memory release of dom element\n    }\n\n    public ngOnInit(): void {\n        if (this.selectable) {\n            this.enableSelecting();\n        }\n\n        this.getFileElement().multiple = this.multiple;\n    }\n\n    public ngOnChanges(changes: SimpleChanges): void {\n        if (changes.accept) {\n            this.getFileElement().setAttribute('accept', changes.accept.currentValue || '*');\n        }\n    }\n\n    private getFileElement(): HTMLInputElement {\n        if (this.fileElement) {\n            return this.fileElement;\n        }\n\n        // elm is a file input\n        if (isFileInput(this.element.nativeElement)) {\n            this.fileElement = this.element.nativeElement;\n\n            return this.fileElement;\n        }\n\n        // create foo file input\n        const label = createInvisibleFileInputWrap(this.document);\n        this.fileElement = label.getElementsByTagName('input')[0];\n        this.fileElement.addEventListener('change', this.changeFn.bind(this));\n        this.element.nativeElement.appendChild(label);\n\n        return this.fileElement;\n    }\n\n    private enableSelecting(): void {\n        const elm = this.element.nativeElement;\n\n        if (isFileInput(elm)) {\n            const bindedHandler = (): void => this.beforeSelect();\n            elm.addEventListener('click', bindedHandler);\n            elm.addEventListener('touchstart', bindedHandler);\n            return;\n        } else {\n            const bindedHandler = (event: Event): boolean => this.clickHandler(event);\n            elm.addEventListener('click', bindedHandler);\n            elm.addEventListener('touchstart', bindedHandler);\n            elm.addEventListener('touchend', bindedHandler);\n        }\n    }\n\n    protected handleFiles(files: File[]): void {\n        const selection: FileSelection = {\n            valid: [],\n            invalid: [],\n        };\n\n        forkJoin(\n            files.map(file =>\n                this.validate(file).pipe(\n                    tap(error => {\n                        if (error) {\n                            selection.invalid.push({\n                                file: file,\n                                error: error,\n                            });\n                        } else {\n                            selection.valid.push(file);\n                        }\n                    }),\n                ),\n            ),\n        ).subscribe(() => {\n            if (selection.valid.length) {\n                this.fileChange.emit(selection.valid[0]);\n            }\n\n            if (selection.valid.length || selection.invalid.length) {\n                this.filesChange.emit(selection);\n\n                if (this.broadcast) {\n                    this.naturalFileService.filesChanged.next(selection);\n                }\n            }\n\n            this.getFileElement().value = '';\n        });\n    }\n\n    /**\n     * Called when input has files\n     */\n    private changeFn(event: Event): void {\n        if (!(event.target instanceof HTMLInputElement)) {\n            return;\n        }\n\n        const fileList = event.target.files;\n\n        if (!fileList) {\n            return;\n        }\n\n        stopEvent(event);\n        this.handleFiles(fileListToArray(fileList));\n    }\n\n    private clickHandler(event: Event): boolean {\n        const elm = this.element.nativeElement;\n        if (elm.getAttribute('disabled') || this.fileSelectionDisabled) {\n            return false;\n        }\n\n        // prevent the click if it is a swipe\n        if (detectSwipe(event)) {\n            return true;\n        }\n\n        const fileElm = this.getFileElement();\n        fileElm.click();\n        this.beforeSelect();\n\n        return false;\n    }\n\n    private beforeSelect(): void {\n        if (!this.fileElement) {\n            return;\n        }\n\n        // if no files in array, be sure browser doesnt prevent reselect of same file (see github issue 27)\n        this.fileElement.value = '';\n    }\n\n    @HostListener('change', ['$event'])\n    public onChange(event: Event): void {\n        const fileList = this.getFileElement().files;\n        const files: File[] = fileList ? fileListToArray(fileList) : eventToFiles(event);\n\n        if (!files.length) {\n            return;\n        }\n\n        stopEvent(event);\n        this.handleFiles(files);\n    }\n\n    private validate(file: File): Observable<string | null> {\n        return forkJoin<Record<string, ObservableInput<boolean>>>({\n            accept: of(acceptType(this.accept, file.type, file.name)),\n            fileSize: of(!(this.maxSize && file.size > this.maxSize)),\n            directory: isDirectory(file),\n        }).pipe(\n            map(result => {\n                for (const [key, value] of Object.entries(result)) {\n                    if (!value) {\n                        return key;\n                    }\n                }\n\n                return null;\n            }),\n        );\n    }\n}\n"]}
|
|
@@ -1,172 +0,0 @@
|
|
|
1
|
-
import { Component, computed, EventEmitter, inject, input, Input, Output, } from '@angular/core';
|
|
2
|
-
import { of, Subject, tap } from 'rxjs';
|
|
3
|
-
import { NaturalFileService } from '../file.service';
|
|
4
|
-
import { CommonModule, DOCUMENT } from '@angular/common';
|
|
5
|
-
import { NaturalAlertService } from '../../alert/alert.service';
|
|
6
|
-
import { NaturalCapitalizePipe } from '../../common/pipes/capitalize.pipe';
|
|
7
|
-
import { NaturalIconDirective } from '../../icon/icon.directive';
|
|
8
|
-
import { MatIconModule } from '@angular/material/icon';
|
|
9
|
-
import { MatRippleModule } from '@angular/material/core';
|
|
10
|
-
import { NaturalFileDropDirective } from '../file-drop.directive';
|
|
11
|
-
import { NaturalBackgroundDensityDirective } from '../../common/directives/background-density.directive';
|
|
12
|
-
import * as i0 from "@angular/core";
|
|
13
|
-
import * as i1 from "@angular/material/core";
|
|
14
|
-
import * as i2 from "@angular/common";
|
|
15
|
-
import * as i3 from "@angular/material/icon";
|
|
16
|
-
// @dynamic
|
|
17
|
-
export class NaturalFileComponent {
|
|
18
|
-
naturalFileService = inject(NaturalFileService);
|
|
19
|
-
alertService = inject(NaturalAlertService);
|
|
20
|
-
document = inject(DOCUMENT);
|
|
21
|
-
height = input(250);
|
|
22
|
-
iconHeight = computed(() => Math.min(this.height() * 0.66, 120));
|
|
23
|
-
fontSize = computed(() => Math.min(this.height() * 0.3, 36));
|
|
24
|
-
action = null;
|
|
25
|
-
backgroundSize = 'contain';
|
|
26
|
-
/**
|
|
27
|
-
* Comma-separated list of unique file type specifiers. Like the native element,
|
|
28
|
-
* it can be a mix of mime-type and file extensions.
|
|
29
|
-
*
|
|
30
|
-
* See https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/file#accept
|
|
31
|
-
*/
|
|
32
|
-
accept = 'image/avif,image/bmp,image/gif,image/heic,image/heif,image/jpeg,image/pjpeg,image/png,image/svg+xml,image/svg,image/webp';
|
|
33
|
-
/**
|
|
34
|
-
* If given, it will be called when a new file is selected. The callback should typically upload the file
|
|
35
|
-
* to the server and link the newly uploaded file to the existing related object.
|
|
36
|
-
*
|
|
37
|
-
* The callback **must** be able to run even if the calling component has been destroyed. That means in most
|
|
38
|
-
* cases you **must** `bind()` the callback explicitly, like so:
|
|
39
|
-
*
|
|
40
|
-
* ```html
|
|
41
|
-
* <natural-file [uploader]="myCallback.bind(this)" />
|
|
42
|
-
* ```
|
|
43
|
-
*
|
|
44
|
-
* Also, you probably **should** set a `[formCtrl]` so that the form is updated automatically, instead of doing
|
|
45
|
-
* it manually within the callback.
|
|
46
|
-
*/
|
|
47
|
-
uploader;
|
|
48
|
-
model = null;
|
|
49
|
-
/**
|
|
50
|
-
* If provided, its value will get updated when the model changes.
|
|
51
|
-
* But its value is never read, so if you want to set a value use `[model]` instead.
|
|
52
|
-
*/
|
|
53
|
-
formCtrl = null;
|
|
54
|
-
/**
|
|
55
|
-
* This **must not** be used to mutate the server, because it is very likely it will never be called if the
|
|
56
|
-
* human navigates away from the page before the upload is finished. Instead, you should use `[uploader]`.
|
|
57
|
-
*/
|
|
58
|
-
modelChange = new EventEmitter();
|
|
59
|
-
imagePreview = '';
|
|
60
|
-
filePreview = null;
|
|
61
|
-
ngOnChanges(changes) {
|
|
62
|
-
if (changes.model && changes.model.previousValue !== changes.model.currentValue) {
|
|
63
|
-
this.updateImage();
|
|
64
|
-
}
|
|
65
|
-
}
|
|
66
|
-
ngOnInit() {
|
|
67
|
-
this.updateImage();
|
|
68
|
-
}
|
|
69
|
-
upload(file) {
|
|
70
|
-
this.model = { file: file };
|
|
71
|
-
this.updateImage();
|
|
72
|
-
if (this.formCtrl) {
|
|
73
|
-
this.formCtrl.setValue(this.model);
|
|
74
|
-
}
|
|
75
|
-
const observable = this.uploader?.(file).pipe(tap(() => this.alertService.info($localize `Mis à jour`))) ?? of(this.model);
|
|
76
|
-
observable.subscribe(result => {
|
|
77
|
-
this.model = result;
|
|
78
|
-
if (this.formCtrl) {
|
|
79
|
-
this.formCtrl.setValue(this.model);
|
|
80
|
-
}
|
|
81
|
-
this.modelChange.emit(this.model);
|
|
82
|
-
});
|
|
83
|
-
}
|
|
84
|
-
getDownloadLink() {
|
|
85
|
-
if (this.action !== 'download') {
|
|
86
|
-
return null;
|
|
87
|
-
}
|
|
88
|
-
return this.naturalFileService.getDownloadLink(this.model);
|
|
89
|
-
}
|
|
90
|
-
updateImage() {
|
|
91
|
-
this.imagePreview = '';
|
|
92
|
-
this.filePreview = null;
|
|
93
|
-
if (!this.model) {
|
|
94
|
-
return;
|
|
95
|
-
}
|
|
96
|
-
if (this.model.file?.type.startsWith('image/')) {
|
|
97
|
-
// Model from upload (before saving)
|
|
98
|
-
this.getBase64(this.model.file).subscribe(result => {
|
|
99
|
-
if (this.model?.file?.type) {
|
|
100
|
-
const content = 'url(data:' + this.model?.file?.type + ';base64,' + result + ')';
|
|
101
|
-
this.imagePreview = content;
|
|
102
|
-
}
|
|
103
|
-
});
|
|
104
|
-
}
|
|
105
|
-
else if (this.model.file) {
|
|
106
|
-
this.filePreview = this.model.file.type.split('/')[1];
|
|
107
|
-
}
|
|
108
|
-
else if (this.model.__typename === 'Image' && this.model.id) {
|
|
109
|
-
// Model image with id, use specific API to render image by size
|
|
110
|
-
const window = this.document.defaultView;
|
|
111
|
-
if (!window) {
|
|
112
|
-
throw new Error('Could not show image preview because `window` is undefined');
|
|
113
|
-
}
|
|
114
|
-
const loc = window.location;
|
|
115
|
-
const height = this.height() ? '/' + this.height() : '';
|
|
116
|
-
// create image url without port to stay compatible with dev mode
|
|
117
|
-
const image = loc.protocol + '//' + loc.hostname + '/api/image/' + this.model.id + height;
|
|
118
|
-
this.imagePreview = image;
|
|
119
|
-
}
|
|
120
|
-
else if (this.model?.mime && ['File', 'AccountingDocument'].includes(this.model.__typename || '')) {
|
|
121
|
-
this.filePreview = this.model.mime.split('/')[1];
|
|
122
|
-
}
|
|
123
|
-
else if (this.model.src) {
|
|
124
|
-
// external url
|
|
125
|
-
this.imagePreview = this.model.src;
|
|
126
|
-
}
|
|
127
|
-
}
|
|
128
|
-
getBase64(file) {
|
|
129
|
-
if (!file) {
|
|
130
|
-
return of('');
|
|
131
|
-
}
|
|
132
|
-
const subject = new Subject();
|
|
133
|
-
const reader = new FileReader();
|
|
134
|
-
reader.addEventListener('load', (ev) => {
|
|
135
|
-
subject.next(btoa(ev.target.result));
|
|
136
|
-
subject.complete();
|
|
137
|
-
});
|
|
138
|
-
reader.readAsBinaryString(file);
|
|
139
|
-
return subject.asObservable();
|
|
140
|
-
}
|
|
141
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalFileComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
142
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.3", type: NaturalFileComponent, isStandalone: true, selector: "natural-file", inputs: { height: { classPropertyName: "height", publicName: "height", isSignal: true, isRequired: false, transformFunction: null }, action: { classPropertyName: "action", publicName: "action", isSignal: false, isRequired: false, transformFunction: null }, backgroundSize: { classPropertyName: "backgroundSize", publicName: "backgroundSize", isSignal: false, isRequired: false, transformFunction: null }, accept: { classPropertyName: "accept", publicName: "accept", isSignal: false, isRequired: false, transformFunction: null }, uploader: { classPropertyName: "uploader", publicName: "uploader", isSignal: false, isRequired: false, transformFunction: null }, model: { classPropertyName: "model", publicName: "model", isSignal: false, isRequired: false, transformFunction: null }, formCtrl: { classPropertyName: "formCtrl", publicName: "formCtrl", isSignal: false, isRequired: false, transformFunction: null } }, outputs: { modelChange: "modelChange" }, host: { properties: { "style.height.px": "height()" } }, usesOnChanges: true, ngImport: i0, template: "<a\n (fileChange)=\"upload($event)\"\n naturalFileDrop\n [selectable]=\"true\"\n [accept]=\"accept\"\n [attr.href]=\"getDownloadLink()\"\n [class.has-action]=\"!!action\"\n [class.suggest-upload]=\"!model && action === 'upload'\"\n [fileSelectionDisabled]=\"action !== 'upload'\"\n [matRippleDisabled]=\"!action\"\n [naturalBackgroundDensity]=\"imagePreview\"\n [style.backgroundSize]=\"backgroundSize\"\n matRipple\n target=\"_blank\"\n>\n @if (filePreview) {\n <div class=\"file-preview\" [style.font-size.px]=\"fontSize()\">\n <mat-icon [size]=\"iconHeight()\" naturalIcon=\"attachment\" />\n {{ filePreview | uppercase }}\n </div>\n }\n\n <div class=\"action-overlay\" [style.font-size.px]=\"fontSize()\">\n @if (action === 'upload') {\n <mat-icon [size]=\"iconHeight()\" naturalIcon=\"cloud_upload\" />\n }\n @if (action === 'download') {\n <mat-icon [size]=\"iconHeight()\" naturalIcon=\"get_app\" />\n }\n {{ action | capitalize }}\n </div>\n</a>\n", styles: [":host{display:flex;position:relative;flex-direction:row;overflow:hidden}:host>a{position:relative;flex:1;background-position:center;background-repeat:no-repeat}:host>a.has-action{cursor:pointer}:host>a.has-action.suggest-upload .action-overlay{opacity:.66}:host>a.has-action:hover .action-overlay,:host>a.has-action.natural-file-over .action-overlay{opacity:1}:host .action-overlay,:host .file-preview{display:flex;position:absolute;inset:0;flex-direction:column;justify-content:center;align-items:center;font-size:36px;line-height:1.3em;text-align:center}:host .action-overlay{opacity:0}:host .action-overlay>div{display:flex;position:absolute;inset:0;justify-content:center;align-items:center;opacity:0}\n"], dependencies: [{ kind: "directive", type: NaturalFileDropDirective, selector: ":not([naturalFileSelect])[naturalFileDrop]", outputs: ["fileOver"] }, { kind: "ngmodule", type: MatRippleModule }, { kind: "directive", type: i1.MatRipple, selector: "[mat-ripple], [matRipple]", inputs: ["matRippleColor", "matRippleUnbounded", "matRippleCentered", "matRippleRadius", "matRippleAnimation", "matRippleDisabled", "matRippleTrigger"], exportAs: ["matRipple"] }, { kind: "ngmodule", type: CommonModule }, { kind: "pipe", type: i2.UpperCasePipe, name: "uppercase" }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i3.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: NaturalIconDirective, selector: "mat-icon[naturalIcon]", inputs: ["naturalIcon", "size"] }, { kind: "pipe", type: NaturalCapitalizePipe, name: "capitalize" }, { kind: "directive", type: NaturalBackgroundDensityDirective, selector: "[naturalBackgroundDensity]", inputs: ["naturalBackgroundDensity"] }] });
|
|
143
|
-
}
|
|
144
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalFileComponent, decorators: [{
|
|
145
|
-
type: Component,
|
|
146
|
-
args: [{ selector: 'natural-file', standalone: true, imports: [
|
|
147
|
-
NaturalFileDropDirective,
|
|
148
|
-
MatRippleModule,
|
|
149
|
-
CommonModule,
|
|
150
|
-
MatIconModule,
|
|
151
|
-
NaturalIconDirective,
|
|
152
|
-
NaturalCapitalizePipe,
|
|
153
|
-
NaturalBackgroundDensityDirective,
|
|
154
|
-
], host: {
|
|
155
|
-
'[style.height.px]': 'height()',
|
|
156
|
-
}, template: "<a\n (fileChange)=\"upload($event)\"\n naturalFileDrop\n [selectable]=\"true\"\n [accept]=\"accept\"\n [attr.href]=\"getDownloadLink()\"\n [class.has-action]=\"!!action\"\n [class.suggest-upload]=\"!model && action === 'upload'\"\n [fileSelectionDisabled]=\"action !== 'upload'\"\n [matRippleDisabled]=\"!action\"\n [naturalBackgroundDensity]=\"imagePreview\"\n [style.backgroundSize]=\"backgroundSize\"\n matRipple\n target=\"_blank\"\n>\n @if (filePreview) {\n <div class=\"file-preview\" [style.font-size.px]=\"fontSize()\">\n <mat-icon [size]=\"iconHeight()\" naturalIcon=\"attachment\" />\n {{ filePreview | uppercase }}\n </div>\n }\n\n <div class=\"action-overlay\" [style.font-size.px]=\"fontSize()\">\n @if (action === 'upload') {\n <mat-icon [size]=\"iconHeight()\" naturalIcon=\"cloud_upload\" />\n }\n @if (action === 'download') {\n <mat-icon [size]=\"iconHeight()\" naturalIcon=\"get_app\" />\n }\n {{ action | capitalize }}\n </div>\n</a>\n", styles: [":host{display:flex;position:relative;flex-direction:row;overflow:hidden}:host>a{position:relative;flex:1;background-position:center;background-repeat:no-repeat}:host>a.has-action{cursor:pointer}:host>a.has-action.suggest-upload .action-overlay{opacity:.66}:host>a.has-action:hover .action-overlay,:host>a.has-action.natural-file-over .action-overlay{opacity:1}:host .action-overlay,:host .file-preview{display:flex;position:absolute;inset:0;flex-direction:column;justify-content:center;align-items:center;font-size:36px;line-height:1.3em;text-align:center}:host .action-overlay{opacity:0}:host .action-overlay>div{display:flex;position:absolute;inset:0;justify-content:center;align-items:center;opacity:0}\n"] }]
|
|
157
|
-
}], propDecorators: { action: [{
|
|
158
|
-
type: Input
|
|
159
|
-
}], backgroundSize: [{
|
|
160
|
-
type: Input
|
|
161
|
-
}], accept: [{
|
|
162
|
-
type: Input
|
|
163
|
-
}], uploader: [{
|
|
164
|
-
type: Input
|
|
165
|
-
}], model: [{
|
|
166
|
-
type: Input
|
|
167
|
-
}], formCtrl: [{
|
|
168
|
-
type: Input
|
|
169
|
-
}], modelChange: [{
|
|
170
|
-
type: Output
|
|
171
|
-
}] } });
|
|
172
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file.component.js","sourceRoot":"","sources":["../../../../../../../projects/natural/src/lib/modules/file/component/file.component.ts","../../../../../../../projects/natural/src/lib/modules/file/component/file.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    Component,\n    computed,\n    EventEmitter,\n    inject,\n    input,\n    Input,\n    OnChanges,\n    OnInit,\n    Output,\n    SimpleChanges,\n} from '@angular/core';\nimport {AbstractControl} from '@angular/forms';\nimport {Observable, of, Subject, tap} from 'rxjs';\nimport {NaturalFileService} from '../file.service';\nimport {CommonModule, DOCUMENT} from '@angular/common';\nimport {FileModel} from '../types';\nimport {NaturalAlertService} from '../../alert/alert.service';\nimport {NaturalCapitalizePipe} from '../../common/pipes/capitalize.pipe';\nimport {NaturalIconDirective} from '../../icon/icon.directive';\nimport {MatIconModule} from '@angular/material/icon';\nimport {MatRippleModule} from '@angular/material/core';\nimport {NaturalFileDropDirective} from '../file-drop.directive';\nimport {NaturalBackgroundDensityDirective} from '../../common/directives/background-density.directive';\n\n// @dynamic\n@Component({\n    selector: 'natural-file',\n    templateUrl: './file.component.html',\n    styleUrl: './file.component.scss',\n    standalone: true,\n    imports: [\n        NaturalFileDropDirective,\n        MatRippleModule,\n        CommonModule,\n        MatIconModule,\n        NaturalIconDirective,\n        NaturalCapitalizePipe,\n        NaturalBackgroundDensityDirective,\n    ],\n    // eslint-disable-next-line @angular-eslint/no-host-metadata-property\n    host: {\n        '[style.height.px]': 'height()',\n    },\n})\nexport class NaturalFileComponent implements OnInit, OnChanges {\n    private readonly naturalFileService = inject(NaturalFileService);\n    private readonly alertService = inject(NaturalAlertService);\n    private readonly document = inject(DOCUMENT);\n\n    public height = input(250);\n    public readonly iconHeight = computed(() => Math.min(this.height() * 0.66, 120));\n    public readonly fontSize = computed(() => Math.min(this.height() * 0.3, 36));\n\n    @Input() public action: 'upload' | 'download' | null = null;\n\n    @Input() public backgroundSize = 'contain';\n\n    /**\n     * Comma-separated list of unique file type specifiers. Like the native element,\n     * it can be a mix of mime-type and file extensions.\n     *\n     * See https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/file#accept\n     */\n    @Input() public accept =\n        'image/avif,image/bmp,image/gif,image/heic,image/heif,image/jpeg,image/pjpeg,image/png,image/svg+xml,image/svg,image/webp';\n\n    /**\n     * If given, it will be called when a new file is selected. The callback should typically upload the file\n     * to the server and link the newly uploaded file to the existing related object.\n     *\n     * The callback **must** be able to run even if the calling component has been destroyed. That means in most\n     * cases you **must** `bind()` the callback explicitly, like so:\n     *\n     * ```html\n     * <natural-file [uploader]=\"myCallback.bind(this)\" />\n     * ```\n     *\n     * Also, you probably **should** set a `[formCtrl]` so that the form is updated automatically, instead of doing\n     * it manually within the callback.\n     */\n    @Input() public uploader?: (file: File) => Observable<FileModel>;\n\n    @Input() public model: FileModel | null = null;\n\n    /**\n     * If provided, its value will get updated when the model changes.\n     * But its value is never read, so if you want to set a value use `[model]` instead.\n     */\n    @Input() public formCtrl: AbstractControl | null | undefined = null;\n\n    /**\n     * This **must not** be used to mutate the server, because it is very likely it will never be called if the\n     * human navigates away from the page before the upload is finished. Instead, you should use `[uploader]`.\n     */\n    @Output() public readonly modelChange = new EventEmitter<FileModel>();\n\n    public imagePreview = '';\n    public filePreview: string | null = null;\n\n    public ngOnChanges(changes: SimpleChanges): void {\n        if (changes.model && changes.model.previousValue !== changes.model.currentValue) {\n            this.updateImage();\n        }\n    }\n\n    public ngOnInit(): void {\n        this.updateImage();\n    }\n\n    public upload(file: File): void {\n        this.model = {file: file};\n        this.updateImage();\n\n        if (this.formCtrl) {\n            this.formCtrl.setValue(this.model);\n        }\n\n        const observable =\n            this.uploader?.(file).pipe(tap(() => this.alertService.info($localize`Mis à jour`))) ?? of(this.model);\n\n        observable.subscribe(result => {\n            this.model = result;\n            if (this.formCtrl) {\n                this.formCtrl.setValue(this.model);\n            }\n\n            this.modelChange.emit(this.model);\n        });\n    }\n\n    public getDownloadLink(): null | string {\n        if (this.action !== 'download') {\n            return null;\n        }\n\n        return this.naturalFileService.getDownloadLink(this.model);\n    }\n\n    private updateImage(): void {\n        this.imagePreview = '';\n        this.filePreview = null;\n        if (!this.model) {\n            return;\n        }\n\n        if (this.model.file?.type.startsWith('image/')) {\n            // Model from upload (before saving)\n            this.getBase64(this.model.file).subscribe(result => {\n                if (this.model?.file?.type) {\n                    const content = 'url(data:' + this.model?.file?.type + ';base64,' + result + ')';\n                    this.imagePreview = content;\n                }\n            });\n        } else if (this.model.file) {\n            this.filePreview = this.model.file.type.split('/')[1];\n        } else if (this.model.__typename === 'Image' && this.model.id) {\n            // Model image with id, use specific API to render image by size\n            const window = this.document.defaultView;\n            if (!window) {\n                throw new Error('Could not show image preview because `window` is undefined');\n            }\n\n            const loc = window.location;\n            const height = this.height() ? '/' + this.height() : '';\n\n            // create image url without port to stay compatible with dev mode\n            const image = loc.protocol + '//' + loc.hostname + '/api/image/' + this.model.id + height;\n            this.imagePreview = image;\n        } else if (this.model?.mime && ['File', 'AccountingDocument'].includes(this.model.__typename || '')) {\n            this.filePreview = this.model.mime.split('/')[1];\n        } else if (this.model.src) {\n            // external url\n            this.imagePreview = this.model.src;\n        }\n    }\n\n    private getBase64(file: File | null): Observable<string> {\n        if (!file) {\n            return of('');\n        }\n\n        const subject = new Subject<string>();\n\n        const reader = new FileReader();\n        reader.addEventListener('load', (ev: any) => {\n            subject.next(btoa(ev.target.result));\n            subject.complete();\n        });\n        reader.readAsBinaryString(file);\n\n        return subject.asObservable();\n    }\n}\n","<a\n    (fileChange)=\"upload($event)\"\n    naturalFileDrop\n    [selectable]=\"true\"\n    [accept]=\"accept\"\n    [attr.href]=\"getDownloadLink()\"\n    [class.has-action]=\"!!action\"\n    [class.suggest-upload]=\"!model && action === 'upload'\"\n    [fileSelectionDisabled]=\"action !== 'upload'\"\n    [matRippleDisabled]=\"!action\"\n    [naturalBackgroundDensity]=\"imagePreview\"\n    [style.backgroundSize]=\"backgroundSize\"\n    matRipple\n    target=\"_blank\"\n>\n    @if (filePreview) {\n        <div class=\"file-preview\" [style.font-size.px]=\"fontSize()\">\n            <mat-icon [size]=\"iconHeight()\" naturalIcon=\"attachment\" />\n            {{ filePreview | uppercase }}\n        </div>\n    }\n\n    <div class=\"action-overlay\" [style.font-size.px]=\"fontSize()\">\n        @if (action === 'upload') {\n            <mat-icon [size]=\"iconHeight()\" naturalIcon=\"cloud_upload\" />\n        }\n        @if (action === 'download') {\n            <mat-icon [size]=\"iconHeight()\" naturalIcon=\"get_app\" />\n        }\n        {{ action | capitalize }}\n    </div>\n</a>\n"]}
|
|
@@ -1,111 +0,0 @@
|
|
|
1
|
-
import { DestroyRef, Directive, EventEmitter, HostBinding, HostListener, inject, Output } from '@angular/core';
|
|
2
|
-
import { NaturalAbstractFile } from './abstract-file';
|
|
3
|
-
import { eventToFiles, stopEvent } from './utils';
|
|
4
|
-
import { asyncScheduler, Subject } from 'rxjs';
|
|
5
|
-
import { throttleTime } from 'rxjs/operators';
|
|
6
|
-
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
|
7
|
-
import * as i0 from "@angular/core";
|
|
8
|
-
/**
|
|
9
|
-
* This directive has all options to select files, and adds support for drag'n'drop.
|
|
10
|
-
*
|
|
11
|
-
* It will add the CSS class `natural-file-over` on the component when a file is
|
|
12
|
-
* dragged over. It is up to the component to have some specific styling by using
|
|
13
|
-
* this class.
|
|
14
|
-
*
|
|
15
|
-
* In most cases you probably also want click-to-select, so you should use:
|
|
16
|
-
*
|
|
17
|
-
* ```html
|
|
18
|
-
* <div naturalFileDrop [selectable]="true"></div>
|
|
19
|
-
* ```
|
|
20
|
-
*/
|
|
21
|
-
export class NaturalFileDropDirective extends NaturalAbstractFile {
|
|
22
|
-
destroyRef = inject(DestroyRef);
|
|
23
|
-
fileOverClass = false;
|
|
24
|
-
/**
|
|
25
|
-
* Emits whenever files are being dragged over
|
|
26
|
-
*/
|
|
27
|
-
fileOver = new EventEmitter();
|
|
28
|
-
rawFileOver = new Subject();
|
|
29
|
-
ngOnInit() {
|
|
30
|
-
super.ngOnInit();
|
|
31
|
-
// Automatically change the class, but not too often to avoid visual
|
|
32
|
-
// flickering in Chrome when hovering across child HTML element of our host.
|
|
33
|
-
// It's not absolutely perfect and if dragging slowly and precisely we can
|
|
34
|
-
// still see flicker, but it should be better for most normal usages.
|
|
35
|
-
this.rawFileOver
|
|
36
|
-
.pipe(takeUntilDestroyed(this.destroyRef), throttleTime(200, asyncScheduler, {
|
|
37
|
-
leading: true,
|
|
38
|
-
trailing: true,
|
|
39
|
-
}))
|
|
40
|
-
.subscribe(fileOver => {
|
|
41
|
-
this.fileOver.emit(fileOver);
|
|
42
|
-
this.fileOverClass = fileOver;
|
|
43
|
-
});
|
|
44
|
-
}
|
|
45
|
-
onDrop(event) {
|
|
46
|
-
if (this.fileSelectionDisabled) {
|
|
47
|
-
return;
|
|
48
|
-
}
|
|
49
|
-
this.closeDrags();
|
|
50
|
-
const files = eventToFiles(event);
|
|
51
|
-
if (!files.length) {
|
|
52
|
-
return;
|
|
53
|
-
}
|
|
54
|
-
stopEvent(event);
|
|
55
|
-
this.handleFiles(files);
|
|
56
|
-
}
|
|
57
|
-
onDragOver(event) {
|
|
58
|
-
if (!this.hasObservers()) {
|
|
59
|
-
return;
|
|
60
|
-
}
|
|
61
|
-
if (this.fileSelectionDisabled) {
|
|
62
|
-
return;
|
|
63
|
-
}
|
|
64
|
-
stopEvent(event);
|
|
65
|
-
// Change cursor
|
|
66
|
-
const transfer = event.dataTransfer;
|
|
67
|
-
if (transfer) {
|
|
68
|
-
transfer.dropEffect = 'copy';
|
|
69
|
-
}
|
|
70
|
-
this.rawFileOver.next(true);
|
|
71
|
-
}
|
|
72
|
-
closeDrags() {
|
|
73
|
-
this.rawFileOver.next(false);
|
|
74
|
-
}
|
|
75
|
-
onDragLeave(event) {
|
|
76
|
-
if (this.fileSelectionDisabled) {
|
|
77
|
-
return;
|
|
78
|
-
}
|
|
79
|
-
stopEvent(event);
|
|
80
|
-
this.closeDrags();
|
|
81
|
-
}
|
|
82
|
-
hasObservers() {
|
|
83
|
-
return (this.fileChange.observed ||
|
|
84
|
-
this.filesChange.observed ||
|
|
85
|
-
(this.broadcast && this.naturalFileService.filesChanged.observed));
|
|
86
|
-
}
|
|
87
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalFileDropDirective, deps: null, target: i0.ɵɵFactoryTarget.Directive });
|
|
88
|
-
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.3", type: NaturalFileDropDirective, isStandalone: true, selector: ":not([naturalFileSelect])[naturalFileDrop]", outputs: { fileOver: "fileOver" }, host: { listeners: { "drop": "onDrop($event)", "dragover": "onDragOver($event)", "dragleave": "onDragLeave($event)" }, properties: { "class.natural-file-over": "this.fileOverClass" } }, usesInheritance: true, ngImport: i0 });
|
|
89
|
-
}
|
|
90
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalFileDropDirective, decorators: [{
|
|
91
|
-
type: Directive,
|
|
92
|
-
args: [{
|
|
93
|
-
selector: ':not([naturalFileSelect])[naturalFileDrop]',
|
|
94
|
-
standalone: true,
|
|
95
|
-
}]
|
|
96
|
-
}], propDecorators: { fileOverClass: [{
|
|
97
|
-
type: HostBinding,
|
|
98
|
-
args: ['class.natural-file-over']
|
|
99
|
-
}], fileOver: [{
|
|
100
|
-
type: Output
|
|
101
|
-
}], onDrop: [{
|
|
102
|
-
type: HostListener,
|
|
103
|
-
args: ['drop', ['$event']]
|
|
104
|
-
}], onDragOver: [{
|
|
105
|
-
type: HostListener,
|
|
106
|
-
args: ['dragover', ['$event']]
|
|
107
|
-
}], onDragLeave: [{
|
|
108
|
-
type: HostListener,
|
|
109
|
-
args: ['dragleave', ['$event']]
|
|
110
|
-
}] } });
|
|
111
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-drop.directive.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/file/file-drop.directive.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,UAAU,EAAE,SAAS,EAAE,YAAY,EAAE,WAAW,EAAE,YAAY,EAAE,MAAM,EAAU,MAAM,EAAC,MAAM,eAAe,CAAC;AACrH,OAAO,EAAC,mBAAmB,EAAC,MAAM,iBAAiB,CAAC;AACpD,OAAO,EAAC,YAAY,EAAE,SAAS,EAAC,MAAM,SAAS,CAAC;AAChD,OAAO,EAAC,cAAc,EAAE,OAAO,EAAC,MAAM,MAAM,CAAC;AAC7C,OAAO,EAAC,YAAY,EAAC,MAAM,gBAAgB,CAAC;AAC5C,OAAO,EAAC,kBAAkB,EAAC,MAAM,4BAA4B,CAAC;;AAE9D;;;;;;;;;;;;GAYG;AAKH,MAAM,OAAO,wBAAyB,SAAQ,mBAAmB;IAC5C,UAAU,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC;IACF,aAAa,GAAG,KAAK,CAAC;IAErE;;OAEG;IACuB,QAAQ,GAAG,IAAI,YAAY,EAAW,CAAC;IAEhD,WAAW,GAAG,IAAI,OAAO,EAAW,CAAC;IAEtC,QAAQ;QACpB,KAAK,CAAC,QAAQ,EAAE,CAAC;QAEjB,oEAAoE;QACpE,4EAA4E;QAC5E,0EAA0E;QAC1E,qEAAqE;QACrE,IAAI,CAAC,WAAW;aACX,IAAI,CACD,kBAAkB,CAAC,IAAI,CAAC,UAAU,CAAC,EACnC,YAAY,CAAC,GAAG,EAAE,cAAc,EAAE;YAC9B,OAAO,EAAE,IAAI;YACb,QAAQ,EAAE,IAAI;SACjB,CAAC,CACL;aACA,SAAS,CAAC,QAAQ,CAAC,EAAE;YAClB,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;YAC7B,IAAI,CAAC,aAAa,GAAG,QAAQ,CAAC;QAClC,CAAC,CAAC,CAAC;IACX,CAAC;IAGM,MAAM,CAAC,KAAgB;QAC1B,IAAI,IAAI,CAAC,qBAAqB,EAAE,CAAC;YAC7B,OAAO;QACX,CAAC;QAED,IAAI,CAAC,UAAU,EAAE,CAAC;QAElB,MAAM,KAAK,GAAG,YAAY,CAAC,KAAK,CAAC,CAAC;QAClC,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC;YAChB,OAAO;QACX,CAAC;QAED,SAAS,CAAC,KAAK,CAAC,CAAC;QACjB,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;IAC5B,CAAC;IAGM,UAAU,CAAC,KAAgB;QAC9B,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE,EAAE,CAAC;YACvB,OAAO;QACX,CAAC;QAED,IAAI,IAAI,CAAC,qBAAqB,EAAE,CAAC;YAC7B,OAAO;QACX,CAAC;QAED,SAAS,CAAC,KAAK,CAAC,CAAC;QAEjB,gBAAgB;QAChB,MAAM,QAAQ,GAAG,KAAK,CAAC,YAAY,CAAC;QACpC,IAAI,QAAQ,EAAE,CAAC;YACX,QAAQ,CAAC,UAAU,GAAG,MAAM,CAAC;QACjC,CAAC;QAED,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAChC,CAAC;IAEO,UAAU;QACd,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACjC,CAAC;IAGM,WAAW,CAAC,KAAgB;QAC/B,IAAI,IAAI,CAAC,qBAAqB,EAAE,CAAC;YAC7B,OAAO;QACX,CAAC;QAED,SAAS,CAAC,KAAK,CAAC,CAAC;QACjB,IAAI,CAAC,UAAU,EAAE,CAAC;IACtB,CAAC;IAEO,YAAY;QAChB,OAAO,CACH,IAAI,CAAC,UAAU,CAAC,QAAQ;YACxB,IAAI,CAAC,WAAW,CAAC,QAAQ;YACzB,CAAC,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,kBAAkB,CAAC,YAAY,CAAC,QAAQ,CAAC,CACpE,CAAC;IACN,CAAC;uGA1FQ,wBAAwB;2FAAxB,wBAAwB;;2FAAxB,wBAAwB;kBAJpC,SAAS;mBAAC;oBACP,QAAQ,EAAE,4CAA4C;oBACtD,UAAU,EAAE,IAAI;iBACnB;8BAGkD,aAAa;sBAA3D,WAAW;uBAAC,yBAAyB;gBAKZ,QAAQ;sBAAjC,MAAM;gBA0BA,MAAM;sBADZ,YAAY;uBAAC,MAAM,EAAE,CAAC,QAAQ,CAAC;gBAkBzB,UAAU;sBADhB,YAAY;uBAAC,UAAU,EAAE,CAAC,QAAQ,CAAC;gBA0B7B,WAAW;sBADjB,YAAY;uBAAC,WAAW,EAAE,CAAC,QAAQ,CAAC","sourcesContent":["import {DestroyRef, Directive, EventEmitter, HostBinding, HostListener, inject, OnInit, Output} from '@angular/core';\nimport {NaturalAbstractFile} from './abstract-file';\nimport {eventToFiles, stopEvent} from './utils';\nimport {asyncScheduler, Subject} from 'rxjs';\nimport {throttleTime} from 'rxjs/operators';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\n\n/**\n * This directive has all options to select files, and adds support for drag'n'drop.\n *\n * It will add the CSS class `natural-file-over` on the component when a file is\n * dragged over. It is up to the component to have some specific styling by using\n * this class.\n *\n * In most cases you probably also want click-to-select, so you should use:\n *\n * ```html\n * <div naturalFileDrop [selectable]=\"true\"></div>\n * ```\n */\n@Directive({\n    selector: ':not([naturalFileSelect])[naturalFileDrop]',\n    standalone: true,\n})\nexport class NaturalFileDropDirective extends NaturalAbstractFile implements OnInit {\n    private readonly destroyRef = inject(DestroyRef);\n    @HostBinding('class.natural-file-over') public fileOverClass = false;\n\n    /**\n     * Emits whenever files are being dragged over\n     */\n    @Output() public readonly fileOver = new EventEmitter<boolean>();\n\n    private readonly rawFileOver = new Subject<boolean>();\n\n    public override ngOnInit(): void {\n        super.ngOnInit();\n\n        // Automatically change the class, but not too often to avoid visual\n        // flickering in Chrome when hovering across child HTML element of our host.\n        // It's not absolutely perfect and if dragging slowly and precisely we can\n        // still see flicker, but it should be better for most normal usages.\n        this.rawFileOver\n            .pipe(\n                takeUntilDestroyed(this.destroyRef),\n                throttleTime(200, asyncScheduler, {\n                    leading: true,\n                    trailing: true,\n                }),\n            )\n            .subscribe(fileOver => {\n                this.fileOver.emit(fileOver);\n                this.fileOverClass = fileOver;\n            });\n    }\n\n    @HostListener('drop', ['$event'])\n    public onDrop(event: DragEvent): void {\n        if (this.fileSelectionDisabled) {\n            return;\n        }\n\n        this.closeDrags();\n\n        const files = eventToFiles(event);\n        if (!files.length) {\n            return;\n        }\n\n        stopEvent(event);\n        this.handleFiles(files);\n    }\n\n    @HostListener('dragover', ['$event'])\n    public onDragOver(event: DragEvent): void {\n        if (!this.hasObservers()) {\n            return;\n        }\n\n        if (this.fileSelectionDisabled) {\n            return;\n        }\n\n        stopEvent(event);\n\n        // Change cursor\n        const transfer = event.dataTransfer;\n        if (transfer) {\n            transfer.dropEffect = 'copy';\n        }\n\n        this.rawFileOver.next(true);\n    }\n\n    private closeDrags(): void {\n        this.rawFileOver.next(false);\n    }\n\n    @HostListener('dragleave', ['$event'])\n    public onDragLeave(event: DragEvent): void {\n        if (this.fileSelectionDisabled) {\n            return;\n        }\n\n        stopEvent(event);\n        this.closeDrags();\n    }\n\n    private hasObservers(): boolean {\n        return (\n            this.fileChange.observed ||\n            this.filesChange.observed ||\n            (this.broadcast && this.naturalFileService.filesChanged.observed)\n        );\n    }\n}\n"]}
|
|
@@ -1,26 +0,0 @@
|
|
|
1
|
-
import { Directive, Input } from '@angular/core';
|
|
2
|
-
import { NaturalAbstractFile } from './abstract-file';
|
|
3
|
-
import * as i0 from "@angular/core";
|
|
4
|
-
/**
|
|
5
|
-
* This directive has all options to select files, except drag'n'drop.
|
|
6
|
-
*/
|
|
7
|
-
export class NaturalFileSelectDirective extends NaturalAbstractFile {
|
|
8
|
-
/**
|
|
9
|
-
* Whether the user can click on the element to select something
|
|
10
|
-
*
|
|
11
|
-
* Override parent to enable it by default
|
|
12
|
-
*/
|
|
13
|
-
selectable = true;
|
|
14
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalFileSelectDirective, deps: null, target: i0.ɵɵFactoryTarget.Directive });
|
|
15
|
-
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.3", type: NaturalFileSelectDirective, isStandalone: true, selector: ":not([naturalFileDrop])[naturalFileSelect]", inputs: { selectable: "selectable" }, usesInheritance: true, ngImport: i0 });
|
|
16
|
-
}
|
|
17
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalFileSelectDirective, decorators: [{
|
|
18
|
-
type: Directive,
|
|
19
|
-
args: [{
|
|
20
|
-
selector: ':not([naturalFileDrop])[naturalFileSelect]',
|
|
21
|
-
standalone: true,
|
|
22
|
-
}]
|
|
23
|
-
}], propDecorators: { selectable: [{
|
|
24
|
-
type: Input
|
|
25
|
-
}] } });
|
|
26
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmlsZS1zZWxlY3QuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmF0dXJhbC9zcmMvbGliL21vZHVsZXMvZmlsZS9maWxlLXNlbGVjdC5kaXJlY3RpdmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFDLFNBQVMsRUFBRSxLQUFLLEVBQUMsTUFBTSxlQUFlLENBQUM7QUFDL0MsT0FBTyxFQUFDLG1CQUFtQixFQUFDLE1BQU0saUJBQWlCLENBQUM7O0FBRXBEOztHQUVHO0FBS0gsTUFBTSxPQUFPLDBCQUEyQixTQUFRLG1CQUFtQjtJQUMvRDs7OztPQUlHO0lBQ3NCLFVBQVUsR0FBRyxJQUFJLENBQUM7dUdBTmxDLDBCQUEwQjsyRkFBMUIsMEJBQTBCOzsyRkFBMUIsMEJBQTBCO2tCQUp0QyxTQUFTO21CQUFDO29CQUNQLFFBQVEsRUFBRSw0Q0FBNEM7b0JBQ3RELFVBQVUsRUFBRSxJQUFJO2lCQUNuQjs4QkFPNEIsVUFBVTtzQkFBbEMsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7RGlyZWN0aXZlLCBJbnB1dH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge05hdHVyYWxBYnN0cmFjdEZpbGV9IGZyb20gJy4vYWJzdHJhY3QtZmlsZSc7XG5cbi8qKlxuICogVGhpcyBkaXJlY3RpdmUgaGFzIGFsbCBvcHRpb25zIHRvIHNlbGVjdCBmaWxlcywgZXhjZXB0IGRyYWcnbidkcm9wLlxuICovXG5ARGlyZWN0aXZlKHtcbiAgICBzZWxlY3RvcjogJzpub3QoW25hdHVyYWxGaWxlRHJvcF0pW25hdHVyYWxGaWxlU2VsZWN0XScsXG4gICAgc3RhbmRhbG9uZTogdHJ1ZSxcbn0pXG5leHBvcnQgY2xhc3MgTmF0dXJhbEZpbGVTZWxlY3REaXJlY3RpdmUgZXh0ZW5kcyBOYXR1cmFsQWJzdHJhY3RGaWxlIHtcbiAgICAvKipcbiAgICAgKiBXaGV0aGVyIHRoZSB1c2VyIGNhbiBjbGljayBvbiB0aGUgZWxlbWVudCB0byBzZWxlY3Qgc29tZXRoaW5nXG4gICAgICpcbiAgICAgKiBPdmVycmlkZSBwYXJlbnQgdG8gZW5hYmxlIGl0IGJ5IGRlZmF1bHRcbiAgICAgKi9cbiAgICBASW5wdXQoKSBwdWJsaWMgb3ZlcnJpZGUgc2VsZWN0YWJsZSA9IHRydWU7XG59XG4iXX0=
|