@theia/plugin-ext 1.53.0-next.4 → 1.53.0-next.55
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +54 -54
- package/lib/common/plugin-api-rpc.d.ts +19 -4
- package/lib/common/plugin-api-rpc.d.ts.map +1 -1
- package/lib/common/plugin-api-rpc.js +4 -2
- package/lib/common/plugin-api-rpc.js.map +1 -1
- package/lib/common/rpc-protocol.d.ts +14 -13
- package/lib/common/rpc-protocol.d.ts.map +1 -1
- package/lib/common/rpc-protocol.js +56 -53
- package/lib/common/rpc-protocol.js.map +1 -1
- package/lib/hosted/browser/hosted-plugin.d.ts +1 -0
- package/lib/hosted/browser/hosted-plugin.d.ts.map +1 -1
- package/lib/hosted/browser/hosted-plugin.js +22 -12
- package/lib/hosted/browser/hosted-plugin.js.map +1 -1
- package/lib/hosted/node/scanners/scanner-theia.d.ts.map +1 -1
- package/lib/hosted/node/scanners/scanner-theia.js +13 -1
- package/lib/hosted/node/scanners/scanner-theia.js.map +1 -1
- package/lib/main/browser/authentication-main.d.ts +6 -5
- package/lib/main/browser/authentication-main.d.ts.map +1 -1
- package/lib/main/browser/authentication-main.js +15 -9
- package/lib/main/browser/authentication-main.js.map +1 -1
- package/lib/main/browser/custom-editors/custom-editor-opener.d.ts +9 -3
- package/lib/main/browser/custom-editors/custom-editor-opener.d.ts.map +1 -1
- package/lib/main/browser/custom-editors/custom-editor-opener.js +120 -31
- package/lib/main/browser/custom-editors/custom-editor-opener.js.map +1 -1
- package/lib/main/browser/custom-editors/custom-editor-widget.d.ts +7 -4
- package/lib/main/browser/custom-editors/custom-editor-widget.d.ts.map +1 -1
- package/lib/main/browser/custom-editors/custom-editor-widget.js +21 -7
- package/lib/main/browser/custom-editors/custom-editor-widget.js.map +1 -1
- package/lib/main/browser/custom-editors/custom-editors-main.d.ts +3 -4
- package/lib/main/browser/custom-editors/custom-editors-main.d.ts.map +1 -1
- package/lib/main/browser/custom-editors/custom-editors-main.js +5 -55
- package/lib/main/browser/custom-editors/custom-editors-main.js.map +1 -1
- package/lib/main/browser/custom-editors/plugin-custom-editor-registry.d.ts +4 -3
- package/lib/main/browser/custom-editors/plugin-custom-editor-registry.d.ts.map +1 -1
- package/lib/main/browser/custom-editors/plugin-custom-editor-registry.js +24 -16
- package/lib/main/browser/custom-editors/plugin-custom-editor-registry.js.map +1 -1
- package/lib/main/browser/main-context.d.ts.map +1 -1
- package/lib/main/browser/main-context.js +3 -0
- package/lib/main/browser/main-context.js.map +1 -1
- package/lib/main/browser/menus/menus-contribution-handler.d.ts.map +1 -1
- package/lib/main/browser/menus/menus-contribution-handler.js +1 -1
- package/lib/main/browser/menus/menus-contribution-handler.js.map +1 -1
- package/lib/main/browser/menus/plugin-menu-command-adapter.d.ts +1 -0
- package/lib/main/browser/menus/plugin-menu-command-adapter.d.ts.map +1 -1
- package/lib/main/browser/menus/plugin-menu-command-adapter.js +22 -1
- package/lib/main/browser/menus/plugin-menu-command-adapter.js.map +1 -1
- package/lib/main/browser/menus/vscode-theia-menu-mappings.d.ts +2 -2
- package/lib/main/browser/menus/vscode-theia-menu-mappings.d.ts.map +1 -1
- package/lib/main/browser/menus/vscode-theia-menu-mappings.js +2 -0
- package/lib/main/browser/menus/vscode-theia-menu-mappings.js.map +1 -1
- package/lib/main/browser/notebooks/notebook-documents-and-editors-main.js +1 -1
- package/lib/main/browser/notebooks/notebook-documents-and-editors-main.js.map +1 -1
- package/lib/main/browser/notebooks/notebook-documents-main.d.ts.map +1 -1
- package/lib/main/browser/notebooks/notebook-documents-main.js +22 -3
- package/lib/main/browser/notebooks/notebook-documents-main.js.map +1 -1
- package/lib/main/browser/notebooks/notebook-editors-main.d.ts +1 -1
- package/lib/main/browser/notebooks/notebook-editors-main.d.ts.map +1 -1
- package/lib/main/browser/notebooks/notebook-editors-main.js +16 -2
- package/lib/main/browser/notebooks/notebook-editors-main.js.map +1 -1
- package/lib/main/browser/notebooks/renderers/cell-output-webview.d.ts +1 -1
- package/lib/main/browser/notebooks/renderers/cell-output-webview.d.ts.map +1 -1
- package/lib/main/browser/notebooks/renderers/cell-output-webview.js +152 -77
- package/lib/main/browser/notebooks/renderers/cell-output-webview.js.map +1 -1
- package/lib/main/browser/notebooks/renderers/output-webview-internal.d.ts.map +1 -1
- package/lib/main/browser/notebooks/renderers/output-webview-internal.js +38 -19
- package/lib/main/browser/notebooks/renderers/output-webview-internal.js.map +1 -1
- package/lib/main/browser/plugin-ext-frontend-module.d.ts.map +1 -1
- package/lib/main/browser/plugin-ext-frontend-module.js +18 -0
- package/lib/main/browser/plugin-ext-frontend-module.js.map +1 -1
- package/lib/main/browser/plugin-icon-theme-service.js +20 -20
- package/lib/main/browser/plugin-shared-style.js +14 -14
- package/lib/main/browser/uri-main.d.ts +16 -0
- package/lib/main/browser/uri-main.d.ts.map +1 -0
- package/lib/main/browser/uri-main.js +64 -0
- package/lib/main/browser/uri-main.js.map +1 -0
- package/lib/main/browser/view/plugin-view-registry.js +3 -3
- package/lib/main/browser/webview/webview-frontend-security-warnings.js +2 -2
- package/lib/main/browser/webview/webview-secondary-window-support.js +10 -10
- package/lib/main/browser/webview/webview.d.ts +1 -3
- package/lib/main/browser/webview/webview.d.ts.map +1 -1
- package/lib/main/browser/webview/webview.js +5 -4
- package/lib/main/browser/webview/webview.js.map +1 -1
- package/lib/main/browser/webviews-main.d.ts.map +1 -1
- package/lib/main/browser/webviews-main.js +6 -1
- package/lib/main/browser/webviews-main.js.map +1 -1
- package/lib/main/browser/workspace-main.d.ts +2 -0
- package/lib/main/browser/workspace-main.d.ts.map +1 -1
- package/lib/main/node/plugin-service.d.ts +1 -1
- package/lib/main/node/webview-backend-security-warnings.js +6 -6
- package/lib/plugin/authentication-ext.d.ts +3 -2
- package/lib/plugin/authentication-ext.d.ts.map +1 -1
- package/lib/plugin/authentication-ext.js +8 -5
- package/lib/plugin/authentication-ext.js.map +1 -1
- package/lib/plugin/custom-editors.d.ts +2 -4
- package/lib/plugin/custom-editors.d.ts.map +1 -1
- package/lib/plugin/custom-editors.js +5 -7
- package/lib/plugin/custom-editors.js.map +1 -1
- package/lib/plugin/env.d.ts +2 -0
- package/lib/plugin/env.d.ts.map +1 -1
- package/lib/plugin/env.js +4 -1
- package/lib/plugin/env.js.map +1 -1
- package/lib/plugin/known-commands.d.ts.map +1 -1
- package/lib/plugin/known-commands.js +1 -0
- package/lib/plugin/known-commands.js.map +1 -1
- package/lib/plugin/languages.d.ts +1 -0
- package/lib/plugin/languages.d.ts.map +1 -1
- package/lib/plugin/notebook/notebook-kernels.d.ts +2 -0
- package/lib/plugin/notebook/notebook-kernels.d.ts.map +1 -1
- package/lib/plugin/plugin-context.d.ts.map +1 -1
- package/lib/plugin/plugin-context.js +17 -6
- package/lib/plugin/plugin-context.js.map +1 -1
- package/lib/plugin/plugin-manager.d.ts.map +1 -1
- package/lib/plugin/plugin-manager.js +2 -1
- package/lib/plugin/plugin-manager.js.map +1 -1
- package/lib/plugin/scm.d.ts +1 -0
- package/lib/plugin/scm.d.ts.map +1 -1
- package/lib/plugin/terminal-ext.d.ts +3 -0
- package/lib/plugin/terminal-ext.d.ts.map +1 -1
- package/lib/plugin/terminal-ext.js +2 -0
- package/lib/plugin/terminal-ext.js.map +1 -1
- package/lib/plugin/type-converters.d.ts +6 -0
- package/lib/plugin/type-converters.d.ts.map +1 -1
- package/lib/plugin/type-converters.js +24 -3
- package/lib/plugin/type-converters.js.map +1 -1
- package/lib/plugin/types-impl.d.ts +8 -2
- package/lib/plugin/types-impl.d.ts.map +1 -1
- package/lib/plugin/types-impl.js +9 -1
- package/lib/plugin/types-impl.js.map +1 -1
- package/lib/plugin/uri-ext.d.ts +37 -0
- package/lib/plugin/uri-ext.d.ts.map +1 -0
- package/lib/plugin/uri-ext.js +50 -0
- package/lib/plugin/uri-ext.js.map +1 -0
- package/lib/plugin/webviews.js +1 -1
- package/lib/plugin/webviews.js.map +1 -1
- package/lib/plugin/workspace.d.ts +4 -0
- package/lib/plugin/workspace.d.ts.map +1 -1
- package/package.json +29 -29
- package/src/common/arrays.ts +70 -70
- package/src/common/assert.ts +23 -23
- package/src/common/cache.ts +51 -51
- package/src/common/character-classifier.ts +73 -73
- package/src/common/collections.ts +54 -54
- package/src/common/commands.ts +19 -19
- package/src/common/connection.ts +137 -137
- package/src/common/disposable-util.ts +39 -39
- package/src/common/editor-options.ts +74 -74
- package/src/common/env.ts +19 -19
- package/src/common/errors.ts +63 -63
- package/src/common/id-generator.ts +26 -26
- package/src/common/index.ts +24 -24
- package/src/common/language-pack-service.ts +34 -34
- package/src/common/link-computer.ts +354 -354
- package/src/common/object-identifier.ts +33 -33
- package/src/common/objects.ts +50 -50
- package/src/common/paths-util.ts +158 -158
- package/src/common/plugin-api-rpc-model.ts +907 -907
- package/src/common/plugin-api-rpc.ts +2752 -2738
- package/src/common/plugin-ext-api-contribution.ts +115 -115
- package/src/common/plugin-identifiers.ts +84 -84
- package/src/common/plugin-protocol.ts +1094 -1094
- package/src/common/reference-map.ts +38 -38
- package/src/common/rpc-protocol.ts +306 -310
- package/src/common/semantic-tokens-dto.ts +182 -182
- package/src/common/test-types.ts +154 -154
- package/src/common/types.ts +129 -129
- package/src/common/uint.ts +37 -37
- package/src/common/uri-components.ts +81 -81
- package/src/hosted/browser/hosted-plugin-watcher.ts +54 -54
- package/src/hosted/browser/hosted-plugin.ts +635 -624
- package/src/hosted/browser/plugin-worker.ts +52 -52
- package/src/hosted/browser/worker/debug-stub.ts +29 -29
- package/src/hosted/browser/worker/plugin-manifest-loader.ts +114 -114
- package/src/hosted/browser/worker/worker-env-ext.ts +40 -40
- package/src/hosted/browser/worker/worker-main.ts +212 -212
- package/src/hosted/browser/worker/worker-plugin-module.ts +80 -80
- package/src/hosted/common/hosted-plugin.ts +456 -456
- package/src/hosted/node/hosted-plugin-cli-contribution.ts +75 -75
- package/src/hosted/node/hosted-plugin-deployer-handler.ts +274 -274
- package/src/hosted/node/hosted-plugin-localization-service.ts +410 -410
- package/src/hosted/node/hosted-plugin-process.ts +248 -248
- package/src/hosted/node/hosted-plugin-protocol.ts +49 -49
- package/src/hosted/node/hosted-plugin.ts +116 -116
- package/src/hosted/node/metadata-scanner.ts +64 -64
- package/src/hosted/node/plugin-activation-events.ts +112 -112
- package/src/hosted/node/plugin-ext-hosted-backend-module.ts +94 -94
- package/src/hosted/node/plugin-host-module.ts +69 -69
- package/src/hosted/node/plugin-host-proxy.ts +82 -82
- package/src/hosted/node/plugin-host-rpc.ts +377 -377
- package/src/hosted/node/plugin-host.ts +110 -110
- package/src/hosted/node/plugin-language-pack-service.ts +43 -43
- package/src/hosted/node/plugin-manifest-loader.ts +32 -32
- package/src/hosted/node/plugin-reader.ts +136 -136
- package/src/hosted/node/plugin-service.ts +197 -197
- package/src/hosted/node/scanners/backend-init-theia.ts +71 -71
- package/src/hosted/node/scanners/file-plugin-uri-factory.ts +32 -32
- package/src/hosted/node/scanners/grammars-reader.ts +57 -57
- package/src/hosted/node/scanners/plugin-uri-factory.ts +33 -33
- package/src/hosted/node/scanners/scanner-theia.ts +963 -952
- package/src/hosted/node-electron/plugin-ext-hosted-electron-backend-module.ts +26 -26
- package/src/hosted/node-electron/scanner-theia-electron.ts +32 -32
- package/src/main/browser/authentication-main.ts +401 -388
- package/src/main/browser/clipboard-main.ts +38 -38
- package/src/main/browser/command-registry-main.ts +130 -130
- package/src/main/browser/commands.ts +104 -104
- package/src/main/browser/comments/comment-glyph-widget.ts +66 -66
- package/src/main/browser/comments/comment-thread-widget.tsx +696 -696
- package/src/main/browser/comments/comments-context-key-service.ts +68 -68
- package/src/main/browser/comments/comments-contribution.ts +268 -268
- package/src/main/browser/comments/comments-decorator.ts +110 -110
- package/src/main/browser/comments/comments-main.ts +482 -482
- package/src/main/browser/comments/comments-service.ts +205 -205
- package/src/main/browser/custom-editors/custom-editor-opener.tsx +205 -105
- package/src/main/browser/custom-editors/custom-editor-service.ts +108 -108
- package/src/main/browser/custom-editors/custom-editor-undo-redo-handler.ts +41 -41
- package/src/main/browser/custom-editors/custom-editor-widget-factory.ts +44 -44
- package/src/main/browser/custom-editors/custom-editor-widget.ts +127 -118
- package/src/main/browser/custom-editors/custom-editors-main.ts +526 -585
- package/src/main/browser/custom-editors/plugin-custom-editor-registry.ts +126 -120
- package/src/main/browser/data-transfer/data-transfer-type-converters.ts +68 -68
- package/src/main/browser/debug/debug-main.ts +397 -397
- package/src/main/browser/debug/plugin-debug-adapter-contribution.ts +48 -48
- package/src/main/browser/debug/plugin-debug-configuration-provider.ts +63 -63
- package/src/main/browser/debug/plugin-debug-service.ts +427 -427
- package/src/main/browser/debug/plugin-debug-session-contribution-registry.ts +76 -76
- package/src/main/browser/debug/plugin-debug-session-factory.ts +115 -115
- package/src/main/browser/decorations/decorations-main.ts +146 -146
- package/src/main/browser/dialogs/modal-notification.ts +112 -112
- package/src/main/browser/dialogs/style/modal-notification.css +123 -123
- package/src/main/browser/dialogs-main.ts +185 -185
- package/src/main/browser/documents-main.ts +275 -275
- package/src/main/browser/editors-and-documents-main.ts +425 -425
- package/src/main/browser/env-main.ts +60 -60
- package/src/main/browser/file-system-main-impl.ts +267 -267
- package/src/main/browser/hierarchy/hierarchy-types-converters.ts +189 -189
- package/src/main/browser/keybindings/keybindings-contribution-handler.ts +66 -66
- package/src/main/browser/label-service-main.ts +51 -51
- package/src/main/browser/languages-main.ts +1439 -1439
- package/src/main/browser/localization-main.ts +34 -34
- package/src/main/browser/main-context.ts +210 -206
- package/src/main/browser/main-file-system-event-service.ts +76 -76
- package/src/main/browser/menus/menus-contribution-handler.ts +172 -171
- package/src/main/browser/menus/plugin-menu-command-adapter.ts +358 -337
- package/src/main/browser/menus/vscode-theia-menu-mappings.ts +118 -116
- package/src/main/browser/message-registry-main.ts +43 -43
- package/src/main/browser/notebooks/notebook-documents-and-editors-main.ts +249 -249
- package/src/main/browser/notebooks/notebook-documents-main.ts +202 -182
- package/src/main/browser/notebooks/notebook-dto.ts +131 -131
- package/src/main/browser/notebooks/notebook-editors-main.ts +88 -75
- package/src/main/browser/notebooks/notebook-kernels-main.ts +339 -339
- package/src/main/browser/notebooks/notebook-renderers-main.ts +47 -47
- package/src/main/browser/notebooks/notebooks-main.ts +164 -164
- package/src/main/browser/notebooks/renderers/cell-output-webview.tsx +452 -375
- package/src/main/browser/notebooks/renderers/output-webview-internal.ts +636 -613
- package/src/main/browser/notebooks/renderers/webview-communication.ts +107 -107
- package/src/main/browser/notification-main.ts +26 -26
- package/src/main/browser/output-channel-registry-main.ts +53 -53
- package/src/main/browser/plugin-authentication-service.ts +71 -71
- package/src/main/browser/plugin-contribution-handler.ts +698 -698
- package/src/main/browser/plugin-ext-frontend-module.ts +291 -272
- package/src/main/browser/plugin-ext-widget.tsx +132 -132
- package/src/main/browser/plugin-frontend-contribution.ts +70 -70
- package/src/main/browser/plugin-frontend-view-contribution.ts +38 -38
- package/src/main/browser/plugin-icon-service.ts +92 -92
- package/src/main/browser/plugin-icon-theme-service.ts +625 -625
- package/src/main/browser/plugin-shared-style.ts +154 -154
- package/src/main/browser/plugin-storage.ts +55 -55
- package/src/main/browser/plugin-terminal-registry.ts +27 -27
- package/src/main/browser/preference-registry-main.ts +123 -123
- package/src/main/browser/quick-open-main.ts +367 -367
- package/src/main/browser/scm-main.ts +472 -472
- package/src/main/browser/secrets-main.ts +82 -82
- package/src/main/browser/selection-provider-command.ts +45 -45
- package/src/main/browser/status-bar-message-registry-main.ts +90 -90
- package/src/main/browser/style/comments.css +345 -345
- package/src/main/browser/style/index.css +84 -84
- package/src/main/browser/style/plugin-sidebar.css +73 -73
- package/src/main/browser/style/tree.css +54 -54
- package/src/main/browser/style/webview.css +55 -55
- package/src/main/browser/tabs/tabs-main.ts +384 -384
- package/src/main/browser/tasks-main.ts +268 -268
- package/src/main/browser/terminal-main.ts +370 -370
- package/src/main/browser/test-main.ts +632 -632
- package/src/main/browser/text-editor-main.ts +478 -478
- package/src/main/browser/text-editor-model-service.ts +111 -111
- package/src/main/browser/text-editors-main.ts +234 -234
- package/src/main/browser/theme-icon-override.ts +246 -246
- package/src/main/browser/theming-main.ts +42 -42
- package/src/main/browser/timeline-main.ts +80 -80
- package/src/main/browser/uri-main.ts +72 -0
- package/src/main/browser/view/dnd-file-content-store.ts +42 -42
- package/src/main/browser/view/plugin-tree-view-node-label-provider.ts +81 -81
- package/src/main/browser/view/plugin-view-registry.ts +967 -967
- package/src/main/browser/view/plugin-view-widget.ts +221 -221
- package/src/main/browser/view/tree-view-decorator-service.ts +51 -51
- package/src/main/browser/view/tree-view-widget.tsx +931 -931
- package/src/main/browser/view/tree-views-main.ts +205 -205
- package/src/main/browser/view/view-context-key-service.ts +64 -64
- package/src/main/browser/webview/pre/fake.html +14 -14
- package/src/main/browser/webview/pre/host.js +130 -130
- package/src/main/browser/webview/pre/index.html +17 -17
- package/src/main/browser/webview/pre/main.js +682 -682
- package/src/main/browser/webview/pre/service-worker.js +296 -295
- package/src/main/browser/webview/webview-context-keys.ts +62 -62
- package/src/main/browser/webview/webview-environment.ts +87 -87
- package/src/main/browser/webview/webview-frontend-security-warnings.ts +59 -59
- package/src/main/browser/webview/webview-preferences.ts +72 -72
- package/src/main/browser/webview/webview-resource-cache.ts +88 -88
- package/src/main/browser/webview/webview-secondary-window-support.ts +47 -47
- package/src/main/browser/webview/webview-theme-data-provider.ts +124 -124
- package/src/main/browser/webview/webview.ts +718 -719
- package/src/main/browser/webview-views/webview-views-main.ts +154 -154
- package/src/main/browser/webview-views/webview-views.ts +43 -43
- package/src/main/browser/webviews-main.ts +287 -282
- package/src/main/browser/window-activity-tracker.ts +96 -96
- package/src/main/browser/window-state-main.ts +85 -85
- package/src/main/browser/workspace-main.ts +424 -424
- package/src/main/common/basic-message-registry-main.ts +53 -53
- package/src/main/common/basic-notification-main.ts +86 -86
- package/src/main/common/env-main.ts +44 -44
- package/src/main/common/plugin-paths-protocol.ts +26 -26
- package/src/main/common/plugin-theia-environment.ts +36 -36
- package/src/main/common/webview-protocol.ts +28 -28
- package/src/main/electron-browser/plugin-ext-frontend-electron-module.ts +25 -25
- package/src/main/electron-browser/webview/electron-webview-widget-factory.ts +59 -59
- package/src/main/node/errors.spec.ts +37 -37
- package/src/main/node/handlers/plugin-theia-directory-handler.ts +137 -137
- package/src/main/node/handlers/plugin-theia-file-handler.ts +66 -66
- package/src/main/node/paths/const.ts +21 -21
- package/src/main/node/paths/plugin-paths-service.ts +163 -163
- package/src/main/node/plugin-cli-contribution.ts +85 -85
- package/src/main/node/plugin-deployer-contribution.ts +35 -35
- package/src/main/node/plugin-deployer-directory-handler-context-impl.ts +45 -45
- package/src/main/node/plugin-deployer-entry-impl.ts +132 -132
- package/src/main/node/plugin-deployer-file-handler-context-impl.ts +33 -33
- package/src/main/node/plugin-deployer-impl.ts +375 -375
- package/src/main/node/plugin-deployer-proxy-entry-impl.ts +96 -96
- package/src/main/node/plugin-deployer-resolver-context-impl.ts +55 -55
- package/src/main/node/plugin-ext-backend-module.ts +106 -106
- package/src/main/node/plugin-github-resolver.ts +139 -139
- package/src/main/node/plugin-http-resolver.ts +92 -92
- package/src/main/node/plugin-localization-server.ts +42 -42
- package/src/main/node/plugin-mgmt-cli-contribution.ts +64 -64
- package/src/main/node/plugin-remote-cli-contribution.ts +36 -36
- package/src/main/node/plugin-remote-copy-contribution.ts +36 -36
- package/src/main/node/plugin-server-handler.ts +69 -69
- package/src/main/node/plugin-service.ts +97 -97
- package/src/main/node/plugin-theia-deployer-participant.ts +32 -32
- package/src/main/node/plugin-uninstallation-manager.ts +74 -74
- package/src/main/node/plugins-key-value-storage.spec.ts +110 -110
- package/src/main/node/plugins-key-value-storage.ts +161 -161
- package/src/main/node/resolvers/local-directory-plugin-deployer-resolver.ts +37 -37
- package/src/main/node/resolvers/local-plugin-deployer-resolver.ts +56 -56
- package/src/main/node/temp-dir-util.ts +36 -36
- package/src/main/node/webview-backend-security-warnings.ts +45 -45
- package/src/main/style/status-bar.css +35 -35
- package/src/plugin/authentication-ext.ts +140 -136
- package/src/plugin/clipboard-ext.ts +43 -43
- package/src/plugin/command-registry.ts +219 -219
- package/src/plugin/comments.ts +549 -549
- package/src/plugin/custom-editors.ts +334 -336
- package/src/plugin/debug/debug-ext.ts +549 -549
- package/src/plugin/debug/plugin-debug-adapter-creator.ts +50 -50
- package/src/plugin/debug/plugin-debug-adapter-session.ts +106 -106
- package/src/plugin/debug/plugin-debug-adapter-tracker.ts +85 -85
- package/src/plugin/decorations.ts +140 -140
- package/src/plugin/dialogs.ts +96 -96
- package/src/plugin/document-data.ts +366 -366
- package/src/plugin/documents.ts +283 -283
- package/src/plugin/editors-and-documents.ts +176 -176
- package/src/plugin/env.ts +134 -129
- package/src/plugin/file-system-event-service-ext-impl.ts +256 -256
- package/src/plugin/file-system-ext-impl.ts +415 -415
- package/src/plugin/known-commands.spec.ts +50 -50
- package/src/plugin/known-commands.ts +429 -428
- package/src/plugin/label-service.ts +36 -36
- package/src/plugin/languages/call-hierarchy.ts +124 -124
- package/src/plugin/languages/code-action.ts +162 -162
- package/src/plugin/languages/color.ts +75 -75
- package/src/plugin/languages/completion.ts +183 -183
- package/src/plugin/languages/declaration.ts +72 -72
- package/src/plugin/languages/definition.ts +73 -73
- package/src/plugin/languages/diagnostics.ts +325 -325
- package/src/plugin/languages/document-drop-edit.ts +44 -44
- package/src/plugin/languages/document-formatting.ts +47 -47
- package/src/plugin/languages/document-highlight.ts +61 -61
- package/src/plugin/languages/evaluatable-expression.ts +47 -47
- package/src/plugin/languages/folding.ts +46 -46
- package/src/plugin/languages/hover.ts +58 -58
- package/src/plugin/languages/implementation.ts +73 -73
- package/src/plugin/languages/inlay-hints.ts +149 -149
- package/src/plugin/languages/inline-completion.ts +126 -126
- package/src/plugin/languages/inline-values.ts +50 -50
- package/src/plugin/languages/lens.ts +102 -102
- package/src/plugin/languages/link-provider.ts +81 -81
- package/src/plugin/languages/linked-editing-range.ts +48 -48
- package/src/plugin/languages/on-type-formatting.ts +50 -50
- package/src/plugin/languages/outline.ts +126 -126
- package/src/plugin/languages/range-formatting.ts +48 -48
- package/src/plugin/languages/reference.ts +58 -58
- package/src/plugin/languages/rename.ts +130 -130
- package/src/plugin/languages/selection-range.ts +80 -80
- package/src/plugin/languages/semantic-highlighting.ts +211 -211
- package/src/plugin/languages/signature.ts +82 -82
- package/src/plugin/languages/type-definition.ts +73 -73
- package/src/plugin/languages/type-hierarchy.ts +117 -117
- package/src/plugin/languages/util.ts +26 -26
- package/src/plugin/languages/workspace-symbol.ts +66 -66
- package/src/plugin/languages-utils.ts +68 -68
- package/src/plugin/languages.ts +1022 -1022
- package/src/plugin/localization-ext.ts +89 -89
- package/src/plugin/markdown-string.ts +115 -115
- package/src/plugin/message-registry.ts +70 -70
- package/src/plugin/node/debug/debug.spec.ts +98 -98
- package/src/plugin/node/debug/plugin-node-debug-adapter-creator.ts +167 -167
- package/src/plugin/node/env-node-ext.ts +64 -64
- package/src/plugin/node/plugin-container-module.ts +165 -165
- package/src/plugin/notebook/notebook-document.ts +446 -446
- package/src/plugin/notebook/notebook-documents.ts +58 -58
- package/src/plugin/notebook/notebook-editor.ts +116 -116
- package/src/plugin/notebook/notebook-editors.ts +71 -71
- package/src/plugin/notebook/notebook-kernels.ts +631 -631
- package/src/plugin/notebook/notebook-renderers.ts +71 -71
- package/src/plugin/notebook/notebooks.ts +449 -449
- package/src/plugin/notification.ts +80 -80
- package/src/plugin/output-channel/log-output-channel.ts +108 -108
- package/src/plugin/output-channel/output-channel-item.ts +73 -73
- package/src/plugin/output-channel-registry.ts +52 -52
- package/src/plugin/path.spec.ts +40 -40
- package/src/plugin/path.ts +68 -68
- package/src/plugin/plugin-context.ts +1606 -1595
- package/src/plugin/plugin-icon-path.ts +53 -53
- package/src/plugin/plugin-manager.ts +508 -507
- package/src/plugin/plugin-storage.ts +138 -138
- package/src/plugin/preference-registry.spec.ts +288 -288
- package/src/plugin/preference-registry.ts +335 -335
- package/src/plugin/prefix-sum-computer.ts +218 -218
- package/src/plugin/quick-open.ts +735 -735
- package/src/plugin/scm.ts +919 -919
- package/src/plugin/secrets-ext.ts +104 -104
- package/src/plugin/status-bar/status-bar-item.ts +193 -193
- package/src/plugin/status-bar-message-registry.ts +103 -103
- package/src/plugin/tabs.ts +431 -431
- package/src/plugin/tasks/task-provider.ts +57 -57
- package/src/plugin/tasks/tasks.ts +252 -252
- package/src/plugin/telemetry-ext.ts +298 -298
- package/src/plugin/terminal-ext.ts +569 -566
- package/src/plugin/test-item.ts +174 -174
- package/src/plugin/tests.ts +545 -545
- package/src/plugin/text-editor.ts +581 -581
- package/src/plugin/text-editors.ts +157 -157
- package/src/plugin/theming.ts +73 -73
- package/src/plugin/timeline.ts +186 -186
- package/src/plugin/tree/tree-views.ts +682 -682
- package/src/plugin/type-converters.spec.ts +476 -476
- package/src/plugin/type-converters.ts +1768 -1748
- package/src/plugin/types-impl.spec.ts +85 -85
- package/src/plugin/types-impl.ts +4011 -4001
- package/src/plugin/uri-ext.ts +60 -0
- package/src/plugin/webview-views.ts +228 -228
- package/src/plugin/webviews.ts +468 -468
- package/src/plugin/window-state.ts +75 -75
- package/src/plugin/word-helper.ts +162 -162
- package/src/plugin/workspace.ts +505 -505
- package/src/plugin-ext-backend-electron-module.ts +24 -24
- package/src/plugin-ext-backend-module.ts +24 -24
- package/src/plugin-ext-frontend-electron-module.ts +19 -19
- package/src/plugin-ext-frontend-module.ts +19 -19
- package/lib/common/proxy-handler.d.ts +0 -73
- package/lib/common/proxy-handler.d.ts.map +0 -1
- package/lib/common/proxy-handler.js +0 -106
- package/lib/common/proxy-handler.js.map +0 -1
- package/src/common/proxy-handler.ts +0 -143
|
@@ -1,625 +1,625 @@
|
|
|
1
|
-
// *****************************************************************************
|
|
2
|
-
// Copyright (C) 2019 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
|
-
* Copyright (c) Microsoft Corporation. All rights reserved.
|
|
18
|
-
* Licensed under the MIT License. See License.txt in the project root for license information.
|
|
19
|
-
*--------------------------------------------------------------------------------------------*/
|
|
20
|
-
// some code is copied and modified from:
|
|
21
|
-
// https://github.com/microsoft/vscode/blob/7cf4cca47aa025a590fc939af54932042302be63/src/vs/workbench/services/themes/browser/fileIconThemeData.ts
|
|
22
|
-
|
|
23
|
-
import debounce = require('@theia/core/shared/lodash.debounce');
|
|
24
|
-
import * as jsoncparser from 'jsonc-parser';
|
|
25
|
-
import { injectable, inject, postConstruct } from '@theia/core/shared/inversify';
|
|
26
|
-
import { IconThemeService, IconTheme, IconThemeDefinition } from '@theia/core/lib/browser/icon-theme-service';
|
|
27
|
-
import { IconThemeContribution, DeployedPlugin, UiTheme, getPluginId } from '../../common/plugin-protocol';
|
|
28
|
-
import URI from '@theia/core/lib/common/uri';
|
|
29
|
-
import { Disposable, DisposableCollection } from '@theia/core/lib/common/disposable';
|
|
30
|
-
import { Emitter } from '@theia/core/lib/common/event';
|
|
31
|
-
import { RecursivePartial } from '@theia/core/lib/common/types';
|
|
32
|
-
import { LabelProviderContribution, DidChangeLabelEvent, LabelProvider, URIIconReference } from '@theia/core/lib/browser/label-provider';
|
|
33
|
-
import { ThemeType } from '@theia/core/lib/common/theme';
|
|
34
|
-
import { FileStatNode, DirNode } from '@theia/filesystem/lib/browser';
|
|
35
|
-
import { WorkspaceRootNode } from '@theia/navigator/lib/browser/navigator-tree';
|
|
36
|
-
import { Endpoint } from '@theia/core/lib/browser/endpoint';
|
|
37
|
-
import { FileService } from '@theia/filesystem/lib/browser/file-service';
|
|
38
|
-
import { FileStat, FileChangeType } from '@theia/filesystem/lib/common/files';
|
|
39
|
-
import { WorkspaceService } from '@theia/workspace/lib/browser';
|
|
40
|
-
import { StandaloneServices } from '@theia/monaco-editor-core/esm/vs/editor/standalone/browser/standaloneServices';
|
|
41
|
-
import { ILanguageService } from '@theia/monaco-editor-core/esm/vs/editor/common/languages/language';
|
|
42
|
-
import { LanguageService } from '@theia/core/lib/browser/language-service';
|
|
43
|
-
import { DEFAULT_ICON_SIZE, PLUGIN_FILE_ICON_CLASS } from './plugin-shared-style';
|
|
44
|
-
|
|
45
|
-
export interface PluginIconDefinition {
|
|
46
|
-
iconPath: string;
|
|
47
|
-
fontColor: string;
|
|
48
|
-
fontCharacter: string;
|
|
49
|
-
fontSize: string;
|
|
50
|
-
fontId: string;
|
|
51
|
-
}
|
|
52
|
-
|
|
53
|
-
export interface PluginFontDefinition {
|
|
54
|
-
id: string;
|
|
55
|
-
weight: string;
|
|
56
|
-
style: string;
|
|
57
|
-
size: string;
|
|
58
|
-
src: { path: string; format: string; }[];
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
export interface PluginIconsAssociation {
|
|
62
|
-
folder?: string;
|
|
63
|
-
file?: string;
|
|
64
|
-
folderExpanded?: string;
|
|
65
|
-
rootFolder?: string;
|
|
66
|
-
rootFolderExpanded?: string;
|
|
67
|
-
folderNames?: { [folderName: string]: string; };
|
|
68
|
-
folderNamesExpanded?: { [folderName: string]: string; };
|
|
69
|
-
fileExtensions?: { [extension: string]: string; };
|
|
70
|
-
fileNames?: { [fileName: string]: string; };
|
|
71
|
-
languageIds?: { [languageId: string]: string; };
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
export interface PluginIconDefinitions {
|
|
75
|
-
[key: string]: PluginIconDefinition
|
|
76
|
-
}
|
|
77
|
-
|
|
78
|
-
export interface PluginIconThemeDocument extends PluginIconsAssociation {
|
|
79
|
-
iconDefinitions: PluginIconDefinitions;
|
|
80
|
-
fonts: PluginFontDefinition[];
|
|
81
|
-
light?: PluginIconsAssociation;
|
|
82
|
-
highContrast?: PluginIconsAssociation;
|
|
83
|
-
hidesExplorerArrows?: boolean;
|
|
84
|
-
showLanguageModeIcons?: boolean;
|
|
85
|
-
}
|
|
86
|
-
|
|
87
|
-
export const PluginIconThemeFactory = Symbol('PluginIconThemeFactory');
|
|
88
|
-
export type PluginIconThemeFactory = (definition: PluginIconThemeDefinition) => PluginIconTheme;
|
|
89
|
-
|
|
90
|
-
@injectable()
|
|
91
|
-
export class PluginIconThemeDefinition implements IconThemeDefinition, IconThemeContribution {
|
|
92
|
-
id: string;
|
|
93
|
-
label: string;
|
|
94
|
-
description?: string;
|
|
95
|
-
uri: string;
|
|
96
|
-
uiTheme?: UiTheme;
|
|
97
|
-
pluginId: string;
|
|
98
|
-
packageUri: string;
|
|
99
|
-
hasFileIcons?: boolean;
|
|
100
|
-
hasFolderIcons?: boolean;
|
|
101
|
-
hidesExplorerArrows?: boolean;
|
|
102
|
-
showLanguageModeIcons?: boolean;
|
|
103
|
-
}
|
|
104
|
-
|
|
105
|
-
class PluginLanguageIconInfo {
|
|
106
|
-
hasSpecificFileIcons: boolean = false;
|
|
107
|
-
coveredLanguages: { [languageId: string]: boolean } = {};
|
|
108
|
-
};
|
|
109
|
-
|
|
110
|
-
@injectable()
|
|
111
|
-
export class PluginIconTheme extends PluginIconThemeDefinition implements IconTheme, Disposable {
|
|
112
|
-
|
|
113
|
-
@inject(FileService)
|
|
114
|
-
protected readonly fileService: FileService;
|
|
115
|
-
|
|
116
|
-
@inject(LabelProvider)
|
|
117
|
-
protected readonly labelProvider: LabelProvider;
|
|
118
|
-
|
|
119
|
-
@inject(PluginIconThemeDefinition)
|
|
120
|
-
protected readonly definition: PluginIconThemeDefinition;
|
|
121
|
-
|
|
122
|
-
@inject(WorkspaceService)
|
|
123
|
-
protected readonly workspaceService: WorkspaceService;
|
|
124
|
-
|
|
125
|
-
@inject(LanguageService)
|
|
126
|
-
protected readonly languageService: LanguageService;
|
|
127
|
-
|
|
128
|
-
protected readonly onDidChangeEmitter = new Emitter<DidChangeLabelEvent>();
|
|
129
|
-
readonly onDidChange = this.onDidChangeEmitter.event;
|
|
130
|
-
|
|
131
|
-
protected readonly toDeactivate = new DisposableCollection();
|
|
132
|
-
protected readonly toUnload = new DisposableCollection();
|
|
133
|
-
protected readonly toDisposeStyleElement = new DisposableCollection();
|
|
134
|
-
protected readonly toDispose = new DisposableCollection(
|
|
135
|
-
this.toDeactivate, this.toDisposeStyleElement, this.toUnload, this.onDidChangeEmitter
|
|
136
|
-
);
|
|
137
|
-
|
|
138
|
-
protected packageRootUri: URI;
|
|
139
|
-
protected locationUri: URI;
|
|
140
|
-
|
|
141
|
-
protected styleSheetContent: string | undefined;
|
|
142
|
-
protected readonly icons = new Set<string>();
|
|
143
|
-
|
|
144
|
-
@postConstruct()
|
|
145
|
-
protected init(): void {
|
|
146
|
-
Object.assign(this, this.definition);
|
|
147
|
-
this.packageRootUri = new URI(this.packageUri);
|
|
148
|
-
this.locationUri = new URI(this.uri).parent;
|
|
149
|
-
}
|
|
150
|
-
|
|
151
|
-
dispose(): void {
|
|
152
|
-
this.toDispose.dispose();
|
|
153
|
-
}
|
|
154
|
-
|
|
155
|
-
protected fireDidChange(): void {
|
|
156
|
-
this.onDidChangeEmitter.fire({ affects: () => true });
|
|
157
|
-
}
|
|
158
|
-
|
|
159
|
-
activate(): Disposable {
|
|
160
|
-
if (!this.toDeactivate.disposed) {
|
|
161
|
-
return this.toDeactivate;
|
|
162
|
-
}
|
|
163
|
-
this.toDeactivate.push(Disposable.create(() => this.fireDidChange()));
|
|
164
|
-
this.doActivate();
|
|
165
|
-
return this.toDeactivate;
|
|
166
|
-
}
|
|
167
|
-
|
|
168
|
-
protected async doActivate(): Promise<void> {
|
|
169
|
-
await this.load();
|
|
170
|
-
this.updateStyleElement();
|
|
171
|
-
}
|
|
172
|
-
|
|
173
|
-
protected updateStyleElement(): void {
|
|
174
|
-
this.toDisposeStyleElement.dispose();
|
|
175
|
-
if (this.toDeactivate.disposed || !this.styleSheetContent) {
|
|
176
|
-
return;
|
|
177
|
-
}
|
|
178
|
-
const styleElement = document.createElement('style');
|
|
179
|
-
styleElement.type = 'text/css';
|
|
180
|
-
styleElement.className = 'theia-icon-theme';
|
|
181
|
-
styleElement.innerText = this.styleSheetContent;
|
|
182
|
-
document.head.appendChild(styleElement);
|
|
183
|
-
const toRemoveStyleElement = Disposable.create(() => styleElement.remove());
|
|
184
|
-
this.toDisposeStyleElement.push(toRemoveStyleElement);
|
|
185
|
-
this.toDeactivate.push(toRemoveStyleElement);
|
|
186
|
-
this.fireDidChange();
|
|
187
|
-
}
|
|
188
|
-
|
|
189
|
-
protected reload = debounce(() => {
|
|
190
|
-
this.toUnload.dispose();
|
|
191
|
-
this.doActivate();
|
|
192
|
-
}, 50);
|
|
193
|
-
|
|
194
|
-
/**
|
|
195
|
-
* This should be aligned with
|
|
196
|
-
* https://github.com/microsoft/vscode/blob/7cf4cca47aa025a590fc939af54932042302be63/src/vs/workbench/services/themes/browser/fileIconThemeData.ts#L201
|
|
197
|
-
*/
|
|
198
|
-
protected async load(): Promise<void> {
|
|
199
|
-
if (this.styleSheetContent !== undefined) {
|
|
200
|
-
return;
|
|
201
|
-
}
|
|
202
|
-
this.styleSheetContent = '';
|
|
203
|
-
this.toUnload.push(Disposable.create(() => {
|
|
204
|
-
this.styleSheetContent = undefined;
|
|
205
|
-
this.hasFileIcons = undefined;
|
|
206
|
-
this.hasFolderIcons = undefined;
|
|
207
|
-
this.hidesExplorerArrows = undefined;
|
|
208
|
-
this.icons.clear();
|
|
209
|
-
}));
|
|
210
|
-
|
|
211
|
-
const uri = new URI(this.uri);
|
|
212
|
-
const result = await this.fileService.read(uri);
|
|
213
|
-
const content = result.value;
|
|
214
|
-
const json: RecursivePartial<PluginIconThemeDocument> = jsoncparser.parse(content, undefined, { disallowComments: false });
|
|
215
|
-
this.hidesExplorerArrows = !!json.hidesExplorerArrows;
|
|
216
|
-
|
|
217
|
-
const toUnwatch = this.fileService.watch(uri);
|
|
218
|
-
if (this.toUnload.disposed) {
|
|
219
|
-
toUnwatch.dispose();
|
|
220
|
-
} else {
|
|
221
|
-
this.toUnload.push(toUnwatch);
|
|
222
|
-
this.toUnload.push(this.fileService.onDidFilesChange(e => {
|
|
223
|
-
if (e.contains(uri, FileChangeType.ADDED) || e.contains(uri, FileChangeType.UPDATED)) {
|
|
224
|
-
this.reload();
|
|
225
|
-
}
|
|
226
|
-
}));
|
|
227
|
-
}
|
|
228
|
-
|
|
229
|
-
const iconDefinitions = json.iconDefinitions;
|
|
230
|
-
if (!iconDefinitions) {
|
|
231
|
-
return;
|
|
232
|
-
}
|
|
233
|
-
const definitionSelectors = new Map<string, string[]>();
|
|
234
|
-
const acceptSelector = (themeType: ThemeType, definitionId: string, ...icons: string[]) => {
|
|
235
|
-
if (!iconDefinitions[definitionId]) {
|
|
236
|
-
return;
|
|
237
|
-
}
|
|
238
|
-
let selector = '';
|
|
239
|
-
for (const icon of icons) {
|
|
240
|
-
if (icon) {
|
|
241
|
-
selector += '.' + icon;
|
|
242
|
-
this.icons.add(icon);
|
|
243
|
-
}
|
|
244
|
-
}
|
|
245
|
-
if (!selector) {
|
|
246
|
-
return;
|
|
247
|
-
}
|
|
248
|
-
const selectors = definitionSelectors.get(definitionId) || [];
|
|
249
|
-
if (themeType !== 'dark') {
|
|
250
|
-
selector = '.theia-' + themeType + ' ' + selector;
|
|
251
|
-
}
|
|
252
|
-
selectors.push(selector + '::before');
|
|
253
|
-
definitionSelectors.set(definitionId, selectors);
|
|
254
|
-
};
|
|
255
|
-
|
|
256
|
-
let iconInfo = this.collectSelectors(json, acceptSelector.bind(undefined, 'dark'));
|
|
257
|
-
if (json.light) {
|
|
258
|
-
iconInfo = this.collectSelectors(json.light, acceptSelector.bind(undefined, 'light'));
|
|
259
|
-
}
|
|
260
|
-
if (json.highContrast) {
|
|
261
|
-
iconInfo = this.collectSelectors(json.highContrast, acceptSelector.bind(undefined, 'hc'));
|
|
262
|
-
}
|
|
263
|
-
|
|
264
|
-
const showLanguageModeIcons = this.showLanguageModeIcons === true
|
|
265
|
-
|| json.showLanguageModeIcons === true
|
|
266
|
-
|| (iconInfo.hasSpecificFileIcons && json.showLanguageModeIcons !== false);
|
|
267
|
-
|
|
268
|
-
const fonts = json.fonts;
|
|
269
|
-
if (Array.isArray(fonts)) {
|
|
270
|
-
for (const font of fonts) {
|
|
271
|
-
if (font) {
|
|
272
|
-
let src = '';
|
|
273
|
-
if (Array.isArray(font.src)) {
|
|
274
|
-
for (const srcLocation of font.src) {
|
|
275
|
-
if (srcLocation && srcLocation.path) {
|
|
276
|
-
const cssUrl = this.toCSSUrl(srcLocation.path);
|
|
277
|
-
if (cssUrl) {
|
|
278
|
-
if (src) {
|
|
279
|
-
src += ', ';
|
|
280
|
-
}
|
|
281
|
-
src += `${cssUrl} format('${srcLocation.format}')`;
|
|
282
|
-
}
|
|
283
|
-
}
|
|
284
|
-
}
|
|
285
|
-
}
|
|
286
|
-
if (src) {
|
|
287
|
-
this.styleSheetContent += `@font-face {
|
|
288
|
-
src: ${src};
|
|
289
|
-
font-family: '${font.id}';
|
|
290
|
-
font-weight: ${font.weight};
|
|
291
|
-
font-style: ${font.style};
|
|
292
|
-
}
|
|
293
|
-
`;
|
|
294
|
-
}
|
|
295
|
-
}
|
|
296
|
-
}
|
|
297
|
-
const firstFont = fonts[0];
|
|
298
|
-
if (firstFont && firstFont.id) {
|
|
299
|
-
this.styleSheetContent += `.${this.fileIcon}::before, .${this.folderIcon}::before, .${this.rootFolderIcon}::before {
|
|
300
|
-
font-family: '${firstFont.id}';
|
|
301
|
-
font-size: ${firstFont.size || '150%'};
|
|
302
|
-
-webkit-font-smoothing: antialiased;
|
|
303
|
-
-moz-osx-font-smoothing: grayscale;
|
|
304
|
-
vertical-align: top;
|
|
305
|
-
}
|
|
306
|
-
`;
|
|
307
|
-
}
|
|
308
|
-
}
|
|
309
|
-
|
|
310
|
-
for (const definitionId of definitionSelectors.keys()) {
|
|
311
|
-
const iconDefinition = iconDefinitions[definitionId];
|
|
312
|
-
const selectors = definitionSelectors.get(definitionId);
|
|
313
|
-
if (selectors && iconDefinition) {
|
|
314
|
-
const cssUrl = this.toCSSUrl(iconDefinition.iconPath);
|
|
315
|
-
if (cssUrl) {
|
|
316
|
-
this.styleSheetContent += `${selectors.join(', ')} {
|
|
317
|
-
content: ' ';
|
|
318
|
-
background-image: ${cssUrl};
|
|
319
|
-
background-size: ${DEFAULT_ICON_SIZE}px;
|
|
320
|
-
background-position: left center;
|
|
321
|
-
background-repeat: no-repeat;
|
|
322
|
-
}
|
|
323
|
-
`;
|
|
324
|
-
}
|
|
325
|
-
if (iconDefinition.fontCharacter || iconDefinition.fontColor) {
|
|
326
|
-
let body = '';
|
|
327
|
-
if (iconDefinition.fontColor) {
|
|
328
|
-
body += ` color: ${iconDefinition.fontColor};`;
|
|
329
|
-
}
|
|
330
|
-
if (iconDefinition.fontCharacter) {
|
|
331
|
-
body += ` content: '${iconDefinition.fontCharacter}';`;
|
|
332
|
-
}
|
|
333
|
-
if (iconDefinition.fontSize) {
|
|
334
|
-
body += ` font-size: ${iconDefinition.fontSize};`;
|
|
335
|
-
}
|
|
336
|
-
if (iconDefinition.fontId) {
|
|
337
|
-
body += ` font-family: ${iconDefinition.fontId};`;
|
|
338
|
-
}
|
|
339
|
-
this.styleSheetContent += `${selectors.join(', ')} {${body} }\n`;
|
|
340
|
-
}
|
|
341
|
-
}
|
|
342
|
-
}
|
|
343
|
-
|
|
344
|
-
if (showLanguageModeIcons) {
|
|
345
|
-
for (const language of this.languageService.languages) {
|
|
346
|
-
// only show language icons if there are no more specific icons in the style document
|
|
347
|
-
if (!iconInfo.coveredLanguages[language.id]) {
|
|
348
|
-
const icon = this.languageService.getIcon(language.id);
|
|
349
|
-
if (icon) {
|
|
350
|
-
this.icons.add(this.fileIcon);
|
|
351
|
-
this.icons.add(this.languageIcon(language.id));
|
|
352
|
-
this.icons.add(icon);
|
|
353
|
-
}
|
|
354
|
-
}
|
|
355
|
-
}
|
|
356
|
-
}
|
|
357
|
-
}
|
|
358
|
-
|
|
359
|
-
protected toCSSUrl(iconPath: string | undefined): string | undefined {
|
|
360
|
-
if (!iconPath) {
|
|
361
|
-
return undefined;
|
|
362
|
-
}
|
|
363
|
-
const iconUri = this.locationUri.resolve(iconPath);
|
|
364
|
-
const relativePath = this.packageRootUri.path.relative(iconUri.path.normalize());
|
|
365
|
-
return relativePath && `url('${new Endpoint({
|
|
366
|
-
path: `hostedPlugin/${this.pluginId}/${encodeURIComponent(relativePath.normalize().toString())}`
|
|
367
|
-
}).getRestUrl().toString()}')`;
|
|
368
|
-
}
|
|
369
|
-
|
|
370
|
-
protected escapeCSS(value: string): string {
|
|
371
|
-
value = value.replace(/[^\-a-zA-Z0-9]/g, '-');
|
|
372
|
-
if (value.charAt(0).match(/[0-9\-]/)) {
|
|
373
|
-
value = '-' + value;
|
|
374
|
-
}
|
|
375
|
-
return value;
|
|
376
|
-
}
|
|
377
|
-
|
|
378
|
-
protected readonly fileIcon = PLUGIN_FILE_ICON_CLASS;
|
|
379
|
-
protected readonly folderIcon = 'theia-plugin-folder-icon';
|
|
380
|
-
protected readonly folderExpandedIcon = 'theia-plugin-folder-expanded-icon';
|
|
381
|
-
protected readonly rootFolderIcon = 'theia-plugin-root-folder-icon';
|
|
382
|
-
protected readonly rootFolderExpandedIcon = 'theia-plugin-root-folder-expanded-icon';
|
|
383
|
-
protected folderNameIcon(folderName: string): string {
|
|
384
|
-
return 'theia-plugin-' + this.escapeCSS(folderName.toLowerCase()) + '-folder-name-icon';
|
|
385
|
-
}
|
|
386
|
-
protected expandedFolderNameIcon(folderName: string): string {
|
|
387
|
-
return 'theia-plugin-' + this.escapeCSS(folderName.toLowerCase()) + '-expanded-folder-name-icon';
|
|
388
|
-
}
|
|
389
|
-
protected fileNameIcon(fileName: string): string[] {
|
|
390
|
-
fileName = fileName.toLowerCase();
|
|
391
|
-
const extIndex = fileName.indexOf('.');
|
|
392
|
-
const icons = extIndex !== -1 ? this.fileExtensionIcon(fileName.substring(extIndex + 1)) : [];
|
|
393
|
-
icons.unshift('theia-plugin-' + this.escapeCSS(fileName) + '-file-name-icon');
|
|
394
|
-
return icons;
|
|
395
|
-
}
|
|
396
|
-
protected fileExtensionIcon(fileExtension: string): string[] {
|
|
397
|
-
fileExtension = fileExtension.toString();
|
|
398
|
-
const icons = [];
|
|
399
|
-
const segments = fileExtension.split('.');
|
|
400
|
-
if (segments.length) {
|
|
401
|
-
if (segments.length) {
|
|
402
|
-
for (let i = 0; i < segments.length; i++) {
|
|
403
|
-
icons.push('theia-plugin-' + this.escapeCSS(segments.slice(i).join('.')) + '-ext-file-icon');
|
|
404
|
-
}
|
|
405
|
-
icons.push('theia-plugin-ext-file-icon'); // extra segment to increase file-ext score
|
|
406
|
-
}
|
|
407
|
-
}
|
|
408
|
-
return icons;
|
|
409
|
-
}
|
|
410
|
-
protected languageIcon(languageId: string): string {
|
|
411
|
-
return 'theia-plugin-' + this.escapeCSS(languageId) + '-lang-file-icon';
|
|
412
|
-
}
|
|
413
|
-
|
|
414
|
-
protected collectSelectors(associations: RecursivePartial<PluginIconsAssociation>, accept: (definitionId: string, ...icons: string[]) => void): PluginLanguageIconInfo {
|
|
415
|
-
const iconInfo = new PluginLanguageIconInfo();
|
|
416
|
-
if (associations.folder) {
|
|
417
|
-
accept(associations.folder, this.folderIcon);
|
|
418
|
-
if (associations.folderExpanded === undefined) {
|
|
419
|
-
// Use the same icon for expanded state (issue #12727). Check for
|
|
420
|
-
// undefined folderExpanded property to allow for
|
|
421
|
-
// "folderExpanded": null in case a developer really wants that
|
|
422
|
-
accept(associations.folder, this.folderExpandedIcon);
|
|
423
|
-
}
|
|
424
|
-
this.hasFolderIcons = true;
|
|
425
|
-
}
|
|
426
|
-
if (associations.folderExpanded) {
|
|
427
|
-
accept(associations.folderExpanded, this.folderExpandedIcon);
|
|
428
|
-
this.hasFolderIcons = true;
|
|
429
|
-
}
|
|
430
|
-
const rootFolder = associations.rootFolder || associations.folder;
|
|
431
|
-
if (rootFolder) {
|
|
432
|
-
accept(rootFolder, this.rootFolderIcon);
|
|
433
|
-
this.hasFolderIcons = true;
|
|
434
|
-
}
|
|
435
|
-
const rootFolderExpanded = associations.rootFolderExpanded || associations.folderExpanded;
|
|
436
|
-
if (rootFolderExpanded) {
|
|
437
|
-
accept(rootFolderExpanded, this.rootFolderExpandedIcon);
|
|
438
|
-
this.hasFolderIcons = true;
|
|
439
|
-
}
|
|
440
|
-
if (associations.file) {
|
|
441
|
-
accept(associations.file, this.fileIcon);
|
|
442
|
-
this.hasFileIcons = true;
|
|
443
|
-
}
|
|
444
|
-
const folderNames = associations.folderNames;
|
|
445
|
-
if (folderNames) {
|
|
446
|
-
// eslint-disable-next-line guard-for-in
|
|
447
|
-
for (const folderName in folderNames) {
|
|
448
|
-
accept(folderNames[folderName]!, this.folderNameIcon(folderName), this.folderIcon);
|
|
449
|
-
this.hasFolderIcons = true;
|
|
450
|
-
}
|
|
451
|
-
}
|
|
452
|
-
const folderNamesExpanded = associations.folderNamesExpanded;
|
|
453
|
-
if (folderNamesExpanded) {
|
|
454
|
-
// eslint-disable-next-line guard-for-in
|
|
455
|
-
for (const folderName in folderNamesExpanded) {
|
|
456
|
-
accept(folderNamesExpanded[folderName]!, this.expandedFolderNameIcon(folderName), this.folderExpandedIcon);
|
|
457
|
-
this.hasFolderIcons = true;
|
|
458
|
-
}
|
|
459
|
-
}
|
|
460
|
-
const languageIds = associations.languageIds;
|
|
461
|
-
if (languageIds) {
|
|
462
|
-
if (!languageIds.jsonc && languageIds.json) {
|
|
463
|
-
languageIds.jsonc = languageIds.json;
|
|
464
|
-
}
|
|
465
|
-
// eslint-disable-next-line guard-for-in
|
|
466
|
-
for (const languageId in languageIds) {
|
|
467
|
-
accept(languageIds[languageId]!, this.languageIcon(languageId), this.fileIcon);
|
|
468
|
-
this.hasFileIcons = true;
|
|
469
|
-
iconInfo.hasSpecificFileIcons = true;
|
|
470
|
-
iconInfo.coveredLanguages[languageId] = true;
|
|
471
|
-
}
|
|
472
|
-
}
|
|
473
|
-
const fileExtensions = associations.fileExtensions;
|
|
474
|
-
if (fileExtensions) {
|
|
475
|
-
// eslint-disable-next-line guard-for-in
|
|
476
|
-
for (const fileExtension in fileExtensions) {
|
|
477
|
-
accept(fileExtensions[fileExtension]!, ...this.fileExtensionIcon(fileExtension), this.fileIcon);
|
|
478
|
-
this.hasFileIcons = true;
|
|
479
|
-
iconInfo.hasSpecificFileIcons = true;
|
|
480
|
-
}
|
|
481
|
-
}
|
|
482
|
-
const fileNames = associations.fileNames;
|
|
483
|
-
if (fileNames) {
|
|
484
|
-
// eslint-disable-next-line guard-for-in
|
|
485
|
-
for (const fileName in fileNames) {
|
|
486
|
-
accept(fileNames[fileName]!, ...this.fileNameIcon(fileName), this.fileIcon);
|
|
487
|
-
this.hasFileIcons = true;
|
|
488
|
-
iconInfo.hasSpecificFileIcons = true;
|
|
489
|
-
}
|
|
490
|
-
}
|
|
491
|
-
return iconInfo;
|
|
492
|
-
}
|
|
493
|
-
|
|
494
|
-
/**
|
|
495
|
-
* This should be aligned with
|
|
496
|
-
* https://github.com/microsoft/vscode/blob/7cf4cca47aa025a590fc939af54932042302be63/src/vs/editor/common/services/getIconClasses.ts#L5
|
|
497
|
-
*/
|
|
498
|
-
getIcon(element: URI | URIIconReference | FileStat | FileStatNode | WorkspaceRootNode): string {
|
|
499
|
-
let icon = '';
|
|
500
|
-
for (const className of this.getClassNames(element)) {
|
|
501
|
-
if (this.icons.has(className)) {
|
|
502
|
-
if (icon) {
|
|
503
|
-
icon += ' ';
|
|
504
|
-
}
|
|
505
|
-
icon += className;
|
|
506
|
-
}
|
|
507
|
-
}
|
|
508
|
-
return icon;
|
|
509
|
-
}
|
|
510
|
-
|
|
511
|
-
protected getClassNames(element: URI | URIIconReference | FileStat | FileStatNode | WorkspaceRootNode): string[] {
|
|
512
|
-
if (WorkspaceRootNode.is(element)) {
|
|
513
|
-
const name = this.labelProvider.getName(element);
|
|
514
|
-
if (element.expanded) {
|
|
515
|
-
return [this.rootFolderExpandedIcon, this.expandedFolderNameIcon(name)];
|
|
516
|
-
}
|
|
517
|
-
return [this.rootFolderIcon, this.folderNameIcon(name)];
|
|
518
|
-
}
|
|
519
|
-
if (DirNode.is(element)) {
|
|
520
|
-
if (element.expanded) {
|
|
521
|
-
const name = this.labelProvider.getName(element);
|
|
522
|
-
return [this.folderExpandedIcon, this.expandedFolderNameIcon(name)];
|
|
523
|
-
}
|
|
524
|
-
return this.getFolderClassNames(element);
|
|
525
|
-
}
|
|
526
|
-
if (FileStatNode.is(element)) {
|
|
527
|
-
return this.getFileClassNames(element, element.fileStat.resource.toString());
|
|
528
|
-
}
|
|
529
|
-
if (FileStat.is(element)) {
|
|
530
|
-
if (element.isDirectory) {
|
|
531
|
-
return this.getFolderClassNames(element);
|
|
532
|
-
}
|
|
533
|
-
return this.getFileClassNames(element, element.resource.toString());
|
|
534
|
-
}
|
|
535
|
-
if (URIIconReference.is(element)) {
|
|
536
|
-
if (element.id === 'folder') {
|
|
537
|
-
return this.getFolderClassNames(element);
|
|
538
|
-
}
|
|
539
|
-
return this.getFileClassNames(element, element.uri && element.uri.toString());
|
|
540
|
-
}
|
|
541
|
-
return this.getFileClassNames(element, element.toString());
|
|
542
|
-
}
|
|
543
|
-
|
|
544
|
-
protected getFolderClassNames(element: object): string[] {
|
|
545
|
-
const name = this.labelProvider.getName(element);
|
|
546
|
-
return [this.folderIcon, this.folderNameIcon(name)];
|
|
547
|
-
}
|
|
548
|
-
|
|
549
|
-
protected getFileClassNames(element: object, uri?: string): string[] {
|
|
550
|
-
const name = this.labelProvider.getName(element);
|
|
551
|
-
const classNames = this.fileNameIcon(name);
|
|
552
|
-
if (uri) {
|
|
553
|
-
const parsedURI = new URI(uri);
|
|
554
|
-
const isRoot = this.workspaceService.getWorkspaceRootUri(new URI(uri))?.isEqual(parsedURI);
|
|
555
|
-
if (isRoot) {
|
|
556
|
-
classNames.unshift(this.rootFolderIcon);
|
|
557
|
-
} else {
|
|
558
|
-
classNames.unshift(this.fileIcon);
|
|
559
|
-
}
|
|
560
|
-
const language = StandaloneServices.get(ILanguageService).createByFilepathOrFirstLine(parsedURI['codeUri']);
|
|
561
|
-
classNames.push(this.languageIcon(language.languageId));
|
|
562
|
-
const defaultLanguageIcon = this.languageService.getIcon(language.languageId);
|
|
563
|
-
if (defaultLanguageIcon) {
|
|
564
|
-
classNames.push(defaultLanguageIcon);
|
|
565
|
-
}
|
|
566
|
-
}
|
|
567
|
-
return classNames;
|
|
568
|
-
}
|
|
569
|
-
|
|
570
|
-
}
|
|
571
|
-
|
|
572
|
-
@injectable()
|
|
573
|
-
export class PluginIconThemeService implements LabelProviderContribution {
|
|
574
|
-
|
|
575
|
-
@inject(IconThemeService)
|
|
576
|
-
protected readonly iconThemeService: IconThemeService;
|
|
577
|
-
|
|
578
|
-
@inject(PluginIconThemeFactory)
|
|
579
|
-
protected readonly iconThemeFactory: PluginIconThemeFactory;
|
|
580
|
-
|
|
581
|
-
protected readonly onDidChangeEmitter = new Emitter<DidChangeLabelEvent>();
|
|
582
|
-
readonly onDidChange = this.onDidChangeEmitter.event;
|
|
583
|
-
|
|
584
|
-
protected fireDidChange(): void {
|
|
585
|
-
this.onDidChangeEmitter.fire({ affects: () => true });
|
|
586
|
-
}
|
|
587
|
-
|
|
588
|
-
register(contribution: IconThemeContribution, plugin: DeployedPlugin): Disposable {
|
|
589
|
-
const pluginId = getPluginId(plugin.metadata.model);
|
|
590
|
-
const packageUri = plugin.metadata.model.packageUri;
|
|
591
|
-
const iconTheme = this.iconThemeFactory({
|
|
592
|
-
id: contribution.id,
|
|
593
|
-
label: contribution.label || new URI(contribution.uri).path.base,
|
|
594
|
-
description: contribution.description,
|
|
595
|
-
uri: contribution.uri,
|
|
596
|
-
uiTheme: contribution.uiTheme,
|
|
597
|
-
pluginId,
|
|
598
|
-
packageUri
|
|
599
|
-
});
|
|
600
|
-
return new DisposableCollection(
|
|
601
|
-
iconTheme,
|
|
602
|
-
iconTheme.onDidChange(() => this.fireDidChange()),
|
|
603
|
-
this.iconThemeService.register(iconTheme)
|
|
604
|
-
);
|
|
605
|
-
}
|
|
606
|
-
|
|
607
|
-
canHandle(element: object): number {
|
|
608
|
-
const current = this.iconThemeService.getDefinition(this.iconThemeService.current);
|
|
609
|
-
if (current instanceof PluginIconTheme && (
|
|
610
|
-
(element instanceof URI && element.scheme === 'file') || URIIconReference.is(element) || FileStat.is(element) || FileStatNode.is(element)
|
|
611
|
-
)) {
|
|
612
|
-
return Number.MAX_SAFE_INTEGER;
|
|
613
|
-
}
|
|
614
|
-
return 0;
|
|
615
|
-
}
|
|
616
|
-
|
|
617
|
-
getIcon(element: URI | URIIconReference | FileStat | FileStatNode | WorkspaceRootNode): string | undefined {
|
|
618
|
-
const current = this.iconThemeService.getDefinition(this.iconThemeService.current);
|
|
619
|
-
if (current instanceof PluginIconTheme) {
|
|
620
|
-
return current.getIcon(element);
|
|
621
|
-
}
|
|
622
|
-
return undefined;
|
|
623
|
-
}
|
|
624
|
-
|
|
625
|
-
}
|
|
1
|
+
// *****************************************************************************
|
|
2
|
+
// Copyright (C) 2019 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
|
+
* Copyright (c) Microsoft Corporation. All rights reserved.
|
|
18
|
+
* Licensed under the MIT License. See License.txt in the project root for license information.
|
|
19
|
+
*--------------------------------------------------------------------------------------------*/
|
|
20
|
+
// some code is copied and modified from:
|
|
21
|
+
// https://github.com/microsoft/vscode/blob/7cf4cca47aa025a590fc939af54932042302be63/src/vs/workbench/services/themes/browser/fileIconThemeData.ts
|
|
22
|
+
|
|
23
|
+
import debounce = require('@theia/core/shared/lodash.debounce');
|
|
24
|
+
import * as jsoncparser from 'jsonc-parser';
|
|
25
|
+
import { injectable, inject, postConstruct } from '@theia/core/shared/inversify';
|
|
26
|
+
import { IconThemeService, IconTheme, IconThemeDefinition } from '@theia/core/lib/browser/icon-theme-service';
|
|
27
|
+
import { IconThemeContribution, DeployedPlugin, UiTheme, getPluginId } from '../../common/plugin-protocol';
|
|
28
|
+
import URI from '@theia/core/lib/common/uri';
|
|
29
|
+
import { Disposable, DisposableCollection } from '@theia/core/lib/common/disposable';
|
|
30
|
+
import { Emitter } from '@theia/core/lib/common/event';
|
|
31
|
+
import { RecursivePartial } from '@theia/core/lib/common/types';
|
|
32
|
+
import { LabelProviderContribution, DidChangeLabelEvent, LabelProvider, URIIconReference } from '@theia/core/lib/browser/label-provider';
|
|
33
|
+
import { ThemeType } from '@theia/core/lib/common/theme';
|
|
34
|
+
import { FileStatNode, DirNode } from '@theia/filesystem/lib/browser';
|
|
35
|
+
import { WorkspaceRootNode } from '@theia/navigator/lib/browser/navigator-tree';
|
|
36
|
+
import { Endpoint } from '@theia/core/lib/browser/endpoint';
|
|
37
|
+
import { FileService } from '@theia/filesystem/lib/browser/file-service';
|
|
38
|
+
import { FileStat, FileChangeType } from '@theia/filesystem/lib/common/files';
|
|
39
|
+
import { WorkspaceService } from '@theia/workspace/lib/browser';
|
|
40
|
+
import { StandaloneServices } from '@theia/monaco-editor-core/esm/vs/editor/standalone/browser/standaloneServices';
|
|
41
|
+
import { ILanguageService } from '@theia/monaco-editor-core/esm/vs/editor/common/languages/language';
|
|
42
|
+
import { LanguageService } from '@theia/core/lib/browser/language-service';
|
|
43
|
+
import { DEFAULT_ICON_SIZE, PLUGIN_FILE_ICON_CLASS } from './plugin-shared-style';
|
|
44
|
+
|
|
45
|
+
export interface PluginIconDefinition {
|
|
46
|
+
iconPath: string;
|
|
47
|
+
fontColor: string;
|
|
48
|
+
fontCharacter: string;
|
|
49
|
+
fontSize: string;
|
|
50
|
+
fontId: string;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
export interface PluginFontDefinition {
|
|
54
|
+
id: string;
|
|
55
|
+
weight: string;
|
|
56
|
+
style: string;
|
|
57
|
+
size: string;
|
|
58
|
+
src: { path: string; format: string; }[];
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
export interface PluginIconsAssociation {
|
|
62
|
+
folder?: string;
|
|
63
|
+
file?: string;
|
|
64
|
+
folderExpanded?: string;
|
|
65
|
+
rootFolder?: string;
|
|
66
|
+
rootFolderExpanded?: string;
|
|
67
|
+
folderNames?: { [folderName: string]: string; };
|
|
68
|
+
folderNamesExpanded?: { [folderName: string]: string; };
|
|
69
|
+
fileExtensions?: { [extension: string]: string; };
|
|
70
|
+
fileNames?: { [fileName: string]: string; };
|
|
71
|
+
languageIds?: { [languageId: string]: string; };
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
export interface PluginIconDefinitions {
|
|
75
|
+
[key: string]: PluginIconDefinition
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
export interface PluginIconThemeDocument extends PluginIconsAssociation {
|
|
79
|
+
iconDefinitions: PluginIconDefinitions;
|
|
80
|
+
fonts: PluginFontDefinition[];
|
|
81
|
+
light?: PluginIconsAssociation;
|
|
82
|
+
highContrast?: PluginIconsAssociation;
|
|
83
|
+
hidesExplorerArrows?: boolean;
|
|
84
|
+
showLanguageModeIcons?: boolean;
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
export const PluginIconThemeFactory = Symbol('PluginIconThemeFactory');
|
|
88
|
+
export type PluginIconThemeFactory = (definition: PluginIconThemeDefinition) => PluginIconTheme;
|
|
89
|
+
|
|
90
|
+
@injectable()
|
|
91
|
+
export class PluginIconThemeDefinition implements IconThemeDefinition, IconThemeContribution {
|
|
92
|
+
id: string;
|
|
93
|
+
label: string;
|
|
94
|
+
description?: string;
|
|
95
|
+
uri: string;
|
|
96
|
+
uiTheme?: UiTheme;
|
|
97
|
+
pluginId: string;
|
|
98
|
+
packageUri: string;
|
|
99
|
+
hasFileIcons?: boolean;
|
|
100
|
+
hasFolderIcons?: boolean;
|
|
101
|
+
hidesExplorerArrows?: boolean;
|
|
102
|
+
showLanguageModeIcons?: boolean;
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
class PluginLanguageIconInfo {
|
|
106
|
+
hasSpecificFileIcons: boolean = false;
|
|
107
|
+
coveredLanguages: { [languageId: string]: boolean } = {};
|
|
108
|
+
};
|
|
109
|
+
|
|
110
|
+
@injectable()
|
|
111
|
+
export class PluginIconTheme extends PluginIconThemeDefinition implements IconTheme, Disposable {
|
|
112
|
+
|
|
113
|
+
@inject(FileService)
|
|
114
|
+
protected readonly fileService: FileService;
|
|
115
|
+
|
|
116
|
+
@inject(LabelProvider)
|
|
117
|
+
protected readonly labelProvider: LabelProvider;
|
|
118
|
+
|
|
119
|
+
@inject(PluginIconThemeDefinition)
|
|
120
|
+
protected readonly definition: PluginIconThemeDefinition;
|
|
121
|
+
|
|
122
|
+
@inject(WorkspaceService)
|
|
123
|
+
protected readonly workspaceService: WorkspaceService;
|
|
124
|
+
|
|
125
|
+
@inject(LanguageService)
|
|
126
|
+
protected readonly languageService: LanguageService;
|
|
127
|
+
|
|
128
|
+
protected readonly onDidChangeEmitter = new Emitter<DidChangeLabelEvent>();
|
|
129
|
+
readonly onDidChange = this.onDidChangeEmitter.event;
|
|
130
|
+
|
|
131
|
+
protected readonly toDeactivate = new DisposableCollection();
|
|
132
|
+
protected readonly toUnload = new DisposableCollection();
|
|
133
|
+
protected readonly toDisposeStyleElement = new DisposableCollection();
|
|
134
|
+
protected readonly toDispose = new DisposableCollection(
|
|
135
|
+
this.toDeactivate, this.toDisposeStyleElement, this.toUnload, this.onDidChangeEmitter
|
|
136
|
+
);
|
|
137
|
+
|
|
138
|
+
protected packageRootUri: URI;
|
|
139
|
+
protected locationUri: URI;
|
|
140
|
+
|
|
141
|
+
protected styleSheetContent: string | undefined;
|
|
142
|
+
protected readonly icons = new Set<string>();
|
|
143
|
+
|
|
144
|
+
@postConstruct()
|
|
145
|
+
protected init(): void {
|
|
146
|
+
Object.assign(this, this.definition);
|
|
147
|
+
this.packageRootUri = new URI(this.packageUri);
|
|
148
|
+
this.locationUri = new URI(this.uri).parent;
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
dispose(): void {
|
|
152
|
+
this.toDispose.dispose();
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
protected fireDidChange(): void {
|
|
156
|
+
this.onDidChangeEmitter.fire({ affects: () => true });
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
activate(): Disposable {
|
|
160
|
+
if (!this.toDeactivate.disposed) {
|
|
161
|
+
return this.toDeactivate;
|
|
162
|
+
}
|
|
163
|
+
this.toDeactivate.push(Disposable.create(() => this.fireDidChange()));
|
|
164
|
+
this.doActivate();
|
|
165
|
+
return this.toDeactivate;
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
protected async doActivate(): Promise<void> {
|
|
169
|
+
await this.load();
|
|
170
|
+
this.updateStyleElement();
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
protected updateStyleElement(): void {
|
|
174
|
+
this.toDisposeStyleElement.dispose();
|
|
175
|
+
if (this.toDeactivate.disposed || !this.styleSheetContent) {
|
|
176
|
+
return;
|
|
177
|
+
}
|
|
178
|
+
const styleElement = document.createElement('style');
|
|
179
|
+
styleElement.type = 'text/css';
|
|
180
|
+
styleElement.className = 'theia-icon-theme';
|
|
181
|
+
styleElement.innerText = this.styleSheetContent;
|
|
182
|
+
document.head.appendChild(styleElement);
|
|
183
|
+
const toRemoveStyleElement = Disposable.create(() => styleElement.remove());
|
|
184
|
+
this.toDisposeStyleElement.push(toRemoveStyleElement);
|
|
185
|
+
this.toDeactivate.push(toRemoveStyleElement);
|
|
186
|
+
this.fireDidChange();
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
protected reload = debounce(() => {
|
|
190
|
+
this.toUnload.dispose();
|
|
191
|
+
this.doActivate();
|
|
192
|
+
}, 50);
|
|
193
|
+
|
|
194
|
+
/**
|
|
195
|
+
* This should be aligned with
|
|
196
|
+
* https://github.com/microsoft/vscode/blob/7cf4cca47aa025a590fc939af54932042302be63/src/vs/workbench/services/themes/browser/fileIconThemeData.ts#L201
|
|
197
|
+
*/
|
|
198
|
+
protected async load(): Promise<void> {
|
|
199
|
+
if (this.styleSheetContent !== undefined) {
|
|
200
|
+
return;
|
|
201
|
+
}
|
|
202
|
+
this.styleSheetContent = '';
|
|
203
|
+
this.toUnload.push(Disposable.create(() => {
|
|
204
|
+
this.styleSheetContent = undefined;
|
|
205
|
+
this.hasFileIcons = undefined;
|
|
206
|
+
this.hasFolderIcons = undefined;
|
|
207
|
+
this.hidesExplorerArrows = undefined;
|
|
208
|
+
this.icons.clear();
|
|
209
|
+
}));
|
|
210
|
+
|
|
211
|
+
const uri = new URI(this.uri);
|
|
212
|
+
const result = await this.fileService.read(uri);
|
|
213
|
+
const content = result.value;
|
|
214
|
+
const json: RecursivePartial<PluginIconThemeDocument> = jsoncparser.parse(content, undefined, { disallowComments: false });
|
|
215
|
+
this.hidesExplorerArrows = !!json.hidesExplorerArrows;
|
|
216
|
+
|
|
217
|
+
const toUnwatch = this.fileService.watch(uri);
|
|
218
|
+
if (this.toUnload.disposed) {
|
|
219
|
+
toUnwatch.dispose();
|
|
220
|
+
} else {
|
|
221
|
+
this.toUnload.push(toUnwatch);
|
|
222
|
+
this.toUnload.push(this.fileService.onDidFilesChange(e => {
|
|
223
|
+
if (e.contains(uri, FileChangeType.ADDED) || e.contains(uri, FileChangeType.UPDATED)) {
|
|
224
|
+
this.reload();
|
|
225
|
+
}
|
|
226
|
+
}));
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
const iconDefinitions = json.iconDefinitions;
|
|
230
|
+
if (!iconDefinitions) {
|
|
231
|
+
return;
|
|
232
|
+
}
|
|
233
|
+
const definitionSelectors = new Map<string, string[]>();
|
|
234
|
+
const acceptSelector = (themeType: ThemeType, definitionId: string, ...icons: string[]) => {
|
|
235
|
+
if (!iconDefinitions[definitionId]) {
|
|
236
|
+
return;
|
|
237
|
+
}
|
|
238
|
+
let selector = '';
|
|
239
|
+
for (const icon of icons) {
|
|
240
|
+
if (icon) {
|
|
241
|
+
selector += '.' + icon;
|
|
242
|
+
this.icons.add(icon);
|
|
243
|
+
}
|
|
244
|
+
}
|
|
245
|
+
if (!selector) {
|
|
246
|
+
return;
|
|
247
|
+
}
|
|
248
|
+
const selectors = definitionSelectors.get(definitionId) || [];
|
|
249
|
+
if (themeType !== 'dark') {
|
|
250
|
+
selector = '.theia-' + themeType + ' ' + selector;
|
|
251
|
+
}
|
|
252
|
+
selectors.push(selector + '::before');
|
|
253
|
+
definitionSelectors.set(definitionId, selectors);
|
|
254
|
+
};
|
|
255
|
+
|
|
256
|
+
let iconInfo = this.collectSelectors(json, acceptSelector.bind(undefined, 'dark'));
|
|
257
|
+
if (json.light) {
|
|
258
|
+
iconInfo = this.collectSelectors(json.light, acceptSelector.bind(undefined, 'light'));
|
|
259
|
+
}
|
|
260
|
+
if (json.highContrast) {
|
|
261
|
+
iconInfo = this.collectSelectors(json.highContrast, acceptSelector.bind(undefined, 'hc'));
|
|
262
|
+
}
|
|
263
|
+
|
|
264
|
+
const showLanguageModeIcons = this.showLanguageModeIcons === true
|
|
265
|
+
|| json.showLanguageModeIcons === true
|
|
266
|
+
|| (iconInfo.hasSpecificFileIcons && json.showLanguageModeIcons !== false);
|
|
267
|
+
|
|
268
|
+
const fonts = json.fonts;
|
|
269
|
+
if (Array.isArray(fonts)) {
|
|
270
|
+
for (const font of fonts) {
|
|
271
|
+
if (font) {
|
|
272
|
+
let src = '';
|
|
273
|
+
if (Array.isArray(font.src)) {
|
|
274
|
+
for (const srcLocation of font.src) {
|
|
275
|
+
if (srcLocation && srcLocation.path) {
|
|
276
|
+
const cssUrl = this.toCSSUrl(srcLocation.path);
|
|
277
|
+
if (cssUrl) {
|
|
278
|
+
if (src) {
|
|
279
|
+
src += ', ';
|
|
280
|
+
}
|
|
281
|
+
src += `${cssUrl} format('${srcLocation.format}')`;
|
|
282
|
+
}
|
|
283
|
+
}
|
|
284
|
+
}
|
|
285
|
+
}
|
|
286
|
+
if (src) {
|
|
287
|
+
this.styleSheetContent += `@font-face {
|
|
288
|
+
src: ${src};
|
|
289
|
+
font-family: '${font.id}';
|
|
290
|
+
font-weight: ${font.weight};
|
|
291
|
+
font-style: ${font.style};
|
|
292
|
+
}
|
|
293
|
+
`;
|
|
294
|
+
}
|
|
295
|
+
}
|
|
296
|
+
}
|
|
297
|
+
const firstFont = fonts[0];
|
|
298
|
+
if (firstFont && firstFont.id) {
|
|
299
|
+
this.styleSheetContent += `.${this.fileIcon}::before, .${this.folderIcon}::before, .${this.rootFolderIcon}::before {
|
|
300
|
+
font-family: '${firstFont.id}';
|
|
301
|
+
font-size: ${firstFont.size || '150%'};
|
|
302
|
+
-webkit-font-smoothing: antialiased;
|
|
303
|
+
-moz-osx-font-smoothing: grayscale;
|
|
304
|
+
vertical-align: top;
|
|
305
|
+
}
|
|
306
|
+
`;
|
|
307
|
+
}
|
|
308
|
+
}
|
|
309
|
+
|
|
310
|
+
for (const definitionId of definitionSelectors.keys()) {
|
|
311
|
+
const iconDefinition = iconDefinitions[definitionId];
|
|
312
|
+
const selectors = definitionSelectors.get(definitionId);
|
|
313
|
+
if (selectors && iconDefinition) {
|
|
314
|
+
const cssUrl = this.toCSSUrl(iconDefinition.iconPath);
|
|
315
|
+
if (cssUrl) {
|
|
316
|
+
this.styleSheetContent += `${selectors.join(', ')} {
|
|
317
|
+
content: ' ';
|
|
318
|
+
background-image: ${cssUrl};
|
|
319
|
+
background-size: ${DEFAULT_ICON_SIZE}px;
|
|
320
|
+
background-position: left center;
|
|
321
|
+
background-repeat: no-repeat;
|
|
322
|
+
}
|
|
323
|
+
`;
|
|
324
|
+
}
|
|
325
|
+
if (iconDefinition.fontCharacter || iconDefinition.fontColor) {
|
|
326
|
+
let body = '';
|
|
327
|
+
if (iconDefinition.fontColor) {
|
|
328
|
+
body += ` color: ${iconDefinition.fontColor};`;
|
|
329
|
+
}
|
|
330
|
+
if (iconDefinition.fontCharacter) {
|
|
331
|
+
body += ` content: '${iconDefinition.fontCharacter}';`;
|
|
332
|
+
}
|
|
333
|
+
if (iconDefinition.fontSize) {
|
|
334
|
+
body += ` font-size: ${iconDefinition.fontSize};`;
|
|
335
|
+
}
|
|
336
|
+
if (iconDefinition.fontId) {
|
|
337
|
+
body += ` font-family: ${iconDefinition.fontId};`;
|
|
338
|
+
}
|
|
339
|
+
this.styleSheetContent += `${selectors.join(', ')} {${body} }\n`;
|
|
340
|
+
}
|
|
341
|
+
}
|
|
342
|
+
}
|
|
343
|
+
|
|
344
|
+
if (showLanguageModeIcons) {
|
|
345
|
+
for (const language of this.languageService.languages) {
|
|
346
|
+
// only show language icons if there are no more specific icons in the style document
|
|
347
|
+
if (!iconInfo.coveredLanguages[language.id]) {
|
|
348
|
+
const icon = this.languageService.getIcon(language.id);
|
|
349
|
+
if (icon) {
|
|
350
|
+
this.icons.add(this.fileIcon);
|
|
351
|
+
this.icons.add(this.languageIcon(language.id));
|
|
352
|
+
this.icons.add(icon);
|
|
353
|
+
}
|
|
354
|
+
}
|
|
355
|
+
}
|
|
356
|
+
}
|
|
357
|
+
}
|
|
358
|
+
|
|
359
|
+
protected toCSSUrl(iconPath: string | undefined): string | undefined {
|
|
360
|
+
if (!iconPath) {
|
|
361
|
+
return undefined;
|
|
362
|
+
}
|
|
363
|
+
const iconUri = this.locationUri.resolve(iconPath);
|
|
364
|
+
const relativePath = this.packageRootUri.path.relative(iconUri.path.normalize());
|
|
365
|
+
return relativePath && `url('${new Endpoint({
|
|
366
|
+
path: `hostedPlugin/${this.pluginId}/${encodeURIComponent(relativePath.normalize().toString())}`
|
|
367
|
+
}).getRestUrl().toString()}')`;
|
|
368
|
+
}
|
|
369
|
+
|
|
370
|
+
protected escapeCSS(value: string): string {
|
|
371
|
+
value = value.replace(/[^\-a-zA-Z0-9]/g, '-');
|
|
372
|
+
if (value.charAt(0).match(/[0-9\-]/)) {
|
|
373
|
+
value = '-' + value;
|
|
374
|
+
}
|
|
375
|
+
return value;
|
|
376
|
+
}
|
|
377
|
+
|
|
378
|
+
protected readonly fileIcon = PLUGIN_FILE_ICON_CLASS;
|
|
379
|
+
protected readonly folderIcon = 'theia-plugin-folder-icon';
|
|
380
|
+
protected readonly folderExpandedIcon = 'theia-plugin-folder-expanded-icon';
|
|
381
|
+
protected readonly rootFolderIcon = 'theia-plugin-root-folder-icon';
|
|
382
|
+
protected readonly rootFolderExpandedIcon = 'theia-plugin-root-folder-expanded-icon';
|
|
383
|
+
protected folderNameIcon(folderName: string): string {
|
|
384
|
+
return 'theia-plugin-' + this.escapeCSS(folderName.toLowerCase()) + '-folder-name-icon';
|
|
385
|
+
}
|
|
386
|
+
protected expandedFolderNameIcon(folderName: string): string {
|
|
387
|
+
return 'theia-plugin-' + this.escapeCSS(folderName.toLowerCase()) + '-expanded-folder-name-icon';
|
|
388
|
+
}
|
|
389
|
+
protected fileNameIcon(fileName: string): string[] {
|
|
390
|
+
fileName = fileName.toLowerCase();
|
|
391
|
+
const extIndex = fileName.indexOf('.');
|
|
392
|
+
const icons = extIndex !== -1 ? this.fileExtensionIcon(fileName.substring(extIndex + 1)) : [];
|
|
393
|
+
icons.unshift('theia-plugin-' + this.escapeCSS(fileName) + '-file-name-icon');
|
|
394
|
+
return icons;
|
|
395
|
+
}
|
|
396
|
+
protected fileExtensionIcon(fileExtension: string): string[] {
|
|
397
|
+
fileExtension = fileExtension.toString();
|
|
398
|
+
const icons = [];
|
|
399
|
+
const segments = fileExtension.split('.');
|
|
400
|
+
if (segments.length) {
|
|
401
|
+
if (segments.length) {
|
|
402
|
+
for (let i = 0; i < segments.length; i++) {
|
|
403
|
+
icons.push('theia-plugin-' + this.escapeCSS(segments.slice(i).join('.')) + '-ext-file-icon');
|
|
404
|
+
}
|
|
405
|
+
icons.push('theia-plugin-ext-file-icon'); // extra segment to increase file-ext score
|
|
406
|
+
}
|
|
407
|
+
}
|
|
408
|
+
return icons;
|
|
409
|
+
}
|
|
410
|
+
protected languageIcon(languageId: string): string {
|
|
411
|
+
return 'theia-plugin-' + this.escapeCSS(languageId) + '-lang-file-icon';
|
|
412
|
+
}
|
|
413
|
+
|
|
414
|
+
protected collectSelectors(associations: RecursivePartial<PluginIconsAssociation>, accept: (definitionId: string, ...icons: string[]) => void): PluginLanguageIconInfo {
|
|
415
|
+
const iconInfo = new PluginLanguageIconInfo();
|
|
416
|
+
if (associations.folder) {
|
|
417
|
+
accept(associations.folder, this.folderIcon);
|
|
418
|
+
if (associations.folderExpanded === undefined) {
|
|
419
|
+
// Use the same icon for expanded state (issue #12727). Check for
|
|
420
|
+
// undefined folderExpanded property to allow for
|
|
421
|
+
// "folderExpanded": null in case a developer really wants that
|
|
422
|
+
accept(associations.folder, this.folderExpandedIcon);
|
|
423
|
+
}
|
|
424
|
+
this.hasFolderIcons = true;
|
|
425
|
+
}
|
|
426
|
+
if (associations.folderExpanded) {
|
|
427
|
+
accept(associations.folderExpanded, this.folderExpandedIcon);
|
|
428
|
+
this.hasFolderIcons = true;
|
|
429
|
+
}
|
|
430
|
+
const rootFolder = associations.rootFolder || associations.folder;
|
|
431
|
+
if (rootFolder) {
|
|
432
|
+
accept(rootFolder, this.rootFolderIcon);
|
|
433
|
+
this.hasFolderIcons = true;
|
|
434
|
+
}
|
|
435
|
+
const rootFolderExpanded = associations.rootFolderExpanded || associations.folderExpanded;
|
|
436
|
+
if (rootFolderExpanded) {
|
|
437
|
+
accept(rootFolderExpanded, this.rootFolderExpandedIcon);
|
|
438
|
+
this.hasFolderIcons = true;
|
|
439
|
+
}
|
|
440
|
+
if (associations.file) {
|
|
441
|
+
accept(associations.file, this.fileIcon);
|
|
442
|
+
this.hasFileIcons = true;
|
|
443
|
+
}
|
|
444
|
+
const folderNames = associations.folderNames;
|
|
445
|
+
if (folderNames) {
|
|
446
|
+
// eslint-disable-next-line guard-for-in
|
|
447
|
+
for (const folderName in folderNames) {
|
|
448
|
+
accept(folderNames[folderName]!, this.folderNameIcon(folderName), this.folderIcon);
|
|
449
|
+
this.hasFolderIcons = true;
|
|
450
|
+
}
|
|
451
|
+
}
|
|
452
|
+
const folderNamesExpanded = associations.folderNamesExpanded;
|
|
453
|
+
if (folderNamesExpanded) {
|
|
454
|
+
// eslint-disable-next-line guard-for-in
|
|
455
|
+
for (const folderName in folderNamesExpanded) {
|
|
456
|
+
accept(folderNamesExpanded[folderName]!, this.expandedFolderNameIcon(folderName), this.folderExpandedIcon);
|
|
457
|
+
this.hasFolderIcons = true;
|
|
458
|
+
}
|
|
459
|
+
}
|
|
460
|
+
const languageIds = associations.languageIds;
|
|
461
|
+
if (languageIds) {
|
|
462
|
+
if (!languageIds.jsonc && languageIds.json) {
|
|
463
|
+
languageIds.jsonc = languageIds.json;
|
|
464
|
+
}
|
|
465
|
+
// eslint-disable-next-line guard-for-in
|
|
466
|
+
for (const languageId in languageIds) {
|
|
467
|
+
accept(languageIds[languageId]!, this.languageIcon(languageId), this.fileIcon);
|
|
468
|
+
this.hasFileIcons = true;
|
|
469
|
+
iconInfo.hasSpecificFileIcons = true;
|
|
470
|
+
iconInfo.coveredLanguages[languageId] = true;
|
|
471
|
+
}
|
|
472
|
+
}
|
|
473
|
+
const fileExtensions = associations.fileExtensions;
|
|
474
|
+
if (fileExtensions) {
|
|
475
|
+
// eslint-disable-next-line guard-for-in
|
|
476
|
+
for (const fileExtension in fileExtensions) {
|
|
477
|
+
accept(fileExtensions[fileExtension]!, ...this.fileExtensionIcon(fileExtension), this.fileIcon);
|
|
478
|
+
this.hasFileIcons = true;
|
|
479
|
+
iconInfo.hasSpecificFileIcons = true;
|
|
480
|
+
}
|
|
481
|
+
}
|
|
482
|
+
const fileNames = associations.fileNames;
|
|
483
|
+
if (fileNames) {
|
|
484
|
+
// eslint-disable-next-line guard-for-in
|
|
485
|
+
for (const fileName in fileNames) {
|
|
486
|
+
accept(fileNames[fileName]!, ...this.fileNameIcon(fileName), this.fileIcon);
|
|
487
|
+
this.hasFileIcons = true;
|
|
488
|
+
iconInfo.hasSpecificFileIcons = true;
|
|
489
|
+
}
|
|
490
|
+
}
|
|
491
|
+
return iconInfo;
|
|
492
|
+
}
|
|
493
|
+
|
|
494
|
+
/**
|
|
495
|
+
* This should be aligned with
|
|
496
|
+
* https://github.com/microsoft/vscode/blob/7cf4cca47aa025a590fc939af54932042302be63/src/vs/editor/common/services/getIconClasses.ts#L5
|
|
497
|
+
*/
|
|
498
|
+
getIcon(element: URI | URIIconReference | FileStat | FileStatNode | WorkspaceRootNode): string {
|
|
499
|
+
let icon = '';
|
|
500
|
+
for (const className of this.getClassNames(element)) {
|
|
501
|
+
if (this.icons.has(className)) {
|
|
502
|
+
if (icon) {
|
|
503
|
+
icon += ' ';
|
|
504
|
+
}
|
|
505
|
+
icon += className;
|
|
506
|
+
}
|
|
507
|
+
}
|
|
508
|
+
return icon;
|
|
509
|
+
}
|
|
510
|
+
|
|
511
|
+
protected getClassNames(element: URI | URIIconReference | FileStat | FileStatNode | WorkspaceRootNode): string[] {
|
|
512
|
+
if (WorkspaceRootNode.is(element)) {
|
|
513
|
+
const name = this.labelProvider.getName(element);
|
|
514
|
+
if (element.expanded) {
|
|
515
|
+
return [this.rootFolderExpandedIcon, this.expandedFolderNameIcon(name)];
|
|
516
|
+
}
|
|
517
|
+
return [this.rootFolderIcon, this.folderNameIcon(name)];
|
|
518
|
+
}
|
|
519
|
+
if (DirNode.is(element)) {
|
|
520
|
+
if (element.expanded) {
|
|
521
|
+
const name = this.labelProvider.getName(element);
|
|
522
|
+
return [this.folderExpandedIcon, this.expandedFolderNameIcon(name)];
|
|
523
|
+
}
|
|
524
|
+
return this.getFolderClassNames(element);
|
|
525
|
+
}
|
|
526
|
+
if (FileStatNode.is(element)) {
|
|
527
|
+
return this.getFileClassNames(element, element.fileStat.resource.toString());
|
|
528
|
+
}
|
|
529
|
+
if (FileStat.is(element)) {
|
|
530
|
+
if (element.isDirectory) {
|
|
531
|
+
return this.getFolderClassNames(element);
|
|
532
|
+
}
|
|
533
|
+
return this.getFileClassNames(element, element.resource.toString());
|
|
534
|
+
}
|
|
535
|
+
if (URIIconReference.is(element)) {
|
|
536
|
+
if (element.id === 'folder') {
|
|
537
|
+
return this.getFolderClassNames(element);
|
|
538
|
+
}
|
|
539
|
+
return this.getFileClassNames(element, element.uri && element.uri.toString());
|
|
540
|
+
}
|
|
541
|
+
return this.getFileClassNames(element, element.toString());
|
|
542
|
+
}
|
|
543
|
+
|
|
544
|
+
protected getFolderClassNames(element: object): string[] {
|
|
545
|
+
const name = this.labelProvider.getName(element);
|
|
546
|
+
return [this.folderIcon, this.folderNameIcon(name)];
|
|
547
|
+
}
|
|
548
|
+
|
|
549
|
+
protected getFileClassNames(element: object, uri?: string): string[] {
|
|
550
|
+
const name = this.labelProvider.getName(element);
|
|
551
|
+
const classNames = this.fileNameIcon(name);
|
|
552
|
+
if (uri) {
|
|
553
|
+
const parsedURI = new URI(uri);
|
|
554
|
+
const isRoot = this.workspaceService.getWorkspaceRootUri(new URI(uri))?.isEqual(parsedURI);
|
|
555
|
+
if (isRoot) {
|
|
556
|
+
classNames.unshift(this.rootFolderIcon);
|
|
557
|
+
} else {
|
|
558
|
+
classNames.unshift(this.fileIcon);
|
|
559
|
+
}
|
|
560
|
+
const language = StandaloneServices.get(ILanguageService).createByFilepathOrFirstLine(parsedURI['codeUri']);
|
|
561
|
+
classNames.push(this.languageIcon(language.languageId));
|
|
562
|
+
const defaultLanguageIcon = this.languageService.getIcon(language.languageId);
|
|
563
|
+
if (defaultLanguageIcon) {
|
|
564
|
+
classNames.push(defaultLanguageIcon);
|
|
565
|
+
}
|
|
566
|
+
}
|
|
567
|
+
return classNames;
|
|
568
|
+
}
|
|
569
|
+
|
|
570
|
+
}
|
|
571
|
+
|
|
572
|
+
@injectable()
|
|
573
|
+
export class PluginIconThemeService implements LabelProviderContribution {
|
|
574
|
+
|
|
575
|
+
@inject(IconThemeService)
|
|
576
|
+
protected readonly iconThemeService: IconThemeService;
|
|
577
|
+
|
|
578
|
+
@inject(PluginIconThemeFactory)
|
|
579
|
+
protected readonly iconThemeFactory: PluginIconThemeFactory;
|
|
580
|
+
|
|
581
|
+
protected readonly onDidChangeEmitter = new Emitter<DidChangeLabelEvent>();
|
|
582
|
+
readonly onDidChange = this.onDidChangeEmitter.event;
|
|
583
|
+
|
|
584
|
+
protected fireDidChange(): void {
|
|
585
|
+
this.onDidChangeEmitter.fire({ affects: () => true });
|
|
586
|
+
}
|
|
587
|
+
|
|
588
|
+
register(contribution: IconThemeContribution, plugin: DeployedPlugin): Disposable {
|
|
589
|
+
const pluginId = getPluginId(plugin.metadata.model);
|
|
590
|
+
const packageUri = plugin.metadata.model.packageUri;
|
|
591
|
+
const iconTheme = this.iconThemeFactory({
|
|
592
|
+
id: contribution.id,
|
|
593
|
+
label: contribution.label || new URI(contribution.uri).path.base,
|
|
594
|
+
description: contribution.description,
|
|
595
|
+
uri: contribution.uri,
|
|
596
|
+
uiTheme: contribution.uiTheme,
|
|
597
|
+
pluginId,
|
|
598
|
+
packageUri
|
|
599
|
+
});
|
|
600
|
+
return new DisposableCollection(
|
|
601
|
+
iconTheme,
|
|
602
|
+
iconTheme.onDidChange(() => this.fireDidChange()),
|
|
603
|
+
this.iconThemeService.register(iconTheme)
|
|
604
|
+
);
|
|
605
|
+
}
|
|
606
|
+
|
|
607
|
+
canHandle(element: object): number {
|
|
608
|
+
const current = this.iconThemeService.getDefinition(this.iconThemeService.current);
|
|
609
|
+
if (current instanceof PluginIconTheme && (
|
|
610
|
+
(element instanceof URI && element.scheme === 'file') || URIIconReference.is(element) || FileStat.is(element) || FileStatNode.is(element)
|
|
611
|
+
)) {
|
|
612
|
+
return Number.MAX_SAFE_INTEGER;
|
|
613
|
+
}
|
|
614
|
+
return 0;
|
|
615
|
+
}
|
|
616
|
+
|
|
617
|
+
getIcon(element: URI | URIIconReference | FileStat | FileStatNode | WorkspaceRootNode): string | undefined {
|
|
618
|
+
const current = this.iconThemeService.getDefinition(this.iconThemeService.current);
|
|
619
|
+
if (current instanceof PluginIconTheme) {
|
|
620
|
+
return current.getIcon(element);
|
|
621
|
+
}
|
|
622
|
+
return undefined;
|
|
623
|
+
}
|
|
624
|
+
|
|
625
|
+
}
|