@theia/filesystem 1.48.0 → 1.48.1
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/lib/browser/breadcrumbs/filepath-breadcrumb.d.ts +15 -15
- package/lib/browser/breadcrumbs/filepath-breadcrumb.js +41 -41
- package/lib/browser/breadcrumbs/filepath-breadcrumbs-container.d.ts +13 -13
- package/lib/browser/breadcrumbs/filepath-breadcrumbs-container.js +70 -70
- package/lib/browser/breadcrumbs/filepath-breadcrumbs-contribution.d.ts +27 -27
- package/lib/browser/breadcrumbs/filepath-breadcrumbs-contribution.js +118 -118
- package/lib/browser/download/file-download-command-contribution.d.ts +18 -18
- package/lib/browser/download/file-download-command-contribution.js +75 -75
- package/lib/browser/download/file-download-frontend-module.d.ts +3 -3
- package/lib/browser/download/file-download-frontend-module.js +25 -25
- package/lib/browser/download/file-download-service.d.ts +28 -28
- package/lib/browser/download/file-download-service.js +167 -167
- package/lib/browser/file-dialog/file-dialog-container.d.ts +5 -5
- package/lib/browser/file-dialog/file-dialog-container.js +60 -60
- package/lib/browser/file-dialog/file-dialog-hidden-files-renderer.d.ts +15 -15
- package/lib/browser/file-dialog/file-dialog-hidden-files-renderer.js +59 -59
- package/lib/browser/file-dialog/file-dialog-model.d.ts +25 -25
- package/lib/browser/file-dialog/file-dialog-model.js +100 -100
- package/lib/browser/file-dialog/file-dialog-module.d.ts +3 -3
- package/lib/browser/file-dialog/file-dialog-module.js +45 -45
- package/lib/browser/file-dialog/file-dialog-service.d.ts +32 -32
- package/lib/browser/file-dialog/file-dialog-service.js +101 -101
- package/lib/browser/file-dialog/file-dialog-tree-filters-renderer.d.ts +39 -39
- package/lib/browser/file-dialog/file-dialog-tree-filters-renderer.js +81 -81
- package/lib/browser/file-dialog/file-dialog-tree.d.ts +26 -26
- package/lib/browser/file-dialog/file-dialog-tree.js +83 -83
- package/lib/browser/file-dialog/file-dialog-widget.d.ts +15 -15
- package/lib/browser/file-dialog/file-dialog-widget.js +75 -75
- package/lib/browser/file-dialog/file-dialog.d.ts +129 -129
- package/lib/browser/file-dialog/file-dialog.js +351 -351
- package/lib/browser/file-dialog/index.d.ts +4 -4
- package/lib/browser/file-dialog/index.js +22 -22
- package/lib/browser/file-resource.d.ts +67 -67
- package/lib/browser/file-resource.js +369 -369
- package/lib/browser/file-selection.d.ts +14 -14
- package/lib/browser/file-selection.js +36 -36
- package/lib/browser/file-service.d.ts +425 -425
- package/lib/browser/file-service.js +1366 -1366
- package/lib/browser/file-tree/file-tree-container.d.ts +3 -3
- package/lib/browser/file-tree/file-tree-container.js +35 -35
- package/lib/browser/file-tree/file-tree-decorator-adapter.d.ts +30 -30
- package/lib/browser/file-tree/file-tree-decorator-adapter.js +169 -169
- package/lib/browser/file-tree/file-tree-label-provider.d.ts +12 -12
- package/lib/browser/file-tree/file-tree-label-provider.js +54 -54
- package/lib/browser/file-tree/file-tree-model.d.ts +37 -37
- package/lib/browser/file-tree/file-tree-model.js +217 -217
- package/lib/browser/file-tree/file-tree-widget.d.ts +44 -44
- package/lib/browser/file-tree/file-tree-widget.js +317 -317
- package/lib/browser/file-tree/file-tree.d.ts +46 -46
- package/lib/browser/file-tree/file-tree.js +176 -176
- package/lib/browser/file-tree/index.d.ts +6 -6
- package/lib/browser/file-tree/index.js +24 -24
- package/lib/browser/file-upload-service.d.ts +109 -109
- package/lib/browser/file-upload-service.js +434 -434
- package/lib/browser/filesystem-frontend-contribution.d.ts +74 -74
- package/lib/browser/filesystem-frontend-contribution.js +356 -356
- package/lib/browser/filesystem-frontend-module.d.ts +5 -5
- package/lib/browser/filesystem-frontend-module.js +64 -64
- package/lib/browser/filesystem-preferences.d.ts +29 -29
- package/lib/browser/filesystem-preferences.js +110 -110
- package/lib/browser/filesystem-save-resource-service.d.ts +29 -29
- package/lib/browser/filesystem-save-resource-service.js +135 -135
- package/lib/browser/filesystem-watcher-error-handler.d.ts +10 -10
- package/lib/browser/filesystem-watcher-error-handler.js +61 -61
- package/lib/browser/index.d.ts +5 -5
- package/lib/browser/index.js +23 -23
- package/lib/browser/location/index.d.ts +2 -2
- package/lib/browser/location/index.js +20 -20
- package/lib/browser/location/location-renderer.d.ts +100 -100
- package/lib/browser/location/location-renderer.js +343 -343
- package/lib/browser/location/location-service.d.ts +5 -5
- package/lib/browser/location/location-service.js +17 -17
- package/lib/browser/remote-file-service-contribution.d.ts +6 -6
- package/lib/browser/remote-file-service-contribution.js +39 -39
- package/lib/browser-only/browser-only-filesystem-frontend-module.d.ts +3 -3
- package/lib/browser-only/browser-only-filesystem-frontend-module.js +40 -40
- package/lib/browser-only/browser-only-filesystem-provider-server.d.ts +11 -11
- package/lib/browser-only/browser-only-filesystem-provider-server.js +39 -39
- package/lib/browser-only/browserfs-filesystem-initialization.d.ts +12 -12
- package/lib/browser-only/browserfs-filesystem-initialization.js +54 -54
- package/lib/browser-only/browserfs-filesystem-provider.d.ts +45 -45
- package/lib/browser-only/browserfs-filesystem-provider.js +439 -439
- package/lib/common/delegating-file-system-provider.d.ts +76 -76
- package/lib/common/delegating-file-system-provider.js +168 -168
- package/lib/common/download/file-download-data.d.ts +6 -6
- package/lib/common/download/file-download-data.js +26 -26
- package/lib/common/file-upload.d.ts +1 -1
- package/lib/common/file-upload.js +19 -19
- package/lib/common/files.d.ts +659 -659
- package/lib/common/files.js +355 -355
- package/lib/common/files.spec.d.ts +1 -1
- package/lib/common/files.spec.js +51 -51
- package/lib/common/filesystem-utils.d.ts +14 -14
- package/lib/common/filesystem-utils.js +63 -63
- package/lib/common/filesystem-utils.spec.d.ts +1 -1
- package/lib/common/filesystem-utils.spec.js +378 -378
- package/lib/common/filesystem-watcher-protocol.d.ts +71 -71
- package/lib/common/filesystem-watcher-protocol.js +20 -20
- package/lib/common/filesystem.d.ts +22 -22
- package/lib/common/filesystem.js +42 -42
- package/lib/common/index.d.ts +2 -2
- package/lib/common/index.js +20 -20
- package/lib/common/io.d.ts +19 -19
- package/lib/common/io.js +110 -110
- package/lib/common/remote-file-system-provider.d.ts +173 -173
- package/lib/common/remote-file-system-provider.js +435 -435
- package/lib/electron-browser/file-dialog/electron-file-dialog-module.d.ts +3 -3
- package/lib/electron-browser/file-dialog/electron-file-dialog-module.js +24 -24
- package/lib/electron-browser/file-dialog/electron-file-dialog-service.d.ts +19 -19
- package/lib/electron-browser/file-dialog/electron-file-dialog-service.js +148 -148
- package/lib/electron-browser/preload.d.ts +1 -1
- package/lib/electron-browser/preload.js +30 -30
- package/lib/electron-common/electron-api.d.ts +34 -34
- package/lib/electron-common/electron-api.js +20 -20
- package/lib/electron-main/electron-api-main.d.ts +5 -5
- package/lib/electron-main/electron-api-main.js +73 -73
- package/lib/electron-main/electron-main-module.d.ts +3 -3
- package/lib/electron-main/electron-main-module.js +24 -24
- package/lib/node/disk-file-system-provider.d.ts +72 -72
- package/lib/node/disk-file-system-provider.js +787 -787
- package/lib/node/disk-file-system-provider.spec.d.ts +1 -1
- package/lib/node/disk-file-system-provider.spec.js +122 -122
- package/lib/node/download/directory-archiver.d.ts +9 -9
- package/lib/node/download/directory-archiver.js +127 -127
- package/lib/node/download/directory-archiver.spec.d.ts +1 -1
- package/lib/node/download/directory-archiver.spec.js +97 -97
- package/lib/node/download/file-download-backend-module.d.ts +3 -3
- package/lib/node/download/file-download-backend-module.js +32 -32
- package/lib/node/download/file-download-cache.d.ts +21 -21
- package/lib/node/download/file-download-cache.js +82 -82
- package/lib/node/download/file-download-endpoint.d.ts +11 -11
- package/lib/node/download/file-download-endpoint.js +67 -67
- package/lib/node/download/file-download-handler.d.ts +50 -50
- package/lib/node/download/file-download-handler.js +307 -307
- package/lib/node/download/test/mock-directory-archiver.d.ts +7 -7
- package/lib/node/download/test/mock-directory-archiver.js +29 -29
- package/lib/node/file-change-collection.d.ts +22 -22
- package/lib/node/file-change-collection.js +77 -77
- package/lib/node/file-change-collection.spec.d.ts +1 -1
- package/lib/node/file-change-collection.spec.js +90 -90
- package/lib/node/filesystem-backend-module.d.ts +26 -26
- package/lib/node/filesystem-backend-module.js +120 -120
- package/lib/node/filesystem-watcher-client.d.ts +23 -23
- package/lib/node/filesystem-watcher-client.js +75 -75
- package/lib/node/filesystem-watcher-dispatcher.d.ts +23 -23
- package/lib/node/filesystem-watcher-dispatcher.js +80 -80
- package/lib/node/node-file-upload-service.d.ts +16 -16
- package/lib/node/node-file-upload-service.js +79 -79
- package/lib/node/nsfw-watcher/index.d.ts +3 -3
- package/lib/node/nsfw-watcher/index.js +39 -39
- package/lib/node/nsfw-watcher/nsfw-filesystem-service.d.ts +191 -191
- package/lib/node/nsfw-watcher/nsfw-filesystem-service.js +405 -405
- package/lib/node/nsfw-watcher/nsfw-filesystem-watcher.spec.d.ts +1 -1
- package/lib/node/nsfw-watcher/nsfw-filesystem-watcher.spec.js +151 -151
- package/lib/node/nsfw-watcher/nsfw-options.d.ts +6 -6
- package/lib/node/nsfw-watcher/nsfw-options.js +22 -22
- package/package.json +4 -4
- 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.ts +390 -390
- package/src/browser/file-selection.ts +44 -44
- package/src/browser/file-service.ts +1841 -1841
- 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 +539 -539
- package/src/browser/filesystem-frontend-contribution.ts +381 -381
- package/src/browser/filesystem-frontend-module.ts +77 -77
- package/src/browser/filesystem-preferences.ts +139 -139
- package/src/browser/filesystem-save-resource-service.ts +125 -125
- 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 +404 -404
- 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 +996 -996
- 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,344 +1,344 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
// *****************************************************************************
|
|
3
|
-
// Copyright (C) 2017 TypeFox and others.
|
|
4
|
-
//
|
|
5
|
-
// This program and the accompanying materials are made available under the
|
|
6
|
-
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
7
|
-
// http://www.eclipse.org/legal/epl-2.0.
|
|
8
|
-
//
|
|
9
|
-
// This Source Code may also be made available under the following Secondary
|
|
10
|
-
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
11
|
-
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
12
|
-
// with the GNU Classpath Exception which is available at
|
|
13
|
-
// https://www.gnu.org/software/classpath/license.html.
|
|
14
|
-
//
|
|
15
|
-
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
16
|
-
// *****************************************************************************
|
|
17
|
-
var LocationListRenderer_1;
|
|
18
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
19
|
-
exports.LocationListRenderer = exports.LocationListRendererOptions = exports.LocationListRendererFactory = void 0;
|
|
20
|
-
const tslib_1 = require("tslib");
|
|
21
|
-
const uri_1 = require("@theia/core/lib/common/uri");
|
|
22
|
-
const React = require("@theia/core/shared/react");
|
|
23
|
-
const file_service_1 = require("../file-service");
|
|
24
|
-
const common_1 = require("@theia/core/lib/common");
|
|
25
|
-
const inversify_1 = require("@theia/core/shared/inversify");
|
|
26
|
-
const env_variables_1 = require("@theia/core/lib/common/env-variables");
|
|
27
|
-
const react_renderer_1 = require("@theia/core/lib/browser/widgets/react-renderer");
|
|
28
|
-
const browser_1 = require("@theia/core/lib/browser");
|
|
29
|
-
class ResolvedDirectoryCache {
|
|
30
|
-
constructor(fileService) {
|
|
31
|
-
this.fileService = fileService;
|
|
32
|
-
this.pendingResolvedDirectories = new Map();
|
|
33
|
-
this.cachedDirectories = new Map();
|
|
34
|
-
this.directoryResolvedEmitter = new common_1.Emitter();
|
|
35
|
-
this.onDirectoryDidResolve = this.directoryResolvedEmitter.event;
|
|
36
|
-
}
|
|
37
|
-
tryResolveChildDirectories(inputAsURI) {
|
|
38
|
-
const parentDirectory = inputAsURI.path.dir.toString();
|
|
39
|
-
const cachedDirectories = this.cachedDirectories.get(parentDirectory);
|
|
40
|
-
const pendingDirectories = this.pendingResolvedDirectories.get(parentDirectory);
|
|
41
|
-
if (cachedDirectories) {
|
|
42
|
-
return cachedDirectories;
|
|
43
|
-
}
|
|
44
|
-
else if (!pendingDirectories) {
|
|
45
|
-
this.pendingResolvedDirectories.set(parentDirectory, this.createResolutionPromise(parentDirectory));
|
|
46
|
-
}
|
|
47
|
-
return undefined;
|
|
48
|
-
}
|
|
49
|
-
async createResolutionPromise(directoryToResolve) {
|
|
50
|
-
return this.fileService.resolve(new uri_1.default(directoryToResolve)).then(({ children }) => {
|
|
51
|
-
if (children) {
|
|
52
|
-
const childDirectories = children.filter(child => child.isDirectory)
|
|
53
|
-
.map(directory => `${directory.resource.path}/`);
|
|
54
|
-
this.cachedDirectories.set(directoryToResolve, childDirectories);
|
|
55
|
-
this.directoryResolvedEmitter.fire({ parent: directoryToResolve, children: childDirectories });
|
|
56
|
-
}
|
|
57
|
-
}).catch(e => {
|
|
58
|
-
// no-op
|
|
59
|
-
});
|
|
60
|
-
}
|
|
61
|
-
}
|
|
62
|
-
exports.LocationListRendererFactory = Symbol('LocationListRendererFactory');
|
|
63
|
-
exports.LocationListRendererOptions = Symbol('LocationListRendererOptions');
|
|
64
|
-
let LocationListRenderer = LocationListRenderer_1 = class LocationListRenderer extends react_renderer_1.ReactRenderer {
|
|
65
|
-
constructor(options) {
|
|
66
|
-
super(options.host);
|
|
67
|
-
this.options = options;
|
|
68
|
-
this.toDisposeOnNewCache = new common_1.DisposableCollection();
|
|
69
|
-
this._doShowTextInput = false;
|
|
70
|
-
this.doAttemptAutocomplete = true;
|
|
71
|
-
this.doAfterRender = () => {
|
|
72
|
-
const locationList = this.locationList;
|
|
73
|
-
const locationListTextInput = this.locationTextInput;
|
|
74
|
-
if (locationList) {
|
|
75
|
-
const currentLocation = this.service.location;
|
|
76
|
-
locationList.value = currentLocation ? currentLocation.toString() : '';
|
|
77
|
-
}
|
|
78
|
-
else if (locationListTextInput) {
|
|
79
|
-
locationListTextInput.focus();
|
|
80
|
-
}
|
|
81
|
-
};
|
|
82
|
-
this.handleLocationChanged = (e) => this.onLocationChanged(e);
|
|
83
|
-
this.handleTextInputOnChange = (e) => this.trySuggestDirectory(e);
|
|
84
|
-
this.handleTextInputKeyDown = (e) => this.handleControlKeys(e);
|
|
85
|
-
this.handleIconKeyDown = (e) => this.toggleInputOnKeyDown(e);
|
|
86
|
-
this.handleTextInputOnBlur = () => this.toggleToSelectInput();
|
|
87
|
-
this.handleTextInputMouseDown = (e) => this.toggleToTextInputOnMouseDown(e);
|
|
88
|
-
this.service = options.model;
|
|
89
|
-
this.doLoadDrives();
|
|
90
|
-
this.doAfterRender = this.doAfterRender.bind(this);
|
|
91
|
-
}
|
|
92
|
-
get doShowTextInput() {
|
|
93
|
-
return this._doShowTextInput;
|
|
94
|
-
}
|
|
95
|
-
set doShowTextInput(doShow) {
|
|
96
|
-
this._doShowTextInput = doShow;
|
|
97
|
-
if (doShow) {
|
|
98
|
-
this.initResolveDirectoryCache();
|
|
99
|
-
}
|
|
100
|
-
}
|
|
101
|
-
init() {
|
|
102
|
-
this.doInit();
|
|
103
|
-
}
|
|
104
|
-
async doInit() {
|
|
105
|
-
const homeDirWithPrefix = await this.variablesServer.getHomeDirUri();
|
|
106
|
-
this.homeDir = (new uri_1.default(homeDirWithPrefix)).path.toString();
|
|
107
|
-
}
|
|
108
|
-
render() {
|
|
109
|
-
this.hostRoot.render(this.doRender());
|
|
110
|
-
}
|
|
111
|
-
initResolveDirectoryCache() {
|
|
112
|
-
this.toDisposeOnNewCache.dispose();
|
|
113
|
-
this.directoryCache = new ResolvedDirectoryCache(this.fileService);
|
|
114
|
-
this.toDisposeOnNewCache.push(this.directoryCache.onDirectoryDidResolve(({ parent, children }) => {
|
|
115
|
-
if (this.locationTextInput) {
|
|
116
|
-
const expandedPath = common_1.Path.untildify(this.locationTextInput.value, this.homeDir);
|
|
117
|
-
const inputParent = (new uri_1.default(expandedPath)).path.dir.toString();
|
|
118
|
-
if (inputParent === parent) {
|
|
119
|
-
this.tryRenderFirstMatch(this.locationTextInput, children);
|
|
120
|
-
}
|
|
121
|
-
}
|
|
122
|
-
}));
|
|
123
|
-
}
|
|
124
|
-
doRender() {
|
|
125
|
-
return (React.createElement(React.Fragment, null,
|
|
126
|
-
this.renderInputIcon(),
|
|
127
|
-
this.doShowTextInput
|
|
128
|
-
? this.renderTextInput()
|
|
129
|
-
: this.renderSelectInput()));
|
|
130
|
-
}
|
|
131
|
-
renderInputIcon() {
|
|
132
|
-
return (React.createElement("span", {
|
|
133
|
-
// onMouseDown is used since it will fire before 'onBlur'. This prevents
|
|
134
|
-
// a re-render when textinput is in focus and user clicks toggle icon
|
|
135
|
-
onMouseDown: this.handleTextInputMouseDown, onKeyDown: this.handleIconKeyDown, className: LocationListRenderer_1.Styles.LOCATION_INPUT_TOGGLE_CLASS, tabIndex: 0, id: `${this.doShowTextInput ? 'text-input' : 'select-input'}`, title: this.doShowTextInput
|
|
136
|
-
? LocationListRenderer_1.Tooltips.TOGGLE_SELECT_INPUT
|
|
137
|
-
: LocationListRenderer_1.Tooltips.TOGGLE_TEXT_INPUT, ref: this.doAfterRender },
|
|
138
|
-
React.createElement("i", { className: (0, browser_1.codicon)(this.doShowTextInput ? 'folder-opened' : 'edit') })));
|
|
139
|
-
}
|
|
140
|
-
renderTextInput() {
|
|
141
|
-
var _a;
|
|
142
|
-
return (React.createElement("input", { className: 'theia-select ' + LocationListRenderer_1.Styles.LOCATION_TEXT_INPUT_CLASS, defaultValue: (_a = this.service.location) === null || _a === void 0 ? void 0 : _a.path.fsPath(), onBlur: this.handleTextInputOnBlur, onChange: this.handleTextInputOnChange, onKeyDown: this.handleTextInputKeyDown, spellCheck: false }));
|
|
143
|
-
}
|
|
144
|
-
renderSelectInput() {
|
|
145
|
-
const options = this.collectLocations().map(value => this.renderLocation(value));
|
|
146
|
-
return (React.createElement("select", { className: `theia-select ${LocationListRenderer_1.Styles.LOCATION_LIST_CLASS}`, onChange: this.handleLocationChanged }, ...options));
|
|
147
|
-
}
|
|
148
|
-
toggleInputOnKeyDown(e) {
|
|
149
|
-
if (e.key === 'Enter') {
|
|
150
|
-
this.doShowTextInput = true;
|
|
151
|
-
this.render();
|
|
152
|
-
}
|
|
153
|
-
}
|
|
154
|
-
toggleToTextInputOnMouseDown(e) {
|
|
155
|
-
if (e.currentTarget.id === 'select-input') {
|
|
156
|
-
e.preventDefault();
|
|
157
|
-
this.doShowTextInput = true;
|
|
158
|
-
this.render();
|
|
159
|
-
}
|
|
160
|
-
}
|
|
161
|
-
toggleToSelectInput() {
|
|
162
|
-
if (this.doShowTextInput) {
|
|
163
|
-
this.doShowTextInput = false;
|
|
164
|
-
this.render();
|
|
165
|
-
}
|
|
166
|
-
}
|
|
167
|
-
/**
|
|
168
|
-
* Collects the available locations based on the currently selected, and appends the available drives to it.
|
|
169
|
-
*/
|
|
170
|
-
collectLocations() {
|
|
171
|
-
const location = this.service.location;
|
|
172
|
-
const locations = (!!location ? location.allLocations : []).map(uri => ({ uri }));
|
|
173
|
-
if (this._drives) {
|
|
174
|
-
const drives = this._drives.map(uri => ({ uri, isDrive: true }));
|
|
175
|
-
// `URI.allLocations` returns with the URI without the trailing slash unlike `FileUri.create(fsPath)`.
|
|
176
|
-
// to be able to compare file:///path/to/resource with file:///path/to/resource/.
|
|
177
|
-
const toUriString = (uri) => {
|
|
178
|
-
const toString = uri.toString();
|
|
179
|
-
return toString.endsWith('/') ? toString.slice(0, -1) : toString;
|
|
180
|
-
};
|
|
181
|
-
drives.forEach(drive => {
|
|
182
|
-
const index = locations.findIndex(loc => toUriString(loc.uri) === toUriString(drive.uri));
|
|
183
|
-
// Ignore drives which are already discovered as a location based on the current model root URI.
|
|
184
|
-
if (index === -1) {
|
|
185
|
-
// Make sure, it does not have the trailing slash.
|
|
186
|
-
locations.push({ uri: new uri_1.default(toUriString(drive.uri)), isDrive: true });
|
|
187
|
-
}
|
|
188
|
-
else {
|
|
189
|
-
// This is necessary for Windows to be able to show `/e:/` as a drive and `c:` as "non-drive" in the same way.
|
|
190
|
-
// `URI.path.toString()` Vs. `URI.displayName` behaves a bit differently on Windows.
|
|
191
|
-
// https://github.com/eclipse-theia/theia/pull/3038#issuecomment-425944189
|
|
192
|
-
locations[index].isDrive = true;
|
|
193
|
-
}
|
|
194
|
-
});
|
|
195
|
-
}
|
|
196
|
-
this.doLoadDrives();
|
|
197
|
-
return locations;
|
|
198
|
-
}
|
|
199
|
-
/**
|
|
200
|
-
* Asynchronously loads the drives (if not yet available) and triggers a UI update on success with the new values.
|
|
201
|
-
*/
|
|
202
|
-
doLoadDrives() {
|
|
203
|
-
if (!this._drives) {
|
|
204
|
-
this.service.drives().then(drives => {
|
|
205
|
-
// If the `drives` are empty, something already went wrong.
|
|
206
|
-
if (drives.length > 0) {
|
|
207
|
-
this._drives = drives;
|
|
208
|
-
this.render();
|
|
209
|
-
}
|
|
210
|
-
});
|
|
211
|
-
}
|
|
212
|
-
}
|
|
213
|
-
renderLocation(location) {
|
|
214
|
-
const { uri, isDrive } = location;
|
|
215
|
-
const value = uri.toString();
|
|
216
|
-
return React.createElement("option", { value: value, key: uri.toString() }, isDrive ? uri.path.fsPath() : uri.displayName);
|
|
217
|
-
}
|
|
218
|
-
onLocationChanged(e) {
|
|
219
|
-
const locationList = this.locationList;
|
|
220
|
-
if (locationList) {
|
|
221
|
-
const value = locationList.value;
|
|
222
|
-
const uri = new uri_1.default(value);
|
|
223
|
-
this.trySetNewLocation(uri);
|
|
224
|
-
e.preventDefault();
|
|
225
|
-
e.stopPropagation();
|
|
226
|
-
}
|
|
227
|
-
}
|
|
228
|
-
trySetNewLocation(newLocation) {
|
|
229
|
-
var _a;
|
|
230
|
-
if (this.lastUniqueTextInputLocation === undefined) {
|
|
231
|
-
this.lastUniqueTextInputLocation = this.service.location;
|
|
232
|
-
}
|
|
233
|
-
// prevent consecutive repeated locations from being added to location history
|
|
234
|
-
if (((_a = this.lastUniqueTextInputLocation) === null || _a === void 0 ? void 0 : _a.path.toString()) !== newLocation.path.toString()) {
|
|
235
|
-
this.lastUniqueTextInputLocation = newLocation;
|
|
236
|
-
this.service.location = newLocation;
|
|
237
|
-
}
|
|
238
|
-
}
|
|
239
|
-
trySuggestDirectory(e) {
|
|
240
|
-
if (this.doAttemptAutocomplete) {
|
|
241
|
-
const inputElement = e.currentTarget;
|
|
242
|
-
const { value } = inputElement;
|
|
243
|
-
if ((value.startsWith('/') || value.startsWith('~/')) && value.slice(-1) !== '/') {
|
|
244
|
-
const expandedPath = common_1.Path.untildify(value, this.homeDir);
|
|
245
|
-
const valueAsURI = new uri_1.default(expandedPath);
|
|
246
|
-
const autocompleteDirectories = this.directoryCache.tryResolveChildDirectories(valueAsURI);
|
|
247
|
-
if (autocompleteDirectories) {
|
|
248
|
-
this.tryRenderFirstMatch(inputElement, autocompleteDirectories);
|
|
249
|
-
}
|
|
250
|
-
}
|
|
251
|
-
}
|
|
252
|
-
}
|
|
253
|
-
tryRenderFirstMatch(inputElement, children) {
|
|
254
|
-
const { value, selectionStart } = inputElement;
|
|
255
|
-
if (this.locationTextInput) {
|
|
256
|
-
const expandedPath = common_1.Path.untildify(value, this.homeDir);
|
|
257
|
-
const firstMatch = children === null || children === void 0 ? void 0 : children.find(child => child.includes(expandedPath));
|
|
258
|
-
if (firstMatch) {
|
|
259
|
-
const contractedPath = value.startsWith('~') ? common_1.Path.tildify(firstMatch, this.homeDir) : firstMatch;
|
|
260
|
-
this.locationTextInput.value = contractedPath;
|
|
261
|
-
this.locationTextInput.selectionStart = selectionStart;
|
|
262
|
-
this.locationTextInput.selectionEnd = firstMatch.length;
|
|
263
|
-
}
|
|
264
|
-
}
|
|
265
|
-
}
|
|
266
|
-
handleControlKeys(e) {
|
|
267
|
-
this.doAttemptAutocomplete = e.key !== 'Backspace';
|
|
268
|
-
if (e.key === 'Enter') {
|
|
269
|
-
const locationTextInput = this.locationTextInput;
|
|
270
|
-
if (locationTextInput) {
|
|
271
|
-
// expand '~' if present and remove extra whitespace and any trailing slashes or periods.
|
|
272
|
-
const sanitizedInput = locationTextInput.value.trim().replace(/[\/\\.]*$/, '');
|
|
273
|
-
const untildifiedInput = common_1.Path.untildify(sanitizedInput, this.homeDir);
|
|
274
|
-
const uri = new uri_1.default(untildifiedInput);
|
|
275
|
-
this.trySetNewLocation(uri);
|
|
276
|
-
this.toggleToSelectInput();
|
|
277
|
-
}
|
|
278
|
-
}
|
|
279
|
-
else if (e.key === 'Escape') {
|
|
280
|
-
this.toggleToSelectInput();
|
|
281
|
-
}
|
|
282
|
-
else if (e.key === 'Tab') {
|
|
283
|
-
e.preventDefault();
|
|
284
|
-
const textInput = this.locationTextInput;
|
|
285
|
-
if (textInput) {
|
|
286
|
-
textInput.selectionStart = textInput.value.length;
|
|
287
|
-
}
|
|
288
|
-
}
|
|
289
|
-
e.stopPropagation();
|
|
290
|
-
}
|
|
291
|
-
get locationList() {
|
|
292
|
-
const locationList = this.host.getElementsByClassName(LocationListRenderer_1.Styles.LOCATION_LIST_CLASS)[0];
|
|
293
|
-
if (locationList instanceof HTMLSelectElement) {
|
|
294
|
-
return locationList;
|
|
295
|
-
}
|
|
296
|
-
return undefined;
|
|
297
|
-
}
|
|
298
|
-
get locationTextInput() {
|
|
299
|
-
const locationTextInput = this.host.getElementsByClassName(LocationListRenderer_1.Styles.LOCATION_TEXT_INPUT_CLASS)[0];
|
|
300
|
-
if (locationTextInput instanceof HTMLInputElement) {
|
|
301
|
-
return locationTextInput;
|
|
302
|
-
}
|
|
303
|
-
return undefined;
|
|
304
|
-
}
|
|
305
|
-
dispose() {
|
|
306
|
-
super.dispose();
|
|
307
|
-
this.toDisposeOnNewCache.dispose();
|
|
308
|
-
}
|
|
309
|
-
};
|
|
310
|
-
(0, tslib_1.__decorate)([
|
|
311
|
-
(0, inversify_1.inject)(file_service_1.FileService),
|
|
312
|
-
(0, tslib_1.__metadata)("design:type", file_service_1.FileService)
|
|
313
|
-
], LocationListRenderer.prototype, "fileService", void 0);
|
|
314
|
-
(0, tslib_1.__decorate)([
|
|
315
|
-
(0, inversify_1.inject)(env_variables_1.EnvVariablesServer),
|
|
316
|
-
(0, tslib_1.__metadata)("design:type", Object)
|
|
317
|
-
], LocationListRenderer.prototype, "variablesServer", void 0);
|
|
318
|
-
(0, tslib_1.__decorate)([
|
|
319
|
-
(0, inversify_1.postConstruct)(),
|
|
320
|
-
(0, tslib_1.__metadata)("design:type", Function),
|
|
321
|
-
(0, tslib_1.__metadata)("design:paramtypes", []),
|
|
322
|
-
(0, tslib_1.__metadata)("design:returntype", void 0)
|
|
323
|
-
], LocationListRenderer.prototype, "init", null);
|
|
324
|
-
LocationListRenderer = LocationListRenderer_1 = (0, tslib_1.__decorate)([
|
|
325
|
-
(0, inversify_1.injectable)(),
|
|
326
|
-
(0, tslib_1.__param)(0, (0, inversify_1.inject)(exports.LocationListRendererOptions)),
|
|
327
|
-
(0, tslib_1.__metadata)("design:paramtypes", [Object])
|
|
328
|
-
], LocationListRenderer);
|
|
329
|
-
exports.LocationListRenderer = LocationListRenderer;
|
|
330
|
-
(function (LocationListRenderer) {
|
|
331
|
-
let Styles;
|
|
332
|
-
(function (Styles) {
|
|
333
|
-
Styles.LOCATION_LIST_CLASS = 'theia-LocationList';
|
|
334
|
-
Styles.LOCATION_INPUT_TOGGLE_CLASS = 'theia-LocationInputToggle';
|
|
335
|
-
Styles.LOCATION_TEXT_INPUT_CLASS = 'theia-LocationTextInput';
|
|
336
|
-
})(Styles = LocationListRenderer.Styles || (LocationListRenderer.Styles = {}));
|
|
337
|
-
let Tooltips;
|
|
338
|
-
(function (Tooltips) {
|
|
339
|
-
Tooltips.TOGGLE_TEXT_INPUT = 'Switch to text-based input';
|
|
340
|
-
Tooltips.TOGGLE_SELECT_INPUT = 'Switch to location list';
|
|
341
|
-
})(Tooltips = LocationListRenderer.Tooltips || (LocationListRenderer.Tooltips = {}));
|
|
342
|
-
})(LocationListRenderer = exports.LocationListRenderer || (exports.LocationListRenderer = {}));
|
|
343
|
-
exports.LocationListRenderer = LocationListRenderer;
|
|
1
|
+
"use strict";
|
|
2
|
+
// *****************************************************************************
|
|
3
|
+
// Copyright (C) 2017 TypeFox and others.
|
|
4
|
+
//
|
|
5
|
+
// This program and the accompanying materials are made available under the
|
|
6
|
+
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
7
|
+
// http://www.eclipse.org/legal/epl-2.0.
|
|
8
|
+
//
|
|
9
|
+
// This Source Code may also be made available under the following Secondary
|
|
10
|
+
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
11
|
+
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
12
|
+
// with the GNU Classpath Exception which is available at
|
|
13
|
+
// https://www.gnu.org/software/classpath/license.html.
|
|
14
|
+
//
|
|
15
|
+
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
16
|
+
// *****************************************************************************
|
|
17
|
+
var LocationListRenderer_1;
|
|
18
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
19
|
+
exports.LocationListRenderer = exports.LocationListRendererOptions = exports.LocationListRendererFactory = void 0;
|
|
20
|
+
const tslib_1 = require("tslib");
|
|
21
|
+
const uri_1 = require("@theia/core/lib/common/uri");
|
|
22
|
+
const React = require("@theia/core/shared/react");
|
|
23
|
+
const file_service_1 = require("../file-service");
|
|
24
|
+
const common_1 = require("@theia/core/lib/common");
|
|
25
|
+
const inversify_1 = require("@theia/core/shared/inversify");
|
|
26
|
+
const env_variables_1 = require("@theia/core/lib/common/env-variables");
|
|
27
|
+
const react_renderer_1 = require("@theia/core/lib/browser/widgets/react-renderer");
|
|
28
|
+
const browser_1 = require("@theia/core/lib/browser");
|
|
29
|
+
class ResolvedDirectoryCache {
|
|
30
|
+
constructor(fileService) {
|
|
31
|
+
this.fileService = fileService;
|
|
32
|
+
this.pendingResolvedDirectories = new Map();
|
|
33
|
+
this.cachedDirectories = new Map();
|
|
34
|
+
this.directoryResolvedEmitter = new common_1.Emitter();
|
|
35
|
+
this.onDirectoryDidResolve = this.directoryResolvedEmitter.event;
|
|
36
|
+
}
|
|
37
|
+
tryResolveChildDirectories(inputAsURI) {
|
|
38
|
+
const parentDirectory = inputAsURI.path.dir.toString();
|
|
39
|
+
const cachedDirectories = this.cachedDirectories.get(parentDirectory);
|
|
40
|
+
const pendingDirectories = this.pendingResolvedDirectories.get(parentDirectory);
|
|
41
|
+
if (cachedDirectories) {
|
|
42
|
+
return cachedDirectories;
|
|
43
|
+
}
|
|
44
|
+
else if (!pendingDirectories) {
|
|
45
|
+
this.pendingResolvedDirectories.set(parentDirectory, this.createResolutionPromise(parentDirectory));
|
|
46
|
+
}
|
|
47
|
+
return undefined;
|
|
48
|
+
}
|
|
49
|
+
async createResolutionPromise(directoryToResolve) {
|
|
50
|
+
return this.fileService.resolve(new uri_1.default(directoryToResolve)).then(({ children }) => {
|
|
51
|
+
if (children) {
|
|
52
|
+
const childDirectories = children.filter(child => child.isDirectory)
|
|
53
|
+
.map(directory => `${directory.resource.path}/`);
|
|
54
|
+
this.cachedDirectories.set(directoryToResolve, childDirectories);
|
|
55
|
+
this.directoryResolvedEmitter.fire({ parent: directoryToResolve, children: childDirectories });
|
|
56
|
+
}
|
|
57
|
+
}).catch(e => {
|
|
58
|
+
// no-op
|
|
59
|
+
});
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
exports.LocationListRendererFactory = Symbol('LocationListRendererFactory');
|
|
63
|
+
exports.LocationListRendererOptions = Symbol('LocationListRendererOptions');
|
|
64
|
+
let LocationListRenderer = LocationListRenderer_1 = class LocationListRenderer extends react_renderer_1.ReactRenderer {
|
|
65
|
+
constructor(options) {
|
|
66
|
+
super(options.host);
|
|
67
|
+
this.options = options;
|
|
68
|
+
this.toDisposeOnNewCache = new common_1.DisposableCollection();
|
|
69
|
+
this._doShowTextInput = false;
|
|
70
|
+
this.doAttemptAutocomplete = true;
|
|
71
|
+
this.doAfterRender = () => {
|
|
72
|
+
const locationList = this.locationList;
|
|
73
|
+
const locationListTextInput = this.locationTextInput;
|
|
74
|
+
if (locationList) {
|
|
75
|
+
const currentLocation = this.service.location;
|
|
76
|
+
locationList.value = currentLocation ? currentLocation.toString() : '';
|
|
77
|
+
}
|
|
78
|
+
else if (locationListTextInput) {
|
|
79
|
+
locationListTextInput.focus();
|
|
80
|
+
}
|
|
81
|
+
};
|
|
82
|
+
this.handleLocationChanged = (e) => this.onLocationChanged(e);
|
|
83
|
+
this.handleTextInputOnChange = (e) => this.trySuggestDirectory(e);
|
|
84
|
+
this.handleTextInputKeyDown = (e) => this.handleControlKeys(e);
|
|
85
|
+
this.handleIconKeyDown = (e) => this.toggleInputOnKeyDown(e);
|
|
86
|
+
this.handleTextInputOnBlur = () => this.toggleToSelectInput();
|
|
87
|
+
this.handleTextInputMouseDown = (e) => this.toggleToTextInputOnMouseDown(e);
|
|
88
|
+
this.service = options.model;
|
|
89
|
+
this.doLoadDrives();
|
|
90
|
+
this.doAfterRender = this.doAfterRender.bind(this);
|
|
91
|
+
}
|
|
92
|
+
get doShowTextInput() {
|
|
93
|
+
return this._doShowTextInput;
|
|
94
|
+
}
|
|
95
|
+
set doShowTextInput(doShow) {
|
|
96
|
+
this._doShowTextInput = doShow;
|
|
97
|
+
if (doShow) {
|
|
98
|
+
this.initResolveDirectoryCache();
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
init() {
|
|
102
|
+
this.doInit();
|
|
103
|
+
}
|
|
104
|
+
async doInit() {
|
|
105
|
+
const homeDirWithPrefix = await this.variablesServer.getHomeDirUri();
|
|
106
|
+
this.homeDir = (new uri_1.default(homeDirWithPrefix)).path.toString();
|
|
107
|
+
}
|
|
108
|
+
render() {
|
|
109
|
+
this.hostRoot.render(this.doRender());
|
|
110
|
+
}
|
|
111
|
+
initResolveDirectoryCache() {
|
|
112
|
+
this.toDisposeOnNewCache.dispose();
|
|
113
|
+
this.directoryCache = new ResolvedDirectoryCache(this.fileService);
|
|
114
|
+
this.toDisposeOnNewCache.push(this.directoryCache.onDirectoryDidResolve(({ parent, children }) => {
|
|
115
|
+
if (this.locationTextInput) {
|
|
116
|
+
const expandedPath = common_1.Path.untildify(this.locationTextInput.value, this.homeDir);
|
|
117
|
+
const inputParent = (new uri_1.default(expandedPath)).path.dir.toString();
|
|
118
|
+
if (inputParent === parent) {
|
|
119
|
+
this.tryRenderFirstMatch(this.locationTextInput, children);
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
}));
|
|
123
|
+
}
|
|
124
|
+
doRender() {
|
|
125
|
+
return (React.createElement(React.Fragment, null,
|
|
126
|
+
this.renderInputIcon(),
|
|
127
|
+
this.doShowTextInput
|
|
128
|
+
? this.renderTextInput()
|
|
129
|
+
: this.renderSelectInput()));
|
|
130
|
+
}
|
|
131
|
+
renderInputIcon() {
|
|
132
|
+
return (React.createElement("span", {
|
|
133
|
+
// onMouseDown is used since it will fire before 'onBlur'. This prevents
|
|
134
|
+
// a re-render when textinput is in focus and user clicks toggle icon
|
|
135
|
+
onMouseDown: this.handleTextInputMouseDown, onKeyDown: this.handleIconKeyDown, className: LocationListRenderer_1.Styles.LOCATION_INPUT_TOGGLE_CLASS, tabIndex: 0, id: `${this.doShowTextInput ? 'text-input' : 'select-input'}`, title: this.doShowTextInput
|
|
136
|
+
? LocationListRenderer_1.Tooltips.TOGGLE_SELECT_INPUT
|
|
137
|
+
: LocationListRenderer_1.Tooltips.TOGGLE_TEXT_INPUT, ref: this.doAfterRender },
|
|
138
|
+
React.createElement("i", { className: (0, browser_1.codicon)(this.doShowTextInput ? 'folder-opened' : 'edit') })));
|
|
139
|
+
}
|
|
140
|
+
renderTextInput() {
|
|
141
|
+
var _a;
|
|
142
|
+
return (React.createElement("input", { className: 'theia-select ' + LocationListRenderer_1.Styles.LOCATION_TEXT_INPUT_CLASS, defaultValue: (_a = this.service.location) === null || _a === void 0 ? void 0 : _a.path.fsPath(), onBlur: this.handleTextInputOnBlur, onChange: this.handleTextInputOnChange, onKeyDown: this.handleTextInputKeyDown, spellCheck: false }));
|
|
143
|
+
}
|
|
144
|
+
renderSelectInput() {
|
|
145
|
+
const options = this.collectLocations().map(value => this.renderLocation(value));
|
|
146
|
+
return (React.createElement("select", { className: `theia-select ${LocationListRenderer_1.Styles.LOCATION_LIST_CLASS}`, onChange: this.handleLocationChanged }, ...options));
|
|
147
|
+
}
|
|
148
|
+
toggleInputOnKeyDown(e) {
|
|
149
|
+
if (e.key === 'Enter') {
|
|
150
|
+
this.doShowTextInput = true;
|
|
151
|
+
this.render();
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
toggleToTextInputOnMouseDown(e) {
|
|
155
|
+
if (e.currentTarget.id === 'select-input') {
|
|
156
|
+
e.preventDefault();
|
|
157
|
+
this.doShowTextInput = true;
|
|
158
|
+
this.render();
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
toggleToSelectInput() {
|
|
162
|
+
if (this.doShowTextInput) {
|
|
163
|
+
this.doShowTextInput = false;
|
|
164
|
+
this.render();
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
/**
|
|
168
|
+
* Collects the available locations based on the currently selected, and appends the available drives to it.
|
|
169
|
+
*/
|
|
170
|
+
collectLocations() {
|
|
171
|
+
const location = this.service.location;
|
|
172
|
+
const locations = (!!location ? location.allLocations : []).map(uri => ({ uri }));
|
|
173
|
+
if (this._drives) {
|
|
174
|
+
const drives = this._drives.map(uri => ({ uri, isDrive: true }));
|
|
175
|
+
// `URI.allLocations` returns with the URI without the trailing slash unlike `FileUri.create(fsPath)`.
|
|
176
|
+
// to be able to compare file:///path/to/resource with file:///path/to/resource/.
|
|
177
|
+
const toUriString = (uri) => {
|
|
178
|
+
const toString = uri.toString();
|
|
179
|
+
return toString.endsWith('/') ? toString.slice(0, -1) : toString;
|
|
180
|
+
};
|
|
181
|
+
drives.forEach(drive => {
|
|
182
|
+
const index = locations.findIndex(loc => toUriString(loc.uri) === toUriString(drive.uri));
|
|
183
|
+
// Ignore drives which are already discovered as a location based on the current model root URI.
|
|
184
|
+
if (index === -1) {
|
|
185
|
+
// Make sure, it does not have the trailing slash.
|
|
186
|
+
locations.push({ uri: new uri_1.default(toUriString(drive.uri)), isDrive: true });
|
|
187
|
+
}
|
|
188
|
+
else {
|
|
189
|
+
// This is necessary for Windows to be able to show `/e:/` as a drive and `c:` as "non-drive" in the same way.
|
|
190
|
+
// `URI.path.toString()` Vs. `URI.displayName` behaves a bit differently on Windows.
|
|
191
|
+
// https://github.com/eclipse-theia/theia/pull/3038#issuecomment-425944189
|
|
192
|
+
locations[index].isDrive = true;
|
|
193
|
+
}
|
|
194
|
+
});
|
|
195
|
+
}
|
|
196
|
+
this.doLoadDrives();
|
|
197
|
+
return locations;
|
|
198
|
+
}
|
|
199
|
+
/**
|
|
200
|
+
* Asynchronously loads the drives (if not yet available) and triggers a UI update on success with the new values.
|
|
201
|
+
*/
|
|
202
|
+
doLoadDrives() {
|
|
203
|
+
if (!this._drives) {
|
|
204
|
+
this.service.drives().then(drives => {
|
|
205
|
+
// If the `drives` are empty, something already went wrong.
|
|
206
|
+
if (drives.length > 0) {
|
|
207
|
+
this._drives = drives;
|
|
208
|
+
this.render();
|
|
209
|
+
}
|
|
210
|
+
});
|
|
211
|
+
}
|
|
212
|
+
}
|
|
213
|
+
renderLocation(location) {
|
|
214
|
+
const { uri, isDrive } = location;
|
|
215
|
+
const value = uri.toString();
|
|
216
|
+
return React.createElement("option", { value: value, key: uri.toString() }, isDrive ? uri.path.fsPath() : uri.displayName);
|
|
217
|
+
}
|
|
218
|
+
onLocationChanged(e) {
|
|
219
|
+
const locationList = this.locationList;
|
|
220
|
+
if (locationList) {
|
|
221
|
+
const value = locationList.value;
|
|
222
|
+
const uri = new uri_1.default(value);
|
|
223
|
+
this.trySetNewLocation(uri);
|
|
224
|
+
e.preventDefault();
|
|
225
|
+
e.stopPropagation();
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
trySetNewLocation(newLocation) {
|
|
229
|
+
var _a;
|
|
230
|
+
if (this.lastUniqueTextInputLocation === undefined) {
|
|
231
|
+
this.lastUniqueTextInputLocation = this.service.location;
|
|
232
|
+
}
|
|
233
|
+
// prevent consecutive repeated locations from being added to location history
|
|
234
|
+
if (((_a = this.lastUniqueTextInputLocation) === null || _a === void 0 ? void 0 : _a.path.toString()) !== newLocation.path.toString()) {
|
|
235
|
+
this.lastUniqueTextInputLocation = newLocation;
|
|
236
|
+
this.service.location = newLocation;
|
|
237
|
+
}
|
|
238
|
+
}
|
|
239
|
+
trySuggestDirectory(e) {
|
|
240
|
+
if (this.doAttemptAutocomplete) {
|
|
241
|
+
const inputElement = e.currentTarget;
|
|
242
|
+
const { value } = inputElement;
|
|
243
|
+
if ((value.startsWith('/') || value.startsWith('~/')) && value.slice(-1) !== '/') {
|
|
244
|
+
const expandedPath = common_1.Path.untildify(value, this.homeDir);
|
|
245
|
+
const valueAsURI = new uri_1.default(expandedPath);
|
|
246
|
+
const autocompleteDirectories = this.directoryCache.tryResolveChildDirectories(valueAsURI);
|
|
247
|
+
if (autocompleteDirectories) {
|
|
248
|
+
this.tryRenderFirstMatch(inputElement, autocompleteDirectories);
|
|
249
|
+
}
|
|
250
|
+
}
|
|
251
|
+
}
|
|
252
|
+
}
|
|
253
|
+
tryRenderFirstMatch(inputElement, children) {
|
|
254
|
+
const { value, selectionStart } = inputElement;
|
|
255
|
+
if (this.locationTextInput) {
|
|
256
|
+
const expandedPath = common_1.Path.untildify(value, this.homeDir);
|
|
257
|
+
const firstMatch = children === null || children === void 0 ? void 0 : children.find(child => child.includes(expandedPath));
|
|
258
|
+
if (firstMatch) {
|
|
259
|
+
const contractedPath = value.startsWith('~') ? common_1.Path.tildify(firstMatch, this.homeDir) : firstMatch;
|
|
260
|
+
this.locationTextInput.value = contractedPath;
|
|
261
|
+
this.locationTextInput.selectionStart = selectionStart;
|
|
262
|
+
this.locationTextInput.selectionEnd = firstMatch.length;
|
|
263
|
+
}
|
|
264
|
+
}
|
|
265
|
+
}
|
|
266
|
+
handleControlKeys(e) {
|
|
267
|
+
this.doAttemptAutocomplete = e.key !== 'Backspace';
|
|
268
|
+
if (e.key === 'Enter') {
|
|
269
|
+
const locationTextInput = this.locationTextInput;
|
|
270
|
+
if (locationTextInput) {
|
|
271
|
+
// expand '~' if present and remove extra whitespace and any trailing slashes or periods.
|
|
272
|
+
const sanitizedInput = locationTextInput.value.trim().replace(/[\/\\.]*$/, '');
|
|
273
|
+
const untildifiedInput = common_1.Path.untildify(sanitizedInput, this.homeDir);
|
|
274
|
+
const uri = new uri_1.default(untildifiedInput);
|
|
275
|
+
this.trySetNewLocation(uri);
|
|
276
|
+
this.toggleToSelectInput();
|
|
277
|
+
}
|
|
278
|
+
}
|
|
279
|
+
else if (e.key === 'Escape') {
|
|
280
|
+
this.toggleToSelectInput();
|
|
281
|
+
}
|
|
282
|
+
else if (e.key === 'Tab') {
|
|
283
|
+
e.preventDefault();
|
|
284
|
+
const textInput = this.locationTextInput;
|
|
285
|
+
if (textInput) {
|
|
286
|
+
textInput.selectionStart = textInput.value.length;
|
|
287
|
+
}
|
|
288
|
+
}
|
|
289
|
+
e.stopPropagation();
|
|
290
|
+
}
|
|
291
|
+
get locationList() {
|
|
292
|
+
const locationList = this.host.getElementsByClassName(LocationListRenderer_1.Styles.LOCATION_LIST_CLASS)[0];
|
|
293
|
+
if (locationList instanceof HTMLSelectElement) {
|
|
294
|
+
return locationList;
|
|
295
|
+
}
|
|
296
|
+
return undefined;
|
|
297
|
+
}
|
|
298
|
+
get locationTextInput() {
|
|
299
|
+
const locationTextInput = this.host.getElementsByClassName(LocationListRenderer_1.Styles.LOCATION_TEXT_INPUT_CLASS)[0];
|
|
300
|
+
if (locationTextInput instanceof HTMLInputElement) {
|
|
301
|
+
return locationTextInput;
|
|
302
|
+
}
|
|
303
|
+
return undefined;
|
|
304
|
+
}
|
|
305
|
+
dispose() {
|
|
306
|
+
super.dispose();
|
|
307
|
+
this.toDisposeOnNewCache.dispose();
|
|
308
|
+
}
|
|
309
|
+
};
|
|
310
|
+
(0, tslib_1.__decorate)([
|
|
311
|
+
(0, inversify_1.inject)(file_service_1.FileService),
|
|
312
|
+
(0, tslib_1.__metadata)("design:type", file_service_1.FileService)
|
|
313
|
+
], LocationListRenderer.prototype, "fileService", void 0);
|
|
314
|
+
(0, tslib_1.__decorate)([
|
|
315
|
+
(0, inversify_1.inject)(env_variables_1.EnvVariablesServer),
|
|
316
|
+
(0, tslib_1.__metadata)("design:type", Object)
|
|
317
|
+
], LocationListRenderer.prototype, "variablesServer", void 0);
|
|
318
|
+
(0, tslib_1.__decorate)([
|
|
319
|
+
(0, inversify_1.postConstruct)(),
|
|
320
|
+
(0, tslib_1.__metadata)("design:type", Function),
|
|
321
|
+
(0, tslib_1.__metadata)("design:paramtypes", []),
|
|
322
|
+
(0, tslib_1.__metadata)("design:returntype", void 0)
|
|
323
|
+
], LocationListRenderer.prototype, "init", null);
|
|
324
|
+
LocationListRenderer = LocationListRenderer_1 = (0, tslib_1.__decorate)([
|
|
325
|
+
(0, inversify_1.injectable)(),
|
|
326
|
+
(0, tslib_1.__param)(0, (0, inversify_1.inject)(exports.LocationListRendererOptions)),
|
|
327
|
+
(0, tslib_1.__metadata)("design:paramtypes", [Object])
|
|
328
|
+
], LocationListRenderer);
|
|
329
|
+
exports.LocationListRenderer = LocationListRenderer;
|
|
330
|
+
(function (LocationListRenderer) {
|
|
331
|
+
let Styles;
|
|
332
|
+
(function (Styles) {
|
|
333
|
+
Styles.LOCATION_LIST_CLASS = 'theia-LocationList';
|
|
334
|
+
Styles.LOCATION_INPUT_TOGGLE_CLASS = 'theia-LocationInputToggle';
|
|
335
|
+
Styles.LOCATION_TEXT_INPUT_CLASS = 'theia-LocationTextInput';
|
|
336
|
+
})(Styles = LocationListRenderer.Styles || (LocationListRenderer.Styles = {}));
|
|
337
|
+
let Tooltips;
|
|
338
|
+
(function (Tooltips) {
|
|
339
|
+
Tooltips.TOGGLE_TEXT_INPUT = 'Switch to text-based input';
|
|
340
|
+
Tooltips.TOGGLE_SELECT_INPUT = 'Switch to location list';
|
|
341
|
+
})(Tooltips = LocationListRenderer.Tooltips || (LocationListRenderer.Tooltips = {}));
|
|
342
|
+
})(LocationListRenderer = exports.LocationListRenderer || (exports.LocationListRenderer = {}));
|
|
343
|
+
exports.LocationListRenderer = LocationListRenderer;
|
|
344
344
|
//# sourceMappingURL=location-renderer.js.map
|