@theia/filesystem 1.34.2 → 1.34.3

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