@theia/debug 1.45.0 → 1.46.0-next.72

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 (251) hide show
  1. package/README.md +62 -62
  2. package/lib/browser/breakpoint/breakpoint-manager.d.ts +70 -70
  3. package/lib/browser/breakpoint/breakpoint-manager.js +328 -328
  4. package/lib/browser/breakpoint/breakpoint-marker.d.ts +41 -41
  5. package/lib/browser/breakpoint/breakpoint-marker.js +87 -87
  6. package/lib/browser/console/debug-console-contribution.d.ts +40 -40
  7. package/lib/browser/console/debug-console-contribution.js +231 -231
  8. package/lib/browser/console/debug-console-items.d.ts +93 -93
  9. package/lib/browser/console/debug-console-items.js +319 -319
  10. package/lib/browser/console/debug-console-session.d.ts +31 -31
  11. package/lib/browser/console/debug-console-session.js +199 -199
  12. package/lib/browser/debug-call-stack-item-type-key.d.ts +3 -3
  13. package/lib/browser/debug-call-stack-item-type-key.js +19 -19
  14. package/lib/browser/debug-configuration-manager.d.ts +102 -102
  15. package/lib/browser/debug-configuration-manager.js +552 -552
  16. package/lib/browser/debug-configuration-model.d.ts +29 -29
  17. package/lib/browser/debug-configuration-model.js +79 -79
  18. package/lib/browser/debug-contribution.d.ts +22 -22
  19. package/lib/browser/debug-contribution.js +19 -19
  20. package/lib/browser/debug-frontend-application-contribution.d.ts +222 -222
  21. package/lib/browser/debug-frontend-application-contribution.js +1404 -1404
  22. package/lib/browser/debug-frontend-module.d.ts +4 -4
  23. package/lib/browser/debug-frontend-module.d.ts.map +1 -1
  24. package/lib/browser/debug-frontend-module.js +105 -104
  25. package/lib/browser/debug-frontend-module.js.map +1 -1
  26. package/lib/browser/debug-package.spec.js +18 -18
  27. package/lib/browser/debug-preferences.d.ts +17 -17
  28. package/lib/browser/debug-preferences.js +89 -89
  29. package/lib/browser/debug-prefix-configuration.d.ts +51 -51
  30. package/lib/browser/debug-prefix-configuration.js +208 -208
  31. package/lib/browser/debug-resource.d.ts +14 -14
  32. package/lib/browser/debug-resource.js +65 -65
  33. package/lib/browser/debug-schema-updater.d.ts +15 -15
  34. package/lib/browser/debug-schema-updater.js +168 -168
  35. package/lib/browser/debug-session-connection.d.ts +110 -110
  36. package/lib/browser/debug-session-connection.js +258 -258
  37. package/lib/browser/debug-session-contribution.d.ts +73 -73
  38. package/lib/browser/debug-session-contribution.js +145 -145
  39. package/lib/browser/debug-session-manager.d.ts +137 -137
  40. package/lib/browser/debug-session-manager.js +599 -599
  41. package/lib/browser/debug-session-manager.js.map +1 -1
  42. package/lib/browser/debug-session-options.d.ts +48 -48
  43. package/lib/browser/debug-session-options.js +85 -85
  44. package/lib/browser/debug-session.d.ts +178 -178
  45. package/lib/browser/debug-session.js +871 -871
  46. package/lib/browser/debug-tab-bar-decorator.d.ts +16 -16
  47. package/lib/browser/debug-tab-bar-decorator.js +70 -70
  48. package/lib/browser/debug-watch-manager.d.ts +21 -21
  49. package/lib/browser/debug-watch-manager.js +93 -93
  50. package/lib/browser/disassembly-view/disassembly-view-accessibility-provider.d.ts +6 -6
  51. package/lib/browser/disassembly-view/disassembly-view-accessibility-provider.js +38 -38
  52. package/lib/browser/disassembly-view/disassembly-view-breakpoint-renderer.d.ts +19 -19
  53. package/lib/browser/disassembly-view/disassembly-view-breakpoint-renderer.js +110 -110
  54. package/lib/browser/disassembly-view/disassembly-view-contribution.d.ts +24 -24
  55. package/lib/browser/disassembly-view/disassembly-view-contribution.js +131 -131
  56. package/lib/browser/disassembly-view/disassembly-view-instruction-renderer.d.ts +38 -38
  57. package/lib/browser/disassembly-view/disassembly-view-instruction-renderer.js +222 -222
  58. package/lib/browser/disassembly-view/disassembly-view-instruction-renderer.js.map +1 -1
  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 +453 -452
  65. package/lib/browser/disassembly-view/disassembly-view-widget.js.map +1 -1
  66. package/lib/browser/editor/debug-breakpoint-widget.d.ts +53 -53
  67. package/lib/browser/editor/debug-breakpoint-widget.js +261 -261
  68. package/lib/browser/editor/debug-editor-model.d.ts +91 -93
  69. package/lib/browser/editor/debug-editor-model.d.ts.map +1 -1
  70. package/lib/browser/editor/debug-editor-model.js +515 -516
  71. package/lib/browser/editor/debug-editor-model.js.map +1 -1
  72. package/lib/browser/editor/debug-editor-service.d.ts +35 -35
  73. package/lib/browser/editor/debug-editor-service.js +202 -202
  74. package/lib/browser/editor/debug-editor.d.ts +3 -3
  75. package/lib/browser/editor/debug-editor.js +19 -19
  76. package/lib/browser/editor/debug-exception-widget.d.ts +29 -29
  77. package/lib/browser/editor/debug-exception-widget.js +114 -114
  78. package/lib/browser/editor/debug-expression-provider.d.ts +11 -11
  79. package/lib/browser/editor/debug-expression-provider.js +81 -81
  80. package/lib/browser/editor/debug-hover-source.d.ts +18 -18
  81. package/lib/browser/editor/debug-hover-source.js +117 -117
  82. package/lib/browser/editor/debug-hover-widget.d.ts +47 -47
  83. package/lib/browser/editor/debug-hover-widget.js +276 -276
  84. package/lib/browser/editor/debug-inline-value-decorator.d.ts +19 -21
  85. package/lib/browser/editor/debug-inline-value-decorator.d.ts.map +1 -1
  86. package/lib/browser/editor/debug-inline-value-decorator.js +333 -337
  87. package/lib/browser/editor/debug-inline-value-decorator.js.map +1 -1
  88. package/lib/browser/model/debug-breakpoint.d.ts +47 -47
  89. package/lib/browser/model/debug-breakpoint.js +109 -109
  90. package/lib/browser/model/debug-function-breakpoint.d.ts +18 -18
  91. package/lib/browser/model/debug-function-breakpoint.js +92 -92
  92. package/lib/browser/model/debug-instruction-breakpoint.d.ts +14 -14
  93. package/lib/browser/model/debug-instruction-breakpoint.js +65 -65
  94. package/lib/browser/model/debug-source-breakpoint.d.ts +37 -37
  95. package/lib/browser/model/debug-source-breakpoint.js +217 -217
  96. package/lib/browser/model/debug-source.d.ts +24 -24
  97. package/lib/browser/model/debug-source.js +80 -80
  98. package/lib/browser/model/debug-stack-frame.d.ts +41 -41
  99. package/lib/browser/model/debug-stack-frame.js +150 -150
  100. package/lib/browser/model/debug-thread.d.ts +62 -62
  101. package/lib/browser/model/debug-thread.js +237 -237
  102. package/lib/browser/preferences/launch-preferences.d.ts +4 -4
  103. package/lib/browser/preferences/launch-preferences.js +38 -38
  104. package/lib/browser/view/debug-action.d.ts +16 -16
  105. package/lib/browser/view/debug-action.js +43 -43
  106. package/lib/browser/view/debug-breakpoints-source.d.ts +9 -9
  107. package/lib/browser/view/debug-breakpoints-source.js +70 -70
  108. package/lib/browser/view/debug-breakpoints-widget.d.ts +22 -22
  109. package/lib/browser/view/debug-breakpoints-widget.js +87 -87
  110. package/lib/browser/view/debug-configuration-select.d.ts +64 -64
  111. package/lib/browser/view/debug-configuration-select.js +216 -216
  112. package/lib/browser/view/debug-configuration-widget.d.ts +30 -30
  113. package/lib/browser/view/debug-configuration-widget.js +135 -135
  114. package/lib/browser/view/debug-exception-breakpoint.d.ts +14 -14
  115. package/lib/browser/view/debug-exception-breakpoint.js +61 -61
  116. package/lib/browser/view/debug-session-widget.d.ts +28 -28
  117. package/lib/browser/view/debug-session-widget.js +134 -134
  118. package/lib/browser/view/debug-stack-frames-source.d.ts +17 -17
  119. package/lib/browser/view/debug-stack-frames-source.js +89 -89
  120. package/lib/browser/view/debug-stack-frames-widget.d.ts +24 -24
  121. package/lib/browser/view/debug-stack-frames-widget.js +154 -154
  122. package/lib/browser/view/debug-threads-source.d.ts +8 -8
  123. package/lib/browser/view/debug-threads-source.js +63 -63
  124. package/lib/browser/view/debug-threads-widget.d.ts +26 -26
  125. package/lib/browser/view/debug-threads-widget.js +145 -145
  126. package/lib/browser/view/debug-toolbar-widget.d.ts +34 -34
  127. package/lib/browser/view/debug-toolbar-widget.js +156 -156
  128. package/lib/browser/view/debug-variables-source.d.ts +9 -9
  129. package/lib/browser/view/debug-variables-source.js +60 -60
  130. package/lib/browser/view/debug-variables-widget.d.ts +16 -16
  131. package/lib/browser/view/debug-variables-widget.js +78 -78
  132. package/lib/browser/view/debug-view-model.d.ts +51 -51
  133. package/lib/browser/view/debug-view-model.js +220 -220
  134. package/lib/browser/view/debug-watch-expression.d.ts +30 -30
  135. package/lib/browser/view/debug-watch-expression.js +73 -73
  136. package/lib/browser/view/debug-watch-source.d.ts +9 -9
  137. package/lib/browser/view/debug-watch-source.js +58 -58
  138. package/lib/browser/view/debug-watch-widget.d.ts +16 -16
  139. package/lib/browser/view/debug-watch-widget.js +78 -78
  140. package/lib/browser/view/debug-widget.d.ts +23 -23
  141. package/lib/browser/view/debug-widget.js +108 -108
  142. package/lib/common/debug-adapter-contribution-registry.d.ts +59 -59
  143. package/lib/common/debug-adapter-contribution-registry.js +212 -212
  144. package/lib/common/debug-adapter-session.d.ts +18 -18
  145. package/lib/common/debug-adapter-session.js +78 -78
  146. package/lib/common/debug-common.d.ts +1 -1
  147. package/lib/common/debug-common.js +29 -29
  148. package/lib/common/debug-compound.d.ts +14 -14
  149. package/lib/common/debug-compound.js +27 -27
  150. package/lib/common/debug-configuration.d.ts +70 -70
  151. package/lib/common/debug-configuration.js +31 -31
  152. package/lib/common/debug-model.d.ts +152 -152
  153. package/lib/common/debug-model.js +35 -35
  154. package/lib/common/debug-service.d.ts +123 -123
  155. package/lib/common/debug-service.js +66 -66
  156. package/lib/common/debug-uri-utils.d.ts +23 -23
  157. package/lib/common/debug-uri-utils.js +26 -26
  158. package/lib/common/inline-debug-adapter.d.ts +18 -18
  159. package/lib/common/inline-debug-adapter.js +44 -44
  160. package/lib/node/debug-adapter-factory.d.ts +19 -19
  161. package/lib/node/debug-adapter-factory.js +96 -96
  162. package/lib/node/debug-adapter-session-manager.d.ts +37 -37
  163. package/lib/node/debug-adapter-session-manager.js +111 -111
  164. package/lib/node/debug-backend-module.d.ts +3 -3
  165. package/lib/node/debug-backend-module.js +41 -41
  166. package/lib/node/debug-service-impl.d.ts +30 -30
  167. package/lib/node/debug-service-impl.js +124 -124
  168. package/lib/node/stream-debug-adapter.d.ts +36 -36
  169. package/lib/node/stream-debug-adapter.js +112 -112
  170. package/package.json +16 -16
  171. package/src/browser/breakpoint/breakpoint-manager.ts +369 -369
  172. package/src/browser/breakpoint/breakpoint-marker.ts +104 -104
  173. package/src/browser/console/debug-console-contribution.tsx +240 -240
  174. package/src/browser/console/debug-console-items.tsx +384 -384
  175. package/src/browser/console/debug-console-session.ts +205 -205
  176. package/src/browser/debug-call-stack-item-type-key.ts +20 -20
  177. package/src/browser/debug-configuration-manager.ts +591 -591
  178. package/src/browser/debug-configuration-model.ts +100 -100
  179. package/src/browser/debug-contribution.ts +43 -43
  180. package/src/browser/debug-frontend-application-contribution.ts +1551 -1551
  181. package/src/browser/debug-frontend-module.ts +133 -132
  182. package/src/browser/debug-package.spec.ts +20 -20
  183. package/src/browser/debug-preferences.ts +98 -98
  184. package/src/browser/debug-prefix-configuration.ts +195 -195
  185. package/src/browser/debug-resource.ts +59 -59
  186. package/src/browser/debug-schema-updater.ts +149 -149
  187. package/src/browser/debug-session-connection.ts +357 -357
  188. package/src/browser/debug-session-contribution.ts +150 -150
  189. package/src/browser/debug-session-manager.ts +682 -682
  190. package/src/browser/debug-session-options.ts +114 -114
  191. package/src/browser/debug-session.tsx +947 -947
  192. package/src/browser/debug-tab-bar-decorator.ts +57 -57
  193. package/src/browser/debug-watch-manager.ts +93 -93
  194. package/src/browser/disassembly-view/disassembly-view-accessibility-provider.ts +43 -43
  195. package/src/browser/disassembly-view/disassembly-view-breakpoint-renderer.ts +119 -119
  196. package/src/browser/disassembly-view/disassembly-view-contribution.ts +109 -109
  197. package/src/browser/disassembly-view/disassembly-view-instruction-renderer.ts +245 -245
  198. package/src/browser/disassembly-view/disassembly-view-table-delegate.ts +39 -39
  199. package/src/browser/disassembly-view/disassembly-view-utilities.ts +55 -55
  200. package/src/browser/disassembly-view/disassembly-view-widget.ts +463 -463
  201. package/src/browser/editor/debug-breakpoint-widget.tsx +293 -293
  202. package/src/browser/editor/debug-editor-model.ts +529 -528
  203. package/src/browser/editor/debug-editor-service.ts +192 -192
  204. package/src/browser/editor/debug-editor.ts +20 -20
  205. package/src/browser/editor/debug-exception-widget.tsx +122 -122
  206. package/src/browser/editor/debug-expression-provider.ts +78 -78
  207. package/src/browser/editor/debug-hover-source.tsx +105 -105
  208. package/src/browser/editor/debug-hover-widget.ts +298 -298
  209. package/src/browser/editor/debug-inline-value-decorator.ts +373 -377
  210. package/src/browser/model/debug-breakpoint.tsx +151 -151
  211. package/src/browser/model/debug-function-breakpoint.tsx +101 -101
  212. package/src/browser/model/debug-instruction-breakpoint.tsx +68 -68
  213. package/src/browser/model/debug-source-breakpoint.tsx +237 -237
  214. package/src/browser/model/debug-source.ts +93 -93
  215. package/src/browser/model/debug-stack-frame.tsx +170 -170
  216. package/src/browser/model/debug-thread.tsx +285 -285
  217. package/src/browser/preferences/launch-preferences.ts +38 -38
  218. package/src/browser/style/index.css +453 -453
  219. package/src/browser/view/debug-action.tsx +57 -57
  220. package/src/browser/view/debug-breakpoints-source.tsx +53 -53
  221. package/src/browser/view/debug-breakpoints-widget.ts +71 -71
  222. package/src/browser/view/debug-configuration-select.tsx +269 -269
  223. package/src/browser/view/debug-configuration-widget.tsx +121 -121
  224. package/src/browser/view/debug-exception-breakpoint.tsx +68 -68
  225. package/src/browser/view/debug-session-widget.ts +124 -124
  226. package/src/browser/view/debug-stack-frames-source.tsx +75 -75
  227. package/src/browser/view/debug-stack-frames-widget.ts +135 -135
  228. package/src/browser/view/debug-threads-source.tsx +48 -48
  229. package/src/browser/view/debug-threads-widget.ts +125 -125
  230. package/src/browser/view/debug-toolbar-widget.tsx +145 -145
  231. package/src/browser/view/debug-variables-source.ts +43 -43
  232. package/src/browser/view/debug-variables-widget.ts +61 -61
  233. package/src/browser/view/debug-view-model.ts +230 -230
  234. package/src/browser/view/debug-watch-expression.tsx +88 -88
  235. package/src/browser/view/debug-watch-source.ts +41 -41
  236. package/src/browser/view/debug-watch-widget.ts +61 -61
  237. package/src/browser/view/debug-widget.ts +97 -97
  238. package/src/common/debug-adapter-contribution-registry.ts +206 -206
  239. package/src/common/debug-adapter-session.ts +102 -102
  240. package/src/common/debug-common.ts +19 -19
  241. package/src/common/debug-compound.ts +33 -33
  242. package/src/common/debug-configuration.ts +108 -108
  243. package/src/common/debug-model.ts +200 -200
  244. package/src/common/debug-service.ts +184 -184
  245. package/src/common/debug-uri-utils.ts +24 -24
  246. package/src/common/inline-debug-adapter.ts +47 -47
  247. package/src/node/debug-adapter-factory.ts +107 -107
  248. package/src/node/debug-adapter-session-manager.ts +106 -106
  249. package/src/node/debug-backend-module.ts +57 -57
  250. package/src/node/debug-service-impl.ts +119 -119
  251. package/src/node/stream-debug-adapter.ts +126 -126
@@ -1,528 +1,529 @@
1
- // *****************************************************************************
2
- // Copyright (C) 2018 TypeFox and others.
3
- //
4
- // This program and the accompanying materials are made available under the
5
- // terms of the Eclipse Public License v. 2.0 which is available at
6
- // http://www.eclipse.org/legal/epl-2.0.
7
- //
8
- // This Source Code may also be made available under the following Secondary
9
- // Licenses when the conditions for such availability set forth in the Eclipse
10
- // Public License v. 2.0 are satisfied: GNU General Public License, version 2
11
- // with the GNU Classpath Exception which is available at
12
- // https://www.gnu.org/software/classpath/license.html.
13
- //
14
- // SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
15
- // *****************************************************************************
16
-
17
- import debounce = require('p-debounce');
18
- import { injectable, inject, postConstruct, interfaces, Container } from '@theia/core/shared/inversify';
19
- import * as monaco from '@theia/monaco-editor-core';
20
- import { IConfigurationService } from '@theia/monaco-editor-core/esm/vs/platform/configuration/common/configuration';
21
- import { StandaloneCodeEditor } from '@theia/monaco-editor-core/esm/vs/editor/standalone/browser/standaloneCodeEditor';
22
- import { IDecorationOptions } from '@theia/monaco-editor-core/esm/vs/editor/common/editorCommon';
23
- import { IEditorHoverOptions } from '@theia/monaco-editor-core/esm/vs/editor/common/config/editorOptions';
24
- import URI from '@theia/core/lib/common/uri';
25
- import { Disposable, DisposableCollection, MenuPath, isOSX } from '@theia/core';
26
- import { ContextMenuRenderer } from '@theia/core/lib/browser';
27
- import { MonacoConfigurationService } from '@theia/monaco/lib/browser/monaco-frontend-module';
28
- import { BreakpointManager, SourceBreakpointsChangeEvent } from '../breakpoint/breakpoint-manager';
29
- import { DebugSourceBreakpoint } from '../model/debug-source-breakpoint';
30
- import { DebugSessionManager } from '../debug-session-manager';
31
- import { SourceBreakpoint } from '../breakpoint/breakpoint-marker';
32
- import { DebugEditor } from './debug-editor';
33
- import { DebugHoverWidget, createDebugHoverWidgetContainer } from './debug-hover-widget';
34
- import { DebugBreakpointWidget } from './debug-breakpoint-widget';
35
- import { DebugExceptionWidget } from './debug-exception-widget';
36
- import { DebugProtocol } from '@vscode/debugprotocol';
37
- import { DebugInlineValueDecorator, INLINE_VALUE_DECORATION_KEY } from './debug-inline-value-decorator';
38
-
39
- export const DebugEditorModelFactory = Symbol('DebugEditorModelFactory');
40
- export type DebugEditorModelFactory = (editor: DebugEditor) => DebugEditorModel;
41
-
42
- @injectable()
43
- export class DebugEditorModel implements Disposable {
44
-
45
- static createContainer(parent: interfaces.Container, editor: DebugEditor): Container {
46
- const child = createDebugHoverWidgetContainer(parent, editor);
47
- child.bind(DebugEditorModel).toSelf();
48
- child.bind(DebugBreakpointWidget).toSelf();
49
- child.bind(DebugExceptionWidget).toSelf();
50
- return child;
51
- }
52
- static createModel(parent: interfaces.Container, editor: DebugEditor): DebugEditorModel {
53
- return DebugEditorModel.createContainer(parent, editor).get(DebugEditorModel);
54
- }
55
-
56
- static CONTEXT_MENU: MenuPath = ['debug-editor-context-menu'];
57
-
58
- protected readonly toDispose = new DisposableCollection();
59
- protected readonly toDisposeOnUpdate = new DisposableCollection();
60
-
61
- protected uri: URI;
62
-
63
- protected breakpointDecorations: string[] = [];
64
- protected breakpointRanges = new Map<string, [monaco.Range, SourceBreakpoint]>();
65
-
66
- protected currentBreakpointDecorations: string[] = [];
67
-
68
- protected editorDecorations: string[] = [];
69
- protected topFrameRange: monaco.Range | undefined;
70
-
71
- protected updatingDecorations = false;
72
-
73
- @inject(DebugHoverWidget)
74
- readonly hover: DebugHoverWidget;
75
-
76
- @inject(DebugEditor)
77
- readonly editor: DebugEditor;
78
-
79
- @inject(BreakpointManager)
80
- readonly breakpoints: BreakpointManager;
81
-
82
- @inject(DebugSessionManager)
83
- readonly sessions: DebugSessionManager;
84
-
85
- @inject(ContextMenuRenderer)
86
- readonly contextMenu: ContextMenuRenderer;
87
-
88
- @inject(DebugBreakpointWidget)
89
- readonly breakpointWidget: DebugBreakpointWidget;
90
-
91
- @inject(DebugExceptionWidget)
92
- readonly exceptionWidget: DebugExceptionWidget;
93
-
94
- @inject(DebugInlineValueDecorator)
95
- readonly inlineValueDecorator: DebugInlineValueDecorator;
96
-
97
- @inject(MonacoConfigurationService)
98
- readonly configurationService: IConfigurationService;
99
-
100
- @inject(DebugSessionManager)
101
- protected readonly sessionManager: DebugSessionManager;
102
-
103
- @postConstruct()
104
- protected init(): void {
105
- this.uri = new URI(this.editor.getControl().getModel()!.uri.toString());
106
- this.toDispose.pushAll([
107
- this.hover,
108
- this.breakpointWidget,
109
- this.exceptionWidget,
110
- this.editor.getControl().onMouseDown(event => this.handleMouseDown(event)),
111
- this.editor.getControl().onMouseMove(event => this.handleMouseMove(event)),
112
- this.editor.getControl().onMouseLeave(event => this.handleMouseLeave(event)),
113
- this.editor.getControl().onKeyDown(() => this.hover.hide({ immediate: false })),
114
- this.editor.getControl().onDidChangeModelContent(() => this.update()),
115
- this.editor.getControl().getModel()!.onDidChangeDecorations(() => this.updateBreakpoints()),
116
- this.editor.onDidResize(e => this.breakpointWidget.inputSize = e),
117
- this.sessions.onDidChange(() => this.update()),
118
- this.toDisposeOnUpdate,
119
- this.sessionManager.onDidChangeBreakpoints(({ session, uri }) => {
120
- if ((!session || session === this.sessionManager.currentSession) && uri.isEqual(this.uri)) {
121
- this.render();
122
- }
123
- }),
124
- this.breakpoints.onDidChangeBreakpoints(event => this.closeBreakpointIfAffected(event)),
125
- ]);
126
- this.update();
127
- this.render();
128
- }
129
-
130
- dispose(): void {
131
- this.toDispose.dispose();
132
- }
133
-
134
- protected readonly update = debounce(async () => {
135
- if (this.toDispose.disposed) {
136
- return;
137
- }
138
- this.toDisposeOnUpdate.dispose();
139
- this.toggleExceptionWidget();
140
- await this.updateEditorDecorations();
141
- this.updateEditorHover();
142
- }, 100);
143
-
144
- /**
145
- * To disable the default editor-contribution hover from Code when
146
- * the editor has the `currentFrame`. Otherwise, both `textdocument/hover`
147
- * and the debug hovers are visible at the same time when hovering over a symbol.
148
- */
149
- protected async updateEditorHover(): Promise<void> {
150
- if (this.sessions.isCurrentEditorFrame(this.uri)) {
151
- const codeEditor = this.editor.getControl();
152
- codeEditor.updateOptions({ hover: { enabled: false } });
153
- this.toDisposeOnUpdate.push(Disposable.create(() => {
154
- const model = codeEditor.getModel()!;
155
- const overrides = {
156
- resource: model.uri,
157
- overrideIdentifier: model.getLanguageId(),
158
- };
159
- const { enabled, delay, sticky } = this.configurationService.getValue<IEditorHoverOptions>('editor.hover', overrides);
160
- codeEditor.updateOptions({
161
- hover: {
162
- enabled,
163
- delay,
164
- sticky
165
- }
166
- });
167
- }));
168
- }
169
- }
170
-
171
- protected async updateEditorDecorations(): Promise<void> {
172
- const [newFrameDecorations, inlineValueDecorations] = await Promise.all([
173
- this.createFrameDecorations(),
174
- this.createInlineValueDecorations()
175
- ]);
176
- const codeEditor = this.editor.getControl() as unknown as StandaloneCodeEditor;
177
- codeEditor.removeDecorations([INLINE_VALUE_DECORATION_KEY]);
178
- codeEditor.setDecorationsByType('Inline debug decorations', INLINE_VALUE_DECORATION_KEY, inlineValueDecorations);
179
- this.editorDecorations = this.deltaDecorations(this.editorDecorations, newFrameDecorations);
180
- }
181
-
182
- protected async createInlineValueDecorations(): Promise<IDecorationOptions[]> {
183
- if (!this.sessions.isCurrentEditorFrame(this.uri)) {
184
- return [];
185
- }
186
- const { currentFrame } = this.sessions;
187
- return this.inlineValueDecorator.calculateDecorations(this, currentFrame);
188
- }
189
-
190
- protected createFrameDecorations(): monaco.editor.IModelDeltaDecoration[] {
191
- const { currentFrame, topFrame } = this.sessions;
192
- if (!currentFrame) {
193
- return [];
194
- }
195
-
196
- if (!this.sessions.isCurrentEditorFrame(this.uri)) {
197
- return [];
198
- }
199
-
200
- const decorations: monaco.editor.IModelDeltaDecoration[] = [];
201
- const columnUntilEOLRange = new monaco.Range(currentFrame.raw.line, currentFrame.raw.column, currentFrame.raw.line, 1 << 30);
202
- const range = new monaco.Range(currentFrame.raw.line, currentFrame.raw.column, currentFrame.raw.line, currentFrame.raw.column + 1);
203
-
204
- if (topFrame === currentFrame) {
205
- decorations.push({
206
- options: DebugEditorModel.TOP_STACK_FRAME_MARGIN,
207
- range
208
- });
209
- decorations.push({
210
- options: DebugEditorModel.TOP_STACK_FRAME_DECORATION,
211
- range: columnUntilEOLRange
212
- });
213
- const { topFrameRange } = this;
214
- if (topFrameRange && topFrameRange.startLineNumber === currentFrame.raw.line && topFrameRange.startColumn !== currentFrame.raw.column) {
215
- decorations.push({
216
- options: DebugEditorModel.TOP_STACK_FRAME_INLINE_DECORATION,
217
- range: columnUntilEOLRange
218
- });
219
- }
220
- this.topFrameRange = columnUntilEOLRange;
221
- } else {
222
- decorations.push({
223
- options: DebugEditorModel.FOCUSED_STACK_FRAME_MARGIN,
224
- range
225
- });
226
- decorations.push({
227
- options: DebugEditorModel.FOCUSED_STACK_FRAME_DECORATION,
228
- range: columnUntilEOLRange
229
- });
230
- }
231
- return decorations;
232
- }
233
-
234
- protected async toggleExceptionWidget(): Promise<void> {
235
- const { currentFrame } = this.sessions;
236
- if (!currentFrame) {
237
- return;
238
- }
239
- if (!this.sessions.isCurrentEditorFrame(this.uri)) {
240
- this.exceptionWidget.hide();
241
- return;
242
- }
243
- const info = await currentFrame.thread.getExceptionInfo();
244
- if (!info) {
245
- this.exceptionWidget.hide();
246
- return;
247
- }
248
- this.exceptionWidget.show({
249
- info,
250
- lineNumber: currentFrame.raw.line,
251
- column: currentFrame.raw.column
252
- });
253
- }
254
-
255
- render(): void {
256
- this.renderBreakpoints();
257
- this.renderCurrentBreakpoints();
258
- }
259
- protected renderBreakpoints(): void {
260
- const breakpoints = this.breakpoints.getBreakpoints(this.uri);
261
- const decorations = this.createBreakpointDecorations(breakpoints);
262
- this.breakpointDecorations = this.deltaDecorations(this.breakpointDecorations, decorations);
263
- this.updateBreakpointRanges(breakpoints);
264
- }
265
- protected createBreakpointDecorations(breakpoints: SourceBreakpoint[]): monaco.editor.IModelDeltaDecoration[] {
266
- return breakpoints.map(breakpoint => this.createBreakpointDecoration(breakpoint));
267
- }
268
- protected createBreakpointDecoration(breakpoint: SourceBreakpoint): monaco.editor.IModelDeltaDecoration {
269
- const lineNumber = breakpoint.raw.line;
270
- const column = breakpoint.raw.column;
271
- const range = typeof column === 'number' ? new monaco.Range(lineNumber, column, lineNumber, column + 1) : new monaco.Range(lineNumber, 1, lineNumber, 2);
272
- return {
273
- range,
274
- options: {
275
- stickiness: DebugEditorModel.STICKINESS
276
- }
277
- };
278
- }
279
-
280
- protected updateBreakpointRanges(breakpoints: SourceBreakpoint[]): void {
281
- this.breakpointRanges.clear();
282
- for (let i = 0; i < this.breakpointDecorations.length; i++) {
283
- const decoration = this.breakpointDecorations[i];
284
- const breakpoint = breakpoints[i];
285
- const range = this.editor.getControl().getModel()!.getDecorationRange(decoration)!;
286
- this.breakpointRanges.set(decoration, [range, breakpoint]);
287
- }
288
- }
289
-
290
- protected renderCurrentBreakpoints(): void {
291
- const decorations = this.createCurrentBreakpointDecorations();
292
- this.currentBreakpointDecorations = this.deltaDecorations(this.currentBreakpointDecorations, decorations);
293
- }
294
- protected createCurrentBreakpointDecorations(): monaco.editor.IModelDeltaDecoration[] {
295
- const breakpoints = this.sessions.getBreakpoints(this.uri);
296
- return breakpoints.map(breakpoint => this.createCurrentBreakpointDecoration(breakpoint));
297
- }
298
- protected createCurrentBreakpointDecoration(breakpoint: DebugSourceBreakpoint): monaco.editor.IModelDeltaDecoration {
299
- const lineNumber = breakpoint.line;
300
- const column = breakpoint.column;
301
- const range = typeof column === 'number' ? new monaco.Range(lineNumber, column, lineNumber, column + 1) : new monaco.Range(lineNumber, 1, lineNumber, 1);
302
- const { className, message } = breakpoint.getDecoration();
303
- const renderInline = typeof column === 'number' && (column > this.editor.getControl().getModel()!.getLineFirstNonWhitespaceColumn(lineNumber));
304
- return {
305
- range,
306
- options: {
307
- glyphMarginClassName: className,
308
- glyphMarginHoverMessage: message.map(value => ({ value })),
309
- stickiness: DebugEditorModel.STICKINESS,
310
- beforeContentClassName: renderInline ? `theia-debug-breakpoint-column codicon ${className}` : undefined
311
- }
312
- };
313
- }
314
-
315
- protected updateBreakpoints(): void {
316
- if (this.areBreakpointsAffected()) {
317
- const breakpoints = this.createBreakpoints();
318
- this.breakpoints.setBreakpoints(this.uri, breakpoints);
319
- }
320
- }
321
- protected areBreakpointsAffected(): boolean {
322
- if (this.updatingDecorations || !this.editor.getControl().getModel()) {
323
- return false;
324
- }
325
- for (const decoration of this.breakpointDecorations) {
326
- const range = this.editor.getControl().getModel()!.getDecorationRange(decoration);
327
- const oldRange = this.breakpointRanges.get(decoration)![0];
328
- if (!range || !range.equalsRange(oldRange)) {
329
- return true;
330
- }
331
- }
332
- return false;
333
- }
334
- protected createBreakpoints(): SourceBreakpoint[] {
335
- const { uri } = this;
336
- const lines = new Set<number>();
337
- const breakpoints: SourceBreakpoint[] = [];
338
- for (const decoration of this.breakpointDecorations) {
339
- const range = this.editor.getControl().getModel()!.getDecorationRange(decoration);
340
- if (range && !lines.has(range.startLineNumber)) {
341
- const line = range.startLineNumber;
342
- const column = range.startColumn;
343
- const oldBreakpoint = this.breakpointRanges.get(decoration)?.[1];
344
- const isLineBreakpoint = oldBreakpoint?.raw.line !== undefined && oldBreakpoint?.raw.column === undefined;
345
- const change = isLineBreakpoint ? { line } : { line, column };
346
- const breakpoint = SourceBreakpoint.create(uri, change, oldBreakpoint);
347
- breakpoints.push(breakpoint);
348
- lines.add(line);
349
- }
350
- }
351
- return breakpoints;
352
- }
353
-
354
- get position(): monaco.Position {
355
- return this.editor.getControl().getPosition()!;
356
- }
357
- getBreakpoint(position: monaco.Position = this.position): DebugSourceBreakpoint | undefined {
358
- return this.getInlineBreakpoint(position) || this.getLineBreakpoints(position)[0];
359
- }
360
-
361
- getInlineBreakpoint(position: monaco.Position = this.position): DebugSourceBreakpoint | undefined {
362
- return this.sessions.getInlineBreakpoint(this.uri, position.lineNumber, position.column);
363
- }
364
-
365
- protected getLineBreakpoints(position: monaco.Position = this.position): DebugSourceBreakpoint[] {
366
- return this.sessions.getLineBreakpoints(this.uri, position.lineNumber);
367
- }
368
-
369
- protected addBreakpoint(raw: DebugProtocol.SourceBreakpoint): void {
370
- this.breakpoints.addBreakpoint(SourceBreakpoint.create(this.uri, raw));
371
- }
372
-
373
- toggleBreakpoint(position: monaco.Position = this.position): void {
374
- const { lineNumber } = position;
375
- const breakpoints = this.getLineBreakpoints(position);
376
- if (breakpoints.length) {
377
- for (const breakpoint of breakpoints) {
378
- breakpoint.remove();
379
- }
380
- } else {
381
- this.addBreakpoint({ line: lineNumber });
382
- }
383
- }
384
-
385
- addInlineBreakpoint(): void {
386
- const { position } = this;
387
- const { lineNumber, column } = position;
388
- const breakpoint = this.getInlineBreakpoint(position);
389
- if (breakpoint) {
390
- return;
391
- }
392
- this.addBreakpoint({ line: lineNumber, column });
393
- }
394
-
395
- acceptBreakpoint(): void {
396
- const { position, values } = this.breakpointWidget;
397
- if (position && values) {
398
- const breakpoint = position.column > 0 ? this.getInlineBreakpoint(position) : this.getLineBreakpoints(position)[0];
399
- if (breakpoint) {
400
- breakpoint.updateOrigins(values);
401
- } else {
402
- const { lineNumber } = position;
403
- const column = position.column > 0 ? position.column : undefined;
404
- this.addBreakpoint({ line: lineNumber, column, ...values });
405
- }
406
- this.breakpointWidget.hide();
407
- }
408
- }
409
-
410
- protected handleMouseDown(event: monaco.editor.IEditorMouseEvent): void {
411
- if (event.target && event.target.type === monaco.editor.MouseTargetType.GUTTER_GLYPH_MARGIN) {
412
- if (!event.event.rightButton) {
413
- this.toggleBreakpoint(event.target.position!);
414
- }
415
- }
416
- this.hintBreakpoint(event);
417
- }
418
- protected handleMouseMove(event: monaco.editor.IEditorMouseEvent): void {
419
- this.showHover(event);
420
- this.hintBreakpoint(event);
421
- }
422
- protected handleMouseLeave(event: monaco.editor.IPartialEditorMouseEvent): void {
423
- this.hideHover(event);
424
- this.deltaHintDecorations([]);
425
- }
426
-
427
- protected hintDecorations: string[] = [];
428
- protected hintBreakpoint(event: monaco.editor.IEditorMouseEvent): void {
429
- const hintDecorations = this.createHintDecorations(event);
430
- this.deltaHintDecorations(hintDecorations);
431
- }
432
- protected deltaHintDecorations(hintDecorations: monaco.editor.IModelDeltaDecoration[]): void {
433
- this.hintDecorations = this.deltaDecorations(this.hintDecorations, hintDecorations);
434
- }
435
- protected createHintDecorations(event: monaco.editor.IEditorMouseEvent): monaco.editor.IModelDeltaDecoration[] {
436
- if (event.target && event.target.type === monaco.editor.MouseTargetType.GUTTER_GLYPH_MARGIN && event.target.position) {
437
- const lineNumber = event.target.position.lineNumber;
438
- if (this.getLineBreakpoints(event.target.position).length) {
439
- return [];
440
- }
441
- return [{
442
- range: new monaco.Range(lineNumber, 1, lineNumber, 1),
443
- options: DebugEditorModel.BREAKPOINT_HINT_DECORATION
444
- }];
445
- }
446
- return [];
447
- }
448
-
449
- protected closeBreakpointIfAffected({ uri, removed }: SourceBreakpointsChangeEvent): void {
450
- if (!uri.isEqual(this.uri)) {
451
- return;
452
- }
453
- const position = this.breakpointWidget.position;
454
- if (!position) {
455
- return;
456
- }
457
- for (const breakpoint of removed) {
458
- if (breakpoint.raw.line === position.lineNumber) {
459
- this.breakpointWidget.hide();
460
- break;
461
- }
462
- }
463
- }
464
-
465
- protected showHover(mouseEvent: monaco.editor.IEditorMouseEvent): void {
466
- const targetType = mouseEvent.target.type;
467
- const stopKey = isOSX ? 'metaKey' : 'ctrlKey';
468
-
469
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
470
- if (targetType === monaco.editor.MouseTargetType.CONTENT_WIDGET && mouseEvent.target.detail === this.hover.getId() && !(<any>mouseEvent.event)[stopKey]) {
471
- // mouse moved on top of debug hover widget
472
- return;
473
- }
474
- if (targetType === monaco.editor.MouseTargetType.CONTENT_TEXT) {
475
- this.hover.show({
476
- selection: mouseEvent.target.range!,
477
- immediate: false
478
- });
479
- } else {
480
- this.hover.hide({ immediate: false });
481
- }
482
- }
483
- protected hideHover({ event }: monaco.editor.IPartialEditorMouseEvent): void {
484
- const rect = this.hover.getDomNode().getBoundingClientRect();
485
- if (event.posx < rect.left || event.posx > rect.right || event.posy < rect.top || event.posy > rect.bottom) {
486
- this.hover.hide({ immediate: false });
487
- }
488
- }
489
-
490
- protected deltaDecorations(oldDecorations: string[], newDecorations: monaco.editor.IModelDeltaDecoration[]): string[] {
491
- this.updatingDecorations = true;
492
- try {
493
- return this.editor.getControl().deltaDecorations(oldDecorations, newDecorations);
494
- } finally {
495
- this.updatingDecorations = false;
496
- }
497
- }
498
-
499
- static STICKINESS = monaco.editor.TrackedRangeStickiness.NeverGrowsWhenTypingAtEdges;
500
-
501
- static BREAKPOINT_HINT_DECORATION: monaco.editor.IModelDecorationOptions = {
502
- glyphMarginClassName: 'codicon-debug-hint',
503
- stickiness: DebugEditorModel.STICKINESS
504
- };
505
-
506
- static TOP_STACK_FRAME_MARGIN: monaco.editor.IModelDecorationOptions = {
507
- glyphMarginClassName: 'codicon-debug-stackframe',
508
- stickiness: DebugEditorModel.STICKINESS
509
- };
510
- static FOCUSED_STACK_FRAME_MARGIN: monaco.editor.IModelDecorationOptions = {
511
- glyphMarginClassName: 'codicon-debug-stackframe-focused',
512
- stickiness: DebugEditorModel.STICKINESS
513
- };
514
- static TOP_STACK_FRAME_DECORATION: monaco.editor.IModelDecorationOptions = {
515
- isWholeLine: true,
516
- className: 'theia-debug-top-stack-frame-line',
517
- stickiness: DebugEditorModel.STICKINESS
518
- };
519
- static TOP_STACK_FRAME_INLINE_DECORATION: monaco.editor.IModelDecorationOptions = {
520
- beforeContentClassName: 'theia-debug-top-stack-frame-column'
521
- };
522
- static FOCUSED_STACK_FRAME_DECORATION: monaco.editor.IModelDecorationOptions = {
523
- isWholeLine: true,
524
- className: 'theia-debug-focused-stack-frame-line',
525
- stickiness: DebugEditorModel.STICKINESS
526
- };
527
-
528
- }
1
+ // *****************************************************************************
2
+ // Copyright (C) 2018 TypeFox and others.
3
+ //
4
+ // This program and the accompanying materials are made available under the
5
+ // terms of the Eclipse Public License v. 2.0 which is available at
6
+ // http://www.eclipse.org/legal/epl-2.0.
7
+ //
8
+ // This Source Code may also be made available under the following Secondary
9
+ // Licenses when the conditions for such availability set forth in the Eclipse
10
+ // Public License v. 2.0 are satisfied: GNU General Public License, version 2
11
+ // with the GNU Classpath Exception which is available at
12
+ // https://www.gnu.org/software/classpath/license.html.
13
+ //
14
+ // SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
15
+ // *****************************************************************************
16
+
17
+ import debounce = require('p-debounce');
18
+ import { injectable, inject, postConstruct, interfaces, Container } from '@theia/core/shared/inversify';
19
+ import * as monaco from '@theia/monaco-editor-core';
20
+ import { IConfigurationService } from '@theia/monaco-editor-core/esm/vs/platform/configuration/common/configuration';
21
+ import { StandaloneCodeEditor } from '@theia/monaco-editor-core/esm/vs/editor/standalone/browser/standaloneCodeEditor';
22
+ import { IDecorationOptions } from '@theia/monaco-editor-core/esm/vs/editor/common/editorCommon';
23
+ import { IEditorHoverOptions } from '@theia/monaco-editor-core/esm/vs/editor/common/config/editorOptions';
24
+ import URI from '@theia/core/lib/common/uri';
25
+ import { Disposable, DisposableCollection, MenuPath, isOSX } from '@theia/core';
26
+ import { ContextMenuRenderer } from '@theia/core/lib/browser';
27
+ import { BreakpointManager, SourceBreakpointsChangeEvent } from '../breakpoint/breakpoint-manager';
28
+ import { DebugSourceBreakpoint } from '../model/debug-source-breakpoint';
29
+ import { DebugSessionManager } from '../debug-session-manager';
30
+ import { SourceBreakpoint } from '../breakpoint/breakpoint-marker';
31
+ import { DebugEditor } from './debug-editor';
32
+ import { DebugHoverWidget, createDebugHoverWidgetContainer } from './debug-hover-widget';
33
+ import { DebugBreakpointWidget } from './debug-breakpoint-widget';
34
+ import { DebugExceptionWidget } from './debug-exception-widget';
35
+ import { DebugProtocol } from '@vscode/debugprotocol';
36
+ import { DebugInlineValueDecorator, INLINE_VALUE_DECORATION_KEY } from './debug-inline-value-decorator';
37
+ import { StandaloneServices } from '@theia/monaco-editor-core/esm/vs/editor/standalone/browser/standaloneServices';
38
+
39
+ export const DebugEditorModelFactory = Symbol('DebugEditorModelFactory');
40
+ export type DebugEditorModelFactory = (editor: DebugEditor) => DebugEditorModel;
41
+
42
+ @injectable()
43
+ export class DebugEditorModel implements Disposable {
44
+
45
+ static createContainer(parent: interfaces.Container, editor: DebugEditor): Container {
46
+ const child = createDebugHoverWidgetContainer(parent, editor);
47
+ child.bind(DebugEditorModel).toSelf();
48
+ child.bind(DebugBreakpointWidget).toSelf();
49
+ child.bind(DebugExceptionWidget).toSelf();
50
+ return child;
51
+ }
52
+ static createModel(parent: interfaces.Container, editor: DebugEditor): DebugEditorModel {
53
+ return DebugEditorModel.createContainer(parent, editor).get(DebugEditorModel);
54
+ }
55
+
56
+ static CONTEXT_MENU: MenuPath = ['debug-editor-context-menu'];
57
+
58
+ protected readonly toDispose = new DisposableCollection();
59
+ protected readonly toDisposeOnUpdate = new DisposableCollection();
60
+
61
+ protected uri: URI;
62
+
63
+ protected breakpointDecorations: string[] = [];
64
+ protected breakpointRanges = new Map<string, [monaco.Range, SourceBreakpoint]>();
65
+
66
+ protected currentBreakpointDecorations: string[] = [];
67
+
68
+ protected editorDecorations: string[] = [];
69
+ protected topFrameRange: monaco.Range | undefined;
70
+
71
+ protected updatingDecorations = false;
72
+
73
+ @inject(DebugHoverWidget)
74
+ readonly hover: DebugHoverWidget;
75
+
76
+ @inject(DebugEditor)
77
+ readonly editor: DebugEditor;
78
+
79
+ @inject(BreakpointManager)
80
+ readonly breakpoints: BreakpointManager;
81
+
82
+ @inject(DebugSessionManager)
83
+ readonly sessions: DebugSessionManager;
84
+
85
+ @inject(ContextMenuRenderer)
86
+ readonly contextMenu: ContextMenuRenderer;
87
+
88
+ @inject(DebugBreakpointWidget)
89
+ readonly breakpointWidget: DebugBreakpointWidget;
90
+
91
+ @inject(DebugExceptionWidget)
92
+ readonly exceptionWidget: DebugExceptionWidget;
93
+
94
+ @inject(DebugInlineValueDecorator)
95
+ readonly inlineValueDecorator: DebugInlineValueDecorator;
96
+
97
+ @inject(DebugSessionManager)
98
+ protected readonly sessionManager: DebugSessionManager;
99
+
100
+ @postConstruct()
101
+ protected init(): void {
102
+ this.uri = new URI(this.editor.getControl().getModel()!.uri.toString());
103
+ this.toDispose.pushAll([
104
+ this.hover,
105
+ this.breakpointWidget,
106
+ this.exceptionWidget,
107
+ this.editor.getControl().onMouseDown(event => this.handleMouseDown(event)),
108
+ this.editor.getControl().onMouseMove(event => this.handleMouseMove(event)),
109
+ this.editor.getControl().onMouseLeave(event => this.handleMouseLeave(event)),
110
+ this.editor.getControl().onKeyDown(() => this.hover.hide({ immediate: false })),
111
+ this.editor.getControl().onDidChangeModelContent(() => this.update()),
112
+ this.editor.getControl().getModel()!.onDidChangeDecorations(() => this.updateBreakpoints()),
113
+ this.editor.onDidResize(e => this.breakpointWidget.inputSize = e),
114
+ this.sessions.onDidChange(() => this.update()),
115
+ this.toDisposeOnUpdate,
116
+ this.sessionManager.onDidChangeBreakpoints(({ session, uri }) => {
117
+ if ((!session || session === this.sessionManager.currentSession) && uri.isEqual(this.uri)) {
118
+ this.render();
119
+ }
120
+ }),
121
+ this.breakpoints.onDidChangeBreakpoints(event => this.closeBreakpointIfAffected(event)),
122
+ ]);
123
+ this.update();
124
+ this.render();
125
+ }
126
+
127
+ dispose(): void {
128
+ this.toDispose.dispose();
129
+ }
130
+
131
+ protected readonly update = debounce(async () => {
132
+ if (this.toDispose.disposed) {
133
+ return;
134
+ }
135
+ this.toDisposeOnUpdate.dispose();
136
+ this.toggleExceptionWidget();
137
+ await this.updateEditorDecorations();
138
+ this.updateEditorHover();
139
+ }, 100);
140
+
141
+ /**
142
+ * To disable the default editor-contribution hover from Code when
143
+ * the editor has the `currentFrame`. Otherwise, both `textdocument/hover`
144
+ * and the debug hovers are visible at the same time when hovering over a symbol.
145
+ */
146
+ protected async updateEditorHover(): Promise<void> {
147
+ if (this.sessions.isCurrentEditorFrame(this.uri)) {
148
+ const codeEditor = this.editor.getControl();
149
+ codeEditor.updateOptions({ hover: { enabled: false } });
150
+ this.toDisposeOnUpdate.push(Disposable.create(() => {
151
+ const model = codeEditor.getModel()!;
152
+ const overrides = {
153
+ resource: model.uri,
154
+ overrideIdentifier: model.getLanguageId(),
155
+ };
156
+ const { enabled, delay, sticky } = StandaloneServices.get(IConfigurationService).getValue<IEditorHoverOptions>('editor.hover', overrides);
157
+ codeEditor.updateOptions({
158
+ hover: {
159
+ enabled,
160
+ delay,
161
+ sticky
162
+ }
163
+ });
164
+ }));
165
+ }
166
+ }
167
+
168
+ protected async updateEditorDecorations(): Promise<void> {
169
+ const [newFrameDecorations, inlineValueDecorations] = await Promise.all([
170
+ this.createFrameDecorations(),
171
+ this.createInlineValueDecorations()
172
+ ]);
173
+ const codeEditor = this.editor.getControl() as unknown as StandaloneCodeEditor;
174
+ codeEditor.removeDecorations([INLINE_VALUE_DECORATION_KEY]);
175
+ codeEditor.setDecorationsByType('Inline debug decorations', INLINE_VALUE_DECORATION_KEY, inlineValueDecorations);
176
+ this.editorDecorations = this.deltaDecorations(this.editorDecorations, newFrameDecorations);
177
+ }
178
+
179
+ protected async createInlineValueDecorations(): Promise<IDecorationOptions[]> {
180
+ if (!this.sessions.isCurrentEditorFrame(this.uri)) {
181
+ return [];
182
+ }
183
+ const { currentFrame } = this.sessions;
184
+ return this.inlineValueDecorator.calculateDecorations(this, currentFrame);
185
+ }
186
+
187
+ protected createFrameDecorations(): monaco.editor.IModelDeltaDecoration[] {
188
+ const { currentFrame, topFrame } = this.sessions;
189
+ if (!currentFrame) {
190
+ return [];
191
+ }
192
+
193
+ if (!currentFrame.thread.stopped) {
194
+ return [];
195
+ }
196
+
197
+ if (!this.sessions.isCurrentEditorFrame(this.uri)) {
198
+ return [];
199
+ }
200
+
201
+ const decorations: monaco.editor.IModelDeltaDecoration[] = [];
202
+ const columnUntilEOLRange = new monaco.Range(currentFrame.raw.line, currentFrame.raw.column, currentFrame.raw.line, 1 << 30);
203
+ const range = new monaco.Range(currentFrame.raw.line, currentFrame.raw.column, currentFrame.raw.line, currentFrame.raw.column + 1);
204
+
205
+ if (topFrame === currentFrame) {
206
+ decorations.push({
207
+ options: DebugEditorModel.TOP_STACK_FRAME_MARGIN,
208
+ range
209
+ });
210
+ decorations.push({
211
+ options: DebugEditorModel.TOP_STACK_FRAME_DECORATION,
212
+ range: columnUntilEOLRange
213
+ });
214
+ const { topFrameRange } = this;
215
+ if (topFrameRange && topFrameRange.startLineNumber === currentFrame.raw.line && topFrameRange.startColumn !== currentFrame.raw.column) {
216
+ decorations.push({
217
+ options: DebugEditorModel.TOP_STACK_FRAME_INLINE_DECORATION,
218
+ range: columnUntilEOLRange
219
+ });
220
+ }
221
+ this.topFrameRange = columnUntilEOLRange;
222
+ } else {
223
+ decorations.push({
224
+ options: DebugEditorModel.FOCUSED_STACK_FRAME_MARGIN,
225
+ range
226
+ });
227
+ decorations.push({
228
+ options: DebugEditorModel.FOCUSED_STACK_FRAME_DECORATION,
229
+ range: columnUntilEOLRange
230
+ });
231
+ }
232
+ return decorations;
233
+ }
234
+
235
+ protected async toggleExceptionWidget(): Promise<void> {
236
+ const { currentFrame } = this.sessions;
237
+ if (!currentFrame) {
238
+ return;
239
+ }
240
+ if (!this.sessions.isCurrentEditorFrame(this.uri)) {
241
+ this.exceptionWidget.hide();
242
+ return;
243
+ }
244
+ const info = await currentFrame.thread.getExceptionInfo();
245
+ if (!info) {
246
+ this.exceptionWidget.hide();
247
+ return;
248
+ }
249
+ this.exceptionWidget.show({
250
+ info,
251
+ lineNumber: currentFrame.raw.line,
252
+ column: currentFrame.raw.column
253
+ });
254
+ }
255
+
256
+ render(): void {
257
+ this.renderBreakpoints();
258
+ this.renderCurrentBreakpoints();
259
+ }
260
+ protected renderBreakpoints(): void {
261
+ const breakpoints = this.breakpoints.getBreakpoints(this.uri);
262
+ const decorations = this.createBreakpointDecorations(breakpoints);
263
+ this.breakpointDecorations = this.deltaDecorations(this.breakpointDecorations, decorations);
264
+ this.updateBreakpointRanges(breakpoints);
265
+ }
266
+ protected createBreakpointDecorations(breakpoints: SourceBreakpoint[]): monaco.editor.IModelDeltaDecoration[] {
267
+ return breakpoints.map(breakpoint => this.createBreakpointDecoration(breakpoint));
268
+ }
269
+ protected createBreakpointDecoration(breakpoint: SourceBreakpoint): monaco.editor.IModelDeltaDecoration {
270
+ const lineNumber = breakpoint.raw.line;
271
+ const column = breakpoint.raw.column;
272
+ const range = typeof column === 'number' ? new monaco.Range(lineNumber, column, lineNumber, column + 1) : new monaco.Range(lineNumber, 1, lineNumber, 2);
273
+ return {
274
+ range,
275
+ options: {
276
+ stickiness: DebugEditorModel.STICKINESS
277
+ }
278
+ };
279
+ }
280
+
281
+ protected updateBreakpointRanges(breakpoints: SourceBreakpoint[]): void {
282
+ this.breakpointRanges.clear();
283
+ for (let i = 0; i < this.breakpointDecorations.length; i++) {
284
+ const decoration = this.breakpointDecorations[i];
285
+ const breakpoint = breakpoints[i];
286
+ const range = this.editor.getControl().getModel()!.getDecorationRange(decoration)!;
287
+ this.breakpointRanges.set(decoration, [range, breakpoint]);
288
+ }
289
+ }
290
+
291
+ protected renderCurrentBreakpoints(): void {
292
+ const decorations = this.createCurrentBreakpointDecorations();
293
+ this.currentBreakpointDecorations = this.deltaDecorations(this.currentBreakpointDecorations, decorations);
294
+ }
295
+ protected createCurrentBreakpointDecorations(): monaco.editor.IModelDeltaDecoration[] {
296
+ const breakpoints = this.sessions.getBreakpoints(this.uri);
297
+ return breakpoints.map(breakpoint => this.createCurrentBreakpointDecoration(breakpoint));
298
+ }
299
+ protected createCurrentBreakpointDecoration(breakpoint: DebugSourceBreakpoint): monaco.editor.IModelDeltaDecoration {
300
+ const lineNumber = breakpoint.line;
301
+ const column = breakpoint.column;
302
+ const range = typeof column === 'number' ? new monaco.Range(lineNumber, column, lineNumber, column + 1) : new monaco.Range(lineNumber, 1, lineNumber, 1);
303
+ const { className, message } = breakpoint.getDecoration();
304
+ const renderInline = typeof column === 'number' && (column > this.editor.getControl().getModel()!.getLineFirstNonWhitespaceColumn(lineNumber));
305
+ return {
306
+ range,
307
+ options: {
308
+ glyphMarginClassName: className,
309
+ glyphMarginHoverMessage: message.map(value => ({ value })),
310
+ stickiness: DebugEditorModel.STICKINESS,
311
+ beforeContentClassName: renderInline ? `theia-debug-breakpoint-column codicon ${className}` : undefined
312
+ }
313
+ };
314
+ }
315
+
316
+ protected updateBreakpoints(): void {
317
+ if (this.areBreakpointsAffected()) {
318
+ const breakpoints = this.createBreakpoints();
319
+ this.breakpoints.setBreakpoints(this.uri, breakpoints);
320
+ }
321
+ }
322
+ protected areBreakpointsAffected(): boolean {
323
+ if (this.updatingDecorations || !this.editor.getControl().getModel()) {
324
+ return false;
325
+ }
326
+ for (const decoration of this.breakpointDecorations) {
327
+ const range = this.editor.getControl().getModel()!.getDecorationRange(decoration);
328
+ const oldRange = this.breakpointRanges.get(decoration)![0];
329
+ if (!range || !range.equalsRange(oldRange)) {
330
+ return true;
331
+ }
332
+ }
333
+ return false;
334
+ }
335
+ protected createBreakpoints(): SourceBreakpoint[] {
336
+ const { uri } = this;
337
+ const lines = new Set<number>();
338
+ const breakpoints: SourceBreakpoint[] = [];
339
+ for (const decoration of this.breakpointDecorations) {
340
+ const range = this.editor.getControl().getModel()!.getDecorationRange(decoration);
341
+ if (range && !lines.has(range.startLineNumber)) {
342
+ const line = range.startLineNumber;
343
+ const column = range.startColumn;
344
+ const oldBreakpoint = this.breakpointRanges.get(decoration)?.[1];
345
+ const isLineBreakpoint = oldBreakpoint?.raw.line !== undefined && oldBreakpoint?.raw.column === undefined;
346
+ const change = isLineBreakpoint ? { line } : { line, column };
347
+ const breakpoint = SourceBreakpoint.create(uri, change, oldBreakpoint);
348
+ breakpoints.push(breakpoint);
349
+ lines.add(line);
350
+ }
351
+ }
352
+ return breakpoints;
353
+ }
354
+
355
+ get position(): monaco.Position {
356
+ return this.editor.getControl().getPosition()!;
357
+ }
358
+ getBreakpoint(position: monaco.Position = this.position): DebugSourceBreakpoint | undefined {
359
+ return this.getInlineBreakpoint(position) || this.getLineBreakpoints(position)[0];
360
+ }
361
+
362
+ getInlineBreakpoint(position: monaco.Position = this.position): DebugSourceBreakpoint | undefined {
363
+ return this.sessions.getInlineBreakpoint(this.uri, position.lineNumber, position.column);
364
+ }
365
+
366
+ protected getLineBreakpoints(position: monaco.Position = this.position): DebugSourceBreakpoint[] {
367
+ return this.sessions.getLineBreakpoints(this.uri, position.lineNumber);
368
+ }
369
+
370
+ protected addBreakpoint(raw: DebugProtocol.SourceBreakpoint): void {
371
+ this.breakpoints.addBreakpoint(SourceBreakpoint.create(this.uri, raw));
372
+ }
373
+
374
+ toggleBreakpoint(position: monaco.Position = this.position): void {
375
+ const { lineNumber } = position;
376
+ const breakpoints = this.getLineBreakpoints(position);
377
+ if (breakpoints.length) {
378
+ for (const breakpoint of breakpoints) {
379
+ breakpoint.remove();
380
+ }
381
+ } else {
382
+ this.addBreakpoint({ line: lineNumber });
383
+ }
384
+ }
385
+
386
+ addInlineBreakpoint(): void {
387
+ const { position } = this;
388
+ const { lineNumber, column } = position;
389
+ const breakpoint = this.getInlineBreakpoint(position);
390
+ if (breakpoint) {
391
+ return;
392
+ }
393
+ this.addBreakpoint({ line: lineNumber, column });
394
+ }
395
+
396
+ acceptBreakpoint(): void {
397
+ const { position, values } = this.breakpointWidget;
398
+ if (position && values) {
399
+ const breakpoint = position.column > 0 ? this.getInlineBreakpoint(position) : this.getLineBreakpoints(position)[0];
400
+ if (breakpoint) {
401
+ breakpoint.updateOrigins(values);
402
+ } else {
403
+ const { lineNumber } = position;
404
+ const column = position.column > 0 ? position.column : undefined;
405
+ this.addBreakpoint({ line: lineNumber, column, ...values });
406
+ }
407
+ this.breakpointWidget.hide();
408
+ }
409
+ }
410
+
411
+ protected handleMouseDown(event: monaco.editor.IEditorMouseEvent): void {
412
+ if (event.target && event.target.type === monaco.editor.MouseTargetType.GUTTER_GLYPH_MARGIN) {
413
+ if (!event.event.rightButton) {
414
+ this.toggleBreakpoint(event.target.position!);
415
+ }
416
+ }
417
+ this.hintBreakpoint(event);
418
+ }
419
+ protected handleMouseMove(event: monaco.editor.IEditorMouseEvent): void {
420
+ this.showHover(event);
421
+ this.hintBreakpoint(event);
422
+ }
423
+ protected handleMouseLeave(event: monaco.editor.IPartialEditorMouseEvent): void {
424
+ this.hideHover(event);
425
+ this.deltaHintDecorations([]);
426
+ }
427
+
428
+ protected hintDecorations: string[] = [];
429
+ protected hintBreakpoint(event: monaco.editor.IEditorMouseEvent): void {
430
+ const hintDecorations = this.createHintDecorations(event);
431
+ this.deltaHintDecorations(hintDecorations);
432
+ }
433
+ protected deltaHintDecorations(hintDecorations: monaco.editor.IModelDeltaDecoration[]): void {
434
+ this.hintDecorations = this.deltaDecorations(this.hintDecorations, hintDecorations);
435
+ }
436
+ protected createHintDecorations(event: monaco.editor.IEditorMouseEvent): monaco.editor.IModelDeltaDecoration[] {
437
+ if (event.target && event.target.type === monaco.editor.MouseTargetType.GUTTER_GLYPH_MARGIN && event.target.position) {
438
+ const lineNumber = event.target.position.lineNumber;
439
+ if (this.getLineBreakpoints(event.target.position).length) {
440
+ return [];
441
+ }
442
+ return [{
443
+ range: new monaco.Range(lineNumber, 1, lineNumber, 1),
444
+ options: DebugEditorModel.BREAKPOINT_HINT_DECORATION
445
+ }];
446
+ }
447
+ return [];
448
+ }
449
+
450
+ protected closeBreakpointIfAffected({ uri, removed }: SourceBreakpointsChangeEvent): void {
451
+ if (!uri.isEqual(this.uri)) {
452
+ return;
453
+ }
454
+ const position = this.breakpointWidget.position;
455
+ if (!position) {
456
+ return;
457
+ }
458
+ for (const breakpoint of removed) {
459
+ if (breakpoint.raw.line === position.lineNumber) {
460
+ this.breakpointWidget.hide();
461
+ break;
462
+ }
463
+ }
464
+ }
465
+
466
+ protected showHover(mouseEvent: monaco.editor.IEditorMouseEvent): void {
467
+ const targetType = mouseEvent.target.type;
468
+ const stopKey = isOSX ? 'metaKey' : 'ctrlKey';
469
+
470
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
471
+ if (targetType === monaco.editor.MouseTargetType.CONTENT_WIDGET && mouseEvent.target.detail === this.hover.getId() && !(<any>mouseEvent.event)[stopKey]) {
472
+ // mouse moved on top of debug hover widget
473
+ return;
474
+ }
475
+ if (targetType === monaco.editor.MouseTargetType.CONTENT_TEXT) {
476
+ this.hover.show({
477
+ selection: mouseEvent.target.range!,
478
+ immediate: false
479
+ });
480
+ } else {
481
+ this.hover.hide({ immediate: false });
482
+ }
483
+ }
484
+ protected hideHover({ event }: monaco.editor.IPartialEditorMouseEvent): void {
485
+ const rect = this.hover.getDomNode().getBoundingClientRect();
486
+ if (event.posx < rect.left || event.posx > rect.right || event.posy < rect.top || event.posy > rect.bottom) {
487
+ this.hover.hide({ immediate: false });
488
+ }
489
+ }
490
+
491
+ protected deltaDecorations(oldDecorations: string[], newDecorations: monaco.editor.IModelDeltaDecoration[]): string[] {
492
+ this.updatingDecorations = true;
493
+ try {
494
+ return this.editor.getControl().deltaDecorations(oldDecorations, newDecorations);
495
+ } finally {
496
+ this.updatingDecorations = false;
497
+ }
498
+ }
499
+
500
+ static STICKINESS = monaco.editor.TrackedRangeStickiness.NeverGrowsWhenTypingAtEdges;
501
+
502
+ static BREAKPOINT_HINT_DECORATION: monaco.editor.IModelDecorationOptions = {
503
+ glyphMarginClassName: 'codicon-debug-hint',
504
+ stickiness: DebugEditorModel.STICKINESS
505
+ };
506
+
507
+ static TOP_STACK_FRAME_MARGIN: monaco.editor.IModelDecorationOptions = {
508
+ glyphMarginClassName: 'codicon-debug-stackframe',
509
+ stickiness: DebugEditorModel.STICKINESS
510
+ };
511
+ static FOCUSED_STACK_FRAME_MARGIN: monaco.editor.IModelDecorationOptions = {
512
+ glyphMarginClassName: 'codicon-debug-stackframe-focused',
513
+ stickiness: DebugEditorModel.STICKINESS
514
+ };
515
+ static TOP_STACK_FRAME_DECORATION: monaco.editor.IModelDecorationOptions = {
516
+ isWholeLine: true,
517
+ className: 'theia-debug-top-stack-frame-line',
518
+ stickiness: DebugEditorModel.STICKINESS
519
+ };
520
+ static TOP_STACK_FRAME_INLINE_DECORATION: monaco.editor.IModelDecorationOptions = {
521
+ beforeContentClassName: 'theia-debug-top-stack-frame-column'
522
+ };
523
+ static FOCUSED_STACK_FRAME_DECORATION: monaco.editor.IModelDecorationOptions = {
524
+ isWholeLine: true,
525
+ className: 'theia-debug-focused-stack-frame-line',
526
+ stickiness: DebugEditorModel.STICKINESS
527
+ };
528
+
529
+ }