@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.
- package/assets/mat-icons.d.ts +5705 -0
- package/components/filemanager/file-grid/file-grid.component.d.ts +97 -0
- package/components/filemanager/filemanager.component.d.ts +206 -0
- package/components/filemanager/folder-rename/folder-rename.component.d.ts +18 -0
- package/components/filemanager/helpers.d.ts +5 -0
- package/components/filemanager/icon-resolver.d.ts +13 -0
- package/components/filemanager/textextensions.d.ts +3 -0
- package/components/filemanager/toolbar/breadcrumb/breadcrumb.component.d.ts +16 -0
- package/components/filemanager/toolbar/icon-button/icon-button.component.d.ts +11 -0
- package/components/filemanager/toolbar/toolbar.component.d.ts +30 -0
- package/components/filemanager/tree-view/tree-view.component.d.ts +21 -0
- package/components/tabulator/tabulator.component.d.ts +30 -0
- package/components/types.d.ts +16 -0
- package/components/vscode/ts-type-resolver/dependency-parser.d.ts +6 -0
- package/components/vscode/ts-type-resolver/dummy-source-cache.d.ts +7 -0
- package/components/vscode/ts-type-resolver/import-resolver.d.ts +28 -0
- package/components/vscode/ts-type-resolver/main.d.ts +22 -0
- package/components/vscode/ts-type-resolver/recursion-depth.d.ts +11 -0
- package/components/vscode/ts-type-resolver/types.d.ts +179 -0
- package/components/vscode/ts-type-resolver/unpkg-source-resolver.d.ts +5 -0
- package/components/vscode/ts-type-resolver/update-emitter.d.ts +2 -0
- package/components/vscode/vscode.component.d.ts +57 -0
- package/esm2020/assets/mat-icons.mjs +5705 -0
- package/esm2020/components/filemanager/file-grid/file-grid.component.mjs +672 -0
- package/esm2020/components/filemanager/filemanager.component.mjs +301 -0
- package/esm2020/components/filemanager/folder-rename/folder-rename.component.mjs +57 -0
- package/esm2020/components/filemanager/helpers.mjs +26 -0
- package/esm2020/components/filemanager/icon-resolver.mjs +155 -0
- package/esm2020/components/filemanager/textextensions.mjs +294 -0
- package/esm2020/components/filemanager/toolbar/breadcrumb/breadcrumb.component.mjs +26 -0
- package/esm2020/components/filemanager/toolbar/icon-button/icon-button.component.mjs +34 -0
- package/esm2020/components/filemanager/toolbar/toolbar.component.mjs +163 -0
- package/esm2020/components/filemanager/tree-view/tree-view.component.mjs +53 -0
- package/esm2020/components/lazy-loader/lazy-loader.module.mjs +3 -3
- package/esm2020/components/lazy-loader/lazy-loader.service.mjs +1 -1
- package/esm2020/components/menu/menu.component.mjs +3 -4
- package/esm2020/components/tabulator/tabulator.component.mjs +92 -0
- package/esm2020/components/types.mjs +3 -0
- package/esm2020/components/vscode/ts-type-resolver/dependency-parser.mjs +91 -0
- package/esm2020/components/vscode/ts-type-resolver/dummy-source-cache.mjs +15 -0
- package/esm2020/components/vscode/ts-type-resolver/import-resolver.mjs +311 -0
- package/esm2020/components/vscode/ts-type-resolver/main.mjs +112 -0
- package/esm2020/components/vscode/ts-type-resolver/recursion-depth.mjs +21 -0
- package/esm2020/components/vscode/ts-type-resolver/types.mjs +14 -0
- package/esm2020/components/vscode/ts-type-resolver/unpkg-source-resolver.mjs +21 -0
- package/esm2020/components/vscode/ts-type-resolver/update-emitter.mjs +37 -0
- package/esm2020/components/vscode/vscode.component.mjs +230 -0
- package/esm2020/directives/menu.directive.mjs +19 -18
- package/esm2020/public-api.mjs +6 -1
- package/esm2020/types/menu.mjs +1 -1
- package/fesm2015/dotglitch-ngx-common-folder-rename.component-d039534b.mjs +79 -0
- package/fesm2015/dotglitch-ngx-common-folder-rename.component-d039534b.mjs.map +1 -0
- package/fesm2015/dotglitch-ngx-common.mjs +8361 -28
- package/fesm2015/dotglitch-ngx-common.mjs.map +1 -1
- package/fesm2020/dotglitch-ngx-common-folder-rename.component-ba3ebd0a.mjs +78 -0
- package/fesm2020/dotglitch-ngx-common-folder-rename.component-ba3ebd0a.mjs.map +1 -0
- package/fesm2020/dotglitch-ngx-common.mjs +8330 -27
- package/fesm2020/dotglitch-ngx-common.mjs.map +1 -1
- package/package.json +6 -2
- package/public-api.d.ts +4 -0
- package/types/menu.d.ts +4 -0
|
@@ -0,0 +1,301 @@
|
|
|
1
|
+
import { Component, EventEmitter, Inject, Input, Optional, Output, ViewChild, ViewChildren } from '@angular/core';
|
|
2
|
+
import { NgForOf, NgIf } from '@angular/common';
|
|
3
|
+
import { MatTabsModule } from '@angular/material/tabs';
|
|
4
|
+
import { MatIconModule } from '@angular/material/icon';
|
|
5
|
+
import { MatDrawerContainer, MatSidenavModule } from '@angular/material/sidenav';
|
|
6
|
+
import { AngularSplitModule } from 'angular-split';
|
|
7
|
+
import { FileGridComponent } from './file-grid/file-grid.component';
|
|
8
|
+
import { ToolbarComponent } from './toolbar/toolbar.component';
|
|
9
|
+
import { TreeViewComponent } from './tree-view/tree-view.component';
|
|
10
|
+
import { NGX_WEB_COMPONENTS_CONFIG } from '../types';
|
|
11
|
+
import { IconResolver } from './icon-resolver';
|
|
12
|
+
import * as i0 from "@angular/core";
|
|
13
|
+
import * as i1 from "../lazy-loader/lazy-loader.service";
|
|
14
|
+
import * as i2 from "../../public-api";
|
|
15
|
+
import * as i3 from "angular-split";
|
|
16
|
+
import * as i4 from "@angular/material/tabs";
|
|
17
|
+
import * as i5 from "@angular/material/icon";
|
|
18
|
+
import * as i6 from "@angular/material/sidenav";
|
|
19
|
+
export class FilemanagerComponent {
|
|
20
|
+
get currentFileGrid() { return this.fileGrids[this.tabIndex]; }
|
|
21
|
+
constructor(libConfig = {}, lazyLoader, viewContainer, fetch) {
|
|
22
|
+
this.libConfig = libConfig;
|
|
23
|
+
this.lazyLoader = lazyLoader;
|
|
24
|
+
this.viewContainer = viewContainer;
|
|
25
|
+
this.fetch = fetch;
|
|
26
|
+
this.config = {
|
|
27
|
+
apiSettings: {
|
|
28
|
+
listEntriesUrl: `/api/filesystem/`,
|
|
29
|
+
uploadEntryUrl: `/api/filesystem/`,
|
|
30
|
+
downloadEntryUrl: `/api/filesystem/`,
|
|
31
|
+
deleteEntryUrl: `/api/filesystem/`,
|
|
32
|
+
createDirectoryUrl: `/api/filesystem/folder`,
|
|
33
|
+
renameEntryUrl: `/api/filesystem/rename`
|
|
34
|
+
}
|
|
35
|
+
};
|
|
36
|
+
this.gridSize = "normal";
|
|
37
|
+
this.value = [];
|
|
38
|
+
this.valueChange = new EventEmitter();
|
|
39
|
+
this.gridValues = [];
|
|
40
|
+
/**
|
|
41
|
+
* Emits when focused files change.
|
|
42
|
+
* Only available in `focusFiles` mode.
|
|
43
|
+
*/
|
|
44
|
+
this.focusedFilesChange = new EventEmitter();
|
|
45
|
+
/**
|
|
46
|
+
* Emits when a file is uploaded.
|
|
47
|
+
*/
|
|
48
|
+
this.fileUpload = new EventEmitter();
|
|
49
|
+
/**
|
|
50
|
+
* Emits when a file is downloaded.
|
|
51
|
+
*/
|
|
52
|
+
this.fileDownload = new EventEmitter();
|
|
53
|
+
this.fileRename = new EventEmitter();
|
|
54
|
+
this.fileDelete = new EventEmitter();
|
|
55
|
+
this.fileCopy = new EventEmitter();
|
|
56
|
+
this.filePaste = new EventEmitter();
|
|
57
|
+
this.fileSelect = new EventEmitter();
|
|
58
|
+
this.fileDblClick = new EventEmitter();
|
|
59
|
+
this.folderSelect = new EventEmitter();
|
|
60
|
+
this.folderDblClick = new EventEmitter();
|
|
61
|
+
/**
|
|
62
|
+
* Emits when multiple file selections change.
|
|
63
|
+
*/
|
|
64
|
+
this.filesSelect = new EventEmitter();
|
|
65
|
+
/**
|
|
66
|
+
* Emits when any selection changes, single or multiple files.
|
|
67
|
+
*/
|
|
68
|
+
// @Output() selectionChange = new EventEmitter<FSDescriptor[]>();
|
|
69
|
+
this.showHiddenFiles = false;
|
|
70
|
+
this.showSidebar = true;
|
|
71
|
+
this.sidebarOverlay = false;
|
|
72
|
+
this.width = 0;
|
|
73
|
+
this.isHomeAncestor = false;
|
|
74
|
+
this.currentTab = {};
|
|
75
|
+
this.tabIndex = 0;
|
|
76
|
+
this.tabs = [];
|
|
77
|
+
lazyLoader.registerComponent({
|
|
78
|
+
id: "folder-rename",
|
|
79
|
+
group: "@dotglitch/ngx-web-components",
|
|
80
|
+
load: () => import('./folder-rename/folder-rename.component')
|
|
81
|
+
});
|
|
82
|
+
this.iconResolver = new IconResolver(libConfig.assetPath);
|
|
83
|
+
}
|
|
84
|
+
ngOnInit() {
|
|
85
|
+
this.initTab(this.config.path);
|
|
86
|
+
this.currentTab = this.tabs[0];
|
|
87
|
+
}
|
|
88
|
+
ngAfterViewInit() {
|
|
89
|
+
this.onResize();
|
|
90
|
+
setTimeout(() => this.onResize(), 250);
|
|
91
|
+
}
|
|
92
|
+
onTreeViewLoadChildren({ item, cb }) {
|
|
93
|
+
const url = this.config.apiSettings.listEntriesUrlTemplate
|
|
94
|
+
? this.config.apiSettings.listEntriesUrlTemplate(item.path + item.name)
|
|
95
|
+
: this.config.apiSettings.listEntriesUrl;
|
|
96
|
+
this.fetch.post(url, { path: item.path + item.name, showHidden: this.showHiddenFiles })
|
|
97
|
+
.then((data) => {
|
|
98
|
+
const dirs = data.dirs;
|
|
99
|
+
cb(dirs);
|
|
100
|
+
});
|
|
101
|
+
}
|
|
102
|
+
initTab(path) {
|
|
103
|
+
this.tabs.push(this.currentTab = {
|
|
104
|
+
id: crypto.randomUUID(),
|
|
105
|
+
label: this.getTabLabel(path),
|
|
106
|
+
breadcrumb: this.calcBreadcrumb(path),
|
|
107
|
+
path,
|
|
108
|
+
selection: [],
|
|
109
|
+
viewMode: this.mode || 'grid',
|
|
110
|
+
historyIndex: 0,
|
|
111
|
+
history: [],
|
|
112
|
+
sidebarItems: [],
|
|
113
|
+
sortOrder: 'a-z'
|
|
114
|
+
});
|
|
115
|
+
this.tabIndex = this.tabs.length;
|
|
116
|
+
}
|
|
117
|
+
closeTab(tab) {
|
|
118
|
+
this.tabs.splice(this.tabs.findIndex(t => t.id == tab.id), 1);
|
|
119
|
+
}
|
|
120
|
+
calcBreadcrumb(path) {
|
|
121
|
+
if (!path)
|
|
122
|
+
return null;
|
|
123
|
+
path = path.replace("#/", '/');
|
|
124
|
+
// If we're acting like we're in a changed root, we wipe out
|
|
125
|
+
// breadcrumbs below the root
|
|
126
|
+
if (this.config.chrootPath) {
|
|
127
|
+
path = path.replace(this.config.chrootPath, '');
|
|
128
|
+
const parts = path.split('/');
|
|
129
|
+
path = path.replace(/^\//, this.config.chrootPath);
|
|
130
|
+
return parts.map((p, i) => {
|
|
131
|
+
const path = parts.slice(0, i + 1).join('/');
|
|
132
|
+
return {
|
|
133
|
+
id: (this.config.chrootPath + (path || '/')).replace(/\/+/g, '/'),
|
|
134
|
+
label: p || ""
|
|
135
|
+
};
|
|
136
|
+
});
|
|
137
|
+
}
|
|
138
|
+
else {
|
|
139
|
+
const parts = path.split('/');
|
|
140
|
+
return parts.map((p, i) => {
|
|
141
|
+
const path = parts.slice(0, i + 1).join('/');
|
|
142
|
+
return {
|
|
143
|
+
id: path || '/',
|
|
144
|
+
label: p || ""
|
|
145
|
+
};
|
|
146
|
+
});
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
onBreadcrumbClick(crumb) {
|
|
150
|
+
if (crumb.id) {
|
|
151
|
+
this.currentTab.path = crumb.id;
|
|
152
|
+
this.currentTab.breadcrumb = this.calcBreadcrumb(crumb.id);
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
onTabPathChange(tab) {
|
|
156
|
+
tab.label = this.getTabLabel(tab.path);
|
|
157
|
+
tab.breadcrumb = this.calcBreadcrumb(tab.path);
|
|
158
|
+
tab.historyIndex++;
|
|
159
|
+
tab.history.push(tab.path);
|
|
160
|
+
tab.history.splice(typeof this.config.maxHistoryLength == 'number' ? this.config.maxHistoryLength : 50);
|
|
161
|
+
}
|
|
162
|
+
onTreeViewSelect(item) {
|
|
163
|
+
this.currentTab.path = item.path + item.name;
|
|
164
|
+
}
|
|
165
|
+
onTabLoadFiles(tab, files) {
|
|
166
|
+
if (tab.sidebarItems.length == 0) {
|
|
167
|
+
tab.sidebarItems = files.filter(f => f.kind == "directory");
|
|
168
|
+
return;
|
|
169
|
+
}
|
|
170
|
+
const currentItems = tab.sidebarItems;
|
|
171
|
+
const dirItems = files.filter(f => f.kind == "directory");
|
|
172
|
+
function recurse(items) {
|
|
173
|
+
return items.find(i => tab.path?.startsWith(i.path));
|
|
174
|
+
}
|
|
175
|
+
const target = recurse(currentItems);
|
|
176
|
+
if (target)
|
|
177
|
+
target['_children'] = dirItems;
|
|
178
|
+
tab.sidebarItems = currentItems;
|
|
179
|
+
}
|
|
180
|
+
onGridValueChange() {
|
|
181
|
+
this.value = this.gridValues.flat(1);
|
|
182
|
+
this.valueChange.emit(this.value);
|
|
183
|
+
}
|
|
184
|
+
getTabLabel(path) {
|
|
185
|
+
return path?.split('/').filter(p => p).pop();
|
|
186
|
+
}
|
|
187
|
+
async onResize() {
|
|
188
|
+
// Trigger re-calculation of the view
|
|
189
|
+
this.fileGrids.forEach(g => g.resize());
|
|
190
|
+
const el = this.viewContainer.element.nativeElement;
|
|
191
|
+
const bounds = el.getBoundingClientRect();
|
|
192
|
+
this.width = bounds.width;
|
|
193
|
+
// If the view area is less than 650px wide, use overlay the sidebar panel
|
|
194
|
+
this.sidebarOverlay = bounds.width < 650;
|
|
195
|
+
if (this.sidebarOverlay == false && [...this.drawer._drawers][0].opened) {
|
|
196
|
+
this.drawer.close();
|
|
197
|
+
}
|
|
198
|
+
}
|
|
199
|
+
async onResizeEnd() {
|
|
200
|
+
this.onResize();
|
|
201
|
+
setTimeout(() => this.onResize(), 250);
|
|
202
|
+
}
|
|
203
|
+
getSelection() {
|
|
204
|
+
if (this.currentTab.viewMode == "grid") {
|
|
205
|
+
return this.currentTab.selection;
|
|
206
|
+
}
|
|
207
|
+
else {
|
|
208
|
+
return this.value;
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
clearSelection() {
|
|
212
|
+
this.fileGrids.forEach(g => g.clearSelection());
|
|
213
|
+
}
|
|
214
|
+
// Tell the child grid to refresh it's sorting
|
|
215
|
+
refreshSorting() {
|
|
216
|
+
this.fileGrids.forEach(g => g.sort());
|
|
217
|
+
}
|
|
218
|
+
refreshData() {
|
|
219
|
+
this.fileGrids.forEach(g => g.loadFolder());
|
|
220
|
+
}
|
|
221
|
+
getFileData(file) {
|
|
222
|
+
let url = this.config.apiSettings.renameEntryUrlTemplate
|
|
223
|
+
? this.config.apiSettings.renameEntryUrlTemplate(file.path + file.name)
|
|
224
|
+
: this.config.apiSettings.renameEntryUrl;
|
|
225
|
+
url = (url.includes('?') ? '&' : '?') + `dir=${file.path}&file=${file.name}`;
|
|
226
|
+
return url;
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
FilemanagerComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FilemanagerComponent, deps: [{ token: NGX_WEB_COMPONENTS_CONFIG, optional: true }, { token: i1.LazyLoaderService }, { token: i0.ViewContainerRef }, { token: i2.Fetch }], target: i0.ɵɵFactoryTarget.Component });
|
|
230
|
+
FilemanagerComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: FilemanagerComponent, isStandalone: true, selector: "app-filemanager", inputs: { config: "config", gridSize: "gridSize", mode: "mode", contextTags: "contextTags", value: "value" }, outputs: { valueChange: "valueChange", focusedFilesChange: "focusedFilesChange", fileUpload: "fileUpload", fileDownload: "fileDownload", fileRename: "fileRename", fileDelete: "fileDelete", fileCopy: "fileCopy", filePaste: "filePaste", fileSelect: "fileSelect", fileDblClick: "fileDblClick", folderSelect: "folderSelect", folderDblClick: "folderDblClick", filesSelect: "filesSelect" }, viewQueries: [{ propertyName: "tabGroup", first: true, predicate: ["tabGroup"], descendants: true }, { propertyName: "treeView", first: true, predicate: TreeViewComponent, descendants: true }, { propertyName: "toolbar", first: true, predicate: ToolbarComponent, descendants: true }, { propertyName: "drawer", first: true, predicate: MatDrawerContainer, descendants: true }, { propertyName: "fileGrids", predicate: FileGridComponent, descendants: true }], ngImport: i0, template: "<app-toolbar\n [currentTab]=\"currentTab\"\n [(showHiddenFiles)]=\"showHiddenFiles\"\n [(showSidebar)]=\"showSidebar\"\n [config]=\"config\"\n [showBareMinimum]=\"width < 500 && config.showBreadcrumb != false\"\n (onBreadcrumbClick)=\"onBreadcrumbClick($event)\"\n/>\n\n<mat-drawer-container>\n <mat-drawer #drawer mode=\"over\">\n <app-toolbar\n *ngIf=\"width < 500 && config.showBreadcrumb != false\"\n [currentTab]=\"currentTab\"\n [(showHiddenFiles)]=\"showHiddenFiles\"\n [(showSidebar)]=\"showSidebar\"\n [config]=\"config\"\n [drawerMode]=\"true\"\n (onBreadcrumbClick)=\"onBreadcrumbClick($event)\"\n />\n <app-tree-view\n [data]=\"currentTab.sidebarItems\"\n (loadChildren)=\"onTreeViewLoadChildren($event)\"\n (click)=\"onTreeViewSelect($event)\"\n />\n </mat-drawer>\n <mat-drawer-content>\n <as-split direction=\"horizontal\" [gutterSize]=\"sidebarOverlay ? 0 : 2\" [unit]=\"'pixel'\" (dragEnd)=\"onResizeEnd()\">\n <as-split-area\n style=\"background-color: var(--filemanager-sidebar-background-color, #2c2c2c)\"\n [size]=\"215\"\n [minSize]=\"215\"\n [maxSize]=\"width/4\"\n [visible]=\"!sidebarOverlay && config.showTreeview != false\"\n >\n <div class=\"sidebar\"\n *ngIf=\"!sidebarOverlay && config.showTreeview != false\"\n (click)=\"currentTab.selection = []\"\n >\n <app-tree-view\n [data]=\"currentTab.sidebarItems\"\n (loadChildren)=\"onTreeViewLoadChildren($event)\"\n (click)=\"onTreeViewSelect($event)\"\n />\n </div>\n </as-split-area>\n <as-split-area style=\"background-color: var(--filemanager-background-color, #272727);\">\n <div class=\"content\"\n [style.--tab-height]=\"tabs.length > 1 ? '36px' : '0'\"\n >\n <mat-tab-group #tabGroup (selectedTabChange)=\"currentTab = tabs[$event.index]\" [(selectedIndex)]=\"tabIndex\">\n <mat-tab *ngFor=\"let tab of tabs; let i = index\">\n <ng-template mat-tab-label>\n {{tab.label}}\n <mat-icon\n class=\"close\"\n (click)=\"closeTab(tab)\">\n close\n </mat-icon>\n </ng-template>\n\n <ng-template matTabContent>\n <app-file-grid\n [viewMode]=\"tab.viewMode\"\n [tab]=\"tab\"\n [config]=\"config\"\n [gridSize]=\"gridSize\"\n [sortOrder]=\"tab.sortOrder\"\n [(selection)]=\"tab.selection\"\n [(path)]=\"tab.path\"\n (pathChange)=\"onTabPathChange(tab)\"\n [(value)]=\"gridValues[i]\"\n (valueChange)=\"onGridValueChange()\"\n (loadFiles)=\"onTabLoadFiles(tab, $event)\"\n\n (fileSelect)=\"fileSelect.next($event)\"\n (fileDblClick)=\"fileDblClick.next($event)\"\n (folderSelect)=\"folderSelect.next($event)\"\n (folderDblClick)=\"folderDblClick.next($event)\"\n />\n </ng-template>\n </mat-tab>\n </mat-tab-group>\n </div>\n </as-split-area>\n </as-split>\n </mat-drawer-content>\n</mat-drawer-container>\n\n", styles: [":host{display:flex;flex-direction:column;height:100%;width:100%}:host .content{display:flex;height:100%;overflow:hidden}:host ::ng-deep .as-split-gutter{background-color:var(--gutter-color, #121212)!important}:host ::ng-deep .mat-mdc-tab-header{height:var(--tab-height, 0);transition:height .25s ease}:host ::ng-deep .mat-mdc-tab-header .mdc-tab{height:var(--tab-height, 0);background-color:var(--filemanager-tab-background-color, #303030)}:host ::ng-deep .mat-mdc-tab .mdc-tab__content{position:static}:host ::ng-deep .mat-mdc-tab:not(.mat-mdc-tab-disabled).mdc-tab--active .mdc-tab__text-label,:host ::ng-deep .mat-mdc-tab-link:not(.mat-mdc-tab-disabled).mdc-tab--active .mdc-tab__text-label{color:var(--text-subdued-color)}.sidebar{border-right:1px solid transparent;transition:background-color .2s cubic-bezier(0,0,.2,1),border-color .2s ease}.sidebar ul{list-style-type:none;text-align:left;margin:0;padding:0;display:flex;flex-direction:column}.sidebar ul li{display:flex;align-items:center;transition:all .2s ease;font-size:14px;padding:6px 6px 6px 24px}.sidebar ul li mat-icon{margin-right:8px}.sidebar ul li:hover{background-color:#343434}.sidebar ul .tooltip{background-color:#101010cc;border:1px solid #414141;opacity:0;pointer-events:none;transition:all .2s ease}.active .sidebar{background-color:#2c2c2c;border-right-color:#1c1c1c}.active .sidebar li:hover{background-color:#343434}.close{position:absolute;right:12px;color:#666;transition:color .25s ease}.close:hover{color:#ccc}\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: AngularSplitModule }, { kind: "component", type: i3.SplitComponent, selector: "as-split", inputs: ["direction", "unit", "gutterSize", "gutterStep", "restrictMove", "useTransition", "disabled", "dir", "gutterDblClickDuration", "gutterClickDeltaPx", "gutterAriaLabel"], outputs: ["transitionEnd", "dragStart", "dragEnd", "gutterClick", "gutterDblClick"], exportAs: ["asSplit"] }, { kind: "directive", type: i3.SplitAreaDirective, selector: "as-split-area, [as-split-area]", inputs: ["order", "size", "minSize", "maxSize", "lockSize", "visible"], exportAs: ["asSplitArea"] }, { kind: "component", type: FileGridComponent, selector: "app-file-grid", inputs: ["path", "config", "showHiddenFiles", "viewMode", "gridSize", "tab", "selection", "value", "sortOrder"], outputs: ["pathChange", "fileSelect", "fileDblClick", "folderSelect", "folderDblClick", "newTab", "loadFiles", "selectionChange", "valueChange"] }, { kind: "ngmodule", type: MatTabsModule }, { kind: "directive", type: i4.MatTabContent, selector: "[matTabContent]" }, { kind: "directive", type: i4.MatTabLabel, selector: "[mat-tab-label], [matTabLabel]" }, { kind: "component", type: i4.MatTab, selector: "mat-tab", inputs: ["disabled"], exportAs: ["matTab"] }, { kind: "component", type: i4.MatTabGroup, selector: "mat-tab-group", inputs: ["color", "disableRipple", "fitInkBarToContent", "mat-stretch-tabs"], exportAs: ["matTabGroup"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i5.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: MatSidenavModule }, { kind: "component", type: i6.MatDrawer, selector: "mat-drawer", inputs: ["position", "mode", "disableClose", "autoFocus", "opened"], outputs: ["openedChange", "opened", "openedStart", "closed", "closedStart", "positionChanged"], exportAs: ["matDrawer"] }, { kind: "component", type: i6.MatDrawerContainer, selector: "mat-drawer-container", inputs: ["autosize", "hasBackdrop"], outputs: ["backdropClick"], exportAs: ["matDrawerContainer"] }, { kind: "component", type: i6.MatDrawerContent, selector: "mat-drawer-content" }, { kind: "component", type: ToolbarComponent, selector: "app-toolbar", inputs: ["config", "currentTab", "showHiddenFiles", "showSidebar", "drawerMode", "showBareMinimum"], outputs: ["onBreadcrumbClick", "showHiddenFilesChange", "showSidebarChange"] }, { kind: "component", type: TreeViewComponent, selector: "app-tree-view", inputs: ["rowTemplate", "data"], outputs: ["click", "loadChildren"] }] });
|
|
231
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FilemanagerComponent, decorators: [{
|
|
232
|
+
type: Component,
|
|
233
|
+
args: [{ selector: 'app-filemanager', imports: [
|
|
234
|
+
NgIf,
|
|
235
|
+
NgForOf,
|
|
236
|
+
AngularSplitModule,
|
|
237
|
+
FileGridComponent,
|
|
238
|
+
MatTabsModule,
|
|
239
|
+
MatIconModule,
|
|
240
|
+
MatSidenavModule,
|
|
241
|
+
ToolbarComponent,
|
|
242
|
+
TreeViewComponent
|
|
243
|
+
], standalone: true, template: "<app-toolbar\n [currentTab]=\"currentTab\"\n [(showHiddenFiles)]=\"showHiddenFiles\"\n [(showSidebar)]=\"showSidebar\"\n [config]=\"config\"\n [showBareMinimum]=\"width < 500 && config.showBreadcrumb != false\"\n (onBreadcrumbClick)=\"onBreadcrumbClick($event)\"\n/>\n\n<mat-drawer-container>\n <mat-drawer #drawer mode=\"over\">\n <app-toolbar\n *ngIf=\"width < 500 && config.showBreadcrumb != false\"\n [currentTab]=\"currentTab\"\n [(showHiddenFiles)]=\"showHiddenFiles\"\n [(showSidebar)]=\"showSidebar\"\n [config]=\"config\"\n [drawerMode]=\"true\"\n (onBreadcrumbClick)=\"onBreadcrumbClick($event)\"\n />\n <app-tree-view\n [data]=\"currentTab.sidebarItems\"\n (loadChildren)=\"onTreeViewLoadChildren($event)\"\n (click)=\"onTreeViewSelect($event)\"\n />\n </mat-drawer>\n <mat-drawer-content>\n <as-split direction=\"horizontal\" [gutterSize]=\"sidebarOverlay ? 0 : 2\" [unit]=\"'pixel'\" (dragEnd)=\"onResizeEnd()\">\n <as-split-area\n style=\"background-color: var(--filemanager-sidebar-background-color, #2c2c2c)\"\n [size]=\"215\"\n [minSize]=\"215\"\n [maxSize]=\"width/4\"\n [visible]=\"!sidebarOverlay && config.showTreeview != false\"\n >\n <div class=\"sidebar\"\n *ngIf=\"!sidebarOverlay && config.showTreeview != false\"\n (click)=\"currentTab.selection = []\"\n >\n <app-tree-view\n [data]=\"currentTab.sidebarItems\"\n (loadChildren)=\"onTreeViewLoadChildren($event)\"\n (click)=\"onTreeViewSelect($event)\"\n />\n </div>\n </as-split-area>\n <as-split-area style=\"background-color: var(--filemanager-background-color, #272727);\">\n <div class=\"content\"\n [style.--tab-height]=\"tabs.length > 1 ? '36px' : '0'\"\n >\n <mat-tab-group #tabGroup (selectedTabChange)=\"currentTab = tabs[$event.index]\" [(selectedIndex)]=\"tabIndex\">\n <mat-tab *ngFor=\"let tab of tabs; let i = index\">\n <ng-template mat-tab-label>\n {{tab.label}}\n <mat-icon\n class=\"close\"\n (click)=\"closeTab(tab)\">\n close\n </mat-icon>\n </ng-template>\n\n <ng-template matTabContent>\n <app-file-grid\n [viewMode]=\"tab.viewMode\"\n [tab]=\"tab\"\n [config]=\"config\"\n [gridSize]=\"gridSize\"\n [sortOrder]=\"tab.sortOrder\"\n [(selection)]=\"tab.selection\"\n [(path)]=\"tab.path\"\n (pathChange)=\"onTabPathChange(tab)\"\n [(value)]=\"gridValues[i]\"\n (valueChange)=\"onGridValueChange()\"\n (loadFiles)=\"onTabLoadFiles(tab, $event)\"\n\n (fileSelect)=\"fileSelect.next($event)\"\n (fileDblClick)=\"fileDblClick.next($event)\"\n (folderSelect)=\"folderSelect.next($event)\"\n (folderDblClick)=\"folderDblClick.next($event)\"\n />\n </ng-template>\n </mat-tab>\n </mat-tab-group>\n </div>\n </as-split-area>\n </as-split>\n </mat-drawer-content>\n</mat-drawer-container>\n\n", styles: [":host{display:flex;flex-direction:column;height:100%;width:100%}:host .content{display:flex;height:100%;overflow:hidden}:host ::ng-deep .as-split-gutter{background-color:var(--gutter-color, #121212)!important}:host ::ng-deep .mat-mdc-tab-header{height:var(--tab-height, 0);transition:height .25s ease}:host ::ng-deep .mat-mdc-tab-header .mdc-tab{height:var(--tab-height, 0);background-color:var(--filemanager-tab-background-color, #303030)}:host ::ng-deep .mat-mdc-tab .mdc-tab__content{position:static}:host ::ng-deep .mat-mdc-tab:not(.mat-mdc-tab-disabled).mdc-tab--active .mdc-tab__text-label,:host ::ng-deep .mat-mdc-tab-link:not(.mat-mdc-tab-disabled).mdc-tab--active .mdc-tab__text-label{color:var(--text-subdued-color)}.sidebar{border-right:1px solid transparent;transition:background-color .2s cubic-bezier(0,0,.2,1),border-color .2s ease}.sidebar ul{list-style-type:none;text-align:left;margin:0;padding:0;display:flex;flex-direction:column}.sidebar ul li{display:flex;align-items:center;transition:all .2s ease;font-size:14px;padding:6px 6px 6px 24px}.sidebar ul li mat-icon{margin-right:8px}.sidebar ul li:hover{background-color:#343434}.sidebar ul .tooltip{background-color:#101010cc;border:1px solid #414141;opacity:0;pointer-events:none;transition:all .2s ease}.active .sidebar{background-color:#2c2c2c;border-right-color:#1c1c1c}.active .sidebar li:hover{background-color:#343434}.close{position:absolute;right:12px;color:#666;transition:color .25s ease}.close:hover{color:#ccc}\n"] }]
|
|
244
|
+
}], ctorParameters: function () { return [{ type: undefined, decorators: [{
|
|
245
|
+
type: Optional
|
|
246
|
+
}, {
|
|
247
|
+
type: Inject,
|
|
248
|
+
args: [NGX_WEB_COMPONENTS_CONFIG]
|
|
249
|
+
}] }, { type: i1.LazyLoaderService }, { type: i0.ViewContainerRef }, { type: i2.Fetch }]; }, propDecorators: { tabGroup: [{
|
|
250
|
+
type: ViewChild,
|
|
251
|
+
args: ['tabGroup']
|
|
252
|
+
}], fileGrids: [{
|
|
253
|
+
type: ViewChildren,
|
|
254
|
+
args: [FileGridComponent]
|
|
255
|
+
}], treeView: [{
|
|
256
|
+
type: ViewChild,
|
|
257
|
+
args: [TreeViewComponent]
|
|
258
|
+
}], toolbar: [{
|
|
259
|
+
type: ViewChild,
|
|
260
|
+
args: [ToolbarComponent]
|
|
261
|
+
}], drawer: [{
|
|
262
|
+
type: ViewChild,
|
|
263
|
+
args: [MatDrawerContainer]
|
|
264
|
+
}], config: [{
|
|
265
|
+
type: Input
|
|
266
|
+
}], gridSize: [{
|
|
267
|
+
type: Input
|
|
268
|
+
}], mode: [{
|
|
269
|
+
type: Input
|
|
270
|
+
}], contextTags: [{
|
|
271
|
+
type: Input
|
|
272
|
+
}], value: [{
|
|
273
|
+
type: Input
|
|
274
|
+
}], valueChange: [{
|
|
275
|
+
type: Output
|
|
276
|
+
}], focusedFilesChange: [{
|
|
277
|
+
type: Output
|
|
278
|
+
}], fileUpload: [{
|
|
279
|
+
type: Output
|
|
280
|
+
}], fileDownload: [{
|
|
281
|
+
type: Output
|
|
282
|
+
}], fileRename: [{
|
|
283
|
+
type: Output
|
|
284
|
+
}], fileDelete: [{
|
|
285
|
+
type: Output
|
|
286
|
+
}], fileCopy: [{
|
|
287
|
+
type: Output
|
|
288
|
+
}], filePaste: [{
|
|
289
|
+
type: Output
|
|
290
|
+
}], fileSelect: [{
|
|
291
|
+
type: Output
|
|
292
|
+
}], fileDblClick: [{
|
|
293
|
+
type: Output
|
|
294
|
+
}], folderSelect: [{
|
|
295
|
+
type: Output
|
|
296
|
+
}], folderDblClick: [{
|
|
297
|
+
type: Output
|
|
298
|
+
}], filesSelect: [{
|
|
299
|
+
type: Output
|
|
300
|
+
}] } });
|
|
301
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"filemanager.component.js","sourceRoot":"","sources":["../../../../../packages/common/src/components/filemanager/filemanager.component.ts","../../../../../packages/common/src/components/filemanager/filemanager.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,EAAE,KAAK,EAAU,QAAQ,EAAE,MAAM,EAAE,SAAS,EAAE,YAAY,EAAoB,MAAM,eAAe,CAAC;AAC5I,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,iBAAiB,CAAC;AAChD,OAAO,EAAe,aAAa,EAAE,MAAM,wBAAwB,CAAC;AACpE,OAAO,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAC;AACvD,OAAO,EAAE,kBAAkB,EAAE,gBAAgB,EAAE,MAAM,2BAA2B,CAAC;AACjF,OAAO,EAAE,kBAAkB,EAAE,MAAM,eAAe,CAAC;AAEnD,OAAO,EAAE,iBAAiB,EAAE,MAAM,iCAAiC,CAAC;AACpE,OAAO,EAAE,gBAAgB,EAAE,MAAM,6BAA6B,CAAC;AAC/D,OAAO,EAAE,iBAAiB,EAAE,MAAM,iCAAiC,CAAC;AACpE,OAAO,EAAe,yBAAyB,EAA0B,MAAM,UAAU,CAAC;AAC1F,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;;;;;;;;AAqJ/C,MAAM,OAAO,oBAAoB;IAoE7B,IAAI,eAAe,KAAK,OAAO,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAA,CAAC,CAAC;IAM9D,YAC4D,YAAoC,EAAE,EAC7E,UAA6B,EACtC,aAA+B,EAC/B,KAAY;QAHoC,cAAS,GAAT,SAAS,CAA6B;QAC7E,eAAU,GAAV,UAAU,CAAmB;QACtC,kBAAa,GAAb,aAAa,CAAkB;QAC/B,UAAK,GAAL,KAAK,CAAO;QAvEf,WAAM,GAAgC;YAC3C,WAAW,EAAE;gBACT,cAAc,EAAE,kBAAkB;gBAClC,cAAc,EAAE,kBAAkB;gBAClC,gBAAgB,EAAE,kBAAkB;gBACpC,cAAc,EAAE,kBAAkB;gBAClC,kBAAkB,EAAE,wBAAwB;gBAC5C,cAAc,EAAE,wBAAwB;aAC3C;SACJ,CAAC;QAEO,aAAQ,GAAiC,QAAQ,CAAC;QAKlD,UAAK,GAAmB,EAAE,CAAC;QAC1B,gBAAW,GAAG,IAAI,YAAY,EAAkB,CAAC;QAE3D,eAAU,GAAqB,EAAE,CAAC;QAElC;;;WAGG;QACO,uBAAkB,GAAG,IAAI,YAAY,EAAE,CAAC;QAClD;;WAEG;QACO,eAAU,GAAG,IAAI,YAAY,EAAE,CAAC;QAC1C;;WAEG;QACO,iBAAY,GAAG,IAAI,YAAY,EAAE,CAAC;QAClC,eAAU,GAAG,IAAI,YAAY,EAAE,CAAC;QAChC,eAAU,GAAG,IAAI,YAAY,EAAE,CAAC;QAChC,aAAQ,GAAG,IAAI,YAAY,EAAE,CAAC;QAC9B,cAAS,GAAG,IAAI,YAAY,EAAE,CAAC;QAE/B,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;QAEnE;;WAEG;QACO,gBAAW,GAAG,IAAI,YAAY,EAAkB,CAAC;QAC3D;;WAEG;QACH,kEAAkE;QAElE,oBAAe,GAAG,KAAK,CAAC;QACxB,gBAAW,GAAG,IAAI,CAAC;QACnB,mBAAc,GAAG,KAAK,CAAC;QACvB,UAAK,GAAG,CAAC,CAAC;QAEV,mBAAc,GAAG,KAAK,CAAC;QAEvB,eAAU,GAAgB,EAAS,CAAC;QAEpC,aAAQ,GAAG,CAAC,CAAC;QACb,SAAI,GAAkB,EAAE,CAAC;QAUrB,UAAU,CAAC,iBAAiB,CAAC;YACzB,EAAE,EAAE,eAAe;YACnB,KAAK,EAAE,+BAA+B;YACtC,IAAI,EAAE,GAAG,EAAE,CAAC,MAAM,CAAC,yCAAyC,CAAC;SAChE,CAAC,CAAA;QAEF,IAAI,CAAC,YAAY,GAAG,IAAI,YAAY,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;IAC9D,CAAC;IAED,QAAQ;QACJ,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QAC/B,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IACnC,CAAC;IAED,eAAe;QACX,IAAI,CAAC,QAAQ,EAAE,CAAC;QAEhB,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,QAAQ,EAAE,EAAE,GAAG,CAAC,CAAC;IAC3C,CAAC;IAED,sBAAsB,CAAC,EAAC,IAAI,EAAE,EAAE,EAAC;QAC7B,MAAM,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,sBAAsB;YAChD,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,sBAAsB,CAAC,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC;YACvE,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,cAAc,CAAA;QAElD,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,EAAE,EAAE,IAAI,EAAE,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,IAAI,EAAE,UAAU,EAAE,IAAI,CAAC,eAAe,EAAE,CAAC;aAClF,IAAI,CAAC,CAAC,IAAS,EAAE,EAAE;YAChB,MAAM,IAAI,GAA0B,IAAI,CAAC,IAAI,CAAC;YAC9C,EAAE,CAAC,IAAI,CAAC,CAAC;QACb,CAAC,CAAC,CAAA;IACV,CAAC;IAED,OAAO,CAAC,IAAY;QAChB,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,GAAG;YAC7B,EAAE,EAAE,MAAM,CAAC,UAAU,EAAE;YACvB,KAAK,EAAE,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC;YAC7B,UAAU,EAAE,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC;YACrC,IAAI;YACJ,SAAS,EAAE,EAAE;YACb,QAAQ,EAAE,IAAI,CAAC,IAAI,IAAI,MAAM;YAC7B,YAAY,EAAE,CAAC;YACf,OAAO,EAAE,EAAE;YACX,YAAY,EAAE,EAAE;YAChB,SAAS,EAAE,KAAK;SACnB,CAAC,CAAC;QACH,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC;IACrC,CAAC;IAED,QAAQ,CAAC,GAAgB;QACrB,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,IAAI,GAAG,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;IAClE,CAAC;IAED,cAAc,CAAC,IAAY;QACvB,IAAI,CAAC,IAAI;YAAE,OAAO,IAAI,CAAC;QAEvB,IAAI,GAAG,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC;QAE/B,4DAA4D;QAC5D,6BAA6B;QAC7B,IAAI,IAAI,CAAC,MAAM,CAAC,UAAU,EAAE;YACxB,IAAI,GAAG,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,EAAE,EAAE,CAAC,CAAC;YAChD,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YAE9B,IAAI,GAAG,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC;YACnD,OAAO,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE;gBACtB,MAAM,IAAI,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;gBAE7C,OAAO;oBACH,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,GAAG,CAAC,IAAI,IAAI,GAAG,CAAC,CAAC,CAAC,OAAO,CAAC,MAAM,EAAE,GAAG,CAAC;oBACjE,KAAK,EAAE,CAAC,IAAI,EAAE;iBACjB,CAAC;YACN,CAAC,CAAC,CAAC;SACN;aACI;YACD,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YAE9B,OAAO,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE;gBACtB,MAAM,IAAI,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;gBAE7C,OAAO;oBACH,EAAE,EAAE,IAAI,IAAI,GAAG;oBACf,KAAK,EAAE,CAAC,IAAI,EAAE;iBACjB,CAAC;YACN,CAAC,CAAC,CAAC;SACN;IACL,CAAC;IAED,iBAAiB,CAAC,KAAK;QACnB,IAAI,KAAK,CAAC,EAAE,EAAE;YACV,IAAI,CAAC,UAAU,CAAC,IAAI,GAAG,KAAK,CAAC,EAAE,CAAC;YAChC,IAAI,CAAC,UAAU,CAAC,UAAU,GAAG,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;SAC9D;IACL,CAAC;IAED,eAAe,CAAC,GAAgB;QAC5B,GAAG,CAAC,KAAK,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACvC,GAAG,CAAC,UAAU,GAAG,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAE/C,GAAG,CAAC,YAAY,EAAE,CAAC;QACnB,GAAG,CAAC,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAC3B,GAAG,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,IAAI,CAAC,MAAM,CAAC,gBAAgB,IAAI,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,gBAAgB,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;IAC5G,CAAC;IAED,gBAAgB,CAAC,IAAkB;QAC/B,IAAI,CAAC,UAAU,CAAC,IAAI,GAAG,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC;IACjD,CAAC;IAED,cAAc,CAAC,GAAgB,EAAE,KAAqB;QAClD,IAAI,GAAG,CAAC,YAAY,CAAC,MAAM,IAAI,CAAC,EAAE;YAC9B,GAAG,CAAC,YAAY,GAAG,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,IAAI,WAAW,CAAC,CAAC;YAC5D,OAAO;SACV;QAED,MAAM,YAAY,GAAG,GAAG,CAAC,YAAY,CAAC;QACtC,MAAM,QAAQ,GAAG,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,IAAI,WAAW,CAAC,CAAC;QAE1D,SAAS,OAAO,CAAC,KAAK;YAClB,OAAO,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,IAAI,EAAE,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;QACzD,CAAC;QACD,MAAM,MAAM,GAAG,OAAO,CAAC,YAAY,CAAC,CAAC;QAErC,IAAI,MAAM;YACN,MAAM,CAAC,WAAW,CAAC,GAAG,QAAQ,CAAC;QAEnC,GAAG,CAAC,YAAY,GAAG,YAAY,CAAC;IACpC,CAAC;IAED,iBAAiB;QACb,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QACrC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;IACrC,CAAC;IAED,WAAW,CAAC,IAAY;QACpB,OAAO,IAAI,EAAE,KAAK,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,GAAG,EAAE,CAAC;IACjD,CAAC;IAED,KAAK,CAAC,QAAQ;QACV,qCAAqC;QACrC,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC;QAExC,MAAM,EAAE,GAAG,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,aAA4B,CAAC;QACnE,MAAM,MAAM,GAAG,EAAE,CAAC,qBAAqB,EAAE,CAAC;QAC1C,IAAI,CAAC,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC;QAE1B,0EAA0E;QAC1E,IAAI,CAAC,cAAc,GAAG,MAAM,CAAC,KAAK,GAAG,GAAG,CAAC;QACzC,IAAI,IAAI,CAAC,cAAc,IAAI,KAAK,IAAI,CAAC,GAAG,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE;YACrE,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC;SACvB;IACL,CAAC;IAED,KAAK,CAAC,WAAW;QACb,IAAI,CAAC,QAAQ,EAAE,CAAC;QAEhB,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,QAAQ,EAAE,EAAE,GAAG,CAAC,CAAC;IAC3C,CAAC;IAED,YAAY;QACR,IAAI,IAAI,CAAC,UAAU,CAAC,QAAQ,IAAI,MAAM,EAAE;YACpC,OAAO,IAAI,CAAC,UAAU,CAAC,SAAS,CAAA;SACnC;aACI;YACD,OAAO,IAAI,CAAC,KAAK,CAAC;SACrB;IACL,CAAC;IAED,cAAc;QACV,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,cAAc,EAAE,CAAC,CAAC;IACpD,CAAC;IAED,8CAA8C;IAC9C,cAAc;QACV,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC;IAC1C,CAAC;IAED,WAAW;QACP,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,UAAU,EAAE,CAAC,CAAC;IAChD,CAAC;IAED,WAAW,CAAC,IAAoB;QAC5B,IAAI,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,sBAAsB;YACpD,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,sBAAsB,CAAC,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC;YACvE,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,cAAc,CAAC;QAE7C,GAAG,GAAG,CAAC,GAAG,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,OAAO,IAAI,CAAC,IAAI,SAAS,IAAI,CAAC,IAAI,EAAE,CAAC;QAE7E,OAAO,GAAG,CAAC;IACf,CAAC;;iHA3QQ,oBAAoB,kBA2EL,yBAAyB;qGA3ExC,oBAAoB,2rBAGlB,iBAAiB,0EACjB,gBAAgB,yEAChB,kBAAkB,+DAHf,iBAAiB,gDClKnC,koIAyFA,ihDD2DQ,IAAI,6FACJ,OAAO,kHACP,kBAAkB,skBAClB,iBAAiB,4TACjB,aAAa,6dACb,aAAa,mLACb,gBAAgB,2iBAChB,gBAAgB,2OAChB,iBAAiB;2FAIZ,oBAAoB;kBAjBhC,SAAS;+BACI,iBAAiB,WAGlB;wBACL,IAAI;wBACJ,OAAO;wBACP,kBAAkB;wBAClB,iBAAiB;wBACjB,aAAa;wBACb,aAAa;wBACb,gBAAgB;wBAChB,gBAAgB;wBAChB,iBAAiB;qBACpB,cACW,IAAI;;0BA6EX,QAAQ;;0BAAI,MAAM;2BAAC,yBAAyB;+HA1E1B,QAAQ;sBAA9B,SAAS;uBAAC,UAAU;gBACY,SAAS;sBAAzC,YAAY;uBAAC,iBAAiB;gBACD,QAAQ;sBAArC,SAAS;uBAAC,iBAAiB;gBACC,OAAO;sBAAnC,SAAS;uBAAC,gBAAgB;gBACI,MAAM;sBAApC,SAAS;uBAAC,kBAAkB;gBAEpB,MAAM;sBAAd,KAAK;gBAWG,QAAQ;sBAAhB,KAAK;gBACG,IAAI;sBAAZ,KAAK;gBACG,WAAW;sBAAnB,KAAK;gBAGG,KAAK;sBAAb,KAAK;gBACI,WAAW;sBAApB,MAAM;gBAQG,kBAAkB;sBAA3B,MAAM;gBAIG,UAAU;sBAAnB,MAAM;gBAIG,YAAY;sBAArB,MAAM;gBACG,UAAU;sBAAnB,MAAM;gBACG,UAAU;sBAAnB,MAAM;gBACG,QAAQ;sBAAjB,MAAM;gBACG,SAAS;sBAAlB,MAAM;gBAEG,UAAU;sBAAnB,MAAM;gBACG,YAAY;sBAArB,MAAM;gBACG,YAAY;sBAArB,MAAM;gBACG,cAAc;sBAAvB,MAAM;gBAKG,WAAW;sBAApB,MAAM","sourcesContent":["import { Component, EventEmitter, Inject, Input, OnInit, Optional, Output, ViewChild, ViewChildren, ViewContainerRef } from '@angular/core';\nimport { NgForOf, NgIf } from '@angular/common';\nimport { MatTabGroup, MatTabsModule } from '@angular/material/tabs';\nimport { MatIconModule } from '@angular/material/icon';\nimport { MatDrawerContainer, MatSidenavModule } from '@angular/material/sidenav';\nimport { AngularSplitModule } from 'angular-split';\n\nimport { FileGridComponent } from './file-grid/file-grid.component';\nimport { ToolbarComponent } from './toolbar/toolbar.component';\nimport { TreeViewComponent } from './tree-view/tree-view.component';\nimport { FileSorting, NGX_WEB_COMPONENTS_CONFIG, NgxWebComponentsConfig } from '../types';\nimport { IconResolver } from './icon-resolver';\nimport { LazyLoaderService, NGX_LAZY_LOADER_CONFIG } from '../lazy-loader/lazy-loader.service';\nimport { Fetch } from '../../public-api';\n\n// TODO:\n/**\n * Multiple music / video / image files selected turns into a playlist\n * Dragging music / video / image queues the file(s)\n * Can save and edit a list of files as playlist\n * Can \"loop\" \"randomize\"\n */\n\nexport type DirectoryDescriptor = {\n    kind: \"directory\",\n    path: string,\n    name: string\n    contents: {\n        dirs: DirectoryDescriptor[],\n        files: FileDescriptor[]\n    }\n}\n\nexport type FileDescriptor = {\n    kind: \"file\",\n    stats: {\n        size: number;\n        // Size for zipped file\n        compressedSize?: number,\n        atimeMs: number;\n        mtimeMs: number;\n        ctimeMs: number;\n        birthtimeMs: number;\n    },\n    path: string\n    name: string,\n    ext: string,\n    // Comment for entries in a zip file.\n    comment?: string\n};\n\nexport type FSDescriptor = DirectoryDescriptor | FileDescriptor;\n\nexport type FileViewTab = {\n    id: string,\n    label: string,\n    breadcrumb: {\n        id: string,\n        label: string\n    }[],\n    path: string,\n    selection: FSDescriptor[],\n    viewMode: \"grid\" | \"list\",\n    historyIndex: number,\n    history: string[],\n    sidebarItems: FSDescriptor[],\n    sortOrder: FileSorting\n}\n\nexport type NgxFileManagerConfiguration = Partial<{\n    /**\n     * Initial path\n     */\n    path: string,\n\n    /**\n     * Custom root path (will appear as '/').\n     * Defaults to '/'.\n     *\n     * User cannot view outside of this path.\n     *   (Not to be used as a security measure)\n     */\n    chrootPath: string,\n\n    /**\n     * Restrict users to only navigate around to subpaths of the specified `path`\n     */\n    navigateOnlyToDescendants: boolean,\n    showBreadcrumb: boolean,\n    showTreeview: boolean,\n\n    /**\n     * Name of the \"root\" path `/`\n     * Defaults to \"Storage\"\n     */\n    rootName: string,\n\n    /**\n     * Maximum number of items to be stored in history.\n     */\n    maxHistoryLength: number,\n\n    apiSettings: {\n        listEntriesUrl?: string,\n        listEntriesUrlTemplate?: (path: string) => string,\n        downloadEntryUrl?: string,\n        downloadEntryUrlTemplate?: (path: string) => string,\n        uploadEntryUrl?: string,\n        uploadEntryUrlTemplate?: (path: string) => string,\n        deleteEntryUrl?: string,\n        deleteEntryUrlTemplate?: (path: string) => string,\n        renameEntryUrl?: string,\n        renameEntryUrlTemplate?: (path: string) => string,\n        createDirectoryUrl?: string,\n        createDirectoryUrlTemplate?: (path: string) => string\n    },\n\n    /**\n     * The path that images are loaded from.\n     * Default value `/assets/dotglitch/webcomponents/`\n     */\n    assetPath: string,\n    sidebarLocationStrategy: \"known\" | \"currentDirectory\",\n\n    iconResolver: (file: FSDescriptor) => string,\n\n    imageSize: \"normal\" | \"small\" | \"huge\",\n\n    /**\n     *\n     */\n    selectionMode: \"single\" | \"multiple\",\n\n    /**\n     * This determines if the filemanager shows selected entries\n     *\n     * If set to `focusFiles`, file paths that match from the provided `focusedFiles`\n     * will be highlighted, and can be selected / deselected.\n     */\n    mode: \"focusFiles\" | \"normal\",\n    focusedFiles: string[]\n}>\n\n@Component({\n    selector: 'app-filemanager',\n    templateUrl: './filemanager.component.html',\n    styleUrls: ['./filemanager.component.scss'],\n    imports: [\n        NgIf,\n        NgForOf,\n        AngularSplitModule,\n        FileGridComponent,\n        MatTabsModule,\n        MatIconModule,\n        MatSidenavModule,\n        ToolbarComponent,\n        TreeViewComponent\n    ],\n    standalone: true\n})\nexport class FilemanagerComponent implements OnInit {\n    @ViewChild('tabGroup') tabGroup: MatTabGroup;\n    @ViewChildren(FileGridComponent) fileGrids: FileGridComponent[];\n    @ViewChild(TreeViewComponent) treeView: TreeViewComponent;\n    @ViewChild(ToolbarComponent) toolbar: ToolbarComponent;\n    @ViewChild(MatDrawerContainer) drawer: MatDrawerContainer;\n\n    @Input() config: NgxFileManagerConfiguration = {\n        apiSettings: {\n            listEntriesUrl: `/api/filesystem/`,\n            uploadEntryUrl: `/api/filesystem/`,\n            downloadEntryUrl: `/api/filesystem/`,\n            deleteEntryUrl: `/api/filesystem/`,\n            createDirectoryUrl: `/api/filesystem/folder`,\n            renameEntryUrl: `/api/filesystem/rename`\n        }\n    };\n\n    @Input() gridSize: \"small\" | \"normal\" | \"large\" = \"normal\";\n    @Input() mode: \"grid\" | \"list\";\n    @Input() contextTags: {[key: string]: string};\n\n\n    @Input() value: FSDescriptor[] = [];\n    @Output() valueChange = new EventEmitter<FSDescriptor[]>();\n\n    gridValues: FSDescriptor[][] = [];\n\n    /**\n     * Emits when focused files change.\n     * Only available in `focusFiles` mode.\n     */\n    @Output() focusedFilesChange = new EventEmitter();\n    /**\n     * Emits when a file is uploaded.\n     */\n    @Output() fileUpload = new EventEmitter();\n    /**\n     * Emits when a file is downloaded.\n     */\n    @Output() fileDownload = new EventEmitter();\n    @Output() fileRename = new EventEmitter();\n    @Output() fileDelete = new EventEmitter();\n    @Output() fileCopy = new EventEmitter();\n    @Output() filePaste = new EventEmitter();\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\n    /**\n     * Emits when multiple file selections change.\n     */\n    @Output() filesSelect = new EventEmitter<FSDescriptor[]>();\n    /**\n     * Emits when any selection changes, single or multiple files.\n     */\n    // @Output() selectionChange = new EventEmitter<FSDescriptor[]>();\n\n    showHiddenFiles = false;\n    showSidebar = true;\n    sidebarOverlay = false;\n    width = 0;\n\n    isHomeAncestor = false;\n\n    currentTab: FileViewTab = {} as any;\n    get currentFileGrid() { return this.fileGrids[this.tabIndex] }\n    tabIndex = 0;\n    tabs: FileViewTab[] = [];\n\n    iconResolver: IconResolver;\n\n    constructor (\n        @Optional() @Inject(NGX_WEB_COMPONENTS_CONFIG) readonly libConfig: NgxWebComponentsConfig = {},\n        private readonly lazyLoader: LazyLoaderService,\n        private viewContainer: ViewContainerRef,\n        private fetch: Fetch\n    ) {\n        lazyLoader.registerComponent({\n            id: \"folder-rename\",\n            group: \"@dotglitch/ngx-web-components\",\n            load: () => import('./folder-rename/folder-rename.component')\n        })\n\n        this.iconResolver = new IconResolver(libConfig.assetPath);\n    }\n\n    ngOnInit(): void {\n        this.initTab(this.config.path);\n        this.currentTab = this.tabs[0];\n    }\n\n    ngAfterViewInit() {\n        this.onResize();\n\n        setTimeout(() => this.onResize(), 250);\n    }\n\n    onTreeViewLoadChildren({item, cb}) {\n        const url = this.config.apiSettings.listEntriesUrlTemplate\n                  ? this.config.apiSettings.listEntriesUrlTemplate(item.path + item.name)\n                  : this.config.apiSettings.listEntriesUrl\n\n        this.fetch.post(url, { path: item.path + item.name, showHidden: this.showHiddenFiles })\n            .then((data: any) => {\n                const dirs: DirectoryDescriptor[] = data.dirs;\n                cb(dirs);\n            })\n    }\n\n    initTab(path: string) {\n        this.tabs.push(this.currentTab = {\n            id: crypto.randomUUID(),\n            label: this.getTabLabel(path),\n            breadcrumb: this.calcBreadcrumb(path),\n            path,\n            selection: [],\n            viewMode: this.mode || 'grid',\n            historyIndex: 0,\n            history: [],\n            sidebarItems: [],\n            sortOrder: 'a-z'\n        });\n        this.tabIndex = this.tabs.length;\n    }\n\n    closeTab(tab: FileViewTab) {\n        this.tabs.splice(this.tabs.findIndex(t => t.id == tab.id), 1);\n    }\n\n    calcBreadcrumb(path: string) {\n        if (!path) return null;\n\n        path = path.replace(\"#/\", '/');\n\n        // If we're acting like we're in a changed root, we wipe out\n        // breadcrumbs below the root\n        if (this.config.chrootPath) {\n            path = path.replace(this.config.chrootPath, '');\n            const parts = path.split('/');\n\n            path = path.replace(/^\\//, this.config.chrootPath);\n            return parts.map((p, i) => {\n                const path = parts.slice(0, i + 1).join('/');\n\n                return {\n                    id: (this.config.chrootPath + (path || '/')).replace(/\\/+/g, '/'),\n                    label: p || \"\"\n                };\n            });\n        }\n        else {\n            const parts = path.split('/');\n\n            return parts.map((p, i) => {\n                const path = parts.slice(0, i + 1).join('/');\n\n                return {\n                    id: path || '/',\n                    label: p || \"\"\n                };\n            });\n        }\n    }\n\n    onBreadcrumbClick(crumb) {\n        if (crumb.id) {\n            this.currentTab.path = crumb.id;\n            this.currentTab.breadcrumb = this.calcBreadcrumb(crumb.id);\n        }\n    }\n\n    onTabPathChange(tab: FileViewTab) {\n        tab.label = this.getTabLabel(tab.path);\n        tab.breadcrumb = this.calcBreadcrumb(tab.path);\n\n        tab.historyIndex++;\n        tab.history.push(tab.path);\n        tab.history.splice(typeof this.config.maxHistoryLength == 'number' ? this.config.maxHistoryLength : 50);\n    }\n\n    onTreeViewSelect(item: FSDescriptor) {\n        this.currentTab.path = item.path + item.name;\n    }\n\n    onTabLoadFiles(tab: FileViewTab, files: FSDescriptor[]) {\n        if (tab.sidebarItems.length == 0) {\n            tab.sidebarItems = files.filter(f => f.kind == \"directory\");\n            return;\n        }\n\n        const currentItems = tab.sidebarItems;\n        const dirItems = files.filter(f => f.kind == \"directory\");\n\n        function recurse(items) {\n            return items.find(i => tab.path?.startsWith(i.path));\n        }\n        const target = recurse(currentItems);\n\n        if (target)\n            target['_children'] = dirItems;\n\n        tab.sidebarItems = currentItems;\n    }\n\n    onGridValueChange() {\n        this.value = this.gridValues.flat(1);\n        this.valueChange.emit(this.value)\n    }\n\n    getTabLabel(path: string) {\n        return path?.split('/').filter(p => p).pop();\n    }\n\n    async onResize() {\n        // Trigger re-calculation of the view\n        this.fileGrids.forEach(g => g.resize());\n\n        const el = this.viewContainer.element.nativeElement as HTMLElement;\n        const bounds = el.getBoundingClientRect();\n        this.width = bounds.width;\n\n        // If the view area is less than 650px wide, use overlay the sidebar panel\n        this.sidebarOverlay = bounds.width < 650;\n        if (this.sidebarOverlay == false && [...this.drawer._drawers][0].opened) {\n            this.drawer.close();\n        }\n    }\n\n    async onResizeEnd() {\n        this.onResize();\n\n        setTimeout(() => this.onResize(), 250);\n    }\n\n    getSelection() {\n        if (this.currentTab.viewMode == \"grid\") {\n            return this.currentTab.selection\n        }\n        else {\n            return this.value;\n        }\n    }\n\n    clearSelection() {\n        this.fileGrids.forEach(g => g.clearSelection());\n    }\n\n    // Tell the child grid to refresh it's sorting\n    refreshSorting() {\n        this.fileGrids.forEach(g => g.sort());\n    }\n\n    refreshData() {\n        this.fileGrids.forEach(g => g.loadFolder());\n    }\n\n    getFileData(file: FileDescriptor) {\n        let url = this.config.apiSettings.renameEntryUrlTemplate\n            ? this.config.apiSettings.renameEntryUrlTemplate(file.path + file.name)\n            : this.config.apiSettings.renameEntryUrl;\n\n        url = (url.includes('?') ? '&' : '?') + `dir=${file.path}&file=${file.name}`;\n\n        return url;\n    }\n}\n","<app-toolbar\n    [currentTab]=\"currentTab\"\n    [(showHiddenFiles)]=\"showHiddenFiles\"\n    [(showSidebar)]=\"showSidebar\"\n    [config]=\"config\"\n    [showBareMinimum]=\"width < 500 && config.showBreadcrumb != false\"\n    (onBreadcrumbClick)=\"onBreadcrumbClick($event)\"\n/>\n\n<mat-drawer-container>\n    <mat-drawer #drawer mode=\"over\">\n        <app-toolbar\n            *ngIf=\"width < 500 && config.showBreadcrumb != false\"\n            [currentTab]=\"currentTab\"\n            [(showHiddenFiles)]=\"showHiddenFiles\"\n            [(showSidebar)]=\"showSidebar\"\n            [config]=\"config\"\n            [drawerMode]=\"true\"\n            (onBreadcrumbClick)=\"onBreadcrumbClick($event)\"\n        />\n        <app-tree-view\n            [data]=\"currentTab.sidebarItems\"\n            (loadChildren)=\"onTreeViewLoadChildren($event)\"\n            (click)=\"onTreeViewSelect($event)\"\n        />\n    </mat-drawer>\n    <mat-drawer-content>\n        <as-split direction=\"horizontal\" [gutterSize]=\"sidebarOverlay ? 0 : 2\" [unit]=\"'pixel'\" (dragEnd)=\"onResizeEnd()\">\n            <as-split-area\n                style=\"background-color: var(--filemanager-sidebar-background-color, #2c2c2c)\"\n                [size]=\"215\"\n                [minSize]=\"215\"\n                [maxSize]=\"width/4\"\n                [visible]=\"!sidebarOverlay && config.showTreeview != false\"\n            >\n                <div class=\"sidebar\"\n                    *ngIf=\"!sidebarOverlay && config.showTreeview != false\"\n                    (click)=\"currentTab.selection = []\"\n                >\n                    <app-tree-view\n                        [data]=\"currentTab.sidebarItems\"\n                        (loadChildren)=\"onTreeViewLoadChildren($event)\"\n                        (click)=\"onTreeViewSelect($event)\"\n                    />\n                </div>\n            </as-split-area>\n            <as-split-area style=\"background-color: var(--filemanager-background-color, #272727);\">\n                <div class=\"content\"\n                    [style.--tab-height]=\"tabs.length > 1 ? '36px' : '0'\"\n                >\n                    <mat-tab-group #tabGroup (selectedTabChange)=\"currentTab = tabs[$event.index]\" [(selectedIndex)]=\"tabIndex\">\n                        <mat-tab *ngFor=\"let tab of tabs; let i = index\">\n                            <ng-template mat-tab-label>\n                                {{tab.label}}\n                                <mat-icon\n                                    class=\"close\"\n                                    (click)=\"closeTab(tab)\">\n                                    close\n                                </mat-icon>\n                            </ng-template>\n\n                            <ng-template matTabContent>\n                                <app-file-grid\n                                    [viewMode]=\"tab.viewMode\"\n                                    [tab]=\"tab\"\n                                    [config]=\"config\"\n                                    [gridSize]=\"gridSize\"\n                                    [sortOrder]=\"tab.sortOrder\"\n                                    [(selection)]=\"tab.selection\"\n                                    [(path)]=\"tab.path\"\n                                    (pathChange)=\"onTabPathChange(tab)\"\n                                    [(value)]=\"gridValues[i]\"\n                                    (valueChange)=\"onGridValueChange()\"\n                                    (loadFiles)=\"onTabLoadFiles(tab, $event)\"\n\n                                    (fileSelect)=\"fileSelect.next($event)\"\n                                    (fileDblClick)=\"fileDblClick.next($event)\"\n                                    (folderSelect)=\"folderSelect.next($event)\"\n                                    (folderDblClick)=\"folderDblClick.next($event)\"\n                                />\n                            </ng-template>\n                        </mat-tab>\n                    </mat-tab-group>\n                </div>\n            </as-split-area>\n        </as-split>\n    </mat-drawer-content>\n</mat-drawer-container>\n\n"]}
|
|
@@ -0,0 +1,57 @@
|
|
|
1
|
+
import { Component, Input } from '@angular/core';
|
|
2
|
+
import { FormsModule } from '@angular/forms';
|
|
3
|
+
import { MatButtonModule } from '@angular/material/button';
|
|
4
|
+
import { MatIconModule } from '@angular/material/icon';
|
|
5
|
+
import { MatInputModule } from '@angular/material/input';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "@angular/material/dialog";
|
|
8
|
+
import * as i2 from "../../../public-api";
|
|
9
|
+
import * as i3 from "@angular/material/input";
|
|
10
|
+
import * as i4 from "@angular/material/form-field";
|
|
11
|
+
import * as i5 from "@angular/material/icon";
|
|
12
|
+
import * as i6 from "@angular/material/button";
|
|
13
|
+
import * as i7 from "@angular/forms";
|
|
14
|
+
export class FolderRenameComponent {
|
|
15
|
+
constructor(dialog, fetch) {
|
|
16
|
+
this.dialog = dialog;
|
|
17
|
+
this.fetch = fetch;
|
|
18
|
+
this.name = '';
|
|
19
|
+
this.isRename = false;
|
|
20
|
+
}
|
|
21
|
+
ngOnInit() {
|
|
22
|
+
this.isRename = this.name.length > 0;
|
|
23
|
+
}
|
|
24
|
+
onSave() {
|
|
25
|
+
if (this.isRename) {
|
|
26
|
+
const url = this.config.apiSettings.renameEntryUrlTemplate
|
|
27
|
+
? this.config.apiSettings.renameEntryUrlTemplate(this.path + this.name)
|
|
28
|
+
: this.config.apiSettings.renameEntryUrl;
|
|
29
|
+
this.fetch.post(url, { path: this.path + '/' + this.name });
|
|
30
|
+
}
|
|
31
|
+
else {
|
|
32
|
+
const url = this.config.apiSettings.createDirectoryUrlTemplate
|
|
33
|
+
? this.config.apiSettings.createDirectoryUrlTemplate(this.path + this.name)
|
|
34
|
+
: this.config.apiSettings.createDirectoryUrl;
|
|
35
|
+
this.fetch.post(url, { path: this.path + '/' + this.name });
|
|
36
|
+
}
|
|
37
|
+
this.dialog.close(true);
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
FolderRenameComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FolderRenameComponent, deps: [{ token: i1.MatDialogRef }, { token: i2.Fetch }], target: i0.ɵɵFactoryTarget.Component });
|
|
41
|
+
FolderRenameComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: FolderRenameComponent, isStandalone: true, selector: "app-folder-rename", inputs: { path: "path", name: "name", config: "config" }, ngImport: i0, template: "<mat-form-field style=\"width: 280px;\">\n <mat-label>Name</mat-label>\n <input matInput type=\"text\" [(ngModel)]=\"name\" (keyup)=\"$event.key == 'Enter' && onSave()\">\n</mat-form-field>\n\n<div style=\"align-self: end;\">\n <button mat-flat-button (click)=\"onSave()\"><mat-icon>check</mat-icon>Ok</button>\n <button mat-flat-button (click)=\"dialog.close()\"><mat-icon>cancel</mat-icon>Cancel</button>\n</div>\n", styles: [":host{display:flex;flex-direction:column;padding:24px}\n"], dependencies: [{ kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i3.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "component", type: i4.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i4.MatLabel, selector: "mat-label" }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i5.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i6.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i7.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i7.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i7.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }] });
|
|
42
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FolderRenameComponent, decorators: [{
|
|
43
|
+
type: Component,
|
|
44
|
+
args: [{ selector: 'app-folder-rename', imports: [
|
|
45
|
+
MatInputModule,
|
|
46
|
+
MatIconModule,
|
|
47
|
+
MatButtonModule,
|
|
48
|
+
FormsModule
|
|
49
|
+
], standalone: true, template: "<mat-form-field style=\"width: 280px;\">\n <mat-label>Name</mat-label>\n <input matInput type=\"text\" [(ngModel)]=\"name\" (keyup)=\"$event.key == 'Enter' && onSave()\">\n</mat-form-field>\n\n<div style=\"align-self: end;\">\n <button mat-flat-button (click)=\"onSave()\"><mat-icon>check</mat-icon>Ok</button>\n <button mat-flat-button (click)=\"dialog.close()\"><mat-icon>cancel</mat-icon>Cancel</button>\n</div>\n", styles: [":host{display:flex;flex-direction:column;padding:24px}\n"] }]
|
|
50
|
+
}], ctorParameters: function () { return [{ type: i1.MatDialogRef }, { type: i2.Fetch }]; }, propDecorators: { path: [{
|
|
51
|
+
type: Input
|
|
52
|
+
}], name: [{
|
|
53
|
+
type: Input
|
|
54
|
+
}], config: [{
|
|
55
|
+
type: Input
|
|
56
|
+
}] } });
|
|
57
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9sZGVyLXJlbmFtZS5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9jb21tb24vc3JjL2NvbXBvbmVudHMvZmlsZW1hbmFnZXIvZm9sZGVyLXJlbmFtZS9mb2xkZXItcmVuYW1lLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL2NvbW1vbi9zcmMvY29tcG9uZW50cy9maWxlbWFuYWdlci9mb2xkZXItcmVuYW1lL2ZvbGRlci1yZW5hbWUuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQVUsTUFBTSxlQUFlLENBQUM7QUFDekQsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQzdDLE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQztBQUUzRCxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sd0JBQXdCLENBQUM7QUFDdkQsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLHlCQUF5QixDQUFDOzs7Ozs7Ozs7QUFnQnpELE1BQU0sT0FBTyxxQkFBcUI7SUFROUIsWUFDVyxNQUF5QixFQUN4QixLQUFZO1FBRGIsV0FBTSxHQUFOLE1BQU0sQ0FBbUI7UUFDeEIsVUFBSyxHQUFMLEtBQUssQ0FBTztRQVBmLFNBQUksR0FBVyxFQUFFLENBQUM7UUFHM0IsYUFBUSxHQUFHLEtBQUssQ0FBQztJQUtiLENBQUM7SUFFTCxRQUFRO1FBQ0osSUFBSSxDQUFDLFFBQVEsR0FBRyxJQUFJLENBQUMsSUFBSSxDQUFDLE1BQU0sR0FBRyxDQUFDLENBQUM7SUFDekMsQ0FBQztJQUVELE1BQU07UUFDRixJQUFJLElBQUksQ0FBQyxRQUFRLEVBQUU7WUFDZixNQUFNLEdBQUcsR0FBRyxJQUFJLENBQUMsTUFBTSxDQUFDLFdBQVcsQ0FBQyxzQkFBc0I7Z0JBQ3RELENBQUMsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLFdBQVcsQ0FBQyxzQkFBc0IsQ0FBQyxJQUFJLENBQUMsSUFBSSxHQUFHLElBQUksQ0FBQyxJQUFJLENBQUM7Z0JBQ3ZFLENBQUMsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLFdBQVcsQ0FBQyxjQUFjLENBQUE7WUFFNUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsR0FBRyxFQUFFLEVBQUUsSUFBSSxFQUFFLElBQUksQ0FBQyxJQUFJLEdBQUcsR0FBRyxHQUFHLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQyxDQUFDO1NBQy9EO2FBQ0k7WUFDRCxNQUFNLEdBQUcsR0FBRyxJQUFJLENBQUMsTUFBTSxDQUFDLFdBQVcsQ0FBQywwQkFBMEI7Z0JBQ3BELENBQUMsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLFdBQVcsQ0FBQywwQkFBMEIsQ0FBQyxJQUFJLENBQUMsSUFBSSxHQUFHLElBQUksQ0FBQyxJQUFJLENBQUM7Z0JBQzNFLENBQUMsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLFdBQVcsQ0FBQyxrQkFBa0IsQ0FBQTtZQUV0RCxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxHQUFHLEVBQUUsRUFBQyxJQUFJLEVBQUUsSUFBSSxDQUFDLElBQUksR0FBRSxHQUFHLEdBQUUsSUFBSSxDQUFDLElBQUksRUFBQyxDQUFDLENBQUM7U0FDM0Q7UUFDRCxJQUFJLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUM1QixDQUFDOztrSEFqQ1EscUJBQXFCO3NHQUFyQixxQkFBcUIsdUlDckJsQyw4YUFTQSxpSERLUSxjQUFjLHNuQkFDZCxhQUFhLG1MQUNiLGVBQWUsMlFBQ2YsV0FBVzsyRkFJTixxQkFBcUI7a0JBWmpDLFNBQVM7K0JBQ0ksbUJBQW1CLFdBR3BCO3dCQUNMLGNBQWM7d0JBQ2QsYUFBYTt3QkFDYixlQUFlO3dCQUNmLFdBQVc7cUJBQ2QsY0FDVyxJQUFJO3VIQUlQLElBQUk7c0JBQVosS0FBSztnQkFDRyxJQUFJO3NCQUFaLEtBQUs7Z0JBQ0csTUFBTTtzQkFBZCxLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50LCBJbnB1dCwgT25Jbml0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBGb3Jtc01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2Zvcm1zJztcbmltcG9ydCB7IE1hdEJ1dHRvbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2J1dHRvbic7XG5pbXBvcnQgeyBNYXREaWFsb2dSZWYgfSBmcm9tICdAYW5ndWxhci9tYXRlcmlhbC9kaWFsb2cnO1xuaW1wb3J0IHsgTWF0SWNvbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2ljb24nO1xuaW1wb3J0IHsgTWF0SW5wdXRNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9tYXRlcmlhbC9pbnB1dCc7XG5pbXBvcnQgeyBOZ3hGaWxlTWFuYWdlckNvbmZpZ3VyYXRpb24gfSBmcm9tICcuLi9maWxlbWFuYWdlci5jb21wb25lbnQnO1xuaW1wb3J0IHsgRmV0Y2ggfSBmcm9tICcuLi8uLi8uLi9wdWJsaWMtYXBpJztcblxuQENvbXBvbmVudCh7XG4gICAgc2VsZWN0b3I6ICdhcHAtZm9sZGVyLXJlbmFtZScsXG4gICAgdGVtcGxhdGVVcmw6ICcuL2ZvbGRlci1yZW5hbWUuY29tcG9uZW50Lmh0bWwnLFxuICAgIHN0eWxlVXJsczogWycuL2ZvbGRlci1yZW5hbWUuY29tcG9uZW50LnNjc3MnXSxcbiAgICBpbXBvcnRzOiBbXG4gICAgICAgIE1hdElucHV0TW9kdWxlLFxuICAgICAgICBNYXRJY29uTW9kdWxlLFxuICAgICAgICBNYXRCdXR0b25Nb2R1bGUsXG4gICAgICAgIEZvcm1zTW9kdWxlXG4gICAgXSxcbiAgICBzdGFuZGFsb25lOiB0cnVlXG59KVxuZXhwb3J0IGNsYXNzIEZvbGRlclJlbmFtZUNvbXBvbmVudCBpbXBsZW1lbnRzIE9uSW5pdCB7XG5cbiAgICBASW5wdXQoKSBwYXRoOiBzdHJpbmc7XG4gICAgQElucHV0KCkgbmFtZTogc3RyaW5nID0gJyc7XG4gICAgQElucHV0KCkgY29uZmlnOiBOZ3hGaWxlTWFuYWdlckNvbmZpZ3VyYXRpb247XG5cbiAgICBpc1JlbmFtZSA9IGZhbHNlO1xuXG4gICAgY29uc3RydWN0b3IoXG4gICAgICAgIHB1YmxpYyBkaWFsb2c6IE1hdERpYWxvZ1JlZjxhbnk+LFxuICAgICAgICBwcml2YXRlIGZldGNoOiBGZXRjaFxuICAgICkgeyB9XG5cbiAgICBuZ09uSW5pdCgpIHtcbiAgICAgICAgdGhpcy5pc1JlbmFtZSA9IHRoaXMubmFtZS5sZW5ndGggPiAwO1xuICAgIH1cblxuICAgIG9uU2F2ZSgpIHtcbiAgICAgICAgaWYgKHRoaXMuaXNSZW5hbWUpIHtcbiAgICAgICAgICAgIGNvbnN0IHVybCA9IHRoaXMuY29uZmlnLmFwaVNldHRpbmdzLnJlbmFtZUVudHJ5VXJsVGVtcGxhdGVcbiAgICAgICAgICAgICAgICA/IHRoaXMuY29uZmlnLmFwaVNldHRpbmdzLnJlbmFtZUVudHJ5VXJsVGVtcGxhdGUodGhpcy5wYXRoICsgdGhpcy5uYW1lKVxuICAgICAgICAgICAgICAgIDogdGhpcy5jb25maWcuYXBpU2V0dGluZ3MucmVuYW1lRW50cnlVcmxcblxuICAgICAgICAgICAgdGhpcy5mZXRjaC5wb3N0KHVybCwgeyBwYXRoOiB0aGlzLnBhdGggKyAnLycgKyB0aGlzLm5hbWUgfSk7XG4gICAgICAgIH1cbiAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICBjb25zdCB1cmwgPSB0aGlzLmNvbmZpZy5hcGlTZXR0aW5ncy5jcmVhdGVEaXJlY3RvcnlVcmxUZW1wbGF0ZVxuICAgICAgICAgICAgICAgICAgICAgID8gdGhpcy5jb25maWcuYXBpU2V0dGluZ3MuY3JlYXRlRGlyZWN0b3J5VXJsVGVtcGxhdGUodGhpcy5wYXRoICsgdGhpcy5uYW1lKVxuICAgICAgICAgICAgICAgICAgICAgIDogdGhpcy5jb25maWcuYXBpU2V0dGluZ3MuY3JlYXRlRGlyZWN0b3J5VXJsXG5cbiAgICAgICAgICAgIHRoaXMuZmV0Y2gucG9zdCh1cmwsIHtwYXRoOiB0aGlzLnBhdGggKycvJysgdGhpcy5uYW1lfSk7XG4gICAgICAgIH1cbiAgICAgICAgdGhpcy5kaWFsb2cuY2xvc2UodHJ1ZSk7XG4gICAgfVxufVxuIiwiPG1hdC1mb3JtLWZpZWxkIHN0eWxlPVwid2lkdGg6IDI4MHB4O1wiPlxuICAgIDxtYXQtbGFiZWw+TmFtZTwvbWF0LWxhYmVsPlxuICAgIDxpbnB1dCBtYXRJbnB1dCB0eXBlPVwidGV4dFwiIFsobmdNb2RlbCldPVwibmFtZVwiIChrZXl1cCk9XCIkZXZlbnQua2V5ID09ICdFbnRlcicgJiYgb25TYXZlKClcIj5cbjwvbWF0LWZvcm0tZmllbGQ+XG5cbjxkaXYgc3R5bGU9XCJhbGlnbi1zZWxmOiBlbmQ7XCI+XG4gICAgPGJ1dHRvbiBtYXQtZmxhdC1idXR0b24gKGNsaWNrKT1cIm9uU2F2ZSgpXCI+PG1hdC1pY29uPmNoZWNrPC9tYXQtaWNvbj5PazwvYnV0dG9uPlxuICAgIDxidXR0b24gbWF0LWZsYXQtYnV0dG9uIChjbGljayk9XCJkaWFsb2cuY2xvc2UoKVwiPjxtYXQtaWNvbj5jYW5jZWw8L21hdC1pY29uPkNhbmNlbDwvYnV0dG9uPlxuPC9kaXY+XG4iXX0=
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
export const uploadFile = (fetch, config, currentDirectory, targetPath, contextTags = {}) => new Promise(r => {
|
|
2
|
+
const inEl = document.createElement('input');
|
|
3
|
+
inEl.setAttribute('type', 'file');
|
|
4
|
+
inEl.setAttribute('multiple', '');
|
|
5
|
+
inEl.click();
|
|
6
|
+
let formData = new FormData();
|
|
7
|
+
inEl.addEventListener('change', () => {
|
|
8
|
+
Object.keys(inEl.files).forEach(k => {
|
|
9
|
+
const file = inEl.files[k];
|
|
10
|
+
const name = file.name;
|
|
11
|
+
formData.append(name, file);
|
|
12
|
+
});
|
|
13
|
+
formData.append("data", JSON.stringify({
|
|
14
|
+
path: currentDirectory,
|
|
15
|
+
...contextTags
|
|
16
|
+
}));
|
|
17
|
+
const url = config.apiSettings.uploadEntryUrlTemplate
|
|
18
|
+
? config.apiSettings.uploadEntryUrlTemplate(targetPath ?? currentDirectory)
|
|
19
|
+
: config.apiSettings.uploadEntryUrl;
|
|
20
|
+
r(fetch.post(url, formData).then(res => {
|
|
21
|
+
inEl.remove();
|
|
22
|
+
return res;
|
|
23
|
+
}));
|
|
24
|
+
});
|
|
25
|
+
});
|
|
26
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGVscGVycy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL2NvbW1vbi9zcmMvY29tcG9uZW50cy9maWxlbWFuYWdlci9oZWxwZXJzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUdBLE1BQU0sQ0FBQyxNQUFNLFVBQVUsR0FBRyxDQUNsQixLQUFZLEVBQ1osTUFBbUMsRUFDbkMsZ0JBQXdCLEVBQ3hCLFVBQW1CLEVBQ25CLGNBQXVDLEVBQUUsRUFDM0MsRUFBRSxDQUFDLElBQUksT0FBTyxDQUFDLENBQUMsQ0FBQyxFQUFFO0lBRXJCLE1BQU0sSUFBSSxHQUFHLFFBQVEsQ0FBQyxhQUFhLENBQUMsT0FBTyxDQUFDLENBQUM7SUFDN0MsSUFBSSxDQUFDLFlBQVksQ0FBQyxNQUFNLEVBQUUsTUFBTSxDQUFDLENBQUM7SUFDbEMsSUFBSSxDQUFDLFlBQVksQ0FBQyxVQUFVLEVBQUUsRUFBRSxDQUFDLENBQUM7SUFDbEMsSUFBSSxDQUFDLEtBQUssRUFBRSxDQUFDO0lBRWIsSUFBSSxRQUFRLEdBQUcsSUFBSSxRQUFRLEVBQUUsQ0FBQztJQUU5QixJQUFJLENBQUMsZ0JBQWdCLENBQUMsUUFBUSxFQUFFLEdBQUcsRUFBRTtRQUNqQyxNQUFNLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLEVBQUU7WUFDaEMsTUFBTSxJQUFJLEdBTU4sSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztZQUVsQixNQUFNLElBQUksR0FBRyxJQUFJLENBQUMsSUFBSSxDQUFDO1lBQ3ZCLFFBQVEsQ0FBQyxNQUFNLENBQUMsSUFBSSxFQUFFLElBQVcsQ0FBQyxDQUFDO1FBQ3ZDLENBQUMsQ0FBQyxDQUFDO1FBQ0gsUUFBUSxDQUFDLE1BQU0sQ0FBQyxNQUFNLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQztZQUNuQyxJQUFJLEVBQUUsZ0JBQWdCO1lBQ3RCLEdBQUcsV0FBVztTQUNqQixDQUFDLENBQUMsQ0FBQztRQUVKLE1BQU0sR0FBRyxHQUFHLE1BQU0sQ0FBQyxXQUFXLENBQUMsc0JBQXNCO1lBQ2pELENBQUMsQ0FBQyxNQUFNLENBQUMsV0FBVyxDQUFDLHNCQUFzQixDQUFDLFVBQVUsSUFBSSxnQkFBZ0IsQ0FBQztZQUMzRSxDQUFDLENBQUMsTUFBTSxDQUFDLFdBQVcsQ0FBQyxjQUFjLENBQUM7UUFFeEMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsR0FBRyxFQUFFLFFBQVEsQ0FBQyxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsRUFBRTtZQUNuQyxJQUFJLENBQUMsTUFBTSxFQUFFLENBQUM7WUFDZCxPQUFPLEdBQUcsQ0FBQztRQUNmLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDUixDQUFDLENBQUMsQ0FBQztBQUNQLENBQUMsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRmV0Y2ggfSBmcm9tICcuLi8uLi9wdWJsaWMtYXBpJztcbmltcG9ydCB7IE5neEZpbGVNYW5hZ2VyQ29uZmlndXJhdGlvbiB9IGZyb20gJy4vZmlsZW1hbmFnZXIuY29tcG9uZW50JztcblxuZXhwb3J0IGNvbnN0IHVwbG9hZEZpbGUgPSAoXG4gICAgICAgIGZldGNoOiBGZXRjaCxcbiAgICAgICAgY29uZmlnOiBOZ3hGaWxlTWFuYWdlckNvbmZpZ3VyYXRpb24sXG4gICAgICAgIGN1cnJlbnREaXJlY3Rvcnk6IHN0cmluZyxcbiAgICAgICAgdGFyZ2V0UGF0aD86IHN0cmluZyxcbiAgICAgICAgY29udGV4dFRhZ3M6IHtba2V5OiBzdHJpbmddOiBzdHJpbmd9ID0ge31cbiAgICApID0+IG5ldyBQcm9taXNlKHIgPT4ge1xuXG4gICAgY29uc3QgaW5FbCA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoJ2lucHV0Jyk7XG4gICAgaW5FbC5zZXRBdHRyaWJ1dGUoJ3R5cGUnLCAnZmlsZScpO1xuICAgIGluRWwuc2V0QXR0cmlidXRlKCdtdWx0aXBsZScsICcnKTtcbiAgICBpbkVsLmNsaWNrKCk7XG5cbiAgICBsZXQgZm9ybURhdGEgPSBuZXcgRm9ybURhdGEoKTtcblxuICAgIGluRWwuYWRkRXZlbnRMaXN0ZW5lcignY2hhbmdlJywgKCkgPT4ge1xuICAgICAgICBPYmplY3Qua2V5cyhpbkVsLmZpbGVzKS5mb3JFYWNoKGsgPT4ge1xuICAgICAgICAgICAgY29uc3QgZmlsZToge1xuICAgICAgICAgICAgICAgIGxhc3RNb2RpZmllZDogbnVtYmVyLFxuICAgICAgICAgICAgICAgIGxhc3RNb2RpZmllZERhdGU6IERhdGUsXG4gICAgICAgICAgICAgICAgbmFtZTogc3RyaW5nLFxuICAgICAgICAgICAgICAgIHNpemU6IG51bWJlcixcbiAgICAgICAgICAgICAgICB0eXBlOiBzdHJpbmc7XG4gICAgICAgICAgICB9ID0gaW5FbC5maWxlc1trXTtcblxuICAgICAgICAgICAgY29uc3QgbmFtZSA9IGZpbGUubmFtZTtcbiAgICAgICAgICAgIGZvcm1EYXRhLmFwcGVuZChuYW1lLCBmaWxlIGFzIGFueSk7XG4gICAgICAgIH0pO1xuICAgICAgICBmb3JtRGF0YS5hcHBlbmQoXCJkYXRhXCIsIEpTT04uc3RyaW5naWZ5KHtcbiAgICAgICAgICAgIHBhdGg6IGN1cnJlbnREaXJlY3RvcnksXG4gICAgICAgICAgICAuLi5jb250ZXh0VGFnc1xuICAgICAgICB9KSk7XG5cbiAgICAgICAgY29uc3QgdXJsID0gY29uZmlnLmFwaVNldHRpbmdzLnVwbG9hZEVudHJ5VXJsVGVtcGxhdGVcbiAgICAgICAgICAgID8gY29uZmlnLmFwaVNldHRpbmdzLnVwbG9hZEVudHJ5VXJsVGVtcGxhdGUodGFyZ2V0UGF0aCA/PyBjdXJyZW50RGlyZWN0b3J5KVxuICAgICAgICAgICAgOiBjb25maWcuYXBpU2V0dGluZ3MudXBsb2FkRW50cnlVcmw7XG5cbiAgICAgICAgcihmZXRjaC5wb3N0KHVybCwgZm9ybURhdGEpLnRoZW4ocmVzID0+IHtcbiAgICAgICAgICAgIGluRWwucmVtb3ZlKCk7XG4gICAgICAgICAgICByZXR1cm4gcmVzO1xuICAgICAgICB9KSk7XG4gICAgfSk7XG59KTtcblxuIl19
|