@theia/core 1.48.1 → 1.48.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 (1497) hide show
  1. package/README.md +6 -6
  2. package/i18n/nls.cs.json +541 -541
  3. package/i18n/nls.de.json +541 -541
  4. package/i18n/nls.es.json +541 -541
  5. package/i18n/nls.fr.json +541 -541
  6. package/i18n/nls.hu.json +541 -541
  7. package/i18n/nls.it.json +541 -541
  8. package/i18n/nls.ja.json +541 -541
  9. package/i18n/nls.json +541 -541
  10. package/i18n/nls.pl.json +541 -541
  11. package/i18n/nls.pt-br.json +541 -541
  12. package/i18n/nls.pt-pt.json +541 -541
  13. package/i18n/nls.ru.json +541 -541
  14. package/i18n/nls.zh-cn.json +541 -541
  15. package/lib/browser/about-dialog.d.ts +33 -33
  16. package/lib/browser/about-dialog.js +125 -125
  17. package/lib/browser/authentication-service.d.ts +144 -144
  18. package/lib/browser/authentication-service.js +324 -324
  19. package/lib/browser/breadcrumbs/breadcrumb-popup-container.d.ts +34 -34
  20. package/lib/browser/breadcrumbs/breadcrumb-popup-container.js +106 -106
  21. package/lib/browser/breadcrumbs/breadcrumb-renderer.d.ts +12 -12
  22. package/lib/browser/breadcrumbs/breadcrumb-renderer.js +37 -37
  23. package/lib/browser/breadcrumbs/breadcrumbs-constants.d.ts +49 -49
  24. package/lib/browser/breadcrumbs/breadcrumbs-constants.js +27 -27
  25. package/lib/browser/breadcrumbs/breadcrumbs-renderer.d.ts +43 -43
  26. package/lib/browser/breadcrumbs/breadcrumbs-renderer.js +179 -179
  27. package/lib/browser/breadcrumbs/breadcrumbs-service.d.ts +32 -32
  28. package/lib/browser/breadcrumbs/breadcrumbs-service.js +111 -111
  29. package/lib/browser/breadcrumbs/index.d.ts +5 -5
  30. package/lib/browser/breadcrumbs/index.js +23 -23
  31. package/lib/browser/browser-clipboard-service.d.ts +23 -23
  32. package/lib/browser/browser-clipboard-service.js +107 -107
  33. package/lib/browser/browser.d.ts +37 -37
  34. package/lib/browser/browser.js +196 -196
  35. package/lib/browser/clipboard-service.d.ts +6 -6
  36. package/lib/browser/clipboard-service.js +19 -19
  37. package/lib/browser/color-application-contribution.d.ts +26 -26
  38. package/lib/browser/color-application-contribution.js +110 -110
  39. package/lib/browser/color-registry.d.ts +14 -14
  40. package/lib/browser/color-registry.js +58 -58
  41. package/lib/browser/command-open-handler.d.ts +9 -9
  42. package/lib/browser/command-open-handler.js +58 -58
  43. package/lib/browser/common-frontend-contribution.d.ts +212 -212
  44. package/lib/browser/common-frontend-contribution.js +2338 -2338
  45. package/lib/browser/common-styling-participants.d.ts +27 -27
  46. package/lib/browser/common-styling-participants.js +365 -365
  47. package/lib/browser/connection-status-service.d.ts +77 -77
  48. package/lib/browser/connection-status-service.js +198 -198
  49. package/lib/browser/connection-status-service.spec.d.ts +1 -1
  50. package/lib/browser/connection-status-service.spec.js +161 -161
  51. package/lib/browser/context-key-service.d.ts +82 -82
  52. package/lib/browser/context-key-service.js +83 -83
  53. package/lib/browser/context-menu-renderer.d.ts +54 -54
  54. package/lib/browser/context-menu-renderer.js +87 -87
  55. package/lib/browser/core-preferences.d.ts +37 -37
  56. package/lib/browser/core-preferences.js +300 -300
  57. package/lib/browser/credentials-service.d.ts +35 -35
  58. package/lib/browser/credentials-service.js +80 -80
  59. package/lib/browser/decoration-style.d.ts +6 -6
  60. package/lib/browser/decoration-style.js +64 -64
  61. package/lib/browser/decorations-service.d.ts +30 -30
  62. package/lib/browser/decorations-service.js +157 -157
  63. package/lib/browser/dialogs/react-dialog.d.ts +17 -17
  64. package/lib/browser/dialogs/react-dialog.js +49 -49
  65. package/lib/browser/dialogs.d.ts +135 -135
  66. package/lib/browser/dialogs.js +446 -446
  67. package/lib/browser/diff-uris.d.ts +17 -17
  68. package/lib/browser/diff-uris.js +115 -115
  69. package/lib/browser/encoding-registry.d.ts +18 -18
  70. package/lib/browser/encoding-registry.js +88 -88
  71. package/lib/browser/endpoint.d.ts +42 -42
  72. package/lib/browser/endpoint.js +120 -120
  73. package/lib/browser/endpoint.spec.d.ts +1 -1
  74. package/lib/browser/endpoint.spec.js +122 -122
  75. package/lib/browser/external-uri-service.d.ts +24 -24
  76. package/lib/browser/external-uri-service.js +72 -72
  77. package/lib/browser/frontend-application-bindings.d.ts +5 -5
  78. package/lib/browser/frontend-application-bindings.js +56 -56
  79. package/lib/browser/frontend-application-config-provider.d.ts +8 -8
  80. package/lib/browser/frontend-application-config-provider.js +46 -46
  81. package/lib/browser/frontend-application-config-provider.spec.d.ts +1 -1
  82. package/lib/browser/frontend-application-config-provider.spec.js +40 -40
  83. package/lib/browser/frontend-application-contribution.d.ts +75 -75
  84. package/lib/browser/frontend-application-contribution.js +45 -45
  85. package/lib/browser/frontend-application-module.d.ts +8 -8
  86. package/lib/browser/frontend-application-module.js +355 -355
  87. package/lib/browser/frontend-application-state.d.ts +19 -19
  88. package/lib/browser/frontend-application-state.js +71 -71
  89. package/lib/browser/frontend-application.d.ts +86 -86
  90. package/lib/browser/frontend-application.js +308 -308
  91. package/lib/browser/hover-service.d.ts +50 -50
  92. package/lib/browser/hover-service.js +199 -199
  93. package/lib/browser/http-open-handler.d.ts +14 -14
  94. package/lib/browser/http-open-handler.js +46 -46
  95. package/lib/browser/i18n/i18n-frontend-module.d.ts +3 -3
  96. package/lib/browser/i18n/i18n-frontend-module.js +25 -25
  97. package/lib/browser/i18n/language-quick-pick-service.d.ts +15 -15
  98. package/lib/browser/i18n/language-quick-pick-service.js +136 -136
  99. package/lib/browser/icon-registry.d.ts +56 -56
  100. package/lib/browser/icon-registry.js +24 -24
  101. package/lib/browser/icon-theme-contribution.d.ts +23 -23
  102. package/lib/browser/icon-theme-contribution.js +64 -64
  103. package/lib/browser/icon-theme-service.d.ts +63 -63
  104. package/lib/browser/icon-theme-service.js +198 -198
  105. package/lib/browser/index.d.ts +31 -31
  106. package/lib/browser/index.js +49 -49
  107. package/lib/browser/json-schema-store.d.ts +37 -37
  108. package/lib/browser/json-schema-store.js +111 -111
  109. package/lib/browser/keybinding.d.ts +288 -288
  110. package/lib/browser/keybinding.js +677 -677
  111. package/lib/browser/keybinding.js.map +1 -1
  112. package/lib/browser/keybinding.spec.d.ts +1 -1
  113. package/lib/browser/keybinding.spec.js +451 -451
  114. package/lib/browser/keyboard/browser-keyboard-frontend-contribution.d.ts +13 -13
  115. package/lib/browser/keyboard/browser-keyboard-frontend-contribution.js +110 -110
  116. package/lib/browser/keyboard/browser-keyboard-layout-provider.d.ts +91 -91
  117. package/lib/browser/keyboard/browser-keyboard-layout-provider.js +418 -418
  118. package/lib/browser/keyboard/browser-keyboard-layout-provider.spec.d.ts +1 -1
  119. package/lib/browser/keyboard/browser-keyboard-layout-provider.spec.js +152 -152
  120. package/lib/browser/keyboard/browser-keyboard-module.d.ts +3 -3
  121. package/lib/browser/keyboard/browser-keyboard-module.js +30 -30
  122. package/lib/browser/keyboard/index.d.ts +4 -4
  123. package/lib/browser/keyboard/index.js +22 -22
  124. package/lib/browser/keyboard/keyboard-layout-service.d.ts +49 -49
  125. package/lib/browser/keyboard/keyboard-layout-service.js +427 -427
  126. package/lib/browser/keyboard/keyboard-layout-service.spec.d.ts +1 -1
  127. package/lib/browser/keyboard/keyboard-layout-service.spec.js +110 -110
  128. package/lib/browser/keyboard/keys.d.ts +4 -4
  129. package/lib/browser/keyboard/keys.js +22 -22
  130. package/lib/browser/keyboard/keys.spec.d.ts +1 -1
  131. package/lib/browser/keyboard/keys.spec.js +205 -205
  132. package/lib/browser/keys.d.ts +2 -2
  133. package/lib/browser/keys.js +28 -28
  134. package/lib/browser/label-parser.d.ts +34 -34
  135. package/lib/browser/label-parser.js +107 -107
  136. package/lib/browser/label-parser.spec.d.ts +1 -1
  137. package/lib/browser/label-parser.spec.js +146 -146
  138. package/lib/browser/label-provider.d.ts +140 -140
  139. package/lib/browser/label-provider.js +310 -310
  140. package/lib/browser/label-provider.spec.d.ts +1 -1
  141. package/lib/browser/label-provider.spec.js +53 -53
  142. package/lib/browser/language-icon-provider.d.ts +14 -14
  143. package/lib/browser/language-icon-provider.js +65 -65
  144. package/lib/browser/language-service.d.ts +38 -38
  145. package/lib/browser/language-service.js +66 -66
  146. package/lib/browser/logger-frontend-module.d.ts +2 -2
  147. package/lib/browser/logger-frontend-module.js +63 -63
  148. package/lib/browser/markdown-rendering/markdown-renderer.d.ts +39 -39
  149. package/lib/browser/markdown-rendering/markdown-renderer.js +72 -72
  150. package/lib/browser/menu/browser-context-menu-renderer.d.ts +12 -12
  151. package/lib/browser/menu/browser-context-menu-renderer.js +51 -51
  152. package/lib/browser/menu/browser-menu-module.d.ts +3 -3
  153. package/lib/browser/menu/browser-menu-module.js +28 -28
  154. package/lib/browser/menu/browser-menu-plugin.d.ts +103 -103
  155. package/lib/browser/menu/browser-menu-plugin.js +439 -439
  156. package/lib/browser/menu/context-menu-context.d.ts +7 -7
  157. package/lib/browser/menu/context-menu-context.js +42 -42
  158. package/lib/browser/messaging/connection-source.d.ts +8 -8
  159. package/lib/browser/messaging/connection-source.js +19 -19
  160. package/lib/browser/messaging/frontend-id-provider.d.ts +12 -12
  161. package/lib/browser/messaging/frontend-id-provider.js +34 -34
  162. package/lib/browser/messaging/index.d.ts +1 -1
  163. package/lib/browser/messaging/index.js +19 -19
  164. package/lib/browser/messaging/messaging-frontend-module.d.ts +2 -2
  165. package/lib/browser/messaging/messaging-frontend-module.js +41 -41
  166. package/lib/browser/messaging/service-connection-provider.d.ts +47 -47
  167. package/lib/browser/messaging/service-connection-provider.js +106 -106
  168. package/lib/browser/messaging/ws-connection-provider.d.ts +12 -12
  169. package/lib/browser/messaging/ws-connection-provider.js +49 -49
  170. package/lib/browser/messaging/ws-connection-source.d.ts +42 -42
  171. package/lib/browser/messaging/ws-connection-source.js +217 -217
  172. package/lib/browser/mime-service.d.ts +7 -7
  173. package/lib/browser/mime-service.js +30 -30
  174. package/lib/browser/navigatable-types.d.ts +33 -33
  175. package/lib/browser/navigatable-types.js +63 -63
  176. package/lib/browser/navigatable.d.ts +8 -8
  177. package/lib/browser/navigatable.js +38 -38
  178. package/lib/browser/opener-service.d.ts +79 -79
  179. package/lib/browser/opener-service.js +79 -79
  180. package/lib/browser/opener-service.spec.d.ts +1 -1
  181. package/lib/browser/opener-service.spec.js +46 -46
  182. package/lib/browser/performance/frontend-stopwatch.d.ts +20 -20
  183. package/lib/browser/performance/frontend-stopwatch.js +66 -66
  184. package/lib/browser/performance/index.d.ts +2 -2
  185. package/lib/browser/performance/index.js +20 -20
  186. package/lib/browser/performance/measurement-frontend-bindings.d.ts +19 -19
  187. package/lib/browser/performance/measurement-frontend-bindings.js +32 -32
  188. package/lib/browser/preferences/index.d.ts +7 -7
  189. package/lib/browser/preferences/index.js +25 -25
  190. package/lib/browser/preferences/injectable-preference-proxy.d.ts +51 -51
  191. package/lib/browser/preferences/injectable-preference-proxy.js +276 -276
  192. package/lib/browser/preferences/preference-configurations.d.ts +22 -22
  193. package/lib/browser/preferences/preference-configurations.js +72 -72
  194. package/lib/browser/preferences/preference-contribution.d.ts +111 -111
  195. package/lib/browser/preferences/preference-contribution.js +386 -386
  196. package/lib/browser/preferences/preference-language-override-service.d.ts +39 -39
  197. package/lib/browser/preferences/preference-language-override-service.js +107 -107
  198. package/lib/browser/preferences/preference-provider.d.ts +132 -132
  199. package/lib/browser/preferences/preference-provider.js +219 -219
  200. package/lib/browser/preferences/preference-provider.spec.d.ts +1 -1
  201. package/lib/browser/preferences/preference-provider.spec.js +33 -33
  202. package/lib/browser/preferences/preference-proxy.d.ts +150 -150
  203. package/lib/browser/preferences/preference-proxy.js +229 -229
  204. package/lib/browser/preferences/preference-proxy.spec.d.ts +1 -1
  205. package/lib/browser/preferences/preference-proxy.spec.js +337 -337
  206. package/lib/browser/preferences/preference-schema-provider.spec.d.ts +1 -1
  207. package/lib/browser/preferences/preference-schema-provider.spec.js +115 -115
  208. package/lib/browser/preferences/preference-scope.d.ts +2 -2
  209. package/lib/browser/preferences/preference-scope.js +20 -20
  210. package/lib/browser/preferences/preference-service.d.ts +278 -278
  211. package/lib/browser/preferences/preference-service.js +350 -350
  212. package/lib/browser/preferences/preference-service.spec.d.ts +1 -1
  213. package/lib/browser/preferences/preference-service.spec.js +505 -505
  214. package/lib/browser/preferences/preference-validation-service.d.ts +57 -57
  215. package/lib/browser/preferences/preference-validation-service.js +346 -346
  216. package/lib/browser/preferences/preference-validation-service.spec.d.ts +1 -1
  217. package/lib/browser/preferences/preference-validation-service.spec.js +330 -330
  218. package/lib/browser/preferences/test/index.d.ts +3 -3
  219. package/lib/browser/preferences/test/index.js +21 -21
  220. package/lib/browser/preferences/test/mock-preference-provider.d.ts +16 -16
  221. package/lib/browser/preferences/test/mock-preference-provider.js +46 -46
  222. package/lib/browser/preferences/test/mock-preference-proxy.d.ts +3 -3
  223. package/lib/browser/preferences/test/mock-preference-proxy.js +49 -49
  224. package/lib/browser/preferences/test/mock-preference-service.d.ts +28 -28
  225. package/lib/browser/preferences/test/mock-preference-service.js +59 -59
  226. package/lib/browser/preload/i18n-preload-contribution.d.ts +6 -6
  227. package/lib/browser/preload/i18n-preload-contribution.js +54 -54
  228. package/lib/browser/preload/os-preload-contribution.d.ts +6 -6
  229. package/lib/browser/preload/os-preload-contribution.js +40 -40
  230. package/lib/browser/preload/preload-module.d.ts +3 -3
  231. package/lib/browser/preload/preload-module.js +38 -38
  232. package/lib/browser/preload/preloader.d.ts +11 -11
  233. package/lib/browser/preload/preloader.js +36 -36
  234. package/lib/browser/preload/theme-preload-contribution.d.ts +4 -4
  235. package/lib/browser/preload/theme-preload-contribution.js +34 -34
  236. package/lib/browser/progress-bar-factory.d.ts +11 -11
  237. package/lib/browser/progress-bar-factory.js +20 -20
  238. package/lib/browser/progress-bar.d.ts +15 -15
  239. package/lib/browser/progress-bar.js +81 -81
  240. package/lib/browser/progress-client.d.ts +12 -12
  241. package/lib/browser/progress-client.js +53 -53
  242. package/lib/browser/progress-location-service.d.ts +21 -21
  243. package/lib/browser/progress-location-service.js +91 -91
  244. package/lib/browser/progress-location-service.spec.d.ts +1 -1
  245. package/lib/browser/progress-location-service.spec.js +44 -44
  246. package/lib/browser/progress-status-bar-item.d.ts +16 -16
  247. package/lib/browser/progress-status-bar-item.js +81 -81
  248. package/lib/browser/quick-input/index.d.ts +8 -8
  249. package/lib/browser/quick-input/index.js +26 -26
  250. package/lib/browser/quick-input/quick-access.d.ts +53 -53
  251. package/lib/browser/quick-input/quick-access.js +20 -20
  252. package/lib/browser/quick-input/quick-command-frontend-contribution.d.ts +11 -11
  253. package/lib/browser/quick-input/quick-command-frontend-contribution.js +96 -96
  254. package/lib/browser/quick-input/quick-command-service.d.ts +59 -59
  255. package/lib/browser/quick-input/quick-command-service.js +225 -225
  256. package/lib/browser/quick-input/quick-help-service.d.ts +11 -11
  257. package/lib/browser/quick-input/quick-help-service.js +83 -83
  258. package/lib/browser/quick-input/quick-input-frontend-contribution.d.ts +7 -7
  259. package/lib/browser/quick-input/quick-input-frontend-contribution.js +38 -38
  260. package/lib/browser/quick-input/quick-input-service.d.ts +1 -1
  261. package/lib/browser/quick-input/quick-input-service.js +19 -19
  262. package/lib/browser/quick-input/quick-input-service.spec.d.ts +1 -1
  263. package/lib/browser/quick-input/quick-input-service.spec.js +159 -159
  264. package/lib/browser/quick-input/quick-pick-service-impl.d.ts +30 -30
  265. package/lib/browser/quick-input/quick-pick-service-impl.js +69 -69
  266. package/lib/browser/quick-input/quick-view-service.d.ts +23 -23
  267. package/lib/browser/quick-input/quick-view-service.js +78 -78
  268. package/lib/browser/request/browser-request-module.d.ts +18 -18
  269. package/lib/browser/request/browser-request-module.js +23 -23
  270. package/lib/browser/request/browser-request-service.d.ts +41 -41
  271. package/lib/browser/request/browser-request-service.js +172 -172
  272. package/lib/browser/resource-context-key.d.ts +19 -19
  273. package/lib/browser/resource-context-key.js +75 -75
  274. package/lib/browser/save-resource-service.d.ts +35 -35
  275. package/lib/browser/save-resource-service.js +64 -64
  276. package/lib/browser/saveable.d.ts +124 -124
  277. package/lib/browser/saveable.js +274 -274
  278. package/lib/browser/secondary-window-handler.d.ts +65 -65
  279. package/lib/browser/secondary-window-handler.js +193 -193
  280. package/lib/browser/shell/additional-views-menu-widget.d.ts +19 -19
  281. package/lib/browser/shell/additional-views-menu-widget.js +73 -73
  282. package/lib/browser/shell/application-shell-mouse-tracker.d.ts +30 -30
  283. package/lib/browser/shell/application-shell-mouse-tracker.js +99 -99
  284. package/lib/browser/shell/application-shell.d.ts +574 -574
  285. package/lib/browser/shell/application-shell.js +1992 -1992
  286. package/lib/browser/shell/current-widget-command-adapter.d.ts +23 -23
  287. package/lib/browser/shell/current-widget-command-adapter.js +41 -41
  288. package/lib/browser/shell/index.d.ts +7 -7
  289. package/lib/browser/shell/index.js +25 -25
  290. package/lib/browser/shell/shell-layout-restorer.d.ts +123 -123
  291. package/lib/browser/shell/shell-layout-restorer.js +336 -336
  292. package/lib/browser/shell/side-panel-handler.d.ts +286 -286
  293. package/lib/browser/shell/side-panel-handler.js +659 -659
  294. package/lib/browser/shell/side-panel-toolbar.d.ts +25 -25
  295. package/lib/browser/shell/side-panel-toolbar.js +96 -96
  296. package/lib/browser/shell/sidebar-bottom-menu-widget.d.ts +9 -9
  297. package/lib/browser/shell/sidebar-bottom-menu-widget.js +41 -41
  298. package/lib/browser/shell/sidebar-menu-widget.d.ts +38 -38
  299. package/lib/browser/shell/sidebar-menu-widget.js +113 -113
  300. package/lib/browser/shell/sidebar-top-menu-widget.d.ts +6 -6
  301. package/lib/browser/shell/sidebar-top-menu-widget.js +30 -30
  302. package/lib/browser/shell/split-panels.d.ts +42 -42
  303. package/lib/browser/shell/split-panels.js +172 -172
  304. package/lib/browser/shell/tab-bar-decorator.d.ts +40 -40
  305. package/lib/browser/shell/tab-bar-decorator.js +87 -87
  306. package/lib/browser/shell/tab-bar-toolbar/index.d.ts +3 -3
  307. package/lib/browser/shell/tab-bar-toolbar/index.js +21 -21
  308. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-menu-adapters.d.ts +15 -15
  309. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-menu-adapters.js +35 -35
  310. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-registry.d.ts +95 -95
  311. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-registry.js +245 -245
  312. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-types.d.ts +136 -136
  313. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-types.js +88 -88
  314. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar.d.ts +89 -89
  315. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar.js +414 -414
  316. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar.spec.d.ts +1 -1
  317. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar.spec.js +51 -51
  318. package/lib/browser/shell/tab-bars.d.ts +341 -341
  319. package/lib/browser/shell/tab-bars.js +1312 -1312
  320. package/lib/browser/shell/tab-bars.spec.d.ts +1 -1
  321. package/lib/browser/shell/tab-bars.spec.js +56 -56
  322. package/lib/browser/shell/theia-dock-panel.d.ts +59 -59
  323. package/lib/browser/shell/theia-dock-panel.js +226 -226
  324. package/lib/browser/shell/view-contribution.d.ts +43 -43
  325. package/lib/browser/shell/view-contribution.js +159 -159
  326. package/lib/browser/source-tree/index.d.ts +3 -3
  327. package/lib/browser/source-tree/index.js +21 -21
  328. package/lib/browser/source-tree/source-tree-widget.d.ts +21 -21
  329. package/lib/browser/source-tree/source-tree-widget.js +110 -110
  330. package/lib/browser/source-tree/source-tree.d.ts +36 -36
  331. package/lib/browser/source-tree/source-tree.js +129 -129
  332. package/lib/browser/source-tree/tree-source.d.ts +35 -35
  333. package/lib/browser/source-tree/tree-source.js +53 -53
  334. package/lib/browser/status-bar/index.d.ts +5 -5
  335. package/lib/browser/status-bar/index.js +31 -31
  336. package/lib/browser/status-bar/status-bar-types.d.ts +72 -72
  337. package/lib/browser/status-bar/status-bar-types.js +25 -25
  338. package/lib/browser/status-bar/status-bar-view-model.d.ts +35 -35
  339. package/lib/browser/status-bar/status-bar-view-model.js +207 -207
  340. package/lib/browser/status-bar/status-bar.d.ts +34 -34
  341. package/lib/browser/status-bar/status-bar.js +181 -181
  342. package/lib/browser/storage-service.d.ts +35 -35
  343. package/lib/browser/storage-service.js +122 -122
  344. package/lib/browser/storage-service.spec.d.ts +1 -1
  345. package/lib/browser/storage-service.spec.js +68 -68
  346. package/lib/browser/styling-service.d.ts +29 -29
  347. package/lib/browser/styling-service.js +86 -86
  348. package/lib/browser/supported-encodings.d.ts +9 -9
  349. package/lib/browser/supported-encodings.js +263 -263
  350. package/lib/browser/test/jsdom.d.ts +8 -8
  351. package/lib/browser/test/jsdom.js +69 -69
  352. package/lib/browser/test/mock-connection-status-service.d.ts +5 -5
  353. package/lib/browser/test/mock-connection-status-service.js +32 -32
  354. package/lib/browser/test/mock-env-variables-server.d.ts +12 -12
  355. package/lib/browser/test/mock-env-variables-server.js +42 -42
  356. package/lib/browser/test/mock-opener-service.d.ts +8 -8
  357. package/lib/browser/test/mock-opener-service.js +35 -35
  358. package/lib/browser/test/mock-storage-service.d.ts +11 -11
  359. package/lib/browser/test/mock-storage-service.js +49 -49
  360. package/lib/browser/theming.d.ts +50 -50
  361. package/lib/browser/theming.js +203 -203
  362. package/lib/browser/tooltip-service.d.ts +34 -34
  363. package/lib/browser/tooltip-service.js +76 -76
  364. package/lib/browser/tree/fuzzy-search.d.ts +60 -60
  365. package/lib/browser/tree/fuzzy-search.js +75 -75
  366. package/lib/browser/tree/fuzzy-search.spec.d.ts +1 -1
  367. package/lib/browser/tree/fuzzy-search.spec.js +89 -89
  368. package/lib/browser/tree/index.d.ts +13 -13
  369. package/lib/browser/tree/index.js +31 -31
  370. package/lib/browser/tree/search-box-debounce.d.ts +33 -33
  371. package/lib/browser/tree/search-box-debounce.js +75 -75
  372. package/lib/browser/tree/search-box.d.ts +105 -105
  373. package/lib/browser/tree/search-box.js +265 -265
  374. package/lib/browser/tree/test/mock-selectable-tree-model.d.ts +14 -14
  375. package/lib/browser/tree/test/mock-selectable-tree-model.js +102 -102
  376. package/lib/browser/tree/test/mock-tree-model.d.ts +12 -12
  377. package/lib/browser/tree/test/mock-tree-model.js +129 -129
  378. package/lib/browser/tree/test/tree-test-container.d.ts +2 -2
  379. package/lib/browser/tree/test/tree-test-container.js +52 -52
  380. package/lib/browser/tree/tree-compression/compressed-tree-expansion-service.d.ts +8 -8
  381. package/lib/browser/tree/tree-compression/compressed-tree-expansion-service.js +59 -59
  382. package/lib/browser/tree/tree-compression/compressed-tree-model.d.ts +23 -23
  383. package/lib/browser/tree/tree-compression/compressed-tree-model.js +89 -89
  384. package/lib/browser/tree/tree-compression/compressed-tree-widget.d.ts +38 -38
  385. package/lib/browser/tree/tree-compression/compressed-tree-widget.js +187 -187
  386. package/lib/browser/tree/tree-compression/index.d.ts +4 -4
  387. package/lib/browser/tree/tree-compression/index.js +22 -22
  388. package/lib/browser/tree/tree-compression/tree-compression-service.d.ts +58 -58
  389. package/lib/browser/tree/tree-compression/tree-compression-service.js +98 -98
  390. package/lib/browser/tree/tree-consistency.spec.d.ts +1 -1
  391. package/lib/browser/tree/tree-consistency.spec.js +99 -99
  392. package/lib/browser/tree/tree-container.d.ts +51 -51
  393. package/lib/browser/tree/tree-container.js +115 -115
  394. package/lib/browser/tree/tree-container.spec.d.ts +1 -1
  395. package/lib/browser/tree/tree-container.spec.js +45 -45
  396. package/lib/browser/tree/tree-decorator.d.ts +143 -143
  397. package/lib/browser/tree/tree-decorator.js +115 -115
  398. package/lib/browser/tree/tree-decorator.spec.d.ts +1 -1
  399. package/lib/browser/tree/tree-decorator.spec.js +145 -145
  400. package/lib/browser/tree/tree-expansion.d.ts +65 -65
  401. package/lib/browser/tree/tree-expansion.js +124 -124
  402. package/lib/browser/tree/tree-expansion.spec.d.ts +1 -1
  403. package/lib/browser/tree/tree-expansion.spec.js +155 -155
  404. package/lib/browser/tree/tree-focus-service.d.ts +19 -19
  405. package/lib/browser/tree/tree-focus-service.js +53 -53
  406. package/lib/browser/tree/tree-iterator.d.ts +94 -94
  407. package/lib/browser/tree/tree-iterator.js +223 -223
  408. package/lib/browser/tree/tree-iterator.spec.d.ts +1 -1
  409. package/lib/browser/tree/tree-iterator.spec.js +150 -150
  410. package/lib/browser/tree/tree-label-provider.d.ts +8 -8
  411. package/lib/browser/tree/tree-label-provider.js +39 -39
  412. package/lib/browser/tree/tree-model.d.ts +202 -202
  413. package/lib/browser/tree/tree-model.js +395 -395
  414. package/lib/browser/tree/tree-navigation.d.ts +10 -10
  415. package/lib/browser/tree/tree-navigation.js +57 -57
  416. package/lib/browser/tree/tree-preference.d.ts +10 -10
  417. package/lib/browser/tree/tree-preference.js +46 -46
  418. package/lib/browser/tree/tree-search.d.ts +36 -36
  419. package/lib/browser/tree/tree-search.js +130 -130
  420. package/lib/browser/tree/tree-selectable.spec.d.ts +1 -1
  421. package/lib/browser/tree/tree-selectable.spec.js +146 -146
  422. package/lib/browser/tree/tree-selection-impl.d.ts +44 -44
  423. package/lib/browser/tree/tree-selection-impl.js +158 -158
  424. package/lib/browser/tree/tree-selection-state.d.ts +48 -48
  425. package/lib/browser/tree/tree-selection-state.js +227 -227
  426. package/lib/browser/tree/tree-selection-state.spec.d.ts +1 -1
  427. package/lib/browser/tree/tree-selection-state.spec.js +420 -420
  428. package/lib/browser/tree/tree-selection.d.ts +86 -86
  429. package/lib/browser/tree/tree-selection.js +86 -86
  430. package/lib/browser/tree/tree-view-welcome-widget.d.ts +45 -45
  431. package/lib/browser/tree/tree-view-welcome-widget.js +192 -192
  432. package/lib/browser/tree/tree-widget-selection.d.ts +11 -11
  433. package/lib/browser/tree/tree-widget-selection.js +46 -46
  434. package/lib/browser/tree/tree-widget.d.ts +559 -559
  435. package/lib/browser/tree/tree-widget.js +1352 -1352
  436. package/lib/browser/tree/tree.d.ts +180 -180
  437. package/lib/browser/tree/tree.js +288 -288
  438. package/lib/browser/tree/tree.spec.d.ts +1 -1
  439. package/lib/browser/tree/tree.spec.js +226 -226
  440. package/lib/browser/user-working-directory-provider.d.ts +23 -23
  441. package/lib/browser/user-working-directory-provider.js +76 -76
  442. package/lib/browser/view-container.d.ts +312 -312
  443. package/lib/browser/view-container.js +1458 -1458
  444. package/lib/browser/widget-decoration.d.ts +299 -299
  445. package/lib/browser/widget-decoration.js +140 -140
  446. package/lib/browser/widget-manager.d.ts +172 -172
  447. package/lib/browser/widget-manager.js +222 -222
  448. package/lib/browser/widget-manager.spec.d.ts +1 -1
  449. package/lib/browser/widget-manager.spec.js +91 -91
  450. package/lib/browser/widget-open-handler.d.ts +88 -88
  451. package/lib/browser/widget-open-handler.js +137 -137
  452. package/lib/browser/widgets/alert-message.d.ts +23 -23
  453. package/lib/browser/widgets/alert-message.js +38 -38
  454. package/lib/browser/widgets/enhanced-preview-widget.d.ts +6 -6
  455. package/lib/browser/widgets/enhanced-preview-widget.js +26 -26
  456. package/lib/browser/widgets/extractable-widget.d.ts +13 -13
  457. package/lib/browser/widgets/extractable-widget.js +26 -26
  458. package/lib/browser/widgets/index.d.ts +4 -4
  459. package/lib/browser/widgets/index.js +22 -22
  460. package/lib/browser/widgets/previewable-widget.d.ts +8 -8
  461. package/lib/browser/widgets/previewable-widget.js +30 -30
  462. package/lib/browser/widgets/react-renderer.d.ts +14 -14
  463. package/lib/browser/widgets/react-renderer.js +48 -48
  464. package/lib/browser/widgets/react-widget.d.ts +16 -16
  465. package/lib/browser/widgets/react-widget.js +46 -46
  466. package/lib/browser/widgets/select-component.d.ts +53 -53
  467. package/lib/browser/widgets/select-component.js +307 -307
  468. package/lib/browser/widgets/widget.d.ts +108 -108
  469. package/lib/browser/widgets/widget.js +367 -367
  470. package/lib/browser/window/browser-window-module.d.ts +3 -3
  471. package/lib/browser/window/browser-window-module.js +32 -32
  472. package/lib/browser/window/default-secondary-window-service.d.ts +25 -25
  473. package/lib/browser/window/default-secondary-window-service.js +187 -187
  474. package/lib/browser/window/default-window-service.d.ts +50 -50
  475. package/lib/browser/window/default-window-service.js +167 -167
  476. package/lib/browser/window/default-window-service.spec.d.ts +1 -1
  477. package/lib/browser/window/default-window-service.spec.js +80 -80
  478. package/lib/browser/window/secondary-window-service.d.ts +20 -20
  479. package/lib/browser/window/secondary-window-service.js +19 -19
  480. package/lib/browser/window/test/mock-window-service.d.ts +11 -11
  481. package/lib/browser/window/test/mock-window-service.js +34 -34
  482. package/lib/browser/window/window-service.d.ts +53 -53
  483. package/lib/browser/window/window-service.js +22 -22
  484. package/lib/browser/window/window-title-service.d.ts +31 -31
  485. package/lib/browser/window/window-title-service.js +112 -112
  486. package/lib/browser/window/window-title-updater.d.ts +19 -19
  487. package/lib/browser/window/window-title-updater.js +99 -99
  488. package/lib/browser/window-contribution.d.ts +14 -14
  489. package/lib/browser/window-contribution.js +62 -62
  490. package/lib/browser-only/frontend-only-application-module.d.ts +4 -4
  491. package/lib/browser-only/frontend-only-application-module.js +114 -114
  492. package/lib/browser-only/i18n/i18n-frontend-only-module.d.ts +3 -3
  493. package/lib/browser-only/i18n/i18n-frontend-only-module.js +34 -34
  494. package/lib/browser-only/logger-frontend-only-module.d.ts +2 -2
  495. package/lib/browser-only/logger-frontend-only-module.js +60 -60
  496. package/lib/browser-only/messaging/frontend-only-service-connection-provider.d.ts +13 -13
  497. package/lib/browser-only/messaging/frontend-only-service-connection-provider.js +51 -51
  498. package/lib/browser-only/messaging/messaging-frontend-only-module.d.ts +2 -2
  499. package/lib/browser-only/messaging/messaging-frontend-only-module.js +47 -47
  500. package/lib/browser-only/preload/frontend-only-preload-module.d.ts +3 -3
  501. package/lib/browser-only/preload/frontend-only-preload-module.js +51 -51
  502. package/lib/common/accessibility.d.ts +16 -16
  503. package/lib/common/accessibility.js +17 -17
  504. package/lib/common/application-error.d.ts +20 -20
  505. package/lib/common/application-error.js +62 -62
  506. package/lib/common/application-error.spec.d.ts +1 -1
  507. package/lib/common/application-error.spec.js +26 -26
  508. package/lib/common/application-protocol.d.ts +20 -20
  509. package/lib/common/application-protocol.js +20 -20
  510. package/lib/common/array-utils.d.ts +55 -55
  511. package/lib/common/array-utils.js +118 -118
  512. package/lib/common/buffer.d.ts +43 -43
  513. package/lib/common/buffer.js +205 -205
  514. package/lib/common/cancellation.d.ts +24 -24
  515. package/lib/common/cancellation.js +142 -142
  516. package/lib/common/char-code.d.ts +401 -401
  517. package/lib/common/char-code.js +22 -22
  518. package/lib/common/collections.d.ts +29 -29
  519. package/lib/common/collections.js +121 -121
  520. package/lib/common/color.d.ts +71 -71
  521. package/lib/common/color.js +43 -43
  522. package/lib/common/command.d.ts +232 -232
  523. package/lib/common/command.js +353 -353
  524. package/lib/common/command.spec.d.ts +1 -1
  525. package/lib/common/command.spec.js +173 -173
  526. package/lib/common/contribution-filter/contribution-filter-registry.d.ts +17 -17
  527. package/lib/common/contribution-filter/contribution-filter-registry.js +77 -77
  528. package/lib/common/contribution-filter/contribution-filter.d.ts +41 -41
  529. package/lib/common/contribution-filter/contribution-filter.js +20 -20
  530. package/lib/common/contribution-filter/filter.d.ts +6 -6
  531. package/lib/common/contribution-filter/filter.js +19 -19
  532. package/lib/common/contribution-filter/index.d.ts +3 -3
  533. package/lib/common/contribution-filter/index.js +21 -21
  534. package/lib/common/contribution-provider.d.ts +20 -20
  535. package/lib/common/contribution-provider.js +82 -82
  536. package/lib/common/disposable.d.ts +80 -80
  537. package/lib/common/disposable.js +173 -173
  538. package/lib/common/disposable.spec.d.ts +1 -1
  539. package/lib/common/disposable.spec.js +80 -80
  540. package/lib/common/encoding-service.d.ts +38 -38
  541. package/lib/common/encoding-service.js +302 -302
  542. package/lib/common/encodings.d.ts +7 -7
  543. package/lib/common/encodings.js +25 -25
  544. package/lib/common/env-variables/env-variables-protocol.d.ts +20 -20
  545. package/lib/common/env-variables/env-variables-protocol.js +20 -20
  546. package/lib/common/env-variables/index.d.ts +1 -1
  547. package/lib/common/env-variables/index.js +19 -19
  548. package/lib/common/event.d.ts +115 -115
  549. package/lib/common/event.js +385 -385
  550. package/lib/common/event.spec.d.ts +1 -1
  551. package/lib/common/event.spec.js +29 -29
  552. package/lib/common/file-uri.d.ts +14 -14
  553. package/lib/common/file-uri.js +62 -62
  554. package/lib/common/frontend-application-state.d.ts +15 -15
  555. package/lib/common/frontend-application-state.js +33 -33
  556. package/lib/common/glob.d.ts +50 -50
  557. package/lib/common/glob.js +591 -591
  558. package/lib/common/hash.d.ts +7 -7
  559. package/lib/common/hash.js +84 -84
  560. package/lib/common/i18n/localization-server.d.ts +6 -6
  561. package/lib/common/i18n/localization-server.js +20 -20
  562. package/lib/common/i18n/localization.d.ts +34 -34
  563. package/lib/common/i18n/localization.js +59 -59
  564. package/lib/common/index.d.ts +35 -35
  565. package/lib/common/index.js +56 -56
  566. package/lib/common/json-schema.d.ts +72 -72
  567. package/lib/common/json-schema.js +17 -17
  568. package/lib/common/key-store.d.ts +12 -12
  569. package/lib/common/key-store.js +20 -20
  570. package/lib/common/keybinding.d.ts +75 -75
  571. package/lib/common/keybinding.js +113 -113
  572. package/lib/common/keyboard/keyboard-layout-provider.d.ts +26 -26
  573. package/lib/common/keyboard/keyboard-layout-provider.js +22 -22
  574. package/lib/common/keys.d.ts +253 -253
  575. package/lib/common/keys.js +634 -634
  576. package/lib/common/label-protocol.d.ts +13 -13
  577. package/lib/common/label-protocol.js +17 -17
  578. package/lib/common/logger-protocol.d.ts +41 -41
  579. package/lib/common/logger-protocol.js +100 -100
  580. package/lib/common/logger-watcher.d.ts +9 -9
  581. package/lib/common/logger-watcher.js +49 -49
  582. package/lib/common/logger.d.ts +207 -207
  583. package/lib/common/logger.js +204 -204
  584. package/lib/common/logger.spec.d.ts +1 -1
  585. package/lib/common/logger.spec.js +41 -41
  586. package/lib/common/lsp-types.d.ts +9 -9
  587. package/lib/common/lsp-types.js +30 -30
  588. package/lib/common/markdown-rendering/icon-utilities.d.ts +7 -7
  589. package/lib/common/markdown-rendering/icon-utilities.js +32 -32
  590. package/lib/common/markdown-rendering/index.d.ts +2 -2
  591. package/lib/common/markdown-rendering/index.js +20 -20
  592. package/lib/common/markdown-rendering/markdown-string.d.ts +46 -46
  593. package/lib/common/markdown-rendering/markdown-string.js +126 -126
  594. package/lib/common/menu/action-menu-node.d.ts +19 -19
  595. package/lib/common/menu/action-menu-node.js +56 -56
  596. package/lib/common/menu/composite-menu-node.d.ts +46 -46
  597. package/lib/common/menu/composite-menu-node.js +93 -93
  598. package/lib/common/menu/composite-menu-node.spec.d.ts +1 -1
  599. package/lib/common/menu/composite-menu-node.spec.js +67 -67
  600. package/lib/common/menu/index.d.ts +5 -5
  601. package/lib/common/menu/index.js +23 -23
  602. package/lib/common/menu/menu-adapter.d.ts +35 -35
  603. package/lib/common/menu/menu-adapter.js +92 -92
  604. package/lib/common/menu/menu-model-registry.d.ts +145 -145
  605. package/lib/common/menu/menu-model-registry.js +275 -275
  606. package/lib/common/menu/menu-types.d.ts +148 -148
  607. package/lib/common/menu/menu-types.js +93 -93
  608. package/lib/common/menu/menu.spec.d.ts +1 -1
  609. package/lib/common/menu/menu.spec.js +96 -96
  610. package/lib/common/message-rpc/channel.d.ts +123 -123
  611. package/lib/common/message-rpc/channel.js +230 -230
  612. package/lib/common/message-rpc/channel.spec.d.ts +8 -8
  613. package/lib/common/message-rpc/channel.spec.js +79 -79
  614. package/lib/common/message-rpc/index.d.ts +3 -3
  615. package/lib/common/message-rpc/index.js +24 -24
  616. package/lib/common/message-rpc/message-buffer.d.ts +51 -51
  617. package/lib/common/message-rpc/message-buffer.js +59 -59
  618. package/lib/common/message-rpc/msg-pack-extension-manager.d.ts +22 -22
  619. package/lib/common/message-rpc/msg-pack-extension-manager.js +57 -57
  620. package/lib/common/message-rpc/rpc-message-encoder.d.ts +103 -103
  621. package/lib/common/message-rpc/rpc-message-encoder.js +113 -113
  622. package/lib/common/message-rpc/rpc-message-encoder.spec.d.ts +1 -1
  623. package/lib/common/message-rpc/rpc-message-encoder.spec.js +58 -58
  624. package/lib/common/message-rpc/rpc-protocol.d.ts +67 -67
  625. package/lib/common/message-rpc/rpc-protocol.js +209 -209
  626. package/lib/common/message-rpc/uint8-array-message-buffer.d.ts +52 -52
  627. package/lib/common/message-rpc/uint8-array-message-buffer.js +174 -174
  628. package/lib/common/message-rpc/uint8-array-message-buffer.spec.d.ts +1 -1
  629. package/lib/common/message-rpc/uint8-array-message-buffer.spec.js +38 -38
  630. package/lib/common/message-service-protocol.d.ts +110 -110
  631. package/lib/common/message-service-protocol.js +73 -73
  632. package/lib/common/message-service.d.ts +134 -134
  633. package/lib/common/message-service.js +151 -151
  634. package/lib/common/messaging/connection-error-handler.d.ts +28 -28
  635. package/lib/common/messaging/connection-error-handler.js +47 -47
  636. package/lib/common/messaging/connection-management.d.ts +24 -24
  637. package/lib/common/messaging/connection-management.js +37 -37
  638. package/lib/common/messaging/handler.d.ts +7 -7
  639. package/lib/common/messaging/handler.js +20 -20
  640. package/lib/common/messaging/index.d.ts +3 -3
  641. package/lib/common/messaging/index.js +21 -21
  642. package/lib/common/messaging/proxy-factory.d.ts +180 -180
  643. package/lib/common/messaging/proxy-factory.js +280 -280
  644. package/lib/common/messaging/proxy-factory.spec.d.ts +1 -1
  645. package/lib/common/messaging/proxy-factory.spec.js +99 -99
  646. package/lib/common/messaging/socket-write-buffer.d.ts +10 -10
  647. package/lib/common/messaging/socket-write-buffer.js +49 -49
  648. package/lib/common/messaging/web-socket-channel.d.ts +19 -19
  649. package/lib/common/messaging/web-socket-channel.js +64 -64
  650. package/lib/common/nls.d.ts +15 -15
  651. package/lib/common/nls.js +135 -135
  652. package/lib/common/numbers.d.ts +5 -5
  653. package/lib/common/numbers.js +23 -23
  654. package/lib/common/objects.d.ts +8 -8
  655. package/lib/common/objects.js +118 -118
  656. package/lib/common/objects.spec.d.ts +1 -1
  657. package/lib/common/objects.spec.js +101 -101
  658. package/lib/common/os.d.ts +31 -31
  659. package/lib/common/os.js +73 -73
  660. package/lib/common/path.d.ts +93 -93
  661. package/lib/common/path.js +307 -307
  662. package/lib/common/path.spec.d.ts +1 -1
  663. package/lib/common/path.spec.js +350 -350
  664. package/lib/common/paths.d.ts +24 -24
  665. package/lib/common/paths.js +226 -226
  666. package/lib/common/performance/index.d.ts +3 -3
  667. package/lib/common/performance/index.js +21 -21
  668. package/lib/common/performance/measurement-protocol.d.ts +64 -64
  669. package/lib/common/performance/measurement-protocol.js +70 -70
  670. package/lib/common/performance/measurement.d.ts +111 -111
  671. package/lib/common/performance/measurement.js +17 -17
  672. package/lib/common/performance/stopwatch.d.ts +75 -75
  673. package/lib/common/performance/stopwatch.js +148 -148
  674. package/lib/common/preferences/preference-schema.d.ts +52 -52
  675. package/lib/common/preferences/preference-schema.js +58 -58
  676. package/lib/common/preferences/preference-scope.d.ts +19 -19
  677. package/lib/common/preferences/preference-scope.js +71 -71
  678. package/lib/common/preferences/preference-scope.spec.d.ts +1 -1
  679. package/lib/common/preferences/preference-scope.spec.js +44 -44
  680. package/lib/common/prioritizeable.d.ts +16 -16
  681. package/lib/common/prioritizeable.js +55 -55
  682. package/lib/common/progress-service-protocol.d.ts +13 -13
  683. package/lib/common/progress-service-protocol.js +19 -19
  684. package/lib/common/progress-service.d.ts +13 -13
  685. package/lib/common/progress-service.js +88 -88
  686. package/lib/common/promise-util.d.ts +51 -51
  687. package/lib/common/promise-util.js +132 -132
  688. package/lib/common/promise-util.spec.d.ts +1 -1
  689. package/lib/common/promise-util.spec.js +80 -80
  690. package/lib/common/quick-pick-service.d.ts +270 -270
  691. package/lib/common/quick-pick-service.js +133 -133
  692. package/lib/common/reference.d.ts +95 -95
  693. package/lib/common/reference.js +207 -207
  694. package/lib/common/reference.spec.d.ts +1 -1
  695. package/lib/common/reference.spec.js +125 -125
  696. package/lib/common/resource.d.ts +207 -207
  697. package/lib/common/resource.js +322 -322
  698. package/lib/common/selection-command-handler.d.ts +24 -24
  699. package/lib/common/selection-command-handler.js +83 -83
  700. package/lib/common/selection-service.d.ts +18 -18
  701. package/lib/common/selection-service.js +43 -43
  702. package/lib/common/selection-service.spec.d.ts +1 -1
  703. package/lib/common/selection-service.spec.js +38 -38
  704. package/lib/common/selection.d.ts +9 -9
  705. package/lib/common/selection.js +47 -47
  706. package/lib/common/severity.d.ts +15 -15
  707. package/lib/common/severity.js +113 -113
  708. package/lib/common/stream.d.ts +176 -176
  709. package/lib/common/stream.js +474 -474
  710. package/lib/common/strings.d.ts +24 -24
  711. package/lib/common/strings.js +215 -215
  712. package/lib/common/telemetry.d.ts +19 -19
  713. package/lib/common/telemetry.js +24 -24
  714. package/lib/common/ternary-search-tree.d.ts +50 -50
  715. package/lib/common/ternary-search-tree.js +381 -381
  716. package/lib/common/test/expect.d.ts +2 -2
  717. package/lib/common/test/expect.js +34 -34
  718. package/lib/common/test/mock-logger.d.ts +27 -27
  719. package/lib/common/test/mock-logger.js +99 -99
  720. package/lib/common/test/mock-menu.d.ts +7 -7
  721. package/lib/common/test/mock-menu.js +34 -34
  722. package/lib/common/test/mock-resource-provider.d.ts +6 -6
  723. package/lib/common/test/mock-resource-provider.js +36 -36
  724. package/lib/common/theme.d.ts +40 -40
  725. package/lib/common/theme.js +22 -22
  726. package/lib/common/types.d.ts +58 -58
  727. package/lib/common/types.js +126 -126
  728. package/lib/common/types.spec.d.ts +1 -1
  729. package/lib/common/types.spec.js +73 -73
  730. package/lib/common/uri-command-handler.d.ts +53 -53
  731. package/lib/common/uri-command-handler.js +99 -99
  732. package/lib/common/uri-command-handler.spec.d.ts +1 -1
  733. package/lib/common/uri-command-handler.spec.js +80 -80
  734. package/lib/common/uri.d.ts +87 -87
  735. package/lib/common/uri.js +236 -236
  736. package/lib/common/uri.spec.d.ts +1 -1
  737. package/lib/common/uri.spec.js +230 -230
  738. package/lib/common/uuid.d.ts +8 -8
  739. package/lib/common/uuid.js +44 -44
  740. package/lib/common/version.d.ts +1 -1
  741. package/lib/common/version.js +19 -19
  742. package/lib/common/view-column.d.ts +17 -17
  743. package/lib/common/view-column.js +36 -36
  744. package/lib/common/window.d.ts +16 -16
  745. package/lib/common/window.js +22 -22
  746. package/lib/electron-browser/electron-clipboard-service.d.ts +5 -5
  747. package/lib/electron-browser/electron-clipboard-service.js +33 -33
  748. package/lib/electron-browser/keyboard/electron-keyboard-layout-change-notifier.d.ts +11 -11
  749. package/lib/electron-browser/keyboard/electron-keyboard-layout-change-notifier.js +46 -46
  750. package/lib/electron-browser/keyboard/electron-keyboard-module.d.ts +3 -3
  751. package/lib/electron-browser/keyboard/electron-keyboard-module.js +26 -26
  752. package/lib/electron-browser/menu/electron-context-menu-renderer.d.ts +30 -30
  753. package/lib/electron-browser/menu/electron-context-menu-renderer.js +128 -128
  754. package/lib/electron-browser/menu/electron-main-menu-factory.d.ts +59 -59
  755. package/lib/electron-browser/menu/electron-main-menu-factory.js +274 -274
  756. package/lib/electron-browser/menu/electron-menu-contribution.d.ts +78 -78
  757. package/lib/electron-browser/menu/electron-menu-contribution.js +471 -471
  758. package/lib/electron-browser/menu/electron-menu-module.d.ts +3 -3
  759. package/lib/electron-browser/menu/electron-menu-module.js +39 -39
  760. package/lib/electron-browser/messaging/electron-frontend-id-provider.d.ts +4 -4
  761. package/lib/electron-browser/messaging/electron-frontend-id-provider.js +29 -29
  762. package/lib/electron-browser/messaging/electron-ipc-connection-source.d.ts +23 -23
  763. package/lib/electron-browser/messaging/electron-ipc-connection-source.js +59 -59
  764. package/lib/electron-browser/messaging/electron-local-ws-connection-source.d.ts +6 -6
  765. package/lib/electron-browser/messaging/electron-local-ws-connection-source.js +49 -49
  766. package/lib/electron-browser/messaging/electron-messaging-frontend-module.d.ts +2 -2
  767. package/lib/electron-browser/messaging/electron-messaging-frontend-module.js +75 -75
  768. package/lib/electron-browser/messaging/electron-ws-connection-source.d.ts +11 -11
  769. package/lib/electron-browser/messaging/electron-ws-connection-source.js +42 -42
  770. package/lib/electron-browser/preload.d.ts +1 -1
  771. package/lib/electron-browser/preload.js +201 -201
  772. package/lib/electron-browser/preload.js.map +1 -1
  773. package/lib/electron-browser/request/electron-browser-request-module.d.ts +18 -18
  774. package/lib/electron-browser/request/electron-browser-request-module.js +26 -26
  775. package/lib/electron-browser/token/electron-token-frontend-module.d.ts +3 -3
  776. package/lib/electron-browser/token/electron-token-frontend-module.js +22 -22
  777. package/lib/electron-browser/window/electron-frontend-application-state.d.ts +4 -4
  778. package/lib/electron-browser/window/electron-frontend-application-state.js +31 -31
  779. package/lib/electron-browser/window/electron-secondary-window-service.d.ts +7 -7
  780. package/lib/electron-browser/window/electron-secondary-window-service.js +38 -38
  781. package/lib/electron-browser/window/electron-window-module.d.ts +3 -3
  782. package/lib/electron-browser/window/electron-window-module.js +39 -39
  783. package/lib/electron-browser/window/electron-window-preferences.d.ts +18 -18
  784. package/lib/electron-browser/window/electron-window-preferences.js +72 -72
  785. package/lib/electron-browser/window/electron-window-service.d.ts +31 -31
  786. package/lib/electron-browser/window/electron-window-service.js +116 -116
  787. package/lib/electron-browser/window/electron-window-service.js.map +1 -1
  788. package/lib/electron-common/electron-api.d.ts +104 -104
  789. package/lib/electron-common/electron-api.d.ts.map +1 -1
  790. package/lib/electron-common/electron-api.js +56 -56
  791. package/lib/electron-common/electron-main-window-service.d.ts +7 -7
  792. package/lib/electron-common/electron-main-window-service.js +20 -20
  793. package/lib/electron-common/electron-token.d.ts +10 -10
  794. package/lib/electron-common/electron-token.js +27 -27
  795. package/lib/electron-main/electron-api-main.d.ts +22 -22
  796. package/lib/electron-main/electron-api-main.d.ts.map +1 -1
  797. package/lib/electron-main/electron-api-main.js +296 -294
  798. package/lib/electron-main/electron-api-main.js.map +1 -1
  799. package/lib/electron-main/electron-main-application-module.d.ts +3 -3
  800. package/lib/electron-main/electron-main-application-module.js +56 -56
  801. package/lib/electron-main/electron-main-application.d.ts +159 -159
  802. package/lib/electron-main/electron-main-application.js +603 -603
  803. package/lib/electron-main/electron-main-constants.d.ts +6 -6
  804. package/lib/electron-main/electron-main-constants.js +19 -19
  805. package/lib/electron-main/electron-main-window-service-impl.d.ts +8 -8
  806. package/lib/electron-main/electron-main-window-service-impl.js +46 -46
  807. package/lib/electron-main/electron-security-token-service.d.ts +5 -5
  808. package/lib/electron-main/electron-security-token-service.js +41 -41
  809. package/lib/electron-main/event-utils.d.ts +8 -8
  810. package/lib/electron-main/event-utils.js +33 -33
  811. package/lib/electron-main/messaging/electron-connection-handler.d.ts +5 -5
  812. package/lib/electron-main/messaging/electron-connection-handler.js +23 -23
  813. package/lib/electron-main/messaging/electron-messaging-contribution.d.ts +41 -41
  814. package/lib/electron-main/messaging/electron-messaging-contribution.js +142 -142
  815. package/lib/electron-main/messaging/electron-messaging-service.d.ts +17 -17
  816. package/lib/electron-main/messaging/electron-messaging-service.js +22 -22
  817. package/lib/electron-main/theia-electron-window.d.ts +53 -53
  818. package/lib/electron-main/theia-electron-window.js +186 -186
  819. package/lib/electron-node/hosting/electron-backend-hosting-module.d.ts +3 -3
  820. package/lib/electron-node/hosting/electron-backend-hosting-module.js +24 -24
  821. package/lib/electron-node/hosting/electron-ws-origin-validator.d.ts +8 -8
  822. package/lib/electron-node/hosting/electron-ws-origin-validator.js +40 -40
  823. package/lib/electron-node/keyboard/electron-backend-keyboard-module.d.ts +3 -3
  824. package/lib/electron-node/keyboard/electron-backend-keyboard-module.js +26 -26
  825. package/lib/electron-node/keyboard/electron-keyboard-layout-provider.d.ts +5 -5
  826. package/lib/electron-node/keyboard/electron-keyboard-layout-provider.js +36 -36
  827. package/lib/electron-node/request/electron-backend-request-module.d.ts +18 -18
  828. package/lib/electron-node/request/electron-backend-request-module.js +23 -23
  829. package/lib/electron-node/request/electron-backend-request-service.d.ts +22 -22
  830. package/lib/electron-node/request/electron-backend-request-service.js +81 -81
  831. package/lib/electron-node/token/electron-token-backend-contribution.d.ts +15 -15
  832. package/lib/electron-node/token/electron-token-backend-contribution.js +50 -50
  833. package/lib/electron-node/token/electron-token-backend-module.d.ts +3 -3
  834. package/lib/electron-node/token/electron-token-backend-module.js +28 -28
  835. package/lib/electron-node/token/electron-token-validator.d.ts +29 -29
  836. package/lib/electron-node/token/electron-token-validator.js +95 -95
  837. package/lib/node/application-server.d.ts +10 -10
  838. package/lib/node/application-server.js +55 -55
  839. package/lib/node/backend-application-config-provider.d.ts +7 -7
  840. package/lib/node/backend-application-config-provider.js +45 -45
  841. package/lib/node/backend-application-config-provider.spec.d.ts +1 -1
  842. package/lib/node/backend-application-config-provider.spec.js +28 -28
  843. package/lib/node/backend-application-module.d.ts +2 -2
  844. package/lib/node/backend-application-module.js +108 -108
  845. package/lib/node/backend-application.d.ts +108 -108
  846. package/lib/node/backend-application.js +290 -290
  847. package/lib/node/backend-remote-service.d.ts +3 -3
  848. package/lib/node/backend-remote-service.js +29 -29
  849. package/lib/node/cli.d.ts +17 -17
  850. package/lib/node/cli.js +59 -59
  851. package/lib/node/cli.spec.d.ts +1 -1
  852. package/lib/node/cli.spec.js +89 -89
  853. package/lib/node/console-logger-server.d.ts +15 -15
  854. package/lib/node/console-logger-server.js +82 -82
  855. package/lib/node/console-logger-server.spec.d.ts +1 -1
  856. package/lib/node/console-logger-server.spec.js +61 -61
  857. package/lib/node/debug.d.ts +1 -1
  858. package/lib/node/debug.js +28 -28
  859. package/lib/node/dynamic-require.d.ts +7 -7
  860. package/lib/node/dynamic-require.js +56 -56
  861. package/lib/node/env-variables/env-variables-server.d.ts +23 -23
  862. package/lib/node/env-variables/env-variables-server.js +117 -117
  863. package/lib/node/env-variables/index.d.ts +1 -1
  864. package/lib/node/env-variables/index.js +19 -19
  865. package/lib/node/environment-utils.d.ts +21 -21
  866. package/lib/node/environment-utils.js +72 -72
  867. package/lib/node/environment-utils.spec.d.ts +1 -1
  868. package/lib/node/environment-utils.spec.js +77 -77
  869. package/lib/node/file-uri.spec.d.ts +1 -1
  870. package/lib/node/file-uri.spec.js +66 -66
  871. package/lib/node/filesystem-locking.d.ts +23 -23
  872. package/lib/node/filesystem-locking.js +65 -65
  873. package/lib/node/hosting/backend-application-hosts.d.ts +23 -23
  874. package/lib/node/hosting/backend-application-hosts.js +69 -69
  875. package/lib/node/hosting/backend-hosting-module.d.ts +3 -3
  876. package/lib/node/hosting/backend-hosting-module.js +26 -26
  877. package/lib/node/hosting/ws-origin-validator.d.ts +8 -8
  878. package/lib/node/hosting/ws-origin-validator.js +39 -39
  879. package/lib/node/i18n/i18n-backend-module.d.ts +3 -3
  880. package/lib/node/i18n/i18n-backend-module.js +38 -38
  881. package/lib/node/i18n/localization-contribution.d.ts +18 -18
  882. package/lib/node/i18n/localization-contribution.js +108 -108
  883. package/lib/node/i18n/localization-provider.d.ts +23 -23
  884. package/lib/node/i18n/localization-provider.js +109 -109
  885. package/lib/node/i18n/localization-server.d.ts +14 -14
  886. package/lib/node/i18n/localization-server.js +54 -54
  887. package/lib/node/i18n/theia-localization-contribution.d.ts +4 -4
  888. package/lib/node/i18n/theia-localization-contribution.js +40 -40
  889. package/lib/node/index.d.ts +6 -6
  890. package/lib/node/index.js +26 -26
  891. package/lib/node/key-store-server.d.ts +27 -27
  892. package/lib/node/key-store-server.js +145 -145
  893. package/lib/node/logger-backend-module.d.ts +9 -9
  894. package/lib/node/logger-backend-module.js +78 -78
  895. package/lib/node/logger-cli-contribution.d.ts +33 -33
  896. package/lib/node/logger-cli-contribution.js +147 -147
  897. package/lib/node/logger-cli-contribution.spec.d.ts +1 -1
  898. package/lib/node/logger-cli-contribution.spec.js +207 -207
  899. package/lib/node/main.d.ts +7 -7
  900. package/lib/node/main.js +29 -29
  901. package/lib/node/messaging/binary-message-pipe.d.ts +44 -44
  902. package/lib/node/messaging/binary-message-pipe.js +151 -151
  903. package/lib/node/messaging/connection-container-module.d.ts +53 -53
  904. package/lib/node/messaging/connection-container-module.js +81 -81
  905. package/lib/node/messaging/default-messaging-service.d.ts +28 -28
  906. package/lib/node/messaging/default-messaging-service.js +131 -131
  907. package/lib/node/messaging/frontend-connection-service.d.ts +6 -6
  908. package/lib/node/messaging/frontend-connection-service.js +18 -18
  909. package/lib/node/messaging/index.d.ts +3 -3
  910. package/lib/node/messaging/index.js +21 -21
  911. package/lib/node/messaging/ipc-bootstrap.d.ts +1 -1
  912. package/lib/node/messaging/ipc-bootstrap.js +24 -24
  913. package/lib/node/messaging/ipc-channel.d.ts +17 -17
  914. package/lib/node/messaging/ipc-channel.js +69 -69
  915. package/lib/node/messaging/ipc-connection-provider.d.ts +21 -21
  916. package/lib/node/messaging/ipc-connection-provider.js +92 -92
  917. package/lib/node/messaging/ipc-protocol.d.ts +19 -19
  918. package/lib/node/messaging/ipc-protocol.js +68 -68
  919. package/lib/node/messaging/messaging-backend-module.d.ts +2 -2
  920. package/lib/node/messaging/messaging-backend-module.js +52 -52
  921. package/lib/node/messaging/messaging-listeners.d.ts +27 -27
  922. package/lib/node/messaging/messaging-listeners.js +45 -45
  923. package/lib/node/messaging/messaging-service.d.ts +28 -28
  924. package/lib/node/messaging/messaging-service.js +24 -24
  925. package/lib/node/messaging/test/test-web-socket-channel.d.ts +12 -12
  926. package/lib/node/messaging/test/test-web-socket-channel.js +54 -54
  927. package/lib/node/messaging/websocket-endpoint.d.ts +20 -20
  928. package/lib/node/messaging/websocket-endpoint.js +80 -80
  929. package/lib/node/messaging/websocket-frontend-connection-service.d.ts +30 -30
  930. package/lib/node/messaging/websocket-frontend-connection-service.js +169 -169
  931. package/lib/node/os-backend-provider.d.ts +4 -4
  932. package/lib/node/os-backend-provider.js +30 -30
  933. package/lib/node/performance/index.d.ts +2 -2
  934. package/lib/node/performance/index.js +20 -20
  935. package/lib/node/performance/measurement-backend-bindings.d.ts +19 -19
  936. package/lib/node/performance/measurement-backend-bindings.js +30 -30
  937. package/lib/node/performance/node-stopwatch.d.ts +20 -20
  938. package/lib/node/performance/node-stopwatch.js +43 -43
  939. package/lib/node/process-utils.d.ts +13 -13
  940. package/lib/node/process-utils.js +105 -105
  941. package/lib/node/process-utils.spec.d.ts +1 -1
  942. package/lib/node/process-utils.spec.js +44 -44
  943. package/lib/node/request/backend-request-facade.d.ts +22 -22
  944. package/lib/node/request/backend-request-facade.js +41 -41
  945. package/lib/node/request/backend-request-module.d.ts +18 -18
  946. package/lib/node/request/backend-request-module.js +24 -24
  947. package/lib/node/request/proxy-cli-contribution.d.ts +27 -27
  948. package/lib/node/request/proxy-cli-contribution.js +64 -64
  949. package/lib/node/ws-request-validators.d.ts +25 -25
  950. package/lib/node/ws-request-validators.js +52 -52
  951. package/package.json +6 -6
  952. package/shared/ajv/index.d.ts +2 -2
  953. package/shared/markdown-it.d.ts +2 -2
  954. package/shared/markdown-it.js +1 -1
  955. package/shared/reflect-metadata/index.d.ts +1 -1
  956. package/shared/reflect-metadata/index.js +1 -1
  957. package/shared/vscode-languageserver-types/index.d.ts +1 -1
  958. package/src/browser/about-dialog.tsx +137 -137
  959. package/src/browser/authentication-service.ts +456 -456
  960. package/src/browser/breadcrumbs/breadcrumb-popup-container.ts +101 -101
  961. package/src/browser/breadcrumbs/breadcrumb-renderer.tsx +41 -41
  962. package/src/browser/breadcrumbs/breadcrumbs-constants.ts +79 -79
  963. package/src/browser/breadcrumbs/breadcrumbs-renderer.tsx +185 -185
  964. package/src/browser/breadcrumbs/breadcrumbs-service.ts +108 -108
  965. package/src/browser/breadcrumbs/index.ts +21 -21
  966. package/src/browser/browser-clipboard-service.ts +122 -122
  967. package/src/browser/browser.ts +225 -225
  968. package/src/browser/clipboard-service.ts +23 -23
  969. package/src/browser/color-application-contribution.ts +110 -110
  970. package/src/browser/color-registry.ts +60 -60
  971. package/src/browser/command-open-handler.ts +54 -54
  972. package/src/browser/common-frontend-contribution.ts +2569 -2569
  973. package/src/browser/common-styling-participants.ts +361 -361
  974. package/src/browser/connection-status-service.spec.ts +200 -200
  975. package/src/browser/connection-status-service.ts +216 -216
  976. package/src/browser/context-key-service.ts +142 -142
  977. package/src/browser/context-menu-renderer.ts +124 -124
  978. package/src/browser/core-preferences.ts +334 -334
  979. package/src/browser/credentials-service.ts +106 -106
  980. package/src/browser/decoration-style.ts +65 -65
  981. package/src/browser/decorations-service.ts +209 -209
  982. package/src/browser/dialogs/react-dialog.tsx +56 -56
  983. package/src/browser/dialogs.ts +534 -534
  984. package/src/browser/diff-uris.ts +117 -117
  985. package/src/browser/encoding-registry.ts +97 -97
  986. package/src/browser/endpoint.spec.ts +148 -148
  987. package/src/browser/endpoint.ts +136 -136
  988. package/src/browser/external-uri-service.ts +79 -79
  989. package/src/browser/file-icons-js.d.ts +20 -20
  990. package/src/browser/frontend-application-bindings.ts +62 -62
  991. package/src/browser/frontend-application-config-provider.spec.ts +45 -45
  992. package/src/browser/frontend-application-config-provider.ts +50 -50
  993. package/src/browser/frontend-application-contribution.ts +110 -110
  994. package/src/browser/frontend-application-module.ts +460 -460
  995. package/src/browser/frontend-application-state.ts +74 -74
  996. package/src/browser/frontend-application.ts +326 -326
  997. package/src/browser/hover-service.ts +218 -218
  998. package/src/browser/http-open-handler.ts +47 -47
  999. package/src/browser/i18n/i18n-frontend-module.ts +27 -27
  1000. package/src/browser/i18n/language-quick-pick-service.ts +130 -130
  1001. package/src/browser/icon-registry.ts +87 -87
  1002. package/src/browser/icon-theme-contribution.ts +64 -64
  1003. package/src/browser/icon-theme-service.ts +217 -217
  1004. package/src/browser/icons/CollapseAll.svg +7 -7
  1005. package/src/browser/icons/CollapseAll_inverse.svg +7 -7
  1006. package/src/browser/icons/Refresh.svg +7 -7
  1007. package/src/browser/icons/Refresh_inverse.svg +7 -7
  1008. package/src/browser/icons/add-inverse.svg +4 -4
  1009. package/src/browser/icons/add.svg +4 -4
  1010. package/src/browser/icons/arrow-down-bright.svg +6 -6
  1011. package/src/browser/icons/arrow-down-dark.svg +6 -6
  1012. package/src/browser/icons/arrow-up-bright.svg +6 -6
  1013. package/src/browser/icons/arrow-up-dark.svg +6 -6
  1014. package/src/browser/icons/case-sensitive-dark.svg +16 -16
  1015. package/src/browser/icons/case-sensitive.svg +16 -16
  1016. package/src/browser/icons/chevron-right-dark.svg +5 -5
  1017. package/src/browser/icons/chevron-right-light.svg +6 -6
  1018. package/src/browser/icons/circle-bright.svg +7 -7
  1019. package/src/browser/icons/circle-dark.svg +7 -7
  1020. package/src/browser/icons/clear-search-results-dark.svg +7 -7
  1021. package/src/browser/icons/clear-search-results.svg +7 -7
  1022. package/src/browser/icons/close-all-bright.svg +7 -7
  1023. package/src/browser/icons/close-all-dark.svg +7 -7
  1024. package/src/browser/icons/close-bright.svg +7 -7
  1025. package/src/browser/icons/close-dark.svg +7 -7
  1026. package/src/browser/icons/collapse.svg +4 -4
  1027. package/src/browser/icons/edit-json-dark.svg +6 -6
  1028. package/src/browser/icons/edit-json.svg +6 -6
  1029. package/src/browser/icons/expand.svg +4 -4
  1030. package/src/browser/icons/loading-dark.svg +6 -6
  1031. package/src/browser/icons/loading-light.svg +6 -6
  1032. package/src/browser/icons/open-change-bright.svg +3 -3
  1033. package/src/browser/icons/open-change-dark.svg +4 -4
  1034. package/src/browser/icons/open-file-bright.svg +4 -4
  1035. package/src/browser/icons/open-file-dark.svg +4 -4
  1036. package/src/browser/icons/preview-bright.svg +3 -3
  1037. package/src/browser/icons/preview-dark.svg +3 -3
  1038. package/src/browser/icons/regex-dark.svg +10 -10
  1039. package/src/browser/icons/regex.svg +10 -10
  1040. package/src/browser/icons/remove-all-inverse.svg +4 -4
  1041. package/src/browser/icons/remove-all.svg +4 -4
  1042. package/src/browser/icons/replace-all-inverse.svg +13 -13
  1043. package/src/browser/icons/replace-all.svg +13 -13
  1044. package/src/browser/icons/replace-inverse.svg +15 -15
  1045. package/src/browser/icons/replace.svg +15 -15
  1046. package/src/browser/icons/whole-word-dark.svg +19 -19
  1047. package/src/browser/icons/whole-word.svg +19 -19
  1048. package/src/browser/index.ts +47 -47
  1049. package/src/browser/json-schema-store.ts +127 -127
  1050. package/src/browser/keybinding.spec.ts +553 -553
  1051. package/src/browser/keybinding.ts +759 -759
  1052. package/src/browser/keyboard/browser-keyboard-frontend-contribution.ts +108 -108
  1053. package/src/browser/keyboard/browser-keyboard-layout-provider.spec.ts +171 -171
  1054. package/src/browser/keyboard/browser-keyboard-layout-provider.ts +469 -469
  1055. package/src/browser/keyboard/browser-keyboard-module.ts +30 -30
  1056. package/src/browser/keyboard/index.ts +20 -20
  1057. package/src/browser/keyboard/keyboard-layout-service.spec.ts +121 -121
  1058. package/src/browser/keyboard/keyboard-layout-service.ts +455 -455
  1059. package/src/browser/keyboard/keys.spec.ts +258 -258
  1060. package/src/browser/keyboard/keys.ts +20 -20
  1061. package/src/browser/keys.ts +21 -21
  1062. package/src/browser/label-parser.spec.ts +165 -165
  1063. package/src/browser/label-parser.ts +108 -108
  1064. package/src/browser/label-provider.spec.ts +62 -62
  1065. package/src/browser/label-provider.ts +385 -385
  1066. package/src/browser/language-icon-provider.ts +55 -55
  1067. package/src/browser/language-service.ts +77 -77
  1068. package/src/browser/logger-frontend-module.ts +65 -65
  1069. package/src/browser/markdown-rendering/markdown-renderer.ts +98 -98
  1070. package/src/browser/menu/browser-context-menu-renderer.ts +48 -48
  1071. package/src/browser/menu/browser-menu-module.ts +28 -28
  1072. package/src/browser/menu/browser-menu-plugin.ts +484 -484
  1073. package/src/browser/menu/context-menu-context.ts +41 -41
  1074. package/src/browser/messaging/connection-source.ts +26 -26
  1075. package/src/browser/messaging/frontend-id-provider.ts +37 -37
  1076. package/src/browser/messaging/index.ts +17 -17
  1077. package/src/browser/messaging/messaging-frontend-module.ts +41 -41
  1078. package/src/browser/messaging/service-connection-provider.ts +126 -126
  1079. package/src/browser/messaging/ws-connection-provider.ts +48 -48
  1080. package/src/browser/messaging/ws-connection-source.ts +230 -230
  1081. package/src/browser/mime-service.ts +30 -30
  1082. package/src/browser/navigatable-types.ts +81 -81
  1083. package/src/browser/navigatable.ts +39 -39
  1084. package/src/browser/opener-service.spec.ts +49 -49
  1085. package/src/browser/opener-service.ts +146 -146
  1086. package/src/browser/performance/frontend-stopwatch.ts +65 -65
  1087. package/src/browser/performance/index.ts +18 -18
  1088. package/src/browser/performance/measurement-frontend-bindings.ts +31 -31
  1089. package/src/browser/preferences/index.ts +23 -23
  1090. package/src/browser/preferences/injectable-preference-proxy.ts +283 -283
  1091. package/src/browser/preferences/preference-configurations.ts +82 -82
  1092. package/src/browser/preferences/preference-contribution.ts +436 -436
  1093. package/src/browser/preferences/preference-language-override-service.ts +111 -111
  1094. package/src/browser/preferences/preference-provider.spec.ts +36 -36
  1095. package/src/browser/preferences/preference-provider.ts +277 -277
  1096. package/src/browser/preferences/preference-proxy.spec.ts +367 -367
  1097. package/src/browser/preferences/preference-proxy.ts +367 -367
  1098. package/src/browser/preferences/preference-schema-provider.spec.ts +130 -130
  1099. package/src/browser/preferences/preference-scope.ts +18 -18
  1100. package/src/browser/preferences/preference-service.spec.ts +613 -613
  1101. package/src/browser/preferences/preference-service.ts +594 -594
  1102. package/src/browser/preferences/preference-validation-service.spec.ts +334 -334
  1103. package/src/browser/preferences/preference-validation-service.ts +358 -358
  1104. package/src/browser/preferences/test/index.ts +19 -19
  1105. package/src/browser/preferences/test/mock-preference-provider.ts +50 -50
  1106. package/src/browser/preferences/test/mock-preference-proxy.ts +48 -48
  1107. package/src/browser/preferences/test/mock-preference-service.ts +63 -63
  1108. package/src/browser/preload/i18n-preload-contribution.ts +50 -50
  1109. package/src/browser/preload/os-preload-contribution.ts +37 -37
  1110. package/src/browser/preload/preload-module.ts +45 -45
  1111. package/src/browser/preload/preloader.ts +37 -37
  1112. package/src/browser/preload/theme-preload-contribution.ts +31 -31
  1113. package/src/browser/progress-bar-factory.ts +29 -29
  1114. package/src/browser/progress-bar.ts +76 -76
  1115. package/src/browser/progress-client.ts +53 -53
  1116. package/src/browser/progress-location-service.spec.ts +50 -50
  1117. package/src/browser/progress-location-service.ts +96 -96
  1118. package/src/browser/progress-status-bar-item.ts +83 -83
  1119. package/src/browser/quick-input/index.ts +23 -23
  1120. package/src/browser/quick-input/quick-access.ts +75 -75
  1121. package/src/browser/quick-input/quick-command-frontend-contribution.ts +89 -89
  1122. package/src/browser/quick-input/quick-command-service.ts +246 -246
  1123. package/src/browser/quick-input/quick-help-service.ts +87 -87
  1124. package/src/browser/quick-input/quick-input-frontend-contribution.ts +33 -33
  1125. package/src/browser/quick-input/quick-input-service.spec.ts +176 -176
  1126. package/src/browser/quick-input/quick-input-service.ts +17 -17
  1127. package/src/browser/quick-input/quick-pick-service-impl.ts +69 -69
  1128. package/src/browser/quick-input/quick-view-service.ts +83 -83
  1129. package/src/browser/request/browser-request-module.ts +23 -23
  1130. package/src/browser/request/browser-request-service.ts +172 -172
  1131. package/src/browser/resource-context-key.ts +77 -77
  1132. package/src/browser/save-resource-service.ts +60 -60
  1133. package/src/browser/saveable.ts +365 -365
  1134. package/src/browser/secondary-window-handler.ts +211 -211
  1135. package/src/browser/shell/additional-views-menu-widget.tsx +71 -71
  1136. package/src/browser/shell/application-shell-mouse-tracker.ts +103 -103
  1137. package/src/browser/shell/application-shell.ts +2261 -2261
  1138. package/src/browser/shell/current-widget-command-adapter.ts +57 -57
  1139. package/src/browser/shell/index.ts +23 -23
  1140. package/src/browser/shell/shell-layout-restorer.ts +399 -399
  1141. package/src/browser/shell/side-panel-handler.ts +793 -793
  1142. package/src/browser/shell/side-panel-toolbar.ts +111 -111
  1143. package/src/browser/shell/sidebar-bottom-menu-widget.tsx +39 -39
  1144. package/src/browser/shell/sidebar-menu-widget.tsx +140 -140
  1145. package/src/browser/shell/sidebar-top-menu-widget.tsx +26 -26
  1146. package/src/browser/shell/split-panels.ts +190 -190
  1147. package/src/browser/shell/tab-bar-decorator.ts +106 -106
  1148. package/src/browser/shell/tab-bar-toolbar/index.ts +19 -19
  1149. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar-menu-adapters.ts +31 -31
  1150. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar-registry.ts +256 -256
  1151. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar-types.ts +207 -207
  1152. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar.spec.ts +62 -62
  1153. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar.tsx +428 -428
  1154. package/src/browser/shell/tab-bars.spec.ts +63 -63
  1155. package/src/browser/shell/tab-bars.ts +1485 -1485
  1156. package/src/browser/shell/theia-dock-panel.ts +255 -255
  1157. package/src/browser/shell/view-contribution.ts +177 -177
  1158. package/src/browser/source-tree/index.ts +19 -19
  1159. package/src/browser/source-tree/source-tree-widget.tsx +107 -107
  1160. package/src/browser/source-tree/source-tree.ts +146 -146
  1161. package/src/browser/source-tree/tree-source.ts +73 -73
  1162. package/src/browser/status-bar/index.ts +29 -29
  1163. package/src/browser/status-bar/status-bar-types.ts +97 -97
  1164. package/src/browser/status-bar/status-bar-view-model.ts +209 -209
  1165. package/src/browser/status-bar/status-bar.tsx +189 -189
  1166. package/src/browser/storage-service.spec.ts +76 -76
  1167. package/src/browser/storage-service.ts +129 -129
  1168. package/src/browser/style/about.css +36 -36
  1169. package/src/browser/style/alert-messages.css +62 -62
  1170. package/src/browser/style/ansi.css +88 -88
  1171. package/src/browser/style/breadcrumbs.css +130 -130
  1172. package/src/browser/style/dialog.css +126 -126
  1173. package/src/browser/style/dockpanel.css +76 -76
  1174. package/src/browser/style/hover-service.css +101 -101
  1175. package/src/browser/style/icons.css +61 -61
  1176. package/src/browser/style/index.css +351 -351
  1177. package/src/browser/style/materialcolors.css +278 -278
  1178. package/src/browser/style/menus.css +230 -230
  1179. package/src/browser/style/notification.css +39 -39
  1180. package/src/browser/style/os.css +87 -87
  1181. package/src/browser/style/progress-bar.css +43 -43
  1182. package/src/browser/style/quick-title-bar.css +45 -45
  1183. package/src/browser/style/scrollbars.css +172 -172
  1184. package/src/browser/style/search-box.css +123 -123
  1185. package/src/browser/style/select-component.css +107 -107
  1186. package/src/browser/style/sidepanel.css +364 -364
  1187. package/src/browser/style/status-bar.css +127 -127
  1188. package/src/browser/style/tabs.css +647 -647
  1189. package/src/browser/style/tooltip.css +28 -28
  1190. package/src/browser/style/tree-decorators.css +81 -81
  1191. package/src/browser/style/tree.css +232 -232
  1192. package/src/browser/style/view-container.css +194 -194
  1193. package/src/browser/style/widget.css +19 -19
  1194. package/src/browser/styling-service.ts +96 -96
  1195. package/src/browser/supported-encodings.ts +262 -262
  1196. package/src/browser/test/jsdom.ts +69 -69
  1197. package/src/browser/test/mock-connection-status-service.ts +33 -33
  1198. package/src/browser/test/mock-env-variables-server.ts +47 -47
  1199. package/src/browser/test/mock-opener-service.ts +34 -34
  1200. package/src/browser/test/mock-storage-service.ts +49 -49
  1201. package/src/browser/theming.ts +206 -206
  1202. package/src/browser/tooltip-service.tsx +96 -96
  1203. package/src/browser/tree/fuzzy-search.spec.ts +99 -99
  1204. package/src/browser/tree/fuzzy-search.ts +136 -136
  1205. package/src/browser/tree/index.ts +29 -29
  1206. package/src/browser/tree/search-box-debounce.ts +96 -96
  1207. package/src/browser/tree/search-box.ts +355 -355
  1208. package/src/browser/tree/test/mock-selectable-tree-model.ts +109 -109
  1209. package/src/browser/tree/test/mock-tree-model.ts +136 -136
  1210. package/src/browser/tree/test/tree-test-container.ts +50 -50
  1211. package/src/browser/tree/tree-compression/compressed-tree-expansion-service.ts +46 -46
  1212. package/src/browser/tree/tree-compression/compressed-tree-model.ts +88 -88
  1213. package/src/browser/tree/tree-compression/compressed-tree-widget.tsx +203 -203
  1214. package/src/browser/tree/tree-compression/index.ts +20 -20
  1215. package/src/browser/tree/tree-compression/tree-compression-service.ts +125 -125
  1216. package/src/browser/tree/tree-compression/tree-compression.css +28 -28
  1217. package/src/browser/tree/tree-consistency.spec.ts +105 -105
  1218. package/src/browser/tree/tree-container.spec.ts +45 -45
  1219. package/src/browser/tree/tree-container.ts +155 -155
  1220. package/src/browser/tree/tree-decorator.spec.ts +162 -162
  1221. package/src/browser/tree/tree-decorator.ts +238 -238
  1222. package/src/browser/tree/tree-expansion.spec.ts +173 -173
  1223. package/src/browser/tree/tree-expansion.ts +165 -165
  1224. package/src/browser/tree/tree-focus-service.ts +55 -55
  1225. package/src/browser/tree/tree-iterator.spec.ts +170 -170
  1226. package/src/browser/tree/tree-iterator.ts +256 -256
  1227. package/src/browser/tree/tree-label-provider.ts +40 -40
  1228. package/src/browser/tree/tree-model.ts +562 -562
  1229. package/src/browser/tree/tree-navigation.ts +58 -58
  1230. package/src/browser/tree/tree-preference.ts +50 -50
  1231. package/src/browser/tree/tree-search.ts +128 -128
  1232. package/src/browser/tree/tree-selectable.spec.ts +152 -152
  1233. package/src/browser/tree/tree-selection-impl.ts +176 -176
  1234. package/src/browser/tree/tree-selection-state.spec.ts +462 -462
  1235. package/src/browser/tree/tree-selection-state.ts +245 -245
  1236. package/src/browser/tree/tree-selection.ts +159 -159
  1237. package/src/browser/tree/tree-view-welcome-widget.tsx +263 -263
  1238. package/src/browser/tree/tree-widget-selection.ts +45 -45
  1239. package/src/browser/tree/tree-widget.tsx +1585 -1585
  1240. package/src/browser/tree/tree.spec.ts +241 -241
  1241. package/src/browser/tree/tree.ts +425 -425
  1242. package/src/browser/user-working-directory-provider.ts +77 -77
  1243. package/src/browser/view-container.ts +1640 -1640
  1244. package/src/browser/widget-decoration.ts +358 -358
  1245. package/src/browser/widget-manager.spec.ts +102 -102
  1246. package/src/browser/widget-manager.ts +310 -310
  1247. package/src/browser/widget-open-handler.ts +165 -165
  1248. package/src/browser/widgets/alert-message.tsx +56 -56
  1249. package/src/browser/widgets/enhanced-preview-widget.ts +27 -27
  1250. package/src/browser/widgets/extractable-widget.ts +33 -33
  1251. package/src/browser/widgets/index.ts +20 -20
  1252. package/src/browser/widgets/previewable-widget.ts +31 -31
  1253. package/src/browser/widgets/react-renderer.tsx +50 -50
  1254. package/src/browser/widgets/react-widget.tsx +51 -51
  1255. package/src/browser/widgets/select-component.tsx +367 -367
  1256. package/src/browser/widgets/widget.ts +406 -406
  1257. package/src/browser/window/browser-window-module.ts +32 -32
  1258. package/src/browser/window/default-secondary-window-service.ts +185 -185
  1259. package/src/browser/window/default-window-service.spec.ts +78 -78
  1260. package/src/browser/window/default-window-service.ts +171 -171
  1261. package/src/browser/window/secondary-window-service.ts +39 -39
  1262. package/src/browser/window/test/mock-window-service.ts +29 -29
  1263. package/src/browser/window/window-service.ts +78 -78
  1264. package/src/browser/window/window-title-service.ts +107 -107
  1265. package/src/browser/window/window-title-updater.ts +95 -95
  1266. package/src/browser/window-contribution.ts +64 -64
  1267. package/src/browser-only/frontend-only-application-module.ts +115 -115
  1268. package/src/browser-only/i18n/i18n-frontend-only-module.ts +37 -37
  1269. package/src/browser-only/logger-frontend-only-module.ts +63 -63
  1270. package/src/browser-only/messaging/frontend-only-service-connection-provider.ts +39 -39
  1271. package/src/browser-only/messaging/messaging-frontend-only-module.ts +42 -42
  1272. package/src/browser-only/preload/frontend-only-preload-module.ts +49 -49
  1273. package/src/common/accessibility.ts +33 -33
  1274. package/src/common/application-error.spec.ts +27 -27
  1275. package/src/common/application-error.ts +76 -76
  1276. package/src/common/application-protocol.ts +41 -41
  1277. package/src/common/array-utils.ts +129 -129
  1278. package/src/common/buffer.ts +228 -228
  1279. package/src/common/cancellation.ts +163 -163
  1280. package/src/common/char-code.ts +438 -438
  1281. package/src/common/collections.ts +125 -125
  1282. package/src/common/color.ts +103 -103
  1283. package/src/common/command.spec.ts +208 -208
  1284. package/src/common/command.ts +485 -485
  1285. package/src/common/contribution-filter/contribution-filter-registry.ts +79 -79
  1286. package/src/common/contribution-filter/contribution-filter.ts +64 -64
  1287. package/src/common/contribution-filter/filter.ts +23 -23
  1288. package/src/common/contribution-filter/index.ts +19 -19
  1289. package/src/common/contribution-provider.ts +96 -96
  1290. package/src/common/disposable.spec.ts +94 -94
  1291. package/src/common/disposable.ts +188 -188
  1292. package/src/common/encoding-service.ts +380 -380
  1293. package/src/common/encodings.ts +24 -24
  1294. package/src/common/env-variables/env-variables-protocol.ts +38 -38
  1295. package/src/common/env-variables/index.ts +17 -17
  1296. package/src/common/event.spec.ts +32 -32
  1297. package/src/common/event.ts +469 -469
  1298. package/src/common/file-uri.ts +61 -61
  1299. package/src/common/frontend-application-state.ts +38 -38
  1300. package/src/common/glob.ts +741 -741
  1301. package/src/common/hash.ts +85 -85
  1302. package/src/common/i18n/localization-server.ts +25 -25
  1303. package/src/common/i18n/localization.ts +80 -80
  1304. package/src/common/i18n/nls.metadata.json +31314 -31314
  1305. package/src/common/index.ts +51 -51
  1306. package/src/common/json-schema.ts +106 -106
  1307. package/src/common/key-store.ts +26 -26
  1308. package/src/common/keybinding.ts +152 -152
  1309. package/src/common/keyboard/keyboard-layout-provider.ts +51 -51
  1310. package/src/common/keys.ts +694 -694
  1311. package/src/common/label-protocol.ts +35 -35
  1312. package/src/common/logger-protocol.ts +119 -119
  1313. package/src/common/logger-watcher.ts +48 -48
  1314. package/src/common/logger.spec.ts +46 -46
  1315. package/src/common/logger.ts +389 -389
  1316. package/src/common/lsp-types.ts +34 -34
  1317. package/src/common/markdown-rendering/icon-utilities.ts +30 -30
  1318. package/src/common/markdown-rendering/index.ts +18 -18
  1319. package/src/common/markdown-rendering/markdown-string.ts +152 -152
  1320. package/src/common/menu/action-menu-node.ts +65 -65
  1321. package/src/common/menu/composite-menu-node.spec.ts +67 -67
  1322. package/src/common/menu/composite-menu-node.ts +114 -114
  1323. package/src/common/menu/index.ts +21 -21
  1324. package/src/common/menu/menu-adapter.ts +103 -103
  1325. package/src/common/menu/menu-model-registry.ts +343 -343
  1326. package/src/common/menu/menu-types.ts +219 -219
  1327. package/src/common/menu/menu.spec.ts +101 -101
  1328. package/src/common/message-rpc/channel.spec.ts +88 -88
  1329. package/src/common/message-rpc/channel.ts +300 -300
  1330. package/src/common/message-rpc/index.ts +22 -22
  1331. package/src/common/message-rpc/message-buffer.ts +105 -105
  1332. package/src/common/message-rpc/msg-pack-extension-manager.ts +70 -70
  1333. package/src/common/message-rpc/rpc-message-encoder.spec.ts +65 -65
  1334. package/src/common/message-rpc/rpc-message-encoder.ts +190 -190
  1335. package/src/common/message-rpc/rpc-protocol.ts +255 -255
  1336. package/src/common/message-rpc/uint8-array-message-buffer.spec.ts +41 -41
  1337. package/src/common/message-rpc/uint8-array-message-buffer.ts +213 -213
  1338. package/src/common/message-service-protocol.ts +148 -148
  1339. package/src/common/message-service.ts +226 -226
  1340. package/src/common/messaging/connection-error-handler.ts +73 -73
  1341. package/src/common/messaging/connection-management.ts +43 -43
  1342. package/src/common/messaging/handler.ts +26 -26
  1343. package/src/common/messaging/index.ts +19 -19
  1344. package/src/common/messaging/proxy-factory.spec.ts +108 -108
  1345. package/src/common/messaging/proxy-factory.ts +336 -336
  1346. package/src/common/messaging/socket-write-buffer.ts +52 -52
  1347. package/src/common/messaging/web-socket-channel.ts +76 -76
  1348. package/src/common/nls.ts +151 -151
  1349. package/src/common/numbers.ts +21 -21
  1350. package/src/common/objects.spec.ts +112 -112
  1351. package/src/common/objects.ts +123 -123
  1352. package/src/common/os.ts +82 -82
  1353. package/src/common/path.spec.ts +415 -415
  1354. package/src/common/path.ts +334 -334
  1355. package/src/common/paths.ts +250 -250
  1356. package/src/common/performance/index.ts +19 -19
  1357. package/src/common/performance/measurement-protocol.ts +104 -104
  1358. package/src/common/performance/measurement.ts +130 -130
  1359. package/src/common/performance/stopwatch.ts +183 -183
  1360. package/src/common/preferences/preference-schema.ts +95 -95
  1361. package/src/common/preferences/preference-scope.spec.ts +48 -48
  1362. package/src/common/preferences/preference-scope.ts +68 -68
  1363. package/src/common/prioritizeable.ts +58 -58
  1364. package/src/common/progress-service-protocol.ts +35 -35
  1365. package/src/common/progress-service.ts +82 -82
  1366. package/src/common/promise-util.spec.ts +102 -102
  1367. package/src/common/promise-util.ts +143 -143
  1368. package/src/common/quick-pick-service.ts +356 -356
  1369. package/src/common/reference.spec.ts +145 -145
  1370. package/src/common/reference.ts +230 -230
  1371. package/src/common/resource.ts +430 -430
  1372. package/src/common/selection-command-handler.ts +101 -101
  1373. package/src/common/selection-service.spec.ts +43 -43
  1374. package/src/common/selection-service.ts +49 -49
  1375. package/src/common/selection.ts +50 -50
  1376. package/src/common/severity.ts +111 -111
  1377. package/src/common/stream.ts +718 -718
  1378. package/src/common/strings.ts +231 -231
  1379. package/src/common/telemetry.ts +45 -45
  1380. package/src/common/ternary-search-tree.ts +417 -417
  1381. package/src/common/test/expect.ts +34 -34
  1382. package/src/common/test/mock-logger.ts +118 -118
  1383. package/src/common/test/mock-menu.ts +35 -35
  1384. package/src/common/test/mock-resource-provider.ts +33 -33
  1385. package/src/common/theme.ts +68 -68
  1386. package/src/common/types.spec.ts +86 -86
  1387. package/src/common/types.ts +140 -140
  1388. package/src/common/uri-command-handler.spec.ts +90 -90
  1389. package/src/common/uri-command-handler.ts +148 -148
  1390. package/src/common/uri.spec.ts +278 -278
  1391. package/src/common/uri.ts +279 -279
  1392. package/src/common/uuid.ts +45 -45
  1393. package/src/common/version.ts +17 -17
  1394. package/src/common/view-column.ts +33 -33
  1395. package/src/common/window.ts +34 -34
  1396. package/src/electron-browser/electron-clipboard-service.ts +32 -32
  1397. package/src/electron-browser/keyboard/electron-keyboard-layout-change-notifier.ts +39 -39
  1398. package/src/electron-browser/keyboard/electron-keyboard-module.ts +28 -28
  1399. package/src/electron-browser/menu/electron-context-menu-renderer.ts +122 -122
  1400. package/src/electron-browser/menu/electron-main-menu-factory.ts +335 -335
  1401. package/src/electron-browser/menu/electron-menu-contribution.ts +506 -506
  1402. package/src/electron-browser/menu/electron-menu-module.ts +40 -40
  1403. package/src/electron-browser/menu/electron-menu-style.css +110 -110
  1404. package/src/electron-browser/messaging/electron-frontend-id-provider.ts +25 -25
  1405. package/src/electron-browser/messaging/electron-ipc-connection-source.ts +65 -65
  1406. package/src/electron-browser/messaging/electron-local-ws-connection-source.ts +45 -45
  1407. package/src/electron-browser/messaging/electron-messaging-frontend-module.ts +78 -78
  1408. package/src/electron-browser/messaging/electron-ws-connection-source.ts +38 -38
  1409. package/src/electron-browser/preload.ts +246 -246
  1410. package/src/electron-browser/request/electron-browser-request-module.ts +26 -26
  1411. package/src/electron-browser/token/electron-token-frontend-module.ts +22 -22
  1412. package/src/electron-browser/window/electron-frontend-application-state.ts +26 -26
  1413. package/src/electron-browser/window/electron-secondary-window-service.ts +35 -35
  1414. package/src/electron-browser/window/electron-window-module.ts +41 -41
  1415. package/src/electron-browser/window/electron-window-preferences.ts +76 -76
  1416. package/src/electron-browser/window/electron-window-service.ts +109 -109
  1417. package/src/electron-common/electron-api.ts +148 -148
  1418. package/src/electron-common/electron-main-window-service.ts +24 -24
  1419. package/src/electron-common/electron-token.ts +27 -27
  1420. package/src/electron-main/electron-api-main.ts +342 -340
  1421. package/src/electron-main/electron-main-application-module.ts +65 -65
  1422. package/src/electron-main/electron-main-application.ts +685 -685
  1423. package/src/electron-main/electron-main-constants.ts +22 -22
  1424. package/src/electron-main/electron-main-window-service-impl.ts +44 -44
  1425. package/src/electron-main/electron-security-token-service.ts +36 -36
  1426. package/src/electron-main/event-utils.ts +36 -36
  1427. package/src/electron-main/messaging/electron-connection-handler.ts +21 -21
  1428. package/src/electron-main/messaging/electron-messaging-contribution.ts +143 -143
  1429. package/src/electron-main/messaging/electron-messaging-service.ts +35 -35
  1430. package/src/electron-main/theia-electron-window.ts +202 -202
  1431. package/src/electron-node/hosting/electron-backend-hosting-module.ts +24 -24
  1432. package/src/electron-node/hosting/electron-ws-origin-validator.ts +37 -37
  1433. package/src/electron-node/keyboard/electron-backend-keyboard-module.ts +30 -30
  1434. package/src/electron-node/keyboard/electron-keyboard-layout-provider.ts +35 -35
  1435. package/src/electron-node/request/electron-backend-request-module.ts +23 -23
  1436. package/src/electron-node/request/electron-backend-request-service.ts +78 -78
  1437. package/src/electron-node/token/electron-token-backend-contribution.ts +48 -48
  1438. package/src/electron-node/token/electron-token-backend-module.ts +28 -28
  1439. package/src/electron-node/token/electron-token-validator.ts +93 -93
  1440. package/src/node/application-server.ts +55 -55
  1441. package/src/node/backend-application-config-provider.spec.ts +29 -29
  1442. package/src/node/backend-application-config-provider.ts +48 -48
  1443. package/src/node/backend-application-module.ts +140 -140
  1444. package/src/node/backend-application.ts +382 -382
  1445. package/src/node/backend-remote-service.ts +25 -25
  1446. package/src/node/cli.spec.ts +94 -94
  1447. package/src/node/cli.ts +63 -63
  1448. package/src/node/console-logger-server.spec.ts +59 -59
  1449. package/src/node/console-logger-server.ts +76 -76
  1450. package/src/node/debug.ts +30 -30
  1451. package/src/node/dynamic-require.ts +56 -56
  1452. package/src/node/env-variables/env-variables-server.ts +120 -120
  1453. package/src/node/env-variables/index.ts +17 -17
  1454. package/src/node/environment-utils.spec.ts +92 -92
  1455. package/src/node/environment-utils.ts +66 -66
  1456. package/src/node/file-uri.spec.ts +76 -76
  1457. package/src/node/filesystem-locking.ts +77 -77
  1458. package/src/node/hosting/backend-application-hosts.ts +60 -60
  1459. package/src/node/hosting/backend-hosting-module.ts +26 -26
  1460. package/src/node/hosting/ws-origin-validator.ts +36 -36
  1461. package/src/node/i18n/i18n-backend-module.ts +42 -42
  1462. package/src/node/i18n/localization-contribution.ts +112 -112
  1463. package/src/node/i18n/localization-provider.ts +125 -125
  1464. package/src/node/i18n/localization-server.ts +52 -52
  1465. package/src/node/i18n/theia-localization-contribution.ts +36 -36
  1466. package/src/node/index.ts +22 -22
  1467. package/src/node/key-store-server.ts +162 -162
  1468. package/src/node/logger-backend-module.ts +88 -88
  1469. package/src/node/logger-cli-contribution.spec.ts +245 -245
  1470. package/src/node/logger-cli-contribution.ts +168 -168
  1471. package/src/node/main.ts +33 -33
  1472. package/src/node/messaging/binary-message-pipe.ts +168 -168
  1473. package/src/node/messaging/connection-container-module.ts +96 -96
  1474. package/src/node/messaging/default-messaging-service.ts +129 -129
  1475. package/src/node/messaging/frontend-connection-service.ts +24 -24
  1476. package/src/node/messaging/index.ts +19 -19
  1477. package/src/node/messaging/ipc-bootstrap.ts +27 -27
  1478. package/src/node/messaging/ipc-channel.ts +77 -77
  1479. package/src/node/messaging/ipc-connection-provider.ts +107 -107
  1480. package/src/node/messaging/ipc-protocol.ts +76 -76
  1481. package/src/node/messaging/messaging-backend-module.ts +52 -52
  1482. package/src/node/messaging/messaging-listeners.ts +52 -52
  1483. package/src/node/messaging/messaging-service.ts +46 -46
  1484. package/src/node/messaging/test/test-web-socket-channel.ts +61 -61
  1485. package/src/node/messaging/websocket-endpoint.ts +79 -79
  1486. package/src/node/messaging/websocket-frontend-connection-service.ts +176 -176
  1487. package/src/node/os-backend-provider.ts +25 -25
  1488. package/src/node/performance/index.ts +18 -18
  1489. package/src/node/performance/measurement-backend-bindings.ts +35 -35
  1490. package/src/node/performance/node-stopwatch.ts +40 -40
  1491. package/src/node/process-utils.spec.ts +48 -48
  1492. package/src/node/process-utils.ts +102 -102
  1493. package/src/node/request/backend-request-facade.ts +39 -39
  1494. package/src/node/request/backend-request-module.ts +25 -25
  1495. package/src/node/request/proxy-cli-contribution.ts +65 -65
  1496. package/src/node/ws-request-validators.ts +56 -56
  1497. package/src/typings/native-keymap.d.ts +108 -108
@@ -1,2261 +1,2261 @@
1
- // *****************************************************************************
2
- // Copyright (C) 2018 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 { injectable, inject, optional, postConstruct } from 'inversify';
18
- import { ArrayExt, find, toArray, each } from '@phosphor/algorithm';
19
- import {
20
- BoxLayout, BoxPanel, DockLayout, DockPanel, FocusTracker, Layout, Panel, SplitLayout,
21
- SplitPanel, TabBar, Widget, Title
22
- } from '@phosphor/widgets';
23
- import { Message } from '@phosphor/messaging';
24
- import { IDragEvent } from '@phosphor/dragdrop';
25
- import { RecursivePartial, Event as CommonEvent, DisposableCollection, Disposable, environment, isObject } from '../../common';
26
- import { animationFrame } from '../browser';
27
- import { Saveable, SaveableWidget, SaveOptions, SaveableSource } from '../saveable';
28
- import { StatusBarImpl, StatusBarEntry, StatusBarAlignment } from '../status-bar/status-bar';
29
- import { TheiaDockPanel, BOTTOM_AREA_ID, MAIN_AREA_ID } from './theia-dock-panel';
30
- import { SidePanelHandler, SidePanel, SidePanelHandlerFactory } from './side-panel-handler';
31
- import { TabBarRendererFactory, SHELL_TABBAR_CONTEXT_MENU, ScrollableTabBar, ToolbarAwareTabBar } from './tab-bars';
32
- import { SplitPositionHandler, SplitPositionOptions } from './split-panels';
33
- import { FrontendApplicationStateService } from '../frontend-application-state';
34
- import { TabBarToolbarRegistry, TabBarToolbarFactory } from './tab-bar-toolbar';
35
- import { ContextKeyService } from '../context-key-service';
36
- import { Emitter } from '../../common/event';
37
- import { waitForRevealed, waitForClosed, PINNED_CLASS } from '../widgets';
38
- import { CorePreferences } from '../core-preferences';
39
- import { BreadcrumbsRendererFactory } from '../breadcrumbs/breadcrumbs-renderer';
40
- import { Deferred } from '../../common/promise-util';
41
- import { SaveResourceService } from '../save-resource-service';
42
- import { nls } from '../../common/nls';
43
- import { SecondaryWindowHandler } from '../secondary-window-handler';
44
- import URI from '../../common/uri';
45
- import { OpenerService } from '../opener-service';
46
- import { PreviewableWidget } from '../widgets/previewable-widget';
47
- import { WindowService } from '../window/window-service';
48
-
49
- /** The class name added to ApplicationShell instances. */
50
- const APPLICATION_SHELL_CLASS = 'theia-ApplicationShell';
51
- /** The class name added to the main and bottom area panels. */
52
- const MAIN_BOTTOM_AREA_CLASS = 'theia-app-centers';
53
- /** Status bar entry identifier for the bottom panel toggle button. */
54
- const BOTTOM_PANEL_TOGGLE_ID = 'bottom-panel-toggle';
55
- /** The class name added to the main area panel. */
56
- const MAIN_AREA_CLASS = 'theia-app-main';
57
- /** The class name added to the bottom area panel. */
58
- const BOTTOM_AREA_CLASS = 'theia-app-bottom';
59
-
60
- export type ApplicationShellLayoutVersion =
61
- /** layout versioning is introduced, unversioned layout are not compatible */
62
- 2.0 |
63
- /** view containers are introduced, backward compatible to 2.0 */
64
- 3.0 |
65
- /** git history view is replaced by a more generic scm history view, backward compatible to 3.0 */
66
- 4.0 |
67
- /** Replace custom/font-awesome icons with codicons */
68
- 5.0 |
69
- /** added the ability to drag and drop view parts between view containers */
70
- 6.0;
71
-
72
- /**
73
- * When a version is increased, make sure to introduce a migration (ApplicationShellLayoutMigration) to this version.
74
- */
75
- export const applicationShellLayoutVersion: ApplicationShellLayoutVersion = 5.0;
76
-
77
- export const ApplicationShellOptions = Symbol('ApplicationShellOptions');
78
- export const DockPanelRendererFactory = Symbol('DockPanelRendererFactory');
79
- export interface DockPanelRendererFactory {
80
- (): DockPanelRenderer
81
- }
82
-
83
- /**
84
- * A renderer for dock panels that supports context menus on tabs.
85
- */
86
- @injectable()
87
- export class DockPanelRenderer implements DockLayout.IRenderer {
88
-
89
- @inject(TheiaDockPanel.Factory)
90
- protected readonly dockPanelFactory: TheiaDockPanel.Factory;
91
-
92
- readonly tabBarClasses: string[] = [];
93
-
94
- private readonly onDidCreateTabBarEmitter = new Emitter<TabBar<Widget>>();
95
-
96
- constructor(
97
- @inject(TabBarRendererFactory) protected readonly tabBarRendererFactory: TabBarRendererFactory,
98
- @inject(TabBarToolbarRegistry) protected readonly tabBarToolbarRegistry: TabBarToolbarRegistry,
99
- @inject(TabBarToolbarFactory) protected readonly tabBarToolbarFactory: TabBarToolbarFactory,
100
- @inject(BreadcrumbsRendererFactory) protected readonly breadcrumbsRendererFactory: BreadcrumbsRendererFactory,
101
- @inject(CorePreferences) protected readonly corePreferences: CorePreferences
102
- ) { }
103
-
104
- get onDidCreateTabBar(): CommonEvent<TabBar<Widget>> {
105
- return this.onDidCreateTabBarEmitter.event;
106
- }
107
-
108
- createTabBar(): TabBar<Widget> {
109
- const getDynamicTabOptions: () => ScrollableTabBar.Options | undefined = () => {
110
- if (this.corePreferences.get('workbench.tab.shrinkToFit.enabled')) {
111
- return {
112
- minimumTabSize: this.corePreferences.get('workbench.tab.shrinkToFit.minimumSize'),
113
- defaultTabSize: this.corePreferences.get('workbench.tab.shrinkToFit.defaultSize')
114
- };
115
- } else {
116
- return undefined;
117
- }
118
- };
119
-
120
- const renderer = this.tabBarRendererFactory();
121
- const tabBar = new ToolbarAwareTabBar(
122
- this.tabBarToolbarRegistry,
123
- this.tabBarToolbarFactory,
124
- this.breadcrumbsRendererFactory,
125
- {
126
- renderer,
127
- // Scroll bar options
128
- handlers: ['drag-thumb', 'keyboard', 'wheel', 'touch'],
129
- useBothWheelAxes: true,
130
- scrollXMarginOffset: 4,
131
- suppressScrollY: true
132
- },
133
- getDynamicTabOptions());
134
- this.tabBarClasses.forEach(c => tabBar.addClass(c));
135
- renderer.tabBar = tabBar;
136
- renderer.contextMenuPath = SHELL_TABBAR_CONTEXT_MENU;
137
- tabBar.currentChanged.connect(this.onCurrentTabChanged, this);
138
- const prefChangeDisposable = this.corePreferences.onPreferenceChanged(change => {
139
- if (change.preferenceName === 'workbench.tab.shrinkToFit.enabled' ||
140
- change.preferenceName === 'workbench.tab.shrinkToFit.minimumSize' ||
141
- change.preferenceName === 'workbench.tab.shrinkToFit.defaultSize') {
142
- tabBar.dynamicTabOptions = getDynamicTabOptions();
143
- }
144
- });
145
- tabBar.disposed.connect(() => {
146
- prefChangeDisposable.dispose();
147
- renderer.dispose();
148
- });
149
- this.onDidCreateTabBarEmitter.fire(tabBar);
150
- return tabBar;
151
- }
152
-
153
- createHandle(): HTMLDivElement {
154
- return DockPanel.defaultRenderer.createHandle();
155
- }
156
-
157
- protected onCurrentTabChanged(sender: ToolbarAwareTabBar, { currentIndex }: TabBar.ICurrentChangedArgs<Widget>): void {
158
- if (currentIndex >= 0) {
159
- sender.revealTab(currentIndex);
160
- }
161
- }
162
- }
163
-
164
- /**
165
- * Data stored while dragging widgets in the shell.
166
- */
167
- interface WidgetDragState {
168
- startTime: number;
169
- leftExpanded: boolean;
170
- rightExpanded: boolean;
171
- bottomExpanded: boolean;
172
- lastDragOver?: IDragEvent;
173
- leaveTimeout?: number;
174
- }
175
-
176
- /**
177
- * The application shell manages the top-level widgets of the application. Use this class to
178
- * add, remove, or activate a widget.
179
- */
180
- @injectable()
181
- export class ApplicationShell extends Widget {
182
-
183
- /**
184
- * The dock panel in the main shell area. This is where editors usually go to.
185
- */
186
- mainPanel: TheiaDockPanel;
187
-
188
- /**
189
- * The dock panel in the bottom shell area. In contrast to the main panel, the bottom panel
190
- * can be collapsed and expanded.
191
- */
192
- bottomPanel: TheiaDockPanel;
193
-
194
- /**
195
- * Handler for the left side panel. The primary application views go here, such as the
196
- * file explorer and the git view.
197
- */
198
- leftPanelHandler: SidePanelHandler;
199
-
200
- /**
201
- * Handler for the right side panel. The secondary application views go here, such as the
202
- * outline view.
203
- */
204
- rightPanelHandler: SidePanelHandler;
205
-
206
- /**
207
- * General options for the application shell.
208
- */
209
- protected options: ApplicationShell.Options;
210
-
211
- /**
212
- * The fixed-size panel shown on top. This one usually holds the main menu.
213
- */
214
- topPanel: Panel;
215
-
216
- /**
217
- * The current state of the bottom panel.
218
- */
219
- protected readonly bottomPanelState: SidePanel.State = {
220
- empty: true,
221
- expansion: SidePanel.ExpansionState.collapsed,
222
- pendingUpdate: Promise.resolve()
223
- };
224
-
225
- private readonly tracker = new FocusTracker<Widget>();
226
- private dragState?: WidgetDragState;
227
- additionalDraggedUris: URI[] | undefined;
228
-
229
- @inject(ContextKeyService)
230
- protected readonly contextKeyService: ContextKeyService;
231
-
232
- @inject(OpenerService)
233
- protected readonly openerService: OpenerService;
234
-
235
- protected readonly onDidAddWidgetEmitter = new Emitter<Widget>();
236
- readonly onDidAddWidget = this.onDidAddWidgetEmitter.event;
237
- protected fireDidAddWidget(widget: Widget): void {
238
- this.onDidAddWidgetEmitter.fire(widget);
239
- }
240
-
241
- protected readonly onDidRemoveWidgetEmitter = new Emitter<Widget>();
242
- readonly onDidRemoveWidget = this.onDidRemoveWidgetEmitter.event;
243
- protected fireDidRemoveWidget(widget: Widget): void {
244
- this.onDidRemoveWidgetEmitter.fire(widget);
245
- }
246
-
247
- protected readonly onDidChangeActiveWidgetEmitter = new Emitter<FocusTracker.IChangedArgs<Widget>>();
248
- readonly onDidChangeActiveWidget = this.onDidChangeActiveWidgetEmitter.event;
249
-
250
- protected readonly onDidChangeCurrentWidgetEmitter = new Emitter<FocusTracker.IChangedArgs<Widget>>();
251
- readonly onDidChangeCurrentWidget = this.onDidChangeCurrentWidgetEmitter.event;
252
-
253
- protected readonly onDidDoubleClickMainAreaEmitter = new Emitter<void>();
254
- readonly onDidDoubleClickMainArea = this.onDidDoubleClickMainAreaEmitter.event;
255
-
256
- @inject(TheiaDockPanel.Factory)
257
- protected readonly dockPanelFactory: TheiaDockPanel.Factory;
258
-
259
- private _mainPanelRenderer: DockPanelRenderer;
260
- get mainPanelRenderer(): DockPanelRenderer {
261
- return this._mainPanelRenderer;
262
- }
263
-
264
- /**
265
- * Construct a new application shell.
266
- */
267
- constructor(
268
- @inject(DockPanelRendererFactory) protected dockPanelRendererFactory: () => DockPanelRenderer,
269
- @inject(StatusBarImpl) protected readonly statusBar: StatusBarImpl,
270
- @inject(SidePanelHandlerFactory) protected readonly sidePanelHandlerFactory: () => SidePanelHandler,
271
- @inject(SplitPositionHandler) protected splitPositionHandler: SplitPositionHandler,
272
- @inject(FrontendApplicationStateService) protected readonly applicationStateService: FrontendApplicationStateService,
273
- @inject(ApplicationShellOptions) @optional() options: RecursivePartial<ApplicationShell.Options> = {},
274
- @inject(CorePreferences) protected readonly corePreferences: CorePreferences,
275
- @inject(SaveResourceService) protected readonly saveResourceService: SaveResourceService,
276
- @inject(SecondaryWindowHandler) protected readonly secondaryWindowHandler: SecondaryWindowHandler,
277
- @inject(WindowService) protected readonly windowService: WindowService
278
- ) {
279
- super(options as Widget.IOptions);
280
-
281
- // Merge the user-defined application options with the default options
282
- this.options = {
283
- bottomPanel: {
284
- ...ApplicationShell.DEFAULT_OPTIONS.bottomPanel,
285
- ...options?.bottomPanel || {}
286
- },
287
- leftPanel: {
288
- ...ApplicationShell.DEFAULT_OPTIONS.leftPanel,
289
- ...options?.leftPanel || {}
290
- },
291
- rightPanel: {
292
- ...ApplicationShell.DEFAULT_OPTIONS.rightPanel,
293
- ...options?.rightPanel || {}
294
- }
295
- };
296
- }
297
-
298
- @postConstruct()
299
- protected init(): void {
300
- this.initializeShell();
301
- this.initSidebarVisibleKeyContext();
302
- this.initFocusKeyContexts();
303
-
304
- if (!environment.electron.is()) {
305
- this.corePreferences.ready.then(() => {
306
- this.setTopPanelVisibility(this.corePreferences['window.menuBarVisibility']);
307
- });
308
- this.corePreferences.onPreferenceChanged(preference => {
309
- if (preference.preferenceName === 'window.menuBarVisibility') {
310
- this.setTopPanelVisibility(preference.newValue);
311
- }
312
- });
313
- }
314
-
315
- this.corePreferences.onPreferenceChanged(preference => {
316
- if (preference.preferenceName === 'window.tabbar.enhancedPreview') {
317
- this.allTabBars.forEach(tabBar => {
318
- tabBar.update();
319
- });
320
- }
321
- });
322
- }
323
-
324
- protected initializeShell(): void {
325
- this.addClass(APPLICATION_SHELL_CLASS);
326
- this.id = 'theia-app-shell';
327
-
328
- this.mainPanel = this.createMainPanel();
329
- this.topPanel = this.createTopPanel();
330
- this.bottomPanel = this.createBottomPanel();
331
-
332
- this.leftPanelHandler = this.sidePanelHandlerFactory();
333
- this.leftPanelHandler.create('left', this.options.leftPanel);
334
- this.leftPanelHandler.dockPanel.widgetAdded.connect((_, widget) => this.fireDidAddWidget(widget));
335
- this.leftPanelHandler.dockPanel.widgetRemoved.connect((_, widget) => this.fireDidRemoveWidget(widget));
336
-
337
- this.rightPanelHandler = this.sidePanelHandlerFactory();
338
- this.rightPanelHandler.create('right', this.options.rightPanel);
339
- this.rightPanelHandler.dockPanel.widgetAdded.connect((_, widget) => this.fireDidAddWidget(widget));
340
- this.rightPanelHandler.dockPanel.widgetRemoved.connect((_, widget) => this.fireDidRemoveWidget(widget));
341
-
342
- this.secondaryWindowHandler.init(this);
343
- this.secondaryWindowHandler.onDidAddWidget(([widget, window]) => this.fireDidAddWidget(widget));
344
- this.secondaryWindowHandler.onDidRemoveWidget(([widget, window]) => this.fireDidRemoveWidget(widget));
345
-
346
- this.layout = this.createLayout();
347
-
348
- this.tracker.currentChanged.connect(this.onCurrentChanged, this);
349
- this.tracker.activeChanged.connect(this.onActiveChanged, this);
350
- }
351
-
352
- protected initSidebarVisibleKeyContext(): void {
353
- const leftSideBarPanel = this.leftPanelHandler.dockPanel;
354
- const sidebarVisibleKey = this.contextKeyService.createKey('sidebarVisible', leftSideBarPanel.isVisible);
355
- const onAfterShow = leftSideBarPanel['onAfterShow'].bind(leftSideBarPanel);
356
- leftSideBarPanel['onAfterShow'] = (msg: Message) => {
357
- onAfterShow(msg);
358
- sidebarVisibleKey.set(true);
359
- };
360
- const onAfterHide = leftSideBarPanel['onAfterHide'].bind(leftSideBarPanel);
361
- leftSideBarPanel['onAfterHide'] = (msg: Message) => {
362
- onAfterHide(msg);
363
- sidebarVisibleKey.set(false);
364
- };
365
- }
366
-
367
- protected initFocusKeyContexts(): void {
368
- const sideBarFocus = this.contextKeyService.createKey<boolean>('sideBarFocus', false);
369
- const panelFocus = this.contextKeyService.createKey<boolean>('panelFocus', false);
370
- const updateFocusContextKeys = () => {
371
- const area = this.activeWidget && this.getAreaFor(this.activeWidget);
372
- sideBarFocus.set(area === 'left');
373
- panelFocus.set(area === 'main');
374
- };
375
- updateFocusContextKeys();
376
- this.onDidChangeActiveWidget(updateFocusContextKeys);
377
- }
378
-
379
- protected setTopPanelVisibility(preference: string): void {
380
- const hiddenPreferences = ['compact', 'hidden'];
381
- this.topPanel.setHidden(hiddenPreferences.includes(preference));
382
- }
383
-
384
- protected override onBeforeAttach(msg: Message): void {
385
- document.addEventListener('p-dragenter', this, true);
386
- document.addEventListener('p-dragover', this, true);
387
- document.addEventListener('p-dragleave', this, true);
388
- document.addEventListener('p-drop', this, true);
389
- }
390
-
391
- protected override onAfterDetach(msg: Message): void {
392
- document.removeEventListener('p-dragenter', this, true);
393
- document.removeEventListener('p-dragover', this, true);
394
- document.removeEventListener('p-dragleave', this, true);
395
- document.removeEventListener('p-drop', this, true);
396
- }
397
-
398
- handleEvent(event: Event): void {
399
- switch (event.type) {
400
- case 'p-dragenter':
401
- this.onDragEnter(event as IDragEvent);
402
- break;
403
- case 'p-dragover':
404
- this.onDragOver(event as IDragEvent);
405
- break;
406
- case 'p-drop':
407
- this.onDrop(event as IDragEvent);
408
- break;
409
- case 'p-dragleave':
410
- this.onDragLeave(event as IDragEvent);
411
- break;
412
- }
413
- }
414
-
415
- protected onDragEnter({ mimeData }: IDragEvent): void {
416
- if (!this.dragState) {
417
- if (mimeData && mimeData.hasData('application/vnd.phosphor.widget-factory')) {
418
- // The drag contains a widget, so we'll track it and expand side panels as needed
419
- this.dragState = {
420
- startTime: performance.now(),
421
- leftExpanded: false,
422
- rightExpanded: false,
423
- bottomExpanded: false
424
- };
425
- }
426
- }
427
- }
428
-
429
- protected onDragOver(event: IDragEvent): void {
430
- const state = this.dragState;
431
- if (state) {
432
- state.lastDragOver = event;
433
- if (state.leaveTimeout) {
434
- window.clearTimeout(state.leaveTimeout);
435
- state.leaveTimeout = undefined;
436
- }
437
- const { clientX, clientY } = event;
438
- const { offsetLeft, offsetTop, clientWidth, clientHeight } = this.node;
439
-
440
- // Don't expand any side panels right after the drag has started
441
- const allowExpansion = performance.now() - state.startTime >= 500;
442
- const expLeft = allowExpansion && clientX >= offsetLeft
443
- && clientX <= offsetLeft + this.options.leftPanel.expandThreshold;
444
- const expRight = allowExpansion && clientX <= offsetLeft + clientWidth
445
- && clientX >= offsetLeft + clientWidth - this.options.rightPanel.expandThreshold;
446
- const expBottom = allowExpansion && !expLeft && !expRight && clientY <= offsetTop + clientHeight
447
- && clientY >= offsetTop + clientHeight - this.options.bottomPanel.expandThreshold;
448
- // eslint-disable-next-line no-null/no-null
449
- if (expLeft && !state.leftExpanded && this.leftPanelHandler.tabBar.currentTitle === null) {
450
- // The mouse cursor is moved close to the left border
451
- this.leftPanelHandler.expand();
452
- this.leftPanelHandler.state.pendingUpdate.then(() => this.dispatchMouseMove());
453
- state.leftExpanded = true;
454
- } else if (!expLeft && state.leftExpanded) {
455
- // The mouse cursor is moved away from the left border
456
- this.leftPanelHandler.collapse();
457
- state.leftExpanded = false;
458
- }
459
- // eslint-disable-next-line no-null/no-null
460
- if (expRight && !state.rightExpanded && this.rightPanelHandler.tabBar.currentTitle === null) {
461
- // The mouse cursor is moved close to the right border
462
- this.rightPanelHandler.expand();
463
- this.rightPanelHandler.state.pendingUpdate.then(() => this.dispatchMouseMove());
464
- state.rightExpanded = true;
465
- } else if (!expRight && state.rightExpanded) {
466
- // The mouse cursor is moved away from the right border
467
- this.rightPanelHandler.collapse();
468
- state.rightExpanded = false;
469
- }
470
- if (expBottom && !state.bottomExpanded && this.bottomPanel.isHidden) {
471
- // The mouse cursor is moved close to the bottom border
472
- this.expandBottomPanel();
473
- this.bottomPanelState.pendingUpdate.then(() => this.dispatchMouseMove());
474
- state.bottomExpanded = true;
475
- } else if (!expBottom && state.bottomExpanded) {
476
- // The mouse cursor is moved away from the bottom border
477
- this.collapseBottomPanel();
478
- state.bottomExpanded = false;
479
- }
480
- }
481
- }
482
-
483
- /**
484
- * This method is called after a side panel has been expanded while dragging a widget. It fires
485
- * a `mousemove` event so that the drag overlay markers are updated correctly in all dock panels.
486
- */
487
- private dispatchMouseMove(): void {
488
- if (this.dragState && this.dragState.lastDragOver) {
489
- const { clientX, clientY } = this.dragState.lastDragOver;
490
- const event = document.createEvent('MouseEvent');
491
- event.initMouseEvent('mousemove', true, true, window, 0, 0, 0,
492
- // eslint-disable-next-line no-null/no-null
493
- clientX, clientY, false, false, false, false, 0, null);
494
- document.dispatchEvent(event);
495
- }
496
- }
497
-
498
- protected onDrop(event: IDragEvent): void {
499
- const state = this.dragState;
500
- if (state) {
501
- if (state.leaveTimeout) {
502
- window.clearTimeout(state.leaveTimeout);
503
- }
504
- this.dragState = undefined;
505
- window.requestAnimationFrame(() => {
506
- // Clean up the side panel state in the next frame
507
- if (this.leftPanelHandler.dockPanel.isEmpty) {
508
- this.leftPanelHandler.collapse();
509
- }
510
- if (this.rightPanelHandler.dockPanel.isEmpty) {
511
- this.rightPanelHandler.collapse();
512
- }
513
- if (this.bottomPanel.isEmpty) {
514
- this.collapseBottomPanel();
515
- }
516
- });
517
- }
518
- }
519
-
520
- protected onDragLeave(event: IDragEvent): void {
521
- const state = this.dragState;
522
- if (state) {
523
- state.lastDragOver = undefined;
524
- if (state.leaveTimeout) {
525
- window.clearTimeout(state.leaveTimeout);
526
- }
527
- state.leaveTimeout = window.setTimeout(() => {
528
- this.dragState = undefined;
529
- if (state.leftExpanded || this.leftPanelHandler.dockPanel.isEmpty) {
530
- this.leftPanelHandler.collapse();
531
- }
532
- if (state.rightExpanded || this.rightPanelHandler.dockPanel.isEmpty) {
533
- this.rightPanelHandler.collapse();
534
- }
535
- if (state.bottomExpanded || this.bottomPanel.isEmpty) {
536
- this.collapseBottomPanel();
537
- }
538
- }, 100);
539
- }
540
- }
541
-
542
- /**
543
- * Create the dock panel in the main shell area.
544
- */
545
- protected createMainPanel(): TheiaDockPanel {
546
- const renderer = this.dockPanelRendererFactory();
547
- renderer.tabBarClasses.push(MAIN_BOTTOM_AREA_CLASS);
548
- renderer.tabBarClasses.push(MAIN_AREA_CLASS);
549
- this._mainPanelRenderer = renderer;
550
- const dockPanel = this.dockPanelFactory({
551
- mode: 'multiple-document',
552
- renderer,
553
- spacing: 0
554
- });
555
- dockPanel.id = MAIN_AREA_ID;
556
- dockPanel.widgetAdded.connect((_, widget) => this.fireDidAddWidget(widget));
557
- dockPanel.widgetRemoved.connect((_, widget) => this.fireDidRemoveWidget(widget));
558
-
559
- const openUri = async (fileUri: URI) => {
560
- try {
561
- const opener = await this.openerService.getOpener(fileUri);
562
- opener.open(fileUri);
563
- } catch (e) {
564
- console.info(`no opener found for '${fileUri}'`);
565
- }
566
- };
567
-
568
- dockPanel.node.addEventListener('drop', event => {
569
- if (event.dataTransfer) {
570
- const uris = this.additionalDraggedUris || ApplicationShell.getDraggedEditorUris(event.dataTransfer);
571
- if (uris.length > 0) {
572
- uris.forEach(openUri);
573
- } else if (event.dataTransfer.files?.length > 0) {
574
- // the files were dragged from the outside the workspace
575
- Array.from(event.dataTransfer.files).forEach(file => {
576
- if (file.path) {
577
- const fileUri = URI.fromComponents({
578
- scheme: 'file',
579
- path: file.path,
580
- authority: '',
581
- query: '',
582
- fragment: ''
583
- });
584
- openUri(fileUri);
585
- }
586
- });
587
- }
588
- }
589
- });
590
-
591
- dockPanel.node.addEventListener('dblclick', event => {
592
- const el = event.target as Element;
593
- if (el.id === MAIN_AREA_ID || el.classList.contains('p-TabBar-content')) {
594
- this.onDidDoubleClickMainAreaEmitter.fire();
595
- }
596
- });
597
-
598
- const handler = (e: DragEvent) => {
599
- if (e.dataTransfer) {
600
- e.dataTransfer.dropEffect = 'link';
601
- e.preventDefault();
602
- e.stopPropagation();
603
- }
604
- };
605
- dockPanel.node.addEventListener('dragover', handler);
606
- dockPanel.node.addEventListener('dragenter', handler);
607
-
608
- return dockPanel;
609
- }
610
-
611
- addAdditionalDraggedEditorUris(uris: URI[]): void {
612
- this.additionalDraggedUris = uris;
613
- }
614
-
615
- clearAdditionalDraggedEditorUris(): void {
616
- this.additionalDraggedUris = undefined;
617
- }
618
-
619
- protected static getDraggedEditorUris(dataTransfer: DataTransfer): URI[] {
620
- const data = dataTransfer.getData('theia-editor-dnd');
621
- return data ? data.split('\n').map(entry => new URI(entry)) : [];
622
- }
623
-
624
- static setDraggedEditorUris(dataTransfer: DataTransfer, uris: URI[]): void {
625
- dataTransfer.setData('theia-editor-dnd', uris.map(uri => uri.toString()).join('\n'));
626
- }
627
-
628
- /**
629
- * Create the dock panel in the bottom shell area.
630
- */
631
- protected createBottomPanel(): TheiaDockPanel {
632
- const renderer = this.dockPanelRendererFactory();
633
- renderer.tabBarClasses.push(MAIN_BOTTOM_AREA_CLASS);
634
- renderer.tabBarClasses.push(BOTTOM_AREA_CLASS);
635
- const dockPanel = this.dockPanelFactory({
636
- mode: 'multiple-document',
637
- renderer,
638
- spacing: 0
639
- });
640
- dockPanel.id = BOTTOM_AREA_ID;
641
- dockPanel.widgetAdded.connect((sender, widget) => {
642
- this.refreshBottomPanelToggleButton();
643
- });
644
- dockPanel.widgetRemoved.connect((sender, widget) => {
645
- if (sender.isEmpty) {
646
- this.collapseBottomPanel();
647
- }
648
- this.refreshBottomPanelToggleButton();
649
- }, this);
650
- dockPanel.node.addEventListener('p-dragenter', event => {
651
- // Make sure that the main panel hides its overlay when the bottom panel is expanded
652
- this.mainPanel.overlay.hide(0);
653
- });
654
- dockPanel.hide();
655
- dockPanel.widgetAdded.connect((_, widget) => this.fireDidAddWidget(widget));
656
- dockPanel.widgetRemoved.connect((_, widget) => this.fireDidRemoveWidget(widget));
657
- return dockPanel;
658
- }
659
-
660
- /**
661
- * Create the top panel, which is used to hold the main menu.
662
- */
663
- protected createTopPanel(): Panel {
664
- const topPanel = new Panel();
665
- topPanel.id = 'theia-top-panel';
666
- topPanel.hide();
667
- return topPanel;
668
- }
669
-
670
- /**
671
- * Create a box layout to assemble the application shell layout.
672
- */
673
- protected createBoxLayout(widgets: Widget[], stretch?: number[], options?: BoxPanel.IOptions): BoxLayout {
674
- const boxLayout = new BoxLayout(options);
675
- for (let i = 0; i < widgets.length; i++) {
676
- if (stretch !== undefined && i < stretch.length) {
677
- BoxPanel.setStretch(widgets[i], stretch[i]);
678
- }
679
- boxLayout.addWidget(widgets[i]);
680
- }
681
- return boxLayout;
682
- }
683
-
684
- /**
685
- * Create a split layout to assemble the application shell layout.
686
- */
687
- protected createSplitLayout(widgets: Widget[], stretch?: number[], options?: Partial<SplitLayout.IOptions>): SplitLayout {
688
- let optParam: SplitLayout.IOptions = { renderer: SplitPanel.defaultRenderer, };
689
- if (options) {
690
- optParam = { ...optParam, ...options };
691
- }
692
- const splitLayout = new SplitLayout(optParam);
693
- for (let i = 0; i < widgets.length; i++) {
694
- if (stretch !== undefined && i < stretch.length) {
695
- SplitPanel.setStretch(widgets[i], stretch[i]);
696
- }
697
- splitLayout.addWidget(widgets[i]);
698
- }
699
- return splitLayout;
700
- }
701
-
702
- /**
703
- * Assemble the application shell layout. Override this method in order to change the arrangement
704
- * of the main area and the side panels.
705
- */
706
- protected createLayout(): Layout {
707
- const bottomSplitLayout = this.createSplitLayout(
708
- [this.mainPanel, this.bottomPanel],
709
- [1, 0],
710
- { orientation: 'vertical', spacing: 0 }
711
- );
712
- const panelForBottomArea = new SplitPanel({ layout: bottomSplitLayout });
713
- panelForBottomArea.id = 'theia-bottom-split-panel';
714
-
715
- const leftRightSplitLayout = this.createSplitLayout(
716
- [this.leftPanelHandler.container, panelForBottomArea, this.rightPanelHandler.container],
717
- [0, 1, 0],
718
- { orientation: 'horizontal', spacing: 0 }
719
- );
720
- const panelForSideAreas = new SplitPanel({ layout: leftRightSplitLayout });
721
- panelForSideAreas.id = 'theia-left-right-split-panel';
722
-
723
- return this.createBoxLayout(
724
- [this.topPanel, panelForSideAreas, this.statusBar],
725
- [0, 1, 0],
726
- { direction: 'top-to-bottom', spacing: 0 }
727
- );
728
- }
729
-
730
- /**
731
- * Create an object that describes the current shell layout. This object may contain references
732
- * to widgets; these need to be transformed before the layout can be serialized.
733
- */
734
- getLayoutData(): ApplicationShell.LayoutData {
735
- return {
736
- version: applicationShellLayoutVersion,
737
- mainPanel: this.mainPanel.saveLayout(),
738
- mainPanelPinned: this.getPinnedMainWidgets(),
739
- bottomPanel: {
740
- config: this.bottomPanel.saveLayout(),
741
- pinned: this.getPinnedBottomWidgets(),
742
- size: this.bottomPanel.isVisible ? this.getBottomPanelSize() : this.bottomPanelState.lastPanelSize,
743
- expanded: this.isExpanded('bottom')
744
- },
745
- leftPanel: this.leftPanelHandler.getLayoutData(),
746
- rightPanel: this.rightPanelHandler.getLayoutData(),
747
- activeWidgetId: this.activeWidget ? this.activeWidget.id : undefined
748
- };
749
- }
750
-
751
- // Get an array corresponding to main panel widgets' pinned state.
752
- getPinnedMainWidgets(): boolean[] {
753
- const pinned: boolean[] = [];
754
-
755
- toArray(this.mainPanel.widgets()).forEach((a, i) => {
756
- pinned[i] = a.title.className.includes(PINNED_CLASS);
757
- });
758
-
759
- return pinned;
760
- }
761
-
762
- // Get an array corresponding to bottom panel widgets' pinned state.
763
- getPinnedBottomWidgets(): boolean[] {
764
- const pinned: boolean[] = [];
765
-
766
- toArray(this.bottomPanel.widgets()).forEach((a, i) => {
767
- pinned[i] = a.title.className.includes(PINNED_CLASS);
768
- });
769
-
770
- return pinned;
771
- }
772
-
773
- /**
774
- * Compute the current height of the bottom panel. This implementation assumes that the container
775
- * of the bottom panel is a `SplitPanel`.
776
- */
777
- protected getBottomPanelSize(): number | undefined {
778
- const parent = this.bottomPanel.parent;
779
- if (parent instanceof SplitPanel && parent.isVisible) {
780
- const index = parent.widgets.indexOf(this.bottomPanel) - 1;
781
- if (index >= 0) {
782
- const handle = parent.handles[index];
783
- if (!handle.classList.contains('p-mod-hidden')) {
784
- const parentHeight = parent.node.clientHeight;
785
- return parentHeight - handle.offsetTop;
786
- }
787
- }
788
- }
789
- }
790
-
791
- /**
792
- * Determine the default size to apply when the bottom panel is expanded for the first time.
793
- */
794
- protected getDefaultBottomPanelSize(): number | undefined {
795
- const parent = this.bottomPanel.parent;
796
- if (parent && parent.isVisible) {
797
- return parent.node.clientHeight * this.options.bottomPanel.initialSizeRatio;
798
- }
799
- }
800
-
801
- /**
802
- * Apply a shell layout that has been previously created with `getLayoutData`.
803
- */
804
- async setLayoutData(layoutData: ApplicationShell.LayoutData): Promise<void> {
805
- const { mainPanel, mainPanelPinned, bottomPanel, leftPanel, rightPanel, activeWidgetId } = layoutData;
806
- if (leftPanel) {
807
- this.leftPanelHandler.setLayoutData(leftPanel);
808
- this.registerWithFocusTracker(leftPanel);
809
- }
810
- if (rightPanel) {
811
- this.rightPanelHandler.setLayoutData(rightPanel);
812
- this.registerWithFocusTracker(rightPanel);
813
- }
814
- // Proceed with the bottom panel once the side panels are set up
815
- await Promise.all([this.leftPanelHandler.state.pendingUpdate, this.rightPanelHandler.state.pendingUpdate]);
816
- if (bottomPanel) {
817
- if (bottomPanel.config) {
818
- this.bottomPanel.restoreLayout(bottomPanel.config);
819
- this.registerWithFocusTracker(bottomPanel.config.main);
820
- }
821
- if (bottomPanel.size) {
822
- this.bottomPanelState.lastPanelSize = bottomPanel.size;
823
- }
824
- if (bottomPanel.expanded) {
825
- this.expandBottomPanel();
826
- } else {
827
- this.collapseBottomPanel();
828
- }
829
- const widgets = toArray(this.bottomPanel.widgets());
830
- this.bottomPanel.markActiveTabBar(widgets[0]?.title);
831
- if (bottomPanel.pinned && bottomPanel.pinned.length === widgets.length) {
832
- widgets.forEach((a, i) => {
833
- if (bottomPanel.pinned![i]) {
834
- a.title.className += ` ${PINNED_CLASS}`;
835
- a.title.closable = false;
836
- }
837
- });
838
- }
839
- this.refreshBottomPanelToggleButton();
840
- }
841
- // Proceed with the main panel once all others are set up
842
- await this.bottomPanelState.pendingUpdate;
843
- if (mainPanel) {
844
- this.mainPanel.restoreLayout(mainPanel);
845
- this.registerWithFocusTracker(mainPanel.main);
846
- const widgets = toArray(this.mainPanel.widgets());
847
- // We don't store information about the last active tabbar
848
- // So we simply mark the first as being active
849
- this.mainPanel.markActiveTabBar(widgets[0]?.title);
850
- if (mainPanelPinned && mainPanelPinned.length === widgets.length) {
851
- widgets.forEach((a, i) => {
852
- if (mainPanelPinned[i]) {
853
- a.title.className += ` ${PINNED_CLASS}`;
854
- a.title.closable = false;
855
- }
856
- });
857
- }
858
- }
859
- if (activeWidgetId) {
860
- this.activateWidget(activeWidgetId);
861
- }
862
- }
863
-
864
- /**
865
- * Modify the height of the bottom panel. This implementation assumes that the container of the
866
- * bottom panel is a `SplitPanel`.
867
- */
868
- protected setBottomPanelSize(size: number): Promise<void> {
869
- const enableAnimation = this.applicationStateService.state === 'ready';
870
- const options: SplitPositionOptions = {
871
- side: 'bottom',
872
- duration: enableAnimation ? this.options.bottomPanel.expandDuration : 0,
873
- referenceWidget: this.bottomPanel
874
- };
875
- const promise = this.splitPositionHandler.setSidePanelSize(this.bottomPanel, size, options);
876
- const result = new Promise<void>(resolve => {
877
- // Resolve the resulting promise in any case, regardless of whether resizing was successful
878
- promise.then(() => resolve(), () => resolve());
879
- });
880
- this.bottomPanelState.pendingUpdate = this.bottomPanelState.pendingUpdate.then(() => result);
881
- return result;
882
- }
883
-
884
- /**
885
- * A promise that is resolved when all currently pending updates are done.
886
- */
887
- get pendingUpdates(): Promise<void> {
888
- return Promise.all([
889
- this.bottomPanelState.pendingUpdate,
890
- this.leftPanelHandler.state.pendingUpdate,
891
- this.rightPanelHandler.state.pendingUpdate
892
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
893
- ]) as Promise<any>;
894
- }
895
-
896
- /**
897
- * Track all widgets that are referenced by the given layout data.
898
- */
899
- protected registerWithFocusTracker(data: DockLayout.ITabAreaConfig | DockLayout.ISplitAreaConfig | SidePanel.LayoutData | null): void {
900
- if (data) {
901
- if (data.type === 'tab-area') {
902
- for (const widget of data.widgets) {
903
- if (widget) {
904
- this.track(widget);
905
- }
906
- }
907
- } else if (data.type === 'split-area') {
908
- for (const child of data.children) {
909
- this.registerWithFocusTracker(child);
910
- }
911
- } else if (data.type === 'sidepanel' && data.items) {
912
- for (const item of data.items) {
913
- if (item.widget) {
914
- this.track(item.widget);
915
- }
916
- }
917
- }
918
- }
919
- }
920
-
921
- /**
922
- * Add a widget to the application shell. The given widget must have a unique `id` property,
923
- * which will be used as the DOM id.
924
- *
925
- * Widgets are removed from the shell by calling their `close` or `dispose` methods.
926
- *
927
- * Widgets added to the top area are not tracked regarding the _current_ and _active_ states.
928
- */
929
- async addWidget(widget: Widget, options?: Readonly<ApplicationShell.WidgetOptions>): Promise<void> {
930
- if (!widget.id) {
931
- console.error('Widgets added to the application shell must have a unique id property.');
932
- return;
933
- }
934
- const { area, addOptions } = this.getInsertionOptions(options);
935
- const sidePanelOptions: SidePanel.WidgetOptions = { rank: options?.rank };
936
- switch (area) {
937
- case 'main':
938
- this.mainPanel.addWidget(widget, addOptions);
939
- break;
940
- case 'top':
941
- this.topPanel.addWidget(widget);
942
- break;
943
- case 'bottom':
944
- this.bottomPanel.addWidget(widget, addOptions);
945
- break;
946
- case 'left':
947
- this.leftPanelHandler.addWidget(widget, sidePanelOptions);
948
- break;
949
- case 'right':
950
- this.rightPanelHandler.addWidget(widget, sidePanelOptions);
951
- break;
952
- case 'secondaryWindow':
953
- /** At the moment, widgets are only moved to this area (i.e. a secondary window) by moving them from one of the other areas. */
954
- throw new Error('Widgets cannot be added directly to a secondary window');
955
- default:
956
- throw new Error('Unexpected area: ' + options?.area);
957
- }
958
- if (area !== 'top') {
959
- this.track(widget);
960
- }
961
- }
962
-
963
- getInsertionOptions(options?: Readonly<ApplicationShell.WidgetOptions>): { area: string; addOptions: DockLayout.IAddOptions; } {
964
- let ref: Widget | undefined = options?.ref;
965
- let area: ApplicationShell.Area = options?.area || 'main';
966
- if (!ref && (area === 'main' || area === 'bottom')) {
967
- const tabBar = this.getTabBarFor(area);
968
- ref = tabBar && tabBar.currentTitle && tabBar.currentTitle.owner || undefined;
969
- }
970
- // make sure that ref belongs to area
971
- area = ref && this.getAreaFor(ref) || area;
972
- const addOptions: DockPanel.IAddOptions = {};
973
- if (ApplicationShell.isOpenToSideMode(options?.mode)) {
974
- const areaPanel = area === 'main' ? this.mainPanel : area === 'bottom' ? this.bottomPanel : undefined;
975
- const sideRef = areaPanel && ref && (options?.mode === 'open-to-left' ?
976
- areaPanel.previousTabBarWidget(ref) :
977
- areaPanel.nextTabBarWidget(ref));
978
- if (sideRef) {
979
- addOptions.ref = sideRef;
980
- } else {
981
- addOptions.ref = ref;
982
- addOptions.mode = options?.mode === 'open-to-left' ? 'split-left' : 'split-right';
983
- }
984
- } else {
985
- addOptions.ref = ref;
986
- addOptions.mode = options?.mode;
987
- }
988
- return { area, addOptions };
989
- }
990
-
991
- /**
992
- * The widgets contained in the given shell area.
993
- */
994
- getWidgets(area: ApplicationShell.Area): Widget[] {
995
- switch (area) {
996
- case 'main':
997
- return toArray(this.mainPanel.widgets());
998
- case 'top':
999
- return toArray(this.topPanel.widgets);
1000
- case 'bottom':
1001
- return toArray(this.bottomPanel.widgets());
1002
- case 'left':
1003
- return toArray(this.leftPanelHandler.dockPanel.widgets());
1004
- case 'right':
1005
- return toArray(this.rightPanelHandler.dockPanel.widgets());
1006
- case 'secondaryWindow':
1007
- return toArray(this.secondaryWindowHandler.widgets);
1008
- default:
1009
- throw new Error('Illegal argument: ' + area);
1010
- }
1011
- }
1012
-
1013
- /**
1014
- * Find the widget that contains the given HTML element. The returned widget may be one
1015
- * that is managed by the application shell, or one that is embedded in another widget and
1016
- * not directly managed by the shell, or a tab bar.
1017
- */
1018
- findWidgetForElement(element: HTMLElement): Widget | undefined {
1019
- let widgetNode: HTMLElement | null = element;
1020
- while (widgetNode && !widgetNode.classList.contains('p-Widget')) {
1021
- widgetNode = widgetNode.parentElement;
1022
- }
1023
- if (widgetNode) {
1024
- return this.findWidgetForNode(widgetNode, this);
1025
- }
1026
- return undefined;
1027
- }
1028
-
1029
- private findWidgetForNode(widgetNode: HTMLElement, widget: Widget): Widget | undefined {
1030
- if (widget.node === widgetNode) {
1031
- return widget;
1032
- }
1033
- let result: Widget | undefined;
1034
- each(widget.children(), child => {
1035
- result = this.findWidgetForNode(widgetNode, child);
1036
- return !result;
1037
- });
1038
- return result;
1039
- }
1040
-
1041
- /**
1042
- * Finds the title widget from the tab-bar.
1043
- * @param tabBar used for providing an array of titles.
1044
- * @returns the selected title widget, else returns the currentTitle or undefined.
1045
- */
1046
- findTitle(tabBar: TabBar<Widget>, event?: Event): Title<Widget> | undefined {
1047
- if (event?.target instanceof HTMLElement) {
1048
- const tabNode = event.target;
1049
-
1050
- const titleIndex = Array.from(tabBar.contentNode.getElementsByClassName('p-TabBar-tab'))
1051
- .findIndex(node => node.contains(tabNode));
1052
-
1053
- if (titleIndex !== -1) {
1054
- return tabBar.titles[titleIndex];
1055
- }
1056
-
1057
- }
1058
- return tabBar.currentTitle || undefined;
1059
- }
1060
-
1061
- /**
1062
- * Finds the tab-bar widget.
1063
- * @returns the selected tab-bar, else returns the currentTabBar.
1064
- */
1065
- findTabBar(event?: Event): TabBar<Widget> | undefined {
1066
- if (event?.target instanceof HTMLElement) {
1067
- const tabBar = this.findWidgetForElement(event.target);
1068
- if (tabBar instanceof TabBar) {
1069
- return tabBar;
1070
- }
1071
- }
1072
- return this.currentTabBar;
1073
- }
1074
-
1075
- /**
1076
- * @returns the widget whose title has been targeted by a DOM event on a tabbar, or undefined if none can be found.
1077
- */
1078
- findTargetedWidget(event?: Event): Widget | undefined {
1079
- if (event) {
1080
- const tab = this.findTabBar(event);
1081
- const title = tab && this.findTitle(tab, event);
1082
- return title && title.owner;
1083
- }
1084
- }
1085
-
1086
- /**
1087
- * The current widget in the application shell. The current widget is the last widget that
1088
- * was active and not yet closed. See the remarks to `activeWidget` on what _active_ means.
1089
- */
1090
- get currentWidget(): Widget | undefined {
1091
- return this.tracker.currentWidget || undefined;
1092
- }
1093
-
1094
- /**
1095
- * The active widget in the application shell. The active widget is the one that has focus
1096
- * (either the widget itself or any of its contents).
1097
- *
1098
- * _Note:_ Focus is taken by a widget through the `onActivateRequest` method. It is up to the
1099
- * widget implementation which DOM element will get the focus. The default implementation
1100
- * does not take any focus; in that case the widget is never returned by this property.
1101
- */
1102
- get activeWidget(): Widget | undefined {
1103
- return this.tracker.activeWidget || undefined;
1104
- }
1105
-
1106
- /**
1107
- * Returns the last active widget in the given shell area.
1108
- */
1109
- getCurrentWidget(area: ApplicationShell.Area): Widget | undefined {
1110
- let title: Title<Widget> | null | undefined;
1111
- switch (area) {
1112
- case 'main':
1113
- title = this.mainPanel.currentTitle;
1114
- break;
1115
- case 'bottom':
1116
- title = this.bottomPanel.currentTitle;
1117
- break;
1118
- case 'left':
1119
- title = this.leftPanelHandler.tabBar.currentTitle;
1120
- break;
1121
- case 'right':
1122
- title = this.rightPanelHandler.tabBar.currentTitle;
1123
- break;
1124
- case 'secondaryWindow':
1125
- // The current widget in a secondary window is not tracked.
1126
- return undefined;
1127
- default:
1128
- throw new Error('Illegal argument: ' + area);
1129
- }
1130
- return title ? title.owner : undefined;
1131
- }
1132
-
1133
- /**
1134
- * Handle a change to the current widget.
1135
- */
1136
- private onCurrentChanged(sender: FocusTracker<Widget>, args: FocusTracker.IChangedArgs<Widget>): void {
1137
- this.onDidChangeCurrentWidgetEmitter.fire(args);
1138
- }
1139
-
1140
- protected readonly toDisposeOnActiveChanged = new DisposableCollection();
1141
-
1142
- /**
1143
- * Handle a change to the active widget.
1144
- */
1145
- private onActiveChanged(sender: FocusTracker<Widget>, args: FocusTracker.IChangedArgs<Widget>): void {
1146
- this.toDisposeOnActiveChanged.dispose();
1147
- const { newValue, oldValue } = args;
1148
- if (oldValue) {
1149
- let w: Widget | null = oldValue;
1150
- while (w) {
1151
- // Remove the mark of the previously active widget
1152
- w.title.className = w.title.className.replace(' theia-mod-active', '');
1153
- w = w.parent;
1154
- }
1155
- // Reset the z-index to the default
1156
- // eslint-disable-next-line no-null/no-null
1157
- this.setZIndex(oldValue.node, null);
1158
- }
1159
- if (newValue) {
1160
- let w: Widget | null = newValue;
1161
- while (w) {
1162
- // Mark the tab of the active widget
1163
- w.title.className += ' theia-mod-active';
1164
- w = w.parent;
1165
- }
1166
- // Reveal the title of the active widget in its tab bar
1167
- const tabBar = this.getTabBarFor(newValue);
1168
- if (tabBar instanceof ScrollableTabBar) {
1169
- const index = tabBar.titles.indexOf(newValue.title);
1170
- if (index >= 0) {
1171
- tabBar.revealTab(index);
1172
- }
1173
- }
1174
- const widget = this.toTrackedStack(newValue.id).pop();
1175
- const panel = this.findPanel(widget);
1176
- if (panel) {
1177
- // if widget was undefined, we wouldn't have gotten a panel back before
1178
- panel.markAsCurrent(widget!.title);
1179
- }
1180
- // Add checks to ensure that the 'sash' for left panel is displayed correctly
1181
- if (newValue.node.className === 'p-Widget theia-view-container p-DockPanel-widget') {
1182
- // Set the z-index so elements with `position: fixed` contained in the active widget are displayed correctly
1183
- this.setZIndex(newValue.node, '1');
1184
- }
1185
-
1186
- // activate another widget if an active widget will be closed
1187
- const onCloseRequest = newValue['onCloseRequest'];
1188
- newValue['onCloseRequest'] = msg => {
1189
- const currentTabBar = this.currentTabBar;
1190
- if (currentTabBar) {
1191
- const recentlyUsedInTabBar = currentTabBar['_previousTitle'] as TabBar<Widget>['currentTitle'];
1192
- if (recentlyUsedInTabBar && recentlyUsedInTabBar.owner !== newValue) {
1193
- currentTabBar.currentIndex = ArrayExt.firstIndexOf(currentTabBar.titles, recentlyUsedInTabBar);
1194
- if (currentTabBar.currentTitle) {
1195
- this.activateWidget(currentTabBar.currentTitle.owner.id);
1196
- }
1197
- } else if (!this.activateNextTabInTabBar(currentTabBar)) {
1198
- if (!this.activatePreviousTabBar(currentTabBar)) {
1199
- this.activateNextTabBar(currentTabBar);
1200
- }
1201
- }
1202
- }
1203
- newValue['onCloseRequest'] = onCloseRequest;
1204
- newValue['onCloseRequest'](msg);
1205
- };
1206
- this.toDisposeOnActiveChanged.push(Disposable.create(() => newValue['onCloseRequest'] = onCloseRequest));
1207
- if (PreviewableWidget.is(newValue)) {
1208
- newValue.loaded = true;
1209
- }
1210
- }
1211
- this.onDidChangeActiveWidgetEmitter.fire(args);
1212
- }
1213
-
1214
- /**
1215
- * Set the z-index of the given element and its ancestors to the value `z`.
1216
- */
1217
- private setZIndex(element: HTMLElement, z: string | null): void {
1218
- element.style.zIndex = z || '';
1219
- const parent = element.parentElement;
1220
- if (parent && parent !== this.node) {
1221
- this.setZIndex(parent, z);
1222
- }
1223
- }
1224
-
1225
- /**
1226
- * Track the given widget so it is considered in the `current` and `active` state of the shell.
1227
- */
1228
- protected track(widget: Widget): void {
1229
- if (this.tracker.widgets.indexOf(widget) !== -1) {
1230
- return;
1231
- }
1232
- this.tracker.add(widget);
1233
- this.checkActivation(widget);
1234
- Saveable.apply(
1235
- widget,
1236
- () => this.widgets.filter((maybeSaveable): maybeSaveable is Widget & SaveableSource => !!Saveable.get(maybeSaveable)),
1237
- async (toSave, options) => {
1238
- await this.saveResourceService.save(toSave, options);
1239
- },
1240
- );
1241
- if (ApplicationShell.TrackableWidgetProvider.is(widget)) {
1242
- for (const toTrack of widget.getTrackableWidgets()) {
1243
- this.track(toTrack);
1244
- }
1245
- if (widget.onDidChangeTrackableWidgets) {
1246
- widget.onDidChangeTrackableWidgets(widgets => widgets.forEach(w => this.track(w)));
1247
- }
1248
- }
1249
- }
1250
-
1251
- /**
1252
- * @returns an array of Widgets, all of which are tracked by the focus tracker
1253
- * The first member of the array is the widget whose id is passed in, and the other widgets
1254
- * are its tracked parents in ascending order
1255
- */
1256
- protected toTrackedStack(id: string): Widget[] {
1257
- const tracked = new Map<string, Widget>(this.tracker.widgets.map(w => [w.id, w] as [string, Widget]));
1258
- let current = tracked.get(id);
1259
- const stack: Widget[] = [];
1260
- while (current) {
1261
- if (tracked.has(current.id)) {
1262
- stack.push(current);
1263
- }
1264
- current = current.parent || undefined;
1265
- }
1266
- return stack;
1267
- }
1268
-
1269
- /**
1270
- * Activate a widget in the application shell. This makes the widget visible and usually
1271
- * also assigns focus to it.
1272
- *
1273
- * _Note:_ Focus is taken by a widget through the `onActivateRequest` method. It is up to the
1274
- * widget implementation which DOM element will get the focus. The default implementation
1275
- * does not take any focus.
1276
- *
1277
- * @returns the activated widget if it was found
1278
- */
1279
- async activateWidget(id: string): Promise<Widget | undefined> {
1280
- const stack = this.toTrackedStack(id);
1281
- let current = stack.pop();
1282
- if (current && !this.doActivateWidget(current.id)) {
1283
- return undefined;
1284
- }
1285
- while (current && stack.length) {
1286
- const child = stack.pop()!;
1287
- if (ApplicationShell.TrackableWidgetProvider.is(current) && current.activateWidget) {
1288
- current = current.activateWidget(child.id);
1289
- } else {
1290
- child.activate();
1291
- current = child;
1292
- }
1293
- }
1294
- if (!current) {
1295
- return undefined;
1296
- }
1297
- return Promise.all([
1298
- this.waitForActivation(current.id),
1299
- waitForRevealed(current),
1300
- this.pendingUpdates
1301
- ]).then(() => current, () => undefined);
1302
- }
1303
-
1304
- waitForActivation(id: string): Promise<void> {
1305
- if (this.activeWidget && this.activeWidget.id === id) {
1306
- return Promise.resolve();
1307
- }
1308
- const activation = new Deferred();
1309
- const success = this.onDidChangeActiveWidget(() => {
1310
- if (this.activeWidget && this.activeWidget.id === id) {
1311
- activation.resolve();
1312
- }
1313
- });
1314
- const failure = setTimeout(() => activation.reject(new Error(`Widget with id '${id}' failed to activate.`)), this.activationTimeout + 250);
1315
- return activation.promise.finally(() => {
1316
- success.dispose();
1317
- clearTimeout(failure);
1318
- });
1319
- }
1320
-
1321
- /**
1322
- * Activate top-level area widget.
1323
- */
1324
- protected doActivateWidget(id: string): Widget | undefined {
1325
- let widget = find(this.mainPanel.widgets(), w => w.id === id);
1326
- if (widget) {
1327
- this.mainPanel.activateWidget(widget);
1328
- }
1329
- if (!widget) {
1330
- widget = find(this.bottomPanel.widgets(), w => w.id === id);
1331
- if (widget) {
1332
- this.expandBottomPanel();
1333
- this.bottomPanel.activateWidget(widget);
1334
- }
1335
- }
1336
- if (!widget) {
1337
- widget = this.leftPanelHandler.activate(id);
1338
- }
1339
-
1340
- if (!widget) {
1341
- widget = this.rightPanelHandler.activate(id);
1342
- }
1343
- if (widget) {
1344
- this.windowService.focus();
1345
- return widget;
1346
- }
1347
- return this.secondaryWindowHandler.activateWidget(id);
1348
- }
1349
-
1350
- /**
1351
- * Focus is taken by a widget through the `onActivateRequest` method. It is up to the
1352
- * widget implementation which DOM element will get the focus. The default implementation
1353
- * of Widget does not take any focus. This method can help finding such problems by logging
1354
- * a warning in case a widget was explicitly activated, but did not trigger a change of the
1355
- * `activeWidget` property.
1356
- */
1357
- private checkActivation(widget: Widget): Widget {
1358
- const onActivateRequest = widget['onActivateRequest'].bind(widget);
1359
- widget['onActivateRequest'] = (msg: Message) => {
1360
- onActivateRequest(msg);
1361
- this.assertActivated(widget);
1362
- };
1363
- return widget;
1364
- }
1365
-
1366
- private readonly activationTimeout = 2000;
1367
- private readonly toDisposeOnActivationCheck = new DisposableCollection();
1368
- private assertActivated(widget: Widget): void {
1369
- this.toDisposeOnActivationCheck.dispose();
1370
-
1371
- const onDispose = () => this.toDisposeOnActivationCheck.dispose();
1372
- widget.disposed.connect(onDispose);
1373
- this.toDisposeOnActivationCheck.push(Disposable.create(() => widget.disposed.disconnect(onDispose)));
1374
-
1375
- let start = 0;
1376
- const step: FrameRequestCallback = timestamp => {
1377
- const activeElement = widget.node.ownerDocument.activeElement;
1378
- if (activeElement && widget.node.contains(activeElement)) {
1379
- return;
1380
- }
1381
- if (!start) {
1382
- start = timestamp;
1383
- }
1384
- const delta = timestamp - start;
1385
- if (delta < this.activationTimeout) {
1386
- request = window.requestAnimationFrame(step);
1387
- } else {
1388
- console.warn(`Widget was activated, but did not accept focus after ${this.activationTimeout}ms: ${widget.id}`);
1389
- }
1390
- };
1391
- let request = window.requestAnimationFrame(step);
1392
- this.toDisposeOnActivationCheck.push(Disposable.create(() => window.cancelAnimationFrame(request)));
1393
- }
1394
-
1395
- /**
1396
- * Reveal a widget in the application shell. This makes the widget visible,
1397
- * but does not activate it.
1398
- *
1399
- * @returns the revealed widget if it was found
1400
- */
1401
- async revealWidget(id: string): Promise<Widget | undefined> {
1402
- const stack = this.toTrackedStack(id);
1403
- let current = stack.pop();
1404
- if (current && !this.doRevealWidget(current.id)) {
1405
- return undefined;
1406
- }
1407
- while (current && stack.length) {
1408
- const child = stack.pop()!;
1409
- if (ApplicationShell.TrackableWidgetProvider.is(current) && current.revealWidget) {
1410
- current = current.revealWidget(child.id);
1411
- } else {
1412
- current = child;
1413
- }
1414
- }
1415
- if (!current) {
1416
- return undefined;
1417
- }
1418
- await Promise.all([
1419
- waitForRevealed(current),
1420
- this.pendingUpdates
1421
- ]);
1422
- return current;
1423
- }
1424
-
1425
- /**
1426
- * Reveal top-level area widget.
1427
- */
1428
- protected doRevealWidget(id: string): Widget | undefined {
1429
- let widget = find(this.mainPanel.widgets(), w => w.id === id);
1430
- if (!widget) {
1431
- widget = find(this.bottomPanel.widgets(), w => w.id === id);
1432
- if (widget) {
1433
- this.expandBottomPanel();
1434
- }
1435
- }
1436
- if (widget) {
1437
- const tabBar = this.getTabBarFor(widget);
1438
- if (tabBar) {
1439
- tabBar.currentTitle = widget.title;
1440
- }
1441
- }
1442
- if (!widget) {
1443
- widget = this.leftPanelHandler.expand(id);
1444
- }
1445
- if (!widget) {
1446
- widget = this.rightPanelHandler.expand(id);
1447
- }
1448
- if (widget) {
1449
- this.windowService.focus();
1450
- return widget;
1451
- } else {
1452
- return this.secondaryWindowHandler.revealWidget(id);
1453
- }
1454
- }
1455
-
1456
- /**
1457
- * Expand the named side panel area. This makes sure that the panel is visible, even if there
1458
- * are no widgets in it. If the panel is already visible, nothing happens. If the panel is currently
1459
- * collapsed (see `collapsePanel`) and it contains widgets, the widgets are revealed that were
1460
- * visible before it was collapsed.
1461
- */
1462
- expandPanel(area: ApplicationShell.Area): void {
1463
- switch (area) {
1464
- case 'bottom':
1465
- this.expandBottomPanel();
1466
- break;
1467
- case 'left':
1468
- this.leftPanelHandler.expand();
1469
- break;
1470
- case 'right':
1471
- this.rightPanelHandler.expand();
1472
- break;
1473
- default:
1474
- throw new Error('Area cannot be expanded: ' + area);
1475
- }
1476
- }
1477
-
1478
- /**
1479
- * Adjusts the size of the given area in the application shell.
1480
- *
1481
- * @param size the desired size of the panel in pixels.
1482
- * @param area the area to resize.
1483
- */
1484
- resize(size: number, area: ApplicationShell.Area): void {
1485
- switch (area) {
1486
- case 'bottom':
1487
- if (this.bottomPanel.isHidden) {
1488
- this.bottomPanelState.lastPanelSize = size;
1489
- } else {
1490
- this.setBottomPanelSize(size);
1491
- }
1492
- break;
1493
- case 'left':
1494
- this.leftPanelHandler.resize(size);
1495
- break;
1496
- case 'right':
1497
- this.rightPanelHandler.resize(size);
1498
- break;
1499
- default:
1500
- throw new Error('Area cannot be resized: ' + area);
1501
- }
1502
- }
1503
-
1504
- /**
1505
- * Expand the bottom panel. See `expandPanel` regarding the exact behavior.
1506
- */
1507
- protected expandBottomPanel(): void {
1508
- const bottomPanel = this.bottomPanel;
1509
- if (bottomPanel.isHidden) {
1510
- let relativeSizes: number[] | undefined;
1511
- const parent = bottomPanel.parent;
1512
- if (parent instanceof SplitPanel) {
1513
- relativeSizes = parent.relativeSizes();
1514
- }
1515
- bottomPanel.show();
1516
- if (relativeSizes && parent instanceof SplitPanel) {
1517
- // Make sure that the expansion animation starts at the smallest possible size
1518
- parent.setRelativeSizes(relativeSizes);
1519
- }
1520
-
1521
- let size: number | undefined;
1522
- if (bottomPanel.isEmpty) {
1523
- bottomPanel.node.style.minHeight = '0';
1524
- size = this.options.bottomPanel.emptySize;
1525
- } else if (this.bottomPanelState.lastPanelSize) {
1526
- size = this.bottomPanelState.lastPanelSize;
1527
- } else {
1528
- size = this.getDefaultBottomPanelSize();
1529
- }
1530
- if (size) {
1531
- this.bottomPanelState.expansion = SidePanel.ExpansionState.expanding;
1532
- this.setBottomPanelSize(size).then(() => {
1533
- if (this.bottomPanelState.expansion === SidePanel.ExpansionState.expanding) {
1534
- this.bottomPanelState.expansion = SidePanel.ExpansionState.expanded;
1535
- }
1536
- });
1537
- } else {
1538
- this.bottomPanelState.expansion = SidePanel.ExpansionState.expanded;
1539
- }
1540
- }
1541
- }
1542
-
1543
- /**
1544
- * Collapse the named side panel area. This makes sure that the panel is hidden,
1545
- * increasing the space that is available for other shell areas.
1546
- */
1547
- collapsePanel(area: ApplicationShell.Area): Promise<void> {
1548
- switch (area) {
1549
- case 'bottom':
1550
- return this.collapseBottomPanel();
1551
- case 'left':
1552
- return this.leftPanelHandler.collapse();
1553
- case 'right':
1554
- return this.rightPanelHandler.collapse();
1555
- default:
1556
- throw new Error('Area cannot be collapsed: ' + area);
1557
- }
1558
- }
1559
-
1560
- /**
1561
- * Collapse the bottom panel. All contained widgets are hidden, but not closed.
1562
- * They can be restored by calling `expandBottomPanel`.
1563
- */
1564
- protected collapseBottomPanel(): Promise<void> {
1565
- const bottomPanel = this.bottomPanel;
1566
- if (bottomPanel.isHidden) {
1567
- return Promise.resolve();
1568
- }
1569
- if (this.bottomPanelState.expansion === SidePanel.ExpansionState.expanded) {
1570
- const size = this.getBottomPanelSize();
1571
- if (size) {
1572
- this.bottomPanelState.lastPanelSize = size;
1573
- }
1574
- }
1575
- this.bottomPanelState.expansion = SidePanel.ExpansionState.collapsed;
1576
- bottomPanel.hide();
1577
- return animationFrame();
1578
- }
1579
-
1580
- /**
1581
- * Refresh the toggle button for the bottom panel. This implementation creates a status bar entry
1582
- * and refers to the command `core.toggle.bottom.panel`.
1583
- */
1584
- protected refreshBottomPanelToggleButton(): void {
1585
- if (this.bottomPanel.isEmpty) {
1586
- this.statusBar.removeElement(BOTTOM_PANEL_TOGGLE_ID);
1587
- } else {
1588
- const label = nls.localize('theia/core/common/collapseBottomPanel', 'Toggle Bottom Panel');
1589
- const element: StatusBarEntry = {
1590
- name: label,
1591
- text: '$(codicon-window)',
1592
- alignment: StatusBarAlignment.RIGHT,
1593
- tooltip: label,
1594
- command: 'core.toggle.bottom.panel',
1595
- accessibilityInformation: {
1596
- label: label,
1597
- role: 'button'
1598
- },
1599
- priority: -1000
1600
- };
1601
- this.statusBar.setElement(BOTTOM_PANEL_TOGGLE_ID, element);
1602
- }
1603
- }
1604
-
1605
- /**
1606
- * Check whether the named side panel area is expanded (returns `true`) or collapsed (returns `false`).
1607
- */
1608
- isExpanded(area: ApplicationShell.Area): boolean {
1609
- switch (area) {
1610
- case 'bottom':
1611
- return this.bottomPanelState.expansion === SidePanel.ExpansionState.expanded;
1612
- case 'left':
1613
- return this.leftPanelHandler.state.expansion === SidePanel.ExpansionState.expanded;
1614
- case 'right':
1615
- return this.rightPanelHandler.state.expansion === SidePanel.ExpansionState.expanded;
1616
- default:
1617
- return true;
1618
- }
1619
- }
1620
-
1621
- /**
1622
- * Close all tabs or a selection of tabs in a specific part of the application shell.
1623
- *
1624
- * @param tabBarOrArea
1625
- * Either the name of a shell area or a `TabBar` that is contained in such an area.
1626
- * @param filter
1627
- * If undefined, all tabs are closed; otherwise only those tabs that match the filter are closed.
1628
- */
1629
- async closeTabs(tabBarOrArea: TabBar<Widget> | ApplicationShell.Area,
1630
- filter?: (title: Title<Widget>, index: number) => boolean): Promise<void> {
1631
- const titles: Array<Title<Widget>> = this.getWidgetTitles(tabBarOrArea, filter);
1632
- if (titles.length) {
1633
- await this.closeMany(titles.map(title => title.owner));
1634
- }
1635
- }
1636
-
1637
- saveTabs(tabBarOrArea: TabBar<Widget> | ApplicationShell.Area,
1638
- filter?: (title: Title<Widget>, index: number) => boolean): void {
1639
-
1640
- const titles = this.getWidgetTitles(tabBarOrArea, filter);
1641
- for (let i = 0; i < titles.length; i++) {
1642
- const widget = titles[i].owner;
1643
- const saveable = Saveable.get(widget);
1644
- saveable?.save();
1645
- }
1646
- }
1647
-
1648
- /**
1649
- * Collects all widget titles for the given tab bar or area and optionally filters them.
1650
- *
1651
- * @param tabBarOrArea The tab bar or area to retrieve the widget titles for
1652
- * @param filter The filter to apply to the result
1653
- * @returns The filtered array of widget titles or an empty array
1654
- */
1655
- protected getWidgetTitles(tabBarOrArea: TabBar<Widget> | ApplicationShell.Area,
1656
- filter?: (title: Title<Widget>, index: number) => boolean): Title<Widget>[] {
1657
-
1658
- const titles: Title<Widget>[] = [];
1659
- if (tabBarOrArea === 'main') {
1660
- this.mainAreaTabBars.forEach(tabbar => titles.push(...toArray(tabbar.titles)));
1661
- } else if (tabBarOrArea === 'bottom') {
1662
- this.bottomAreaTabBars.forEach(tabbar => titles.push(...toArray(tabbar.titles)));
1663
- } else if (tabBarOrArea === 'secondaryWindow') {
1664
- titles.push(...this.secondaryWindowHandler.widgets.map(w => w.title));
1665
- } else if (typeof tabBarOrArea === 'string') {
1666
- const tabbar = this.getTabBarFor(tabBarOrArea);
1667
- if (tabbar) {
1668
- titles.push(...toArray(tabbar.titles));
1669
- }
1670
- } else if (tabBarOrArea) {
1671
- titles.push(...toArray(tabBarOrArea.titles));
1672
- }
1673
-
1674
- return filter ? titles.filter(filter) : titles;
1675
- }
1676
-
1677
- /**
1678
- * @param targets the widgets to be closed
1679
- * @return an array of all the widgets that were actually closed.
1680
- */
1681
- async closeMany(targets: Widget[], options?: ApplicationShell.CloseOptions): Promise<Widget[]> {
1682
- if (options?.save === false || await Saveable.confirmSaveBeforeClose(targets, this.widgets.filter(widget => !targets.includes(widget)))) {
1683
- return (await Promise.all(targets.map(target => this.closeWidget(target.id, options)))).filter((widget): widget is Widget => widget !== undefined);
1684
- }
1685
- return [];
1686
- }
1687
-
1688
- /**
1689
- * @returns the widget that was closed, if any, `undefined` otherwise.
1690
- *
1691
- * If your use case requires closing multiple widgets, use {@link ApplicationShell#closeMany} instead. That method handles closing saveable widgets more reliably.
1692
- */
1693
- async closeWidget(id: string, options?: ApplicationShell.CloseOptions): Promise<Widget | undefined> {
1694
- // TODO handle save for composite widgets, i.e. the preference widget has 2 editors
1695
- const stack = this.toTrackedStack(id);
1696
- const current = stack.pop();
1697
- if (!current) {
1698
- return undefined;
1699
- }
1700
- const saveableOptions = options && { shouldSave: () => options.save };
1701
- const pendingClose = SaveableWidget.is(current)
1702
- ? current.closeWithSaving(saveableOptions)
1703
- : (current.close(), waitForClosed(current));
1704
- await Promise.all([
1705
- pendingClose,
1706
- this.pendingUpdates
1707
- ]);
1708
- return stack[0] || current;
1709
- }
1710
-
1711
- /**
1712
- * The shell area name of the currently active tab, or undefined.
1713
- */
1714
- get currentTabArea(): ApplicationShell.Area | undefined {
1715
- const currentWidget = this.currentWidget;
1716
- if (currentWidget) {
1717
- return this.getAreaFor(currentWidget);
1718
- }
1719
- }
1720
-
1721
- /**
1722
- * Determine the name of the shell area where the given widget resides. The result is
1723
- * undefined if the widget does not reside directly in the shell.
1724
- */
1725
- getAreaFor(input: TabBar<Widget> | Widget): ApplicationShell.Area | undefined {
1726
- if (input instanceof TabBar) {
1727
- if (find(this.mainPanel.tabBars(), tb => tb === input)) {
1728
- return 'main';
1729
- }
1730
- if (find(this.bottomPanel.tabBars(), tb => tb === input)) {
1731
- return 'bottom';
1732
- }
1733
- if (this.leftPanelHandler.tabBar === input) {
1734
- return 'left';
1735
- }
1736
- if (this.rightPanelHandler.tabBar === input) {
1737
- return 'right';
1738
- }
1739
- }
1740
- const widget = this.toTrackedStack(input.id).pop();
1741
- if (!widget) {
1742
- return undefined;
1743
- }
1744
- const title = widget.title;
1745
- const mainPanelTabBar = this.mainPanel.findTabBar(title);
1746
- if (mainPanelTabBar) {
1747
- return 'main';
1748
- }
1749
- const bottomPanelTabBar = this.bottomPanel.findTabBar(title);
1750
- if (bottomPanelTabBar) {
1751
- return 'bottom';
1752
- }
1753
- if (ArrayExt.firstIndexOf(this.leftPanelHandler.tabBar.titles, title) > -1) {
1754
- return 'left';
1755
- }
1756
- if (ArrayExt.firstIndexOf(this.rightPanelHandler.tabBar.titles, title) > -1) {
1757
- return 'right';
1758
- }
1759
- if (this.secondaryWindowHandler.widgets.includes(widget)) {
1760
- return 'secondaryWindow';
1761
- }
1762
- return undefined;
1763
- }
1764
-
1765
- protected getAreaPanelFor(input: Widget): DockPanel | undefined {
1766
- const widget = this.toTrackedStack(input.id).pop();
1767
- if (!widget) {
1768
- return undefined;
1769
- }
1770
- return this.findPanel(widget);
1771
- }
1772
-
1773
- /**
1774
- * Find the shell panel this top-level widget is part of
1775
- */
1776
- protected findPanel(widget: Widget | undefined): TheiaDockPanel | undefined {
1777
- if (!widget) {
1778
- return undefined;
1779
- }
1780
- const title = widget.title;
1781
- const mainPanelTabBar = this.mainPanel.findTabBar(title);
1782
- if (mainPanelTabBar) {
1783
- return this.mainPanel;
1784
- }
1785
- const bottomPanelTabBar = this.bottomPanel.findTabBar(title);
1786
- if (bottomPanelTabBar) {
1787
- return this.bottomPanel;
1788
- }
1789
- if (ArrayExt.firstIndexOf(this.leftPanelHandler.tabBar.titles, title) > -1) {
1790
- return this.leftPanelHandler.dockPanel;
1791
- }
1792
- if (ArrayExt.firstIndexOf(this.rightPanelHandler.tabBar.titles, title) > -1) {
1793
- return this.rightPanelHandler.dockPanel;
1794
- }
1795
- return undefined;
1796
- }
1797
-
1798
- /**
1799
- * Return the tab bar that has the currently active widget, or undefined.
1800
- */
1801
- get currentTabBar(): TabBar<Widget> | undefined {
1802
- const currentWidget = this.currentWidget;
1803
- if (currentWidget) {
1804
- return this.getTabBarFor(currentWidget);
1805
- }
1806
- }
1807
-
1808
- /**
1809
- * Return the tab bar in the given shell area, or the tab bar that has the given widget, or undefined.
1810
- */
1811
- getTabBarFor(widgetOrArea: Widget | ApplicationShell.Area): TabBar<Widget> | undefined {
1812
- if (typeof widgetOrArea === 'string') {
1813
- switch (widgetOrArea) {
1814
- case 'main':
1815
- return this.mainPanel.currentTabBar;
1816
- case 'bottom':
1817
- return this.bottomPanel.currentTabBar;
1818
- case 'left':
1819
- return this.leftPanelHandler.tabBar;
1820
- case 'right':
1821
- return this.rightPanelHandler.tabBar;
1822
- case 'secondaryWindow':
1823
- // Secondary windows don't have a tab bar
1824
- return undefined;
1825
- default:
1826
- throw new Error('Illegal argument: ' + widgetOrArea);
1827
- }
1828
- }
1829
- const widget = this.toTrackedStack(widgetOrArea.id).pop();
1830
- if (!widget) {
1831
- return undefined;
1832
- }
1833
- const widgetTitle = widget.title;
1834
- const mainPanelTabBar = this.mainPanel.findTabBar(widgetTitle);
1835
- if (mainPanelTabBar) {
1836
- return mainPanelTabBar;
1837
- }
1838
- const bottomPanelTabBar = this.bottomPanel.findTabBar(widgetTitle);
1839
- if (bottomPanelTabBar) {
1840
- return bottomPanelTabBar;
1841
- }
1842
- const leftPanelTabBar = this.leftPanelHandler.tabBar;
1843
- if (ArrayExt.firstIndexOf(leftPanelTabBar.titles, widgetTitle) > -1) {
1844
- return leftPanelTabBar;
1845
- }
1846
- const rightPanelTabBar = this.rightPanelHandler.tabBar;
1847
- if (ArrayExt.firstIndexOf(rightPanelTabBar.titles, widgetTitle) > -1) {
1848
- return rightPanelTabBar;
1849
- }
1850
- return undefined;
1851
- }
1852
-
1853
- /**
1854
- * The tab bars contained in the main shell area. If there is no widget in the main area, the
1855
- * returned array is empty.
1856
- */
1857
- get mainAreaTabBars(): TabBar<Widget>[] {
1858
- return toArray(this.mainPanel.tabBars());
1859
- }
1860
-
1861
- /**
1862
- * The tab bars contained in the bottom shell area. If there is no widget in the bottom area,
1863
- * the returned array is empty.
1864
- */
1865
- get bottomAreaTabBars(): TabBar<Widget>[] {
1866
- return toArray(this.bottomPanel.tabBars());
1867
- }
1868
-
1869
- /**
1870
- * The tab bars contained in all shell areas.
1871
- */
1872
- get allTabBars(): TabBar<Widget>[] {
1873
- return [...this.mainAreaTabBars, ...this.bottomAreaTabBars, this.leftPanelHandler.tabBar, this.rightPanelHandler.tabBar];
1874
- }
1875
-
1876
- /*
1877
- * Activate the next tab in the current tab bar.
1878
- */
1879
- activateNextTabInTabBar(current: TabBar<Widget> | undefined = this.currentTabBar): boolean {
1880
- const index = this.nextTabIndexInTabBar(current);
1881
- if (!current || index === -1) {
1882
- return false;
1883
- }
1884
- current.currentIndex = index;
1885
- if (current.currentTitle) {
1886
- this.activateWidget(current.currentTitle.owner.id);
1887
- }
1888
- return true;
1889
- }
1890
-
1891
- nextTabIndexInTabBar(current: TabBar<Widget> | undefined = this.currentTabBar): number {
1892
- if (!current || current.titles.length <= 1) {
1893
- return -1;
1894
- }
1895
- const index = current.currentIndex;
1896
- if (index === -1) {
1897
- return -1;
1898
- }
1899
- if (index < current.titles.length - 1) {
1900
- return index + 1;
1901
- }
1902
- return 0;
1903
- }
1904
-
1905
- activateNextTab(): boolean {
1906
- const current = this.currentTabBar;
1907
- if (current) {
1908
- const ci = current.currentIndex;
1909
- if (ci !== -1) {
1910
- if (ci < current.titles.length - 1) {
1911
- current.currentIndex += 1;
1912
- if (current.currentTitle) {
1913
- this.activateWidget(current.currentTitle.owner.id);
1914
- }
1915
- return true;
1916
- } else if (ci === current.titles.length - 1) {
1917
- return this.activateNextTabBar(current);
1918
- }
1919
- }
1920
- }
1921
- return false;
1922
- }
1923
-
1924
- activateNextTabBar(current: TabBar<Widget> | undefined = this.currentTabBar): boolean {
1925
- const nextBar = this.nextTabBar(current);
1926
- if (nextBar) {
1927
- nextBar.currentIndex = 0;
1928
- if (nextBar.currentTitle) {
1929
- this.activateWidget(nextBar.currentTitle.owner.id);
1930
- }
1931
- return true;
1932
- }
1933
- return false;
1934
- }
1935
-
1936
- /**
1937
- * Return the tab bar next to the given tab bar; return the given tab bar if there is no adjacent one.
1938
- */
1939
- nextTabBar(current: TabBar<Widget> | undefined = this.currentTabBar): TabBar<Widget> | undefined {
1940
- let bars = toArray(this.bottomPanel.tabBars());
1941
- let len = bars.length;
1942
- let ci = ArrayExt.firstIndexOf(bars, current);
1943
- if (ci < 0) {
1944
- bars = toArray(this.mainPanel.tabBars());
1945
- len = bars.length;
1946
- ci = ArrayExt.firstIndexOf(bars, current);
1947
- }
1948
- if (ci >= 0 && ci < len - 1) {
1949
- return bars[ci + 1];
1950
- } else if (ci >= 0 && ci === len - 1) {
1951
- return bars[0];
1952
- } else {
1953
- return current;
1954
- }
1955
- }
1956
-
1957
- /*
1958
- * Activate the previous tab in the current tab bar.
1959
- */
1960
- activatePreviousTabInTabBar(current: TabBar<Widget> | undefined = this.currentTabBar): boolean {
1961
- const index = this.previousTabIndexInTabBar(current);
1962
- if (!current || index === -1) {
1963
- return false;
1964
- }
1965
- current.currentIndex = index;
1966
- if (current.currentTitle) {
1967
- this.activateWidget(current.currentTitle.owner.id);
1968
- }
1969
- return true;
1970
- }
1971
-
1972
- previousTabIndexInTabBar(current: TabBar<Widget> | undefined = this.currentTabBar): number {
1973
- if (!current || current.titles.length <= 1) {
1974
- return -1;
1975
- }
1976
- const index = current.currentIndex;
1977
- if (index === -1) {
1978
- return -1;
1979
- }
1980
- if (index > 0) {
1981
- return index - 1;
1982
- }
1983
- return current.titles.length - 1;
1984
- }
1985
-
1986
- activatePreviousTab(): boolean {
1987
- const current = this.currentTabBar;
1988
- if (current) {
1989
- const ci = current.currentIndex;
1990
- if (ci !== -1) {
1991
- if (ci > 0) {
1992
- current.currentIndex -= 1;
1993
- if (current.currentTitle) {
1994
- this.activateWidget(current.currentTitle.owner.id);
1995
- }
1996
- return true;
1997
- } else if (ci === 0) {
1998
- if (current && current.titles.length > 0) {
1999
- current.currentIndex = current.titles.length - 1;
2000
- if (current.currentTitle) {
2001
- this.activateWidget(current.currentTitle.owner.id);
2002
- }
2003
- return true;
2004
- }
2005
- return this.activatePreviousTabBar(current);
2006
- }
2007
- }
2008
- }
2009
- return false;
2010
- }
2011
-
2012
- activatePreviousTabBar(current: TabBar<Widget> | undefined = this.currentTabBar): boolean {
2013
- const prevBar = this.previousTabBar(current);
2014
- if (!prevBar) {
2015
- return false;
2016
- }
2017
- if (!prevBar.currentTitle) {
2018
- prevBar.currentIndex = prevBar.titles.length - 1;
2019
- }
2020
- if (prevBar.currentTitle) {
2021
- this.activateWidget(prevBar.currentTitle.owner.id);
2022
- }
2023
- return true;
2024
- }
2025
-
2026
- /**
2027
- * Return the tab bar previous to the given tab bar; return the given tab bar if there is no adjacent one.
2028
- */
2029
- previousTabBar(current: TabBar<Widget> | undefined = this.currentTabBar): TabBar<Widget> | undefined {
2030
- const bars = toArray(this.mainPanel.tabBars());
2031
- const len = bars.length;
2032
- const ci = ArrayExt.firstIndexOf(bars, current);
2033
- if (ci > 0) {
2034
- return bars[ci - 1];
2035
- } else if (ci === 0) {
2036
- return bars[len - 1];
2037
- } else {
2038
- return current;
2039
- }
2040
- }
2041
-
2042
- /**
2043
- * Test whether the current widget is dirty.
2044
- */
2045
- canSave(): boolean {
2046
- return this.saveResourceService.canSave(this.currentWidget);
2047
- }
2048
-
2049
- /**
2050
- * Save the current widget if it is dirty.
2051
- */
2052
- async save(options?: SaveOptions): Promise<void> {
2053
- await this.saveResourceService.save(this.currentWidget, options);
2054
- }
2055
-
2056
- /**
2057
- * Test whether there is a dirty widget.
2058
- */
2059
- canSaveAll(): boolean {
2060
- return this.tracker.widgets.some(widget => this.saveResourceService.canSave(widget));
2061
- }
2062
-
2063
- /**
2064
- * Save all dirty widgets.
2065
- */
2066
- async saveAll(options?: SaveOptions): Promise<void> {
2067
- for (const widget of this.widgets) {
2068
- if (this.saveResourceService.canSaveNotSaveAs(widget)) {
2069
- await this.saveResourceService.save(widget, options);
2070
- }
2071
- }
2072
- }
2073
-
2074
- /**
2075
- * Returns a snapshot of all tracked widgets to allow async modifications.
2076
- */
2077
- get widgets(): ReadonlyArray<Widget> {
2078
- return [...this.tracker.widgets];
2079
- }
2080
-
2081
- getWidgetById(id: string): Widget | undefined {
2082
- for (const widget of this.tracker.widgets) {
2083
- if (widget.id === id) {
2084
- return widget;
2085
- }
2086
- }
2087
- return undefined;
2088
- }
2089
-
2090
- canToggleMaximized(widget: Widget | undefined = this.currentWidget): boolean {
2091
- const area = widget && this.getAreaFor(widget);
2092
- return area === 'main' || area === 'bottom';
2093
- }
2094
-
2095
- toggleMaximized(widget: Widget | undefined = this.currentWidget): void {
2096
- const area = widget && this.getAreaPanelFor(widget);
2097
- if (area instanceof TheiaDockPanel && (area === this.mainPanel || area === this.bottomPanel)) {
2098
- area.toggleMaximized();
2099
- this.revealWidget(widget!.id);
2100
- }
2101
- }
2102
-
2103
- }
2104
-
2105
- /**
2106
- * The namespace for `ApplicationShell` class statics.
2107
- */
2108
- export namespace ApplicationShell {
2109
- /**
2110
- * The areas of the application shell where widgets can reside.
2111
- */
2112
- export type Area = 'main' | 'top' | 'left' | 'right' | 'bottom' | 'secondaryWindow';
2113
-
2114
- export const areaLabels: Record<Area, string> = {
2115
- main: nls.localizeByDefault('Main'),
2116
- top: nls.localizeByDefault('Top'),
2117
- left: nls.localizeByDefault('Left'),
2118
- right: nls.localizeByDefault('Right'),
2119
- bottom: nls.localizeByDefault('Bottom'),
2120
- secondaryWindow: nls.localize('theia/shell-area/secondary', 'Secondary Window'),
2121
- };
2122
-
2123
- /**
2124
- * The _side areas_ are those shell areas that can be collapsed and expanded,
2125
- * i.e. `left`, `right`, and `bottom`.
2126
- */
2127
- export function isSideArea(area?: string): area is 'left' | 'right' | 'bottom' {
2128
- return area === 'left' || area === 'right' || area === 'bottom';
2129
- }
2130
-
2131
- export function isValidArea(area?: unknown): area is ApplicationShell.Area {
2132
- const areas = ['main', 'top', 'left', 'right', 'bottom', 'secondaryWindow'];
2133
- return typeof area === 'string' && areas.includes(area);
2134
- }
2135
-
2136
- /**
2137
- * General options for the application shell. These are passed on construction and can be modified
2138
- * through dependency injection (`ApplicationShellOptions` symbol).
2139
- */
2140
- export interface Options extends Widget.IOptions {
2141
- bottomPanel: BottomPanelOptions;
2142
- leftPanel: SidePanel.Options;
2143
- rightPanel: SidePanel.Options;
2144
- }
2145
-
2146
- export interface BottomPanelOptions extends SidePanel.Options {
2147
- }
2148
-
2149
- /**
2150
- * The default values for application shell options.
2151
- */
2152
- export const DEFAULT_OPTIONS = Object.freeze(<Options>{
2153
- bottomPanel: Object.freeze(<BottomPanelOptions>{
2154
- emptySize: 140,
2155
- expandThreshold: 160,
2156
- expandDuration: 0,
2157
- initialSizeRatio: 0.382
2158
- }),
2159
- leftPanel: Object.freeze(<SidePanel.Options>{
2160
- emptySize: 140,
2161
- expandThreshold: 140,
2162
- expandDuration: 0,
2163
- initialSizeRatio: 0.191
2164
- }),
2165
- rightPanel: Object.freeze(<SidePanel.Options>{
2166
- emptySize: 140,
2167
- expandThreshold: 140,
2168
- expandDuration: 0,
2169
- initialSizeRatio: 0.191
2170
- })
2171
- });
2172
-
2173
- /**
2174
- * Whether a widget should be opened to the side tab bar relatively to the reference widget.
2175
- */
2176
- export type OpenToSideMode = 'open-to-left' | 'open-to-right';
2177
-
2178
- export function isOpenToSideMode(mode: unknown): mode is OpenToSideMode {
2179
- return mode === 'open-to-left' || mode === 'open-to-right';
2180
- }
2181
-
2182
- /**
2183
- * Options for adding a widget to the application shell.
2184
- */
2185
- export interface WidgetOptions extends SidePanel.WidgetOptions {
2186
- /**
2187
- * The area of the application shell where the widget will reside.
2188
- */
2189
- area?: Area;
2190
- /**
2191
- * The insertion mode for adding the widget.
2192
- *
2193
- * The default is `'tab-after'`.
2194
- */
2195
- mode?: DockLayout.InsertMode | OpenToSideMode
2196
- /**
2197
- * The reference widget for the insert location.
2198
- *
2199
- * The default is `undefined`.
2200
- */
2201
- ref?: Widget;
2202
- }
2203
-
2204
- export interface CloseOptions {
2205
- /**
2206
- * if optional then a user will be prompted
2207
- * if undefined then close will be canceled
2208
- * if true then will be saved on close
2209
- * if false then won't be saved on close
2210
- */
2211
- save?: boolean | undefined
2212
- }
2213
-
2214
- /**
2215
- * Data to save and load the application shell layout.
2216
- */
2217
- export interface LayoutData {
2218
- version?: string | ApplicationShellLayoutVersion,
2219
- mainPanel?: DockPanel.ILayoutConfig;
2220
- mainPanelPinned?: boolean[];
2221
- bottomPanel?: BottomPanelLayoutData;
2222
- leftPanel?: SidePanel.LayoutData;
2223
- rightPanel?: SidePanel.LayoutData;
2224
- activeWidgetId?: string;
2225
- }
2226
-
2227
- /**
2228
- * Data to save and load the bottom panel layout.
2229
- */
2230
- export interface BottomPanelLayoutData {
2231
- config?: DockPanel.ILayoutConfig;
2232
- size?: number;
2233
- expanded?: boolean;
2234
- pinned?: boolean[];
2235
- }
2236
-
2237
- /**
2238
- * Exposes widgets which activation state should be tracked by shell.
2239
- */
2240
- export interface TrackableWidgetProvider {
2241
- getTrackableWidgets(): Widget[]
2242
- readonly onDidChangeTrackableWidgets?: CommonEvent<Widget[]>
2243
- /**
2244
- * Make visible and focus a trackable widget for the given id.
2245
- * If not implemented then `activate` request will be sent to a child widget directly.
2246
- */
2247
- activateWidget?(id: string): Widget | undefined;
2248
- /**
2249
- * Make visible a trackable widget for the given id.
2250
- * If not implemented then a widget should be always visible when an owner is visible.
2251
- */
2252
- revealWidget?(id: string): Widget | undefined;
2253
- }
2254
-
2255
- export namespace TrackableWidgetProvider {
2256
- export function is(widget: unknown): widget is TrackableWidgetProvider {
2257
- return isObject(widget) && 'getTrackableWidgets' in widget;
2258
- }
2259
- }
2260
-
2261
- }
1
+ // *****************************************************************************
2
+ // Copyright (C) 2018 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 { injectable, inject, optional, postConstruct } from 'inversify';
18
+ import { ArrayExt, find, toArray, each } from '@phosphor/algorithm';
19
+ import {
20
+ BoxLayout, BoxPanel, DockLayout, DockPanel, FocusTracker, Layout, Panel, SplitLayout,
21
+ SplitPanel, TabBar, Widget, Title
22
+ } from '@phosphor/widgets';
23
+ import { Message } from '@phosphor/messaging';
24
+ import { IDragEvent } from '@phosphor/dragdrop';
25
+ import { RecursivePartial, Event as CommonEvent, DisposableCollection, Disposable, environment, isObject } from '../../common';
26
+ import { animationFrame } from '../browser';
27
+ import { Saveable, SaveableWidget, SaveOptions, SaveableSource } from '../saveable';
28
+ import { StatusBarImpl, StatusBarEntry, StatusBarAlignment } from '../status-bar/status-bar';
29
+ import { TheiaDockPanel, BOTTOM_AREA_ID, MAIN_AREA_ID } from './theia-dock-panel';
30
+ import { SidePanelHandler, SidePanel, SidePanelHandlerFactory } from './side-panel-handler';
31
+ import { TabBarRendererFactory, SHELL_TABBAR_CONTEXT_MENU, ScrollableTabBar, ToolbarAwareTabBar } from './tab-bars';
32
+ import { SplitPositionHandler, SplitPositionOptions } from './split-panels';
33
+ import { FrontendApplicationStateService } from '../frontend-application-state';
34
+ import { TabBarToolbarRegistry, TabBarToolbarFactory } from './tab-bar-toolbar';
35
+ import { ContextKeyService } from '../context-key-service';
36
+ import { Emitter } from '../../common/event';
37
+ import { waitForRevealed, waitForClosed, PINNED_CLASS } from '../widgets';
38
+ import { CorePreferences } from '../core-preferences';
39
+ import { BreadcrumbsRendererFactory } from '../breadcrumbs/breadcrumbs-renderer';
40
+ import { Deferred } from '../../common/promise-util';
41
+ import { SaveResourceService } from '../save-resource-service';
42
+ import { nls } from '../../common/nls';
43
+ import { SecondaryWindowHandler } from '../secondary-window-handler';
44
+ import URI from '../../common/uri';
45
+ import { OpenerService } from '../opener-service';
46
+ import { PreviewableWidget } from '../widgets/previewable-widget';
47
+ import { WindowService } from '../window/window-service';
48
+
49
+ /** The class name added to ApplicationShell instances. */
50
+ const APPLICATION_SHELL_CLASS = 'theia-ApplicationShell';
51
+ /** The class name added to the main and bottom area panels. */
52
+ const MAIN_BOTTOM_AREA_CLASS = 'theia-app-centers';
53
+ /** Status bar entry identifier for the bottom panel toggle button. */
54
+ const BOTTOM_PANEL_TOGGLE_ID = 'bottom-panel-toggle';
55
+ /** The class name added to the main area panel. */
56
+ const MAIN_AREA_CLASS = 'theia-app-main';
57
+ /** The class name added to the bottom area panel. */
58
+ const BOTTOM_AREA_CLASS = 'theia-app-bottom';
59
+
60
+ export type ApplicationShellLayoutVersion =
61
+ /** layout versioning is introduced, unversioned layout are not compatible */
62
+ 2.0 |
63
+ /** view containers are introduced, backward compatible to 2.0 */
64
+ 3.0 |
65
+ /** git history view is replaced by a more generic scm history view, backward compatible to 3.0 */
66
+ 4.0 |
67
+ /** Replace custom/font-awesome icons with codicons */
68
+ 5.0 |
69
+ /** added the ability to drag and drop view parts between view containers */
70
+ 6.0;
71
+
72
+ /**
73
+ * When a version is increased, make sure to introduce a migration (ApplicationShellLayoutMigration) to this version.
74
+ */
75
+ export const applicationShellLayoutVersion: ApplicationShellLayoutVersion = 5.0;
76
+
77
+ export const ApplicationShellOptions = Symbol('ApplicationShellOptions');
78
+ export const DockPanelRendererFactory = Symbol('DockPanelRendererFactory');
79
+ export interface DockPanelRendererFactory {
80
+ (): DockPanelRenderer
81
+ }
82
+
83
+ /**
84
+ * A renderer for dock panels that supports context menus on tabs.
85
+ */
86
+ @injectable()
87
+ export class DockPanelRenderer implements DockLayout.IRenderer {
88
+
89
+ @inject(TheiaDockPanel.Factory)
90
+ protected readonly dockPanelFactory: TheiaDockPanel.Factory;
91
+
92
+ readonly tabBarClasses: string[] = [];
93
+
94
+ private readonly onDidCreateTabBarEmitter = new Emitter<TabBar<Widget>>();
95
+
96
+ constructor(
97
+ @inject(TabBarRendererFactory) protected readonly tabBarRendererFactory: TabBarRendererFactory,
98
+ @inject(TabBarToolbarRegistry) protected readonly tabBarToolbarRegistry: TabBarToolbarRegistry,
99
+ @inject(TabBarToolbarFactory) protected readonly tabBarToolbarFactory: TabBarToolbarFactory,
100
+ @inject(BreadcrumbsRendererFactory) protected readonly breadcrumbsRendererFactory: BreadcrumbsRendererFactory,
101
+ @inject(CorePreferences) protected readonly corePreferences: CorePreferences
102
+ ) { }
103
+
104
+ get onDidCreateTabBar(): CommonEvent<TabBar<Widget>> {
105
+ return this.onDidCreateTabBarEmitter.event;
106
+ }
107
+
108
+ createTabBar(): TabBar<Widget> {
109
+ const getDynamicTabOptions: () => ScrollableTabBar.Options | undefined = () => {
110
+ if (this.corePreferences.get('workbench.tab.shrinkToFit.enabled')) {
111
+ return {
112
+ minimumTabSize: this.corePreferences.get('workbench.tab.shrinkToFit.minimumSize'),
113
+ defaultTabSize: this.corePreferences.get('workbench.tab.shrinkToFit.defaultSize')
114
+ };
115
+ } else {
116
+ return undefined;
117
+ }
118
+ };
119
+
120
+ const renderer = this.tabBarRendererFactory();
121
+ const tabBar = new ToolbarAwareTabBar(
122
+ this.tabBarToolbarRegistry,
123
+ this.tabBarToolbarFactory,
124
+ this.breadcrumbsRendererFactory,
125
+ {
126
+ renderer,
127
+ // Scroll bar options
128
+ handlers: ['drag-thumb', 'keyboard', 'wheel', 'touch'],
129
+ useBothWheelAxes: true,
130
+ scrollXMarginOffset: 4,
131
+ suppressScrollY: true
132
+ },
133
+ getDynamicTabOptions());
134
+ this.tabBarClasses.forEach(c => tabBar.addClass(c));
135
+ renderer.tabBar = tabBar;
136
+ renderer.contextMenuPath = SHELL_TABBAR_CONTEXT_MENU;
137
+ tabBar.currentChanged.connect(this.onCurrentTabChanged, this);
138
+ const prefChangeDisposable = this.corePreferences.onPreferenceChanged(change => {
139
+ if (change.preferenceName === 'workbench.tab.shrinkToFit.enabled' ||
140
+ change.preferenceName === 'workbench.tab.shrinkToFit.minimumSize' ||
141
+ change.preferenceName === 'workbench.tab.shrinkToFit.defaultSize') {
142
+ tabBar.dynamicTabOptions = getDynamicTabOptions();
143
+ }
144
+ });
145
+ tabBar.disposed.connect(() => {
146
+ prefChangeDisposable.dispose();
147
+ renderer.dispose();
148
+ });
149
+ this.onDidCreateTabBarEmitter.fire(tabBar);
150
+ return tabBar;
151
+ }
152
+
153
+ createHandle(): HTMLDivElement {
154
+ return DockPanel.defaultRenderer.createHandle();
155
+ }
156
+
157
+ protected onCurrentTabChanged(sender: ToolbarAwareTabBar, { currentIndex }: TabBar.ICurrentChangedArgs<Widget>): void {
158
+ if (currentIndex >= 0) {
159
+ sender.revealTab(currentIndex);
160
+ }
161
+ }
162
+ }
163
+
164
+ /**
165
+ * Data stored while dragging widgets in the shell.
166
+ */
167
+ interface WidgetDragState {
168
+ startTime: number;
169
+ leftExpanded: boolean;
170
+ rightExpanded: boolean;
171
+ bottomExpanded: boolean;
172
+ lastDragOver?: IDragEvent;
173
+ leaveTimeout?: number;
174
+ }
175
+
176
+ /**
177
+ * The application shell manages the top-level widgets of the application. Use this class to
178
+ * add, remove, or activate a widget.
179
+ */
180
+ @injectable()
181
+ export class ApplicationShell extends Widget {
182
+
183
+ /**
184
+ * The dock panel in the main shell area. This is where editors usually go to.
185
+ */
186
+ mainPanel: TheiaDockPanel;
187
+
188
+ /**
189
+ * The dock panel in the bottom shell area. In contrast to the main panel, the bottom panel
190
+ * can be collapsed and expanded.
191
+ */
192
+ bottomPanel: TheiaDockPanel;
193
+
194
+ /**
195
+ * Handler for the left side panel. The primary application views go here, such as the
196
+ * file explorer and the git view.
197
+ */
198
+ leftPanelHandler: SidePanelHandler;
199
+
200
+ /**
201
+ * Handler for the right side panel. The secondary application views go here, such as the
202
+ * outline view.
203
+ */
204
+ rightPanelHandler: SidePanelHandler;
205
+
206
+ /**
207
+ * General options for the application shell.
208
+ */
209
+ protected options: ApplicationShell.Options;
210
+
211
+ /**
212
+ * The fixed-size panel shown on top. This one usually holds the main menu.
213
+ */
214
+ topPanel: Panel;
215
+
216
+ /**
217
+ * The current state of the bottom panel.
218
+ */
219
+ protected readonly bottomPanelState: SidePanel.State = {
220
+ empty: true,
221
+ expansion: SidePanel.ExpansionState.collapsed,
222
+ pendingUpdate: Promise.resolve()
223
+ };
224
+
225
+ private readonly tracker = new FocusTracker<Widget>();
226
+ private dragState?: WidgetDragState;
227
+ additionalDraggedUris: URI[] | undefined;
228
+
229
+ @inject(ContextKeyService)
230
+ protected readonly contextKeyService: ContextKeyService;
231
+
232
+ @inject(OpenerService)
233
+ protected readonly openerService: OpenerService;
234
+
235
+ protected readonly onDidAddWidgetEmitter = new Emitter<Widget>();
236
+ readonly onDidAddWidget = this.onDidAddWidgetEmitter.event;
237
+ protected fireDidAddWidget(widget: Widget): void {
238
+ this.onDidAddWidgetEmitter.fire(widget);
239
+ }
240
+
241
+ protected readonly onDidRemoveWidgetEmitter = new Emitter<Widget>();
242
+ readonly onDidRemoveWidget = this.onDidRemoveWidgetEmitter.event;
243
+ protected fireDidRemoveWidget(widget: Widget): void {
244
+ this.onDidRemoveWidgetEmitter.fire(widget);
245
+ }
246
+
247
+ protected readonly onDidChangeActiveWidgetEmitter = new Emitter<FocusTracker.IChangedArgs<Widget>>();
248
+ readonly onDidChangeActiveWidget = this.onDidChangeActiveWidgetEmitter.event;
249
+
250
+ protected readonly onDidChangeCurrentWidgetEmitter = new Emitter<FocusTracker.IChangedArgs<Widget>>();
251
+ readonly onDidChangeCurrentWidget = this.onDidChangeCurrentWidgetEmitter.event;
252
+
253
+ protected readonly onDidDoubleClickMainAreaEmitter = new Emitter<void>();
254
+ readonly onDidDoubleClickMainArea = this.onDidDoubleClickMainAreaEmitter.event;
255
+
256
+ @inject(TheiaDockPanel.Factory)
257
+ protected readonly dockPanelFactory: TheiaDockPanel.Factory;
258
+
259
+ private _mainPanelRenderer: DockPanelRenderer;
260
+ get mainPanelRenderer(): DockPanelRenderer {
261
+ return this._mainPanelRenderer;
262
+ }
263
+
264
+ /**
265
+ * Construct a new application shell.
266
+ */
267
+ constructor(
268
+ @inject(DockPanelRendererFactory) protected dockPanelRendererFactory: () => DockPanelRenderer,
269
+ @inject(StatusBarImpl) protected readonly statusBar: StatusBarImpl,
270
+ @inject(SidePanelHandlerFactory) protected readonly sidePanelHandlerFactory: () => SidePanelHandler,
271
+ @inject(SplitPositionHandler) protected splitPositionHandler: SplitPositionHandler,
272
+ @inject(FrontendApplicationStateService) protected readonly applicationStateService: FrontendApplicationStateService,
273
+ @inject(ApplicationShellOptions) @optional() options: RecursivePartial<ApplicationShell.Options> = {},
274
+ @inject(CorePreferences) protected readonly corePreferences: CorePreferences,
275
+ @inject(SaveResourceService) protected readonly saveResourceService: SaveResourceService,
276
+ @inject(SecondaryWindowHandler) protected readonly secondaryWindowHandler: SecondaryWindowHandler,
277
+ @inject(WindowService) protected readonly windowService: WindowService
278
+ ) {
279
+ super(options as Widget.IOptions);
280
+
281
+ // Merge the user-defined application options with the default options
282
+ this.options = {
283
+ bottomPanel: {
284
+ ...ApplicationShell.DEFAULT_OPTIONS.bottomPanel,
285
+ ...options?.bottomPanel || {}
286
+ },
287
+ leftPanel: {
288
+ ...ApplicationShell.DEFAULT_OPTIONS.leftPanel,
289
+ ...options?.leftPanel || {}
290
+ },
291
+ rightPanel: {
292
+ ...ApplicationShell.DEFAULT_OPTIONS.rightPanel,
293
+ ...options?.rightPanel || {}
294
+ }
295
+ };
296
+ }
297
+
298
+ @postConstruct()
299
+ protected init(): void {
300
+ this.initializeShell();
301
+ this.initSidebarVisibleKeyContext();
302
+ this.initFocusKeyContexts();
303
+
304
+ if (!environment.electron.is()) {
305
+ this.corePreferences.ready.then(() => {
306
+ this.setTopPanelVisibility(this.corePreferences['window.menuBarVisibility']);
307
+ });
308
+ this.corePreferences.onPreferenceChanged(preference => {
309
+ if (preference.preferenceName === 'window.menuBarVisibility') {
310
+ this.setTopPanelVisibility(preference.newValue);
311
+ }
312
+ });
313
+ }
314
+
315
+ this.corePreferences.onPreferenceChanged(preference => {
316
+ if (preference.preferenceName === 'window.tabbar.enhancedPreview') {
317
+ this.allTabBars.forEach(tabBar => {
318
+ tabBar.update();
319
+ });
320
+ }
321
+ });
322
+ }
323
+
324
+ protected initializeShell(): void {
325
+ this.addClass(APPLICATION_SHELL_CLASS);
326
+ this.id = 'theia-app-shell';
327
+
328
+ this.mainPanel = this.createMainPanel();
329
+ this.topPanel = this.createTopPanel();
330
+ this.bottomPanel = this.createBottomPanel();
331
+
332
+ this.leftPanelHandler = this.sidePanelHandlerFactory();
333
+ this.leftPanelHandler.create('left', this.options.leftPanel);
334
+ this.leftPanelHandler.dockPanel.widgetAdded.connect((_, widget) => this.fireDidAddWidget(widget));
335
+ this.leftPanelHandler.dockPanel.widgetRemoved.connect((_, widget) => this.fireDidRemoveWidget(widget));
336
+
337
+ this.rightPanelHandler = this.sidePanelHandlerFactory();
338
+ this.rightPanelHandler.create('right', this.options.rightPanel);
339
+ this.rightPanelHandler.dockPanel.widgetAdded.connect((_, widget) => this.fireDidAddWidget(widget));
340
+ this.rightPanelHandler.dockPanel.widgetRemoved.connect((_, widget) => this.fireDidRemoveWidget(widget));
341
+
342
+ this.secondaryWindowHandler.init(this);
343
+ this.secondaryWindowHandler.onDidAddWidget(([widget, window]) => this.fireDidAddWidget(widget));
344
+ this.secondaryWindowHandler.onDidRemoveWidget(([widget, window]) => this.fireDidRemoveWidget(widget));
345
+
346
+ this.layout = this.createLayout();
347
+
348
+ this.tracker.currentChanged.connect(this.onCurrentChanged, this);
349
+ this.tracker.activeChanged.connect(this.onActiveChanged, this);
350
+ }
351
+
352
+ protected initSidebarVisibleKeyContext(): void {
353
+ const leftSideBarPanel = this.leftPanelHandler.dockPanel;
354
+ const sidebarVisibleKey = this.contextKeyService.createKey('sidebarVisible', leftSideBarPanel.isVisible);
355
+ const onAfterShow = leftSideBarPanel['onAfterShow'].bind(leftSideBarPanel);
356
+ leftSideBarPanel['onAfterShow'] = (msg: Message) => {
357
+ onAfterShow(msg);
358
+ sidebarVisibleKey.set(true);
359
+ };
360
+ const onAfterHide = leftSideBarPanel['onAfterHide'].bind(leftSideBarPanel);
361
+ leftSideBarPanel['onAfterHide'] = (msg: Message) => {
362
+ onAfterHide(msg);
363
+ sidebarVisibleKey.set(false);
364
+ };
365
+ }
366
+
367
+ protected initFocusKeyContexts(): void {
368
+ const sideBarFocus = this.contextKeyService.createKey<boolean>('sideBarFocus', false);
369
+ const panelFocus = this.contextKeyService.createKey<boolean>('panelFocus', false);
370
+ const updateFocusContextKeys = () => {
371
+ const area = this.activeWidget && this.getAreaFor(this.activeWidget);
372
+ sideBarFocus.set(area === 'left');
373
+ panelFocus.set(area === 'main');
374
+ };
375
+ updateFocusContextKeys();
376
+ this.onDidChangeActiveWidget(updateFocusContextKeys);
377
+ }
378
+
379
+ protected setTopPanelVisibility(preference: string): void {
380
+ const hiddenPreferences = ['compact', 'hidden'];
381
+ this.topPanel.setHidden(hiddenPreferences.includes(preference));
382
+ }
383
+
384
+ protected override onBeforeAttach(msg: Message): void {
385
+ document.addEventListener('p-dragenter', this, true);
386
+ document.addEventListener('p-dragover', this, true);
387
+ document.addEventListener('p-dragleave', this, true);
388
+ document.addEventListener('p-drop', this, true);
389
+ }
390
+
391
+ protected override onAfterDetach(msg: Message): void {
392
+ document.removeEventListener('p-dragenter', this, true);
393
+ document.removeEventListener('p-dragover', this, true);
394
+ document.removeEventListener('p-dragleave', this, true);
395
+ document.removeEventListener('p-drop', this, true);
396
+ }
397
+
398
+ handleEvent(event: Event): void {
399
+ switch (event.type) {
400
+ case 'p-dragenter':
401
+ this.onDragEnter(event as IDragEvent);
402
+ break;
403
+ case 'p-dragover':
404
+ this.onDragOver(event as IDragEvent);
405
+ break;
406
+ case 'p-drop':
407
+ this.onDrop(event as IDragEvent);
408
+ break;
409
+ case 'p-dragleave':
410
+ this.onDragLeave(event as IDragEvent);
411
+ break;
412
+ }
413
+ }
414
+
415
+ protected onDragEnter({ mimeData }: IDragEvent): void {
416
+ if (!this.dragState) {
417
+ if (mimeData && mimeData.hasData('application/vnd.phosphor.widget-factory')) {
418
+ // The drag contains a widget, so we'll track it and expand side panels as needed
419
+ this.dragState = {
420
+ startTime: performance.now(),
421
+ leftExpanded: false,
422
+ rightExpanded: false,
423
+ bottomExpanded: false
424
+ };
425
+ }
426
+ }
427
+ }
428
+
429
+ protected onDragOver(event: IDragEvent): void {
430
+ const state = this.dragState;
431
+ if (state) {
432
+ state.lastDragOver = event;
433
+ if (state.leaveTimeout) {
434
+ window.clearTimeout(state.leaveTimeout);
435
+ state.leaveTimeout = undefined;
436
+ }
437
+ const { clientX, clientY } = event;
438
+ const { offsetLeft, offsetTop, clientWidth, clientHeight } = this.node;
439
+
440
+ // Don't expand any side panels right after the drag has started
441
+ const allowExpansion = performance.now() - state.startTime >= 500;
442
+ const expLeft = allowExpansion && clientX >= offsetLeft
443
+ && clientX <= offsetLeft + this.options.leftPanel.expandThreshold;
444
+ const expRight = allowExpansion && clientX <= offsetLeft + clientWidth
445
+ && clientX >= offsetLeft + clientWidth - this.options.rightPanel.expandThreshold;
446
+ const expBottom = allowExpansion && !expLeft && !expRight && clientY <= offsetTop + clientHeight
447
+ && clientY >= offsetTop + clientHeight - this.options.bottomPanel.expandThreshold;
448
+ // eslint-disable-next-line no-null/no-null
449
+ if (expLeft && !state.leftExpanded && this.leftPanelHandler.tabBar.currentTitle === null) {
450
+ // The mouse cursor is moved close to the left border
451
+ this.leftPanelHandler.expand();
452
+ this.leftPanelHandler.state.pendingUpdate.then(() => this.dispatchMouseMove());
453
+ state.leftExpanded = true;
454
+ } else if (!expLeft && state.leftExpanded) {
455
+ // The mouse cursor is moved away from the left border
456
+ this.leftPanelHandler.collapse();
457
+ state.leftExpanded = false;
458
+ }
459
+ // eslint-disable-next-line no-null/no-null
460
+ if (expRight && !state.rightExpanded && this.rightPanelHandler.tabBar.currentTitle === null) {
461
+ // The mouse cursor is moved close to the right border
462
+ this.rightPanelHandler.expand();
463
+ this.rightPanelHandler.state.pendingUpdate.then(() => this.dispatchMouseMove());
464
+ state.rightExpanded = true;
465
+ } else if (!expRight && state.rightExpanded) {
466
+ // The mouse cursor is moved away from the right border
467
+ this.rightPanelHandler.collapse();
468
+ state.rightExpanded = false;
469
+ }
470
+ if (expBottom && !state.bottomExpanded && this.bottomPanel.isHidden) {
471
+ // The mouse cursor is moved close to the bottom border
472
+ this.expandBottomPanel();
473
+ this.bottomPanelState.pendingUpdate.then(() => this.dispatchMouseMove());
474
+ state.bottomExpanded = true;
475
+ } else if (!expBottom && state.bottomExpanded) {
476
+ // The mouse cursor is moved away from the bottom border
477
+ this.collapseBottomPanel();
478
+ state.bottomExpanded = false;
479
+ }
480
+ }
481
+ }
482
+
483
+ /**
484
+ * This method is called after a side panel has been expanded while dragging a widget. It fires
485
+ * a `mousemove` event so that the drag overlay markers are updated correctly in all dock panels.
486
+ */
487
+ private dispatchMouseMove(): void {
488
+ if (this.dragState && this.dragState.lastDragOver) {
489
+ const { clientX, clientY } = this.dragState.lastDragOver;
490
+ const event = document.createEvent('MouseEvent');
491
+ event.initMouseEvent('mousemove', true, true, window, 0, 0, 0,
492
+ // eslint-disable-next-line no-null/no-null
493
+ clientX, clientY, false, false, false, false, 0, null);
494
+ document.dispatchEvent(event);
495
+ }
496
+ }
497
+
498
+ protected onDrop(event: IDragEvent): void {
499
+ const state = this.dragState;
500
+ if (state) {
501
+ if (state.leaveTimeout) {
502
+ window.clearTimeout(state.leaveTimeout);
503
+ }
504
+ this.dragState = undefined;
505
+ window.requestAnimationFrame(() => {
506
+ // Clean up the side panel state in the next frame
507
+ if (this.leftPanelHandler.dockPanel.isEmpty) {
508
+ this.leftPanelHandler.collapse();
509
+ }
510
+ if (this.rightPanelHandler.dockPanel.isEmpty) {
511
+ this.rightPanelHandler.collapse();
512
+ }
513
+ if (this.bottomPanel.isEmpty) {
514
+ this.collapseBottomPanel();
515
+ }
516
+ });
517
+ }
518
+ }
519
+
520
+ protected onDragLeave(event: IDragEvent): void {
521
+ const state = this.dragState;
522
+ if (state) {
523
+ state.lastDragOver = undefined;
524
+ if (state.leaveTimeout) {
525
+ window.clearTimeout(state.leaveTimeout);
526
+ }
527
+ state.leaveTimeout = window.setTimeout(() => {
528
+ this.dragState = undefined;
529
+ if (state.leftExpanded || this.leftPanelHandler.dockPanel.isEmpty) {
530
+ this.leftPanelHandler.collapse();
531
+ }
532
+ if (state.rightExpanded || this.rightPanelHandler.dockPanel.isEmpty) {
533
+ this.rightPanelHandler.collapse();
534
+ }
535
+ if (state.bottomExpanded || this.bottomPanel.isEmpty) {
536
+ this.collapseBottomPanel();
537
+ }
538
+ }, 100);
539
+ }
540
+ }
541
+
542
+ /**
543
+ * Create the dock panel in the main shell area.
544
+ */
545
+ protected createMainPanel(): TheiaDockPanel {
546
+ const renderer = this.dockPanelRendererFactory();
547
+ renderer.tabBarClasses.push(MAIN_BOTTOM_AREA_CLASS);
548
+ renderer.tabBarClasses.push(MAIN_AREA_CLASS);
549
+ this._mainPanelRenderer = renderer;
550
+ const dockPanel = this.dockPanelFactory({
551
+ mode: 'multiple-document',
552
+ renderer,
553
+ spacing: 0
554
+ });
555
+ dockPanel.id = MAIN_AREA_ID;
556
+ dockPanel.widgetAdded.connect((_, widget) => this.fireDidAddWidget(widget));
557
+ dockPanel.widgetRemoved.connect((_, widget) => this.fireDidRemoveWidget(widget));
558
+
559
+ const openUri = async (fileUri: URI) => {
560
+ try {
561
+ const opener = await this.openerService.getOpener(fileUri);
562
+ opener.open(fileUri);
563
+ } catch (e) {
564
+ console.info(`no opener found for '${fileUri}'`);
565
+ }
566
+ };
567
+
568
+ dockPanel.node.addEventListener('drop', event => {
569
+ if (event.dataTransfer) {
570
+ const uris = this.additionalDraggedUris || ApplicationShell.getDraggedEditorUris(event.dataTransfer);
571
+ if (uris.length > 0) {
572
+ uris.forEach(openUri);
573
+ } else if (event.dataTransfer.files?.length > 0) {
574
+ // the files were dragged from the outside the workspace
575
+ Array.from(event.dataTransfer.files).forEach(file => {
576
+ if (file.path) {
577
+ const fileUri = URI.fromComponents({
578
+ scheme: 'file',
579
+ path: file.path,
580
+ authority: '',
581
+ query: '',
582
+ fragment: ''
583
+ });
584
+ openUri(fileUri);
585
+ }
586
+ });
587
+ }
588
+ }
589
+ });
590
+
591
+ dockPanel.node.addEventListener('dblclick', event => {
592
+ const el = event.target as Element;
593
+ if (el.id === MAIN_AREA_ID || el.classList.contains('p-TabBar-content')) {
594
+ this.onDidDoubleClickMainAreaEmitter.fire();
595
+ }
596
+ });
597
+
598
+ const handler = (e: DragEvent) => {
599
+ if (e.dataTransfer) {
600
+ e.dataTransfer.dropEffect = 'link';
601
+ e.preventDefault();
602
+ e.stopPropagation();
603
+ }
604
+ };
605
+ dockPanel.node.addEventListener('dragover', handler);
606
+ dockPanel.node.addEventListener('dragenter', handler);
607
+
608
+ return dockPanel;
609
+ }
610
+
611
+ addAdditionalDraggedEditorUris(uris: URI[]): void {
612
+ this.additionalDraggedUris = uris;
613
+ }
614
+
615
+ clearAdditionalDraggedEditorUris(): void {
616
+ this.additionalDraggedUris = undefined;
617
+ }
618
+
619
+ protected static getDraggedEditorUris(dataTransfer: DataTransfer): URI[] {
620
+ const data = dataTransfer.getData('theia-editor-dnd');
621
+ return data ? data.split('\n').map(entry => new URI(entry)) : [];
622
+ }
623
+
624
+ static setDraggedEditorUris(dataTransfer: DataTransfer, uris: URI[]): void {
625
+ dataTransfer.setData('theia-editor-dnd', uris.map(uri => uri.toString()).join('\n'));
626
+ }
627
+
628
+ /**
629
+ * Create the dock panel in the bottom shell area.
630
+ */
631
+ protected createBottomPanel(): TheiaDockPanel {
632
+ const renderer = this.dockPanelRendererFactory();
633
+ renderer.tabBarClasses.push(MAIN_BOTTOM_AREA_CLASS);
634
+ renderer.tabBarClasses.push(BOTTOM_AREA_CLASS);
635
+ const dockPanel = this.dockPanelFactory({
636
+ mode: 'multiple-document',
637
+ renderer,
638
+ spacing: 0
639
+ });
640
+ dockPanel.id = BOTTOM_AREA_ID;
641
+ dockPanel.widgetAdded.connect((sender, widget) => {
642
+ this.refreshBottomPanelToggleButton();
643
+ });
644
+ dockPanel.widgetRemoved.connect((sender, widget) => {
645
+ if (sender.isEmpty) {
646
+ this.collapseBottomPanel();
647
+ }
648
+ this.refreshBottomPanelToggleButton();
649
+ }, this);
650
+ dockPanel.node.addEventListener('p-dragenter', event => {
651
+ // Make sure that the main panel hides its overlay when the bottom panel is expanded
652
+ this.mainPanel.overlay.hide(0);
653
+ });
654
+ dockPanel.hide();
655
+ dockPanel.widgetAdded.connect((_, widget) => this.fireDidAddWidget(widget));
656
+ dockPanel.widgetRemoved.connect((_, widget) => this.fireDidRemoveWidget(widget));
657
+ return dockPanel;
658
+ }
659
+
660
+ /**
661
+ * Create the top panel, which is used to hold the main menu.
662
+ */
663
+ protected createTopPanel(): Panel {
664
+ const topPanel = new Panel();
665
+ topPanel.id = 'theia-top-panel';
666
+ topPanel.hide();
667
+ return topPanel;
668
+ }
669
+
670
+ /**
671
+ * Create a box layout to assemble the application shell layout.
672
+ */
673
+ protected createBoxLayout(widgets: Widget[], stretch?: number[], options?: BoxPanel.IOptions): BoxLayout {
674
+ const boxLayout = new BoxLayout(options);
675
+ for (let i = 0; i < widgets.length; i++) {
676
+ if (stretch !== undefined && i < stretch.length) {
677
+ BoxPanel.setStretch(widgets[i], stretch[i]);
678
+ }
679
+ boxLayout.addWidget(widgets[i]);
680
+ }
681
+ return boxLayout;
682
+ }
683
+
684
+ /**
685
+ * Create a split layout to assemble the application shell layout.
686
+ */
687
+ protected createSplitLayout(widgets: Widget[], stretch?: number[], options?: Partial<SplitLayout.IOptions>): SplitLayout {
688
+ let optParam: SplitLayout.IOptions = { renderer: SplitPanel.defaultRenderer, };
689
+ if (options) {
690
+ optParam = { ...optParam, ...options };
691
+ }
692
+ const splitLayout = new SplitLayout(optParam);
693
+ for (let i = 0; i < widgets.length; i++) {
694
+ if (stretch !== undefined && i < stretch.length) {
695
+ SplitPanel.setStretch(widgets[i], stretch[i]);
696
+ }
697
+ splitLayout.addWidget(widgets[i]);
698
+ }
699
+ return splitLayout;
700
+ }
701
+
702
+ /**
703
+ * Assemble the application shell layout. Override this method in order to change the arrangement
704
+ * of the main area and the side panels.
705
+ */
706
+ protected createLayout(): Layout {
707
+ const bottomSplitLayout = this.createSplitLayout(
708
+ [this.mainPanel, this.bottomPanel],
709
+ [1, 0],
710
+ { orientation: 'vertical', spacing: 0 }
711
+ );
712
+ const panelForBottomArea = new SplitPanel({ layout: bottomSplitLayout });
713
+ panelForBottomArea.id = 'theia-bottom-split-panel';
714
+
715
+ const leftRightSplitLayout = this.createSplitLayout(
716
+ [this.leftPanelHandler.container, panelForBottomArea, this.rightPanelHandler.container],
717
+ [0, 1, 0],
718
+ { orientation: 'horizontal', spacing: 0 }
719
+ );
720
+ const panelForSideAreas = new SplitPanel({ layout: leftRightSplitLayout });
721
+ panelForSideAreas.id = 'theia-left-right-split-panel';
722
+
723
+ return this.createBoxLayout(
724
+ [this.topPanel, panelForSideAreas, this.statusBar],
725
+ [0, 1, 0],
726
+ { direction: 'top-to-bottom', spacing: 0 }
727
+ );
728
+ }
729
+
730
+ /**
731
+ * Create an object that describes the current shell layout. This object may contain references
732
+ * to widgets; these need to be transformed before the layout can be serialized.
733
+ */
734
+ getLayoutData(): ApplicationShell.LayoutData {
735
+ return {
736
+ version: applicationShellLayoutVersion,
737
+ mainPanel: this.mainPanel.saveLayout(),
738
+ mainPanelPinned: this.getPinnedMainWidgets(),
739
+ bottomPanel: {
740
+ config: this.bottomPanel.saveLayout(),
741
+ pinned: this.getPinnedBottomWidgets(),
742
+ size: this.bottomPanel.isVisible ? this.getBottomPanelSize() : this.bottomPanelState.lastPanelSize,
743
+ expanded: this.isExpanded('bottom')
744
+ },
745
+ leftPanel: this.leftPanelHandler.getLayoutData(),
746
+ rightPanel: this.rightPanelHandler.getLayoutData(),
747
+ activeWidgetId: this.activeWidget ? this.activeWidget.id : undefined
748
+ };
749
+ }
750
+
751
+ // Get an array corresponding to main panel widgets' pinned state.
752
+ getPinnedMainWidgets(): boolean[] {
753
+ const pinned: boolean[] = [];
754
+
755
+ toArray(this.mainPanel.widgets()).forEach((a, i) => {
756
+ pinned[i] = a.title.className.includes(PINNED_CLASS);
757
+ });
758
+
759
+ return pinned;
760
+ }
761
+
762
+ // Get an array corresponding to bottom panel widgets' pinned state.
763
+ getPinnedBottomWidgets(): boolean[] {
764
+ const pinned: boolean[] = [];
765
+
766
+ toArray(this.bottomPanel.widgets()).forEach((a, i) => {
767
+ pinned[i] = a.title.className.includes(PINNED_CLASS);
768
+ });
769
+
770
+ return pinned;
771
+ }
772
+
773
+ /**
774
+ * Compute the current height of the bottom panel. This implementation assumes that the container
775
+ * of the bottom panel is a `SplitPanel`.
776
+ */
777
+ protected getBottomPanelSize(): number | undefined {
778
+ const parent = this.bottomPanel.parent;
779
+ if (parent instanceof SplitPanel && parent.isVisible) {
780
+ const index = parent.widgets.indexOf(this.bottomPanel) - 1;
781
+ if (index >= 0) {
782
+ const handle = parent.handles[index];
783
+ if (!handle.classList.contains('p-mod-hidden')) {
784
+ const parentHeight = parent.node.clientHeight;
785
+ return parentHeight - handle.offsetTop;
786
+ }
787
+ }
788
+ }
789
+ }
790
+
791
+ /**
792
+ * Determine the default size to apply when the bottom panel is expanded for the first time.
793
+ */
794
+ protected getDefaultBottomPanelSize(): number | undefined {
795
+ const parent = this.bottomPanel.parent;
796
+ if (parent && parent.isVisible) {
797
+ return parent.node.clientHeight * this.options.bottomPanel.initialSizeRatio;
798
+ }
799
+ }
800
+
801
+ /**
802
+ * Apply a shell layout that has been previously created with `getLayoutData`.
803
+ */
804
+ async setLayoutData(layoutData: ApplicationShell.LayoutData): Promise<void> {
805
+ const { mainPanel, mainPanelPinned, bottomPanel, leftPanel, rightPanel, activeWidgetId } = layoutData;
806
+ if (leftPanel) {
807
+ this.leftPanelHandler.setLayoutData(leftPanel);
808
+ this.registerWithFocusTracker(leftPanel);
809
+ }
810
+ if (rightPanel) {
811
+ this.rightPanelHandler.setLayoutData(rightPanel);
812
+ this.registerWithFocusTracker(rightPanel);
813
+ }
814
+ // Proceed with the bottom panel once the side panels are set up
815
+ await Promise.all([this.leftPanelHandler.state.pendingUpdate, this.rightPanelHandler.state.pendingUpdate]);
816
+ if (bottomPanel) {
817
+ if (bottomPanel.config) {
818
+ this.bottomPanel.restoreLayout(bottomPanel.config);
819
+ this.registerWithFocusTracker(bottomPanel.config.main);
820
+ }
821
+ if (bottomPanel.size) {
822
+ this.bottomPanelState.lastPanelSize = bottomPanel.size;
823
+ }
824
+ if (bottomPanel.expanded) {
825
+ this.expandBottomPanel();
826
+ } else {
827
+ this.collapseBottomPanel();
828
+ }
829
+ const widgets = toArray(this.bottomPanel.widgets());
830
+ this.bottomPanel.markActiveTabBar(widgets[0]?.title);
831
+ if (bottomPanel.pinned && bottomPanel.pinned.length === widgets.length) {
832
+ widgets.forEach((a, i) => {
833
+ if (bottomPanel.pinned![i]) {
834
+ a.title.className += ` ${PINNED_CLASS}`;
835
+ a.title.closable = false;
836
+ }
837
+ });
838
+ }
839
+ this.refreshBottomPanelToggleButton();
840
+ }
841
+ // Proceed with the main panel once all others are set up
842
+ await this.bottomPanelState.pendingUpdate;
843
+ if (mainPanel) {
844
+ this.mainPanel.restoreLayout(mainPanel);
845
+ this.registerWithFocusTracker(mainPanel.main);
846
+ const widgets = toArray(this.mainPanel.widgets());
847
+ // We don't store information about the last active tabbar
848
+ // So we simply mark the first as being active
849
+ this.mainPanel.markActiveTabBar(widgets[0]?.title);
850
+ if (mainPanelPinned && mainPanelPinned.length === widgets.length) {
851
+ widgets.forEach((a, i) => {
852
+ if (mainPanelPinned[i]) {
853
+ a.title.className += ` ${PINNED_CLASS}`;
854
+ a.title.closable = false;
855
+ }
856
+ });
857
+ }
858
+ }
859
+ if (activeWidgetId) {
860
+ this.activateWidget(activeWidgetId);
861
+ }
862
+ }
863
+
864
+ /**
865
+ * Modify the height of the bottom panel. This implementation assumes that the container of the
866
+ * bottom panel is a `SplitPanel`.
867
+ */
868
+ protected setBottomPanelSize(size: number): Promise<void> {
869
+ const enableAnimation = this.applicationStateService.state === 'ready';
870
+ const options: SplitPositionOptions = {
871
+ side: 'bottom',
872
+ duration: enableAnimation ? this.options.bottomPanel.expandDuration : 0,
873
+ referenceWidget: this.bottomPanel
874
+ };
875
+ const promise = this.splitPositionHandler.setSidePanelSize(this.bottomPanel, size, options);
876
+ const result = new Promise<void>(resolve => {
877
+ // Resolve the resulting promise in any case, regardless of whether resizing was successful
878
+ promise.then(() => resolve(), () => resolve());
879
+ });
880
+ this.bottomPanelState.pendingUpdate = this.bottomPanelState.pendingUpdate.then(() => result);
881
+ return result;
882
+ }
883
+
884
+ /**
885
+ * A promise that is resolved when all currently pending updates are done.
886
+ */
887
+ get pendingUpdates(): Promise<void> {
888
+ return Promise.all([
889
+ this.bottomPanelState.pendingUpdate,
890
+ this.leftPanelHandler.state.pendingUpdate,
891
+ this.rightPanelHandler.state.pendingUpdate
892
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
893
+ ]) as Promise<any>;
894
+ }
895
+
896
+ /**
897
+ * Track all widgets that are referenced by the given layout data.
898
+ */
899
+ protected registerWithFocusTracker(data: DockLayout.ITabAreaConfig | DockLayout.ISplitAreaConfig | SidePanel.LayoutData | null): void {
900
+ if (data) {
901
+ if (data.type === 'tab-area') {
902
+ for (const widget of data.widgets) {
903
+ if (widget) {
904
+ this.track(widget);
905
+ }
906
+ }
907
+ } else if (data.type === 'split-area') {
908
+ for (const child of data.children) {
909
+ this.registerWithFocusTracker(child);
910
+ }
911
+ } else if (data.type === 'sidepanel' && data.items) {
912
+ for (const item of data.items) {
913
+ if (item.widget) {
914
+ this.track(item.widget);
915
+ }
916
+ }
917
+ }
918
+ }
919
+ }
920
+
921
+ /**
922
+ * Add a widget to the application shell. The given widget must have a unique `id` property,
923
+ * which will be used as the DOM id.
924
+ *
925
+ * Widgets are removed from the shell by calling their `close` or `dispose` methods.
926
+ *
927
+ * Widgets added to the top area are not tracked regarding the _current_ and _active_ states.
928
+ */
929
+ async addWidget(widget: Widget, options?: Readonly<ApplicationShell.WidgetOptions>): Promise<void> {
930
+ if (!widget.id) {
931
+ console.error('Widgets added to the application shell must have a unique id property.');
932
+ return;
933
+ }
934
+ const { area, addOptions } = this.getInsertionOptions(options);
935
+ const sidePanelOptions: SidePanel.WidgetOptions = { rank: options?.rank };
936
+ switch (area) {
937
+ case 'main':
938
+ this.mainPanel.addWidget(widget, addOptions);
939
+ break;
940
+ case 'top':
941
+ this.topPanel.addWidget(widget);
942
+ break;
943
+ case 'bottom':
944
+ this.bottomPanel.addWidget(widget, addOptions);
945
+ break;
946
+ case 'left':
947
+ this.leftPanelHandler.addWidget(widget, sidePanelOptions);
948
+ break;
949
+ case 'right':
950
+ this.rightPanelHandler.addWidget(widget, sidePanelOptions);
951
+ break;
952
+ case 'secondaryWindow':
953
+ /** At the moment, widgets are only moved to this area (i.e. a secondary window) by moving them from one of the other areas. */
954
+ throw new Error('Widgets cannot be added directly to a secondary window');
955
+ default:
956
+ throw new Error('Unexpected area: ' + options?.area);
957
+ }
958
+ if (area !== 'top') {
959
+ this.track(widget);
960
+ }
961
+ }
962
+
963
+ getInsertionOptions(options?: Readonly<ApplicationShell.WidgetOptions>): { area: string; addOptions: DockLayout.IAddOptions; } {
964
+ let ref: Widget | undefined = options?.ref;
965
+ let area: ApplicationShell.Area = options?.area || 'main';
966
+ if (!ref && (area === 'main' || area === 'bottom')) {
967
+ const tabBar = this.getTabBarFor(area);
968
+ ref = tabBar && tabBar.currentTitle && tabBar.currentTitle.owner || undefined;
969
+ }
970
+ // make sure that ref belongs to area
971
+ area = ref && this.getAreaFor(ref) || area;
972
+ const addOptions: DockPanel.IAddOptions = {};
973
+ if (ApplicationShell.isOpenToSideMode(options?.mode)) {
974
+ const areaPanel = area === 'main' ? this.mainPanel : area === 'bottom' ? this.bottomPanel : undefined;
975
+ const sideRef = areaPanel && ref && (options?.mode === 'open-to-left' ?
976
+ areaPanel.previousTabBarWidget(ref) :
977
+ areaPanel.nextTabBarWidget(ref));
978
+ if (sideRef) {
979
+ addOptions.ref = sideRef;
980
+ } else {
981
+ addOptions.ref = ref;
982
+ addOptions.mode = options?.mode === 'open-to-left' ? 'split-left' : 'split-right';
983
+ }
984
+ } else {
985
+ addOptions.ref = ref;
986
+ addOptions.mode = options?.mode;
987
+ }
988
+ return { area, addOptions };
989
+ }
990
+
991
+ /**
992
+ * The widgets contained in the given shell area.
993
+ */
994
+ getWidgets(area: ApplicationShell.Area): Widget[] {
995
+ switch (area) {
996
+ case 'main':
997
+ return toArray(this.mainPanel.widgets());
998
+ case 'top':
999
+ return toArray(this.topPanel.widgets);
1000
+ case 'bottom':
1001
+ return toArray(this.bottomPanel.widgets());
1002
+ case 'left':
1003
+ return toArray(this.leftPanelHandler.dockPanel.widgets());
1004
+ case 'right':
1005
+ return toArray(this.rightPanelHandler.dockPanel.widgets());
1006
+ case 'secondaryWindow':
1007
+ return toArray(this.secondaryWindowHandler.widgets);
1008
+ default:
1009
+ throw new Error('Illegal argument: ' + area);
1010
+ }
1011
+ }
1012
+
1013
+ /**
1014
+ * Find the widget that contains the given HTML element. The returned widget may be one
1015
+ * that is managed by the application shell, or one that is embedded in another widget and
1016
+ * not directly managed by the shell, or a tab bar.
1017
+ */
1018
+ findWidgetForElement(element: HTMLElement): Widget | undefined {
1019
+ let widgetNode: HTMLElement | null = element;
1020
+ while (widgetNode && !widgetNode.classList.contains('p-Widget')) {
1021
+ widgetNode = widgetNode.parentElement;
1022
+ }
1023
+ if (widgetNode) {
1024
+ return this.findWidgetForNode(widgetNode, this);
1025
+ }
1026
+ return undefined;
1027
+ }
1028
+
1029
+ private findWidgetForNode(widgetNode: HTMLElement, widget: Widget): Widget | undefined {
1030
+ if (widget.node === widgetNode) {
1031
+ return widget;
1032
+ }
1033
+ let result: Widget | undefined;
1034
+ each(widget.children(), child => {
1035
+ result = this.findWidgetForNode(widgetNode, child);
1036
+ return !result;
1037
+ });
1038
+ return result;
1039
+ }
1040
+
1041
+ /**
1042
+ * Finds the title widget from the tab-bar.
1043
+ * @param tabBar used for providing an array of titles.
1044
+ * @returns the selected title widget, else returns the currentTitle or undefined.
1045
+ */
1046
+ findTitle(tabBar: TabBar<Widget>, event?: Event): Title<Widget> | undefined {
1047
+ if (event?.target instanceof HTMLElement) {
1048
+ const tabNode = event.target;
1049
+
1050
+ const titleIndex = Array.from(tabBar.contentNode.getElementsByClassName('p-TabBar-tab'))
1051
+ .findIndex(node => node.contains(tabNode));
1052
+
1053
+ if (titleIndex !== -1) {
1054
+ return tabBar.titles[titleIndex];
1055
+ }
1056
+
1057
+ }
1058
+ return tabBar.currentTitle || undefined;
1059
+ }
1060
+
1061
+ /**
1062
+ * Finds the tab-bar widget.
1063
+ * @returns the selected tab-bar, else returns the currentTabBar.
1064
+ */
1065
+ findTabBar(event?: Event): TabBar<Widget> | undefined {
1066
+ if (event?.target instanceof HTMLElement) {
1067
+ const tabBar = this.findWidgetForElement(event.target);
1068
+ if (tabBar instanceof TabBar) {
1069
+ return tabBar;
1070
+ }
1071
+ }
1072
+ return this.currentTabBar;
1073
+ }
1074
+
1075
+ /**
1076
+ * @returns the widget whose title has been targeted by a DOM event on a tabbar, or undefined if none can be found.
1077
+ */
1078
+ findTargetedWidget(event?: Event): Widget | undefined {
1079
+ if (event) {
1080
+ const tab = this.findTabBar(event);
1081
+ const title = tab && this.findTitle(tab, event);
1082
+ return title && title.owner;
1083
+ }
1084
+ }
1085
+
1086
+ /**
1087
+ * The current widget in the application shell. The current widget is the last widget that
1088
+ * was active and not yet closed. See the remarks to `activeWidget` on what _active_ means.
1089
+ */
1090
+ get currentWidget(): Widget | undefined {
1091
+ return this.tracker.currentWidget || undefined;
1092
+ }
1093
+
1094
+ /**
1095
+ * The active widget in the application shell. The active widget is the one that has focus
1096
+ * (either the widget itself or any of its contents).
1097
+ *
1098
+ * _Note:_ Focus is taken by a widget through the `onActivateRequest` method. It is up to the
1099
+ * widget implementation which DOM element will get the focus. The default implementation
1100
+ * does not take any focus; in that case the widget is never returned by this property.
1101
+ */
1102
+ get activeWidget(): Widget | undefined {
1103
+ return this.tracker.activeWidget || undefined;
1104
+ }
1105
+
1106
+ /**
1107
+ * Returns the last active widget in the given shell area.
1108
+ */
1109
+ getCurrentWidget(area: ApplicationShell.Area): Widget | undefined {
1110
+ let title: Title<Widget> | null | undefined;
1111
+ switch (area) {
1112
+ case 'main':
1113
+ title = this.mainPanel.currentTitle;
1114
+ break;
1115
+ case 'bottom':
1116
+ title = this.bottomPanel.currentTitle;
1117
+ break;
1118
+ case 'left':
1119
+ title = this.leftPanelHandler.tabBar.currentTitle;
1120
+ break;
1121
+ case 'right':
1122
+ title = this.rightPanelHandler.tabBar.currentTitle;
1123
+ break;
1124
+ case 'secondaryWindow':
1125
+ // The current widget in a secondary window is not tracked.
1126
+ return undefined;
1127
+ default:
1128
+ throw new Error('Illegal argument: ' + area);
1129
+ }
1130
+ return title ? title.owner : undefined;
1131
+ }
1132
+
1133
+ /**
1134
+ * Handle a change to the current widget.
1135
+ */
1136
+ private onCurrentChanged(sender: FocusTracker<Widget>, args: FocusTracker.IChangedArgs<Widget>): void {
1137
+ this.onDidChangeCurrentWidgetEmitter.fire(args);
1138
+ }
1139
+
1140
+ protected readonly toDisposeOnActiveChanged = new DisposableCollection();
1141
+
1142
+ /**
1143
+ * Handle a change to the active widget.
1144
+ */
1145
+ private onActiveChanged(sender: FocusTracker<Widget>, args: FocusTracker.IChangedArgs<Widget>): void {
1146
+ this.toDisposeOnActiveChanged.dispose();
1147
+ const { newValue, oldValue } = args;
1148
+ if (oldValue) {
1149
+ let w: Widget | null = oldValue;
1150
+ while (w) {
1151
+ // Remove the mark of the previously active widget
1152
+ w.title.className = w.title.className.replace(' theia-mod-active', '');
1153
+ w = w.parent;
1154
+ }
1155
+ // Reset the z-index to the default
1156
+ // eslint-disable-next-line no-null/no-null
1157
+ this.setZIndex(oldValue.node, null);
1158
+ }
1159
+ if (newValue) {
1160
+ let w: Widget | null = newValue;
1161
+ while (w) {
1162
+ // Mark the tab of the active widget
1163
+ w.title.className += ' theia-mod-active';
1164
+ w = w.parent;
1165
+ }
1166
+ // Reveal the title of the active widget in its tab bar
1167
+ const tabBar = this.getTabBarFor(newValue);
1168
+ if (tabBar instanceof ScrollableTabBar) {
1169
+ const index = tabBar.titles.indexOf(newValue.title);
1170
+ if (index >= 0) {
1171
+ tabBar.revealTab(index);
1172
+ }
1173
+ }
1174
+ const widget = this.toTrackedStack(newValue.id).pop();
1175
+ const panel = this.findPanel(widget);
1176
+ if (panel) {
1177
+ // if widget was undefined, we wouldn't have gotten a panel back before
1178
+ panel.markAsCurrent(widget!.title);
1179
+ }
1180
+ // Add checks to ensure that the 'sash' for left panel is displayed correctly
1181
+ if (newValue.node.className === 'p-Widget theia-view-container p-DockPanel-widget') {
1182
+ // Set the z-index so elements with `position: fixed` contained in the active widget are displayed correctly
1183
+ this.setZIndex(newValue.node, '1');
1184
+ }
1185
+
1186
+ // activate another widget if an active widget will be closed
1187
+ const onCloseRequest = newValue['onCloseRequest'];
1188
+ newValue['onCloseRequest'] = msg => {
1189
+ const currentTabBar = this.currentTabBar;
1190
+ if (currentTabBar) {
1191
+ const recentlyUsedInTabBar = currentTabBar['_previousTitle'] as TabBar<Widget>['currentTitle'];
1192
+ if (recentlyUsedInTabBar && recentlyUsedInTabBar.owner !== newValue) {
1193
+ currentTabBar.currentIndex = ArrayExt.firstIndexOf(currentTabBar.titles, recentlyUsedInTabBar);
1194
+ if (currentTabBar.currentTitle) {
1195
+ this.activateWidget(currentTabBar.currentTitle.owner.id);
1196
+ }
1197
+ } else if (!this.activateNextTabInTabBar(currentTabBar)) {
1198
+ if (!this.activatePreviousTabBar(currentTabBar)) {
1199
+ this.activateNextTabBar(currentTabBar);
1200
+ }
1201
+ }
1202
+ }
1203
+ newValue['onCloseRequest'] = onCloseRequest;
1204
+ newValue['onCloseRequest'](msg);
1205
+ };
1206
+ this.toDisposeOnActiveChanged.push(Disposable.create(() => newValue['onCloseRequest'] = onCloseRequest));
1207
+ if (PreviewableWidget.is(newValue)) {
1208
+ newValue.loaded = true;
1209
+ }
1210
+ }
1211
+ this.onDidChangeActiveWidgetEmitter.fire(args);
1212
+ }
1213
+
1214
+ /**
1215
+ * Set the z-index of the given element and its ancestors to the value `z`.
1216
+ */
1217
+ private setZIndex(element: HTMLElement, z: string | null): void {
1218
+ element.style.zIndex = z || '';
1219
+ const parent = element.parentElement;
1220
+ if (parent && parent !== this.node) {
1221
+ this.setZIndex(parent, z);
1222
+ }
1223
+ }
1224
+
1225
+ /**
1226
+ * Track the given widget so it is considered in the `current` and `active` state of the shell.
1227
+ */
1228
+ protected track(widget: Widget): void {
1229
+ if (this.tracker.widgets.indexOf(widget) !== -1) {
1230
+ return;
1231
+ }
1232
+ this.tracker.add(widget);
1233
+ this.checkActivation(widget);
1234
+ Saveable.apply(
1235
+ widget,
1236
+ () => this.widgets.filter((maybeSaveable): maybeSaveable is Widget & SaveableSource => !!Saveable.get(maybeSaveable)),
1237
+ async (toSave, options) => {
1238
+ await this.saveResourceService.save(toSave, options);
1239
+ },
1240
+ );
1241
+ if (ApplicationShell.TrackableWidgetProvider.is(widget)) {
1242
+ for (const toTrack of widget.getTrackableWidgets()) {
1243
+ this.track(toTrack);
1244
+ }
1245
+ if (widget.onDidChangeTrackableWidgets) {
1246
+ widget.onDidChangeTrackableWidgets(widgets => widgets.forEach(w => this.track(w)));
1247
+ }
1248
+ }
1249
+ }
1250
+
1251
+ /**
1252
+ * @returns an array of Widgets, all of which are tracked by the focus tracker
1253
+ * The first member of the array is the widget whose id is passed in, and the other widgets
1254
+ * are its tracked parents in ascending order
1255
+ */
1256
+ protected toTrackedStack(id: string): Widget[] {
1257
+ const tracked = new Map<string, Widget>(this.tracker.widgets.map(w => [w.id, w] as [string, Widget]));
1258
+ let current = tracked.get(id);
1259
+ const stack: Widget[] = [];
1260
+ while (current) {
1261
+ if (tracked.has(current.id)) {
1262
+ stack.push(current);
1263
+ }
1264
+ current = current.parent || undefined;
1265
+ }
1266
+ return stack;
1267
+ }
1268
+
1269
+ /**
1270
+ * Activate a widget in the application shell. This makes the widget visible and usually
1271
+ * also assigns focus to it.
1272
+ *
1273
+ * _Note:_ Focus is taken by a widget through the `onActivateRequest` method. It is up to the
1274
+ * widget implementation which DOM element will get the focus. The default implementation
1275
+ * does not take any focus.
1276
+ *
1277
+ * @returns the activated widget if it was found
1278
+ */
1279
+ async activateWidget(id: string): Promise<Widget | undefined> {
1280
+ const stack = this.toTrackedStack(id);
1281
+ let current = stack.pop();
1282
+ if (current && !this.doActivateWidget(current.id)) {
1283
+ return undefined;
1284
+ }
1285
+ while (current && stack.length) {
1286
+ const child = stack.pop()!;
1287
+ if (ApplicationShell.TrackableWidgetProvider.is(current) && current.activateWidget) {
1288
+ current = current.activateWidget(child.id);
1289
+ } else {
1290
+ child.activate();
1291
+ current = child;
1292
+ }
1293
+ }
1294
+ if (!current) {
1295
+ return undefined;
1296
+ }
1297
+ return Promise.all([
1298
+ this.waitForActivation(current.id),
1299
+ waitForRevealed(current),
1300
+ this.pendingUpdates
1301
+ ]).then(() => current, () => undefined);
1302
+ }
1303
+
1304
+ waitForActivation(id: string): Promise<void> {
1305
+ if (this.activeWidget && this.activeWidget.id === id) {
1306
+ return Promise.resolve();
1307
+ }
1308
+ const activation = new Deferred();
1309
+ const success = this.onDidChangeActiveWidget(() => {
1310
+ if (this.activeWidget && this.activeWidget.id === id) {
1311
+ activation.resolve();
1312
+ }
1313
+ });
1314
+ const failure = setTimeout(() => activation.reject(new Error(`Widget with id '${id}' failed to activate.`)), this.activationTimeout + 250);
1315
+ return activation.promise.finally(() => {
1316
+ success.dispose();
1317
+ clearTimeout(failure);
1318
+ });
1319
+ }
1320
+
1321
+ /**
1322
+ * Activate top-level area widget.
1323
+ */
1324
+ protected doActivateWidget(id: string): Widget | undefined {
1325
+ let widget = find(this.mainPanel.widgets(), w => w.id === id);
1326
+ if (widget) {
1327
+ this.mainPanel.activateWidget(widget);
1328
+ }
1329
+ if (!widget) {
1330
+ widget = find(this.bottomPanel.widgets(), w => w.id === id);
1331
+ if (widget) {
1332
+ this.expandBottomPanel();
1333
+ this.bottomPanel.activateWidget(widget);
1334
+ }
1335
+ }
1336
+ if (!widget) {
1337
+ widget = this.leftPanelHandler.activate(id);
1338
+ }
1339
+
1340
+ if (!widget) {
1341
+ widget = this.rightPanelHandler.activate(id);
1342
+ }
1343
+ if (widget) {
1344
+ this.windowService.focus();
1345
+ return widget;
1346
+ }
1347
+ return this.secondaryWindowHandler.activateWidget(id);
1348
+ }
1349
+
1350
+ /**
1351
+ * Focus is taken by a widget through the `onActivateRequest` method. It is up to the
1352
+ * widget implementation which DOM element will get the focus. The default implementation
1353
+ * of Widget does not take any focus. This method can help finding such problems by logging
1354
+ * a warning in case a widget was explicitly activated, but did not trigger a change of the
1355
+ * `activeWidget` property.
1356
+ */
1357
+ private checkActivation(widget: Widget): Widget {
1358
+ const onActivateRequest = widget['onActivateRequest'].bind(widget);
1359
+ widget['onActivateRequest'] = (msg: Message) => {
1360
+ onActivateRequest(msg);
1361
+ this.assertActivated(widget);
1362
+ };
1363
+ return widget;
1364
+ }
1365
+
1366
+ private readonly activationTimeout = 2000;
1367
+ private readonly toDisposeOnActivationCheck = new DisposableCollection();
1368
+ private assertActivated(widget: Widget): void {
1369
+ this.toDisposeOnActivationCheck.dispose();
1370
+
1371
+ const onDispose = () => this.toDisposeOnActivationCheck.dispose();
1372
+ widget.disposed.connect(onDispose);
1373
+ this.toDisposeOnActivationCheck.push(Disposable.create(() => widget.disposed.disconnect(onDispose)));
1374
+
1375
+ let start = 0;
1376
+ const step: FrameRequestCallback = timestamp => {
1377
+ const activeElement = widget.node.ownerDocument.activeElement;
1378
+ if (activeElement && widget.node.contains(activeElement)) {
1379
+ return;
1380
+ }
1381
+ if (!start) {
1382
+ start = timestamp;
1383
+ }
1384
+ const delta = timestamp - start;
1385
+ if (delta < this.activationTimeout) {
1386
+ request = window.requestAnimationFrame(step);
1387
+ } else {
1388
+ console.warn(`Widget was activated, but did not accept focus after ${this.activationTimeout}ms: ${widget.id}`);
1389
+ }
1390
+ };
1391
+ let request = window.requestAnimationFrame(step);
1392
+ this.toDisposeOnActivationCheck.push(Disposable.create(() => window.cancelAnimationFrame(request)));
1393
+ }
1394
+
1395
+ /**
1396
+ * Reveal a widget in the application shell. This makes the widget visible,
1397
+ * but does not activate it.
1398
+ *
1399
+ * @returns the revealed widget if it was found
1400
+ */
1401
+ async revealWidget(id: string): Promise<Widget | undefined> {
1402
+ const stack = this.toTrackedStack(id);
1403
+ let current = stack.pop();
1404
+ if (current && !this.doRevealWidget(current.id)) {
1405
+ return undefined;
1406
+ }
1407
+ while (current && stack.length) {
1408
+ const child = stack.pop()!;
1409
+ if (ApplicationShell.TrackableWidgetProvider.is(current) && current.revealWidget) {
1410
+ current = current.revealWidget(child.id);
1411
+ } else {
1412
+ current = child;
1413
+ }
1414
+ }
1415
+ if (!current) {
1416
+ return undefined;
1417
+ }
1418
+ await Promise.all([
1419
+ waitForRevealed(current),
1420
+ this.pendingUpdates
1421
+ ]);
1422
+ return current;
1423
+ }
1424
+
1425
+ /**
1426
+ * Reveal top-level area widget.
1427
+ */
1428
+ protected doRevealWidget(id: string): Widget | undefined {
1429
+ let widget = find(this.mainPanel.widgets(), w => w.id === id);
1430
+ if (!widget) {
1431
+ widget = find(this.bottomPanel.widgets(), w => w.id === id);
1432
+ if (widget) {
1433
+ this.expandBottomPanel();
1434
+ }
1435
+ }
1436
+ if (widget) {
1437
+ const tabBar = this.getTabBarFor(widget);
1438
+ if (tabBar) {
1439
+ tabBar.currentTitle = widget.title;
1440
+ }
1441
+ }
1442
+ if (!widget) {
1443
+ widget = this.leftPanelHandler.expand(id);
1444
+ }
1445
+ if (!widget) {
1446
+ widget = this.rightPanelHandler.expand(id);
1447
+ }
1448
+ if (widget) {
1449
+ this.windowService.focus();
1450
+ return widget;
1451
+ } else {
1452
+ return this.secondaryWindowHandler.revealWidget(id);
1453
+ }
1454
+ }
1455
+
1456
+ /**
1457
+ * Expand the named side panel area. This makes sure that the panel is visible, even if there
1458
+ * are no widgets in it. If the panel is already visible, nothing happens. If the panel is currently
1459
+ * collapsed (see `collapsePanel`) and it contains widgets, the widgets are revealed that were
1460
+ * visible before it was collapsed.
1461
+ */
1462
+ expandPanel(area: ApplicationShell.Area): void {
1463
+ switch (area) {
1464
+ case 'bottom':
1465
+ this.expandBottomPanel();
1466
+ break;
1467
+ case 'left':
1468
+ this.leftPanelHandler.expand();
1469
+ break;
1470
+ case 'right':
1471
+ this.rightPanelHandler.expand();
1472
+ break;
1473
+ default:
1474
+ throw new Error('Area cannot be expanded: ' + area);
1475
+ }
1476
+ }
1477
+
1478
+ /**
1479
+ * Adjusts the size of the given area in the application shell.
1480
+ *
1481
+ * @param size the desired size of the panel in pixels.
1482
+ * @param area the area to resize.
1483
+ */
1484
+ resize(size: number, area: ApplicationShell.Area): void {
1485
+ switch (area) {
1486
+ case 'bottom':
1487
+ if (this.bottomPanel.isHidden) {
1488
+ this.bottomPanelState.lastPanelSize = size;
1489
+ } else {
1490
+ this.setBottomPanelSize(size);
1491
+ }
1492
+ break;
1493
+ case 'left':
1494
+ this.leftPanelHandler.resize(size);
1495
+ break;
1496
+ case 'right':
1497
+ this.rightPanelHandler.resize(size);
1498
+ break;
1499
+ default:
1500
+ throw new Error('Area cannot be resized: ' + area);
1501
+ }
1502
+ }
1503
+
1504
+ /**
1505
+ * Expand the bottom panel. See `expandPanel` regarding the exact behavior.
1506
+ */
1507
+ protected expandBottomPanel(): void {
1508
+ const bottomPanel = this.bottomPanel;
1509
+ if (bottomPanel.isHidden) {
1510
+ let relativeSizes: number[] | undefined;
1511
+ const parent = bottomPanel.parent;
1512
+ if (parent instanceof SplitPanel) {
1513
+ relativeSizes = parent.relativeSizes();
1514
+ }
1515
+ bottomPanel.show();
1516
+ if (relativeSizes && parent instanceof SplitPanel) {
1517
+ // Make sure that the expansion animation starts at the smallest possible size
1518
+ parent.setRelativeSizes(relativeSizes);
1519
+ }
1520
+
1521
+ let size: number | undefined;
1522
+ if (bottomPanel.isEmpty) {
1523
+ bottomPanel.node.style.minHeight = '0';
1524
+ size = this.options.bottomPanel.emptySize;
1525
+ } else if (this.bottomPanelState.lastPanelSize) {
1526
+ size = this.bottomPanelState.lastPanelSize;
1527
+ } else {
1528
+ size = this.getDefaultBottomPanelSize();
1529
+ }
1530
+ if (size) {
1531
+ this.bottomPanelState.expansion = SidePanel.ExpansionState.expanding;
1532
+ this.setBottomPanelSize(size).then(() => {
1533
+ if (this.bottomPanelState.expansion === SidePanel.ExpansionState.expanding) {
1534
+ this.bottomPanelState.expansion = SidePanel.ExpansionState.expanded;
1535
+ }
1536
+ });
1537
+ } else {
1538
+ this.bottomPanelState.expansion = SidePanel.ExpansionState.expanded;
1539
+ }
1540
+ }
1541
+ }
1542
+
1543
+ /**
1544
+ * Collapse the named side panel area. This makes sure that the panel is hidden,
1545
+ * increasing the space that is available for other shell areas.
1546
+ */
1547
+ collapsePanel(area: ApplicationShell.Area): Promise<void> {
1548
+ switch (area) {
1549
+ case 'bottom':
1550
+ return this.collapseBottomPanel();
1551
+ case 'left':
1552
+ return this.leftPanelHandler.collapse();
1553
+ case 'right':
1554
+ return this.rightPanelHandler.collapse();
1555
+ default:
1556
+ throw new Error('Area cannot be collapsed: ' + area);
1557
+ }
1558
+ }
1559
+
1560
+ /**
1561
+ * Collapse the bottom panel. All contained widgets are hidden, but not closed.
1562
+ * They can be restored by calling `expandBottomPanel`.
1563
+ */
1564
+ protected collapseBottomPanel(): Promise<void> {
1565
+ const bottomPanel = this.bottomPanel;
1566
+ if (bottomPanel.isHidden) {
1567
+ return Promise.resolve();
1568
+ }
1569
+ if (this.bottomPanelState.expansion === SidePanel.ExpansionState.expanded) {
1570
+ const size = this.getBottomPanelSize();
1571
+ if (size) {
1572
+ this.bottomPanelState.lastPanelSize = size;
1573
+ }
1574
+ }
1575
+ this.bottomPanelState.expansion = SidePanel.ExpansionState.collapsed;
1576
+ bottomPanel.hide();
1577
+ return animationFrame();
1578
+ }
1579
+
1580
+ /**
1581
+ * Refresh the toggle button for the bottom panel. This implementation creates a status bar entry
1582
+ * and refers to the command `core.toggle.bottom.panel`.
1583
+ */
1584
+ protected refreshBottomPanelToggleButton(): void {
1585
+ if (this.bottomPanel.isEmpty) {
1586
+ this.statusBar.removeElement(BOTTOM_PANEL_TOGGLE_ID);
1587
+ } else {
1588
+ const label = nls.localize('theia/core/common/collapseBottomPanel', 'Toggle Bottom Panel');
1589
+ const element: StatusBarEntry = {
1590
+ name: label,
1591
+ text: '$(codicon-window)',
1592
+ alignment: StatusBarAlignment.RIGHT,
1593
+ tooltip: label,
1594
+ command: 'core.toggle.bottom.panel',
1595
+ accessibilityInformation: {
1596
+ label: label,
1597
+ role: 'button'
1598
+ },
1599
+ priority: -1000
1600
+ };
1601
+ this.statusBar.setElement(BOTTOM_PANEL_TOGGLE_ID, element);
1602
+ }
1603
+ }
1604
+
1605
+ /**
1606
+ * Check whether the named side panel area is expanded (returns `true`) or collapsed (returns `false`).
1607
+ */
1608
+ isExpanded(area: ApplicationShell.Area): boolean {
1609
+ switch (area) {
1610
+ case 'bottom':
1611
+ return this.bottomPanelState.expansion === SidePanel.ExpansionState.expanded;
1612
+ case 'left':
1613
+ return this.leftPanelHandler.state.expansion === SidePanel.ExpansionState.expanded;
1614
+ case 'right':
1615
+ return this.rightPanelHandler.state.expansion === SidePanel.ExpansionState.expanded;
1616
+ default:
1617
+ return true;
1618
+ }
1619
+ }
1620
+
1621
+ /**
1622
+ * Close all tabs or a selection of tabs in a specific part of the application shell.
1623
+ *
1624
+ * @param tabBarOrArea
1625
+ * Either the name of a shell area or a `TabBar` that is contained in such an area.
1626
+ * @param filter
1627
+ * If undefined, all tabs are closed; otherwise only those tabs that match the filter are closed.
1628
+ */
1629
+ async closeTabs(tabBarOrArea: TabBar<Widget> | ApplicationShell.Area,
1630
+ filter?: (title: Title<Widget>, index: number) => boolean): Promise<void> {
1631
+ const titles: Array<Title<Widget>> = this.getWidgetTitles(tabBarOrArea, filter);
1632
+ if (titles.length) {
1633
+ await this.closeMany(titles.map(title => title.owner));
1634
+ }
1635
+ }
1636
+
1637
+ saveTabs(tabBarOrArea: TabBar<Widget> | ApplicationShell.Area,
1638
+ filter?: (title: Title<Widget>, index: number) => boolean): void {
1639
+
1640
+ const titles = this.getWidgetTitles(tabBarOrArea, filter);
1641
+ for (let i = 0; i < titles.length; i++) {
1642
+ const widget = titles[i].owner;
1643
+ const saveable = Saveable.get(widget);
1644
+ saveable?.save();
1645
+ }
1646
+ }
1647
+
1648
+ /**
1649
+ * Collects all widget titles for the given tab bar or area and optionally filters them.
1650
+ *
1651
+ * @param tabBarOrArea The tab bar or area to retrieve the widget titles for
1652
+ * @param filter The filter to apply to the result
1653
+ * @returns The filtered array of widget titles or an empty array
1654
+ */
1655
+ protected getWidgetTitles(tabBarOrArea: TabBar<Widget> | ApplicationShell.Area,
1656
+ filter?: (title: Title<Widget>, index: number) => boolean): Title<Widget>[] {
1657
+
1658
+ const titles: Title<Widget>[] = [];
1659
+ if (tabBarOrArea === 'main') {
1660
+ this.mainAreaTabBars.forEach(tabbar => titles.push(...toArray(tabbar.titles)));
1661
+ } else if (tabBarOrArea === 'bottom') {
1662
+ this.bottomAreaTabBars.forEach(tabbar => titles.push(...toArray(tabbar.titles)));
1663
+ } else if (tabBarOrArea === 'secondaryWindow') {
1664
+ titles.push(...this.secondaryWindowHandler.widgets.map(w => w.title));
1665
+ } else if (typeof tabBarOrArea === 'string') {
1666
+ const tabbar = this.getTabBarFor(tabBarOrArea);
1667
+ if (tabbar) {
1668
+ titles.push(...toArray(tabbar.titles));
1669
+ }
1670
+ } else if (tabBarOrArea) {
1671
+ titles.push(...toArray(tabBarOrArea.titles));
1672
+ }
1673
+
1674
+ return filter ? titles.filter(filter) : titles;
1675
+ }
1676
+
1677
+ /**
1678
+ * @param targets the widgets to be closed
1679
+ * @return an array of all the widgets that were actually closed.
1680
+ */
1681
+ async closeMany(targets: Widget[], options?: ApplicationShell.CloseOptions): Promise<Widget[]> {
1682
+ if (options?.save === false || await Saveable.confirmSaveBeforeClose(targets, this.widgets.filter(widget => !targets.includes(widget)))) {
1683
+ return (await Promise.all(targets.map(target => this.closeWidget(target.id, options)))).filter((widget): widget is Widget => widget !== undefined);
1684
+ }
1685
+ return [];
1686
+ }
1687
+
1688
+ /**
1689
+ * @returns the widget that was closed, if any, `undefined` otherwise.
1690
+ *
1691
+ * If your use case requires closing multiple widgets, use {@link ApplicationShell#closeMany} instead. That method handles closing saveable widgets more reliably.
1692
+ */
1693
+ async closeWidget(id: string, options?: ApplicationShell.CloseOptions): Promise<Widget | undefined> {
1694
+ // TODO handle save for composite widgets, i.e. the preference widget has 2 editors
1695
+ const stack = this.toTrackedStack(id);
1696
+ const current = stack.pop();
1697
+ if (!current) {
1698
+ return undefined;
1699
+ }
1700
+ const saveableOptions = options && { shouldSave: () => options.save };
1701
+ const pendingClose = SaveableWidget.is(current)
1702
+ ? current.closeWithSaving(saveableOptions)
1703
+ : (current.close(), waitForClosed(current));
1704
+ await Promise.all([
1705
+ pendingClose,
1706
+ this.pendingUpdates
1707
+ ]);
1708
+ return stack[0] || current;
1709
+ }
1710
+
1711
+ /**
1712
+ * The shell area name of the currently active tab, or undefined.
1713
+ */
1714
+ get currentTabArea(): ApplicationShell.Area | undefined {
1715
+ const currentWidget = this.currentWidget;
1716
+ if (currentWidget) {
1717
+ return this.getAreaFor(currentWidget);
1718
+ }
1719
+ }
1720
+
1721
+ /**
1722
+ * Determine the name of the shell area where the given widget resides. The result is
1723
+ * undefined if the widget does not reside directly in the shell.
1724
+ */
1725
+ getAreaFor(input: TabBar<Widget> | Widget): ApplicationShell.Area | undefined {
1726
+ if (input instanceof TabBar) {
1727
+ if (find(this.mainPanel.tabBars(), tb => tb === input)) {
1728
+ return 'main';
1729
+ }
1730
+ if (find(this.bottomPanel.tabBars(), tb => tb === input)) {
1731
+ return 'bottom';
1732
+ }
1733
+ if (this.leftPanelHandler.tabBar === input) {
1734
+ return 'left';
1735
+ }
1736
+ if (this.rightPanelHandler.tabBar === input) {
1737
+ return 'right';
1738
+ }
1739
+ }
1740
+ const widget = this.toTrackedStack(input.id).pop();
1741
+ if (!widget) {
1742
+ return undefined;
1743
+ }
1744
+ const title = widget.title;
1745
+ const mainPanelTabBar = this.mainPanel.findTabBar(title);
1746
+ if (mainPanelTabBar) {
1747
+ return 'main';
1748
+ }
1749
+ const bottomPanelTabBar = this.bottomPanel.findTabBar(title);
1750
+ if (bottomPanelTabBar) {
1751
+ return 'bottom';
1752
+ }
1753
+ if (ArrayExt.firstIndexOf(this.leftPanelHandler.tabBar.titles, title) > -1) {
1754
+ return 'left';
1755
+ }
1756
+ if (ArrayExt.firstIndexOf(this.rightPanelHandler.tabBar.titles, title) > -1) {
1757
+ return 'right';
1758
+ }
1759
+ if (this.secondaryWindowHandler.widgets.includes(widget)) {
1760
+ return 'secondaryWindow';
1761
+ }
1762
+ return undefined;
1763
+ }
1764
+
1765
+ protected getAreaPanelFor(input: Widget): DockPanel | undefined {
1766
+ const widget = this.toTrackedStack(input.id).pop();
1767
+ if (!widget) {
1768
+ return undefined;
1769
+ }
1770
+ return this.findPanel(widget);
1771
+ }
1772
+
1773
+ /**
1774
+ * Find the shell panel this top-level widget is part of
1775
+ */
1776
+ protected findPanel(widget: Widget | undefined): TheiaDockPanel | undefined {
1777
+ if (!widget) {
1778
+ return undefined;
1779
+ }
1780
+ const title = widget.title;
1781
+ const mainPanelTabBar = this.mainPanel.findTabBar(title);
1782
+ if (mainPanelTabBar) {
1783
+ return this.mainPanel;
1784
+ }
1785
+ const bottomPanelTabBar = this.bottomPanel.findTabBar(title);
1786
+ if (bottomPanelTabBar) {
1787
+ return this.bottomPanel;
1788
+ }
1789
+ if (ArrayExt.firstIndexOf(this.leftPanelHandler.tabBar.titles, title) > -1) {
1790
+ return this.leftPanelHandler.dockPanel;
1791
+ }
1792
+ if (ArrayExt.firstIndexOf(this.rightPanelHandler.tabBar.titles, title) > -1) {
1793
+ return this.rightPanelHandler.dockPanel;
1794
+ }
1795
+ return undefined;
1796
+ }
1797
+
1798
+ /**
1799
+ * Return the tab bar that has the currently active widget, or undefined.
1800
+ */
1801
+ get currentTabBar(): TabBar<Widget> | undefined {
1802
+ const currentWidget = this.currentWidget;
1803
+ if (currentWidget) {
1804
+ return this.getTabBarFor(currentWidget);
1805
+ }
1806
+ }
1807
+
1808
+ /**
1809
+ * Return the tab bar in the given shell area, or the tab bar that has the given widget, or undefined.
1810
+ */
1811
+ getTabBarFor(widgetOrArea: Widget | ApplicationShell.Area): TabBar<Widget> | undefined {
1812
+ if (typeof widgetOrArea === 'string') {
1813
+ switch (widgetOrArea) {
1814
+ case 'main':
1815
+ return this.mainPanel.currentTabBar;
1816
+ case 'bottom':
1817
+ return this.bottomPanel.currentTabBar;
1818
+ case 'left':
1819
+ return this.leftPanelHandler.tabBar;
1820
+ case 'right':
1821
+ return this.rightPanelHandler.tabBar;
1822
+ case 'secondaryWindow':
1823
+ // Secondary windows don't have a tab bar
1824
+ return undefined;
1825
+ default:
1826
+ throw new Error('Illegal argument: ' + widgetOrArea);
1827
+ }
1828
+ }
1829
+ const widget = this.toTrackedStack(widgetOrArea.id).pop();
1830
+ if (!widget) {
1831
+ return undefined;
1832
+ }
1833
+ const widgetTitle = widget.title;
1834
+ const mainPanelTabBar = this.mainPanel.findTabBar(widgetTitle);
1835
+ if (mainPanelTabBar) {
1836
+ return mainPanelTabBar;
1837
+ }
1838
+ const bottomPanelTabBar = this.bottomPanel.findTabBar(widgetTitle);
1839
+ if (bottomPanelTabBar) {
1840
+ return bottomPanelTabBar;
1841
+ }
1842
+ const leftPanelTabBar = this.leftPanelHandler.tabBar;
1843
+ if (ArrayExt.firstIndexOf(leftPanelTabBar.titles, widgetTitle) > -1) {
1844
+ return leftPanelTabBar;
1845
+ }
1846
+ const rightPanelTabBar = this.rightPanelHandler.tabBar;
1847
+ if (ArrayExt.firstIndexOf(rightPanelTabBar.titles, widgetTitle) > -1) {
1848
+ return rightPanelTabBar;
1849
+ }
1850
+ return undefined;
1851
+ }
1852
+
1853
+ /**
1854
+ * The tab bars contained in the main shell area. If there is no widget in the main area, the
1855
+ * returned array is empty.
1856
+ */
1857
+ get mainAreaTabBars(): TabBar<Widget>[] {
1858
+ return toArray(this.mainPanel.tabBars());
1859
+ }
1860
+
1861
+ /**
1862
+ * The tab bars contained in the bottom shell area. If there is no widget in the bottom area,
1863
+ * the returned array is empty.
1864
+ */
1865
+ get bottomAreaTabBars(): TabBar<Widget>[] {
1866
+ return toArray(this.bottomPanel.tabBars());
1867
+ }
1868
+
1869
+ /**
1870
+ * The tab bars contained in all shell areas.
1871
+ */
1872
+ get allTabBars(): TabBar<Widget>[] {
1873
+ return [...this.mainAreaTabBars, ...this.bottomAreaTabBars, this.leftPanelHandler.tabBar, this.rightPanelHandler.tabBar];
1874
+ }
1875
+
1876
+ /*
1877
+ * Activate the next tab in the current tab bar.
1878
+ */
1879
+ activateNextTabInTabBar(current: TabBar<Widget> | undefined = this.currentTabBar): boolean {
1880
+ const index = this.nextTabIndexInTabBar(current);
1881
+ if (!current || index === -1) {
1882
+ return false;
1883
+ }
1884
+ current.currentIndex = index;
1885
+ if (current.currentTitle) {
1886
+ this.activateWidget(current.currentTitle.owner.id);
1887
+ }
1888
+ return true;
1889
+ }
1890
+
1891
+ nextTabIndexInTabBar(current: TabBar<Widget> | undefined = this.currentTabBar): number {
1892
+ if (!current || current.titles.length <= 1) {
1893
+ return -1;
1894
+ }
1895
+ const index = current.currentIndex;
1896
+ if (index === -1) {
1897
+ return -1;
1898
+ }
1899
+ if (index < current.titles.length - 1) {
1900
+ return index + 1;
1901
+ }
1902
+ return 0;
1903
+ }
1904
+
1905
+ activateNextTab(): boolean {
1906
+ const current = this.currentTabBar;
1907
+ if (current) {
1908
+ const ci = current.currentIndex;
1909
+ if (ci !== -1) {
1910
+ if (ci < current.titles.length - 1) {
1911
+ current.currentIndex += 1;
1912
+ if (current.currentTitle) {
1913
+ this.activateWidget(current.currentTitle.owner.id);
1914
+ }
1915
+ return true;
1916
+ } else if (ci === current.titles.length - 1) {
1917
+ return this.activateNextTabBar(current);
1918
+ }
1919
+ }
1920
+ }
1921
+ return false;
1922
+ }
1923
+
1924
+ activateNextTabBar(current: TabBar<Widget> | undefined = this.currentTabBar): boolean {
1925
+ const nextBar = this.nextTabBar(current);
1926
+ if (nextBar) {
1927
+ nextBar.currentIndex = 0;
1928
+ if (nextBar.currentTitle) {
1929
+ this.activateWidget(nextBar.currentTitle.owner.id);
1930
+ }
1931
+ return true;
1932
+ }
1933
+ return false;
1934
+ }
1935
+
1936
+ /**
1937
+ * Return the tab bar next to the given tab bar; return the given tab bar if there is no adjacent one.
1938
+ */
1939
+ nextTabBar(current: TabBar<Widget> | undefined = this.currentTabBar): TabBar<Widget> | undefined {
1940
+ let bars = toArray(this.bottomPanel.tabBars());
1941
+ let len = bars.length;
1942
+ let ci = ArrayExt.firstIndexOf(bars, current);
1943
+ if (ci < 0) {
1944
+ bars = toArray(this.mainPanel.tabBars());
1945
+ len = bars.length;
1946
+ ci = ArrayExt.firstIndexOf(bars, current);
1947
+ }
1948
+ if (ci >= 0 && ci < len - 1) {
1949
+ return bars[ci + 1];
1950
+ } else if (ci >= 0 && ci === len - 1) {
1951
+ return bars[0];
1952
+ } else {
1953
+ return current;
1954
+ }
1955
+ }
1956
+
1957
+ /*
1958
+ * Activate the previous tab in the current tab bar.
1959
+ */
1960
+ activatePreviousTabInTabBar(current: TabBar<Widget> | undefined = this.currentTabBar): boolean {
1961
+ const index = this.previousTabIndexInTabBar(current);
1962
+ if (!current || index === -1) {
1963
+ return false;
1964
+ }
1965
+ current.currentIndex = index;
1966
+ if (current.currentTitle) {
1967
+ this.activateWidget(current.currentTitle.owner.id);
1968
+ }
1969
+ return true;
1970
+ }
1971
+
1972
+ previousTabIndexInTabBar(current: TabBar<Widget> | undefined = this.currentTabBar): number {
1973
+ if (!current || current.titles.length <= 1) {
1974
+ return -1;
1975
+ }
1976
+ const index = current.currentIndex;
1977
+ if (index === -1) {
1978
+ return -1;
1979
+ }
1980
+ if (index > 0) {
1981
+ return index - 1;
1982
+ }
1983
+ return current.titles.length - 1;
1984
+ }
1985
+
1986
+ activatePreviousTab(): boolean {
1987
+ const current = this.currentTabBar;
1988
+ if (current) {
1989
+ const ci = current.currentIndex;
1990
+ if (ci !== -1) {
1991
+ if (ci > 0) {
1992
+ current.currentIndex -= 1;
1993
+ if (current.currentTitle) {
1994
+ this.activateWidget(current.currentTitle.owner.id);
1995
+ }
1996
+ return true;
1997
+ } else if (ci === 0) {
1998
+ if (current && current.titles.length > 0) {
1999
+ current.currentIndex = current.titles.length - 1;
2000
+ if (current.currentTitle) {
2001
+ this.activateWidget(current.currentTitle.owner.id);
2002
+ }
2003
+ return true;
2004
+ }
2005
+ return this.activatePreviousTabBar(current);
2006
+ }
2007
+ }
2008
+ }
2009
+ return false;
2010
+ }
2011
+
2012
+ activatePreviousTabBar(current: TabBar<Widget> | undefined = this.currentTabBar): boolean {
2013
+ const prevBar = this.previousTabBar(current);
2014
+ if (!prevBar) {
2015
+ return false;
2016
+ }
2017
+ if (!prevBar.currentTitle) {
2018
+ prevBar.currentIndex = prevBar.titles.length - 1;
2019
+ }
2020
+ if (prevBar.currentTitle) {
2021
+ this.activateWidget(prevBar.currentTitle.owner.id);
2022
+ }
2023
+ return true;
2024
+ }
2025
+
2026
+ /**
2027
+ * Return the tab bar previous to the given tab bar; return the given tab bar if there is no adjacent one.
2028
+ */
2029
+ previousTabBar(current: TabBar<Widget> | undefined = this.currentTabBar): TabBar<Widget> | undefined {
2030
+ const bars = toArray(this.mainPanel.tabBars());
2031
+ const len = bars.length;
2032
+ const ci = ArrayExt.firstIndexOf(bars, current);
2033
+ if (ci > 0) {
2034
+ return bars[ci - 1];
2035
+ } else if (ci === 0) {
2036
+ return bars[len - 1];
2037
+ } else {
2038
+ return current;
2039
+ }
2040
+ }
2041
+
2042
+ /**
2043
+ * Test whether the current widget is dirty.
2044
+ */
2045
+ canSave(): boolean {
2046
+ return this.saveResourceService.canSave(this.currentWidget);
2047
+ }
2048
+
2049
+ /**
2050
+ * Save the current widget if it is dirty.
2051
+ */
2052
+ async save(options?: SaveOptions): Promise<void> {
2053
+ await this.saveResourceService.save(this.currentWidget, options);
2054
+ }
2055
+
2056
+ /**
2057
+ * Test whether there is a dirty widget.
2058
+ */
2059
+ canSaveAll(): boolean {
2060
+ return this.tracker.widgets.some(widget => this.saveResourceService.canSave(widget));
2061
+ }
2062
+
2063
+ /**
2064
+ * Save all dirty widgets.
2065
+ */
2066
+ async saveAll(options?: SaveOptions): Promise<void> {
2067
+ for (const widget of this.widgets) {
2068
+ if (this.saveResourceService.canSaveNotSaveAs(widget)) {
2069
+ await this.saveResourceService.save(widget, options);
2070
+ }
2071
+ }
2072
+ }
2073
+
2074
+ /**
2075
+ * Returns a snapshot of all tracked widgets to allow async modifications.
2076
+ */
2077
+ get widgets(): ReadonlyArray<Widget> {
2078
+ return [...this.tracker.widgets];
2079
+ }
2080
+
2081
+ getWidgetById(id: string): Widget | undefined {
2082
+ for (const widget of this.tracker.widgets) {
2083
+ if (widget.id === id) {
2084
+ return widget;
2085
+ }
2086
+ }
2087
+ return undefined;
2088
+ }
2089
+
2090
+ canToggleMaximized(widget: Widget | undefined = this.currentWidget): boolean {
2091
+ const area = widget && this.getAreaFor(widget);
2092
+ return area === 'main' || area === 'bottom';
2093
+ }
2094
+
2095
+ toggleMaximized(widget: Widget | undefined = this.currentWidget): void {
2096
+ const area = widget && this.getAreaPanelFor(widget);
2097
+ if (area instanceof TheiaDockPanel && (area === this.mainPanel || area === this.bottomPanel)) {
2098
+ area.toggleMaximized();
2099
+ this.revealWidget(widget!.id);
2100
+ }
2101
+ }
2102
+
2103
+ }
2104
+
2105
+ /**
2106
+ * The namespace for `ApplicationShell` class statics.
2107
+ */
2108
+ export namespace ApplicationShell {
2109
+ /**
2110
+ * The areas of the application shell where widgets can reside.
2111
+ */
2112
+ export type Area = 'main' | 'top' | 'left' | 'right' | 'bottom' | 'secondaryWindow';
2113
+
2114
+ export const areaLabels: Record<Area, string> = {
2115
+ main: nls.localizeByDefault('Main'),
2116
+ top: nls.localizeByDefault('Top'),
2117
+ left: nls.localizeByDefault('Left'),
2118
+ right: nls.localizeByDefault('Right'),
2119
+ bottom: nls.localizeByDefault('Bottom'),
2120
+ secondaryWindow: nls.localize('theia/shell-area/secondary', 'Secondary Window'),
2121
+ };
2122
+
2123
+ /**
2124
+ * The _side areas_ are those shell areas that can be collapsed and expanded,
2125
+ * i.e. `left`, `right`, and `bottom`.
2126
+ */
2127
+ export function isSideArea(area?: string): area is 'left' | 'right' | 'bottom' {
2128
+ return area === 'left' || area === 'right' || area === 'bottom';
2129
+ }
2130
+
2131
+ export function isValidArea(area?: unknown): area is ApplicationShell.Area {
2132
+ const areas = ['main', 'top', 'left', 'right', 'bottom', 'secondaryWindow'];
2133
+ return typeof area === 'string' && areas.includes(area);
2134
+ }
2135
+
2136
+ /**
2137
+ * General options for the application shell. These are passed on construction and can be modified
2138
+ * through dependency injection (`ApplicationShellOptions` symbol).
2139
+ */
2140
+ export interface Options extends Widget.IOptions {
2141
+ bottomPanel: BottomPanelOptions;
2142
+ leftPanel: SidePanel.Options;
2143
+ rightPanel: SidePanel.Options;
2144
+ }
2145
+
2146
+ export interface BottomPanelOptions extends SidePanel.Options {
2147
+ }
2148
+
2149
+ /**
2150
+ * The default values for application shell options.
2151
+ */
2152
+ export const DEFAULT_OPTIONS = Object.freeze(<Options>{
2153
+ bottomPanel: Object.freeze(<BottomPanelOptions>{
2154
+ emptySize: 140,
2155
+ expandThreshold: 160,
2156
+ expandDuration: 0,
2157
+ initialSizeRatio: 0.382
2158
+ }),
2159
+ leftPanel: Object.freeze(<SidePanel.Options>{
2160
+ emptySize: 140,
2161
+ expandThreshold: 140,
2162
+ expandDuration: 0,
2163
+ initialSizeRatio: 0.191
2164
+ }),
2165
+ rightPanel: Object.freeze(<SidePanel.Options>{
2166
+ emptySize: 140,
2167
+ expandThreshold: 140,
2168
+ expandDuration: 0,
2169
+ initialSizeRatio: 0.191
2170
+ })
2171
+ });
2172
+
2173
+ /**
2174
+ * Whether a widget should be opened to the side tab bar relatively to the reference widget.
2175
+ */
2176
+ export type OpenToSideMode = 'open-to-left' | 'open-to-right';
2177
+
2178
+ export function isOpenToSideMode(mode: unknown): mode is OpenToSideMode {
2179
+ return mode === 'open-to-left' || mode === 'open-to-right';
2180
+ }
2181
+
2182
+ /**
2183
+ * Options for adding a widget to the application shell.
2184
+ */
2185
+ export interface WidgetOptions extends SidePanel.WidgetOptions {
2186
+ /**
2187
+ * The area of the application shell where the widget will reside.
2188
+ */
2189
+ area?: Area;
2190
+ /**
2191
+ * The insertion mode for adding the widget.
2192
+ *
2193
+ * The default is `'tab-after'`.
2194
+ */
2195
+ mode?: DockLayout.InsertMode | OpenToSideMode
2196
+ /**
2197
+ * The reference widget for the insert location.
2198
+ *
2199
+ * The default is `undefined`.
2200
+ */
2201
+ ref?: Widget;
2202
+ }
2203
+
2204
+ export interface CloseOptions {
2205
+ /**
2206
+ * if optional then a user will be prompted
2207
+ * if undefined then close will be canceled
2208
+ * if true then will be saved on close
2209
+ * if false then won't be saved on close
2210
+ */
2211
+ save?: boolean | undefined
2212
+ }
2213
+
2214
+ /**
2215
+ * Data to save and load the application shell layout.
2216
+ */
2217
+ export interface LayoutData {
2218
+ version?: string | ApplicationShellLayoutVersion,
2219
+ mainPanel?: DockPanel.ILayoutConfig;
2220
+ mainPanelPinned?: boolean[];
2221
+ bottomPanel?: BottomPanelLayoutData;
2222
+ leftPanel?: SidePanel.LayoutData;
2223
+ rightPanel?: SidePanel.LayoutData;
2224
+ activeWidgetId?: string;
2225
+ }
2226
+
2227
+ /**
2228
+ * Data to save and load the bottom panel layout.
2229
+ */
2230
+ export interface BottomPanelLayoutData {
2231
+ config?: DockPanel.ILayoutConfig;
2232
+ size?: number;
2233
+ expanded?: boolean;
2234
+ pinned?: boolean[];
2235
+ }
2236
+
2237
+ /**
2238
+ * Exposes widgets which activation state should be tracked by shell.
2239
+ */
2240
+ export interface TrackableWidgetProvider {
2241
+ getTrackableWidgets(): Widget[]
2242
+ readonly onDidChangeTrackableWidgets?: CommonEvent<Widget[]>
2243
+ /**
2244
+ * Make visible and focus a trackable widget for the given id.
2245
+ * If not implemented then `activate` request will be sent to a child widget directly.
2246
+ */
2247
+ activateWidget?(id: string): Widget | undefined;
2248
+ /**
2249
+ * Make visible a trackable widget for the given id.
2250
+ * If not implemented then a widget should be always visible when an owner is visible.
2251
+ */
2252
+ revealWidget?(id: string): Widget | undefined;
2253
+ }
2254
+
2255
+ export namespace TrackableWidgetProvider {
2256
+ export function is(widget: unknown): widget is TrackableWidgetProvider {
2257
+ return isObject(widget) && 'getTrackableWidgets' in widget;
2258
+ }
2259
+ }
2260
+
2261
+ }