@dotglitch/ngx-common 1.0.21 → 1.0.23

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (61) hide show
  1. package/assets/mat-icons.d.ts +5705 -0
  2. package/components/filemanager/file-grid/file-grid.component.d.ts +97 -0
  3. package/components/filemanager/filemanager.component.d.ts +206 -0
  4. package/components/filemanager/folder-rename/folder-rename.component.d.ts +18 -0
  5. package/components/filemanager/helpers.d.ts +5 -0
  6. package/components/filemanager/icon-resolver.d.ts +13 -0
  7. package/components/filemanager/textextensions.d.ts +3 -0
  8. package/components/filemanager/toolbar/breadcrumb/breadcrumb.component.d.ts +16 -0
  9. package/components/filemanager/toolbar/icon-button/icon-button.component.d.ts +11 -0
  10. package/components/filemanager/toolbar/toolbar.component.d.ts +30 -0
  11. package/components/filemanager/tree-view/tree-view.component.d.ts +21 -0
  12. package/components/tabulator/tabulator.component.d.ts +30 -0
  13. package/components/types.d.ts +16 -0
  14. package/components/vscode/ts-type-resolver/dependency-parser.d.ts +6 -0
  15. package/components/vscode/ts-type-resolver/dummy-source-cache.d.ts +7 -0
  16. package/components/vscode/ts-type-resolver/import-resolver.d.ts +28 -0
  17. package/components/vscode/ts-type-resolver/main.d.ts +22 -0
  18. package/components/vscode/ts-type-resolver/recursion-depth.d.ts +11 -0
  19. package/components/vscode/ts-type-resolver/types.d.ts +179 -0
  20. package/components/vscode/ts-type-resolver/unpkg-source-resolver.d.ts +5 -0
  21. package/components/vscode/ts-type-resolver/update-emitter.d.ts +2 -0
  22. package/components/vscode/vscode.component.d.ts +57 -0
  23. package/esm2020/assets/mat-icons.mjs +5705 -0
  24. package/esm2020/components/filemanager/file-grid/file-grid.component.mjs +672 -0
  25. package/esm2020/components/filemanager/filemanager.component.mjs +301 -0
  26. package/esm2020/components/filemanager/folder-rename/folder-rename.component.mjs +57 -0
  27. package/esm2020/components/filemanager/helpers.mjs +26 -0
  28. package/esm2020/components/filemanager/icon-resolver.mjs +155 -0
  29. package/esm2020/components/filemanager/textextensions.mjs +294 -0
  30. package/esm2020/components/filemanager/toolbar/breadcrumb/breadcrumb.component.mjs +26 -0
  31. package/esm2020/components/filemanager/toolbar/icon-button/icon-button.component.mjs +34 -0
  32. package/esm2020/components/filemanager/toolbar/toolbar.component.mjs +163 -0
  33. package/esm2020/components/filemanager/tree-view/tree-view.component.mjs +53 -0
  34. package/esm2020/components/lazy-loader/lazy-loader.module.mjs +3 -3
  35. package/esm2020/components/lazy-loader/lazy-loader.service.mjs +1 -1
  36. package/esm2020/components/menu/menu.component.mjs +3 -4
  37. package/esm2020/components/tabulator/tabulator.component.mjs +92 -0
  38. package/esm2020/components/types.mjs +3 -0
  39. package/esm2020/components/vscode/ts-type-resolver/dependency-parser.mjs +91 -0
  40. package/esm2020/components/vscode/ts-type-resolver/dummy-source-cache.mjs +15 -0
  41. package/esm2020/components/vscode/ts-type-resolver/import-resolver.mjs +311 -0
  42. package/esm2020/components/vscode/ts-type-resolver/main.mjs +112 -0
  43. package/esm2020/components/vscode/ts-type-resolver/recursion-depth.mjs +21 -0
  44. package/esm2020/components/vscode/ts-type-resolver/types.mjs +14 -0
  45. package/esm2020/components/vscode/ts-type-resolver/unpkg-source-resolver.mjs +21 -0
  46. package/esm2020/components/vscode/ts-type-resolver/update-emitter.mjs +37 -0
  47. package/esm2020/components/vscode/vscode.component.mjs +230 -0
  48. package/esm2020/directives/menu.directive.mjs +19 -18
  49. package/esm2020/public-api.mjs +6 -1
  50. package/esm2020/types/menu.mjs +1 -1
  51. package/fesm2015/dotglitch-ngx-common-folder-rename.component-d039534b.mjs +79 -0
  52. package/fesm2015/dotglitch-ngx-common-folder-rename.component-d039534b.mjs.map +1 -0
  53. package/fesm2015/dotglitch-ngx-common.mjs +8361 -28
  54. package/fesm2015/dotglitch-ngx-common.mjs.map +1 -1
  55. package/fesm2020/dotglitch-ngx-common-folder-rename.component-ba3ebd0a.mjs +78 -0
  56. package/fesm2020/dotglitch-ngx-common-folder-rename.component-ba3ebd0a.mjs.map +1 -0
  57. package/fesm2020/dotglitch-ngx-common.mjs +8330 -27
  58. package/fesm2020/dotglitch-ngx-common.mjs.map +1 -1
  59. package/package.json +6 -2
  60. package/public-api.d.ts +4 -0
  61. package/types/menu.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":"AAAA,OAAO,EAAE,SAAS,EAAU,YAAY,EAAE,MAAM,EAAE,KAAK,EAAE,SAAS,EAAgC,WAAW,EAAqB,eAAe,EAAE,MAAM,eAAe,CAAC;AACzK,OAAO,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAC;AACvD,OAAO,EAAqB,iBAAiB,EAAE,MAAM,4BAA4B,CAAC;AAClF,OAAO,EAAE,cAAc,EAAE,MAAM,yBAAyB,CAAC;AACzD,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,iBAAiB,CAAC;AAC1D,OAAO,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC;AAGzD,OAAO,EAAE,iBAAiB,EAAE,MAAM,eAAe,CAAC;AAGlD,OAAO,EAAE,YAAY,EAAE,MAAM,kBAAkB,CAAC;AAChD,OAAO,EAAE,kBAAkB,EAAE,MAAM,qCAAqC,CAAC;AACzE,OAAO,EAAE,oBAAoB,EAAE,MAAM,gCAAgC,CAAC;AACtE,OAAO,EAAE,UAAU,EAAE,MAAM,YAAY,CAAC;AACxC,OAAO,EAAyC,aAAa,EAAY,QAAQ,EAAE,MAAM,qBAAqB,CAAC;;;;;;;;;;AAG/G,MAAM,SAAS,GAAG,CAAC,EAAE,GAAG,EAAE,CAAC,CAAC;AAqB5B,MAAM,OAAO,iBAAiB;IAQ1B,IAAa,IAAI,CAAC,KAAa;QAC3B,IAAI,CAAC,KAAK;YAAE,OAAO;QAEnB,IAAI,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,MAAM,CAAC,yBAAyB,EAAE;YACrD,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,GAAG,CAAC;gBACtB,KAAK,GAAG,GAAG,GAAG,KAAK,CAAC;YACxB,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC;gBACnC,OAAO;SACd;QAED,IAAI,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC;QAEtB,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QAEnB,IAAI,IAAI,IAAI,KAAK,EAAE;YACf,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAChC,IAAI,IAAI,CAAC,MAAM,CAAC,WAAW;gBACvB,IAAI,CAAC,UAAU,EAAE,CAAA;SACxB;IACL,CAAC;IACD,IAAI,IAAI,KAAI,OAAO,IAAI,CAAC,KAAK,CAAA,CAAA,CAAC;IA2H9B,eAAe,CAAC,IAAI,EAAE,MAAM;QACxB,kCAAkC;QAClC,yBAAyB;QACzB,8BAA8B;QAC9B,2CAA2C;QAC3C,kBAAkB;QAClB,kBAAkB;QAClB,MAAM;IACV,CAAC;IAID,IAAI,SAAS,KAAK,OAAO,IAAI,CAAC,WAAW,CAAC,SAAS,CAAA,CAAC,CAAC;IACrD,YACqB,KAAY,EACZ,QAAyB,EACzB,MAAqB,EACrB,SAAoB,EACpB,WAAiC,EACjC,cAAiC,EACjC,SAAuB;QANvB,UAAK,GAAL,KAAK,CAAO;QACZ,aAAQ,GAAR,QAAQ,CAAiB;QACzB,WAAM,GAAN,MAAM,CAAe;QACrB,cAAS,GAAT,SAAS,CAAW;QACpB,gBAAW,GAAX,WAAW,CAAsB;QACjC,mBAAc,GAAd,cAAc,CAAmB;QACjC,cAAS,GAAT,SAAS,CAAc;QA9IlC,eAAU,GAAG,IAAI,YAAY,EAAU,CAAC;QAEzC,WAAM,GAAgC,EAAE,CAAC;QACzC,oBAAe,GAAG,KAAK,CAAC;QACxB,aAAQ,GAAoB,MAAM,CAAC;QACnC,aAAQ,GAAiC,QAAQ,CAAC;QAGjD,eAAU,GAAG,IAAI,YAAY,EAAkB,CAAC;QAChD,iBAAY,GAAG,IAAI,YAAY,EAAkB,CAAC;QAClD,iBAAY,GAAG,IAAI,YAAY,EAAuB,CAAC;QACvD,mBAAc,GAAG,IAAI,YAAY,EAAuB,CAAC;QACzD,WAAM,GAAG,IAAI,YAAY,EAAqB,CAAC;QAC/C,cAAS,GAAG,IAAI,YAAY,EAAkB,CAAC;QAEzD,sBAAiB,GAAmB,EAAE,CAAC;QAC9B,cAAS,GAAmB,EAAE,CAAC;QAC9B,oBAAe,GAAG,IAAI,YAAY,EAAkB,CAAC;QAEtD,UAAK,GAAmB,EAAE,CAAC;QAC1B,gBAAW,GAAG,IAAI,YAAY,EAAkB,CAAC;QAI3D,kBAAa,GAAY,EAAE,CAAC;QAEX,YAAO,GAAG;YACvB,KAAK,EAAE,CAAC,CAAiB,EAAE,CAAiB,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,GAAG,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;YACzE,KAAK,EAAE,CAAC,CAAiB,EAAE,CAAiB,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,GAAG,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;YACzE,SAAS,EAAE,CAAC,CAAiB,EAAE,CAAiB,EAAE,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,GAAG,CAAC,CAAC,KAAK,CAAC,OAAO;YACtF,UAAU,EAAE,CAAC,CAAiB,EAAE,CAAiB,EAAE,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,GAAG,CAAC,CAAC,KAAK,CAAC,OAAO;YACvF,MAAM,EAAE,CAAC,CAAiB,EAAE,CAAiB,EAAE,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,GAAG,CAAC,CAAC,KAAK,CAAC,IAAI;YAC7E,MAAM,EAAE,CAAC,CAAiB,EAAE,CAAiB,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;SACrI,CAAC;QACO,cAAS,GAAgB,KAAK,CAAC;QAExC,gBAAW,GAAG,CAAC,CAAC;QAEhB,mDAAmD;QACnD,cAAS,GAAG,IAAI,CAAC;QAEjB,uBAAkB,GAAG,KAAK,CAAC;QAC3B,iBAAY,GAAG,KAAK,CAAC;QAErB,eAAU,GAAG,KAAK,CAAC;QACnB,eAAU,GAAG,KAAK,CAAC;QAEV,YAAO,GAAG;YACf,EAAE,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE;YAC7B,EAAE,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAC;YAC5B,EAAE,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAC;YAC5B,EAAE,EAAE,EAAE,OAAO,EAAE,KAAK,EAAE,OAAO,EAAC;YAC9B,EAAE,EAAE,EAAE,OAAO,EAAE,KAAK,EAAE,OAAO,EAAC;YAC9B,EAAE,EAAE,EAAE,aAAa,EAAE,KAAK,EAAE,aAAa,EAAC;YAC1C,EAAE,EAAE,EAAE,UAAU,EAAE,KAAK,EAAE,UAAU,EAAC;YACpC,EAAE,EAAE,EAAE,UAAU,EAAE,KAAK,EAAE,UAAU,EAAC;YACpC,EAAE,EAAE,EAAE,gBAAgB,EAAE,KAAK,EAAE,iBAAiB,EAAC;YACjD,EAAE,EAAE,EAAE,UAAU,EAAE,KAAK,EAAE,UAAU,EAAC;YACpC,EAAE,EAAE,EAAE,SAAS,EAAE,KAAK,EAAE,SAAS,EAAC;YAClC,EAAE,EAAE,EAAE,SAAS,EAAE,KAAK,EAAE,SAAS,EAAC;YAClC,EAAE,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAC;YAC5B,EAAE,EAAE,EAAE,eAAe,EAAE,KAAK,EAAE,eAAe,EAAC;SACjD,CAAC;QAEF,SAAI,GAAG;YACH,EAAE,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE;YAC7B,EAAE,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE;YAC7B,EAAE,EAAE,EAAE,UAAU,EAAE,KAAK,EAAE,UAAU,EAAE;YACrC,EAAE,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE;SAChC,CAAC;QAEF,sBAAiB,GAA6B;YAC1C;gBACI,KAAK,EAAE,YAAY;gBACnB,iCAAiC;gBACjC,IAAI,EAAE,mBAAmB;gBACzB,MAAM,EAAE,CAAC,IAAI,EAAE,EAAE;oBACb,sCAAsC;oBACtC,qBAAqB;oBACrB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,eAAe,EAAE,+BAA+B,EAAE;wBAC/D,MAAM,EAAE,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,IAAI,IAAI,CAAC,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,IAAI,EAAE,EAAE,MAAM,EAAE,IAAI,CAAC,MAAM,EAAE;qBACzF,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC,CAAA;gBACnC,CAAC;aACJ;YACD;gBACI,KAAK,EAAE,aAAa;gBACpB,2BAA2B;gBAC3B,IAAI,EAAE,aAAa;gBACnB,MAAM,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,KAAK,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,IAAI,GAAG,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE;oBAC1I,IAAI,CAAC,UAAU,EAAE,CAAC;gBACtB,CAAC,CAAC;aACL;YACD,WAAW;YACX,IAAI;YACJ,kCAAkC;YAClC,wBAAwB;YACxB,6BAA6B;YAC7B,yBAAyB;YACzB,QAAQ;YACR,KAAK;YACL;gBACI,KAAK,EAAE,cAAc;gBACrB,aAAa,EAAE,QAAQ;gBACvB,IAAI,EAAE,YAAY;gBAClB,MAAM,EAAE,CAAC,GAAG,EAAE,EAAE;oBACZ,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;oBACpC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,gBAAgB,EAAE,CAAC;oBAC7C,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;gBAC9C,CAAC;aACJ;YACD,eAAe;YACf,IAAI;YACJ,6BAA6B;YAC7B,4BAA4B;YAC5B,yBAAyB;YAEzB,QAAQ;YACR,IAAI;SACP,CAAC;QAEF,oBAAe,GAA6B,EAAE,CAAC;QAif/C,sBAAiB,GAAG,CAAC,CAAC,IAAmB,EAAE,eAAmB,EAAE,UAAyB,EAAE,EAAE;YACzF,kBAAkB;YAClB,MAAM,IAAI,GAAG,IAAI,CAAC,OAAO,EAAkB,CAAC;YAC5C,OAAO;;oEAEqD,IAAI,CAAC,OAAO,CAAC,CAAC,IAAI;uCAC/C,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,eAAe,CAAC,IAAI,EAAE,IAAI,CAAC,YAAY,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC;;SAE5G,CAAC;QACN,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QAheT,IAAI,CAAC,YAAY,GAAG,IAAI,YAAY,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;QAE/D,yBAAyB;QACzB,QAAQ,CAAC,YAAY,CAAC;YAClB,GAAG,EAAE,GAAG;YACR,IAAI,EAAE,IAAI;SACb,CAAC,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE;YACf,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;YACpC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC7C,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAC9C,CAAC,CAAC,CAAC;QAEH,2CAA2C;QAC3C,QAAQ,CAAC,YAAY,CAAC;YAClB,GAAG,EAAE,GAAG;YACR,IAAI,EAAE,IAAI;SACb,CAAC,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE;QAEnB,CAAC,CAAC,CAAC;QAEH,kCAAkC;QAClC,QAAQ,CAAC,YAAY,CAAC;YAClB,GAAG,EAAE,GAAG;YACR,IAAI,EAAE,IAAI;YACV,SAAS,EAAE,IAAI;SAClB,CAAC,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE;YACf,IAAI,CAAC,eAAe,GAAG,CAAC,IAAI,CAAC,eAAe,CAAC;QACjD,CAAC,CAAC,CAAC;QAEH,8BAA8B;QAC9B,QAAQ,CAAC,YAAY,CAAC;YAClB,GAAG,EAAE,IAAI;SACZ,CAAC,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE;YACf,0BAA0B;QAC9B,CAAC,CAAC,CAAC;QAEH,+BAA+B;QAC/B,QAAQ,CAAC,YAAY,CAAC;YAClB,GAAG,EAAE,OAAO;SACf,CAAC,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE;YACf,MAAM,KAAK,GAAG,IAAI,CAAC,iBAAiB,CAAC,MAAM,CAAC,EAAE,CAAC,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,IAAI,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC;YAC/F,8CAA8C;QAClD,CAAC,CAAC,CAAC;QAEH,kCAAkC;QAClC,QAAQ,CAAC,YAAY,CAAC;YAClB,GAAG,EAAE,QAAQ;SAChB,CAAC,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE;YACf,MAAM,KAAK,GAAG,IAAI,CAAC,iBAAiB,CAAC,MAAM,CAAC,EAAE,CAAC,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,IAAI,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC;QACnG,CAAC,CAAC,CAAC;IACP,CAAC;IAED,KAAK,CAAC,QAAQ;QACV,qBAAqB;IACzB,CAAC;IAED,eAAe;QACX,IAAI,CAAC,eAAe,GAAG;YACnB;gBACI,KAAK,EAAE,UAAU;gBACjB,IAAI,EAAE,UAAU;gBAChB,MAAM,EAAE,CAAC,IAAI,EAAE,EAAE;oBACb,IAAI,MAAM,GAAG,GAAG,MAAM,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,gBAAgB,EAAE,CAAC;oBAC3E,IAAI,IAAI,GAAG,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC;oBAEjC,IAAI,IAAI,CAAC,IAAI,IAAI,WAAW,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC;wBAC/C,IAAI,IAAI,GAAG,CAAC;oBAEhB,MAAM,IAAI,GAAG,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,QAAQ,IAAI,mBAAmB,CAAC;oBAC7E,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;oBACpB,uBAAuB;oBACvB,IAAI,IAAI,GAAG,QAAQ,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC;oBACvC,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC;oBAC1B,IAAI,CAAC,IAAI,GAAG,MAAM,CAAC;oBACnB,IAAI,CAAC,KAAK,EAAE,CAAC;oBACb,IAAI,CAAC,MAAM,EAAE,CAAC;oBACd,IAAI,CAAC,WAAW,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;gBAC7C,CAAC;aACJ;YACD;gBACI,KAAK,EAAE,iBAAiB;gBACxB,IAAI,EAAE,aAAa;gBACnB,SAAS,EAAE,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,IAAI,WAAW;gBAC7C,MAAM,EAAE,CAAC,IAAI,EAAE,EAAE;oBACb,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC;gBACpD,CAAC;aACJ;YACD,IAAI;YACJ,yCAAyC;YACzC,gDAAgD;YAChD,+BAA+B;YAC/B,yBAAyB;YAEzB,SAAS;YACT,KAAK;YACL,WAAW;YACX,IAAI;YACJ,oBAAoB;YACpB,2BAA2B;YAC3B,gCAAgC;YAChC,yBAAyB;YACzB,SAAS;YACT,KAAK;YACL,IAAI;YACJ,qBAAqB;YACrB,yBAAyB;YACzB,gCAAgC;YAChC,sEAAsE;YACtE,KAAK;YACL,IAAI;YACJ,2BAA2B;YAC3B,+BAA+B;YAC/B,+BAA+B;YAC/B,yBAAyB;YAEzB,SAAS;YACT,KAAK;YACL,IAAI;YACJ,2BAA2B;YAC3B,2BAA2B;YAC3B,+BAA+B;YAC/B,yBAAyB;YAEzB,SAAS;YACT,KAAK;YACL;gBACI,KAAK,EAAE,QAAQ;gBACf,IAAI,EAAE,QAAQ;gBACd,wBAAwB;gBACxB,SAAS,EAAE,IAAI,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC;gBAC5C,MAAM,EAAE,CAAC,GAAG,EAAE,EAAE;oBACZ,MAAM,IAAI,GAAG,GAAG,CAAC,IAAI,GAAG,GAAG,CAAC,IAAI,CAAC;oBAEjC,MAAM,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,sBAAsB;wBACtD,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,sBAAsB,CAAC,IAAI,CAAC;wBACtD,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,cAAc,CAAA;oBAE5C,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC;yBACjB,IAAI,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC,CAAA;gBACtC,CAAC;aACJ;YACD,IAAI;YACJ,2BAA2B;YAC3B,8BAA8B;YAC9B,qFAAqF;YACrF,yBAAyB;YACzB,+FAA+F;YAC/F,6CAA6C;YAC7C,SAAS;YACT,KAAK;YACL;gBACI,KAAK,EAAE,QAAQ;gBACf,IAAI,EAAE,2BAA2B;gBACjC,SAAS,EAAE,IAAI,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC;gBAC5C,uBAAuB;gBACvB,MAAM,EAAE,CAAC,IAAI,EAAE,EAAE;oBACb,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,eAAe,EAAE,+BAA+B,EAAE;wBAC/D,MAAM,EAAE,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,IAAI,IAAI,CAAC,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,IAAI,EAAE,EAAE,MAAM,EAAE,IAAI,CAAC,MAAM,EAAE;qBACzF,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC,CAAA;gBACnC,CAAC;aACJ;YAED,eAAe;YACf,gBAAgB;YAChB,IAAI;YACJ,6BAA6B;YAC7B,0BAA0B;YAC1B,+BAA+B;YAC/B,6FAA6F;YAC7F,yBAAyB;YACzB,kBAAkB;YAClB,SAAS;YACT,KAAK;YACL,IAAI;YACJ,8BAA8B;YAC9B,0BAA0B;YAC1B,+BAA+B;YAC/B,6FAA6F;YAC7F,yBAAyB;YACzB,kBAAkB;YAClB,SAAS;YACT,KAAK;YACL,IAAI;YACJ,4BAA4B;YAC5B,0BAA0B;YAC1B,+BAA+B;YAC/B,iDAAiD;YACjD,yBAAyB;YACzB,kBAAkB;YAClB,SAAS;YACT,KAAK;YACL;gBACI,KAAK,EAAE,UAAU;gBACjB,IAAI,EAAE,eAAe;gBACrB,UAAU,EAAE,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,IAAI,MAAM;gBACzC,QAAQ,EAAE;oBACN;wBACI,KAAK,EAAE,KAAK;wBACZ,MAAM,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,IAAI,GAAG,GAAG,CAAC,IAAI,EAAE,KAAK,CAAC;qBACpE;oBACD;wBACI,KAAK,EAAE,MAAM;wBACb,MAAM,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,IAAI,GAAG,GAAG,CAAC,IAAI,EAAE,MAAM,CAAC;qBACrE;oBACD;wBACI,KAAK,EAAE,QAAQ;wBACf,MAAM,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,IAAI,GAAG,GAAG,CAAC,IAAI,EAAE,QAAQ,CAAC;qBACvE;oBACD;wBACI,KAAK,EAAE,QAAQ;wBACf,MAAM,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,IAAI,GAAG,GAAG,CAAC,IAAI,EAAE,QAAQ,CAAC;qBACvE;iBACJ;gBACD,SAAS,EAAE,CAAC,IAAI,EAAE,EAAE;oBAChB,OAAO,KAAK,CAAC;oBACb,OAAO,CAAC,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,IAAI,IAAI,MAAM,CAAC;gBAClD,CAAC;aACJ;YACD,IAAI;YACJ,qBAAqB;YACrB,oBAAoB;YACpB,+BAA+B;YAC/B,yBAAyB;YAEzB,SAAS;YACT,KAAK;YACL,eAAe;YACf,IAAI;YACJ,6BAA6B;YAC7B,4BAA4B;YAC5B,yBAAyB;YAEzB,SAAS;YACT,IAAI;SACP,CAAC;IACN,CAAC;IAED,KAAK,CAAC,UAAU;QACZ,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;QACvB,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC;QAExB,MAAM,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,sBAAsB;YACtD,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,sBAAsB,CAAC,IAAI,CAAC,IAAI,CAAC;YAC3D,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,cAAc,CAAA;QAE5C,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,EAAE,EAAE,IAAI,EAAE,IAAI,CAAC,IAAI,EAAE,UAAU,EAAE,IAAI,CAAC,eAAe,EAAE,CAAC;aACtE,IAAI,CAAC,CAAC,IAAS,EAAE,EAAE;YAChB,MAAM,KAAK,GAAqB,IAAI,EAAE,KAAK,IAAI,EAAE,CAAC;YAClD,MAAM,IAAI,GAA0B,IAAI,EAAE,IAAI,IAAI,EAAE,CAAC;YACrD,MAAM,WAAW,GAAG,KAAK,CAAC,MAAM,CAAC,IAAW,CAAmB,CAAC;YAEhE,WAAW,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE;gBACpB,CAAC,CAAC,OAAO,CAAC,GAAG,IAAI,CAAC,YAAY,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC;gBAC9C,IAAI,CAAC,CAAC,IAAI,IAAI,MAAM,EAAE;oBAClB,CAAC,CAAC,QAAQ,CAAC,GAAG,IAAI,IAAI,CAAC,CAAC,CAAC,KAAK,EAAE,OAAO,CAAC,EAAE,cAAc,EAAE,CAAC;oBAC3D,CAAC,CAAC,QAAQ,CAAC,GAAG,IAAI,IAAI,CAAC,CAAC,CAAC,KAAK,EAAE,OAAO,CAAC,EAAE,cAAc,EAAE,CAAC;oBAC3D,CAAC,CAAC,OAAO,CAAC,GAAG,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC;iBAClD;YACL,CAAC,CAAC,CAAC;YAEH,IAAI,CAAC,iBAAiB,GAAG,WAAW,CAAC;YAErC,IAAI,CAAC,WAAW,EAAE,CAAC;YACnB,IAAI,CAAC,MAAM,EAAE,CAAC;YACd,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YAEjC,IAAI,IAAI,CAAC,aAAa,CAAC,MAAM,GAAG,CAAC;gBAC7B,IAAI,CAAC,QAAQ,EAAE,CAAC;YAEpB,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,EAAE,EAAE,GAAG,CAAC,CAAC;YACrC,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,EAAE,EAAE,GAAG,CAAC,CAAC;YACrC,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,EAAE,EAAE,IAAI,CAAC,CAAC;YACtC,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,EAAE,EAAE,IAAI,CAAC,CAAC;YACtC,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,EAAE,EAAE,IAAI,CAAC,CAAC;QAC1C,CAAC,CAAC;aACD,KAAK,CAAC,CAAC,CAAC,EAAE,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;aAC5B,OAAO,CAAC,GAAG,EAAE;YACV,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;YACvB,UAAU,CAAC,GAAG,EAAE;gBACZ,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC;YAC5B,CAAC,EAAE,GAAG,CAAC,CAAC;QACZ,CAAC,CAAC,CAAA;IACV,CAAC;IAED,QAAQ;QACJ,IAAI,QAAQ,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;QAElC,IAAI,CAAC,aAAa,GAAG,EAAE,CAAC;QACxB,MAAM,GAAG,GAAG,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,IAAI,CAAC,WAAW,CAAC,CAAC;QAC1D,MAAM,UAAU,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC;QAEtC,KAAK,IAAI,GAAG,GAAG,CAAC,EAAE,GAAG,GAAG,UAAU,EAAE,GAAG,EAAE,EAAE;YACvC,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC;gBACxB,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC;YAEjC,KAAK,IAAI,CAAC,GAAG,GAAG,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC,GAAG,CAAC,GAAG,GAAG,CAAC,CAAC,GAAG,IAAI,CAAC,WAAW,IAAI,CAAC,GAAG,QAAQ,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;gBAC/F,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC;aAC7C;SACJ;IACL,CAAC;IAED,QAAQ,CAAC,IAAkB,EAAE,GAAG;QAC5B,GAAG,CAAC,eAAe,EAAE,CAAC;QAEtB,IAAI,IAAI,CAAC,QAAQ,CAAC,cAAc,EAAE;YAC9B,IAAI,KAAK,GAAG,IAAI,CAAC,iBAAiB,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,IAAI,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;YACjG,IAAI,GAAG,GAAG,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;YAE/C,IAAI,KAAK,IAAI,CAAC,CAAC;gBACX,KAAK,GAAG,GAAG,CAAC;YAEhB,IAAI,KAAK,GAAG,KAAK,GAAG,GAAG;gBACnB,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC,GAAG,EAAE,KAAK,GAAG,CAAC,CAAC;gBAC9C,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC,KAAK,EAAE,GAAG,GAAG,CAAC,CAAC,CAAC;YAEnD,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;SAC1B;aACI,IAAI,IAAI,CAAC,QAAQ,CAAC,aAAa,EAAE;YAClC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,IAAI,CAAC;gBAC9B,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;iBACzB,4CAA4C;gBAC7C,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,CAAC;SAC9D;;YAEG,IAAI,CAAC,SAAS,GAAG,CAAC,IAAI,CAAC,CAAC;QAE5B,IAAI,IAAI,CAAC,SAAS,CAAC,MAAM,IAAI,CAAC,EAAE;YAC5B,IAAI,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,IAAI,WAAW;gBACrC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC;;gBAE1C,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC;SAC/C;QAED,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAC1C,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,gBAAgB,EAAE,CAAC;IACjD,CAAC;IAED,WAAW,CAAC,IAAkB;QAC1B,OAAO,CAAC,GAAG,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;QACxB,IAAI,IAAI,CAAC,IAAI,IAAI,WAAW,EAAE;YAC1B,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAC/B,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC;SACrC;QACD,gCAAgC;QAChC,oCAAoC;QACpC,yCAAyC;QACzC,IAAI;aACC;YACD,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAC7B,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;SAC9B;IACL,CAAC;IAED,QAAQ,CAAC,IAAI,EAAE,KAAwB;QACnC,IAAI,CAAC,QAAQ,CAAC,GAAG,KAAK,CAAC,OAAO,CAAC;QAE/B,oEAAoE;QACpE,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE;YACb,IAAI,CAAC,KAAK,GAAG,EAAE,CAAC;SACnB;QAED,IAAI,KAAK,CAAC,OAAO,EAAE;YACf,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;SACzB;aACI;YACD,MAAM,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,IAAI,CAAC,CAAC;YAC/C,IAAI,CAAC,IAAI,CAAC;gBACN,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;SAC/B;QACD,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACtC,CAAC;IAED,KAAK,CAAC,cAAc;QAChB,IAAI,CAAC,KAAK,GAAG,EAAE,CAAC;QAChB,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAElC,IAAI,CAAC,SAAS,EAAE,KAAK,EAAE,OAAO,EAAE,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,UAAU,EAAE,CAAC,SAAS,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC;IAC/F,CAAC;IAEO,WAAW;QACf,OAAO,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,iBAAiB,EAAE,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,IAAI,WAAW,CAAC;aACrF,MAAM,CAAC,IAAI,CAAC,iBAAiB,EAAE,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,IAAI,MAAM,CAAC;aACxD,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAClB,CAAC;IAC9B,CAAC;IAEO,gBAAgB;QACpB,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,IAAI,WAAW,CAAC,CAAC,MAAM,CAAC;QAC1E,MAAM,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,IAAI,MAAM,CAAC,CAAC,MAAM,CAAC;QAEtE,IAAI,QAAQ,GAAG,SAAS,IAAI,CAAC;YAAE,OAAO,EAAE,CAAC;QAEzC,MAAM,SAAS,GACX,IAAI,CAAC,iBAAiB;aACjB,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,IAAI,MAAM,CAAC;aAC7B,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC;aACxD,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;QAE9D,IAAI,QAAQ,GAAG,SAAS,IAAI,CAAC;YACzB,OAAO,IAAI,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,eAAe,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,GAAG,CAAC;QAErF,IAAI,QAAQ,GAAG,CAAC,IAAI,SAAS,IAAI,CAAC;YAC9B,OAAO,IAAI,QAAQ,oBAAoB,CAAC;QAC5C,IAAI,SAAS,GAAG,CAAC,IAAI,QAAQ,IAAI,CAAC;YAC9B,OAAO,GAAG,SAAS,oBAAoB,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,GAAG,CAAC;QAE5E,OAAO,GAAG,QAAQ,UAAU,QAAQ,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,cAAc,SAAS,cAAc,SAAS,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,cAAc,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,GAAG,CAAC;IACrK,CAAC;IAED,aAAa,CAAC,KAAa,EAAE,QAAQ,GAAG,CAAC;QACrC,IAAI,CAAC,CAAC,KAAK;YAAE,OAAO,SAAS,CAAC;QAE9B,MAAM,CAAC,GAAG,IAAI,CAAC;QACf,MAAM,EAAE,GAAG,QAAQ,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC;QACvC,MAAM,KAAK,GAAG,CAAC,OAAO,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;QAExE,MAAM,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;QAEpD,OAAO,GAAG,UAAU,CAAC,CAAC,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,IAAI,KAAK,CAAC,CAAC,CAAC,EAAE,CAAC;IAC7E,CAAC;IAEM,MAAM;QACT,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE;YAChB,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,EAAE,EAAE,EAAE,CAAC,CAAC;YACpC,OAAO;SACV;QAAA,CAAC;QAEF,MAAM,MAAM,GAAI,IAAI,CAAC,QAAQ,CAAC,aAA6B,CAAC,qBAAqB,EAAE,CAAC;QAEpF,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,GAAG,SAAS,CAAC,CAAC;QACzD,IAAI,WAAW,IAAI,IAAI,CAAC,WAAW,EAAE;YACjC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,GAAG,SAAS,CAAC,CAAC;YACxD,IAAI,IAAI,CAAC,WAAW,GAAG,GAAG;gBACtB,IAAI,CAAC,WAAW,GAAG,CAAC,CAAC;YAEzB,IAAI,CAAC,QAAQ,EAAE,CAAC;SACnB;QAED,IAAI,IAAI,CAAC,aAAa,EAAE,MAAM,IAAI,CAAC;YAC/B,IAAI,CAAC,QAAQ,EAAE,CAAC;IACxB,CAAC;IAED,WAAW,CAAC,GAAc,EAAE,IAAkB;QAC1C,MAAM,MAAM,GAAG,GAAG,MAAM,CAAC,MAAM,gCAAgC,IAAI,CAAC,IAAI,SAAS,IAAI,CAAC,IAAI,EAAE,CAAC;QAE7F,GAAG,CAAC,YAAY,CAAC,SAAS,EAAE,CAAC;QAC7B,qDAAqD;QACrD,sEAAsE;QACtE,GAAG,CAAC,YAAY,CAAC,OAAO,CAAC,YAAY,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC;IACtD,CAAC;IAED,MAAM,CAAC,EAAE;QACL,EAAE,CAAC,cAAc,EAAE,CAAC;QAEpB,IAAI,EAAE,CAAC,YAAY,CAAC,KAAK,EAAE;YACvB,2DAA2D;YAC3D,CAAC,GAAG,EAAE,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,CAAC,EAAE,EAAE;gBAC/C,6CAA6C;gBAC7C,IAAI,IAAI,CAAC,IAAI,KAAK,MAAM,EAAE;oBACtB,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,EAAE,CAAC;oBAC9B,OAAO,CAAC,GAAG,CAAC,UAAU,CAAC,YAAY,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;iBACnD;YACD,CAAC,CAAC,CAAC;SACN;aAAM;YACH,mDAAmD;YACnD,CAAC,GAAG,EAAE,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,CAAC,EAAE,EAAE;gBAC/C,OAAO,CAAC,GAAG,CAAC,UAAU,CAAC,YAAY,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;YAChD,CAAC,CAAC,CAAC;SACN;IACL,CAAC;IAaD,QAAQ,CAAC,EAAC,KAAK,EAAE,GAAG,EAAC;QACjB,QAAQ,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,eAAe,EAAE,GAAG,CAAC,OAAO,EAAE,EAAE,KAAK,CAAC,CAAC;IACzE,CAAC;IAED,UAAU,CAAC,EAAC,KAAK,EAAE,GAAG,EAAE,IAAI,EAAC;QACzB,uEAAuE;QACvE,6CAA6C;QAE7C,MAAM,KAAK,GAAG,GAAG,CAAC,UAAU,EAAE,CAAC;QAC/B,IAAI,KAAK,GAAG,KAAK,CAAC,SAAS,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;QACjD,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,KAAK,CAAC;QAExB,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE;YACb,IAAI,CAAC,KAAK,GAAG,EAAE,CAAC;SACnB;QAED,IAAI,CAAC,KAAK,EAAE;YACR,KAAK,CAAC,SAAS,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;YAChC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;SACzB;aACI;YACD,KAAK,CAAC,SAAS,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC;YACnC,MAAM,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,IAAI,CAAC,CAAC;YAC/C,IAAI,CAAC,IAAI,CAAC;gBACN,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;SAC/B;QAED,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACtC,CAAC;IAED,IAAI;QACA,IAAI,CAAC,WAAW,EAAE,CAAC;IACvB,CAAC;;8GAjrBQ,iBAAiB;kGAAjB,iBAAiB,gqBAEf,kBAAkB,8HAEQ,WAAW,6BC3CpD,w0GAwFA,2hFD/DQ,IAAI,6FACJ,OAAO,kHAEP,aAAa,8BACb,iBAAiB,gjBACjB,cAAc,8BACd,iBAAiB,gLACjB,oBAAoB,yNACpB,kBAAkB,gMAClB,aAAa,mLACb,eAAe;2FAIV,iBAAiB;kBAnB7B,SAAS;+BACI,eAAe,WAGhB;wBACL,IAAI;wBACJ,OAAO;wBACP,QAAQ;wBACR,aAAa;wBACb,iBAAiB;wBACjB,cAAc;wBACd,iBAAiB;wBACjB,oBAAoB;wBACpB,kBAAkB;wBAClB,aAAa;wBACb,eAAe;qBAClB,cACW,IAAI;4QAGW,QAAQ;sBAAlC,SAAS;uBAAC,cAAc;gBACM,SAAS;sBAAvC,SAAS;uBAAC,kBAAkB;gBAEuB,cAAc;sBAAjE,SAAS;uBAAC,gBAAgB,EAAE,EAAE,IAAI,EAAE,WAAW,EAAE;gBAIrC,IAAI;sBAAhB,KAAK;gBAqBI,UAAU;sBAAnB,MAAM;gBAEE,MAAM;sBAAd,KAAK;gBACG,eAAe;sBAAvB,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBACG,GAAG;sBAAX,KAAK;gBAEI,UAAU;sBAAnB,MAAM;gBACG,YAAY;sBAArB,MAAM;gBACG,YAAY;sBAArB,MAAM;gBACG,cAAc;sBAAvB,MAAM;gBACG,MAAM;sBAAf,MAAM;gBACG,SAAS;sBAAlB,MAAM;gBAGE,SAAS;sBAAjB,KAAK;gBACI,eAAe;sBAAxB,MAAM;gBAEE,KAAK;sBAAb,KAAK;gBACI,WAAW;sBAApB,MAAM;gBAcE,SAAS;sBAAjB,KAAK","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"]}