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