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

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 (689) hide show
  1. package/README.md +7 -7
  2. package/i18n/nls.cs.json +582 -552
  3. package/i18n/nls.de.json +582 -552
  4. package/i18n/nls.es.json +582 -552
  5. package/i18n/nls.fr.json +582 -552
  6. package/i18n/nls.hu.json +582 -552
  7. package/i18n/nls.it.json +582 -552
  8. package/i18n/nls.ja.json +582 -552
  9. package/i18n/nls.json +582 -552
  10. package/i18n/nls.ko.json +582 -0
  11. package/i18n/nls.pl.json +582 -552
  12. package/i18n/nls.pt-br.json +582 -552
  13. package/i18n/nls.ru.json +582 -552
  14. package/i18n/nls.tr.json +582 -0
  15. package/i18n/nls.zh-cn.json +582 -552
  16. package/i18n/nls.zh-tw.json +582 -0
  17. package/lib/browser/authentication-service.d.ts +15 -14
  18. package/lib/browser/authentication-service.d.ts.map +1 -1
  19. package/lib/browser/authentication-service.js +5 -5
  20. package/lib/browser/authentication-service.js.map +1 -1
  21. package/lib/browser/catalog.json +6889 -0
  22. package/lib/browser/common-frontend-contribution.js +3 -3
  23. package/lib/browser/common-styling-participants.js +166 -166
  24. package/lib/browser/context-key-service.d.ts +3 -2
  25. package/lib/browser/context-key-service.d.ts.map +1 -1
  26. package/lib/browser/context-key-service.js.map +1 -1
  27. package/lib/browser/core-preferences.d.ts.map +1 -1
  28. package/lib/browser/core-preferences.js +9 -0
  29. package/lib/browser/core-preferences.js.map +1 -1
  30. package/lib/browser/frontend-application-module.d.ts.map +1 -1
  31. package/lib/browser/frontend-application-module.js.map +1 -1
  32. package/lib/browser/json-schema-store.d.ts +0 -3
  33. package/lib/browser/json-schema-store.d.ts.map +1 -1
  34. package/lib/browser/json-schema-store.js +2 -12
  35. package/lib/browser/json-schema-store.js.map +1 -1
  36. package/lib/browser/open-with-service.d.ts +13 -1
  37. package/lib/browser/open-with-service.d.ts.map +1 -1
  38. package/lib/browser/open-with-service.js +48 -9
  39. package/lib/browser/open-with-service.js.map +1 -1
  40. package/lib/browser/opener-service.d.ts +8 -0
  41. package/lib/browser/opener-service.d.ts.map +1 -1
  42. package/lib/browser/opener-service.js +18 -3
  43. package/lib/browser/opener-service.js.map +1 -1
  44. package/lib/browser/progress-location-service.spec.js +7 -7
  45. package/lib/browser/saveable-service.d.ts.map +1 -1
  46. package/lib/browser/saveable-service.js +6 -2
  47. package/lib/browser/saveable-service.js.map +1 -1
  48. package/lib/browser/saveable.d.ts +17 -1
  49. package/lib/browser/saveable.d.ts.map +1 -1
  50. package/lib/browser/saveable.js +62 -1
  51. package/lib/browser/saveable.js.map +1 -1
  52. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-menu-adapters.d.ts +5 -4
  53. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-menu-adapters.d.ts.map +1 -1
  54. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-menu-adapters.js +2 -1
  55. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-menu-adapters.js.map +1 -1
  56. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-registry.d.ts +6 -16
  57. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-registry.d.ts.map +1 -1
  58. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-registry.js +12 -29
  59. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-registry.js.map +1 -1
  60. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-types.d.ts +43 -78
  61. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-types.d.ts.map +1 -1
  62. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-types.js +8 -39
  63. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-types.js.map +1 -1
  64. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar.d.ts +10 -10
  65. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar.d.ts.map +1 -1
  66. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar.js +43 -32
  67. package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar.js.map +1 -1
  68. package/lib/browser/storage-service.js +3 -3
  69. package/lib/browser/tree/tree.spec.js +75 -75
  70. package/lib/browser/view-container.d.ts +2 -2
  71. package/lib/browser/view-container.d.ts.map +1 -1
  72. package/lib/browser/view-container.js.map +1 -1
  73. package/lib/browser/widget-open-handler.d.ts +4 -1
  74. package/lib/browser/widget-open-handler.d.ts.map +1 -1
  75. package/lib/browser/widget-open-handler.js.map +1 -1
  76. package/lib/browser/widgets/extractable-widget.js +1 -1
  77. package/lib/browser/widgets/extractable-widget.js.map +1 -1
  78. package/lib/browser/widgets/index.d.ts +1 -0
  79. package/lib/browser/widgets/index.d.ts.map +1 -1
  80. package/lib/browser/widgets/index.js +1 -0
  81. package/lib/browser/widgets/index.js.map +1 -1
  82. package/lib/browser/widgets/split-widget.d.ts +45 -0
  83. package/lib/browser/widgets/split-widget.d.ts.map +1 -0
  84. package/lib/browser/widgets/split-widget.js +126 -0
  85. package/lib/browser/widgets/split-widget.js.map +1 -0
  86. package/lib/common/event.d.ts +2 -0
  87. package/lib/common/event.d.ts.map +1 -1
  88. package/lib/common/event.js +4 -0
  89. package/lib/common/event.js.map +1 -1
  90. package/lib/common/glob.d.ts +4 -4
  91. package/lib/common/json-schema.d.ts +2 -0
  92. package/lib/common/json-schema.d.ts.map +1 -1
  93. package/lib/common/menu/menu-types.d.ts.map +1 -1
  94. package/lib/common/menu/menu-types.js.map +1 -1
  95. package/lib/electron-browser/electron-uri-handler.d.ts +6 -0
  96. package/lib/electron-browser/electron-uri-handler.d.ts.map +1 -0
  97. package/lib/electron-browser/electron-uri-handler.js +49 -0
  98. package/lib/electron-browser/electron-uri-handler.js.map +1 -0
  99. package/lib/electron-browser/menu/electron-main-menu-factory.d.ts +1 -1
  100. package/lib/electron-browser/menu/electron-main-menu-factory.d.ts.map +1 -1
  101. package/lib/electron-browser/menu/electron-main-menu-factory.js +6 -6
  102. package/lib/electron-browser/menu/electron-main-menu-factory.js.map +1 -1
  103. package/lib/electron-browser/preload.d.ts.map +1 -1
  104. package/lib/electron-browser/preload.js +12 -0
  105. package/lib/electron-browser/preload.js.map +1 -1
  106. package/lib/electron-browser/window/electron-window-module.d.ts.map +1 -1
  107. package/lib/electron-browser/window/electron-window-module.js +3 -0
  108. package/lib/electron-browser/window/electron-window-module.js.map +1 -1
  109. package/lib/electron-browser/window/external-app-open-handler.js +1 -1
  110. package/lib/electron-browser/window/external-app-open-handler.js.map +1 -1
  111. package/lib/electron-common/electron-api.d.ts +2 -0
  112. package/lib/electron-common/electron-api.d.ts.map +1 -1
  113. package/lib/electron-common/electron-api.js +2 -1
  114. package/lib/electron-common/electron-api.js.map +1 -1
  115. package/lib/electron-main/electron-api-main.d.ts +2 -0
  116. package/lib/electron-main/electron-api-main.d.ts.map +1 -1
  117. package/lib/electron-main/electron-api-main.js +27 -3
  118. package/lib/electron-main/electron-api-main.js.map +1 -1
  119. package/lib/electron-main/electron-main-application.d.ts +5 -3
  120. package/lib/electron-main/electron-main-application.d.ts.map +1 -1
  121. package/lib/electron-main/electron-main-application.js +57 -14
  122. package/lib/electron-main/electron-main-application.js.map +1 -1
  123. package/lib/electron-main/theia-electron-window.d.ts +1 -0
  124. package/lib/electron-main/theia-electron-window.d.ts.map +1 -1
  125. package/lib/electron-main/theia-electron-window.js +3 -0
  126. package/lib/electron-main/theia-electron-window.js.map +1 -1
  127. package/lib/node/i18n/theia-localization-contribution.d.ts.map +1 -1
  128. package/lib/node/i18n/theia-localization-contribution.js +12 -8
  129. package/lib/node/i18n/theia-localization-contribution.js.map +1 -1
  130. package/lib/node/process-utils.spec.js +8 -8
  131. package/package.json +10 -8
  132. package/shared/ajv/index.d.ts +2 -2
  133. package/shared/markdown-it.d.ts +2 -2
  134. package/shared/markdown-it.js +1 -1
  135. package/shared/reflect-metadata/index.d.ts +1 -1
  136. package/shared/reflect-metadata/index.js +1 -1
  137. package/shared/vscode-languageserver-types/index.d.ts +1 -1
  138. package/src/browser/about-dialog.tsx +137 -137
  139. package/src/browser/authentication-service.ts +467 -468
  140. package/src/browser/breadcrumbs/breadcrumb-popup-container.ts +101 -101
  141. package/src/browser/breadcrumbs/breadcrumb-renderer.tsx +41 -41
  142. package/src/browser/breadcrumbs/breadcrumbs-constants.ts +79 -79
  143. package/src/browser/breadcrumbs/breadcrumbs-renderer.tsx +185 -185
  144. package/src/browser/breadcrumbs/breadcrumbs-service.ts +108 -108
  145. package/src/browser/breadcrumbs/index.ts +21 -21
  146. package/src/browser/browser-clipboard-service.ts +122 -122
  147. package/src/browser/browser.ts +239 -239
  148. package/src/browser/clipboard-service.ts +23 -23
  149. package/src/browser/color-application-contribution.ts +110 -110
  150. package/src/browser/color-registry.ts +60 -60
  151. package/src/browser/command-open-handler.ts +54 -54
  152. package/src/browser/common-frontend-contribution.ts +2680 -2680
  153. package/src/browser/common-styling-participants.ts +361 -361
  154. package/src/browser/connection-status-service.spec.ts +200 -200
  155. package/src/browser/connection-status-service.ts +216 -216
  156. package/src/browser/context-key-service.ts +142 -142
  157. package/src/browser/context-menu-renderer.ts +124 -124
  158. package/src/browser/core-preferences.ts +343 -334
  159. package/src/browser/credentials-service.ts +106 -106
  160. package/src/browser/decoration-style.ts +65 -65
  161. package/src/browser/decorations-service.ts +209 -209
  162. package/src/browser/dialogs/react-dialog.tsx +56 -56
  163. package/src/browser/dialogs.ts +534 -534
  164. package/src/browser/diff-uris.ts +117 -117
  165. package/src/browser/encoding-registry.ts +97 -97
  166. package/src/browser/endpoint.spec.ts +148 -148
  167. package/src/browser/endpoint.ts +136 -136
  168. package/src/browser/external-uri-service.ts +79 -79
  169. package/src/browser/file-icons-js.d.ts +20 -20
  170. package/src/browser/frontend-application-bindings.ts +62 -62
  171. package/src/browser/frontend-application-config-provider.spec.ts +45 -45
  172. package/src/browser/frontend-application-config-provider.ts +50 -50
  173. package/src/browser/frontend-application-contribution.ts +110 -110
  174. package/src/browser/frontend-application-module.ts +474 -475
  175. package/src/browser/frontend-application-state.ts +74 -74
  176. package/src/browser/frontend-application.ts +326 -326
  177. package/src/browser/hover-service.ts +218 -218
  178. package/src/browser/http-open-handler.ts +49 -49
  179. package/src/browser/i18n/i18n-frontend-module.ts +27 -27
  180. package/src/browser/i18n/language-quick-pick-service.ts +130 -130
  181. package/src/browser/icon-registry.ts +87 -87
  182. package/src/browser/icon-theme-contribution.ts +64 -64
  183. package/src/browser/icon-theme-service.ts +217 -217
  184. package/src/browser/icons/CollapseAll.svg +7 -7
  185. package/src/browser/icons/CollapseAll_inverse.svg +7 -7
  186. package/src/browser/icons/Refresh.svg +7 -7
  187. package/src/browser/icons/Refresh_inverse.svg +7 -7
  188. package/src/browser/icons/add-inverse.svg +4 -4
  189. package/src/browser/icons/add.svg +4 -4
  190. package/src/browser/icons/arrow-down-bright.svg +6 -6
  191. package/src/browser/icons/arrow-down-dark.svg +6 -6
  192. package/src/browser/icons/arrow-up-bright.svg +6 -6
  193. package/src/browser/icons/arrow-up-dark.svg +6 -6
  194. package/src/browser/icons/case-sensitive-dark.svg +16 -16
  195. package/src/browser/icons/case-sensitive.svg +16 -16
  196. package/src/browser/icons/chevron-right-dark.svg +5 -5
  197. package/src/browser/icons/chevron-right-light.svg +6 -6
  198. package/src/browser/icons/circle-bright.svg +7 -7
  199. package/src/browser/icons/circle-dark.svg +7 -7
  200. package/src/browser/icons/clear-search-results-dark.svg +7 -7
  201. package/src/browser/icons/clear-search-results.svg +7 -7
  202. package/src/browser/icons/close-all-bright.svg +7 -7
  203. package/src/browser/icons/close-all-dark.svg +7 -7
  204. package/src/browser/icons/close-bright.svg +7 -7
  205. package/src/browser/icons/close-dark.svg +7 -7
  206. package/src/browser/icons/collapse.svg +4 -4
  207. package/src/browser/icons/edit-json-dark.svg +6 -6
  208. package/src/browser/icons/edit-json.svg +6 -6
  209. package/src/browser/icons/expand.svg +4 -4
  210. package/src/browser/icons/loading-dark.svg +6 -6
  211. package/src/browser/icons/loading-light.svg +6 -6
  212. package/src/browser/icons/open-change-bright.svg +3 -3
  213. package/src/browser/icons/open-change-dark.svg +4 -4
  214. package/src/browser/icons/open-file-bright.svg +4 -4
  215. package/src/browser/icons/open-file-dark.svg +4 -4
  216. package/src/browser/icons/preview-bright.svg +3 -3
  217. package/src/browser/icons/preview-dark.svg +3 -3
  218. package/src/browser/icons/regex-dark.svg +10 -10
  219. package/src/browser/icons/regex.svg +10 -10
  220. package/src/browser/icons/remove-all-inverse.svg +4 -4
  221. package/src/browser/icons/remove-all.svg +4 -4
  222. package/src/browser/icons/replace-all-inverse.svg +13 -13
  223. package/src/browser/icons/replace-all.svg +13 -13
  224. package/src/browser/icons/replace-inverse.svg +15 -15
  225. package/src/browser/icons/replace.svg +15 -15
  226. package/src/browser/icons/whole-word-dark.svg +19 -19
  227. package/src/browser/icons/whole-word.svg +19 -19
  228. package/src/browser/index.ts +50 -50
  229. package/src/browser/json-schema-store.ts +118 -127
  230. package/src/browser/keybinding.spec.ts +554 -554
  231. package/src/browser/keybinding.ts +759 -759
  232. package/src/browser/keyboard/browser-keyboard-frontend-contribution.ts +108 -108
  233. package/src/browser/keyboard/browser-keyboard-layout-provider.spec.ts +171 -171
  234. package/src/browser/keyboard/browser-keyboard-layout-provider.ts +469 -469
  235. package/src/browser/keyboard/browser-keyboard-module.ts +30 -30
  236. package/src/browser/keyboard/index.ts +20 -20
  237. package/src/browser/keyboard/keyboard-layout-service.spec.ts +121 -121
  238. package/src/browser/keyboard/keyboard-layout-service.ts +455 -455
  239. package/src/browser/keyboard/keys.spec.ts +258 -258
  240. package/src/browser/keyboard/keys.ts +20 -20
  241. package/src/browser/keys.ts +21 -21
  242. package/src/browser/label-parser.spec.ts +165 -165
  243. package/src/browser/label-parser.ts +108 -108
  244. package/src/browser/label-provider.spec.ts +62 -62
  245. package/src/browser/label-provider.ts +385 -385
  246. package/src/browser/language-icon-provider.ts +55 -55
  247. package/src/browser/language-service.ts +77 -77
  248. package/src/browser/logger-frontend-module.ts +65 -65
  249. package/src/browser/markdown-rendering/markdown-renderer.ts +98 -98
  250. package/src/browser/menu/browser-context-menu-renderer.ts +48 -48
  251. package/src/browser/menu/browser-menu-module.ts +28 -28
  252. package/src/browser/menu/browser-menu-plugin.ts +491 -491
  253. package/src/browser/menu/context-menu-context.ts +41 -41
  254. package/src/browser/messaging/connection-source.ts +26 -26
  255. package/src/browser/messaging/frontend-id-provider.ts +37 -37
  256. package/src/browser/messaging/index.ts +18 -18
  257. package/src/browser/messaging/messaging-frontend-module.ts +41 -41
  258. package/src/browser/messaging/service-connection-provider.ts +140 -140
  259. package/src/browser/messaging/ws-connection-provider.ts +49 -49
  260. package/src/browser/messaging/ws-connection-source.ts +230 -230
  261. package/src/browser/mime-service.ts +30 -30
  262. package/src/browser/navigatable-types.ts +81 -81
  263. package/src/browser/navigatable.ts +39 -39
  264. package/src/browser/open-with-service.ts +140 -93
  265. package/src/browser/opener-service.spec.ts +49 -49
  266. package/src/browser/opener-service.ts +169 -146
  267. package/src/browser/performance/frontend-stopwatch.ts +65 -65
  268. package/src/browser/performance/index.ts +18 -18
  269. package/src/browser/performance/measurement-frontend-bindings.ts +31 -31
  270. package/src/browser/preferences/index.ts +23 -23
  271. package/src/browser/preferences/injectable-preference-proxy.ts +283 -283
  272. package/src/browser/preferences/preference-configurations.ts +82 -82
  273. package/src/browser/preferences/preference-contribution.ts +436 -436
  274. package/src/browser/preferences/preference-language-override-service.ts +111 -111
  275. package/src/browser/preferences/preference-provider.spec.ts +36 -36
  276. package/src/browser/preferences/preference-provider.ts +277 -277
  277. package/src/browser/preferences/preference-proxy.spec.ts +367 -367
  278. package/src/browser/preferences/preference-proxy.ts +367 -367
  279. package/src/browser/preferences/preference-schema-provider.spec.ts +130 -130
  280. package/src/browser/preferences/preference-scope.ts +18 -18
  281. package/src/browser/preferences/preference-service.spec.ts +613 -613
  282. package/src/browser/preferences/preference-service.ts +594 -594
  283. package/src/browser/preferences/preference-validation-service.spec.ts +334 -334
  284. package/src/browser/preferences/preference-validation-service.ts +358 -358
  285. package/src/browser/preferences/test/index.ts +19 -19
  286. package/src/browser/preferences/test/mock-preference-provider.ts +50 -50
  287. package/src/browser/preferences/test/mock-preference-proxy.ts +48 -48
  288. package/src/browser/preferences/test/mock-preference-service.ts +63 -63
  289. package/src/browser/preload/i18n-preload-contribution.ts +50 -50
  290. package/src/browser/preload/os-preload-contribution.ts +37 -37
  291. package/src/browser/preload/preload-module.ts +45 -45
  292. package/src/browser/preload/preloader.ts +37 -37
  293. package/src/browser/preload/theme-preload-contribution.ts +31 -31
  294. package/src/browser/progress-bar-factory.ts +29 -29
  295. package/src/browser/progress-bar.ts +76 -76
  296. package/src/browser/progress-client.ts +53 -53
  297. package/src/browser/progress-location-service.spec.ts +50 -50
  298. package/src/browser/progress-location-service.ts +96 -96
  299. package/src/browser/progress-status-bar-item.ts +83 -83
  300. package/src/browser/quick-input/index.ts +23 -23
  301. package/src/browser/quick-input/quick-access.ts +75 -75
  302. package/src/browser/quick-input/quick-command-frontend-contribution.ts +89 -89
  303. package/src/browser/quick-input/quick-command-service.ts +246 -246
  304. package/src/browser/quick-input/quick-help-service.ts +87 -87
  305. package/src/browser/quick-input/quick-input-frontend-contribution.ts +33 -33
  306. package/src/browser/quick-input/quick-input-service.spec.ts +176 -176
  307. package/src/browser/quick-input/quick-input-service.ts +17 -17
  308. package/src/browser/quick-input/quick-pick-service-impl.ts +69 -69
  309. package/src/browser/quick-input/quick-view-service.ts +83 -83
  310. package/src/browser/request/browser-request-module.ts +23 -23
  311. package/src/browser/request/browser-request-service.ts +172 -172
  312. package/src/browser/resource-context-key.ts +77 -77
  313. package/src/browser/saveable-service.ts +332 -328
  314. package/src/browser/saveable.ts +395 -327
  315. package/src/browser/secondary-window-handler.ts +211 -211
  316. package/src/browser/shell/additional-views-menu-widget.tsx +71 -71
  317. package/src/browser/shell/application-shell-mouse-tracker.ts +103 -103
  318. package/src/browser/shell/application-shell.ts +2271 -2271
  319. package/src/browser/shell/current-widget-command-adapter.ts +57 -57
  320. package/src/browser/shell/index.ts +23 -23
  321. package/src/browser/shell/shell-layout-restorer.ts +399 -399
  322. package/src/browser/shell/side-panel-handler.ts +794 -794
  323. package/src/browser/shell/side-panel-toolbar.ts +111 -111
  324. package/src/browser/shell/sidebar-bottom-menu-widget.tsx +39 -39
  325. package/src/browser/shell/sidebar-menu-widget.tsx +183 -183
  326. package/src/browser/shell/sidebar-top-menu-widget.tsx +26 -26
  327. package/src/browser/shell/split-panels.ts +191 -191
  328. package/src/browser/shell/tab-bar-decorator.ts +106 -106
  329. package/src/browser/shell/tab-bar-toolbar/index.ts +19 -19
  330. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar-menu-adapters.ts +31 -31
  331. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar-registry.ts +242 -256
  332. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar-types.ts +149 -207
  333. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar.spec.ts +62 -62
  334. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar.tsx +443 -428
  335. package/src/browser/shell/tab-bars.spec.ts +63 -63
  336. package/src/browser/shell/tab-bars.ts +1468 -1468
  337. package/src/browser/shell/theia-dock-panel.ts +265 -265
  338. package/src/browser/shell/view-column-service.ts +125 -125
  339. package/src/browser/shell/view-contribution.ts +178 -178
  340. package/src/browser/source-tree/index.ts +19 -19
  341. package/src/browser/source-tree/source-tree-widget.tsx +107 -107
  342. package/src/browser/source-tree/source-tree.ts +146 -146
  343. package/src/browser/source-tree/tree-source.ts +73 -73
  344. package/src/browser/status-bar/index.ts +29 -29
  345. package/src/browser/status-bar/status-bar-types.ts +97 -97
  346. package/src/browser/status-bar/status-bar-view-model.ts +209 -209
  347. package/src/browser/status-bar/status-bar.tsx +189 -189
  348. package/src/browser/storage-service.spec.ts +76 -76
  349. package/src/browser/storage-service.ts +129 -129
  350. package/src/browser/style/about.css +36 -36
  351. package/src/browser/style/alert-messages.css +62 -62
  352. package/src/browser/style/ansi.css +88 -88
  353. package/src/browser/style/breadcrumbs.css +130 -130
  354. package/src/browser/style/dialog.css +126 -126
  355. package/src/browser/style/dockpanel.css +76 -76
  356. package/src/browser/style/hover-service.css +101 -101
  357. package/src/browser/style/icons.css +61 -61
  358. package/src/browser/style/index.css +353 -352
  359. package/src/browser/style/materialcolors.css +278 -278
  360. package/src/browser/style/menus.css +230 -230
  361. package/src/browser/style/notification.css +39 -39
  362. package/src/browser/style/os.css +87 -87
  363. package/src/browser/style/progress-bar.css +43 -43
  364. package/src/browser/style/quick-title-bar.css +45 -45
  365. package/src/browser/style/scrollbars.css +172 -172
  366. package/src/browser/style/search-box.css +123 -123
  367. package/src/browser/style/select-component.css +107 -107
  368. package/src/browser/style/sidepanel.css +367 -367
  369. package/src/browser/style/split-widget.css +38 -0
  370. package/src/browser/style/status-bar.css +127 -127
  371. package/src/browser/style/tabs.css +647 -658
  372. package/src/browser/style/tooltip.css +28 -28
  373. package/src/browser/style/tree-decorators.css +81 -81
  374. package/src/browser/style/tree.css +232 -232
  375. package/src/browser/style/view-container.css +187 -194
  376. package/src/browser/style/widget.css +19 -19
  377. package/src/browser/styling-service.ts +96 -96
  378. package/src/browser/supported-encodings.ts +262 -262
  379. package/src/browser/test/jsdom.ts +69 -69
  380. package/src/browser/test/mock-connection-status-service.ts +33 -33
  381. package/src/browser/test/mock-env-variables-server.ts +47 -47
  382. package/src/browser/test/mock-opener-service.ts +34 -34
  383. package/src/browser/test/mock-storage-service.ts +49 -49
  384. package/src/browser/theming.ts +206 -206
  385. package/src/browser/tooltip-service.tsx +96 -96
  386. package/src/browser/tree/fuzzy-search.spec.ts +99 -99
  387. package/src/browser/tree/fuzzy-search.ts +136 -136
  388. package/src/browser/tree/index.ts +29 -29
  389. package/src/browser/tree/search-box-debounce.ts +96 -96
  390. package/src/browser/tree/search-box.ts +355 -355
  391. package/src/browser/tree/test/mock-selectable-tree-model.ts +109 -109
  392. package/src/browser/tree/test/mock-tree-model.ts +136 -136
  393. package/src/browser/tree/test/tree-test-container.ts +50 -50
  394. package/src/browser/tree/tree-compression/compressed-tree-expansion-service.ts +46 -46
  395. package/src/browser/tree/tree-compression/compressed-tree-model.ts +88 -88
  396. package/src/browser/tree/tree-compression/compressed-tree-widget.tsx +203 -203
  397. package/src/browser/tree/tree-compression/index.ts +20 -20
  398. package/src/browser/tree/tree-compression/tree-compression-service.ts +125 -125
  399. package/src/browser/tree/tree-compression/tree-compression.css +28 -28
  400. package/src/browser/tree/tree-consistency.spec.ts +105 -105
  401. package/src/browser/tree/tree-container.spec.ts +45 -45
  402. package/src/browser/tree/tree-container.ts +155 -155
  403. package/src/browser/tree/tree-decorator.spec.ts +162 -162
  404. package/src/browser/tree/tree-decorator.ts +238 -238
  405. package/src/browser/tree/tree-expansion.spec.ts +173 -173
  406. package/src/browser/tree/tree-expansion.ts +165 -165
  407. package/src/browser/tree/tree-focus-service.ts +55 -55
  408. package/src/browser/tree/tree-iterator.spec.ts +170 -170
  409. package/src/browser/tree/tree-iterator.ts +256 -256
  410. package/src/browser/tree/tree-label-provider.ts +40 -40
  411. package/src/browser/tree/tree-model.ts +562 -562
  412. package/src/browser/tree/tree-navigation.ts +58 -58
  413. package/src/browser/tree/tree-preference.ts +50 -50
  414. package/src/browser/tree/tree-search.ts +128 -128
  415. package/src/browser/tree/tree-selectable.spec.ts +152 -152
  416. package/src/browser/tree/tree-selection-impl.ts +176 -176
  417. package/src/browser/tree/tree-selection-state.spec.ts +462 -462
  418. package/src/browser/tree/tree-selection-state.ts +245 -245
  419. package/src/browser/tree/tree-selection.ts +159 -159
  420. package/src/browser/tree/tree-view-welcome-widget.tsx +263 -263
  421. package/src/browser/tree/tree-widget-selection.ts +45 -45
  422. package/src/browser/tree/tree-widget.tsx +1591 -1591
  423. package/src/browser/tree/tree.spec.ts +241 -241
  424. package/src/browser/tree/tree.ts +425 -425
  425. package/src/browser/undo-redo-handler.ts +85 -85
  426. package/src/browser/user-working-directory-provider.ts +77 -77
  427. package/src/browser/view-container.ts +1640 -1640
  428. package/src/browser/widget-decoration.ts +358 -358
  429. package/src/browser/widget-manager.spec.ts +102 -102
  430. package/src/browser/widget-manager.ts +318 -318
  431. package/src/browser/widget-open-handler.ts +168 -165
  432. package/src/browser/widgets/alert-message.tsx +56 -56
  433. package/src/browser/widgets/enhanced-preview-widget.ts +27 -27
  434. package/src/browser/widgets/extractable-widget.ts +33 -33
  435. package/src/browser/widgets/index.ts +21 -20
  436. package/src/browser/widgets/previewable-widget.ts +31 -31
  437. package/src/browser/widgets/react-renderer.tsx +53 -53
  438. package/src/browser/widgets/react-widget.tsx +51 -51
  439. package/src/browser/widgets/select-component.tsx +367 -367
  440. package/src/browser/widgets/split-widget.ts +163 -0
  441. package/src/browser/widgets/widget.ts +406 -406
  442. package/src/browser/window/browser-window-module.ts +32 -32
  443. package/src/browser/window/default-secondary-window-service.ts +189 -189
  444. package/src/browser/window/default-window-service.spec.ts +78 -78
  445. package/src/browser/window/default-window-service.ts +171 -171
  446. package/src/browser/window/secondary-window-service.ts +39 -39
  447. package/src/browser/window/test/mock-window-service.ts +29 -29
  448. package/src/browser/window/window-service.ts +78 -78
  449. package/src/browser/window/window-title-service.ts +107 -107
  450. package/src/browser/window/window-title-updater.ts +95 -95
  451. package/src/browser/window-contribution.ts +64 -64
  452. package/src/browser-only/frontend-only-application-module.ts +116 -116
  453. package/src/browser-only/i18n/i18n-frontend-only-module.ts +37 -37
  454. package/src/browser-only/logger-frontend-only-module.ts +63 -63
  455. package/src/browser-only/messaging/frontend-only-service-connection-provider.ts +39 -39
  456. package/src/browser-only/messaging/messaging-frontend-only-module.ts +42 -42
  457. package/src/browser-only/preload/frontend-only-preload-module.ts +49 -49
  458. package/src/common/accessibility.ts +33 -33
  459. package/src/common/application-error.spec.ts +27 -27
  460. package/src/common/application-error.ts +76 -76
  461. package/src/common/application-protocol.ts +42 -42
  462. package/src/common/array-utils.ts +129 -129
  463. package/src/common/buffer.ts +228 -228
  464. package/src/common/cancellation.ts +163 -163
  465. package/src/common/char-code.ts +438 -438
  466. package/src/common/collections.ts +125 -125
  467. package/src/common/color.ts +103 -103
  468. package/src/common/command.spec.ts +208 -208
  469. package/src/common/command.ts +489 -489
  470. package/src/common/contribution-filter/contribution-filter-registry.ts +79 -79
  471. package/src/common/contribution-filter/contribution-filter.ts +64 -64
  472. package/src/common/contribution-filter/filter.ts +23 -23
  473. package/src/common/contribution-filter/index.ts +19 -19
  474. package/src/common/contribution-provider.ts +96 -96
  475. package/src/common/disposable.spec.ts +94 -94
  476. package/src/common/disposable.ts +188 -188
  477. package/src/common/encoding-service.ts +380 -380
  478. package/src/common/encodings.ts +24 -24
  479. package/src/common/env-variables/env-variables-protocol.ts +38 -38
  480. package/src/common/env-variables/index.ts +17 -17
  481. package/src/common/event.spec.ts +32 -32
  482. package/src/common/event.ts +493 -487
  483. package/src/common/file-uri.ts +61 -61
  484. package/src/common/frontend-application-state.ts +38 -38
  485. package/src/common/glob.ts +741 -741
  486. package/src/common/hash.ts +85 -85
  487. package/src/common/i18n/localization-server.ts +25 -25
  488. package/src/common/i18n/localization.ts +80 -80
  489. package/src/common/i18n/nls.metadata.json +34112 -34112
  490. package/src/common/index.ts +51 -51
  491. package/src/common/json-schema.ts +108 -106
  492. package/src/common/key-store.ts +26 -26
  493. package/src/common/keybinding.ts +152 -152
  494. package/src/common/keyboard/keyboard-layout-provider.ts +51 -51
  495. package/src/common/keys.ts +694 -694
  496. package/src/common/label-protocol.ts +35 -35
  497. package/src/common/logger-protocol.ts +119 -119
  498. package/src/common/logger-watcher.ts +48 -48
  499. package/src/common/logger.spec.ts +46 -46
  500. package/src/common/logger.ts +389 -389
  501. package/src/common/lsp-types.ts +34 -34
  502. package/src/common/markdown-rendering/icon-utilities.ts +30 -30
  503. package/src/common/markdown-rendering/index.ts +18 -18
  504. package/src/common/markdown-rendering/markdown-string.ts +152 -152
  505. package/src/common/menu/action-menu-node.ts +65 -65
  506. package/src/common/menu/composite-menu-node.spec.ts +67 -67
  507. package/src/common/menu/composite-menu-node.ts +114 -114
  508. package/src/common/menu/index.ts +21 -21
  509. package/src/common/menu/menu-adapter.ts +103 -103
  510. package/src/common/menu/menu-model-registry.ts +374 -374
  511. package/src/common/menu/menu-types.ts +220 -219
  512. package/src/common/menu/menu.spec.ts +101 -101
  513. package/src/common/message-rpc/channel.spec.ts +88 -88
  514. package/src/common/message-rpc/channel.ts +300 -300
  515. package/src/common/message-rpc/index.ts +22 -22
  516. package/src/common/message-rpc/message-buffer.ts +105 -105
  517. package/src/common/message-rpc/msg-pack-extension-manager.ts +70 -70
  518. package/src/common/message-rpc/rpc-message-encoder.spec.ts +65 -65
  519. package/src/common/message-rpc/rpc-message-encoder.ts +190 -190
  520. package/src/common/message-rpc/rpc-protocol.ts +255 -255
  521. package/src/common/message-rpc/uint8-array-message-buffer.spec.ts +41 -41
  522. package/src/common/message-rpc/uint8-array-message-buffer.ts +213 -213
  523. package/src/common/message-service-protocol.ts +148 -148
  524. package/src/common/message-service.ts +226 -226
  525. package/src/common/messaging/connection-error-handler.ts +73 -73
  526. package/src/common/messaging/connection-management.ts +43 -43
  527. package/src/common/messaging/handler.ts +26 -26
  528. package/src/common/messaging/index.ts +19 -19
  529. package/src/common/messaging/proxy-factory.spec.ts +108 -108
  530. package/src/common/messaging/proxy-factory.ts +336 -336
  531. package/src/common/messaging/socket-write-buffer.ts +52 -52
  532. package/src/common/messaging/web-socket-channel.ts +76 -76
  533. package/src/common/nls.ts +151 -151
  534. package/src/common/numbers.ts +21 -21
  535. package/src/common/objects.spec.ts +112 -112
  536. package/src/common/objects.ts +123 -123
  537. package/src/common/os.ts +82 -82
  538. package/src/common/path.spec.ts +415 -415
  539. package/src/common/path.ts +334 -334
  540. package/src/common/paths.ts +250 -250
  541. package/src/common/performance/index.ts +19 -19
  542. package/src/common/performance/measurement-protocol.ts +104 -104
  543. package/src/common/performance/measurement.ts +130 -130
  544. package/src/common/performance/stopwatch.ts +183 -183
  545. package/src/common/preferences/preference-schema.ts +101 -101
  546. package/src/common/preferences/preference-scope.spec.ts +48 -48
  547. package/src/common/preferences/preference-scope.ts +68 -68
  548. package/src/common/prioritizeable.ts +58 -58
  549. package/src/common/progress-service-protocol.ts +35 -35
  550. package/src/common/progress-service.ts +82 -82
  551. package/src/common/promise-util.spec.ts +102 -102
  552. package/src/common/promise-util.ts +143 -143
  553. package/src/common/quick-pick-service.ts +353 -353
  554. package/src/common/reference.spec.ts +145 -145
  555. package/src/common/reference.ts +230 -230
  556. package/src/common/resource.ts +430 -430
  557. package/src/common/selection-command-handler.ts +101 -101
  558. package/src/common/selection-service.spec.ts +43 -43
  559. package/src/common/selection-service.ts +49 -49
  560. package/src/common/selection.ts +50 -50
  561. package/src/common/severity.ts +111 -111
  562. package/src/common/stream.ts +718 -718
  563. package/src/common/strings.ts +231 -231
  564. package/src/common/telemetry.ts +45 -45
  565. package/src/common/ternary-search-tree.ts +417 -417
  566. package/src/common/test/expect.ts +34 -34
  567. package/src/common/test/mock-logger.ts +118 -118
  568. package/src/common/test/mock-menu.ts +35 -35
  569. package/src/common/test/mock-resource-provider.ts +33 -33
  570. package/src/common/theme.ts +68 -68
  571. package/src/common/types.spec.ts +86 -86
  572. package/src/common/types.ts +140 -140
  573. package/src/common/uri-command-handler.spec.ts +90 -90
  574. package/src/common/uri-command-handler.ts +148 -148
  575. package/src/common/uri.spec.ts +278 -278
  576. package/src/common/uri.ts +279 -279
  577. package/src/common/uuid.ts +45 -45
  578. package/src/common/version.ts +17 -17
  579. package/src/common/view-column.ts +33 -33
  580. package/src/common/window.ts +34 -34
  581. package/src/electron-browser/electron-clipboard-service.ts +32 -32
  582. package/src/electron-browser/electron-uri-handler.ts +42 -0
  583. package/src/electron-browser/keyboard/electron-keyboard-layout-change-notifier.ts +39 -39
  584. package/src/electron-browser/keyboard/electron-keyboard-module.ts +28 -28
  585. package/src/electron-browser/menu/electron-context-menu-renderer.ts +122 -122
  586. package/src/electron-browser/menu/electron-main-menu-factory.ts +339 -338
  587. package/src/electron-browser/menu/electron-menu-contribution.ts +506 -506
  588. package/src/electron-browser/menu/electron-menu-module.ts +40 -40
  589. package/src/electron-browser/menu/electron-menu-style.css +110 -110
  590. package/src/electron-browser/messaging/electron-frontend-id-provider.ts +25 -25
  591. package/src/electron-browser/messaging/electron-ipc-connection-source.ts +65 -65
  592. package/src/electron-browser/messaging/electron-local-ws-connection-source.ts +45 -45
  593. package/src/electron-browser/messaging/electron-messaging-frontend-module.ts +78 -78
  594. package/src/electron-browser/messaging/electron-ws-connection-source.ts +38 -38
  595. package/src/electron-browser/preload.ts +264 -249
  596. package/src/electron-browser/request/electron-browser-request-module.ts +26 -26
  597. package/src/electron-browser/token/electron-token-frontend-module.ts +22 -22
  598. package/src/electron-browser/window/electron-frontend-application-state.ts +26 -26
  599. package/src/electron-browser/window/electron-secondary-window-service.ts +35 -35
  600. package/src/electron-browser/window/electron-window-module.ts +48 -45
  601. package/src/electron-browser/window/electron-window-preferences.ts +76 -76
  602. package/src/electron-browser/window/electron-window-service.ts +109 -109
  603. package/src/electron-browser/window/external-app-open-handler.ts +42 -42
  604. package/src/electron-common/electron-api.ts +157 -154
  605. package/src/electron-common/electron-main-window-service.ts +24 -24
  606. package/src/electron-common/electron-token.ts +27 -27
  607. package/src/electron-main/electron-api-main.ts +373 -347
  608. package/src/electron-main/electron-main-application-module.ts +65 -65
  609. package/src/electron-main/electron-main-application.ts +860 -818
  610. package/src/electron-main/electron-main-constants.ts +23 -23
  611. package/src/electron-main/electron-main-window-service-impl.ts +44 -44
  612. package/src/electron-main/electron-security-token-service.ts +36 -36
  613. package/src/electron-main/event-utils.ts +36 -36
  614. package/src/electron-main/messaging/electron-connection-handler.ts +21 -21
  615. package/src/electron-main/messaging/electron-messaging-contribution.ts +143 -143
  616. package/src/electron-main/messaging/electron-messaging-service.ts +35 -35
  617. package/src/electron-main/theia-electron-window.ts +219 -214
  618. package/src/electron-node/cli/electron-backend-cli-module.ts +24 -24
  619. package/src/electron-node/cli/electron-cli-contribution.ts +35 -35
  620. package/src/electron-node/hosting/electron-backend-hosting-module.ts +24 -24
  621. package/src/electron-node/hosting/electron-ws-origin-validator.ts +37 -37
  622. package/src/electron-node/keyboard/electron-backend-keyboard-module.ts +30 -30
  623. package/src/electron-node/keyboard/electron-keyboard-layout-provider.ts +35 -35
  624. package/src/electron-node/request/electron-backend-request-module.ts +23 -23
  625. package/src/electron-node/request/electron-backend-request-service.ts +78 -78
  626. package/src/electron-node/token/electron-token-backend-contribution.ts +48 -48
  627. package/src/electron-node/token/electron-token-backend-module.ts +28 -28
  628. package/src/electron-node/token/electron-token-validator.ts +93 -93
  629. package/src/node/application-server.ts +59 -59
  630. package/src/node/backend-application-config-provider.spec.ts +29 -29
  631. package/src/node/backend-application-config-provider.ts +48 -48
  632. package/src/node/backend-application-module.ts +139 -139
  633. package/src/node/backend-application.ts +374 -374
  634. package/src/node/cli.spec.ts +94 -94
  635. package/src/node/cli.ts +63 -63
  636. package/src/node/console-logger-server.spec.ts +59 -59
  637. package/src/node/console-logger-server.ts +76 -76
  638. package/src/node/debug.ts +30 -30
  639. package/src/node/dynamic-require.ts +56 -56
  640. package/src/node/env-variables/env-variables-server.ts +123 -123
  641. package/src/node/env-variables/index.ts +17 -17
  642. package/src/node/environment-utils.spec.ts +92 -92
  643. package/src/node/environment-utils.ts +66 -66
  644. package/src/node/file-uri.spec.ts +76 -76
  645. package/src/node/filesystem-locking.ts +77 -77
  646. package/src/node/hosting/backend-application-hosts.ts +60 -60
  647. package/src/node/hosting/backend-hosting-module.ts +26 -26
  648. package/src/node/hosting/ws-origin-validator.ts +36 -36
  649. package/src/node/i18n/i18n-backend-module.ts +42 -42
  650. package/src/node/i18n/localization-contribution.ts +112 -112
  651. package/src/node/i18n/localization-provider.ts +125 -125
  652. package/src/node/i18n/localization-server.ts +52 -52
  653. package/src/node/i18n/theia-localization-contribution.ts +40 -36
  654. package/src/node/index.ts +22 -22
  655. package/src/node/key-store-server.ts +162 -162
  656. package/src/node/logger-backend-module.ts +88 -88
  657. package/src/node/logger-cli-contribution.spec.ts +245 -245
  658. package/src/node/logger-cli-contribution.ts +168 -168
  659. package/src/node/main.ts +33 -33
  660. package/src/node/messaging/binary-message-pipe.ts +168 -168
  661. package/src/node/messaging/connection-container-module.ts +96 -96
  662. package/src/node/messaging/default-messaging-service.ts +129 -129
  663. package/src/node/messaging/frontend-connection-service.ts +24 -24
  664. package/src/node/messaging/index.ts +19 -19
  665. package/src/node/messaging/ipc-bootstrap.ts +27 -27
  666. package/src/node/messaging/ipc-channel.ts +77 -77
  667. package/src/node/messaging/ipc-connection-provider.ts +107 -107
  668. package/src/node/messaging/ipc-protocol.ts +76 -76
  669. package/src/node/messaging/messaging-backend-module.ts +52 -52
  670. package/src/node/messaging/messaging-listeners.ts +52 -52
  671. package/src/node/messaging/messaging-service.ts +46 -46
  672. package/src/node/messaging/test/test-web-socket-channel.ts +61 -61
  673. package/src/node/messaging/websocket-endpoint.ts +79 -79
  674. package/src/node/messaging/websocket-frontend-connection-service.ts +186 -186
  675. package/src/node/os-backend-provider.ts +25 -25
  676. package/src/node/performance/index.ts +18 -18
  677. package/src/node/performance/measurement-backend-bindings.ts +35 -35
  678. package/src/node/performance/node-stopwatch.ts +40 -40
  679. package/src/node/process-utils.spec.ts +48 -48
  680. package/src/node/process-utils.ts +102 -102
  681. package/src/node/remote/backend-remote-service.ts +25 -25
  682. package/src/node/remote/remote-cli-contribution.ts +34 -34
  683. package/src/node/remote/remote-copy-contribution.ts +45 -45
  684. package/src/node/request/backend-request-facade.ts +39 -39
  685. package/src/node/request/backend-request-module.ts +25 -25
  686. package/src/node/request/proxy-cli-contribution.ts +65 -65
  687. package/src/node/ws-request-validators.ts +56 -56
  688. package/src/typings/native-keymap.d.ts +108 -108
  689. package/i18n/nls.pt-pt.json +0 -552
@@ -1,1468 +1,1468 @@
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 PerfectScrollbar from 'perfect-scrollbar';
18
- import { TabBar, Title, Widget } from '@phosphor/widgets';
19
- import { VirtualElement, h, VirtualDOM, ElementInlineStyle } from '@phosphor/virtualdom';
20
- import { Disposable, DisposableCollection, MenuPath, notEmpty, SelectionService, CommandService, nls, ArrayUtils } from '../../common';
21
- import { ContextMenuRenderer } from '../context-menu-renderer';
22
- import { Signal, Slot } from '@phosphor/signaling';
23
- import { Message, MessageLoop } from '@phosphor/messaging';
24
- import { ArrayExt } from '@phosphor/algorithm';
25
- import { ElementExt } from '@phosphor/domutils';
26
- import { TabBarToolbarRegistry, TabBarToolbar } from './tab-bar-toolbar';
27
- import { TheiaDockPanel, MAIN_AREA_ID, BOTTOM_AREA_ID } from './theia-dock-panel';
28
- import { WidgetDecoration } from '../widget-decoration';
29
- import { TabBarDecoratorService } from './tab-bar-decorator';
30
- import { IconThemeService } from '../icon-theme-service';
31
- import { BreadcrumbsRenderer, BreadcrumbsRendererFactory } from '../breadcrumbs/breadcrumbs-renderer';
32
- import { NavigatableWidget } from '../navigatable-types';
33
- import { IDragEvent } from '@phosphor/dragdrop';
34
- import { LOCKED_CLASS, PINNED_CLASS } from '../widgets/widget';
35
- import { CorePreferences } from '../core-preferences';
36
- import { HoverService } from '../hover-service';
37
- import { Root, createRoot } from 'react-dom/client';
38
- import { SelectComponent } from '../widgets/select-component';
39
- import { createElement } from 'react';
40
- import { PreviewableWidget } from '../widgets/previewable-widget';
41
- import { EnhancedPreviewWidget } from '../widgets/enhanced-preview-widget';
42
- import { ContextKeyService } from '../context-key-service';
43
-
44
- /** The class name added to hidden content nodes, which are required to render vertical side bars. */
45
- const HIDDEN_CONTENT_CLASS = 'theia-TabBar-hidden-content';
46
-
47
- /** Menu path for tab bars used throughout the application shell. */
48
- export const SHELL_TABBAR_CONTEXT_MENU: MenuPath = ['shell-tabbar-context-menu'];
49
- export const SHELL_TABBAR_CONTEXT_CLOSE: MenuPath = [...SHELL_TABBAR_CONTEXT_MENU, '0_close'];
50
- export const SHELL_TABBAR_CONTEXT_COPY: MenuPath = [...SHELL_TABBAR_CONTEXT_MENU, '1_copy'];
51
- // Kept here in anticipation of tab pinning behavior implemented in tab-bars.ts
52
- export const SHELL_TABBAR_CONTEXT_PIN: MenuPath = [...SHELL_TABBAR_CONTEXT_MENU, '4_pin'];
53
- export const SHELL_TABBAR_CONTEXT_SPLIT: MenuPath = [...SHELL_TABBAR_CONTEXT_MENU, '5_split'];
54
-
55
- export const TabBarRendererFactory = Symbol('TabBarRendererFactory');
56
- export type TabBarRendererFactory = () => TabBarRenderer;
57
-
58
- /**
59
- * Size information of DOM elements used for rendering tabs in side bars.
60
- */
61
- export interface SizeData {
62
- width: number;
63
- height: number;
64
- }
65
-
66
- /**
67
- * Extension of the rendering data used for tabs in side bars of the application shell.
68
- */
69
- export interface SideBarRenderData extends TabBar.IRenderData<Widget> {
70
- labelSize?: SizeData;
71
- iconSize?: SizeData;
72
- paddingTop?: number;
73
- paddingBottom?: number;
74
- visible?: boolean
75
- }
76
-
77
- export interface ScrollableRenderData extends TabBar.IRenderData<Widget> {
78
- tabWidth?: number;
79
- }
80
-
81
- /**
82
- * A tab bar renderer that offers a context menu. In addition, this renderer is able to
83
- * set an explicit position and size on the icon and label of each tab in a side bar.
84
- * This is necessary because the elements of side bar tabs are rotated using the CSS
85
- * `transform` property, disrupting the browser's ability to arrange those elements
86
- * automatically.
87
- */
88
- export class TabBarRenderer extends TabBar.Renderer {
89
- /**
90
- * The menu path used to render the context menu.
91
- */
92
- contextMenuPath?: MenuPath;
93
-
94
- protected readonly toDispose = new DisposableCollection();
95
-
96
- // TODO refactor shell, rendered should only receive props with event handlers
97
- // events should be handled by clients, like ApplicationShell
98
- // right now it is mess: (1) client logic belong to renderer, (2) cyclic dependencies between renderers and clients
99
- constructor(
100
- protected readonly contextMenuRenderer?: ContextMenuRenderer,
101
- protected readonly decoratorService?: TabBarDecoratorService,
102
- protected readonly iconThemeService?: IconThemeService,
103
- protected readonly selectionService?: SelectionService,
104
- protected readonly commandService?: CommandService,
105
- protected readonly corePreferences?: CorePreferences,
106
- protected readonly hoverService?: HoverService,
107
- protected readonly contextKeyService?: ContextKeyService,
108
- ) {
109
- super();
110
- if (this.decoratorService) {
111
- this.toDispose.push(Disposable.create(() => this.resetDecorations()));
112
- this.toDispose.push(this.decoratorService.onDidChangeDecorations(() => this.resetDecorations()));
113
- }
114
- if (this.iconThemeService) {
115
- this.toDispose.push(this.iconThemeService.onDidChangeCurrent(() => {
116
- if (this._tabBar) {
117
- this._tabBar.update();
118
- }
119
- }));
120
- }
121
- }
122
-
123
- dispose(): void {
124
- this.toDispose.dispose();
125
- }
126
-
127
- protected _tabBar?: TabBar<Widget>;
128
- protected readonly toDisposeOnTabBar = new DisposableCollection();
129
- /**
130
- * A reference to the tab bar is required in order to activate it when a context menu
131
- * is requested.
132
- */
133
- set tabBar(tabBar: TabBar<Widget> | undefined) {
134
- if (this.toDispose.disposed) {
135
- throw new Error('disposed');
136
- }
137
- if (this._tabBar === tabBar) {
138
- return;
139
- }
140
- this.toDisposeOnTabBar.dispose();
141
- this.toDispose.push(this.toDisposeOnTabBar);
142
- this._tabBar = tabBar;
143
- if (tabBar) {
144
- const listener: Slot<Widget, TabBar.ITabCloseRequestedArgs<Widget>> = (_, { title }) => this.resetDecorations(title);
145
- tabBar.tabCloseRequested.connect(listener);
146
- this.toDisposeOnTabBar.push(Disposable.create(() => tabBar.tabCloseRequested.disconnect(listener)));
147
- }
148
- this.resetDecorations();
149
- }
150
- get tabBar(): TabBar<Widget> | undefined {
151
- return this._tabBar;
152
- }
153
-
154
- /**
155
- * Render tabs with the default DOM structure, but additionally register a context menu listener.
156
- * @param {SideBarRenderData} data Data used to render the tab.
157
- * @param {boolean} isInSidePanel An optional check which determines if the tab is in the side-panel.
158
- * @param {boolean} isPartOfHiddenTabBar An optional check which determines if the tab is in the hidden horizontal tab bar.
159
- * @returns {VirtualElement} The virtual element of the rendered tab.
160
- */
161
- override renderTab(data: SideBarRenderData, isInSidePanel?: boolean, isPartOfHiddenTabBar?: boolean): VirtualElement {
162
- const title = data.title;
163
- const id = this.createTabId(title, isPartOfHiddenTabBar);
164
- const key = this.createTabKey(data);
165
- const style = this.createTabStyle(data);
166
- const className = this.createTabClass(data);
167
- const dataset = this.createTabDataset(data);
168
- const closeIconTitle = data.title.className.includes(PINNED_CLASS)
169
- ? nls.localizeByDefault('Unpin')
170
- : nls.localizeByDefault('Close');
171
-
172
- const hover = this.tabBar && (this.tabBar.orientation === 'horizontal' && this.corePreferences?.['window.tabbar.enhancedPreview'] === 'classic')
173
- ? { title: title.caption }
174
- : {
175
- onmouseenter: this.handleMouseEnterEvent
176
- };
177
-
178
- return h.li(
179
- {
180
- ...hover,
181
- key, className, id, style, dataset,
182
- oncontextmenu: this.handleContextMenuEvent,
183
- ondblclick: this.handleDblClickEvent,
184
- onauxclick: (e: MouseEvent) => {
185
- // If user closes the tab using mouse wheel, nothing should be pasted to an active editor
186
- e.preventDefault();
187
- }
188
- },
189
- h.div(
190
- { className: 'theia-tab-icon-label' },
191
- this.renderIcon(data, isInSidePanel),
192
- this.renderLabel(data, isInSidePanel),
193
- this.renderTailDecorations(data, isInSidePanel),
194
- this.renderBadge(data, isInSidePanel),
195
- this.renderLock(data, isInSidePanel)
196
- ),
197
- h.div({
198
- className: 'p-TabBar-tabCloseIcon action-label',
199
- title: closeIconTitle,
200
- onclick: this.handleCloseClickEvent
201
- })
202
- );
203
- }
204
-
205
- override createTabClass(data: SideBarRenderData): string {
206
- let tabClass = super.createTabClass(data);
207
- if (!(data.visible ?? true)) {
208
- tabClass += ' p-mod-invisible';
209
- }
210
- return tabClass;
211
- }
212
-
213
- /**
214
- * Generate ID for an entry in the tab bar
215
- * @param {Title<Widget>} title Title of the widget controlled by this tab bar
216
- * @param {boolean} isPartOfHiddenTabBar Tells us if this entry is part of the hidden horizontal tab bar.
217
- * If yes, add a suffix to differentiate it's ID from the entry in the visible tab bar
218
- * @returns {string} DOM element ID
219
- */
220
- createTabId(title: Title<Widget>, isPartOfHiddenTabBar = false): string {
221
- return 'shell-tab-' + title.owner.id + (isPartOfHiddenTabBar ? '-hidden' : '');
222
- }
223
-
224
- /**
225
- * If size information is available for the label and icon, set an explicit height on the tab.
226
- * The height value also considers padding, which should be derived from CSS settings.
227
- */
228
- override createTabStyle(data: SideBarRenderData & ScrollableRenderData): ElementInlineStyle {
229
- const zIndex = `${data.zIndex}`;
230
- const labelSize = data.labelSize;
231
- const iconSize = data.iconSize;
232
- let height: string | undefined;
233
- let width: string | undefined;
234
- if (labelSize || iconSize) {
235
- const labelHeight = labelSize ? (this.tabBar && this.tabBar.orientation === 'horizontal' ? labelSize.height : labelSize.width) : 0;
236
- const iconHeight = iconSize ? iconSize.height : 0;
237
- let paddingTop = data.paddingTop || 0;
238
- if (labelHeight > 0 && iconHeight > 0) {
239
- // Leave some extra space between icon and label
240
- paddingTop = paddingTop * 1.5;
241
- }
242
- const paddingBottom = data.paddingBottom || 0;
243
- height = `${labelHeight + iconHeight + paddingTop + paddingBottom}px`;
244
- }
245
- if (data.tabWidth) {
246
- width = `${data.tabWidth}px`;
247
- } else {
248
- width = '';
249
- }
250
- return { zIndex, height, minWidth: width, maxWidth: width };
251
- }
252
-
253
- /**
254
- * If size information is available for the label, set it as inline style.
255
- * Tab padding and icon size are also considered in the `top` position.
256
- * @param {SideBarRenderData} data Data used to render the tab.
257
- * @param {boolean} isInSidePanel An optional check which determines if the tab is in the side-panel.
258
- * @returns {VirtualElement} The virtual element of the rendered label.
259
- */
260
- override renderLabel(data: SideBarRenderData, isInSidePanel?: boolean): VirtualElement {
261
- const labelSize = data.labelSize;
262
- const iconSize = data.iconSize;
263
- let width: string | undefined;
264
- let height: string | undefined;
265
- let top: string | undefined;
266
- if (labelSize) {
267
- width = `${labelSize.width}px`;
268
- height = `${labelSize.height}px`;
269
- }
270
- if (data.paddingTop || iconSize) {
271
- const iconHeight = iconSize ? iconSize.height : 0;
272
- let paddingTop = data.paddingTop || 0;
273
- if (iconHeight > 0) {
274
- // Leave some extra space between icon and label
275
- paddingTop = paddingTop * 1.5;
276
- }
277
- top = `${paddingTop + iconHeight}px`;
278
- }
279
- const style: ElementInlineStyle = { width, height, top };
280
- // No need to check for duplicate labels if the tab is rendered in the side panel (title is not displayed),
281
- // or if there are less than two files in the tab bar.
282
- if (isInSidePanel || (this.tabBar && this.tabBar.titles.length < 2)) {
283
- return h.div({ className: 'p-TabBar-tabLabel', style }, data.title.label);
284
- }
285
- const originalToDisplayedMap = this.findDuplicateLabels([...this.tabBar!.titles]);
286
- const labelDetails: string | undefined = originalToDisplayedMap.get(data.title.caption);
287
- if (labelDetails) {
288
- return h.div({ className: 'p-TabBar-tabLabelWrapper' },
289
- h.div({ className: 'p-TabBar-tabLabel', style }, data.title.label),
290
- h.div({ className: 'p-TabBar-tabLabelDetails', style }, labelDetails));
291
- }
292
- return h.div({ className: 'p-TabBar-tabLabel', style }, data.title.label);
293
- }
294
-
295
- protected renderTailDecorations(renderData: SideBarRenderData, isInSidePanel?: boolean): VirtualElement[] {
296
- if (!this.corePreferences?.get('workbench.editor.decorations.badges')) {
297
- return [];
298
- }
299
- const tailDecorations = ArrayUtils.coalesce(this.getDecorationData(renderData.title, 'tailDecorations')).flat();
300
- if (tailDecorations === undefined || tailDecorations.length === 0) {
301
- return [];
302
- }
303
- let dotDecoration: WidgetDecoration.TailDecoration.AnyPartial | undefined;
304
- const otherDecorations: WidgetDecoration.TailDecoration.AnyPartial[] = [];
305
- tailDecorations.reverse().forEach(decoration => {
306
- const partial = decoration as WidgetDecoration.TailDecoration.AnyPartial;
307
- if (WidgetDecoration.TailDecoration.isDotDecoration(partial)) {
308
- dotDecoration ||= partial;
309
- } else if (partial.data || partial.icon || partial.iconClass) {
310
- otherDecorations.push(partial);
311
- }
312
- });
313
- const decorationsToRender = dotDecoration ? [dotDecoration, ...otherDecorations] : otherDecorations;
314
- return decorationsToRender.map((decoration, index) => {
315
- const { tooltip, data, fontData, color, icon, iconClass } = decoration;
316
- const iconToRender = icon ?? iconClass;
317
- const className = ['p-TabBar-tail', 'flex'].join(' ');
318
- const style = fontData ? fontData : color ? { color } : undefined;
319
- const content = (data ? data : iconToRender
320
- ? h.span({ className: this.getIconClass(iconToRender, iconToRender === 'circle' ? [WidgetDecoration.Styles.DECORATOR_SIZE_CLASS] : []) })
321
- : '') + (index !== decorationsToRender.length - 1 ? ',' : '');
322
- return h.span({ key: ('tailDecoration_' + index), className, style, title: tooltip ?? content }, content);
323
- });
324
- }
325
-
326
- renderBadge(data: SideBarRenderData, isInSidePanel?: boolean): VirtualElement {
327
- const totalBadge = this.getDecorationData(data.title, 'badge').reduce((sum, badge) => sum! + badge!, 0);
328
- if (!totalBadge) {
329
- return h.div({});
330
- }
331
- const limitedBadge = totalBadge >= 100 ? '99+' : totalBadge;
332
- return isInSidePanel
333
- ? h.div({ className: 'theia-badge-decorator-sidebar' }, `${limitedBadge}`)
334
- : h.div({ className: 'theia-badge-decorator-horizontal' }, `${limitedBadge}`);
335
- }
336
-
337
- renderLock(data: SideBarRenderData, isInSidePanel?: boolean): VirtualElement {
338
- return !isInSidePanel && data.title.className.includes(LOCKED_CLASS)
339
- ? h.div({ className: 'p-TabBar-tabLock' })
340
- : h.div({});
341
- }
342
-
343
- protected readonly decorations = new Map<Title<Widget>, WidgetDecoration.Data[]>();
344
-
345
- protected resetDecorations(title?: Title<Widget>): void {
346
- if (title) {
347
- this.decorations.delete(title);
348
- } else {
349
- this.decorations.clear();
350
- }
351
- if (this.tabBar) {
352
- this.tabBar.update();
353
- }
354
- }
355
-
356
- /**
357
- * Get all available decorations of a given tab.
358
- * @param {string} title The widget title.
359
- */
360
- protected getDecorations(title: Title<Widget>): WidgetDecoration.Data[] {
361
- if (this.tabBar && this.decoratorService) {
362
- const owner: { resetTabBarDecorations?: () => void; } & Widget = title.owner;
363
- if (!owner.resetTabBarDecorations) {
364
- owner.resetTabBarDecorations = () => this.decorations.delete(title);
365
- title.owner.disposed.connect(owner.resetTabBarDecorations);
366
- }
367
-
368
- const decorations = this.decorations.get(title) || this.decoratorService.getDecorations(title);
369
- this.decorations.set(title, decorations);
370
- return decorations;
371
- }
372
- return [];
373
- }
374
-
375
- /**
376
- * Get the decoration data given the tab URI and the decoration data type.
377
- * @param {string} title The title.
378
- * @param {K} key The type of the decoration data.
379
- */
380
- protected getDecorationData<K extends keyof WidgetDecoration.Data>(title: Title<Widget>, key: K): WidgetDecoration.Data[K][] {
381
- return this.getDecorations(title).filter(data => data[key] !== undefined).map(data => data[key]);
382
- }
383
-
384
- /**
385
- * Get the class of an icon.
386
- * @param {string | string[]} iconName The name of the icon.
387
- * @param {string[]} additionalClasses Additional classes of the icon.
388
- */
389
- protected getIconClass(iconName: string | string[], additionalClasses: string[] = []): string {
390
- const iconClass = (typeof iconName === 'string') ? ['a', 'fa', `fa-${iconName}`] : ['a'].concat(iconName);
391
- return iconClass.concat(additionalClasses).join(' ');
392
- }
393
-
394
- /**
395
- * Find duplicate labels from the currently opened tabs in the tab bar.
396
- * Return the appropriate partial paths that can distinguish the identical labels.
397
- *
398
- * E.g., a/p/index.ts => a/..., b/p/index.ts => b/...
399
- *
400
- * To prevent excessively long path displayed, show at maximum three levels from the end by default.
401
- * @param {Title<Widget>[]} titles Array of titles in the current tab bar.
402
- * @returns {Map<string, string>} A map from each tab's original path to its displayed partial path.
403
- */
404
- findDuplicateLabels(titles: Title<Widget>[]): Map<string, string> {
405
- // Filter from all tabs to group them by the distinct label (file name).
406
- // E.g., 'foo.js' => {0 (index) => 'a/b/foo.js', '2 => a/c/foo.js' },
407
- // 'bar.js' => {1 => 'a/d/bar.js', ...}
408
- const labelGroups = new Map<string, Map<number, string>>();
409
- titles.forEach((title, index) => {
410
- if (!labelGroups.has(title.label)) {
411
- labelGroups.set(title.label, new Map<number, string>());
412
- }
413
- labelGroups.get(title.label)!.set(index, title.caption);
414
- });
415
-
416
- const originalToDisplayedMap = new Map<string, string>();
417
- // Parse each group of editors with the same label.
418
- labelGroups.forEach(labelGroup => {
419
- // Filter to get groups that have duplicates.
420
- if (labelGroup.size > 1) {
421
- const paths: string[][] = [];
422
- let maxPathLength = 0;
423
- labelGroup.forEach((pathStr, index) => {
424
- const steps = pathStr.split('/');
425
- maxPathLength = Math.max(maxPathLength, steps.length);
426
- paths[index] = (steps.slice(0, steps.length - 1));
427
- // By default, show at maximum three levels from the end.
428
- let defaultDisplayedPath = steps.slice(-4, -1).join('/');
429
- if (steps.length > 4) {
430
- defaultDisplayedPath = '.../' + defaultDisplayedPath;
431
- }
432
- originalToDisplayedMap.set(pathStr, defaultDisplayedPath);
433
- });
434
-
435
- // Iterate through the steps of the path from the left to find the step that can distinguish it.
436
- // E.g., ['root', 'foo', 'c'], ['root', 'bar', 'd'] => 'foo', 'bar'
437
- let i = 0;
438
- while (i < maxPathLength - 1) {
439
- // Store indexes of all paths that have the identical element in each step.
440
- const stepOccurrences = new Map<string, number[]>();
441
- // Compare the current step of all paths
442
- paths.forEach((path, index) => {
443
- const step = path[i];
444
- if (path.length > 0) {
445
- if (i > path.length - 1) {
446
- paths[index] = [];
447
- } else if (!stepOccurrences.has(step)) {
448
- stepOccurrences.set(step, [index]);
449
- } else {
450
- stepOccurrences.get(step)!.push(index);
451
- }
452
- }
453
- });
454
- // Set the displayed path for each tab.
455
- stepOccurrences.forEach((indexArr, displayedPath) => {
456
- if (indexArr.length === 1) {
457
- const originalPath = labelGroup.get(indexArr[0]);
458
- if (originalPath) {
459
- const originalElements = originalPath.split('/');
460
- const displayedElements = displayedPath.split('/');
461
- if (originalElements.slice(-2)[0] !== displayedElements.slice(-1)[0]) {
462
- displayedPath += '/...';
463
- }
464
- if (originalElements[0] !== displayedElements[0]) {
465
- displayedPath = '.../' + displayedPath;
466
- }
467
- originalToDisplayedMap.set(originalPath, displayedPath);
468
- paths[indexArr[0]] = [];
469
- }
470
- }
471
- });
472
- i++;
473
- }
474
- }
475
- });
476
- return originalToDisplayedMap;
477
- }
478
-
479
- /**
480
- * If size information is available for the icon, set it as inline style. Tab padding
481
- * is also considered in the `top` position.
482
- * @param {SideBarRenderData} data Data used to render the tab icon.
483
- * @param {boolean} isInSidePanel An optional check which determines if the tab is in the side-panel.
484
- */
485
- override renderIcon(data: SideBarRenderData, isInSidePanel?: boolean): VirtualElement {
486
- if (!isInSidePanel && this.iconThemeService && this.iconThemeService.current === 'none') {
487
- return h.div();
488
- }
489
- let top: string | undefined;
490
- if (data.paddingTop) {
491
- top = `${data.paddingTop || 0}px`;
492
- }
493
- const style: ElementInlineStyle = { top };
494
- const baseClassName = this.createIconClass(data);
495
-
496
- const overlayIcons: VirtualElement[] = [];
497
- const decorationData = this.getDecorationData(data.title, 'iconOverlay');
498
-
499
- // Check if the tab has decoration markers to be rendered on top.
500
- if (decorationData.length > 0) {
501
- const baseIcon: VirtualElement = h.div({ className: baseClassName, style }, data.title.iconLabel);
502
- const wrapperClassName: string = WidgetDecoration.Styles.ICON_WRAPPER_CLASS;
503
- const decoratorSizeClassName: string = isInSidePanel ? WidgetDecoration.Styles.DECORATOR_SIDEBAR_SIZE_CLASS : WidgetDecoration.Styles.DECORATOR_SIZE_CLASS;
504
-
505
- decorationData
506
- .filter(notEmpty)
507
- .map(overlay => [overlay.position, overlay] as [WidgetDecoration.IconOverlayPosition, WidgetDecoration.IconOverlay | WidgetDecoration.IconClassOverlay])
508
- .forEach(([position, overlay]) => {
509
- const iconAdditionalClasses: string[] = [decoratorSizeClassName, WidgetDecoration.IconOverlayPosition.getStyle(position, isInSidePanel)];
510
- const overlayIconStyle = (color?: string) => {
511
- if (color === undefined) {
512
- return {};
513
- }
514
- return { color };
515
- };
516
- // Parse the optional background (if it exists) of the overlay icon.
517
- if (overlay.background) {
518
- const backgroundIconClassName = this.getIconClass(overlay.background.shape, iconAdditionalClasses);
519
- overlayIcons.push(
520
- h.div({ key: data.title.label + '-background', className: backgroundIconClassName, style: overlayIconStyle(overlay.background.color) })
521
- );
522
- }
523
- // Parse the overlay icon.
524
- const overlayIcon = (overlay as WidgetDecoration.IconOverlay).icon || (overlay as WidgetDecoration.IconClassOverlay).iconClass;
525
- const overlayIconClassName = this.getIconClass(overlayIcon, iconAdditionalClasses);
526
- overlayIcons.push(
527
- h.span({ key: data.title.label, className: overlayIconClassName, style: overlayIconStyle(overlay.color) })
528
- );
529
- });
530
- return h.div({ className: wrapperClassName, style }, [baseIcon, ...overlayIcons]);
531
- }
532
- return h.div({ className: baseClassName, style }, data.title.iconLabel);
533
- }
534
-
535
- protected renderEnhancedPreview = (title: Title<Widget>) => {
536
- const hoverBox = document.createElement('div');
537
- hoverBox.classList.add('theia-horizontal-tabBar-hover-div');
538
- const labelElement = document.createElement('p');
539
- labelElement.classList.add('theia-horizontal-tabBar-hover-title');
540
- labelElement.textContent = title.label;
541
- hoverBox.append(labelElement);
542
- const widget = title.owner;
543
- if (EnhancedPreviewWidget.is(widget)) {
544
- const enhancedPreviewNode = widget.getEnhancedPreviewNode();
545
- if (enhancedPreviewNode) {
546
- hoverBox.appendChild(enhancedPreviewNode);
547
- }
548
- } else if (title.caption) {
549
- const captionElement = document.createElement('p');
550
- captionElement.classList.add('theia-horizontal-tabBar-hover-caption');
551
- captionElement.textContent = title.caption;
552
- hoverBox.appendChild(captionElement);
553
- }
554
- return hoverBox;
555
- };
556
-
557
- protected renderVisualPreview(desiredWidth: number, title: Title<Widget>): HTMLElement | undefined {
558
- const widget = title.owner;
559
- // Check that the widget is not currently shown, is a PreviewableWidget and it was already loaded before
560
- if (this.tabBar && this.tabBar.currentTitle !== title && PreviewableWidget.isPreviewable(widget)) {
561
- const html = document.getElementById(widget.id);
562
- if (html) {
563
- const previewNode: Node | undefined = widget.getPreviewNode();
564
- if (previewNode) {
565
- const clonedNode = previewNode.cloneNode(true);
566
- const visualPreviewDiv = document.createElement('div');
567
- visualPreviewDiv.classList.add('enhanced-preview-container');
568
- // Add the clonedNode and get it from the children to have a HTMLElement instead of a Node
569
- visualPreviewDiv.append(clonedNode);
570
- const visualPreview = visualPreviewDiv.children.item(visualPreviewDiv.children.length - 1);
571
- if (visualPreview instanceof HTMLElement) {
572
- visualPreview.classList.remove('p-mod-hidden');
573
- visualPreview.classList.add('enhanced-preview');
574
- visualPreview.id = `preview:${widget.id}`;
575
-
576
- // Use the current visible editor as a fallback if not available
577
- const height: number = visualPreview.style.height === '' ? this.tabBar.currentTitle!.owner.node.offsetHeight : parseFloat(visualPreview.style.height);
578
- const width: number = visualPreview.style.width === '' ? this.tabBar.currentTitle!.owner.node.offsetWidth : parseFloat(visualPreview.style.width);
579
- const desiredRatio = 9 / 16;
580
- const desiredHeight = desiredWidth * desiredRatio;
581
- const ratio = height / width;
582
- visualPreviewDiv.style.width = `${desiredWidth}px`;
583
- visualPreviewDiv.style.height = `${desiredHeight}px`;
584
-
585
- // If the view is wider than the desiredRatio scale the width and crop the height. If the view is longer its the other way around.
586
- const scale = ratio < desiredRatio ? (desiredHeight / height) : (desiredWidth / width);
587
- visualPreview.style.transform = `scale(${scale},${scale})`;
588
- visualPreview.style.removeProperty('top');
589
- visualPreview.style.removeProperty('left');
590
-
591
- // Copy canvases (They are cloned empty)
592
- const originalCanvases = html.getElementsByTagName('canvas');
593
- const previewCanvases = visualPreview.getElementsByTagName('canvas');
594
- // If this is not given, something went wrong during the cloning
595
- if (originalCanvases.length === previewCanvases.length) {
596
- for (let i = 0; i < originalCanvases.length; i++) {
597
- previewCanvases[i].getContext('2d')?.drawImage(originalCanvases[i], 0, 0);
598
- }
599
- }
600
-
601
- return visualPreviewDiv;
602
- }
603
- }
604
- }
605
- }
606
- return undefined;
607
- }
608
-
609
- protected handleMouseEnterEvent = (event: MouseEvent) => {
610
- if (this.tabBar && this.hoverService && event.currentTarget instanceof HTMLElement) {
611
- const id = event.currentTarget.id;
612
- const title = this.tabBar.titles.find(t => this.createTabId(t) === id);
613
- if (title) {
614
- if (this.tabBar.orientation === 'horizontal') {
615
- this.hoverService.requestHover({
616
- content: this.renderEnhancedPreview(title),
617
- target: event.currentTarget,
618
- position: 'bottom',
619
- cssClasses: ['extended-tab-preview'],
620
- visualPreview: this.corePreferences?.['window.tabbar.enhancedPreview'] === 'visual' ? width => this.renderVisualPreview(width, title) : undefined
621
- });
622
- } else if (title.caption) {
623
- this.hoverService.requestHover({
624
- content: title.caption,
625
- target: event.currentTarget,
626
- position: 'right'
627
- });
628
- }
629
- }
630
- }
631
- };
632
-
633
- protected handleContextMenuEvent = (event: MouseEvent) => {
634
- if (this.contextMenuRenderer && this.contextMenuPath && event.currentTarget instanceof HTMLElement) {
635
- event.stopPropagation();
636
- event.preventDefault();
637
- let widget: Widget | undefined = undefined;
638
- if (this.tabBar) {
639
- const titleIndex = Array.from(this.tabBar.contentNode.getElementsByClassName('p-TabBar-tab'))
640
- .findIndex(node => node.contains(event.currentTarget as HTMLElement));
641
- if (titleIndex !== -1) {
642
- widget = this.tabBar.titles[titleIndex].owner;
643
- }
644
- }
645
-
646
- const oldSelection = this.selectionService?.selection;
647
- if (widget && this.selectionService) {
648
- this.selectionService.selection = NavigatableWidget.is(widget) ? { uri: widget.getResourceUri() } : widget;
649
- }
650
-
651
- const contextKeyServiceOverlay = this.contextKeyService?.createOverlay([['isTerminalTab', widget && 'terminalId' in widget]]);
652
- this.contextMenuRenderer.render({
653
- menuPath: this.contextMenuPath!,
654
- anchor: event,
655
- args: [event],
656
- contextKeyService: contextKeyServiceOverlay,
657
- // We'd like to wait until the command triggered by the context menu has been run, but this should let it get through the preamble, at least.
658
- onHide: () => setTimeout(() => { if (this.selectionService) { this.selectionService.selection = oldSelection; } })
659
- });
660
- }
661
- };
662
-
663
- protected handleCloseClickEvent = (event: MouseEvent) => {
664
- if (this.tabBar && event.currentTarget instanceof HTMLElement) {
665
- const id = event.currentTarget.parentElement!.id;
666
- const title = this.tabBar.titles.find(t => this.createTabId(t) === id);
667
- if (title?.closable === false && title?.className.includes(PINNED_CLASS) && this.commandService) {
668
- this.commandService.executeCommand('workbench.action.unpinEditor', event);
669
- }
670
- }
671
- };
672
-
673
- protected handleDblClickEvent = (event: MouseEvent) => {
674
- if (!this.corePreferences?.get('workbench.tab.maximize')) {
675
- return;
676
- }
677
- if (this.tabBar && event.currentTarget instanceof HTMLElement) {
678
- const id = event.currentTarget.id;
679
- const title = this.tabBar.titles.find(t => this.createTabId(t) === id);
680
- const area = title?.owner.parent;
681
- if (area instanceof TheiaDockPanel && (area.id === BOTTOM_AREA_ID || area.id === MAIN_AREA_ID)) {
682
- area.toggleMaximized();
683
- }
684
- }
685
- };
686
-
687
- }
688
-
689
- /**
690
- * A specialized tab bar for the main and bottom areas.
691
- */
692
- export class ScrollableTabBar extends TabBar<Widget> {
693
-
694
- protected scrollBar?: PerfectScrollbar;
695
-
696
- protected scrollBarFactory: () => PerfectScrollbar;
697
- protected pendingReveal?: Promise<void>;
698
- protected isMouseOver = false;
699
- protected needsRecompute = false;
700
- protected tabSize = 0;
701
- protected _dynamicTabOptions?: ScrollableTabBar.Options;
702
- protected contentContainer: HTMLElement;
703
- protected topRow: HTMLElement;
704
-
705
- protected readonly toDispose = new DisposableCollection();
706
- protected openTabsContainer: HTMLDivElement;
707
- protected openTabsRoot: Root;
708
-
709
- constructor(options?: TabBar.IOptions<Widget> & PerfectScrollbar.Options, dynamicTabOptions?: ScrollableTabBar.Options) {
710
- super(options);
711
- this.scrollBarFactory = () => new PerfectScrollbar(this.scrollbarHost, options);
712
- this._dynamicTabOptions = dynamicTabOptions;
713
- this.rewireDOM();
714
- }
715
-
716
- set dynamicTabOptions(options: ScrollableTabBar.Options | undefined) {
717
- this._dynamicTabOptions = options;
718
- this.updateTabs();
719
- }
720
-
721
- get dynamicTabOptions(): ScrollableTabBar.Options | undefined {
722
- return this._dynamicTabOptions;
723
- }
724
-
725
- override dispose(): void {
726
- if (this.isDisposed) {
727
- return;
728
- }
729
- super.dispose();
730
- this.toDispose.dispose();
731
- }
732
-
733
- /**
734
- * Restructures the DOM defined in PhosphorJS.
735
- *
736
- * By default the tabs (`li`) are contained in the `this.contentNode` (`ul`) which is wrapped in a `div` (`this.node`).
737
- * Instead of this structure, we add a container for the `this.contentNode` and for the toolbar.
738
- * The scrollbar will only work for the `ul` part but it does not affect the toolbar, so it can be on the right hand-side.
739
- */
740
- protected rewireDOM(): void {
741
- const contentNode = this.node.getElementsByClassName(ScrollableTabBar.Styles.TAB_BAR_CONTENT)[0];
742
- if (!contentNode) {
743
- throw new Error("'this.node' does not have the content as a direct child with class name 'p-TabBar-content'.");
744
- }
745
- this.node.removeChild(contentNode);
746
- this.contentContainer = document.createElement('div');
747
- this.contentContainer.classList.add(ScrollableTabBar.Styles.TAB_BAR_CONTENT_CONTAINER);
748
- this.contentContainer.appendChild(contentNode);
749
-
750
- this.topRow = document.createElement('div');
751
- this.topRow.classList.add('theia-tabBar-tab-row');
752
- this.topRow.appendChild(this.contentContainer);
753
-
754
- this.openTabsContainer = document.createElement('div');
755
- this.openTabsContainer.classList.add('theia-tabBar-open-tabs');
756
- this.openTabsRoot = createRoot(this.openTabsContainer);
757
- this.topRow.appendChild(this.openTabsContainer);
758
-
759
- this.node.appendChild(this.topRow);
760
- }
761
-
762
- protected override onAfterAttach(msg: Message): void {
763
- if (!this.scrollBar) {
764
- this.scrollBar = this.scrollBarFactory();
765
- }
766
- this.node.addEventListener('mouseenter', () => { this.isMouseOver = true; });
767
- this.node.addEventListener('mouseleave', () => {
768
- this.isMouseOver = false;
769
- if (this.needsRecompute) {
770
- this.updateTabs();
771
- }
772
- });
773
-
774
- super.onAfterAttach(msg);
775
- }
776
-
777
- protected override onBeforeDetach(msg: Message): void {
778
- super.onBeforeDetach(msg);
779
- if (this.scrollBar) {
780
- this.scrollBar.destroy();
781
- this.scrollBar = undefined;
782
- }
783
- }
784
-
785
- protected override onUpdateRequest(msg: Message): void {
786
- this.updateTabs();
787
- }
788
-
789
- protected updateTabs(): void {
790
- const content = [];
791
- if (this.dynamicTabOptions) {
792
-
793
- this.openTabsRoot.render(createElement(SelectComponent, {
794
- options: this.titles,
795
- onChange: (option, index) => {
796
- this.currentIndex = index;
797
- },
798
- alignment: 'right'
799
- }));
800
-
801
- if (this.isMouseOver) {
802
- this.needsRecompute = true;
803
- } else {
804
- this.needsRecompute = false;
805
- if (this.orientation === 'horizontal') {
806
- let availableWidth = this.scrollbarHost.clientWidth;
807
- let effectiveWidth = availableWidth;
808
- if (!this.openTabsContainer.classList.contains('p-mod-hidden')) {
809
- availableWidth += this.openTabsContainer.getBoundingClientRect().width;
810
- }
811
- if (this.dynamicTabOptions.minimumTabSize * this.titles.length <= availableWidth) {
812
- effectiveWidth += this.openTabsContainer.getBoundingClientRect().width;
813
- this.openTabsContainer.classList.add('p-mod-hidden');
814
- } else {
815
- this.openTabsContainer.classList.remove('p-mod-hidden');
816
- }
817
- this.tabSize = Math.max(Math.min(effectiveWidth / this.titles.length,
818
- this.dynamicTabOptions.defaultTabSize), this.dynamicTabOptions.minimumTabSize);
819
- }
820
- }
821
- this.node.classList.add('dynamic-tabs');
822
- } else {
823
- this.openTabsContainer.classList.add('p-mod-hidden');
824
- this.node.classList.remove('dynamic-tabs');
825
- }
826
- for (let i = 0, n = this.titles.length; i < n; ++i) {
827
- const title = this.titles[i];
828
- const current = title === this.currentTitle;
829
- const zIndex = current ? n : n - i - 1;
830
- const renderData: ScrollableRenderData = { title: title, current: current, zIndex: zIndex };
831
- if (this.dynamicTabOptions && this.orientation === 'horizontal') {
832
- renderData.tabWidth = this.tabSize;
833
- }
834
- content[i] = this.renderer.renderTab(renderData);
835
- }
836
- VirtualDOM.render(content, this.contentNode);
837
- if (this.scrollBar) {
838
- if (!(this.dynamicTabOptions && this.isMouseOver)) {
839
- this.scrollBar.update();
840
- }
841
- }
842
- }
843
-
844
- protected override onResize(msg: Widget.ResizeMessage): void {
845
- super.onResize(msg);
846
- if (this.dynamicTabOptions) {
847
- this.updateTabs();
848
- }
849
- if (this.scrollBar) {
850
- if (this.currentIndex >= 0) {
851
- this.revealTab(this.currentIndex);
852
- }
853
- this.scrollBar.update();
854
- }
855
- }
856
-
857
- /**
858
- * Reveal the tab with the given index by moving the scroll bar if necessary.
859
- */
860
- revealTab(index: number): Promise<void> {
861
- if (this.pendingReveal) {
862
- // A reveal has already been scheduled
863
- return this.pendingReveal;
864
- }
865
- const result = new Promise<void>((resolve, reject) => {
866
- // The tab might not have been created yet, so wait until the next frame
867
- window.requestAnimationFrame(() => {
868
- const tab = this.contentNode.children[index] as HTMLElement;
869
- if (tab && this.isVisible) {
870
- const parent = this.scrollbarHost;
871
- if (this.orientation === 'horizontal') {
872
- const scroll = parent.scrollLeft;
873
- const left = tab.offsetLeft;
874
- if (scroll > left) {
875
- parent.scrollLeft = left;
876
- } else {
877
- const right = left + tab.clientWidth - parent.clientWidth;
878
- if (scroll < right && tab.clientWidth < parent.clientWidth) {
879
- parent.scrollLeft = right;
880
- }
881
- }
882
- } else {
883
- const scroll = parent.scrollTop;
884
- const top = tab.offsetTop;
885
- if (scroll > top) {
886
- parent.scrollTop = top;
887
- } else {
888
- const bottom = top + tab.clientHeight - parent.clientHeight;
889
- if (scroll < bottom && tab.clientHeight < parent.clientHeight) {
890
- parent.scrollTop = bottom;
891
- }
892
- }
893
- }
894
- }
895
- if (this.pendingReveal === result) {
896
- this.pendingReveal = undefined;
897
- }
898
- resolve();
899
- });
900
- });
901
- this.pendingReveal = result;
902
- return result;
903
- }
904
-
905
- /**
906
- * Overrides the `contentNode` property getter in PhosphorJS' TabBar.
907
- */
908
- // @ts-expect-error TS2611 `TabBar<T>.contentNode` is declared as `readonly contentNode` but is implemented as a getter.
909
- get contentNode(): HTMLUListElement {
910
- return this.tabBarContainer.getElementsByClassName(ToolbarAwareTabBar.Styles.TAB_BAR_CONTENT)[0] as HTMLUListElement;
911
- }
912
-
913
- /**
914
- * Overrides the scrollable host from the parent class.
915
- */
916
- protected get scrollbarHost(): HTMLElement {
917
- return this.tabBarContainer;
918
- }
919
-
920
- protected get tabBarContainer(): HTMLElement {
921
- return this.node.getElementsByClassName(ToolbarAwareTabBar.Styles.TAB_BAR_CONTENT_CONTAINER)[0] as HTMLElement;
922
- }
923
- }
924
-
925
- export namespace ScrollableTabBar {
926
-
927
- export interface Options {
928
- minimumTabSize: number;
929
- defaultTabSize: number;
930
- }
931
- export namespace Styles {
932
-
933
- export const TAB_BAR_CONTENT = 'p-TabBar-content';
934
- export const TAB_BAR_CONTENT_CONTAINER = 'p-TabBar-content-container';
935
-
936
- }
937
- }
938
-
939
- /**
940
- * Specialized scrollable tab-bar which comes with toolbar support.
941
- * Instead of the following DOM structure.
942
- *
943
- * +-------------------------+
944
- * |[TAB_0][TAB_1][TAB_2][TAB|
945
- * +-------------Scrollable--+
946
- *
947
- * There is a dedicated HTML element for toolbar which does **not** contained in the scrollable element.
948
- *
949
- * +-------------------------+-----------------+
950
- * |[TAB_0][TAB_1][TAB_2][TAB| Toolbar |
951
- * +-------------Scrollable--+-Non-Scrollable-+
952
- *
953
- */
954
- export class ToolbarAwareTabBar extends ScrollableTabBar {
955
- protected toolbar: TabBarToolbar | undefined;
956
- protected breadcrumbsContainer: HTMLElement;
957
- protected readonly breadcrumbsRenderer: BreadcrumbsRenderer;
958
-
959
- constructor(
960
- protected readonly tabBarToolbarRegistry: TabBarToolbarRegistry,
961
- protected readonly tabBarToolbarFactory: () => TabBarToolbar,
962
- protected readonly breadcrumbsRendererFactory: BreadcrumbsRendererFactory,
963
- options?: TabBar.IOptions<Widget> & PerfectScrollbar.Options,
964
- dynamicTabOptions?: ScrollableTabBar.Options
965
- ) {
966
- super(options, dynamicTabOptions);
967
- this.breadcrumbsRenderer = this.breadcrumbsRendererFactory();
968
- this.addBreadcrumbs();
969
- this.toolbar = this.tabBarToolbarFactory();
970
- this.toDispose.push(this.tabBarToolbarRegistry.onDidChange(() => this.update()));
971
- this.toDispose.push(this.breadcrumbsRenderer);
972
- this.toDispose.push(this.breadcrumbsRenderer.onDidChangeActiveState(active => {
973
- this.node.classList.toggle('theia-tabBar-multirow', active);
974
- if (this.parent) {
975
- MessageLoop.sendMessage(this.parent, new Message('fit-request'));
976
- }
977
- }));
978
- this.node.classList.toggle('theia-tabBar-multirow', this.breadcrumbsRenderer.active);
979
- const handler = () => this.updateBreadcrumbs();
980
- this.currentChanged.connect(handler);
981
- this.toDispose.push(Disposable.create(() => this.currentChanged.disconnect(handler)));
982
- }
983
-
984
- protected async updateBreadcrumbs(): Promise<void> {
985
- const current = this.currentTitle?.owner;
986
- const uri = NavigatableWidget.is(current) ? current.getResourceUri() : undefined;
987
- await this.breadcrumbsRenderer.refresh(uri);
988
- }
989
-
990
- protected override onAfterAttach(msg: Message): void {
991
- if (this.toolbar) {
992
- if (this.toolbar.isAttached) {
993
- Widget.detach(this.toolbar);
994
- }
995
- Widget.attach(this.toolbar, this.topRow);
996
- if (this.breadcrumbsContainer) {
997
- this.node.appendChild(this.breadcrumbsContainer);
998
- }
999
- this.updateBreadcrumbs();
1000
- }
1001
- super.onAfterAttach(msg);
1002
- }
1003
-
1004
- protected override onBeforeDetach(msg: Message): void {
1005
- if (this.toolbar && this.toolbar.isAttached) {
1006
- this.toolbar.dispose();
1007
- }
1008
- super.onBeforeDetach(msg);
1009
- }
1010
-
1011
- protected override onUpdateRequest(msg: Message): void {
1012
- super.onUpdateRequest(msg);
1013
- this.updateToolbar();
1014
- }
1015
-
1016
- protected updateToolbar(): void {
1017
- if (!this.toolbar) {
1018
- return;
1019
- }
1020
- const widget = this.currentTitle?.owner ?? undefined;
1021
- this.toolbar.updateTarget(widget);
1022
- this.updateTabs();
1023
- }
1024
-
1025
- override handleEvent(event: Event): void {
1026
- if (event instanceof MouseEvent) {
1027
- if (this.toolbar && this.toolbar.shouldHandleMouseEvent(event) || this.isOver(event, this.openTabsContainer)) {
1028
- // if the mouse event is over the toolbar part don't handle it.
1029
- return;
1030
- }
1031
- }
1032
- super.handleEvent(event);
1033
- }
1034
-
1035
- protected isOver(event: Event, element: Element): boolean {
1036
- return element && event.target instanceof Element && element.contains(event.target);
1037
- }
1038
-
1039
- /**
1040
- * Restructures the DOM defined in PhosphorJS.
1041
- *
1042
- * By default the tabs (`li`) are contained in the `this.contentNode` (`ul`) which is wrapped in a `div` (`this.node`).
1043
- * Instead of this structure, we add a container for the `this.contentNode` and for the toolbar.
1044
- * The scrollbar will only work for the `ul` part but it does not affect the toolbar, so it can be on the right hand-side.
1045
- */
1046
- protected addBreadcrumbs(): void {
1047
- this.breadcrumbsContainer = document.createElement('div');
1048
- this.breadcrumbsContainer.classList.add('theia-tabBar-breadcrumb-row');
1049
- this.breadcrumbsContainer.appendChild(this.breadcrumbsRenderer.host);
1050
- this.node.appendChild(this.breadcrumbsContainer);
1051
- }
1052
- }
1053
-
1054
- /**
1055
- * A specialized tab bar for side areas.
1056
- */
1057
- export class SideTabBar extends ScrollableTabBar {
1058
-
1059
- protected static readonly DRAG_THRESHOLD = 5;
1060
-
1061
- /**
1062
- * Emitted when a tab is added to the tab bar.
1063
- */
1064
- readonly tabAdded = new Signal<this, { title: Title<Widget> }>(this);
1065
- /**
1066
- * Side panels can be collapsed by clicking on the currently selected tab. This signal is
1067
- * emitted when the mouse is released on the selected tab without initiating a drag.
1068
- */
1069
- readonly collapseRequested = new Signal<this, Title<Widget>>(this);
1070
-
1071
- /**
1072
- * Emitted when the set of overflowing/hidden tabs changes.
1073
- */
1074
- readonly tabsOverflowChanged = new Signal<this, { titles: Title<Widget>[], startIndex: number }>(this);
1075
-
1076
- protected mouseData?: {
1077
- pressX: number,
1078
- pressY: number,
1079
- mouseDownTabIndex: number
1080
- };
1081
-
1082
- protected tabsOverflowData?: {
1083
- titles: Title<Widget>[],
1084
- startIndex: number
1085
- };
1086
-
1087
- constructor(options?: TabBar.IOptions<Widget> & PerfectScrollbar.Options) {
1088
- super(options);
1089
-
1090
- // Create the hidden content node (see `hiddenContentNode` for explanation)
1091
- const hiddenContent = document.createElement('ul');
1092
- hiddenContent.className = HIDDEN_CONTENT_CLASS;
1093
- this.node.appendChild(hiddenContent);
1094
- }
1095
-
1096
- /**
1097
- * Tab bars of the left and right side panel are arranged vertically by rotating their labels.
1098
- * Rotation is realized with the CSS `transform` property, which disrupts the browser's ability
1099
- * to arrange the involved elements automatically. Therefore the elements are arranged explicitly
1100
- * by the TabBarRenderer using inline `height` and `top` styles. However, the size of labels
1101
- * must still be computed by the browser, so the rendering is performed in two steps: first the
1102
- * tab bar is rendered horizontally inside a _hidden content node_, then it is rendered again
1103
- * vertically inside the proper content node. After the first step, size information is gathered
1104
- * from all labels so it can be applied during the second step.
1105
- */
1106
- get hiddenContentNode(): HTMLUListElement {
1107
- return this.node.getElementsByClassName(HIDDEN_CONTENT_CLASS)[0] as HTMLUListElement;
1108
- }
1109
-
1110
- override insertTab(index: number, value: Title<Widget> | Title.IOptions<Widget>): Title<Widget> {
1111
- const result = super.insertTab(index, value);
1112
- this.tabAdded.emit({ title: result });
1113
- return result;
1114
- }
1115
-
1116
- protected override onAfterAttach(msg: Message): void {
1117
- this.updateTabs();
1118
- this.node.addEventListener('p-dragenter', this);
1119
- this.node.addEventListener('p-dragover', this);
1120
- this.node.addEventListener('p-dragleave', this);
1121
- document.addEventListener('p-drop', this);
1122
- }
1123
-
1124
- protected override onAfterDetach(msg: Message): void {
1125
- super.onAfterDetach(msg);
1126
- this.node.removeEventListener('p-dragenter', this);
1127
- this.node.removeEventListener('p-dragover', this);
1128
- this.node.removeEventListener('p-dragleave', this);
1129
- document.removeEventListener('p-drop', this);
1130
- }
1131
-
1132
- protected override onUpdateRequest(msg: Message): void {
1133
- this.updateTabs();
1134
- }
1135
-
1136
- protected override onResize(msg: Widget.ResizeMessage): void {
1137
- // Tabs need to be updated if there are already overflowing tabs or the current tabs don't fit
1138
- if (this.tabsOverflowData || this.node.clientHeight < this.contentNode.clientHeight) {
1139
- this.updateTabs();
1140
- }
1141
- }
1142
-
1143
- /**
1144
- * Reveal the tab with the given index by moving it into the non-overflowing tabBar section
1145
- * if necessary.
1146
- */
1147
- override revealTab(index: number): Promise<void> {
1148
- if (this.pendingReveal) {
1149
- // A reveal has already been scheduled
1150
- return this.pendingReveal;
1151
- }
1152
- const result = new Promise<void>(resolve => {
1153
- // The tab might not have been created yet, so wait until the next frame
1154
- window.requestAnimationFrame(() => {
1155
- if (this.tabsOverflowData && index >= this.tabsOverflowData.startIndex) {
1156
- const title = this.titles[index];
1157
- this.insertTab(this.tabsOverflowData.startIndex - 1, title);
1158
- }
1159
-
1160
- if (this.pendingReveal === result) {
1161
- this.pendingReveal = undefined;
1162
- }
1163
- resolve();
1164
- });
1165
- });
1166
- this.pendingReveal = result;
1167
- return result;
1168
- }
1169
-
1170
- /**
1171
- * Render the tab bar in the _hidden content node_ (see `hiddenContentNode` for explanation),
1172
- * then gather size information for labels and render it again in the proper content node.
1173
- */
1174
- protected override updateTabs(): void {
1175
- if (this.isAttached) {
1176
- // Render into the invisible node
1177
- this.renderTabs(this.hiddenContentNode);
1178
- // Await a rendering frame
1179
- window.requestAnimationFrame(() => {
1180
- const hiddenContent = this.hiddenContentNode;
1181
- const n = hiddenContent.children.length;
1182
- const renderData = new Array<Partial<SideBarRenderData>>(n);
1183
- for (let i = 0; i < n; i++) {
1184
- const hiddenTab = hiddenContent.children[i];
1185
- // Extract tab padding, and margin from the computed style
1186
- const tabStyle = window.getComputedStyle(hiddenTab);
1187
- const rd: Partial<SideBarRenderData> = {
1188
- paddingTop: parseFloat(tabStyle.paddingTop!),
1189
- paddingBottom: parseFloat(tabStyle.paddingBottom!)
1190
- };
1191
- // Extract label size from the DOM
1192
- const labelElements = hiddenTab.getElementsByClassName('p-TabBar-tabLabel');
1193
- if (labelElements.length === 1) {
1194
- const label = labelElements[0];
1195
- rd.labelSize = { width: label.clientWidth, height: label.clientHeight };
1196
- }
1197
- // Extract icon size from the DOM
1198
- const iconElements = hiddenTab.getElementsByClassName('p-TabBar-tabIcon');
1199
- if (iconElements.length === 1) {
1200
- const icon = iconElements[0];
1201
- rd.iconSize = { width: icon.clientWidth, height: icon.clientHeight };
1202
- }
1203
-
1204
- renderData[i] = rd;
1205
- }
1206
- // Render into the visible node
1207
- this.renderTabs(this.contentNode, renderData);
1208
- this.computeOverflowingTabsData();
1209
- });
1210
- }
1211
- }
1212
-
1213
- protected computeOverflowingTabsData(): void {
1214
- // ensure that render tabs has completed
1215
- window.requestAnimationFrame(() => {
1216
- const startIndex = this.hideOverflowingTabs();
1217
- if (startIndex === -1) {
1218
- if (this.tabsOverflowData) {
1219
- this.tabsOverflowData = undefined;
1220
- this.tabsOverflowChanged.emit({ titles: [], startIndex });
1221
- }
1222
- return;
1223
- }
1224
- const newOverflowingTabs = this.titles.slice(startIndex);
1225
-
1226
- if (!this.tabsOverflowData) {
1227
- this.tabsOverflowData = { titles: newOverflowingTabs, startIndex };
1228
- this.tabsOverflowChanged.emit(this.tabsOverflowData);
1229
- return;
1230
- }
1231
-
1232
- if ((newOverflowingTabs.length !== this.tabsOverflowData?.titles.length ?? 0) ||
1233
- newOverflowingTabs.find((newTitle, i) => newTitle !== this.tabsOverflowData?.titles[i]) !== undefined) {
1234
- this.tabsOverflowData = { titles: newOverflowingTabs, startIndex };
1235
- this.tabsOverflowChanged.emit(this.tabsOverflowData);
1236
- }
1237
- });
1238
- }
1239
-
1240
- /**
1241
- * Hide overflowing tabs and return the index of the first hidden tab.
1242
- */
1243
- protected hideOverflowingTabs(): number {
1244
- const availableHeight = this.node.clientHeight;
1245
- const invisibleClass = 'p-mod-invisible';
1246
- let startIndex = -1;
1247
- const n = this.contentNode.children.length;
1248
- for (let i = 0; i < n; i++) {
1249
- const tab = this.contentNode.children[i] as HTMLLIElement;
1250
- if (tab.offsetTop + tab.offsetHeight >= availableHeight) {
1251
- tab.classList.add(invisibleClass);
1252
- if (startIndex === -1) {
1253
- startIndex = i;
1254
- /* If only one element is overflowing and the additional menu widget is visible (i.e. this.tabsOverflowData is set)
1255
- * there might already be enough space to show the last tab. In this case, we need to include the size of the
1256
- * additional menu widget and recheck if the last tab is visible */
1257
- if (startIndex === n - 1 && this.tabsOverflowData) {
1258
- const additionalViewsMenu = this.node.parentElement?.querySelector('.theia-additional-views-menu') as HTMLDivElement;
1259
- if (tab.offsetTop + tab.offsetHeight < availableHeight + additionalViewsMenu.offsetHeight) {
1260
- tab.classList.remove(invisibleClass);
1261
- startIndex = -1;
1262
- }
1263
- }
1264
- }
1265
- } else {
1266
- tab.classList.remove(invisibleClass);
1267
- }
1268
- }
1269
- return startIndex;
1270
- }
1271
-
1272
- /**
1273
- * Render the tab bar using the given DOM element as host. The optional `renderData` is forwarded
1274
- * to the TabBarRenderer.
1275
- */
1276
- protected renderTabs(host: HTMLElement, renderData?: Partial<SideBarRenderData>[]): void {
1277
- const titles = this.titles;
1278
- const n = titles.length;
1279
- const renderer = this.renderer as TabBarRenderer;
1280
- const currentTitle = this.currentTitle;
1281
- const content = new Array<VirtualElement>(n);
1282
- for (let i = 0; i < n; i++) {
1283
- const title = titles[i];
1284
- const current = title === currentTitle;
1285
- const zIndex = current ? n : n - i - 1;
1286
- let rd: SideBarRenderData;
1287
- if (renderData && i < renderData.length) {
1288
- rd = { title, current, zIndex, ...renderData[i] };
1289
- } else {
1290
- rd = { title, current, zIndex };
1291
- }
1292
- // Based on how renderTabs() is called, assume renderData will be undefined when invoked for this.hiddenContentNode
1293
- content[i] = renderer.renderTab(rd, true, renderData === undefined);
1294
- }
1295
- VirtualDOM.render(content, host);
1296
- }
1297
-
1298
- /**
1299
- * The following event processing is used to generate `collapseRequested` signals
1300
- * when the mouse goes up on the currently selected tab without too much movement
1301
- * between `mousedown` and `mouseup`. The movement threshold is the same that
1302
- * is used by the superclass to detect a drag event. The `allowDeselect` option
1303
- * of the TabBar constructor cannot be used here because it is triggered when the
1304
- * mouse goes down, and thus collides with dragging.
1305
- */
1306
- override handleEvent(event: Event): void {
1307
- switch (event.type) {
1308
- case 'mousedown':
1309
- this.onMouseDown(event as MouseEvent);
1310
- super.handleEvent(event);
1311
- break;
1312
- case 'mouseup':
1313
- super.handleEvent(event);
1314
- this.onMouseUp(event as MouseEvent);
1315
- break;
1316
- case 'mousemove':
1317
- this.onMouseMove(event as MouseEvent);
1318
- super.handleEvent(event);
1319
- break;
1320
- case 'p-dragenter':
1321
- this.onDragEnter(event as IDragEvent);
1322
- break;
1323
- case 'p-dragover':
1324
- this.onDragOver(event as IDragEvent);
1325
- break;
1326
- case 'p-dragleave': case 'p-drop':
1327
- this.cancelViewContainerDND();
1328
- break;
1329
- default:
1330
- super.handleEvent(event);
1331
- }
1332
- }
1333
-
1334
- protected onMouseDown(event: MouseEvent): void {
1335
- // Check for left mouse button and current mouse status
1336
- if (event.button !== 0 || this.mouseData) {
1337
- return;
1338
- }
1339
-
1340
- // Check whether the mouse went down on the current tab
1341
- const tabs = this.contentNode.children;
1342
- const index = ArrayExt.findFirstIndex(tabs, tab => ElementExt.hitTest(tab, event.clientX, event.clientY));
1343
- if (index < 0 || index !== this.currentIndex) {
1344
- return;
1345
- }
1346
-
1347
- // Check whether the close button was clicked
1348
- const icon = tabs[index].querySelector(this.renderer.closeIconSelector);
1349
- if (icon && icon.contains(event.target as HTMLElement)) {
1350
- return;
1351
- }
1352
-
1353
- this.mouseData = {
1354
- pressX: event.clientX,
1355
- pressY: event.clientY,
1356
- mouseDownTabIndex: index
1357
- };
1358
- }
1359
-
1360
- protected onMouseUp(event: MouseEvent): void {
1361
- // Check for left mouse button and current mouse status
1362
- if (event.button !== 0 || !this.mouseData) {
1363
- return;
1364
- }
1365
-
1366
- // Check whether the mouse went up on the current tab
1367
- const mouseDownTabIndex = this.mouseData.mouseDownTabIndex;
1368
- this.mouseData = undefined;
1369
- const tabs = this.contentNode.children;
1370
- const index = ArrayExt.findFirstIndex(tabs, tab => ElementExt.hitTest(tab, event.clientX, event.clientY));
1371
- if (index < 0 || index !== mouseDownTabIndex) {
1372
- return;
1373
- }
1374
-
1375
- // Collapse the side bar
1376
- this.collapseRequested.emit(this.titles[index]);
1377
- }
1378
-
1379
- protected onMouseMove(event: MouseEvent): void {
1380
- // Check for left mouse button and current mouse status
1381
- if (event.button !== 0 || !this.mouseData) {
1382
- return;
1383
- }
1384
-
1385
- const data = this.mouseData;
1386
- const dx = Math.abs(event.clientX - data.pressX);
1387
- const dy = Math.abs(event.clientY - data.pressY);
1388
- const threshold = SideTabBar.DRAG_THRESHOLD;
1389
- if (dx >= threshold || dy >= threshold) {
1390
- this.mouseData = undefined;
1391
- }
1392
- }
1393
-
1394
- toCancelViewContainerDND = new DisposableCollection();
1395
- protected cancelViewContainerDND = () => {
1396
- this.toCancelViewContainerDND.dispose();
1397
- };
1398
-
1399
- /**
1400
- * Handles `viewContainerPart` drag enter.
1401
- */
1402
- protected onDragEnter = (event: IDragEvent) => {
1403
- this.cancelViewContainerDND();
1404
- if (event.mimeData.getData('application/vnd.phosphor.view-container-factory')) {
1405
- event.preventDefault();
1406
- event.stopPropagation();
1407
- }
1408
- };
1409
-
1410
- /**
1411
- * Handle `viewContainerPart` drag over,
1412
- * Defines the appropriate `dropAction` and opens the tab on which the mouse stands on for more than 800 ms.
1413
- */
1414
- protected onDragOver = (event: IDragEvent) => {
1415
- const factory = event.mimeData.getData('application/vnd.phosphor.view-container-factory');
1416
- const widget = factory && factory();
1417
- if (!widget) {
1418
- event.dropAction = 'none';
1419
- return;
1420
- }
1421
- event.preventDefault();
1422
- event.stopPropagation();
1423
- if (!this.toCancelViewContainerDND.disposed) {
1424
- event.dropAction = event.proposedAction;
1425
- return;
1426
- }
1427
-
1428
- const { target, clientX, clientY } = event;
1429
- if (target instanceof HTMLElement) {
1430
- if (widget.options.disableDraggingToOtherContainers || widget.viewContainer.disableDNDBetweenContainers) {
1431
- event.dropAction = 'none';
1432
- target.classList.add('theia-cursor-no-drop');
1433
- this.toCancelViewContainerDND.push(Disposable.create(() => {
1434
- target.classList.remove('theia-cursor-no-drop');
1435
- }));
1436
- } else {
1437
- event.dropAction = event.proposedAction;
1438
- }
1439
- const { top, bottom, left, right, height } = target.getBoundingClientRect();
1440
- const mouseOnTop = (clientY - top) < (height / 2);
1441
- const dropTargetClass = `drop-target-${mouseOnTop ? 'top' : 'bottom'}`;
1442
- const tabs = this.contentNode.children;
1443
- const targetTab = ArrayExt.findFirstValue(tabs, t => ElementExt.hitTest(t, clientX, clientY));
1444
- if (!targetTab) {
1445
- return;
1446
- }
1447
- targetTab.classList.add(dropTargetClass);
1448
- this.toCancelViewContainerDND.push(Disposable.create(() => {
1449
- if (targetTab) {
1450
- targetTab.classList.remove(dropTargetClass);
1451
- }
1452
- }));
1453
- const openTabTimer = setTimeout(() => {
1454
- const title = this.titles.find(t => (this.renderer as TabBarRenderer).createTabId(t) === targetTab.id);
1455
- if (title) {
1456
- const mouseStillOnTab = clientX >= left && clientX <= right && clientY >= top && clientY <= bottom;
1457
- if (mouseStillOnTab) {
1458
- this.currentTitle = title;
1459
- }
1460
- }
1461
- }, 800);
1462
- this.toCancelViewContainerDND.push(Disposable.create(() => {
1463
- clearTimeout(openTabTimer);
1464
- }));
1465
- }
1466
- };
1467
-
1468
- }
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 PerfectScrollbar from 'perfect-scrollbar';
18
+ import { TabBar, Title, Widget } from '@phosphor/widgets';
19
+ import { VirtualElement, h, VirtualDOM, ElementInlineStyle } from '@phosphor/virtualdom';
20
+ import { Disposable, DisposableCollection, MenuPath, notEmpty, SelectionService, CommandService, nls, ArrayUtils } from '../../common';
21
+ import { ContextMenuRenderer } from '../context-menu-renderer';
22
+ import { Signal, Slot } from '@phosphor/signaling';
23
+ import { Message, MessageLoop } from '@phosphor/messaging';
24
+ import { ArrayExt } from '@phosphor/algorithm';
25
+ import { ElementExt } from '@phosphor/domutils';
26
+ import { TabBarToolbarRegistry, TabBarToolbar } from './tab-bar-toolbar';
27
+ import { TheiaDockPanel, MAIN_AREA_ID, BOTTOM_AREA_ID } from './theia-dock-panel';
28
+ import { WidgetDecoration } from '../widget-decoration';
29
+ import { TabBarDecoratorService } from './tab-bar-decorator';
30
+ import { IconThemeService } from '../icon-theme-service';
31
+ import { BreadcrumbsRenderer, BreadcrumbsRendererFactory } from '../breadcrumbs/breadcrumbs-renderer';
32
+ import { NavigatableWidget } from '../navigatable-types';
33
+ import { IDragEvent } from '@phosphor/dragdrop';
34
+ import { LOCKED_CLASS, PINNED_CLASS } from '../widgets/widget';
35
+ import { CorePreferences } from '../core-preferences';
36
+ import { HoverService } from '../hover-service';
37
+ import { Root, createRoot } from 'react-dom/client';
38
+ import { SelectComponent } from '../widgets/select-component';
39
+ import { createElement } from 'react';
40
+ import { PreviewableWidget } from '../widgets/previewable-widget';
41
+ import { EnhancedPreviewWidget } from '../widgets/enhanced-preview-widget';
42
+ import { ContextKeyService } from '../context-key-service';
43
+
44
+ /** The class name added to hidden content nodes, which are required to render vertical side bars. */
45
+ const HIDDEN_CONTENT_CLASS = 'theia-TabBar-hidden-content';
46
+
47
+ /** Menu path for tab bars used throughout the application shell. */
48
+ export const SHELL_TABBAR_CONTEXT_MENU: MenuPath = ['shell-tabbar-context-menu'];
49
+ export const SHELL_TABBAR_CONTEXT_CLOSE: MenuPath = [...SHELL_TABBAR_CONTEXT_MENU, '0_close'];
50
+ export const SHELL_TABBAR_CONTEXT_COPY: MenuPath = [...SHELL_TABBAR_CONTEXT_MENU, '1_copy'];
51
+ // Kept here in anticipation of tab pinning behavior implemented in tab-bars.ts
52
+ export const SHELL_TABBAR_CONTEXT_PIN: MenuPath = [...SHELL_TABBAR_CONTEXT_MENU, '4_pin'];
53
+ export const SHELL_TABBAR_CONTEXT_SPLIT: MenuPath = [...SHELL_TABBAR_CONTEXT_MENU, '5_split'];
54
+
55
+ export const TabBarRendererFactory = Symbol('TabBarRendererFactory');
56
+ export type TabBarRendererFactory = () => TabBarRenderer;
57
+
58
+ /**
59
+ * Size information of DOM elements used for rendering tabs in side bars.
60
+ */
61
+ export interface SizeData {
62
+ width: number;
63
+ height: number;
64
+ }
65
+
66
+ /**
67
+ * Extension of the rendering data used for tabs in side bars of the application shell.
68
+ */
69
+ export interface SideBarRenderData extends TabBar.IRenderData<Widget> {
70
+ labelSize?: SizeData;
71
+ iconSize?: SizeData;
72
+ paddingTop?: number;
73
+ paddingBottom?: number;
74
+ visible?: boolean
75
+ }
76
+
77
+ export interface ScrollableRenderData extends TabBar.IRenderData<Widget> {
78
+ tabWidth?: number;
79
+ }
80
+
81
+ /**
82
+ * A tab bar renderer that offers a context menu. In addition, this renderer is able to
83
+ * set an explicit position and size on the icon and label of each tab in a side bar.
84
+ * This is necessary because the elements of side bar tabs are rotated using the CSS
85
+ * `transform` property, disrupting the browser's ability to arrange those elements
86
+ * automatically.
87
+ */
88
+ export class TabBarRenderer extends TabBar.Renderer {
89
+ /**
90
+ * The menu path used to render the context menu.
91
+ */
92
+ contextMenuPath?: MenuPath;
93
+
94
+ protected readonly toDispose = new DisposableCollection();
95
+
96
+ // TODO refactor shell, rendered should only receive props with event handlers
97
+ // events should be handled by clients, like ApplicationShell
98
+ // right now it is mess: (1) client logic belong to renderer, (2) cyclic dependencies between renderers and clients
99
+ constructor(
100
+ protected readonly contextMenuRenderer?: ContextMenuRenderer,
101
+ protected readonly decoratorService?: TabBarDecoratorService,
102
+ protected readonly iconThemeService?: IconThemeService,
103
+ protected readonly selectionService?: SelectionService,
104
+ protected readonly commandService?: CommandService,
105
+ protected readonly corePreferences?: CorePreferences,
106
+ protected readonly hoverService?: HoverService,
107
+ protected readonly contextKeyService?: ContextKeyService,
108
+ ) {
109
+ super();
110
+ if (this.decoratorService) {
111
+ this.toDispose.push(Disposable.create(() => this.resetDecorations()));
112
+ this.toDispose.push(this.decoratorService.onDidChangeDecorations(() => this.resetDecorations()));
113
+ }
114
+ if (this.iconThemeService) {
115
+ this.toDispose.push(this.iconThemeService.onDidChangeCurrent(() => {
116
+ if (this._tabBar) {
117
+ this._tabBar.update();
118
+ }
119
+ }));
120
+ }
121
+ }
122
+
123
+ dispose(): void {
124
+ this.toDispose.dispose();
125
+ }
126
+
127
+ protected _tabBar?: TabBar<Widget>;
128
+ protected readonly toDisposeOnTabBar = new DisposableCollection();
129
+ /**
130
+ * A reference to the tab bar is required in order to activate it when a context menu
131
+ * is requested.
132
+ */
133
+ set tabBar(tabBar: TabBar<Widget> | undefined) {
134
+ if (this.toDispose.disposed) {
135
+ throw new Error('disposed');
136
+ }
137
+ if (this._tabBar === tabBar) {
138
+ return;
139
+ }
140
+ this.toDisposeOnTabBar.dispose();
141
+ this.toDispose.push(this.toDisposeOnTabBar);
142
+ this._tabBar = tabBar;
143
+ if (tabBar) {
144
+ const listener: Slot<Widget, TabBar.ITabCloseRequestedArgs<Widget>> = (_, { title }) => this.resetDecorations(title);
145
+ tabBar.tabCloseRequested.connect(listener);
146
+ this.toDisposeOnTabBar.push(Disposable.create(() => tabBar.tabCloseRequested.disconnect(listener)));
147
+ }
148
+ this.resetDecorations();
149
+ }
150
+ get tabBar(): TabBar<Widget> | undefined {
151
+ return this._tabBar;
152
+ }
153
+
154
+ /**
155
+ * Render tabs with the default DOM structure, but additionally register a context menu listener.
156
+ * @param {SideBarRenderData} data Data used to render the tab.
157
+ * @param {boolean} isInSidePanel An optional check which determines if the tab is in the side-panel.
158
+ * @param {boolean} isPartOfHiddenTabBar An optional check which determines if the tab is in the hidden horizontal tab bar.
159
+ * @returns {VirtualElement} The virtual element of the rendered tab.
160
+ */
161
+ override renderTab(data: SideBarRenderData, isInSidePanel?: boolean, isPartOfHiddenTabBar?: boolean): VirtualElement {
162
+ const title = data.title;
163
+ const id = this.createTabId(title, isPartOfHiddenTabBar);
164
+ const key = this.createTabKey(data);
165
+ const style = this.createTabStyle(data);
166
+ const className = this.createTabClass(data);
167
+ const dataset = this.createTabDataset(data);
168
+ const closeIconTitle = data.title.className.includes(PINNED_CLASS)
169
+ ? nls.localizeByDefault('Unpin')
170
+ : nls.localizeByDefault('Close');
171
+
172
+ const hover = this.tabBar && (this.tabBar.orientation === 'horizontal' && this.corePreferences?.['window.tabbar.enhancedPreview'] === 'classic')
173
+ ? { title: title.caption }
174
+ : {
175
+ onmouseenter: this.handleMouseEnterEvent
176
+ };
177
+
178
+ return h.li(
179
+ {
180
+ ...hover,
181
+ key, className, id, style, dataset,
182
+ oncontextmenu: this.handleContextMenuEvent,
183
+ ondblclick: this.handleDblClickEvent,
184
+ onauxclick: (e: MouseEvent) => {
185
+ // If user closes the tab using mouse wheel, nothing should be pasted to an active editor
186
+ e.preventDefault();
187
+ }
188
+ },
189
+ h.div(
190
+ { className: 'theia-tab-icon-label' },
191
+ this.renderIcon(data, isInSidePanel),
192
+ this.renderLabel(data, isInSidePanel),
193
+ this.renderTailDecorations(data, isInSidePanel),
194
+ this.renderBadge(data, isInSidePanel),
195
+ this.renderLock(data, isInSidePanel)
196
+ ),
197
+ h.div({
198
+ className: 'p-TabBar-tabCloseIcon action-label',
199
+ title: closeIconTitle,
200
+ onclick: this.handleCloseClickEvent
201
+ })
202
+ );
203
+ }
204
+
205
+ override createTabClass(data: SideBarRenderData): string {
206
+ let tabClass = super.createTabClass(data);
207
+ if (!(data.visible ?? true)) {
208
+ tabClass += ' p-mod-invisible';
209
+ }
210
+ return tabClass;
211
+ }
212
+
213
+ /**
214
+ * Generate ID for an entry in the tab bar
215
+ * @param {Title<Widget>} title Title of the widget controlled by this tab bar
216
+ * @param {boolean} isPartOfHiddenTabBar Tells us if this entry is part of the hidden horizontal tab bar.
217
+ * If yes, add a suffix to differentiate it's ID from the entry in the visible tab bar
218
+ * @returns {string} DOM element ID
219
+ */
220
+ createTabId(title: Title<Widget>, isPartOfHiddenTabBar = false): string {
221
+ return 'shell-tab-' + title.owner.id + (isPartOfHiddenTabBar ? '-hidden' : '');
222
+ }
223
+
224
+ /**
225
+ * If size information is available for the label and icon, set an explicit height on the tab.
226
+ * The height value also considers padding, which should be derived from CSS settings.
227
+ */
228
+ override createTabStyle(data: SideBarRenderData & ScrollableRenderData): ElementInlineStyle {
229
+ const zIndex = `${data.zIndex}`;
230
+ const labelSize = data.labelSize;
231
+ const iconSize = data.iconSize;
232
+ let height: string | undefined;
233
+ let width: string | undefined;
234
+ if (labelSize || iconSize) {
235
+ const labelHeight = labelSize ? (this.tabBar && this.tabBar.orientation === 'horizontal' ? labelSize.height : labelSize.width) : 0;
236
+ const iconHeight = iconSize ? iconSize.height : 0;
237
+ let paddingTop = data.paddingTop || 0;
238
+ if (labelHeight > 0 && iconHeight > 0) {
239
+ // Leave some extra space between icon and label
240
+ paddingTop = paddingTop * 1.5;
241
+ }
242
+ const paddingBottom = data.paddingBottom || 0;
243
+ height = `${labelHeight + iconHeight + paddingTop + paddingBottom}px`;
244
+ }
245
+ if (data.tabWidth) {
246
+ width = `${data.tabWidth}px`;
247
+ } else {
248
+ width = '';
249
+ }
250
+ return { zIndex, height, minWidth: width, maxWidth: width };
251
+ }
252
+
253
+ /**
254
+ * If size information is available for the label, set it as inline style.
255
+ * Tab padding and icon size are also considered in the `top` position.
256
+ * @param {SideBarRenderData} data Data used to render the tab.
257
+ * @param {boolean} isInSidePanel An optional check which determines if the tab is in the side-panel.
258
+ * @returns {VirtualElement} The virtual element of the rendered label.
259
+ */
260
+ override renderLabel(data: SideBarRenderData, isInSidePanel?: boolean): VirtualElement {
261
+ const labelSize = data.labelSize;
262
+ const iconSize = data.iconSize;
263
+ let width: string | undefined;
264
+ let height: string | undefined;
265
+ let top: string | undefined;
266
+ if (labelSize) {
267
+ width = `${labelSize.width}px`;
268
+ height = `${labelSize.height}px`;
269
+ }
270
+ if (data.paddingTop || iconSize) {
271
+ const iconHeight = iconSize ? iconSize.height : 0;
272
+ let paddingTop = data.paddingTop || 0;
273
+ if (iconHeight > 0) {
274
+ // Leave some extra space between icon and label
275
+ paddingTop = paddingTop * 1.5;
276
+ }
277
+ top = `${paddingTop + iconHeight}px`;
278
+ }
279
+ const style: ElementInlineStyle = { width, height, top };
280
+ // No need to check for duplicate labels if the tab is rendered in the side panel (title is not displayed),
281
+ // or if there are less than two files in the tab bar.
282
+ if (isInSidePanel || (this.tabBar && this.tabBar.titles.length < 2)) {
283
+ return h.div({ className: 'p-TabBar-tabLabel', style }, data.title.label);
284
+ }
285
+ const originalToDisplayedMap = this.findDuplicateLabels([...this.tabBar!.titles]);
286
+ const labelDetails: string | undefined = originalToDisplayedMap.get(data.title.caption);
287
+ if (labelDetails) {
288
+ return h.div({ className: 'p-TabBar-tabLabelWrapper' },
289
+ h.div({ className: 'p-TabBar-tabLabel', style }, data.title.label),
290
+ h.div({ className: 'p-TabBar-tabLabelDetails', style }, labelDetails));
291
+ }
292
+ return h.div({ className: 'p-TabBar-tabLabel', style }, data.title.label);
293
+ }
294
+
295
+ protected renderTailDecorations(renderData: SideBarRenderData, isInSidePanel?: boolean): VirtualElement[] {
296
+ if (!this.corePreferences?.get('workbench.editor.decorations.badges')) {
297
+ return [];
298
+ }
299
+ const tailDecorations = ArrayUtils.coalesce(this.getDecorationData(renderData.title, 'tailDecorations')).flat();
300
+ if (tailDecorations === undefined || tailDecorations.length === 0) {
301
+ return [];
302
+ }
303
+ let dotDecoration: WidgetDecoration.TailDecoration.AnyPartial | undefined;
304
+ const otherDecorations: WidgetDecoration.TailDecoration.AnyPartial[] = [];
305
+ tailDecorations.reverse().forEach(decoration => {
306
+ const partial = decoration as WidgetDecoration.TailDecoration.AnyPartial;
307
+ if (WidgetDecoration.TailDecoration.isDotDecoration(partial)) {
308
+ dotDecoration ||= partial;
309
+ } else if (partial.data || partial.icon || partial.iconClass) {
310
+ otherDecorations.push(partial);
311
+ }
312
+ });
313
+ const decorationsToRender = dotDecoration ? [dotDecoration, ...otherDecorations] : otherDecorations;
314
+ return decorationsToRender.map((decoration, index) => {
315
+ const { tooltip, data, fontData, color, icon, iconClass } = decoration;
316
+ const iconToRender = icon ?? iconClass;
317
+ const className = ['p-TabBar-tail', 'flex'].join(' ');
318
+ const style = fontData ? fontData : color ? { color } : undefined;
319
+ const content = (data ? data : iconToRender
320
+ ? h.span({ className: this.getIconClass(iconToRender, iconToRender === 'circle' ? [WidgetDecoration.Styles.DECORATOR_SIZE_CLASS] : []) })
321
+ : '') + (index !== decorationsToRender.length - 1 ? ',' : '');
322
+ return h.span({ key: ('tailDecoration_' + index), className, style, title: tooltip ?? content }, content);
323
+ });
324
+ }
325
+
326
+ renderBadge(data: SideBarRenderData, isInSidePanel?: boolean): VirtualElement {
327
+ const totalBadge = this.getDecorationData(data.title, 'badge').reduce((sum, badge) => sum! + badge!, 0);
328
+ if (!totalBadge) {
329
+ return h.div({});
330
+ }
331
+ const limitedBadge = totalBadge >= 100 ? '99+' : totalBadge;
332
+ return isInSidePanel
333
+ ? h.div({ className: 'theia-badge-decorator-sidebar' }, `${limitedBadge}`)
334
+ : h.div({ className: 'theia-badge-decorator-horizontal' }, `${limitedBadge}`);
335
+ }
336
+
337
+ renderLock(data: SideBarRenderData, isInSidePanel?: boolean): VirtualElement {
338
+ return !isInSidePanel && data.title.className.includes(LOCKED_CLASS)
339
+ ? h.div({ className: 'p-TabBar-tabLock' })
340
+ : h.div({});
341
+ }
342
+
343
+ protected readonly decorations = new Map<Title<Widget>, WidgetDecoration.Data[]>();
344
+
345
+ protected resetDecorations(title?: Title<Widget>): void {
346
+ if (title) {
347
+ this.decorations.delete(title);
348
+ } else {
349
+ this.decorations.clear();
350
+ }
351
+ if (this.tabBar) {
352
+ this.tabBar.update();
353
+ }
354
+ }
355
+
356
+ /**
357
+ * Get all available decorations of a given tab.
358
+ * @param {string} title The widget title.
359
+ */
360
+ protected getDecorations(title: Title<Widget>): WidgetDecoration.Data[] {
361
+ if (this.tabBar && this.decoratorService) {
362
+ const owner: { resetTabBarDecorations?: () => void; } & Widget = title.owner;
363
+ if (!owner.resetTabBarDecorations) {
364
+ owner.resetTabBarDecorations = () => this.decorations.delete(title);
365
+ title.owner.disposed.connect(owner.resetTabBarDecorations);
366
+ }
367
+
368
+ const decorations = this.decorations.get(title) || this.decoratorService.getDecorations(title);
369
+ this.decorations.set(title, decorations);
370
+ return decorations;
371
+ }
372
+ return [];
373
+ }
374
+
375
+ /**
376
+ * Get the decoration data given the tab URI and the decoration data type.
377
+ * @param {string} title The title.
378
+ * @param {K} key The type of the decoration data.
379
+ */
380
+ protected getDecorationData<K extends keyof WidgetDecoration.Data>(title: Title<Widget>, key: K): WidgetDecoration.Data[K][] {
381
+ return this.getDecorations(title).filter(data => data[key] !== undefined).map(data => data[key]);
382
+ }
383
+
384
+ /**
385
+ * Get the class of an icon.
386
+ * @param {string | string[]} iconName The name of the icon.
387
+ * @param {string[]} additionalClasses Additional classes of the icon.
388
+ */
389
+ protected getIconClass(iconName: string | string[], additionalClasses: string[] = []): string {
390
+ const iconClass = (typeof iconName === 'string') ? ['a', 'fa', `fa-${iconName}`] : ['a'].concat(iconName);
391
+ return iconClass.concat(additionalClasses).join(' ');
392
+ }
393
+
394
+ /**
395
+ * Find duplicate labels from the currently opened tabs in the tab bar.
396
+ * Return the appropriate partial paths that can distinguish the identical labels.
397
+ *
398
+ * E.g., a/p/index.ts => a/..., b/p/index.ts => b/...
399
+ *
400
+ * To prevent excessively long path displayed, show at maximum three levels from the end by default.
401
+ * @param {Title<Widget>[]} titles Array of titles in the current tab bar.
402
+ * @returns {Map<string, string>} A map from each tab's original path to its displayed partial path.
403
+ */
404
+ findDuplicateLabels(titles: Title<Widget>[]): Map<string, string> {
405
+ // Filter from all tabs to group them by the distinct label (file name).
406
+ // E.g., 'foo.js' => {0 (index) => 'a/b/foo.js', '2 => a/c/foo.js' },
407
+ // 'bar.js' => {1 => 'a/d/bar.js', ...}
408
+ const labelGroups = new Map<string, Map<number, string>>();
409
+ titles.forEach((title, index) => {
410
+ if (!labelGroups.has(title.label)) {
411
+ labelGroups.set(title.label, new Map<number, string>());
412
+ }
413
+ labelGroups.get(title.label)!.set(index, title.caption);
414
+ });
415
+
416
+ const originalToDisplayedMap = new Map<string, string>();
417
+ // Parse each group of editors with the same label.
418
+ labelGroups.forEach(labelGroup => {
419
+ // Filter to get groups that have duplicates.
420
+ if (labelGroup.size > 1) {
421
+ const paths: string[][] = [];
422
+ let maxPathLength = 0;
423
+ labelGroup.forEach((pathStr, index) => {
424
+ const steps = pathStr.split('/');
425
+ maxPathLength = Math.max(maxPathLength, steps.length);
426
+ paths[index] = (steps.slice(0, steps.length - 1));
427
+ // By default, show at maximum three levels from the end.
428
+ let defaultDisplayedPath = steps.slice(-4, -1).join('/');
429
+ if (steps.length > 4) {
430
+ defaultDisplayedPath = '.../' + defaultDisplayedPath;
431
+ }
432
+ originalToDisplayedMap.set(pathStr, defaultDisplayedPath);
433
+ });
434
+
435
+ // Iterate through the steps of the path from the left to find the step that can distinguish it.
436
+ // E.g., ['root', 'foo', 'c'], ['root', 'bar', 'd'] => 'foo', 'bar'
437
+ let i = 0;
438
+ while (i < maxPathLength - 1) {
439
+ // Store indexes of all paths that have the identical element in each step.
440
+ const stepOccurrences = new Map<string, number[]>();
441
+ // Compare the current step of all paths
442
+ paths.forEach((path, index) => {
443
+ const step = path[i];
444
+ if (path.length > 0) {
445
+ if (i > path.length - 1) {
446
+ paths[index] = [];
447
+ } else if (!stepOccurrences.has(step)) {
448
+ stepOccurrences.set(step, [index]);
449
+ } else {
450
+ stepOccurrences.get(step)!.push(index);
451
+ }
452
+ }
453
+ });
454
+ // Set the displayed path for each tab.
455
+ stepOccurrences.forEach((indexArr, displayedPath) => {
456
+ if (indexArr.length === 1) {
457
+ const originalPath = labelGroup.get(indexArr[0]);
458
+ if (originalPath) {
459
+ const originalElements = originalPath.split('/');
460
+ const displayedElements = displayedPath.split('/');
461
+ if (originalElements.slice(-2)[0] !== displayedElements.slice(-1)[0]) {
462
+ displayedPath += '/...';
463
+ }
464
+ if (originalElements[0] !== displayedElements[0]) {
465
+ displayedPath = '.../' + displayedPath;
466
+ }
467
+ originalToDisplayedMap.set(originalPath, displayedPath);
468
+ paths[indexArr[0]] = [];
469
+ }
470
+ }
471
+ });
472
+ i++;
473
+ }
474
+ }
475
+ });
476
+ return originalToDisplayedMap;
477
+ }
478
+
479
+ /**
480
+ * If size information is available for the icon, set it as inline style. Tab padding
481
+ * is also considered in the `top` position.
482
+ * @param {SideBarRenderData} data Data used to render the tab icon.
483
+ * @param {boolean} isInSidePanel An optional check which determines if the tab is in the side-panel.
484
+ */
485
+ override renderIcon(data: SideBarRenderData, isInSidePanel?: boolean): VirtualElement {
486
+ if (!isInSidePanel && this.iconThemeService && this.iconThemeService.current === 'none') {
487
+ return h.div();
488
+ }
489
+ let top: string | undefined;
490
+ if (data.paddingTop) {
491
+ top = `${data.paddingTop || 0}px`;
492
+ }
493
+ const style: ElementInlineStyle = { top };
494
+ const baseClassName = this.createIconClass(data);
495
+
496
+ const overlayIcons: VirtualElement[] = [];
497
+ const decorationData = this.getDecorationData(data.title, 'iconOverlay');
498
+
499
+ // Check if the tab has decoration markers to be rendered on top.
500
+ if (decorationData.length > 0) {
501
+ const baseIcon: VirtualElement = h.div({ className: baseClassName, style }, data.title.iconLabel);
502
+ const wrapperClassName: string = WidgetDecoration.Styles.ICON_WRAPPER_CLASS;
503
+ const decoratorSizeClassName: string = isInSidePanel ? WidgetDecoration.Styles.DECORATOR_SIDEBAR_SIZE_CLASS : WidgetDecoration.Styles.DECORATOR_SIZE_CLASS;
504
+
505
+ decorationData
506
+ .filter(notEmpty)
507
+ .map(overlay => [overlay.position, overlay] as [WidgetDecoration.IconOverlayPosition, WidgetDecoration.IconOverlay | WidgetDecoration.IconClassOverlay])
508
+ .forEach(([position, overlay]) => {
509
+ const iconAdditionalClasses: string[] = [decoratorSizeClassName, WidgetDecoration.IconOverlayPosition.getStyle(position, isInSidePanel)];
510
+ const overlayIconStyle = (color?: string) => {
511
+ if (color === undefined) {
512
+ return {};
513
+ }
514
+ return { color };
515
+ };
516
+ // Parse the optional background (if it exists) of the overlay icon.
517
+ if (overlay.background) {
518
+ const backgroundIconClassName = this.getIconClass(overlay.background.shape, iconAdditionalClasses);
519
+ overlayIcons.push(
520
+ h.div({ key: data.title.label + '-background', className: backgroundIconClassName, style: overlayIconStyle(overlay.background.color) })
521
+ );
522
+ }
523
+ // Parse the overlay icon.
524
+ const overlayIcon = (overlay as WidgetDecoration.IconOverlay).icon || (overlay as WidgetDecoration.IconClassOverlay).iconClass;
525
+ const overlayIconClassName = this.getIconClass(overlayIcon, iconAdditionalClasses);
526
+ overlayIcons.push(
527
+ h.span({ key: data.title.label, className: overlayIconClassName, style: overlayIconStyle(overlay.color) })
528
+ );
529
+ });
530
+ return h.div({ className: wrapperClassName, style }, [baseIcon, ...overlayIcons]);
531
+ }
532
+ return h.div({ className: baseClassName, style }, data.title.iconLabel);
533
+ }
534
+
535
+ protected renderEnhancedPreview = (title: Title<Widget>) => {
536
+ const hoverBox = document.createElement('div');
537
+ hoverBox.classList.add('theia-horizontal-tabBar-hover-div');
538
+ const labelElement = document.createElement('p');
539
+ labelElement.classList.add('theia-horizontal-tabBar-hover-title');
540
+ labelElement.textContent = title.label;
541
+ hoverBox.append(labelElement);
542
+ const widget = title.owner;
543
+ if (EnhancedPreviewWidget.is(widget)) {
544
+ const enhancedPreviewNode = widget.getEnhancedPreviewNode();
545
+ if (enhancedPreviewNode) {
546
+ hoverBox.appendChild(enhancedPreviewNode);
547
+ }
548
+ } else if (title.caption) {
549
+ const captionElement = document.createElement('p');
550
+ captionElement.classList.add('theia-horizontal-tabBar-hover-caption');
551
+ captionElement.textContent = title.caption;
552
+ hoverBox.appendChild(captionElement);
553
+ }
554
+ return hoverBox;
555
+ };
556
+
557
+ protected renderVisualPreview(desiredWidth: number, title: Title<Widget>): HTMLElement | undefined {
558
+ const widget = title.owner;
559
+ // Check that the widget is not currently shown, is a PreviewableWidget and it was already loaded before
560
+ if (this.tabBar && this.tabBar.currentTitle !== title && PreviewableWidget.isPreviewable(widget)) {
561
+ const html = document.getElementById(widget.id);
562
+ if (html) {
563
+ const previewNode: Node | undefined = widget.getPreviewNode();
564
+ if (previewNode) {
565
+ const clonedNode = previewNode.cloneNode(true);
566
+ const visualPreviewDiv = document.createElement('div');
567
+ visualPreviewDiv.classList.add('enhanced-preview-container');
568
+ // Add the clonedNode and get it from the children to have a HTMLElement instead of a Node
569
+ visualPreviewDiv.append(clonedNode);
570
+ const visualPreview = visualPreviewDiv.children.item(visualPreviewDiv.children.length - 1);
571
+ if (visualPreview instanceof HTMLElement) {
572
+ visualPreview.classList.remove('p-mod-hidden');
573
+ visualPreview.classList.add('enhanced-preview');
574
+ visualPreview.id = `preview:${widget.id}`;
575
+
576
+ // Use the current visible editor as a fallback if not available
577
+ const height: number = visualPreview.style.height === '' ? this.tabBar.currentTitle!.owner.node.offsetHeight : parseFloat(visualPreview.style.height);
578
+ const width: number = visualPreview.style.width === '' ? this.tabBar.currentTitle!.owner.node.offsetWidth : parseFloat(visualPreview.style.width);
579
+ const desiredRatio = 9 / 16;
580
+ const desiredHeight = desiredWidth * desiredRatio;
581
+ const ratio = height / width;
582
+ visualPreviewDiv.style.width = `${desiredWidth}px`;
583
+ visualPreviewDiv.style.height = `${desiredHeight}px`;
584
+
585
+ // If the view is wider than the desiredRatio scale the width and crop the height. If the view is longer its the other way around.
586
+ const scale = ratio < desiredRatio ? (desiredHeight / height) : (desiredWidth / width);
587
+ visualPreview.style.transform = `scale(${scale},${scale})`;
588
+ visualPreview.style.removeProperty('top');
589
+ visualPreview.style.removeProperty('left');
590
+
591
+ // Copy canvases (They are cloned empty)
592
+ const originalCanvases = html.getElementsByTagName('canvas');
593
+ const previewCanvases = visualPreview.getElementsByTagName('canvas');
594
+ // If this is not given, something went wrong during the cloning
595
+ if (originalCanvases.length === previewCanvases.length) {
596
+ for (let i = 0; i < originalCanvases.length; i++) {
597
+ previewCanvases[i].getContext('2d')?.drawImage(originalCanvases[i], 0, 0);
598
+ }
599
+ }
600
+
601
+ return visualPreviewDiv;
602
+ }
603
+ }
604
+ }
605
+ }
606
+ return undefined;
607
+ }
608
+
609
+ protected handleMouseEnterEvent = (event: MouseEvent) => {
610
+ if (this.tabBar && this.hoverService && event.currentTarget instanceof HTMLElement) {
611
+ const id = event.currentTarget.id;
612
+ const title = this.tabBar.titles.find(t => this.createTabId(t) === id);
613
+ if (title) {
614
+ if (this.tabBar.orientation === 'horizontal') {
615
+ this.hoverService.requestHover({
616
+ content: this.renderEnhancedPreview(title),
617
+ target: event.currentTarget,
618
+ position: 'bottom',
619
+ cssClasses: ['extended-tab-preview'],
620
+ visualPreview: this.corePreferences?.['window.tabbar.enhancedPreview'] === 'visual' ? width => this.renderVisualPreview(width, title) : undefined
621
+ });
622
+ } else if (title.caption) {
623
+ this.hoverService.requestHover({
624
+ content: title.caption,
625
+ target: event.currentTarget,
626
+ position: 'right'
627
+ });
628
+ }
629
+ }
630
+ }
631
+ };
632
+
633
+ protected handleContextMenuEvent = (event: MouseEvent) => {
634
+ if (this.contextMenuRenderer && this.contextMenuPath && event.currentTarget instanceof HTMLElement) {
635
+ event.stopPropagation();
636
+ event.preventDefault();
637
+ let widget: Widget | undefined = undefined;
638
+ if (this.tabBar) {
639
+ const titleIndex = Array.from(this.tabBar.contentNode.getElementsByClassName('p-TabBar-tab'))
640
+ .findIndex(node => node.contains(event.currentTarget as HTMLElement));
641
+ if (titleIndex !== -1) {
642
+ widget = this.tabBar.titles[titleIndex].owner;
643
+ }
644
+ }
645
+
646
+ const oldSelection = this.selectionService?.selection;
647
+ if (widget && this.selectionService) {
648
+ this.selectionService.selection = NavigatableWidget.is(widget) ? { uri: widget.getResourceUri() } : widget;
649
+ }
650
+
651
+ const contextKeyServiceOverlay = this.contextKeyService?.createOverlay([['isTerminalTab', widget && 'terminalId' in widget]]);
652
+ this.contextMenuRenderer.render({
653
+ menuPath: this.contextMenuPath!,
654
+ anchor: event,
655
+ args: [event],
656
+ contextKeyService: contextKeyServiceOverlay,
657
+ // We'd like to wait until the command triggered by the context menu has been run, but this should let it get through the preamble, at least.
658
+ onHide: () => setTimeout(() => { if (this.selectionService) { this.selectionService.selection = oldSelection; } })
659
+ });
660
+ }
661
+ };
662
+
663
+ protected handleCloseClickEvent = (event: MouseEvent) => {
664
+ if (this.tabBar && event.currentTarget instanceof HTMLElement) {
665
+ const id = event.currentTarget.parentElement!.id;
666
+ const title = this.tabBar.titles.find(t => this.createTabId(t) === id);
667
+ if (title?.closable === false && title?.className.includes(PINNED_CLASS) && this.commandService) {
668
+ this.commandService.executeCommand('workbench.action.unpinEditor', event);
669
+ }
670
+ }
671
+ };
672
+
673
+ protected handleDblClickEvent = (event: MouseEvent) => {
674
+ if (!this.corePreferences?.get('workbench.tab.maximize')) {
675
+ return;
676
+ }
677
+ if (this.tabBar && event.currentTarget instanceof HTMLElement) {
678
+ const id = event.currentTarget.id;
679
+ const title = this.tabBar.titles.find(t => this.createTabId(t) === id);
680
+ const area = title?.owner.parent;
681
+ if (area instanceof TheiaDockPanel && (area.id === BOTTOM_AREA_ID || area.id === MAIN_AREA_ID)) {
682
+ area.toggleMaximized();
683
+ }
684
+ }
685
+ };
686
+
687
+ }
688
+
689
+ /**
690
+ * A specialized tab bar for the main and bottom areas.
691
+ */
692
+ export class ScrollableTabBar extends TabBar<Widget> {
693
+
694
+ protected scrollBar?: PerfectScrollbar;
695
+
696
+ protected scrollBarFactory: () => PerfectScrollbar;
697
+ protected pendingReveal?: Promise<void>;
698
+ protected isMouseOver = false;
699
+ protected needsRecompute = false;
700
+ protected tabSize = 0;
701
+ protected _dynamicTabOptions?: ScrollableTabBar.Options;
702
+ protected contentContainer: HTMLElement;
703
+ protected topRow: HTMLElement;
704
+
705
+ protected readonly toDispose = new DisposableCollection();
706
+ protected openTabsContainer: HTMLDivElement;
707
+ protected openTabsRoot: Root;
708
+
709
+ constructor(options?: TabBar.IOptions<Widget> & PerfectScrollbar.Options, dynamicTabOptions?: ScrollableTabBar.Options) {
710
+ super(options);
711
+ this.scrollBarFactory = () => new PerfectScrollbar(this.scrollbarHost, options);
712
+ this._dynamicTabOptions = dynamicTabOptions;
713
+ this.rewireDOM();
714
+ }
715
+
716
+ set dynamicTabOptions(options: ScrollableTabBar.Options | undefined) {
717
+ this._dynamicTabOptions = options;
718
+ this.updateTabs();
719
+ }
720
+
721
+ get dynamicTabOptions(): ScrollableTabBar.Options | undefined {
722
+ return this._dynamicTabOptions;
723
+ }
724
+
725
+ override dispose(): void {
726
+ if (this.isDisposed) {
727
+ return;
728
+ }
729
+ super.dispose();
730
+ this.toDispose.dispose();
731
+ }
732
+
733
+ /**
734
+ * Restructures the DOM defined in PhosphorJS.
735
+ *
736
+ * By default the tabs (`li`) are contained in the `this.contentNode` (`ul`) which is wrapped in a `div` (`this.node`).
737
+ * Instead of this structure, we add a container for the `this.contentNode` and for the toolbar.
738
+ * The scrollbar will only work for the `ul` part but it does not affect the toolbar, so it can be on the right hand-side.
739
+ */
740
+ protected rewireDOM(): void {
741
+ const contentNode = this.node.getElementsByClassName(ScrollableTabBar.Styles.TAB_BAR_CONTENT)[0];
742
+ if (!contentNode) {
743
+ throw new Error("'this.node' does not have the content as a direct child with class name 'p-TabBar-content'.");
744
+ }
745
+ this.node.removeChild(contentNode);
746
+ this.contentContainer = document.createElement('div');
747
+ this.contentContainer.classList.add(ScrollableTabBar.Styles.TAB_BAR_CONTENT_CONTAINER);
748
+ this.contentContainer.appendChild(contentNode);
749
+
750
+ this.topRow = document.createElement('div');
751
+ this.topRow.classList.add('theia-tabBar-tab-row');
752
+ this.topRow.appendChild(this.contentContainer);
753
+
754
+ this.openTabsContainer = document.createElement('div');
755
+ this.openTabsContainer.classList.add('theia-tabBar-open-tabs');
756
+ this.openTabsRoot = createRoot(this.openTabsContainer);
757
+ this.topRow.appendChild(this.openTabsContainer);
758
+
759
+ this.node.appendChild(this.topRow);
760
+ }
761
+
762
+ protected override onAfterAttach(msg: Message): void {
763
+ if (!this.scrollBar) {
764
+ this.scrollBar = this.scrollBarFactory();
765
+ }
766
+ this.node.addEventListener('mouseenter', () => { this.isMouseOver = true; });
767
+ this.node.addEventListener('mouseleave', () => {
768
+ this.isMouseOver = false;
769
+ if (this.needsRecompute) {
770
+ this.updateTabs();
771
+ }
772
+ });
773
+
774
+ super.onAfterAttach(msg);
775
+ }
776
+
777
+ protected override onBeforeDetach(msg: Message): void {
778
+ super.onBeforeDetach(msg);
779
+ if (this.scrollBar) {
780
+ this.scrollBar.destroy();
781
+ this.scrollBar = undefined;
782
+ }
783
+ }
784
+
785
+ protected override onUpdateRequest(msg: Message): void {
786
+ this.updateTabs();
787
+ }
788
+
789
+ protected updateTabs(): void {
790
+ const content = [];
791
+ if (this.dynamicTabOptions) {
792
+
793
+ this.openTabsRoot.render(createElement(SelectComponent, {
794
+ options: this.titles,
795
+ onChange: (option, index) => {
796
+ this.currentIndex = index;
797
+ },
798
+ alignment: 'right'
799
+ }));
800
+
801
+ if (this.isMouseOver) {
802
+ this.needsRecompute = true;
803
+ } else {
804
+ this.needsRecompute = false;
805
+ if (this.orientation === 'horizontal') {
806
+ let availableWidth = this.scrollbarHost.clientWidth;
807
+ let effectiveWidth = availableWidth;
808
+ if (!this.openTabsContainer.classList.contains('p-mod-hidden')) {
809
+ availableWidth += this.openTabsContainer.getBoundingClientRect().width;
810
+ }
811
+ if (this.dynamicTabOptions.minimumTabSize * this.titles.length <= availableWidth) {
812
+ effectiveWidth += this.openTabsContainer.getBoundingClientRect().width;
813
+ this.openTabsContainer.classList.add('p-mod-hidden');
814
+ } else {
815
+ this.openTabsContainer.classList.remove('p-mod-hidden');
816
+ }
817
+ this.tabSize = Math.max(Math.min(effectiveWidth / this.titles.length,
818
+ this.dynamicTabOptions.defaultTabSize), this.dynamicTabOptions.minimumTabSize);
819
+ }
820
+ }
821
+ this.node.classList.add('dynamic-tabs');
822
+ } else {
823
+ this.openTabsContainer.classList.add('p-mod-hidden');
824
+ this.node.classList.remove('dynamic-tabs');
825
+ }
826
+ for (let i = 0, n = this.titles.length; i < n; ++i) {
827
+ const title = this.titles[i];
828
+ const current = title === this.currentTitle;
829
+ const zIndex = current ? n : n - i - 1;
830
+ const renderData: ScrollableRenderData = { title: title, current: current, zIndex: zIndex };
831
+ if (this.dynamicTabOptions && this.orientation === 'horizontal') {
832
+ renderData.tabWidth = this.tabSize;
833
+ }
834
+ content[i] = this.renderer.renderTab(renderData);
835
+ }
836
+ VirtualDOM.render(content, this.contentNode);
837
+ if (this.scrollBar) {
838
+ if (!(this.dynamicTabOptions && this.isMouseOver)) {
839
+ this.scrollBar.update();
840
+ }
841
+ }
842
+ }
843
+
844
+ protected override onResize(msg: Widget.ResizeMessage): void {
845
+ super.onResize(msg);
846
+ if (this.dynamicTabOptions) {
847
+ this.updateTabs();
848
+ }
849
+ if (this.scrollBar) {
850
+ if (this.currentIndex >= 0) {
851
+ this.revealTab(this.currentIndex);
852
+ }
853
+ this.scrollBar.update();
854
+ }
855
+ }
856
+
857
+ /**
858
+ * Reveal the tab with the given index by moving the scroll bar if necessary.
859
+ */
860
+ revealTab(index: number): Promise<void> {
861
+ if (this.pendingReveal) {
862
+ // A reveal has already been scheduled
863
+ return this.pendingReveal;
864
+ }
865
+ const result = new Promise<void>((resolve, reject) => {
866
+ // The tab might not have been created yet, so wait until the next frame
867
+ window.requestAnimationFrame(() => {
868
+ const tab = this.contentNode.children[index] as HTMLElement;
869
+ if (tab && this.isVisible) {
870
+ const parent = this.scrollbarHost;
871
+ if (this.orientation === 'horizontal') {
872
+ const scroll = parent.scrollLeft;
873
+ const left = tab.offsetLeft;
874
+ if (scroll > left) {
875
+ parent.scrollLeft = left;
876
+ } else {
877
+ const right = left + tab.clientWidth - parent.clientWidth;
878
+ if (scroll < right && tab.clientWidth < parent.clientWidth) {
879
+ parent.scrollLeft = right;
880
+ }
881
+ }
882
+ } else {
883
+ const scroll = parent.scrollTop;
884
+ const top = tab.offsetTop;
885
+ if (scroll > top) {
886
+ parent.scrollTop = top;
887
+ } else {
888
+ const bottom = top + tab.clientHeight - parent.clientHeight;
889
+ if (scroll < bottom && tab.clientHeight < parent.clientHeight) {
890
+ parent.scrollTop = bottom;
891
+ }
892
+ }
893
+ }
894
+ }
895
+ if (this.pendingReveal === result) {
896
+ this.pendingReveal = undefined;
897
+ }
898
+ resolve();
899
+ });
900
+ });
901
+ this.pendingReveal = result;
902
+ return result;
903
+ }
904
+
905
+ /**
906
+ * Overrides the `contentNode` property getter in PhosphorJS' TabBar.
907
+ */
908
+ // @ts-expect-error TS2611 `TabBar<T>.contentNode` is declared as `readonly contentNode` but is implemented as a getter.
909
+ get contentNode(): HTMLUListElement {
910
+ return this.tabBarContainer.getElementsByClassName(ToolbarAwareTabBar.Styles.TAB_BAR_CONTENT)[0] as HTMLUListElement;
911
+ }
912
+
913
+ /**
914
+ * Overrides the scrollable host from the parent class.
915
+ */
916
+ protected get scrollbarHost(): HTMLElement {
917
+ return this.tabBarContainer;
918
+ }
919
+
920
+ protected get tabBarContainer(): HTMLElement {
921
+ return this.node.getElementsByClassName(ToolbarAwareTabBar.Styles.TAB_BAR_CONTENT_CONTAINER)[0] as HTMLElement;
922
+ }
923
+ }
924
+
925
+ export namespace ScrollableTabBar {
926
+
927
+ export interface Options {
928
+ minimumTabSize: number;
929
+ defaultTabSize: number;
930
+ }
931
+ export namespace Styles {
932
+
933
+ export const TAB_BAR_CONTENT = 'p-TabBar-content';
934
+ export const TAB_BAR_CONTENT_CONTAINER = 'p-TabBar-content-container';
935
+
936
+ }
937
+ }
938
+
939
+ /**
940
+ * Specialized scrollable tab-bar which comes with toolbar support.
941
+ * Instead of the following DOM structure.
942
+ *
943
+ * +-------------------------+
944
+ * |[TAB_0][TAB_1][TAB_2][TAB|
945
+ * +-------------Scrollable--+
946
+ *
947
+ * There is a dedicated HTML element for toolbar which does **not** contained in the scrollable element.
948
+ *
949
+ * +-------------------------+-----------------+
950
+ * |[TAB_0][TAB_1][TAB_2][TAB| Toolbar |
951
+ * +-------------Scrollable--+-Non-Scrollable-+
952
+ *
953
+ */
954
+ export class ToolbarAwareTabBar extends ScrollableTabBar {
955
+ protected toolbar: TabBarToolbar | undefined;
956
+ protected breadcrumbsContainer: HTMLElement;
957
+ protected readonly breadcrumbsRenderer: BreadcrumbsRenderer;
958
+
959
+ constructor(
960
+ protected readonly tabBarToolbarRegistry: TabBarToolbarRegistry,
961
+ protected readonly tabBarToolbarFactory: () => TabBarToolbar,
962
+ protected readonly breadcrumbsRendererFactory: BreadcrumbsRendererFactory,
963
+ options?: TabBar.IOptions<Widget> & PerfectScrollbar.Options,
964
+ dynamicTabOptions?: ScrollableTabBar.Options
965
+ ) {
966
+ super(options, dynamicTabOptions);
967
+ this.breadcrumbsRenderer = this.breadcrumbsRendererFactory();
968
+ this.addBreadcrumbs();
969
+ this.toolbar = this.tabBarToolbarFactory();
970
+ this.toDispose.push(this.tabBarToolbarRegistry.onDidChange(() => this.update()));
971
+ this.toDispose.push(this.breadcrumbsRenderer);
972
+ this.toDispose.push(this.breadcrumbsRenderer.onDidChangeActiveState(active => {
973
+ this.node.classList.toggle('theia-tabBar-multirow', active);
974
+ if (this.parent) {
975
+ MessageLoop.sendMessage(this.parent, new Message('fit-request'));
976
+ }
977
+ }));
978
+ this.node.classList.toggle('theia-tabBar-multirow', this.breadcrumbsRenderer.active);
979
+ const handler = () => this.updateBreadcrumbs();
980
+ this.currentChanged.connect(handler);
981
+ this.toDispose.push(Disposable.create(() => this.currentChanged.disconnect(handler)));
982
+ }
983
+
984
+ protected async updateBreadcrumbs(): Promise<void> {
985
+ const current = this.currentTitle?.owner;
986
+ const uri = NavigatableWidget.is(current) ? current.getResourceUri() : undefined;
987
+ await this.breadcrumbsRenderer.refresh(uri);
988
+ }
989
+
990
+ protected override onAfterAttach(msg: Message): void {
991
+ if (this.toolbar) {
992
+ if (this.toolbar.isAttached) {
993
+ Widget.detach(this.toolbar);
994
+ }
995
+ Widget.attach(this.toolbar, this.topRow);
996
+ if (this.breadcrumbsContainer) {
997
+ this.node.appendChild(this.breadcrumbsContainer);
998
+ }
999
+ this.updateBreadcrumbs();
1000
+ }
1001
+ super.onAfterAttach(msg);
1002
+ }
1003
+
1004
+ protected override onBeforeDetach(msg: Message): void {
1005
+ if (this.toolbar && this.toolbar.isAttached) {
1006
+ this.toolbar.dispose();
1007
+ }
1008
+ super.onBeforeDetach(msg);
1009
+ }
1010
+
1011
+ protected override onUpdateRequest(msg: Message): void {
1012
+ super.onUpdateRequest(msg);
1013
+ this.updateToolbar();
1014
+ }
1015
+
1016
+ protected updateToolbar(): void {
1017
+ if (!this.toolbar) {
1018
+ return;
1019
+ }
1020
+ const widget = this.currentTitle?.owner ?? undefined;
1021
+ this.toolbar.updateTarget(widget);
1022
+ this.updateTabs();
1023
+ }
1024
+
1025
+ override handleEvent(event: Event): void {
1026
+ if (event instanceof MouseEvent) {
1027
+ if (this.toolbar && this.toolbar.shouldHandleMouseEvent(event) || this.isOver(event, this.openTabsContainer)) {
1028
+ // if the mouse event is over the toolbar part don't handle it.
1029
+ return;
1030
+ }
1031
+ }
1032
+ super.handleEvent(event);
1033
+ }
1034
+
1035
+ protected isOver(event: Event, element: Element): boolean {
1036
+ return element && event.target instanceof Element && element.contains(event.target);
1037
+ }
1038
+
1039
+ /**
1040
+ * Restructures the DOM defined in PhosphorJS.
1041
+ *
1042
+ * By default the tabs (`li`) are contained in the `this.contentNode` (`ul`) which is wrapped in a `div` (`this.node`).
1043
+ * Instead of this structure, we add a container for the `this.contentNode` and for the toolbar.
1044
+ * The scrollbar will only work for the `ul` part but it does not affect the toolbar, so it can be on the right hand-side.
1045
+ */
1046
+ protected addBreadcrumbs(): void {
1047
+ this.breadcrumbsContainer = document.createElement('div');
1048
+ this.breadcrumbsContainer.classList.add('theia-tabBar-breadcrumb-row');
1049
+ this.breadcrumbsContainer.appendChild(this.breadcrumbsRenderer.host);
1050
+ this.node.appendChild(this.breadcrumbsContainer);
1051
+ }
1052
+ }
1053
+
1054
+ /**
1055
+ * A specialized tab bar for side areas.
1056
+ */
1057
+ export class SideTabBar extends ScrollableTabBar {
1058
+
1059
+ protected static readonly DRAG_THRESHOLD = 5;
1060
+
1061
+ /**
1062
+ * Emitted when a tab is added to the tab bar.
1063
+ */
1064
+ readonly tabAdded = new Signal<this, { title: Title<Widget> }>(this);
1065
+ /**
1066
+ * Side panels can be collapsed by clicking on the currently selected tab. This signal is
1067
+ * emitted when the mouse is released on the selected tab without initiating a drag.
1068
+ */
1069
+ readonly collapseRequested = new Signal<this, Title<Widget>>(this);
1070
+
1071
+ /**
1072
+ * Emitted when the set of overflowing/hidden tabs changes.
1073
+ */
1074
+ readonly tabsOverflowChanged = new Signal<this, { titles: Title<Widget>[], startIndex: number }>(this);
1075
+
1076
+ protected mouseData?: {
1077
+ pressX: number,
1078
+ pressY: number,
1079
+ mouseDownTabIndex: number
1080
+ };
1081
+
1082
+ protected tabsOverflowData?: {
1083
+ titles: Title<Widget>[],
1084
+ startIndex: number
1085
+ };
1086
+
1087
+ constructor(options?: TabBar.IOptions<Widget> & PerfectScrollbar.Options) {
1088
+ super(options);
1089
+
1090
+ // Create the hidden content node (see `hiddenContentNode` for explanation)
1091
+ const hiddenContent = document.createElement('ul');
1092
+ hiddenContent.className = HIDDEN_CONTENT_CLASS;
1093
+ this.node.appendChild(hiddenContent);
1094
+ }
1095
+
1096
+ /**
1097
+ * Tab bars of the left and right side panel are arranged vertically by rotating their labels.
1098
+ * Rotation is realized with the CSS `transform` property, which disrupts the browser's ability
1099
+ * to arrange the involved elements automatically. Therefore the elements are arranged explicitly
1100
+ * by the TabBarRenderer using inline `height` and `top` styles. However, the size of labels
1101
+ * must still be computed by the browser, so the rendering is performed in two steps: first the
1102
+ * tab bar is rendered horizontally inside a _hidden content node_, then it is rendered again
1103
+ * vertically inside the proper content node. After the first step, size information is gathered
1104
+ * from all labels so it can be applied during the second step.
1105
+ */
1106
+ get hiddenContentNode(): HTMLUListElement {
1107
+ return this.node.getElementsByClassName(HIDDEN_CONTENT_CLASS)[0] as HTMLUListElement;
1108
+ }
1109
+
1110
+ override insertTab(index: number, value: Title<Widget> | Title.IOptions<Widget>): Title<Widget> {
1111
+ const result = super.insertTab(index, value);
1112
+ this.tabAdded.emit({ title: result });
1113
+ return result;
1114
+ }
1115
+
1116
+ protected override onAfterAttach(msg: Message): void {
1117
+ this.updateTabs();
1118
+ this.node.addEventListener('p-dragenter', this);
1119
+ this.node.addEventListener('p-dragover', this);
1120
+ this.node.addEventListener('p-dragleave', this);
1121
+ document.addEventListener('p-drop', this);
1122
+ }
1123
+
1124
+ protected override onAfterDetach(msg: Message): void {
1125
+ super.onAfterDetach(msg);
1126
+ this.node.removeEventListener('p-dragenter', this);
1127
+ this.node.removeEventListener('p-dragover', this);
1128
+ this.node.removeEventListener('p-dragleave', this);
1129
+ document.removeEventListener('p-drop', this);
1130
+ }
1131
+
1132
+ protected override onUpdateRequest(msg: Message): void {
1133
+ this.updateTabs();
1134
+ }
1135
+
1136
+ protected override onResize(msg: Widget.ResizeMessage): void {
1137
+ // Tabs need to be updated if there are already overflowing tabs or the current tabs don't fit
1138
+ if (this.tabsOverflowData || this.node.clientHeight < this.contentNode.clientHeight) {
1139
+ this.updateTabs();
1140
+ }
1141
+ }
1142
+
1143
+ /**
1144
+ * Reveal the tab with the given index by moving it into the non-overflowing tabBar section
1145
+ * if necessary.
1146
+ */
1147
+ override revealTab(index: number): Promise<void> {
1148
+ if (this.pendingReveal) {
1149
+ // A reveal has already been scheduled
1150
+ return this.pendingReveal;
1151
+ }
1152
+ const result = new Promise<void>(resolve => {
1153
+ // The tab might not have been created yet, so wait until the next frame
1154
+ window.requestAnimationFrame(() => {
1155
+ if (this.tabsOverflowData && index >= this.tabsOverflowData.startIndex) {
1156
+ const title = this.titles[index];
1157
+ this.insertTab(this.tabsOverflowData.startIndex - 1, title);
1158
+ }
1159
+
1160
+ if (this.pendingReveal === result) {
1161
+ this.pendingReveal = undefined;
1162
+ }
1163
+ resolve();
1164
+ });
1165
+ });
1166
+ this.pendingReveal = result;
1167
+ return result;
1168
+ }
1169
+
1170
+ /**
1171
+ * Render the tab bar in the _hidden content node_ (see `hiddenContentNode` for explanation),
1172
+ * then gather size information for labels and render it again in the proper content node.
1173
+ */
1174
+ protected override updateTabs(): void {
1175
+ if (this.isAttached) {
1176
+ // Render into the invisible node
1177
+ this.renderTabs(this.hiddenContentNode);
1178
+ // Await a rendering frame
1179
+ window.requestAnimationFrame(() => {
1180
+ const hiddenContent = this.hiddenContentNode;
1181
+ const n = hiddenContent.children.length;
1182
+ const renderData = new Array<Partial<SideBarRenderData>>(n);
1183
+ for (let i = 0; i < n; i++) {
1184
+ const hiddenTab = hiddenContent.children[i];
1185
+ // Extract tab padding, and margin from the computed style
1186
+ const tabStyle = window.getComputedStyle(hiddenTab);
1187
+ const rd: Partial<SideBarRenderData> = {
1188
+ paddingTop: parseFloat(tabStyle.paddingTop!),
1189
+ paddingBottom: parseFloat(tabStyle.paddingBottom!)
1190
+ };
1191
+ // Extract label size from the DOM
1192
+ const labelElements = hiddenTab.getElementsByClassName('p-TabBar-tabLabel');
1193
+ if (labelElements.length === 1) {
1194
+ const label = labelElements[0];
1195
+ rd.labelSize = { width: label.clientWidth, height: label.clientHeight };
1196
+ }
1197
+ // Extract icon size from the DOM
1198
+ const iconElements = hiddenTab.getElementsByClassName('p-TabBar-tabIcon');
1199
+ if (iconElements.length === 1) {
1200
+ const icon = iconElements[0];
1201
+ rd.iconSize = { width: icon.clientWidth, height: icon.clientHeight };
1202
+ }
1203
+
1204
+ renderData[i] = rd;
1205
+ }
1206
+ // Render into the visible node
1207
+ this.renderTabs(this.contentNode, renderData);
1208
+ this.computeOverflowingTabsData();
1209
+ });
1210
+ }
1211
+ }
1212
+
1213
+ protected computeOverflowingTabsData(): void {
1214
+ // ensure that render tabs has completed
1215
+ window.requestAnimationFrame(() => {
1216
+ const startIndex = this.hideOverflowingTabs();
1217
+ if (startIndex === -1) {
1218
+ if (this.tabsOverflowData) {
1219
+ this.tabsOverflowData = undefined;
1220
+ this.tabsOverflowChanged.emit({ titles: [], startIndex });
1221
+ }
1222
+ return;
1223
+ }
1224
+ const newOverflowingTabs = this.titles.slice(startIndex);
1225
+
1226
+ if (!this.tabsOverflowData) {
1227
+ this.tabsOverflowData = { titles: newOverflowingTabs, startIndex };
1228
+ this.tabsOverflowChanged.emit(this.tabsOverflowData);
1229
+ return;
1230
+ }
1231
+
1232
+ if ((newOverflowingTabs.length !== this.tabsOverflowData?.titles.length ?? 0) ||
1233
+ newOverflowingTabs.find((newTitle, i) => newTitle !== this.tabsOverflowData?.titles[i]) !== undefined) {
1234
+ this.tabsOverflowData = { titles: newOverflowingTabs, startIndex };
1235
+ this.tabsOverflowChanged.emit(this.tabsOverflowData);
1236
+ }
1237
+ });
1238
+ }
1239
+
1240
+ /**
1241
+ * Hide overflowing tabs and return the index of the first hidden tab.
1242
+ */
1243
+ protected hideOverflowingTabs(): number {
1244
+ const availableHeight = this.node.clientHeight;
1245
+ const invisibleClass = 'p-mod-invisible';
1246
+ let startIndex = -1;
1247
+ const n = this.contentNode.children.length;
1248
+ for (let i = 0; i < n; i++) {
1249
+ const tab = this.contentNode.children[i] as HTMLLIElement;
1250
+ if (tab.offsetTop + tab.offsetHeight >= availableHeight) {
1251
+ tab.classList.add(invisibleClass);
1252
+ if (startIndex === -1) {
1253
+ startIndex = i;
1254
+ /* If only one element is overflowing and the additional menu widget is visible (i.e. this.tabsOverflowData is set)
1255
+ * there might already be enough space to show the last tab. In this case, we need to include the size of the
1256
+ * additional menu widget and recheck if the last tab is visible */
1257
+ if (startIndex === n - 1 && this.tabsOverflowData) {
1258
+ const additionalViewsMenu = this.node.parentElement?.querySelector('.theia-additional-views-menu') as HTMLDivElement;
1259
+ if (tab.offsetTop + tab.offsetHeight < availableHeight + additionalViewsMenu.offsetHeight) {
1260
+ tab.classList.remove(invisibleClass);
1261
+ startIndex = -1;
1262
+ }
1263
+ }
1264
+ }
1265
+ } else {
1266
+ tab.classList.remove(invisibleClass);
1267
+ }
1268
+ }
1269
+ return startIndex;
1270
+ }
1271
+
1272
+ /**
1273
+ * Render the tab bar using the given DOM element as host. The optional `renderData` is forwarded
1274
+ * to the TabBarRenderer.
1275
+ */
1276
+ protected renderTabs(host: HTMLElement, renderData?: Partial<SideBarRenderData>[]): void {
1277
+ const titles = this.titles;
1278
+ const n = titles.length;
1279
+ const renderer = this.renderer as TabBarRenderer;
1280
+ const currentTitle = this.currentTitle;
1281
+ const content = new Array<VirtualElement>(n);
1282
+ for (let i = 0; i < n; i++) {
1283
+ const title = titles[i];
1284
+ const current = title === currentTitle;
1285
+ const zIndex = current ? n : n - i - 1;
1286
+ let rd: SideBarRenderData;
1287
+ if (renderData && i < renderData.length) {
1288
+ rd = { title, current, zIndex, ...renderData[i] };
1289
+ } else {
1290
+ rd = { title, current, zIndex };
1291
+ }
1292
+ // Based on how renderTabs() is called, assume renderData will be undefined when invoked for this.hiddenContentNode
1293
+ content[i] = renderer.renderTab(rd, true, renderData === undefined);
1294
+ }
1295
+ VirtualDOM.render(content, host);
1296
+ }
1297
+
1298
+ /**
1299
+ * The following event processing is used to generate `collapseRequested` signals
1300
+ * when the mouse goes up on the currently selected tab without too much movement
1301
+ * between `mousedown` and `mouseup`. The movement threshold is the same that
1302
+ * is used by the superclass to detect a drag event. The `allowDeselect` option
1303
+ * of the TabBar constructor cannot be used here because it is triggered when the
1304
+ * mouse goes down, and thus collides with dragging.
1305
+ */
1306
+ override handleEvent(event: Event): void {
1307
+ switch (event.type) {
1308
+ case 'mousedown':
1309
+ this.onMouseDown(event as MouseEvent);
1310
+ super.handleEvent(event);
1311
+ break;
1312
+ case 'mouseup':
1313
+ super.handleEvent(event);
1314
+ this.onMouseUp(event as MouseEvent);
1315
+ break;
1316
+ case 'mousemove':
1317
+ this.onMouseMove(event as MouseEvent);
1318
+ super.handleEvent(event);
1319
+ break;
1320
+ case 'p-dragenter':
1321
+ this.onDragEnter(event as IDragEvent);
1322
+ break;
1323
+ case 'p-dragover':
1324
+ this.onDragOver(event as IDragEvent);
1325
+ break;
1326
+ case 'p-dragleave': case 'p-drop':
1327
+ this.cancelViewContainerDND();
1328
+ break;
1329
+ default:
1330
+ super.handleEvent(event);
1331
+ }
1332
+ }
1333
+
1334
+ protected onMouseDown(event: MouseEvent): void {
1335
+ // Check for left mouse button and current mouse status
1336
+ if (event.button !== 0 || this.mouseData) {
1337
+ return;
1338
+ }
1339
+
1340
+ // Check whether the mouse went down on the current tab
1341
+ const tabs = this.contentNode.children;
1342
+ const index = ArrayExt.findFirstIndex(tabs, tab => ElementExt.hitTest(tab, event.clientX, event.clientY));
1343
+ if (index < 0 || index !== this.currentIndex) {
1344
+ return;
1345
+ }
1346
+
1347
+ // Check whether the close button was clicked
1348
+ const icon = tabs[index].querySelector(this.renderer.closeIconSelector);
1349
+ if (icon && icon.contains(event.target as HTMLElement)) {
1350
+ return;
1351
+ }
1352
+
1353
+ this.mouseData = {
1354
+ pressX: event.clientX,
1355
+ pressY: event.clientY,
1356
+ mouseDownTabIndex: index
1357
+ };
1358
+ }
1359
+
1360
+ protected onMouseUp(event: MouseEvent): void {
1361
+ // Check for left mouse button and current mouse status
1362
+ if (event.button !== 0 || !this.mouseData) {
1363
+ return;
1364
+ }
1365
+
1366
+ // Check whether the mouse went up on the current tab
1367
+ const mouseDownTabIndex = this.mouseData.mouseDownTabIndex;
1368
+ this.mouseData = undefined;
1369
+ const tabs = this.contentNode.children;
1370
+ const index = ArrayExt.findFirstIndex(tabs, tab => ElementExt.hitTest(tab, event.clientX, event.clientY));
1371
+ if (index < 0 || index !== mouseDownTabIndex) {
1372
+ return;
1373
+ }
1374
+
1375
+ // Collapse the side bar
1376
+ this.collapseRequested.emit(this.titles[index]);
1377
+ }
1378
+
1379
+ protected onMouseMove(event: MouseEvent): void {
1380
+ // Check for left mouse button and current mouse status
1381
+ if (event.button !== 0 || !this.mouseData) {
1382
+ return;
1383
+ }
1384
+
1385
+ const data = this.mouseData;
1386
+ const dx = Math.abs(event.clientX - data.pressX);
1387
+ const dy = Math.abs(event.clientY - data.pressY);
1388
+ const threshold = SideTabBar.DRAG_THRESHOLD;
1389
+ if (dx >= threshold || dy >= threshold) {
1390
+ this.mouseData = undefined;
1391
+ }
1392
+ }
1393
+
1394
+ toCancelViewContainerDND = new DisposableCollection();
1395
+ protected cancelViewContainerDND = () => {
1396
+ this.toCancelViewContainerDND.dispose();
1397
+ };
1398
+
1399
+ /**
1400
+ * Handles `viewContainerPart` drag enter.
1401
+ */
1402
+ protected onDragEnter = (event: IDragEvent) => {
1403
+ this.cancelViewContainerDND();
1404
+ if (event.mimeData.getData('application/vnd.phosphor.view-container-factory')) {
1405
+ event.preventDefault();
1406
+ event.stopPropagation();
1407
+ }
1408
+ };
1409
+
1410
+ /**
1411
+ * Handle `viewContainerPart` drag over,
1412
+ * Defines the appropriate `dropAction` and opens the tab on which the mouse stands on for more than 800 ms.
1413
+ */
1414
+ protected onDragOver = (event: IDragEvent) => {
1415
+ const factory = event.mimeData.getData('application/vnd.phosphor.view-container-factory');
1416
+ const widget = factory && factory();
1417
+ if (!widget) {
1418
+ event.dropAction = 'none';
1419
+ return;
1420
+ }
1421
+ event.preventDefault();
1422
+ event.stopPropagation();
1423
+ if (!this.toCancelViewContainerDND.disposed) {
1424
+ event.dropAction = event.proposedAction;
1425
+ return;
1426
+ }
1427
+
1428
+ const { target, clientX, clientY } = event;
1429
+ if (target instanceof HTMLElement) {
1430
+ if (widget.options.disableDraggingToOtherContainers || widget.viewContainer.disableDNDBetweenContainers) {
1431
+ event.dropAction = 'none';
1432
+ target.classList.add('theia-cursor-no-drop');
1433
+ this.toCancelViewContainerDND.push(Disposable.create(() => {
1434
+ target.classList.remove('theia-cursor-no-drop');
1435
+ }));
1436
+ } else {
1437
+ event.dropAction = event.proposedAction;
1438
+ }
1439
+ const { top, bottom, left, right, height } = target.getBoundingClientRect();
1440
+ const mouseOnTop = (clientY - top) < (height / 2);
1441
+ const dropTargetClass = `drop-target-${mouseOnTop ? 'top' : 'bottom'}`;
1442
+ const tabs = this.contentNode.children;
1443
+ const targetTab = ArrayExt.findFirstValue(tabs, t => ElementExt.hitTest(t, clientX, clientY));
1444
+ if (!targetTab) {
1445
+ return;
1446
+ }
1447
+ targetTab.classList.add(dropTargetClass);
1448
+ this.toCancelViewContainerDND.push(Disposable.create(() => {
1449
+ if (targetTab) {
1450
+ targetTab.classList.remove(dropTargetClass);
1451
+ }
1452
+ }));
1453
+ const openTabTimer = setTimeout(() => {
1454
+ const title = this.titles.find(t => (this.renderer as TabBarRenderer).createTabId(t) === targetTab.id);
1455
+ if (title) {
1456
+ const mouseStillOnTab = clientX >= left && clientX <= right && clientY >= top && clientY <= bottom;
1457
+ if (mouseStillOnTab) {
1458
+ this.currentTitle = title;
1459
+ }
1460
+ }
1461
+ }, 800);
1462
+ this.toCancelViewContainerDND.push(Disposable.create(() => {
1463
+ clearTimeout(openTabTimer);
1464
+ }));
1465
+ }
1466
+ };
1467
+
1468
+ }