@theia/core 1.48.0 → 1.48.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +6 -6
- package/i18n/nls.cs.json +541 -541
- package/i18n/nls.de.json +541 -541
- package/i18n/nls.es.json +541 -541
- package/i18n/nls.fr.json +541 -541
- package/i18n/nls.hu.json +541 -541
- package/i18n/nls.it.json +541 -541
- package/i18n/nls.ja.json +541 -541
- package/i18n/nls.json +541 -541
- package/i18n/nls.pl.json +541 -541
- package/i18n/nls.pt-br.json +541 -541
- package/i18n/nls.pt-pt.json +541 -541
- package/i18n/nls.ru.json +541 -541
- package/i18n/nls.zh-cn.json +541 -541
- package/lib/browser/about-dialog.d.ts +33 -33
- package/lib/browser/about-dialog.js +125 -125
- package/lib/browser/authentication-service.d.ts +144 -144
- package/lib/browser/authentication-service.js +324 -324
- package/lib/browser/breadcrumbs/breadcrumb-popup-container.d.ts +34 -34
- package/lib/browser/breadcrumbs/breadcrumb-popup-container.js +106 -106
- package/lib/browser/breadcrumbs/breadcrumb-renderer.d.ts +12 -12
- package/lib/browser/breadcrumbs/breadcrumb-renderer.js +37 -37
- package/lib/browser/breadcrumbs/breadcrumbs-constants.d.ts +49 -49
- package/lib/browser/breadcrumbs/breadcrumbs-constants.js +27 -27
- package/lib/browser/breadcrumbs/breadcrumbs-renderer.d.ts +43 -43
- package/lib/browser/breadcrumbs/breadcrumbs-renderer.js +179 -179
- package/lib/browser/breadcrumbs/breadcrumbs-service.d.ts +32 -32
- package/lib/browser/breadcrumbs/breadcrumbs-service.js +111 -111
- package/lib/browser/breadcrumbs/index.d.ts +5 -5
- package/lib/browser/breadcrumbs/index.js +23 -23
- package/lib/browser/browser-clipboard-service.d.ts +23 -23
- package/lib/browser/browser-clipboard-service.js +107 -107
- package/lib/browser/browser.d.ts +37 -37
- package/lib/browser/browser.js +196 -196
- package/lib/browser/clipboard-service.d.ts +6 -6
- package/lib/browser/clipboard-service.js +19 -19
- package/lib/browser/color-application-contribution.d.ts +26 -26
- package/lib/browser/color-application-contribution.js +110 -110
- package/lib/browser/color-registry.d.ts +14 -14
- package/lib/browser/color-registry.js +58 -58
- package/lib/browser/command-open-handler.d.ts +9 -9
- package/lib/browser/command-open-handler.js +58 -58
- package/lib/browser/common-frontend-contribution.d.ts +212 -212
- package/lib/browser/common-frontend-contribution.js +2338 -2338
- package/lib/browser/common-styling-participants.d.ts +27 -27
- package/lib/browser/common-styling-participants.js +365 -365
- package/lib/browser/connection-status-service.d.ts +77 -77
- package/lib/browser/connection-status-service.js +198 -198
- package/lib/browser/connection-status-service.spec.d.ts +1 -1
- package/lib/browser/connection-status-service.spec.js +161 -161
- package/lib/browser/context-key-service.d.ts +82 -82
- package/lib/browser/context-key-service.js +83 -83
- package/lib/browser/context-menu-renderer.d.ts +54 -54
- package/lib/browser/context-menu-renderer.js +87 -87
- package/lib/browser/core-preferences.d.ts +37 -37
- package/lib/browser/core-preferences.js +300 -300
- package/lib/browser/credentials-service.d.ts +35 -35
- package/lib/browser/credentials-service.js +80 -80
- package/lib/browser/decoration-style.d.ts +6 -6
- package/lib/browser/decoration-style.js +64 -64
- package/lib/browser/decorations-service.d.ts +30 -30
- package/lib/browser/decorations-service.js +157 -157
- package/lib/browser/dialogs/react-dialog.d.ts +17 -17
- package/lib/browser/dialogs/react-dialog.js +49 -49
- package/lib/browser/dialogs.d.ts +135 -135
- package/lib/browser/dialogs.js +446 -446
- package/lib/browser/diff-uris.d.ts +17 -17
- package/lib/browser/diff-uris.js +115 -115
- package/lib/browser/encoding-registry.d.ts +18 -18
- package/lib/browser/encoding-registry.js +88 -88
- package/lib/browser/endpoint.d.ts +42 -42
- package/lib/browser/endpoint.js +120 -120
- package/lib/browser/endpoint.spec.d.ts +1 -1
- package/lib/browser/endpoint.spec.js +122 -122
- package/lib/browser/external-uri-service.d.ts +24 -24
- package/lib/browser/external-uri-service.js +72 -72
- package/lib/browser/frontend-application-bindings.d.ts +5 -5
- package/lib/browser/frontend-application-bindings.js +56 -56
- package/lib/browser/frontend-application-config-provider.d.ts +8 -8
- package/lib/browser/frontend-application-config-provider.js +46 -46
- package/lib/browser/frontend-application-config-provider.spec.d.ts +1 -1
- package/lib/browser/frontend-application-config-provider.spec.js +40 -40
- package/lib/browser/frontend-application-contribution.d.ts +75 -75
- package/lib/browser/frontend-application-contribution.js +45 -45
- package/lib/browser/frontend-application-module.d.ts +8 -8
- package/lib/browser/frontend-application-module.js +355 -355
- package/lib/browser/frontend-application-state.d.ts +19 -19
- package/lib/browser/frontend-application-state.js +71 -71
- package/lib/browser/frontend-application.d.ts +86 -86
- package/lib/browser/frontend-application.js +308 -308
- package/lib/browser/hover-service.d.ts +50 -50
- package/lib/browser/hover-service.js +199 -199
- package/lib/browser/http-open-handler.d.ts +14 -14
- package/lib/browser/http-open-handler.js +46 -46
- package/lib/browser/i18n/i18n-frontend-module.d.ts +3 -3
- package/lib/browser/i18n/i18n-frontend-module.js +25 -25
- package/lib/browser/i18n/language-quick-pick-service.d.ts +15 -15
- package/lib/browser/i18n/language-quick-pick-service.js +136 -136
- package/lib/browser/icon-registry.d.ts +56 -56
- package/lib/browser/icon-registry.js +24 -24
- package/lib/browser/icon-theme-contribution.d.ts +23 -23
- package/lib/browser/icon-theme-contribution.js +64 -64
- package/lib/browser/icon-theme-service.d.ts +63 -63
- package/lib/browser/icon-theme-service.js +198 -198
- package/lib/browser/index.d.ts +31 -31
- package/lib/browser/index.js +49 -49
- package/lib/browser/json-schema-store.d.ts +37 -37
- package/lib/browser/json-schema-store.js +111 -111
- package/lib/browser/keybinding.d.ts +288 -288
- package/lib/browser/keybinding.js +677 -677
- package/lib/browser/keybinding.spec.d.ts +1 -1
- package/lib/browser/keybinding.spec.js +451 -451
- package/lib/browser/keyboard/browser-keyboard-frontend-contribution.d.ts +13 -13
- package/lib/browser/keyboard/browser-keyboard-frontend-contribution.js +110 -110
- package/lib/browser/keyboard/browser-keyboard-layout-provider.d.ts +91 -91
- package/lib/browser/keyboard/browser-keyboard-layout-provider.js +418 -418
- package/lib/browser/keyboard/browser-keyboard-layout-provider.spec.d.ts +1 -1
- package/lib/browser/keyboard/browser-keyboard-layout-provider.spec.js +152 -152
- package/lib/browser/keyboard/browser-keyboard-module.d.ts +3 -3
- package/lib/browser/keyboard/browser-keyboard-module.js +30 -30
- package/lib/browser/keyboard/index.d.ts +4 -4
- package/lib/browser/keyboard/index.js +22 -22
- package/lib/browser/keyboard/keyboard-layout-service.d.ts +49 -49
- package/lib/browser/keyboard/keyboard-layout-service.js +427 -427
- package/lib/browser/keyboard/keyboard-layout-service.spec.d.ts +1 -1
- package/lib/browser/keyboard/keyboard-layout-service.spec.js +110 -110
- package/lib/browser/keyboard/keys.d.ts +4 -4
- package/lib/browser/keyboard/keys.js +22 -22
- package/lib/browser/keyboard/keys.spec.d.ts +1 -1
- package/lib/browser/keyboard/keys.spec.js +205 -205
- package/lib/browser/keys.d.ts +2 -2
- package/lib/browser/keys.js +28 -28
- package/lib/browser/label-parser.d.ts +34 -34
- package/lib/browser/label-parser.js +107 -107
- package/lib/browser/label-parser.spec.d.ts +1 -1
- package/lib/browser/label-parser.spec.js +146 -146
- package/lib/browser/label-provider.d.ts +140 -140
- package/lib/browser/label-provider.js +310 -310
- package/lib/browser/label-provider.spec.d.ts +1 -1
- package/lib/browser/label-provider.spec.js +53 -53
- package/lib/browser/language-icon-provider.d.ts +14 -14
- package/lib/browser/language-icon-provider.js +65 -65
- package/lib/browser/language-service.d.ts +38 -38
- package/lib/browser/language-service.js +66 -66
- package/lib/browser/logger-frontend-module.d.ts +2 -2
- package/lib/browser/logger-frontend-module.js +63 -63
- package/lib/browser/markdown-rendering/markdown-renderer.d.ts +39 -39
- package/lib/browser/markdown-rendering/markdown-renderer.js +72 -72
- package/lib/browser/menu/browser-context-menu-renderer.d.ts +12 -12
- package/lib/browser/menu/browser-context-menu-renderer.js +51 -51
- package/lib/browser/menu/browser-menu-module.d.ts +3 -3
- package/lib/browser/menu/browser-menu-module.js +28 -28
- package/lib/browser/menu/browser-menu-plugin.d.ts +103 -103
- package/lib/browser/menu/browser-menu-plugin.js +439 -439
- package/lib/browser/menu/context-menu-context.d.ts +7 -7
- package/lib/browser/menu/context-menu-context.js +42 -42
- package/lib/browser/messaging/connection-source.d.ts +8 -8
- package/lib/browser/messaging/connection-source.js +19 -19
- package/lib/browser/messaging/frontend-id-provider.d.ts +12 -12
- package/lib/browser/messaging/frontend-id-provider.js +34 -34
- package/lib/browser/messaging/index.d.ts +1 -1
- package/lib/browser/messaging/index.js +19 -19
- package/lib/browser/messaging/messaging-frontend-module.d.ts +2 -2
- package/lib/browser/messaging/messaging-frontend-module.js +41 -41
- package/lib/browser/messaging/service-connection-provider.d.ts +47 -47
- package/lib/browser/messaging/service-connection-provider.js +106 -106
- package/lib/browser/messaging/ws-connection-provider.d.ts +12 -12
- package/lib/browser/messaging/ws-connection-provider.js +49 -49
- package/lib/browser/messaging/ws-connection-source.d.ts +42 -42
- package/lib/browser/messaging/ws-connection-source.js +217 -217
- package/lib/browser/mime-service.d.ts +7 -7
- package/lib/browser/mime-service.js +30 -30
- package/lib/browser/navigatable-types.d.ts +33 -33
- package/lib/browser/navigatable-types.js +63 -63
- package/lib/browser/navigatable.d.ts +8 -8
- package/lib/browser/navigatable.js +38 -38
- package/lib/browser/opener-service.d.ts +79 -79
- package/lib/browser/opener-service.js +79 -79
- package/lib/browser/opener-service.spec.d.ts +1 -1
- package/lib/browser/opener-service.spec.js +46 -46
- package/lib/browser/performance/frontend-stopwatch.d.ts +20 -20
- package/lib/browser/performance/frontend-stopwatch.js +66 -66
- package/lib/browser/performance/index.d.ts +2 -2
- package/lib/browser/performance/index.js +20 -20
- package/lib/browser/performance/measurement-frontend-bindings.d.ts +19 -19
- package/lib/browser/performance/measurement-frontend-bindings.js +32 -32
- package/lib/browser/preferences/index.d.ts +7 -7
- package/lib/browser/preferences/index.js +25 -25
- package/lib/browser/preferences/injectable-preference-proxy.d.ts +51 -51
- package/lib/browser/preferences/injectable-preference-proxy.js +276 -276
- package/lib/browser/preferences/preference-configurations.d.ts +22 -22
- package/lib/browser/preferences/preference-configurations.js +72 -72
- package/lib/browser/preferences/preference-contribution.d.ts +111 -111
- package/lib/browser/preferences/preference-contribution.js +386 -386
- package/lib/browser/preferences/preference-language-override-service.d.ts +39 -39
- package/lib/browser/preferences/preference-language-override-service.js +107 -107
- package/lib/browser/preferences/preference-provider.d.ts +132 -132
- package/lib/browser/preferences/preference-provider.js +219 -219
- package/lib/browser/preferences/preference-provider.spec.d.ts +1 -1
- package/lib/browser/preferences/preference-provider.spec.js +33 -33
- package/lib/browser/preferences/preference-proxy.d.ts +150 -150
- package/lib/browser/preferences/preference-proxy.js +229 -229
- package/lib/browser/preferences/preference-proxy.spec.d.ts +1 -1
- package/lib/browser/preferences/preference-proxy.spec.js +337 -337
- package/lib/browser/preferences/preference-schema-provider.spec.d.ts +1 -1
- package/lib/browser/preferences/preference-schema-provider.spec.js +115 -115
- package/lib/browser/preferences/preference-scope.d.ts +2 -2
- package/lib/browser/preferences/preference-scope.js +20 -20
- package/lib/browser/preferences/preference-service.d.ts +278 -278
- package/lib/browser/preferences/preference-service.js +350 -350
- package/lib/browser/preferences/preference-service.spec.d.ts +1 -1
- package/lib/browser/preferences/preference-service.spec.js +505 -505
- package/lib/browser/preferences/preference-validation-service.d.ts +57 -57
- package/lib/browser/preferences/preference-validation-service.js +346 -346
- package/lib/browser/preferences/preference-validation-service.spec.d.ts +1 -1
- package/lib/browser/preferences/preference-validation-service.spec.js +330 -330
- package/lib/browser/preferences/test/index.d.ts +3 -3
- package/lib/browser/preferences/test/index.js +21 -21
- package/lib/browser/preferences/test/mock-preference-provider.d.ts +16 -16
- package/lib/browser/preferences/test/mock-preference-provider.js +46 -46
- package/lib/browser/preferences/test/mock-preference-proxy.d.ts +3 -3
- package/lib/browser/preferences/test/mock-preference-proxy.js +49 -49
- package/lib/browser/preferences/test/mock-preference-service.d.ts +28 -28
- package/lib/browser/preferences/test/mock-preference-service.js +59 -59
- package/lib/browser/preload/i18n-preload-contribution.d.ts +6 -6
- package/lib/browser/preload/i18n-preload-contribution.js +54 -54
- package/lib/browser/preload/os-preload-contribution.d.ts +6 -6
- package/lib/browser/preload/os-preload-contribution.js +40 -40
- package/lib/browser/preload/preload-module.d.ts +3 -3
- package/lib/browser/preload/preload-module.js +38 -38
- package/lib/browser/preload/preloader.d.ts +11 -11
- package/lib/browser/preload/preloader.js +36 -36
- package/lib/browser/preload/theme-preload-contribution.d.ts +4 -4
- package/lib/browser/preload/theme-preload-contribution.js +34 -34
- package/lib/browser/progress-bar-factory.d.ts +11 -11
- package/lib/browser/progress-bar-factory.js +20 -20
- package/lib/browser/progress-bar.d.ts +15 -15
- package/lib/browser/progress-bar.js +81 -81
- package/lib/browser/progress-client.d.ts +12 -12
- package/lib/browser/progress-client.js +53 -53
- package/lib/browser/progress-location-service.d.ts +21 -21
- package/lib/browser/progress-location-service.js +91 -91
- package/lib/browser/progress-location-service.spec.d.ts +1 -1
- package/lib/browser/progress-location-service.spec.js +44 -44
- package/lib/browser/progress-status-bar-item.d.ts +16 -16
- package/lib/browser/progress-status-bar-item.js +81 -81
- package/lib/browser/quick-input/index.d.ts +8 -8
- package/lib/browser/quick-input/index.js +26 -26
- package/lib/browser/quick-input/quick-access.d.ts +53 -53
- package/lib/browser/quick-input/quick-access.js +20 -20
- package/lib/browser/quick-input/quick-command-frontend-contribution.d.ts +11 -11
- package/lib/browser/quick-input/quick-command-frontend-contribution.js +96 -96
- package/lib/browser/quick-input/quick-command-service.d.ts +59 -59
- package/lib/browser/quick-input/quick-command-service.js +225 -225
- package/lib/browser/quick-input/quick-help-service.d.ts +11 -11
- package/lib/browser/quick-input/quick-help-service.js +83 -83
- package/lib/browser/quick-input/quick-input-frontend-contribution.d.ts +7 -7
- package/lib/browser/quick-input/quick-input-frontend-contribution.js +38 -38
- package/lib/browser/quick-input/quick-input-service.d.ts +1 -1
- package/lib/browser/quick-input/quick-input-service.js +19 -19
- package/lib/browser/quick-input/quick-input-service.spec.d.ts +1 -1
- package/lib/browser/quick-input/quick-input-service.spec.js +159 -159
- package/lib/browser/quick-input/quick-pick-service-impl.d.ts +30 -30
- package/lib/browser/quick-input/quick-pick-service-impl.js +69 -69
- package/lib/browser/quick-input/quick-view-service.d.ts +23 -23
- package/lib/browser/quick-input/quick-view-service.js +78 -78
- package/lib/browser/request/browser-request-module.d.ts +18 -18
- package/lib/browser/request/browser-request-module.js +23 -23
- package/lib/browser/request/browser-request-service.d.ts +41 -41
- package/lib/browser/request/browser-request-service.js +172 -172
- package/lib/browser/resource-context-key.d.ts +19 -19
- package/lib/browser/resource-context-key.js +75 -75
- package/lib/browser/save-resource-service.d.ts +35 -35
- package/lib/browser/save-resource-service.js +64 -64
- package/lib/browser/saveable.d.ts +124 -124
- package/lib/browser/saveable.js +274 -274
- package/lib/browser/secondary-window-handler.d.ts +65 -65
- package/lib/browser/secondary-window-handler.js +193 -193
- package/lib/browser/shell/additional-views-menu-widget.d.ts +19 -19
- package/lib/browser/shell/additional-views-menu-widget.js +73 -73
- package/lib/browser/shell/application-shell-mouse-tracker.d.ts +30 -30
- package/lib/browser/shell/application-shell-mouse-tracker.js +99 -99
- package/lib/browser/shell/application-shell.d.ts +574 -574
- package/lib/browser/shell/application-shell.js +1992 -1992
- package/lib/browser/shell/current-widget-command-adapter.d.ts +23 -23
- package/lib/browser/shell/current-widget-command-adapter.js +41 -41
- package/lib/browser/shell/index.d.ts +7 -7
- package/lib/browser/shell/index.js +25 -25
- package/lib/browser/shell/shell-layout-restorer.d.ts +123 -123
- package/lib/browser/shell/shell-layout-restorer.js +336 -336
- package/lib/browser/shell/side-panel-handler.d.ts +286 -286
- package/lib/browser/shell/side-panel-handler.js +659 -659
- package/lib/browser/shell/side-panel-toolbar.d.ts +25 -25
- package/lib/browser/shell/side-panel-toolbar.js +96 -96
- package/lib/browser/shell/sidebar-bottom-menu-widget.d.ts +9 -9
- package/lib/browser/shell/sidebar-bottom-menu-widget.js +41 -41
- package/lib/browser/shell/sidebar-menu-widget.d.ts +38 -38
- package/lib/browser/shell/sidebar-menu-widget.js +113 -113
- package/lib/browser/shell/sidebar-top-menu-widget.d.ts +6 -6
- package/lib/browser/shell/sidebar-top-menu-widget.js +30 -30
- package/lib/browser/shell/split-panels.d.ts +42 -42
- package/lib/browser/shell/split-panels.js +172 -172
- package/lib/browser/shell/tab-bar-decorator.d.ts +40 -40
- package/lib/browser/shell/tab-bar-decorator.js +87 -87
- package/lib/browser/shell/tab-bar-toolbar/index.d.ts +3 -3
- package/lib/browser/shell/tab-bar-toolbar/index.js +21 -21
- package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-menu-adapters.d.ts +15 -15
- package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-menu-adapters.js +35 -35
- package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-registry.d.ts +95 -95
- package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-registry.js +245 -245
- package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-types.d.ts +136 -136
- package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar-types.js +88 -88
- package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar.d.ts +89 -89
- package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar.js +414 -414
- package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar.spec.d.ts +1 -1
- package/lib/browser/shell/tab-bar-toolbar/tab-bar-toolbar.spec.js +51 -51
- package/lib/browser/shell/tab-bars.d.ts +341 -341
- package/lib/browser/shell/tab-bars.js +1312 -1312
- package/lib/browser/shell/tab-bars.spec.d.ts +1 -1
- package/lib/browser/shell/tab-bars.spec.js +56 -56
- package/lib/browser/shell/theia-dock-panel.d.ts +59 -59
- package/lib/browser/shell/theia-dock-panel.js +226 -226
- package/lib/browser/shell/view-contribution.d.ts +43 -43
- package/lib/browser/shell/view-contribution.js +159 -159
- package/lib/browser/source-tree/index.d.ts +3 -3
- package/lib/browser/source-tree/index.js +21 -21
- package/lib/browser/source-tree/source-tree-widget.d.ts +21 -21
- package/lib/browser/source-tree/source-tree-widget.js +110 -110
- package/lib/browser/source-tree/source-tree.d.ts +36 -36
- package/lib/browser/source-tree/source-tree.js +129 -129
- package/lib/browser/source-tree/tree-source.d.ts +35 -35
- package/lib/browser/source-tree/tree-source.js +53 -53
- package/lib/browser/status-bar/index.d.ts +5 -5
- package/lib/browser/status-bar/index.js +31 -31
- package/lib/browser/status-bar/status-bar-types.d.ts +72 -72
- package/lib/browser/status-bar/status-bar-types.js +25 -25
- package/lib/browser/status-bar/status-bar-view-model.d.ts +35 -35
- package/lib/browser/status-bar/status-bar-view-model.js +207 -207
- package/lib/browser/status-bar/status-bar.d.ts +34 -34
- package/lib/browser/status-bar/status-bar.js +181 -181
- package/lib/browser/storage-service.d.ts +35 -35
- package/lib/browser/storage-service.js +122 -122
- package/lib/browser/storage-service.spec.d.ts +1 -1
- package/lib/browser/storage-service.spec.js +68 -68
- package/lib/browser/styling-service.d.ts +29 -29
- package/lib/browser/styling-service.js +86 -86
- package/lib/browser/supported-encodings.d.ts +9 -9
- package/lib/browser/supported-encodings.js +263 -263
- package/lib/browser/test/jsdom.d.ts +8 -8
- package/lib/browser/test/jsdom.js +69 -69
- package/lib/browser/test/mock-connection-status-service.d.ts +5 -5
- package/lib/browser/test/mock-connection-status-service.js +32 -32
- package/lib/browser/test/mock-env-variables-server.d.ts +12 -12
- package/lib/browser/test/mock-env-variables-server.js +42 -42
- package/lib/browser/test/mock-opener-service.d.ts +8 -8
- package/lib/browser/test/mock-opener-service.js +35 -35
- package/lib/browser/test/mock-storage-service.d.ts +11 -11
- package/lib/browser/test/mock-storage-service.js +49 -49
- package/lib/browser/theming.d.ts +50 -50
- package/lib/browser/theming.js +203 -203
- package/lib/browser/tooltip-service.d.ts +34 -34
- package/lib/browser/tooltip-service.js +76 -76
- package/lib/browser/tree/fuzzy-search.d.ts +60 -60
- package/lib/browser/tree/fuzzy-search.js +75 -75
- package/lib/browser/tree/fuzzy-search.spec.d.ts +1 -1
- package/lib/browser/tree/fuzzy-search.spec.js +89 -89
- package/lib/browser/tree/index.d.ts +13 -13
- package/lib/browser/tree/index.js +31 -31
- package/lib/browser/tree/search-box-debounce.d.ts +33 -33
- package/lib/browser/tree/search-box-debounce.js +75 -75
- package/lib/browser/tree/search-box.d.ts +105 -105
- package/lib/browser/tree/search-box.js +265 -265
- package/lib/browser/tree/test/mock-selectable-tree-model.d.ts +14 -14
- package/lib/browser/tree/test/mock-selectable-tree-model.js +102 -102
- package/lib/browser/tree/test/mock-tree-model.d.ts +12 -12
- package/lib/browser/tree/test/mock-tree-model.js +129 -129
- package/lib/browser/tree/test/tree-test-container.d.ts +2 -2
- package/lib/browser/tree/test/tree-test-container.js +52 -52
- package/lib/browser/tree/tree-compression/compressed-tree-expansion-service.d.ts +8 -8
- package/lib/browser/tree/tree-compression/compressed-tree-expansion-service.js +59 -59
- package/lib/browser/tree/tree-compression/compressed-tree-model.d.ts +23 -23
- package/lib/browser/tree/tree-compression/compressed-tree-model.js +89 -89
- package/lib/browser/tree/tree-compression/compressed-tree-widget.d.ts +38 -38
- package/lib/browser/tree/tree-compression/compressed-tree-widget.js +187 -187
- package/lib/browser/tree/tree-compression/index.d.ts +4 -4
- package/lib/browser/tree/tree-compression/index.js +22 -22
- package/lib/browser/tree/tree-compression/tree-compression-service.d.ts +58 -58
- package/lib/browser/tree/tree-compression/tree-compression-service.js +98 -98
- package/lib/browser/tree/tree-consistency.spec.d.ts +1 -1
- package/lib/browser/tree/tree-consistency.spec.js +99 -99
- package/lib/browser/tree/tree-container.d.ts +51 -51
- package/lib/browser/tree/tree-container.js +115 -115
- package/lib/browser/tree/tree-container.spec.d.ts +1 -1
- package/lib/browser/tree/tree-container.spec.js +45 -45
- package/lib/browser/tree/tree-decorator.d.ts +143 -143
- package/lib/browser/tree/tree-decorator.js +115 -115
- package/lib/browser/tree/tree-decorator.spec.d.ts +1 -1
- package/lib/browser/tree/tree-decorator.spec.js +145 -145
- package/lib/browser/tree/tree-expansion.d.ts +65 -65
- package/lib/browser/tree/tree-expansion.js +124 -124
- package/lib/browser/tree/tree-expansion.spec.d.ts +1 -1
- package/lib/browser/tree/tree-expansion.spec.js +155 -155
- package/lib/browser/tree/tree-focus-service.d.ts +19 -19
- package/lib/browser/tree/tree-focus-service.js +53 -53
- package/lib/browser/tree/tree-iterator.d.ts +94 -94
- package/lib/browser/tree/tree-iterator.js +223 -223
- package/lib/browser/tree/tree-iterator.spec.d.ts +1 -1
- package/lib/browser/tree/tree-iterator.spec.js +150 -150
- package/lib/browser/tree/tree-label-provider.d.ts +8 -8
- package/lib/browser/tree/tree-label-provider.js +39 -39
- package/lib/browser/tree/tree-model.d.ts +202 -202
- package/lib/browser/tree/tree-model.js +395 -395
- package/lib/browser/tree/tree-navigation.d.ts +10 -10
- package/lib/browser/tree/tree-navigation.js +57 -57
- package/lib/browser/tree/tree-preference.d.ts +10 -10
- package/lib/browser/tree/tree-preference.js +46 -46
- package/lib/browser/tree/tree-search.d.ts +36 -36
- package/lib/browser/tree/tree-search.js +130 -130
- package/lib/browser/tree/tree-selectable.spec.d.ts +1 -1
- package/lib/browser/tree/tree-selectable.spec.js +146 -146
- package/lib/browser/tree/tree-selection-impl.d.ts +44 -44
- package/lib/browser/tree/tree-selection-impl.js +158 -158
- package/lib/browser/tree/tree-selection-state.d.ts +48 -48
- package/lib/browser/tree/tree-selection-state.js +227 -227
- package/lib/browser/tree/tree-selection-state.spec.d.ts +1 -1
- package/lib/browser/tree/tree-selection-state.spec.js +420 -420
- package/lib/browser/tree/tree-selection.d.ts +86 -86
- package/lib/browser/tree/tree-selection.js +86 -86
- package/lib/browser/tree/tree-view-welcome-widget.d.ts +45 -45
- package/lib/browser/tree/tree-view-welcome-widget.js +192 -192
- package/lib/browser/tree/tree-widget-selection.d.ts +11 -11
- package/lib/browser/tree/tree-widget-selection.js +46 -46
- package/lib/browser/tree/tree-widget.d.ts +559 -559
- package/lib/browser/tree/tree-widget.js +1352 -1352
- package/lib/browser/tree/tree.d.ts +180 -180
- package/lib/browser/tree/tree.js +288 -288
- package/lib/browser/tree/tree.spec.d.ts +1 -1
- package/lib/browser/tree/tree.spec.js +226 -226
- package/lib/browser/user-working-directory-provider.d.ts +23 -23
- package/lib/browser/user-working-directory-provider.js +76 -76
- package/lib/browser/view-container.d.ts +312 -312
- package/lib/browser/view-container.js +1458 -1458
- package/lib/browser/widget-decoration.d.ts +299 -299
- package/lib/browser/widget-decoration.js +140 -140
- package/lib/browser/widget-manager.d.ts +172 -172
- package/lib/browser/widget-manager.js +222 -222
- package/lib/browser/widget-manager.spec.d.ts +1 -1
- package/lib/browser/widget-manager.spec.js +91 -91
- package/lib/browser/widget-open-handler.d.ts +88 -88
- package/lib/browser/widget-open-handler.js +137 -137
- package/lib/browser/widgets/alert-message.d.ts +23 -23
- package/lib/browser/widgets/alert-message.js +38 -38
- package/lib/browser/widgets/enhanced-preview-widget.d.ts +6 -6
- package/lib/browser/widgets/enhanced-preview-widget.js +26 -26
- package/lib/browser/widgets/extractable-widget.d.ts +13 -13
- package/lib/browser/widgets/extractable-widget.js +26 -26
- package/lib/browser/widgets/index.d.ts +4 -4
- package/lib/browser/widgets/index.js +22 -22
- package/lib/browser/widgets/previewable-widget.d.ts +8 -8
- package/lib/browser/widgets/previewable-widget.js +30 -30
- package/lib/browser/widgets/react-renderer.d.ts +14 -14
- package/lib/browser/widgets/react-renderer.js +48 -48
- package/lib/browser/widgets/react-widget.d.ts +16 -16
- package/lib/browser/widgets/react-widget.js +46 -46
- package/lib/browser/widgets/select-component.d.ts +53 -53
- package/lib/browser/widgets/select-component.js +307 -307
- package/lib/browser/widgets/widget.d.ts +108 -108
- package/lib/browser/widgets/widget.js +367 -367
- package/lib/browser/window/browser-window-module.d.ts +3 -3
- package/lib/browser/window/browser-window-module.js +32 -32
- package/lib/browser/window/default-secondary-window-service.d.ts +25 -25
- package/lib/browser/window/default-secondary-window-service.js +187 -187
- package/lib/browser/window/default-window-service.d.ts +50 -50
- package/lib/browser/window/default-window-service.js +167 -167
- package/lib/browser/window/default-window-service.spec.d.ts +1 -1
- package/lib/browser/window/default-window-service.spec.js +80 -80
- package/lib/browser/window/secondary-window-service.d.ts +20 -20
- package/lib/browser/window/secondary-window-service.js +19 -19
- package/lib/browser/window/test/mock-window-service.d.ts +11 -11
- package/lib/browser/window/test/mock-window-service.js +34 -34
- package/lib/browser/window/window-service.d.ts +53 -53
- package/lib/browser/window/window-service.js +22 -22
- package/lib/browser/window/window-title-service.d.ts +31 -31
- package/lib/browser/window/window-title-service.js +112 -112
- package/lib/browser/window/window-title-updater.d.ts +19 -19
- package/lib/browser/window/window-title-updater.js +99 -99
- package/lib/browser/window-contribution.d.ts +14 -14
- package/lib/browser/window-contribution.js +62 -62
- package/lib/browser-only/frontend-only-application-module.d.ts +4 -4
- package/lib/browser-only/frontend-only-application-module.js +114 -114
- package/lib/browser-only/i18n/i18n-frontend-only-module.d.ts +3 -3
- package/lib/browser-only/i18n/i18n-frontend-only-module.js +34 -34
- package/lib/browser-only/logger-frontend-only-module.d.ts +2 -2
- package/lib/browser-only/logger-frontend-only-module.js +60 -60
- package/lib/browser-only/messaging/frontend-only-service-connection-provider.d.ts +13 -13
- package/lib/browser-only/messaging/frontend-only-service-connection-provider.js +51 -51
- package/lib/browser-only/messaging/messaging-frontend-only-module.d.ts +2 -2
- package/lib/browser-only/messaging/messaging-frontend-only-module.js +47 -47
- package/lib/browser-only/preload/frontend-only-preload-module.d.ts +3 -3
- package/lib/browser-only/preload/frontend-only-preload-module.js +51 -51
- package/lib/common/accessibility.d.ts +16 -16
- package/lib/common/accessibility.js +17 -17
- package/lib/common/application-error.d.ts +20 -20
- package/lib/common/application-error.js +62 -62
- package/lib/common/application-error.spec.d.ts +1 -1
- package/lib/common/application-error.spec.js +26 -26
- package/lib/common/application-protocol.d.ts +20 -20
- package/lib/common/application-protocol.js +20 -20
- package/lib/common/array-utils.d.ts +55 -55
- package/lib/common/array-utils.js +118 -118
- package/lib/common/buffer.d.ts +43 -43
- package/lib/common/buffer.js +205 -205
- package/lib/common/cancellation.d.ts +24 -24
- package/lib/common/cancellation.js +142 -142
- package/lib/common/char-code.d.ts +401 -401
- package/lib/common/char-code.js +22 -22
- package/lib/common/collections.d.ts +29 -29
- package/lib/common/collections.js +121 -121
- package/lib/common/color.d.ts +71 -71
- package/lib/common/color.js +43 -43
- package/lib/common/command.d.ts +232 -232
- package/lib/common/command.js +353 -353
- package/lib/common/command.spec.d.ts +1 -1
- package/lib/common/command.spec.js +173 -173
- package/lib/common/contribution-filter/contribution-filter-registry.d.ts +17 -17
- package/lib/common/contribution-filter/contribution-filter-registry.js +77 -77
- package/lib/common/contribution-filter/contribution-filter.d.ts +41 -41
- package/lib/common/contribution-filter/contribution-filter.js +20 -20
- package/lib/common/contribution-filter/filter.d.ts +6 -6
- package/lib/common/contribution-filter/filter.js +19 -19
- package/lib/common/contribution-filter/index.d.ts +3 -3
- package/lib/common/contribution-filter/index.js +21 -21
- package/lib/common/contribution-provider.d.ts +20 -20
- package/lib/common/contribution-provider.js +82 -82
- package/lib/common/disposable.d.ts +80 -80
- package/lib/common/disposable.js +173 -173
- package/lib/common/disposable.spec.d.ts +1 -1
- package/lib/common/disposable.spec.js +80 -80
- package/lib/common/encoding-service.d.ts +38 -38
- package/lib/common/encoding-service.js +302 -302
- package/lib/common/encodings.d.ts +7 -7
- package/lib/common/encodings.js +25 -25
- package/lib/common/env-variables/env-variables-protocol.d.ts +20 -20
- package/lib/common/env-variables/env-variables-protocol.js +20 -20
- package/lib/common/env-variables/index.d.ts +1 -1
- package/lib/common/env-variables/index.js +19 -19
- package/lib/common/event.d.ts +115 -115
- package/lib/common/event.js +385 -385
- package/lib/common/event.spec.d.ts +1 -1
- package/lib/common/event.spec.js +29 -29
- package/lib/common/file-uri.d.ts +14 -14
- package/lib/common/file-uri.js +62 -62
- package/lib/common/frontend-application-state.d.ts +15 -15
- package/lib/common/frontend-application-state.js +33 -33
- package/lib/common/glob.d.ts +50 -50
- package/lib/common/glob.js +591 -591
- package/lib/common/hash.d.ts +7 -7
- package/lib/common/hash.js +84 -84
- package/lib/common/i18n/localization-server.d.ts +6 -6
- package/lib/common/i18n/localization-server.js +20 -20
- package/lib/common/i18n/localization.d.ts +34 -34
- package/lib/common/i18n/localization.js +59 -59
- package/lib/common/index.d.ts +35 -35
- package/lib/common/index.js +56 -56
- package/lib/common/json-schema.d.ts +72 -72
- package/lib/common/json-schema.js +17 -17
- package/lib/common/key-store.d.ts +12 -12
- package/lib/common/key-store.js +20 -20
- package/lib/common/keybinding.d.ts +75 -75
- package/lib/common/keybinding.js +113 -113
- package/lib/common/keyboard/keyboard-layout-provider.d.ts +26 -26
- package/lib/common/keyboard/keyboard-layout-provider.js +22 -22
- package/lib/common/keys.d.ts +253 -253
- package/lib/common/keys.js +634 -634
- package/lib/common/label-protocol.d.ts +13 -13
- package/lib/common/label-protocol.js +17 -17
- package/lib/common/logger-protocol.d.ts +41 -41
- package/lib/common/logger-protocol.js +100 -100
- package/lib/common/logger-watcher.d.ts +9 -9
- package/lib/common/logger-watcher.js +49 -49
- package/lib/common/logger.d.ts +207 -207
- package/lib/common/logger.js +204 -204
- package/lib/common/logger.spec.d.ts +1 -1
- package/lib/common/logger.spec.js +41 -41
- package/lib/common/lsp-types.d.ts +9 -9
- package/lib/common/lsp-types.js +30 -30
- package/lib/common/markdown-rendering/icon-utilities.d.ts +7 -7
- package/lib/common/markdown-rendering/icon-utilities.js +32 -32
- package/lib/common/markdown-rendering/index.d.ts +2 -2
- package/lib/common/markdown-rendering/index.js +20 -20
- package/lib/common/markdown-rendering/markdown-string.d.ts +46 -46
- package/lib/common/markdown-rendering/markdown-string.js +126 -126
- package/lib/common/menu/action-menu-node.d.ts +19 -19
- package/lib/common/menu/action-menu-node.js +56 -56
- package/lib/common/menu/composite-menu-node.d.ts +46 -46
- package/lib/common/menu/composite-menu-node.js +93 -93
- package/lib/common/menu/composite-menu-node.spec.d.ts +1 -1
- package/lib/common/menu/composite-menu-node.spec.js +67 -67
- package/lib/common/menu/index.d.ts +5 -5
- package/lib/common/menu/index.js +23 -23
- package/lib/common/menu/menu-adapter.d.ts +35 -35
- package/lib/common/menu/menu-adapter.js +92 -92
- package/lib/common/menu/menu-model-registry.d.ts +145 -145
- package/lib/common/menu/menu-model-registry.js +275 -275
- package/lib/common/menu/menu-types.d.ts +148 -148
- package/lib/common/menu/menu-types.js +93 -93
- package/lib/common/menu/menu.spec.d.ts +1 -1
- package/lib/common/menu/menu.spec.js +96 -96
- package/lib/common/message-rpc/channel.d.ts +123 -123
- package/lib/common/message-rpc/channel.js +230 -230
- package/lib/common/message-rpc/channel.spec.d.ts +8 -8
- package/lib/common/message-rpc/channel.spec.js +79 -79
- package/lib/common/message-rpc/index.d.ts +3 -3
- package/lib/common/message-rpc/index.js +24 -24
- package/lib/common/message-rpc/message-buffer.d.ts +51 -51
- package/lib/common/message-rpc/message-buffer.js +59 -59
- package/lib/common/message-rpc/msg-pack-extension-manager.d.ts +22 -22
- package/lib/common/message-rpc/msg-pack-extension-manager.js +57 -57
- package/lib/common/message-rpc/rpc-message-encoder.d.ts +103 -103
- package/lib/common/message-rpc/rpc-message-encoder.js +113 -113
- package/lib/common/message-rpc/rpc-message-encoder.spec.d.ts +1 -1
- package/lib/common/message-rpc/rpc-message-encoder.spec.js +58 -58
- package/lib/common/message-rpc/rpc-protocol.d.ts +67 -67
- package/lib/common/message-rpc/rpc-protocol.js +209 -209
- package/lib/common/message-rpc/uint8-array-message-buffer.d.ts +52 -52
- package/lib/common/message-rpc/uint8-array-message-buffer.js +174 -174
- package/lib/common/message-rpc/uint8-array-message-buffer.spec.d.ts +1 -1
- package/lib/common/message-rpc/uint8-array-message-buffer.spec.js +38 -38
- package/lib/common/message-service-protocol.d.ts +110 -110
- package/lib/common/message-service-protocol.js +73 -73
- package/lib/common/message-service.d.ts +134 -134
- package/lib/common/message-service.js +151 -151
- package/lib/common/messaging/connection-error-handler.d.ts +28 -28
- package/lib/common/messaging/connection-error-handler.js +47 -47
- package/lib/common/messaging/connection-management.d.ts +24 -24
- package/lib/common/messaging/connection-management.js +37 -37
- package/lib/common/messaging/handler.d.ts +7 -7
- package/lib/common/messaging/handler.js +20 -20
- package/lib/common/messaging/index.d.ts +3 -3
- package/lib/common/messaging/index.js +21 -21
- package/lib/common/messaging/proxy-factory.d.ts +180 -180
- package/lib/common/messaging/proxy-factory.js +280 -280
- package/lib/common/messaging/proxy-factory.spec.d.ts +1 -1
- package/lib/common/messaging/proxy-factory.spec.js +99 -99
- package/lib/common/messaging/socket-write-buffer.d.ts +10 -10
- package/lib/common/messaging/socket-write-buffer.js +49 -49
- package/lib/common/messaging/web-socket-channel.d.ts +19 -19
- package/lib/common/messaging/web-socket-channel.js +64 -64
- package/lib/common/nls.d.ts +15 -15
- package/lib/common/nls.js +135 -135
- package/lib/common/numbers.d.ts +5 -5
- package/lib/common/numbers.js +23 -23
- package/lib/common/objects.d.ts +8 -8
- package/lib/common/objects.js +118 -118
- package/lib/common/objects.spec.d.ts +1 -1
- package/lib/common/objects.spec.js +101 -101
- package/lib/common/os.d.ts +31 -31
- package/lib/common/os.js +73 -73
- package/lib/common/path.d.ts +93 -93
- package/lib/common/path.js +307 -307
- package/lib/common/path.spec.d.ts +1 -1
- package/lib/common/path.spec.js +350 -350
- package/lib/common/paths.d.ts +24 -24
- package/lib/common/paths.js +226 -226
- package/lib/common/performance/index.d.ts +3 -3
- package/lib/common/performance/index.js +21 -21
- package/lib/common/performance/measurement-protocol.d.ts +64 -64
- package/lib/common/performance/measurement-protocol.js +70 -70
- package/lib/common/performance/measurement.d.ts +111 -111
- package/lib/common/performance/measurement.js +17 -17
- package/lib/common/performance/stopwatch.d.ts +75 -75
- package/lib/common/performance/stopwatch.js +148 -148
- package/lib/common/preferences/preference-schema.d.ts +52 -52
- package/lib/common/preferences/preference-schema.js +58 -58
- package/lib/common/preferences/preference-scope.d.ts +19 -19
- package/lib/common/preferences/preference-scope.js +71 -71
- package/lib/common/preferences/preference-scope.spec.d.ts +1 -1
- package/lib/common/preferences/preference-scope.spec.js +44 -44
- package/lib/common/prioritizeable.d.ts +16 -16
- package/lib/common/prioritizeable.js +55 -55
- package/lib/common/progress-service-protocol.d.ts +13 -13
- package/lib/common/progress-service-protocol.js +19 -19
- package/lib/common/progress-service.d.ts +13 -13
- package/lib/common/progress-service.js +88 -88
- package/lib/common/promise-util.d.ts +51 -51
- package/lib/common/promise-util.js +132 -132
- package/lib/common/promise-util.spec.d.ts +1 -1
- package/lib/common/promise-util.spec.js +80 -80
- package/lib/common/quick-pick-service.d.ts +270 -270
- package/lib/common/quick-pick-service.js +133 -133
- package/lib/common/reference.d.ts +95 -95
- package/lib/common/reference.js +207 -207
- package/lib/common/reference.spec.d.ts +1 -1
- package/lib/common/reference.spec.js +125 -125
- package/lib/common/resource.d.ts +207 -207
- package/lib/common/resource.js +322 -322
- package/lib/common/selection-command-handler.d.ts +24 -24
- package/lib/common/selection-command-handler.js +83 -83
- package/lib/common/selection-service.d.ts +18 -18
- package/lib/common/selection-service.js +43 -43
- package/lib/common/selection-service.spec.d.ts +1 -1
- package/lib/common/selection-service.spec.js +38 -38
- package/lib/common/selection.d.ts +9 -9
- package/lib/common/selection.js +47 -47
- package/lib/common/severity.d.ts +15 -15
- package/lib/common/severity.js +113 -113
- package/lib/common/stream.d.ts +176 -176
- package/lib/common/stream.js +474 -474
- package/lib/common/strings.d.ts +24 -24
- package/lib/common/strings.js +215 -215
- package/lib/common/telemetry.d.ts +19 -19
- package/lib/common/telemetry.js +24 -24
- package/lib/common/ternary-search-tree.d.ts +50 -50
- package/lib/common/ternary-search-tree.js +381 -381
- package/lib/common/test/expect.d.ts +2 -2
- package/lib/common/test/expect.js +34 -34
- package/lib/common/test/mock-logger.d.ts +27 -27
- package/lib/common/test/mock-logger.js +99 -99
- package/lib/common/test/mock-menu.d.ts +7 -7
- package/lib/common/test/mock-menu.js +34 -34
- package/lib/common/test/mock-resource-provider.d.ts +6 -6
- package/lib/common/test/mock-resource-provider.js +36 -36
- package/lib/common/theme.d.ts +40 -40
- package/lib/common/theme.js +22 -22
- package/lib/common/types.d.ts +58 -58
- package/lib/common/types.js +126 -126
- package/lib/common/types.spec.d.ts +1 -1
- package/lib/common/types.spec.js +73 -73
- package/lib/common/uri-command-handler.d.ts +53 -53
- package/lib/common/uri-command-handler.js +99 -99
- package/lib/common/uri-command-handler.spec.d.ts +1 -1
- package/lib/common/uri-command-handler.spec.js +80 -80
- package/lib/common/uri.d.ts +87 -87
- package/lib/common/uri.js +236 -236
- package/lib/common/uri.spec.d.ts +1 -1
- package/lib/common/uri.spec.js +230 -230
- package/lib/common/uuid.d.ts +8 -8
- package/lib/common/uuid.js +44 -44
- package/lib/common/version.d.ts +1 -1
- package/lib/common/version.js +19 -19
- package/lib/common/view-column.d.ts +17 -17
- package/lib/common/view-column.js +36 -36
- package/lib/common/window.d.ts +16 -16
- package/lib/common/window.js +22 -22
- package/lib/electron-browser/electron-clipboard-service.d.ts +5 -5
- package/lib/electron-browser/electron-clipboard-service.js +33 -33
- package/lib/electron-browser/keyboard/electron-keyboard-layout-change-notifier.d.ts +11 -11
- package/lib/electron-browser/keyboard/electron-keyboard-layout-change-notifier.js +46 -46
- package/lib/electron-browser/keyboard/electron-keyboard-module.d.ts +3 -3
- package/lib/electron-browser/keyboard/electron-keyboard-module.js +26 -26
- package/lib/electron-browser/menu/electron-context-menu-renderer.d.ts +30 -30
- package/lib/electron-browser/menu/electron-context-menu-renderer.js +128 -128
- package/lib/electron-browser/menu/electron-main-menu-factory.d.ts +59 -59
- package/lib/electron-browser/menu/electron-main-menu-factory.js +274 -274
- package/lib/electron-browser/menu/electron-menu-contribution.d.ts +78 -78
- package/lib/electron-browser/menu/electron-menu-contribution.js +471 -471
- package/lib/electron-browser/menu/electron-menu-module.d.ts +3 -3
- package/lib/electron-browser/menu/electron-menu-module.js +39 -39
- package/lib/electron-browser/messaging/electron-frontend-id-provider.d.ts +4 -4
- package/lib/electron-browser/messaging/electron-frontend-id-provider.js +29 -29
- package/lib/electron-browser/messaging/electron-ipc-connection-source.d.ts +23 -23
- package/lib/electron-browser/messaging/electron-ipc-connection-source.js +59 -59
- package/lib/electron-browser/messaging/electron-local-ws-connection-source.d.ts +6 -6
- package/lib/electron-browser/messaging/electron-local-ws-connection-source.js +49 -49
- package/lib/electron-browser/messaging/electron-messaging-frontend-module.d.ts +2 -2
- package/lib/electron-browser/messaging/electron-messaging-frontend-module.js +75 -75
- package/lib/electron-browser/messaging/electron-ws-connection-source.d.ts +11 -11
- package/lib/electron-browser/messaging/electron-ws-connection-source.js +42 -42
- package/lib/electron-browser/preload.d.ts +1 -1
- package/lib/electron-browser/preload.js +201 -201
- package/lib/electron-browser/request/electron-browser-request-module.d.ts +18 -18
- package/lib/electron-browser/request/electron-browser-request-module.js +26 -26
- package/lib/electron-browser/token/electron-token-frontend-module.d.ts +3 -3
- package/lib/electron-browser/token/electron-token-frontend-module.js +22 -22
- package/lib/electron-browser/window/electron-frontend-application-state.d.ts +4 -4
- package/lib/electron-browser/window/electron-frontend-application-state.js +31 -31
- package/lib/electron-browser/window/electron-secondary-window-service.d.ts +7 -7
- package/lib/electron-browser/window/electron-secondary-window-service.js +38 -38
- package/lib/electron-browser/window/electron-window-module.d.ts +3 -3
- package/lib/electron-browser/window/electron-window-module.js +39 -39
- package/lib/electron-browser/window/electron-window-preferences.d.ts +18 -18
- package/lib/electron-browser/window/electron-window-preferences.js +72 -72
- package/lib/electron-browser/window/electron-window-service.d.ts +31 -31
- package/lib/electron-browser/window/electron-window-service.js +116 -116
- package/lib/electron-common/electron-api.d.ts +104 -104
- package/lib/electron-common/electron-api.js +56 -56
- package/lib/electron-common/electron-main-window-service.d.ts +7 -7
- package/lib/electron-common/electron-main-window-service.js +20 -20
- package/lib/electron-common/electron-token.d.ts +10 -10
- package/lib/electron-common/electron-token.js +27 -27
- package/lib/electron-main/electron-api-main.d.ts +22 -22
- package/lib/electron-main/electron-api-main.js +294 -294
- package/lib/electron-main/electron-main-application-module.d.ts +3 -3
- package/lib/electron-main/electron-main-application-module.js +56 -56
- package/lib/electron-main/electron-main-application.d.ts +159 -159
- package/lib/electron-main/electron-main-application.js +603 -603
- package/lib/electron-main/electron-main-constants.d.ts +6 -6
- package/lib/electron-main/electron-main-constants.js +19 -19
- package/lib/electron-main/electron-main-window-service-impl.d.ts +8 -8
- package/lib/electron-main/electron-main-window-service-impl.js +46 -46
- package/lib/electron-main/electron-security-token-service.d.ts +5 -5
- package/lib/electron-main/electron-security-token-service.js +41 -41
- package/lib/electron-main/event-utils.d.ts +8 -8
- package/lib/electron-main/event-utils.js +33 -33
- package/lib/electron-main/messaging/electron-connection-handler.d.ts +5 -5
- package/lib/electron-main/messaging/electron-connection-handler.js +23 -23
- package/lib/electron-main/messaging/electron-messaging-contribution.d.ts +41 -41
- package/lib/electron-main/messaging/electron-messaging-contribution.js +142 -142
- package/lib/electron-main/messaging/electron-messaging-service.d.ts +17 -17
- package/lib/electron-main/messaging/electron-messaging-service.js +22 -22
- package/lib/electron-main/theia-electron-window.d.ts +53 -53
- package/lib/electron-main/theia-electron-window.js +186 -186
- package/lib/electron-node/hosting/electron-backend-hosting-module.d.ts +3 -3
- package/lib/electron-node/hosting/electron-backend-hosting-module.js +24 -24
- package/lib/electron-node/hosting/electron-ws-origin-validator.d.ts +8 -8
- package/lib/electron-node/hosting/electron-ws-origin-validator.js +40 -40
- package/lib/electron-node/keyboard/electron-backend-keyboard-module.d.ts +3 -3
- package/lib/electron-node/keyboard/electron-backend-keyboard-module.js +26 -26
- package/lib/electron-node/keyboard/electron-keyboard-layout-provider.d.ts +5 -5
- package/lib/electron-node/keyboard/electron-keyboard-layout-provider.js +36 -36
- package/lib/electron-node/request/electron-backend-request-module.d.ts +18 -18
- package/lib/electron-node/request/electron-backend-request-module.js +23 -23
- package/lib/electron-node/request/electron-backend-request-service.d.ts +22 -22
- package/lib/electron-node/request/electron-backend-request-service.js +81 -81
- package/lib/electron-node/token/electron-token-backend-contribution.d.ts +15 -15
- package/lib/electron-node/token/electron-token-backend-contribution.js +50 -50
- package/lib/electron-node/token/electron-token-backend-module.d.ts +3 -3
- package/lib/electron-node/token/electron-token-backend-module.js +28 -28
- package/lib/electron-node/token/electron-token-validator.d.ts +29 -29
- package/lib/electron-node/token/electron-token-validator.js +95 -95
- package/lib/node/application-server.d.ts +10 -10
- package/lib/node/application-server.js +55 -55
- package/lib/node/backend-application-config-provider.d.ts +7 -7
- package/lib/node/backend-application-config-provider.js +45 -45
- package/lib/node/backend-application-config-provider.spec.d.ts +1 -1
- package/lib/node/backend-application-config-provider.spec.js +28 -28
- package/lib/node/backend-application-module.d.ts +2 -2
- package/lib/node/backend-application-module.js +108 -108
- package/lib/node/backend-application.d.ts +108 -108
- package/lib/node/backend-application.js +290 -290
- package/lib/node/backend-remote-service.d.ts +3 -3
- package/lib/node/backend-remote-service.js +29 -29
- package/lib/node/cli.d.ts +17 -17
- package/lib/node/cli.js +59 -59
- package/lib/node/cli.spec.d.ts +1 -1
- package/lib/node/cli.spec.js +89 -89
- package/lib/node/console-logger-server.d.ts +15 -15
- package/lib/node/console-logger-server.js +82 -82
- package/lib/node/console-logger-server.spec.d.ts +1 -1
- package/lib/node/console-logger-server.spec.js +61 -61
- package/lib/node/debug.d.ts +1 -1
- package/lib/node/debug.js +28 -28
- package/lib/node/dynamic-require.d.ts +7 -7
- package/lib/node/dynamic-require.js +56 -56
- package/lib/node/env-variables/env-variables-server.d.ts +23 -23
- package/lib/node/env-variables/env-variables-server.js +117 -117
- package/lib/node/env-variables/index.d.ts +1 -1
- package/lib/node/env-variables/index.js +19 -19
- package/lib/node/environment-utils.d.ts +21 -21
- package/lib/node/environment-utils.js +72 -72
- package/lib/node/environment-utils.spec.d.ts +1 -1
- package/lib/node/environment-utils.spec.js +77 -77
- package/lib/node/file-uri.spec.d.ts +1 -1
- package/lib/node/file-uri.spec.js +66 -66
- package/lib/node/filesystem-locking.d.ts +23 -23
- package/lib/node/filesystem-locking.js +65 -65
- package/lib/node/hosting/backend-application-hosts.d.ts +23 -23
- package/lib/node/hosting/backend-application-hosts.js +69 -69
- package/lib/node/hosting/backend-hosting-module.d.ts +3 -3
- package/lib/node/hosting/backend-hosting-module.js +26 -26
- package/lib/node/hosting/ws-origin-validator.d.ts +8 -8
- package/lib/node/hosting/ws-origin-validator.js +39 -39
- package/lib/node/i18n/i18n-backend-module.d.ts +3 -3
- package/lib/node/i18n/i18n-backend-module.js +38 -38
- package/lib/node/i18n/localization-contribution.d.ts +18 -18
- package/lib/node/i18n/localization-contribution.js +108 -108
- package/lib/node/i18n/localization-provider.d.ts +23 -23
- package/lib/node/i18n/localization-provider.js +109 -109
- package/lib/node/i18n/localization-server.d.ts +14 -14
- package/lib/node/i18n/localization-server.js +54 -54
- package/lib/node/i18n/theia-localization-contribution.d.ts +4 -4
- package/lib/node/i18n/theia-localization-contribution.js +40 -40
- package/lib/node/index.d.ts +6 -6
- package/lib/node/index.js +26 -26
- package/lib/node/key-store-server.d.ts +27 -27
- package/lib/node/key-store-server.js +145 -145
- package/lib/node/logger-backend-module.d.ts +9 -9
- package/lib/node/logger-backend-module.js +78 -78
- package/lib/node/logger-cli-contribution.d.ts +33 -33
- package/lib/node/logger-cli-contribution.js +147 -147
- package/lib/node/logger-cli-contribution.spec.d.ts +1 -1
- package/lib/node/logger-cli-contribution.spec.js +207 -207
- package/lib/node/main.d.ts +7 -7
- package/lib/node/main.js +29 -29
- package/lib/node/messaging/binary-message-pipe.d.ts +44 -44
- package/lib/node/messaging/binary-message-pipe.js +151 -151
- package/lib/node/messaging/connection-container-module.d.ts +53 -53
- package/lib/node/messaging/connection-container-module.js +81 -81
- package/lib/node/messaging/default-messaging-service.d.ts +28 -28
- package/lib/node/messaging/default-messaging-service.js +131 -131
- package/lib/node/messaging/frontend-connection-service.d.ts +6 -6
- package/lib/node/messaging/frontend-connection-service.js +18 -18
- package/lib/node/messaging/index.d.ts +3 -3
- package/lib/node/messaging/index.js +21 -21
- package/lib/node/messaging/ipc-bootstrap.d.ts +1 -1
- package/lib/node/messaging/ipc-bootstrap.js +24 -24
- package/lib/node/messaging/ipc-channel.d.ts +17 -17
- package/lib/node/messaging/ipc-channel.js +69 -69
- package/lib/node/messaging/ipc-connection-provider.d.ts +21 -21
- package/lib/node/messaging/ipc-connection-provider.js +92 -92
- package/lib/node/messaging/ipc-protocol.d.ts +19 -19
- package/lib/node/messaging/ipc-protocol.js +68 -68
- package/lib/node/messaging/messaging-backend-module.d.ts +2 -2
- package/lib/node/messaging/messaging-backend-module.js +52 -52
- package/lib/node/messaging/messaging-listeners.d.ts +27 -27
- package/lib/node/messaging/messaging-listeners.js +45 -45
- package/lib/node/messaging/messaging-service.d.ts +28 -28
- package/lib/node/messaging/messaging-service.js +24 -24
- package/lib/node/messaging/test/test-web-socket-channel.d.ts +12 -12
- package/lib/node/messaging/test/test-web-socket-channel.js +54 -54
- package/lib/node/messaging/websocket-endpoint.d.ts +20 -20
- package/lib/node/messaging/websocket-endpoint.js +80 -80
- package/lib/node/messaging/websocket-frontend-connection-service.d.ts +30 -30
- package/lib/node/messaging/websocket-frontend-connection-service.js +169 -169
- package/lib/node/os-backend-provider.d.ts +4 -4
- package/lib/node/os-backend-provider.js +30 -30
- package/lib/node/performance/index.d.ts +2 -2
- package/lib/node/performance/index.js +20 -20
- package/lib/node/performance/measurement-backend-bindings.d.ts +19 -19
- package/lib/node/performance/measurement-backend-bindings.js +30 -30
- package/lib/node/performance/node-stopwatch.d.ts +20 -20
- package/lib/node/performance/node-stopwatch.js +43 -43
- package/lib/node/process-utils.d.ts +13 -13
- package/lib/node/process-utils.js +105 -105
- package/lib/node/process-utils.spec.d.ts +1 -1
- package/lib/node/process-utils.spec.js +44 -44
- package/lib/node/request/backend-request-facade.d.ts +22 -22
- package/lib/node/request/backend-request-facade.js +41 -41
- package/lib/node/request/backend-request-module.d.ts +18 -18
- package/lib/node/request/backend-request-module.js +24 -24
- package/lib/node/request/proxy-cli-contribution.d.ts +27 -27
- package/lib/node/request/proxy-cli-contribution.js +64 -64
- package/lib/node/ws-request-validators.d.ts +25 -25
- package/lib/node/ws-request-validators.js +52 -52
- package/package.json +6 -6
- package/shared/ajv/index.d.ts +2 -2
- package/shared/markdown-it.d.ts +2 -2
- package/shared/markdown-it.js +1 -1
- package/shared/reflect-metadata/index.d.ts +1 -1
- package/shared/reflect-metadata/index.js +1 -1
- package/shared/vscode-languageserver-types/index.d.ts +1 -1
- package/src/browser/about-dialog.tsx +137 -137
- package/src/browser/authentication-service.ts +456 -456
- package/src/browser/breadcrumbs/breadcrumb-popup-container.ts +101 -101
- package/src/browser/breadcrumbs/breadcrumb-renderer.tsx +41 -41
- package/src/browser/breadcrumbs/breadcrumbs-constants.ts +79 -79
- package/src/browser/breadcrumbs/breadcrumbs-renderer.tsx +185 -185
- package/src/browser/breadcrumbs/breadcrumbs-service.ts +108 -108
- package/src/browser/breadcrumbs/index.ts +21 -21
- package/src/browser/browser-clipboard-service.ts +122 -122
- package/src/browser/browser.ts +225 -225
- package/src/browser/clipboard-service.ts +23 -23
- package/src/browser/color-application-contribution.ts +110 -110
- package/src/browser/color-registry.ts +60 -60
- package/src/browser/command-open-handler.ts +54 -54
- package/src/browser/common-frontend-contribution.ts +2569 -2569
- package/src/browser/common-styling-participants.ts +361 -361
- package/src/browser/connection-status-service.spec.ts +200 -200
- package/src/browser/connection-status-service.ts +216 -216
- package/src/browser/context-key-service.ts +142 -142
- package/src/browser/context-menu-renderer.ts +124 -124
- package/src/browser/core-preferences.ts +334 -334
- package/src/browser/credentials-service.ts +106 -106
- package/src/browser/decoration-style.ts +65 -65
- package/src/browser/decorations-service.ts +209 -209
- package/src/browser/dialogs/react-dialog.tsx +56 -56
- package/src/browser/dialogs.ts +534 -534
- package/src/browser/diff-uris.ts +117 -117
- package/src/browser/encoding-registry.ts +97 -97
- package/src/browser/endpoint.spec.ts +148 -148
- package/src/browser/endpoint.ts +136 -136
- package/src/browser/external-uri-service.ts +79 -79
- package/src/browser/file-icons-js.d.ts +20 -20
- package/src/browser/frontend-application-bindings.ts +62 -62
- package/src/browser/frontend-application-config-provider.spec.ts +45 -45
- package/src/browser/frontend-application-config-provider.ts +50 -50
- package/src/browser/frontend-application-contribution.ts +110 -110
- package/src/browser/frontend-application-module.ts +460 -460
- package/src/browser/frontend-application-state.ts +74 -74
- package/src/browser/frontend-application.ts +326 -326
- package/src/browser/hover-service.ts +218 -218
- package/src/browser/http-open-handler.ts +47 -47
- package/src/browser/i18n/i18n-frontend-module.ts +27 -27
- package/src/browser/i18n/language-quick-pick-service.ts +130 -130
- package/src/browser/icon-registry.ts +87 -87
- package/src/browser/icon-theme-contribution.ts +64 -64
- package/src/browser/icon-theme-service.ts +217 -217
- package/src/browser/icons/CollapseAll.svg +7 -7
- package/src/browser/icons/CollapseAll_inverse.svg +7 -7
- package/src/browser/icons/Refresh.svg +7 -7
- package/src/browser/icons/Refresh_inverse.svg +7 -7
- package/src/browser/icons/add-inverse.svg +4 -4
- package/src/browser/icons/add.svg +4 -4
- package/src/browser/icons/arrow-down-bright.svg +6 -6
- package/src/browser/icons/arrow-down-dark.svg +6 -6
- package/src/browser/icons/arrow-up-bright.svg +6 -6
- package/src/browser/icons/arrow-up-dark.svg +6 -6
- package/src/browser/icons/case-sensitive-dark.svg +16 -16
- package/src/browser/icons/case-sensitive.svg +16 -16
- package/src/browser/icons/chevron-right-dark.svg +5 -5
- package/src/browser/icons/chevron-right-light.svg +6 -6
- package/src/browser/icons/circle-bright.svg +7 -7
- package/src/browser/icons/circle-dark.svg +7 -7
- package/src/browser/icons/clear-search-results-dark.svg +7 -7
- package/src/browser/icons/clear-search-results.svg +7 -7
- package/src/browser/icons/close-all-bright.svg +7 -7
- package/src/browser/icons/close-all-dark.svg +7 -7
- package/src/browser/icons/close-bright.svg +7 -7
- package/src/browser/icons/close-dark.svg +7 -7
- package/src/browser/icons/collapse.svg +4 -4
- package/src/browser/icons/edit-json-dark.svg +6 -6
- package/src/browser/icons/edit-json.svg +6 -6
- package/src/browser/icons/expand.svg +4 -4
- package/src/browser/icons/loading-dark.svg +6 -6
- package/src/browser/icons/loading-light.svg +6 -6
- package/src/browser/icons/open-change-bright.svg +3 -3
- package/src/browser/icons/open-change-dark.svg +4 -4
- package/src/browser/icons/open-file-bright.svg +4 -4
- package/src/browser/icons/open-file-dark.svg +4 -4
- package/src/browser/icons/preview-bright.svg +3 -3
- package/src/browser/icons/preview-dark.svg +3 -3
- package/src/browser/icons/regex-dark.svg +10 -10
- package/src/browser/icons/regex.svg +10 -10
- package/src/browser/icons/remove-all-inverse.svg +4 -4
- package/src/browser/icons/remove-all.svg +4 -4
- package/src/browser/icons/replace-all-inverse.svg +13 -13
- package/src/browser/icons/replace-all.svg +13 -13
- package/src/browser/icons/replace-inverse.svg +15 -15
- package/src/browser/icons/replace.svg +15 -15
- package/src/browser/icons/whole-word-dark.svg +19 -19
- package/src/browser/icons/whole-word.svg +19 -19
- package/src/browser/index.ts +47 -47
- package/src/browser/json-schema-store.ts +127 -127
- package/src/browser/keybinding.spec.ts +553 -553
- package/src/browser/keybinding.ts +759 -759
- package/src/browser/keyboard/browser-keyboard-frontend-contribution.ts +108 -108
- package/src/browser/keyboard/browser-keyboard-layout-provider.spec.ts +171 -171
- package/src/browser/keyboard/browser-keyboard-layout-provider.ts +469 -469
- package/src/browser/keyboard/browser-keyboard-module.ts +30 -30
- package/src/browser/keyboard/index.ts +20 -20
- package/src/browser/keyboard/keyboard-layout-service.spec.ts +121 -121
- package/src/browser/keyboard/keyboard-layout-service.ts +455 -455
- package/src/browser/keyboard/keys.spec.ts +258 -258
- package/src/browser/keyboard/keys.ts +20 -20
- package/src/browser/keys.ts +21 -21
- package/src/browser/label-parser.spec.ts +165 -165
- package/src/browser/label-parser.ts +108 -108
- package/src/browser/label-provider.spec.ts +62 -62
- package/src/browser/label-provider.ts +385 -385
- package/src/browser/language-icon-provider.ts +55 -55
- package/src/browser/language-service.ts +77 -77
- package/src/browser/logger-frontend-module.ts +65 -65
- package/src/browser/markdown-rendering/markdown-renderer.ts +98 -98
- package/src/browser/menu/browser-context-menu-renderer.ts +48 -48
- package/src/browser/menu/browser-menu-module.ts +28 -28
- package/src/browser/menu/browser-menu-plugin.ts +484 -484
- package/src/browser/menu/context-menu-context.ts +41 -41
- package/src/browser/messaging/connection-source.ts +26 -26
- package/src/browser/messaging/frontend-id-provider.ts +37 -37
- package/src/browser/messaging/index.ts +17 -17
- package/src/browser/messaging/messaging-frontend-module.ts +41 -41
- package/src/browser/messaging/service-connection-provider.ts +126 -126
- package/src/browser/messaging/ws-connection-provider.ts +48 -48
- package/src/browser/messaging/ws-connection-source.ts +230 -230
- package/src/browser/mime-service.ts +30 -30
- package/src/browser/navigatable-types.ts +81 -81
- package/src/browser/navigatable.ts +39 -39
- package/src/browser/opener-service.spec.ts +49 -49
- package/src/browser/opener-service.ts +146 -146
- package/src/browser/performance/frontend-stopwatch.ts +65 -65
- package/src/browser/performance/index.ts +18 -18
- package/src/browser/performance/measurement-frontend-bindings.ts +31 -31
- package/src/browser/preferences/index.ts +23 -23
- package/src/browser/preferences/injectable-preference-proxy.ts +283 -283
- package/src/browser/preferences/preference-configurations.ts +82 -82
- package/src/browser/preferences/preference-contribution.ts +436 -436
- package/src/browser/preferences/preference-language-override-service.ts +111 -111
- package/src/browser/preferences/preference-provider.spec.ts +36 -36
- package/src/browser/preferences/preference-provider.ts +277 -277
- package/src/browser/preferences/preference-proxy.spec.ts +367 -367
- package/src/browser/preferences/preference-proxy.ts +367 -367
- package/src/browser/preferences/preference-schema-provider.spec.ts +130 -130
- package/src/browser/preferences/preference-scope.ts +18 -18
- package/src/browser/preferences/preference-service.spec.ts +613 -613
- package/src/browser/preferences/preference-service.ts +594 -594
- package/src/browser/preferences/preference-validation-service.spec.ts +334 -334
- package/src/browser/preferences/preference-validation-service.ts +358 -358
- package/src/browser/preferences/test/index.ts +19 -19
- package/src/browser/preferences/test/mock-preference-provider.ts +50 -50
- package/src/browser/preferences/test/mock-preference-proxy.ts +48 -48
- package/src/browser/preferences/test/mock-preference-service.ts +63 -63
- package/src/browser/preload/i18n-preload-contribution.ts +50 -50
- package/src/browser/preload/os-preload-contribution.ts +37 -37
- package/src/browser/preload/preload-module.ts +45 -45
- package/src/browser/preload/preloader.ts +37 -37
- package/src/browser/preload/theme-preload-contribution.ts +31 -31
- package/src/browser/progress-bar-factory.ts +29 -29
- package/src/browser/progress-bar.ts +76 -76
- package/src/browser/progress-client.ts +53 -53
- package/src/browser/progress-location-service.spec.ts +50 -50
- package/src/browser/progress-location-service.ts +96 -96
- package/src/browser/progress-status-bar-item.ts +83 -83
- package/src/browser/quick-input/index.ts +23 -23
- package/src/browser/quick-input/quick-access.ts +75 -75
- package/src/browser/quick-input/quick-command-frontend-contribution.ts +89 -89
- package/src/browser/quick-input/quick-command-service.ts +246 -246
- package/src/browser/quick-input/quick-help-service.ts +87 -87
- package/src/browser/quick-input/quick-input-frontend-contribution.ts +33 -33
- package/src/browser/quick-input/quick-input-service.spec.ts +176 -176
- package/src/browser/quick-input/quick-input-service.ts +17 -17
- package/src/browser/quick-input/quick-pick-service-impl.ts +69 -69
- package/src/browser/quick-input/quick-view-service.ts +83 -83
- package/src/browser/request/browser-request-module.ts +23 -23
- package/src/browser/request/browser-request-service.ts +172 -172
- package/src/browser/resource-context-key.ts +77 -77
- package/src/browser/save-resource-service.ts +60 -60
- package/src/browser/saveable.ts +365 -365
- package/src/browser/secondary-window-handler.ts +211 -211
- package/src/browser/shell/additional-views-menu-widget.tsx +71 -71
- package/src/browser/shell/application-shell-mouse-tracker.ts +103 -103
- package/src/browser/shell/application-shell.ts +2261 -2261
- package/src/browser/shell/current-widget-command-adapter.ts +57 -57
- package/src/browser/shell/index.ts +23 -23
- package/src/browser/shell/shell-layout-restorer.ts +399 -399
- package/src/browser/shell/side-panel-handler.ts +793 -793
- package/src/browser/shell/side-panel-toolbar.ts +111 -111
- package/src/browser/shell/sidebar-bottom-menu-widget.tsx +39 -39
- package/src/browser/shell/sidebar-menu-widget.tsx +140 -140
- package/src/browser/shell/sidebar-top-menu-widget.tsx +26 -26
- package/src/browser/shell/split-panels.ts +190 -190
- package/src/browser/shell/tab-bar-decorator.ts +106 -106
- package/src/browser/shell/tab-bar-toolbar/index.ts +19 -19
- package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar-menu-adapters.ts +31 -31
- package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar-registry.ts +256 -256
- package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar-types.ts +207 -207
- package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar.spec.ts +62 -62
- package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar.tsx +428 -428
- package/src/browser/shell/tab-bars.spec.ts +63 -63
- package/src/browser/shell/tab-bars.ts +1485 -1485
- package/src/browser/shell/theia-dock-panel.ts +255 -255
- package/src/browser/shell/view-contribution.ts +177 -177
- package/src/browser/source-tree/index.ts +19 -19
- package/src/browser/source-tree/source-tree-widget.tsx +107 -107
- package/src/browser/source-tree/source-tree.ts +146 -146
- package/src/browser/source-tree/tree-source.ts +73 -73
- package/src/browser/status-bar/index.ts +29 -29
- package/src/browser/status-bar/status-bar-types.ts +97 -97
- package/src/browser/status-bar/status-bar-view-model.ts +209 -209
- package/src/browser/status-bar/status-bar.tsx +189 -189
- package/src/browser/storage-service.spec.ts +76 -76
- package/src/browser/storage-service.ts +129 -129
- package/src/browser/style/about.css +36 -36
- package/src/browser/style/alert-messages.css +62 -62
- package/src/browser/style/ansi.css +88 -88
- package/src/browser/style/breadcrumbs.css +130 -130
- package/src/browser/style/dialog.css +126 -126
- package/src/browser/style/dockpanel.css +76 -76
- package/src/browser/style/hover-service.css +101 -101
- package/src/browser/style/icons.css +61 -61
- package/src/browser/style/index.css +351 -351
- package/src/browser/style/materialcolors.css +278 -278
- package/src/browser/style/menus.css +230 -230
- package/src/browser/style/notification.css +39 -39
- package/src/browser/style/os.css +87 -87
- package/src/browser/style/progress-bar.css +43 -43
- package/src/browser/style/quick-title-bar.css +45 -45
- package/src/browser/style/scrollbars.css +172 -172
- package/src/browser/style/search-box.css +123 -123
- package/src/browser/style/select-component.css +107 -107
- package/src/browser/style/sidepanel.css +364 -364
- package/src/browser/style/status-bar.css +127 -127
- package/src/browser/style/tabs.css +647 -647
- package/src/browser/style/tooltip.css +28 -28
- package/src/browser/style/tree-decorators.css +81 -81
- package/src/browser/style/tree.css +232 -232
- package/src/browser/style/view-container.css +194 -194
- package/src/browser/style/widget.css +19 -19
- package/src/browser/styling-service.ts +96 -96
- package/src/browser/supported-encodings.ts +262 -262
- package/src/browser/test/jsdom.ts +69 -69
- package/src/browser/test/mock-connection-status-service.ts +33 -33
- package/src/browser/test/mock-env-variables-server.ts +47 -47
- package/src/browser/test/mock-opener-service.ts +34 -34
- package/src/browser/test/mock-storage-service.ts +49 -49
- package/src/browser/theming.ts +206 -206
- package/src/browser/tooltip-service.tsx +96 -96
- package/src/browser/tree/fuzzy-search.spec.ts +99 -99
- package/src/browser/tree/fuzzy-search.ts +136 -136
- package/src/browser/tree/index.ts +29 -29
- package/src/browser/tree/search-box-debounce.ts +96 -96
- package/src/browser/tree/search-box.ts +355 -355
- package/src/browser/tree/test/mock-selectable-tree-model.ts +109 -109
- package/src/browser/tree/test/mock-tree-model.ts +136 -136
- package/src/browser/tree/test/tree-test-container.ts +50 -50
- package/src/browser/tree/tree-compression/compressed-tree-expansion-service.ts +46 -46
- package/src/browser/tree/tree-compression/compressed-tree-model.ts +88 -88
- package/src/browser/tree/tree-compression/compressed-tree-widget.tsx +203 -203
- package/src/browser/tree/tree-compression/index.ts +20 -20
- package/src/browser/tree/tree-compression/tree-compression-service.ts +125 -125
- package/src/browser/tree/tree-compression/tree-compression.css +28 -28
- package/src/browser/tree/tree-consistency.spec.ts +105 -105
- package/src/browser/tree/tree-container.spec.ts +45 -45
- package/src/browser/tree/tree-container.ts +155 -155
- package/src/browser/tree/tree-decorator.spec.ts +162 -162
- package/src/browser/tree/tree-decorator.ts +238 -238
- package/src/browser/tree/tree-expansion.spec.ts +173 -173
- package/src/browser/tree/tree-expansion.ts +165 -165
- package/src/browser/tree/tree-focus-service.ts +55 -55
- package/src/browser/tree/tree-iterator.spec.ts +170 -170
- package/src/browser/tree/tree-iterator.ts +256 -256
- package/src/browser/tree/tree-label-provider.ts +40 -40
- package/src/browser/tree/tree-model.ts +562 -562
- package/src/browser/tree/tree-navigation.ts +58 -58
- package/src/browser/tree/tree-preference.ts +50 -50
- package/src/browser/tree/tree-search.ts +128 -128
- package/src/browser/tree/tree-selectable.spec.ts +152 -152
- package/src/browser/tree/tree-selection-impl.ts +176 -176
- package/src/browser/tree/tree-selection-state.spec.ts +462 -462
- package/src/browser/tree/tree-selection-state.ts +245 -245
- package/src/browser/tree/tree-selection.ts +159 -159
- package/src/browser/tree/tree-view-welcome-widget.tsx +263 -263
- package/src/browser/tree/tree-widget-selection.ts +45 -45
- package/src/browser/tree/tree-widget.tsx +1585 -1585
- package/src/browser/tree/tree.spec.ts +241 -241
- package/src/browser/tree/tree.ts +425 -425
- package/src/browser/user-working-directory-provider.ts +77 -77
- package/src/browser/view-container.ts +1640 -1640
- package/src/browser/widget-decoration.ts +358 -358
- package/src/browser/widget-manager.spec.ts +102 -102
- package/src/browser/widget-manager.ts +310 -310
- package/src/browser/widget-open-handler.ts +165 -165
- package/src/browser/widgets/alert-message.tsx +56 -56
- package/src/browser/widgets/enhanced-preview-widget.ts +27 -27
- package/src/browser/widgets/extractable-widget.ts +33 -33
- package/src/browser/widgets/index.ts +20 -20
- package/src/browser/widgets/previewable-widget.ts +31 -31
- package/src/browser/widgets/react-renderer.tsx +50 -50
- package/src/browser/widgets/react-widget.tsx +51 -51
- package/src/browser/widgets/select-component.tsx +367 -367
- package/src/browser/widgets/widget.ts +406 -406
- package/src/browser/window/browser-window-module.ts +32 -32
- package/src/browser/window/default-secondary-window-service.ts +185 -185
- package/src/browser/window/default-window-service.spec.ts +78 -78
- package/src/browser/window/default-window-service.ts +171 -171
- package/src/browser/window/secondary-window-service.ts +39 -39
- package/src/browser/window/test/mock-window-service.ts +29 -29
- package/src/browser/window/window-service.ts +78 -78
- package/src/browser/window/window-title-service.ts +107 -107
- package/src/browser/window/window-title-updater.ts +95 -95
- package/src/browser/window-contribution.ts +64 -64
- package/src/browser-only/frontend-only-application-module.ts +115 -115
- package/src/browser-only/i18n/i18n-frontend-only-module.ts +37 -37
- package/src/browser-only/logger-frontend-only-module.ts +63 -63
- package/src/browser-only/messaging/frontend-only-service-connection-provider.ts +39 -39
- package/src/browser-only/messaging/messaging-frontend-only-module.ts +42 -42
- package/src/browser-only/preload/frontend-only-preload-module.ts +49 -49
- package/src/common/accessibility.ts +33 -33
- package/src/common/application-error.spec.ts +27 -27
- package/src/common/application-error.ts +76 -76
- package/src/common/application-protocol.ts +41 -41
- package/src/common/array-utils.ts +129 -129
- package/src/common/buffer.ts +228 -228
- package/src/common/cancellation.ts +163 -163
- package/src/common/char-code.ts +438 -438
- package/src/common/collections.ts +125 -125
- package/src/common/color.ts +103 -103
- package/src/common/command.spec.ts +208 -208
- package/src/common/command.ts +485 -485
- package/src/common/contribution-filter/contribution-filter-registry.ts +79 -79
- package/src/common/contribution-filter/contribution-filter.ts +64 -64
- package/src/common/contribution-filter/filter.ts +23 -23
- package/src/common/contribution-filter/index.ts +19 -19
- package/src/common/contribution-provider.ts +96 -96
- package/src/common/disposable.spec.ts +94 -94
- package/src/common/disposable.ts +188 -188
- package/src/common/encoding-service.ts +380 -380
- package/src/common/encodings.ts +24 -24
- package/src/common/env-variables/env-variables-protocol.ts +38 -38
- package/src/common/env-variables/index.ts +17 -17
- package/src/common/event.spec.ts +32 -32
- package/src/common/event.ts +469 -469
- package/src/common/file-uri.ts +61 -61
- package/src/common/frontend-application-state.ts +38 -38
- package/src/common/glob.ts +741 -741
- package/src/common/hash.ts +85 -85
- package/src/common/i18n/localization-server.ts +25 -25
- package/src/common/i18n/localization.ts +80 -80
- package/src/common/i18n/nls.metadata.json +31314 -31314
- package/src/common/index.ts +51 -51
- package/src/common/json-schema.ts +106 -106
- package/src/common/key-store.ts +26 -26
- package/src/common/keybinding.ts +152 -152
- package/src/common/keyboard/keyboard-layout-provider.ts +51 -51
- package/src/common/keys.ts +694 -694
- package/src/common/label-protocol.ts +35 -35
- package/src/common/logger-protocol.ts +119 -119
- package/src/common/logger-watcher.ts +48 -48
- package/src/common/logger.spec.ts +46 -46
- package/src/common/logger.ts +389 -389
- package/src/common/lsp-types.ts +34 -34
- package/src/common/markdown-rendering/icon-utilities.ts +30 -30
- package/src/common/markdown-rendering/index.ts +18 -18
- package/src/common/markdown-rendering/markdown-string.ts +152 -152
- package/src/common/menu/action-menu-node.ts +65 -65
- package/src/common/menu/composite-menu-node.spec.ts +67 -67
- package/src/common/menu/composite-menu-node.ts +114 -114
- package/src/common/menu/index.ts +21 -21
- package/src/common/menu/menu-adapter.ts +103 -103
- package/src/common/menu/menu-model-registry.ts +343 -343
- package/src/common/menu/menu-types.ts +219 -219
- package/src/common/menu/menu.spec.ts +101 -101
- package/src/common/message-rpc/channel.spec.ts +88 -88
- package/src/common/message-rpc/channel.ts +300 -300
- package/src/common/message-rpc/index.ts +22 -22
- package/src/common/message-rpc/message-buffer.ts +105 -105
- package/src/common/message-rpc/msg-pack-extension-manager.ts +70 -70
- package/src/common/message-rpc/rpc-message-encoder.spec.ts +65 -65
- package/src/common/message-rpc/rpc-message-encoder.ts +190 -190
- package/src/common/message-rpc/rpc-protocol.ts +255 -255
- package/src/common/message-rpc/uint8-array-message-buffer.spec.ts +41 -41
- package/src/common/message-rpc/uint8-array-message-buffer.ts +213 -213
- package/src/common/message-service-protocol.ts +148 -148
- package/src/common/message-service.ts +226 -226
- package/src/common/messaging/connection-error-handler.ts +73 -73
- package/src/common/messaging/connection-management.ts +43 -43
- package/src/common/messaging/handler.ts +26 -26
- package/src/common/messaging/index.ts +19 -19
- package/src/common/messaging/proxy-factory.spec.ts +108 -108
- package/src/common/messaging/proxy-factory.ts +336 -336
- package/src/common/messaging/socket-write-buffer.ts +52 -52
- package/src/common/messaging/web-socket-channel.ts +76 -76
- package/src/common/nls.ts +151 -151
- package/src/common/numbers.ts +21 -21
- package/src/common/objects.spec.ts +112 -112
- package/src/common/objects.ts +123 -123
- package/src/common/os.ts +82 -82
- package/src/common/path.spec.ts +415 -415
- package/src/common/path.ts +334 -334
- package/src/common/paths.ts +250 -250
- package/src/common/performance/index.ts +19 -19
- package/src/common/performance/measurement-protocol.ts +104 -104
- package/src/common/performance/measurement.ts +130 -130
- package/src/common/performance/stopwatch.ts +183 -183
- package/src/common/preferences/preference-schema.ts +95 -95
- package/src/common/preferences/preference-scope.spec.ts +48 -48
- package/src/common/preferences/preference-scope.ts +68 -68
- package/src/common/prioritizeable.ts +58 -58
- package/src/common/progress-service-protocol.ts +35 -35
- package/src/common/progress-service.ts +82 -82
- package/src/common/promise-util.spec.ts +102 -102
- package/src/common/promise-util.ts +143 -143
- package/src/common/quick-pick-service.ts +356 -356
- package/src/common/reference.spec.ts +145 -145
- package/src/common/reference.ts +230 -230
- package/src/common/resource.ts +430 -430
- package/src/common/selection-command-handler.ts +101 -101
- package/src/common/selection-service.spec.ts +43 -43
- package/src/common/selection-service.ts +49 -49
- package/src/common/selection.ts +50 -50
- package/src/common/severity.ts +111 -111
- package/src/common/stream.ts +718 -718
- package/src/common/strings.ts +231 -231
- package/src/common/telemetry.ts +45 -45
- package/src/common/ternary-search-tree.ts +417 -417
- package/src/common/test/expect.ts +34 -34
- package/src/common/test/mock-logger.ts +118 -118
- package/src/common/test/mock-menu.ts +35 -35
- package/src/common/test/mock-resource-provider.ts +33 -33
- package/src/common/theme.ts +68 -68
- package/src/common/types.spec.ts +86 -86
- package/src/common/types.ts +140 -140
- package/src/common/uri-command-handler.spec.ts +90 -90
- package/src/common/uri-command-handler.ts +148 -148
- package/src/common/uri.spec.ts +278 -278
- package/src/common/uri.ts +279 -279
- package/src/common/uuid.ts +45 -45
- package/src/common/version.ts +17 -17
- package/src/common/view-column.ts +33 -33
- package/src/common/window.ts +34 -34
- package/src/electron-browser/electron-clipboard-service.ts +32 -32
- package/src/electron-browser/keyboard/electron-keyboard-layout-change-notifier.ts +39 -39
- package/src/electron-browser/keyboard/electron-keyboard-module.ts +28 -28
- package/src/electron-browser/menu/electron-context-menu-renderer.ts +122 -122
- package/src/electron-browser/menu/electron-main-menu-factory.ts +335 -335
- package/src/electron-browser/menu/electron-menu-contribution.ts +506 -506
- package/src/electron-browser/menu/electron-menu-module.ts +40 -40
- package/src/electron-browser/menu/electron-menu-style.css +110 -110
- package/src/electron-browser/messaging/electron-frontend-id-provider.ts +25 -25
- package/src/electron-browser/messaging/electron-ipc-connection-source.ts +65 -65
- package/src/electron-browser/messaging/electron-local-ws-connection-source.ts +45 -45
- package/src/electron-browser/messaging/electron-messaging-frontend-module.ts +78 -78
- package/src/electron-browser/messaging/electron-ws-connection-source.ts +38 -38
- package/src/electron-browser/preload.ts +246 -246
- package/src/electron-browser/request/electron-browser-request-module.ts +26 -26
- package/src/electron-browser/token/electron-token-frontend-module.ts +22 -22
- package/src/electron-browser/window/electron-frontend-application-state.ts +26 -26
- package/src/electron-browser/window/electron-secondary-window-service.ts +35 -35
- package/src/electron-browser/window/electron-window-module.ts +41 -41
- package/src/electron-browser/window/electron-window-preferences.ts +76 -76
- package/src/electron-browser/window/electron-window-service.ts +109 -109
- package/src/electron-common/electron-api.ts +148 -148
- package/src/electron-common/electron-main-window-service.ts +24 -24
- package/src/electron-common/electron-token.ts +27 -27
- package/src/electron-main/electron-api-main.ts +340 -340
- package/src/electron-main/electron-main-application-module.ts +65 -65
- package/src/electron-main/electron-main-application.ts +685 -685
- package/src/electron-main/electron-main-constants.ts +22 -22
- package/src/electron-main/electron-main-window-service-impl.ts +44 -44
- package/src/electron-main/electron-security-token-service.ts +36 -36
- package/src/electron-main/event-utils.ts +36 -36
- package/src/electron-main/messaging/electron-connection-handler.ts +21 -21
- package/src/electron-main/messaging/electron-messaging-contribution.ts +143 -143
- package/src/electron-main/messaging/electron-messaging-service.ts +35 -35
- package/src/electron-main/theia-electron-window.ts +202 -202
- package/src/electron-node/hosting/electron-backend-hosting-module.ts +24 -24
- package/src/electron-node/hosting/electron-ws-origin-validator.ts +37 -37
- package/src/electron-node/keyboard/electron-backend-keyboard-module.ts +30 -30
- package/src/electron-node/keyboard/electron-keyboard-layout-provider.ts +35 -35
- package/src/electron-node/request/electron-backend-request-module.ts +23 -23
- package/src/electron-node/request/electron-backend-request-service.ts +78 -78
- package/src/electron-node/token/electron-token-backend-contribution.ts +48 -48
- package/src/electron-node/token/electron-token-backend-module.ts +28 -28
- package/src/electron-node/token/electron-token-validator.ts +93 -93
- package/src/node/application-server.ts +55 -55
- package/src/node/backend-application-config-provider.spec.ts +29 -29
- package/src/node/backend-application-config-provider.ts +48 -48
- package/src/node/backend-application-module.ts +140 -140
- package/src/node/backend-application.ts +382 -382
- package/src/node/backend-remote-service.ts +25 -25
- package/src/node/cli.spec.ts +94 -94
- package/src/node/cli.ts +63 -63
- package/src/node/console-logger-server.spec.ts +59 -59
- package/src/node/console-logger-server.ts +76 -76
- package/src/node/debug.ts +30 -30
- package/src/node/dynamic-require.ts +56 -56
- package/src/node/env-variables/env-variables-server.ts +120 -120
- package/src/node/env-variables/index.ts +17 -17
- package/src/node/environment-utils.spec.ts +92 -92
- package/src/node/environment-utils.ts +66 -66
- package/src/node/file-uri.spec.ts +76 -76
- package/src/node/filesystem-locking.ts +77 -77
- package/src/node/hosting/backend-application-hosts.ts +60 -60
- package/src/node/hosting/backend-hosting-module.ts +26 -26
- package/src/node/hosting/ws-origin-validator.ts +36 -36
- package/src/node/i18n/i18n-backend-module.ts +42 -42
- package/src/node/i18n/localization-contribution.ts +112 -112
- package/src/node/i18n/localization-provider.ts +125 -125
- package/src/node/i18n/localization-server.ts +52 -52
- package/src/node/i18n/theia-localization-contribution.ts +36 -36
- package/src/node/index.ts +22 -22
- package/src/node/key-store-server.ts +162 -162
- package/src/node/logger-backend-module.ts +88 -88
- package/src/node/logger-cli-contribution.spec.ts +245 -245
- package/src/node/logger-cli-contribution.ts +168 -168
- package/src/node/main.ts +33 -33
- package/src/node/messaging/binary-message-pipe.ts +168 -168
- package/src/node/messaging/connection-container-module.ts +96 -96
- package/src/node/messaging/default-messaging-service.ts +129 -129
- package/src/node/messaging/frontend-connection-service.ts +24 -24
- package/src/node/messaging/index.ts +19 -19
- package/src/node/messaging/ipc-bootstrap.ts +27 -27
- package/src/node/messaging/ipc-channel.ts +77 -77
- package/src/node/messaging/ipc-connection-provider.ts +107 -107
- package/src/node/messaging/ipc-protocol.ts +76 -76
- package/src/node/messaging/messaging-backend-module.ts +52 -52
- package/src/node/messaging/messaging-listeners.ts +52 -52
- package/src/node/messaging/messaging-service.ts +46 -46
- package/src/node/messaging/test/test-web-socket-channel.ts +61 -61
- package/src/node/messaging/websocket-endpoint.ts +79 -79
- package/src/node/messaging/websocket-frontend-connection-service.ts +176 -176
- package/src/node/os-backend-provider.ts +25 -25
- package/src/node/performance/index.ts +18 -18
- package/src/node/performance/measurement-backend-bindings.ts +35 -35
- package/src/node/performance/node-stopwatch.ts +40 -40
- package/src/node/process-utils.spec.ts +48 -48
- package/src/node/process-utils.ts +102 -102
- package/src/node/request/backend-request-facade.ts +39 -39
- package/src/node/request/backend-request-module.ts +25 -25
- package/src/node/request/proxy-cli-contribution.ts +65 -65
- package/src/node/ws-request-validators.ts +56 -56
- package/src/typings/native-keymap.d.ts +108 -108
|
@@ -1,1585 +1,1585 @@
|
|
|
1
|
-
// *****************************************************************************
|
|
2
|
-
// Copyright (C) 2018 TypeFox and others.
|
|
3
|
-
//
|
|
4
|
-
// This program and the accompanying materials are made available under the
|
|
5
|
-
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
6
|
-
// http://www.eclipse.org/legal/epl-2.0.
|
|
7
|
-
//
|
|
8
|
-
// This Source Code may also be made available under the following Secondary
|
|
9
|
-
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
10
|
-
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
11
|
-
// with the GNU Classpath Exception which is available at
|
|
12
|
-
// https://www.gnu.org/software/classpath/license.html.
|
|
13
|
-
//
|
|
14
|
-
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
15
|
-
// *****************************************************************************
|
|
16
|
-
|
|
17
|
-
import { injectable, inject, postConstruct } from 'inversify';
|
|
18
|
-
import { Message } from '@phosphor/messaging';
|
|
19
|
-
import { Disposable, MenuPath, SelectionService } from '../../common';
|
|
20
|
-
import { Key, KeyCode, KeyModifier } from '../keyboard/keys';
|
|
21
|
-
import { ContextMenuRenderer } from '../context-menu-renderer';
|
|
22
|
-
import { StatefulWidget } from '../shell';
|
|
23
|
-
import {
|
|
24
|
-
EXPANSION_TOGGLE_CLASS, SELECTED_CLASS, COLLAPSED_CLASS, FOCUS_CLASS, BUSY_CLASS, CODICON_TREE_ITEM_CLASSES, CODICON_LOADING_CLASSES, Widget, UnsafeWidgetUtilities
|
|
25
|
-
} from '../widgets';
|
|
26
|
-
import { TreeNode, CompositeTreeNode } from './tree';
|
|
27
|
-
import { TreeModel } from './tree-model';
|
|
28
|
-
import { ExpandableTreeNode } from './tree-expansion';
|
|
29
|
-
import { SelectableTreeNode, TreeSelection } from './tree-selection';
|
|
30
|
-
import { TreeDecoratorService, TreeDecoration, DecoratedTreeNode } from './tree-decorator';
|
|
31
|
-
import { notEmpty } from '../../common/objects';
|
|
32
|
-
import { isOSX } from '../../common/os';
|
|
33
|
-
import { ReactWidget } from '../widgets/react-widget';
|
|
34
|
-
import * as React from 'react';
|
|
35
|
-
import { Virtuoso, VirtuosoHandle } from 'react-virtuoso';
|
|
36
|
-
import { TopDownTreeIterator } from './tree-iterator';
|
|
37
|
-
import { SearchBox, SearchBoxFactory, SearchBoxProps } from './search-box';
|
|
38
|
-
import { TreeSearch } from './tree-search';
|
|
39
|
-
import { ElementExt } from '@phosphor/domutils';
|
|
40
|
-
import { TreeWidgetSelection } from './tree-widget-selection';
|
|
41
|
-
import { MaybePromise } from '../../common/types';
|
|
42
|
-
import { LabelProvider } from '../label-provider';
|
|
43
|
-
import { CorePreferences } from '../core-preferences';
|
|
44
|
-
import { TreeFocusService } from './tree-focus-service';
|
|
45
|
-
import { useEffect } from 'react';
|
|
46
|
-
import { PreferenceService, PreferenceChange } from '../preferences';
|
|
47
|
-
import { PREFERENCE_NAME_TREE_INDENT } from './tree-preference';
|
|
48
|
-
|
|
49
|
-
const debounce = require('lodash.debounce');
|
|
50
|
-
|
|
51
|
-
export const TREE_CLASS = 'theia-Tree';
|
|
52
|
-
export const TREE_CONTAINER_CLASS = 'theia-TreeContainer';
|
|
53
|
-
export const TREE_NODE_CLASS = 'theia-TreeNode';
|
|
54
|
-
export const TREE_NODE_CONTENT_CLASS = 'theia-TreeNodeContent';
|
|
55
|
-
export const TREE_NODE_INFO_CLASS = 'theia-TreeNodeInfo';
|
|
56
|
-
export const TREE_NODE_TAIL_CLASS = 'theia-TreeNodeTail';
|
|
57
|
-
export const TREE_NODE_SEGMENT_CLASS = 'theia-TreeNodeSegment';
|
|
58
|
-
export const TREE_NODE_SEGMENT_GROW_CLASS = 'theia-TreeNodeSegmentGrow';
|
|
59
|
-
|
|
60
|
-
export const EXPANDABLE_TREE_NODE_CLASS = 'theia-ExpandableTreeNode';
|
|
61
|
-
export const COMPOSITE_TREE_NODE_CLASS = 'theia-CompositeTreeNode';
|
|
62
|
-
export const TREE_NODE_CAPTION_CLASS = 'theia-TreeNodeCaption';
|
|
63
|
-
export const TREE_NODE_INDENT_GUIDE_CLASS = 'theia-tree-node-indent';
|
|
64
|
-
|
|
65
|
-
export const TreeProps = Symbol('TreeProps');
|
|
66
|
-
|
|
67
|
-
/**
|
|
68
|
-
* Representation of tree properties.
|
|
69
|
-
*/
|
|
70
|
-
export interface TreeProps {
|
|
71
|
-
|
|
72
|
-
/**
|
|
73
|
-
* The path of the context menu that one can use to contribute context menu items to the tree widget.
|
|
74
|
-
*/
|
|
75
|
-
readonly contextMenuPath?: MenuPath;
|
|
76
|
-
|
|
77
|
-
/**
|
|
78
|
-
* The size of the padding (in pixels) for the root node of the tree.
|
|
79
|
-
*/
|
|
80
|
-
readonly leftPadding: number;
|
|
81
|
-
|
|
82
|
-
readonly expansionTogglePadding: number;
|
|
83
|
-
|
|
84
|
-
/**
|
|
85
|
-
* `true` if the tree widget support multi-selection. Otherwise, `false`. Defaults to `false`.
|
|
86
|
-
*/
|
|
87
|
-
readonly multiSelect?: boolean;
|
|
88
|
-
|
|
89
|
-
/**
|
|
90
|
-
* `true` if the tree widget support searching. Otherwise, `false`. Defaults to `false`.
|
|
91
|
-
*/
|
|
92
|
-
readonly search?: boolean
|
|
93
|
-
|
|
94
|
-
/**
|
|
95
|
-
* `true` if the tree widget should be virtualized searching. Otherwise, `false`. Defaults to `true`.
|
|
96
|
-
*/
|
|
97
|
-
readonly virtualized?: boolean
|
|
98
|
-
|
|
99
|
-
/**
|
|
100
|
-
* `true` if the selected node should be auto scrolled only if the widget is active. Otherwise, `false`. Defaults to `false`.
|
|
101
|
-
*/
|
|
102
|
-
readonly scrollIfActive?: boolean
|
|
103
|
-
|
|
104
|
-
/**
|
|
105
|
-
* `true` if a tree widget contributes to the global selection. Defaults to `false`.
|
|
106
|
-
*/
|
|
107
|
-
readonly globalSelection?: boolean;
|
|
108
|
-
|
|
109
|
-
/**
|
|
110
|
-
* `true` if the tree widget supports expansion only when clicking the expansion toggle. Defaults to `false`.
|
|
111
|
-
*/
|
|
112
|
-
readonly expandOnlyOnExpansionToggleClick?: boolean;
|
|
113
|
-
|
|
114
|
-
}
|
|
115
|
-
|
|
116
|
-
/**
|
|
117
|
-
* Representation of node properties.
|
|
118
|
-
*/
|
|
119
|
-
export interface NodeProps {
|
|
120
|
-
|
|
121
|
-
/**
|
|
122
|
-
* A root relative number representing the hierarchical depth of the actual node. Root is `0`, its children have `1` and so on.
|
|
123
|
-
*/
|
|
124
|
-
readonly depth: number;
|
|
125
|
-
|
|
126
|
-
}
|
|
127
|
-
|
|
128
|
-
/**
|
|
129
|
-
* The default tree properties.
|
|
130
|
-
*/
|
|
131
|
-
export const defaultTreeProps: TreeProps = {
|
|
132
|
-
leftPadding: 8,
|
|
133
|
-
expansionTogglePadding: 22
|
|
134
|
-
};
|
|
135
|
-
|
|
136
|
-
export namespace TreeWidget {
|
|
137
|
-
|
|
138
|
-
/**
|
|
139
|
-
* Bare minimum common interface of the keyboard and the mouse event with respect to the key maskings.
|
|
140
|
-
*/
|
|
141
|
-
export interface ModifierAwareEvent {
|
|
142
|
-
/**
|
|
143
|
-
* Determines if the modifier aware event has the `meta` key masking.
|
|
144
|
-
*/
|
|
145
|
-
readonly metaKey: boolean;
|
|
146
|
-
/**
|
|
147
|
-
* Determines if the modifier aware event has the `ctrl` key masking.
|
|
148
|
-
*/
|
|
149
|
-
readonly ctrlKey: boolean;
|
|
150
|
-
/**
|
|
151
|
-
* Determines if the modifier aware event has the `shift` key masking.
|
|
152
|
-
*/
|
|
153
|
-
readonly shiftKey: boolean;
|
|
154
|
-
}
|
|
155
|
-
|
|
156
|
-
}
|
|
157
|
-
|
|
158
|
-
@injectable()
|
|
159
|
-
export class TreeWidget extends ReactWidget implements StatefulWidget {
|
|
160
|
-
|
|
161
|
-
protected searchBox: SearchBox;
|
|
162
|
-
protected searchHighlights: Map<string, TreeDecoration.CaptionHighlight>;
|
|
163
|
-
|
|
164
|
-
@inject(TreeDecoratorService)
|
|
165
|
-
protected readonly decoratorService: TreeDecoratorService;
|
|
166
|
-
@inject(TreeSearch)
|
|
167
|
-
protected readonly treeSearch: TreeSearch;
|
|
168
|
-
@inject(SearchBoxFactory)
|
|
169
|
-
protected readonly searchBoxFactory: SearchBoxFactory;
|
|
170
|
-
@inject(TreeFocusService)
|
|
171
|
-
protected readonly focusService: TreeFocusService;
|
|
172
|
-
|
|
173
|
-
protected decorations: Map<string, TreeDecoration.Data[]> = new Map();
|
|
174
|
-
|
|
175
|
-
@inject(SelectionService)
|
|
176
|
-
protected readonly selectionService: SelectionService;
|
|
177
|
-
|
|
178
|
-
@inject(PreferenceService)
|
|
179
|
-
protected readonly preferenceService: PreferenceService;
|
|
180
|
-
|
|
181
|
-
@inject(LabelProvider)
|
|
182
|
-
protected readonly labelProvider: LabelProvider;
|
|
183
|
-
|
|
184
|
-
@inject(CorePreferences)
|
|
185
|
-
protected readonly corePreferences: CorePreferences;
|
|
186
|
-
|
|
187
|
-
protected shouldScrollToRow = true;
|
|
188
|
-
|
|
189
|
-
protected treeIndent: number = 8;
|
|
190
|
-
|
|
191
|
-
constructor(
|
|
192
|
-
@inject(TreeProps) readonly props: TreeProps,
|
|
193
|
-
@inject(TreeModel) readonly model: TreeModel,
|
|
194
|
-
@inject(ContextMenuRenderer) protected readonly contextMenuRenderer: ContextMenuRenderer,
|
|
195
|
-
) {
|
|
196
|
-
super();
|
|
197
|
-
this.scrollOptions = {
|
|
198
|
-
suppressScrollX: true,
|
|
199
|
-
minScrollbarLength: 35
|
|
200
|
-
};
|
|
201
|
-
this.addClass(TREE_CLASS);
|
|
202
|
-
this.node.tabIndex = 0;
|
|
203
|
-
}
|
|
204
|
-
|
|
205
|
-
@postConstruct()
|
|
206
|
-
protected init(): void {
|
|
207
|
-
this.treeIndent = this.preferenceService.get(PREFERENCE_NAME_TREE_INDENT, this.treeIndent);
|
|
208
|
-
if (this.props.search) {
|
|
209
|
-
this.searchBox = this.searchBoxFactory({ ...SearchBoxProps.DEFAULT, showButtons: true, showFilter: true });
|
|
210
|
-
this.searchBox.node.addEventListener('focus', () => {
|
|
211
|
-
this.node.focus();
|
|
212
|
-
});
|
|
213
|
-
this.toDispose.pushAll([
|
|
214
|
-
this.searchBox,
|
|
215
|
-
this.searchBox.onTextChange(async data => {
|
|
216
|
-
await this.treeSearch.filter(data);
|
|
217
|
-
this.searchHighlights = this.treeSearch.getHighlights();
|
|
218
|
-
this.searchBox.updateHighlightInfo({
|
|
219
|
-
filterText: data,
|
|
220
|
-
total: this.rows.size,
|
|
221
|
-
matched: this.searchHighlights.size
|
|
222
|
-
});
|
|
223
|
-
this.update();
|
|
224
|
-
}),
|
|
225
|
-
this.searchBox.onClose(data => this.treeSearch.filter(undefined)),
|
|
226
|
-
this.searchBox.onNext(() => {
|
|
227
|
-
// Enable next selection if there are currently highlights.
|
|
228
|
-
if (this.searchHighlights.size > 1) {
|
|
229
|
-
this.model.selectNextNode();
|
|
230
|
-
}
|
|
231
|
-
}),
|
|
232
|
-
this.searchBox.onPrevious(() => {
|
|
233
|
-
// Enable previous selection if there are currently highlights.
|
|
234
|
-
if (this.searchHighlights.size > 1) {
|
|
235
|
-
this.model.selectPrevNode();
|
|
236
|
-
}
|
|
237
|
-
}),
|
|
238
|
-
this.searchBox.onFilterToggled(e => {
|
|
239
|
-
this.updateRows();
|
|
240
|
-
}),
|
|
241
|
-
this.treeSearch,
|
|
242
|
-
this.treeSearch.onFilteredNodesChanged(nodes => {
|
|
243
|
-
if (this.searchBox.isFiltering) {
|
|
244
|
-
this.updateRows();
|
|
245
|
-
}
|
|
246
|
-
const node = nodes.find(SelectableTreeNode.is);
|
|
247
|
-
if (node) {
|
|
248
|
-
this.model.selectNode(node);
|
|
249
|
-
}
|
|
250
|
-
}),
|
|
251
|
-
]);
|
|
252
|
-
}
|
|
253
|
-
this.node.addEventListener('mousedown', this.handleMiddleClickEvent.bind(this));
|
|
254
|
-
this.node.addEventListener('mouseup', this.handleMiddleClickEvent.bind(this));
|
|
255
|
-
this.node.addEventListener('auxclick', this.handleMiddleClickEvent.bind(this));
|
|
256
|
-
this.toDispose.pushAll([
|
|
257
|
-
this.model,
|
|
258
|
-
this.model.onChanged(() => this.updateRows()),
|
|
259
|
-
this.model.onSelectionChanged(() => this.scheduleUpdateScrollToRow({ resize: false })),
|
|
260
|
-
this.focusService.onDidChangeFocus(() => this.scheduleUpdateScrollToRow({ resize: false })),
|
|
261
|
-
this.model.onDidChangeBusy(() => this.update()),
|
|
262
|
-
this.model.onDidUpdate(() => this.update()),
|
|
263
|
-
this.model.onNodeRefreshed(() => this.updateDecorations()),
|
|
264
|
-
this.model.onExpansionChanged(() => this.updateDecorations()),
|
|
265
|
-
this.decoratorService,
|
|
266
|
-
this.decoratorService.onDidChangeDecorations(() => this.updateDecorations()),
|
|
267
|
-
this.labelProvider.onDidChange(e => {
|
|
268
|
-
for (const row of this.rows.values()) {
|
|
269
|
-
if (e.affects(row)) {
|
|
270
|
-
this.update();
|
|
271
|
-
return;
|
|
272
|
-
}
|
|
273
|
-
}
|
|
274
|
-
}),
|
|
275
|
-
this.preferenceService.onPreferenceChanged((event: PreferenceChange) => {
|
|
276
|
-
if (event.preferenceName === PREFERENCE_NAME_TREE_INDENT) {
|
|
277
|
-
this.treeIndent = event.newValue;
|
|
278
|
-
this.update();
|
|
279
|
-
}
|
|
280
|
-
})
|
|
281
|
-
]);
|
|
282
|
-
setTimeout(() => {
|
|
283
|
-
this.updateRows();
|
|
284
|
-
this.updateDecorations();
|
|
285
|
-
});
|
|
286
|
-
if (this.props.globalSelection) {
|
|
287
|
-
this.toDispose.pushAll([
|
|
288
|
-
this.model.onSelectionChanged(() => {
|
|
289
|
-
if (this.node.contains(document.activeElement)) {
|
|
290
|
-
this.updateGlobalSelection();
|
|
291
|
-
}
|
|
292
|
-
}),
|
|
293
|
-
this.focusService.onDidChangeFocus(focus => {
|
|
294
|
-
if (focus && this.node.contains(document.activeElement) && this.model.selectedNodes[0] !== focus && this.model.selectedNodes.includes(focus)) {
|
|
295
|
-
this.updateGlobalSelection();
|
|
296
|
-
}
|
|
297
|
-
}),
|
|
298
|
-
Disposable.create(() => {
|
|
299
|
-
const selection = this.selectionService.selection;
|
|
300
|
-
if (TreeWidgetSelection.isSource(selection, this)) {
|
|
301
|
-
this.selectionService.selection = undefined;
|
|
302
|
-
}
|
|
303
|
-
})
|
|
304
|
-
]);
|
|
305
|
-
}
|
|
306
|
-
this.toDispose.push(this.corePreferences.onPreferenceChanged(preference => {
|
|
307
|
-
if (preference.preferenceName === 'workbench.tree.renderIndentGuides') {
|
|
308
|
-
this.update();
|
|
309
|
-
}
|
|
310
|
-
}));
|
|
311
|
-
}
|
|
312
|
-
|
|
313
|
-
/**
|
|
314
|
-
* Update the global selection for the tree.
|
|
315
|
-
*/
|
|
316
|
-
protected updateGlobalSelection(): void {
|
|
317
|
-
this.selectionService.selection = TreeWidgetSelection.create(this);
|
|
318
|
-
}
|
|
319
|
-
|
|
320
|
-
protected rows = new Map<string, TreeWidget.NodeRow>();
|
|
321
|
-
protected updateRows = debounce(() => this.doUpdateRows(), 10);
|
|
322
|
-
protected doUpdateRows(): void {
|
|
323
|
-
const root = this.model.root;
|
|
324
|
-
const rowsToUpdate: Array<[string, TreeWidget.NodeRow]> = [];
|
|
325
|
-
if (root) {
|
|
326
|
-
const depths = new Map<CompositeTreeNode | undefined, number>();
|
|
327
|
-
let index = 0;
|
|
328
|
-
for (const node of new TopDownTreeIterator(root, {
|
|
329
|
-
pruneCollapsed: true,
|
|
330
|
-
pruneSiblings: true
|
|
331
|
-
})) {
|
|
332
|
-
if (this.shouldDisplayNode(node)) {
|
|
333
|
-
const depth = this.getDepthForNode(node, depths);
|
|
334
|
-
if (CompositeTreeNode.is(node)) {
|
|
335
|
-
depths.set(node, depth);
|
|
336
|
-
}
|
|
337
|
-
rowsToUpdate.push([node.id, this.toNodeRow(node, index++, depth)]);
|
|
338
|
-
}
|
|
339
|
-
}
|
|
340
|
-
}
|
|
341
|
-
this.rows = new Map(rowsToUpdate);
|
|
342
|
-
this.updateScrollToRow();
|
|
343
|
-
}
|
|
344
|
-
|
|
345
|
-
protected getDepthForNode(node: TreeNode, depths: Map<CompositeTreeNode | undefined, number>): number {
|
|
346
|
-
const parentDepth = depths.get(node.parent);
|
|
347
|
-
return parentDepth === undefined ? 0 : TreeNode.isVisible(node.parent) ? parentDepth + 1 : parentDepth;
|
|
348
|
-
}
|
|
349
|
-
|
|
350
|
-
protected toNodeRow(node: TreeNode, index: number, depth: number): TreeWidget.NodeRow {
|
|
351
|
-
return { node, index, depth };
|
|
352
|
-
}
|
|
353
|
-
|
|
354
|
-
protected shouldDisplayNode(node: TreeNode): boolean {
|
|
355
|
-
return TreeNode.isVisible(node) && (!this.searchBox?.isFiltering || this.treeSearch.passesFilters(node));
|
|
356
|
-
}
|
|
357
|
-
|
|
358
|
-
/**
|
|
359
|
-
* Row index to ensure visibility.
|
|
360
|
-
* - Used to forcefully scroll if necessary.
|
|
361
|
-
*/
|
|
362
|
-
protected scrollToRow: number | undefined;
|
|
363
|
-
/**
|
|
364
|
-
* Update the `scrollToRow`.
|
|
365
|
-
* @param updateOptions the tree widget force update options.
|
|
366
|
-
*/
|
|
367
|
-
protected updateScrollToRow(): void {
|
|
368
|
-
this.scrollToRow = this.getScrollToRow();
|
|
369
|
-
this.update();
|
|
370
|
-
}
|
|
371
|
-
|
|
372
|
-
protected scheduleUpdateScrollToRow = debounce(this.updateScrollToRow);
|
|
373
|
-
|
|
374
|
-
/**
|
|
375
|
-
* Get the `scrollToRow`.
|
|
376
|
-
*
|
|
377
|
-
* @returns the `scrollToRow` if available.
|
|
378
|
-
*/
|
|
379
|
-
protected getScrollToRow(): number | undefined {
|
|
380
|
-
if (!this.shouldScrollToRow) {
|
|
381
|
-
return undefined;
|
|
382
|
-
}
|
|
383
|
-
const { focusedNode } = this.focusService;
|
|
384
|
-
return focusedNode && this.rows.get(focusedNode.id)?.index;
|
|
385
|
-
}
|
|
386
|
-
|
|
387
|
-
/**
|
|
388
|
-
* Update tree decorations.
|
|
389
|
-
* - Updating decorations are debounced in order to limit the number of expensive updates.
|
|
390
|
-
*/
|
|
391
|
-
protected readonly updateDecorations = debounce(() => this.doUpdateDecorations(), 150);
|
|
392
|
-
protected async doUpdateDecorations(): Promise<void> {
|
|
393
|
-
this.decorations = await this.decoratorService.getDecorations(this.model);
|
|
394
|
-
this.update();
|
|
395
|
-
}
|
|
396
|
-
|
|
397
|
-
protected override onActivateRequest(msg: Message): void {
|
|
398
|
-
super.onActivateRequest(msg);
|
|
399
|
-
this.node.focus({ preventScroll: true });
|
|
400
|
-
}
|
|
401
|
-
|
|
402
|
-
/**
|
|
403
|
-
* Actually focus the tree node.
|
|
404
|
-
*/
|
|
405
|
-
protected doFocus(): void {
|
|
406
|
-
if (!this.model.selectedNodes.length) {
|
|
407
|
-
const node = this.getNodeToFocus();
|
|
408
|
-
if (SelectableTreeNode.is(node)) {
|
|
409
|
-
this.model.selectNode(node);
|
|
410
|
-
}
|
|
411
|
-
}
|
|
412
|
-
}
|
|
413
|
-
|
|
414
|
-
/**
|
|
415
|
-
* Get the tree node to focus.
|
|
416
|
-
*
|
|
417
|
-
* @returns the node to focus if available.
|
|
418
|
-
*/
|
|
419
|
-
protected getNodeToFocus(): SelectableTreeNode | undefined {
|
|
420
|
-
const { focusedNode } = this.focusService;
|
|
421
|
-
if (focusedNode) {
|
|
422
|
-
return focusedNode;
|
|
423
|
-
}
|
|
424
|
-
const { root } = this.model;
|
|
425
|
-
if (SelectableTreeNode.isVisible(root)) {
|
|
426
|
-
return root;
|
|
427
|
-
}
|
|
428
|
-
return this.model.getNextSelectableNode(root);
|
|
429
|
-
}
|
|
430
|
-
|
|
431
|
-
protected override onUpdateRequest(msg: Message): void {
|
|
432
|
-
if (!this.isAttached || !this.isVisible) {
|
|
433
|
-
return;
|
|
434
|
-
}
|
|
435
|
-
super.onUpdateRequest(msg);
|
|
436
|
-
}
|
|
437
|
-
|
|
438
|
-
protected override onResize(msg: Widget.ResizeMessage): void {
|
|
439
|
-
super.onResize(msg);
|
|
440
|
-
this.update();
|
|
441
|
-
}
|
|
442
|
-
|
|
443
|
-
protected render(): React.ReactNode {
|
|
444
|
-
return React.createElement('div', this.createContainerAttributes(), this.renderTree(this.model));
|
|
445
|
-
}
|
|
446
|
-
|
|
447
|
-
/**
|
|
448
|
-
* Create the container attributes for the widget.
|
|
449
|
-
*/
|
|
450
|
-
protected createContainerAttributes(): React.HTMLAttributes<HTMLElement> {
|
|
451
|
-
const classNames = [TREE_CONTAINER_CLASS];
|
|
452
|
-
if (!this.rows.size) {
|
|
453
|
-
classNames.push('empty');
|
|
454
|
-
}
|
|
455
|
-
if (this.model.selectedNodes.length === 0 && !this.focusService.focusedNode) {
|
|
456
|
-
classNames.push('focused');
|
|
457
|
-
}
|
|
458
|
-
return {
|
|
459
|
-
className: classNames.join(' '),
|
|
460
|
-
onContextMenu: event => this.handleContextMenuEvent(this.getContainerTreeNode(), event)
|
|
461
|
-
};
|
|
462
|
-
}
|
|
463
|
-
/**
|
|
464
|
-
* Get the container tree node.
|
|
465
|
-
*
|
|
466
|
-
* @returns the tree node for the container if available.
|
|
467
|
-
*/
|
|
468
|
-
protected getContainerTreeNode(): TreeNode | undefined {
|
|
469
|
-
return this.model.root;
|
|
470
|
-
}
|
|
471
|
-
|
|
472
|
-
protected ScrollingRowRenderer: React.FC<{ rows: TreeWidget.NodeRow[] }> = ({ rows }) => {
|
|
473
|
-
useEffect(() => this.scrollToSelected());
|
|
474
|
-
return <>{rows.map(row => <div key={row.index}>{this.renderNodeRow(row)}</div>)}</>;
|
|
475
|
-
};
|
|
476
|
-
|
|
477
|
-
protected view: TreeWidget.View | undefined;
|
|
478
|
-
/**
|
|
479
|
-
* Render the tree widget.
|
|
480
|
-
* @param model the tree model.
|
|
481
|
-
*/
|
|
482
|
-
protected renderTree(model: TreeModel): React.ReactNode {
|
|
483
|
-
if (model.root) {
|
|
484
|
-
const rows = Array.from(this.rows.values());
|
|
485
|
-
if (this.props.virtualized === false) {
|
|
486
|
-
return <this.ScrollingRowRenderer rows={rows} />;
|
|
487
|
-
}
|
|
488
|
-
return <TreeWidget.View
|
|
489
|
-
ref={view => this.view = (view || undefined)}
|
|
490
|
-
width={this.node.offsetWidth}
|
|
491
|
-
height={this.node.offsetHeight}
|
|
492
|
-
rows={rows}
|
|
493
|
-
renderNodeRow={this.renderNodeRow}
|
|
494
|
-
scrollToRow={this.scrollToRow}
|
|
495
|
-
/>;
|
|
496
|
-
}
|
|
497
|
-
// eslint-disable-next-line no-null/no-null
|
|
498
|
-
return null;
|
|
499
|
-
}
|
|
500
|
-
|
|
501
|
-
scrollArea: Element = this.node;
|
|
502
|
-
/**
|
|
503
|
-
* Scroll to the selected tree node.
|
|
504
|
-
*/
|
|
505
|
-
protected scrollToSelected(): void {
|
|
506
|
-
if (this.props.scrollIfActive === true && !this.node.contains(document.activeElement)) {
|
|
507
|
-
return;
|
|
508
|
-
}
|
|
509
|
-
const focus = this.node.getElementsByClassName(FOCUS_CLASS)[0];
|
|
510
|
-
if (focus) {
|
|
511
|
-
ElementExt.scrollIntoViewIfNeeded(this.scrollArea, focus);
|
|
512
|
-
} else {
|
|
513
|
-
const selected = this.node.getElementsByClassName(SELECTED_CLASS)[0];
|
|
514
|
-
if (selected) {
|
|
515
|
-
ElementExt.scrollIntoViewIfNeeded(this.scrollArea, selected);
|
|
516
|
-
}
|
|
517
|
-
}
|
|
518
|
-
}
|
|
519
|
-
|
|
520
|
-
/**
|
|
521
|
-
* Render the node row.
|
|
522
|
-
*/
|
|
523
|
-
protected readonly renderNodeRow = (row: TreeWidget.NodeRow) => this.doRenderNodeRow(row);
|
|
524
|
-
/**
|
|
525
|
-
* Actually render the node row.
|
|
526
|
-
*/
|
|
527
|
-
protected doRenderNodeRow({ node, depth }: TreeWidget.NodeRow): React.ReactNode {
|
|
528
|
-
return <React.Fragment>
|
|
529
|
-
{this.renderIndent(node, { depth })}
|
|
530
|
-
{this.renderNode(node, { depth })}
|
|
531
|
-
</React.Fragment>;
|
|
532
|
-
}
|
|
533
|
-
|
|
534
|
-
/**
|
|
535
|
-
* Render the tree node given the node properties.
|
|
536
|
-
* @param node the tree node.
|
|
537
|
-
* @param props the node properties.
|
|
538
|
-
*/
|
|
539
|
-
protected renderIcon(node: TreeNode, props: NodeProps): React.ReactNode {
|
|
540
|
-
// eslint-disable-next-line no-null/no-null
|
|
541
|
-
return null;
|
|
542
|
-
}
|
|
543
|
-
|
|
544
|
-
/**
|
|
545
|
-
* Toggle the node.
|
|
546
|
-
*/
|
|
547
|
-
protected readonly toggle = (event: React.MouseEvent<HTMLElement>) => this.doToggle(event);
|
|
548
|
-
/**
|
|
549
|
-
* Actually toggle the tree node.
|
|
550
|
-
* @param event the mouse click event.
|
|
551
|
-
*/
|
|
552
|
-
protected doToggle(event: React.MouseEvent<HTMLElement>): void {
|
|
553
|
-
const nodeId = event.currentTarget.getAttribute('data-node-id');
|
|
554
|
-
if (nodeId) {
|
|
555
|
-
const node = this.model.getNode(nodeId);
|
|
556
|
-
if (node && this.props.expandOnlyOnExpansionToggleClick) {
|
|
557
|
-
if (this.isExpandable(node) && !this.hasShiftMask(event) && !this.hasCtrlCmdMask(event)) {
|
|
558
|
-
this.model.toggleNodeExpansion(node);
|
|
559
|
-
}
|
|
560
|
-
} else {
|
|
561
|
-
this.handleClickEvent(node, event);
|
|
562
|
-
}
|
|
563
|
-
}
|
|
564
|
-
event.stopPropagation();
|
|
565
|
-
}
|
|
566
|
-
|
|
567
|
-
/**
|
|
568
|
-
* Render the node expansion toggle.
|
|
569
|
-
* @param node the tree node.
|
|
570
|
-
* @param props the node properties.
|
|
571
|
-
*/
|
|
572
|
-
protected renderExpansionToggle(node: TreeNode, props: NodeProps): React.ReactNode {
|
|
573
|
-
if (!this.isExpandable(node)) {
|
|
574
|
-
// eslint-disable-next-line no-null/no-null
|
|
575
|
-
return null;
|
|
576
|
-
}
|
|
577
|
-
const classes = [TREE_NODE_SEGMENT_CLASS, EXPANSION_TOGGLE_CLASS];
|
|
578
|
-
if (!node.expanded) {
|
|
579
|
-
classes.push(COLLAPSED_CLASS);
|
|
580
|
-
}
|
|
581
|
-
if (node.busy) {
|
|
582
|
-
classes.push(BUSY_CLASS, ...CODICON_LOADING_CLASSES);
|
|
583
|
-
} else {
|
|
584
|
-
classes.push(...CODICON_TREE_ITEM_CLASSES);
|
|
585
|
-
}
|
|
586
|
-
const className = classes.join(' ');
|
|
587
|
-
return <div
|
|
588
|
-
data-node-id={node.id}
|
|
589
|
-
className={className}
|
|
590
|
-
onClick={this.toggle}
|
|
591
|
-
onDoubleClick={this.handleExpansionToggleDblClickEvent}>
|
|
592
|
-
</div>;
|
|
593
|
-
}
|
|
594
|
-
|
|
595
|
-
/**
|
|
596
|
-
* Render the node expansion toggle.
|
|
597
|
-
* @param node the tree node.
|
|
598
|
-
* @param props the node properties.
|
|
599
|
-
*/
|
|
600
|
-
protected renderCheckbox(node: TreeNode, props: NodeProps): React.ReactNode {
|
|
601
|
-
if (node.checkboxInfo === undefined) {
|
|
602
|
-
// eslint-disable-next-line no-null/no-null
|
|
603
|
-
return null;
|
|
604
|
-
}
|
|
605
|
-
return <input data-node-id={node.id}
|
|
606
|
-
readOnly
|
|
607
|
-
type='checkbox'
|
|
608
|
-
checked={!!node.checkboxInfo.checked}
|
|
609
|
-
title={node.checkboxInfo.tooltip}
|
|
610
|
-
aria-label={node.checkboxInfo.accessibilityInformation?.label}
|
|
611
|
-
role={node.checkboxInfo.accessibilityInformation?.role}
|
|
612
|
-
className='theia-input'
|
|
613
|
-
onClick={event => this.toggleChecked(event)} />;
|
|
614
|
-
}
|
|
615
|
-
|
|
616
|
-
protected toggleChecked(event: React.MouseEvent<HTMLElement>): void {
|
|
617
|
-
const nodeId = event.currentTarget.getAttribute('data-node-id');
|
|
618
|
-
if (nodeId) {
|
|
619
|
-
const node = this.model.getNode(nodeId);
|
|
620
|
-
if (node) {
|
|
621
|
-
this.model.markAsChecked(node, !node.checkboxInfo!.checked);
|
|
622
|
-
} else {
|
|
623
|
-
this.handleClickEvent(node, event);
|
|
624
|
-
}
|
|
625
|
-
}
|
|
626
|
-
event.preventDefault();
|
|
627
|
-
event.stopPropagation();
|
|
628
|
-
}
|
|
629
|
-
/**
|
|
630
|
-
* Render the tree node caption given the node properties.
|
|
631
|
-
* @param node the tree node.
|
|
632
|
-
* @param props the node properties.
|
|
633
|
-
*/
|
|
634
|
-
protected renderCaption(node: TreeNode, props: NodeProps): React.ReactNode {
|
|
635
|
-
const attrs = this.getCaptionAttributes(node, props);
|
|
636
|
-
const children = this.getCaptionChildren(node, props);
|
|
637
|
-
return React.createElement('div', attrs, children);
|
|
638
|
-
}
|
|
639
|
-
|
|
640
|
-
protected getCaptionAttributes(node: TreeNode, props: NodeProps): React.Attributes & React.HTMLAttributes<HTMLElement> {
|
|
641
|
-
const tooltip = this.getDecorationData(node, 'tooltip').filter(notEmpty).join(' • ');
|
|
642
|
-
const classes = [TREE_NODE_SEGMENT_CLASS];
|
|
643
|
-
if (!this.hasTrailingSuffixes(node)) {
|
|
644
|
-
classes.push(TREE_NODE_SEGMENT_GROW_CLASS);
|
|
645
|
-
}
|
|
646
|
-
const className = classes.join(' ');
|
|
647
|
-
let attrs = this.decorateCaption(node, {
|
|
648
|
-
className, id: node.id
|
|
649
|
-
});
|
|
650
|
-
if (tooltip.length > 0) {
|
|
651
|
-
attrs = {
|
|
652
|
-
...attrs,
|
|
653
|
-
title: tooltip
|
|
654
|
-
};
|
|
655
|
-
}
|
|
656
|
-
return attrs;
|
|
657
|
-
}
|
|
658
|
-
|
|
659
|
-
protected getCaptionChildren(node: TreeNode, props: NodeProps): React.ReactNode {
|
|
660
|
-
const children = [];
|
|
661
|
-
const caption = this.toNodeName(node);
|
|
662
|
-
const highlight = this.getDecorationData(node, 'highlight')[0];
|
|
663
|
-
if (highlight) {
|
|
664
|
-
children.push(this.toReactNode(caption, highlight));
|
|
665
|
-
}
|
|
666
|
-
const searchHighlight = this.searchHighlights ? this.searchHighlights.get(node.id) : undefined;
|
|
667
|
-
if (searchHighlight) {
|
|
668
|
-
children.push(...this.toReactNode(caption, searchHighlight));
|
|
669
|
-
} else if (!highlight) {
|
|
670
|
-
children.push(caption);
|
|
671
|
-
}
|
|
672
|
-
return children;
|
|
673
|
-
}
|
|
674
|
-
|
|
675
|
-
/**
|
|
676
|
-
* Update the node given the caption and highlight.
|
|
677
|
-
* @param caption the caption.
|
|
678
|
-
* @param highlight the tree decoration caption highlight.
|
|
679
|
-
*/
|
|
680
|
-
protected toReactNode(caption: string, highlight: TreeDecoration.CaptionHighlight): React.ReactNode[] {
|
|
681
|
-
let style: React.CSSProperties = {};
|
|
682
|
-
if (highlight.color) {
|
|
683
|
-
style = {
|
|
684
|
-
...style,
|
|
685
|
-
color: highlight.color
|
|
686
|
-
};
|
|
687
|
-
}
|
|
688
|
-
if (highlight.backgroundColor) {
|
|
689
|
-
style = {
|
|
690
|
-
...style,
|
|
691
|
-
backgroundColor: highlight.backgroundColor
|
|
692
|
-
};
|
|
693
|
-
}
|
|
694
|
-
const createChildren = (fragment: TreeDecoration.CaptionHighlight.Fragment, index: number) => {
|
|
695
|
-
const { data } = fragment;
|
|
696
|
-
if (fragment.highlight) {
|
|
697
|
-
return <mark className={TreeDecoration.Styles.CAPTION_HIGHLIGHT_CLASS} style={style} key={index}>{data}</mark>;
|
|
698
|
-
} else {
|
|
699
|
-
return data;
|
|
700
|
-
}
|
|
701
|
-
};
|
|
702
|
-
return TreeDecoration.CaptionHighlight.split(caption, highlight).map(createChildren);
|
|
703
|
-
}
|
|
704
|
-
|
|
705
|
-
/**
|
|
706
|
-
* Decorate the tree caption.
|
|
707
|
-
* @param node the tree node.
|
|
708
|
-
* @param attrs the additional attributes.
|
|
709
|
-
*/
|
|
710
|
-
protected decorateCaption(node: TreeNode, attrs: React.HTMLAttributes<HTMLElement>): React.Attributes & React.HTMLAttributes<HTMLElement> {
|
|
711
|
-
const style = this.getDecorationData(node, 'fontData')
|
|
712
|
-
.filter(notEmpty)
|
|
713
|
-
.reverse()
|
|
714
|
-
.map(fontData => this.applyFontStyles({}, fontData))
|
|
715
|
-
.reduce((acc, current) => ({
|
|
716
|
-
...acc,
|
|
717
|
-
...current
|
|
718
|
-
}), {});
|
|
719
|
-
return {
|
|
720
|
-
...attrs,
|
|
721
|
-
style
|
|
722
|
-
};
|
|
723
|
-
}
|
|
724
|
-
|
|
725
|
-
/**
|
|
726
|
-
* Determine if the tree node contains trailing suffixes.
|
|
727
|
-
* @param node the tree node.
|
|
728
|
-
*
|
|
729
|
-
* @returns `true` if the tree node contains trailing suffices.
|
|
730
|
-
*/
|
|
731
|
-
protected hasTrailingSuffixes(node: TreeNode): boolean {
|
|
732
|
-
return this.getDecorationData(node, 'captionSuffixes').filter(notEmpty).reduce((acc, current) => acc.concat(current), []).length > 0;
|
|
733
|
-
}
|
|
734
|
-
|
|
735
|
-
/**
|
|
736
|
-
* Apply font styles to the tree.
|
|
737
|
-
* @param original the original css properties.
|
|
738
|
-
* @param fontData the optional `fontData`.
|
|
739
|
-
*/
|
|
740
|
-
protected applyFontStyles(original: React.CSSProperties, fontData: TreeDecoration.FontData | undefined): React.CSSProperties {
|
|
741
|
-
if (fontData === undefined) {
|
|
742
|
-
return original;
|
|
743
|
-
}
|
|
744
|
-
const modified = { ...original }; // make a copy to mutate
|
|
745
|
-
const { color, style } = fontData;
|
|
746
|
-
if (color) {
|
|
747
|
-
modified.color = color;
|
|
748
|
-
}
|
|
749
|
-
if (style) {
|
|
750
|
-
(Array.isArray(style) ? style : [style]).forEach(s => {
|
|
751
|
-
switch (s) {
|
|
752
|
-
case 'bold':
|
|
753
|
-
modified.fontWeight = s;
|
|
754
|
-
break;
|
|
755
|
-
case 'normal':
|
|
756
|
-
case 'oblique':
|
|
757
|
-
case 'italic':
|
|
758
|
-
modified.fontStyle = s;
|
|
759
|
-
break;
|
|
760
|
-
case 'underline':
|
|
761
|
-
case 'line-through':
|
|
762
|
-
modified.textDecoration = s;
|
|
763
|
-
break;
|
|
764
|
-
default:
|
|
765
|
-
throw new Error(`Unexpected font style: "${s}".`);
|
|
766
|
-
}
|
|
767
|
-
});
|
|
768
|
-
}
|
|
769
|
-
return modified;
|
|
770
|
-
}
|
|
771
|
-
|
|
772
|
-
/**
|
|
773
|
-
* Render caption affixes for the given tree node.
|
|
774
|
-
* @param node the tree node.
|
|
775
|
-
* @param props the node properties.
|
|
776
|
-
* @param affixKey the affix key.
|
|
777
|
-
*/
|
|
778
|
-
protected renderCaptionAffixes(node: TreeNode, props: NodeProps, affixKey: 'captionPrefixes' | 'captionSuffixes'): React.ReactNode {
|
|
779
|
-
const suffix = affixKey === 'captionSuffixes';
|
|
780
|
-
const affixClass = suffix ? TreeDecoration.Styles.CAPTION_SUFFIX_CLASS : TreeDecoration.Styles.CAPTION_PREFIX_CLASS;
|
|
781
|
-
const classes = [TREE_NODE_SEGMENT_CLASS, affixClass];
|
|
782
|
-
const affixes = this.getDecorationData(node, affixKey).filter(notEmpty).reduce((acc, current) => acc.concat(current), []);
|
|
783
|
-
const children: React.ReactNode[] = [];
|
|
784
|
-
for (let i = 0; i < affixes.length; i++) {
|
|
785
|
-
const affix = affixes[i];
|
|
786
|
-
if (suffix && i === affixes.length - 1) {
|
|
787
|
-
classes.push(TREE_NODE_SEGMENT_GROW_CLASS);
|
|
788
|
-
}
|
|
789
|
-
const style = this.applyFontStyles({}, affix.fontData);
|
|
790
|
-
const className = classes.join(' ');
|
|
791
|
-
const key = node.id + '_' + i;
|
|
792
|
-
const attrs = {
|
|
793
|
-
className,
|
|
794
|
-
style,
|
|
795
|
-
key
|
|
796
|
-
};
|
|
797
|
-
children.push(React.createElement('div', attrs, affix.data));
|
|
798
|
-
}
|
|
799
|
-
return <React.Fragment>{children}</React.Fragment>;
|
|
800
|
-
}
|
|
801
|
-
|
|
802
|
-
/**
|
|
803
|
-
* Decorate the tree node icon.
|
|
804
|
-
* @param node the tree node.
|
|
805
|
-
* @param icon the icon.
|
|
806
|
-
*/
|
|
807
|
-
protected decorateIcon(node: TreeNode, icon: React.ReactNode): React.ReactNode {
|
|
808
|
-
if (!icon) {
|
|
809
|
-
return;
|
|
810
|
-
}
|
|
811
|
-
const overlayIcons: React.ReactNode[] = [];
|
|
812
|
-
// if multiple overlays have the same overlay.position attribute, we'll de-duplicate those and only process the first one from the decoration array
|
|
813
|
-
const seenPositions = new Set<TreeDecoration.IconOverlayPosition>();
|
|
814
|
-
const overlays = this.getDecorationData(node, 'iconOverlay').filter(notEmpty);
|
|
815
|
-
|
|
816
|
-
for (const overlay of overlays) {
|
|
817
|
-
if (!seenPositions.has(overlay.position)) {
|
|
818
|
-
seenPositions.add(overlay.position);
|
|
819
|
-
const iconClasses = [TreeDecoration.Styles.DECORATOR_SIZE_CLASS, TreeDecoration.IconOverlayPosition.getStyle(overlay.position)];
|
|
820
|
-
const style = (color?: string) => color === undefined ? {} : { color };
|
|
821
|
-
|
|
822
|
-
if (overlay.background) {
|
|
823
|
-
overlayIcons.push(<span key={node.id + 'bg'} className={this.getIconClass(overlay.background.shape, iconClasses)}
|
|
824
|
-
style={style(overlay.background.color)}></span>);
|
|
825
|
-
}
|
|
826
|
-
|
|
827
|
-
const overlayIcon = 'icon' in overlay ? overlay.icon : overlay.iconClass;
|
|
828
|
-
overlayIcons.push(<span key={node.id} className={this.getIconClass(overlayIcon, iconClasses)} style={style(overlay.color)}></span>);
|
|
829
|
-
}
|
|
830
|
-
}
|
|
831
|
-
|
|
832
|
-
if (overlayIcons.length > 0) {
|
|
833
|
-
return <div className={TreeDecoration.Styles.ICON_WRAPPER_CLASS}>{icon}{overlayIcons}</div>;
|
|
834
|
-
}
|
|
835
|
-
|
|
836
|
-
return icon;
|
|
837
|
-
}
|
|
838
|
-
|
|
839
|
-
/**
|
|
840
|
-
* Render the tree node tail decorations.
|
|
841
|
-
* @param node the tree node.
|
|
842
|
-
* @param props the node properties.
|
|
843
|
-
*/
|
|
844
|
-
protected renderTailDecorations(node: TreeNode, props: NodeProps): React.ReactNode {
|
|
845
|
-
const tailDecorations = this.getDecorationData(node, 'tailDecorations').reduce((acc, current) => acc.concat(current), []);
|
|
846
|
-
if (tailDecorations.length === 0) {
|
|
847
|
-
return;
|
|
848
|
-
}
|
|
849
|
-
return this.renderTailDecorationsForNode(node, props, tailDecorations);
|
|
850
|
-
}
|
|
851
|
-
|
|
852
|
-
protected renderTailDecorationsForNode(node: TreeNode, props: NodeProps, tailDecorations: TreeDecoration.TailDecoration.AnyPartial[]): React.ReactNode {
|
|
853
|
-
let dotDecoration: TreeDecoration.TailDecoration.AnyPartial | undefined;
|
|
854
|
-
const otherDecorations: TreeDecoration.TailDecoration.AnyPartial[] = [];
|
|
855
|
-
tailDecorations.reverse().forEach(decoration => {
|
|
856
|
-
if (TreeDecoration.TailDecoration.isDotDecoration(decoration)) {
|
|
857
|
-
dotDecoration ||= decoration;
|
|
858
|
-
} else if (decoration.data || decoration.icon || decoration.iconClass) {
|
|
859
|
-
otherDecorations.push(decoration);
|
|
860
|
-
}
|
|
861
|
-
});
|
|
862
|
-
const decorationsToRender = dotDecoration ? [dotDecoration, ...otherDecorations] : otherDecorations;
|
|
863
|
-
return <React.Fragment>
|
|
864
|
-
{decorationsToRender.map((decoration, index) => {
|
|
865
|
-
const { tooltip, data, fontData, color, icon, iconClass } = decoration;
|
|
866
|
-
const iconToRender = icon ?? iconClass;
|
|
867
|
-
const className = [TREE_NODE_SEGMENT_CLASS, TREE_NODE_TAIL_CLASS, 'flex'].join(' ');
|
|
868
|
-
const style = fontData ? this.applyFontStyles({}, fontData) : color ? { color } : undefined;
|
|
869
|
-
const content = data ? data : iconToRender
|
|
870
|
-
? <span
|
|
871
|
-
key={node.id + 'icon' + index}
|
|
872
|
-
className={this.getIconClass(iconToRender, iconToRender === 'circle' ? [TreeDecoration.Styles.DECORATOR_SIZE_CLASS] : [])}
|
|
873
|
-
></span>
|
|
874
|
-
: '';
|
|
875
|
-
return <div key={node.id + className + index} className={className} style={style} title={tooltip}>
|
|
876
|
-
{content}{index !== decorationsToRender.length - 1 ? ',' : ''}
|
|
877
|
-
</div>;
|
|
878
|
-
})}
|
|
879
|
-
</React.Fragment>;
|
|
880
|
-
}
|
|
881
|
-
|
|
882
|
-
/**
|
|
883
|
-
* Determine the classes to use for an icon
|
|
884
|
-
* - Assumes a Font Awesome name when passed a single string, otherwise uses the passed string array
|
|
885
|
-
* @param iconName the icon name or list of icon names.
|
|
886
|
-
* @param additionalClasses additional CSS classes.
|
|
887
|
-
*
|
|
888
|
-
* @returns the icon class name.
|
|
889
|
-
*/
|
|
890
|
-
protected getIconClass(iconName: string | string[], additionalClasses: string[] = []): string {
|
|
891
|
-
const iconClass = (typeof iconName === 'string') ? ['a', 'fa', `fa-${iconName}`] : ['a'].concat(iconName);
|
|
892
|
-
return iconClass.concat(additionalClasses).join(' ');
|
|
893
|
-
}
|
|
894
|
-
|
|
895
|
-
/**
|
|
896
|
-
* Render indent for the file tree based on the depth
|
|
897
|
-
* @param node the tree node.
|
|
898
|
-
* @param depth the depth of the tree node.
|
|
899
|
-
*/
|
|
900
|
-
protected renderIndent(node: TreeNode, props: NodeProps): React.ReactNode {
|
|
901
|
-
const renderIndentGuides = this.corePreferences['workbench.tree.renderIndentGuides'];
|
|
902
|
-
if (renderIndentGuides === 'none') {
|
|
903
|
-
return undefined;
|
|
904
|
-
}
|
|
905
|
-
|
|
906
|
-
const indentDivs: React.ReactNode[] = [];
|
|
907
|
-
let current: TreeNode | undefined = node;
|
|
908
|
-
let depth = props.depth;
|
|
909
|
-
while (current && depth) {
|
|
910
|
-
if (this.shouldRenderIndent(current)) {
|
|
911
|
-
const classNames: string[] = [TREE_NODE_INDENT_GUIDE_CLASS];
|
|
912
|
-
if (this.needsActiveIndentGuideline(current)) {
|
|
913
|
-
classNames.push('active');
|
|
914
|
-
} else {
|
|
915
|
-
classNames.push(renderIndentGuides === 'onHover' ? 'hover' : 'always');
|
|
916
|
-
}
|
|
917
|
-
const paddingLeft = this.getDepthPadding(depth);
|
|
918
|
-
indentDivs.unshift(<div key={depth} className={classNames.join(' ')} style={{
|
|
919
|
-
paddingLeft: `${paddingLeft}px`
|
|
920
|
-
}} />);
|
|
921
|
-
depth--;
|
|
922
|
-
}
|
|
923
|
-
current = current.parent;
|
|
924
|
-
}
|
|
925
|
-
return indentDivs;
|
|
926
|
-
}
|
|
927
|
-
|
|
928
|
-
/**
|
|
929
|
-
* Determines whether an indentation div should be rendered for the specified tree node.
|
|
930
|
-
* If there are multiple tree nodes inside of a single rendered row,
|
|
931
|
-
* this method should only return true for the first node.
|
|
932
|
-
*/
|
|
933
|
-
protected shouldRenderIndent(node: TreeNode): boolean {
|
|
934
|
-
return true;
|
|
935
|
-
}
|
|
936
|
-
|
|
937
|
-
protected needsActiveIndentGuideline(node: TreeNode): boolean {
|
|
938
|
-
const parent = node.parent;
|
|
939
|
-
if (!parent || !this.isExpandable(parent)) {
|
|
940
|
-
return false;
|
|
941
|
-
}
|
|
942
|
-
if (SelectableTreeNode.isSelected(parent)) {
|
|
943
|
-
return true;
|
|
944
|
-
}
|
|
945
|
-
if (parent.expanded) {
|
|
946
|
-
for (const sibling of parent.children) {
|
|
947
|
-
if (SelectableTreeNode.isSelected(sibling) && !(this.isExpandable(sibling) && sibling.expanded)) {
|
|
948
|
-
return true;
|
|
949
|
-
}
|
|
950
|
-
}
|
|
951
|
-
}
|
|
952
|
-
return false;
|
|
953
|
-
}
|
|
954
|
-
|
|
955
|
-
/**
|
|
956
|
-
* Render the node given the tree node and node properties.
|
|
957
|
-
* @param node the tree node.
|
|
958
|
-
* @param props the node properties.
|
|
959
|
-
*/
|
|
960
|
-
protected renderNode(node: TreeNode, props: NodeProps): React.ReactNode {
|
|
961
|
-
if (!TreeNode.isVisible(node)) {
|
|
962
|
-
return undefined;
|
|
963
|
-
}
|
|
964
|
-
const attributes = this.createNodeAttributes(node, props);
|
|
965
|
-
const content = <div className={TREE_NODE_CONTENT_CLASS}>
|
|
966
|
-
{this.renderExpansionToggle(node, props)}
|
|
967
|
-
{this.renderCheckbox(node, props)}
|
|
968
|
-
{this.decorateIcon(node, this.renderIcon(node, props))}
|
|
969
|
-
{this.renderCaptionAffixes(node, props, 'captionPrefixes')}
|
|
970
|
-
{this.renderCaption(node, props)}
|
|
971
|
-
{this.renderCaptionAffixes(node, props, 'captionSuffixes')}
|
|
972
|
-
{this.renderTailDecorations(node, props)}
|
|
973
|
-
</div>;
|
|
974
|
-
return React.createElement('div', attributes, content);
|
|
975
|
-
}
|
|
976
|
-
|
|
977
|
-
/**
|
|
978
|
-
* Create node attributes for the tree node given the node properties.
|
|
979
|
-
* @param node the tree node.
|
|
980
|
-
* @param props the node properties.
|
|
981
|
-
*/
|
|
982
|
-
protected createNodeAttributes(node: TreeNode, props: NodeProps): React.Attributes & React.HTMLAttributes<HTMLElement> {
|
|
983
|
-
const className = this.createNodeClassNames(node, props).join(' ');
|
|
984
|
-
const style = this.createNodeStyle(node, props);
|
|
985
|
-
return {
|
|
986
|
-
className,
|
|
987
|
-
style,
|
|
988
|
-
onClick: event => this.handleClickEvent(node, event),
|
|
989
|
-
onDoubleClick: event => this.handleDblClickEvent(node, event),
|
|
990
|
-
onAuxClick: event => this.handleAuxClickEvent(node, event),
|
|
991
|
-
onContextMenu: event => this.handleContextMenuEvent(node, event),
|
|
992
|
-
};
|
|
993
|
-
}
|
|
994
|
-
|
|
995
|
-
/**
|
|
996
|
-
* Create the node class names.
|
|
997
|
-
* @param node the tree node.
|
|
998
|
-
* @param props the node properties.
|
|
999
|
-
*
|
|
1000
|
-
* @returns the list of tree node class names.
|
|
1001
|
-
*/
|
|
1002
|
-
protected createNodeClassNames(node: TreeNode, props: NodeProps): string[] {
|
|
1003
|
-
const classNames = [TREE_NODE_CLASS];
|
|
1004
|
-
if (CompositeTreeNode.is(node)) {
|
|
1005
|
-
classNames.push(COMPOSITE_TREE_NODE_CLASS);
|
|
1006
|
-
}
|
|
1007
|
-
if (this.isExpandable(node)) {
|
|
1008
|
-
classNames.push(EXPANDABLE_TREE_NODE_CLASS);
|
|
1009
|
-
}
|
|
1010
|
-
if (this.rowIsSelected(node, props)) {
|
|
1011
|
-
classNames.push(SELECTED_CLASS);
|
|
1012
|
-
}
|
|
1013
|
-
if (this.focusService.hasFocus(node)) {
|
|
1014
|
-
classNames.push(FOCUS_CLASS);
|
|
1015
|
-
}
|
|
1016
|
-
return classNames;
|
|
1017
|
-
}
|
|
1018
|
-
|
|
1019
|
-
protected rowIsSelected(node: TreeNode, props: NodeProps): boolean {
|
|
1020
|
-
return SelectableTreeNode.isSelected(node);
|
|
1021
|
-
}
|
|
1022
|
-
|
|
1023
|
-
/**
|
|
1024
|
-
* Get the default node style.
|
|
1025
|
-
* @param node the tree node.
|
|
1026
|
-
* @param props the node properties.
|
|
1027
|
-
*
|
|
1028
|
-
* @returns the CSS properties if available.
|
|
1029
|
-
*/
|
|
1030
|
-
protected getDefaultNodeStyle(node: TreeNode, props: NodeProps): React.CSSProperties | undefined {
|
|
1031
|
-
const paddingLeft = this.getPaddingLeft(node, props) + 'px';
|
|
1032
|
-
return { paddingLeft };
|
|
1033
|
-
}
|
|
1034
|
-
|
|
1035
|
-
protected getPaddingLeft(node: TreeNode, props: NodeProps): number {
|
|
1036
|
-
return this.getDepthPadding(props.depth) + (this.needsExpansionTogglePadding(node) ? this.props.expansionTogglePadding : 0);
|
|
1037
|
-
}
|
|
1038
|
-
|
|
1039
|
-
/**
|
|
1040
|
-
* If the node is a composite, a toggle will be rendered.
|
|
1041
|
-
* Otherwise we need to add the width and the left, right padding => 18px
|
|
1042
|
-
*/
|
|
1043
|
-
protected needsExpansionTogglePadding(node: TreeNode): boolean {
|
|
1044
|
-
return !this.isExpandable(node);
|
|
1045
|
-
}
|
|
1046
|
-
|
|
1047
|
-
/**
|
|
1048
|
-
* Create the tree node style.
|
|
1049
|
-
* @param node the tree node.
|
|
1050
|
-
* @param props the node properties.
|
|
1051
|
-
*/
|
|
1052
|
-
protected createNodeStyle(node: TreeNode, props: NodeProps): React.CSSProperties | undefined {
|
|
1053
|
-
return this.decorateNodeStyle(node, this.getDefaultNodeStyle(node, props));
|
|
1054
|
-
}
|
|
1055
|
-
|
|
1056
|
-
/**
|
|
1057
|
-
* Decorate the node style.
|
|
1058
|
-
* @param node the tree node.
|
|
1059
|
-
* @param style the optional CSS properties.
|
|
1060
|
-
*
|
|
1061
|
-
* @returns the CSS styles if available.
|
|
1062
|
-
*/
|
|
1063
|
-
protected decorateNodeStyle(node: TreeNode, style: React.CSSProperties | undefined): React.CSSProperties | undefined {
|
|
1064
|
-
const backgroundColor = this.getDecorationData(node, 'backgroundColor').filter(notEmpty).shift();
|
|
1065
|
-
if (backgroundColor) {
|
|
1066
|
-
style = {
|
|
1067
|
-
...(style || {}),
|
|
1068
|
-
backgroundColor
|
|
1069
|
-
};
|
|
1070
|
-
}
|
|
1071
|
-
return style;
|
|
1072
|
-
}
|
|
1073
|
-
|
|
1074
|
-
/**
|
|
1075
|
-
* Determine if the tree node is expandable.
|
|
1076
|
-
* @param node the tree node.
|
|
1077
|
-
*
|
|
1078
|
-
* @returns `true` if the tree node is expandable.
|
|
1079
|
-
*/
|
|
1080
|
-
protected isExpandable(node: TreeNode): node is ExpandableTreeNode {
|
|
1081
|
-
return ExpandableTreeNode.is(node);
|
|
1082
|
-
}
|
|
1083
|
-
|
|
1084
|
-
/**
|
|
1085
|
-
* Get the tree node decorations.
|
|
1086
|
-
* @param node the tree node.
|
|
1087
|
-
*
|
|
1088
|
-
* @returns the list of tree decoration data.
|
|
1089
|
-
*/
|
|
1090
|
-
protected getDecorations(node: TreeNode): TreeDecoration.Data[] {
|
|
1091
|
-
const decorations: TreeDecoration.Data[] = [];
|
|
1092
|
-
if (DecoratedTreeNode.is(node)) {
|
|
1093
|
-
decorations.push(node.decorationData);
|
|
1094
|
-
}
|
|
1095
|
-
if (this.decorations.has(node.id)) {
|
|
1096
|
-
decorations.push(...this.decorations.get(node.id)!);
|
|
1097
|
-
}
|
|
1098
|
-
return decorations.sort(TreeDecoration.Data.comparePriority);
|
|
1099
|
-
}
|
|
1100
|
-
|
|
1101
|
-
/**
|
|
1102
|
-
* Get the tree decoration data for the given key.
|
|
1103
|
-
* @param node the tree node.
|
|
1104
|
-
* @param key the tree decoration data key.
|
|
1105
|
-
*
|
|
1106
|
-
* @returns the tree decoration data at the given key.
|
|
1107
|
-
*/
|
|
1108
|
-
protected getDecorationData<K extends keyof TreeDecoration.Data>(node: TreeNode, key: K): Required<Pick<TreeDecoration.Data, K>>[K][] {
|
|
1109
|
-
return this.getDecorations(node).filter(data => data[key] !== undefined).map(data => data[key]);
|
|
1110
|
-
}
|
|
1111
|
-
|
|
1112
|
-
/**
|
|
1113
|
-
* Store the last scroll state.
|
|
1114
|
-
*/
|
|
1115
|
-
protected lastScrollState: {
|
|
1116
|
-
/**
|
|
1117
|
-
* The scroll top value.
|
|
1118
|
-
*/
|
|
1119
|
-
scrollTop: number,
|
|
1120
|
-
/**
|
|
1121
|
-
* The scroll left value.
|
|
1122
|
-
*/
|
|
1123
|
-
scrollLeft: number
|
|
1124
|
-
} | undefined;
|
|
1125
|
-
|
|
1126
|
-
/**
|
|
1127
|
-
* Get the scroll container.
|
|
1128
|
-
*/
|
|
1129
|
-
protected override getScrollContainer(): MaybePromise<HTMLElement> {
|
|
1130
|
-
this.toDisposeOnDetach.push(Disposable.create(() => {
|
|
1131
|
-
const { scrollTop, scrollLeft } = this.node;
|
|
1132
|
-
this.lastScrollState = { scrollTop, scrollLeft };
|
|
1133
|
-
}));
|
|
1134
|
-
if (this.lastScrollState) {
|
|
1135
|
-
const { scrollTop, scrollLeft } = this.lastScrollState;
|
|
1136
|
-
this.node.scrollTop = scrollTop;
|
|
1137
|
-
this.node.scrollLeft = scrollLeft;
|
|
1138
|
-
}
|
|
1139
|
-
return this.node;
|
|
1140
|
-
}
|
|
1141
|
-
|
|
1142
|
-
protected override onAfterAttach(msg: Message): void {
|
|
1143
|
-
const up = [
|
|
1144
|
-
Key.ARROW_UP,
|
|
1145
|
-
KeyCode.createKeyCode({ first: Key.ARROW_UP, modifiers: [KeyModifier.Shift] })
|
|
1146
|
-
];
|
|
1147
|
-
const down = [
|
|
1148
|
-
Key.ARROW_DOWN,
|
|
1149
|
-
KeyCode.createKeyCode({ first: Key.ARROW_DOWN, modifiers: [KeyModifier.Shift] })
|
|
1150
|
-
];
|
|
1151
|
-
if (this.props.search) {
|
|
1152
|
-
if (this.searchBox.isAttached) {
|
|
1153
|
-
Widget.detach(this.searchBox);
|
|
1154
|
-
}
|
|
1155
|
-
UnsafeWidgetUtilities.attach(this.searchBox, this.node.parentElement!);
|
|
1156
|
-
this.addKeyListener(this.node, this.searchBox.keyCodePredicate.bind(this.searchBox), this.searchBox.handle.bind(this.searchBox));
|
|
1157
|
-
this.toDisposeOnDetach.push(Disposable.create(() => {
|
|
1158
|
-
Widget.detach(this.searchBox);
|
|
1159
|
-
}));
|
|
1160
|
-
}
|
|
1161
|
-
super.onAfterAttach(msg);
|
|
1162
|
-
this.addKeyListener(this.node, Key.ARROW_LEFT, event => this.handleLeft(event));
|
|
1163
|
-
this.addKeyListener(this.node, Key.ARROW_RIGHT, event => this.handleRight(event));
|
|
1164
|
-
this.addKeyListener(this.node, up, event => this.handleUp(event));
|
|
1165
|
-
this.addKeyListener(this.node, down, event => this.handleDown(event));
|
|
1166
|
-
this.addKeyListener(this.node, Key.ENTER, event => this.handleEnter(event));
|
|
1167
|
-
this.addKeyListener(this.node, Key.SPACE, event => this.handleSpace(event));
|
|
1168
|
-
this.addKeyListener(this.node, Key.ESCAPE, event => this.handleEscape(event));
|
|
1169
|
-
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
1170
|
-
this.addEventListener<any>(this.node, 'ps-scroll-y', (e: Event & { target: { scrollTop: number } }) => {
|
|
1171
|
-
if (this.view && this.view.list) {
|
|
1172
|
-
const { scrollTop } = e.target;
|
|
1173
|
-
this.view.list.scrollTo({
|
|
1174
|
-
top: scrollTop
|
|
1175
|
-
});
|
|
1176
|
-
}
|
|
1177
|
-
});
|
|
1178
|
-
}
|
|
1179
|
-
|
|
1180
|
-
/**
|
|
1181
|
-
* Handle the `left arrow` keyboard event.
|
|
1182
|
-
* @param event the `left arrow` keyboard event.
|
|
1183
|
-
*/
|
|
1184
|
-
protected async handleLeft(event: KeyboardEvent): Promise<void> {
|
|
1185
|
-
if (!!this.props.multiSelect && (this.hasCtrlCmdMask(event) || this.hasShiftMask(event))) {
|
|
1186
|
-
return;
|
|
1187
|
-
}
|
|
1188
|
-
if (!await this.model.collapseNode()) {
|
|
1189
|
-
this.model.selectParent();
|
|
1190
|
-
}
|
|
1191
|
-
}
|
|
1192
|
-
|
|
1193
|
-
/**
|
|
1194
|
-
* Handle the `right arrow` keyboard event.
|
|
1195
|
-
* @param event the `right arrow` keyboard event.
|
|
1196
|
-
*/
|
|
1197
|
-
protected async handleRight(event: KeyboardEvent): Promise<void> {
|
|
1198
|
-
if (!!this.props.multiSelect && (this.hasCtrlCmdMask(event) || this.hasShiftMask(event))) {
|
|
1199
|
-
return;
|
|
1200
|
-
}
|
|
1201
|
-
if (!await this.model.expandNode()) {
|
|
1202
|
-
this.model.selectNextNode();
|
|
1203
|
-
}
|
|
1204
|
-
}
|
|
1205
|
-
|
|
1206
|
-
/**
|
|
1207
|
-
* Handle the `up arrow` keyboard event.
|
|
1208
|
-
* @param event the `up arrow` keyboard event.
|
|
1209
|
-
*/
|
|
1210
|
-
protected handleUp(event: KeyboardEvent): void {
|
|
1211
|
-
if (!!this.props.multiSelect && this.hasShiftMask(event)) {
|
|
1212
|
-
this.model.selectPrevNode(TreeSelection.SelectionType.RANGE);
|
|
1213
|
-
} else {
|
|
1214
|
-
this.model.selectPrevNode();
|
|
1215
|
-
}
|
|
1216
|
-
this.node.focus();
|
|
1217
|
-
}
|
|
1218
|
-
|
|
1219
|
-
/**
|
|
1220
|
-
* Handle the `down arrow` keyboard event.
|
|
1221
|
-
* @param event the `down arrow` keyboard event.
|
|
1222
|
-
*/
|
|
1223
|
-
protected handleDown(event: KeyboardEvent): void {
|
|
1224
|
-
if (!!this.props.multiSelect && this.hasShiftMask(event)) {
|
|
1225
|
-
this.model.selectNextNode(TreeSelection.SelectionType.RANGE);
|
|
1226
|
-
} else {
|
|
1227
|
-
this.model.selectNextNode();
|
|
1228
|
-
}
|
|
1229
|
-
this.node.focus();
|
|
1230
|
-
}
|
|
1231
|
-
|
|
1232
|
-
/**
|
|
1233
|
-
* Handle the `enter key` keyboard event.
|
|
1234
|
-
* - `enter` opens the tree node.
|
|
1235
|
-
* @param event the `enter key` keyboard event.
|
|
1236
|
-
*/
|
|
1237
|
-
protected handleEnter(event: KeyboardEvent): void {
|
|
1238
|
-
this.model.openNode();
|
|
1239
|
-
}
|
|
1240
|
-
|
|
1241
|
-
/**
|
|
1242
|
-
* Handle the `space key` keyboard event.
|
|
1243
|
-
* - If the element has a checkbox, it will be toggled.
|
|
1244
|
-
* - Otherwise, it should be similar to a single-click action.
|
|
1245
|
-
* @param event the `space key` keyboard event.
|
|
1246
|
-
*/
|
|
1247
|
-
protected handleSpace(event: KeyboardEvent): void {
|
|
1248
|
-
const { focusedNode } = this.focusService;
|
|
1249
|
-
if (focusedNode && focusedNode.checkboxInfo) {
|
|
1250
|
-
this.model.markAsChecked(focusedNode, !focusedNode.checkboxInfo.checked);
|
|
1251
|
-
} else if (!this.props.multiSelect || (!event.ctrlKey && !event.metaKey && !event.shiftKey)) {
|
|
1252
|
-
this.tapNode(focusedNode);
|
|
1253
|
-
}
|
|
1254
|
-
}
|
|
1255
|
-
|
|
1256
|
-
protected handleEscape(event: KeyboardEvent): void {
|
|
1257
|
-
if (this.model.selectedNodes.length <= 1) {
|
|
1258
|
-
this.focusService.setFocus(undefined);
|
|
1259
|
-
this.node.focus();
|
|
1260
|
-
}
|
|
1261
|
-
this.model.clearSelection();
|
|
1262
|
-
}
|
|
1263
|
-
|
|
1264
|
-
/**
|
|
1265
|
-
* Handle the single-click mouse event.
|
|
1266
|
-
* @param node the tree node if available.
|
|
1267
|
-
* @param event the mouse single-click event.
|
|
1268
|
-
*/
|
|
1269
|
-
protected handleClickEvent(node: TreeNode | undefined, event: React.MouseEvent<HTMLElement>): void {
|
|
1270
|
-
if (node) {
|
|
1271
|
-
event.stopPropagation();
|
|
1272
|
-
const shiftMask = this.hasShiftMask(event);
|
|
1273
|
-
const ctrlCmdMask = this.hasCtrlCmdMask(event);
|
|
1274
|
-
if (this.props.multiSelect && (shiftMask || ctrlCmdMask) && SelectableTreeNode.is(node)) {
|
|
1275
|
-
if (shiftMask) {
|
|
1276
|
-
this.model.selectRange(node);
|
|
1277
|
-
} else if (ctrlCmdMask) {
|
|
1278
|
-
this.model.toggleNode(node);
|
|
1279
|
-
}
|
|
1280
|
-
} else {
|
|
1281
|
-
this.tapNode(node);
|
|
1282
|
-
}
|
|
1283
|
-
}
|
|
1284
|
-
}
|
|
1285
|
-
|
|
1286
|
-
/**
|
|
1287
|
-
* The effective handler of an unmodified single-click event.
|
|
1288
|
-
*/
|
|
1289
|
-
protected tapNode(node?: TreeNode): void {
|
|
1290
|
-
if (SelectableTreeNode.is(node)) {
|
|
1291
|
-
this.model.selectNode(node);
|
|
1292
|
-
}
|
|
1293
|
-
if (node && !this.props.expandOnlyOnExpansionToggleClick && this.isExpandable(node)) {
|
|
1294
|
-
this.model.toggleNodeExpansion(node);
|
|
1295
|
-
}
|
|
1296
|
-
}
|
|
1297
|
-
|
|
1298
|
-
/**
|
|
1299
|
-
* Handle the double-click mouse event.
|
|
1300
|
-
* @param node the tree node if available.
|
|
1301
|
-
* @param event the double-click mouse event.
|
|
1302
|
-
*/
|
|
1303
|
-
protected handleDblClickEvent(node: TreeNode | undefined, event: React.MouseEvent<HTMLElement>): void {
|
|
1304
|
-
this.model.openNode(node);
|
|
1305
|
-
event.stopPropagation();
|
|
1306
|
-
}
|
|
1307
|
-
|
|
1308
|
-
/**
|
|
1309
|
-
* Handle the middle-click mouse event.
|
|
1310
|
-
* @param node the tree node if available.
|
|
1311
|
-
* @param event the middle-click mouse event.
|
|
1312
|
-
*/
|
|
1313
|
-
protected handleAuxClickEvent(node: TreeNode | undefined, event: React.MouseEvent<HTMLElement>): void {
|
|
1314
|
-
if (event.button === 1) {
|
|
1315
|
-
this.model.openNode(node);
|
|
1316
|
-
if (SelectableTreeNode.is(node)) {
|
|
1317
|
-
this.model.selectNode(node);
|
|
1318
|
-
}
|
|
1319
|
-
}
|
|
1320
|
-
event.stopPropagation();
|
|
1321
|
-
}
|
|
1322
|
-
|
|
1323
|
-
/**
|
|
1324
|
-
* Handle the middle-click mouse event.
|
|
1325
|
-
* @param event the middle-click mouse event.
|
|
1326
|
-
*/
|
|
1327
|
-
protected handleMiddleClickEvent(event: MouseEvent): void {
|
|
1328
|
-
// Prevents auto-scrolling behavior when middle-clicking.
|
|
1329
|
-
if (event.button === 1) {
|
|
1330
|
-
event.preventDefault();
|
|
1331
|
-
}
|
|
1332
|
-
}
|
|
1333
|
-
|
|
1334
|
-
/**
|
|
1335
|
-
* Handle the context menu click event.
|
|
1336
|
-
* - The context menu click event is triggered by the right-click.
|
|
1337
|
-
* @param node the tree node if available.
|
|
1338
|
-
* @param event the right-click mouse event.
|
|
1339
|
-
*/
|
|
1340
|
-
protected handleContextMenuEvent(node: TreeNode | undefined, event: React.MouseEvent<HTMLElement>): void {
|
|
1341
|
-
if (SelectableTreeNode.is(node)) {
|
|
1342
|
-
// Keep the selection for the context menu, if the widget support multi-selection and the right click happens on an already selected node.
|
|
1343
|
-
if (!this.props.multiSelect || !node.selected) {
|
|
1344
|
-
const type = !!this.props.multiSelect && this.hasCtrlCmdMask(event) ? TreeSelection.SelectionType.TOGGLE : TreeSelection.SelectionType.DEFAULT;
|
|
1345
|
-
this.model.addSelection({ node, type });
|
|
1346
|
-
}
|
|
1347
|
-
this.focusService.setFocus(node);
|
|
1348
|
-
const contextMenuPath = this.props.contextMenuPath;
|
|
1349
|
-
if (contextMenuPath) {
|
|
1350
|
-
const { x, y } = event.nativeEvent;
|
|
1351
|
-
const args = this.toContextMenuArgs(node);
|
|
1352
|
-
setTimeout(() => this.contextMenuRenderer.render({
|
|
1353
|
-
menuPath: contextMenuPath,
|
|
1354
|
-
anchor: { x, y },
|
|
1355
|
-
args
|
|
1356
|
-
}), 10);
|
|
1357
|
-
}
|
|
1358
|
-
}
|
|
1359
|
-
event.stopPropagation();
|
|
1360
|
-
event.preventDefault();
|
|
1361
|
-
}
|
|
1362
|
-
|
|
1363
|
-
/**
|
|
1364
|
-
* Handle the double-click mouse event on the expansion toggle.
|
|
1365
|
-
*/
|
|
1366
|
-
protected readonly handleExpansionToggleDblClickEvent = (event: React.MouseEvent<HTMLElement>) => this.doHandleExpansionToggleDblClickEvent(event);
|
|
1367
|
-
/**
|
|
1368
|
-
* Actually handle the double-click mouse event on the expansion toggle.
|
|
1369
|
-
* @param event the double-click mouse event.
|
|
1370
|
-
*/
|
|
1371
|
-
protected doHandleExpansionToggleDblClickEvent(event: React.MouseEvent<HTMLElement>): void {
|
|
1372
|
-
if (this.props.expandOnlyOnExpansionToggleClick) {
|
|
1373
|
-
// Ignore the double-click event.
|
|
1374
|
-
event.stopPropagation();
|
|
1375
|
-
}
|
|
1376
|
-
}
|
|
1377
|
-
|
|
1378
|
-
/**
|
|
1379
|
-
* Convert the tree node to context menu arguments.
|
|
1380
|
-
* @param node the selectable tree node.
|
|
1381
|
-
*/
|
|
1382
|
-
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
1383
|
-
protected toContextMenuArgs(node: SelectableTreeNode): any[] | undefined {
|
|
1384
|
-
return undefined;
|
|
1385
|
-
}
|
|
1386
|
-
|
|
1387
|
-
/**
|
|
1388
|
-
* Determine if the tree modifier aware event has a `ctrlcmd` mask.
|
|
1389
|
-
* @param event the tree modifier aware event.
|
|
1390
|
-
*
|
|
1391
|
-
* @returns `true` if the tree modifier aware event contains the `ctrlcmd` mask.
|
|
1392
|
-
*/
|
|
1393
|
-
protected hasCtrlCmdMask(event: TreeWidget.ModifierAwareEvent): boolean {
|
|
1394
|
-
return isOSX ? event.metaKey : event.ctrlKey;
|
|
1395
|
-
}
|
|
1396
|
-
|
|
1397
|
-
/**
|
|
1398
|
-
* Determine if the tree modifier aware event has a `shift` mask.
|
|
1399
|
-
* @param event the tree modifier aware event.
|
|
1400
|
-
*
|
|
1401
|
-
* @returns `true` if the tree modifier aware event contains the `shift` mask.
|
|
1402
|
-
*/
|
|
1403
|
-
protected hasShiftMask(event: TreeWidget.ModifierAwareEvent): boolean {
|
|
1404
|
-
// Ctrl/Cmd mask overrules the Shift mask.
|
|
1405
|
-
if (this.hasCtrlCmdMask(event)) {
|
|
1406
|
-
return false;
|
|
1407
|
-
}
|
|
1408
|
-
return event.shiftKey;
|
|
1409
|
-
}
|
|
1410
|
-
|
|
1411
|
-
/**
|
|
1412
|
-
* Deflate the tree node for storage.
|
|
1413
|
-
* @param node the tree node.
|
|
1414
|
-
*/
|
|
1415
|
-
protected deflateForStorage(node: TreeNode): object {
|
|
1416
|
-
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
1417
|
-
const copy = Object.assign({}, node) as any;
|
|
1418
|
-
if (copy.parent) {
|
|
1419
|
-
delete copy.parent;
|
|
1420
|
-
}
|
|
1421
|
-
if ('previousSibling' in copy) {
|
|
1422
|
-
delete copy.previousSibling;
|
|
1423
|
-
}
|
|
1424
|
-
if ('nextSibling' in copy) {
|
|
1425
|
-
delete copy.nextSibling;
|
|
1426
|
-
}
|
|
1427
|
-
if ('busy' in copy) {
|
|
1428
|
-
delete copy.busy;
|
|
1429
|
-
}
|
|
1430
|
-
if (CompositeTreeNode.is(node)) {
|
|
1431
|
-
copy.children = [];
|
|
1432
|
-
for (const child of node.children) {
|
|
1433
|
-
copy.children.push(this.deflateForStorage(child));
|
|
1434
|
-
}
|
|
1435
|
-
}
|
|
1436
|
-
return copy;
|
|
1437
|
-
}
|
|
1438
|
-
|
|
1439
|
-
/**
|
|
1440
|
-
* Inflate the tree node from storage.
|
|
1441
|
-
* @param node the tree node.
|
|
1442
|
-
* @param parent the optional tree node.
|
|
1443
|
-
*/
|
|
1444
|
-
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
1445
|
-
protected inflateFromStorage(node: any, parent?: TreeNode): TreeNode {
|
|
1446
|
-
if (node.selected) {
|
|
1447
|
-
node.selected = false;
|
|
1448
|
-
}
|
|
1449
|
-
if (parent) {
|
|
1450
|
-
node.parent = parent;
|
|
1451
|
-
}
|
|
1452
|
-
if (Array.isArray(node.children)) {
|
|
1453
|
-
for (const child of node.children as TreeNode[]) {
|
|
1454
|
-
this.inflateFromStorage(child, node);
|
|
1455
|
-
}
|
|
1456
|
-
}
|
|
1457
|
-
return node;
|
|
1458
|
-
}
|
|
1459
|
-
|
|
1460
|
-
/**
|
|
1461
|
-
* Store the tree state.
|
|
1462
|
-
*/
|
|
1463
|
-
storeState(): object {
|
|
1464
|
-
const decorations = this.decoratorService.deflateDecorators(this.decorations);
|
|
1465
|
-
let state: object = {
|
|
1466
|
-
decorations
|
|
1467
|
-
};
|
|
1468
|
-
if (this.model.root) {
|
|
1469
|
-
state = {
|
|
1470
|
-
...state,
|
|
1471
|
-
root: this.deflateForStorage(this.model.root),
|
|
1472
|
-
model: this.model.storeState(),
|
|
1473
|
-
focusedNodeId: this.focusService.focusedNode?.id
|
|
1474
|
-
};
|
|
1475
|
-
}
|
|
1476
|
-
|
|
1477
|
-
return state;
|
|
1478
|
-
}
|
|
1479
|
-
|
|
1480
|
-
/**
|
|
1481
|
-
* Restore the state.
|
|
1482
|
-
* @param oldState the old state object.
|
|
1483
|
-
*/
|
|
1484
|
-
restoreState(oldState: object): void {
|
|
1485
|
-
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
1486
|
-
const { root, decorations, model, focusedNodeId } = (oldState as any);
|
|
1487
|
-
if (root) {
|
|
1488
|
-
this.model.root = this.inflateFromStorage(root);
|
|
1489
|
-
}
|
|
1490
|
-
if (decorations) {
|
|
1491
|
-
this.decorations = this.decoratorService.inflateDecorators(decorations);
|
|
1492
|
-
}
|
|
1493
|
-
if (model) {
|
|
1494
|
-
this.model.restoreState(model);
|
|
1495
|
-
}
|
|
1496
|
-
if (focusedNodeId) {
|
|
1497
|
-
const candidate = this.model.getNode(focusedNodeId);
|
|
1498
|
-
if (SelectableTreeNode.is(candidate)) {
|
|
1499
|
-
this.focusService.setFocus(candidate);
|
|
1500
|
-
}
|
|
1501
|
-
}
|
|
1502
|
-
}
|
|
1503
|
-
|
|
1504
|
-
protected toNodeIcon(node: TreeNode): string {
|
|
1505
|
-
return this.labelProvider.getIcon(node);
|
|
1506
|
-
}
|
|
1507
|
-
|
|
1508
|
-
protected toNodeName(node: TreeNode): string {
|
|
1509
|
-
return this.labelProvider.getName(node);
|
|
1510
|
-
}
|
|
1511
|
-
|
|
1512
|
-
protected toNodeDescription(node: TreeNode): string {
|
|
1513
|
-
return this.labelProvider.getLongName(node);
|
|
1514
|
-
}
|
|
1515
|
-
protected getDepthPadding(depth: number): number {
|
|
1516
|
-
if (depth === 1) {
|
|
1517
|
-
return this.props.leftPadding;
|
|
1518
|
-
}
|
|
1519
|
-
return depth * this.treeIndent;
|
|
1520
|
-
}
|
|
1521
|
-
}
|
|
1522
|
-
export namespace TreeWidget {
|
|
1523
|
-
/**
|
|
1524
|
-
* Representation of a tree node row.
|
|
1525
|
-
*/
|
|
1526
|
-
export interface NodeRow {
|
|
1527
|
-
/**
|
|
1528
|
-
* The node row index.
|
|
1529
|
-
*/
|
|
1530
|
-
index: number
|
|
1531
|
-
/**
|
|
1532
|
-
* The actual node.
|
|
1533
|
-
*/
|
|
1534
|
-
node: TreeNode
|
|
1535
|
-
/**
|
|
1536
|
-
* A root relative number representing the hierarchical depth of the actual node. Root is `0`, its children have `1` and so on.
|
|
1537
|
-
*/
|
|
1538
|
-
depth: number
|
|
1539
|
-
}
|
|
1540
|
-
/**
|
|
1541
|
-
* Representation of the tree view properties.
|
|
1542
|
-
*/
|
|
1543
|
-
export interface ViewProps {
|
|
1544
|
-
/**
|
|
1545
|
-
* The width property.
|
|
1546
|
-
*/
|
|
1547
|
-
width: number
|
|
1548
|
-
/**
|
|
1549
|
-
* The height property.
|
|
1550
|
-
*/
|
|
1551
|
-
height: number
|
|
1552
|
-
/**
|
|
1553
|
-
* The scroll to row value.
|
|
1554
|
-
*/
|
|
1555
|
-
scrollToRow?: number
|
|
1556
|
-
/**
|
|
1557
|
-
* The list of node rows.
|
|
1558
|
-
*/
|
|
1559
|
-
rows: NodeRow[]
|
|
1560
|
-
renderNodeRow: (row: NodeRow) => React.ReactNode
|
|
1561
|
-
}
|
|
1562
|
-
export class View extends React.Component<ViewProps> {
|
|
1563
|
-
list: VirtuosoHandle | undefined;
|
|
1564
|
-
override render(): React.ReactNode {
|
|
1565
|
-
const { rows, width, height, scrollToRow } = this.props;
|
|
1566
|
-
return <Virtuoso
|
|
1567
|
-
ref={list => {
|
|
1568
|
-
this.list = (list || undefined);
|
|
1569
|
-
if (this.list && scrollToRow !== undefined) {
|
|
1570
|
-
this.list.scrollIntoView({
|
|
1571
|
-
index: scrollToRow,
|
|
1572
|
-
align: 'center'
|
|
1573
|
-
});
|
|
1574
|
-
}
|
|
1575
|
-
}}
|
|
1576
|
-
totalCount={rows.length}
|
|
1577
|
-
itemContent={index => this.props.renderNodeRow(rows[index])}
|
|
1578
|
-
width={width}
|
|
1579
|
-
height={height}
|
|
1580
|
-
// This is a pixel value, it will scan 200px to the top and bottom of the current view
|
|
1581
|
-
overscan={500}
|
|
1582
|
-
/>;
|
|
1583
|
-
}
|
|
1584
|
-
}
|
|
1585
|
-
}
|
|
1
|
+
// *****************************************************************************
|
|
2
|
+
// Copyright (C) 2018 TypeFox and others.
|
|
3
|
+
//
|
|
4
|
+
// This program and the accompanying materials are made available under the
|
|
5
|
+
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
6
|
+
// http://www.eclipse.org/legal/epl-2.0.
|
|
7
|
+
//
|
|
8
|
+
// This Source Code may also be made available under the following Secondary
|
|
9
|
+
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
10
|
+
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
11
|
+
// with the GNU Classpath Exception which is available at
|
|
12
|
+
// https://www.gnu.org/software/classpath/license.html.
|
|
13
|
+
//
|
|
14
|
+
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
15
|
+
// *****************************************************************************
|
|
16
|
+
|
|
17
|
+
import { injectable, inject, postConstruct } from 'inversify';
|
|
18
|
+
import { Message } from '@phosphor/messaging';
|
|
19
|
+
import { Disposable, MenuPath, SelectionService } from '../../common';
|
|
20
|
+
import { Key, KeyCode, KeyModifier } from '../keyboard/keys';
|
|
21
|
+
import { ContextMenuRenderer } from '../context-menu-renderer';
|
|
22
|
+
import { StatefulWidget } from '../shell';
|
|
23
|
+
import {
|
|
24
|
+
EXPANSION_TOGGLE_CLASS, SELECTED_CLASS, COLLAPSED_CLASS, FOCUS_CLASS, BUSY_CLASS, CODICON_TREE_ITEM_CLASSES, CODICON_LOADING_CLASSES, Widget, UnsafeWidgetUtilities
|
|
25
|
+
} from '../widgets';
|
|
26
|
+
import { TreeNode, CompositeTreeNode } from './tree';
|
|
27
|
+
import { TreeModel } from './tree-model';
|
|
28
|
+
import { ExpandableTreeNode } from './tree-expansion';
|
|
29
|
+
import { SelectableTreeNode, TreeSelection } from './tree-selection';
|
|
30
|
+
import { TreeDecoratorService, TreeDecoration, DecoratedTreeNode } from './tree-decorator';
|
|
31
|
+
import { notEmpty } from '../../common/objects';
|
|
32
|
+
import { isOSX } from '../../common/os';
|
|
33
|
+
import { ReactWidget } from '../widgets/react-widget';
|
|
34
|
+
import * as React from 'react';
|
|
35
|
+
import { Virtuoso, VirtuosoHandle } from 'react-virtuoso';
|
|
36
|
+
import { TopDownTreeIterator } from './tree-iterator';
|
|
37
|
+
import { SearchBox, SearchBoxFactory, SearchBoxProps } from './search-box';
|
|
38
|
+
import { TreeSearch } from './tree-search';
|
|
39
|
+
import { ElementExt } from '@phosphor/domutils';
|
|
40
|
+
import { TreeWidgetSelection } from './tree-widget-selection';
|
|
41
|
+
import { MaybePromise } from '../../common/types';
|
|
42
|
+
import { LabelProvider } from '../label-provider';
|
|
43
|
+
import { CorePreferences } from '../core-preferences';
|
|
44
|
+
import { TreeFocusService } from './tree-focus-service';
|
|
45
|
+
import { useEffect } from 'react';
|
|
46
|
+
import { PreferenceService, PreferenceChange } from '../preferences';
|
|
47
|
+
import { PREFERENCE_NAME_TREE_INDENT } from './tree-preference';
|
|
48
|
+
|
|
49
|
+
const debounce = require('lodash.debounce');
|
|
50
|
+
|
|
51
|
+
export const TREE_CLASS = 'theia-Tree';
|
|
52
|
+
export const TREE_CONTAINER_CLASS = 'theia-TreeContainer';
|
|
53
|
+
export const TREE_NODE_CLASS = 'theia-TreeNode';
|
|
54
|
+
export const TREE_NODE_CONTENT_CLASS = 'theia-TreeNodeContent';
|
|
55
|
+
export const TREE_NODE_INFO_CLASS = 'theia-TreeNodeInfo';
|
|
56
|
+
export const TREE_NODE_TAIL_CLASS = 'theia-TreeNodeTail';
|
|
57
|
+
export const TREE_NODE_SEGMENT_CLASS = 'theia-TreeNodeSegment';
|
|
58
|
+
export const TREE_NODE_SEGMENT_GROW_CLASS = 'theia-TreeNodeSegmentGrow';
|
|
59
|
+
|
|
60
|
+
export const EXPANDABLE_TREE_NODE_CLASS = 'theia-ExpandableTreeNode';
|
|
61
|
+
export const COMPOSITE_TREE_NODE_CLASS = 'theia-CompositeTreeNode';
|
|
62
|
+
export const TREE_NODE_CAPTION_CLASS = 'theia-TreeNodeCaption';
|
|
63
|
+
export const TREE_NODE_INDENT_GUIDE_CLASS = 'theia-tree-node-indent';
|
|
64
|
+
|
|
65
|
+
export const TreeProps = Symbol('TreeProps');
|
|
66
|
+
|
|
67
|
+
/**
|
|
68
|
+
* Representation of tree properties.
|
|
69
|
+
*/
|
|
70
|
+
export interface TreeProps {
|
|
71
|
+
|
|
72
|
+
/**
|
|
73
|
+
* The path of the context menu that one can use to contribute context menu items to the tree widget.
|
|
74
|
+
*/
|
|
75
|
+
readonly contextMenuPath?: MenuPath;
|
|
76
|
+
|
|
77
|
+
/**
|
|
78
|
+
* The size of the padding (in pixels) for the root node of the tree.
|
|
79
|
+
*/
|
|
80
|
+
readonly leftPadding: number;
|
|
81
|
+
|
|
82
|
+
readonly expansionTogglePadding: number;
|
|
83
|
+
|
|
84
|
+
/**
|
|
85
|
+
* `true` if the tree widget support multi-selection. Otherwise, `false`. Defaults to `false`.
|
|
86
|
+
*/
|
|
87
|
+
readonly multiSelect?: boolean;
|
|
88
|
+
|
|
89
|
+
/**
|
|
90
|
+
* `true` if the tree widget support searching. Otherwise, `false`. Defaults to `false`.
|
|
91
|
+
*/
|
|
92
|
+
readonly search?: boolean
|
|
93
|
+
|
|
94
|
+
/**
|
|
95
|
+
* `true` if the tree widget should be virtualized searching. Otherwise, `false`. Defaults to `true`.
|
|
96
|
+
*/
|
|
97
|
+
readonly virtualized?: boolean
|
|
98
|
+
|
|
99
|
+
/**
|
|
100
|
+
* `true` if the selected node should be auto scrolled only if the widget is active. Otherwise, `false`. Defaults to `false`.
|
|
101
|
+
*/
|
|
102
|
+
readonly scrollIfActive?: boolean
|
|
103
|
+
|
|
104
|
+
/**
|
|
105
|
+
* `true` if a tree widget contributes to the global selection. Defaults to `false`.
|
|
106
|
+
*/
|
|
107
|
+
readonly globalSelection?: boolean;
|
|
108
|
+
|
|
109
|
+
/**
|
|
110
|
+
* `true` if the tree widget supports expansion only when clicking the expansion toggle. Defaults to `false`.
|
|
111
|
+
*/
|
|
112
|
+
readonly expandOnlyOnExpansionToggleClick?: boolean;
|
|
113
|
+
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
/**
|
|
117
|
+
* Representation of node properties.
|
|
118
|
+
*/
|
|
119
|
+
export interface NodeProps {
|
|
120
|
+
|
|
121
|
+
/**
|
|
122
|
+
* A root relative number representing the hierarchical depth of the actual node. Root is `0`, its children have `1` and so on.
|
|
123
|
+
*/
|
|
124
|
+
readonly depth: number;
|
|
125
|
+
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
/**
|
|
129
|
+
* The default tree properties.
|
|
130
|
+
*/
|
|
131
|
+
export const defaultTreeProps: TreeProps = {
|
|
132
|
+
leftPadding: 8,
|
|
133
|
+
expansionTogglePadding: 22
|
|
134
|
+
};
|
|
135
|
+
|
|
136
|
+
export namespace TreeWidget {
|
|
137
|
+
|
|
138
|
+
/**
|
|
139
|
+
* Bare minimum common interface of the keyboard and the mouse event with respect to the key maskings.
|
|
140
|
+
*/
|
|
141
|
+
export interface ModifierAwareEvent {
|
|
142
|
+
/**
|
|
143
|
+
* Determines if the modifier aware event has the `meta` key masking.
|
|
144
|
+
*/
|
|
145
|
+
readonly metaKey: boolean;
|
|
146
|
+
/**
|
|
147
|
+
* Determines if the modifier aware event has the `ctrl` key masking.
|
|
148
|
+
*/
|
|
149
|
+
readonly ctrlKey: boolean;
|
|
150
|
+
/**
|
|
151
|
+
* Determines if the modifier aware event has the `shift` key masking.
|
|
152
|
+
*/
|
|
153
|
+
readonly shiftKey: boolean;
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
@injectable()
|
|
159
|
+
export class TreeWidget extends ReactWidget implements StatefulWidget {
|
|
160
|
+
|
|
161
|
+
protected searchBox: SearchBox;
|
|
162
|
+
protected searchHighlights: Map<string, TreeDecoration.CaptionHighlight>;
|
|
163
|
+
|
|
164
|
+
@inject(TreeDecoratorService)
|
|
165
|
+
protected readonly decoratorService: TreeDecoratorService;
|
|
166
|
+
@inject(TreeSearch)
|
|
167
|
+
protected readonly treeSearch: TreeSearch;
|
|
168
|
+
@inject(SearchBoxFactory)
|
|
169
|
+
protected readonly searchBoxFactory: SearchBoxFactory;
|
|
170
|
+
@inject(TreeFocusService)
|
|
171
|
+
protected readonly focusService: TreeFocusService;
|
|
172
|
+
|
|
173
|
+
protected decorations: Map<string, TreeDecoration.Data[]> = new Map();
|
|
174
|
+
|
|
175
|
+
@inject(SelectionService)
|
|
176
|
+
protected readonly selectionService: SelectionService;
|
|
177
|
+
|
|
178
|
+
@inject(PreferenceService)
|
|
179
|
+
protected readonly preferenceService: PreferenceService;
|
|
180
|
+
|
|
181
|
+
@inject(LabelProvider)
|
|
182
|
+
protected readonly labelProvider: LabelProvider;
|
|
183
|
+
|
|
184
|
+
@inject(CorePreferences)
|
|
185
|
+
protected readonly corePreferences: CorePreferences;
|
|
186
|
+
|
|
187
|
+
protected shouldScrollToRow = true;
|
|
188
|
+
|
|
189
|
+
protected treeIndent: number = 8;
|
|
190
|
+
|
|
191
|
+
constructor(
|
|
192
|
+
@inject(TreeProps) readonly props: TreeProps,
|
|
193
|
+
@inject(TreeModel) readonly model: TreeModel,
|
|
194
|
+
@inject(ContextMenuRenderer) protected readonly contextMenuRenderer: ContextMenuRenderer,
|
|
195
|
+
) {
|
|
196
|
+
super();
|
|
197
|
+
this.scrollOptions = {
|
|
198
|
+
suppressScrollX: true,
|
|
199
|
+
minScrollbarLength: 35
|
|
200
|
+
};
|
|
201
|
+
this.addClass(TREE_CLASS);
|
|
202
|
+
this.node.tabIndex = 0;
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
@postConstruct()
|
|
206
|
+
protected init(): void {
|
|
207
|
+
this.treeIndent = this.preferenceService.get(PREFERENCE_NAME_TREE_INDENT, this.treeIndent);
|
|
208
|
+
if (this.props.search) {
|
|
209
|
+
this.searchBox = this.searchBoxFactory({ ...SearchBoxProps.DEFAULT, showButtons: true, showFilter: true });
|
|
210
|
+
this.searchBox.node.addEventListener('focus', () => {
|
|
211
|
+
this.node.focus();
|
|
212
|
+
});
|
|
213
|
+
this.toDispose.pushAll([
|
|
214
|
+
this.searchBox,
|
|
215
|
+
this.searchBox.onTextChange(async data => {
|
|
216
|
+
await this.treeSearch.filter(data);
|
|
217
|
+
this.searchHighlights = this.treeSearch.getHighlights();
|
|
218
|
+
this.searchBox.updateHighlightInfo({
|
|
219
|
+
filterText: data,
|
|
220
|
+
total: this.rows.size,
|
|
221
|
+
matched: this.searchHighlights.size
|
|
222
|
+
});
|
|
223
|
+
this.update();
|
|
224
|
+
}),
|
|
225
|
+
this.searchBox.onClose(data => this.treeSearch.filter(undefined)),
|
|
226
|
+
this.searchBox.onNext(() => {
|
|
227
|
+
// Enable next selection if there are currently highlights.
|
|
228
|
+
if (this.searchHighlights.size > 1) {
|
|
229
|
+
this.model.selectNextNode();
|
|
230
|
+
}
|
|
231
|
+
}),
|
|
232
|
+
this.searchBox.onPrevious(() => {
|
|
233
|
+
// Enable previous selection if there are currently highlights.
|
|
234
|
+
if (this.searchHighlights.size > 1) {
|
|
235
|
+
this.model.selectPrevNode();
|
|
236
|
+
}
|
|
237
|
+
}),
|
|
238
|
+
this.searchBox.onFilterToggled(e => {
|
|
239
|
+
this.updateRows();
|
|
240
|
+
}),
|
|
241
|
+
this.treeSearch,
|
|
242
|
+
this.treeSearch.onFilteredNodesChanged(nodes => {
|
|
243
|
+
if (this.searchBox.isFiltering) {
|
|
244
|
+
this.updateRows();
|
|
245
|
+
}
|
|
246
|
+
const node = nodes.find(SelectableTreeNode.is);
|
|
247
|
+
if (node) {
|
|
248
|
+
this.model.selectNode(node);
|
|
249
|
+
}
|
|
250
|
+
}),
|
|
251
|
+
]);
|
|
252
|
+
}
|
|
253
|
+
this.node.addEventListener('mousedown', this.handleMiddleClickEvent.bind(this));
|
|
254
|
+
this.node.addEventListener('mouseup', this.handleMiddleClickEvent.bind(this));
|
|
255
|
+
this.node.addEventListener('auxclick', this.handleMiddleClickEvent.bind(this));
|
|
256
|
+
this.toDispose.pushAll([
|
|
257
|
+
this.model,
|
|
258
|
+
this.model.onChanged(() => this.updateRows()),
|
|
259
|
+
this.model.onSelectionChanged(() => this.scheduleUpdateScrollToRow({ resize: false })),
|
|
260
|
+
this.focusService.onDidChangeFocus(() => this.scheduleUpdateScrollToRow({ resize: false })),
|
|
261
|
+
this.model.onDidChangeBusy(() => this.update()),
|
|
262
|
+
this.model.onDidUpdate(() => this.update()),
|
|
263
|
+
this.model.onNodeRefreshed(() => this.updateDecorations()),
|
|
264
|
+
this.model.onExpansionChanged(() => this.updateDecorations()),
|
|
265
|
+
this.decoratorService,
|
|
266
|
+
this.decoratorService.onDidChangeDecorations(() => this.updateDecorations()),
|
|
267
|
+
this.labelProvider.onDidChange(e => {
|
|
268
|
+
for (const row of this.rows.values()) {
|
|
269
|
+
if (e.affects(row)) {
|
|
270
|
+
this.update();
|
|
271
|
+
return;
|
|
272
|
+
}
|
|
273
|
+
}
|
|
274
|
+
}),
|
|
275
|
+
this.preferenceService.onPreferenceChanged((event: PreferenceChange) => {
|
|
276
|
+
if (event.preferenceName === PREFERENCE_NAME_TREE_INDENT) {
|
|
277
|
+
this.treeIndent = event.newValue;
|
|
278
|
+
this.update();
|
|
279
|
+
}
|
|
280
|
+
})
|
|
281
|
+
]);
|
|
282
|
+
setTimeout(() => {
|
|
283
|
+
this.updateRows();
|
|
284
|
+
this.updateDecorations();
|
|
285
|
+
});
|
|
286
|
+
if (this.props.globalSelection) {
|
|
287
|
+
this.toDispose.pushAll([
|
|
288
|
+
this.model.onSelectionChanged(() => {
|
|
289
|
+
if (this.node.contains(document.activeElement)) {
|
|
290
|
+
this.updateGlobalSelection();
|
|
291
|
+
}
|
|
292
|
+
}),
|
|
293
|
+
this.focusService.onDidChangeFocus(focus => {
|
|
294
|
+
if (focus && this.node.contains(document.activeElement) && this.model.selectedNodes[0] !== focus && this.model.selectedNodes.includes(focus)) {
|
|
295
|
+
this.updateGlobalSelection();
|
|
296
|
+
}
|
|
297
|
+
}),
|
|
298
|
+
Disposable.create(() => {
|
|
299
|
+
const selection = this.selectionService.selection;
|
|
300
|
+
if (TreeWidgetSelection.isSource(selection, this)) {
|
|
301
|
+
this.selectionService.selection = undefined;
|
|
302
|
+
}
|
|
303
|
+
})
|
|
304
|
+
]);
|
|
305
|
+
}
|
|
306
|
+
this.toDispose.push(this.corePreferences.onPreferenceChanged(preference => {
|
|
307
|
+
if (preference.preferenceName === 'workbench.tree.renderIndentGuides') {
|
|
308
|
+
this.update();
|
|
309
|
+
}
|
|
310
|
+
}));
|
|
311
|
+
}
|
|
312
|
+
|
|
313
|
+
/**
|
|
314
|
+
* Update the global selection for the tree.
|
|
315
|
+
*/
|
|
316
|
+
protected updateGlobalSelection(): void {
|
|
317
|
+
this.selectionService.selection = TreeWidgetSelection.create(this);
|
|
318
|
+
}
|
|
319
|
+
|
|
320
|
+
protected rows = new Map<string, TreeWidget.NodeRow>();
|
|
321
|
+
protected updateRows = debounce(() => this.doUpdateRows(), 10);
|
|
322
|
+
protected doUpdateRows(): void {
|
|
323
|
+
const root = this.model.root;
|
|
324
|
+
const rowsToUpdate: Array<[string, TreeWidget.NodeRow]> = [];
|
|
325
|
+
if (root) {
|
|
326
|
+
const depths = new Map<CompositeTreeNode | undefined, number>();
|
|
327
|
+
let index = 0;
|
|
328
|
+
for (const node of new TopDownTreeIterator(root, {
|
|
329
|
+
pruneCollapsed: true,
|
|
330
|
+
pruneSiblings: true
|
|
331
|
+
})) {
|
|
332
|
+
if (this.shouldDisplayNode(node)) {
|
|
333
|
+
const depth = this.getDepthForNode(node, depths);
|
|
334
|
+
if (CompositeTreeNode.is(node)) {
|
|
335
|
+
depths.set(node, depth);
|
|
336
|
+
}
|
|
337
|
+
rowsToUpdate.push([node.id, this.toNodeRow(node, index++, depth)]);
|
|
338
|
+
}
|
|
339
|
+
}
|
|
340
|
+
}
|
|
341
|
+
this.rows = new Map(rowsToUpdate);
|
|
342
|
+
this.updateScrollToRow();
|
|
343
|
+
}
|
|
344
|
+
|
|
345
|
+
protected getDepthForNode(node: TreeNode, depths: Map<CompositeTreeNode | undefined, number>): number {
|
|
346
|
+
const parentDepth = depths.get(node.parent);
|
|
347
|
+
return parentDepth === undefined ? 0 : TreeNode.isVisible(node.parent) ? parentDepth + 1 : parentDepth;
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
protected toNodeRow(node: TreeNode, index: number, depth: number): TreeWidget.NodeRow {
|
|
351
|
+
return { node, index, depth };
|
|
352
|
+
}
|
|
353
|
+
|
|
354
|
+
protected shouldDisplayNode(node: TreeNode): boolean {
|
|
355
|
+
return TreeNode.isVisible(node) && (!this.searchBox?.isFiltering || this.treeSearch.passesFilters(node));
|
|
356
|
+
}
|
|
357
|
+
|
|
358
|
+
/**
|
|
359
|
+
* Row index to ensure visibility.
|
|
360
|
+
* - Used to forcefully scroll if necessary.
|
|
361
|
+
*/
|
|
362
|
+
protected scrollToRow: number | undefined;
|
|
363
|
+
/**
|
|
364
|
+
* Update the `scrollToRow`.
|
|
365
|
+
* @param updateOptions the tree widget force update options.
|
|
366
|
+
*/
|
|
367
|
+
protected updateScrollToRow(): void {
|
|
368
|
+
this.scrollToRow = this.getScrollToRow();
|
|
369
|
+
this.update();
|
|
370
|
+
}
|
|
371
|
+
|
|
372
|
+
protected scheduleUpdateScrollToRow = debounce(this.updateScrollToRow);
|
|
373
|
+
|
|
374
|
+
/**
|
|
375
|
+
* Get the `scrollToRow`.
|
|
376
|
+
*
|
|
377
|
+
* @returns the `scrollToRow` if available.
|
|
378
|
+
*/
|
|
379
|
+
protected getScrollToRow(): number | undefined {
|
|
380
|
+
if (!this.shouldScrollToRow) {
|
|
381
|
+
return undefined;
|
|
382
|
+
}
|
|
383
|
+
const { focusedNode } = this.focusService;
|
|
384
|
+
return focusedNode && this.rows.get(focusedNode.id)?.index;
|
|
385
|
+
}
|
|
386
|
+
|
|
387
|
+
/**
|
|
388
|
+
* Update tree decorations.
|
|
389
|
+
* - Updating decorations are debounced in order to limit the number of expensive updates.
|
|
390
|
+
*/
|
|
391
|
+
protected readonly updateDecorations = debounce(() => this.doUpdateDecorations(), 150);
|
|
392
|
+
protected async doUpdateDecorations(): Promise<void> {
|
|
393
|
+
this.decorations = await this.decoratorService.getDecorations(this.model);
|
|
394
|
+
this.update();
|
|
395
|
+
}
|
|
396
|
+
|
|
397
|
+
protected override onActivateRequest(msg: Message): void {
|
|
398
|
+
super.onActivateRequest(msg);
|
|
399
|
+
this.node.focus({ preventScroll: true });
|
|
400
|
+
}
|
|
401
|
+
|
|
402
|
+
/**
|
|
403
|
+
* Actually focus the tree node.
|
|
404
|
+
*/
|
|
405
|
+
protected doFocus(): void {
|
|
406
|
+
if (!this.model.selectedNodes.length) {
|
|
407
|
+
const node = this.getNodeToFocus();
|
|
408
|
+
if (SelectableTreeNode.is(node)) {
|
|
409
|
+
this.model.selectNode(node);
|
|
410
|
+
}
|
|
411
|
+
}
|
|
412
|
+
}
|
|
413
|
+
|
|
414
|
+
/**
|
|
415
|
+
* Get the tree node to focus.
|
|
416
|
+
*
|
|
417
|
+
* @returns the node to focus if available.
|
|
418
|
+
*/
|
|
419
|
+
protected getNodeToFocus(): SelectableTreeNode | undefined {
|
|
420
|
+
const { focusedNode } = this.focusService;
|
|
421
|
+
if (focusedNode) {
|
|
422
|
+
return focusedNode;
|
|
423
|
+
}
|
|
424
|
+
const { root } = this.model;
|
|
425
|
+
if (SelectableTreeNode.isVisible(root)) {
|
|
426
|
+
return root;
|
|
427
|
+
}
|
|
428
|
+
return this.model.getNextSelectableNode(root);
|
|
429
|
+
}
|
|
430
|
+
|
|
431
|
+
protected override onUpdateRequest(msg: Message): void {
|
|
432
|
+
if (!this.isAttached || !this.isVisible) {
|
|
433
|
+
return;
|
|
434
|
+
}
|
|
435
|
+
super.onUpdateRequest(msg);
|
|
436
|
+
}
|
|
437
|
+
|
|
438
|
+
protected override onResize(msg: Widget.ResizeMessage): void {
|
|
439
|
+
super.onResize(msg);
|
|
440
|
+
this.update();
|
|
441
|
+
}
|
|
442
|
+
|
|
443
|
+
protected render(): React.ReactNode {
|
|
444
|
+
return React.createElement('div', this.createContainerAttributes(), this.renderTree(this.model));
|
|
445
|
+
}
|
|
446
|
+
|
|
447
|
+
/**
|
|
448
|
+
* Create the container attributes for the widget.
|
|
449
|
+
*/
|
|
450
|
+
protected createContainerAttributes(): React.HTMLAttributes<HTMLElement> {
|
|
451
|
+
const classNames = [TREE_CONTAINER_CLASS];
|
|
452
|
+
if (!this.rows.size) {
|
|
453
|
+
classNames.push('empty');
|
|
454
|
+
}
|
|
455
|
+
if (this.model.selectedNodes.length === 0 && !this.focusService.focusedNode) {
|
|
456
|
+
classNames.push('focused');
|
|
457
|
+
}
|
|
458
|
+
return {
|
|
459
|
+
className: classNames.join(' '),
|
|
460
|
+
onContextMenu: event => this.handleContextMenuEvent(this.getContainerTreeNode(), event)
|
|
461
|
+
};
|
|
462
|
+
}
|
|
463
|
+
/**
|
|
464
|
+
* Get the container tree node.
|
|
465
|
+
*
|
|
466
|
+
* @returns the tree node for the container if available.
|
|
467
|
+
*/
|
|
468
|
+
protected getContainerTreeNode(): TreeNode | undefined {
|
|
469
|
+
return this.model.root;
|
|
470
|
+
}
|
|
471
|
+
|
|
472
|
+
protected ScrollingRowRenderer: React.FC<{ rows: TreeWidget.NodeRow[] }> = ({ rows }) => {
|
|
473
|
+
useEffect(() => this.scrollToSelected());
|
|
474
|
+
return <>{rows.map(row => <div key={row.index}>{this.renderNodeRow(row)}</div>)}</>;
|
|
475
|
+
};
|
|
476
|
+
|
|
477
|
+
protected view: TreeWidget.View | undefined;
|
|
478
|
+
/**
|
|
479
|
+
* Render the tree widget.
|
|
480
|
+
* @param model the tree model.
|
|
481
|
+
*/
|
|
482
|
+
protected renderTree(model: TreeModel): React.ReactNode {
|
|
483
|
+
if (model.root) {
|
|
484
|
+
const rows = Array.from(this.rows.values());
|
|
485
|
+
if (this.props.virtualized === false) {
|
|
486
|
+
return <this.ScrollingRowRenderer rows={rows} />;
|
|
487
|
+
}
|
|
488
|
+
return <TreeWidget.View
|
|
489
|
+
ref={view => this.view = (view || undefined)}
|
|
490
|
+
width={this.node.offsetWidth}
|
|
491
|
+
height={this.node.offsetHeight}
|
|
492
|
+
rows={rows}
|
|
493
|
+
renderNodeRow={this.renderNodeRow}
|
|
494
|
+
scrollToRow={this.scrollToRow}
|
|
495
|
+
/>;
|
|
496
|
+
}
|
|
497
|
+
// eslint-disable-next-line no-null/no-null
|
|
498
|
+
return null;
|
|
499
|
+
}
|
|
500
|
+
|
|
501
|
+
scrollArea: Element = this.node;
|
|
502
|
+
/**
|
|
503
|
+
* Scroll to the selected tree node.
|
|
504
|
+
*/
|
|
505
|
+
protected scrollToSelected(): void {
|
|
506
|
+
if (this.props.scrollIfActive === true && !this.node.contains(document.activeElement)) {
|
|
507
|
+
return;
|
|
508
|
+
}
|
|
509
|
+
const focus = this.node.getElementsByClassName(FOCUS_CLASS)[0];
|
|
510
|
+
if (focus) {
|
|
511
|
+
ElementExt.scrollIntoViewIfNeeded(this.scrollArea, focus);
|
|
512
|
+
} else {
|
|
513
|
+
const selected = this.node.getElementsByClassName(SELECTED_CLASS)[0];
|
|
514
|
+
if (selected) {
|
|
515
|
+
ElementExt.scrollIntoViewIfNeeded(this.scrollArea, selected);
|
|
516
|
+
}
|
|
517
|
+
}
|
|
518
|
+
}
|
|
519
|
+
|
|
520
|
+
/**
|
|
521
|
+
* Render the node row.
|
|
522
|
+
*/
|
|
523
|
+
protected readonly renderNodeRow = (row: TreeWidget.NodeRow) => this.doRenderNodeRow(row);
|
|
524
|
+
/**
|
|
525
|
+
* Actually render the node row.
|
|
526
|
+
*/
|
|
527
|
+
protected doRenderNodeRow({ node, depth }: TreeWidget.NodeRow): React.ReactNode {
|
|
528
|
+
return <React.Fragment>
|
|
529
|
+
{this.renderIndent(node, { depth })}
|
|
530
|
+
{this.renderNode(node, { depth })}
|
|
531
|
+
</React.Fragment>;
|
|
532
|
+
}
|
|
533
|
+
|
|
534
|
+
/**
|
|
535
|
+
* Render the tree node given the node properties.
|
|
536
|
+
* @param node the tree node.
|
|
537
|
+
* @param props the node properties.
|
|
538
|
+
*/
|
|
539
|
+
protected renderIcon(node: TreeNode, props: NodeProps): React.ReactNode {
|
|
540
|
+
// eslint-disable-next-line no-null/no-null
|
|
541
|
+
return null;
|
|
542
|
+
}
|
|
543
|
+
|
|
544
|
+
/**
|
|
545
|
+
* Toggle the node.
|
|
546
|
+
*/
|
|
547
|
+
protected readonly toggle = (event: React.MouseEvent<HTMLElement>) => this.doToggle(event);
|
|
548
|
+
/**
|
|
549
|
+
* Actually toggle the tree node.
|
|
550
|
+
* @param event the mouse click event.
|
|
551
|
+
*/
|
|
552
|
+
protected doToggle(event: React.MouseEvent<HTMLElement>): void {
|
|
553
|
+
const nodeId = event.currentTarget.getAttribute('data-node-id');
|
|
554
|
+
if (nodeId) {
|
|
555
|
+
const node = this.model.getNode(nodeId);
|
|
556
|
+
if (node && this.props.expandOnlyOnExpansionToggleClick) {
|
|
557
|
+
if (this.isExpandable(node) && !this.hasShiftMask(event) && !this.hasCtrlCmdMask(event)) {
|
|
558
|
+
this.model.toggleNodeExpansion(node);
|
|
559
|
+
}
|
|
560
|
+
} else {
|
|
561
|
+
this.handleClickEvent(node, event);
|
|
562
|
+
}
|
|
563
|
+
}
|
|
564
|
+
event.stopPropagation();
|
|
565
|
+
}
|
|
566
|
+
|
|
567
|
+
/**
|
|
568
|
+
* Render the node expansion toggle.
|
|
569
|
+
* @param node the tree node.
|
|
570
|
+
* @param props the node properties.
|
|
571
|
+
*/
|
|
572
|
+
protected renderExpansionToggle(node: TreeNode, props: NodeProps): React.ReactNode {
|
|
573
|
+
if (!this.isExpandable(node)) {
|
|
574
|
+
// eslint-disable-next-line no-null/no-null
|
|
575
|
+
return null;
|
|
576
|
+
}
|
|
577
|
+
const classes = [TREE_NODE_SEGMENT_CLASS, EXPANSION_TOGGLE_CLASS];
|
|
578
|
+
if (!node.expanded) {
|
|
579
|
+
classes.push(COLLAPSED_CLASS);
|
|
580
|
+
}
|
|
581
|
+
if (node.busy) {
|
|
582
|
+
classes.push(BUSY_CLASS, ...CODICON_LOADING_CLASSES);
|
|
583
|
+
} else {
|
|
584
|
+
classes.push(...CODICON_TREE_ITEM_CLASSES);
|
|
585
|
+
}
|
|
586
|
+
const className = classes.join(' ');
|
|
587
|
+
return <div
|
|
588
|
+
data-node-id={node.id}
|
|
589
|
+
className={className}
|
|
590
|
+
onClick={this.toggle}
|
|
591
|
+
onDoubleClick={this.handleExpansionToggleDblClickEvent}>
|
|
592
|
+
</div>;
|
|
593
|
+
}
|
|
594
|
+
|
|
595
|
+
/**
|
|
596
|
+
* Render the node expansion toggle.
|
|
597
|
+
* @param node the tree node.
|
|
598
|
+
* @param props the node properties.
|
|
599
|
+
*/
|
|
600
|
+
protected renderCheckbox(node: TreeNode, props: NodeProps): React.ReactNode {
|
|
601
|
+
if (node.checkboxInfo === undefined) {
|
|
602
|
+
// eslint-disable-next-line no-null/no-null
|
|
603
|
+
return null;
|
|
604
|
+
}
|
|
605
|
+
return <input data-node-id={node.id}
|
|
606
|
+
readOnly
|
|
607
|
+
type='checkbox'
|
|
608
|
+
checked={!!node.checkboxInfo.checked}
|
|
609
|
+
title={node.checkboxInfo.tooltip}
|
|
610
|
+
aria-label={node.checkboxInfo.accessibilityInformation?.label}
|
|
611
|
+
role={node.checkboxInfo.accessibilityInformation?.role}
|
|
612
|
+
className='theia-input'
|
|
613
|
+
onClick={event => this.toggleChecked(event)} />;
|
|
614
|
+
}
|
|
615
|
+
|
|
616
|
+
protected toggleChecked(event: React.MouseEvent<HTMLElement>): void {
|
|
617
|
+
const nodeId = event.currentTarget.getAttribute('data-node-id');
|
|
618
|
+
if (nodeId) {
|
|
619
|
+
const node = this.model.getNode(nodeId);
|
|
620
|
+
if (node) {
|
|
621
|
+
this.model.markAsChecked(node, !node.checkboxInfo!.checked);
|
|
622
|
+
} else {
|
|
623
|
+
this.handleClickEvent(node, event);
|
|
624
|
+
}
|
|
625
|
+
}
|
|
626
|
+
event.preventDefault();
|
|
627
|
+
event.stopPropagation();
|
|
628
|
+
}
|
|
629
|
+
/**
|
|
630
|
+
* Render the tree node caption given the node properties.
|
|
631
|
+
* @param node the tree node.
|
|
632
|
+
* @param props the node properties.
|
|
633
|
+
*/
|
|
634
|
+
protected renderCaption(node: TreeNode, props: NodeProps): React.ReactNode {
|
|
635
|
+
const attrs = this.getCaptionAttributes(node, props);
|
|
636
|
+
const children = this.getCaptionChildren(node, props);
|
|
637
|
+
return React.createElement('div', attrs, children);
|
|
638
|
+
}
|
|
639
|
+
|
|
640
|
+
protected getCaptionAttributes(node: TreeNode, props: NodeProps): React.Attributes & React.HTMLAttributes<HTMLElement> {
|
|
641
|
+
const tooltip = this.getDecorationData(node, 'tooltip').filter(notEmpty).join(' • ');
|
|
642
|
+
const classes = [TREE_NODE_SEGMENT_CLASS];
|
|
643
|
+
if (!this.hasTrailingSuffixes(node)) {
|
|
644
|
+
classes.push(TREE_NODE_SEGMENT_GROW_CLASS);
|
|
645
|
+
}
|
|
646
|
+
const className = classes.join(' ');
|
|
647
|
+
let attrs = this.decorateCaption(node, {
|
|
648
|
+
className, id: node.id
|
|
649
|
+
});
|
|
650
|
+
if (tooltip.length > 0) {
|
|
651
|
+
attrs = {
|
|
652
|
+
...attrs,
|
|
653
|
+
title: tooltip
|
|
654
|
+
};
|
|
655
|
+
}
|
|
656
|
+
return attrs;
|
|
657
|
+
}
|
|
658
|
+
|
|
659
|
+
protected getCaptionChildren(node: TreeNode, props: NodeProps): React.ReactNode {
|
|
660
|
+
const children = [];
|
|
661
|
+
const caption = this.toNodeName(node);
|
|
662
|
+
const highlight = this.getDecorationData(node, 'highlight')[0];
|
|
663
|
+
if (highlight) {
|
|
664
|
+
children.push(this.toReactNode(caption, highlight));
|
|
665
|
+
}
|
|
666
|
+
const searchHighlight = this.searchHighlights ? this.searchHighlights.get(node.id) : undefined;
|
|
667
|
+
if (searchHighlight) {
|
|
668
|
+
children.push(...this.toReactNode(caption, searchHighlight));
|
|
669
|
+
} else if (!highlight) {
|
|
670
|
+
children.push(caption);
|
|
671
|
+
}
|
|
672
|
+
return children;
|
|
673
|
+
}
|
|
674
|
+
|
|
675
|
+
/**
|
|
676
|
+
* Update the node given the caption and highlight.
|
|
677
|
+
* @param caption the caption.
|
|
678
|
+
* @param highlight the tree decoration caption highlight.
|
|
679
|
+
*/
|
|
680
|
+
protected toReactNode(caption: string, highlight: TreeDecoration.CaptionHighlight): React.ReactNode[] {
|
|
681
|
+
let style: React.CSSProperties = {};
|
|
682
|
+
if (highlight.color) {
|
|
683
|
+
style = {
|
|
684
|
+
...style,
|
|
685
|
+
color: highlight.color
|
|
686
|
+
};
|
|
687
|
+
}
|
|
688
|
+
if (highlight.backgroundColor) {
|
|
689
|
+
style = {
|
|
690
|
+
...style,
|
|
691
|
+
backgroundColor: highlight.backgroundColor
|
|
692
|
+
};
|
|
693
|
+
}
|
|
694
|
+
const createChildren = (fragment: TreeDecoration.CaptionHighlight.Fragment, index: number) => {
|
|
695
|
+
const { data } = fragment;
|
|
696
|
+
if (fragment.highlight) {
|
|
697
|
+
return <mark className={TreeDecoration.Styles.CAPTION_HIGHLIGHT_CLASS} style={style} key={index}>{data}</mark>;
|
|
698
|
+
} else {
|
|
699
|
+
return data;
|
|
700
|
+
}
|
|
701
|
+
};
|
|
702
|
+
return TreeDecoration.CaptionHighlight.split(caption, highlight).map(createChildren);
|
|
703
|
+
}
|
|
704
|
+
|
|
705
|
+
/**
|
|
706
|
+
* Decorate the tree caption.
|
|
707
|
+
* @param node the tree node.
|
|
708
|
+
* @param attrs the additional attributes.
|
|
709
|
+
*/
|
|
710
|
+
protected decorateCaption(node: TreeNode, attrs: React.HTMLAttributes<HTMLElement>): React.Attributes & React.HTMLAttributes<HTMLElement> {
|
|
711
|
+
const style = this.getDecorationData(node, 'fontData')
|
|
712
|
+
.filter(notEmpty)
|
|
713
|
+
.reverse()
|
|
714
|
+
.map(fontData => this.applyFontStyles({}, fontData))
|
|
715
|
+
.reduce((acc, current) => ({
|
|
716
|
+
...acc,
|
|
717
|
+
...current
|
|
718
|
+
}), {});
|
|
719
|
+
return {
|
|
720
|
+
...attrs,
|
|
721
|
+
style
|
|
722
|
+
};
|
|
723
|
+
}
|
|
724
|
+
|
|
725
|
+
/**
|
|
726
|
+
* Determine if the tree node contains trailing suffixes.
|
|
727
|
+
* @param node the tree node.
|
|
728
|
+
*
|
|
729
|
+
* @returns `true` if the tree node contains trailing suffices.
|
|
730
|
+
*/
|
|
731
|
+
protected hasTrailingSuffixes(node: TreeNode): boolean {
|
|
732
|
+
return this.getDecorationData(node, 'captionSuffixes').filter(notEmpty).reduce((acc, current) => acc.concat(current), []).length > 0;
|
|
733
|
+
}
|
|
734
|
+
|
|
735
|
+
/**
|
|
736
|
+
* Apply font styles to the tree.
|
|
737
|
+
* @param original the original css properties.
|
|
738
|
+
* @param fontData the optional `fontData`.
|
|
739
|
+
*/
|
|
740
|
+
protected applyFontStyles(original: React.CSSProperties, fontData: TreeDecoration.FontData | undefined): React.CSSProperties {
|
|
741
|
+
if (fontData === undefined) {
|
|
742
|
+
return original;
|
|
743
|
+
}
|
|
744
|
+
const modified = { ...original }; // make a copy to mutate
|
|
745
|
+
const { color, style } = fontData;
|
|
746
|
+
if (color) {
|
|
747
|
+
modified.color = color;
|
|
748
|
+
}
|
|
749
|
+
if (style) {
|
|
750
|
+
(Array.isArray(style) ? style : [style]).forEach(s => {
|
|
751
|
+
switch (s) {
|
|
752
|
+
case 'bold':
|
|
753
|
+
modified.fontWeight = s;
|
|
754
|
+
break;
|
|
755
|
+
case 'normal':
|
|
756
|
+
case 'oblique':
|
|
757
|
+
case 'italic':
|
|
758
|
+
modified.fontStyle = s;
|
|
759
|
+
break;
|
|
760
|
+
case 'underline':
|
|
761
|
+
case 'line-through':
|
|
762
|
+
modified.textDecoration = s;
|
|
763
|
+
break;
|
|
764
|
+
default:
|
|
765
|
+
throw new Error(`Unexpected font style: "${s}".`);
|
|
766
|
+
}
|
|
767
|
+
});
|
|
768
|
+
}
|
|
769
|
+
return modified;
|
|
770
|
+
}
|
|
771
|
+
|
|
772
|
+
/**
|
|
773
|
+
* Render caption affixes for the given tree node.
|
|
774
|
+
* @param node the tree node.
|
|
775
|
+
* @param props the node properties.
|
|
776
|
+
* @param affixKey the affix key.
|
|
777
|
+
*/
|
|
778
|
+
protected renderCaptionAffixes(node: TreeNode, props: NodeProps, affixKey: 'captionPrefixes' | 'captionSuffixes'): React.ReactNode {
|
|
779
|
+
const suffix = affixKey === 'captionSuffixes';
|
|
780
|
+
const affixClass = suffix ? TreeDecoration.Styles.CAPTION_SUFFIX_CLASS : TreeDecoration.Styles.CAPTION_PREFIX_CLASS;
|
|
781
|
+
const classes = [TREE_NODE_SEGMENT_CLASS, affixClass];
|
|
782
|
+
const affixes = this.getDecorationData(node, affixKey).filter(notEmpty).reduce((acc, current) => acc.concat(current), []);
|
|
783
|
+
const children: React.ReactNode[] = [];
|
|
784
|
+
for (let i = 0; i < affixes.length; i++) {
|
|
785
|
+
const affix = affixes[i];
|
|
786
|
+
if (suffix && i === affixes.length - 1) {
|
|
787
|
+
classes.push(TREE_NODE_SEGMENT_GROW_CLASS);
|
|
788
|
+
}
|
|
789
|
+
const style = this.applyFontStyles({}, affix.fontData);
|
|
790
|
+
const className = classes.join(' ');
|
|
791
|
+
const key = node.id + '_' + i;
|
|
792
|
+
const attrs = {
|
|
793
|
+
className,
|
|
794
|
+
style,
|
|
795
|
+
key
|
|
796
|
+
};
|
|
797
|
+
children.push(React.createElement('div', attrs, affix.data));
|
|
798
|
+
}
|
|
799
|
+
return <React.Fragment>{children}</React.Fragment>;
|
|
800
|
+
}
|
|
801
|
+
|
|
802
|
+
/**
|
|
803
|
+
* Decorate the tree node icon.
|
|
804
|
+
* @param node the tree node.
|
|
805
|
+
* @param icon the icon.
|
|
806
|
+
*/
|
|
807
|
+
protected decorateIcon(node: TreeNode, icon: React.ReactNode): React.ReactNode {
|
|
808
|
+
if (!icon) {
|
|
809
|
+
return;
|
|
810
|
+
}
|
|
811
|
+
const overlayIcons: React.ReactNode[] = [];
|
|
812
|
+
// if multiple overlays have the same overlay.position attribute, we'll de-duplicate those and only process the first one from the decoration array
|
|
813
|
+
const seenPositions = new Set<TreeDecoration.IconOverlayPosition>();
|
|
814
|
+
const overlays = this.getDecorationData(node, 'iconOverlay').filter(notEmpty);
|
|
815
|
+
|
|
816
|
+
for (const overlay of overlays) {
|
|
817
|
+
if (!seenPositions.has(overlay.position)) {
|
|
818
|
+
seenPositions.add(overlay.position);
|
|
819
|
+
const iconClasses = [TreeDecoration.Styles.DECORATOR_SIZE_CLASS, TreeDecoration.IconOverlayPosition.getStyle(overlay.position)];
|
|
820
|
+
const style = (color?: string) => color === undefined ? {} : { color };
|
|
821
|
+
|
|
822
|
+
if (overlay.background) {
|
|
823
|
+
overlayIcons.push(<span key={node.id + 'bg'} className={this.getIconClass(overlay.background.shape, iconClasses)}
|
|
824
|
+
style={style(overlay.background.color)}></span>);
|
|
825
|
+
}
|
|
826
|
+
|
|
827
|
+
const overlayIcon = 'icon' in overlay ? overlay.icon : overlay.iconClass;
|
|
828
|
+
overlayIcons.push(<span key={node.id} className={this.getIconClass(overlayIcon, iconClasses)} style={style(overlay.color)}></span>);
|
|
829
|
+
}
|
|
830
|
+
}
|
|
831
|
+
|
|
832
|
+
if (overlayIcons.length > 0) {
|
|
833
|
+
return <div className={TreeDecoration.Styles.ICON_WRAPPER_CLASS}>{icon}{overlayIcons}</div>;
|
|
834
|
+
}
|
|
835
|
+
|
|
836
|
+
return icon;
|
|
837
|
+
}
|
|
838
|
+
|
|
839
|
+
/**
|
|
840
|
+
* Render the tree node tail decorations.
|
|
841
|
+
* @param node the tree node.
|
|
842
|
+
* @param props the node properties.
|
|
843
|
+
*/
|
|
844
|
+
protected renderTailDecorations(node: TreeNode, props: NodeProps): React.ReactNode {
|
|
845
|
+
const tailDecorations = this.getDecorationData(node, 'tailDecorations').reduce((acc, current) => acc.concat(current), []);
|
|
846
|
+
if (tailDecorations.length === 0) {
|
|
847
|
+
return;
|
|
848
|
+
}
|
|
849
|
+
return this.renderTailDecorationsForNode(node, props, tailDecorations);
|
|
850
|
+
}
|
|
851
|
+
|
|
852
|
+
protected renderTailDecorationsForNode(node: TreeNode, props: NodeProps, tailDecorations: TreeDecoration.TailDecoration.AnyPartial[]): React.ReactNode {
|
|
853
|
+
let dotDecoration: TreeDecoration.TailDecoration.AnyPartial | undefined;
|
|
854
|
+
const otherDecorations: TreeDecoration.TailDecoration.AnyPartial[] = [];
|
|
855
|
+
tailDecorations.reverse().forEach(decoration => {
|
|
856
|
+
if (TreeDecoration.TailDecoration.isDotDecoration(decoration)) {
|
|
857
|
+
dotDecoration ||= decoration;
|
|
858
|
+
} else if (decoration.data || decoration.icon || decoration.iconClass) {
|
|
859
|
+
otherDecorations.push(decoration);
|
|
860
|
+
}
|
|
861
|
+
});
|
|
862
|
+
const decorationsToRender = dotDecoration ? [dotDecoration, ...otherDecorations] : otherDecorations;
|
|
863
|
+
return <React.Fragment>
|
|
864
|
+
{decorationsToRender.map((decoration, index) => {
|
|
865
|
+
const { tooltip, data, fontData, color, icon, iconClass } = decoration;
|
|
866
|
+
const iconToRender = icon ?? iconClass;
|
|
867
|
+
const className = [TREE_NODE_SEGMENT_CLASS, TREE_NODE_TAIL_CLASS, 'flex'].join(' ');
|
|
868
|
+
const style = fontData ? this.applyFontStyles({}, fontData) : color ? { color } : undefined;
|
|
869
|
+
const content = data ? data : iconToRender
|
|
870
|
+
? <span
|
|
871
|
+
key={node.id + 'icon' + index}
|
|
872
|
+
className={this.getIconClass(iconToRender, iconToRender === 'circle' ? [TreeDecoration.Styles.DECORATOR_SIZE_CLASS] : [])}
|
|
873
|
+
></span>
|
|
874
|
+
: '';
|
|
875
|
+
return <div key={node.id + className + index} className={className} style={style} title={tooltip}>
|
|
876
|
+
{content}{index !== decorationsToRender.length - 1 ? ',' : ''}
|
|
877
|
+
</div>;
|
|
878
|
+
})}
|
|
879
|
+
</React.Fragment>;
|
|
880
|
+
}
|
|
881
|
+
|
|
882
|
+
/**
|
|
883
|
+
* Determine the classes to use for an icon
|
|
884
|
+
* - Assumes a Font Awesome name when passed a single string, otherwise uses the passed string array
|
|
885
|
+
* @param iconName the icon name or list of icon names.
|
|
886
|
+
* @param additionalClasses additional CSS classes.
|
|
887
|
+
*
|
|
888
|
+
* @returns the icon class name.
|
|
889
|
+
*/
|
|
890
|
+
protected getIconClass(iconName: string | string[], additionalClasses: string[] = []): string {
|
|
891
|
+
const iconClass = (typeof iconName === 'string') ? ['a', 'fa', `fa-${iconName}`] : ['a'].concat(iconName);
|
|
892
|
+
return iconClass.concat(additionalClasses).join(' ');
|
|
893
|
+
}
|
|
894
|
+
|
|
895
|
+
/**
|
|
896
|
+
* Render indent for the file tree based on the depth
|
|
897
|
+
* @param node the tree node.
|
|
898
|
+
* @param depth the depth of the tree node.
|
|
899
|
+
*/
|
|
900
|
+
protected renderIndent(node: TreeNode, props: NodeProps): React.ReactNode {
|
|
901
|
+
const renderIndentGuides = this.corePreferences['workbench.tree.renderIndentGuides'];
|
|
902
|
+
if (renderIndentGuides === 'none') {
|
|
903
|
+
return undefined;
|
|
904
|
+
}
|
|
905
|
+
|
|
906
|
+
const indentDivs: React.ReactNode[] = [];
|
|
907
|
+
let current: TreeNode | undefined = node;
|
|
908
|
+
let depth = props.depth;
|
|
909
|
+
while (current && depth) {
|
|
910
|
+
if (this.shouldRenderIndent(current)) {
|
|
911
|
+
const classNames: string[] = [TREE_NODE_INDENT_GUIDE_CLASS];
|
|
912
|
+
if (this.needsActiveIndentGuideline(current)) {
|
|
913
|
+
classNames.push('active');
|
|
914
|
+
} else {
|
|
915
|
+
classNames.push(renderIndentGuides === 'onHover' ? 'hover' : 'always');
|
|
916
|
+
}
|
|
917
|
+
const paddingLeft = this.getDepthPadding(depth);
|
|
918
|
+
indentDivs.unshift(<div key={depth} className={classNames.join(' ')} style={{
|
|
919
|
+
paddingLeft: `${paddingLeft}px`
|
|
920
|
+
}} />);
|
|
921
|
+
depth--;
|
|
922
|
+
}
|
|
923
|
+
current = current.parent;
|
|
924
|
+
}
|
|
925
|
+
return indentDivs;
|
|
926
|
+
}
|
|
927
|
+
|
|
928
|
+
/**
|
|
929
|
+
* Determines whether an indentation div should be rendered for the specified tree node.
|
|
930
|
+
* If there are multiple tree nodes inside of a single rendered row,
|
|
931
|
+
* this method should only return true for the first node.
|
|
932
|
+
*/
|
|
933
|
+
protected shouldRenderIndent(node: TreeNode): boolean {
|
|
934
|
+
return true;
|
|
935
|
+
}
|
|
936
|
+
|
|
937
|
+
protected needsActiveIndentGuideline(node: TreeNode): boolean {
|
|
938
|
+
const parent = node.parent;
|
|
939
|
+
if (!parent || !this.isExpandable(parent)) {
|
|
940
|
+
return false;
|
|
941
|
+
}
|
|
942
|
+
if (SelectableTreeNode.isSelected(parent)) {
|
|
943
|
+
return true;
|
|
944
|
+
}
|
|
945
|
+
if (parent.expanded) {
|
|
946
|
+
for (const sibling of parent.children) {
|
|
947
|
+
if (SelectableTreeNode.isSelected(sibling) && !(this.isExpandable(sibling) && sibling.expanded)) {
|
|
948
|
+
return true;
|
|
949
|
+
}
|
|
950
|
+
}
|
|
951
|
+
}
|
|
952
|
+
return false;
|
|
953
|
+
}
|
|
954
|
+
|
|
955
|
+
/**
|
|
956
|
+
* Render the node given the tree node and node properties.
|
|
957
|
+
* @param node the tree node.
|
|
958
|
+
* @param props the node properties.
|
|
959
|
+
*/
|
|
960
|
+
protected renderNode(node: TreeNode, props: NodeProps): React.ReactNode {
|
|
961
|
+
if (!TreeNode.isVisible(node)) {
|
|
962
|
+
return undefined;
|
|
963
|
+
}
|
|
964
|
+
const attributes = this.createNodeAttributes(node, props);
|
|
965
|
+
const content = <div className={TREE_NODE_CONTENT_CLASS}>
|
|
966
|
+
{this.renderExpansionToggle(node, props)}
|
|
967
|
+
{this.renderCheckbox(node, props)}
|
|
968
|
+
{this.decorateIcon(node, this.renderIcon(node, props))}
|
|
969
|
+
{this.renderCaptionAffixes(node, props, 'captionPrefixes')}
|
|
970
|
+
{this.renderCaption(node, props)}
|
|
971
|
+
{this.renderCaptionAffixes(node, props, 'captionSuffixes')}
|
|
972
|
+
{this.renderTailDecorations(node, props)}
|
|
973
|
+
</div>;
|
|
974
|
+
return React.createElement('div', attributes, content);
|
|
975
|
+
}
|
|
976
|
+
|
|
977
|
+
/**
|
|
978
|
+
* Create node attributes for the tree node given the node properties.
|
|
979
|
+
* @param node the tree node.
|
|
980
|
+
* @param props the node properties.
|
|
981
|
+
*/
|
|
982
|
+
protected createNodeAttributes(node: TreeNode, props: NodeProps): React.Attributes & React.HTMLAttributes<HTMLElement> {
|
|
983
|
+
const className = this.createNodeClassNames(node, props).join(' ');
|
|
984
|
+
const style = this.createNodeStyle(node, props);
|
|
985
|
+
return {
|
|
986
|
+
className,
|
|
987
|
+
style,
|
|
988
|
+
onClick: event => this.handleClickEvent(node, event),
|
|
989
|
+
onDoubleClick: event => this.handleDblClickEvent(node, event),
|
|
990
|
+
onAuxClick: event => this.handleAuxClickEvent(node, event),
|
|
991
|
+
onContextMenu: event => this.handleContextMenuEvent(node, event),
|
|
992
|
+
};
|
|
993
|
+
}
|
|
994
|
+
|
|
995
|
+
/**
|
|
996
|
+
* Create the node class names.
|
|
997
|
+
* @param node the tree node.
|
|
998
|
+
* @param props the node properties.
|
|
999
|
+
*
|
|
1000
|
+
* @returns the list of tree node class names.
|
|
1001
|
+
*/
|
|
1002
|
+
protected createNodeClassNames(node: TreeNode, props: NodeProps): string[] {
|
|
1003
|
+
const classNames = [TREE_NODE_CLASS];
|
|
1004
|
+
if (CompositeTreeNode.is(node)) {
|
|
1005
|
+
classNames.push(COMPOSITE_TREE_NODE_CLASS);
|
|
1006
|
+
}
|
|
1007
|
+
if (this.isExpandable(node)) {
|
|
1008
|
+
classNames.push(EXPANDABLE_TREE_NODE_CLASS);
|
|
1009
|
+
}
|
|
1010
|
+
if (this.rowIsSelected(node, props)) {
|
|
1011
|
+
classNames.push(SELECTED_CLASS);
|
|
1012
|
+
}
|
|
1013
|
+
if (this.focusService.hasFocus(node)) {
|
|
1014
|
+
classNames.push(FOCUS_CLASS);
|
|
1015
|
+
}
|
|
1016
|
+
return classNames;
|
|
1017
|
+
}
|
|
1018
|
+
|
|
1019
|
+
protected rowIsSelected(node: TreeNode, props: NodeProps): boolean {
|
|
1020
|
+
return SelectableTreeNode.isSelected(node);
|
|
1021
|
+
}
|
|
1022
|
+
|
|
1023
|
+
/**
|
|
1024
|
+
* Get the default node style.
|
|
1025
|
+
* @param node the tree node.
|
|
1026
|
+
* @param props the node properties.
|
|
1027
|
+
*
|
|
1028
|
+
* @returns the CSS properties if available.
|
|
1029
|
+
*/
|
|
1030
|
+
protected getDefaultNodeStyle(node: TreeNode, props: NodeProps): React.CSSProperties | undefined {
|
|
1031
|
+
const paddingLeft = this.getPaddingLeft(node, props) + 'px';
|
|
1032
|
+
return { paddingLeft };
|
|
1033
|
+
}
|
|
1034
|
+
|
|
1035
|
+
protected getPaddingLeft(node: TreeNode, props: NodeProps): number {
|
|
1036
|
+
return this.getDepthPadding(props.depth) + (this.needsExpansionTogglePadding(node) ? this.props.expansionTogglePadding : 0);
|
|
1037
|
+
}
|
|
1038
|
+
|
|
1039
|
+
/**
|
|
1040
|
+
* If the node is a composite, a toggle will be rendered.
|
|
1041
|
+
* Otherwise we need to add the width and the left, right padding => 18px
|
|
1042
|
+
*/
|
|
1043
|
+
protected needsExpansionTogglePadding(node: TreeNode): boolean {
|
|
1044
|
+
return !this.isExpandable(node);
|
|
1045
|
+
}
|
|
1046
|
+
|
|
1047
|
+
/**
|
|
1048
|
+
* Create the tree node style.
|
|
1049
|
+
* @param node the tree node.
|
|
1050
|
+
* @param props the node properties.
|
|
1051
|
+
*/
|
|
1052
|
+
protected createNodeStyle(node: TreeNode, props: NodeProps): React.CSSProperties | undefined {
|
|
1053
|
+
return this.decorateNodeStyle(node, this.getDefaultNodeStyle(node, props));
|
|
1054
|
+
}
|
|
1055
|
+
|
|
1056
|
+
/**
|
|
1057
|
+
* Decorate the node style.
|
|
1058
|
+
* @param node the tree node.
|
|
1059
|
+
* @param style the optional CSS properties.
|
|
1060
|
+
*
|
|
1061
|
+
* @returns the CSS styles if available.
|
|
1062
|
+
*/
|
|
1063
|
+
protected decorateNodeStyle(node: TreeNode, style: React.CSSProperties | undefined): React.CSSProperties | undefined {
|
|
1064
|
+
const backgroundColor = this.getDecorationData(node, 'backgroundColor').filter(notEmpty).shift();
|
|
1065
|
+
if (backgroundColor) {
|
|
1066
|
+
style = {
|
|
1067
|
+
...(style || {}),
|
|
1068
|
+
backgroundColor
|
|
1069
|
+
};
|
|
1070
|
+
}
|
|
1071
|
+
return style;
|
|
1072
|
+
}
|
|
1073
|
+
|
|
1074
|
+
/**
|
|
1075
|
+
* Determine if the tree node is expandable.
|
|
1076
|
+
* @param node the tree node.
|
|
1077
|
+
*
|
|
1078
|
+
* @returns `true` if the tree node is expandable.
|
|
1079
|
+
*/
|
|
1080
|
+
protected isExpandable(node: TreeNode): node is ExpandableTreeNode {
|
|
1081
|
+
return ExpandableTreeNode.is(node);
|
|
1082
|
+
}
|
|
1083
|
+
|
|
1084
|
+
/**
|
|
1085
|
+
* Get the tree node decorations.
|
|
1086
|
+
* @param node the tree node.
|
|
1087
|
+
*
|
|
1088
|
+
* @returns the list of tree decoration data.
|
|
1089
|
+
*/
|
|
1090
|
+
protected getDecorations(node: TreeNode): TreeDecoration.Data[] {
|
|
1091
|
+
const decorations: TreeDecoration.Data[] = [];
|
|
1092
|
+
if (DecoratedTreeNode.is(node)) {
|
|
1093
|
+
decorations.push(node.decorationData);
|
|
1094
|
+
}
|
|
1095
|
+
if (this.decorations.has(node.id)) {
|
|
1096
|
+
decorations.push(...this.decorations.get(node.id)!);
|
|
1097
|
+
}
|
|
1098
|
+
return decorations.sort(TreeDecoration.Data.comparePriority);
|
|
1099
|
+
}
|
|
1100
|
+
|
|
1101
|
+
/**
|
|
1102
|
+
* Get the tree decoration data for the given key.
|
|
1103
|
+
* @param node the tree node.
|
|
1104
|
+
* @param key the tree decoration data key.
|
|
1105
|
+
*
|
|
1106
|
+
* @returns the tree decoration data at the given key.
|
|
1107
|
+
*/
|
|
1108
|
+
protected getDecorationData<K extends keyof TreeDecoration.Data>(node: TreeNode, key: K): Required<Pick<TreeDecoration.Data, K>>[K][] {
|
|
1109
|
+
return this.getDecorations(node).filter(data => data[key] !== undefined).map(data => data[key]);
|
|
1110
|
+
}
|
|
1111
|
+
|
|
1112
|
+
/**
|
|
1113
|
+
* Store the last scroll state.
|
|
1114
|
+
*/
|
|
1115
|
+
protected lastScrollState: {
|
|
1116
|
+
/**
|
|
1117
|
+
* The scroll top value.
|
|
1118
|
+
*/
|
|
1119
|
+
scrollTop: number,
|
|
1120
|
+
/**
|
|
1121
|
+
* The scroll left value.
|
|
1122
|
+
*/
|
|
1123
|
+
scrollLeft: number
|
|
1124
|
+
} | undefined;
|
|
1125
|
+
|
|
1126
|
+
/**
|
|
1127
|
+
* Get the scroll container.
|
|
1128
|
+
*/
|
|
1129
|
+
protected override getScrollContainer(): MaybePromise<HTMLElement> {
|
|
1130
|
+
this.toDisposeOnDetach.push(Disposable.create(() => {
|
|
1131
|
+
const { scrollTop, scrollLeft } = this.node;
|
|
1132
|
+
this.lastScrollState = { scrollTop, scrollLeft };
|
|
1133
|
+
}));
|
|
1134
|
+
if (this.lastScrollState) {
|
|
1135
|
+
const { scrollTop, scrollLeft } = this.lastScrollState;
|
|
1136
|
+
this.node.scrollTop = scrollTop;
|
|
1137
|
+
this.node.scrollLeft = scrollLeft;
|
|
1138
|
+
}
|
|
1139
|
+
return this.node;
|
|
1140
|
+
}
|
|
1141
|
+
|
|
1142
|
+
protected override onAfterAttach(msg: Message): void {
|
|
1143
|
+
const up = [
|
|
1144
|
+
Key.ARROW_UP,
|
|
1145
|
+
KeyCode.createKeyCode({ first: Key.ARROW_UP, modifiers: [KeyModifier.Shift] })
|
|
1146
|
+
];
|
|
1147
|
+
const down = [
|
|
1148
|
+
Key.ARROW_DOWN,
|
|
1149
|
+
KeyCode.createKeyCode({ first: Key.ARROW_DOWN, modifiers: [KeyModifier.Shift] })
|
|
1150
|
+
];
|
|
1151
|
+
if (this.props.search) {
|
|
1152
|
+
if (this.searchBox.isAttached) {
|
|
1153
|
+
Widget.detach(this.searchBox);
|
|
1154
|
+
}
|
|
1155
|
+
UnsafeWidgetUtilities.attach(this.searchBox, this.node.parentElement!);
|
|
1156
|
+
this.addKeyListener(this.node, this.searchBox.keyCodePredicate.bind(this.searchBox), this.searchBox.handle.bind(this.searchBox));
|
|
1157
|
+
this.toDisposeOnDetach.push(Disposable.create(() => {
|
|
1158
|
+
Widget.detach(this.searchBox);
|
|
1159
|
+
}));
|
|
1160
|
+
}
|
|
1161
|
+
super.onAfterAttach(msg);
|
|
1162
|
+
this.addKeyListener(this.node, Key.ARROW_LEFT, event => this.handleLeft(event));
|
|
1163
|
+
this.addKeyListener(this.node, Key.ARROW_RIGHT, event => this.handleRight(event));
|
|
1164
|
+
this.addKeyListener(this.node, up, event => this.handleUp(event));
|
|
1165
|
+
this.addKeyListener(this.node, down, event => this.handleDown(event));
|
|
1166
|
+
this.addKeyListener(this.node, Key.ENTER, event => this.handleEnter(event));
|
|
1167
|
+
this.addKeyListener(this.node, Key.SPACE, event => this.handleSpace(event));
|
|
1168
|
+
this.addKeyListener(this.node, Key.ESCAPE, event => this.handleEscape(event));
|
|
1169
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
1170
|
+
this.addEventListener<any>(this.node, 'ps-scroll-y', (e: Event & { target: { scrollTop: number } }) => {
|
|
1171
|
+
if (this.view && this.view.list) {
|
|
1172
|
+
const { scrollTop } = e.target;
|
|
1173
|
+
this.view.list.scrollTo({
|
|
1174
|
+
top: scrollTop
|
|
1175
|
+
});
|
|
1176
|
+
}
|
|
1177
|
+
});
|
|
1178
|
+
}
|
|
1179
|
+
|
|
1180
|
+
/**
|
|
1181
|
+
* Handle the `left arrow` keyboard event.
|
|
1182
|
+
* @param event the `left arrow` keyboard event.
|
|
1183
|
+
*/
|
|
1184
|
+
protected async handleLeft(event: KeyboardEvent): Promise<void> {
|
|
1185
|
+
if (!!this.props.multiSelect && (this.hasCtrlCmdMask(event) || this.hasShiftMask(event))) {
|
|
1186
|
+
return;
|
|
1187
|
+
}
|
|
1188
|
+
if (!await this.model.collapseNode()) {
|
|
1189
|
+
this.model.selectParent();
|
|
1190
|
+
}
|
|
1191
|
+
}
|
|
1192
|
+
|
|
1193
|
+
/**
|
|
1194
|
+
* Handle the `right arrow` keyboard event.
|
|
1195
|
+
* @param event the `right arrow` keyboard event.
|
|
1196
|
+
*/
|
|
1197
|
+
protected async handleRight(event: KeyboardEvent): Promise<void> {
|
|
1198
|
+
if (!!this.props.multiSelect && (this.hasCtrlCmdMask(event) || this.hasShiftMask(event))) {
|
|
1199
|
+
return;
|
|
1200
|
+
}
|
|
1201
|
+
if (!await this.model.expandNode()) {
|
|
1202
|
+
this.model.selectNextNode();
|
|
1203
|
+
}
|
|
1204
|
+
}
|
|
1205
|
+
|
|
1206
|
+
/**
|
|
1207
|
+
* Handle the `up arrow` keyboard event.
|
|
1208
|
+
* @param event the `up arrow` keyboard event.
|
|
1209
|
+
*/
|
|
1210
|
+
protected handleUp(event: KeyboardEvent): void {
|
|
1211
|
+
if (!!this.props.multiSelect && this.hasShiftMask(event)) {
|
|
1212
|
+
this.model.selectPrevNode(TreeSelection.SelectionType.RANGE);
|
|
1213
|
+
} else {
|
|
1214
|
+
this.model.selectPrevNode();
|
|
1215
|
+
}
|
|
1216
|
+
this.node.focus();
|
|
1217
|
+
}
|
|
1218
|
+
|
|
1219
|
+
/**
|
|
1220
|
+
* Handle the `down arrow` keyboard event.
|
|
1221
|
+
* @param event the `down arrow` keyboard event.
|
|
1222
|
+
*/
|
|
1223
|
+
protected handleDown(event: KeyboardEvent): void {
|
|
1224
|
+
if (!!this.props.multiSelect && this.hasShiftMask(event)) {
|
|
1225
|
+
this.model.selectNextNode(TreeSelection.SelectionType.RANGE);
|
|
1226
|
+
} else {
|
|
1227
|
+
this.model.selectNextNode();
|
|
1228
|
+
}
|
|
1229
|
+
this.node.focus();
|
|
1230
|
+
}
|
|
1231
|
+
|
|
1232
|
+
/**
|
|
1233
|
+
* Handle the `enter key` keyboard event.
|
|
1234
|
+
* - `enter` opens the tree node.
|
|
1235
|
+
* @param event the `enter key` keyboard event.
|
|
1236
|
+
*/
|
|
1237
|
+
protected handleEnter(event: KeyboardEvent): void {
|
|
1238
|
+
this.model.openNode();
|
|
1239
|
+
}
|
|
1240
|
+
|
|
1241
|
+
/**
|
|
1242
|
+
* Handle the `space key` keyboard event.
|
|
1243
|
+
* - If the element has a checkbox, it will be toggled.
|
|
1244
|
+
* - Otherwise, it should be similar to a single-click action.
|
|
1245
|
+
* @param event the `space key` keyboard event.
|
|
1246
|
+
*/
|
|
1247
|
+
protected handleSpace(event: KeyboardEvent): void {
|
|
1248
|
+
const { focusedNode } = this.focusService;
|
|
1249
|
+
if (focusedNode && focusedNode.checkboxInfo) {
|
|
1250
|
+
this.model.markAsChecked(focusedNode, !focusedNode.checkboxInfo.checked);
|
|
1251
|
+
} else if (!this.props.multiSelect || (!event.ctrlKey && !event.metaKey && !event.shiftKey)) {
|
|
1252
|
+
this.tapNode(focusedNode);
|
|
1253
|
+
}
|
|
1254
|
+
}
|
|
1255
|
+
|
|
1256
|
+
protected handleEscape(event: KeyboardEvent): void {
|
|
1257
|
+
if (this.model.selectedNodes.length <= 1) {
|
|
1258
|
+
this.focusService.setFocus(undefined);
|
|
1259
|
+
this.node.focus();
|
|
1260
|
+
}
|
|
1261
|
+
this.model.clearSelection();
|
|
1262
|
+
}
|
|
1263
|
+
|
|
1264
|
+
/**
|
|
1265
|
+
* Handle the single-click mouse event.
|
|
1266
|
+
* @param node the tree node if available.
|
|
1267
|
+
* @param event the mouse single-click event.
|
|
1268
|
+
*/
|
|
1269
|
+
protected handleClickEvent(node: TreeNode | undefined, event: React.MouseEvent<HTMLElement>): void {
|
|
1270
|
+
if (node) {
|
|
1271
|
+
event.stopPropagation();
|
|
1272
|
+
const shiftMask = this.hasShiftMask(event);
|
|
1273
|
+
const ctrlCmdMask = this.hasCtrlCmdMask(event);
|
|
1274
|
+
if (this.props.multiSelect && (shiftMask || ctrlCmdMask) && SelectableTreeNode.is(node)) {
|
|
1275
|
+
if (shiftMask) {
|
|
1276
|
+
this.model.selectRange(node);
|
|
1277
|
+
} else if (ctrlCmdMask) {
|
|
1278
|
+
this.model.toggleNode(node);
|
|
1279
|
+
}
|
|
1280
|
+
} else {
|
|
1281
|
+
this.tapNode(node);
|
|
1282
|
+
}
|
|
1283
|
+
}
|
|
1284
|
+
}
|
|
1285
|
+
|
|
1286
|
+
/**
|
|
1287
|
+
* The effective handler of an unmodified single-click event.
|
|
1288
|
+
*/
|
|
1289
|
+
protected tapNode(node?: TreeNode): void {
|
|
1290
|
+
if (SelectableTreeNode.is(node)) {
|
|
1291
|
+
this.model.selectNode(node);
|
|
1292
|
+
}
|
|
1293
|
+
if (node && !this.props.expandOnlyOnExpansionToggleClick && this.isExpandable(node)) {
|
|
1294
|
+
this.model.toggleNodeExpansion(node);
|
|
1295
|
+
}
|
|
1296
|
+
}
|
|
1297
|
+
|
|
1298
|
+
/**
|
|
1299
|
+
* Handle the double-click mouse event.
|
|
1300
|
+
* @param node the tree node if available.
|
|
1301
|
+
* @param event the double-click mouse event.
|
|
1302
|
+
*/
|
|
1303
|
+
protected handleDblClickEvent(node: TreeNode | undefined, event: React.MouseEvent<HTMLElement>): void {
|
|
1304
|
+
this.model.openNode(node);
|
|
1305
|
+
event.stopPropagation();
|
|
1306
|
+
}
|
|
1307
|
+
|
|
1308
|
+
/**
|
|
1309
|
+
* Handle the middle-click mouse event.
|
|
1310
|
+
* @param node the tree node if available.
|
|
1311
|
+
* @param event the middle-click mouse event.
|
|
1312
|
+
*/
|
|
1313
|
+
protected handleAuxClickEvent(node: TreeNode | undefined, event: React.MouseEvent<HTMLElement>): void {
|
|
1314
|
+
if (event.button === 1) {
|
|
1315
|
+
this.model.openNode(node);
|
|
1316
|
+
if (SelectableTreeNode.is(node)) {
|
|
1317
|
+
this.model.selectNode(node);
|
|
1318
|
+
}
|
|
1319
|
+
}
|
|
1320
|
+
event.stopPropagation();
|
|
1321
|
+
}
|
|
1322
|
+
|
|
1323
|
+
/**
|
|
1324
|
+
* Handle the middle-click mouse event.
|
|
1325
|
+
* @param event the middle-click mouse event.
|
|
1326
|
+
*/
|
|
1327
|
+
protected handleMiddleClickEvent(event: MouseEvent): void {
|
|
1328
|
+
// Prevents auto-scrolling behavior when middle-clicking.
|
|
1329
|
+
if (event.button === 1) {
|
|
1330
|
+
event.preventDefault();
|
|
1331
|
+
}
|
|
1332
|
+
}
|
|
1333
|
+
|
|
1334
|
+
/**
|
|
1335
|
+
* Handle the context menu click event.
|
|
1336
|
+
* - The context menu click event is triggered by the right-click.
|
|
1337
|
+
* @param node the tree node if available.
|
|
1338
|
+
* @param event the right-click mouse event.
|
|
1339
|
+
*/
|
|
1340
|
+
protected handleContextMenuEvent(node: TreeNode | undefined, event: React.MouseEvent<HTMLElement>): void {
|
|
1341
|
+
if (SelectableTreeNode.is(node)) {
|
|
1342
|
+
// Keep the selection for the context menu, if the widget support multi-selection and the right click happens on an already selected node.
|
|
1343
|
+
if (!this.props.multiSelect || !node.selected) {
|
|
1344
|
+
const type = !!this.props.multiSelect && this.hasCtrlCmdMask(event) ? TreeSelection.SelectionType.TOGGLE : TreeSelection.SelectionType.DEFAULT;
|
|
1345
|
+
this.model.addSelection({ node, type });
|
|
1346
|
+
}
|
|
1347
|
+
this.focusService.setFocus(node);
|
|
1348
|
+
const contextMenuPath = this.props.contextMenuPath;
|
|
1349
|
+
if (contextMenuPath) {
|
|
1350
|
+
const { x, y } = event.nativeEvent;
|
|
1351
|
+
const args = this.toContextMenuArgs(node);
|
|
1352
|
+
setTimeout(() => this.contextMenuRenderer.render({
|
|
1353
|
+
menuPath: contextMenuPath,
|
|
1354
|
+
anchor: { x, y },
|
|
1355
|
+
args
|
|
1356
|
+
}), 10);
|
|
1357
|
+
}
|
|
1358
|
+
}
|
|
1359
|
+
event.stopPropagation();
|
|
1360
|
+
event.preventDefault();
|
|
1361
|
+
}
|
|
1362
|
+
|
|
1363
|
+
/**
|
|
1364
|
+
* Handle the double-click mouse event on the expansion toggle.
|
|
1365
|
+
*/
|
|
1366
|
+
protected readonly handleExpansionToggleDblClickEvent = (event: React.MouseEvent<HTMLElement>) => this.doHandleExpansionToggleDblClickEvent(event);
|
|
1367
|
+
/**
|
|
1368
|
+
* Actually handle the double-click mouse event on the expansion toggle.
|
|
1369
|
+
* @param event the double-click mouse event.
|
|
1370
|
+
*/
|
|
1371
|
+
protected doHandleExpansionToggleDblClickEvent(event: React.MouseEvent<HTMLElement>): void {
|
|
1372
|
+
if (this.props.expandOnlyOnExpansionToggleClick) {
|
|
1373
|
+
// Ignore the double-click event.
|
|
1374
|
+
event.stopPropagation();
|
|
1375
|
+
}
|
|
1376
|
+
}
|
|
1377
|
+
|
|
1378
|
+
/**
|
|
1379
|
+
* Convert the tree node to context menu arguments.
|
|
1380
|
+
* @param node the selectable tree node.
|
|
1381
|
+
*/
|
|
1382
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
1383
|
+
protected toContextMenuArgs(node: SelectableTreeNode): any[] | undefined {
|
|
1384
|
+
return undefined;
|
|
1385
|
+
}
|
|
1386
|
+
|
|
1387
|
+
/**
|
|
1388
|
+
* Determine if the tree modifier aware event has a `ctrlcmd` mask.
|
|
1389
|
+
* @param event the tree modifier aware event.
|
|
1390
|
+
*
|
|
1391
|
+
* @returns `true` if the tree modifier aware event contains the `ctrlcmd` mask.
|
|
1392
|
+
*/
|
|
1393
|
+
protected hasCtrlCmdMask(event: TreeWidget.ModifierAwareEvent): boolean {
|
|
1394
|
+
return isOSX ? event.metaKey : event.ctrlKey;
|
|
1395
|
+
}
|
|
1396
|
+
|
|
1397
|
+
/**
|
|
1398
|
+
* Determine if the tree modifier aware event has a `shift` mask.
|
|
1399
|
+
* @param event the tree modifier aware event.
|
|
1400
|
+
*
|
|
1401
|
+
* @returns `true` if the tree modifier aware event contains the `shift` mask.
|
|
1402
|
+
*/
|
|
1403
|
+
protected hasShiftMask(event: TreeWidget.ModifierAwareEvent): boolean {
|
|
1404
|
+
// Ctrl/Cmd mask overrules the Shift mask.
|
|
1405
|
+
if (this.hasCtrlCmdMask(event)) {
|
|
1406
|
+
return false;
|
|
1407
|
+
}
|
|
1408
|
+
return event.shiftKey;
|
|
1409
|
+
}
|
|
1410
|
+
|
|
1411
|
+
/**
|
|
1412
|
+
* Deflate the tree node for storage.
|
|
1413
|
+
* @param node the tree node.
|
|
1414
|
+
*/
|
|
1415
|
+
protected deflateForStorage(node: TreeNode): object {
|
|
1416
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
1417
|
+
const copy = Object.assign({}, node) as any;
|
|
1418
|
+
if (copy.parent) {
|
|
1419
|
+
delete copy.parent;
|
|
1420
|
+
}
|
|
1421
|
+
if ('previousSibling' in copy) {
|
|
1422
|
+
delete copy.previousSibling;
|
|
1423
|
+
}
|
|
1424
|
+
if ('nextSibling' in copy) {
|
|
1425
|
+
delete copy.nextSibling;
|
|
1426
|
+
}
|
|
1427
|
+
if ('busy' in copy) {
|
|
1428
|
+
delete copy.busy;
|
|
1429
|
+
}
|
|
1430
|
+
if (CompositeTreeNode.is(node)) {
|
|
1431
|
+
copy.children = [];
|
|
1432
|
+
for (const child of node.children) {
|
|
1433
|
+
copy.children.push(this.deflateForStorage(child));
|
|
1434
|
+
}
|
|
1435
|
+
}
|
|
1436
|
+
return copy;
|
|
1437
|
+
}
|
|
1438
|
+
|
|
1439
|
+
/**
|
|
1440
|
+
* Inflate the tree node from storage.
|
|
1441
|
+
* @param node the tree node.
|
|
1442
|
+
* @param parent the optional tree node.
|
|
1443
|
+
*/
|
|
1444
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
1445
|
+
protected inflateFromStorage(node: any, parent?: TreeNode): TreeNode {
|
|
1446
|
+
if (node.selected) {
|
|
1447
|
+
node.selected = false;
|
|
1448
|
+
}
|
|
1449
|
+
if (parent) {
|
|
1450
|
+
node.parent = parent;
|
|
1451
|
+
}
|
|
1452
|
+
if (Array.isArray(node.children)) {
|
|
1453
|
+
for (const child of node.children as TreeNode[]) {
|
|
1454
|
+
this.inflateFromStorage(child, node);
|
|
1455
|
+
}
|
|
1456
|
+
}
|
|
1457
|
+
return node;
|
|
1458
|
+
}
|
|
1459
|
+
|
|
1460
|
+
/**
|
|
1461
|
+
* Store the tree state.
|
|
1462
|
+
*/
|
|
1463
|
+
storeState(): object {
|
|
1464
|
+
const decorations = this.decoratorService.deflateDecorators(this.decorations);
|
|
1465
|
+
let state: object = {
|
|
1466
|
+
decorations
|
|
1467
|
+
};
|
|
1468
|
+
if (this.model.root) {
|
|
1469
|
+
state = {
|
|
1470
|
+
...state,
|
|
1471
|
+
root: this.deflateForStorage(this.model.root),
|
|
1472
|
+
model: this.model.storeState(),
|
|
1473
|
+
focusedNodeId: this.focusService.focusedNode?.id
|
|
1474
|
+
};
|
|
1475
|
+
}
|
|
1476
|
+
|
|
1477
|
+
return state;
|
|
1478
|
+
}
|
|
1479
|
+
|
|
1480
|
+
/**
|
|
1481
|
+
* Restore the state.
|
|
1482
|
+
* @param oldState the old state object.
|
|
1483
|
+
*/
|
|
1484
|
+
restoreState(oldState: object): void {
|
|
1485
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
1486
|
+
const { root, decorations, model, focusedNodeId } = (oldState as any);
|
|
1487
|
+
if (root) {
|
|
1488
|
+
this.model.root = this.inflateFromStorage(root);
|
|
1489
|
+
}
|
|
1490
|
+
if (decorations) {
|
|
1491
|
+
this.decorations = this.decoratorService.inflateDecorators(decorations);
|
|
1492
|
+
}
|
|
1493
|
+
if (model) {
|
|
1494
|
+
this.model.restoreState(model);
|
|
1495
|
+
}
|
|
1496
|
+
if (focusedNodeId) {
|
|
1497
|
+
const candidate = this.model.getNode(focusedNodeId);
|
|
1498
|
+
if (SelectableTreeNode.is(candidate)) {
|
|
1499
|
+
this.focusService.setFocus(candidate);
|
|
1500
|
+
}
|
|
1501
|
+
}
|
|
1502
|
+
}
|
|
1503
|
+
|
|
1504
|
+
protected toNodeIcon(node: TreeNode): string {
|
|
1505
|
+
return this.labelProvider.getIcon(node);
|
|
1506
|
+
}
|
|
1507
|
+
|
|
1508
|
+
protected toNodeName(node: TreeNode): string {
|
|
1509
|
+
return this.labelProvider.getName(node);
|
|
1510
|
+
}
|
|
1511
|
+
|
|
1512
|
+
protected toNodeDescription(node: TreeNode): string {
|
|
1513
|
+
return this.labelProvider.getLongName(node);
|
|
1514
|
+
}
|
|
1515
|
+
protected getDepthPadding(depth: number): number {
|
|
1516
|
+
if (depth === 1) {
|
|
1517
|
+
return this.props.leftPadding;
|
|
1518
|
+
}
|
|
1519
|
+
return depth * this.treeIndent;
|
|
1520
|
+
}
|
|
1521
|
+
}
|
|
1522
|
+
export namespace TreeWidget {
|
|
1523
|
+
/**
|
|
1524
|
+
* Representation of a tree node row.
|
|
1525
|
+
*/
|
|
1526
|
+
export interface NodeRow {
|
|
1527
|
+
/**
|
|
1528
|
+
* The node row index.
|
|
1529
|
+
*/
|
|
1530
|
+
index: number
|
|
1531
|
+
/**
|
|
1532
|
+
* The actual node.
|
|
1533
|
+
*/
|
|
1534
|
+
node: TreeNode
|
|
1535
|
+
/**
|
|
1536
|
+
* A root relative number representing the hierarchical depth of the actual node. Root is `0`, its children have `1` and so on.
|
|
1537
|
+
*/
|
|
1538
|
+
depth: number
|
|
1539
|
+
}
|
|
1540
|
+
/**
|
|
1541
|
+
* Representation of the tree view properties.
|
|
1542
|
+
*/
|
|
1543
|
+
export interface ViewProps {
|
|
1544
|
+
/**
|
|
1545
|
+
* The width property.
|
|
1546
|
+
*/
|
|
1547
|
+
width: number
|
|
1548
|
+
/**
|
|
1549
|
+
* The height property.
|
|
1550
|
+
*/
|
|
1551
|
+
height: number
|
|
1552
|
+
/**
|
|
1553
|
+
* The scroll to row value.
|
|
1554
|
+
*/
|
|
1555
|
+
scrollToRow?: number
|
|
1556
|
+
/**
|
|
1557
|
+
* The list of node rows.
|
|
1558
|
+
*/
|
|
1559
|
+
rows: NodeRow[]
|
|
1560
|
+
renderNodeRow: (row: NodeRow) => React.ReactNode
|
|
1561
|
+
}
|
|
1562
|
+
export class View extends React.Component<ViewProps> {
|
|
1563
|
+
list: VirtuosoHandle | undefined;
|
|
1564
|
+
override render(): React.ReactNode {
|
|
1565
|
+
const { rows, width, height, scrollToRow } = this.props;
|
|
1566
|
+
return <Virtuoso
|
|
1567
|
+
ref={list => {
|
|
1568
|
+
this.list = (list || undefined);
|
|
1569
|
+
if (this.list && scrollToRow !== undefined) {
|
|
1570
|
+
this.list.scrollIntoView({
|
|
1571
|
+
index: scrollToRow,
|
|
1572
|
+
align: 'center'
|
|
1573
|
+
});
|
|
1574
|
+
}
|
|
1575
|
+
}}
|
|
1576
|
+
totalCount={rows.length}
|
|
1577
|
+
itemContent={index => this.props.renderNodeRow(rows[index])}
|
|
1578
|
+
width={width}
|
|
1579
|
+
height={height}
|
|
1580
|
+
// This is a pixel value, it will scan 200px to the top and bottom of the current view
|
|
1581
|
+
overscan={500}
|
|
1582
|
+
/>;
|
|
1583
|
+
}
|
|
1584
|
+
}
|
|
1585
|
+
}
|