@theia/debug 1.34.3 → 1.34.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (250) hide show
  1. package/LICENSE +641 -641
  2. package/README.md +62 -62
  3. package/bin/download-adapters.js +76 -76
  4. package/lib/browser/breakpoint/breakpoint-manager.d.ts +69 -69
  5. package/lib/browser/breakpoint/breakpoint-manager.js +311 -311
  6. package/lib/browser/breakpoint/breakpoint-marker.d.ts +40 -40
  7. package/lib/browser/breakpoint/breakpoint-marker.js +73 -73
  8. package/lib/browser/console/debug-console-contribution.d.ts +40 -40
  9. package/lib/browser/console/debug-console-contribution.js +227 -227
  10. package/lib/browser/console/debug-console-items.d.ts +93 -93
  11. package/lib/browser/console/debug-console-items.js +318 -318
  12. package/lib/browser/console/debug-console-session.d.ts +31 -31
  13. package/lib/browser/console/debug-console-session.js +199 -199
  14. package/lib/browser/debug-call-stack-item-type-key.d.ts +3 -3
  15. package/lib/browser/debug-call-stack-item-type-key.js +19 -19
  16. package/lib/browser/debug-configuration-manager.d.ts +101 -101
  17. package/lib/browser/debug-configuration-manager.js +521 -521
  18. package/lib/browser/debug-configuration-model.d.ts +29 -29
  19. package/lib/browser/debug-configuration-model.js +79 -79
  20. package/lib/browser/debug-contribution.d.ts +22 -22
  21. package/lib/browser/debug-contribution.js +19 -19
  22. package/lib/browser/debug-frontend-application-contribution.d.ts +214 -214
  23. package/lib/browser/debug-frontend-application-contribution.js +1351 -1351
  24. package/lib/browser/debug-frontend-module.d.ts +4 -4
  25. package/lib/browser/debug-frontend-module.js +108 -108
  26. package/lib/browser/debug-keybinding-contexts.d.ts +24 -24
  27. package/lib/browser/debug-keybinding-contexts.js +85 -85
  28. package/lib/browser/debug-package.spec.js +18 -18
  29. package/lib/browser/debug-preferences.d.ts +17 -17
  30. package/lib/browser/debug-preferences.js +89 -89
  31. package/lib/browser/debug-prefix-configuration.d.ts +50 -50
  32. package/lib/browser/debug-prefix-configuration.js +204 -204
  33. package/lib/browser/debug-resource.d.ts +14 -14
  34. package/lib/browser/debug-resource.js +65 -65
  35. package/lib/browser/debug-schema-updater.d.ts +15 -15
  36. package/lib/browser/debug-schema-updater.js +168 -168
  37. package/lib/browser/debug-session-connection.d.ts +110 -110
  38. package/lib/browser/debug-session-connection.js +258 -258
  39. package/lib/browser/debug-session-contribution.d.ts +73 -73
  40. package/lib/browser/debug-session-contribution.js +145 -145
  41. package/lib/browser/debug-session-manager.d.ts +137 -137
  42. package/lib/browser/debug-session-manager.js +593 -593
  43. package/lib/browser/debug-session-options.d.ts +48 -48
  44. package/lib/browser/debug-session-options.js +85 -85
  45. package/lib/browser/debug-session.d.ts +178 -178
  46. package/lib/browser/debug-session.js +859 -859
  47. package/lib/browser/debug-tab-bar-decorator.d.ts +16 -16
  48. package/lib/browser/debug-tab-bar-decorator.js +70 -70
  49. package/lib/browser/debug-watch-manager.d.ts +21 -21
  50. package/lib/browser/debug-watch-manager.js +93 -93
  51. package/lib/browser/disassembly-view/disassembly-view-accessibility-provider.d.ts +6 -6
  52. package/lib/browser/disassembly-view/disassembly-view-accessibility-provider.js +38 -38
  53. package/lib/browser/disassembly-view/disassembly-view-breakpoint-renderer.d.ts +19 -19
  54. package/lib/browser/disassembly-view/disassembly-view-breakpoint-renderer.js +110 -110
  55. package/lib/browser/disassembly-view/disassembly-view-contribution.d.ts +24 -24
  56. package/lib/browser/disassembly-view/disassembly-view-contribution.js +131 -131
  57. package/lib/browser/disassembly-view/disassembly-view-instruction-renderer.d.ts +38 -38
  58. package/lib/browser/disassembly-view/disassembly-view-instruction-renderer.js +222 -222
  59. package/lib/browser/disassembly-view/disassembly-view-table-delegate.d.ts +15 -15
  60. package/lib/browser/disassembly-view/disassembly-view-table-delegate.js +38 -38
  61. package/lib/browser/disassembly-view/disassembly-view-utilities.d.ts +37 -37
  62. package/lib/browser/disassembly-view/disassembly-view-utilities.js +17 -17
  63. package/lib/browser/disassembly-view/disassembly-view-widget.d.ts +60 -60
  64. package/lib/browser/disassembly-view/disassembly-view-widget.js +452 -452
  65. package/lib/browser/editor/debug-breakpoint-widget.d.ts +51 -51
  66. package/lib/browser/editor/debug-breakpoint-widget.js +245 -245
  67. package/lib/browser/editor/debug-editor-model.d.ts +91 -91
  68. package/lib/browser/editor/debug-editor-model.js +497 -497
  69. package/lib/browser/editor/debug-editor-service.d.ts +37 -37
  70. package/lib/browser/editor/debug-editor-service.js +214 -214
  71. package/lib/browser/editor/debug-editor.d.ts +3 -3
  72. package/lib/browser/editor/debug-editor.js +19 -19
  73. package/lib/browser/editor/debug-exception-widget.d.ts +28 -28
  74. package/lib/browser/editor/debug-exception-widget.js +111 -111
  75. package/lib/browser/editor/debug-expression-provider.d.ts +11 -11
  76. package/lib/browser/editor/debug-expression-provider.js +81 -81
  77. package/lib/browser/editor/debug-hover-source.d.ts +18 -18
  78. package/lib/browser/editor/debug-hover-source.js +117 -117
  79. package/lib/browser/editor/debug-hover-widget.d.ts +47 -47
  80. package/lib/browser/editor/debug-hover-widget.js +267 -267
  81. package/lib/browser/editor/debug-inline-value-decorator.d.ts +21 -21
  82. package/lib/browser/editor/debug-inline-value-decorator.js +337 -337
  83. package/lib/browser/model/debug-breakpoint.d.ts +47 -47
  84. package/lib/browser/model/debug-breakpoint.js +109 -109
  85. package/lib/browser/model/debug-function-breakpoint.d.ts +18 -18
  86. package/lib/browser/model/debug-function-breakpoint.js +92 -92
  87. package/lib/browser/model/debug-instruction-breakpoint.d.ts +14 -14
  88. package/lib/browser/model/debug-instruction-breakpoint.js +65 -65
  89. package/lib/browser/model/debug-source-breakpoint.d.ts +37 -37
  90. package/lib/browser/model/debug-source-breakpoint.js +211 -211
  91. package/lib/browser/model/debug-source.d.ts +24 -24
  92. package/lib/browser/model/debug-source.js +80 -80
  93. package/lib/browser/model/debug-stack-frame.d.ts +41 -41
  94. package/lib/browser/model/debug-stack-frame.js +146 -146
  95. package/lib/browser/model/debug-thread.d.ts +60 -60
  96. package/lib/browser/model/debug-thread.js +200 -200
  97. package/lib/browser/preferences/launch-preferences.d.ts +4 -4
  98. package/lib/browser/preferences/launch-preferences.js +38 -38
  99. package/lib/browser/view/debug-action.d.ts +16 -16
  100. package/lib/browser/view/debug-action.js +43 -43
  101. package/lib/browser/view/debug-breakpoints-source.d.ts +9 -9
  102. package/lib/browser/view/debug-breakpoints-source.js +70 -70
  103. package/lib/browser/view/debug-breakpoints-widget.d.ts +22 -22
  104. package/lib/browser/view/debug-breakpoints-widget.js +87 -87
  105. package/lib/browser/view/debug-configuration-select.d.ts +61 -61
  106. package/lib/browser/view/debug-configuration-select.js +204 -204
  107. package/lib/browser/view/debug-configuration-widget.d.ts +30 -30
  108. package/lib/browser/view/debug-configuration-widget.js +135 -135
  109. package/lib/browser/view/debug-exception-breakpoint.d.ts +13 -13
  110. package/lib/browser/view/debug-exception-breakpoint.js +34 -34
  111. package/lib/browser/view/debug-session-widget.d.ts +28 -28
  112. package/lib/browser/view/debug-session-widget.js +134 -134
  113. package/lib/browser/view/debug-stack-frames-source.d.ts +17 -17
  114. package/lib/browser/view/debug-stack-frames-source.js +89 -89
  115. package/lib/browser/view/debug-stack-frames-widget.d.ts +24 -24
  116. package/lib/browser/view/debug-stack-frames-widget.js +154 -154
  117. package/lib/browser/view/debug-threads-source.d.ts +8 -8
  118. package/lib/browser/view/debug-threads-source.js +63 -63
  119. package/lib/browser/view/debug-threads-widget.d.ts +26 -26
  120. package/lib/browser/view/debug-threads-widget.js +145 -145
  121. package/lib/browser/view/debug-toolbar-widget.d.ts +34 -34
  122. package/lib/browser/view/debug-toolbar-widget.js +156 -156
  123. package/lib/browser/view/debug-variables-source.d.ts +9 -9
  124. package/lib/browser/view/debug-variables-source.js +60 -60
  125. package/lib/browser/view/debug-variables-widget.d.ts +16 -16
  126. package/lib/browser/view/debug-variables-widget.js +78 -78
  127. package/lib/browser/view/debug-view-model.d.ts +51 -51
  128. package/lib/browser/view/debug-view-model.js +220 -220
  129. package/lib/browser/view/debug-watch-expression.d.ts +30 -30
  130. package/lib/browser/view/debug-watch-expression.js +73 -73
  131. package/lib/browser/view/debug-watch-source.d.ts +9 -9
  132. package/lib/browser/view/debug-watch-source.js +58 -58
  133. package/lib/browser/view/debug-watch-widget.d.ts +16 -16
  134. package/lib/browser/view/debug-watch-widget.js +78 -78
  135. package/lib/browser/view/debug-widget.d.ts +23 -23
  136. package/lib/browser/view/debug-widget.js +108 -108
  137. package/lib/common/debug-adapter-contribution-registry.d.ts +59 -59
  138. package/lib/common/debug-adapter-contribution-registry.js +212 -212
  139. package/lib/common/debug-adapter-session.d.ts +18 -18
  140. package/lib/common/debug-adapter-session.js +78 -78
  141. package/lib/common/debug-common.d.ts +1 -1
  142. package/lib/common/debug-common.js +29 -29
  143. package/lib/common/debug-compound.d.ts +14 -14
  144. package/lib/common/debug-compound.js +27 -27
  145. package/lib/common/debug-configuration.d.ts +58 -58
  146. package/lib/common/debug-configuration.js +31 -31
  147. package/lib/common/debug-model.d.ts +152 -152
  148. package/lib/common/debug-model.js +35 -35
  149. package/lib/common/debug-service.d.ts +123 -123
  150. package/lib/common/debug-service.js +66 -66
  151. package/lib/common/debug-uri-utils.d.ts +23 -23
  152. package/lib/common/debug-uri-utils.js +26 -26
  153. package/lib/common/inline-debug-adapter.d.ts +18 -18
  154. package/lib/common/inline-debug-adapter.js +44 -44
  155. package/lib/node/debug-adapter-factory.d.ts +19 -19
  156. package/lib/node/debug-adapter-factory.js +96 -96
  157. package/lib/node/debug-adapter-session-manager.d.ts +37 -37
  158. package/lib/node/debug-adapter-session-manager.js +111 -111
  159. package/lib/node/debug-backend-module.d.ts +3 -3
  160. package/lib/node/debug-backend-module.js +41 -41
  161. package/lib/node/debug-service-impl.d.ts +30 -30
  162. package/lib/node/debug-service-impl.js +124 -124
  163. package/lib/node/stream-debug-adapter.d.ts +36 -36
  164. package/lib/node/stream-debug-adapter.js +112 -112
  165. package/lib/node/vscode/vscode-debug-adapter-contribution.d.ts +47 -47
  166. package/lib/node/vscode/vscode-debug-adapter-contribution.js +221 -221
  167. package/package.json +15 -15
  168. package/src/browser/breakpoint/breakpoint-manager.ts +351 -351
  169. package/src/browser/breakpoint/breakpoint-marker.ts +102 -102
  170. package/src/browser/console/debug-console-contribution.tsx +239 -239
  171. package/src/browser/console/debug-console-items.tsx +383 -383
  172. package/src/browser/console/debug-console-session.ts +205 -205
  173. package/src/browser/debug-call-stack-item-type-key.ts +20 -20
  174. package/src/browser/debug-configuration-manager.ts +559 -559
  175. package/src/browser/debug-configuration-model.ts +100 -100
  176. package/src/browser/debug-contribution.ts +43 -43
  177. package/src/browser/debug-frontend-application-contribution.ts +1503 -1503
  178. package/src/browser/debug-frontend-module.ts +136 -136
  179. package/src/browser/debug-keybinding-contexts.ts +75 -75
  180. package/src/browser/debug-package.spec.ts +20 -20
  181. package/src/browser/debug-preferences.ts +98 -98
  182. package/src/browser/debug-prefix-configuration.ts +190 -190
  183. package/src/browser/debug-resource.ts +59 -59
  184. package/src/browser/debug-schema-updater.ts +149 -149
  185. package/src/browser/debug-session-connection.ts +357 -357
  186. package/src/browser/debug-session-contribution.ts +150 -150
  187. package/src/browser/debug-session-manager.ts +681 -681
  188. package/src/browser/debug-session-options.ts +114 -114
  189. package/src/browser/debug-session.tsx +936 -936
  190. package/src/browser/debug-tab-bar-decorator.ts +57 -57
  191. package/src/browser/debug-watch-manager.ts +93 -93
  192. package/src/browser/disassembly-view/disassembly-view-accessibility-provider.ts +43 -43
  193. package/src/browser/disassembly-view/disassembly-view-breakpoint-renderer.ts +119 -119
  194. package/src/browser/disassembly-view/disassembly-view-contribution.ts +109 -109
  195. package/src/browser/disassembly-view/disassembly-view-instruction-renderer.ts +245 -245
  196. package/src/browser/disassembly-view/disassembly-view-table-delegate.ts +39 -39
  197. package/src/browser/disassembly-view/disassembly-view-utilities.ts +55 -55
  198. package/src/browser/disassembly-view/disassembly-view-widget.ts +463 -463
  199. package/src/browser/editor/debug-breakpoint-widget.tsx +275 -275
  200. package/src/browser/editor/debug-editor-model.ts +508 -508
  201. package/src/browser/editor/debug-editor-service.ts +206 -206
  202. package/src/browser/editor/debug-editor.ts +20 -20
  203. package/src/browser/editor/debug-exception-widget.tsx +118 -118
  204. package/src/browser/editor/debug-expression-provider.ts +78 -78
  205. package/src/browser/editor/debug-hover-source.tsx +105 -105
  206. package/src/browser/editor/debug-hover-widget.ts +280 -280
  207. package/src/browser/editor/debug-inline-value-decorator.ts +377 -377
  208. package/src/browser/model/debug-breakpoint.tsx +151 -151
  209. package/src/browser/model/debug-function-breakpoint.tsx +101 -101
  210. package/src/browser/model/debug-instruction-breakpoint.tsx +68 -68
  211. package/src/browser/model/debug-source-breakpoint.tsx +237 -237
  212. package/src/browser/model/debug-source.ts +93 -93
  213. package/src/browser/model/debug-stack-frame.tsx +170 -170
  214. package/src/browser/model/debug-thread.tsx +240 -240
  215. package/src/browser/preferences/launch-preferences.ts +38 -38
  216. package/src/browser/style/index.css +445 -445
  217. package/src/browser/view/debug-action.tsx +57 -57
  218. package/src/browser/view/debug-breakpoints-source.tsx +53 -53
  219. package/src/browser/view/debug-breakpoints-widget.ts +71 -71
  220. package/src/browser/view/debug-configuration-select.tsx +256 -256
  221. package/src/browser/view/debug-configuration-widget.tsx +121 -121
  222. package/src/browser/view/debug-exception-breakpoint.tsx +43 -43
  223. package/src/browser/view/debug-session-widget.ts +124 -124
  224. package/src/browser/view/debug-stack-frames-source.tsx +75 -75
  225. package/src/browser/view/debug-stack-frames-widget.ts +135 -135
  226. package/src/browser/view/debug-threads-source.tsx +48 -48
  227. package/src/browser/view/debug-threads-widget.ts +125 -125
  228. package/src/browser/view/debug-toolbar-widget.tsx +145 -145
  229. package/src/browser/view/debug-variables-source.ts +43 -43
  230. package/src/browser/view/debug-variables-widget.ts +61 -61
  231. package/src/browser/view/debug-view-model.ts +230 -230
  232. package/src/browser/view/debug-watch-expression.tsx +88 -88
  233. package/src/browser/view/debug-watch-source.ts +41 -41
  234. package/src/browser/view/debug-watch-widget.ts +61 -61
  235. package/src/browser/view/debug-widget.ts +97 -97
  236. package/src/common/debug-adapter-contribution-registry.ts +206 -206
  237. package/src/common/debug-adapter-session.ts +102 -102
  238. package/src/common/debug-common.ts +19 -19
  239. package/src/common/debug-compound.ts +33 -33
  240. package/src/common/debug-configuration.ts +93 -93
  241. package/src/common/debug-model.ts +200 -200
  242. package/src/common/debug-service.ts +184 -184
  243. package/src/common/debug-uri-utils.ts +24 -24
  244. package/src/common/inline-debug-adapter.ts +47 -47
  245. package/src/node/debug-adapter-factory.ts +107 -107
  246. package/src/node/debug-adapter-session-manager.ts +106 -106
  247. package/src/node/debug-backend-module.ts +57 -57
  248. package/src/node/debug-service-impl.ts +119 -119
  249. package/src/node/stream-debug-adapter.ts +126 -126
  250. package/src/node/vscode/vscode-debug-adapter-contribution.ts +252 -252
@@ -1,377 +1,377 @@
1
- // *****************************************************************************
2
- // Copyright (C) 2020 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 WITH Classpath-exception-2.0
15
- // *****************************************************************************
16
-
17
- /*---------------------------------------------------------------------------------------------
18
- * Copyright (c) Microsoft Corporation. All rights reserved.
19
- * Licensed under the MIT License. See License.txt in the project root for license information.
20
- *--------------------------------------------------------------------------------------------*/
21
- // Based on https://github.com/theia-ide/vscode/blob/standalone/0.19.x/src/vs/workbench/contrib/debug/browser/debugEditorContribution.ts
22
-
23
- import { FrontendApplicationContribution } from '@theia/core/lib/browser/frontend-application';
24
- import { inject, injectable } from '@theia/core/shared/inversify';
25
- import * as monaco from '@theia/monaco-editor-core';
26
- import { CancellationTokenSource } from '@theia/monaco-editor-core/esm/vs/base/common/cancellation';
27
- import { DEFAULT_WORD_REGEXP } from '@theia/monaco-editor-core/esm/vs/editor/common/core/wordHelper';
28
- import { IDecorationOptions } from '@theia/monaco-editor-core/esm/vs/editor/common/editorCommon';
29
- import { StandardTokenType } from '@theia/monaco-editor-core/esm/vs/editor/common/encodedTokenAttributes';
30
- import { InlineValueContext } from '@theia/monaco-editor-core/esm/vs/editor/common/languages';
31
- import { ITextModel } from '@theia/monaco-editor-core/esm/vs/editor/common/model';
32
- import { ILanguageFeaturesService } from '@theia/monaco-editor-core/esm/vs/editor/common/services/languageFeatures';
33
- import { StandaloneServices } from '@theia/monaco-editor-core/esm/vs/editor/standalone/browser/standaloneServices';
34
- import { MonacoEditorService } from '@theia/monaco/lib/browser/monaco-editor-service';
35
- import { DebugVariable, ExpressionContainer, ExpressionItem } from '../console/debug-console-items';
36
- import { DebugPreferences } from '../debug-preferences';
37
- import { DebugStackFrame } from '../model/debug-stack-frame';
38
- import { DebugEditorModel } from './debug-editor-model';
39
-
40
- // https://github.com/theia-ide/vscode/blob/standalone/0.19.x/src/vs/workbench/contrib/debug/browser/debugEditorContribution.ts#L40-L43
41
- export const INLINE_VALUE_DECORATION_KEY = 'inlinevaluedecoration';
42
- const MAX_NUM_INLINE_VALUES = 100; // JS Global scope can have 700+ entries. We want to limit ourselves for perf reasons
43
- const MAX_INLINE_DECORATOR_LENGTH = 150; // Max string length of each inline decorator when debugging. If exceeded ... is added
44
- const MAX_TOKENIZATION_LINE_LEN = 500; // If line is too long, then inline values for the line are skipped
45
-
46
- /**
47
- * MAX SMI (SMall Integer) as defined in v8.
48
- * one bit is lost for boxing/unboxing flag.
49
- * one bit is lost for sign flag.
50
- * See https://thibaultlaurens.github.io/javascript/2013/04/29/how-the-v8-engine-works/#tagged-values
51
- */
52
- // https://github.com/theia-ide/vscode/blob/standalone/0.19.x/src/vs/base/common/uint.ts#L7-L13
53
- const MAX_SAFE_SMALL_INTEGER = 1 << 30;
54
-
55
- class InlineSegment {
56
- constructor(public column: number, public text: string) {
57
- }
58
- }
59
-
60
- @injectable()
61
- export class DebugInlineValueDecorator implements FrontendApplicationContribution {
62
-
63
- @inject(MonacoEditorService)
64
- protected readonly editorService: MonacoEditorService;
65
-
66
- @inject(DebugPreferences)
67
- protected readonly preferences: DebugPreferences;
68
-
69
- protected enabled = false;
70
- protected wordToLineNumbersMap: Map<string, monaco.Position[]> | undefined = new Map();
71
-
72
- onStart(): void {
73
- this.editorService.registerDecorationType('Inline debug decorations', INLINE_VALUE_DECORATION_KEY, {});
74
- this.enabled = !!this.preferences['debug.inlineValues'];
75
- this.preferences.onPreferenceChanged(({ preferenceName, newValue }) => {
76
- if (preferenceName === 'debug.inlineValues' && !!newValue !== this.enabled) {
77
- this.enabled = !!newValue;
78
- }
79
- });
80
- }
81
-
82
- async calculateDecorations(debugEditorModel: DebugEditorModel, stackFrame: DebugStackFrame | undefined): Promise<IDecorationOptions[]> {
83
- this.wordToLineNumbersMap = undefined;
84
- const model = debugEditorModel.editor.getControl().getModel() || undefined;
85
- return this.updateInlineValueDecorations(debugEditorModel, model, stackFrame);
86
- }
87
-
88
- // https://github.com/theia-ide/vscode/blob/standalone/0.19.x/src/vs/workbench/contrib/debug/browser/debugEditorContribution.ts#L382-L408
89
- protected async updateInlineValueDecorations(
90
- debugEditorModel: DebugEditorModel,
91
- model: monaco.editor.ITextModel | undefined,
92
- stackFrame: DebugStackFrame | undefined): Promise<IDecorationOptions[]> {
93
-
94
- if (!this.enabled || !model || !stackFrame || !stackFrame.source || model.uri.toString() !== stackFrame.source.uri.toString()) {
95
- return [];
96
- }
97
-
98
- // XXX: Here is a difference between the VS Code's `IStackFrame` and the `DebugProtocol.StackFrame`.
99
- // In DAP, `source` is optional, hence `range` is optional too.
100
- const { range: stackFrameRange } = stackFrame;
101
- if (!stackFrameRange) {
102
- return [];
103
- }
104
-
105
- const scopes = await stackFrame.getMostSpecificScopes(stackFrameRange);
106
- // Get all top level children in the scope chain
107
- const decorationsPerScope = await Promise.all(scopes.map(async scope => {
108
- const children = Array.from(await scope.getElements());
109
- let range = new monaco.Range(0, 0, stackFrameRange.startLineNumber, stackFrameRange.startColumn);
110
- if (scope.range) {
111
- range = range.setStartPosition(scope.range.startLineNumber, scope.range.startColumn);
112
- }
113
-
114
- return this.createInlineValueDecorationsInsideRange(children, range, model, debugEditorModel, stackFrame);
115
- }));
116
-
117
- return decorationsPerScope.reduce((previous, current) => previous.concat(current), []);
118
- }
119
-
120
- // https://github.com/theia-ide/vscode/blob/standalone/0.19.x/src/vs/workbench/contrib/debug/browser/debugEditorContribution.ts#L410-L452
121
- private async createInlineValueDecorationsInsideRange(
122
- expressions: ReadonlyArray<ExpressionContainer>,
123
- range: monaco.Range,
124
- model: monaco.editor.ITextModel,
125
- debugEditorModel: DebugEditorModel,
126
- stackFrame: DebugStackFrame): Promise<IDecorationOptions[]> {
127
-
128
- const decorations: IDecorationOptions[] = [];
129
-
130
- const inlineValuesProvider = StandaloneServices.get(ILanguageFeaturesService).inlineValuesProvider;
131
- const textEditorModel = debugEditorModel.editor.document.textEditorModel;
132
-
133
- if (inlineValuesProvider && inlineValuesProvider.has(textEditorModel)) {
134
-
135
- const findVariable = async (variableName: string, caseSensitiveLookup: boolean): Promise<DebugVariable | undefined> => {
136
- const scopes = await stackFrame.getMostSpecificScopes(stackFrame.range!);
137
- const key = caseSensitiveLookup ? variableName : variableName.toLowerCase();
138
- for (const scope of scopes) {
139
- const expressionContainers = await scope.getElements();
140
- let container = expressionContainers.next();
141
- while (!container.done) {
142
- const debugVariable = container.value;
143
- if (debugVariable && debugVariable instanceof DebugVariable) {
144
- if (caseSensitiveLookup) {
145
- if (debugVariable.name === key) {
146
- return debugVariable;
147
- }
148
- } else {
149
- if (debugVariable.name.toLowerCase() === key) {
150
- return debugVariable;
151
- }
152
- }
153
- }
154
- container = expressionContainers.next();
155
- }
156
- }
157
- return undefined;
158
- };
159
-
160
- const context: InlineValueContext = {
161
- frameId: stackFrame.raw.id,
162
- stoppedLocation: range
163
- };
164
-
165
- const cancellationToken = new CancellationTokenSource().token;
166
- const registeredProviders = inlineValuesProvider.ordered(textEditorModel).reverse();
167
- const visibleRanges = debugEditorModel.editor.getControl().getVisibleRanges();
168
-
169
- const lineDecorations = new Map<number, InlineSegment[]>();
170
-
171
- for (const provider of registeredProviders) {
172
- for (const visibleRange of visibleRanges) {
173
- const result = await provider.provideInlineValues(textEditorModel, visibleRange, context, cancellationToken);
174
- if (result) {
175
- for (const inlineValue of result) {
176
- let text: string | undefined = undefined;
177
- switch (inlineValue.type) {
178
- case 'text':
179
- text = inlineValue.text;
180
- break;
181
- case 'variable': {
182
- let varName = inlineValue.variableName;
183
- if (!varName) {
184
- const lineContent = model.getLineContent(inlineValue.range.startLineNumber);
185
- varName = lineContent.substring(inlineValue.range.startColumn - 1, inlineValue.range.endColumn - 1);
186
- }
187
- const variable = await findVariable(varName, inlineValue.caseSensitiveLookup);
188
- if (variable) {
189
- text = this.formatInlineValue(varName, variable.value);
190
- }
191
- break;
192
- }
193
- case 'expression': {
194
- let expr = inlineValue.expression;
195
- if (!expr) {
196
- const lineContent = model.getLineContent(inlineValue.range.startLineNumber);
197
- expr = lineContent.substring(inlineValue.range.startColumn - 1, inlineValue.range.endColumn - 1);
198
- }
199
- if (expr) {
200
- const expression = new ExpressionItem(expr, () => stackFrame.thread.session);
201
- await expression.evaluate('watch');
202
- if (expression.available) {
203
- text = this.formatInlineValue(expr, expression.value);
204
- }
205
- }
206
- break;
207
- }
208
- }
209
-
210
- if (text) {
211
- const line = inlineValue.range.startLineNumber;
212
- let lineSegments = lineDecorations.get(line);
213
- if (!lineSegments) {
214
- lineSegments = [];
215
- lineDecorations.set(line, lineSegments);
216
- }
217
- if (!lineSegments.some(segment => segment.text === text)) {
218
- lineSegments.push(new InlineSegment(inlineValue.range.startColumn, text));
219
- }
220
- }
221
- }
222
- }
223
- }
224
- };
225
-
226
- // sort line segments and concatenate them into a decoration
227
- const separator = ', ';
228
- lineDecorations.forEach((segments, line) => {
229
- if (segments.length > 0) {
230
- segments = segments.sort((a, b) => a.column - b.column);
231
- const text = segments.map(s => s.text).join(separator);
232
- decorations.push(this.createInlineValueDecoration(line, text));
233
- }
234
- });
235
-
236
- } else { // use fallback if no provider was registered
237
- const lineToNamesMap: Map<number, string[]> = new Map<number, string[]>();
238
- const nameValueMap = new Map<string, string>();
239
- for (const expr of expressions) {
240
- if (expr instanceof DebugVariable) { // XXX: VS Code uses `IExpression` that has `name` and `value`.
241
- nameValueMap.set(expr.name, expr.value);
242
- }
243
- // Limit the size of map. Too large can have a perf impact
244
- if (nameValueMap.size >= MAX_NUM_INLINE_VALUES) {
245
- break;
246
- }
247
- }
248
-
249
- const wordToPositionsMap = this.getWordToPositionsMap(model);
250
-
251
- // Compute unique set of names on each line
252
- nameValueMap.forEach((_, name) => {
253
- const positions = wordToPositionsMap.get(name);
254
- if (positions) {
255
- for (const position of positions) {
256
- if (range.containsPosition(position)) {
257
- if (!lineToNamesMap.has(position.lineNumber)) {
258
- lineToNamesMap.set(position.lineNumber, []);
259
- }
260
-
261
- if (lineToNamesMap.get(position.lineNumber)!.indexOf(name) === -1) {
262
- lineToNamesMap.get(position.lineNumber)!.push(name);
263
- }
264
- }
265
- }
266
- }
267
- });
268
-
269
- // Compute decorators for each line
270
- lineToNamesMap.forEach((names, line) => {
271
- const contentText = names.sort((first, second) => {
272
- const content = model.getLineContent(line);
273
- return content.indexOf(first) - content.indexOf(second);
274
- }).map(name => `${name} = ${nameValueMap.get(name)}`).join(', ');
275
- decorations.push(this.createInlineValueDecoration(line, contentText));
276
- });
277
- }
278
-
279
- return decorations;
280
- }
281
-
282
- protected formatInlineValue(...args: string[]): string {
283
- const valuePattern = '{0} = {1}';
284
- const formatRegExp = /{(\d+)}/g;
285
- if (args.length === 0) {
286
- return valuePattern;
287
- }
288
- return valuePattern.replace(formatRegExp, (match, group) => {
289
- const idx = parseInt(group, 10);
290
- return isNaN(idx) || idx < 0 || idx >= args.length ?
291
- match :
292
- args[idx];
293
- });
294
- }
295
-
296
- // https://github.com/theia-ide/vscode/blob/standalone/0.19.x/src/vs/workbench/contrib/debug/browser/debugEditorContribution.ts#L454-L485
297
- private createInlineValueDecoration(lineNumber: number, contentText: string): IDecorationOptions {
298
- // If decoratorText is too long, trim and add ellipses. This could happen for minified files with everything on a single line
299
- if (contentText.length > MAX_INLINE_DECORATOR_LENGTH) {
300
- contentText = contentText.substring(0, MAX_INLINE_DECORATOR_LENGTH) + '...';
301
- }
302
-
303
- return {
304
- range: {
305
- startLineNumber: lineNumber,
306
- endLineNumber: lineNumber,
307
- startColumn: MAX_SAFE_SMALL_INTEGER,
308
- endColumn: MAX_SAFE_SMALL_INTEGER
309
- },
310
- renderOptions: {
311
- after: {
312
- contentText,
313
- backgroundColor: 'rgba(255, 200, 0, 0.2)',
314
- margin: '10px'
315
- },
316
- dark: {
317
- after: {
318
- color: 'rgba(255, 255, 255, 0.5)',
319
- }
320
- },
321
- light: {
322
- after: {
323
- color: 'rgba(0, 0, 0, 0.5)',
324
- }
325
- }
326
- }
327
- };
328
- }
329
-
330
- // https://github.com/theia-ide/vscode/blob/standalone/0.19.x/src/vs/workbench/contrib/debug/browser/debugEditorContribution.ts#L487-L531
331
- private getWordToPositionsMap(model: monaco.editor.ITextModel | ITextModel): Map<string, monaco.Position[]> {
332
- model = model as ITextModel;
333
- if (!this.wordToLineNumbersMap) {
334
- this.wordToLineNumbersMap = new Map<string, monaco.Position[]>();
335
- if (!model) {
336
- return this.wordToLineNumbersMap;
337
- }
338
-
339
- // For every word in every line, map its ranges for fast lookup
340
- for (let lineNumber = 1, len = model.getLineCount(); lineNumber <= len; ++lineNumber) {
341
- const lineContent = model.getLineContent(lineNumber);
342
-
343
- // If line is too long then skip the line
344
- if (lineContent.length > MAX_TOKENIZATION_LINE_LEN) {
345
- continue;
346
- }
347
-
348
- model.tokenization.forceTokenization(lineNumber);
349
- const lineTokens = model.tokenization.getLineTokens(lineNumber);
350
- for (let tokenIndex = 0, tokenCount = lineTokens.getCount(); tokenIndex < tokenCount; tokenIndex++) {
351
- const tokenStartOffset = lineTokens.getStartOffset(tokenIndex);
352
- const tokenEndOffset = lineTokens.getEndOffset(tokenIndex);
353
- const tokenType = lineTokens.getStandardTokenType(tokenIndex);
354
- const tokenStr = lineContent.substring(tokenStartOffset, tokenEndOffset);
355
-
356
- // Token is a word and not a comment
357
- if (tokenType === StandardTokenType.Other) {
358
- DEFAULT_WORD_REGEXP.lastIndex = 0; // We assume tokens will usually map 1:1 to words if they match
359
- const wordMatch = DEFAULT_WORD_REGEXP.exec(tokenStr);
360
-
361
- if (wordMatch) {
362
- const word = wordMatch[0];
363
- if (!this.wordToLineNumbersMap.has(word)) {
364
- this.wordToLineNumbersMap.set(word, []);
365
- }
366
-
367
- this.wordToLineNumbersMap.get(word)!.push(new monaco.Position(lineNumber, tokenStartOffset));
368
- }
369
- }
370
- }
371
- }
372
- }
373
-
374
- return this.wordToLineNumbersMap;
375
- }
376
-
377
- }
1
+ // *****************************************************************************
2
+ // Copyright (C) 2020 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 WITH Classpath-exception-2.0
15
+ // *****************************************************************************
16
+
17
+ /*---------------------------------------------------------------------------------------------
18
+ * Copyright (c) Microsoft Corporation. All rights reserved.
19
+ * Licensed under the MIT License. See License.txt in the project root for license information.
20
+ *--------------------------------------------------------------------------------------------*/
21
+ // Based on https://github.com/theia-ide/vscode/blob/standalone/0.19.x/src/vs/workbench/contrib/debug/browser/debugEditorContribution.ts
22
+
23
+ import { FrontendApplicationContribution } from '@theia/core/lib/browser/frontend-application';
24
+ import { inject, injectable } from '@theia/core/shared/inversify';
25
+ import * as monaco from '@theia/monaco-editor-core';
26
+ import { CancellationTokenSource } from '@theia/monaco-editor-core/esm/vs/base/common/cancellation';
27
+ import { DEFAULT_WORD_REGEXP } from '@theia/monaco-editor-core/esm/vs/editor/common/core/wordHelper';
28
+ import { IDecorationOptions } from '@theia/monaco-editor-core/esm/vs/editor/common/editorCommon';
29
+ import { StandardTokenType } from '@theia/monaco-editor-core/esm/vs/editor/common/encodedTokenAttributes';
30
+ import { InlineValueContext } from '@theia/monaco-editor-core/esm/vs/editor/common/languages';
31
+ import { ITextModel } from '@theia/monaco-editor-core/esm/vs/editor/common/model';
32
+ import { ILanguageFeaturesService } from '@theia/monaco-editor-core/esm/vs/editor/common/services/languageFeatures';
33
+ import { StandaloneServices } from '@theia/monaco-editor-core/esm/vs/editor/standalone/browser/standaloneServices';
34
+ import { MonacoEditorService } from '@theia/monaco/lib/browser/monaco-editor-service';
35
+ import { DebugVariable, ExpressionContainer, ExpressionItem } from '../console/debug-console-items';
36
+ import { DebugPreferences } from '../debug-preferences';
37
+ import { DebugStackFrame } from '../model/debug-stack-frame';
38
+ import { DebugEditorModel } from './debug-editor-model';
39
+
40
+ // https://github.com/theia-ide/vscode/blob/standalone/0.19.x/src/vs/workbench/contrib/debug/browser/debugEditorContribution.ts#L40-L43
41
+ export const INLINE_VALUE_DECORATION_KEY = 'inlinevaluedecoration';
42
+ const MAX_NUM_INLINE_VALUES = 100; // JS Global scope can have 700+ entries. We want to limit ourselves for perf reasons
43
+ const MAX_INLINE_DECORATOR_LENGTH = 150; // Max string length of each inline decorator when debugging. If exceeded ... is added
44
+ const MAX_TOKENIZATION_LINE_LEN = 500; // If line is too long, then inline values for the line are skipped
45
+
46
+ /**
47
+ * MAX SMI (SMall Integer) as defined in v8.
48
+ * one bit is lost for boxing/unboxing flag.
49
+ * one bit is lost for sign flag.
50
+ * See https://thibaultlaurens.github.io/javascript/2013/04/29/how-the-v8-engine-works/#tagged-values
51
+ */
52
+ // https://github.com/theia-ide/vscode/blob/standalone/0.19.x/src/vs/base/common/uint.ts#L7-L13
53
+ const MAX_SAFE_SMALL_INTEGER = 1 << 30;
54
+
55
+ class InlineSegment {
56
+ constructor(public column: number, public text: string) {
57
+ }
58
+ }
59
+
60
+ @injectable()
61
+ export class DebugInlineValueDecorator implements FrontendApplicationContribution {
62
+
63
+ @inject(MonacoEditorService)
64
+ protected readonly editorService: MonacoEditorService;
65
+
66
+ @inject(DebugPreferences)
67
+ protected readonly preferences: DebugPreferences;
68
+
69
+ protected enabled = false;
70
+ protected wordToLineNumbersMap: Map<string, monaco.Position[]> | undefined = new Map();
71
+
72
+ onStart(): void {
73
+ this.editorService.registerDecorationType('Inline debug decorations', INLINE_VALUE_DECORATION_KEY, {});
74
+ this.enabled = !!this.preferences['debug.inlineValues'];
75
+ this.preferences.onPreferenceChanged(({ preferenceName, newValue }) => {
76
+ if (preferenceName === 'debug.inlineValues' && !!newValue !== this.enabled) {
77
+ this.enabled = !!newValue;
78
+ }
79
+ });
80
+ }
81
+
82
+ async calculateDecorations(debugEditorModel: DebugEditorModel, stackFrame: DebugStackFrame | undefined): Promise<IDecorationOptions[]> {
83
+ this.wordToLineNumbersMap = undefined;
84
+ const model = debugEditorModel.editor.getControl().getModel() || undefined;
85
+ return this.updateInlineValueDecorations(debugEditorModel, model, stackFrame);
86
+ }
87
+
88
+ // https://github.com/theia-ide/vscode/blob/standalone/0.19.x/src/vs/workbench/contrib/debug/browser/debugEditorContribution.ts#L382-L408
89
+ protected async updateInlineValueDecorations(
90
+ debugEditorModel: DebugEditorModel,
91
+ model: monaco.editor.ITextModel | undefined,
92
+ stackFrame: DebugStackFrame | undefined): Promise<IDecorationOptions[]> {
93
+
94
+ if (!this.enabled || !model || !stackFrame || !stackFrame.source || model.uri.toString() !== stackFrame.source.uri.toString()) {
95
+ return [];
96
+ }
97
+
98
+ // XXX: Here is a difference between the VS Code's `IStackFrame` and the `DebugProtocol.StackFrame`.
99
+ // In DAP, `source` is optional, hence `range` is optional too.
100
+ const { range: stackFrameRange } = stackFrame;
101
+ if (!stackFrameRange) {
102
+ return [];
103
+ }
104
+
105
+ const scopes = await stackFrame.getMostSpecificScopes(stackFrameRange);
106
+ // Get all top level children in the scope chain
107
+ const decorationsPerScope = await Promise.all(scopes.map(async scope => {
108
+ const children = Array.from(await scope.getElements());
109
+ let range = new monaco.Range(0, 0, stackFrameRange.startLineNumber, stackFrameRange.startColumn);
110
+ if (scope.range) {
111
+ range = range.setStartPosition(scope.range.startLineNumber, scope.range.startColumn);
112
+ }
113
+
114
+ return this.createInlineValueDecorationsInsideRange(children, range, model, debugEditorModel, stackFrame);
115
+ }));
116
+
117
+ return decorationsPerScope.reduce((previous, current) => previous.concat(current), []);
118
+ }
119
+
120
+ // https://github.com/theia-ide/vscode/blob/standalone/0.19.x/src/vs/workbench/contrib/debug/browser/debugEditorContribution.ts#L410-L452
121
+ private async createInlineValueDecorationsInsideRange(
122
+ expressions: ReadonlyArray<ExpressionContainer>,
123
+ range: monaco.Range,
124
+ model: monaco.editor.ITextModel,
125
+ debugEditorModel: DebugEditorModel,
126
+ stackFrame: DebugStackFrame): Promise<IDecorationOptions[]> {
127
+
128
+ const decorations: IDecorationOptions[] = [];
129
+
130
+ const inlineValuesProvider = StandaloneServices.get(ILanguageFeaturesService).inlineValuesProvider;
131
+ const textEditorModel = debugEditorModel.editor.document.textEditorModel;
132
+
133
+ if (inlineValuesProvider && inlineValuesProvider.has(textEditorModel)) {
134
+
135
+ const findVariable = async (variableName: string, caseSensitiveLookup: boolean): Promise<DebugVariable | undefined> => {
136
+ const scopes = await stackFrame.getMostSpecificScopes(stackFrame.range!);
137
+ const key = caseSensitiveLookup ? variableName : variableName.toLowerCase();
138
+ for (const scope of scopes) {
139
+ const expressionContainers = await scope.getElements();
140
+ let container = expressionContainers.next();
141
+ while (!container.done) {
142
+ const debugVariable = container.value;
143
+ if (debugVariable && debugVariable instanceof DebugVariable) {
144
+ if (caseSensitiveLookup) {
145
+ if (debugVariable.name === key) {
146
+ return debugVariable;
147
+ }
148
+ } else {
149
+ if (debugVariable.name.toLowerCase() === key) {
150
+ return debugVariable;
151
+ }
152
+ }
153
+ }
154
+ container = expressionContainers.next();
155
+ }
156
+ }
157
+ return undefined;
158
+ };
159
+
160
+ const context: InlineValueContext = {
161
+ frameId: stackFrame.raw.id,
162
+ stoppedLocation: range
163
+ };
164
+
165
+ const cancellationToken = new CancellationTokenSource().token;
166
+ const registeredProviders = inlineValuesProvider.ordered(textEditorModel).reverse();
167
+ const visibleRanges = debugEditorModel.editor.getControl().getVisibleRanges();
168
+
169
+ const lineDecorations = new Map<number, InlineSegment[]>();
170
+
171
+ for (const provider of registeredProviders) {
172
+ for (const visibleRange of visibleRanges) {
173
+ const result = await provider.provideInlineValues(textEditorModel, visibleRange, context, cancellationToken);
174
+ if (result) {
175
+ for (const inlineValue of result) {
176
+ let text: string | undefined = undefined;
177
+ switch (inlineValue.type) {
178
+ case 'text':
179
+ text = inlineValue.text;
180
+ break;
181
+ case 'variable': {
182
+ let varName = inlineValue.variableName;
183
+ if (!varName) {
184
+ const lineContent = model.getLineContent(inlineValue.range.startLineNumber);
185
+ varName = lineContent.substring(inlineValue.range.startColumn - 1, inlineValue.range.endColumn - 1);
186
+ }
187
+ const variable = await findVariable(varName, inlineValue.caseSensitiveLookup);
188
+ if (variable) {
189
+ text = this.formatInlineValue(varName, variable.value);
190
+ }
191
+ break;
192
+ }
193
+ case 'expression': {
194
+ let expr = inlineValue.expression;
195
+ if (!expr) {
196
+ const lineContent = model.getLineContent(inlineValue.range.startLineNumber);
197
+ expr = lineContent.substring(inlineValue.range.startColumn - 1, inlineValue.range.endColumn - 1);
198
+ }
199
+ if (expr) {
200
+ const expression = new ExpressionItem(expr, () => stackFrame.thread.session);
201
+ await expression.evaluate('watch');
202
+ if (expression.available) {
203
+ text = this.formatInlineValue(expr, expression.value);
204
+ }
205
+ }
206
+ break;
207
+ }
208
+ }
209
+
210
+ if (text) {
211
+ const line = inlineValue.range.startLineNumber;
212
+ let lineSegments = lineDecorations.get(line);
213
+ if (!lineSegments) {
214
+ lineSegments = [];
215
+ lineDecorations.set(line, lineSegments);
216
+ }
217
+ if (!lineSegments.some(segment => segment.text === text)) {
218
+ lineSegments.push(new InlineSegment(inlineValue.range.startColumn, text));
219
+ }
220
+ }
221
+ }
222
+ }
223
+ }
224
+ };
225
+
226
+ // sort line segments and concatenate them into a decoration
227
+ const separator = ', ';
228
+ lineDecorations.forEach((segments, line) => {
229
+ if (segments.length > 0) {
230
+ segments = segments.sort((a, b) => a.column - b.column);
231
+ const text = segments.map(s => s.text).join(separator);
232
+ decorations.push(this.createInlineValueDecoration(line, text));
233
+ }
234
+ });
235
+
236
+ } else { // use fallback if no provider was registered
237
+ const lineToNamesMap: Map<number, string[]> = new Map<number, string[]>();
238
+ const nameValueMap = new Map<string, string>();
239
+ for (const expr of expressions) {
240
+ if (expr instanceof DebugVariable) { // XXX: VS Code uses `IExpression` that has `name` and `value`.
241
+ nameValueMap.set(expr.name, expr.value);
242
+ }
243
+ // Limit the size of map. Too large can have a perf impact
244
+ if (nameValueMap.size >= MAX_NUM_INLINE_VALUES) {
245
+ break;
246
+ }
247
+ }
248
+
249
+ const wordToPositionsMap = this.getWordToPositionsMap(model);
250
+
251
+ // Compute unique set of names on each line
252
+ nameValueMap.forEach((_, name) => {
253
+ const positions = wordToPositionsMap.get(name);
254
+ if (positions) {
255
+ for (const position of positions) {
256
+ if (range.containsPosition(position)) {
257
+ if (!lineToNamesMap.has(position.lineNumber)) {
258
+ lineToNamesMap.set(position.lineNumber, []);
259
+ }
260
+
261
+ if (lineToNamesMap.get(position.lineNumber)!.indexOf(name) === -1) {
262
+ lineToNamesMap.get(position.lineNumber)!.push(name);
263
+ }
264
+ }
265
+ }
266
+ }
267
+ });
268
+
269
+ // Compute decorators for each line
270
+ lineToNamesMap.forEach((names, line) => {
271
+ const contentText = names.sort((first, second) => {
272
+ const content = model.getLineContent(line);
273
+ return content.indexOf(first) - content.indexOf(second);
274
+ }).map(name => `${name} = ${nameValueMap.get(name)}`).join(', ');
275
+ decorations.push(this.createInlineValueDecoration(line, contentText));
276
+ });
277
+ }
278
+
279
+ return decorations;
280
+ }
281
+
282
+ protected formatInlineValue(...args: string[]): string {
283
+ const valuePattern = '{0} = {1}';
284
+ const formatRegExp = /{(\d+)}/g;
285
+ if (args.length === 0) {
286
+ return valuePattern;
287
+ }
288
+ return valuePattern.replace(formatRegExp, (match, group) => {
289
+ const idx = parseInt(group, 10);
290
+ return isNaN(idx) || idx < 0 || idx >= args.length ?
291
+ match :
292
+ args[idx];
293
+ });
294
+ }
295
+
296
+ // https://github.com/theia-ide/vscode/blob/standalone/0.19.x/src/vs/workbench/contrib/debug/browser/debugEditorContribution.ts#L454-L485
297
+ private createInlineValueDecoration(lineNumber: number, contentText: string): IDecorationOptions {
298
+ // If decoratorText is too long, trim and add ellipses. This could happen for minified files with everything on a single line
299
+ if (contentText.length > MAX_INLINE_DECORATOR_LENGTH) {
300
+ contentText = contentText.substring(0, MAX_INLINE_DECORATOR_LENGTH) + '...';
301
+ }
302
+
303
+ return {
304
+ range: {
305
+ startLineNumber: lineNumber,
306
+ endLineNumber: lineNumber,
307
+ startColumn: MAX_SAFE_SMALL_INTEGER,
308
+ endColumn: MAX_SAFE_SMALL_INTEGER
309
+ },
310
+ renderOptions: {
311
+ after: {
312
+ contentText,
313
+ backgroundColor: 'rgba(255, 200, 0, 0.2)',
314
+ margin: '10px'
315
+ },
316
+ dark: {
317
+ after: {
318
+ color: 'rgba(255, 255, 255, 0.5)',
319
+ }
320
+ },
321
+ light: {
322
+ after: {
323
+ color: 'rgba(0, 0, 0, 0.5)',
324
+ }
325
+ }
326
+ }
327
+ };
328
+ }
329
+
330
+ // https://github.com/theia-ide/vscode/blob/standalone/0.19.x/src/vs/workbench/contrib/debug/browser/debugEditorContribution.ts#L487-L531
331
+ private getWordToPositionsMap(model: monaco.editor.ITextModel | ITextModel): Map<string, monaco.Position[]> {
332
+ model = model as ITextModel;
333
+ if (!this.wordToLineNumbersMap) {
334
+ this.wordToLineNumbersMap = new Map<string, monaco.Position[]>();
335
+ if (!model) {
336
+ return this.wordToLineNumbersMap;
337
+ }
338
+
339
+ // For every word in every line, map its ranges for fast lookup
340
+ for (let lineNumber = 1, len = model.getLineCount(); lineNumber <= len; ++lineNumber) {
341
+ const lineContent = model.getLineContent(lineNumber);
342
+
343
+ // If line is too long then skip the line
344
+ if (lineContent.length > MAX_TOKENIZATION_LINE_LEN) {
345
+ continue;
346
+ }
347
+
348
+ model.tokenization.forceTokenization(lineNumber);
349
+ const lineTokens = model.tokenization.getLineTokens(lineNumber);
350
+ for (let tokenIndex = 0, tokenCount = lineTokens.getCount(); tokenIndex < tokenCount; tokenIndex++) {
351
+ const tokenStartOffset = lineTokens.getStartOffset(tokenIndex);
352
+ const tokenEndOffset = lineTokens.getEndOffset(tokenIndex);
353
+ const tokenType = lineTokens.getStandardTokenType(tokenIndex);
354
+ const tokenStr = lineContent.substring(tokenStartOffset, tokenEndOffset);
355
+
356
+ // Token is a word and not a comment
357
+ if (tokenType === StandardTokenType.Other) {
358
+ DEFAULT_WORD_REGEXP.lastIndex = 0; // We assume tokens will usually map 1:1 to words if they match
359
+ const wordMatch = DEFAULT_WORD_REGEXP.exec(tokenStr);
360
+
361
+ if (wordMatch) {
362
+ const word = wordMatch[0];
363
+ if (!this.wordToLineNumbersMap.has(word)) {
364
+ this.wordToLineNumbersMap.set(word, []);
365
+ }
366
+
367
+ this.wordToLineNumbersMap.get(word)!.push(new monaco.Position(lineNumber, tokenStartOffset));
368
+ }
369
+ }
370
+ }
371
+ }
372
+ }
373
+
374
+ return this.wordToLineNumbersMap;
375
+ }
376
+
377
+ }