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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (581) hide show
  1. package/README.md +6 -6
  2. package/i18n/nls.cs.json +582 -582
  3. package/i18n/nls.de.json +582 -582
  4. package/i18n/nls.es.json +582 -582
  5. package/i18n/nls.fr.json +582 -582
  6. package/i18n/nls.hu.json +582 -582
  7. package/i18n/nls.it.json +582 -582
  8. package/i18n/nls.ja.json +582 -582
  9. package/i18n/nls.json +582 -582
  10. package/i18n/nls.ko.json +582 -582
  11. package/i18n/nls.pl.json +582 -582
  12. package/i18n/nls.pt-br.json +582 -582
  13. package/i18n/nls.ru.json +582 -582
  14. package/i18n/nls.tr.json +582 -582
  15. package/i18n/nls.zh-cn.json +582 -582
  16. package/i18n/nls.zh-tw.json +582 -582
  17. package/lib/browser/catalog.json +54 -6
  18. package/lib/browser/common-frontend-contribution.js +3 -3
  19. package/lib/browser/common-styling-participants.js +166 -166
  20. package/lib/browser/progress-location-service.spec.js +7 -7
  21. package/lib/browser/storage-service.js +3 -3
  22. package/lib/browser/tree/tree.spec.js +75 -75
  23. package/lib/node/process-utils.spec.js +8 -8
  24. package/package.json +4 -4
  25. package/shared/ajv/index.d.ts +2 -2
  26. package/shared/markdown-it.d.ts +2 -2
  27. package/shared/markdown-it.js +1 -1
  28. package/shared/reflect-metadata/index.d.ts +1 -1
  29. package/shared/reflect-metadata/index.js +1 -1
  30. package/shared/vscode-languageserver-types/index.d.ts +1 -1
  31. package/src/browser/about-dialog.tsx +137 -137
  32. package/src/browser/authentication-service.ts +467 -467
  33. package/src/browser/breadcrumbs/breadcrumb-popup-container.ts +101 -101
  34. package/src/browser/breadcrumbs/breadcrumb-renderer.tsx +41 -41
  35. package/src/browser/breadcrumbs/breadcrumbs-constants.ts +79 -79
  36. package/src/browser/breadcrumbs/breadcrumbs-renderer.tsx +185 -185
  37. package/src/browser/breadcrumbs/breadcrumbs-service.ts +108 -108
  38. package/src/browser/breadcrumbs/index.ts +21 -21
  39. package/src/browser/browser-clipboard-service.ts +122 -122
  40. package/src/browser/browser.ts +239 -239
  41. package/src/browser/clipboard-service.ts +23 -23
  42. package/src/browser/color-application-contribution.ts +110 -110
  43. package/src/browser/color-registry.ts +60 -60
  44. package/src/browser/command-open-handler.ts +54 -54
  45. package/src/browser/common-frontend-contribution.ts +2680 -2680
  46. package/src/browser/common-styling-participants.ts +361 -361
  47. package/src/browser/connection-status-service.spec.ts +200 -200
  48. package/src/browser/connection-status-service.ts +216 -216
  49. package/src/browser/context-key-service.ts +142 -142
  50. package/src/browser/context-menu-renderer.ts +124 -124
  51. package/src/browser/core-preferences.ts +343 -343
  52. package/src/browser/credentials-service.ts +106 -106
  53. package/src/browser/decoration-style.ts +65 -65
  54. package/src/browser/decorations-service.ts +209 -209
  55. package/src/browser/dialogs/react-dialog.tsx +56 -56
  56. package/src/browser/dialogs.ts +534 -534
  57. package/src/browser/diff-uris.ts +117 -117
  58. package/src/browser/encoding-registry.ts +97 -97
  59. package/src/browser/endpoint.spec.ts +148 -148
  60. package/src/browser/endpoint.ts +136 -136
  61. package/src/browser/external-uri-service.ts +79 -79
  62. package/src/browser/file-icons-js.d.ts +20 -20
  63. package/src/browser/frontend-application-bindings.ts +62 -62
  64. package/src/browser/frontend-application-config-provider.spec.ts +45 -45
  65. package/src/browser/frontend-application-config-provider.ts +50 -50
  66. package/src/browser/frontend-application-contribution.ts +110 -110
  67. package/src/browser/frontend-application-module.ts +474 -474
  68. package/src/browser/frontend-application-state.ts +74 -74
  69. package/src/browser/frontend-application.ts +326 -326
  70. package/src/browser/hover-service.ts +218 -218
  71. package/src/browser/http-open-handler.ts +49 -49
  72. package/src/browser/i18n/i18n-frontend-module.ts +27 -27
  73. package/src/browser/i18n/language-quick-pick-service.ts +130 -130
  74. package/src/browser/icon-registry.ts +87 -87
  75. package/src/browser/icon-theme-contribution.ts +64 -64
  76. package/src/browser/icon-theme-service.ts +217 -217
  77. package/src/browser/icons/CollapseAll.svg +7 -7
  78. package/src/browser/icons/CollapseAll_inverse.svg +7 -7
  79. package/src/browser/icons/Refresh.svg +7 -7
  80. package/src/browser/icons/Refresh_inverse.svg +7 -7
  81. package/src/browser/icons/add-inverse.svg +4 -4
  82. package/src/browser/icons/add.svg +4 -4
  83. package/src/browser/icons/arrow-down-bright.svg +6 -6
  84. package/src/browser/icons/arrow-down-dark.svg +6 -6
  85. package/src/browser/icons/arrow-up-bright.svg +6 -6
  86. package/src/browser/icons/arrow-up-dark.svg +6 -6
  87. package/src/browser/icons/case-sensitive-dark.svg +16 -16
  88. package/src/browser/icons/case-sensitive.svg +16 -16
  89. package/src/browser/icons/chevron-right-dark.svg +5 -5
  90. package/src/browser/icons/chevron-right-light.svg +6 -6
  91. package/src/browser/icons/circle-bright.svg +7 -7
  92. package/src/browser/icons/circle-dark.svg +7 -7
  93. package/src/browser/icons/clear-search-results-dark.svg +7 -7
  94. package/src/browser/icons/clear-search-results.svg +7 -7
  95. package/src/browser/icons/close-all-bright.svg +7 -7
  96. package/src/browser/icons/close-all-dark.svg +7 -7
  97. package/src/browser/icons/close-bright.svg +7 -7
  98. package/src/browser/icons/close-dark.svg +7 -7
  99. package/src/browser/icons/collapse.svg +4 -4
  100. package/src/browser/icons/edit-json-dark.svg +6 -6
  101. package/src/browser/icons/edit-json.svg +6 -6
  102. package/src/browser/icons/expand.svg +4 -4
  103. package/src/browser/icons/loading-dark.svg +6 -6
  104. package/src/browser/icons/loading-light.svg +6 -6
  105. package/src/browser/icons/open-change-bright.svg +3 -3
  106. package/src/browser/icons/open-change-dark.svg +4 -4
  107. package/src/browser/icons/open-file-bright.svg +4 -4
  108. package/src/browser/icons/open-file-dark.svg +4 -4
  109. package/src/browser/icons/preview-bright.svg +3 -3
  110. package/src/browser/icons/preview-dark.svg +3 -3
  111. package/src/browser/icons/regex-dark.svg +10 -10
  112. package/src/browser/icons/regex.svg +10 -10
  113. package/src/browser/icons/remove-all-inverse.svg +4 -4
  114. package/src/browser/icons/remove-all.svg +4 -4
  115. package/src/browser/icons/replace-all-inverse.svg +13 -13
  116. package/src/browser/icons/replace-all.svg +13 -13
  117. package/src/browser/icons/replace-inverse.svg +15 -15
  118. package/src/browser/icons/replace.svg +15 -15
  119. package/src/browser/icons/whole-word-dark.svg +19 -19
  120. package/src/browser/icons/whole-word.svg +19 -19
  121. package/src/browser/index.ts +50 -50
  122. package/src/browser/json-schema-store.ts +118 -118
  123. package/src/browser/keybinding.spec.ts +554 -554
  124. package/src/browser/keybinding.ts +759 -759
  125. package/src/browser/keyboard/browser-keyboard-frontend-contribution.ts +108 -108
  126. package/src/browser/keyboard/browser-keyboard-layout-provider.spec.ts +171 -171
  127. package/src/browser/keyboard/browser-keyboard-layout-provider.ts +469 -469
  128. package/src/browser/keyboard/browser-keyboard-module.ts +30 -30
  129. package/src/browser/keyboard/index.ts +20 -20
  130. package/src/browser/keyboard/keyboard-layout-service.spec.ts +121 -121
  131. package/src/browser/keyboard/keyboard-layout-service.ts +455 -455
  132. package/src/browser/keyboard/keys.spec.ts +258 -258
  133. package/src/browser/keyboard/keys.ts +20 -20
  134. package/src/browser/keys.ts +21 -21
  135. package/src/browser/label-parser.spec.ts +165 -165
  136. package/src/browser/label-parser.ts +108 -108
  137. package/src/browser/label-provider.spec.ts +62 -62
  138. package/src/browser/label-provider.ts +385 -385
  139. package/src/browser/language-icon-provider.ts +55 -55
  140. package/src/browser/language-service.ts +77 -77
  141. package/src/browser/logger-frontend-module.ts +65 -65
  142. package/src/browser/markdown-rendering/markdown-renderer.ts +98 -98
  143. package/src/browser/menu/browser-context-menu-renderer.ts +48 -48
  144. package/src/browser/menu/browser-menu-module.ts +28 -28
  145. package/src/browser/menu/browser-menu-plugin.ts +491 -491
  146. package/src/browser/menu/context-menu-context.ts +41 -41
  147. package/src/browser/messaging/connection-source.ts +26 -26
  148. package/src/browser/messaging/frontend-id-provider.ts +37 -37
  149. package/src/browser/messaging/index.ts +18 -18
  150. package/src/browser/messaging/messaging-frontend-module.ts +41 -41
  151. package/src/browser/messaging/service-connection-provider.ts +140 -140
  152. package/src/browser/messaging/ws-connection-provider.ts +49 -49
  153. package/src/browser/messaging/ws-connection-source.ts +230 -230
  154. package/src/browser/mime-service.ts +30 -30
  155. package/src/browser/navigatable-types.ts +81 -81
  156. package/src/browser/navigatable.ts +39 -39
  157. package/src/browser/open-with-service.ts +140 -140
  158. package/src/browser/opener-service.spec.ts +49 -49
  159. package/src/browser/opener-service.ts +169 -169
  160. package/src/browser/performance/frontend-stopwatch.ts +65 -65
  161. package/src/browser/performance/index.ts +18 -18
  162. package/src/browser/performance/measurement-frontend-bindings.ts +31 -31
  163. package/src/browser/preferences/index.ts +23 -23
  164. package/src/browser/preferences/injectable-preference-proxy.ts +283 -283
  165. package/src/browser/preferences/preference-configurations.ts +82 -82
  166. package/src/browser/preferences/preference-contribution.ts +436 -436
  167. package/src/browser/preferences/preference-language-override-service.ts +111 -111
  168. package/src/browser/preferences/preference-provider.spec.ts +36 -36
  169. package/src/browser/preferences/preference-provider.ts +277 -277
  170. package/src/browser/preferences/preference-proxy.spec.ts +367 -367
  171. package/src/browser/preferences/preference-proxy.ts +367 -367
  172. package/src/browser/preferences/preference-schema-provider.spec.ts +130 -130
  173. package/src/browser/preferences/preference-scope.ts +18 -18
  174. package/src/browser/preferences/preference-service.spec.ts +613 -613
  175. package/src/browser/preferences/preference-service.ts +594 -594
  176. package/src/browser/preferences/preference-validation-service.spec.ts +334 -334
  177. package/src/browser/preferences/preference-validation-service.ts +358 -358
  178. package/src/browser/preferences/test/index.ts +19 -19
  179. package/src/browser/preferences/test/mock-preference-provider.ts +50 -50
  180. package/src/browser/preferences/test/mock-preference-proxy.ts +48 -48
  181. package/src/browser/preferences/test/mock-preference-service.ts +63 -63
  182. package/src/browser/preload/i18n-preload-contribution.ts +50 -50
  183. package/src/browser/preload/os-preload-contribution.ts +37 -37
  184. package/src/browser/preload/preload-module.ts +45 -45
  185. package/src/browser/preload/preloader.ts +37 -37
  186. package/src/browser/preload/theme-preload-contribution.ts +31 -31
  187. package/src/browser/progress-bar-factory.ts +29 -29
  188. package/src/browser/progress-bar.ts +76 -76
  189. package/src/browser/progress-client.ts +53 -53
  190. package/src/browser/progress-location-service.spec.ts +50 -50
  191. package/src/browser/progress-location-service.ts +96 -96
  192. package/src/browser/progress-status-bar-item.ts +83 -83
  193. package/src/browser/quick-input/index.ts +23 -23
  194. package/src/browser/quick-input/quick-access.ts +75 -75
  195. package/src/browser/quick-input/quick-command-frontend-contribution.ts +89 -89
  196. package/src/browser/quick-input/quick-command-service.ts +246 -246
  197. package/src/browser/quick-input/quick-help-service.ts +87 -87
  198. package/src/browser/quick-input/quick-input-frontend-contribution.ts +33 -33
  199. package/src/browser/quick-input/quick-input-service.spec.ts +176 -176
  200. package/src/browser/quick-input/quick-input-service.ts +17 -17
  201. package/src/browser/quick-input/quick-pick-service-impl.ts +69 -69
  202. package/src/browser/quick-input/quick-view-service.ts +83 -83
  203. package/src/browser/request/browser-request-module.ts +23 -23
  204. package/src/browser/request/browser-request-service.ts +172 -172
  205. package/src/browser/resource-context-key.ts +77 -77
  206. package/src/browser/saveable-service.ts +332 -332
  207. package/src/browser/saveable.ts +395 -395
  208. package/src/browser/secondary-window-handler.ts +211 -211
  209. package/src/browser/shell/additional-views-menu-widget.tsx +71 -71
  210. package/src/browser/shell/application-shell-mouse-tracker.ts +103 -103
  211. package/src/browser/shell/application-shell.ts +2271 -2271
  212. package/src/browser/shell/current-widget-command-adapter.ts +57 -57
  213. package/src/browser/shell/index.ts +23 -23
  214. package/src/browser/shell/shell-layout-restorer.ts +399 -399
  215. package/src/browser/shell/side-panel-handler.ts +794 -794
  216. package/src/browser/shell/side-panel-toolbar.ts +111 -111
  217. package/src/browser/shell/sidebar-bottom-menu-widget.tsx +39 -39
  218. package/src/browser/shell/sidebar-menu-widget.tsx +183 -183
  219. package/src/browser/shell/sidebar-top-menu-widget.tsx +26 -26
  220. package/src/browser/shell/split-panels.ts +191 -191
  221. package/src/browser/shell/tab-bar-decorator.ts +106 -106
  222. package/src/browser/shell/tab-bar-toolbar/index.ts +19 -19
  223. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar-menu-adapters.ts +31 -31
  224. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar-registry.ts +242 -242
  225. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar-types.ts +149 -149
  226. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar.spec.ts +62 -62
  227. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar.tsx +443 -443
  228. package/src/browser/shell/tab-bars.spec.ts +63 -63
  229. package/src/browser/shell/tab-bars.ts +1468 -1468
  230. package/src/browser/shell/theia-dock-panel.ts +265 -265
  231. package/src/browser/shell/view-column-service.ts +125 -125
  232. package/src/browser/shell/view-contribution.ts +178 -178
  233. package/src/browser/source-tree/index.ts +19 -19
  234. package/src/browser/source-tree/source-tree-widget.tsx +107 -107
  235. package/src/browser/source-tree/source-tree.ts +146 -146
  236. package/src/browser/source-tree/tree-source.ts +73 -73
  237. package/src/browser/status-bar/index.ts +29 -29
  238. package/src/browser/status-bar/status-bar-types.ts +97 -97
  239. package/src/browser/status-bar/status-bar-view-model.ts +209 -209
  240. package/src/browser/status-bar/status-bar.tsx +189 -189
  241. package/src/browser/storage-service.spec.ts +76 -76
  242. package/src/browser/storage-service.ts +129 -129
  243. package/src/browser/style/about.css +36 -36
  244. package/src/browser/style/alert-messages.css +62 -62
  245. package/src/browser/style/ansi.css +88 -88
  246. package/src/browser/style/breadcrumbs.css +130 -130
  247. package/src/browser/style/dialog.css +126 -126
  248. package/src/browser/style/dockpanel.css +76 -76
  249. package/src/browser/style/hover-service.css +101 -101
  250. package/src/browser/style/icons.css +61 -61
  251. package/src/browser/style/index.css +353 -353
  252. package/src/browser/style/materialcolors.css +278 -278
  253. package/src/browser/style/menus.css +230 -230
  254. package/src/browser/style/notification.css +39 -39
  255. package/src/browser/style/os.css +87 -87
  256. package/src/browser/style/progress-bar.css +43 -43
  257. package/src/browser/style/quick-title-bar.css +45 -45
  258. package/src/browser/style/scrollbars.css +172 -172
  259. package/src/browser/style/search-box.css +123 -123
  260. package/src/browser/style/select-component.css +107 -107
  261. package/src/browser/style/sidepanel.css +367 -367
  262. package/src/browser/style/split-widget.css +38 -38
  263. package/src/browser/style/status-bar.css +127 -127
  264. package/src/browser/style/tabs.css +647 -647
  265. package/src/browser/style/tooltip.css +28 -28
  266. package/src/browser/style/tree-decorators.css +81 -81
  267. package/src/browser/style/tree.css +232 -232
  268. package/src/browser/style/view-container.css +187 -187
  269. package/src/browser/style/widget.css +19 -19
  270. package/src/browser/styling-service.ts +96 -96
  271. package/src/browser/supported-encodings.ts +262 -262
  272. package/src/browser/test/jsdom.ts +69 -69
  273. package/src/browser/test/mock-connection-status-service.ts +33 -33
  274. package/src/browser/test/mock-env-variables-server.ts +47 -47
  275. package/src/browser/test/mock-opener-service.ts +34 -34
  276. package/src/browser/test/mock-storage-service.ts +49 -49
  277. package/src/browser/theming.ts +206 -206
  278. package/src/browser/tooltip-service.tsx +96 -96
  279. package/src/browser/tree/fuzzy-search.spec.ts +99 -99
  280. package/src/browser/tree/fuzzy-search.ts +136 -136
  281. package/src/browser/tree/index.ts +29 -29
  282. package/src/browser/tree/search-box-debounce.ts +96 -96
  283. package/src/browser/tree/search-box.ts +355 -355
  284. package/src/browser/tree/test/mock-selectable-tree-model.ts +109 -109
  285. package/src/browser/tree/test/mock-tree-model.ts +136 -136
  286. package/src/browser/tree/test/tree-test-container.ts +50 -50
  287. package/src/browser/tree/tree-compression/compressed-tree-expansion-service.ts +46 -46
  288. package/src/browser/tree/tree-compression/compressed-tree-model.ts +88 -88
  289. package/src/browser/tree/tree-compression/compressed-tree-widget.tsx +203 -203
  290. package/src/browser/tree/tree-compression/index.ts +20 -20
  291. package/src/browser/tree/tree-compression/tree-compression-service.ts +125 -125
  292. package/src/browser/tree/tree-compression/tree-compression.css +28 -28
  293. package/src/browser/tree/tree-consistency.spec.ts +105 -105
  294. package/src/browser/tree/tree-container.spec.ts +45 -45
  295. package/src/browser/tree/tree-container.ts +155 -155
  296. package/src/browser/tree/tree-decorator.spec.ts +162 -162
  297. package/src/browser/tree/tree-decorator.ts +238 -238
  298. package/src/browser/tree/tree-expansion.spec.ts +173 -173
  299. package/src/browser/tree/tree-expansion.ts +165 -165
  300. package/src/browser/tree/tree-focus-service.ts +55 -55
  301. package/src/browser/tree/tree-iterator.spec.ts +170 -170
  302. package/src/browser/tree/tree-iterator.ts +256 -256
  303. package/src/browser/tree/tree-label-provider.ts +40 -40
  304. package/src/browser/tree/tree-model.ts +562 -562
  305. package/src/browser/tree/tree-navigation.ts +58 -58
  306. package/src/browser/tree/tree-preference.ts +50 -50
  307. package/src/browser/tree/tree-search.ts +128 -128
  308. package/src/browser/tree/tree-selectable.spec.ts +152 -152
  309. package/src/browser/tree/tree-selection-impl.ts +176 -176
  310. package/src/browser/tree/tree-selection-state.spec.ts +462 -462
  311. package/src/browser/tree/tree-selection-state.ts +245 -245
  312. package/src/browser/tree/tree-selection.ts +159 -159
  313. package/src/browser/tree/tree-view-welcome-widget.tsx +263 -263
  314. package/src/browser/tree/tree-widget-selection.ts +45 -45
  315. package/src/browser/tree/tree-widget.tsx +1591 -1591
  316. package/src/browser/tree/tree.spec.ts +241 -241
  317. package/src/browser/tree/tree.ts +425 -425
  318. package/src/browser/undo-redo-handler.ts +85 -85
  319. package/src/browser/user-working-directory-provider.ts +77 -77
  320. package/src/browser/view-container.ts +1640 -1640
  321. package/src/browser/widget-decoration.ts +358 -358
  322. package/src/browser/widget-manager.spec.ts +102 -102
  323. package/src/browser/widget-manager.ts +318 -318
  324. package/src/browser/widget-open-handler.ts +168 -168
  325. package/src/browser/widgets/alert-message.tsx +56 -56
  326. package/src/browser/widgets/enhanced-preview-widget.ts +27 -27
  327. package/src/browser/widgets/extractable-widget.ts +33 -33
  328. package/src/browser/widgets/index.ts +21 -21
  329. package/src/browser/widgets/previewable-widget.ts +31 -31
  330. package/src/browser/widgets/react-renderer.tsx +53 -53
  331. package/src/browser/widgets/react-widget.tsx +51 -51
  332. package/src/browser/widgets/select-component.tsx +367 -367
  333. package/src/browser/widgets/split-widget.ts +163 -163
  334. package/src/browser/widgets/widget.ts +406 -406
  335. package/src/browser/window/browser-window-module.ts +32 -32
  336. package/src/browser/window/default-secondary-window-service.ts +189 -189
  337. package/src/browser/window/default-window-service.spec.ts +78 -78
  338. package/src/browser/window/default-window-service.ts +171 -171
  339. package/src/browser/window/secondary-window-service.ts +39 -39
  340. package/src/browser/window/test/mock-window-service.ts +29 -29
  341. package/src/browser/window/window-service.ts +78 -78
  342. package/src/browser/window/window-title-service.ts +107 -107
  343. package/src/browser/window/window-title-updater.ts +95 -95
  344. package/src/browser/window-contribution.ts +64 -64
  345. package/src/browser-only/frontend-only-application-module.ts +116 -116
  346. package/src/browser-only/i18n/i18n-frontend-only-module.ts +37 -37
  347. package/src/browser-only/logger-frontend-only-module.ts +63 -63
  348. package/src/browser-only/messaging/frontend-only-service-connection-provider.ts +39 -39
  349. package/src/browser-only/messaging/messaging-frontend-only-module.ts +42 -42
  350. package/src/browser-only/preload/frontend-only-preload-module.ts +49 -49
  351. package/src/common/accessibility.ts +33 -33
  352. package/src/common/application-error.spec.ts +27 -27
  353. package/src/common/application-error.ts +76 -76
  354. package/src/common/application-protocol.ts +42 -42
  355. package/src/common/array-utils.ts +129 -129
  356. package/src/common/buffer.ts +228 -228
  357. package/src/common/cancellation.ts +163 -163
  358. package/src/common/char-code.ts +438 -438
  359. package/src/common/collections.ts +125 -125
  360. package/src/common/color.ts +103 -103
  361. package/src/common/command.spec.ts +208 -208
  362. package/src/common/command.ts +489 -489
  363. package/src/common/contribution-filter/contribution-filter-registry.ts +79 -79
  364. package/src/common/contribution-filter/contribution-filter.ts +64 -64
  365. package/src/common/contribution-filter/filter.ts +23 -23
  366. package/src/common/contribution-filter/index.ts +19 -19
  367. package/src/common/contribution-provider.ts +96 -96
  368. package/src/common/disposable.spec.ts +94 -94
  369. package/src/common/disposable.ts +188 -188
  370. package/src/common/encoding-service.ts +380 -380
  371. package/src/common/encodings.ts +24 -24
  372. package/src/common/env-variables/env-variables-protocol.ts +38 -38
  373. package/src/common/env-variables/index.ts +17 -17
  374. package/src/common/event.spec.ts +32 -32
  375. package/src/common/event.ts +493 -493
  376. package/src/common/file-uri.ts +61 -61
  377. package/src/common/frontend-application-state.ts +38 -38
  378. package/src/common/glob.ts +741 -741
  379. package/src/common/hash.ts +85 -85
  380. package/src/common/i18n/localization-server.ts +25 -25
  381. package/src/common/i18n/localization.ts +80 -80
  382. package/src/common/i18n/nls.metadata.json +34112 -34112
  383. package/src/common/index.ts +51 -51
  384. package/src/common/json-schema.ts +108 -108
  385. package/src/common/key-store.ts +26 -26
  386. package/src/common/keybinding.ts +152 -152
  387. package/src/common/keyboard/keyboard-layout-provider.ts +51 -51
  388. package/src/common/keys.ts +694 -694
  389. package/src/common/label-protocol.ts +35 -35
  390. package/src/common/logger-protocol.ts +119 -119
  391. package/src/common/logger-watcher.ts +48 -48
  392. package/src/common/logger.spec.ts +46 -46
  393. package/src/common/logger.ts +389 -389
  394. package/src/common/lsp-types.ts +34 -34
  395. package/src/common/markdown-rendering/icon-utilities.ts +30 -30
  396. package/src/common/markdown-rendering/index.ts +18 -18
  397. package/src/common/markdown-rendering/markdown-string.ts +152 -152
  398. package/src/common/menu/action-menu-node.ts +65 -65
  399. package/src/common/menu/composite-menu-node.spec.ts +67 -67
  400. package/src/common/menu/composite-menu-node.ts +114 -114
  401. package/src/common/menu/index.ts +21 -21
  402. package/src/common/menu/menu-adapter.ts +103 -103
  403. package/src/common/menu/menu-model-registry.ts +374 -374
  404. package/src/common/menu/menu-types.ts +220 -220
  405. package/src/common/menu/menu.spec.ts +101 -101
  406. package/src/common/message-rpc/channel.spec.ts +88 -88
  407. package/src/common/message-rpc/channel.ts +300 -300
  408. package/src/common/message-rpc/index.ts +22 -22
  409. package/src/common/message-rpc/message-buffer.ts +105 -105
  410. package/src/common/message-rpc/msg-pack-extension-manager.ts +70 -70
  411. package/src/common/message-rpc/rpc-message-encoder.spec.ts +65 -65
  412. package/src/common/message-rpc/rpc-message-encoder.ts +190 -190
  413. package/src/common/message-rpc/rpc-protocol.ts +255 -255
  414. package/src/common/message-rpc/uint8-array-message-buffer.spec.ts +41 -41
  415. package/src/common/message-rpc/uint8-array-message-buffer.ts +213 -213
  416. package/src/common/message-service-protocol.ts +148 -148
  417. package/src/common/message-service.ts +226 -226
  418. package/src/common/messaging/connection-error-handler.ts +73 -73
  419. package/src/common/messaging/connection-management.ts +43 -43
  420. package/src/common/messaging/handler.ts +26 -26
  421. package/src/common/messaging/index.ts +19 -19
  422. package/src/common/messaging/proxy-factory.spec.ts +108 -108
  423. package/src/common/messaging/proxy-factory.ts +336 -336
  424. package/src/common/messaging/socket-write-buffer.ts +52 -52
  425. package/src/common/messaging/web-socket-channel.ts +76 -76
  426. package/src/common/nls.ts +151 -151
  427. package/src/common/numbers.ts +21 -21
  428. package/src/common/objects.spec.ts +112 -112
  429. package/src/common/objects.ts +123 -123
  430. package/src/common/os.ts +82 -82
  431. package/src/common/path.spec.ts +415 -415
  432. package/src/common/path.ts +334 -334
  433. package/src/common/paths.ts +250 -250
  434. package/src/common/performance/index.ts +19 -19
  435. package/src/common/performance/measurement-protocol.ts +104 -104
  436. package/src/common/performance/measurement.ts +130 -130
  437. package/src/common/performance/stopwatch.ts +183 -183
  438. package/src/common/preferences/preference-schema.ts +101 -101
  439. package/src/common/preferences/preference-scope.spec.ts +48 -48
  440. package/src/common/preferences/preference-scope.ts +68 -68
  441. package/src/common/prioritizeable.ts +58 -58
  442. package/src/common/progress-service-protocol.ts +35 -35
  443. package/src/common/progress-service.ts +82 -82
  444. package/src/common/promise-util.spec.ts +102 -102
  445. package/src/common/promise-util.ts +143 -143
  446. package/src/common/quick-pick-service.ts +353 -353
  447. package/src/common/reference.spec.ts +145 -145
  448. package/src/common/reference.ts +230 -230
  449. package/src/common/resource.ts +430 -430
  450. package/src/common/selection-command-handler.ts +101 -101
  451. package/src/common/selection-service.spec.ts +43 -43
  452. package/src/common/selection-service.ts +49 -49
  453. package/src/common/selection.ts +50 -50
  454. package/src/common/severity.ts +111 -111
  455. package/src/common/stream.ts +718 -718
  456. package/src/common/strings.ts +231 -231
  457. package/src/common/telemetry.ts +45 -45
  458. package/src/common/ternary-search-tree.ts +417 -417
  459. package/src/common/test/expect.ts +34 -34
  460. package/src/common/test/mock-logger.ts +118 -118
  461. package/src/common/test/mock-menu.ts +35 -35
  462. package/src/common/test/mock-resource-provider.ts +33 -33
  463. package/src/common/theme.ts +68 -68
  464. package/src/common/types.spec.ts +86 -86
  465. package/src/common/types.ts +140 -140
  466. package/src/common/uri-command-handler.spec.ts +90 -90
  467. package/src/common/uri-command-handler.ts +148 -148
  468. package/src/common/uri.spec.ts +278 -278
  469. package/src/common/uri.ts +279 -279
  470. package/src/common/uuid.ts +45 -45
  471. package/src/common/version.ts +17 -17
  472. package/src/common/view-column.ts +33 -33
  473. package/src/common/window.ts +34 -34
  474. package/src/electron-browser/electron-clipboard-service.ts +32 -32
  475. package/src/electron-browser/electron-uri-handler.ts +42 -42
  476. package/src/electron-browser/keyboard/electron-keyboard-layout-change-notifier.ts +39 -39
  477. package/src/electron-browser/keyboard/electron-keyboard-module.ts +28 -28
  478. package/src/electron-browser/menu/electron-context-menu-renderer.ts +122 -122
  479. package/src/electron-browser/menu/electron-main-menu-factory.ts +339 -339
  480. package/src/electron-browser/menu/electron-menu-contribution.ts +506 -506
  481. package/src/electron-browser/menu/electron-menu-module.ts +40 -40
  482. package/src/electron-browser/menu/electron-menu-style.css +110 -110
  483. package/src/electron-browser/messaging/electron-frontend-id-provider.ts +25 -25
  484. package/src/electron-browser/messaging/electron-ipc-connection-source.ts +65 -65
  485. package/src/electron-browser/messaging/electron-local-ws-connection-source.ts +45 -45
  486. package/src/electron-browser/messaging/electron-messaging-frontend-module.ts +78 -78
  487. package/src/electron-browser/messaging/electron-ws-connection-source.ts +38 -38
  488. package/src/electron-browser/preload.ts +264 -264
  489. package/src/electron-browser/request/electron-browser-request-module.ts +26 -26
  490. package/src/electron-browser/token/electron-token-frontend-module.ts +22 -22
  491. package/src/electron-browser/window/electron-frontend-application-state.ts +26 -26
  492. package/src/electron-browser/window/electron-secondary-window-service.ts +35 -35
  493. package/src/electron-browser/window/electron-window-module.ts +48 -48
  494. package/src/electron-browser/window/electron-window-preferences.ts +76 -76
  495. package/src/electron-browser/window/electron-window-service.ts +109 -109
  496. package/src/electron-browser/window/external-app-open-handler.ts +42 -42
  497. package/src/electron-common/electron-api.ts +157 -157
  498. package/src/electron-common/electron-main-window-service.ts +24 -24
  499. package/src/electron-common/electron-token.ts +27 -27
  500. package/src/electron-main/electron-api-main.ts +373 -373
  501. package/src/electron-main/electron-main-application-module.ts +65 -65
  502. package/src/electron-main/electron-main-application.ts +860 -860
  503. package/src/electron-main/electron-main-constants.ts +23 -23
  504. package/src/electron-main/electron-main-window-service-impl.ts +44 -44
  505. package/src/electron-main/electron-security-token-service.ts +36 -36
  506. package/src/electron-main/event-utils.ts +36 -36
  507. package/src/electron-main/messaging/electron-connection-handler.ts +21 -21
  508. package/src/electron-main/messaging/electron-messaging-contribution.ts +143 -143
  509. package/src/electron-main/messaging/electron-messaging-service.ts +35 -35
  510. package/src/electron-main/theia-electron-window.ts +219 -219
  511. package/src/electron-node/cli/electron-backend-cli-module.ts +24 -24
  512. package/src/electron-node/cli/electron-cli-contribution.ts +35 -35
  513. package/src/electron-node/hosting/electron-backend-hosting-module.ts +24 -24
  514. package/src/electron-node/hosting/electron-ws-origin-validator.ts +37 -37
  515. package/src/electron-node/keyboard/electron-backend-keyboard-module.ts +30 -30
  516. package/src/electron-node/keyboard/electron-keyboard-layout-provider.ts +35 -35
  517. package/src/electron-node/request/electron-backend-request-module.ts +23 -23
  518. package/src/electron-node/request/electron-backend-request-service.ts +78 -78
  519. package/src/electron-node/token/electron-token-backend-contribution.ts +48 -48
  520. package/src/electron-node/token/electron-token-backend-module.ts +28 -28
  521. package/src/electron-node/token/electron-token-validator.ts +93 -93
  522. package/src/node/application-server.ts +59 -59
  523. package/src/node/backend-application-config-provider.spec.ts +29 -29
  524. package/src/node/backend-application-config-provider.ts +48 -48
  525. package/src/node/backend-application-module.ts +139 -139
  526. package/src/node/backend-application.ts +374 -374
  527. package/src/node/cli.spec.ts +94 -94
  528. package/src/node/cli.ts +63 -63
  529. package/src/node/console-logger-server.spec.ts +59 -59
  530. package/src/node/console-logger-server.ts +76 -76
  531. package/src/node/debug.ts +30 -30
  532. package/src/node/dynamic-require.ts +56 -56
  533. package/src/node/env-variables/env-variables-server.ts +123 -123
  534. package/src/node/env-variables/index.ts +17 -17
  535. package/src/node/environment-utils.spec.ts +92 -92
  536. package/src/node/environment-utils.ts +66 -66
  537. package/src/node/file-uri.spec.ts +76 -76
  538. package/src/node/filesystem-locking.ts +77 -77
  539. package/src/node/hosting/backend-application-hosts.ts +60 -60
  540. package/src/node/hosting/backend-hosting-module.ts +26 -26
  541. package/src/node/hosting/ws-origin-validator.ts +36 -36
  542. package/src/node/i18n/i18n-backend-module.ts +42 -42
  543. package/src/node/i18n/localization-contribution.ts +112 -112
  544. package/src/node/i18n/localization-provider.ts +125 -125
  545. package/src/node/i18n/localization-server.ts +52 -52
  546. package/src/node/i18n/theia-localization-contribution.ts +40 -40
  547. package/src/node/index.ts +22 -22
  548. package/src/node/key-store-server.ts +162 -162
  549. package/src/node/logger-backend-module.ts +88 -88
  550. package/src/node/logger-cli-contribution.spec.ts +245 -245
  551. package/src/node/logger-cli-contribution.ts +168 -168
  552. package/src/node/main.ts +33 -33
  553. package/src/node/messaging/binary-message-pipe.ts +168 -168
  554. package/src/node/messaging/connection-container-module.ts +96 -96
  555. package/src/node/messaging/default-messaging-service.ts +129 -129
  556. package/src/node/messaging/frontend-connection-service.ts +24 -24
  557. package/src/node/messaging/index.ts +19 -19
  558. package/src/node/messaging/ipc-bootstrap.ts +27 -27
  559. package/src/node/messaging/ipc-channel.ts +77 -77
  560. package/src/node/messaging/ipc-connection-provider.ts +107 -107
  561. package/src/node/messaging/ipc-protocol.ts +76 -76
  562. package/src/node/messaging/messaging-backend-module.ts +52 -52
  563. package/src/node/messaging/messaging-listeners.ts +52 -52
  564. package/src/node/messaging/messaging-service.ts +46 -46
  565. package/src/node/messaging/test/test-web-socket-channel.ts +61 -61
  566. package/src/node/messaging/websocket-endpoint.ts +79 -79
  567. package/src/node/messaging/websocket-frontend-connection-service.ts +186 -186
  568. package/src/node/os-backend-provider.ts +25 -25
  569. package/src/node/performance/index.ts +18 -18
  570. package/src/node/performance/measurement-backend-bindings.ts +35 -35
  571. package/src/node/performance/node-stopwatch.ts +40 -40
  572. package/src/node/process-utils.spec.ts +48 -48
  573. package/src/node/process-utils.ts +102 -102
  574. package/src/node/remote/backend-remote-service.ts +25 -25
  575. package/src/node/remote/remote-cli-contribution.ts +34 -34
  576. package/src/node/remote/remote-copy-contribution.ts +45 -45
  577. package/src/node/request/backend-request-facade.ts +39 -39
  578. package/src/node/request/backend-request-module.ts +25 -25
  579. package/src/node/request/proxy-cli-contribution.ts +65 -65
  580. package/src/node/ws-request-validators.ts +56 -56
  581. package/src/typings/native-keymap.d.ts +108 -108
@@ -1,759 +1,759 @@
1
- // *****************************************************************************
2
- // Copyright (C) 2017 TypeFox and others.
3
- //
4
- // This program and the accompanying materials are made available under the
5
- // terms of the Eclipse Public License v. 2.0 which is available at
6
- // http://www.eclipse.org/legal/epl-2.0.
7
- //
8
- // This Source Code may also be made available under the following Secondary
9
- // Licenses when the conditions for such availability set forth in the Eclipse
10
- // Public License v. 2.0 are satisfied: GNU General Public License, version 2
11
- // with the GNU Classpath Exception which is available at
12
- // https://www.gnu.org/software/classpath/license.html.
13
- //
14
- // SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
15
- // *****************************************************************************
16
-
17
- import { injectable, inject, named } from 'inversify';
18
- import { isOSX } from '../common/os';
19
- import { Emitter, Event } from '../common/event';
20
- import { CommandRegistry, Command } from '../common/command';
21
- import { Disposable, DisposableCollection } from '../common/disposable';
22
- import { KeyCode, KeySequence, Key } from './keyboard/keys';
23
- import { KeyboardLayoutService } from './keyboard/keyboard-layout-service';
24
- import { ContributionProvider } from '../common/contribution-provider';
25
- import { ILogger } from '../common/logger';
26
- import { StatusBarAlignment, StatusBar } from './status-bar/status-bar';
27
- import { ContextKeyService } from './context-key-service';
28
- import { CorePreferences } from './core-preferences';
29
- import * as common from '../common/keybinding';
30
- import { nls } from '../common/nls';
31
-
32
- export enum KeybindingScope {
33
- DEFAULT,
34
- USER,
35
- WORKSPACE,
36
- END
37
- }
38
- export namespace KeybindingScope {
39
- export const length = KeybindingScope.END - KeybindingScope.DEFAULT;
40
- }
41
-
42
- /**
43
- * @deprecated import from `@theia/core/lib/common/keybinding` instead
44
- */
45
- export type Keybinding = common.Keybinding;
46
- export const Keybinding = common.Keybinding;
47
-
48
- export interface ResolvedKeybinding extends common.Keybinding {
49
- /**
50
- * The KeyboardLayoutService may transform the `keybinding` depending on the
51
- * user's keyboard layout. This property holds the transformed keybinding that
52
- * should be used in the UI. The value is undefined if the KeyboardLayoutService
53
- * has not been called yet to resolve the keybinding.
54
- */
55
- resolved?: KeyCode[];
56
- }
57
-
58
- export interface ScopedKeybinding extends common.Keybinding {
59
- /** Current keybinding scope */
60
- scope: KeybindingScope;
61
- }
62
-
63
- export const KeybindingContribution = Symbol('KeybindingContribution');
64
- /**
65
- * Allows extensions to contribute {@link common.Keybinding}s
66
- */
67
- export interface KeybindingContribution {
68
- /**
69
- * Registers keybindings.
70
- * @param keybindings the keybinding registry.
71
- */
72
- registerKeybindings(keybindings: KeybindingRegistry): void;
73
- }
74
-
75
- export const KeybindingContext = Symbol('KeybindingContext');
76
- export interface KeybindingContext {
77
- /**
78
- * The unique ID of the current context.
79
- */
80
- readonly id: string;
81
-
82
- isEnabled(arg: common.Keybinding): boolean;
83
- }
84
- export namespace KeybindingContexts {
85
-
86
- export const NOOP_CONTEXT: KeybindingContext = {
87
- id: 'noop.keybinding.context',
88
- isEnabled: () => true
89
- };
90
-
91
- export const DEFAULT_CONTEXT: KeybindingContext = {
92
- id: 'default.keybinding.context',
93
- isEnabled: () => false
94
- };
95
- }
96
-
97
- @injectable()
98
- export class KeybindingRegistry {
99
-
100
- static readonly PASSTHROUGH_PSEUDO_COMMAND = 'passthrough';
101
- protected keySequence: KeySequence = [];
102
-
103
- protected readonly contexts: { [id: string]: KeybindingContext } = {};
104
- protected readonly keymaps: ScopedKeybinding[][] = [...Array(KeybindingScope.length)].map(() => []);
105
-
106
- @inject(CorePreferences)
107
- protected readonly corePreferences: CorePreferences;
108
-
109
- @inject(KeyboardLayoutService)
110
- protected readonly keyboardLayoutService: KeyboardLayoutService;
111
-
112
- @inject(ContributionProvider) @named(KeybindingContext)
113
- protected readonly contextProvider: ContributionProvider<KeybindingContext>;
114
-
115
- @inject(CommandRegistry)
116
- protected readonly commandRegistry: CommandRegistry;
117
-
118
- @inject(ContributionProvider) @named(KeybindingContribution)
119
- protected readonly contributions: ContributionProvider<KeybindingContribution>;
120
-
121
- @inject(StatusBar)
122
- protected readonly statusBar: StatusBar;
123
-
124
- @inject(ILogger)
125
- protected readonly logger: ILogger;
126
-
127
- @inject(ContextKeyService)
128
- protected readonly whenContextService: ContextKeyService;
129
-
130
- async onStart(): Promise<void> {
131
- await this.keyboardLayoutService.initialize();
132
- this.keyboardLayoutService.onKeyboardLayoutChanged(newLayout => {
133
- this.clearResolvedKeybindings();
134
- this.keybindingsChanged.fire(undefined);
135
- });
136
- this.registerContext(KeybindingContexts.NOOP_CONTEXT);
137
- this.registerContext(KeybindingContexts.DEFAULT_CONTEXT);
138
- this.registerContext(...this.contextProvider.getContributions());
139
- for (const contribution of this.contributions.getContributions()) {
140
- contribution.registerKeybindings(this);
141
- }
142
- }
143
-
144
- protected keybindingsChanged = new Emitter<void>();
145
-
146
- /**
147
- * Event that is fired when the resolved keybindings change due to a different keyboard layout
148
- * or when a new keymap is being set
149
- */
150
- get onKeybindingsChanged(): Event<void> {
151
- return this.keybindingsChanged.event;
152
- }
153
-
154
- /**
155
- * Registers the keybinding context arguments into the application. Fails when an already registered
156
- * context is being registered.
157
- *
158
- * @param contexts the keybinding contexts to register into the application.
159
- */
160
- protected registerContext(...contexts: KeybindingContext[]): void {
161
- for (const context of contexts) {
162
- const { id } = context;
163
- if (this.contexts[id]) {
164
- this.logger.error(`A keybinding context with ID ${id} is already registered.`);
165
- } else {
166
- this.contexts[id] = context;
167
- }
168
- }
169
- }
170
-
171
- /**
172
- * Register a default keybinding to the registry.
173
- *
174
- * Keybindings registered later have higher priority during evaluation.
175
- *
176
- * @param binding the keybinding to be registered
177
- */
178
- registerKeybinding(binding: common.Keybinding): Disposable {
179
- return this.doRegisterKeybinding(binding);
180
- }
181
-
182
- /**
183
- * Register multiple default keybindings to the registry
184
- *
185
- * @param bindings An array of keybinding to be registered
186
- */
187
- registerKeybindings(...bindings: common.Keybinding[]): Disposable {
188
- return this.doRegisterKeybindings(bindings, KeybindingScope.DEFAULT);
189
- }
190
-
191
- /**
192
- * Unregister all keybindings from the registry that are bound to the key of the given keybinding
193
- *
194
- * @param binding a keybinding specifying the key to be unregistered
195
- */
196
- unregisterKeybinding(binding: common.Keybinding): void;
197
- /**
198
- * Unregister all keybindings with the given key from the registry
199
- *
200
- * @param key a key to be unregistered
201
- */
202
- unregisterKeybinding(key: string): void;
203
- /**
204
- * Unregister all existing keybindings for the given command
205
- * @param command the command to unregister all keybindings for
206
- */
207
- unregisterKeybinding(command: Command): void;
208
-
209
- unregisterKeybinding(arg: common.Keybinding | string | Command): void {
210
- const keymap = this.keymaps[KeybindingScope.DEFAULT];
211
- const filter = Command.is(arg)
212
- ? ({ command }: common.Keybinding) => command === arg.id
213
- : ({ keybinding }: common.Keybinding) => Keybinding.is(arg)
214
- ? keybinding === arg.keybinding
215
- : keybinding === arg;
216
- for (const binding of keymap.filter(filter)) {
217
- const idx = keymap.indexOf(binding);
218
- if (idx !== -1) {
219
- keymap.splice(idx, 1);
220
- }
221
- }
222
- }
223
-
224
- protected doRegisterKeybindings(bindings: common.Keybinding[], scope: KeybindingScope = KeybindingScope.DEFAULT): Disposable {
225
- const toDispose = new DisposableCollection();
226
- for (const binding of bindings) {
227
- toDispose.push(this.doRegisterKeybinding(binding, scope));
228
- }
229
- return toDispose;
230
- }
231
-
232
- protected doRegisterKeybinding(binding: common.Keybinding, scope: KeybindingScope = KeybindingScope.DEFAULT): Disposable {
233
- try {
234
- this.resolveKeybinding(binding);
235
- const scoped = Object.assign(binding, { scope });
236
- this.insertBindingIntoScope(scoped, scope);
237
- return Disposable.create(() => {
238
- const index = this.keymaps[scope].indexOf(scoped);
239
- if (index !== -1) {
240
- this.keymaps[scope].splice(index, 1);
241
- }
242
- });
243
- } catch (error) {
244
- this.logger.warn(`Could not register keybinding:\n ${common.Keybinding.stringify(binding)}\n${error}`);
245
- return Disposable.NULL;
246
- }
247
- }
248
-
249
- /**
250
- * Ensures that keybindings are inserted in order of increasing length of binding to ensure that if a
251
- * user triggers a short keybinding (e.g. ctrl+k), the UI won't wait for a longer one (e.g. ctrl+k enter)
252
- */
253
- protected insertBindingIntoScope(item: common.Keybinding & { scope: KeybindingScope; }, scope: KeybindingScope): void {
254
- const scopedKeymap = this.keymaps[scope];
255
- const getNumberOfKeystrokes = (binding: common.Keybinding): number => (binding.keybinding.trim().match(/\s/g)?.length ?? 0) + 1;
256
- const numberOfKeystrokesInBinding = getNumberOfKeystrokes(item);
257
- const indexOfFirstItemWithEqualStrokes = scopedKeymap.findIndex(existingBinding => getNumberOfKeystrokes(existingBinding) === numberOfKeystrokesInBinding);
258
- if (indexOfFirstItemWithEqualStrokes > -1) {
259
- scopedKeymap.splice(indexOfFirstItemWithEqualStrokes, 0, item);
260
- } else {
261
- scopedKeymap.push(item);
262
- }
263
- }
264
-
265
- /**
266
- * Ensure that the `resolved` property of the given binding is set by calling the KeyboardLayoutService.
267
- */
268
- resolveKeybinding(binding: ResolvedKeybinding): KeyCode[] {
269
- if (!binding.resolved) {
270
- const sequence = KeySequence.parse(binding.keybinding);
271
- binding.resolved = sequence.map(code => this.keyboardLayoutService.resolveKeyCode(code));
272
- }
273
- return binding.resolved;
274
- }
275
-
276
- /**
277
- * Clear all `resolved` properties of registered keybindings so the KeyboardLayoutService is called
278
- * again to resolve them. This is necessary when the user's keyboard layout has changed.
279
- */
280
- protected clearResolvedKeybindings(): void {
281
- for (let i = KeybindingScope.DEFAULT; i < KeybindingScope.END; i++) {
282
- const bindings = this.keymaps[i];
283
- for (let j = 0; j < bindings.length; j++) {
284
- const binding = bindings[j] as ResolvedKeybinding;
285
- binding.resolved = undefined;
286
- }
287
- }
288
- }
289
-
290
- /**
291
- * Checks whether a colliding {@link common.Keybinding} exists in a specific scope.
292
- * @param binding the keybinding to check
293
- * @param scope the keybinding scope to check
294
- * @returns true if there is a colliding keybinding
295
- */
296
- containsKeybindingInScope(binding: common.Keybinding, scope = KeybindingScope.USER): boolean {
297
- const bindingKeySequence = this.resolveKeybinding(binding);
298
- const collisions = this.getKeySequenceCollisions(this.getUsableBindings(this.keymaps[scope]), bindingKeySequence)
299
- .filter(b => b.context === binding.context && !b.when && !binding.when);
300
- if (collisions.full.length > 0) {
301
- return true;
302
- }
303
- if (collisions.partial.length > 0) {
304
- return true;
305
- }
306
- if (collisions.shadow.length > 0) {
307
- return true;
308
- }
309
- return false;
310
- }
311
-
312
- /**
313
- * Get a user visible representation of a {@link common.Keybinding}.
314
- * @returns an array of strings representing all elements of the {@link KeySequence} defined by the {@link common.Keybinding}
315
- * @param keybinding the keybinding
316
- * @param separator the separator to be used to stringify {@link KeyCode}s that are part of the {@link KeySequence}
317
- */
318
- acceleratorFor(keybinding: common.Keybinding, separator: string = ' ', asciiOnly = false): string[] {
319
- const bindingKeySequence = this.resolveKeybinding(keybinding);
320
- return this.acceleratorForSequence(bindingKeySequence, separator, asciiOnly);
321
- }
322
-
323
- /**
324
- * Get a user visible representation of a {@link KeySequence}.
325
- * @returns an array of strings representing all elements of the {@link KeySequence}
326
- * @param keySequence the keysequence
327
- * @param separator the separator to be used to stringify {@link KeyCode}s that are part of the {@link KeySequence}
328
- */
329
- acceleratorForSequence(keySequence: KeySequence, separator: string = ' ', asciiOnly = false): string[] {
330
- return keySequence.map(keyCode => this.acceleratorForKeyCode(keyCode, separator, asciiOnly));
331
- }
332
-
333
- /**
334
- * Get a user visible representation of a key code (a key with modifiers).
335
- * @returns a string representing the {@link KeyCode}
336
- * @param keyCode the keycode
337
- * @param separator the separator used to separate keys (key and modifiers) in the returning string
338
- * @param asciiOnly if `true`, no special characters will be substituted into the string returned. Ensures correct keyboard shortcuts in Electron menus.
339
- */
340
- acceleratorForKeyCode(keyCode: KeyCode, separator: string = ' ', asciiOnly = false): string {
341
- return this.componentsForKeyCode(keyCode, asciiOnly).join(separator);
342
- }
343
-
344
- componentsForKeyCode(keyCode: KeyCode, asciiOnly = false): string[] {
345
- const keyCodeResult = [];
346
- const useSymbols = isOSX && !asciiOnly;
347
- if (keyCode.meta && isOSX) {
348
- keyCodeResult.push(useSymbols ? '⌘' : 'Cmd');
349
- }
350
- if (keyCode.ctrl) {
351
- keyCodeResult.push(useSymbols ? '⌃' : 'Ctrl');
352
- }
353
- if (keyCode.alt) {
354
- keyCodeResult.push(useSymbols ? '⌥' : 'Alt');
355
- }
356
- if (keyCode.shift) {
357
- keyCodeResult.push(useSymbols ? '⇧' : 'Shift');
358
- }
359
- if (keyCode.key) {
360
- keyCodeResult.push(this.acceleratorForKey(keyCode.key, asciiOnly));
361
- }
362
- return keyCodeResult;
363
- }
364
-
365
- /**
366
- * @param asciiOnly if `true`, no special characters will be substituted into the string returned. Ensures correct keyboard shortcuts in Electron menus.
367
- *
368
- * Return a user visible representation of a single key.
369
- */
370
- acceleratorForKey(key: Key, asciiOnly = false): string {
371
- if (isOSX && !asciiOnly) {
372
- if (key === Key.ARROW_LEFT) {
373
- return '←';
374
- }
375
- if (key === Key.ARROW_RIGHT) {
376
- return '→';
377
- }
378
- if (key === Key.ARROW_UP) {
379
- return '↑';
380
- }
381
- if (key === Key.ARROW_DOWN) {
382
- return '↓';
383
- }
384
- }
385
- const keyString = this.keyboardLayoutService.getKeyboardCharacter(key);
386
- if (key.keyCode >= Key.KEY_A.keyCode && key.keyCode <= Key.KEY_Z.keyCode ||
387
- key.keyCode >= Key.F1.keyCode && key.keyCode <= Key.F24.keyCode) {
388
- return keyString.toUpperCase();
389
- } else if (keyString.length > 1) {
390
- return keyString.charAt(0).toUpperCase() + keyString.slice(1);
391
- } else {
392
- return keyString;
393
- }
394
- }
395
-
396
- /**
397
- * Finds collisions for a key sequence inside a list of bindings (error-free)
398
- *
399
- * @param bindings the reference bindings
400
- * @param candidate the sequence to match
401
- */
402
- protected getKeySequenceCollisions(bindings: ScopedKeybinding[], candidate: KeySequence): KeybindingRegistry.KeybindingsResult {
403
- const result = new KeybindingRegistry.KeybindingsResult();
404
- for (const binding of bindings) {
405
- try {
406
- const bindingKeySequence = this.resolveKeybinding(binding);
407
- const compareResult = KeySequence.compare(candidate, bindingKeySequence);
408
- switch (compareResult) {
409
- case KeySequence.CompareResult.FULL: {
410
- result.full.push(binding);
411
- break;
412
- }
413
- case KeySequence.CompareResult.PARTIAL: {
414
- result.partial.push(binding);
415
- break;
416
- }
417
- case KeySequence.CompareResult.SHADOW: {
418
- result.shadow.push(binding);
419
- break;
420
- }
421
- }
422
- } catch (error) {
423
- this.logger.warn(error);
424
- }
425
- }
426
- return result;
427
- }
428
-
429
- /**
430
- * Get all keybindings associated to a commandId.
431
- *
432
- * @param commandId The ID of the command for which we are looking for keybindings.
433
- * @returns an array of {@link ScopedKeybinding}
434
- */
435
- getKeybindingsForCommand(commandId: string): ScopedKeybinding[] {
436
- const result: ScopedKeybinding[] = [];
437
- const disabledBindings = new Set<string>();
438
- for (let scope = KeybindingScope.END - 1; scope >= KeybindingScope.DEFAULT; scope--) {
439
- this.keymaps[scope].forEach(binding => {
440
- if (binding.command?.startsWith('-')) {
441
- disabledBindings.add(JSON.stringify({ command: binding.command.substring(1), binding: binding.keybinding, context: binding.context, when: binding.when }));
442
- } else {
443
- const command = this.commandRegistry.getCommand(binding.command);
444
- if (command
445
- && command.id === commandId
446
- && !disabledBindings.has(JSON.stringify({ command: binding.command, binding: binding.keybinding, context: binding.context, when: binding.when }))) {
447
- result.push({ ...binding, scope });
448
- }
449
- }
450
- });
451
- }
452
- return result;
453
- }
454
-
455
- protected isActive(binding: common.Keybinding): boolean {
456
- /* Pseudo commands like "passthrough" are always active (and not found
457
- in the command registry). */
458
- if (this.isPseudoCommand(binding.command)) {
459
- return true;
460
- }
461
-
462
- const command = this.commandRegistry.getCommand(binding.command);
463
- return !!command && !!this.commandRegistry.getActiveHandler(command.id);
464
- }
465
-
466
- /**
467
- * Tries to execute a keybinding.
468
- *
469
- * @param binding to execute
470
- * @param event keyboard event.
471
- */
472
- protected executeKeyBinding(binding: common.Keybinding, event: KeyboardEvent): void {
473
- if (this.isPseudoCommand(binding.command)) {
474
- /* Don't do anything, let the event propagate. */
475
- } else {
476
- const command = this.commandRegistry.getCommand(binding.command);
477
- if (command) {
478
- if (this.commandRegistry.isEnabled(binding.command, binding.args)) {
479
- this.commandRegistry.executeCommand(binding.command, binding.args)
480
- .catch(e => console.error('Failed to execute command:', e));
481
- }
482
-
483
- /* Note that if a keybinding is in context but the command is
484
- not active we still stop the processing here. */
485
- event.preventDefault();
486
- event.stopPropagation();
487
- }
488
- }
489
- }
490
-
491
- /**
492
- * Only execute if it has no context (global context) or if we're in that context.
493
- */
494
- protected isEnabled(binding: common.Keybinding, event: KeyboardEvent): boolean {
495
- return this.isEnabledInScope(binding, <HTMLElement>event.target);
496
- }
497
-
498
- isEnabledInScope(binding: common.Keybinding, target: HTMLElement | undefined): boolean {
499
- const context = binding.context && this.contexts[binding.context];
500
- if (binding.command && (!this.isPseudoCommand(binding.command) && !this.commandRegistry.isEnabled(binding.command, binding.args))) {
501
- return false;
502
- }
503
- if (context && !context.isEnabled(binding)) {
504
- return false;
505
- }
506
- if (binding.when && !this.whenContextService.match(binding.when, target)) {
507
- return false;
508
- }
509
- return true;
510
- }
511
-
512
- dispatchCommand(id: string, target?: EventTarget): void {
513
- const keybindings = this.getKeybindingsForCommand(id);
514
- if (keybindings.length) {
515
- for (const keyCode of this.resolveKeybinding(keybindings[0])) {
516
- this.dispatchKeyDown(keyCode, target);
517
- }
518
- }
519
- }
520
-
521
- dispatchKeyDown(input: KeyboardEventInit | KeyCode | string, target: EventTarget = document.activeElement || window): void {
522
- const eventInit = this.asKeyboardEventInit(input);
523
- const emulatedKeyboardEvent = new KeyboardEvent('keydown', eventInit);
524
- target.dispatchEvent(emulatedKeyboardEvent);
525
- }
526
- protected asKeyboardEventInit(input: KeyboardEventInit | KeyCode | string): KeyboardEventInit & Partial<{ keyCode: number }> {
527
- if (typeof input === 'string') {
528
- return this.asKeyboardEventInit(KeyCode.createKeyCode(input));
529
- }
530
- if (input instanceof KeyCode) {
531
- return {
532
- metaKey: input.meta,
533
- shiftKey: input.shift,
534
- altKey: input.alt,
535
- ctrlKey: input.ctrl,
536
- code: input.key && input.key.code,
537
- key: (input && input.character) || (input.key && input.key.code),
538
- keyCode: input.key && input.key.keyCode
539
- };
540
- }
541
- return input;
542
- }
543
-
544
- registerEventListeners(win: Window): Disposable {
545
- /* vvv HOTFIX begin vvv
546
- *
547
- * This is a hotfix against issues eclipse/theia#6459 and gitpod-io/gitpod#875 .
548
- * It should be reverted after Theia was updated to the newer Monaco.
549
- */
550
- let inComposition = false;
551
- const compositionStart = () => {
552
- inComposition = true;
553
- };
554
- win.document.addEventListener('compositionstart', compositionStart);
555
-
556
- const compositionEnd = () => {
557
- inComposition = false;
558
- };
559
- win.document.addEventListener('compositionend', compositionEnd);
560
-
561
- const keydown = (event: KeyboardEvent) => {
562
- if (inComposition !== true) {
563
- this.run(event);
564
- }
565
- };
566
- win.document.addEventListener('keydown', keydown, true);
567
-
568
- return Disposable.create(() => {
569
- win.document.removeEventListener('compositionstart', compositionStart);
570
- win.document.removeEventListener('compositionend', compositionEnd);
571
- win.document.removeEventListener('keydown', keydown);
572
- });
573
- }
574
- /**
575
- * Run the command matching to the given keyboard event.
576
- */
577
- run(event: KeyboardEvent): void {
578
- if (event.defaultPrevented) {
579
- return;
580
- }
581
-
582
- const eventDispatch = this.corePreferences['keyboard.dispatch'];
583
- const keyCode = KeyCode.createKeyCode(event, eventDispatch);
584
- /* Keycode is only a modifier, next keycode will be modifier + key.
585
- Ignore this one. */
586
- if (keyCode.isModifierOnly()) {
587
- return;
588
- }
589
-
590
- this.keyboardLayoutService.validateKeyCode(keyCode);
591
- this.keySequence.push(keyCode);
592
- const match = this.matchKeybinding(this.keySequence, event);
593
-
594
- if (match && match.kind === 'partial') {
595
- /* Accumulate the keysequence */
596
- event.preventDefault();
597
- event.stopPropagation();
598
-
599
- this.statusBar.setElement('keybinding-status', {
600
- text: nls.localize('theia/core/keybindingStatus', '{0} was pressed, waiting for more keys', `(${this.acceleratorForSequence(this.keySequence, '+')})`),
601
- alignment: StatusBarAlignment.LEFT,
602
- priority: 2
603
- });
604
- } else {
605
- if (match && match.kind === 'full') {
606
- this.executeKeyBinding(match.binding, event);
607
- }
608
- this.keySequence = [];
609
- this.statusBar.removeElement('keybinding-status');
610
- }
611
- }
612
-
613
- /**
614
- * Match first binding in the current context.
615
- * Keybindings ordered by a scope and by a registration order within the scope.
616
- *
617
- * FIXME:
618
- * This method should run very fast since it happens on each keystroke. We should reconsider how keybindings are stored.
619
- * It should be possible to look up full and partial keybinding for given key sequence for constant time using some kind of tree.
620
- * Such tree should not contain disabled keybindings and be invalidated whenever the registry is changed.
621
- */
622
- matchKeybinding(keySequence: KeySequence, event?: KeyboardEvent): KeybindingRegistry.Match {
623
- let disabled: Set<string> | undefined;
624
- const isEnabled = (binding: ScopedKeybinding) => {
625
- if (event && !this.isEnabled(binding, event)) {
626
- return false;
627
- }
628
- const { command, context, when, keybinding } = binding;
629
- if (!this.isUsable(binding)) {
630
- disabled = disabled || new Set<string>();
631
- disabled.add(JSON.stringify({ command: command.substring(1), context, when, keybinding }));
632
- return false;
633
- }
634
- return !disabled?.has(JSON.stringify({ command, context, when, keybinding }));
635
- };
636
-
637
- for (let scope = KeybindingScope.END; --scope >= KeybindingScope.DEFAULT;) {
638
- for (const binding of this.keymaps[scope]) {
639
- const resolved = this.resolveKeybinding(binding);
640
- const compareResult = KeySequence.compare(keySequence, resolved);
641
- if (compareResult === KeySequence.CompareResult.FULL && isEnabled(binding)) {
642
- return { kind: 'full', binding };
643
- }
644
- if (compareResult === KeySequence.CompareResult.PARTIAL && isEnabled(binding)) {
645
- return { kind: 'partial', binding };
646
- }
647
- }
648
- }
649
- return undefined;
650
- }
651
-
652
- /**
653
- * Returns true if the binding is usable
654
- * @param binding Binding to be checked
655
- */
656
- protected isUsable(binding: common.Keybinding): boolean {
657
- return binding.command.charAt(0) !== '-';
658
- }
659
-
660
- /**
661
- * Return a new filtered array containing only the usable bindings among the input bindings
662
- * @param bindings Bindings to filter
663
- */
664
- protected getUsableBindings<T extends common.Keybinding>(bindings: T[]): T[] {
665
- return bindings.filter(binding => this.isUsable(binding));
666
- }
667
-
668
- /**
669
- * Return true of string a pseudo-command id, in other words a command id
670
- * that has a special meaning and that we won't find in the command
671
- * registry.
672
- *
673
- * @param commandId commandId to test
674
- */
675
- isPseudoCommand(commandId: string): boolean {
676
- return commandId === KeybindingRegistry.PASSTHROUGH_PSEUDO_COMMAND;
677
- }
678
-
679
- /**
680
- * Sets a new keymap replacing all existing {@link common.Keybinding}s in the given scope.
681
- * @param scope the keybinding scope
682
- * @param bindings an array containing the new {@link common.Keybinding}s
683
- */
684
- setKeymap(scope: KeybindingScope, bindings: common.Keybinding[]): void {
685
- this.resetKeybindingsForScope(scope);
686
- this.toResetKeymap.set(scope, this.doRegisterKeybindings(bindings, scope));
687
- this.keybindingsChanged.fire(undefined);
688
- }
689
-
690
- protected readonly toResetKeymap = new Map<KeybindingScope, Disposable>();
691
-
692
- /**
693
- * Reset keybindings for a specific scope
694
- * @param scope scope to reset the keybindings for
695
- */
696
- resetKeybindingsForScope(scope: KeybindingScope): void {
697
- const toReset = this.toResetKeymap.get(scope);
698
- if (toReset) {
699
- toReset.dispose();
700
- }
701
- }
702
-
703
- /**
704
- * Reset keybindings for all scopes(only leaves the default keybindings mapped)
705
- */
706
- resetKeybindings(): void {
707
- for (let i = KeybindingScope.DEFAULT + 1; i < KeybindingScope.END; i++) {
708
- this.keymaps[i] = [];
709
- }
710
- }
711
-
712
- /**
713
- * Get all {@link common.Keybinding}s for a {@link KeybindingScope}.
714
- * @returns an array of {@link common.ScopedKeybinding}
715
- * @param scope the keybinding scope to retrieve the {@link common.Keybinding}s for.
716
- */
717
- getKeybindingsByScope(scope: KeybindingScope): ScopedKeybinding[] {
718
- return this.keymaps[scope];
719
- }
720
- }
721
-
722
- export namespace KeybindingRegistry {
723
- export type Match = {
724
- kind: 'full' | 'partial'
725
- binding: ScopedKeybinding
726
- } | undefined;
727
- export class KeybindingsResult {
728
- full: ScopedKeybinding[] = [];
729
- partial: ScopedKeybinding[] = [];
730
- shadow: ScopedKeybinding[] = [];
731
-
732
- /**
733
- * Merge two results together inside `this`
734
- *
735
- * @param other the other KeybindingsResult to merge with
736
- * @return this
737
- */
738
- merge(other: KeybindingsResult): KeybindingsResult {
739
- this.full.push(...other.full);
740
- this.partial.push(...other.partial);
741
- this.shadow.push(...other.shadow);
742
- return this;
743
- }
744
-
745
- /**
746
- * Returns a new filtered KeybindingsResult
747
- *
748
- * @param fn callback filter on the results
749
- * @return filtered new result
750
- */
751
- filter(fn: (binding: common.Keybinding) => boolean): KeybindingsResult {
752
- const result = new KeybindingsResult();
753
- result.full = this.full.filter(fn);
754
- result.partial = this.partial.filter(fn);
755
- result.shadow = this.shadow.filter(fn);
756
- return result;
757
- }
758
- }
759
- }
1
+ // *****************************************************************************
2
+ // Copyright (C) 2017 TypeFox and others.
3
+ //
4
+ // This program and the accompanying materials are made available under the
5
+ // terms of the Eclipse Public License v. 2.0 which is available at
6
+ // http://www.eclipse.org/legal/epl-2.0.
7
+ //
8
+ // This Source Code may also be made available under the following Secondary
9
+ // Licenses when the conditions for such availability set forth in the Eclipse
10
+ // Public License v. 2.0 are satisfied: GNU General Public License, version 2
11
+ // with the GNU Classpath Exception which is available at
12
+ // https://www.gnu.org/software/classpath/license.html.
13
+ //
14
+ // SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
15
+ // *****************************************************************************
16
+
17
+ import { injectable, inject, named } from 'inversify';
18
+ import { isOSX } from '../common/os';
19
+ import { Emitter, Event } from '../common/event';
20
+ import { CommandRegistry, Command } from '../common/command';
21
+ import { Disposable, DisposableCollection } from '../common/disposable';
22
+ import { KeyCode, KeySequence, Key } from './keyboard/keys';
23
+ import { KeyboardLayoutService } from './keyboard/keyboard-layout-service';
24
+ import { ContributionProvider } from '../common/contribution-provider';
25
+ import { ILogger } from '../common/logger';
26
+ import { StatusBarAlignment, StatusBar } from './status-bar/status-bar';
27
+ import { ContextKeyService } from './context-key-service';
28
+ import { CorePreferences } from './core-preferences';
29
+ import * as common from '../common/keybinding';
30
+ import { nls } from '../common/nls';
31
+
32
+ export enum KeybindingScope {
33
+ DEFAULT,
34
+ USER,
35
+ WORKSPACE,
36
+ END
37
+ }
38
+ export namespace KeybindingScope {
39
+ export const length = KeybindingScope.END - KeybindingScope.DEFAULT;
40
+ }
41
+
42
+ /**
43
+ * @deprecated import from `@theia/core/lib/common/keybinding` instead
44
+ */
45
+ export type Keybinding = common.Keybinding;
46
+ export const Keybinding = common.Keybinding;
47
+
48
+ export interface ResolvedKeybinding extends common.Keybinding {
49
+ /**
50
+ * The KeyboardLayoutService may transform the `keybinding` depending on the
51
+ * user's keyboard layout. This property holds the transformed keybinding that
52
+ * should be used in the UI. The value is undefined if the KeyboardLayoutService
53
+ * has not been called yet to resolve the keybinding.
54
+ */
55
+ resolved?: KeyCode[];
56
+ }
57
+
58
+ export interface ScopedKeybinding extends common.Keybinding {
59
+ /** Current keybinding scope */
60
+ scope: KeybindingScope;
61
+ }
62
+
63
+ export const KeybindingContribution = Symbol('KeybindingContribution');
64
+ /**
65
+ * Allows extensions to contribute {@link common.Keybinding}s
66
+ */
67
+ export interface KeybindingContribution {
68
+ /**
69
+ * Registers keybindings.
70
+ * @param keybindings the keybinding registry.
71
+ */
72
+ registerKeybindings(keybindings: KeybindingRegistry): void;
73
+ }
74
+
75
+ export const KeybindingContext = Symbol('KeybindingContext');
76
+ export interface KeybindingContext {
77
+ /**
78
+ * The unique ID of the current context.
79
+ */
80
+ readonly id: string;
81
+
82
+ isEnabled(arg: common.Keybinding): boolean;
83
+ }
84
+ export namespace KeybindingContexts {
85
+
86
+ export const NOOP_CONTEXT: KeybindingContext = {
87
+ id: 'noop.keybinding.context',
88
+ isEnabled: () => true
89
+ };
90
+
91
+ export const DEFAULT_CONTEXT: KeybindingContext = {
92
+ id: 'default.keybinding.context',
93
+ isEnabled: () => false
94
+ };
95
+ }
96
+
97
+ @injectable()
98
+ export class KeybindingRegistry {
99
+
100
+ static readonly PASSTHROUGH_PSEUDO_COMMAND = 'passthrough';
101
+ protected keySequence: KeySequence = [];
102
+
103
+ protected readonly contexts: { [id: string]: KeybindingContext } = {};
104
+ protected readonly keymaps: ScopedKeybinding[][] = [...Array(KeybindingScope.length)].map(() => []);
105
+
106
+ @inject(CorePreferences)
107
+ protected readonly corePreferences: CorePreferences;
108
+
109
+ @inject(KeyboardLayoutService)
110
+ protected readonly keyboardLayoutService: KeyboardLayoutService;
111
+
112
+ @inject(ContributionProvider) @named(KeybindingContext)
113
+ protected readonly contextProvider: ContributionProvider<KeybindingContext>;
114
+
115
+ @inject(CommandRegistry)
116
+ protected readonly commandRegistry: CommandRegistry;
117
+
118
+ @inject(ContributionProvider) @named(KeybindingContribution)
119
+ protected readonly contributions: ContributionProvider<KeybindingContribution>;
120
+
121
+ @inject(StatusBar)
122
+ protected readonly statusBar: StatusBar;
123
+
124
+ @inject(ILogger)
125
+ protected readonly logger: ILogger;
126
+
127
+ @inject(ContextKeyService)
128
+ protected readonly whenContextService: ContextKeyService;
129
+
130
+ async onStart(): Promise<void> {
131
+ await this.keyboardLayoutService.initialize();
132
+ this.keyboardLayoutService.onKeyboardLayoutChanged(newLayout => {
133
+ this.clearResolvedKeybindings();
134
+ this.keybindingsChanged.fire(undefined);
135
+ });
136
+ this.registerContext(KeybindingContexts.NOOP_CONTEXT);
137
+ this.registerContext(KeybindingContexts.DEFAULT_CONTEXT);
138
+ this.registerContext(...this.contextProvider.getContributions());
139
+ for (const contribution of this.contributions.getContributions()) {
140
+ contribution.registerKeybindings(this);
141
+ }
142
+ }
143
+
144
+ protected keybindingsChanged = new Emitter<void>();
145
+
146
+ /**
147
+ * Event that is fired when the resolved keybindings change due to a different keyboard layout
148
+ * or when a new keymap is being set
149
+ */
150
+ get onKeybindingsChanged(): Event<void> {
151
+ return this.keybindingsChanged.event;
152
+ }
153
+
154
+ /**
155
+ * Registers the keybinding context arguments into the application. Fails when an already registered
156
+ * context is being registered.
157
+ *
158
+ * @param contexts the keybinding contexts to register into the application.
159
+ */
160
+ protected registerContext(...contexts: KeybindingContext[]): void {
161
+ for (const context of contexts) {
162
+ const { id } = context;
163
+ if (this.contexts[id]) {
164
+ this.logger.error(`A keybinding context with ID ${id} is already registered.`);
165
+ } else {
166
+ this.contexts[id] = context;
167
+ }
168
+ }
169
+ }
170
+
171
+ /**
172
+ * Register a default keybinding to the registry.
173
+ *
174
+ * Keybindings registered later have higher priority during evaluation.
175
+ *
176
+ * @param binding the keybinding to be registered
177
+ */
178
+ registerKeybinding(binding: common.Keybinding): Disposable {
179
+ return this.doRegisterKeybinding(binding);
180
+ }
181
+
182
+ /**
183
+ * Register multiple default keybindings to the registry
184
+ *
185
+ * @param bindings An array of keybinding to be registered
186
+ */
187
+ registerKeybindings(...bindings: common.Keybinding[]): Disposable {
188
+ return this.doRegisterKeybindings(bindings, KeybindingScope.DEFAULT);
189
+ }
190
+
191
+ /**
192
+ * Unregister all keybindings from the registry that are bound to the key of the given keybinding
193
+ *
194
+ * @param binding a keybinding specifying the key to be unregistered
195
+ */
196
+ unregisterKeybinding(binding: common.Keybinding): void;
197
+ /**
198
+ * Unregister all keybindings with the given key from the registry
199
+ *
200
+ * @param key a key to be unregistered
201
+ */
202
+ unregisterKeybinding(key: string): void;
203
+ /**
204
+ * Unregister all existing keybindings for the given command
205
+ * @param command the command to unregister all keybindings for
206
+ */
207
+ unregisterKeybinding(command: Command): void;
208
+
209
+ unregisterKeybinding(arg: common.Keybinding | string | Command): void {
210
+ const keymap = this.keymaps[KeybindingScope.DEFAULT];
211
+ const filter = Command.is(arg)
212
+ ? ({ command }: common.Keybinding) => command === arg.id
213
+ : ({ keybinding }: common.Keybinding) => Keybinding.is(arg)
214
+ ? keybinding === arg.keybinding
215
+ : keybinding === arg;
216
+ for (const binding of keymap.filter(filter)) {
217
+ const idx = keymap.indexOf(binding);
218
+ if (idx !== -1) {
219
+ keymap.splice(idx, 1);
220
+ }
221
+ }
222
+ }
223
+
224
+ protected doRegisterKeybindings(bindings: common.Keybinding[], scope: KeybindingScope = KeybindingScope.DEFAULT): Disposable {
225
+ const toDispose = new DisposableCollection();
226
+ for (const binding of bindings) {
227
+ toDispose.push(this.doRegisterKeybinding(binding, scope));
228
+ }
229
+ return toDispose;
230
+ }
231
+
232
+ protected doRegisterKeybinding(binding: common.Keybinding, scope: KeybindingScope = KeybindingScope.DEFAULT): Disposable {
233
+ try {
234
+ this.resolveKeybinding(binding);
235
+ const scoped = Object.assign(binding, { scope });
236
+ this.insertBindingIntoScope(scoped, scope);
237
+ return Disposable.create(() => {
238
+ const index = this.keymaps[scope].indexOf(scoped);
239
+ if (index !== -1) {
240
+ this.keymaps[scope].splice(index, 1);
241
+ }
242
+ });
243
+ } catch (error) {
244
+ this.logger.warn(`Could not register keybinding:\n ${common.Keybinding.stringify(binding)}\n${error}`);
245
+ return Disposable.NULL;
246
+ }
247
+ }
248
+
249
+ /**
250
+ * Ensures that keybindings are inserted in order of increasing length of binding to ensure that if a
251
+ * user triggers a short keybinding (e.g. ctrl+k), the UI won't wait for a longer one (e.g. ctrl+k enter)
252
+ */
253
+ protected insertBindingIntoScope(item: common.Keybinding & { scope: KeybindingScope; }, scope: KeybindingScope): void {
254
+ const scopedKeymap = this.keymaps[scope];
255
+ const getNumberOfKeystrokes = (binding: common.Keybinding): number => (binding.keybinding.trim().match(/\s/g)?.length ?? 0) + 1;
256
+ const numberOfKeystrokesInBinding = getNumberOfKeystrokes(item);
257
+ const indexOfFirstItemWithEqualStrokes = scopedKeymap.findIndex(existingBinding => getNumberOfKeystrokes(existingBinding) === numberOfKeystrokesInBinding);
258
+ if (indexOfFirstItemWithEqualStrokes > -1) {
259
+ scopedKeymap.splice(indexOfFirstItemWithEqualStrokes, 0, item);
260
+ } else {
261
+ scopedKeymap.push(item);
262
+ }
263
+ }
264
+
265
+ /**
266
+ * Ensure that the `resolved` property of the given binding is set by calling the KeyboardLayoutService.
267
+ */
268
+ resolveKeybinding(binding: ResolvedKeybinding): KeyCode[] {
269
+ if (!binding.resolved) {
270
+ const sequence = KeySequence.parse(binding.keybinding);
271
+ binding.resolved = sequence.map(code => this.keyboardLayoutService.resolveKeyCode(code));
272
+ }
273
+ return binding.resolved;
274
+ }
275
+
276
+ /**
277
+ * Clear all `resolved` properties of registered keybindings so the KeyboardLayoutService is called
278
+ * again to resolve them. This is necessary when the user's keyboard layout has changed.
279
+ */
280
+ protected clearResolvedKeybindings(): void {
281
+ for (let i = KeybindingScope.DEFAULT; i < KeybindingScope.END; i++) {
282
+ const bindings = this.keymaps[i];
283
+ for (let j = 0; j < bindings.length; j++) {
284
+ const binding = bindings[j] as ResolvedKeybinding;
285
+ binding.resolved = undefined;
286
+ }
287
+ }
288
+ }
289
+
290
+ /**
291
+ * Checks whether a colliding {@link common.Keybinding} exists in a specific scope.
292
+ * @param binding the keybinding to check
293
+ * @param scope the keybinding scope to check
294
+ * @returns true if there is a colliding keybinding
295
+ */
296
+ containsKeybindingInScope(binding: common.Keybinding, scope = KeybindingScope.USER): boolean {
297
+ const bindingKeySequence = this.resolveKeybinding(binding);
298
+ const collisions = this.getKeySequenceCollisions(this.getUsableBindings(this.keymaps[scope]), bindingKeySequence)
299
+ .filter(b => b.context === binding.context && !b.when && !binding.when);
300
+ if (collisions.full.length > 0) {
301
+ return true;
302
+ }
303
+ if (collisions.partial.length > 0) {
304
+ return true;
305
+ }
306
+ if (collisions.shadow.length > 0) {
307
+ return true;
308
+ }
309
+ return false;
310
+ }
311
+
312
+ /**
313
+ * Get a user visible representation of a {@link common.Keybinding}.
314
+ * @returns an array of strings representing all elements of the {@link KeySequence} defined by the {@link common.Keybinding}
315
+ * @param keybinding the keybinding
316
+ * @param separator the separator to be used to stringify {@link KeyCode}s that are part of the {@link KeySequence}
317
+ */
318
+ acceleratorFor(keybinding: common.Keybinding, separator: string = ' ', asciiOnly = false): string[] {
319
+ const bindingKeySequence = this.resolveKeybinding(keybinding);
320
+ return this.acceleratorForSequence(bindingKeySequence, separator, asciiOnly);
321
+ }
322
+
323
+ /**
324
+ * Get a user visible representation of a {@link KeySequence}.
325
+ * @returns an array of strings representing all elements of the {@link KeySequence}
326
+ * @param keySequence the keysequence
327
+ * @param separator the separator to be used to stringify {@link KeyCode}s that are part of the {@link KeySequence}
328
+ */
329
+ acceleratorForSequence(keySequence: KeySequence, separator: string = ' ', asciiOnly = false): string[] {
330
+ return keySequence.map(keyCode => this.acceleratorForKeyCode(keyCode, separator, asciiOnly));
331
+ }
332
+
333
+ /**
334
+ * Get a user visible representation of a key code (a key with modifiers).
335
+ * @returns a string representing the {@link KeyCode}
336
+ * @param keyCode the keycode
337
+ * @param separator the separator used to separate keys (key and modifiers) in the returning string
338
+ * @param asciiOnly if `true`, no special characters will be substituted into the string returned. Ensures correct keyboard shortcuts in Electron menus.
339
+ */
340
+ acceleratorForKeyCode(keyCode: KeyCode, separator: string = ' ', asciiOnly = false): string {
341
+ return this.componentsForKeyCode(keyCode, asciiOnly).join(separator);
342
+ }
343
+
344
+ componentsForKeyCode(keyCode: KeyCode, asciiOnly = false): string[] {
345
+ const keyCodeResult = [];
346
+ const useSymbols = isOSX && !asciiOnly;
347
+ if (keyCode.meta && isOSX) {
348
+ keyCodeResult.push(useSymbols ? '⌘' : 'Cmd');
349
+ }
350
+ if (keyCode.ctrl) {
351
+ keyCodeResult.push(useSymbols ? '⌃' : 'Ctrl');
352
+ }
353
+ if (keyCode.alt) {
354
+ keyCodeResult.push(useSymbols ? '⌥' : 'Alt');
355
+ }
356
+ if (keyCode.shift) {
357
+ keyCodeResult.push(useSymbols ? '⇧' : 'Shift');
358
+ }
359
+ if (keyCode.key) {
360
+ keyCodeResult.push(this.acceleratorForKey(keyCode.key, asciiOnly));
361
+ }
362
+ return keyCodeResult;
363
+ }
364
+
365
+ /**
366
+ * @param asciiOnly if `true`, no special characters will be substituted into the string returned. Ensures correct keyboard shortcuts in Electron menus.
367
+ *
368
+ * Return a user visible representation of a single key.
369
+ */
370
+ acceleratorForKey(key: Key, asciiOnly = false): string {
371
+ if (isOSX && !asciiOnly) {
372
+ if (key === Key.ARROW_LEFT) {
373
+ return '←';
374
+ }
375
+ if (key === Key.ARROW_RIGHT) {
376
+ return '→';
377
+ }
378
+ if (key === Key.ARROW_UP) {
379
+ return '↑';
380
+ }
381
+ if (key === Key.ARROW_DOWN) {
382
+ return '↓';
383
+ }
384
+ }
385
+ const keyString = this.keyboardLayoutService.getKeyboardCharacter(key);
386
+ if (key.keyCode >= Key.KEY_A.keyCode && key.keyCode <= Key.KEY_Z.keyCode ||
387
+ key.keyCode >= Key.F1.keyCode && key.keyCode <= Key.F24.keyCode) {
388
+ return keyString.toUpperCase();
389
+ } else if (keyString.length > 1) {
390
+ return keyString.charAt(0).toUpperCase() + keyString.slice(1);
391
+ } else {
392
+ return keyString;
393
+ }
394
+ }
395
+
396
+ /**
397
+ * Finds collisions for a key sequence inside a list of bindings (error-free)
398
+ *
399
+ * @param bindings the reference bindings
400
+ * @param candidate the sequence to match
401
+ */
402
+ protected getKeySequenceCollisions(bindings: ScopedKeybinding[], candidate: KeySequence): KeybindingRegistry.KeybindingsResult {
403
+ const result = new KeybindingRegistry.KeybindingsResult();
404
+ for (const binding of bindings) {
405
+ try {
406
+ const bindingKeySequence = this.resolveKeybinding(binding);
407
+ const compareResult = KeySequence.compare(candidate, bindingKeySequence);
408
+ switch (compareResult) {
409
+ case KeySequence.CompareResult.FULL: {
410
+ result.full.push(binding);
411
+ break;
412
+ }
413
+ case KeySequence.CompareResult.PARTIAL: {
414
+ result.partial.push(binding);
415
+ break;
416
+ }
417
+ case KeySequence.CompareResult.SHADOW: {
418
+ result.shadow.push(binding);
419
+ break;
420
+ }
421
+ }
422
+ } catch (error) {
423
+ this.logger.warn(error);
424
+ }
425
+ }
426
+ return result;
427
+ }
428
+
429
+ /**
430
+ * Get all keybindings associated to a commandId.
431
+ *
432
+ * @param commandId The ID of the command for which we are looking for keybindings.
433
+ * @returns an array of {@link ScopedKeybinding}
434
+ */
435
+ getKeybindingsForCommand(commandId: string): ScopedKeybinding[] {
436
+ const result: ScopedKeybinding[] = [];
437
+ const disabledBindings = new Set<string>();
438
+ for (let scope = KeybindingScope.END - 1; scope >= KeybindingScope.DEFAULT; scope--) {
439
+ this.keymaps[scope].forEach(binding => {
440
+ if (binding.command?.startsWith('-')) {
441
+ disabledBindings.add(JSON.stringify({ command: binding.command.substring(1), binding: binding.keybinding, context: binding.context, when: binding.when }));
442
+ } else {
443
+ const command = this.commandRegistry.getCommand(binding.command);
444
+ if (command
445
+ && command.id === commandId
446
+ && !disabledBindings.has(JSON.stringify({ command: binding.command, binding: binding.keybinding, context: binding.context, when: binding.when }))) {
447
+ result.push({ ...binding, scope });
448
+ }
449
+ }
450
+ });
451
+ }
452
+ return result;
453
+ }
454
+
455
+ protected isActive(binding: common.Keybinding): boolean {
456
+ /* Pseudo commands like "passthrough" are always active (and not found
457
+ in the command registry). */
458
+ if (this.isPseudoCommand(binding.command)) {
459
+ return true;
460
+ }
461
+
462
+ const command = this.commandRegistry.getCommand(binding.command);
463
+ return !!command && !!this.commandRegistry.getActiveHandler(command.id);
464
+ }
465
+
466
+ /**
467
+ * Tries to execute a keybinding.
468
+ *
469
+ * @param binding to execute
470
+ * @param event keyboard event.
471
+ */
472
+ protected executeKeyBinding(binding: common.Keybinding, event: KeyboardEvent): void {
473
+ if (this.isPseudoCommand(binding.command)) {
474
+ /* Don't do anything, let the event propagate. */
475
+ } else {
476
+ const command = this.commandRegistry.getCommand(binding.command);
477
+ if (command) {
478
+ if (this.commandRegistry.isEnabled(binding.command, binding.args)) {
479
+ this.commandRegistry.executeCommand(binding.command, binding.args)
480
+ .catch(e => console.error('Failed to execute command:', e));
481
+ }
482
+
483
+ /* Note that if a keybinding is in context but the command is
484
+ not active we still stop the processing here. */
485
+ event.preventDefault();
486
+ event.stopPropagation();
487
+ }
488
+ }
489
+ }
490
+
491
+ /**
492
+ * Only execute if it has no context (global context) or if we're in that context.
493
+ */
494
+ protected isEnabled(binding: common.Keybinding, event: KeyboardEvent): boolean {
495
+ return this.isEnabledInScope(binding, <HTMLElement>event.target);
496
+ }
497
+
498
+ isEnabledInScope(binding: common.Keybinding, target: HTMLElement | undefined): boolean {
499
+ const context = binding.context && this.contexts[binding.context];
500
+ if (binding.command && (!this.isPseudoCommand(binding.command) && !this.commandRegistry.isEnabled(binding.command, binding.args))) {
501
+ return false;
502
+ }
503
+ if (context && !context.isEnabled(binding)) {
504
+ return false;
505
+ }
506
+ if (binding.when && !this.whenContextService.match(binding.when, target)) {
507
+ return false;
508
+ }
509
+ return true;
510
+ }
511
+
512
+ dispatchCommand(id: string, target?: EventTarget): void {
513
+ const keybindings = this.getKeybindingsForCommand(id);
514
+ if (keybindings.length) {
515
+ for (const keyCode of this.resolveKeybinding(keybindings[0])) {
516
+ this.dispatchKeyDown(keyCode, target);
517
+ }
518
+ }
519
+ }
520
+
521
+ dispatchKeyDown(input: KeyboardEventInit | KeyCode | string, target: EventTarget = document.activeElement || window): void {
522
+ const eventInit = this.asKeyboardEventInit(input);
523
+ const emulatedKeyboardEvent = new KeyboardEvent('keydown', eventInit);
524
+ target.dispatchEvent(emulatedKeyboardEvent);
525
+ }
526
+ protected asKeyboardEventInit(input: KeyboardEventInit | KeyCode | string): KeyboardEventInit & Partial<{ keyCode: number }> {
527
+ if (typeof input === 'string') {
528
+ return this.asKeyboardEventInit(KeyCode.createKeyCode(input));
529
+ }
530
+ if (input instanceof KeyCode) {
531
+ return {
532
+ metaKey: input.meta,
533
+ shiftKey: input.shift,
534
+ altKey: input.alt,
535
+ ctrlKey: input.ctrl,
536
+ code: input.key && input.key.code,
537
+ key: (input && input.character) || (input.key && input.key.code),
538
+ keyCode: input.key && input.key.keyCode
539
+ };
540
+ }
541
+ return input;
542
+ }
543
+
544
+ registerEventListeners(win: Window): Disposable {
545
+ /* vvv HOTFIX begin vvv
546
+ *
547
+ * This is a hotfix against issues eclipse/theia#6459 and gitpod-io/gitpod#875 .
548
+ * It should be reverted after Theia was updated to the newer Monaco.
549
+ */
550
+ let inComposition = false;
551
+ const compositionStart = () => {
552
+ inComposition = true;
553
+ };
554
+ win.document.addEventListener('compositionstart', compositionStart);
555
+
556
+ const compositionEnd = () => {
557
+ inComposition = false;
558
+ };
559
+ win.document.addEventListener('compositionend', compositionEnd);
560
+
561
+ const keydown = (event: KeyboardEvent) => {
562
+ if (inComposition !== true) {
563
+ this.run(event);
564
+ }
565
+ };
566
+ win.document.addEventListener('keydown', keydown, true);
567
+
568
+ return Disposable.create(() => {
569
+ win.document.removeEventListener('compositionstart', compositionStart);
570
+ win.document.removeEventListener('compositionend', compositionEnd);
571
+ win.document.removeEventListener('keydown', keydown);
572
+ });
573
+ }
574
+ /**
575
+ * Run the command matching to the given keyboard event.
576
+ */
577
+ run(event: KeyboardEvent): void {
578
+ if (event.defaultPrevented) {
579
+ return;
580
+ }
581
+
582
+ const eventDispatch = this.corePreferences['keyboard.dispatch'];
583
+ const keyCode = KeyCode.createKeyCode(event, eventDispatch);
584
+ /* Keycode is only a modifier, next keycode will be modifier + key.
585
+ Ignore this one. */
586
+ if (keyCode.isModifierOnly()) {
587
+ return;
588
+ }
589
+
590
+ this.keyboardLayoutService.validateKeyCode(keyCode);
591
+ this.keySequence.push(keyCode);
592
+ const match = this.matchKeybinding(this.keySequence, event);
593
+
594
+ if (match && match.kind === 'partial') {
595
+ /* Accumulate the keysequence */
596
+ event.preventDefault();
597
+ event.stopPropagation();
598
+
599
+ this.statusBar.setElement('keybinding-status', {
600
+ text: nls.localize('theia/core/keybindingStatus', '{0} was pressed, waiting for more keys', `(${this.acceleratorForSequence(this.keySequence, '+')})`),
601
+ alignment: StatusBarAlignment.LEFT,
602
+ priority: 2
603
+ });
604
+ } else {
605
+ if (match && match.kind === 'full') {
606
+ this.executeKeyBinding(match.binding, event);
607
+ }
608
+ this.keySequence = [];
609
+ this.statusBar.removeElement('keybinding-status');
610
+ }
611
+ }
612
+
613
+ /**
614
+ * Match first binding in the current context.
615
+ * Keybindings ordered by a scope and by a registration order within the scope.
616
+ *
617
+ * FIXME:
618
+ * This method should run very fast since it happens on each keystroke. We should reconsider how keybindings are stored.
619
+ * It should be possible to look up full and partial keybinding for given key sequence for constant time using some kind of tree.
620
+ * Such tree should not contain disabled keybindings and be invalidated whenever the registry is changed.
621
+ */
622
+ matchKeybinding(keySequence: KeySequence, event?: KeyboardEvent): KeybindingRegistry.Match {
623
+ let disabled: Set<string> | undefined;
624
+ const isEnabled = (binding: ScopedKeybinding) => {
625
+ if (event && !this.isEnabled(binding, event)) {
626
+ return false;
627
+ }
628
+ const { command, context, when, keybinding } = binding;
629
+ if (!this.isUsable(binding)) {
630
+ disabled = disabled || new Set<string>();
631
+ disabled.add(JSON.stringify({ command: command.substring(1), context, when, keybinding }));
632
+ return false;
633
+ }
634
+ return !disabled?.has(JSON.stringify({ command, context, when, keybinding }));
635
+ };
636
+
637
+ for (let scope = KeybindingScope.END; --scope >= KeybindingScope.DEFAULT;) {
638
+ for (const binding of this.keymaps[scope]) {
639
+ const resolved = this.resolveKeybinding(binding);
640
+ const compareResult = KeySequence.compare(keySequence, resolved);
641
+ if (compareResult === KeySequence.CompareResult.FULL && isEnabled(binding)) {
642
+ return { kind: 'full', binding };
643
+ }
644
+ if (compareResult === KeySequence.CompareResult.PARTIAL && isEnabled(binding)) {
645
+ return { kind: 'partial', binding };
646
+ }
647
+ }
648
+ }
649
+ return undefined;
650
+ }
651
+
652
+ /**
653
+ * Returns true if the binding is usable
654
+ * @param binding Binding to be checked
655
+ */
656
+ protected isUsable(binding: common.Keybinding): boolean {
657
+ return binding.command.charAt(0) !== '-';
658
+ }
659
+
660
+ /**
661
+ * Return a new filtered array containing only the usable bindings among the input bindings
662
+ * @param bindings Bindings to filter
663
+ */
664
+ protected getUsableBindings<T extends common.Keybinding>(bindings: T[]): T[] {
665
+ return bindings.filter(binding => this.isUsable(binding));
666
+ }
667
+
668
+ /**
669
+ * Return true of string a pseudo-command id, in other words a command id
670
+ * that has a special meaning and that we won't find in the command
671
+ * registry.
672
+ *
673
+ * @param commandId commandId to test
674
+ */
675
+ isPseudoCommand(commandId: string): boolean {
676
+ return commandId === KeybindingRegistry.PASSTHROUGH_PSEUDO_COMMAND;
677
+ }
678
+
679
+ /**
680
+ * Sets a new keymap replacing all existing {@link common.Keybinding}s in the given scope.
681
+ * @param scope the keybinding scope
682
+ * @param bindings an array containing the new {@link common.Keybinding}s
683
+ */
684
+ setKeymap(scope: KeybindingScope, bindings: common.Keybinding[]): void {
685
+ this.resetKeybindingsForScope(scope);
686
+ this.toResetKeymap.set(scope, this.doRegisterKeybindings(bindings, scope));
687
+ this.keybindingsChanged.fire(undefined);
688
+ }
689
+
690
+ protected readonly toResetKeymap = new Map<KeybindingScope, Disposable>();
691
+
692
+ /**
693
+ * Reset keybindings for a specific scope
694
+ * @param scope scope to reset the keybindings for
695
+ */
696
+ resetKeybindingsForScope(scope: KeybindingScope): void {
697
+ const toReset = this.toResetKeymap.get(scope);
698
+ if (toReset) {
699
+ toReset.dispose();
700
+ }
701
+ }
702
+
703
+ /**
704
+ * Reset keybindings for all scopes(only leaves the default keybindings mapped)
705
+ */
706
+ resetKeybindings(): void {
707
+ for (let i = KeybindingScope.DEFAULT + 1; i < KeybindingScope.END; i++) {
708
+ this.keymaps[i] = [];
709
+ }
710
+ }
711
+
712
+ /**
713
+ * Get all {@link common.Keybinding}s for a {@link KeybindingScope}.
714
+ * @returns an array of {@link common.ScopedKeybinding}
715
+ * @param scope the keybinding scope to retrieve the {@link common.Keybinding}s for.
716
+ */
717
+ getKeybindingsByScope(scope: KeybindingScope): ScopedKeybinding[] {
718
+ return this.keymaps[scope];
719
+ }
720
+ }
721
+
722
+ export namespace KeybindingRegistry {
723
+ export type Match = {
724
+ kind: 'full' | 'partial'
725
+ binding: ScopedKeybinding
726
+ } | undefined;
727
+ export class KeybindingsResult {
728
+ full: ScopedKeybinding[] = [];
729
+ partial: ScopedKeybinding[] = [];
730
+ shadow: ScopedKeybinding[] = [];
731
+
732
+ /**
733
+ * Merge two results together inside `this`
734
+ *
735
+ * @param other the other KeybindingsResult to merge with
736
+ * @return this
737
+ */
738
+ merge(other: KeybindingsResult): KeybindingsResult {
739
+ this.full.push(...other.full);
740
+ this.partial.push(...other.partial);
741
+ this.shadow.push(...other.shadow);
742
+ return this;
743
+ }
744
+
745
+ /**
746
+ * Returns a new filtered KeybindingsResult
747
+ *
748
+ * @param fn callback filter on the results
749
+ * @return filtered new result
750
+ */
751
+ filter(fn: (binding: common.Keybinding) => boolean): KeybindingsResult {
752
+ const result = new KeybindingsResult();
753
+ result.full = this.full.filter(fn);
754
+ result.partial = this.partial.filter(fn);
755
+ result.shadow = this.shadow.filter(fn);
756
+ return result;
757
+ }
758
+ }
759
+ }