@theia/playwright 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 (159) hide show
  1. package/README.md +54 -54
  2. package/lib/index.d.ts +30 -30
  3. package/lib/index.js +48 -48
  4. package/lib/tests/theia-app.test.d.ts +1 -1
  5. package/lib/tests/theia-app.test.js +29 -29
  6. package/lib/tests/theia-application-shell.test.d.ts +1 -1
  7. package/lib/tests/theia-application-shell.test.js +57 -57
  8. package/lib/tests/theia-explorer-view.test.d.ts +1 -1
  9. package/lib/tests/theia-explorer-view.test.js +183 -183
  10. package/lib/tests/theia-getting-started.test.d.ts +1 -1
  11. package/lib/tests/theia-getting-started.test.js +45 -45
  12. package/lib/tests/theia-main-menu.test.d.ts +1 -1
  13. package/lib/tests/theia-main-menu.test.js +111 -111
  14. package/lib/tests/theia-output-view.test.d.ts +1 -1
  15. package/lib/tests/theia-output-view.test.js +78 -78
  16. package/lib/tests/theia-preference-view.test.d.ts +1 -1
  17. package/lib/tests/theia-preference-view.test.js +101 -101
  18. package/lib/tests/theia-problems-view.test.d.ts +1 -1
  19. package/lib/tests/theia-problems-view.test.js +54 -54
  20. package/lib/tests/theia-quick-command.test.d.ts +1 -1
  21. package/lib/tests/theia-quick-command.test.js +69 -69
  22. package/lib/tests/theia-sample-app.test.d.ts +1 -1
  23. package/lib/tests/theia-sample-app.test.js +57 -57
  24. package/lib/tests/theia-status-bar.test.d.ts +1 -1
  25. package/lib/tests/theia-status-bar.test.js +44 -44
  26. package/lib/tests/theia-terminal-view.test.d.ts +1 -1
  27. package/lib/tests/theia-terminal-view.test.js +78 -78
  28. package/lib/tests/theia-text-editor.test.d.ts +1 -1
  29. package/lib/tests/theia-text-editor.test.js +155 -155
  30. package/lib/tests/theia-toolbar.test.d.ts +1 -1
  31. package/lib/tests/theia-toolbar.test.js +61 -61
  32. package/lib/tests/theia-workspace.test.d.ts +1 -1
  33. package/lib/tests/theia-workspace.test.js +72 -72
  34. package/lib/theia-about-dialog.d.ts +4 -4
  35. package/lib/theia-about-dialog.js +26 -26
  36. package/lib/theia-app-loader.d.ts +19 -19
  37. package/lib/theia-app-loader.js +129 -129
  38. package/lib/theia-app.d.ts +50 -50
  39. package/lib/theia-app.js +153 -153
  40. package/lib/theia-context-menu.d.ts +8 -8
  41. package/lib/theia-context-menu.js +37 -37
  42. package/lib/theia-dialog.d.ts +28 -28
  43. package/lib/theia-dialog.js +99 -99
  44. package/lib/theia-editor.d.ts +9 -9
  45. package/lib/theia-editor.js +68 -68
  46. package/lib/theia-explorer-view.d.ts +47 -47
  47. package/lib/theia-explorer-view.js +273 -273
  48. package/lib/theia-main-menu.d.ts +12 -12
  49. package/lib/theia-main-menu.js +53 -53
  50. package/lib/theia-menu-item.d.ts +14 -14
  51. package/lib/theia-menu-item.js +66 -66
  52. package/lib/theia-menu.d.ts +16 -16
  53. package/lib/theia-menu.js +86 -86
  54. package/lib/theia-monaco-editor.d.ts +15 -15
  55. package/lib/theia-monaco-editor.js +75 -75
  56. package/lib/theia-notification-indicator.d.ts +7 -7
  57. package/lib/theia-notification-indicator.js +44 -44
  58. package/lib/theia-notification-overlay.d.ts +22 -22
  59. package/lib/theia-notification-overlay.js +79 -79
  60. package/lib/theia-output-channel.d.ts +24 -24
  61. package/lib/theia-output-channel.js +71 -71
  62. package/lib/theia-output-view.d.ts +9 -9
  63. package/lib/theia-output-view.js +81 -81
  64. package/lib/theia-page-object.d.ts +7 -7
  65. package/lib/theia-page-object.js +27 -27
  66. package/lib/theia-preference-view.d.ts +84 -84
  67. package/lib/theia-preference-view.js +209 -209
  68. package/lib/theia-problem-indicator.d.ts +8 -8
  69. package/lib/theia-problem-indicator.js +38 -38
  70. package/lib/theia-problem-view.d.ts +5 -5
  71. package/lib/theia-problem-view.js +30 -30
  72. package/lib/theia-quick-command-palette.d.ts +12 -12
  73. package/lib/theia-quick-command-palette.js +78 -78
  74. package/lib/theia-rename-dialog.d.ts +5 -5
  75. package/lib/theia-rename-dialog.js +34 -34
  76. package/lib/theia-status-bar.d.ts +13 -13
  77. package/lib/theia-status-bar.js +39 -39
  78. package/lib/theia-status-indicator.d.ts +10 -10
  79. package/lib/theia-status-indicator.js +48 -48
  80. package/lib/theia-terminal.d.ts +13 -13
  81. package/lib/theia-terminal.js +59 -59
  82. package/lib/theia-text-editor.d.ts +26 -26
  83. package/lib/theia-text-editor.js +120 -120
  84. package/lib/theia-toggle-bottom-indicator.d.ts +4 -4
  85. package/lib/theia-toggle-bottom-indicator.js +26 -26
  86. package/lib/theia-toolbar-item.d.ts +10 -10
  87. package/lib/theia-toolbar-item.js +39 -39
  88. package/lib/theia-toolbar.d.ts +19 -19
  89. package/lib/theia-toolbar.js +90 -90
  90. package/lib/theia-tree-node.d.ts +19 -19
  91. package/lib/theia-tree-node.js +72 -72
  92. package/lib/theia-view.d.ts +32 -32
  93. package/lib/theia-view.js +149 -149
  94. package/lib/theia-welcome-view.d.ts +5 -5
  95. package/lib/theia-welcome-view.js +30 -30
  96. package/lib/theia-workspace.d.ts +18 -18
  97. package/lib/theia-workspace.js +69 -69
  98. package/lib/util.d.ts +19 -19
  99. package/lib/util.js +93 -93
  100. package/package.json +2 -2
  101. package/src/index.ts +46 -46
  102. package/src/tests/resources/sample-files1/sample.txt +4 -4
  103. package/src/tests/resources/sample-files1/sampleFolder/sampleFolder1/sampleFolder1-1/sampleFile1-1-1.txt +1 -1
  104. package/src/tests/resources/sample-files1/sampleFolder/sampleFolder1/sampleFolder1-1/sampleFile1-1-2.txt +1 -1
  105. package/src/tests/resources/sample-files1/sampleFolder/sampleFolder1/sampleFolder1-2/sampleFile1-2-1.txt +1 -1
  106. package/src/tests/resources/sample-files1/sampleFolder/sampleFolder1/sampleFolder1-2/sampleFile1-2-2.txt +1 -1
  107. package/src/tests/resources/sample-files1/sampleFolder/sampleFolder2/sampleFolder2-1/sampleFile2-1-1.txt +1 -1
  108. package/src/tests/resources/sample-files1/sampleFolder/sampleFolder2/sampleFolder2-1/sampleFile2-1-2.txt +1 -1
  109. package/src/tests/resources/sample-files1/sampleFolder/sampleFolder2/sampleFolder2-2/sampleFile2-2-1.txt +1 -1
  110. package/src/tests/resources/sample-files1/sampleFolder/sampleFolder2/sampleFolder2-2/sampleFile2-2-2.txt +1 -1
  111. package/src/tests/resources/sample-files2/another-sample.txt +1 -1
  112. package/src/tests/theia-app.test.ts +33 -33
  113. package/src/tests/theia-application-shell.test.ts +67 -67
  114. package/src/tests/theia-explorer-view.test.ts +211 -211
  115. package/src/tests/theia-getting-started.test.ts +50 -50
  116. package/src/tests/theia-main-menu.test.ts +133 -133
  117. package/src/tests/theia-output-view.test.ts +85 -85
  118. package/src/tests/theia-preference-view.test.ts +122 -122
  119. package/src/tests/theia-problems-view.test.ts +64 -64
  120. package/src/tests/theia-quick-command.test.ts +80 -80
  121. package/src/tests/theia-sample-app.test.ts +66 -66
  122. package/src/tests/theia-status-bar.test.ts +52 -52
  123. package/src/tests/theia-terminal-view.test.ts +90 -90
  124. package/src/tests/theia-text-editor.test.ts +187 -187
  125. package/src/tests/theia-toolbar.test.ts +69 -69
  126. package/src/tests/theia-workspace.test.ts +80 -80
  127. package/src/theia-about-dialog.ts +26 -26
  128. package/src/theia-app-loader.ts +167 -167
  129. package/src/theia-app.ts +188 -188
  130. package/src/theia-context-menu.ts +42 -42
  131. package/src/theia-dialog.ts +114 -114
  132. package/src/theia-editor.ts +73 -73
  133. package/src/theia-explorer-view.ts +311 -311
  134. package/src/theia-main-menu.ts +54 -54
  135. package/src/theia-menu-item.ts +75 -75
  136. package/src/theia-menu.ts +96 -96
  137. package/src/theia-monaco-editor.ts +83 -83
  138. package/src/theia-notification-indicator.ts +44 -44
  139. package/src/theia-notification-overlay.ts +94 -94
  140. package/src/theia-output-channel.ts +88 -88
  141. package/src/theia-output-view.ts +87 -87
  142. package/src/theia-page-object.ts +29 -29
  143. package/src/theia-preference-view.ts +240 -240
  144. package/src/theia-problem-indicator.ts +37 -37
  145. package/src/theia-problem-view.ts +30 -30
  146. package/src/theia-quick-command-palette.ts +81 -81
  147. package/src/theia-rename-dialog.ts +35 -35
  148. package/src/theia-status-bar.ts +44 -44
  149. package/src/theia-status-indicator.ts +50 -50
  150. package/src/theia-terminal.ts +69 -69
  151. package/src/theia-text-editor.ts +141 -141
  152. package/src/theia-toggle-bottom-indicator.ts +21 -21
  153. package/src/theia-toolbar-item.ts +41 -41
  154. package/src/theia-toolbar.ts +99 -99
  155. package/src/theia-tree-node.ts +81 -81
  156. package/src/theia-view.ts +177 -177
  157. package/src/theia-welcome-view.ts +31 -31
  158. package/src/theia-workspace.ts +76 -76
  159. package/src/util.ts +91 -91
@@ -1,274 +1,274 @@
1
- "use strict";
2
- // *****************************************************************************
3
- // Copyright (C) 2021 logi.cals GmbH, EclipseSource and others.
4
- //
5
- // This program and the accompanying materials are made available under the
6
- // terms of the Eclipse Public License v. 2.0 which is available at
7
- // http://www.eclipse.org/legal/epl-2.0.
8
- //
9
- // This Source Code may also be made available under the following Secondary
10
- // Licenses when the conditions for such availability set forth in the Eclipse
11
- // Public License v. 2.0 are satisfied: GNU General Public License, version 2
12
- // with the GNU Classpath Exception which is available at
13
- // https://www.gnu.org/software/classpath/license.html.
14
- //
15
- // SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
16
- // *****************************************************************************
17
- Object.defineProperty(exports, "__esModule", { value: true });
18
- exports.TheiaExplorerView = exports.DOT_FILES_FILTER = exports.TheiaExplorerFileStatNode = void 0;
19
- const theia_dialog_1 = require("./theia-dialog");
20
- const theia_rename_dialog_1 = require("./theia-rename-dialog");
21
- const theia_tree_node_1 = require("./theia-tree-node");
22
- const theia_view_1 = require("./theia-view");
23
- const util_1 = require("./util");
24
- const TheiaExplorerViewData = {
25
- tabSelector: '#shell-tab-explorer-view-container',
26
- viewSelector: '#explorer-view-container--files',
27
- viewName: 'Explorer'
28
- };
29
- class TheiaExplorerFileStatNode extends theia_tree_node_1.TheiaTreeNode {
30
- constructor(elementHandle, explorerView) {
31
- super(elementHandle, explorerView.app);
32
- this.elementHandle = elementHandle;
33
- this.explorerView = explorerView;
34
- }
35
- async absolutePath() {
36
- return this.elementHandle.getAttribute('title');
37
- }
38
- async isFile() {
39
- return !await this.isFolder();
40
- }
41
- async isFolder() {
42
- return (0, util_1.elementContainsClass)(this.elementHandle, 'theia-DirNode');
43
- }
44
- async getMenuItemByNamePath(names, nodeSegmentLabel) {
45
- const contextMenu = nodeSegmentLabel ? await this.openContextMenuOnSegment(nodeSegmentLabel) : await this.openContextMenu();
46
- const menuItem = await contextMenu.menuItemByNamePath(...names);
47
- if (!menuItem) {
48
- throw Error('MenuItem could not be retrieved by path');
49
- }
50
- return menuItem;
51
- }
52
- }
53
- exports.TheiaExplorerFileStatNode = TheiaExplorerFileStatNode;
54
- const DOT_FILES_FILTER = async (node) => {
55
- const label = await node.label();
56
- return label ? !label.startsWith('.') : true;
57
- };
58
- exports.DOT_FILES_FILTER = DOT_FILES_FILTER;
59
- class TheiaExplorerView extends theia_view_1.TheiaView {
60
- constructor(app) {
61
- super(TheiaExplorerViewData, app);
62
- }
63
- async activate() {
64
- await super.activate();
65
- const viewElement = await this.viewElement();
66
- await (viewElement === null || viewElement === void 0 ? void 0 : viewElement.waitForSelector('.theia-TreeContainer'));
67
- }
68
- async refresh() {
69
- await this.clickButton('navigator.refresh');
70
- }
71
- async collapseAll() {
72
- await this.clickButton('navigator.collapse.all');
73
- }
74
- async clickButton(id) {
75
- await this.activate();
76
- const viewElement = await this.viewElement();
77
- await (viewElement === null || viewElement === void 0 ? void 0 : viewElement.hover());
78
- const button = await (viewElement === null || viewElement === void 0 ? void 0 : viewElement.waitForSelector(`#${(0, util_1.normalizeId)(id)}`));
79
- await (button === null || button === void 0 ? void 0 : button.click());
80
- }
81
- async visibleFileStatNodes(filterPredicate = (_ => Promise.resolve(true))) {
82
- const viewElement = await this.viewElement();
83
- const handles = await (viewElement === null || viewElement === void 0 ? void 0 : viewElement.$$('.theia-FileStatNode'));
84
- if (handles) {
85
- const nodes = handles.map(handle => new TheiaExplorerFileStatNode(handle, this));
86
- const filteredNodes = [];
87
- for (const node of nodes) {
88
- if ((await filterPredicate(node)) === true) {
89
- filteredNodes.push(node);
90
- }
91
- }
92
- return filteredNodes;
93
- }
94
- return [];
95
- }
96
- async getFileStatNodeByLabel(label, compact = false) {
97
- const file = await this.fileStatNode(label, compact);
98
- if (!file) {
99
- throw Error('File stat node could not be retrieved by path fragments');
100
- }
101
- return file;
102
- }
103
- async fileStatNode(filePath, compact = false) {
104
- return compact ? this.compactFileStatNode(filePath) : this.fileStatNodeBySegments(...filePath.split('/'));
105
- }
106
- async fileStatNodeBySegments(...pathFragments) {
107
- await super.activate();
108
- const viewElement = await this.viewElement();
109
- let currentTreeNode = undefined;
110
- let fragmentsSoFar = '';
111
- for (let index = 0; index < pathFragments.length; index++) {
112
- const fragment = pathFragments[index];
113
- fragmentsSoFar += index !== 0 ? '/' : '';
114
- fragmentsSoFar += fragment;
115
- const selector = this.treeNodeSelector(fragmentsSoFar);
116
- const nextTreeNode = await (viewElement === null || viewElement === void 0 ? void 0 : viewElement.waitForSelector(selector, { state: 'visible' }));
117
- if (!nextTreeNode) {
118
- throw new Error(`Tree node '${selector}' not found in explorer`);
119
- }
120
- currentTreeNode = new TheiaExplorerFileStatNode(nextTreeNode, this);
121
- if (index < pathFragments.length - 1 && await currentTreeNode.isCollapsed()) {
122
- await currentTreeNode.expand();
123
- }
124
- }
125
- return currentTreeNode;
126
- }
127
- async compactFileStatNode(path) {
128
- // default setting `explorer.compactFolders=true` renders folders in a compact form - single child folders will be compressed in a combined tree element
129
- await super.activate();
130
- const viewElement = await this.viewElement();
131
- // check if first segment folder needs to be expanded first (if folder has never been expanded, it will not show the compact folder structure)
132
- await this.waitForVisibleFileNodes();
133
- const firstSegment = path.split('/')[0];
134
- const selector = this.treeNodeSelector(firstSegment);
135
- const folderElement = await (viewElement === null || viewElement === void 0 ? void 0 : viewElement.$(selector));
136
- if (folderElement && await folderElement.isVisible()) {
137
- const folderNode = await (viewElement === null || viewElement === void 0 ? void 0 : viewElement.waitForSelector(selector, { state: 'visible' }));
138
- if (!folderNode) {
139
- throw new Error(`Tree node '${selector}' not found in explorer`);
140
- }
141
- const folderFileStatNode = new TheiaExplorerFileStatNode(folderNode, this);
142
- if (await folderFileStatNode.isCollapsed()) {
143
- await folderFileStatNode.expand();
144
- }
145
- }
146
- // now get tree node via the full path
147
- const fullPathSelector = this.treeNodeSelector(path);
148
- const treeNode = await (viewElement === null || viewElement === void 0 ? void 0 : viewElement.waitForSelector(fullPathSelector, { state: 'visible' }));
149
- if (!treeNode) {
150
- throw new Error(`Tree node '${fullPathSelector}' not found in explorer`);
151
- }
152
- return new TheiaExplorerFileStatNode(treeNode, this);
153
- }
154
- async selectTreeNode(filePath) {
155
- await this.activate();
156
- const treeNode = await this.page.waitForSelector(this.treeNodeSelector(filePath));
157
- if (await this.isTreeNodeSelected(filePath)) {
158
- await treeNode.focus();
159
- }
160
- else {
161
- await treeNode.click({ modifiers: ['Control'] });
162
- // make sure the click has been acted-upon before returning
163
- while (!await this.isTreeNodeSelected(filePath)) {
164
- console.debug('Waiting for clicked tree node to be selected: ' + filePath);
165
- }
166
- }
167
- await this.page.waitForSelector(this.treeNodeSelector(filePath) + '.theia-mod-selected');
168
- }
169
- async isTreeNodeSelected(filePath) {
170
- const treeNode = await this.page.waitForSelector(this.treeNodeSelector(filePath));
171
- return (0, util_1.elementContainsClass)(treeNode, 'theia-mod-selected');
172
- }
173
- treeNodeSelector(filePath) {
174
- return `.theia-FileStatNode:has(#${(0, util_1.normalizeId)(this.treeNodeId(filePath))})`;
175
- }
176
- treeNodeId(filePath) {
177
- const workspacePath = this.app.workspace.path;
178
- const nodeId = `${workspacePath}:${workspacePath}${util_1.OSUtil.fileSeparator}${filePath}`;
179
- if (util_1.OSUtil.isWindows) {
180
- return (0, util_1.urlEncodePath)(nodeId);
181
- }
182
- return nodeId;
183
- }
184
- async clickContextMenuItem(file, path, nodeSegmentLabel) {
185
- await this.activate();
186
- const fileStatNode = await this.fileStatNode(file, !!nodeSegmentLabel);
187
- if (!fileStatNode) {
188
- throw Error('File stat node could not be retrieved by path fragments');
189
- }
190
- const menuItem = await fileStatNode.getMenuItemByNamePath(path, nodeSegmentLabel);
191
- await menuItem.click();
192
- }
193
- async existsNode(path, isDirectory, compact = false) {
194
- const fileStatNode = await this.fileStatNode(path, compact);
195
- if (!fileStatNode) {
196
- return false;
197
- }
198
- if (isDirectory) {
199
- if (!await fileStatNode.isFolder()) {
200
- throw Error(`FileStatNode for '${path}' is not a directory!`);
201
- }
202
- }
203
- else {
204
- if (!await fileStatNode.isFile()) {
205
- throw Error(`FileStatNode for '${path}' is not a file!`);
206
- }
207
- }
208
- return true;
209
- }
210
- async existsFileNode(path) {
211
- return this.existsNode(path, false);
212
- }
213
- async existsDirectoryNode(path, compact = false) {
214
- return this.existsNode(path, true, compact);
215
- }
216
- async waitForTreeNodeVisible(path) {
217
- // wait for tree node to be visible, e.g. after triggering create
218
- const viewElement = await this.viewElement();
219
- await (viewElement === null || viewElement === void 0 ? void 0 : viewElement.waitForSelector(this.treeNodeSelector(path), { state: 'visible' }));
220
- }
221
- async getNumberOfVisibleNodes() {
222
- await this.activate();
223
- await this.refresh();
224
- const fileStatElements = await this.visibleFileStatNodes(exports.DOT_FILES_FILTER);
225
- return fileStatElements.length;
226
- }
227
- async deleteNode(path, confirm = true, nodeSegmentLabel) {
228
- await this.activate();
229
- await this.clickContextMenuItem(path, ['Delete'], nodeSegmentLabel);
230
- const confirmDialog = new theia_dialog_1.TheiaDialog(this.app);
231
- await confirmDialog.waitForVisible();
232
- confirm ? await confirmDialog.clickMainButton() : await confirmDialog.clickSecondaryButton();
233
- await confirmDialog.waitForClosed();
234
- }
235
- async renameNode(path, newName, confirm = true, nodeSegmentLabel) {
236
- await this.activate();
237
- await this.clickContextMenuItem(path, ['Rename'], nodeSegmentLabel);
238
- const renameDialog = new theia_rename_dialog_1.TheiaRenameDialog(this.app);
239
- await renameDialog.waitForVisible();
240
- await renameDialog.enterNewName(newName);
241
- await renameDialog.waitUntilMainButtonIsEnabled();
242
- confirm ? await renameDialog.confirm() : await renameDialog.close();
243
- await renameDialog.waitForClosed();
244
- await this.refresh();
245
- }
246
- async waitForVisible() {
247
- await super.waitForVisible();
248
- await this.page.waitForSelector(this.tabSelector, { state: 'visible' });
249
- }
250
- /**
251
- * Waits until some non-dot file nodes are visible
252
- */
253
- async waitForVisibleFileNodes() {
254
- while ((await this.visibleFileStatNodes(exports.DOT_FILES_FILTER)).length === 0) {
255
- console.debug('Awaiting for tree nodes to appear');
256
- }
257
- }
258
- async waitForFileNodesToIncrease(numberBefore) {
259
- const fileStatNodesSelector = `${this.viewSelector} .theia-FileStatNode`;
260
- await this.page.waitForFunction((predicate) => {
261
- const elements = document.querySelectorAll(predicate.selector);
262
- return !!elements && elements.length > predicate.numberBefore;
263
- }, { selector: fileStatNodesSelector, numberBefore });
264
- }
265
- async waitForFileNodesToDecrease(numberBefore) {
266
- const fileStatNodesSelector = `${this.viewSelector} .theia-FileStatNode`;
267
- await this.page.waitForFunction((predicate) => {
268
- const elements = document.querySelectorAll(predicate.selector);
269
- return !!elements && elements.length < predicate.numberBefore;
270
- }, { selector: fileStatNodesSelector, numberBefore });
271
- }
272
- }
273
- exports.TheiaExplorerView = TheiaExplorerView;
1
+ "use strict";
2
+ // *****************************************************************************
3
+ // Copyright (C) 2021 logi.cals GmbH, EclipseSource and others.
4
+ //
5
+ // This program and the accompanying materials are made available under the
6
+ // terms of the Eclipse Public License v. 2.0 which is available at
7
+ // http://www.eclipse.org/legal/epl-2.0.
8
+ //
9
+ // This Source Code may also be made available under the following Secondary
10
+ // Licenses when the conditions for such availability set forth in the Eclipse
11
+ // Public License v. 2.0 are satisfied: GNU General Public License, version 2
12
+ // with the GNU Classpath Exception which is available at
13
+ // https://www.gnu.org/software/classpath/license.html.
14
+ //
15
+ // SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
16
+ // *****************************************************************************
17
+ Object.defineProperty(exports, "__esModule", { value: true });
18
+ exports.TheiaExplorerView = exports.DOT_FILES_FILTER = exports.TheiaExplorerFileStatNode = void 0;
19
+ const theia_dialog_1 = require("./theia-dialog");
20
+ const theia_rename_dialog_1 = require("./theia-rename-dialog");
21
+ const theia_tree_node_1 = require("./theia-tree-node");
22
+ const theia_view_1 = require("./theia-view");
23
+ const util_1 = require("./util");
24
+ const TheiaExplorerViewData = {
25
+ tabSelector: '#shell-tab-explorer-view-container',
26
+ viewSelector: '#explorer-view-container--files',
27
+ viewName: 'Explorer'
28
+ };
29
+ class TheiaExplorerFileStatNode extends theia_tree_node_1.TheiaTreeNode {
30
+ constructor(elementHandle, explorerView) {
31
+ super(elementHandle, explorerView.app);
32
+ this.elementHandle = elementHandle;
33
+ this.explorerView = explorerView;
34
+ }
35
+ async absolutePath() {
36
+ return this.elementHandle.getAttribute('title');
37
+ }
38
+ async isFile() {
39
+ return !await this.isFolder();
40
+ }
41
+ async isFolder() {
42
+ return (0, util_1.elementContainsClass)(this.elementHandle, 'theia-DirNode');
43
+ }
44
+ async getMenuItemByNamePath(names, nodeSegmentLabel) {
45
+ const contextMenu = nodeSegmentLabel ? await this.openContextMenuOnSegment(nodeSegmentLabel) : await this.openContextMenu();
46
+ const menuItem = await contextMenu.menuItemByNamePath(...names);
47
+ if (!menuItem) {
48
+ throw Error('MenuItem could not be retrieved by path');
49
+ }
50
+ return menuItem;
51
+ }
52
+ }
53
+ exports.TheiaExplorerFileStatNode = TheiaExplorerFileStatNode;
54
+ const DOT_FILES_FILTER = async (node) => {
55
+ const label = await node.label();
56
+ return label ? !label.startsWith('.') : true;
57
+ };
58
+ exports.DOT_FILES_FILTER = DOT_FILES_FILTER;
59
+ class TheiaExplorerView extends theia_view_1.TheiaView {
60
+ constructor(app) {
61
+ super(TheiaExplorerViewData, app);
62
+ }
63
+ async activate() {
64
+ await super.activate();
65
+ const viewElement = await this.viewElement();
66
+ await (viewElement === null || viewElement === void 0 ? void 0 : viewElement.waitForSelector('.theia-TreeContainer'));
67
+ }
68
+ async refresh() {
69
+ await this.clickButton('navigator.refresh');
70
+ }
71
+ async collapseAll() {
72
+ await this.clickButton('navigator.collapse.all');
73
+ }
74
+ async clickButton(id) {
75
+ await this.activate();
76
+ const viewElement = await this.viewElement();
77
+ await (viewElement === null || viewElement === void 0 ? void 0 : viewElement.hover());
78
+ const button = await (viewElement === null || viewElement === void 0 ? void 0 : viewElement.waitForSelector(`#${(0, util_1.normalizeId)(id)}`));
79
+ await (button === null || button === void 0 ? void 0 : button.click());
80
+ }
81
+ async visibleFileStatNodes(filterPredicate = (_ => Promise.resolve(true))) {
82
+ const viewElement = await this.viewElement();
83
+ const handles = await (viewElement === null || viewElement === void 0 ? void 0 : viewElement.$$('.theia-FileStatNode'));
84
+ if (handles) {
85
+ const nodes = handles.map(handle => new TheiaExplorerFileStatNode(handle, this));
86
+ const filteredNodes = [];
87
+ for (const node of nodes) {
88
+ if ((await filterPredicate(node)) === true) {
89
+ filteredNodes.push(node);
90
+ }
91
+ }
92
+ return filteredNodes;
93
+ }
94
+ return [];
95
+ }
96
+ async getFileStatNodeByLabel(label, compact = false) {
97
+ const file = await this.fileStatNode(label, compact);
98
+ if (!file) {
99
+ throw Error('File stat node could not be retrieved by path fragments');
100
+ }
101
+ return file;
102
+ }
103
+ async fileStatNode(filePath, compact = false) {
104
+ return compact ? this.compactFileStatNode(filePath) : this.fileStatNodeBySegments(...filePath.split('/'));
105
+ }
106
+ async fileStatNodeBySegments(...pathFragments) {
107
+ await super.activate();
108
+ const viewElement = await this.viewElement();
109
+ let currentTreeNode = undefined;
110
+ let fragmentsSoFar = '';
111
+ for (let index = 0; index < pathFragments.length; index++) {
112
+ const fragment = pathFragments[index];
113
+ fragmentsSoFar += index !== 0 ? '/' : '';
114
+ fragmentsSoFar += fragment;
115
+ const selector = this.treeNodeSelector(fragmentsSoFar);
116
+ const nextTreeNode = await (viewElement === null || viewElement === void 0 ? void 0 : viewElement.waitForSelector(selector, { state: 'visible' }));
117
+ if (!nextTreeNode) {
118
+ throw new Error(`Tree node '${selector}' not found in explorer`);
119
+ }
120
+ currentTreeNode = new TheiaExplorerFileStatNode(nextTreeNode, this);
121
+ if (index < pathFragments.length - 1 && await currentTreeNode.isCollapsed()) {
122
+ await currentTreeNode.expand();
123
+ }
124
+ }
125
+ return currentTreeNode;
126
+ }
127
+ async compactFileStatNode(path) {
128
+ // default setting `explorer.compactFolders=true` renders folders in a compact form - single child folders will be compressed in a combined tree element
129
+ await super.activate();
130
+ const viewElement = await this.viewElement();
131
+ // check if first segment folder needs to be expanded first (if folder has never been expanded, it will not show the compact folder structure)
132
+ await this.waitForVisibleFileNodes();
133
+ const firstSegment = path.split('/')[0];
134
+ const selector = this.treeNodeSelector(firstSegment);
135
+ const folderElement = await (viewElement === null || viewElement === void 0 ? void 0 : viewElement.$(selector));
136
+ if (folderElement && await folderElement.isVisible()) {
137
+ const folderNode = await (viewElement === null || viewElement === void 0 ? void 0 : viewElement.waitForSelector(selector, { state: 'visible' }));
138
+ if (!folderNode) {
139
+ throw new Error(`Tree node '${selector}' not found in explorer`);
140
+ }
141
+ const folderFileStatNode = new TheiaExplorerFileStatNode(folderNode, this);
142
+ if (await folderFileStatNode.isCollapsed()) {
143
+ await folderFileStatNode.expand();
144
+ }
145
+ }
146
+ // now get tree node via the full path
147
+ const fullPathSelector = this.treeNodeSelector(path);
148
+ const treeNode = await (viewElement === null || viewElement === void 0 ? void 0 : viewElement.waitForSelector(fullPathSelector, { state: 'visible' }));
149
+ if (!treeNode) {
150
+ throw new Error(`Tree node '${fullPathSelector}' not found in explorer`);
151
+ }
152
+ return new TheiaExplorerFileStatNode(treeNode, this);
153
+ }
154
+ async selectTreeNode(filePath) {
155
+ await this.activate();
156
+ const treeNode = await this.page.waitForSelector(this.treeNodeSelector(filePath));
157
+ if (await this.isTreeNodeSelected(filePath)) {
158
+ await treeNode.focus();
159
+ }
160
+ else {
161
+ await treeNode.click({ modifiers: ['Control'] });
162
+ // make sure the click has been acted-upon before returning
163
+ while (!await this.isTreeNodeSelected(filePath)) {
164
+ console.debug('Waiting for clicked tree node to be selected: ' + filePath);
165
+ }
166
+ }
167
+ await this.page.waitForSelector(this.treeNodeSelector(filePath) + '.theia-mod-selected');
168
+ }
169
+ async isTreeNodeSelected(filePath) {
170
+ const treeNode = await this.page.waitForSelector(this.treeNodeSelector(filePath));
171
+ return (0, util_1.elementContainsClass)(treeNode, 'theia-mod-selected');
172
+ }
173
+ treeNodeSelector(filePath) {
174
+ return `.theia-FileStatNode:has(#${(0, util_1.normalizeId)(this.treeNodeId(filePath))})`;
175
+ }
176
+ treeNodeId(filePath) {
177
+ const workspacePath = this.app.workspace.path;
178
+ const nodeId = `${workspacePath}:${workspacePath}${util_1.OSUtil.fileSeparator}${filePath}`;
179
+ if (util_1.OSUtil.isWindows) {
180
+ return (0, util_1.urlEncodePath)(nodeId);
181
+ }
182
+ return nodeId;
183
+ }
184
+ async clickContextMenuItem(file, path, nodeSegmentLabel) {
185
+ await this.activate();
186
+ const fileStatNode = await this.fileStatNode(file, !!nodeSegmentLabel);
187
+ if (!fileStatNode) {
188
+ throw Error('File stat node could not be retrieved by path fragments');
189
+ }
190
+ const menuItem = await fileStatNode.getMenuItemByNamePath(path, nodeSegmentLabel);
191
+ await menuItem.click();
192
+ }
193
+ async existsNode(path, isDirectory, compact = false) {
194
+ const fileStatNode = await this.fileStatNode(path, compact);
195
+ if (!fileStatNode) {
196
+ return false;
197
+ }
198
+ if (isDirectory) {
199
+ if (!await fileStatNode.isFolder()) {
200
+ throw Error(`FileStatNode for '${path}' is not a directory!`);
201
+ }
202
+ }
203
+ else {
204
+ if (!await fileStatNode.isFile()) {
205
+ throw Error(`FileStatNode for '${path}' is not a file!`);
206
+ }
207
+ }
208
+ return true;
209
+ }
210
+ async existsFileNode(path) {
211
+ return this.existsNode(path, false);
212
+ }
213
+ async existsDirectoryNode(path, compact = false) {
214
+ return this.existsNode(path, true, compact);
215
+ }
216
+ async waitForTreeNodeVisible(path) {
217
+ // wait for tree node to be visible, e.g. after triggering create
218
+ const viewElement = await this.viewElement();
219
+ await (viewElement === null || viewElement === void 0 ? void 0 : viewElement.waitForSelector(this.treeNodeSelector(path), { state: 'visible' }));
220
+ }
221
+ async getNumberOfVisibleNodes() {
222
+ await this.activate();
223
+ await this.refresh();
224
+ const fileStatElements = await this.visibleFileStatNodes(exports.DOT_FILES_FILTER);
225
+ return fileStatElements.length;
226
+ }
227
+ async deleteNode(path, confirm = true, nodeSegmentLabel) {
228
+ await this.activate();
229
+ await this.clickContextMenuItem(path, ['Delete'], nodeSegmentLabel);
230
+ const confirmDialog = new theia_dialog_1.TheiaDialog(this.app);
231
+ await confirmDialog.waitForVisible();
232
+ confirm ? await confirmDialog.clickMainButton() : await confirmDialog.clickSecondaryButton();
233
+ await confirmDialog.waitForClosed();
234
+ }
235
+ async renameNode(path, newName, confirm = true, nodeSegmentLabel) {
236
+ await this.activate();
237
+ await this.clickContextMenuItem(path, ['Rename'], nodeSegmentLabel);
238
+ const renameDialog = new theia_rename_dialog_1.TheiaRenameDialog(this.app);
239
+ await renameDialog.waitForVisible();
240
+ await renameDialog.enterNewName(newName);
241
+ await renameDialog.waitUntilMainButtonIsEnabled();
242
+ confirm ? await renameDialog.confirm() : await renameDialog.close();
243
+ await renameDialog.waitForClosed();
244
+ await this.refresh();
245
+ }
246
+ async waitForVisible() {
247
+ await super.waitForVisible();
248
+ await this.page.waitForSelector(this.tabSelector, { state: 'visible' });
249
+ }
250
+ /**
251
+ * Waits until some non-dot file nodes are visible
252
+ */
253
+ async waitForVisibleFileNodes() {
254
+ while ((await this.visibleFileStatNodes(exports.DOT_FILES_FILTER)).length === 0) {
255
+ console.debug('Awaiting for tree nodes to appear');
256
+ }
257
+ }
258
+ async waitForFileNodesToIncrease(numberBefore) {
259
+ const fileStatNodesSelector = `${this.viewSelector} .theia-FileStatNode`;
260
+ await this.page.waitForFunction((predicate) => {
261
+ const elements = document.querySelectorAll(predicate.selector);
262
+ return !!elements && elements.length > predicate.numberBefore;
263
+ }, { selector: fileStatNodesSelector, numberBefore });
264
+ }
265
+ async waitForFileNodesToDecrease(numberBefore) {
266
+ const fileStatNodesSelector = `${this.viewSelector} .theia-FileStatNode`;
267
+ await this.page.waitForFunction((predicate) => {
268
+ const elements = document.querySelectorAll(predicate.selector);
269
+ return !!elements && elements.length < predicate.numberBefore;
270
+ }, { selector: fileStatNodesSelector, numberBefore });
271
+ }
272
+ }
273
+ exports.TheiaExplorerView = TheiaExplorerView;
274
274
  //# sourceMappingURL=theia-explorer-view.js.map
@@ -1,13 +1,13 @@
1
- import { ElementHandle } from '@playwright/test';
2
- import { TheiaMenu } from './theia-menu';
3
- import { TheiaPageObject } from './theia-page-object';
4
- export declare class TheiaMainMenu extends TheiaMenu {
5
- selector: string;
6
- }
7
- export declare class TheiaMenuBar extends TheiaPageObject {
8
- openMenu(menuName: string): Promise<TheiaMainMenu>;
9
- visibleMenuBarItems(): Promise<string[]>;
10
- protected menuBarItem(label?: string): Promise<ElementHandle<SVGElement | HTMLElement> | null>;
11
- protected menuBarItemSelector(label?: string): string;
12
- }
1
+ import { ElementHandle } from '@playwright/test';
2
+ import { TheiaMenu } from './theia-menu';
3
+ import { TheiaPageObject } from './theia-page-object';
4
+ export declare class TheiaMainMenu extends TheiaMenu {
5
+ selector: string;
6
+ }
7
+ export declare class TheiaMenuBar extends TheiaPageObject {
8
+ openMenu(menuName: string): Promise<TheiaMainMenu>;
9
+ visibleMenuBarItems(): Promise<string[]>;
10
+ protected menuBarItem(label?: string): Promise<ElementHandle<SVGElement | HTMLElement> | null>;
11
+ protected menuBarItemSelector(label?: string): string;
12
+ }
13
13
  //# sourceMappingURL=theia-main-menu.d.ts.map