@dotglitch/ngx-common 1.0.21 → 1.0.22
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/assets/mat-icons.d.ts +5705 -0
- package/components/filemanager/file-grid/file-grid.component.d.ts +97 -0
- package/components/filemanager/filemanager.component.d.ts +206 -0
- package/components/filemanager/folder-rename/folder-rename.component.d.ts +18 -0
- package/components/filemanager/helpers.d.ts +5 -0
- package/components/filemanager/icon-resolver.d.ts +13 -0
- package/components/filemanager/textextensions.d.ts +3 -0
- package/components/filemanager/toolbar/breadcrumb/breadcrumb.component.d.ts +16 -0
- package/components/filemanager/toolbar/icon-button/icon-button.component.d.ts +11 -0
- package/components/filemanager/toolbar/toolbar.component.d.ts +30 -0
- package/components/filemanager/tree-view/tree-view.component.d.ts +21 -0
- package/components/tabulator/tabulator.component.d.ts +30 -0
- package/components/types.d.ts +16 -0
- package/components/vscode/ts-type-resolver/dependency-parser.d.ts +6 -0
- package/components/vscode/ts-type-resolver/dummy-source-cache.d.ts +7 -0
- package/components/vscode/ts-type-resolver/import-resolver.d.ts +28 -0
- package/components/vscode/ts-type-resolver/main.d.ts +22 -0
- package/components/vscode/ts-type-resolver/recursion-depth.d.ts +11 -0
- package/components/vscode/ts-type-resolver/types.d.ts +179 -0
- package/components/vscode/ts-type-resolver/unpkg-source-resolver.d.ts +5 -0
- package/components/vscode/ts-type-resolver/update-emitter.d.ts +2 -0
- package/components/vscode/vscode.component.d.ts +57 -0
- package/esm2020/assets/mat-icons.mjs +5705 -0
- package/esm2020/components/filemanager/file-grid/file-grid.component.mjs +672 -0
- package/esm2020/components/filemanager/filemanager.component.mjs +301 -0
- package/esm2020/components/filemanager/folder-rename/folder-rename.component.mjs +57 -0
- package/esm2020/components/filemanager/helpers.mjs +26 -0
- package/esm2020/components/filemanager/icon-resolver.mjs +155 -0
- package/esm2020/components/filemanager/textextensions.mjs +294 -0
- package/esm2020/components/filemanager/toolbar/breadcrumb/breadcrumb.component.mjs +26 -0
- package/esm2020/components/filemanager/toolbar/icon-button/icon-button.component.mjs +34 -0
- package/esm2020/components/filemanager/toolbar/toolbar.component.mjs +163 -0
- package/esm2020/components/filemanager/tree-view/tree-view.component.mjs +53 -0
- package/esm2020/components/lazy-loader/lazy-loader.module.mjs +3 -3
- package/esm2020/components/lazy-loader/lazy-loader.service.mjs +1 -1
- package/esm2020/components/menu/menu.component.mjs +1 -2
- package/esm2020/components/tabulator/tabulator.component.mjs +92 -0
- package/esm2020/components/types.mjs +3 -0
- package/esm2020/components/vscode/ts-type-resolver/dependency-parser.mjs +91 -0
- package/esm2020/components/vscode/ts-type-resolver/dummy-source-cache.mjs +15 -0
- package/esm2020/components/vscode/ts-type-resolver/import-resolver.mjs +311 -0
- package/esm2020/components/vscode/ts-type-resolver/main.mjs +112 -0
- package/esm2020/components/vscode/ts-type-resolver/recursion-depth.mjs +21 -0
- package/esm2020/components/vscode/ts-type-resolver/types.mjs +14 -0
- package/esm2020/components/vscode/ts-type-resolver/unpkg-source-resolver.mjs +21 -0
- package/esm2020/components/vscode/ts-type-resolver/update-emitter.mjs +37 -0
- package/esm2020/components/vscode/vscode.component.mjs +230 -0
- package/esm2020/directives/menu.directive.mjs +19 -18
- package/esm2020/public-api.mjs +6 -1
- package/fesm2015/dotglitch-ngx-common-folder-rename.component-d039534b.mjs +79 -0
- package/fesm2015/dotglitch-ngx-common-folder-rename.component-d039534b.mjs.map +1 -0
- package/fesm2015/dotglitch-ngx-common.mjs +8360 -27
- package/fesm2015/dotglitch-ngx-common.mjs.map +1 -1
- package/fesm2020/dotglitch-ngx-common-folder-rename.component-ba3ebd0a.mjs +78 -0
- package/fesm2020/dotglitch-ngx-common-folder-rename.component-ba3ebd0a.mjs.map +1 -0
- package/fesm2020/dotglitch-ngx-common.mjs +8329 -26
- package/fesm2020/dotglitch-ngx-common.mjs.map +1 -1
- package/package.json +6 -2
- package/public-api.d.ts +4 -0
|
@@ -0,0 +1,672 @@
|
|
|
1
|
+
import { Component, EventEmitter, Output, Input, ViewChild, TemplateRef, SecurityContext } from '@angular/core';
|
|
2
|
+
import { MatTabsModule } from '@angular/material/tabs';
|
|
3
|
+
import { MatCheckboxModule } from '@angular/material/checkbox';
|
|
4
|
+
import { MatInputModule } from '@angular/material/input';
|
|
5
|
+
import { DatePipe, NgForOf, NgIf } from '@angular/common';
|
|
6
|
+
import { ScrollingModule } from '@angular/cdk/scrolling';
|
|
7
|
+
import { NgScrollbarModule } from 'ngx-scrollbar';
|
|
8
|
+
import { IconResolver } from '../icon-resolver';
|
|
9
|
+
import { TabulatorComponent } from '../../tabulator/tabulator.component';
|
|
10
|
+
import { MatProgressBarModule } from '@angular/material/progress-bar';
|
|
11
|
+
import { uploadFile } from '../helpers';
|
|
12
|
+
import { MenuDirective, openMenu } from '../../../public-api';
|
|
13
|
+
import * as i0 from "@angular/core";
|
|
14
|
+
import * as i1 from "../../../public-api";
|
|
15
|
+
import * as i2 from "@angular/material/dialog";
|
|
16
|
+
import * as i3 from "../filemanager.component";
|
|
17
|
+
import * as i4 from "@angular/platform-browser";
|
|
18
|
+
import * as i5 from "ngx-scrollbar";
|
|
19
|
+
import * as i6 from "@angular/material/checkbox";
|
|
20
|
+
import * as i7 from "@angular/material/progress-bar";
|
|
21
|
+
import * as i8 from "@angular/cdk/scrolling";
|
|
22
|
+
const itemWidth = (80 + 20);
|
|
23
|
+
export class FileGridComponent {
|
|
24
|
+
set path(value) {
|
|
25
|
+
if (!value)
|
|
26
|
+
return;
|
|
27
|
+
if (this._path && this.config.navigateOnlyToDescendants) {
|
|
28
|
+
if (!value.startsWith('/'))
|
|
29
|
+
value = '/' + value;
|
|
30
|
+
if (!value.startsWith(this.config.path))
|
|
31
|
+
return;
|
|
32
|
+
}
|
|
33
|
+
let prev = this._path;
|
|
34
|
+
this._path = value;
|
|
35
|
+
if (prev != value) {
|
|
36
|
+
this.pathChange.next(this.path);
|
|
37
|
+
if (this.config.apiSettings)
|
|
38
|
+
this.loadFolder();
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
get path() { return this._path; }
|
|
42
|
+
performChecksum(path, digest) {
|
|
43
|
+
// this.windowManager.openWindow({
|
|
44
|
+
// appId: "checksum",
|
|
45
|
+
// data: { digest, path },
|
|
46
|
+
// workspace: this.windowRef.workspace,
|
|
47
|
+
// width: 600,
|
|
48
|
+
// height: 250
|
|
49
|
+
// });
|
|
50
|
+
}
|
|
51
|
+
get libConfig() { return this.fileManager.libConfig; }
|
|
52
|
+
constructor(fetch, keyboard, dialog, matDialog, fileManager, changeDetector, sanitizer) {
|
|
53
|
+
this.fetch = fetch;
|
|
54
|
+
this.keyboard = keyboard;
|
|
55
|
+
this.dialog = dialog;
|
|
56
|
+
this.matDialog = matDialog;
|
|
57
|
+
this.fileManager = fileManager;
|
|
58
|
+
this.changeDetector = changeDetector;
|
|
59
|
+
this.sanitizer = sanitizer;
|
|
60
|
+
this.pathChange = new EventEmitter();
|
|
61
|
+
this.config = {};
|
|
62
|
+
this.showHiddenFiles = false;
|
|
63
|
+
this.viewMode = "grid";
|
|
64
|
+
this.gridSize = "normal";
|
|
65
|
+
this.fileSelect = new EventEmitter();
|
|
66
|
+
this.fileDblClick = new EventEmitter();
|
|
67
|
+
this.folderSelect = new EventEmitter();
|
|
68
|
+
this.folderDblClick = new EventEmitter();
|
|
69
|
+
this.newTab = new EventEmitter();
|
|
70
|
+
this.loadFiles = new EventEmitter();
|
|
71
|
+
this.directoryContents = [];
|
|
72
|
+
this.selection = [];
|
|
73
|
+
this.selectionChange = new EventEmitter();
|
|
74
|
+
this.value = [];
|
|
75
|
+
this.valueChange = new EventEmitter();
|
|
76
|
+
this.sortedFolders = [];
|
|
77
|
+
this.sorters = {
|
|
78
|
+
"a-z": (a, b) => a.name > b.name ? 1 : -1,
|
|
79
|
+
"z-a": (a, b) => b.name > a.name ? 1 : -1,
|
|
80
|
+
"lastmod": (a, b) => b.stats.mtimeMs - a.stats.mtimeMs,
|
|
81
|
+
"firstmod": (a, b) => a.stats.mtimeMs - b.stats.mtimeMs,
|
|
82
|
+
"size": (a, b) => b.stats.size - a.stats.size,
|
|
83
|
+
"type": (a, b) => a.path.split('.').splice(-1, 1)[0] > b.path.split('.').splice(-1, 1)[0] ? 1 : -1
|
|
84
|
+
};
|
|
85
|
+
this.sortOrder = "a-z";
|
|
86
|
+
this.itemsPerRow = 6;
|
|
87
|
+
// If the current directory is inside of an archive
|
|
88
|
+
this.isArchive = true;
|
|
89
|
+
this.userIsDraggingFile = false;
|
|
90
|
+
this.draggingOver = false;
|
|
91
|
+
this.showLoader = false;
|
|
92
|
+
this.hideLoader = false;
|
|
93
|
+
this.columns = [
|
|
94
|
+
{ id: "name", label: "Name" },
|
|
95
|
+
{ id: "size", label: "Size" },
|
|
96
|
+
{ id: "type", label: "Type" },
|
|
97
|
+
{ id: "owner", label: "Owner" },
|
|
98
|
+
{ id: "group", label: "Group" },
|
|
99
|
+
{ id: "permissions", label: "Permissions" },
|
|
100
|
+
{ id: "location", label: "Location" },
|
|
101
|
+
{ id: "modified", label: "Modified" },
|
|
102
|
+
{ id: "modified--time", label: "Modified - Time" },
|
|
103
|
+
{ id: "accessed", label: "Accessed" },
|
|
104
|
+
{ id: "created", label: "Created" },
|
|
105
|
+
{ id: "recency", label: "Recency" },
|
|
106
|
+
{ id: "star", label: "Star" },
|
|
107
|
+
{ id: "detailed-type", label: "Detailed Type" },
|
|
108
|
+
];
|
|
109
|
+
this.cols = [
|
|
110
|
+
{ id: "name", label: "Name" },
|
|
111
|
+
{ id: "size", label: "Size" },
|
|
112
|
+
{ id: "modified", label: "Modified" },
|
|
113
|
+
{ id: "star", label: "Star" }
|
|
114
|
+
];
|
|
115
|
+
this.folderContextMenu = [
|
|
116
|
+
{
|
|
117
|
+
label: "New Folder",
|
|
118
|
+
// shortcutLabel: "Shift+Ctrl+N",
|
|
119
|
+
icon: "create_new_folder",
|
|
120
|
+
action: (data) => {
|
|
121
|
+
// console.log("New folder goodness");
|
|
122
|
+
// console.log(data);
|
|
123
|
+
this.dialog.open("folder-rename", "@dotglitch/ngx-web-components", {
|
|
124
|
+
inputs: { path: data?.path || this.path, name: data?.name || '', config: this.config }
|
|
125
|
+
}).then(r => this.loadFolder());
|
|
126
|
+
}
|
|
127
|
+
},
|
|
128
|
+
{
|
|
129
|
+
label: "Upload file",
|
|
130
|
+
// shortcutLabel: "Ctrl+D",
|
|
131
|
+
icon: "file_upload",
|
|
132
|
+
action: (evt) => uploadFile(this.fetch, this.config, this._path, evt ? (evt.path + evt.name) : null, this.fileManager.contextTags).then(res => {
|
|
133
|
+
this.loadFolder();
|
|
134
|
+
})
|
|
135
|
+
},
|
|
136
|
+
"separator",
|
|
137
|
+
// {
|
|
138
|
+
// isDisabled: (data) => true,
|
|
139
|
+
// label: "_P_aste",
|
|
140
|
+
// icon: "content_paste",
|
|
141
|
+
// action: (evt) => {
|
|
142
|
+
// }
|
|
143
|
+
// },
|
|
144
|
+
{
|
|
145
|
+
label: "Select _A_ll",
|
|
146
|
+
shortcutLabel: "Ctrl+A",
|
|
147
|
+
icon: "select_all",
|
|
148
|
+
action: (evt) => {
|
|
149
|
+
this.selection = this._sortFilter();
|
|
150
|
+
this.selectionText = this.getSelectionText();
|
|
151
|
+
this.selectionChange.next(this.selection);
|
|
152
|
+
}
|
|
153
|
+
},
|
|
154
|
+
// "separator",
|
|
155
|
+
// {
|
|
156
|
+
// label: "P_r_operties",
|
|
157
|
+
// icon: "find_in_page",
|
|
158
|
+
// action: (evt) => {
|
|
159
|
+
// }
|
|
160
|
+
// }
|
|
161
|
+
];
|
|
162
|
+
this.fileContextMenu = [];
|
|
163
|
+
this.nameCellFormatter = ((cell, formatterParams, onRendered) => {
|
|
164
|
+
// TODO: Sanitize?
|
|
165
|
+
const item = cell.getData();
|
|
166
|
+
return `
|
|
167
|
+
<span style="display: flex; align-items: center">
|
|
168
|
+
<img style="height: 24px; margin-right: 6px" src="${item['_icon'].path}"/>
|
|
169
|
+
<p style="margin: 0">${this.sanitizer.sanitize(SecurityContext.HTML, item['vanityName'] || item.name)}</p>
|
|
170
|
+
</span>
|
|
171
|
+
`;
|
|
172
|
+
}).bind(this);
|
|
173
|
+
this.iconResolver = new IconResolver(this.libConfig.assetPath);
|
|
174
|
+
// ctrl + a => select all
|
|
175
|
+
keyboard.onKeyCommand({
|
|
176
|
+
key: "a",
|
|
177
|
+
ctrl: true,
|
|
178
|
+
}).subscribe(evt => {
|
|
179
|
+
this.selection = this._sortFilter();
|
|
180
|
+
this.selectionText = this.getSelectionText();
|
|
181
|
+
this.selectionChange.next(this.selection);
|
|
182
|
+
});
|
|
183
|
+
// ctrl + c => copy file names to clipboard
|
|
184
|
+
keyboard.onKeyCommand({
|
|
185
|
+
key: "c",
|
|
186
|
+
ctrl: true,
|
|
187
|
+
}).subscribe(evt => {
|
|
188
|
+
});
|
|
189
|
+
// ctrl + h => toggle hidden files
|
|
190
|
+
keyboard.onKeyCommand({
|
|
191
|
+
key: "h",
|
|
192
|
+
ctrl: true,
|
|
193
|
+
interrupt: true
|
|
194
|
+
}).subscribe(evt => {
|
|
195
|
+
this.showHiddenFiles = !this.showHiddenFiles;
|
|
196
|
+
});
|
|
197
|
+
// F2 => Rename selected files
|
|
198
|
+
keyboard.onKeyCommand({
|
|
199
|
+
key: "f2",
|
|
200
|
+
}).subscribe(evt => {
|
|
201
|
+
// Rename selected file(s)
|
|
202
|
+
});
|
|
203
|
+
// Enter => Open selected files
|
|
204
|
+
keyboard.onKeyCommand({
|
|
205
|
+
key: "Enter",
|
|
206
|
+
}).subscribe(evt => {
|
|
207
|
+
const files = this.directoryContents.filter(dc => this.selection.find(i => i.name == dc.name));
|
|
208
|
+
// this.windowManager.openFiles(files as any);
|
|
209
|
+
});
|
|
210
|
+
// Delete => delete selected files
|
|
211
|
+
keyboard.onKeyCommand({
|
|
212
|
+
key: "delete",
|
|
213
|
+
}).subscribe(evt => {
|
|
214
|
+
const files = this.directoryContents.filter(dc => this.selection.find(i => i.name == dc.name));
|
|
215
|
+
});
|
|
216
|
+
}
|
|
217
|
+
async ngOnInit() {
|
|
218
|
+
// this.loadFolder();
|
|
219
|
+
}
|
|
220
|
+
ngAfterViewInit() {
|
|
221
|
+
this.fileContextMenu = [
|
|
222
|
+
{
|
|
223
|
+
label: "Download",
|
|
224
|
+
icon: "download",
|
|
225
|
+
action: (file) => {
|
|
226
|
+
let target = `${window.origin}${this.config.apiSettings.downloadEntryUrl}`;
|
|
227
|
+
let path = file.path + file.name;
|
|
228
|
+
if (file.kind == "directory" && !path.endsWith('/'))
|
|
229
|
+
path += "/";
|
|
230
|
+
target += `${target.includes('?') ? '&' : '?'}path=${path}&ngsw-bypass=true`;
|
|
231
|
+
console.log(target);
|
|
232
|
+
// window.open(target);
|
|
233
|
+
var link = document.createElement("a");
|
|
234
|
+
link.download = file.name;
|
|
235
|
+
link.href = target;
|
|
236
|
+
link.click();
|
|
237
|
+
link.remove();
|
|
238
|
+
this.fileManager.fileDownload.next(file);
|
|
239
|
+
}
|
|
240
|
+
},
|
|
241
|
+
{
|
|
242
|
+
label: "Open in new Tab",
|
|
243
|
+
icon: "open_in_new",
|
|
244
|
+
isVisible: (data) => data.kind == "directory",
|
|
245
|
+
action: (data) => {
|
|
246
|
+
this.fileManager.initTab(data.path + data.name);
|
|
247
|
+
}
|
|
248
|
+
},
|
|
249
|
+
// {
|
|
250
|
+
// label: "Open with Application...",
|
|
251
|
+
// isVisible: (data) => data.kind == "file",
|
|
252
|
+
// shortcutLabel: "Ctrl+D",
|
|
253
|
+
// action: (evt) => {
|
|
254
|
+
// },
|
|
255
|
+
// },
|
|
256
|
+
"separator",
|
|
257
|
+
// {
|
|
258
|
+
// label: "Cut",
|
|
259
|
+
// icon: "content_cut",
|
|
260
|
+
// isDisabled: data => true,
|
|
261
|
+
// action: (evt) => {
|
|
262
|
+
// },
|
|
263
|
+
// },
|
|
264
|
+
// {
|
|
265
|
+
// label: "Copy",
|
|
266
|
+
// icon: "file_copy",
|
|
267
|
+
// isDisabled: data => true,
|
|
268
|
+
// childrenResolver: () => new Promise(r => setTimeout(r, 500000))
|
|
269
|
+
// },
|
|
270
|
+
// {
|
|
271
|
+
// label: "Move To...",
|
|
272
|
+
// icon: "drive_file_move",
|
|
273
|
+
// shortcutLabel: "Ctrl+A",
|
|
274
|
+
// action: (evt) => {
|
|
275
|
+
// },
|
|
276
|
+
// },
|
|
277
|
+
// {
|
|
278
|
+
// label: "Copy To...",
|
|
279
|
+
// icon: "folder_copy",
|
|
280
|
+
// shortcutLabel: "Ctrl+A",
|
|
281
|
+
// action: (evt) => {
|
|
282
|
+
// },
|
|
283
|
+
// },
|
|
284
|
+
{
|
|
285
|
+
label: "Delete",
|
|
286
|
+
icon: "delete",
|
|
287
|
+
// shortcutLabel: "Del",
|
|
288
|
+
isVisible: data => !data.path.includes("#/"),
|
|
289
|
+
action: (evt) => {
|
|
290
|
+
const path = evt.path + evt.name;
|
|
291
|
+
const url = this.config.apiSettings.deleteEntryUrlTemplate
|
|
292
|
+
? this.config.apiSettings.deleteEntryUrlTemplate(path)
|
|
293
|
+
: this.config.apiSettings.deleteEntryUrl;
|
|
294
|
+
this.fetch.delete(url)
|
|
295
|
+
.then(() => this.loadFolder());
|
|
296
|
+
},
|
|
297
|
+
},
|
|
298
|
+
// {
|
|
299
|
+
// label: "Shred file",
|
|
300
|
+
// icon: "delete_forever",
|
|
301
|
+
// isVisible: data => !data.path.includes("#/"), // omit files in compressed dirs
|
|
302
|
+
// action: (evt) => {
|
|
303
|
+
// this.fetch.post(`/api/filesystem/delete?wipe=true`, { files: [evt.path + evt.name]})
|
|
304
|
+
// .then(() => this.loadFolder())
|
|
305
|
+
// },
|
|
306
|
+
// },
|
|
307
|
+
{
|
|
308
|
+
label: "Rename",
|
|
309
|
+
icon: "drive_file_rename_outline",
|
|
310
|
+
isVisible: data => !data.path.includes("#/"),
|
|
311
|
+
// shortcutLabel: "F2",
|
|
312
|
+
action: (data) => {
|
|
313
|
+
this.dialog.open("folder-rename", "@dotglitch/ngx-web-components", {
|
|
314
|
+
inputs: { path: data?.path || this.path, name: data?.name || '', config: this.config }
|
|
315
|
+
}).then(r => this.loadFolder());
|
|
316
|
+
}
|
|
317
|
+
},
|
|
318
|
+
// Extract Here
|
|
319
|
+
// Extract To...
|
|
320
|
+
// {
|
|
321
|
+
// label: "Extract Here",
|
|
322
|
+
// icon: "folder_zip",
|
|
323
|
+
// shortcutLabel: "Ctrl+A",
|
|
324
|
+
// isDisabled: (data) => !(data.kind == "file" && data.ext != ".zip" && isArchive(data)),
|
|
325
|
+
// action: (evt) => {
|
|
326
|
+
// // TODO
|
|
327
|
+
// },
|
|
328
|
+
// },
|
|
329
|
+
// {
|
|
330
|
+
// label: "Extract to...",
|
|
331
|
+
// icon: "folder_zip",
|
|
332
|
+
// shortcutLabel: "Ctrl+A",
|
|
333
|
+
// isDisabled: (data) => !(data.kind == "file" && data.ext != ".zip" && isArchive(data)),
|
|
334
|
+
// action: (evt) => {
|
|
335
|
+
// // TODO
|
|
336
|
+
// },
|
|
337
|
+
// },
|
|
338
|
+
// {
|
|
339
|
+
// label: "Compress...",
|
|
340
|
+
// icon: "folder_zip",
|
|
341
|
+
// shortcutLabel: "Ctrl+A",
|
|
342
|
+
// isDisabled: (data) => data.kind == "file",
|
|
343
|
+
// action: (evt) => {
|
|
344
|
+
// // TODO
|
|
345
|
+
// },
|
|
346
|
+
// },
|
|
347
|
+
{
|
|
348
|
+
label: "Checksum",
|
|
349
|
+
icon: "manage_search",
|
|
350
|
+
isDisabled: (data) => data.kind != "file",
|
|
351
|
+
children: [
|
|
352
|
+
{
|
|
353
|
+
label: "MD5",
|
|
354
|
+
action: (evt) => this.performChecksum(evt.path + evt.name, "md5"),
|
|
355
|
+
},
|
|
356
|
+
{
|
|
357
|
+
label: "SHA1",
|
|
358
|
+
action: (evt) => this.performChecksum(evt.path + evt.name, "sha1"),
|
|
359
|
+
},
|
|
360
|
+
{
|
|
361
|
+
label: "SHA256",
|
|
362
|
+
action: (evt) => this.performChecksum(evt.path + evt.name, "sha256"),
|
|
363
|
+
},
|
|
364
|
+
{
|
|
365
|
+
label: "SHA512",
|
|
366
|
+
action: (evt) => this.performChecksum(evt.path + evt.name, "sha512"),
|
|
367
|
+
},
|
|
368
|
+
],
|
|
369
|
+
isVisible: (data) => {
|
|
370
|
+
return false;
|
|
371
|
+
return !this.isArchive || data.kind == "file";
|
|
372
|
+
},
|
|
373
|
+
},
|
|
374
|
+
// {
|
|
375
|
+
// label: "Star",
|
|
376
|
+
// icon: "star",
|
|
377
|
+
// shortcutLabel: "Ctrl+A",
|
|
378
|
+
// action: (evt) => {
|
|
379
|
+
// },
|
|
380
|
+
// },
|
|
381
|
+
// "separator",
|
|
382
|
+
// {
|
|
383
|
+
// label: "P_r_operties",
|
|
384
|
+
// icon: "find_in_page",
|
|
385
|
+
// action: (evt) => {
|
|
386
|
+
// },
|
|
387
|
+
// }
|
|
388
|
+
];
|
|
389
|
+
}
|
|
390
|
+
async loadFolder() {
|
|
391
|
+
this.showLoader = true;
|
|
392
|
+
this.hideLoader = false;
|
|
393
|
+
const url = this.config.apiSettings.listEntriesUrlTemplate
|
|
394
|
+
? this.config.apiSettings.listEntriesUrlTemplate(this.path)
|
|
395
|
+
: this.config.apiSettings.listEntriesUrl;
|
|
396
|
+
this.fetch.post(url, { path: this.path, showHidden: this.showHiddenFiles })
|
|
397
|
+
.then((data) => {
|
|
398
|
+
const files = data?.files || [];
|
|
399
|
+
const dirs = data?.dirs || [];
|
|
400
|
+
const descriptors = files.concat(dirs);
|
|
401
|
+
descriptors.forEach(f => {
|
|
402
|
+
f['_icon'] = this.iconResolver.resolveIcon(f);
|
|
403
|
+
if (f.kind == "file") {
|
|
404
|
+
f['_ctime'] = new Date(f.stats?.ctimeMs)?.toLocaleString();
|
|
405
|
+
f['_mtime'] = new Date(f.stats?.mtimeMs)?.toLocaleString();
|
|
406
|
+
f['_size'] = this.bytesToString(f.stats?.size);
|
|
407
|
+
}
|
|
408
|
+
});
|
|
409
|
+
this.directoryContents = descriptors;
|
|
410
|
+
this._sortFilter();
|
|
411
|
+
this.resize();
|
|
412
|
+
this.loadFiles.next(descriptors);
|
|
413
|
+
if (this.sortedFolders.length > 0)
|
|
414
|
+
this.flowRows();
|
|
415
|
+
setTimeout(() => this.resize(), 250);
|
|
416
|
+
setTimeout(() => this.resize(), 500);
|
|
417
|
+
setTimeout(() => this.resize(), 1000);
|
|
418
|
+
setTimeout(() => this.resize(), 2500);
|
|
419
|
+
setTimeout(() => this.resize(), 5000);
|
|
420
|
+
})
|
|
421
|
+
.catch(e => console.error(e))
|
|
422
|
+
.finally(() => {
|
|
423
|
+
this.hideLoader = true;
|
|
424
|
+
setTimeout(() => {
|
|
425
|
+
this.showLoader = false;
|
|
426
|
+
}, 200);
|
|
427
|
+
});
|
|
428
|
+
}
|
|
429
|
+
flowRows() {
|
|
430
|
+
let filtered = this._sortFilter();
|
|
431
|
+
this.sortedFolders = [];
|
|
432
|
+
const num = Math.ceil(filtered.length / this.itemsPerRow);
|
|
433
|
+
const iterations = Math.min(num, 100);
|
|
434
|
+
for (let row = 0; row < iterations; row++) {
|
|
435
|
+
if (!this.sortedFolders[row])
|
|
436
|
+
this.sortedFolders[row] = [];
|
|
437
|
+
for (let i = row * this.itemsPerRow; i < (row + 1) * this.itemsPerRow && i < filtered.length; i++) {
|
|
438
|
+
this.sortedFolders[row].push(filtered[i]);
|
|
439
|
+
}
|
|
440
|
+
}
|
|
441
|
+
}
|
|
442
|
+
onSelect(item, evt) {
|
|
443
|
+
evt.stopPropagation();
|
|
444
|
+
if (this.keyboard.isShiftPressed) {
|
|
445
|
+
let start = this.directoryContents.findIndex(i => i.name == this.selection.slice(-1, 1)[0].name);
|
|
446
|
+
let end = this.directoryContents.indexOf(item);
|
|
447
|
+
if (start == -1)
|
|
448
|
+
start = end;
|
|
449
|
+
let items = start > end
|
|
450
|
+
? this.directoryContents.slice(end, start + 1)
|
|
451
|
+
: this.directoryContents.slice(start, end + 1);
|
|
452
|
+
this.selection = items;
|
|
453
|
+
}
|
|
454
|
+
else if (this.keyboard.isCtrlPressed) {
|
|
455
|
+
if (!this.selection.includes(item))
|
|
456
|
+
this.selection.push(item);
|
|
457
|
+
else // Case that we selected the same item twice
|
|
458
|
+
this.selection.splice(this.selection.indexOf(item), 1);
|
|
459
|
+
}
|
|
460
|
+
else
|
|
461
|
+
this.selection = [item];
|
|
462
|
+
if (this.selection.length == 1) {
|
|
463
|
+
if (this.selection[0].kind == "directory")
|
|
464
|
+
this.folderSelect.next(this.selection[0]);
|
|
465
|
+
else
|
|
466
|
+
this.fileSelect.next(this.selection[0]);
|
|
467
|
+
}
|
|
468
|
+
this.selectionChange.next(this.selection);
|
|
469
|
+
this.selectionText = this.getSelectionText();
|
|
470
|
+
}
|
|
471
|
+
onItemClick(file) {
|
|
472
|
+
console.log(file, this);
|
|
473
|
+
if (file.kind == "directory") {
|
|
474
|
+
this.folderDblClick.next(file);
|
|
475
|
+
this.path = file.path + file.name;
|
|
476
|
+
}
|
|
477
|
+
// else if (file.ext == "zip") {
|
|
478
|
+
// this.fileDblClick.next(file);
|
|
479
|
+
// this.path = file.path + file.name;
|
|
480
|
+
// }
|
|
481
|
+
else {
|
|
482
|
+
this.fileDblClick.next(file);
|
|
483
|
+
this.fileSelect.next(file);
|
|
484
|
+
}
|
|
485
|
+
}
|
|
486
|
+
onToggle(item, state) {
|
|
487
|
+
item['_value'] = state.checked;
|
|
488
|
+
// TODO: What causes this to be null when initialized with an array?
|
|
489
|
+
if (!this.value) {
|
|
490
|
+
this.value = [];
|
|
491
|
+
}
|
|
492
|
+
if (state.checked) {
|
|
493
|
+
this.value.push(item);
|
|
494
|
+
}
|
|
495
|
+
else {
|
|
496
|
+
const i = this.value.findIndex(v => v == item);
|
|
497
|
+
if (i >= 0)
|
|
498
|
+
this.value.splice(i, 1);
|
|
499
|
+
}
|
|
500
|
+
this.valueChange.next(this.value);
|
|
501
|
+
}
|
|
502
|
+
async clearSelection() {
|
|
503
|
+
this.value = [];
|
|
504
|
+
this.valueChange.next(this.value);
|
|
505
|
+
this.tabulator?.table?.getRows().forEach(r => r.getElement().classList.remove('selected'));
|
|
506
|
+
}
|
|
507
|
+
_sortFilter() {
|
|
508
|
+
return this.directoryContents = this.directoryContents?.filter(d => d.kind == 'directory')
|
|
509
|
+
.concat(this.directoryContents?.filter(d => d.kind == 'file')
|
|
510
|
+
.sort(this.sorters[this.sortOrder]));
|
|
511
|
+
}
|
|
512
|
+
getSelectionText() {
|
|
513
|
+
const dirCount = this.selection.filter(s => s.kind == "directory").length;
|
|
514
|
+
const fileCount = this.selection.filter(s => s.kind == "file").length;
|
|
515
|
+
if (dirCount + fileCount == 0)
|
|
516
|
+
return "";
|
|
517
|
+
const totalSize = this.directoryContents
|
|
518
|
+
.filter(d => d.kind == "file")
|
|
519
|
+
.filter(d => this.selection?.find(i => i.name == d.name))
|
|
520
|
+
.map(d => d['stats'].size).reduce((a, b) => a + b, 0);
|
|
521
|
+
if (dirCount + fileCount == 1)
|
|
522
|
+
return `"${this.selection[0].name}" selected (${this.bytesToString(totalSize)})`;
|
|
523
|
+
if (dirCount > 0 && fileCount == 0)
|
|
524
|
+
return `"${dirCount}" folders selected`;
|
|
525
|
+
if (fileCount > 0 && dirCount == 0)
|
|
526
|
+
return `${fileCount} items selected (${this.bytesToString(totalSize)})`;
|
|
527
|
+
return `${dirCount} folder${dirCount == 1 ? "" : "s"} selected, ${fileCount} other item${fileCount == 1 ? "" : "s"} selected (${this.bytesToString(totalSize)})`;
|
|
528
|
+
}
|
|
529
|
+
bytesToString(bytes, decimals = 2) {
|
|
530
|
+
if (!+bytes)
|
|
531
|
+
return '0 Bytes';
|
|
532
|
+
const k = 1024;
|
|
533
|
+
const dm = decimals < 0 ? 0 : decimals;
|
|
534
|
+
const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB'];
|
|
535
|
+
const i = Math.floor(Math.log(bytes) / Math.log(k));
|
|
536
|
+
return `${parseFloat((bytes / Math.pow(k, i)).toFixed(dm))} ${sizes[i]}`;
|
|
537
|
+
}
|
|
538
|
+
resize() {
|
|
539
|
+
if (!this.filesRef) {
|
|
540
|
+
setTimeout(() => this.resize(), 25);
|
|
541
|
+
return;
|
|
542
|
+
}
|
|
543
|
+
;
|
|
544
|
+
const bounds = this.filesRef.nativeElement.getBoundingClientRect();
|
|
545
|
+
const newColCount = Math.floor(bounds.width / itemWidth);
|
|
546
|
+
if (newColCount != this.itemsPerRow) {
|
|
547
|
+
this.itemsPerRow = Math.floor(bounds.width / itemWidth);
|
|
548
|
+
if (this.itemsPerRow > 100)
|
|
549
|
+
this.itemsPerRow = 1;
|
|
550
|
+
this.flowRows();
|
|
551
|
+
}
|
|
552
|
+
if (this.sortedFolders?.length == 0)
|
|
553
|
+
this.flowRows();
|
|
554
|
+
}
|
|
555
|
+
onDragStart(evt, item) {
|
|
556
|
+
const target = `${window.origin}/api/filesystem/download?dir=${item.path}&file=${item.name}`;
|
|
557
|
+
evt.dataTransfer.clearData();
|
|
558
|
+
// evt.dataTransfer.setData('text/uri-list', target);
|
|
559
|
+
// evt.dataTransfer.setData('DownloadURL', `text/uri-list:${target}`);
|
|
560
|
+
evt.dataTransfer.setData('text/plain', item.name);
|
|
561
|
+
}
|
|
562
|
+
onDrop(ev) {
|
|
563
|
+
ev.preventDefault();
|
|
564
|
+
if (ev.dataTransfer.items) {
|
|
565
|
+
// Use DataTransferItemList interface to access the file(s)
|
|
566
|
+
[...ev.dataTransfer.items].forEach((item, i) => {
|
|
567
|
+
// If dropped items aren't files, reject them
|
|
568
|
+
if (item.kind === "file") {
|
|
569
|
+
const file = item.getAsFile();
|
|
570
|
+
console.log(`… file[${i}].name = ${file.name}`);
|
|
571
|
+
}
|
|
572
|
+
});
|
|
573
|
+
}
|
|
574
|
+
else {
|
|
575
|
+
// Use DataTransfer interface to access the file(s)
|
|
576
|
+
[...ev.dataTransfer.files].forEach((file, i) => {
|
|
577
|
+
console.log(`… file[${i}].name = ${file.name}`);
|
|
578
|
+
});
|
|
579
|
+
}
|
|
580
|
+
}
|
|
581
|
+
onRowCtx({ event, row }) {
|
|
582
|
+
openMenu(this.matDialog, this.fileContextMenu, row.getData(), event);
|
|
583
|
+
}
|
|
584
|
+
onRowClick({ event, row, data }) {
|
|
585
|
+
// $event.data['_value'] = $event.data['_value'] == true ? false : true
|
|
586
|
+
// console.log(event, row, data, this.value);
|
|
587
|
+
const rowEl = row.getElement();
|
|
588
|
+
let state = rowEl.classList.contains('selected');
|
|
589
|
+
data['_value'] = !state;
|
|
590
|
+
if (!this.value) {
|
|
591
|
+
this.value = [];
|
|
592
|
+
}
|
|
593
|
+
if (!state) {
|
|
594
|
+
rowEl.classList.add('selected');
|
|
595
|
+
this.value.push(data);
|
|
596
|
+
}
|
|
597
|
+
else {
|
|
598
|
+
rowEl.classList.remove('selected');
|
|
599
|
+
const i = this.value.findIndex(v => v == data);
|
|
600
|
+
if (i >= 0)
|
|
601
|
+
this.value.splice(i, 1);
|
|
602
|
+
}
|
|
603
|
+
this.valueChange.next(this.value);
|
|
604
|
+
}
|
|
605
|
+
sort() {
|
|
606
|
+
this._sortFilter();
|
|
607
|
+
}
|
|
608
|
+
}
|
|
609
|
+
FileGridComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FileGridComponent, deps: [{ token: i1.Fetch }, { token: i1.KeyboardService }, { token: i1.DialogService }, { token: i2.MatDialog }, { token: i3.FilemanagerComponent }, { token: i0.ChangeDetectorRef }, { token: i4.DomSanitizer }], target: i0.ɵɵFactoryTarget.Component });
|
|
610
|
+
FileGridComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: FileGridComponent, isStandalone: true, selector: "app-file-grid", inputs: { path: "path", config: "config", showHiddenFiles: "showHiddenFiles", viewMode: "viewMode", gridSize: "gridSize", tab: "tab", selection: "selection", value: "value", sortOrder: "sortOrder" }, outputs: { pathChange: "pathChange", fileSelect: "fileSelect", fileDblClick: "fileDblClick", folderSelect: "folderSelect", folderDblClick: "folderDblClick", newTab: "newTab", loadFiles: "loadFiles", selectionChange: "selectionChange", valueChange: "valueChange" }, viewQueries: [{ propertyName: "filesRef", first: true, predicate: ["fileViewport"], descendants: true }, { propertyName: "tabulator", first: true, predicate: TabulatorComponent, descendants: true }, { propertyName: "renameTemplate", first: true, predicate: ["renameTemplate"], descendants: true, read: TemplateRef }], ngImport: i0, template: "<ng-container *ngIf=\"showLoader\">\n <mat-progress-bar [class.hide]=\"hideLoader\" mode=\"query\"/>\n</ng-container>\n\n<div\n style=\"display: contents\"\n [style.--filemanager-fileicon-backdrop]=\"'url(' + iconResolver.path + '/pop/generic.svg)'\"\n (dragstart)=\"userIsDraggingFile = true\"\n (dragend)=\"userIsDraggingFile = false\"\n (dragover)=\"draggingOver = true\"\n (dragleave)=\"draggingOver = false\"\n (ondrop)=\"onDrop($event)\"\n>\n <!-- <ng-container *ngIf=\"draggingOver\"></ng-container> -->\n <!-- Grid mode -->\n <ng-scrollbar\n *ngIf=\"viewMode == 'grid'\"\n class=\"grid {{gridSize}} {{config.imageSize || 'normal'}}\"\n [class.selectionMode]=\"config.mode == 'focusFiles'\"\n [class.showDropArea]=\"draggingOver\"\n style=\"height: 100%; width: 100%\"\n track=\"vertical\"\n pointerEventsMethod=\"scrollbar\"\n [ngx-contextmenu]=\"folderContextMenu\"\n >\n <div class=\"resize-observer\" #fileViewport></div>\n <cdk-virtual-scroll-viewport\n itemSize=\"134\"\n scrollViewport\n (click)=\"selection = []; selectionText = ''\"\n >\n <div class=\"row\" *cdkVirtualFor=\"let row of sortedFolders\">\n <div class=\"file\"\n *ngFor=\"let item of row\"\n [class.selected]=\"selection.includes(item)\"\n [class.generic]=\"item['_icon'].needsBackdrop\"\n [ngx-contextmenu]=\"fileContextMenu\"\n [ngx-menu-context]=\"item\"\n >\n <mat-checkbox\n #checkbox\n *ngIf=\"config.mode == 'focusFiles' && item.kind == 'file'\"\n [checked]=\"item['_value']\"\n (change)=\"onToggle(item, $event)\"\n />\n <div\n style=\"display: contents\"\n (click)=\"onSelect(item, $event)\"\n (dblclick)=\"onItemClick(item)\"\n (dragstart)=\"onDragStart($event, item)\"\n >\n <img [src]=\"item['_icon'].path\"/>\n <p>{{item['vanityName'] || item.name}}</p>\n </div>\n </div>\n </div>\n </cdk-virtual-scroll-viewport>\n </ng-scrollbar>\n\n <!-- List mode -->\n <div\n *ngIf=\"viewMode == 'list'\"\n style=\"width: 100%; height: 100%\"\n [class.showDropArea]=\"draggingOver\"\n [ngx-contextmenu]=\"folderContextMenu\"\n >\n <app-tabulator\n [dataSource]=\"directoryContents\"\n [columns]=\"[\n { field: 'name', title: 'Name', formatter: nameCellFormatter },\n { field: '_size', title: 'Size' },\n { field: '_ctime', title: 'Created' },\n { field: '_mtime', title: 'Modified' },\n ]\"\n [options]=\"{\n rowHeight: 32\n }\"\n (rowClick)=\"onRowClick($event)\"\n (rowDblClick)=\"onItemClick($event.data)\"\n (rowContext)=\"onRowCtx($event)\"\n />\n </div>\n</div>\n\n<div class=\"select-hint\" *ngIf=\"selectionText\">\n {{selectionText}}\n</div>\n\n", styles: [":host{display:block;height:100%;width:100%;overflow:hidden}.resize-observer{position:absolute;inset:0}.grid .row{flex:1;display:grid;grid-template-columns:repeat(auto-fill,80px);justify-content:space-between;grid-gap:20px;grid-auto-rows:min-content;padding:10px;margin-right:10px}.grid .file{width:80px;z-index:1;position:relative;transition:opacity 50ms ease-in-out;display:flex;flex-direction:column;align-items:center;text-align:center}.grid .file.selected p{background-color:#8ad9d9;color:#000}.grid .file.generic:before{content:\" \";position:absolute;background:var(--filemanager-fileicon-backdrop);background-repeat:no-repeat;width:100%;height:100%;z-index:-1;left:0}.grid .file.generic img{height:44px;width:44px;margin-top:28px;margin-bottom:8px}.grid .file img{-webkit-user-select:none;user-select:none}.grid .file p{height:50px;width:100%;margin:0;padding:2px;font-size:14px;line-height:16px;border-radius:4px;overflow:hidden;display:-webkit-box;-webkit-line-clamp:3;-webkit-box-orient:vertical;text-overflow:ellipsis;overflow-wrap:break-word;transition:background-color 50ms ease-in-out,color 50ms ease-in-out}.grid .file mat-checkbox{position:absolute;left:-32px;top:-16px}:host ::ng-deep .small .file{width:46px}:host ::ng-deep .grid.selectionMode .row{padding-left:30px}:host ::ng-deep .grid.selectionMode .cdk-virtual-scroll-content-wrapper{padding-top:10px}:host ::ng-deep .tabulator .tabulator-row .tabulator-cell:not(:first-of-type){padding-top:9px}.list .row{flex:1;display:flex;align-items:center;padding:5px 10px;height:42px}.list .row p{margin:0}.list .row:hover{background-color:var(--filemanager-row-hover-background-color, #343434)}.list .row.selected p{background-color:#8ad9d9;color:#000;border-radius:5px;padding:0 4px}.list .row.odd{background-color:var(--filemanager-row-alt-background-color, #323232)}.select-hint{position:absolute;font-size:14px;bottom:0;right:0;background-color:#000a;padding:2px 10px;border:1px solid black;border-top-right-radius:5px;border-bottom:none;border-right:none;z-index:9999}.select-hint:hover{display:none}.showDropArea{border:4px dashed rgba(18,18,18,.8);transition:border-color .25s ease}:host ::ng-deep ng-scrollbar.ng-scrollbar{box-sizing:border-box!important;--scrollbar-thumb-color: #666;min-height:200px;min-width:200px}mat-progress-bar{--mdc-linear-progress-track-height: 2px;position:absolute;top:0;left:0;width:100%;animation:dropIn .2s ease;transition:position .2s ease;z-index:100}mat-progress-bar.hide{top:-2px}@keyframes dropIn{0%{top:-2px}to{top:0}}\n"], dependencies: [{ kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "ngmodule", type: MatTabsModule }, { kind: "ngmodule", type: NgScrollbarModule }, { kind: "component", type: i5.NgScrollbar, selector: "ng-scrollbar", inputs: ["disabled", "sensorDisabled", "pointerEventsDisabled", "viewportPropagateMouseMove", "autoHeightDisabled", "autoWidthDisabled", "viewClass", "trackClass", "thumbClass", "minThumbSize", "trackClickScrollDuration", "pointerEventsMethod", "track", "visibility", "appearance", "position", "sensorDebounce", "scrollAuditTime"], outputs: ["updated"], exportAs: ["ngScrollbar"] }, { kind: "directive", type: i5.ScrollViewport, selector: "[scrollViewport]" }, { kind: "ngmodule", type: MatInputModule }, { kind: "ngmodule", type: MatCheckboxModule }, { kind: "component", type: i6.MatCheckbox, selector: "mat-checkbox", inputs: ["disableRipple", "color", "tabIndex"], exportAs: ["matCheckbox"] }, { kind: "ngmodule", type: MatProgressBarModule }, { kind: "component", type: i7.MatProgressBar, selector: "mat-progress-bar", inputs: ["color", "value", "bufferValue", "mode"], outputs: ["animationEnd"], exportAs: ["matProgressBar"] }, { kind: "component", type: TabulatorComponent, selector: "app-tabulator", inputs: ["dataSource", "columns", "key", "options"], outputs: ["cellClick", "cellDblClick", "rowClick", "rowContext", "rowDblClick"] }, { kind: "directive", type: MenuDirective, selector: "[ngx-contextmenu],[ngx-menu],[ngxContextmenu],[ngxMenu]", inputs: ["ngx-menu-context", "ngx-contextmenu", "ngx-menu", "ngx-menu-config"] }, { kind: "ngmodule", type: ScrollingModule }, { kind: "directive", type: i8.CdkFixedSizeVirtualScroll, selector: "cdk-virtual-scroll-viewport[itemSize]", inputs: ["itemSize", "minBufferPx", "maxBufferPx"] }, { kind: "directive", type: i8.CdkVirtualForOf, selector: "[cdkVirtualFor][cdkVirtualForOf]", inputs: ["cdkVirtualForOf", "cdkVirtualForTrackBy", "cdkVirtualForTemplate", "cdkVirtualForTemplateCacheSize"] }, { kind: "component", type: i8.CdkVirtualScrollViewport, selector: "cdk-virtual-scroll-viewport", inputs: ["orientation", "appendOnly"], outputs: ["scrolledIndexChange"] }] });
|
|
611
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FileGridComponent, decorators: [{
|
|
612
|
+
type: Component,
|
|
613
|
+
args: [{ selector: 'app-file-grid', imports: [
|
|
614
|
+
NgIf,
|
|
615
|
+
NgForOf,
|
|
616
|
+
DatePipe,
|
|
617
|
+
MatTabsModule,
|
|
618
|
+
NgScrollbarModule,
|
|
619
|
+
MatInputModule,
|
|
620
|
+
MatCheckboxModule,
|
|
621
|
+
MatProgressBarModule,
|
|
622
|
+
TabulatorComponent,
|
|
623
|
+
MenuDirective,
|
|
624
|
+
ScrollingModule
|
|
625
|
+
], standalone: true, template: "<ng-container *ngIf=\"showLoader\">\n <mat-progress-bar [class.hide]=\"hideLoader\" mode=\"query\"/>\n</ng-container>\n\n<div\n style=\"display: contents\"\n [style.--filemanager-fileicon-backdrop]=\"'url(' + iconResolver.path + '/pop/generic.svg)'\"\n (dragstart)=\"userIsDraggingFile = true\"\n (dragend)=\"userIsDraggingFile = false\"\n (dragover)=\"draggingOver = true\"\n (dragleave)=\"draggingOver = false\"\n (ondrop)=\"onDrop($event)\"\n>\n <!-- <ng-container *ngIf=\"draggingOver\"></ng-container> -->\n <!-- Grid mode -->\n <ng-scrollbar\n *ngIf=\"viewMode == 'grid'\"\n class=\"grid {{gridSize}} {{config.imageSize || 'normal'}}\"\n [class.selectionMode]=\"config.mode == 'focusFiles'\"\n [class.showDropArea]=\"draggingOver\"\n style=\"height: 100%; width: 100%\"\n track=\"vertical\"\n pointerEventsMethod=\"scrollbar\"\n [ngx-contextmenu]=\"folderContextMenu\"\n >\n <div class=\"resize-observer\" #fileViewport></div>\n <cdk-virtual-scroll-viewport\n itemSize=\"134\"\n scrollViewport\n (click)=\"selection = []; selectionText = ''\"\n >\n <div class=\"row\" *cdkVirtualFor=\"let row of sortedFolders\">\n <div class=\"file\"\n *ngFor=\"let item of row\"\n [class.selected]=\"selection.includes(item)\"\n [class.generic]=\"item['_icon'].needsBackdrop\"\n [ngx-contextmenu]=\"fileContextMenu\"\n [ngx-menu-context]=\"item\"\n >\n <mat-checkbox\n #checkbox\n *ngIf=\"config.mode == 'focusFiles' && item.kind == 'file'\"\n [checked]=\"item['_value']\"\n (change)=\"onToggle(item, $event)\"\n />\n <div\n style=\"display: contents\"\n (click)=\"onSelect(item, $event)\"\n (dblclick)=\"onItemClick(item)\"\n (dragstart)=\"onDragStart($event, item)\"\n >\n <img [src]=\"item['_icon'].path\"/>\n <p>{{item['vanityName'] || item.name}}</p>\n </div>\n </div>\n </div>\n </cdk-virtual-scroll-viewport>\n </ng-scrollbar>\n\n <!-- List mode -->\n <div\n *ngIf=\"viewMode == 'list'\"\n style=\"width: 100%; height: 100%\"\n [class.showDropArea]=\"draggingOver\"\n [ngx-contextmenu]=\"folderContextMenu\"\n >\n <app-tabulator\n [dataSource]=\"directoryContents\"\n [columns]=\"[\n { field: 'name', title: 'Name', formatter: nameCellFormatter },\n { field: '_size', title: 'Size' },\n { field: '_ctime', title: 'Created' },\n { field: '_mtime', title: 'Modified' },\n ]\"\n [options]=\"{\n rowHeight: 32\n }\"\n (rowClick)=\"onRowClick($event)\"\n (rowDblClick)=\"onItemClick($event.data)\"\n (rowContext)=\"onRowCtx($event)\"\n />\n </div>\n</div>\n\n<div class=\"select-hint\" *ngIf=\"selectionText\">\n {{selectionText}}\n</div>\n\n", styles: [":host{display:block;height:100%;width:100%;overflow:hidden}.resize-observer{position:absolute;inset:0}.grid .row{flex:1;display:grid;grid-template-columns:repeat(auto-fill,80px);justify-content:space-between;grid-gap:20px;grid-auto-rows:min-content;padding:10px;margin-right:10px}.grid .file{width:80px;z-index:1;position:relative;transition:opacity 50ms ease-in-out;display:flex;flex-direction:column;align-items:center;text-align:center}.grid .file.selected p{background-color:#8ad9d9;color:#000}.grid .file.generic:before{content:\" \";position:absolute;background:var(--filemanager-fileicon-backdrop);background-repeat:no-repeat;width:100%;height:100%;z-index:-1;left:0}.grid .file.generic img{height:44px;width:44px;margin-top:28px;margin-bottom:8px}.grid .file img{-webkit-user-select:none;user-select:none}.grid .file p{height:50px;width:100%;margin:0;padding:2px;font-size:14px;line-height:16px;border-radius:4px;overflow:hidden;display:-webkit-box;-webkit-line-clamp:3;-webkit-box-orient:vertical;text-overflow:ellipsis;overflow-wrap:break-word;transition:background-color 50ms ease-in-out,color 50ms ease-in-out}.grid .file mat-checkbox{position:absolute;left:-32px;top:-16px}:host ::ng-deep .small .file{width:46px}:host ::ng-deep .grid.selectionMode .row{padding-left:30px}:host ::ng-deep .grid.selectionMode .cdk-virtual-scroll-content-wrapper{padding-top:10px}:host ::ng-deep .tabulator .tabulator-row .tabulator-cell:not(:first-of-type){padding-top:9px}.list .row{flex:1;display:flex;align-items:center;padding:5px 10px;height:42px}.list .row p{margin:0}.list .row:hover{background-color:var(--filemanager-row-hover-background-color, #343434)}.list .row.selected p{background-color:#8ad9d9;color:#000;border-radius:5px;padding:0 4px}.list .row.odd{background-color:var(--filemanager-row-alt-background-color, #323232)}.select-hint{position:absolute;font-size:14px;bottom:0;right:0;background-color:#000a;padding:2px 10px;border:1px solid black;border-top-right-radius:5px;border-bottom:none;border-right:none;z-index:9999}.select-hint:hover{display:none}.showDropArea{border:4px dashed rgba(18,18,18,.8);transition:border-color .25s ease}:host ::ng-deep ng-scrollbar.ng-scrollbar{box-sizing:border-box!important;--scrollbar-thumb-color: #666;min-height:200px;min-width:200px}mat-progress-bar{--mdc-linear-progress-track-height: 2px;position:absolute;top:0;left:0;width:100%;animation:dropIn .2s ease;transition:position .2s ease;z-index:100}mat-progress-bar.hide{top:-2px}@keyframes dropIn{0%{top:-2px}to{top:0}}\n"] }]
|
|
626
|
+
}], ctorParameters: function () { return [{ type: i1.Fetch }, { type: i1.KeyboardService }, { type: i1.DialogService }, { type: i2.MatDialog }, { type: i3.FilemanagerComponent }, { type: i0.ChangeDetectorRef }, { type: i4.DomSanitizer }]; }, propDecorators: { filesRef: [{
|
|
627
|
+
type: ViewChild,
|
|
628
|
+
args: ["fileViewport"]
|
|
629
|
+
}], tabulator: [{
|
|
630
|
+
type: ViewChild,
|
|
631
|
+
args: [TabulatorComponent]
|
|
632
|
+
}], renameTemplate: [{
|
|
633
|
+
type: ViewChild,
|
|
634
|
+
args: ['renameTemplate', { read: TemplateRef }]
|
|
635
|
+
}], path: [{
|
|
636
|
+
type: Input
|
|
637
|
+
}], pathChange: [{
|
|
638
|
+
type: Output
|
|
639
|
+
}], config: [{
|
|
640
|
+
type: Input
|
|
641
|
+
}], showHiddenFiles: [{
|
|
642
|
+
type: Input
|
|
643
|
+
}], viewMode: [{
|
|
644
|
+
type: Input
|
|
645
|
+
}], gridSize: [{
|
|
646
|
+
type: Input
|
|
647
|
+
}], tab: [{
|
|
648
|
+
type: Input
|
|
649
|
+
}], fileSelect: [{
|
|
650
|
+
type: Output
|
|
651
|
+
}], fileDblClick: [{
|
|
652
|
+
type: Output
|
|
653
|
+
}], folderSelect: [{
|
|
654
|
+
type: Output
|
|
655
|
+
}], folderDblClick: [{
|
|
656
|
+
type: Output
|
|
657
|
+
}], newTab: [{
|
|
658
|
+
type: Output
|
|
659
|
+
}], loadFiles: [{
|
|
660
|
+
type: Output
|
|
661
|
+
}], selection: [{
|
|
662
|
+
type: Input
|
|
663
|
+
}], selectionChange: [{
|
|
664
|
+
type: Output
|
|
665
|
+
}], value: [{
|
|
666
|
+
type: Input
|
|
667
|
+
}], valueChange: [{
|
|
668
|
+
type: Output
|
|
669
|
+
}], sortOrder: [{
|
|
670
|
+
type: Input
|
|
671
|
+
}] } });
|
|
672
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-grid.component.js","sourceRoot":"","sources":["../../../../../../packages/common/src/components/filemanager/file-grid/file-grid.component.ts","../../../../../../packages/common/src/components/filemanager/file-grid/file-grid.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, EventEmitter, Output, Input, ViewChild, ElementRef, Optional, Inject, TemplateRef, ChangeDetectorRef, SecurityContext } from '@angular/core';\nimport { MatTabsModule } from '@angular/material/tabs';\nimport { MatCheckboxChange, MatCheckboxModule } from '@angular/material/checkbox';\nimport { MatInputModule } from '@angular/material/input';\nimport { DatePipe, NgForOf, NgIf } from '@angular/common';\nimport { ScrollingModule } from '@angular/cdk/scrolling';\nimport { MatDialog, MatDialogRef } from '@angular/material/dialog';\nimport { CellComponent, EmptyCallback } from 'tabulator-tables';\nimport { NgScrollbarModule } from 'ngx-scrollbar';\nimport { DirectoryDescriptor, FileDescriptor, FilemanagerComponent, FileViewTab, FSDescriptor, NgxFileManagerConfiguration } from '../filemanager.component';\nimport { FileSorting } from '../../types';\nimport { IconResolver } from '../icon-resolver';\nimport { TabulatorComponent } from '../../tabulator/tabulator.component';\nimport { MatProgressBarModule } from '@angular/material/progress-bar';\nimport { uploadFile } from '../helpers';\nimport { DialogService, Fetch, KeyboardService, MenuDirective, MenuItem, openMenu } from '../../../public-api';\nimport { DomSanitizer } from '@angular/platform-browser';\n\nconst itemWidth = (80 + 20);\n\n@Component({\n    selector: 'app-file-grid',\n    templateUrl: './file-grid.component.html',\n    styleUrls: ['./file-grid.component.scss'],\n    imports: [\n        NgIf,\n        NgForOf,\n        DatePipe,\n        MatTabsModule,\n        NgScrollbarModule,\n        MatInputModule,\n        MatCheckboxModule,\n        MatProgressBarModule,\n        TabulatorComponent,\n        MenuDirective,\n        ScrollingModule\n    ],\n    standalone: true\n})\nexport class FileGridComponent implements OnInit {\n    @ViewChild(\"fileViewport\") filesRef: ElementRef;\n    @ViewChild(TabulatorComponent) tabulator: TabulatorComponent;\n\n    @ViewChild('renameTemplate', { read: TemplateRef }) renameTemplate: TemplateRef<any>;\n\n\n    private _path: string;\n    @Input() set path(value: string) {\n        if (!value) return;\n\n        if (this._path && this.config.navigateOnlyToDescendants) {\n            if (!value.startsWith('/'))\n                value = '/' + value;\n            if (!value.startsWith(this.config.path))\n                return;\n        }\n\n        let prev = this._path;\n\n        this._path = value;\n\n        if (prev != value) {\n            this.pathChange.next(this.path);\n            if (this.config.apiSettings)\n                this.loadFolder()\n        }\n    }\n    get path() {return this._path}\n    @Output() pathChange = new EventEmitter<string>();\n\n    @Input() config: NgxFileManagerConfiguration = {};\n    @Input() showHiddenFiles = false;\n    @Input() viewMode: \"list\" | \"grid\" = \"grid\";\n    @Input() gridSize: \"small\" | \"normal\" | \"large\" = \"normal\";\n    @Input() tab: FileViewTab;\n\n    @Output() fileSelect = new EventEmitter<FileDescriptor>();\n    @Output() fileDblClick = new EventEmitter<FileDescriptor>();\n    @Output() folderSelect = new EventEmitter<DirectoryDescriptor>();\n    @Output() folderDblClick = new EventEmitter<DirectoryDescriptor>();\n    @Output() newTab = new EventEmitter<{ path: string; }>();\n    @Output() loadFiles = new EventEmitter<FSDescriptor[]>();\n\n    directoryContents: FSDescriptor[] = [];\n    @Input() selection: FSDescriptor[] = [];\n    @Output() selectionChange = new EventEmitter<FSDescriptor[]>();\n\n    @Input() value: FSDescriptor[] = [];\n    @Output() valueChange = new EventEmitter<FSDescriptor[]>();\n\n    selectionText: string;\n\n    sortedFolders: any[][] = [];\n\n    private readonly sorters = {\n        \"a-z\": (a: FileDescriptor, b: FileDescriptor) => a.name > b.name ? 1 : -1,\n        \"z-a\": (a: FileDescriptor, b: FileDescriptor) => b.name > a.name ? 1 : -1,\n        \"lastmod\": (a: FileDescriptor, b: FileDescriptor) => b.stats.mtimeMs - a.stats.mtimeMs,\n        \"firstmod\": (a: FileDescriptor, b: FileDescriptor) => a.stats.mtimeMs - b.stats.mtimeMs,\n        \"size\": (a: FileDescriptor, b: FileDescriptor) => b.stats.size - a.stats.size,\n        \"type\": (a: FileDescriptor, b: FileDescriptor) => a.path.split('.').splice(-1, 1)[0] > b.path.split('.').splice(-1, 1)[0] ? 1 : -1\n    };\n    @Input() sortOrder: FileSorting = \"a-z\";\n\n    itemsPerRow = 6;\n\n    // If the current directory is inside of an archive\n    isArchive = true;\n\n    userIsDraggingFile = false;\n    draggingOver = false;\n\n    showLoader = false;\n    hideLoader = false;\n\n    readonly columns = [\n        { id: \"name\", label: \"Name\" },\n        { id: \"size\", label: \"Size\"},\n        { id: \"type\", label: \"Type\"},\n        { id: \"owner\", label: \"Owner\"},\n        { id: \"group\", label: \"Group\"},\n        { id: \"permissions\", label: \"Permissions\"},\n        { id: \"location\", label: \"Location\"},\n        { id: \"modified\", label: \"Modified\"},\n        { id: \"modified--time\", label: \"Modified - Time\"},\n        { id: \"accessed\", label: \"Accessed\"},\n        { id: \"created\", label: \"Created\"},\n        { id: \"recency\", label: \"Recency\"},\n        { id: \"star\", label: \"Star\"},\n        { id: \"detailed-type\", label: \"Detailed Type\"},\n    ];\n\n    cols = [\n        { id: \"name\", label: \"Name\" },\n        { id: \"size\", label: \"Size\" },\n        { id: \"modified\", label: \"Modified\" },\n        { id: \"star\", label: \"Star\" }\n    ];\n\n    folderContextMenu: MenuItem<FSDescriptor>[] = [\n        {\n            label: \"New Folder\",\n            // shortcutLabel: \"Shift+Ctrl+N\",\n            icon: \"create_new_folder\",\n            action: (data) => {\n                // console.log(\"New folder goodness\");\n                // console.log(data);\n                this.dialog.open(\"folder-rename\", \"@dotglitch/ngx-web-components\", {\n                    inputs: { path: data?.path || this.path, name: data?.name || '', config: this.config }\n                }).then(r => this.loadFolder())\n            }\n        },\n        {\n            label: \"Upload file\",\n            // shortcutLabel: \"Ctrl+D\",\n            icon: \"file_upload\",\n            action: (evt) => uploadFile(this.fetch, this.config, this._path, evt ? (evt.path + evt.name) : null, this.fileManager.contextTags).then(res => {\n                this.loadFolder();\n            })\n        },\n        \"separator\",\n        // {\n        //     isDisabled: (data) => true,\n        //     label: \"_P_aste\",\n        //     icon: \"content_paste\",\n        //     action: (evt) => {\n        //     }\n        // },\n        {\n            label: \"Select _A_ll\",\n            shortcutLabel: \"Ctrl+A\",\n            icon: \"select_all\",\n            action: (evt) => {\n                this.selection = this._sortFilter();\n                this.selectionText = this.getSelectionText();\n                this.selectionChange.next(this.selection);\n            }\n        },\n        // \"separator\",\n        // {\n        //     label: \"P_r_operties\",\n        //     icon: \"find_in_page\",\n        //     action: (evt) => {\n\n        //     }\n        // }\n    ];\n\n    fileContextMenu: MenuItem<FSDescriptor>[] = [];\n\n    performChecksum(path, digest) {\n        // this.windowManager.openWindow({\n        //     appId: \"checksum\",\n        //     data: { digest, path },\n        //     workspace: this.windowRef.workspace,\n        //     width: 600,\n        //     height: 250\n        // });\n    }\n\n    iconResolver: IconResolver;\n\n    get libConfig() { return this.fileManager.libConfig }\n    constructor(\n        private readonly fetch: Fetch,\n        private readonly keyboard: KeyboardService,\n        private readonly dialog: DialogService,\n        private readonly matDialog: MatDialog,\n        private readonly fileManager: FilemanagerComponent,\n        private readonly changeDetector: ChangeDetectorRef,\n        private readonly sanitizer: DomSanitizer\n    ) {\n\n\n        this.iconResolver = new IconResolver(this.libConfig.assetPath);\n\n        // ctrl + a => select all\n        keyboard.onKeyCommand({\n            key: \"a\",\n            ctrl: true,\n        }).subscribe(evt => {\n            this.selection = this._sortFilter();\n            this.selectionText = this.getSelectionText();\n            this.selectionChange.next(this.selection);\n        });\n\n        // ctrl + c => copy file names to clipboard\n        keyboard.onKeyCommand({\n            key: \"c\",\n            ctrl: true,\n        }).subscribe(evt => {\n\n        });\n\n        // ctrl + h => toggle hidden files\n        keyboard.onKeyCommand({\n            key: \"h\",\n            ctrl: true,\n            interrupt: true\n        }).subscribe(evt => {\n            this.showHiddenFiles = !this.showHiddenFiles;\n        });\n\n        // F2 => Rename selected files\n        keyboard.onKeyCommand({\n            key: \"f2\",\n        }).subscribe(evt => {\n            // Rename selected file(s)\n        });\n\n        // Enter => Open selected files\n        keyboard.onKeyCommand({\n            key: \"Enter\",\n        }).subscribe(evt => {\n            const files = this.directoryContents.filter(dc => this.selection.find(i => i.name == dc.name));\n            // this.windowManager.openFiles(files as any);\n        });\n\n        // Delete => delete selected files\n        keyboard.onKeyCommand({\n            key: \"delete\",\n        }).subscribe(evt => {\n            const files = this.directoryContents.filter(dc => this.selection.find(i => i.name == dc.name));\n        });\n    }\n\n    async ngOnInit() {\n        // this.loadFolder();\n    }\n\n    ngAfterViewInit() {\n        this.fileContextMenu = [\n            {\n                label: \"Download\",\n                icon: \"download\",\n                action: (file) => {\n                    let target = `${window.origin}${this.config.apiSettings.downloadEntryUrl}`;\n                    let path = file.path + file.name;\n\n                    if (file.kind == \"directory\" && !path.endsWith('/'))\n                        path += \"/\";\n\n                    target += `${target.includes('?') ? '&' : '?'}path=${path}&ngsw-bypass=true`;\n                    console.log(target);\n                    // window.open(target);\n                    var link = document.createElement(\"a\");\n                    link.download = file.name;\n                    link.href = target;\n                    link.click();\n                    link.remove();\n                    this.fileManager.fileDownload.next(file);\n                }\n            },\n            {\n                label: \"Open in new Tab\",\n                icon: \"open_in_new\",\n                isVisible: (data) => data.kind == \"directory\",\n                action: (data) => {\n                    this.fileManager.initTab(data.path + data.name);\n                }\n            },\n            // {\n            //     label: \"Open with Application...\",\n            //     isVisible: (data) => data.kind == \"file\",\n            //     shortcutLabel: \"Ctrl+D\",\n            //     action: (evt) => {\n\n            //     },\n            // },\n            \"separator\",\n            // {\n            //     label: \"Cut\",\n            //     icon: \"content_cut\",\n            //     isDisabled: data => true,\n            //     action: (evt) => {\n            //     },\n            // },\n            // {\n            //     label: \"Copy\",\n            //     icon: \"file_copy\",\n            //     isDisabled: data => true,\n            //     childrenResolver: () => new Promise(r => setTimeout(r, 500000))\n            // },\n            // {\n            //     label: \"Move To...\",\n            //     icon: \"drive_file_move\",\n            //     shortcutLabel: \"Ctrl+A\",\n            //     action: (evt) => {\n\n            //     },\n            // },\n            // {\n            //     label: \"Copy To...\",\n            //     icon: \"folder_copy\",\n            //     shortcutLabel: \"Ctrl+A\",\n            //     action: (evt) => {\n\n            //     },\n            // },\n            {\n                label: \"Delete\",\n                icon: \"delete\",\n                // shortcutLabel: \"Del\",\n                isVisible: data => !data.path.includes(\"#/\"), // omit files in compressed dirs\n                action: (evt) => {\n                    const path = evt.path + evt.name;\n\n                    const url = this.config.apiSettings.deleteEntryUrlTemplate\n                        ? this.config.apiSettings.deleteEntryUrlTemplate(path)\n                        : this.config.apiSettings.deleteEntryUrl\n\n                    this.fetch.delete(url)\n                        .then(() => this.loadFolder())\n                },\n            },\n            // {\n            //     label: \"Shred file\",\n            //     icon: \"delete_forever\",\n            //     isVisible: data => !data.path.includes(\"#/\"), // omit files in compressed dirs\n            //     action: (evt) => {\n            //         this.fetch.post(`/api/filesystem/delete?wipe=true`, { files: [evt.path + evt.name]})\n            //             .then(() => this.loadFolder())\n            //     },\n            // },\n            {\n                label: \"Rename\",\n                icon: \"drive_file_rename_outline\",\n                isVisible: data => !data.path.includes(\"#/\"), // omit files in compressed dirs\n                // shortcutLabel: \"F2\",\n                action: (data) => {\n                    this.dialog.open(\"folder-rename\", \"@dotglitch/ngx-web-components\", {\n                        inputs: { path: data?.path || this.path, name: data?.name || '', config: this.config }\n                    }).then(r => this.loadFolder())\n                }\n            },\n\n            // Extract Here\n            // Extract To...\n            // {\n            //     label: \"Extract Here\",\n            //     icon: \"folder_zip\",\n            //     shortcutLabel: \"Ctrl+A\",\n            //     isDisabled: (data) => !(data.kind == \"file\" && data.ext != \".zip\" && isArchive(data)),\n            //     action: (evt) => {\n            //         // TODO\n            //     },\n            // },\n            // {\n            //     label: \"Extract to...\",\n            //     icon: \"folder_zip\",\n            //     shortcutLabel: \"Ctrl+A\",\n            //     isDisabled: (data) => !(data.kind == \"file\" && data.ext != \".zip\" && isArchive(data)),\n            //     action: (evt) => {\n            //         // TODO\n            //     },\n            // },\n            // {\n            //     label: \"Compress...\",\n            //     icon: \"folder_zip\",\n            //     shortcutLabel: \"Ctrl+A\",\n            //     isDisabled: (data) => data.kind == \"file\",\n            //     action: (evt) => {\n            //         // TODO\n            //     },\n            // },\n            {\n                label: \"Checksum\",\n                icon: \"manage_search\",\n                isDisabled: (data) => data.kind != \"file\",\n                children: [\n                    {\n                        label: \"MD5\",\n                        action: (evt) => this.performChecksum(evt.path + evt.name, \"md5\"),\n                    },\n                    {\n                        label: \"SHA1\",\n                        action: (evt) => this.performChecksum(evt.path + evt.name, \"sha1\"),\n                    },\n                    {\n                        label: \"SHA256\",\n                        action: (evt) => this.performChecksum(evt.path + evt.name, \"sha256\"),\n                    },\n                    {\n                        label: \"SHA512\",\n                        action: (evt) => this.performChecksum(evt.path + evt.name, \"sha512\"),\n                    },\n                ],\n                isVisible: (data) => {\n                    return false;\n                    return !this.isArchive || data.kind == \"file\";\n                },\n            },\n            // {\n            //     label: \"Star\",\n            //     icon: \"star\",\n            //     shortcutLabel: \"Ctrl+A\",\n            //     action: (evt) => {\n\n            //     },\n            // },\n            // \"separator\",\n            // {\n            //     label: \"P_r_operties\",\n            //     icon: \"find_in_page\",\n            //     action: (evt) => {\n\n            //     },\n            // }\n        ];\n    }\n\n    async loadFolder() {\n        this.showLoader = true;\n        this.hideLoader = false;\n\n        const url = this.config.apiSettings.listEntriesUrlTemplate\n            ? this.config.apiSettings.listEntriesUrlTemplate(this.path)\n            : this.config.apiSettings.listEntriesUrl\n\n        this.fetch.post(url, { path: this.path, showHidden: this.showHiddenFiles })\n            .then((data: any) => {\n                const files: FileDescriptor[] = data?.files || [];\n                const dirs: DirectoryDescriptor[] = data?.dirs || [];\n                const descriptors = files.concat(dirs as any) as FSDescriptor[];\n\n                descriptors.forEach(f => {\n                    f['_icon'] = this.iconResolver.resolveIcon(f);\n                    if (f.kind == \"file\") {\n                        f['_ctime'] = new Date(f.stats?.ctimeMs)?.toLocaleString();\n                        f['_mtime'] = new Date(f.stats?.mtimeMs)?.toLocaleString();\n                        f['_size'] = this.bytesToString(f.stats?.size);\n                    }\n                });\n\n                this.directoryContents = descriptors;\n\n                this._sortFilter();\n                this.resize();\n                this.loadFiles.next(descriptors);\n\n                if (this.sortedFolders.length > 0)\n                    this.flowRows();\n\n                setTimeout(() => this.resize(), 250);\n                setTimeout(() => this.resize(), 500);\n                setTimeout(() => this.resize(), 1000);\n                setTimeout(() => this.resize(), 2500);\n                setTimeout(() => this.resize(), 5000);\n            })\n            .catch(e => console.error(e))\n            .finally(() => {\n                this.hideLoader = true;\n                setTimeout(() => {\n                    this.showLoader = false;\n                }, 200);\n            })\n    }\n\n    flowRows() {\n        let filtered = this._sortFilter();\n\n        this.sortedFolders = [];\n        const num = Math.ceil(filtered.length / this.itemsPerRow);\n        const iterations = Math.min(num, 100);\n\n        for (let row = 0; row < iterations; row++) {\n            if (!this.sortedFolders[row])\n                this.sortedFolders[row] = [];\n\n            for (let i = row * this.itemsPerRow; i < (row + 1) * this.itemsPerRow && i < filtered.length; i++) {\n                this.sortedFolders[row].push(filtered[i]);\n            }\n        }\n    }\n\n    onSelect(item: FSDescriptor, evt) {\n        evt.stopPropagation();\n\n        if (this.keyboard.isShiftPressed) {\n            let start = this.directoryContents.findIndex(i => i.name == this.selection.slice(-1, 1)[0].name);\n            let end = this.directoryContents.indexOf(item);\n\n            if (start == -1)\n                start = end;\n\n            let items = start > end\n                ? this.directoryContents.slice(end, start + 1)\n                : this.directoryContents.slice(start, end + 1);\n\n            this.selection = items;\n        }\n        else if (this.keyboard.isCtrlPressed) {\n            if (!this.selection.includes(item))\n                this.selection.push(item);\n            else // Case that we selected the same item twice\n                this.selection.splice(this.selection.indexOf(item), 1);\n        }\n        else\n            this.selection = [item];\n\n        if (this.selection.length == 1) {\n            if (this.selection[0].kind == \"directory\")\n                this.folderSelect.next(this.selection[0]);\n            else\n                this.fileSelect.next(this.selection[0]);\n        }\n\n        this.selectionChange.next(this.selection);\n        this.selectionText = this.getSelectionText();\n    }\n\n    onItemClick(file: FSDescriptor) {\n        console.log(file, this);\n        if (file.kind == \"directory\") {\n            this.folderDblClick.next(file);\n            this.path = file.path + file.name;\n        }\n        // else if (file.ext == \"zip\") {\n        //     this.fileDblClick.next(file);\n        //     this.path = file.path + file.name;\n        // }\n        else {\n            this.fileDblClick.next(file);\n            this.fileSelect.next(file);\n        }\n    }\n\n    onToggle(item, state: MatCheckboxChange) {\n        item['_value'] = state.checked;\n\n        // TODO: What causes this to be null when initialized with an array?\n        if (!this.value) {\n            this.value = [];\n        }\n\n        if (state.checked) {\n            this.value.push(item);\n        }\n        else {\n            const i = this.value.findIndex(v => v == item);\n            if (i >= 0)\n                this.value.splice(i, 1);\n        }\n        this.valueChange.next(this.value);\n    }\n\n    async clearSelection() {\n        this.value = [];\n        this.valueChange.next(this.value);\n\n        this.tabulator?.table?.getRows().forEach(r => r.getElement().classList.remove('selected'));\n    }\n\n    private _sortFilter(): FileDescriptor[] {\n        return this.directoryContents = this.directoryContents?.filter(d => d.kind == 'directory')\n            .concat(this.directoryContents?.filter(d => d.kind == 'file')\n                .sort(this.sorters[this.sortOrder])\n            ) as FileDescriptor[];\n    }\n\n    private getSelectionText() {\n        const dirCount = this.selection.filter(s => s.kind == \"directory\").length;\n        const fileCount = this.selection.filter(s => s.kind == \"file\").length;\n\n        if (dirCount + fileCount == 0) return \"\";\n\n        const totalSize =\n            this.directoryContents\n                .filter(d => d.kind == \"file\")\n                .filter(d => this.selection?.find(i => i.name == d.name))\n                .map(d => d['stats'].size).reduce((a, b) => a + b, 0);\n\n        if (dirCount + fileCount == 1)\n            return `\"${this.selection[0].name}\" selected (${this.bytesToString(totalSize)})`;\n\n        if (dirCount > 0 && fileCount == 0)\n            return `\"${dirCount}\" folders selected`;\n        if (fileCount > 0 && dirCount == 0)\n            return `${fileCount} items selected (${this.bytesToString(totalSize)})`;\n\n        return `${dirCount} folder${dirCount == 1 ? \"\" : \"s\"} selected, ${fileCount} other item${fileCount == 1 ? \"\" : \"s\"} selected (${this.bytesToString(totalSize)})`;\n    }\n\n    bytesToString(bytes: number, decimals = 2) {\n        if (!+bytes) return '0 Bytes';\n\n        const k = 1024;\n        const dm = decimals < 0 ? 0 : decimals;\n        const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB'];\n\n        const i = Math.floor(Math.log(bytes) / Math.log(k));\n\n        return `${parseFloat((bytes / Math.pow(k, i)).toFixed(dm))} ${sizes[i]}`;\n    }\n\n    public resize() {\n        if (!this.filesRef) {\n            setTimeout(() => this.resize(), 25);\n            return;\n        };\n\n        const bounds = (this.filesRef.nativeElement as HTMLElement).getBoundingClientRect();\n\n        const newColCount = Math.floor(bounds.width / itemWidth);\n        if (newColCount != this.itemsPerRow) {\n            this.itemsPerRow = Math.floor(bounds.width / itemWidth);\n            if (this.itemsPerRow > 100)\n                this.itemsPerRow = 1;\n\n            this.flowRows();\n        }\n\n        if (this.sortedFolders?.length == 0)\n            this.flowRows();\n    }\n\n    onDragStart(evt: DragEvent, item: FSDescriptor) {\n        const target = `${window.origin}/api/filesystem/download?dir=${item.path}&file=${item.name}`;\n\n        evt.dataTransfer.clearData();\n        // evt.dataTransfer.setData('text/uri-list', target);\n        // evt.dataTransfer.setData('DownloadURL', `text/uri-list:${target}`);\n        evt.dataTransfer.setData('text/plain', item.name);\n    }\n\n    onDrop(ev) {\n        ev.preventDefault();\n\n        if (ev.dataTransfer.items) {\n            // Use DataTransferItemList interface to access the file(s)\n            [...ev.dataTransfer.items].forEach((item, i) => {\n            // If dropped items aren't files, reject them\n            if (item.kind === \"file\") {\n                const file = item.getAsFile();\n                console.log(`… file[${i}].name = ${file.name}`);\n            }\n            });\n        } else {\n            // Use DataTransfer interface to access the file(s)\n            [...ev.dataTransfer.files].forEach((file, i) => {\n            console.log(`… file[${i}].name = ${file.name}`);\n            });\n        }\n    }\n\n    nameCellFormatter = ((cell: CellComponent, formatterParams: {}, onRendered: EmptyCallback) => {\n        // TODO: Sanitize?\n        const item = cell.getData() as FSDescriptor;\n        return `\n            <span style=\"display: flex; align-items: center\">\n                <img style=\"height: 24px; margin-right: 6px\" src=\"${item['_icon'].path}\"/>\n                <p style=\"margin: 0\">${this.sanitizer.sanitize(SecurityContext.HTML, item['vanityName'] || item.name)}</p>\n            </span>\n        `;\n    }).bind(this)\n\n    onRowCtx({event, row}) {\n        openMenu(this.matDialog, this.fileContextMenu, row.getData(), event);\n    }\n\n    onRowClick({event, row, data}) {\n        // $event.data['_value'] = $event.data['_value'] == true ? false : true\n        // console.log(event, row, data, this.value);\n\n        const rowEl = row.getElement();\n        let state = rowEl.classList.contains('selected');\n        data['_value'] = !state;\n\n        if (!this.value) {\n            this.value = [];\n        }\n\n        if (!state) {\n            rowEl.classList.add('selected');\n            this.value.push(data);\n        }\n        else {\n            rowEl.classList.remove('selected');\n            const i = this.value.findIndex(v => v == data);\n            if (i >= 0)\n                this.value.splice(i, 1);\n        }\n\n        this.valueChange.next(this.value);\n    }\n\n    sort() {\n        this._sortFilter();\n    }\n}\n\n\n\n\n\n","<ng-container *ngIf=\"showLoader\">\n    <mat-progress-bar [class.hide]=\"hideLoader\" mode=\"query\"/>\n</ng-container>\n\n<div\n    style=\"display: contents\"\n    [style.--filemanager-fileicon-backdrop]=\"'url(' + iconResolver.path + '/pop/generic.svg)'\"\n    (dragstart)=\"userIsDraggingFile = true\"\n    (dragend)=\"userIsDraggingFile = false\"\n    (dragover)=\"draggingOver = true\"\n    (dragleave)=\"draggingOver = false\"\n    (ondrop)=\"onDrop($event)\"\n>\n    <!-- <ng-container *ngIf=\"draggingOver\"></ng-container> -->\n    <!-- Grid mode -->\n    <ng-scrollbar\n        *ngIf=\"viewMode == 'grid'\"\n        class=\"grid {{gridSize}} {{config.imageSize || 'normal'}}\"\n        [class.selectionMode]=\"config.mode == 'focusFiles'\"\n        [class.showDropArea]=\"draggingOver\"\n        style=\"height: 100%; width: 100%\"\n        track=\"vertical\"\n        pointerEventsMethod=\"scrollbar\"\n        [ngx-contextmenu]=\"folderContextMenu\"\n    >\n        <div class=\"resize-observer\" #fileViewport></div>\n        <cdk-virtual-scroll-viewport\n            itemSize=\"134\"\n            scrollViewport\n            (click)=\"selection = []; selectionText = ''\"\n        >\n            <div class=\"row\" *cdkVirtualFor=\"let row of sortedFolders\">\n                <div class=\"file\"\n                    *ngFor=\"let item of row\"\n                    [class.selected]=\"selection.includes(item)\"\n                    [class.generic]=\"item['_icon'].needsBackdrop\"\n                    [ngx-contextmenu]=\"fileContextMenu\"\n                    [ngx-menu-context]=\"item\"\n                >\n                    <mat-checkbox\n                        #checkbox\n                        *ngIf=\"config.mode == 'focusFiles' && item.kind == 'file'\"\n                        [checked]=\"item['_value']\"\n                        (change)=\"onToggle(item, $event)\"\n                    />\n                    <div\n                        style=\"display: contents\"\n                        (click)=\"onSelect(item, $event)\"\n                        (dblclick)=\"onItemClick(item)\"\n                        (dragstart)=\"onDragStart($event, item)\"\n                    >\n                        <img [src]=\"item['_icon'].path\"/>\n                        <p>{{item['vanityName'] || item.name}}</p>\n                    </div>\n                </div>\n            </div>\n        </cdk-virtual-scroll-viewport>\n    </ng-scrollbar>\n\n    <!-- List mode -->\n    <div\n        *ngIf=\"viewMode == 'list'\"\n        style=\"width: 100%; height: 100%\"\n        [class.showDropArea]=\"draggingOver\"\n        [ngx-contextmenu]=\"folderContextMenu\"\n    >\n        <app-tabulator\n            [dataSource]=\"directoryContents\"\n            [columns]=\"[\n                { field: 'name', title: 'Name', formatter: nameCellFormatter },\n                { field: '_size', title: 'Size' },\n                { field: '_ctime', title: 'Created' },\n                { field: '_mtime', title: 'Modified' },\n            ]\"\n            [options]=\"{\n                rowHeight: 32\n            }\"\n            (rowClick)=\"onRowClick($event)\"\n            (rowDblClick)=\"onItemClick($event.data)\"\n            (rowContext)=\"onRowCtx($event)\"\n        />\n    </div>\n</div>\n\n<div class=\"select-hint\" *ngIf=\"selectionText\">\n    {{selectionText}}\n</div>\n\n"]}
|