@theia/monaco 1.45.1 → 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 (222) hide show
  1. package/README.md +100 -100
  2. package/data/monaco-nls.json +1379 -1379
  3. package/data/monaco-themes/vscode/dark_plus.json +201 -201
  4. package/data/monaco-themes/vscode/dark_theia.json +5 -5
  5. package/data/monaco-themes/vscode/dark_vs.json +393 -393
  6. package/data/monaco-themes/vscode/hc_black.json +457 -457
  7. package/data/monaco-themes/vscode/hc_light.json +590 -590
  8. package/data/monaco-themes/vscode/hc_theia.json +5 -5
  9. package/data/monaco-themes/vscode/hc_theia_light.json +5 -5
  10. package/data/monaco-themes/vscode/light_plus.json +202 -202
  11. package/data/monaco-themes/vscode/light_theia.json +10 -10
  12. package/data/monaco-themes/vscode/light_vs.json +421 -421
  13. package/lib/browser/index.d.ts +1 -1
  14. package/lib/browser/index.js +28 -28
  15. package/lib/browser/markdown-renderer/monaco-markdown-renderer.d.ts +20 -24
  16. package/lib/browser/markdown-renderer/monaco-markdown-renderer.d.ts.map +1 -1
  17. package/lib/browser/markdown-renderer/monaco-markdown-renderer.js +122 -137
  18. package/lib/browser/markdown-renderer/monaco-markdown-renderer.js.map +1 -1
  19. package/lib/browser/monaco-bulk-edit-service.d.ts +12 -14
  20. package/lib/browser/monaco-bulk-edit-service.d.ts.map +1 -1
  21. package/lib/browser/monaco-bulk-edit-service.js +66 -66
  22. package/lib/browser/monaco-bulk-edit-service.js.map +1 -1
  23. package/lib/browser/monaco-color-registry.d.ts +16 -16
  24. package/lib/browser/monaco-color-registry.js +82 -82
  25. package/lib/browser/monaco-command-registry.d.ts +19 -19
  26. package/lib/browser/monaco-command-registry.d.ts.map +1 -1
  27. package/lib/browser/monaco-command-registry.js +88 -85
  28. package/lib/browser/monaco-command-registry.js.map +1 -1
  29. package/lib/browser/monaco-command-service.d.ts +21 -26
  30. package/lib/browser/monaco-command-service.d.ts.map +1 -1
  31. package/lib/browser/monaco-command-service.js +93 -92
  32. package/lib/browser/monaco-command-service.js.map +1 -1
  33. package/lib/browser/monaco-command.d.ts +88 -94
  34. package/lib/browser/monaco-command.d.ts.map +1 -1
  35. package/lib/browser/monaco-command.js +294 -307
  36. package/lib/browser/monaco-command.js.map +1 -1
  37. package/lib/browser/monaco-context-key-service.d.ts +22 -22
  38. package/lib/browser/monaco-context-key-service.d.ts.map +1 -1
  39. package/lib/browser/monaco-context-key-service.js +146 -147
  40. package/lib/browser/monaco-context-key-service.js.map +1 -1
  41. package/lib/browser/monaco-context-menu.d.ts +17 -16
  42. package/lib/browser/monaco-context-menu.d.ts.map +1 -1
  43. package/lib/browser/monaco-context-menu.js +112 -102
  44. package/lib/browser/monaco-context-menu.js.map +1 -1
  45. package/lib/browser/monaco-diff-editor.d.ts +29 -29
  46. package/lib/browser/monaco-diff-editor.d.ts.map +1 -1
  47. package/lib/browser/monaco-diff-editor.js +74 -74
  48. package/lib/browser/monaco-diff-editor.js.map +1 -1
  49. package/lib/browser/monaco-diff-navigator-factory.d.ts +6 -7
  50. package/lib/browser/monaco-diff-navigator-factory.d.ts.map +1 -1
  51. package/lib/browser/monaco-diff-navigator-factory.js +45 -66
  52. package/lib/browser/monaco-diff-navigator-factory.js.map +1 -1
  53. package/lib/browser/monaco-editor-model.d.ts +158 -156
  54. package/lib/browser/monaco-editor-model.d.ts.map +1 -1
  55. package/lib/browser/monaco-editor-model.js +562 -559
  56. package/lib/browser/monaco-editor-model.js.map +1 -1
  57. package/lib/browser/monaco-editor-provider.d.ts +70 -89
  58. package/lib/browser/monaco-editor-provider.d.ts.map +1 -1
  59. package/lib/browser/monaco-editor-provider.js +386 -440
  60. package/lib/browser/monaco-editor-provider.js.map +1 -1
  61. package/lib/browser/monaco-editor-service.d.ts +32 -27
  62. package/lib/browser/monaco-editor-service.d.ts.map +1 -1
  63. package/lib/browser/monaco-editor-service.js +158 -155
  64. package/lib/browser/monaco-editor-service.js.map +1 -1
  65. package/lib/browser/monaco-editor-zone-widget.d.ts +43 -43
  66. package/lib/browser/monaco-editor-zone-widget.js +194 -194
  67. package/lib/browser/monaco-editor.d.ts +154 -151
  68. package/lib/browser/monaco-editor.d.ts.map +1 -1
  69. package/lib/browser/monaco-editor.js +542 -524
  70. package/lib/browser/monaco-editor.js.map +1 -1
  71. package/lib/browser/monaco-formatting-conflicts.d.ts +13 -13
  72. package/lib/browser/monaco-formatting-conflicts.js +111 -111
  73. package/lib/browser/monaco-formatting-conflicts.js.map +1 -1
  74. package/lib/browser/monaco-frontend-application-contribution.d.ts +22 -28
  75. package/lib/browser/monaco-frontend-application-contribution.d.ts.map +1 -1
  76. package/lib/browser/monaco-frontend-application-contribution.js +178 -215
  77. package/lib/browser/monaco-frontend-application-contribution.js.map +1 -1
  78. package/lib/browser/monaco-frontend-module.d.ts +7 -7
  79. package/lib/browser/monaco-frontend-module.d.ts.map +1 -1
  80. package/lib/browser/monaco-frontend-module.js +280 -271
  81. package/lib/browser/monaco-frontend-module.js.map +1 -1
  82. package/lib/browser/monaco-gotoline-quick-access.d.ts +11 -11
  83. package/lib/browser/monaco-gotoline-quick-access.js +64 -64
  84. package/lib/browser/monaco-gotosymbol-quick-access.d.ts +16 -16
  85. package/lib/browser/monaco-gotosymbol-quick-access.js +70 -70
  86. package/lib/browser/monaco-icon-registry.d.ts +10 -11
  87. package/lib/browser/monaco-icon-registry.d.ts.map +1 -1
  88. package/lib/browser/monaco-icon-registry.js +52 -50
  89. package/lib/browser/monaco-icon-registry.js.map +1 -1
  90. package/lib/browser/monaco-indexed-db.d.ts +28 -28
  91. package/lib/browser/monaco-indexed-db.js +131 -131
  92. package/lib/browser/monaco-init.d.ts +5 -0
  93. package/lib/browser/monaco-init.d.ts.map +1 -0
  94. package/lib/browser/monaco-init.js +125 -0
  95. package/lib/browser/monaco-init.js.map +1 -0
  96. package/lib/browser/monaco-keybinding.d.ts +16 -16
  97. package/lib/browser/monaco-keybinding.js +134 -134
  98. package/lib/browser/monaco-keybinding.js.map +1 -1
  99. package/lib/browser/monaco-keycode-map.d.ts +2 -2
  100. package/lib/browser/monaco-keycode-map.js +163 -163
  101. package/lib/browser/monaco-keycode-map.js.map +1 -1
  102. package/lib/browser/monaco-languages.d.ts +35 -35
  103. package/lib/browser/monaco-languages.js +180 -180
  104. package/lib/browser/monaco-marker-collection.d.ts +18 -18
  105. package/lib/browser/monaco-marker-collection.js +70 -70
  106. package/lib/browser/monaco-menu.d.ts +20 -20
  107. package/lib/browser/monaco-menu.js +145 -145
  108. package/lib/browser/monaco-mime-service.d.ts +10 -10
  109. package/lib/browser/monaco-mime-service.js +78 -78
  110. package/lib/browser/monaco-outline-contribution.d.ts +78 -78
  111. package/lib/browser/monaco-outline-contribution.js +384 -384
  112. package/lib/browser/monaco-outline-decorator.d.ts +12 -12
  113. package/lib/browser/monaco-outline-decorator.js +67 -67
  114. package/lib/browser/monaco-quick-access-registry.d.ts +11 -11
  115. package/lib/browser/monaco-quick-access-registry.js +99 -99
  116. package/lib/browser/monaco-quick-input-service.d.ts +99 -101
  117. package/lib/browser/monaco-quick-input-service.d.ts.map +1 -1
  118. package/lib/browser/monaco-quick-input-service.js +529 -568
  119. package/lib/browser/monaco-quick-input-service.js.map +1 -1
  120. package/lib/browser/monaco-resolved-keybinding.d.ts +23 -23
  121. package/lib/browser/monaco-resolved-keybinding.d.ts.map +1 -1
  122. package/lib/browser/monaco-resolved-keybinding.js +131 -131
  123. package/lib/browser/monaco-resolved-keybinding.js.map +1 -1
  124. package/lib/browser/monaco-snippet-suggest-provider.d.ts +66 -66
  125. package/lib/browser/monaco-snippet-suggest-provider.js +273 -273
  126. package/lib/browser/monaco-status-bar-contribution.d.ts +17 -17
  127. package/lib/browser/monaco-status-bar-contribution.js +121 -121
  128. package/lib/browser/monaco-text-model-service.d.ts +51 -51
  129. package/lib/browser/monaco-text-model-service.js +189 -189
  130. package/lib/browser/monaco-text-model-service.js.map +1 -1
  131. package/lib/browser/monaco-theming-service.d.ts +67 -67
  132. package/lib/browser/monaco-theming-service.js +171 -171
  133. package/lib/browser/monaco-to-protocol-converter.d.ts +26 -26
  134. package/lib/browser/monaco-to-protocol-converter.js +61 -61
  135. package/lib/browser/monaco-workspace.d.ts +83 -85
  136. package/lib/browser/monaco-workspace.d.ts.map +1 -1
  137. package/lib/browser/monaco-workspace.js +384 -384
  138. package/lib/browser/monaco-workspace.js.map +1 -1
  139. package/lib/browser/protocol-to-monaco-converter.d.ts +31 -31
  140. package/lib/browser/protocol-to-monaco-converter.js +133 -133
  141. package/lib/browser/simple-monaco-editor.d.ts +45 -44
  142. package/lib/browser/simple-monaco-editor.d.ts.map +1 -1
  143. package/lib/browser/simple-monaco-editor.js +184 -176
  144. package/lib/browser/simple-monaco-editor.js.map +1 -1
  145. package/lib/browser/textmate/index.d.ts +4 -4
  146. package/lib/browser/textmate/index.js +31 -31
  147. package/lib/browser/textmate/monaco-textmate-frontend-bindings.d.ts +11 -11
  148. package/lib/browser/textmate/monaco-textmate-frontend-bindings.js +91 -91
  149. package/lib/browser/textmate/monaco-textmate-service.d.ts +30 -30
  150. package/lib/browser/textmate/monaco-textmate-service.js +185 -185
  151. package/lib/browser/textmate/monaco-textmate-service.js.map +1 -1
  152. package/lib/browser/textmate/monaco-theme-registry.d.ts +27 -27
  153. package/lib/browser/textmate/monaco-theme-registry.js +173 -173
  154. package/lib/browser/textmate/monaco-theme-types.d.ts +20 -20
  155. package/lib/browser/textmate/monaco-theme-types.js +21 -21
  156. package/lib/browser/textmate/textmate-contribution.d.ts +9 -9
  157. package/lib/browser/textmate/textmate-contribution.js +27 -27
  158. package/lib/browser/textmate/textmate-registry.d.ts +31 -31
  159. package/lib/browser/textmate/textmate-registry.js +109 -109
  160. package/lib/browser/textmate/textmate-snippet-completion-provider.d.ts +26 -26
  161. package/lib/browser/textmate/textmate-snippet-completion-provider.js +53 -53
  162. package/lib/browser/textmate/textmate-tokenizer.d.ts +22 -22
  163. package/lib/browser/textmate/textmate-tokenizer.js +65 -65
  164. package/lib/browser/workspace-symbol-command.d.ts +55 -55
  165. package/lib/browser/workspace-symbol-command.js +215 -215
  166. package/lib/package.spec.js +25 -25
  167. package/package.json +10 -10
  168. package/src/browser/index.ts +17 -17
  169. package/src/browser/markdown-renderer/monaco-markdown-renderer.ts +109 -117
  170. package/src/browser/monaco-bulk-edit-service.ts +64 -64
  171. package/src/browser/monaco-color-registry.ts +73 -73
  172. package/src/browser/monaco-command-registry.ts +85 -82
  173. package/src/browser/monaco-command-service.ts +90 -99
  174. package/src/browser/monaco-command.ts +303 -313
  175. package/src/browser/monaco-context-key-service.ts +144 -143
  176. package/src/browser/monaco-context-menu.ts +100 -90
  177. package/src/browser/monaco-diff-editor.ts +100 -101
  178. package/src/browser/monaco-diff-navigator-factory.ts +39 -61
  179. package/src/browser/monaco-editor-model.ts +684 -681
  180. package/src/browser/monaco-editor-provider.ts +409 -452
  181. package/src/browser/monaco-editor-service.ts +149 -144
  182. package/src/browser/monaco-editor-zone-widget.ts +250 -250
  183. package/src/browser/monaco-editor.ts +663 -644
  184. package/src/browser/monaco-formatting-conflicts.ts +116 -116
  185. package/src/browser/monaco-frontend-application-contribution.ts +165 -200
  186. package/src/browser/monaco-frontend-module.ts +328 -320
  187. package/src/browser/monaco-gotoline-quick-access.ts +47 -47
  188. package/src/browser/monaco-gotosymbol-quick-access.ts +53 -53
  189. package/src/browser/monaco-icon-registry.ts +49 -47
  190. package/src/browser/monaco-indexed-db.ts +130 -130
  191. package/src/browser/monaco-init.ts +114 -0
  192. package/src/browser/monaco-keybinding.ts +111 -111
  193. package/src/browser/monaco-keycode-map.ts +171 -171
  194. package/src/browser/monaco-languages.ts +177 -177
  195. package/src/browser/monaco-marker-collection.ts +83 -83
  196. package/src/browser/monaco-menu.ts +142 -142
  197. package/src/browser/monaco-mime-service.ts +71 -71
  198. package/src/browser/monaco-outline-contribution.ts +404 -404
  199. package/src/browser/monaco-outline-decorator.ts +66 -66
  200. package/src/browser/monaco-quick-access-registry.ts +112 -112
  201. package/src/browser/monaco-quick-input-service.ts +647 -676
  202. package/src/browser/monaco-resolved-keybinding.ts +162 -162
  203. package/src/browser/monaco-snippet-suggest-provider.ts +306 -306
  204. package/src/browser/monaco-status-bar-contribution.ts +110 -110
  205. package/src/browser/monaco-text-model-service.ts +199 -199
  206. package/src/browser/monaco-theming-service.ts +204 -204
  207. package/src/browser/monaco-to-protocol-converter.ts +71 -71
  208. package/src/browser/monaco-workspace.ts +412 -412
  209. package/src/browser/protocol-to-monaco-converter.ts +158 -158
  210. package/src/browser/simple-monaco-editor.ts +217 -210
  211. package/src/browser/style/index.css +263 -263
  212. package/src/browser/textmate/index.ts +20 -20
  213. package/src/browser/textmate/monaco-textmate-frontend-bindings.ts +90 -90
  214. package/src/browser/textmate/monaco-textmate-service.ts +187 -187
  215. package/src/browser/textmate/monaco-theme-registry.ts +176 -176
  216. package/src/browser/textmate/monaco-theme-types.ts +37 -37
  217. package/src/browser/textmate/textmate-contribution.ts +29 -29
  218. package/src/browser/textmate/textmate-registry.ts +129 -129
  219. package/src/browser/textmate/textmate-snippet-completion-provider.ts +73 -73
  220. package/src/browser/textmate/textmate-tokenizer.ts +84 -84
  221. package/src/browser/workspace-symbol-command.ts +196 -196
  222. package/src/package.spec.ts +28 -28
@@ -1,306 +1,306 @@
1
- // *****************************************************************************
2
- // Copyright (C) 2019 TypeFox and others.
3
- //
4
- // This program and the accompanying materials are made available under the
5
- // terms of the Eclipse Public License v. 2.0 which is available at
6
- // http://www.eclipse.org/legal/epl-2.0.
7
- //
8
- // This Source Code may also be made available under the following Secondary
9
- // Licenses when the conditions for such availability set forth in the Eclipse
10
- // Public License v. 2.0 are satisfied: GNU General Public License, version 2
11
- // with the GNU Classpath Exception which is available at
12
- // https://www.gnu.org/software/classpath/license.html.
13
- //
14
- // SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
15
- // *****************************************************************************
16
- /*---------------------------------------------------------------------------------------------
17
- * Copyright (c) Microsoft Corporation. All rights reserved.
18
- * Licensed under the MIT License. See License.txt in the project root for license information.
19
- *--------------------------------------------------------------------------------------------*/
20
-
21
- import * as jsoncparser from 'jsonc-parser';
22
- import { injectable, inject } from '@theia/core/shared/inversify';
23
- import URI from '@theia/core/lib/common/uri';
24
- import { Disposable, DisposableCollection } from '@theia/core/lib/common/disposable';
25
- import { FileService } from '@theia/filesystem/lib/browser/file-service';
26
- import { FileOperationError } from '@theia/filesystem/lib/common/files';
27
- import * as monaco from '@theia/monaco-editor-core';
28
- import { SnippetParser } from '@theia/monaco-editor-core/esm/vs/editor/contrib/snippet/browser/snippetParser';
29
- import { isObject } from '@theia/core/lib/common';
30
-
31
- @injectable()
32
- export class MonacoSnippetSuggestProvider implements monaco.languages.CompletionItemProvider {
33
-
34
- private static readonly _maxPrefix = 10000;
35
-
36
- @inject(FileService)
37
- protected readonly fileService: FileService;
38
-
39
- protected readonly snippets = new Map<string, Snippet[]>();
40
- protected readonly pendingSnippets = new Map<string, Promise<void>[]>();
41
-
42
- async provideCompletionItems(model: monaco.editor.ITextModel, position: monaco.Position,
43
- context: monaco.languages.CompletionContext): Promise<monaco.languages.CompletionList | undefined> {
44
-
45
- // copied and modified from https://github.com/microsoft/vscode/blob/master/src/vs/workbench/contrib/snippets/browser/snippetCompletionProvider.ts
46
- if (position.column >= MonacoSnippetSuggestProvider._maxPrefix) {
47
- return undefined;
48
- }
49
-
50
- if (context.triggerKind === monaco.languages.CompletionTriggerKind.TriggerCharacter && context.triggerCharacter === ' ') {
51
- // no snippets when suggestions have been triggered by space
52
- return undefined;
53
- }
54
-
55
- const languageId = model.getLanguageId(); // TODO: look up a language id at the position
56
- await this.loadSnippets(languageId);
57
- const snippetsForLanguage = this.snippets.get(languageId) || [];
58
-
59
- const pos = { lineNumber: position.lineNumber, column: 1 };
60
- const lineOffsets: number[] = [];
61
- const linePrefixLow = model.getLineContent(position.lineNumber).substring(0, position.column - 1).toLowerCase();
62
- const endsInWhitespace = linePrefixLow.match(/\s$/);
63
-
64
- while (pos.column < position.column) {
65
- const word = model.getWordAtPosition(pos);
66
- if (word) {
67
- // at a word
68
- lineOffsets.push(word.startColumn - 1);
69
- pos.column = word.endColumn + 1;
70
- if (word.endColumn - 1 < linePrefixLow.length && !/\s/.test(linePrefixLow[word.endColumn - 1])) {
71
- lineOffsets.push(word.endColumn - 1);
72
- }
73
- } else if (!/\s/.test(linePrefixLow[pos.column - 1])) {
74
- // at a none-whitespace character
75
- lineOffsets.push(pos.column - 1);
76
- pos.column += 1;
77
- } else {
78
- // always advance!
79
- pos.column += 1;
80
- }
81
- }
82
-
83
- const availableSnippets = new Set<Snippet>();
84
- snippetsForLanguage.forEach(availableSnippets.add, availableSnippets);
85
- const suggestions: MonacoSnippetSuggestion[] = [];
86
- for (const start of lineOffsets) {
87
- availableSnippets.forEach(snippet => {
88
- if (this.isPatternInWord(linePrefixLow, start, linePrefixLow.length, snippet.prefix.toLowerCase(), 0, snippet.prefix.length)) {
89
- suggestions.push(new MonacoSnippetSuggestion(snippet, monaco.Range.fromPositions(position.delta(0, -(linePrefixLow.length - start)), position)));
90
- availableSnippets.delete(snippet);
91
- }
92
- });
93
- }
94
- if (endsInWhitespace || lineOffsets.length === 0) {
95
- // add remaining snippets when the current prefix ends in whitespace or when no
96
- // interesting positions have been found
97
- availableSnippets.forEach(snippet => {
98
- suggestions.push(new MonacoSnippetSuggestion(snippet, monaco.Range.fromPositions(position)));
99
- });
100
- }
101
-
102
- // disambiguate suggestions with same labels
103
- suggestions.sort(MonacoSnippetSuggestion.compareByLabel);
104
- return { suggestions };
105
- }
106
-
107
- resolveCompletionItem?(item: monaco.languages.CompletionItem, token: monaco.CancellationToken): monaco.languages.CompletionItem {
108
- return item instanceof MonacoSnippetSuggestion ? item.resolve() : item;
109
- }
110
-
111
- protected async loadSnippets(scope: string): Promise<void> {
112
- const pending: Promise<void>[] = [];
113
- pending.push(...(this.pendingSnippets.get(scope) || []));
114
- pending.push(...(this.pendingSnippets.get('*') || []));
115
- if (pending.length) {
116
- await Promise.all(pending);
117
- }
118
- }
119
-
120
- fromURI(uri: string | URI, options: SnippetLoadOptions): Disposable {
121
- const toDispose = new DisposableCollection(Disposable.create(() => { /* mark as not disposed */ }));
122
- const pending = this.loadURI(uri, options, toDispose);
123
- const { language } = options;
124
- const scopes = Array.isArray(language) ? language : !!language ? [language] : ['*'];
125
- for (const scope of scopes) {
126
- const pendingSnippets = this.pendingSnippets.get(scope) || [];
127
- pendingSnippets.push(pending);
128
- this.pendingSnippets.set(scope, pendingSnippets);
129
- toDispose.push(Disposable.create(() => {
130
- const index = pendingSnippets.indexOf(pending);
131
- if (index !== -1) {
132
- pendingSnippets.splice(index, 1);
133
- }
134
- }));
135
- }
136
- return toDispose;
137
- }
138
-
139
- /**
140
- * should NOT throw to prevent load errors on suggest
141
- */
142
- protected async loadURI(uri: string | URI, options: SnippetLoadOptions, toDispose: DisposableCollection): Promise<void> {
143
- try {
144
- const resource = typeof uri === 'string' ? new URI(uri) : uri;
145
- const { value } = await this.fileService.read(resource);
146
- if (toDispose.disposed) {
147
- return;
148
- }
149
- const snippets = value && jsoncparser.parse(value, undefined, { disallowComments: false });
150
- toDispose.push(this.fromJSON(snippets, options));
151
- } catch (e) {
152
- if (!(e instanceof FileOperationError)) {
153
- console.error(e);
154
- }
155
- }
156
- }
157
-
158
- fromJSON(snippets: JsonSerializedSnippets | undefined, { language, source }: SnippetLoadOptions): Disposable {
159
- const toDispose = new DisposableCollection();
160
- this.parseSnippets(snippets, (name, snippet) => {
161
- const { isFileTemplate, prefix, body, description } = snippet;
162
- const parsedBody = Array.isArray(body) ? body.join('\n') : body;
163
- const parsedPrefixes = !prefix ? [''] : Array.isArray(prefix) ? prefix : [prefix];
164
-
165
- if (typeof parsedBody !== 'string') {
166
- return;
167
- }
168
- const scopes: string[] = [];
169
- if (language) {
170
- if (Array.isArray(language)) {
171
- scopes.push(...language);
172
- } else {
173
- scopes.push(language);
174
- }
175
- } else if (typeof snippet.scope === 'string') {
176
- for (const rawScope of snippet.scope.split(',')) {
177
- const scope = rawScope.trim();
178
- if (scope) {
179
- scopes.push(scope);
180
- }
181
- }
182
- }
183
- parsedPrefixes.forEach(parsedPrefix => toDispose.push(this.push({
184
- isFileTemplate: Boolean(isFileTemplate),
185
- scopes,
186
- name,
187
- prefix: parsedPrefix,
188
- description,
189
- body: parsedBody,
190
- source
191
- })));
192
- });
193
- return toDispose;
194
- }
195
- protected parseSnippets(snippets: JsonSerializedSnippets | undefined, accept: (name: string, snippet: JsonSerializedSnippet) => void): void {
196
- for (const [name, scopeOrTemplate] of Object.entries(snippets ?? {})) {
197
- if (JsonSerializedSnippet.is(scopeOrTemplate)) {
198
- accept(name, scopeOrTemplate);
199
- } else {
200
- // eslint-disable-next-line @typescript-eslint/no-shadow
201
- for (const [name, template] of Object.entries(scopeOrTemplate)) {
202
- accept(name, template);
203
- }
204
- }
205
- }
206
- }
207
-
208
- push(...snippets: Snippet[]): Disposable {
209
- const toDispose = new DisposableCollection();
210
- for (const snippet of snippets) {
211
- for (const scope of snippet.scopes) {
212
- const languageSnippets = this.snippets.get(scope) || [];
213
- languageSnippets.push(snippet);
214
- this.snippets.set(scope, languageSnippets);
215
- toDispose.push(Disposable.create(() => {
216
- const index = languageSnippets.indexOf(snippet);
217
- if (index !== -1) {
218
- languageSnippets.splice(index, 1);
219
- }
220
- }));
221
- }
222
- }
223
- return toDispose;
224
- }
225
-
226
- protected isPatternInWord(patternLow: string, patternPos: number, patternLen: number, wordLow: string, wordPos: number, wordLen: number): boolean {
227
- while (patternPos < patternLen && wordPos < wordLen) {
228
- if (patternLow[patternPos] === wordLow[wordPos]) {
229
- patternPos += 1;
230
- }
231
- wordPos += 1;
232
- }
233
- return patternPos === patternLen; // pattern must be exhausted
234
- }
235
-
236
- }
237
-
238
- export interface SnippetLoadOptions {
239
- language?: string | string[]
240
- source: string
241
- }
242
-
243
- export interface JsonSerializedSnippets {
244
- [name: string]: JsonSerializedSnippet | { [name: string]: JsonSerializedSnippet };
245
- }
246
- export interface JsonSerializedSnippet {
247
- isFileTemplate?: boolean;
248
- body: string | string[];
249
- scope?: string;
250
- prefix?: string | string[];
251
- description: string;
252
- }
253
- export namespace JsonSerializedSnippet {
254
- export function is(obj: unknown): obj is JsonSerializedSnippet {
255
- return isObject(obj) && 'body' in obj;
256
- }
257
- }
258
-
259
- export interface Snippet {
260
- readonly isFileTemplate: boolean
261
- readonly scopes: string[]
262
- readonly name: string
263
- readonly prefix: string
264
- readonly description: string
265
- readonly body: string
266
- readonly source: string
267
- }
268
-
269
- export class MonacoSnippetSuggestion implements monaco.languages.CompletionItem {
270
-
271
- readonly label: string;
272
- readonly detail: string;
273
- readonly sortText: string;
274
- readonly noAutoAccept = true;
275
- readonly kind = monaco.languages.CompletionItemKind.Snippet;
276
- readonly insertTextRules = monaco.languages.CompletionItemInsertTextRule.InsertAsSnippet;
277
-
278
- insertText: string;
279
- documentation?: monaco.IMarkdownString;
280
-
281
- constructor(
282
- protected readonly snippet: Snippet,
283
- readonly range: monaco.Range
284
- ) {
285
- this.label = snippet.prefix;
286
- this.detail = `${snippet.description || snippet.name} (${snippet.source})`;
287
- this.insertText = snippet.body;
288
- this.sortText = `z-${snippet.prefix}`;
289
- this.range = range;
290
- }
291
-
292
- protected resolved = false;
293
- resolve(): MonacoSnippetSuggestion {
294
- if (!this.resolved) {
295
- const codeSnippet = new SnippetParser().parse(this.snippet.body).toString();
296
- this.documentation = { value: '```\n' + codeSnippet + '```' };
297
- this.resolved = true;
298
- }
299
- return this;
300
- }
301
-
302
- static compareByLabel(a: MonacoSnippetSuggestion, b: MonacoSnippetSuggestion): number {
303
- return a.label > b.label ? 1 : a.label < b.label ? -1 : 0;
304
- }
305
-
306
- }
1
+ // *****************************************************************************
2
+ // Copyright (C) 2019 TypeFox and others.
3
+ //
4
+ // This program and the accompanying materials are made available under the
5
+ // terms of the Eclipse Public License v. 2.0 which is available at
6
+ // http://www.eclipse.org/legal/epl-2.0.
7
+ //
8
+ // This Source Code may also be made available under the following Secondary
9
+ // Licenses when the conditions for such availability set forth in the Eclipse
10
+ // Public License v. 2.0 are satisfied: GNU General Public License, version 2
11
+ // with the GNU Classpath Exception which is available at
12
+ // https://www.gnu.org/software/classpath/license.html.
13
+ //
14
+ // SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
15
+ // *****************************************************************************
16
+ /*---------------------------------------------------------------------------------------------
17
+ * Copyright (c) Microsoft Corporation. All rights reserved.
18
+ * Licensed under the MIT License. See License.txt in the project root for license information.
19
+ *--------------------------------------------------------------------------------------------*/
20
+
21
+ import * as jsoncparser from 'jsonc-parser';
22
+ import { injectable, inject } from '@theia/core/shared/inversify';
23
+ import URI from '@theia/core/lib/common/uri';
24
+ import { Disposable, DisposableCollection } from '@theia/core/lib/common/disposable';
25
+ import { FileService } from '@theia/filesystem/lib/browser/file-service';
26
+ import { FileOperationError } from '@theia/filesystem/lib/common/files';
27
+ import * as monaco from '@theia/monaco-editor-core';
28
+ import { SnippetParser } from '@theia/monaco-editor-core/esm/vs/editor/contrib/snippet/browser/snippetParser';
29
+ import { isObject } from '@theia/core/lib/common';
30
+
31
+ @injectable()
32
+ export class MonacoSnippetSuggestProvider implements monaco.languages.CompletionItemProvider {
33
+
34
+ private static readonly _maxPrefix = 10000;
35
+
36
+ @inject(FileService)
37
+ protected readonly fileService: FileService;
38
+
39
+ protected readonly snippets = new Map<string, Snippet[]>();
40
+ protected readonly pendingSnippets = new Map<string, Promise<void>[]>();
41
+
42
+ async provideCompletionItems(model: monaco.editor.ITextModel, position: monaco.Position,
43
+ context: monaco.languages.CompletionContext): Promise<monaco.languages.CompletionList | undefined> {
44
+
45
+ // copied and modified from https://github.com/microsoft/vscode/blob/master/src/vs/workbench/contrib/snippets/browser/snippetCompletionProvider.ts
46
+ if (position.column >= MonacoSnippetSuggestProvider._maxPrefix) {
47
+ return undefined;
48
+ }
49
+
50
+ if (context.triggerKind === monaco.languages.CompletionTriggerKind.TriggerCharacter && context.triggerCharacter === ' ') {
51
+ // no snippets when suggestions have been triggered by space
52
+ return undefined;
53
+ }
54
+
55
+ const languageId = model.getLanguageId(); // TODO: look up a language id at the position
56
+ await this.loadSnippets(languageId);
57
+ const snippetsForLanguage = this.snippets.get(languageId) || [];
58
+
59
+ const pos = { lineNumber: position.lineNumber, column: 1 };
60
+ const lineOffsets: number[] = [];
61
+ const linePrefixLow = model.getLineContent(position.lineNumber).substring(0, position.column - 1).toLowerCase();
62
+ const endsInWhitespace = linePrefixLow.match(/\s$/);
63
+
64
+ while (pos.column < position.column) {
65
+ const word = model.getWordAtPosition(pos);
66
+ if (word) {
67
+ // at a word
68
+ lineOffsets.push(word.startColumn - 1);
69
+ pos.column = word.endColumn + 1;
70
+ if (word.endColumn - 1 < linePrefixLow.length && !/\s/.test(linePrefixLow[word.endColumn - 1])) {
71
+ lineOffsets.push(word.endColumn - 1);
72
+ }
73
+ } else if (!/\s/.test(linePrefixLow[pos.column - 1])) {
74
+ // at a none-whitespace character
75
+ lineOffsets.push(pos.column - 1);
76
+ pos.column += 1;
77
+ } else {
78
+ // always advance!
79
+ pos.column += 1;
80
+ }
81
+ }
82
+
83
+ const availableSnippets = new Set<Snippet>();
84
+ snippetsForLanguage.forEach(availableSnippets.add, availableSnippets);
85
+ const suggestions: MonacoSnippetSuggestion[] = [];
86
+ for (const start of lineOffsets) {
87
+ availableSnippets.forEach(snippet => {
88
+ if (this.isPatternInWord(linePrefixLow, start, linePrefixLow.length, snippet.prefix.toLowerCase(), 0, snippet.prefix.length)) {
89
+ suggestions.push(new MonacoSnippetSuggestion(snippet, monaco.Range.fromPositions(position.delta(0, -(linePrefixLow.length - start)), position)));
90
+ availableSnippets.delete(snippet);
91
+ }
92
+ });
93
+ }
94
+ if (endsInWhitespace || lineOffsets.length === 0) {
95
+ // add remaining snippets when the current prefix ends in whitespace or when no
96
+ // interesting positions have been found
97
+ availableSnippets.forEach(snippet => {
98
+ suggestions.push(new MonacoSnippetSuggestion(snippet, monaco.Range.fromPositions(position)));
99
+ });
100
+ }
101
+
102
+ // disambiguate suggestions with same labels
103
+ suggestions.sort(MonacoSnippetSuggestion.compareByLabel);
104
+ return { suggestions };
105
+ }
106
+
107
+ resolveCompletionItem?(item: monaco.languages.CompletionItem, token: monaco.CancellationToken): monaco.languages.CompletionItem {
108
+ return item instanceof MonacoSnippetSuggestion ? item.resolve() : item;
109
+ }
110
+
111
+ protected async loadSnippets(scope: string): Promise<void> {
112
+ const pending: Promise<void>[] = [];
113
+ pending.push(...(this.pendingSnippets.get(scope) || []));
114
+ pending.push(...(this.pendingSnippets.get('*') || []));
115
+ if (pending.length) {
116
+ await Promise.all(pending);
117
+ }
118
+ }
119
+
120
+ fromURI(uri: string | URI, options: SnippetLoadOptions): Disposable {
121
+ const toDispose = new DisposableCollection(Disposable.create(() => { /* mark as not disposed */ }));
122
+ const pending = this.loadURI(uri, options, toDispose);
123
+ const { language } = options;
124
+ const scopes = Array.isArray(language) ? language : !!language ? [language] : ['*'];
125
+ for (const scope of scopes) {
126
+ const pendingSnippets = this.pendingSnippets.get(scope) || [];
127
+ pendingSnippets.push(pending);
128
+ this.pendingSnippets.set(scope, pendingSnippets);
129
+ toDispose.push(Disposable.create(() => {
130
+ const index = pendingSnippets.indexOf(pending);
131
+ if (index !== -1) {
132
+ pendingSnippets.splice(index, 1);
133
+ }
134
+ }));
135
+ }
136
+ return toDispose;
137
+ }
138
+
139
+ /**
140
+ * should NOT throw to prevent load errors on suggest
141
+ */
142
+ protected async loadURI(uri: string | URI, options: SnippetLoadOptions, toDispose: DisposableCollection): Promise<void> {
143
+ try {
144
+ const resource = typeof uri === 'string' ? new URI(uri) : uri;
145
+ const { value } = await this.fileService.read(resource);
146
+ if (toDispose.disposed) {
147
+ return;
148
+ }
149
+ const snippets = value && jsoncparser.parse(value, undefined, { disallowComments: false });
150
+ toDispose.push(this.fromJSON(snippets, options));
151
+ } catch (e) {
152
+ if (!(e instanceof FileOperationError)) {
153
+ console.error(e);
154
+ }
155
+ }
156
+ }
157
+
158
+ fromJSON(snippets: JsonSerializedSnippets | undefined, { language, source }: SnippetLoadOptions): Disposable {
159
+ const toDispose = new DisposableCollection();
160
+ this.parseSnippets(snippets, (name, snippet) => {
161
+ const { isFileTemplate, prefix, body, description } = snippet;
162
+ const parsedBody = Array.isArray(body) ? body.join('\n') : body;
163
+ const parsedPrefixes = !prefix ? [''] : Array.isArray(prefix) ? prefix : [prefix];
164
+
165
+ if (typeof parsedBody !== 'string') {
166
+ return;
167
+ }
168
+ const scopes: string[] = [];
169
+ if (language) {
170
+ if (Array.isArray(language)) {
171
+ scopes.push(...language);
172
+ } else {
173
+ scopes.push(language);
174
+ }
175
+ } else if (typeof snippet.scope === 'string') {
176
+ for (const rawScope of snippet.scope.split(',')) {
177
+ const scope = rawScope.trim();
178
+ if (scope) {
179
+ scopes.push(scope);
180
+ }
181
+ }
182
+ }
183
+ parsedPrefixes.forEach(parsedPrefix => toDispose.push(this.push({
184
+ isFileTemplate: Boolean(isFileTemplate),
185
+ scopes,
186
+ name,
187
+ prefix: parsedPrefix,
188
+ description,
189
+ body: parsedBody,
190
+ source
191
+ })));
192
+ });
193
+ return toDispose;
194
+ }
195
+ protected parseSnippets(snippets: JsonSerializedSnippets | undefined, accept: (name: string, snippet: JsonSerializedSnippet) => void): void {
196
+ for (const [name, scopeOrTemplate] of Object.entries(snippets ?? {})) {
197
+ if (JsonSerializedSnippet.is(scopeOrTemplate)) {
198
+ accept(name, scopeOrTemplate);
199
+ } else {
200
+ // eslint-disable-next-line @typescript-eslint/no-shadow
201
+ for (const [name, template] of Object.entries(scopeOrTemplate)) {
202
+ accept(name, template);
203
+ }
204
+ }
205
+ }
206
+ }
207
+
208
+ push(...snippets: Snippet[]): Disposable {
209
+ const toDispose = new DisposableCollection();
210
+ for (const snippet of snippets) {
211
+ for (const scope of snippet.scopes) {
212
+ const languageSnippets = this.snippets.get(scope) || [];
213
+ languageSnippets.push(snippet);
214
+ this.snippets.set(scope, languageSnippets);
215
+ toDispose.push(Disposable.create(() => {
216
+ const index = languageSnippets.indexOf(snippet);
217
+ if (index !== -1) {
218
+ languageSnippets.splice(index, 1);
219
+ }
220
+ }));
221
+ }
222
+ }
223
+ return toDispose;
224
+ }
225
+
226
+ protected isPatternInWord(patternLow: string, patternPos: number, patternLen: number, wordLow: string, wordPos: number, wordLen: number): boolean {
227
+ while (patternPos < patternLen && wordPos < wordLen) {
228
+ if (patternLow[patternPos] === wordLow[wordPos]) {
229
+ patternPos += 1;
230
+ }
231
+ wordPos += 1;
232
+ }
233
+ return patternPos === patternLen; // pattern must be exhausted
234
+ }
235
+
236
+ }
237
+
238
+ export interface SnippetLoadOptions {
239
+ language?: string | string[]
240
+ source: string
241
+ }
242
+
243
+ export interface JsonSerializedSnippets {
244
+ [name: string]: JsonSerializedSnippet | { [name: string]: JsonSerializedSnippet };
245
+ }
246
+ export interface JsonSerializedSnippet {
247
+ isFileTemplate?: boolean;
248
+ body: string | string[];
249
+ scope?: string;
250
+ prefix?: string | string[];
251
+ description: string;
252
+ }
253
+ export namespace JsonSerializedSnippet {
254
+ export function is(obj: unknown): obj is JsonSerializedSnippet {
255
+ return isObject(obj) && 'body' in obj;
256
+ }
257
+ }
258
+
259
+ export interface Snippet {
260
+ readonly isFileTemplate: boolean
261
+ readonly scopes: string[]
262
+ readonly name: string
263
+ readonly prefix: string
264
+ readonly description: string
265
+ readonly body: string
266
+ readonly source: string
267
+ }
268
+
269
+ export class MonacoSnippetSuggestion implements monaco.languages.CompletionItem {
270
+
271
+ readonly label: string;
272
+ readonly detail: string;
273
+ readonly sortText: string;
274
+ readonly noAutoAccept = true;
275
+ readonly kind = monaco.languages.CompletionItemKind.Snippet;
276
+ readonly insertTextRules = monaco.languages.CompletionItemInsertTextRule.InsertAsSnippet;
277
+
278
+ insertText: string;
279
+ documentation?: monaco.IMarkdownString;
280
+
281
+ constructor(
282
+ protected readonly snippet: Snippet,
283
+ readonly range: monaco.Range
284
+ ) {
285
+ this.label = snippet.prefix;
286
+ this.detail = `${snippet.description || snippet.name} (${snippet.source})`;
287
+ this.insertText = snippet.body;
288
+ this.sortText = `z-${snippet.prefix}`;
289
+ this.range = range;
290
+ }
291
+
292
+ protected resolved = false;
293
+ resolve(): MonacoSnippetSuggestion {
294
+ if (!this.resolved) {
295
+ const codeSnippet = new SnippetParser().parse(this.snippet.body).toString();
296
+ this.documentation = { value: '```\n' + codeSnippet + '```' };
297
+ this.resolved = true;
298
+ }
299
+ return this;
300
+ }
301
+
302
+ static compareByLabel(a: MonacoSnippetSuggestion, b: MonacoSnippetSuggestion): number {
303
+ return a.label > b.label ? 1 : a.label < b.label ? -1 : 0;
304
+ }
305
+
306
+ }