@theia/core 1.53.0-next.55 → 1.53.0-next.64

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 (581) hide show
  1. package/README.md +6 -6
  2. package/i18n/nls.cs.json +582 -582
  3. package/i18n/nls.de.json +582 -582
  4. package/i18n/nls.es.json +582 -582
  5. package/i18n/nls.fr.json +582 -582
  6. package/i18n/nls.hu.json +582 -582
  7. package/i18n/nls.it.json +582 -582
  8. package/i18n/nls.ja.json +582 -582
  9. package/i18n/nls.json +582 -582
  10. package/i18n/nls.ko.json +582 -582
  11. package/i18n/nls.pl.json +582 -582
  12. package/i18n/nls.pt-br.json +582 -582
  13. package/i18n/nls.ru.json +582 -582
  14. package/i18n/nls.tr.json +582 -582
  15. package/i18n/nls.zh-cn.json +582 -582
  16. package/i18n/nls.zh-tw.json +582 -582
  17. package/lib/browser/catalog.json +54 -6
  18. package/lib/browser/common-frontend-contribution.js +3 -3
  19. package/lib/browser/common-styling-participants.js +166 -166
  20. package/lib/browser/progress-location-service.spec.js +7 -7
  21. package/lib/browser/storage-service.js +3 -3
  22. package/lib/browser/tree/tree.spec.js +75 -75
  23. package/lib/node/process-utils.spec.js +8 -8
  24. package/package.json +4 -4
  25. package/shared/ajv/index.d.ts +2 -2
  26. package/shared/markdown-it.d.ts +2 -2
  27. package/shared/markdown-it.js +1 -1
  28. package/shared/reflect-metadata/index.d.ts +1 -1
  29. package/shared/reflect-metadata/index.js +1 -1
  30. package/shared/vscode-languageserver-types/index.d.ts +1 -1
  31. package/src/browser/about-dialog.tsx +137 -137
  32. package/src/browser/authentication-service.ts +467 -467
  33. package/src/browser/breadcrumbs/breadcrumb-popup-container.ts +101 -101
  34. package/src/browser/breadcrumbs/breadcrumb-renderer.tsx +41 -41
  35. package/src/browser/breadcrumbs/breadcrumbs-constants.ts +79 -79
  36. package/src/browser/breadcrumbs/breadcrumbs-renderer.tsx +185 -185
  37. package/src/browser/breadcrumbs/breadcrumbs-service.ts +108 -108
  38. package/src/browser/breadcrumbs/index.ts +21 -21
  39. package/src/browser/browser-clipboard-service.ts +122 -122
  40. package/src/browser/browser.ts +239 -239
  41. package/src/browser/clipboard-service.ts +23 -23
  42. package/src/browser/color-application-contribution.ts +110 -110
  43. package/src/browser/color-registry.ts +60 -60
  44. package/src/browser/command-open-handler.ts +54 -54
  45. package/src/browser/common-frontend-contribution.ts +2680 -2680
  46. package/src/browser/common-styling-participants.ts +361 -361
  47. package/src/browser/connection-status-service.spec.ts +200 -200
  48. package/src/browser/connection-status-service.ts +216 -216
  49. package/src/browser/context-key-service.ts +142 -142
  50. package/src/browser/context-menu-renderer.ts +124 -124
  51. package/src/browser/core-preferences.ts +343 -343
  52. package/src/browser/credentials-service.ts +106 -106
  53. package/src/browser/decoration-style.ts +65 -65
  54. package/src/browser/decorations-service.ts +209 -209
  55. package/src/browser/dialogs/react-dialog.tsx +56 -56
  56. package/src/browser/dialogs.ts +534 -534
  57. package/src/browser/diff-uris.ts +117 -117
  58. package/src/browser/encoding-registry.ts +97 -97
  59. package/src/browser/endpoint.spec.ts +148 -148
  60. package/src/browser/endpoint.ts +136 -136
  61. package/src/browser/external-uri-service.ts +79 -79
  62. package/src/browser/file-icons-js.d.ts +20 -20
  63. package/src/browser/frontend-application-bindings.ts +62 -62
  64. package/src/browser/frontend-application-config-provider.spec.ts +45 -45
  65. package/src/browser/frontend-application-config-provider.ts +50 -50
  66. package/src/browser/frontend-application-contribution.ts +110 -110
  67. package/src/browser/frontend-application-module.ts +474 -474
  68. package/src/browser/frontend-application-state.ts +74 -74
  69. package/src/browser/frontend-application.ts +326 -326
  70. package/src/browser/hover-service.ts +218 -218
  71. package/src/browser/http-open-handler.ts +49 -49
  72. package/src/browser/i18n/i18n-frontend-module.ts +27 -27
  73. package/src/browser/i18n/language-quick-pick-service.ts +130 -130
  74. package/src/browser/icon-registry.ts +87 -87
  75. package/src/browser/icon-theme-contribution.ts +64 -64
  76. package/src/browser/icon-theme-service.ts +217 -217
  77. package/src/browser/icons/CollapseAll.svg +7 -7
  78. package/src/browser/icons/CollapseAll_inverse.svg +7 -7
  79. package/src/browser/icons/Refresh.svg +7 -7
  80. package/src/browser/icons/Refresh_inverse.svg +7 -7
  81. package/src/browser/icons/add-inverse.svg +4 -4
  82. package/src/browser/icons/add.svg +4 -4
  83. package/src/browser/icons/arrow-down-bright.svg +6 -6
  84. package/src/browser/icons/arrow-down-dark.svg +6 -6
  85. package/src/browser/icons/arrow-up-bright.svg +6 -6
  86. package/src/browser/icons/arrow-up-dark.svg +6 -6
  87. package/src/browser/icons/case-sensitive-dark.svg +16 -16
  88. package/src/browser/icons/case-sensitive.svg +16 -16
  89. package/src/browser/icons/chevron-right-dark.svg +5 -5
  90. package/src/browser/icons/chevron-right-light.svg +6 -6
  91. package/src/browser/icons/circle-bright.svg +7 -7
  92. package/src/browser/icons/circle-dark.svg +7 -7
  93. package/src/browser/icons/clear-search-results-dark.svg +7 -7
  94. package/src/browser/icons/clear-search-results.svg +7 -7
  95. package/src/browser/icons/close-all-bright.svg +7 -7
  96. package/src/browser/icons/close-all-dark.svg +7 -7
  97. package/src/browser/icons/close-bright.svg +7 -7
  98. package/src/browser/icons/close-dark.svg +7 -7
  99. package/src/browser/icons/collapse.svg +4 -4
  100. package/src/browser/icons/edit-json-dark.svg +6 -6
  101. package/src/browser/icons/edit-json.svg +6 -6
  102. package/src/browser/icons/expand.svg +4 -4
  103. package/src/browser/icons/loading-dark.svg +6 -6
  104. package/src/browser/icons/loading-light.svg +6 -6
  105. package/src/browser/icons/open-change-bright.svg +3 -3
  106. package/src/browser/icons/open-change-dark.svg +4 -4
  107. package/src/browser/icons/open-file-bright.svg +4 -4
  108. package/src/browser/icons/open-file-dark.svg +4 -4
  109. package/src/browser/icons/preview-bright.svg +3 -3
  110. package/src/browser/icons/preview-dark.svg +3 -3
  111. package/src/browser/icons/regex-dark.svg +10 -10
  112. package/src/browser/icons/regex.svg +10 -10
  113. package/src/browser/icons/remove-all-inverse.svg +4 -4
  114. package/src/browser/icons/remove-all.svg +4 -4
  115. package/src/browser/icons/replace-all-inverse.svg +13 -13
  116. package/src/browser/icons/replace-all.svg +13 -13
  117. package/src/browser/icons/replace-inverse.svg +15 -15
  118. package/src/browser/icons/replace.svg +15 -15
  119. package/src/browser/icons/whole-word-dark.svg +19 -19
  120. package/src/browser/icons/whole-word.svg +19 -19
  121. package/src/browser/index.ts +50 -50
  122. package/src/browser/json-schema-store.ts +118 -118
  123. package/src/browser/keybinding.spec.ts +554 -554
  124. package/src/browser/keybinding.ts +759 -759
  125. package/src/browser/keyboard/browser-keyboard-frontend-contribution.ts +108 -108
  126. package/src/browser/keyboard/browser-keyboard-layout-provider.spec.ts +171 -171
  127. package/src/browser/keyboard/browser-keyboard-layout-provider.ts +469 -469
  128. package/src/browser/keyboard/browser-keyboard-module.ts +30 -30
  129. package/src/browser/keyboard/index.ts +20 -20
  130. package/src/browser/keyboard/keyboard-layout-service.spec.ts +121 -121
  131. package/src/browser/keyboard/keyboard-layout-service.ts +455 -455
  132. package/src/browser/keyboard/keys.spec.ts +258 -258
  133. package/src/browser/keyboard/keys.ts +20 -20
  134. package/src/browser/keys.ts +21 -21
  135. package/src/browser/label-parser.spec.ts +165 -165
  136. package/src/browser/label-parser.ts +108 -108
  137. package/src/browser/label-provider.spec.ts +62 -62
  138. package/src/browser/label-provider.ts +385 -385
  139. package/src/browser/language-icon-provider.ts +55 -55
  140. package/src/browser/language-service.ts +77 -77
  141. package/src/browser/logger-frontend-module.ts +65 -65
  142. package/src/browser/markdown-rendering/markdown-renderer.ts +98 -98
  143. package/src/browser/menu/browser-context-menu-renderer.ts +48 -48
  144. package/src/browser/menu/browser-menu-module.ts +28 -28
  145. package/src/browser/menu/browser-menu-plugin.ts +491 -491
  146. package/src/browser/menu/context-menu-context.ts +41 -41
  147. package/src/browser/messaging/connection-source.ts +26 -26
  148. package/src/browser/messaging/frontend-id-provider.ts +37 -37
  149. package/src/browser/messaging/index.ts +18 -18
  150. package/src/browser/messaging/messaging-frontend-module.ts +41 -41
  151. package/src/browser/messaging/service-connection-provider.ts +140 -140
  152. package/src/browser/messaging/ws-connection-provider.ts +49 -49
  153. package/src/browser/messaging/ws-connection-source.ts +230 -230
  154. package/src/browser/mime-service.ts +30 -30
  155. package/src/browser/navigatable-types.ts +81 -81
  156. package/src/browser/navigatable.ts +39 -39
  157. package/src/browser/open-with-service.ts +140 -140
  158. package/src/browser/opener-service.spec.ts +49 -49
  159. package/src/browser/opener-service.ts +169 -169
  160. package/src/browser/performance/frontend-stopwatch.ts +65 -65
  161. package/src/browser/performance/index.ts +18 -18
  162. package/src/browser/performance/measurement-frontend-bindings.ts +31 -31
  163. package/src/browser/preferences/index.ts +23 -23
  164. package/src/browser/preferences/injectable-preference-proxy.ts +283 -283
  165. package/src/browser/preferences/preference-configurations.ts +82 -82
  166. package/src/browser/preferences/preference-contribution.ts +436 -436
  167. package/src/browser/preferences/preference-language-override-service.ts +111 -111
  168. package/src/browser/preferences/preference-provider.spec.ts +36 -36
  169. package/src/browser/preferences/preference-provider.ts +277 -277
  170. package/src/browser/preferences/preference-proxy.spec.ts +367 -367
  171. package/src/browser/preferences/preference-proxy.ts +367 -367
  172. package/src/browser/preferences/preference-schema-provider.spec.ts +130 -130
  173. package/src/browser/preferences/preference-scope.ts +18 -18
  174. package/src/browser/preferences/preference-service.spec.ts +613 -613
  175. package/src/browser/preferences/preference-service.ts +594 -594
  176. package/src/browser/preferences/preference-validation-service.spec.ts +334 -334
  177. package/src/browser/preferences/preference-validation-service.ts +358 -358
  178. package/src/browser/preferences/test/index.ts +19 -19
  179. package/src/browser/preferences/test/mock-preference-provider.ts +50 -50
  180. package/src/browser/preferences/test/mock-preference-proxy.ts +48 -48
  181. package/src/browser/preferences/test/mock-preference-service.ts +63 -63
  182. package/src/browser/preload/i18n-preload-contribution.ts +50 -50
  183. package/src/browser/preload/os-preload-contribution.ts +37 -37
  184. package/src/browser/preload/preload-module.ts +45 -45
  185. package/src/browser/preload/preloader.ts +37 -37
  186. package/src/browser/preload/theme-preload-contribution.ts +31 -31
  187. package/src/browser/progress-bar-factory.ts +29 -29
  188. package/src/browser/progress-bar.ts +76 -76
  189. package/src/browser/progress-client.ts +53 -53
  190. package/src/browser/progress-location-service.spec.ts +50 -50
  191. package/src/browser/progress-location-service.ts +96 -96
  192. package/src/browser/progress-status-bar-item.ts +83 -83
  193. package/src/browser/quick-input/index.ts +23 -23
  194. package/src/browser/quick-input/quick-access.ts +75 -75
  195. package/src/browser/quick-input/quick-command-frontend-contribution.ts +89 -89
  196. package/src/browser/quick-input/quick-command-service.ts +246 -246
  197. package/src/browser/quick-input/quick-help-service.ts +87 -87
  198. package/src/browser/quick-input/quick-input-frontend-contribution.ts +33 -33
  199. package/src/browser/quick-input/quick-input-service.spec.ts +176 -176
  200. package/src/browser/quick-input/quick-input-service.ts +17 -17
  201. package/src/browser/quick-input/quick-pick-service-impl.ts +69 -69
  202. package/src/browser/quick-input/quick-view-service.ts +83 -83
  203. package/src/browser/request/browser-request-module.ts +23 -23
  204. package/src/browser/request/browser-request-service.ts +172 -172
  205. package/src/browser/resource-context-key.ts +77 -77
  206. package/src/browser/saveable-service.ts +332 -332
  207. package/src/browser/saveable.ts +395 -395
  208. package/src/browser/secondary-window-handler.ts +211 -211
  209. package/src/browser/shell/additional-views-menu-widget.tsx +71 -71
  210. package/src/browser/shell/application-shell-mouse-tracker.ts +103 -103
  211. package/src/browser/shell/application-shell.ts +2271 -2271
  212. package/src/browser/shell/current-widget-command-adapter.ts +57 -57
  213. package/src/browser/shell/index.ts +23 -23
  214. package/src/browser/shell/shell-layout-restorer.ts +399 -399
  215. package/src/browser/shell/side-panel-handler.ts +794 -794
  216. package/src/browser/shell/side-panel-toolbar.ts +111 -111
  217. package/src/browser/shell/sidebar-bottom-menu-widget.tsx +39 -39
  218. package/src/browser/shell/sidebar-menu-widget.tsx +183 -183
  219. package/src/browser/shell/sidebar-top-menu-widget.tsx +26 -26
  220. package/src/browser/shell/split-panels.ts +191 -191
  221. package/src/browser/shell/tab-bar-decorator.ts +106 -106
  222. package/src/browser/shell/tab-bar-toolbar/index.ts +19 -19
  223. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar-menu-adapters.ts +31 -31
  224. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar-registry.ts +242 -242
  225. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar-types.ts +149 -149
  226. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar.spec.ts +62 -62
  227. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar.tsx +443 -443
  228. package/src/browser/shell/tab-bars.spec.ts +63 -63
  229. package/src/browser/shell/tab-bars.ts +1468 -1468
  230. package/src/browser/shell/theia-dock-panel.ts +265 -265
  231. package/src/browser/shell/view-column-service.ts +125 -125
  232. package/src/browser/shell/view-contribution.ts +178 -178
  233. package/src/browser/source-tree/index.ts +19 -19
  234. package/src/browser/source-tree/source-tree-widget.tsx +107 -107
  235. package/src/browser/source-tree/source-tree.ts +146 -146
  236. package/src/browser/source-tree/tree-source.ts +73 -73
  237. package/src/browser/status-bar/index.ts +29 -29
  238. package/src/browser/status-bar/status-bar-types.ts +97 -97
  239. package/src/browser/status-bar/status-bar-view-model.ts +209 -209
  240. package/src/browser/status-bar/status-bar.tsx +189 -189
  241. package/src/browser/storage-service.spec.ts +76 -76
  242. package/src/browser/storage-service.ts +129 -129
  243. package/src/browser/style/about.css +36 -36
  244. package/src/browser/style/alert-messages.css +62 -62
  245. package/src/browser/style/ansi.css +88 -88
  246. package/src/browser/style/breadcrumbs.css +130 -130
  247. package/src/browser/style/dialog.css +126 -126
  248. package/src/browser/style/dockpanel.css +76 -76
  249. package/src/browser/style/hover-service.css +101 -101
  250. package/src/browser/style/icons.css +61 -61
  251. package/src/browser/style/index.css +353 -353
  252. package/src/browser/style/materialcolors.css +278 -278
  253. package/src/browser/style/menus.css +230 -230
  254. package/src/browser/style/notification.css +39 -39
  255. package/src/browser/style/os.css +87 -87
  256. package/src/browser/style/progress-bar.css +43 -43
  257. package/src/browser/style/quick-title-bar.css +45 -45
  258. package/src/browser/style/scrollbars.css +172 -172
  259. package/src/browser/style/search-box.css +123 -123
  260. package/src/browser/style/select-component.css +107 -107
  261. package/src/browser/style/sidepanel.css +367 -367
  262. package/src/browser/style/split-widget.css +38 -38
  263. package/src/browser/style/status-bar.css +127 -127
  264. package/src/browser/style/tabs.css +647 -647
  265. package/src/browser/style/tooltip.css +28 -28
  266. package/src/browser/style/tree-decorators.css +81 -81
  267. package/src/browser/style/tree.css +232 -232
  268. package/src/browser/style/view-container.css +187 -187
  269. package/src/browser/style/widget.css +19 -19
  270. package/src/browser/styling-service.ts +96 -96
  271. package/src/browser/supported-encodings.ts +262 -262
  272. package/src/browser/test/jsdom.ts +69 -69
  273. package/src/browser/test/mock-connection-status-service.ts +33 -33
  274. package/src/browser/test/mock-env-variables-server.ts +47 -47
  275. package/src/browser/test/mock-opener-service.ts +34 -34
  276. package/src/browser/test/mock-storage-service.ts +49 -49
  277. package/src/browser/theming.ts +206 -206
  278. package/src/browser/tooltip-service.tsx +96 -96
  279. package/src/browser/tree/fuzzy-search.spec.ts +99 -99
  280. package/src/browser/tree/fuzzy-search.ts +136 -136
  281. package/src/browser/tree/index.ts +29 -29
  282. package/src/browser/tree/search-box-debounce.ts +96 -96
  283. package/src/browser/tree/search-box.ts +355 -355
  284. package/src/browser/tree/test/mock-selectable-tree-model.ts +109 -109
  285. package/src/browser/tree/test/mock-tree-model.ts +136 -136
  286. package/src/browser/tree/test/tree-test-container.ts +50 -50
  287. package/src/browser/tree/tree-compression/compressed-tree-expansion-service.ts +46 -46
  288. package/src/browser/tree/tree-compression/compressed-tree-model.ts +88 -88
  289. package/src/browser/tree/tree-compression/compressed-tree-widget.tsx +203 -203
  290. package/src/browser/tree/tree-compression/index.ts +20 -20
  291. package/src/browser/tree/tree-compression/tree-compression-service.ts +125 -125
  292. package/src/browser/tree/tree-compression/tree-compression.css +28 -28
  293. package/src/browser/tree/tree-consistency.spec.ts +105 -105
  294. package/src/browser/tree/tree-container.spec.ts +45 -45
  295. package/src/browser/tree/tree-container.ts +155 -155
  296. package/src/browser/tree/tree-decorator.spec.ts +162 -162
  297. package/src/browser/tree/tree-decorator.ts +238 -238
  298. package/src/browser/tree/tree-expansion.spec.ts +173 -173
  299. package/src/browser/tree/tree-expansion.ts +165 -165
  300. package/src/browser/tree/tree-focus-service.ts +55 -55
  301. package/src/browser/tree/tree-iterator.spec.ts +170 -170
  302. package/src/browser/tree/tree-iterator.ts +256 -256
  303. package/src/browser/tree/tree-label-provider.ts +40 -40
  304. package/src/browser/tree/tree-model.ts +562 -562
  305. package/src/browser/tree/tree-navigation.ts +58 -58
  306. package/src/browser/tree/tree-preference.ts +50 -50
  307. package/src/browser/tree/tree-search.ts +128 -128
  308. package/src/browser/tree/tree-selectable.spec.ts +152 -152
  309. package/src/browser/tree/tree-selection-impl.ts +176 -176
  310. package/src/browser/tree/tree-selection-state.spec.ts +462 -462
  311. package/src/browser/tree/tree-selection-state.ts +245 -245
  312. package/src/browser/tree/tree-selection.ts +159 -159
  313. package/src/browser/tree/tree-view-welcome-widget.tsx +263 -263
  314. package/src/browser/tree/tree-widget-selection.ts +45 -45
  315. package/src/browser/tree/tree-widget.tsx +1591 -1591
  316. package/src/browser/tree/tree.spec.ts +241 -241
  317. package/src/browser/tree/tree.ts +425 -425
  318. package/src/browser/undo-redo-handler.ts +85 -85
  319. package/src/browser/user-working-directory-provider.ts +77 -77
  320. package/src/browser/view-container.ts +1640 -1640
  321. package/src/browser/widget-decoration.ts +358 -358
  322. package/src/browser/widget-manager.spec.ts +102 -102
  323. package/src/browser/widget-manager.ts +318 -318
  324. package/src/browser/widget-open-handler.ts +168 -168
  325. package/src/browser/widgets/alert-message.tsx +56 -56
  326. package/src/browser/widgets/enhanced-preview-widget.ts +27 -27
  327. package/src/browser/widgets/extractable-widget.ts +33 -33
  328. package/src/browser/widgets/index.ts +21 -21
  329. package/src/browser/widgets/previewable-widget.ts +31 -31
  330. package/src/browser/widgets/react-renderer.tsx +53 -53
  331. package/src/browser/widgets/react-widget.tsx +51 -51
  332. package/src/browser/widgets/select-component.tsx +367 -367
  333. package/src/browser/widgets/split-widget.ts +163 -163
  334. package/src/browser/widgets/widget.ts +406 -406
  335. package/src/browser/window/browser-window-module.ts +32 -32
  336. package/src/browser/window/default-secondary-window-service.ts +189 -189
  337. package/src/browser/window/default-window-service.spec.ts +78 -78
  338. package/src/browser/window/default-window-service.ts +171 -171
  339. package/src/browser/window/secondary-window-service.ts +39 -39
  340. package/src/browser/window/test/mock-window-service.ts +29 -29
  341. package/src/browser/window/window-service.ts +78 -78
  342. package/src/browser/window/window-title-service.ts +107 -107
  343. package/src/browser/window/window-title-updater.ts +95 -95
  344. package/src/browser/window-contribution.ts +64 -64
  345. package/src/browser-only/frontend-only-application-module.ts +116 -116
  346. package/src/browser-only/i18n/i18n-frontend-only-module.ts +37 -37
  347. package/src/browser-only/logger-frontend-only-module.ts +63 -63
  348. package/src/browser-only/messaging/frontend-only-service-connection-provider.ts +39 -39
  349. package/src/browser-only/messaging/messaging-frontend-only-module.ts +42 -42
  350. package/src/browser-only/preload/frontend-only-preload-module.ts +49 -49
  351. package/src/common/accessibility.ts +33 -33
  352. package/src/common/application-error.spec.ts +27 -27
  353. package/src/common/application-error.ts +76 -76
  354. package/src/common/application-protocol.ts +42 -42
  355. package/src/common/array-utils.ts +129 -129
  356. package/src/common/buffer.ts +228 -228
  357. package/src/common/cancellation.ts +163 -163
  358. package/src/common/char-code.ts +438 -438
  359. package/src/common/collections.ts +125 -125
  360. package/src/common/color.ts +103 -103
  361. package/src/common/command.spec.ts +208 -208
  362. package/src/common/command.ts +489 -489
  363. package/src/common/contribution-filter/contribution-filter-registry.ts +79 -79
  364. package/src/common/contribution-filter/contribution-filter.ts +64 -64
  365. package/src/common/contribution-filter/filter.ts +23 -23
  366. package/src/common/contribution-filter/index.ts +19 -19
  367. package/src/common/contribution-provider.ts +96 -96
  368. package/src/common/disposable.spec.ts +94 -94
  369. package/src/common/disposable.ts +188 -188
  370. package/src/common/encoding-service.ts +380 -380
  371. package/src/common/encodings.ts +24 -24
  372. package/src/common/env-variables/env-variables-protocol.ts +38 -38
  373. package/src/common/env-variables/index.ts +17 -17
  374. package/src/common/event.spec.ts +32 -32
  375. package/src/common/event.ts +493 -493
  376. package/src/common/file-uri.ts +61 -61
  377. package/src/common/frontend-application-state.ts +38 -38
  378. package/src/common/glob.ts +741 -741
  379. package/src/common/hash.ts +85 -85
  380. package/src/common/i18n/localization-server.ts +25 -25
  381. package/src/common/i18n/localization.ts +80 -80
  382. package/src/common/i18n/nls.metadata.json +34112 -34112
  383. package/src/common/index.ts +51 -51
  384. package/src/common/json-schema.ts +108 -108
  385. package/src/common/key-store.ts +26 -26
  386. package/src/common/keybinding.ts +152 -152
  387. package/src/common/keyboard/keyboard-layout-provider.ts +51 -51
  388. package/src/common/keys.ts +694 -694
  389. package/src/common/label-protocol.ts +35 -35
  390. package/src/common/logger-protocol.ts +119 -119
  391. package/src/common/logger-watcher.ts +48 -48
  392. package/src/common/logger.spec.ts +46 -46
  393. package/src/common/logger.ts +389 -389
  394. package/src/common/lsp-types.ts +34 -34
  395. package/src/common/markdown-rendering/icon-utilities.ts +30 -30
  396. package/src/common/markdown-rendering/index.ts +18 -18
  397. package/src/common/markdown-rendering/markdown-string.ts +152 -152
  398. package/src/common/menu/action-menu-node.ts +65 -65
  399. package/src/common/menu/composite-menu-node.spec.ts +67 -67
  400. package/src/common/menu/composite-menu-node.ts +114 -114
  401. package/src/common/menu/index.ts +21 -21
  402. package/src/common/menu/menu-adapter.ts +103 -103
  403. package/src/common/menu/menu-model-registry.ts +374 -374
  404. package/src/common/menu/menu-types.ts +220 -220
  405. package/src/common/menu/menu.spec.ts +101 -101
  406. package/src/common/message-rpc/channel.spec.ts +88 -88
  407. package/src/common/message-rpc/channel.ts +300 -300
  408. package/src/common/message-rpc/index.ts +22 -22
  409. package/src/common/message-rpc/message-buffer.ts +105 -105
  410. package/src/common/message-rpc/msg-pack-extension-manager.ts +70 -70
  411. package/src/common/message-rpc/rpc-message-encoder.spec.ts +65 -65
  412. package/src/common/message-rpc/rpc-message-encoder.ts +190 -190
  413. package/src/common/message-rpc/rpc-protocol.ts +255 -255
  414. package/src/common/message-rpc/uint8-array-message-buffer.spec.ts +41 -41
  415. package/src/common/message-rpc/uint8-array-message-buffer.ts +213 -213
  416. package/src/common/message-service-protocol.ts +148 -148
  417. package/src/common/message-service.ts +226 -226
  418. package/src/common/messaging/connection-error-handler.ts +73 -73
  419. package/src/common/messaging/connection-management.ts +43 -43
  420. package/src/common/messaging/handler.ts +26 -26
  421. package/src/common/messaging/index.ts +19 -19
  422. package/src/common/messaging/proxy-factory.spec.ts +108 -108
  423. package/src/common/messaging/proxy-factory.ts +336 -336
  424. package/src/common/messaging/socket-write-buffer.ts +52 -52
  425. package/src/common/messaging/web-socket-channel.ts +76 -76
  426. package/src/common/nls.ts +151 -151
  427. package/src/common/numbers.ts +21 -21
  428. package/src/common/objects.spec.ts +112 -112
  429. package/src/common/objects.ts +123 -123
  430. package/src/common/os.ts +82 -82
  431. package/src/common/path.spec.ts +415 -415
  432. package/src/common/path.ts +334 -334
  433. package/src/common/paths.ts +250 -250
  434. package/src/common/performance/index.ts +19 -19
  435. package/src/common/performance/measurement-protocol.ts +104 -104
  436. package/src/common/performance/measurement.ts +130 -130
  437. package/src/common/performance/stopwatch.ts +183 -183
  438. package/src/common/preferences/preference-schema.ts +101 -101
  439. package/src/common/preferences/preference-scope.spec.ts +48 -48
  440. package/src/common/preferences/preference-scope.ts +68 -68
  441. package/src/common/prioritizeable.ts +58 -58
  442. package/src/common/progress-service-protocol.ts +35 -35
  443. package/src/common/progress-service.ts +82 -82
  444. package/src/common/promise-util.spec.ts +102 -102
  445. package/src/common/promise-util.ts +143 -143
  446. package/src/common/quick-pick-service.ts +353 -353
  447. package/src/common/reference.spec.ts +145 -145
  448. package/src/common/reference.ts +230 -230
  449. package/src/common/resource.ts +430 -430
  450. package/src/common/selection-command-handler.ts +101 -101
  451. package/src/common/selection-service.spec.ts +43 -43
  452. package/src/common/selection-service.ts +49 -49
  453. package/src/common/selection.ts +50 -50
  454. package/src/common/severity.ts +111 -111
  455. package/src/common/stream.ts +718 -718
  456. package/src/common/strings.ts +231 -231
  457. package/src/common/telemetry.ts +45 -45
  458. package/src/common/ternary-search-tree.ts +417 -417
  459. package/src/common/test/expect.ts +34 -34
  460. package/src/common/test/mock-logger.ts +118 -118
  461. package/src/common/test/mock-menu.ts +35 -35
  462. package/src/common/test/mock-resource-provider.ts +33 -33
  463. package/src/common/theme.ts +68 -68
  464. package/src/common/types.spec.ts +86 -86
  465. package/src/common/types.ts +140 -140
  466. package/src/common/uri-command-handler.spec.ts +90 -90
  467. package/src/common/uri-command-handler.ts +148 -148
  468. package/src/common/uri.spec.ts +278 -278
  469. package/src/common/uri.ts +279 -279
  470. package/src/common/uuid.ts +45 -45
  471. package/src/common/version.ts +17 -17
  472. package/src/common/view-column.ts +33 -33
  473. package/src/common/window.ts +34 -34
  474. package/src/electron-browser/electron-clipboard-service.ts +32 -32
  475. package/src/electron-browser/electron-uri-handler.ts +42 -42
  476. package/src/electron-browser/keyboard/electron-keyboard-layout-change-notifier.ts +39 -39
  477. package/src/electron-browser/keyboard/electron-keyboard-module.ts +28 -28
  478. package/src/electron-browser/menu/electron-context-menu-renderer.ts +122 -122
  479. package/src/electron-browser/menu/electron-main-menu-factory.ts +339 -339
  480. package/src/electron-browser/menu/electron-menu-contribution.ts +506 -506
  481. package/src/electron-browser/menu/electron-menu-module.ts +40 -40
  482. package/src/electron-browser/menu/electron-menu-style.css +110 -110
  483. package/src/electron-browser/messaging/electron-frontend-id-provider.ts +25 -25
  484. package/src/electron-browser/messaging/electron-ipc-connection-source.ts +65 -65
  485. package/src/electron-browser/messaging/electron-local-ws-connection-source.ts +45 -45
  486. package/src/electron-browser/messaging/electron-messaging-frontend-module.ts +78 -78
  487. package/src/electron-browser/messaging/electron-ws-connection-source.ts +38 -38
  488. package/src/electron-browser/preload.ts +264 -264
  489. package/src/electron-browser/request/electron-browser-request-module.ts +26 -26
  490. package/src/electron-browser/token/electron-token-frontend-module.ts +22 -22
  491. package/src/electron-browser/window/electron-frontend-application-state.ts +26 -26
  492. package/src/electron-browser/window/electron-secondary-window-service.ts +35 -35
  493. package/src/electron-browser/window/electron-window-module.ts +48 -48
  494. package/src/electron-browser/window/electron-window-preferences.ts +76 -76
  495. package/src/electron-browser/window/electron-window-service.ts +109 -109
  496. package/src/electron-browser/window/external-app-open-handler.ts +42 -42
  497. package/src/electron-common/electron-api.ts +157 -157
  498. package/src/electron-common/electron-main-window-service.ts +24 -24
  499. package/src/electron-common/electron-token.ts +27 -27
  500. package/src/electron-main/electron-api-main.ts +373 -373
  501. package/src/electron-main/electron-main-application-module.ts +65 -65
  502. package/src/electron-main/electron-main-application.ts +860 -860
  503. package/src/electron-main/electron-main-constants.ts +23 -23
  504. package/src/electron-main/electron-main-window-service-impl.ts +44 -44
  505. package/src/electron-main/electron-security-token-service.ts +36 -36
  506. package/src/electron-main/event-utils.ts +36 -36
  507. package/src/electron-main/messaging/electron-connection-handler.ts +21 -21
  508. package/src/electron-main/messaging/electron-messaging-contribution.ts +143 -143
  509. package/src/electron-main/messaging/electron-messaging-service.ts +35 -35
  510. package/src/electron-main/theia-electron-window.ts +219 -219
  511. package/src/electron-node/cli/electron-backend-cli-module.ts +24 -24
  512. package/src/electron-node/cli/electron-cli-contribution.ts +35 -35
  513. package/src/electron-node/hosting/electron-backend-hosting-module.ts +24 -24
  514. package/src/electron-node/hosting/electron-ws-origin-validator.ts +37 -37
  515. package/src/electron-node/keyboard/electron-backend-keyboard-module.ts +30 -30
  516. package/src/electron-node/keyboard/electron-keyboard-layout-provider.ts +35 -35
  517. package/src/electron-node/request/electron-backend-request-module.ts +23 -23
  518. package/src/electron-node/request/electron-backend-request-service.ts +78 -78
  519. package/src/electron-node/token/electron-token-backend-contribution.ts +48 -48
  520. package/src/electron-node/token/electron-token-backend-module.ts +28 -28
  521. package/src/electron-node/token/electron-token-validator.ts +93 -93
  522. package/src/node/application-server.ts +59 -59
  523. package/src/node/backend-application-config-provider.spec.ts +29 -29
  524. package/src/node/backend-application-config-provider.ts +48 -48
  525. package/src/node/backend-application-module.ts +139 -139
  526. package/src/node/backend-application.ts +374 -374
  527. package/src/node/cli.spec.ts +94 -94
  528. package/src/node/cli.ts +63 -63
  529. package/src/node/console-logger-server.spec.ts +59 -59
  530. package/src/node/console-logger-server.ts +76 -76
  531. package/src/node/debug.ts +30 -30
  532. package/src/node/dynamic-require.ts +56 -56
  533. package/src/node/env-variables/env-variables-server.ts +123 -123
  534. package/src/node/env-variables/index.ts +17 -17
  535. package/src/node/environment-utils.spec.ts +92 -92
  536. package/src/node/environment-utils.ts +66 -66
  537. package/src/node/file-uri.spec.ts +76 -76
  538. package/src/node/filesystem-locking.ts +77 -77
  539. package/src/node/hosting/backend-application-hosts.ts +60 -60
  540. package/src/node/hosting/backend-hosting-module.ts +26 -26
  541. package/src/node/hosting/ws-origin-validator.ts +36 -36
  542. package/src/node/i18n/i18n-backend-module.ts +42 -42
  543. package/src/node/i18n/localization-contribution.ts +112 -112
  544. package/src/node/i18n/localization-provider.ts +125 -125
  545. package/src/node/i18n/localization-server.ts +52 -52
  546. package/src/node/i18n/theia-localization-contribution.ts +40 -40
  547. package/src/node/index.ts +22 -22
  548. package/src/node/key-store-server.ts +162 -162
  549. package/src/node/logger-backend-module.ts +88 -88
  550. package/src/node/logger-cli-contribution.spec.ts +245 -245
  551. package/src/node/logger-cli-contribution.ts +168 -168
  552. package/src/node/main.ts +33 -33
  553. package/src/node/messaging/binary-message-pipe.ts +168 -168
  554. package/src/node/messaging/connection-container-module.ts +96 -96
  555. package/src/node/messaging/default-messaging-service.ts +129 -129
  556. package/src/node/messaging/frontend-connection-service.ts +24 -24
  557. package/src/node/messaging/index.ts +19 -19
  558. package/src/node/messaging/ipc-bootstrap.ts +27 -27
  559. package/src/node/messaging/ipc-channel.ts +77 -77
  560. package/src/node/messaging/ipc-connection-provider.ts +107 -107
  561. package/src/node/messaging/ipc-protocol.ts +76 -76
  562. package/src/node/messaging/messaging-backend-module.ts +52 -52
  563. package/src/node/messaging/messaging-listeners.ts +52 -52
  564. package/src/node/messaging/messaging-service.ts +46 -46
  565. package/src/node/messaging/test/test-web-socket-channel.ts +61 -61
  566. package/src/node/messaging/websocket-endpoint.ts +79 -79
  567. package/src/node/messaging/websocket-frontend-connection-service.ts +186 -186
  568. package/src/node/os-backend-provider.ts +25 -25
  569. package/src/node/performance/index.ts +18 -18
  570. package/src/node/performance/measurement-backend-bindings.ts +35 -35
  571. package/src/node/performance/node-stopwatch.ts +40 -40
  572. package/src/node/process-utils.spec.ts +48 -48
  573. package/src/node/process-utils.ts +102 -102
  574. package/src/node/remote/backend-remote-service.ts +25 -25
  575. package/src/node/remote/remote-cli-contribution.ts +34 -34
  576. package/src/node/remote/remote-copy-contribution.ts +45 -45
  577. package/src/node/request/backend-request-facade.ts +39 -39
  578. package/src/node/request/backend-request-module.ts +25 -25
  579. package/src/node/request/proxy-cli-contribution.ts +65 -65
  580. package/src/node/ws-request-validators.ts +56 -56
  581. package/src/typings/native-keymap.d.ts +108 -108
@@ -1,794 +1,794 @@
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 } from 'inversify';
18
- import { find, map, toArray, some } from '@phosphor/algorithm';
19
- import { TabBar, Widget, DockPanel, Title, Panel, BoxPanel, BoxLayout, SplitPanel, PanelLayout } from '@phosphor/widgets';
20
- import { MimeData } from '@phosphor/coreutils';
21
- import { Drag } from '@phosphor/dragdrop';
22
- import { AttachedProperty } from '@phosphor/properties';
23
- import { TabBarRendererFactory, TabBarRenderer, SHELL_TABBAR_CONTEXT_MENU, SideTabBar } from './tab-bars';
24
- import { SidebarMenuWidget, SidebarMenu, SidebarBottomMenuWidgetFactory, SidebarTopMenuWidgetFactory } from './sidebar-menu-widget';
25
- import { SplitPositionHandler, SplitPositionOptions } from './split-panels';
26
- import { animationFrame } from '../browser';
27
- import { FrontendApplicationStateService } from '../frontend-application-state';
28
- import { TheiaDockPanel } from './theia-dock-panel';
29
- import { SidePanelToolbar } from './side-panel-toolbar';
30
- import { TabBarToolbarRegistry, TabBarToolbarFactory, TabBarToolbar } from './tab-bar-toolbar';
31
- import { DisposableCollection, Disposable } from '../../common/disposable';
32
- import { ContextMenuRenderer } from '../context-menu-renderer';
33
- import { MenuPath } from '../../common/menu';
34
- import { SidebarBottomMenuWidget } from './sidebar-bottom-menu-widget';
35
- import { SidebarTopMenuWidget } from './sidebar-top-menu-widget';
36
- import { PINNED_CLASS } from '../widgets';
37
- import { AdditionalViewsMenuWidget, AdditionalViewsMenuWidgetFactory } from './additional-views-menu-widget';
38
-
39
- /** The class name added to the left and right area panels. */
40
- export const LEFT_RIGHT_AREA_CLASS = 'theia-app-sides';
41
-
42
- /** The class name added to collapsed side panels. */
43
- const COLLAPSED_CLASS = 'theia-mod-collapsed';
44
-
45
- export const SidePanelHandlerFactory = Symbol('SidePanelHandlerFactory');
46
-
47
- export const SIDE_PANEL_TOOLBAR_CONTEXT_MENU: MenuPath = ['SIDE_PANEL_TOOLBAR_CONTEXT_MENU'];
48
-
49
- /**
50
- * A class which manages a dock panel and a related side bar. This is used for the left and right
51
- * panel of the application shell.
52
- */
53
- @injectable()
54
- export class SidePanelHandler {
55
-
56
- /**
57
- * A property that can be attached to widgets in order to determine the insertion index
58
- * of their title in the tab bar.
59
- */
60
- protected static readonly rankProperty = new AttachedProperty<Widget, number | undefined>({
61
- name: 'sidePanelRank',
62
- create: () => undefined
63
- });
64
-
65
- /**
66
- * The tab bar displays the titles of the widgets in the side panel. Visibility of the widgets
67
- * is controlled entirely through tab selection: a widget is revealed by setting the `currentTitle`
68
- * accordingly in the tab bar, and the panel is hidden by setting that property to `null`. The
69
- * tab bar itself remains visible as long as there is at least one widget.
70
- */
71
- tabBar: SideTabBar;
72
- /**
73
- * Conditional menu placed below the tabBar. Manages overflowing/hidden tabs.
74
- * Is only visible if there are overflowing tabs.
75
- */
76
- additionalViewsMenu: AdditionalViewsMenuWidget;
77
- /**
78
- * The menu placed on the sidebar top.
79
- * Displayed as icons.
80
- * Open menus when on clicks.
81
- */
82
- topMenu: SidebarMenuWidget;
83
- /**
84
- * The menu placed on the sidebar bottom.
85
- * Displayed as icons.
86
- * Open menus when on clicks.
87
- */
88
- bottomMenu: SidebarMenuWidget;
89
- /**
90
- * A tool bar, which displays a title and widget specific command buttons.
91
- */
92
- toolBar: SidePanelToolbar;
93
- /**
94
- * The widget container is a dock panel in `single-document` mode, which means that the panel
95
- * cannot be split.
96
- */
97
- dockPanel: TheiaDockPanel;
98
- /**
99
- * The panel that contains the tab bar and the dock panel. This one is hidden whenever the dock
100
- * panel is empty.
101
- */
102
- container: Panel;
103
- /**
104
- * The current state of the side panel.
105
- */
106
- readonly state: SidePanel.State = {
107
- empty: true,
108
- expansion: SidePanel.ExpansionState.collapsed,
109
- pendingUpdate: Promise.resolve()
110
- };
111
-
112
- /**
113
- * The shell area where the panel is placed. This property should not be modified directly, but
114
- * only by calling `create`.
115
- */
116
- protected side: 'left' | 'right';
117
- /**
118
- * Options that control the behavior of the side panel.
119
- */
120
- protected options: SidePanel.Options;
121
-
122
- @inject(TabBarToolbarRegistry) protected tabBarToolBarRegistry: TabBarToolbarRegistry;
123
- @inject(TabBarToolbarFactory) protected tabBarToolBarFactory: () => TabBarToolbar;
124
- @inject(TabBarRendererFactory) protected tabBarRendererFactory: () => TabBarRenderer;
125
- @inject(SidebarTopMenuWidgetFactory) protected sidebarTopWidgetFactory: () => SidebarTopMenuWidget;
126
- @inject(SidebarBottomMenuWidgetFactory) protected sidebarBottomWidgetFactory: () => SidebarBottomMenuWidget;
127
- @inject(AdditionalViewsMenuWidgetFactory) protected additionalViewsMenuFactory: AdditionalViewsMenuWidgetFactory;
128
- @inject(SplitPositionHandler) protected splitPositionHandler: SplitPositionHandler;
129
- @inject(FrontendApplicationStateService) protected readonly applicationStateService: FrontendApplicationStateService;
130
- @inject(TheiaDockPanel.Factory) protected readonly dockPanelFactory: TheiaDockPanel.Factory;
131
-
132
- @inject(ContextMenuRenderer)
133
- protected readonly contextMenuRenderer: ContextMenuRenderer;
134
-
135
- /**
136
- * Create the side bar and dock panel widgets.
137
- */
138
- create(side: 'left' | 'right', options: SidePanel.Options): void {
139
- this.side = side;
140
- this.options = options;
141
- this.topMenu = this.createSidebarTopMenu();
142
- this.tabBar = this.createSideBar();
143
- this.additionalViewsMenu = this.createAdditionalViewsWidget();
144
- this.bottomMenu = this.createSidebarBottomMenu();
145
- this.toolBar = this.createToolbar();
146
- this.dockPanel = this.createSidePanel();
147
- this.container = this.createContainer();
148
-
149
- this.refresh();
150
- }
151
-
152
- protected createSideBar(): SideTabBar {
153
- const side = this.side;
154
- const tabBarRenderer = this.tabBarRendererFactory();
155
- const sideBar = new SideTabBar({
156
- // Tab bar options
157
- orientation: side === 'left' || side === 'right' ? 'vertical' : 'horizontal',
158
- insertBehavior: 'none',
159
- removeBehavior: 'select-previous-tab',
160
- allowDeselect: false,
161
- tabsMovable: true,
162
- renderer: tabBarRenderer,
163
- // Scroll bar options
164
- handlers: ['drag-thumb', 'keyboard', 'wheel', 'touch'],
165
- useBothWheelAxes: true,
166
- scrollYMarginOffset: 8,
167
- suppressScrollX: true
168
- });
169
- tabBarRenderer.tabBar = sideBar;
170
- sideBar.disposed.connect(() => tabBarRenderer.dispose());
171
- tabBarRenderer.contextMenuPath = SHELL_TABBAR_CONTEXT_MENU;
172
- sideBar.addClass('theia-app-' + side);
173
- sideBar.addClass(LEFT_RIGHT_AREA_CLASS);
174
-
175
- sideBar.tabAdded.connect((sender, { title }) => {
176
- const widget = title.owner;
177
- if (!some(this.dockPanel.widgets(), w => w === widget)) {
178
- this.dockPanel.addWidget(widget);
179
- }
180
- }, this);
181
- sideBar.tabActivateRequested.connect((sender, { title }) => title.owner.activate());
182
- sideBar.tabCloseRequested.connect((sender, { title }) => title.owner.close());
183
- sideBar.collapseRequested.connect(() => this.collapse(), this);
184
- sideBar.currentChanged.connect(this.onCurrentTabChanged, this);
185
- sideBar.tabDetachRequested.connect(this.onTabDetachRequested, this);
186
- sideBar.tabsOverflowChanged.connect(this.onTabsOverflowChanged, this);
187
- return sideBar;
188
- }
189
-
190
- protected createSidePanel(): TheiaDockPanel {
191
- const sidePanel = this.dockPanelFactory({
192
- mode: 'single-document'
193
- });
194
- sidePanel.id = 'theia-' + this.side + '-side-panel';
195
- sidePanel.addClass('theia-side-panel');
196
-
197
- sidePanel.widgetActivated.connect((sender, widget) => {
198
- this.tabBar.currentTitle = widget.title;
199
- }, this);
200
- sidePanel.widgetAdded.connect(this.onWidgetAdded, this);
201
- sidePanel.widgetRemoved.connect(this.onWidgetRemoved, this);
202
- return sidePanel;
203
- }
204
-
205
- protected createToolbar(): SidePanelToolbar {
206
- const toolbar = new SidePanelToolbar(this.tabBarToolBarRegistry, this.tabBarToolBarFactory, this.side);
207
- toolbar.onContextMenu(e => this.showContextMenu(e));
208
- return toolbar;
209
- }
210
-
211
- protected createAdditionalViewsWidget(): AdditionalViewsMenuWidget {
212
- const widget = this.additionalViewsMenuFactory(this.side);
213
- widget.addClass('theia-sidebar-menu');
214
- widget.addClass('theia-additional-views-menu');
215
- return widget;
216
- }
217
-
218
- protected createSidebarTopMenu(): SidebarTopMenuWidget {
219
- return this.createSidebarMenu(this.sidebarTopWidgetFactory);
220
- }
221
-
222
- protected createSidebarBottomMenu(): SidebarBottomMenuWidget {
223
- return this.createSidebarMenu(this.sidebarBottomWidgetFactory);
224
- }
225
-
226
- protected createSidebarMenu<T extends SidebarMenuWidget>(factory: () => T): T {
227
- const menu = factory();
228
- menu.addClass('theia-sidebar-menu');
229
- return menu;
230
- }
231
-
232
- protected showContextMenu(e: MouseEvent): void {
233
- const title = this.tabBar.currentTitle;
234
- if (!title) {
235
- return;
236
- }
237
- e.stopPropagation();
238
- e.preventDefault();
239
-
240
- this.contextMenuRenderer.render({
241
- args: [title.owner],
242
- menuPath: SIDE_PANEL_TOOLBAR_CONTEXT_MENU,
243
- anchor: e
244
- });
245
- }
246
-
247
- protected createContainer(): Panel {
248
- const contentBox = new BoxLayout({ direction: 'top-to-bottom', spacing: 0 });
249
- BoxPanel.setStretch(this.toolBar, 0);
250
- contentBox.addWidget(this.toolBar);
251
- BoxPanel.setStretch(this.dockPanel, 1);
252
- contentBox.addWidget(this.dockPanel);
253
- const contentPanel = new BoxPanel({ layout: contentBox });
254
-
255
- const side = this.side;
256
- let direction: BoxLayout.Direction;
257
- switch (side) {
258
- case 'left':
259
- direction = 'left-to-right';
260
- break;
261
- case 'right':
262
- direction = 'right-to-left';
263
- break;
264
- default:
265
- throw new Error('Illegal argument: ' + side);
266
- }
267
- const containerLayout = new BoxLayout({ direction, spacing: 0 });
268
- const sidebarContainerLayout = new PanelLayout();
269
- const sidebarContainer = new Panel({ layout: sidebarContainerLayout });
270
- sidebarContainer.addClass('theia-app-sidebar-container');
271
- sidebarContainerLayout.addWidget(this.topMenu);
272
- sidebarContainerLayout.addWidget(this.tabBar);
273
- sidebarContainerLayout.addWidget(this.additionalViewsMenu);
274
- sidebarContainerLayout.addWidget(this.bottomMenu);
275
-
276
- BoxPanel.setStretch(sidebarContainer, 0);
277
- BoxPanel.setStretch(contentPanel, 1);
278
- containerLayout.addWidget(sidebarContainer);
279
- containerLayout.addWidget(contentPanel);
280
- const boxPanel = new BoxPanel({ layout: containerLayout });
281
- boxPanel.id = 'theia-' + side + '-content-panel';
282
- return boxPanel;
283
- }
284
-
285
- /**
286
- * Create an object that describes the current side panel layout. This object may contain references
287
- * to widgets; these need to be transformed before the layout can be serialized.
288
- */
289
- getLayoutData(): SidePanel.LayoutData {
290
- const currentTitle = this.tabBar.currentTitle;
291
- const items = toArray(map(this.tabBar.titles, title => <SidePanel.WidgetItem>{
292
- widget: title.owner,
293
- rank: SidePanelHandler.rankProperty.get(title.owner),
294
- expanded: title === currentTitle,
295
- pinned: title.className.includes(PINNED_CLASS)
296
- }));
297
- // eslint-disable-next-line no-null/no-null
298
- const size = currentTitle !== null ? this.getPanelSize() : this.state.lastPanelSize;
299
- return { type: 'sidepanel', items, size };
300
- }
301
-
302
- /**
303
- * Apply a side panel layout that has been previously created with `getLayoutData`.
304
- */
305
- setLayoutData(layoutData: SidePanel.LayoutData): void {
306
- // eslint-disable-next-line no-null/no-null
307
- this.tabBar.currentTitle = null;
308
-
309
- let currentTitle: Title<Widget> | undefined;
310
- if (layoutData.items) {
311
- for (const { widget, rank, expanded, pinned } of layoutData.items) {
312
- if (widget) {
313
- if (rank) {
314
- SidePanelHandler.rankProperty.set(widget, rank);
315
- }
316
- if (expanded) {
317
- currentTitle = widget.title;
318
- }
319
- if (pinned) {
320
- widget.title.className += ` ${PINNED_CLASS}`;
321
- widget.title.closable = false;
322
- }
323
- // Add the widgets directly to the tab bar in the same order as they are stored
324
- this.tabBar.addTab(widget.title);
325
- }
326
- }
327
- }
328
- if (layoutData.size) {
329
- this.state.lastPanelSize = layoutData.size;
330
- }
331
-
332
- // If the layout data contains an expanded item, update the currentTitle property
333
- // This implies a refresh through the `currentChanged` signal
334
- if (currentTitle) {
335
- this.tabBar.currentTitle = currentTitle;
336
- } else {
337
- this.refresh();
338
- }
339
- }
340
-
341
- /**
342
- * Activate a widget residing in the side panel by ID.
343
- *
344
- * @returns the activated widget if it was found
345
- */
346
- activate(id: string): Widget | undefined {
347
- const widget = this.expand(id);
348
- if (widget) {
349
- widget.activate();
350
- }
351
- return widget;
352
- }
353
-
354
- /**
355
- * Expand a widget residing in the side panel by ID. If no ID is given and the panel is
356
- * currently collapsed, the last active tab of this side panel is expanded. If no tab
357
- * was expanded previously, the first one is taken.
358
- *
359
- * @returns the expanded widget if it was found
360
- */
361
- expand(id?: string): Widget | undefined {
362
- if (id) {
363
- const widget = find(this.dockPanel.widgets(), w => w.id === id);
364
- if (widget) {
365
- this.tabBar.currentTitle = widget.title;
366
- }
367
- return widget;
368
- } else if (this.tabBar.currentTitle) {
369
- return this.tabBar.currentTitle.owner;
370
- } else if (this.tabBar.titles.length > 0) {
371
- let index = this.state.lastActiveTabIndex;
372
- if (!index) {
373
- index = 0;
374
- } else if (index >= this.tabBar.titles.length) {
375
- index = this.tabBar.titles.length - 1;
376
- }
377
- const title = this.tabBar.titles[index];
378
- this.tabBar.currentTitle = title;
379
- return title.owner;
380
- } else {
381
- // Reveal the tab bar and dock panel even if there is no widget
382
- // The next call to `refreshVisibility` will collapse them again
383
- this.state.expansion = SidePanel.ExpansionState.expanding;
384
- let relativeSizes: number[] | undefined;
385
- const parent = this.container.parent;
386
- if (parent instanceof SplitPanel) {
387
- relativeSizes = parent.relativeSizes();
388
- }
389
- this.container.removeClass(COLLAPSED_CLASS);
390
- this.container.show();
391
- this.tabBar.show();
392
- this.dockPanel.node.style.minWidth = '0';
393
- this.dockPanel.show();
394
- if (relativeSizes && parent instanceof SplitPanel) {
395
- // Make sure that the expansion animation starts at zero size
396
- parent.setRelativeSizes(relativeSizes);
397
- }
398
- this.setPanelSize(this.options.emptySize).then(() => {
399
- if (this.state.expansion === SidePanel.ExpansionState.expanding) {
400
- this.state.expansion = SidePanel.ExpansionState.expanded;
401
- }
402
- });
403
- }
404
- }
405
-
406
- /**
407
- * Collapse the sidebar so no items are expanded.
408
- */
409
- collapse(): Promise<void> {
410
- if (this.tabBar.currentTitle) {
411
- // eslint-disable-next-line no-null/no-null
412
- this.tabBar.currentTitle = null;
413
- } else {
414
- this.refresh();
415
- }
416
- return animationFrame();
417
- }
418
-
419
- /**
420
- * Add a widget and its title to the dock panel and side bar.
421
- *
422
- * If the widget is already added, it will be moved.
423
- */
424
- addWidget(widget: Widget, options: SidePanel.WidgetOptions): void {
425
- if (options.rank) {
426
- SidePanelHandler.rankProperty.set(widget, options.rank);
427
- }
428
- this.dockPanel.addWidget(widget);
429
- }
430
-
431
- /**
432
- * Add a menu to the sidebar top.
433
- *
434
- * If the menu is already added, it will be ignored.
435
- */
436
- addTopMenu(menu: SidebarMenu): void {
437
- this.topMenu.addMenu(menu);
438
- }
439
-
440
- /**
441
- * Remove a menu from the sidebar top.
442
- *
443
- * @param menuId id of the menu to remove
444
- */
445
- removeTopMenu(menuId: string): void {
446
- this.topMenu.removeMenu(menuId);
447
- }
448
-
449
- /**
450
- * Add a menu to the sidebar bottom.
451
- *
452
- * If the menu is already added, it will be ignored.
453
- */
454
- addBottomMenu(menu: SidebarMenu): void {
455
- this.bottomMenu.addMenu(menu);
456
- }
457
-
458
- /**
459
- * Remove a menu from the sidebar bottom.
460
- *
461
- * @param menuId id of the menu to remove
462
- */
463
- removeBottomMenu(menuId: string): void {
464
- this.bottomMenu.removeMenu(menuId);
465
- }
466
-
467
- // should be a property to preserve fn identity
468
- protected updateToolbarTitle = (): void => {
469
- const currentTitle = this.tabBar && this.tabBar.currentTitle;
470
- this.toolBar.toolbarTitle = currentTitle || undefined;
471
- };
472
-
473
- /**
474
- * Refresh the visibility of the side bar and dock panel.
475
- */
476
- refresh(): void {
477
- const container = this.container;
478
- const parent = container.parent;
479
- const tabBar = this.tabBar;
480
- const dockPanel = this.dockPanel;
481
- const isEmpty = tabBar.titles.length === 0;
482
- const currentTitle = tabBar.currentTitle;
483
- // eslint-disable-next-line no-null/no-null
484
- const hideDockPanel = currentTitle === null;
485
- this.updateSashState(this.container, hideDockPanel);
486
- let relativeSizes: number[] | undefined;
487
-
488
- if (hideDockPanel) {
489
- container.addClass(COLLAPSED_CLASS);
490
- if (this.state.expansion === SidePanel.ExpansionState.expanded && !this.state.empty) {
491
- // Update the lastPanelSize property
492
- const size = this.getPanelSize();
493
- if (size) {
494
- this.state.lastPanelSize = size;
495
- }
496
- }
497
- this.state.expansion = SidePanel.ExpansionState.collapsed;
498
- } else {
499
- container.removeClass(COLLAPSED_CLASS);
500
- let size: number | undefined;
501
- if (this.state.expansion !== SidePanel.ExpansionState.expanded) {
502
- if (this.state.lastPanelSize) {
503
- size = this.state.lastPanelSize;
504
- } else {
505
- size = this.getDefaultPanelSize();
506
- }
507
- }
508
- if (size) {
509
- // Restore the panel size to the last known size or the default size
510
- this.state.expansion = SidePanel.ExpansionState.expanding;
511
- if (parent instanceof SplitPanel) {
512
- relativeSizes = parent.relativeSizes();
513
- }
514
- this.setPanelSize(size).then(() => {
515
- if (this.state.expansion === SidePanel.ExpansionState.expanding) {
516
- this.state.expansion = SidePanel.ExpansionState.expanded;
517
- }
518
- });
519
- } else {
520
- this.state.expansion = SidePanel.ExpansionState.expanded;
521
- }
522
- }
523
- container.setHidden(isEmpty && hideDockPanel);
524
- tabBar.setHidden(isEmpty);
525
- dockPanel.setHidden(hideDockPanel);
526
- this.state.empty = isEmpty;
527
- if (currentTitle) {
528
- dockPanel.selectWidget(currentTitle.owner);
529
- }
530
- if (relativeSizes && parent instanceof SplitPanel) {
531
- // Make sure that the expansion animation starts at the smallest possible size
532
- parent.setRelativeSizes(relativeSizes);
533
- }
534
- }
535
-
536
- /**
537
- * Sets the size of the side panel.
538
- *
539
- * @param size the desired size (width) of the panel in pixels.
540
- */
541
- resize(size: number): void {
542
- if (this.dockPanel.isHidden) {
543
- this.state.lastPanelSize = size;
544
- } else {
545
- this.setPanelSize(size);
546
- }
547
- }
548
-
549
- /**
550
- * Compute the current width of the panel. This implementation assumes that the parent of
551
- * the panel container is a `SplitPanel`.
552
- */
553
- protected getPanelSize(): number | undefined {
554
- const parent = this.container.parent;
555
- if (parent instanceof SplitPanel && parent.isVisible) {
556
- const index = parent.widgets.indexOf(this.container);
557
- if (this.side === 'left') {
558
- const handle = parent.handles[index];
559
- if (!handle.classList.contains('p-mod-hidden')) {
560
- return handle.offsetLeft;
561
- }
562
- } else if (this.side === 'right') {
563
- const handle = parent.handles[index - 1];
564
- if (!handle.classList.contains('p-mod-hidden')) {
565
- const parentWidth = parent.node.clientWidth;
566
- return parentWidth - handle.offsetLeft;
567
- }
568
- }
569
- }
570
- }
571
-
572
- /**
573
- * Determine the default size to apply when the panel is expanded for the first time.
574
- */
575
- protected getDefaultPanelSize(): number | undefined {
576
- const parent = this.container.parent;
577
- if (parent && parent.isVisible) {
578
- return parent.node.clientWidth * this.options.initialSizeRatio;
579
- }
580
- }
581
-
582
- /**
583
- * Modify the width of the panel. This implementation assumes that the parent of the panel
584
- * container is a `SplitPanel`.
585
- */
586
- protected setPanelSize(size: number): Promise<void> {
587
- const enableAnimation = this.applicationStateService.state === 'ready';
588
- const options: SplitPositionOptions = {
589
- side: this.side,
590
- duration: enableAnimation ? this.options.expandDuration : 0,
591
- referenceWidget: this.dockPanel
592
- };
593
- const promise = this.splitPositionHandler.setSidePanelSize(this.container, size, options);
594
- const result = new Promise<void>(resolve => {
595
- // Resolve the resulting promise in any case, regardless of whether resizing was successful
596
- promise.then(() => resolve(), () => resolve());
597
- });
598
- this.state.pendingUpdate = this.state.pendingUpdate.then(() => result);
599
- return result;
600
- }
601
-
602
- protected readonly toDisposeOnCurrentTabChanged = new DisposableCollection();
603
-
604
- /**
605
- * Handle a `currentChanged` signal from the sidebar. The side panel is refreshed so it displays
606
- * the new selected widget.
607
- */
608
- protected onCurrentTabChanged(sender: SideTabBar, { currentTitle, currentIndex }: TabBar.ICurrentChangedArgs<Widget>): void {
609
- this.toDisposeOnCurrentTabChanged.dispose();
610
- if (currentTitle) {
611
- this.updateToolbarTitle();
612
- currentTitle.changed.connect(this.updateToolbarTitle);
613
- this.toDisposeOnCurrentTabChanged.push(Disposable.create(() => currentTitle.changed.disconnect(this.updateToolbarTitle)));
614
- }
615
- if (currentIndex >= 0) {
616
- this.state.lastActiveTabIndex = currentIndex;
617
- sender.revealTab(currentIndex);
618
- }
619
- this.refresh();
620
- }
621
-
622
- /**
623
- * Handle a `tabDetachRequested` signal from the sidebar. A drag is started so the widget can be
624
- * moved to another application shell area.
625
- */
626
- protected onTabDetachRequested(sender: SideTabBar,
627
- { title, tab, clientX, clientY }: TabBar.ITabDetachRequestedArgs<Widget>): void {
628
- // Release the tab bar's hold on the mouse
629
- sender.releaseMouse();
630
-
631
- // Clone the selected tab and use that as drag image
632
- const clonedTab = tab.cloneNode(true) as HTMLElement;
633
- clonedTab.style.width = '';
634
- clonedTab.style.height = '';
635
- const label = clonedTab.getElementsByClassName('p-TabBar-tabLabel')[0] as HTMLElement;
636
- label.style.width = '';
637
- label.style.height = '';
638
-
639
- // Create and start a drag to move the selected tab to another panel
640
- const mimeData = new MimeData();
641
- mimeData.setData('application/vnd.phosphor.widget-factory', () => title.owner);
642
- const drag = new Drag({
643
- mimeData,
644
- dragImage: clonedTab,
645
- proposedAction: 'move',
646
- supportedActions: 'move',
647
- });
648
-
649
- tab.classList.add('p-mod-hidden');
650
- drag.start(clientX, clientY).then(() => {
651
- // The promise is resolved when the drag has ended
652
- tab.classList.remove('p-mod-hidden');
653
- });
654
- }
655
-
656
- protected onTabsOverflowChanged(sender: SideTabBar, event: { titles: Title<Widget>[], startIndex: number }): void {
657
- if (event.startIndex > 0 && event.startIndex <= sender.currentIndex) {
658
- sender.revealTab(sender.currentIndex);
659
- } else {
660
- this.additionalViewsMenu.updateAdditionalViews(sender, event);
661
- }
662
- }
663
-
664
- /*
665
- * Handle the `widgetAdded` signal from the dock panel. The widget's title is inserted into the
666
- * tab bar according to the `rankProperty` value that may be attached to the widget.
667
- */
668
- protected onWidgetAdded(sender: DockPanel, widget: Widget): void {
669
- const titles = this.tabBar.titles;
670
- if (!find(titles, t => t.owner === widget)) {
671
- const rank = SidePanelHandler.rankProperty.get(widget);
672
- let index = titles.length;
673
- if (rank !== undefined) {
674
- for (let i = index - 1; i >= 0; i--) {
675
- const r = SidePanelHandler.rankProperty.get(titles[i].owner);
676
- if (r !== undefined && r > rank) {
677
- index = i;
678
- }
679
- }
680
- }
681
- this.tabBar.insertTab(index, widget.title);
682
- this.refresh();
683
- }
684
- }
685
-
686
- /*
687
- * Handle the `widgetRemoved` signal from the dock panel. The widget's title is also removed
688
- * from the tab bar.
689
- */
690
- protected onWidgetRemoved(sender: DockPanel, widget: Widget): void {
691
- this.tabBar.removeTab(widget.title);
692
- this.refresh();
693
- }
694
-
695
- protected updateSashState(sidePanelElement: Panel | null, sidePanelCollapsed: boolean): void {
696
- if (sidePanelElement) {
697
- // Hide the sash when the left/right side panel is collapsed
698
- if (sidePanelElement.id === 'theia-left-content-panel' && sidePanelElement.node.nextElementSibling) {
699
- sidePanelElement.node.nextElementSibling.classList.toggle('sash-hidden', sidePanelCollapsed);
700
- } else if (sidePanelElement.id === 'theia-right-content-panel' && sidePanelElement.node.previousElementSibling) {
701
- sidePanelElement.node.previousElementSibling.classList.toggle('sash-hidden', sidePanelCollapsed);
702
- }
703
- }
704
- }
705
-
706
- }
707
-
708
- export namespace SidePanel {
709
- /**
710
- * Options that control the behavior of side panels.
711
- */
712
- export interface Options {
713
- /**
714
- * When a widget is being dragged and the distance of the mouse cursor to the shell border
715
- * is below this threshold, the respective side panel is expanded so the widget can be dropped
716
- * into that panel. Set this to `-1` to disable expanding the side panel while dragging.
717
- */
718
- expandThreshold: number;
719
- /**
720
- * The duration in milliseconds of the animation shown when a side panel is expanded.
721
- * Set this to `0` to disable expansion animation.
722
- */
723
- expandDuration: number;
724
- /**
725
- * The ratio of the available shell size to use as initial size for the side panel.
726
- */
727
- initialSizeRatio: number
728
- /**
729
- * How large the panel should be when it's expanded and empty.
730
- */
731
- emptySize: number;
732
- }
733
-
734
- /**
735
- * The options for adding a widget to a side panel.
736
- */
737
- export interface WidgetOptions {
738
- /**
739
- * The rank order of the widget among its siblings.
740
- */
741
- rank?: number;
742
- }
743
-
744
- /**
745
- * Data to save and load the layout of a side panel.
746
- */
747
- export interface LayoutData {
748
- type: 'sidepanel',
749
- items?: WidgetItem[];
750
- size?: number;
751
- }
752
-
753
- /**
754
- * Data structure used to save and restore the side panel layout.
755
- */
756
- export interface WidgetItem extends WidgetOptions {
757
- /** Can be undefined in case the widget could not be restored. */
758
- widget?: Widget;
759
- expanded?: boolean;
760
- pinned?: boolean;
761
- }
762
-
763
- export interface State {
764
- /**
765
- * Indicates whether the panel is empty.
766
- */
767
- empty: boolean;
768
- /**
769
- * Indicates whether the panel is expanded, collapsed, or in a transition between the two.
770
- */
771
- expansion: ExpansionState;
772
- /**
773
- * A promise that is resolved when the currently pending side panel updates are done.
774
- */
775
- pendingUpdate: Promise<void>;
776
- /**
777
- * The index of the last tab that was selected. When the panel is expanded, it tries to restore
778
- * the tab selection to the previous state.
779
- */
780
- lastActiveTabIndex?: number;
781
- /**
782
- * The width or height of the panel before it was collapsed. When the panel is expanded, it tries
783
- * to restore its size to this value.
784
- */
785
- lastPanelSize?: number;
786
- }
787
-
788
- export enum ExpansionState {
789
- collapsed = 'collapsed',
790
- expanding = 'expanding',
791
- expanded = 'expanded',
792
- collapsing = 'collapsing'
793
- }
794
- }
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 } from 'inversify';
18
+ import { find, map, toArray, some } from '@phosphor/algorithm';
19
+ import { TabBar, Widget, DockPanel, Title, Panel, BoxPanel, BoxLayout, SplitPanel, PanelLayout } from '@phosphor/widgets';
20
+ import { MimeData } from '@phosphor/coreutils';
21
+ import { Drag } from '@phosphor/dragdrop';
22
+ import { AttachedProperty } from '@phosphor/properties';
23
+ import { TabBarRendererFactory, TabBarRenderer, SHELL_TABBAR_CONTEXT_MENU, SideTabBar } from './tab-bars';
24
+ import { SidebarMenuWidget, SidebarMenu, SidebarBottomMenuWidgetFactory, SidebarTopMenuWidgetFactory } from './sidebar-menu-widget';
25
+ import { SplitPositionHandler, SplitPositionOptions } from './split-panels';
26
+ import { animationFrame } from '../browser';
27
+ import { FrontendApplicationStateService } from '../frontend-application-state';
28
+ import { TheiaDockPanel } from './theia-dock-panel';
29
+ import { SidePanelToolbar } from './side-panel-toolbar';
30
+ import { TabBarToolbarRegistry, TabBarToolbarFactory, TabBarToolbar } from './tab-bar-toolbar';
31
+ import { DisposableCollection, Disposable } from '../../common/disposable';
32
+ import { ContextMenuRenderer } from '../context-menu-renderer';
33
+ import { MenuPath } from '../../common/menu';
34
+ import { SidebarBottomMenuWidget } from './sidebar-bottom-menu-widget';
35
+ import { SidebarTopMenuWidget } from './sidebar-top-menu-widget';
36
+ import { PINNED_CLASS } from '../widgets';
37
+ import { AdditionalViewsMenuWidget, AdditionalViewsMenuWidgetFactory } from './additional-views-menu-widget';
38
+
39
+ /** The class name added to the left and right area panels. */
40
+ export const LEFT_RIGHT_AREA_CLASS = 'theia-app-sides';
41
+
42
+ /** The class name added to collapsed side panels. */
43
+ const COLLAPSED_CLASS = 'theia-mod-collapsed';
44
+
45
+ export const SidePanelHandlerFactory = Symbol('SidePanelHandlerFactory');
46
+
47
+ export const SIDE_PANEL_TOOLBAR_CONTEXT_MENU: MenuPath = ['SIDE_PANEL_TOOLBAR_CONTEXT_MENU'];
48
+
49
+ /**
50
+ * A class which manages a dock panel and a related side bar. This is used for the left and right
51
+ * panel of the application shell.
52
+ */
53
+ @injectable()
54
+ export class SidePanelHandler {
55
+
56
+ /**
57
+ * A property that can be attached to widgets in order to determine the insertion index
58
+ * of their title in the tab bar.
59
+ */
60
+ protected static readonly rankProperty = new AttachedProperty<Widget, number | undefined>({
61
+ name: 'sidePanelRank',
62
+ create: () => undefined
63
+ });
64
+
65
+ /**
66
+ * The tab bar displays the titles of the widgets in the side panel. Visibility of the widgets
67
+ * is controlled entirely through tab selection: a widget is revealed by setting the `currentTitle`
68
+ * accordingly in the tab bar, and the panel is hidden by setting that property to `null`. The
69
+ * tab bar itself remains visible as long as there is at least one widget.
70
+ */
71
+ tabBar: SideTabBar;
72
+ /**
73
+ * Conditional menu placed below the tabBar. Manages overflowing/hidden tabs.
74
+ * Is only visible if there are overflowing tabs.
75
+ */
76
+ additionalViewsMenu: AdditionalViewsMenuWidget;
77
+ /**
78
+ * The menu placed on the sidebar top.
79
+ * Displayed as icons.
80
+ * Open menus when on clicks.
81
+ */
82
+ topMenu: SidebarMenuWidget;
83
+ /**
84
+ * The menu placed on the sidebar bottom.
85
+ * Displayed as icons.
86
+ * Open menus when on clicks.
87
+ */
88
+ bottomMenu: SidebarMenuWidget;
89
+ /**
90
+ * A tool bar, which displays a title and widget specific command buttons.
91
+ */
92
+ toolBar: SidePanelToolbar;
93
+ /**
94
+ * The widget container is a dock panel in `single-document` mode, which means that the panel
95
+ * cannot be split.
96
+ */
97
+ dockPanel: TheiaDockPanel;
98
+ /**
99
+ * The panel that contains the tab bar and the dock panel. This one is hidden whenever the dock
100
+ * panel is empty.
101
+ */
102
+ container: Panel;
103
+ /**
104
+ * The current state of the side panel.
105
+ */
106
+ readonly state: SidePanel.State = {
107
+ empty: true,
108
+ expansion: SidePanel.ExpansionState.collapsed,
109
+ pendingUpdate: Promise.resolve()
110
+ };
111
+
112
+ /**
113
+ * The shell area where the panel is placed. This property should not be modified directly, but
114
+ * only by calling `create`.
115
+ */
116
+ protected side: 'left' | 'right';
117
+ /**
118
+ * Options that control the behavior of the side panel.
119
+ */
120
+ protected options: SidePanel.Options;
121
+
122
+ @inject(TabBarToolbarRegistry) protected tabBarToolBarRegistry: TabBarToolbarRegistry;
123
+ @inject(TabBarToolbarFactory) protected tabBarToolBarFactory: () => TabBarToolbar;
124
+ @inject(TabBarRendererFactory) protected tabBarRendererFactory: () => TabBarRenderer;
125
+ @inject(SidebarTopMenuWidgetFactory) protected sidebarTopWidgetFactory: () => SidebarTopMenuWidget;
126
+ @inject(SidebarBottomMenuWidgetFactory) protected sidebarBottomWidgetFactory: () => SidebarBottomMenuWidget;
127
+ @inject(AdditionalViewsMenuWidgetFactory) protected additionalViewsMenuFactory: AdditionalViewsMenuWidgetFactory;
128
+ @inject(SplitPositionHandler) protected splitPositionHandler: SplitPositionHandler;
129
+ @inject(FrontendApplicationStateService) protected readonly applicationStateService: FrontendApplicationStateService;
130
+ @inject(TheiaDockPanel.Factory) protected readonly dockPanelFactory: TheiaDockPanel.Factory;
131
+
132
+ @inject(ContextMenuRenderer)
133
+ protected readonly contextMenuRenderer: ContextMenuRenderer;
134
+
135
+ /**
136
+ * Create the side bar and dock panel widgets.
137
+ */
138
+ create(side: 'left' | 'right', options: SidePanel.Options): void {
139
+ this.side = side;
140
+ this.options = options;
141
+ this.topMenu = this.createSidebarTopMenu();
142
+ this.tabBar = this.createSideBar();
143
+ this.additionalViewsMenu = this.createAdditionalViewsWidget();
144
+ this.bottomMenu = this.createSidebarBottomMenu();
145
+ this.toolBar = this.createToolbar();
146
+ this.dockPanel = this.createSidePanel();
147
+ this.container = this.createContainer();
148
+
149
+ this.refresh();
150
+ }
151
+
152
+ protected createSideBar(): SideTabBar {
153
+ const side = this.side;
154
+ const tabBarRenderer = this.tabBarRendererFactory();
155
+ const sideBar = new SideTabBar({
156
+ // Tab bar options
157
+ orientation: side === 'left' || side === 'right' ? 'vertical' : 'horizontal',
158
+ insertBehavior: 'none',
159
+ removeBehavior: 'select-previous-tab',
160
+ allowDeselect: false,
161
+ tabsMovable: true,
162
+ renderer: tabBarRenderer,
163
+ // Scroll bar options
164
+ handlers: ['drag-thumb', 'keyboard', 'wheel', 'touch'],
165
+ useBothWheelAxes: true,
166
+ scrollYMarginOffset: 8,
167
+ suppressScrollX: true
168
+ });
169
+ tabBarRenderer.tabBar = sideBar;
170
+ sideBar.disposed.connect(() => tabBarRenderer.dispose());
171
+ tabBarRenderer.contextMenuPath = SHELL_TABBAR_CONTEXT_MENU;
172
+ sideBar.addClass('theia-app-' + side);
173
+ sideBar.addClass(LEFT_RIGHT_AREA_CLASS);
174
+
175
+ sideBar.tabAdded.connect((sender, { title }) => {
176
+ const widget = title.owner;
177
+ if (!some(this.dockPanel.widgets(), w => w === widget)) {
178
+ this.dockPanel.addWidget(widget);
179
+ }
180
+ }, this);
181
+ sideBar.tabActivateRequested.connect((sender, { title }) => title.owner.activate());
182
+ sideBar.tabCloseRequested.connect((sender, { title }) => title.owner.close());
183
+ sideBar.collapseRequested.connect(() => this.collapse(), this);
184
+ sideBar.currentChanged.connect(this.onCurrentTabChanged, this);
185
+ sideBar.tabDetachRequested.connect(this.onTabDetachRequested, this);
186
+ sideBar.tabsOverflowChanged.connect(this.onTabsOverflowChanged, this);
187
+ return sideBar;
188
+ }
189
+
190
+ protected createSidePanel(): TheiaDockPanel {
191
+ const sidePanel = this.dockPanelFactory({
192
+ mode: 'single-document'
193
+ });
194
+ sidePanel.id = 'theia-' + this.side + '-side-panel';
195
+ sidePanel.addClass('theia-side-panel');
196
+
197
+ sidePanel.widgetActivated.connect((sender, widget) => {
198
+ this.tabBar.currentTitle = widget.title;
199
+ }, this);
200
+ sidePanel.widgetAdded.connect(this.onWidgetAdded, this);
201
+ sidePanel.widgetRemoved.connect(this.onWidgetRemoved, this);
202
+ return sidePanel;
203
+ }
204
+
205
+ protected createToolbar(): SidePanelToolbar {
206
+ const toolbar = new SidePanelToolbar(this.tabBarToolBarRegistry, this.tabBarToolBarFactory, this.side);
207
+ toolbar.onContextMenu(e => this.showContextMenu(e));
208
+ return toolbar;
209
+ }
210
+
211
+ protected createAdditionalViewsWidget(): AdditionalViewsMenuWidget {
212
+ const widget = this.additionalViewsMenuFactory(this.side);
213
+ widget.addClass('theia-sidebar-menu');
214
+ widget.addClass('theia-additional-views-menu');
215
+ return widget;
216
+ }
217
+
218
+ protected createSidebarTopMenu(): SidebarTopMenuWidget {
219
+ return this.createSidebarMenu(this.sidebarTopWidgetFactory);
220
+ }
221
+
222
+ protected createSidebarBottomMenu(): SidebarBottomMenuWidget {
223
+ return this.createSidebarMenu(this.sidebarBottomWidgetFactory);
224
+ }
225
+
226
+ protected createSidebarMenu<T extends SidebarMenuWidget>(factory: () => T): T {
227
+ const menu = factory();
228
+ menu.addClass('theia-sidebar-menu');
229
+ return menu;
230
+ }
231
+
232
+ protected showContextMenu(e: MouseEvent): void {
233
+ const title = this.tabBar.currentTitle;
234
+ if (!title) {
235
+ return;
236
+ }
237
+ e.stopPropagation();
238
+ e.preventDefault();
239
+
240
+ this.contextMenuRenderer.render({
241
+ args: [title.owner],
242
+ menuPath: SIDE_PANEL_TOOLBAR_CONTEXT_MENU,
243
+ anchor: e
244
+ });
245
+ }
246
+
247
+ protected createContainer(): Panel {
248
+ const contentBox = new BoxLayout({ direction: 'top-to-bottom', spacing: 0 });
249
+ BoxPanel.setStretch(this.toolBar, 0);
250
+ contentBox.addWidget(this.toolBar);
251
+ BoxPanel.setStretch(this.dockPanel, 1);
252
+ contentBox.addWidget(this.dockPanel);
253
+ const contentPanel = new BoxPanel({ layout: contentBox });
254
+
255
+ const side = this.side;
256
+ let direction: BoxLayout.Direction;
257
+ switch (side) {
258
+ case 'left':
259
+ direction = 'left-to-right';
260
+ break;
261
+ case 'right':
262
+ direction = 'right-to-left';
263
+ break;
264
+ default:
265
+ throw new Error('Illegal argument: ' + side);
266
+ }
267
+ const containerLayout = new BoxLayout({ direction, spacing: 0 });
268
+ const sidebarContainerLayout = new PanelLayout();
269
+ const sidebarContainer = new Panel({ layout: sidebarContainerLayout });
270
+ sidebarContainer.addClass('theia-app-sidebar-container');
271
+ sidebarContainerLayout.addWidget(this.topMenu);
272
+ sidebarContainerLayout.addWidget(this.tabBar);
273
+ sidebarContainerLayout.addWidget(this.additionalViewsMenu);
274
+ sidebarContainerLayout.addWidget(this.bottomMenu);
275
+
276
+ BoxPanel.setStretch(sidebarContainer, 0);
277
+ BoxPanel.setStretch(contentPanel, 1);
278
+ containerLayout.addWidget(sidebarContainer);
279
+ containerLayout.addWidget(contentPanel);
280
+ const boxPanel = new BoxPanel({ layout: containerLayout });
281
+ boxPanel.id = 'theia-' + side + '-content-panel';
282
+ return boxPanel;
283
+ }
284
+
285
+ /**
286
+ * Create an object that describes the current side panel layout. This object may contain references
287
+ * to widgets; these need to be transformed before the layout can be serialized.
288
+ */
289
+ getLayoutData(): SidePanel.LayoutData {
290
+ const currentTitle = this.tabBar.currentTitle;
291
+ const items = toArray(map(this.tabBar.titles, title => <SidePanel.WidgetItem>{
292
+ widget: title.owner,
293
+ rank: SidePanelHandler.rankProperty.get(title.owner),
294
+ expanded: title === currentTitle,
295
+ pinned: title.className.includes(PINNED_CLASS)
296
+ }));
297
+ // eslint-disable-next-line no-null/no-null
298
+ const size = currentTitle !== null ? this.getPanelSize() : this.state.lastPanelSize;
299
+ return { type: 'sidepanel', items, size };
300
+ }
301
+
302
+ /**
303
+ * Apply a side panel layout that has been previously created with `getLayoutData`.
304
+ */
305
+ setLayoutData(layoutData: SidePanel.LayoutData): void {
306
+ // eslint-disable-next-line no-null/no-null
307
+ this.tabBar.currentTitle = null;
308
+
309
+ let currentTitle: Title<Widget> | undefined;
310
+ if (layoutData.items) {
311
+ for (const { widget, rank, expanded, pinned } of layoutData.items) {
312
+ if (widget) {
313
+ if (rank) {
314
+ SidePanelHandler.rankProperty.set(widget, rank);
315
+ }
316
+ if (expanded) {
317
+ currentTitle = widget.title;
318
+ }
319
+ if (pinned) {
320
+ widget.title.className += ` ${PINNED_CLASS}`;
321
+ widget.title.closable = false;
322
+ }
323
+ // Add the widgets directly to the tab bar in the same order as they are stored
324
+ this.tabBar.addTab(widget.title);
325
+ }
326
+ }
327
+ }
328
+ if (layoutData.size) {
329
+ this.state.lastPanelSize = layoutData.size;
330
+ }
331
+
332
+ // If the layout data contains an expanded item, update the currentTitle property
333
+ // This implies a refresh through the `currentChanged` signal
334
+ if (currentTitle) {
335
+ this.tabBar.currentTitle = currentTitle;
336
+ } else {
337
+ this.refresh();
338
+ }
339
+ }
340
+
341
+ /**
342
+ * Activate a widget residing in the side panel by ID.
343
+ *
344
+ * @returns the activated widget if it was found
345
+ */
346
+ activate(id: string): Widget | undefined {
347
+ const widget = this.expand(id);
348
+ if (widget) {
349
+ widget.activate();
350
+ }
351
+ return widget;
352
+ }
353
+
354
+ /**
355
+ * Expand a widget residing in the side panel by ID. If no ID is given and the panel is
356
+ * currently collapsed, the last active tab of this side panel is expanded. If no tab
357
+ * was expanded previously, the first one is taken.
358
+ *
359
+ * @returns the expanded widget if it was found
360
+ */
361
+ expand(id?: string): Widget | undefined {
362
+ if (id) {
363
+ const widget = find(this.dockPanel.widgets(), w => w.id === id);
364
+ if (widget) {
365
+ this.tabBar.currentTitle = widget.title;
366
+ }
367
+ return widget;
368
+ } else if (this.tabBar.currentTitle) {
369
+ return this.tabBar.currentTitle.owner;
370
+ } else if (this.tabBar.titles.length > 0) {
371
+ let index = this.state.lastActiveTabIndex;
372
+ if (!index) {
373
+ index = 0;
374
+ } else if (index >= this.tabBar.titles.length) {
375
+ index = this.tabBar.titles.length - 1;
376
+ }
377
+ const title = this.tabBar.titles[index];
378
+ this.tabBar.currentTitle = title;
379
+ return title.owner;
380
+ } else {
381
+ // Reveal the tab bar and dock panel even if there is no widget
382
+ // The next call to `refreshVisibility` will collapse them again
383
+ this.state.expansion = SidePanel.ExpansionState.expanding;
384
+ let relativeSizes: number[] | undefined;
385
+ const parent = this.container.parent;
386
+ if (parent instanceof SplitPanel) {
387
+ relativeSizes = parent.relativeSizes();
388
+ }
389
+ this.container.removeClass(COLLAPSED_CLASS);
390
+ this.container.show();
391
+ this.tabBar.show();
392
+ this.dockPanel.node.style.minWidth = '0';
393
+ this.dockPanel.show();
394
+ if (relativeSizes && parent instanceof SplitPanel) {
395
+ // Make sure that the expansion animation starts at zero size
396
+ parent.setRelativeSizes(relativeSizes);
397
+ }
398
+ this.setPanelSize(this.options.emptySize).then(() => {
399
+ if (this.state.expansion === SidePanel.ExpansionState.expanding) {
400
+ this.state.expansion = SidePanel.ExpansionState.expanded;
401
+ }
402
+ });
403
+ }
404
+ }
405
+
406
+ /**
407
+ * Collapse the sidebar so no items are expanded.
408
+ */
409
+ collapse(): Promise<void> {
410
+ if (this.tabBar.currentTitle) {
411
+ // eslint-disable-next-line no-null/no-null
412
+ this.tabBar.currentTitle = null;
413
+ } else {
414
+ this.refresh();
415
+ }
416
+ return animationFrame();
417
+ }
418
+
419
+ /**
420
+ * Add a widget and its title to the dock panel and side bar.
421
+ *
422
+ * If the widget is already added, it will be moved.
423
+ */
424
+ addWidget(widget: Widget, options: SidePanel.WidgetOptions): void {
425
+ if (options.rank) {
426
+ SidePanelHandler.rankProperty.set(widget, options.rank);
427
+ }
428
+ this.dockPanel.addWidget(widget);
429
+ }
430
+
431
+ /**
432
+ * Add a menu to the sidebar top.
433
+ *
434
+ * If the menu is already added, it will be ignored.
435
+ */
436
+ addTopMenu(menu: SidebarMenu): void {
437
+ this.topMenu.addMenu(menu);
438
+ }
439
+
440
+ /**
441
+ * Remove a menu from the sidebar top.
442
+ *
443
+ * @param menuId id of the menu to remove
444
+ */
445
+ removeTopMenu(menuId: string): void {
446
+ this.topMenu.removeMenu(menuId);
447
+ }
448
+
449
+ /**
450
+ * Add a menu to the sidebar bottom.
451
+ *
452
+ * If the menu is already added, it will be ignored.
453
+ */
454
+ addBottomMenu(menu: SidebarMenu): void {
455
+ this.bottomMenu.addMenu(menu);
456
+ }
457
+
458
+ /**
459
+ * Remove a menu from the sidebar bottom.
460
+ *
461
+ * @param menuId id of the menu to remove
462
+ */
463
+ removeBottomMenu(menuId: string): void {
464
+ this.bottomMenu.removeMenu(menuId);
465
+ }
466
+
467
+ // should be a property to preserve fn identity
468
+ protected updateToolbarTitle = (): void => {
469
+ const currentTitle = this.tabBar && this.tabBar.currentTitle;
470
+ this.toolBar.toolbarTitle = currentTitle || undefined;
471
+ };
472
+
473
+ /**
474
+ * Refresh the visibility of the side bar and dock panel.
475
+ */
476
+ refresh(): void {
477
+ const container = this.container;
478
+ const parent = container.parent;
479
+ const tabBar = this.tabBar;
480
+ const dockPanel = this.dockPanel;
481
+ const isEmpty = tabBar.titles.length === 0;
482
+ const currentTitle = tabBar.currentTitle;
483
+ // eslint-disable-next-line no-null/no-null
484
+ const hideDockPanel = currentTitle === null;
485
+ this.updateSashState(this.container, hideDockPanel);
486
+ let relativeSizes: number[] | undefined;
487
+
488
+ if (hideDockPanel) {
489
+ container.addClass(COLLAPSED_CLASS);
490
+ if (this.state.expansion === SidePanel.ExpansionState.expanded && !this.state.empty) {
491
+ // Update the lastPanelSize property
492
+ const size = this.getPanelSize();
493
+ if (size) {
494
+ this.state.lastPanelSize = size;
495
+ }
496
+ }
497
+ this.state.expansion = SidePanel.ExpansionState.collapsed;
498
+ } else {
499
+ container.removeClass(COLLAPSED_CLASS);
500
+ let size: number | undefined;
501
+ if (this.state.expansion !== SidePanel.ExpansionState.expanded) {
502
+ if (this.state.lastPanelSize) {
503
+ size = this.state.lastPanelSize;
504
+ } else {
505
+ size = this.getDefaultPanelSize();
506
+ }
507
+ }
508
+ if (size) {
509
+ // Restore the panel size to the last known size or the default size
510
+ this.state.expansion = SidePanel.ExpansionState.expanding;
511
+ if (parent instanceof SplitPanel) {
512
+ relativeSizes = parent.relativeSizes();
513
+ }
514
+ this.setPanelSize(size).then(() => {
515
+ if (this.state.expansion === SidePanel.ExpansionState.expanding) {
516
+ this.state.expansion = SidePanel.ExpansionState.expanded;
517
+ }
518
+ });
519
+ } else {
520
+ this.state.expansion = SidePanel.ExpansionState.expanded;
521
+ }
522
+ }
523
+ container.setHidden(isEmpty && hideDockPanel);
524
+ tabBar.setHidden(isEmpty);
525
+ dockPanel.setHidden(hideDockPanel);
526
+ this.state.empty = isEmpty;
527
+ if (currentTitle) {
528
+ dockPanel.selectWidget(currentTitle.owner);
529
+ }
530
+ if (relativeSizes && parent instanceof SplitPanel) {
531
+ // Make sure that the expansion animation starts at the smallest possible size
532
+ parent.setRelativeSizes(relativeSizes);
533
+ }
534
+ }
535
+
536
+ /**
537
+ * Sets the size of the side panel.
538
+ *
539
+ * @param size the desired size (width) of the panel in pixels.
540
+ */
541
+ resize(size: number): void {
542
+ if (this.dockPanel.isHidden) {
543
+ this.state.lastPanelSize = size;
544
+ } else {
545
+ this.setPanelSize(size);
546
+ }
547
+ }
548
+
549
+ /**
550
+ * Compute the current width of the panel. This implementation assumes that the parent of
551
+ * the panel container is a `SplitPanel`.
552
+ */
553
+ protected getPanelSize(): number | undefined {
554
+ const parent = this.container.parent;
555
+ if (parent instanceof SplitPanel && parent.isVisible) {
556
+ const index = parent.widgets.indexOf(this.container);
557
+ if (this.side === 'left') {
558
+ const handle = parent.handles[index];
559
+ if (!handle.classList.contains('p-mod-hidden')) {
560
+ return handle.offsetLeft;
561
+ }
562
+ } else if (this.side === 'right') {
563
+ const handle = parent.handles[index - 1];
564
+ if (!handle.classList.contains('p-mod-hidden')) {
565
+ const parentWidth = parent.node.clientWidth;
566
+ return parentWidth - handle.offsetLeft;
567
+ }
568
+ }
569
+ }
570
+ }
571
+
572
+ /**
573
+ * Determine the default size to apply when the panel is expanded for the first time.
574
+ */
575
+ protected getDefaultPanelSize(): number | undefined {
576
+ const parent = this.container.parent;
577
+ if (parent && parent.isVisible) {
578
+ return parent.node.clientWidth * this.options.initialSizeRatio;
579
+ }
580
+ }
581
+
582
+ /**
583
+ * Modify the width of the panel. This implementation assumes that the parent of the panel
584
+ * container is a `SplitPanel`.
585
+ */
586
+ protected setPanelSize(size: number): Promise<void> {
587
+ const enableAnimation = this.applicationStateService.state === 'ready';
588
+ const options: SplitPositionOptions = {
589
+ side: this.side,
590
+ duration: enableAnimation ? this.options.expandDuration : 0,
591
+ referenceWidget: this.dockPanel
592
+ };
593
+ const promise = this.splitPositionHandler.setSidePanelSize(this.container, size, options);
594
+ const result = new Promise<void>(resolve => {
595
+ // Resolve the resulting promise in any case, regardless of whether resizing was successful
596
+ promise.then(() => resolve(), () => resolve());
597
+ });
598
+ this.state.pendingUpdate = this.state.pendingUpdate.then(() => result);
599
+ return result;
600
+ }
601
+
602
+ protected readonly toDisposeOnCurrentTabChanged = new DisposableCollection();
603
+
604
+ /**
605
+ * Handle a `currentChanged` signal from the sidebar. The side panel is refreshed so it displays
606
+ * the new selected widget.
607
+ */
608
+ protected onCurrentTabChanged(sender: SideTabBar, { currentTitle, currentIndex }: TabBar.ICurrentChangedArgs<Widget>): void {
609
+ this.toDisposeOnCurrentTabChanged.dispose();
610
+ if (currentTitle) {
611
+ this.updateToolbarTitle();
612
+ currentTitle.changed.connect(this.updateToolbarTitle);
613
+ this.toDisposeOnCurrentTabChanged.push(Disposable.create(() => currentTitle.changed.disconnect(this.updateToolbarTitle)));
614
+ }
615
+ if (currentIndex >= 0) {
616
+ this.state.lastActiveTabIndex = currentIndex;
617
+ sender.revealTab(currentIndex);
618
+ }
619
+ this.refresh();
620
+ }
621
+
622
+ /**
623
+ * Handle a `tabDetachRequested` signal from the sidebar. A drag is started so the widget can be
624
+ * moved to another application shell area.
625
+ */
626
+ protected onTabDetachRequested(sender: SideTabBar,
627
+ { title, tab, clientX, clientY }: TabBar.ITabDetachRequestedArgs<Widget>): void {
628
+ // Release the tab bar's hold on the mouse
629
+ sender.releaseMouse();
630
+
631
+ // Clone the selected tab and use that as drag image
632
+ const clonedTab = tab.cloneNode(true) as HTMLElement;
633
+ clonedTab.style.width = '';
634
+ clonedTab.style.height = '';
635
+ const label = clonedTab.getElementsByClassName('p-TabBar-tabLabel')[0] as HTMLElement;
636
+ label.style.width = '';
637
+ label.style.height = '';
638
+
639
+ // Create and start a drag to move the selected tab to another panel
640
+ const mimeData = new MimeData();
641
+ mimeData.setData('application/vnd.phosphor.widget-factory', () => title.owner);
642
+ const drag = new Drag({
643
+ mimeData,
644
+ dragImage: clonedTab,
645
+ proposedAction: 'move',
646
+ supportedActions: 'move',
647
+ });
648
+
649
+ tab.classList.add('p-mod-hidden');
650
+ drag.start(clientX, clientY).then(() => {
651
+ // The promise is resolved when the drag has ended
652
+ tab.classList.remove('p-mod-hidden');
653
+ });
654
+ }
655
+
656
+ protected onTabsOverflowChanged(sender: SideTabBar, event: { titles: Title<Widget>[], startIndex: number }): void {
657
+ if (event.startIndex > 0 && event.startIndex <= sender.currentIndex) {
658
+ sender.revealTab(sender.currentIndex);
659
+ } else {
660
+ this.additionalViewsMenu.updateAdditionalViews(sender, event);
661
+ }
662
+ }
663
+
664
+ /*
665
+ * Handle the `widgetAdded` signal from the dock panel. The widget's title is inserted into the
666
+ * tab bar according to the `rankProperty` value that may be attached to the widget.
667
+ */
668
+ protected onWidgetAdded(sender: DockPanel, widget: Widget): void {
669
+ const titles = this.tabBar.titles;
670
+ if (!find(titles, t => t.owner === widget)) {
671
+ const rank = SidePanelHandler.rankProperty.get(widget);
672
+ let index = titles.length;
673
+ if (rank !== undefined) {
674
+ for (let i = index - 1; i >= 0; i--) {
675
+ const r = SidePanelHandler.rankProperty.get(titles[i].owner);
676
+ if (r !== undefined && r > rank) {
677
+ index = i;
678
+ }
679
+ }
680
+ }
681
+ this.tabBar.insertTab(index, widget.title);
682
+ this.refresh();
683
+ }
684
+ }
685
+
686
+ /*
687
+ * Handle the `widgetRemoved` signal from the dock panel. The widget's title is also removed
688
+ * from the tab bar.
689
+ */
690
+ protected onWidgetRemoved(sender: DockPanel, widget: Widget): void {
691
+ this.tabBar.removeTab(widget.title);
692
+ this.refresh();
693
+ }
694
+
695
+ protected updateSashState(sidePanelElement: Panel | null, sidePanelCollapsed: boolean): void {
696
+ if (sidePanelElement) {
697
+ // Hide the sash when the left/right side panel is collapsed
698
+ if (sidePanelElement.id === 'theia-left-content-panel' && sidePanelElement.node.nextElementSibling) {
699
+ sidePanelElement.node.nextElementSibling.classList.toggle('sash-hidden', sidePanelCollapsed);
700
+ } else if (sidePanelElement.id === 'theia-right-content-panel' && sidePanelElement.node.previousElementSibling) {
701
+ sidePanelElement.node.previousElementSibling.classList.toggle('sash-hidden', sidePanelCollapsed);
702
+ }
703
+ }
704
+ }
705
+
706
+ }
707
+
708
+ export namespace SidePanel {
709
+ /**
710
+ * Options that control the behavior of side panels.
711
+ */
712
+ export interface Options {
713
+ /**
714
+ * When a widget is being dragged and the distance of the mouse cursor to the shell border
715
+ * is below this threshold, the respective side panel is expanded so the widget can be dropped
716
+ * into that panel. Set this to `-1` to disable expanding the side panel while dragging.
717
+ */
718
+ expandThreshold: number;
719
+ /**
720
+ * The duration in milliseconds of the animation shown when a side panel is expanded.
721
+ * Set this to `0` to disable expansion animation.
722
+ */
723
+ expandDuration: number;
724
+ /**
725
+ * The ratio of the available shell size to use as initial size for the side panel.
726
+ */
727
+ initialSizeRatio: number
728
+ /**
729
+ * How large the panel should be when it's expanded and empty.
730
+ */
731
+ emptySize: number;
732
+ }
733
+
734
+ /**
735
+ * The options for adding a widget to a side panel.
736
+ */
737
+ export interface WidgetOptions {
738
+ /**
739
+ * The rank order of the widget among its siblings.
740
+ */
741
+ rank?: number;
742
+ }
743
+
744
+ /**
745
+ * Data to save and load the layout of a side panel.
746
+ */
747
+ export interface LayoutData {
748
+ type: 'sidepanel',
749
+ items?: WidgetItem[];
750
+ size?: number;
751
+ }
752
+
753
+ /**
754
+ * Data structure used to save and restore the side panel layout.
755
+ */
756
+ export interface WidgetItem extends WidgetOptions {
757
+ /** Can be undefined in case the widget could not be restored. */
758
+ widget?: Widget;
759
+ expanded?: boolean;
760
+ pinned?: boolean;
761
+ }
762
+
763
+ export interface State {
764
+ /**
765
+ * Indicates whether the panel is empty.
766
+ */
767
+ empty: boolean;
768
+ /**
769
+ * Indicates whether the panel is expanded, collapsed, or in a transition between the two.
770
+ */
771
+ expansion: ExpansionState;
772
+ /**
773
+ * A promise that is resolved when the currently pending side panel updates are done.
774
+ */
775
+ pendingUpdate: Promise<void>;
776
+ /**
777
+ * The index of the last tab that was selected. When the panel is expanded, it tries to restore
778
+ * the tab selection to the previous state.
779
+ */
780
+ lastActiveTabIndex?: number;
781
+ /**
782
+ * The width or height of the panel before it was collapsed. When the panel is expanded, it tries
783
+ * to restore its size to this value.
784
+ */
785
+ lastPanelSize?: number;
786
+ }
787
+
788
+ export enum ExpansionState {
789
+ collapsed = 'collapsed',
790
+ expanding = 'expanding',
791
+ expanded = 'expanded',
792
+ collapsing = 'collapsing'
793
+ }
794
+ }