@theia/filesystem 1.53.0-next.55 → 1.53.0-next.64
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/README.md +30 -30
- package/package.json +3 -3
- package/src/browser/breadcrumbs/filepath-breadcrumb.ts +43 -43
- package/src/browser/breadcrumbs/filepath-breadcrumbs-container.ts +65 -65
- package/src/browser/breadcrumbs/filepath-breadcrumbs-contribution.ts +129 -129
- package/src/browser/download/file-download-command-contribution.ts +83 -83
- package/src/browser/download/file-download-frontend-module.ts +25 -25
- package/src/browser/download/file-download-service.ts +179 -179
- package/src/browser/file-dialog/file-dialog-container.ts +67 -67
- package/src/browser/file-dialog/file-dialog-hidden-files-renderer.tsx +59 -59
- package/src/browser/file-dialog/file-dialog-model.ts +96 -96
- package/src/browser/file-dialog/file-dialog-module.ts +44 -44
- package/src/browser/file-dialog/file-dialog-service.ts +99 -99
- package/src/browser/file-dialog/file-dialog-tree-filters-renderer.tsx +100 -100
- package/src/browser/file-dialog/file-dialog-tree.ts +89 -89
- package/src/browser/file-dialog/file-dialog-widget.ts +75 -75
- package/src/browser/file-dialog/file-dialog.ts +434 -434
- package/src/browser/file-dialog/index.ts +20 -20
- package/src/browser/file-resource.spec.ts +255 -255
- package/src/browser/file-resource.ts +402 -402
- package/src/browser/file-selection.ts +44 -44
- package/src/browser/file-service.ts +1845 -1845
- package/src/browser/file-tree/file-tree-container.ts +36 -36
- package/src/browser/file-tree/file-tree-decorator-adapter.ts +159 -159
- package/src/browser/file-tree/file-tree-label-provider.ts +53 -53
- package/src/browser/file-tree/file-tree-model.ts +212 -212
- package/src/browser/file-tree/file-tree-widget.tsx +327 -327
- package/src/browser/file-tree/file-tree.ts +183 -183
- package/src/browser/file-tree/index.ts +22 -22
- package/src/browser/file-upload-service.ts +547 -547
- package/src/browser/filesystem-frontend-contribution.ts +396 -396
- package/src/browser/filesystem-frontend-module.ts +77 -77
- package/src/browser/filesystem-preferences.ts +139 -139
- package/src/browser/filesystem-saveable-service.ts +138 -138
- package/src/browser/filesystem-watcher-error-handler.ts +60 -60
- package/src/browser/index.ts +21 -21
- package/src/browser/location/index.ts +18 -18
- package/src/browser/location/location-renderer.tsx +406 -406
- package/src/browser/location/location-service.ts +22 -22
- package/src/browser/remote-file-service-contribution.ts +38 -38
- package/src/browser/style/file-dialog.css +208 -208
- package/src/browser/style/file-icons.css +64 -64
- package/src/browser/style/filepath-breadcrumbs.css +20 -20
- package/src/browser/style/index.css +36 -36
- package/src/browser-only/browser-only-filesystem-frontend-module.ts +38 -38
- package/src/browser-only/browser-only-filesystem-provider-server.ts +32 -32
- package/src/browser-only/browserfs-filesystem-initialization.ts +61 -61
- package/src/browser-only/browserfs-filesystem-provider.ts +462 -462
- package/src/common/delegating-file-system-provider.ts +226 -226
- package/src/common/download/README.md +30 -30
- package/src/common/download/file-download-data.ts +27 -27
- package/src/common/file-upload.ts +17 -17
- package/src/common/files.spec.ts +51 -51
- package/src/common/files.ts +997 -997
- package/src/common/filesystem-utils.spec.ts +411 -411
- package/src/common/filesystem-utils.ts +64 -64
- package/src/common/filesystem-watcher-protocol.ts +96 -96
- package/src/common/filesystem.ts +43 -43
- package/src/common/index.ts +18 -18
- package/src/common/io.ts +150 -150
- package/src/common/remote-file-system-provider.ts +549 -549
- package/src/electron-browser/file-dialog/electron-file-dialog-module.ts +24 -24
- package/src/electron-browser/file-dialog/electron-file-dialog-service.ts +165 -165
- package/src/electron-browser/preload.ts +31 -31
- package/src/electron-common/electron-api.ts +55 -55
- package/src/electron-main/electron-api-main.ts +78 -78
- package/src/electron-main/electron-main-module.ts +23 -23
- package/src/node/disk-file-system-provider.spec.ts +142 -142
- package/src/node/disk-file-system-provider.ts +915 -915
- package/src/node/download/directory-archiver.spec.ts +104 -104
- package/src/node/download/directory-archiver.ts +126 -126
- package/src/node/download/file-download-backend-module.ts +32 -32
- package/src/node/download/file-download-cache.ts +88 -88
- package/src/node/download/file-download-endpoint.ts +63 -63
- package/src/node/download/file-download-handler.ts +304 -304
- package/src/node/download/test/mock-directory-archiver.ts +30 -30
- package/src/node/file-change-collection.spec.ts +110 -110
- package/src/node/file-change-collection.ts +78 -78
- package/src/node/filesystem-backend-module.ts +140 -140
- package/src/node/filesystem-watcher-client.ts +72 -72
- package/src/node/filesystem-watcher-dispatcher.ts +82 -82
- package/src/node/node-file-upload-service.ts +80 -80
- package/src/node/nsfw-watcher/index.ts +45 -45
- package/src/node/nsfw-watcher/nsfw-filesystem-service.ts +481 -481
- package/src/node/nsfw-watcher/nsfw-filesystem-watcher.spec.ts +182 -182
- package/src/node/nsfw-watcher/nsfw-options.ts +23 -23
- package/src/typings/dom.webkit.d.ts +77 -77
- package/src/typings/mv/index.d.ts +21 -21
- package/src/typings/nsfw/index.d.ts +18 -18
- package/src/typings/trash/index.d.ts +20 -20
|
@@ -1,99 +1,99 @@
|
|
|
1
|
-
// *****************************************************************************
|
|
2
|
-
// Copyright (C) 2018 Ericsson and others.
|
|
3
|
-
//
|
|
4
|
-
// This program and the accompanying materials are made available under the
|
|
5
|
-
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
6
|
-
// http://www.eclipse.org/legal/epl-2.0.
|
|
7
|
-
//
|
|
8
|
-
// This Source Code may also be made available under the following Secondary
|
|
9
|
-
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
10
|
-
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
11
|
-
// with the GNU Classpath Exception which is available at
|
|
12
|
-
// https://www.gnu.org/software/classpath/license.html.
|
|
13
|
-
//
|
|
14
|
-
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
15
|
-
// *****************************************************************************
|
|
16
|
-
|
|
17
|
-
import { injectable, inject } from '@theia/core/shared/inversify';
|
|
18
|
-
import URI from '@theia/core/lib/common/uri';
|
|
19
|
-
import { MaybeArray, UNTITLED_SCHEME, nls } from '@theia/core/lib/common';
|
|
20
|
-
import { LabelProvider } from '@theia/core/lib/browser';
|
|
21
|
-
import { FileStat } from '../../common/files';
|
|
22
|
-
import { DirNode } from '../file-tree';
|
|
23
|
-
import { OpenFileDialogFactory, OpenFileDialogProps, SaveFileDialogFactory, SaveFileDialogProps } from './file-dialog';
|
|
24
|
-
import { FileService } from '../file-service';
|
|
25
|
-
import { EnvVariablesServer } from '@theia/core/lib/common/env-variables';
|
|
26
|
-
import { UserWorkingDirectoryProvider } from '@theia/core/lib/browser/user-working-directory-provider';
|
|
27
|
-
|
|
28
|
-
export const FileDialogService = Symbol('FileDialogService');
|
|
29
|
-
export interface FileDialogService {
|
|
30
|
-
|
|
31
|
-
showOpenDialog(props: OpenFileDialogProps & { canSelectMany: true }, folder?: FileStat): Promise<MaybeArray<URI> | undefined>;
|
|
32
|
-
showOpenDialog(props: OpenFileDialogProps, folder?: FileStat): Promise<URI | undefined>;
|
|
33
|
-
showOpenDialog(props: OpenFileDialogProps, folder?: FileStat): Promise<MaybeArray<URI> | undefined>;
|
|
34
|
-
|
|
35
|
-
showSaveDialog(props: SaveFileDialogProps, folder?: FileStat): Promise<URI | undefined>
|
|
36
|
-
|
|
37
|
-
}
|
|
38
|
-
|
|
39
|
-
@injectable()
|
|
40
|
-
export class DefaultFileDialogService implements FileDialogService {
|
|
41
|
-
|
|
42
|
-
@inject(EnvVariablesServer)
|
|
43
|
-
protected readonly environments: EnvVariablesServer;
|
|
44
|
-
|
|
45
|
-
@inject(FileService)
|
|
46
|
-
protected readonly fileService: FileService;
|
|
47
|
-
|
|
48
|
-
@inject(OpenFileDialogFactory) protected readonly openFileDialogFactory: OpenFileDialogFactory;
|
|
49
|
-
@inject(LabelProvider) protected readonly labelProvider: LabelProvider;
|
|
50
|
-
@inject(SaveFileDialogFactory) protected readonly saveFileDialogFactory: SaveFileDialogFactory;
|
|
51
|
-
@inject(UserWorkingDirectoryProvider) protected readonly rootProvider: UserWorkingDirectoryProvider;
|
|
52
|
-
|
|
53
|
-
async showOpenDialog(props: OpenFileDialogProps & { canSelectMany: true }, folder?: FileStat): Promise<MaybeArray<URI> | undefined>;
|
|
54
|
-
async showOpenDialog(props: OpenFileDialogProps, folder?: FileStat): Promise<URI | undefined>;
|
|
55
|
-
async showOpenDialog(props: OpenFileDialogProps, folder?: FileStat): Promise<MaybeArray<URI> | undefined> {
|
|
56
|
-
const title = props.title || nls.localizeByDefault('Open');
|
|
57
|
-
const rootNode = await this.getRootNode(folder);
|
|
58
|
-
if (rootNode) {
|
|
59
|
-
const dialog = this.openFileDialogFactory(Object.assign(props, { title }));
|
|
60
|
-
await dialog.model.navigateTo(rootNode);
|
|
61
|
-
const value = await dialog.open();
|
|
62
|
-
if (value) {
|
|
63
|
-
if (!Array.isArray(value)) {
|
|
64
|
-
return value.uri;
|
|
65
|
-
}
|
|
66
|
-
return value.map(node => node.uri);
|
|
67
|
-
}
|
|
68
|
-
}
|
|
69
|
-
return undefined;
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
async showSaveDialog(props: SaveFileDialogProps, folder?: FileStat): Promise<URI | undefined> {
|
|
73
|
-
const title = props.title || nls.localizeByDefault('Save');
|
|
74
|
-
const rootNode = await this.getRootNode(folder);
|
|
75
|
-
if (rootNode) {
|
|
76
|
-
const dialog = this.saveFileDialogFactory(Object.assign(props, { title }));
|
|
77
|
-
await dialog.model.navigateTo(rootNode);
|
|
78
|
-
return dialog.open();
|
|
79
|
-
}
|
|
80
|
-
return undefined;
|
|
81
|
-
}
|
|
82
|
-
|
|
83
|
-
protected async getRootNode(folderToOpen?: FileStat): Promise<DirNode | undefined> {
|
|
84
|
-
const folderExists = folderToOpen
|
|
85
|
-
&& folderToOpen.resource.scheme !== UNTITLED_SCHEME
|
|
86
|
-
&& await this.fileService.exists(folderToOpen.resource);
|
|
87
|
-
const folder = folderToOpen && folderExists ? folderToOpen : {
|
|
88
|
-
resource: await this.rootProvider.getUserWorkingDir(),
|
|
89
|
-
isDirectory: true
|
|
90
|
-
};
|
|
91
|
-
const folderUri = folder.resource;
|
|
92
|
-
const rootUri = folder.isDirectory ? folderUri : folderUri.parent;
|
|
93
|
-
try {
|
|
94
|
-
const rootStat = await this.fileService.resolve(rootUri);
|
|
95
|
-
return DirNode.createRoot(rootStat);
|
|
96
|
-
} catch { }
|
|
97
|
-
return undefined;
|
|
98
|
-
}
|
|
99
|
-
}
|
|
1
|
+
// *****************************************************************************
|
|
2
|
+
// Copyright (C) 2018 Ericsson and others.
|
|
3
|
+
//
|
|
4
|
+
// This program and the accompanying materials are made available under the
|
|
5
|
+
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
6
|
+
// http://www.eclipse.org/legal/epl-2.0.
|
|
7
|
+
//
|
|
8
|
+
// This Source Code may also be made available under the following Secondary
|
|
9
|
+
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
10
|
+
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
11
|
+
// with the GNU Classpath Exception which is available at
|
|
12
|
+
// https://www.gnu.org/software/classpath/license.html.
|
|
13
|
+
//
|
|
14
|
+
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
15
|
+
// *****************************************************************************
|
|
16
|
+
|
|
17
|
+
import { injectable, inject } from '@theia/core/shared/inversify';
|
|
18
|
+
import URI from '@theia/core/lib/common/uri';
|
|
19
|
+
import { MaybeArray, UNTITLED_SCHEME, nls } from '@theia/core/lib/common';
|
|
20
|
+
import { LabelProvider } from '@theia/core/lib/browser';
|
|
21
|
+
import { FileStat } from '../../common/files';
|
|
22
|
+
import { DirNode } from '../file-tree';
|
|
23
|
+
import { OpenFileDialogFactory, OpenFileDialogProps, SaveFileDialogFactory, SaveFileDialogProps } from './file-dialog';
|
|
24
|
+
import { FileService } from '../file-service';
|
|
25
|
+
import { EnvVariablesServer } from '@theia/core/lib/common/env-variables';
|
|
26
|
+
import { UserWorkingDirectoryProvider } from '@theia/core/lib/browser/user-working-directory-provider';
|
|
27
|
+
|
|
28
|
+
export const FileDialogService = Symbol('FileDialogService');
|
|
29
|
+
export interface FileDialogService {
|
|
30
|
+
|
|
31
|
+
showOpenDialog(props: OpenFileDialogProps & { canSelectMany: true }, folder?: FileStat): Promise<MaybeArray<URI> | undefined>;
|
|
32
|
+
showOpenDialog(props: OpenFileDialogProps, folder?: FileStat): Promise<URI | undefined>;
|
|
33
|
+
showOpenDialog(props: OpenFileDialogProps, folder?: FileStat): Promise<MaybeArray<URI> | undefined>;
|
|
34
|
+
|
|
35
|
+
showSaveDialog(props: SaveFileDialogProps, folder?: FileStat): Promise<URI | undefined>
|
|
36
|
+
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
@injectable()
|
|
40
|
+
export class DefaultFileDialogService implements FileDialogService {
|
|
41
|
+
|
|
42
|
+
@inject(EnvVariablesServer)
|
|
43
|
+
protected readonly environments: EnvVariablesServer;
|
|
44
|
+
|
|
45
|
+
@inject(FileService)
|
|
46
|
+
protected readonly fileService: FileService;
|
|
47
|
+
|
|
48
|
+
@inject(OpenFileDialogFactory) protected readonly openFileDialogFactory: OpenFileDialogFactory;
|
|
49
|
+
@inject(LabelProvider) protected readonly labelProvider: LabelProvider;
|
|
50
|
+
@inject(SaveFileDialogFactory) protected readonly saveFileDialogFactory: SaveFileDialogFactory;
|
|
51
|
+
@inject(UserWorkingDirectoryProvider) protected readonly rootProvider: UserWorkingDirectoryProvider;
|
|
52
|
+
|
|
53
|
+
async showOpenDialog(props: OpenFileDialogProps & { canSelectMany: true }, folder?: FileStat): Promise<MaybeArray<URI> | undefined>;
|
|
54
|
+
async showOpenDialog(props: OpenFileDialogProps, folder?: FileStat): Promise<URI | undefined>;
|
|
55
|
+
async showOpenDialog(props: OpenFileDialogProps, folder?: FileStat): Promise<MaybeArray<URI> | undefined> {
|
|
56
|
+
const title = props.title || nls.localizeByDefault('Open');
|
|
57
|
+
const rootNode = await this.getRootNode(folder);
|
|
58
|
+
if (rootNode) {
|
|
59
|
+
const dialog = this.openFileDialogFactory(Object.assign(props, { title }));
|
|
60
|
+
await dialog.model.navigateTo(rootNode);
|
|
61
|
+
const value = await dialog.open();
|
|
62
|
+
if (value) {
|
|
63
|
+
if (!Array.isArray(value)) {
|
|
64
|
+
return value.uri;
|
|
65
|
+
}
|
|
66
|
+
return value.map(node => node.uri);
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
return undefined;
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
async showSaveDialog(props: SaveFileDialogProps, folder?: FileStat): Promise<URI | undefined> {
|
|
73
|
+
const title = props.title || nls.localizeByDefault('Save');
|
|
74
|
+
const rootNode = await this.getRootNode(folder);
|
|
75
|
+
if (rootNode) {
|
|
76
|
+
const dialog = this.saveFileDialogFactory(Object.assign(props, { title }));
|
|
77
|
+
await dialog.model.navigateTo(rootNode);
|
|
78
|
+
return dialog.open();
|
|
79
|
+
}
|
|
80
|
+
return undefined;
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
protected async getRootNode(folderToOpen?: FileStat): Promise<DirNode | undefined> {
|
|
84
|
+
const folderExists = folderToOpen
|
|
85
|
+
&& folderToOpen.resource.scheme !== UNTITLED_SCHEME
|
|
86
|
+
&& await this.fileService.exists(folderToOpen.resource);
|
|
87
|
+
const folder = folderToOpen && folderExists ? folderToOpen : {
|
|
88
|
+
resource: await this.rootProvider.getUserWorkingDir(),
|
|
89
|
+
isDirectory: true
|
|
90
|
+
};
|
|
91
|
+
const folderUri = folder.resource;
|
|
92
|
+
const rootUri = folder.isDirectory ? folderUri : folderUri.parent;
|
|
93
|
+
try {
|
|
94
|
+
const rootStat = await this.fileService.resolve(rootUri);
|
|
95
|
+
return DirNode.createRoot(rootStat);
|
|
96
|
+
} catch { }
|
|
97
|
+
return undefined;
|
|
98
|
+
}
|
|
99
|
+
}
|
|
@@ -1,100 +1,100 @@
|
|
|
1
|
-
// *****************************************************************************
|
|
2
|
-
// Copyright (C) 2018 Red Hat, Inc. and others.
|
|
3
|
-
//
|
|
4
|
-
// This program and the accompanying materials are made available under the
|
|
5
|
-
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
6
|
-
// http://www.eclipse.org/legal/epl-2.0.
|
|
7
|
-
//
|
|
8
|
-
// This Source Code may also be made available under the following Secondary
|
|
9
|
-
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
10
|
-
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
11
|
-
// with the GNU Classpath Exception which is available at
|
|
12
|
-
// https://www.gnu.org/software/classpath/license.html.
|
|
13
|
-
//
|
|
14
|
-
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
15
|
-
// *****************************************************************************
|
|
16
|
-
|
|
17
|
-
import { ReactRenderer } from '@theia/core/lib/browser/widgets/react-renderer';
|
|
18
|
-
import { FileDialogTree } from './file-dialog-tree';
|
|
19
|
-
import * as React from '@theia/core/shared/react';
|
|
20
|
-
import { inject, injectable } from '@theia/core/shared/inversify';
|
|
21
|
-
|
|
22
|
-
export const FILE_TREE_FILTERS_LIST_CLASS = 'theia-FileTreeFiltersList';
|
|
23
|
-
|
|
24
|
-
/**
|
|
25
|
-
* A set of file filters that are used by the dialog. Each entry is a human readable label,
|
|
26
|
-
* like "TypeScript", and an array of extensions, e.g.
|
|
27
|
-
* ```ts
|
|
28
|
-
* {
|
|
29
|
-
* 'Images': ['png', 'jpg']
|
|
30
|
-
* 'TypeScript': ['ts', 'tsx']
|
|
31
|
-
* }
|
|
32
|
-
* ```
|
|
33
|
-
*/
|
|
34
|
-
export class FileDialogTreeFilters {
|
|
35
|
-
[name: string]: string[];
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
export const FileDialogTreeFiltersRendererFactory = Symbol('FileDialogTreeFiltersRendererFactory');
|
|
39
|
-
export interface FileDialogTreeFiltersRendererFactory {
|
|
40
|
-
(options: FileDialogTreeFiltersRendererOptions): FileDialogTreeFiltersRenderer;
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
export const FileDialogTreeFiltersRendererOptions = Symbol('FileDialogTreeFiltersRendererOptions');
|
|
44
|
-
export interface FileDialogTreeFiltersRendererOptions {
|
|
45
|
-
suppliedFilters: FileDialogTreeFilters;
|
|
46
|
-
fileDialogTree: FileDialogTree;
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
@injectable()
|
|
50
|
-
export class FileDialogTreeFiltersRenderer extends ReactRenderer {
|
|
51
|
-
|
|
52
|
-
readonly appliedFilters: FileDialogTreeFilters;
|
|
53
|
-
readonly suppliedFilters: FileDialogTreeFilters;
|
|
54
|
-
readonly fileDialogTree: FileDialogTree;
|
|
55
|
-
|
|
56
|
-
constructor(
|
|
57
|
-
@inject(FileDialogTreeFiltersRendererOptions) readonly options: FileDialogTreeFiltersRendererOptions
|
|
58
|
-
) {
|
|
59
|
-
super();
|
|
60
|
-
this.suppliedFilters = options.suppliedFilters;
|
|
61
|
-
this.fileDialogTree = options.fileDialogTree;
|
|
62
|
-
this.appliedFilters = { ...this.suppliedFilters, 'All Files': [], };
|
|
63
|
-
}
|
|
64
|
-
|
|
65
|
-
protected readonly handleFilterChanged = (e: React.ChangeEvent<HTMLSelectElement>) => this.onFilterChanged(e);
|
|
66
|
-
|
|
67
|
-
protected override doRender(): React.ReactNode {
|
|
68
|
-
if (!this.appliedFilters) {
|
|
69
|
-
return undefined;
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
const options = Object.keys(this.appliedFilters).map(value => this.renderLocation(value));
|
|
73
|
-
return <select className={'theia-select ' + FILE_TREE_FILTERS_LIST_CLASS} onChange={this.handleFilterChanged}>{...options}</select>;
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
protected renderLocation(value: string): React.ReactNode {
|
|
77
|
-
return <option value={value} key={value}>{value}</option>;
|
|
78
|
-
}
|
|
79
|
-
|
|
80
|
-
protected onFilterChanged(e: React.ChangeEvent<HTMLSelectElement>): void {
|
|
81
|
-
const locationList = this.locationList;
|
|
82
|
-
if (locationList) {
|
|
83
|
-
const value = locationList.value;
|
|
84
|
-
const filters = this.appliedFilters[value];
|
|
85
|
-
this.fileDialogTree.setFilter(filters);
|
|
86
|
-
}
|
|
87
|
-
|
|
88
|
-
e.preventDefault();
|
|
89
|
-
e.stopPropagation();
|
|
90
|
-
}
|
|
91
|
-
|
|
92
|
-
get locationList(): HTMLSelectElement | undefined {
|
|
93
|
-
const locationList = this.host.getElementsByClassName(FILE_TREE_FILTERS_LIST_CLASS)[0];
|
|
94
|
-
if (locationList instanceof HTMLSelectElement) {
|
|
95
|
-
return locationList;
|
|
96
|
-
}
|
|
97
|
-
return undefined;
|
|
98
|
-
}
|
|
99
|
-
|
|
100
|
-
}
|
|
1
|
+
// *****************************************************************************
|
|
2
|
+
// Copyright (C) 2018 Red Hat, Inc. and others.
|
|
3
|
+
//
|
|
4
|
+
// This program and the accompanying materials are made available under the
|
|
5
|
+
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
6
|
+
// http://www.eclipse.org/legal/epl-2.0.
|
|
7
|
+
//
|
|
8
|
+
// This Source Code may also be made available under the following Secondary
|
|
9
|
+
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
10
|
+
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
11
|
+
// with the GNU Classpath Exception which is available at
|
|
12
|
+
// https://www.gnu.org/software/classpath/license.html.
|
|
13
|
+
//
|
|
14
|
+
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
15
|
+
// *****************************************************************************
|
|
16
|
+
|
|
17
|
+
import { ReactRenderer } from '@theia/core/lib/browser/widgets/react-renderer';
|
|
18
|
+
import { FileDialogTree } from './file-dialog-tree';
|
|
19
|
+
import * as React from '@theia/core/shared/react';
|
|
20
|
+
import { inject, injectable } from '@theia/core/shared/inversify';
|
|
21
|
+
|
|
22
|
+
export const FILE_TREE_FILTERS_LIST_CLASS = 'theia-FileTreeFiltersList';
|
|
23
|
+
|
|
24
|
+
/**
|
|
25
|
+
* A set of file filters that are used by the dialog. Each entry is a human readable label,
|
|
26
|
+
* like "TypeScript", and an array of extensions, e.g.
|
|
27
|
+
* ```ts
|
|
28
|
+
* {
|
|
29
|
+
* 'Images': ['png', 'jpg']
|
|
30
|
+
* 'TypeScript': ['ts', 'tsx']
|
|
31
|
+
* }
|
|
32
|
+
* ```
|
|
33
|
+
*/
|
|
34
|
+
export class FileDialogTreeFilters {
|
|
35
|
+
[name: string]: string[];
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
export const FileDialogTreeFiltersRendererFactory = Symbol('FileDialogTreeFiltersRendererFactory');
|
|
39
|
+
export interface FileDialogTreeFiltersRendererFactory {
|
|
40
|
+
(options: FileDialogTreeFiltersRendererOptions): FileDialogTreeFiltersRenderer;
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
export const FileDialogTreeFiltersRendererOptions = Symbol('FileDialogTreeFiltersRendererOptions');
|
|
44
|
+
export interface FileDialogTreeFiltersRendererOptions {
|
|
45
|
+
suppliedFilters: FileDialogTreeFilters;
|
|
46
|
+
fileDialogTree: FileDialogTree;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
@injectable()
|
|
50
|
+
export class FileDialogTreeFiltersRenderer extends ReactRenderer {
|
|
51
|
+
|
|
52
|
+
readonly appliedFilters: FileDialogTreeFilters;
|
|
53
|
+
readonly suppliedFilters: FileDialogTreeFilters;
|
|
54
|
+
readonly fileDialogTree: FileDialogTree;
|
|
55
|
+
|
|
56
|
+
constructor(
|
|
57
|
+
@inject(FileDialogTreeFiltersRendererOptions) readonly options: FileDialogTreeFiltersRendererOptions
|
|
58
|
+
) {
|
|
59
|
+
super();
|
|
60
|
+
this.suppliedFilters = options.suppliedFilters;
|
|
61
|
+
this.fileDialogTree = options.fileDialogTree;
|
|
62
|
+
this.appliedFilters = { ...this.suppliedFilters, 'All Files': [], };
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
protected readonly handleFilterChanged = (e: React.ChangeEvent<HTMLSelectElement>) => this.onFilterChanged(e);
|
|
66
|
+
|
|
67
|
+
protected override doRender(): React.ReactNode {
|
|
68
|
+
if (!this.appliedFilters) {
|
|
69
|
+
return undefined;
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
const options = Object.keys(this.appliedFilters).map(value => this.renderLocation(value));
|
|
73
|
+
return <select className={'theia-select ' + FILE_TREE_FILTERS_LIST_CLASS} onChange={this.handleFilterChanged}>{...options}</select>;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
protected renderLocation(value: string): React.ReactNode {
|
|
77
|
+
return <option value={value} key={value}>{value}</option>;
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
protected onFilterChanged(e: React.ChangeEvent<HTMLSelectElement>): void {
|
|
81
|
+
const locationList = this.locationList;
|
|
82
|
+
if (locationList) {
|
|
83
|
+
const value = locationList.value;
|
|
84
|
+
const filters = this.appliedFilters[value];
|
|
85
|
+
this.fileDialogTree.setFilter(filters);
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
e.preventDefault();
|
|
89
|
+
e.stopPropagation();
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
get locationList(): HTMLSelectElement | undefined {
|
|
93
|
+
const locationList = this.host.getElementsByClassName(FILE_TREE_FILTERS_LIST_CLASS)[0];
|
|
94
|
+
if (locationList instanceof HTMLSelectElement) {
|
|
95
|
+
return locationList;
|
|
96
|
+
}
|
|
97
|
+
return undefined;
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
}
|
|
@@ -1,89 +1,89 @@
|
|
|
1
|
-
// *****************************************************************************
|
|
2
|
-
// Copyright (C) 2018 Red Hat, Inc. and others.
|
|
3
|
-
//
|
|
4
|
-
// This program and the accompanying materials are made available under the
|
|
5
|
-
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
6
|
-
// http://www.eclipse.org/legal/epl-2.0.
|
|
7
|
-
//
|
|
8
|
-
// This Source Code may also be made available under the following Secondary
|
|
9
|
-
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
10
|
-
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
11
|
-
// with the GNU Classpath Exception which is available at
|
|
12
|
-
// https://www.gnu.org/software/classpath/license.html.
|
|
13
|
-
//
|
|
14
|
-
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
15
|
-
// *****************************************************************************
|
|
16
|
-
|
|
17
|
-
import { injectable } from '@theia/core/shared/inversify';
|
|
18
|
-
import { DirNode, FileTree } from '../file-tree';
|
|
19
|
-
import { TreeNode, CompositeTreeNode } from '@theia/core/lib/browser/tree/tree';
|
|
20
|
-
import { FileStat } from '../../common/files';
|
|
21
|
-
|
|
22
|
-
@injectable()
|
|
23
|
-
export class FileDialogTree extends FileTree {
|
|
24
|
-
protected _showHidden = false;
|
|
25
|
-
set showHidden(show: boolean) {
|
|
26
|
-
this._showHidden = show;
|
|
27
|
-
this.refresh();
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
get showHidden(): boolean {
|
|
31
|
-
return this._showHidden;
|
|
32
|
-
}
|
|
33
|
-
|
|
34
|
-
protected isHiddenFile = (fileStat: FileStat): boolean => {
|
|
35
|
-
const { name } = fileStat;
|
|
36
|
-
const filename = name ?? '';
|
|
37
|
-
const isHidden = filename.startsWith('.');
|
|
38
|
-
return isHidden;
|
|
39
|
-
};
|
|
40
|
-
/**
|
|
41
|
-
* Extensions for files to be shown
|
|
42
|
-
*/
|
|
43
|
-
protected fileExtensions: string[] = [];
|
|
44
|
-
|
|
45
|
-
/**
|
|
46
|
-
* Sets extensions for filtering files
|
|
47
|
-
*
|
|
48
|
-
* @param fileExtensions array of extensions
|
|
49
|
-
*/
|
|
50
|
-
setFilter(fileExtensions: string[]): void {
|
|
51
|
-
this.fileExtensions = fileExtensions.slice();
|
|
52
|
-
this.refresh();
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
protected override async toNodes(fileStat: FileStat, parent: CompositeTreeNode): Promise<TreeNode[]> {
|
|
56
|
-
if (!fileStat.children) {
|
|
57
|
-
return [];
|
|
58
|
-
}
|
|
59
|
-
|
|
60
|
-
const result = await Promise.all(
|
|
61
|
-
fileStat.children
|
|
62
|
-
.filter(child => this.isVisible(child))
|
|
63
|
-
.map(child => this.toNode(child, parent))
|
|
64
|
-
);
|
|
65
|
-
|
|
66
|
-
return result.sort(DirNode.compare);
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
/**
|
|
70
|
-
* Determines whether file or folder can be shown
|
|
71
|
-
*
|
|
72
|
-
* @param fileStat resource to check
|
|
73
|
-
*/
|
|
74
|
-
protected isVisible(fileStat: FileStat): boolean {
|
|
75
|
-
if (!this._showHidden && this.isHiddenFile(fileStat)) {
|
|
76
|
-
return false;
|
|
77
|
-
}
|
|
78
|
-
if (fileStat.isDirectory) {
|
|
79
|
-
return true;
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
if (this.fileExtensions.length === 0) {
|
|
83
|
-
return true;
|
|
84
|
-
}
|
|
85
|
-
|
|
86
|
-
return !this.fileExtensions.every(value => fileStat.resource.path.ext !== '.' + value);
|
|
87
|
-
}
|
|
88
|
-
|
|
89
|
-
}
|
|
1
|
+
// *****************************************************************************
|
|
2
|
+
// Copyright (C) 2018 Red Hat, Inc. and others.
|
|
3
|
+
//
|
|
4
|
+
// This program and the accompanying materials are made available under the
|
|
5
|
+
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
6
|
+
// http://www.eclipse.org/legal/epl-2.0.
|
|
7
|
+
//
|
|
8
|
+
// This Source Code may also be made available under the following Secondary
|
|
9
|
+
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
10
|
+
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
11
|
+
// with the GNU Classpath Exception which is available at
|
|
12
|
+
// https://www.gnu.org/software/classpath/license.html.
|
|
13
|
+
//
|
|
14
|
+
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
15
|
+
// *****************************************************************************
|
|
16
|
+
|
|
17
|
+
import { injectable } from '@theia/core/shared/inversify';
|
|
18
|
+
import { DirNode, FileTree } from '../file-tree';
|
|
19
|
+
import { TreeNode, CompositeTreeNode } from '@theia/core/lib/browser/tree/tree';
|
|
20
|
+
import { FileStat } from '../../common/files';
|
|
21
|
+
|
|
22
|
+
@injectable()
|
|
23
|
+
export class FileDialogTree extends FileTree {
|
|
24
|
+
protected _showHidden = false;
|
|
25
|
+
set showHidden(show: boolean) {
|
|
26
|
+
this._showHidden = show;
|
|
27
|
+
this.refresh();
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
get showHidden(): boolean {
|
|
31
|
+
return this._showHidden;
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
protected isHiddenFile = (fileStat: FileStat): boolean => {
|
|
35
|
+
const { name } = fileStat;
|
|
36
|
+
const filename = name ?? '';
|
|
37
|
+
const isHidden = filename.startsWith('.');
|
|
38
|
+
return isHidden;
|
|
39
|
+
};
|
|
40
|
+
/**
|
|
41
|
+
* Extensions for files to be shown
|
|
42
|
+
*/
|
|
43
|
+
protected fileExtensions: string[] = [];
|
|
44
|
+
|
|
45
|
+
/**
|
|
46
|
+
* Sets extensions for filtering files
|
|
47
|
+
*
|
|
48
|
+
* @param fileExtensions array of extensions
|
|
49
|
+
*/
|
|
50
|
+
setFilter(fileExtensions: string[]): void {
|
|
51
|
+
this.fileExtensions = fileExtensions.slice();
|
|
52
|
+
this.refresh();
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
protected override async toNodes(fileStat: FileStat, parent: CompositeTreeNode): Promise<TreeNode[]> {
|
|
56
|
+
if (!fileStat.children) {
|
|
57
|
+
return [];
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
const result = await Promise.all(
|
|
61
|
+
fileStat.children
|
|
62
|
+
.filter(child => this.isVisible(child))
|
|
63
|
+
.map(child => this.toNode(child, parent))
|
|
64
|
+
);
|
|
65
|
+
|
|
66
|
+
return result.sort(DirNode.compare);
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
/**
|
|
70
|
+
* Determines whether file or folder can be shown
|
|
71
|
+
*
|
|
72
|
+
* @param fileStat resource to check
|
|
73
|
+
*/
|
|
74
|
+
protected isVisible(fileStat: FileStat): boolean {
|
|
75
|
+
if (!this._showHidden && this.isHiddenFile(fileStat)) {
|
|
76
|
+
return false;
|
|
77
|
+
}
|
|
78
|
+
if (fileStat.isDirectory) {
|
|
79
|
+
return true;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
if (this.fileExtensions.length === 0) {
|
|
83
|
+
return true;
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
return !this.fileExtensions.every(value => fileStat.resource.path.ext !== '.' + value);
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
}
|