1 | ;
|
2 | var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
3 | function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
4 | return new (P || (P = Promise))(function (resolve, reject) {
|
5 | function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
6 | function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
7 | function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
8 | step((generator = generator.apply(thisArg, _arguments || [])).next());
|
9 | });
|
10 | };
|
11 | Object.defineProperty(exports, "__esModule", { value: true });
|
12 | exports.TextEditorSyncAdapter = void 0;
|
13 | const convert_1 = require("../convert");
|
14 | const languageclient_1 = require("../languageclient");
|
15 | const apply_edit_adapter_1 = require("./apply-edit-adapter");
|
16 | const atom_1 = require("atom");
|
17 | const Utils = require("../utils");
|
18 | /**
|
19 | * Public: Synchronizes the documents between Atom and the language server by notifying each end of changes, opening,
|
20 | * closing and other events as well as sending and applying changes either in whole or in part depending on what the
|
21 | * language server supports.
|
22 | */
|
23 | class DocumentSyncAdapter {
|
24 | /**
|
25 | * Public: Create a new {DocumentSyncAdapter} for the given language server.
|
26 | *
|
27 | * @param _connection A {LanguageClientConnection} to the language server to be kept in sync.
|
28 | * @param documentSync The document syncing options.
|
29 | * @param _editorSelector A predicate function that takes a {TextEditor} and returns a {boolean} indicating whether
|
30 | * this adapter should care about the contents of the editor.
|
31 | * @param _getLanguageIdFromEditor A function that returns a {string} of `languageId` used for `textDocument/didOpen`
|
32 | * notification.
|
33 | */
|
34 | constructor(_connection, _editorSelector, documentSync, _reportBusyWhile, _getLanguageIdFromEditor) {
|
35 | this._connection = _connection;
|
36 | this._editorSelector = _editorSelector;
|
37 | this._reportBusyWhile = _reportBusyWhile;
|
38 | this._getLanguageIdFromEditor = _getLanguageIdFromEditor;
|
39 | this._disposable = new atom_1.CompositeDisposable();
|
40 | this._editors = new WeakMap();
|
41 | this._versions = new Map();
|
42 | if (typeof documentSync === "object") {
|
43 | this._documentSync = documentSync;
|
44 | }
|
45 | else {
|
46 | this._documentSync = {
|
47 | change: documentSync || languageclient_1.TextDocumentSyncKind.Full,
|
48 | };
|
49 | }
|
50 | this._disposable.add(atom.textEditors.observe(this.observeTextEditor.bind(this)));
|
51 | }
|
52 | /**
|
53 | * Public: Determine whether this adapter can be used to adapt a language server based on the serverCapabilities
|
54 | * matrix textDocumentSync capability either being Full or Incremental.
|
55 | *
|
56 | * @param serverCapabilities The {ServerCapabilities} of the language server to consider.
|
57 | * @returns A {Boolean} indicating adapter can adapt the server based on the given serverCapabilities.
|
58 | */
|
59 | static canAdapt(serverCapabilities) {
|
60 | return this.canAdaptV2(serverCapabilities) || this.canAdaptV3(serverCapabilities);
|
61 | }
|
62 | static canAdaptV2(serverCapabilities) {
|
63 | return (serverCapabilities.textDocumentSync === languageclient_1.TextDocumentSyncKind.Incremental ||
|
64 | serverCapabilities.textDocumentSync === languageclient_1.TextDocumentSyncKind.Full);
|
65 | }
|
66 | static canAdaptV3(serverCapabilities) {
|
67 | const options = serverCapabilities.textDocumentSync;
|
68 | return (options !== null &&
|
69 | typeof options === "object" &&
|
70 | (options.change === languageclient_1.TextDocumentSyncKind.Incremental || options.change === languageclient_1.TextDocumentSyncKind.Full));
|
71 | }
|
72 | /** Dispose this adapter ensuring any resources are freed and events unhooked. */
|
73 | dispose() {
|
74 | this._disposable.dispose();
|
75 | }
|
76 | /**
|
77 | * Examine a {TextEditor} and decide if we wish to observe it. If so ensure that we stop observing it when it is
|
78 | * closed or otherwise destroyed.
|
79 | *
|
80 | * @param editor A {TextEditor} to consider for observation.
|
81 | */
|
82 | observeTextEditor(editor) {
|
83 | const listener = editor.observeGrammar((_grammar) => this._handleGrammarChange(editor));
|
84 | this._disposable.add(editor.onDidDestroy(() => {
|
85 | this._disposable.remove(listener);
|
86 | listener.dispose();
|
87 | }));
|
88 | this._disposable.add(listener);
|
89 | if (!this._editors.has(editor) && this._editorSelector(editor)) {
|
90 | this._handleNewEditor(editor);
|
91 | }
|
92 | }
|
93 | _handleGrammarChange(editor) {
|
94 | const sync = this._editors.get(editor);
|
95 | if (sync != null && !this._editorSelector(editor)) {
|
96 | this._editors.delete(editor);
|
97 | this._disposable.remove(sync);
|
98 | sync.didClose();
|
99 | sync.dispose();
|
100 | }
|
101 | else if (sync == null && this._editorSelector(editor)) {
|
102 | this._handleNewEditor(editor);
|
103 | }
|
104 | }
|
105 | _handleNewEditor(editor) {
|
106 | const sync = new TextEditorSyncAdapter(editor, this._connection, this._documentSync, this._versions, this._reportBusyWhile, this._getLanguageIdFromEditor);
|
107 | this._editors.set(editor, sync);
|
108 | this._disposable.add(sync);
|
109 | this._disposable.add(editor.onDidDestroy(() => {
|
110 | const destroyedSync = this._editors.get(editor);
|
111 | if (destroyedSync) {
|
112 | this._editors.delete(editor);
|
113 | this._disposable.remove(destroyedSync);
|
114 | destroyedSync.dispose();
|
115 | }
|
116 | }));
|
117 | }
|
118 | getEditorSyncAdapter(editor) {
|
119 | return this._editors.get(editor);
|
120 | }
|
121 | }
|
122 | exports.default = DocumentSyncAdapter;
|
123 | /** Public: Keep a single {TextEditor} in sync with a given language server. */
|
124 | class TextEditorSyncAdapter {
|
125 | /**
|
126 | * Public: Create a {TextEditorSyncAdapter} in sync with a given language server.
|
127 | *
|
128 | * @param _editor A {TextEditor} to keep in sync.
|
129 | * @param _connection A {LanguageClientConnection} to a language server to keep in sync.
|
130 | * @param _documentSync The document syncing options.
|
131 | */
|
132 | constructor(_editor, _connection, _documentSync, _versions, _reportBusyWhile, _getLanguageIdFromEditor) {
|
133 | this._editor = _editor;
|
134 | this._connection = _connection;
|
135 | this._documentSync = _documentSync;
|
136 | this._versions = _versions;
|
137 | this._reportBusyWhile = _reportBusyWhile;
|
138 | this._getLanguageIdFromEditor = _getLanguageIdFromEditor;
|
139 | this._disposable = new atom_1.CompositeDisposable();
|
140 | this._fakeDidChangeWatchedFiles = atom.project.onDidChangeFiles == null;
|
141 | const changeTracking = this.setupChangeTracking(_documentSync);
|
142 | if (changeTracking != null) {
|
143 | this._disposable.add(changeTracking);
|
144 | }
|
145 | // These handlers are attached only if server supports them
|
146 | if (_documentSync.willSave) {
|
147 | this._disposable.add(_editor.getBuffer().onWillSave(this.willSave.bind(this)));
|
148 | }
|
149 | if (_documentSync.willSaveWaitUntil) {
|
150 | this._disposable.add(_editor.getBuffer().onWillSave(this.willSaveWaitUntil.bind(this)));
|
151 | }
|
152 | // Send close notifications unless it's explicitly disabled
|
153 | if (_documentSync.openClose !== false) {
|
154 | this._disposable.add(_editor.onDidDestroy(this.didClose.bind(this)));
|
155 | }
|
156 | this._disposable.add(_editor.onDidSave(this.didSave.bind(this)), _editor.onDidChangePath(this.didRename.bind(this)));
|
157 | this._currentUri = this.getEditorUri();
|
158 | if (_documentSync.openClose !== false) {
|
159 | this.didOpen();
|
160 | }
|
161 | }
|
162 | /** The change tracking disposable listener that will ensure that changes are sent to the language server as appropriate. */
|
163 | setupChangeTracking(documentSync) {
|
164 | switch (documentSync.change) {
|
165 | case languageclient_1.TextDocumentSyncKind.Full:
|
166 | return this._editor.onDidChange(this.sendFullChanges.bind(this));
|
167 | case languageclient_1.TextDocumentSyncKind.Incremental:
|
168 | return this._editor.getBuffer().onDidChangeText(this.sendIncrementalChanges.bind(this));
|
169 | }
|
170 | return null;
|
171 | }
|
172 | /** Dispose this adapter ensuring any resources are freed and events unhooked. */
|
173 | dispose() {
|
174 | this._disposable.dispose();
|
175 | }
|
176 | /** Get the languageId field that will be sent to the language server by simply using the `_getLanguageIdFromEditor`. */
|
177 | getLanguageId() {
|
178 | return this._getLanguageIdFromEditor(this._editor);
|
179 | }
|
180 | /**
|
181 | * Public: Create a {VersionedTextDocumentIdentifier} for the document observed by this adapter including both the Uri
|
182 | * and the current Version.
|
183 | */
|
184 | getVersionedTextDocumentIdentifier() {
|
185 | return {
|
186 | uri: this.getEditorUri(),
|
187 | version: this._getVersion(this._editor.getPath() || ""),
|
188 | };
|
189 | }
|
190 | /** Public: Send the entire document to the language server. This is used when operating in Full (1) sync mode. */
|
191 | sendFullChanges() {
|
192 | if (!this._isPrimaryAdapter()) {
|
193 | return;
|
194 | } // Multiple editors, we are not first
|
195 | this._bumpVersion();
|
196 | this._connection.didChangeTextDocument({
|
197 | textDocument: this.getVersionedTextDocumentIdentifier(),
|
198 | contentChanges: [{ text: this._editor.getText() }],
|
199 | });
|
200 | }
|
201 | /**
|
202 | * Public: Send the incremental text changes to the language server. This is used when operating in Incremental (2) sync mode.
|
203 | *
|
204 | * @param event The event fired by Atom to indicate the document has stopped changing including a list of changes
|
205 | * since the last time this event fired for this text editor. NOTE: The order of changes in the event is guaranteed
|
206 | * top to bottom. Language server expects this in reverse.
|
207 | */
|
208 | sendIncrementalChanges(event) {
|
209 | if (event.changes.length > 0) {
|
210 | if (!this._isPrimaryAdapter()) {
|
211 | return;
|
212 | } // Multiple editors, we are not first
|
213 | this._bumpVersion();
|
214 | this._connection.didChangeTextDocument({
|
215 | textDocument: this.getVersionedTextDocumentIdentifier(),
|
216 | contentChanges: event.changes.map(TextEditorSyncAdapter.textEditToContentChange).reverse(),
|
217 | });
|
218 | }
|
219 | }
|
220 | /**
|
221 | * Public: Convert an Atom {TextEditEvent} to a language server {TextDocumentContentChangeEvent} object.
|
222 | *
|
223 | * @param change The Atom {TextEditEvent} to convert.
|
224 | * @returns A {TextDocumentContentChangeEvent} that represents the converted {TextEditEvent}.
|
225 | */
|
226 | static textEditToContentChange(change) {
|
227 | return {
|
228 | range: convert_1.default.atomRangeToLSRange(change.oldRange),
|
229 | rangeLength: change.oldText.length,
|
230 | text: change.newText,
|
231 | };
|
232 | }
|
233 | _isPrimaryAdapter() {
|
234 | const lowestIdForBuffer = Math.min(...atom.workspace
|
235 | .getTextEditors()
|
236 | .filter((t) => t.getBuffer() === this._editor.getBuffer())
|
237 | .map((t) => t.id));
|
238 | return lowestIdForBuffer === this._editor.id;
|
239 | }
|
240 | _bumpVersion() {
|
241 | const filePath = this._editor.getPath();
|
242 | if (filePath == null) {
|
243 | return;
|
244 | }
|
245 | this._versions.set(filePath, this._getVersion(filePath) + 1);
|
246 | }
|
247 | /**
|
248 | * Ensure when the document is opened we send notification to the language server so it can load it in and keep track
|
249 | * of diagnostics etc.
|
250 | */
|
251 | didOpen() {
|
252 | const filePath = this._editor.getPath();
|
253 | if (filePath == null) {
|
254 | return;
|
255 | } // Not yet saved
|
256 | if (!this._isPrimaryAdapter()) {
|
257 | return;
|
258 | } // Multiple editors, we are not first
|
259 | this._connection.didOpenTextDocument({
|
260 | textDocument: {
|
261 | uri: this.getEditorUri(),
|
262 | languageId: this.getLanguageId().toLowerCase(),
|
263 | version: this._getVersion(filePath),
|
264 | text: this._editor.getText(),
|
265 | },
|
266 | });
|
267 | }
|
268 | _getVersion(filePath) {
|
269 | return this._versions.get(filePath) || 1;
|
270 | }
|
271 | /** Called when the {TextEditor} is closed and sends the 'didCloseTextDocument' notification to the connected language server. */
|
272 | didClose() {
|
273 | if (this._editor.getPath() == null) {
|
274 | return;
|
275 | } // Not yet saved
|
276 | const fileStillOpen = atom.workspace.getTextEditors().find((t) => t.getBuffer() === this._editor.getBuffer());
|
277 | if (fileStillOpen) {
|
278 | return; // Other windows or editors still have this file open
|
279 | }
|
280 | this._connection.didCloseTextDocument({ textDocument: { uri: this.getEditorUri() } });
|
281 | }
|
282 | /** Called just before the {TextEditor} saves and sends the 'willSaveTextDocument' notification to the connected language server. */
|
283 | willSave() {
|
284 | if (!this._isPrimaryAdapter()) {
|
285 | return;
|
286 | }
|
287 | const uri = this.getEditorUri();
|
288 | this._connection.willSaveTextDocument({
|
289 | textDocument: { uri },
|
290 | reason: languageclient_1.TextDocumentSaveReason.Manual,
|
291 | });
|
292 | }
|
293 | /**
|
294 | * Called just before the {TextEditor} saves, sends the 'willSaveWaitUntilTextDocument' request to the connected
|
295 | * language server and waits for the response before saving the buffer.
|
296 | */
|
297 | willSaveWaitUntil() {
|
298 | return __awaiter(this, void 0, void 0, function* () {
|
299 | if (!this._isPrimaryAdapter()) {
|
300 | return Promise.resolve();
|
301 | }
|
302 | const buffer = this._editor.getBuffer();
|
303 | const uri = this.getEditorUri();
|
304 | const title = this._editor.getLongTitle();
|
305 | const applyEditsOrTimeout = Utils.promiseWithTimeout(2500, // 2.5 seconds timeout
|
306 | this._connection.willSaveWaitUntilTextDocument({
|
307 | textDocument: { uri },
|
308 | reason: languageclient_1.TextDocumentSaveReason.Manual,
|
309 | }))
|
310 | .then((edits) => {
|
311 | const cursor = this._editor.getCursorBufferPosition();
|
312 | apply_edit_adapter_1.default.applyEdits(buffer, convert_1.default.convertLsTextEdits(edits));
|
313 | this._editor.setCursorBufferPosition(cursor);
|
314 | })
|
315 | .catch((err) => {
|
316 | atom.notifications.addError("On-save action failed", {
|
317 | description: `Failed to apply edits to ${title}`,
|
318 | detail: err.message,
|
319 | });
|
320 | return;
|
321 | });
|
322 | const withBusySignal = this._reportBusyWhile(`Applying on-save edits for ${title}`, () => applyEditsOrTimeout);
|
323 | return withBusySignal || applyEditsOrTimeout;
|
324 | });
|
325 | }
|
326 | /**
|
327 | * Called when the {TextEditor} saves and sends the 'didSaveTextDocument' notification to the connected language
|
328 | * server. Note: Right now this also sends the `didChangeWatchedFiles` notification as well but that will be sent from
|
329 | * elsewhere soon.
|
330 | */
|
331 | didSave() {
|
332 | if (!this._isPrimaryAdapter()) {
|
333 | return;
|
334 | }
|
335 | const uri = this.getEditorUri();
|
336 | const didSaveNotification = {
|
337 | textDocument: { uri, version: this._getVersion(uri) },
|
338 | };
|
339 | if (typeof this._documentSync.save === "object" && this._documentSync.save.includeText) {
|
340 | didSaveNotification.text = this._editor.getText();
|
341 | }
|
342 | this._connection.didSaveTextDocument(didSaveNotification);
|
343 | if (this._fakeDidChangeWatchedFiles) {
|
344 | this._connection.didChangeWatchedFiles({
|
345 | changes: [{ uri, type: languageclient_1.FileChangeType.Changed }],
|
346 | });
|
347 | }
|
348 | }
|
349 | didRename() {
|
350 | if (!this._isPrimaryAdapter()) {
|
351 | return;
|
352 | }
|
353 | const oldUri = this._currentUri;
|
354 | this._currentUri = this.getEditorUri();
|
355 | if (!oldUri) {
|
356 | return; // Didn't previously have a name
|
357 | }
|
358 | if (this._documentSync.openClose !== false) {
|
359 | this._connection.didCloseTextDocument({ textDocument: { uri: oldUri } });
|
360 | }
|
361 | if (this._fakeDidChangeWatchedFiles) {
|
362 | this._connection.didChangeWatchedFiles({
|
363 | changes: [
|
364 | { uri: oldUri, type: languageclient_1.FileChangeType.Deleted },
|
365 | { uri: this._currentUri, type: languageclient_1.FileChangeType.Created },
|
366 | ],
|
367 | });
|
368 | }
|
369 | // Send an equivalent open event for this editor, which will now use the new
|
370 | // file path.
|
371 | if (this._documentSync.openClose !== false) {
|
372 | this.didOpen();
|
373 | }
|
374 | }
|
375 | /** Public: Obtain the current {TextEditor} path and convert it to a Uri. */
|
376 | getEditorUri() {
|
377 | return convert_1.default.pathToUri(this._editor.getPath() || "");
|
378 | }
|
379 | }
|
380 | exports.TextEditorSyncAdapter = TextEditorSyncAdapter;
|
381 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"document-sync-adapter.js","sourceRoot":"","sources":["../../../lib/adapters/document-sync-adapter.ts"],"names":[],"mappings":"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","sourcesContent":["import Convert from \"../convert\"\nimport {\n  LanguageClientConnection,\n  FileChangeType,\n  TextDocumentSaveReason,\n  TextDocumentSyncKind,\n  TextDocumentSyncOptions,\n  TextDocumentContentChangeEvent,\n  VersionedTextDocumentIdentifier,\n  ServerCapabilities,\n  DidSaveTextDocumentParams,\n} from \"../languageclient\"\nimport ApplyEditAdapter from \"./apply-edit-adapter\"\nimport { CompositeDisposable, Disposable, TextEditor, BufferStoppedChangingEvent, TextChange } from \"atom\"\nimport * as Utils from \"../utils\"\n\n/**\n * Public: Synchronizes the documents between Atom and the language server by notifying each end of changes, opening,\n * closing and other events as well as sending and applying changes either in whole or in part depending on what the\n * language server supports.\n */\nexport default class DocumentSyncAdapter {\n  private _disposable = new CompositeDisposable()\n  public _documentSync: TextDocumentSyncOptions\n  private _editors: WeakMap<TextEditor, TextEditorSyncAdapter> = new WeakMap()\n  private _versions: Map<string, number> = new Map()\n\n  /**\n   * Public: Determine whether this adapter can be used to adapt a language server based on the serverCapabilities\n   * matrix textDocumentSync capability either being Full or Incremental.\n   *\n   * @param serverCapabilities The {ServerCapabilities} of the language server to consider.\n   * @returns A {Boolean} indicating adapter can adapt the server based on the given serverCapabilities.\n   */\n  public static canAdapt(serverCapabilities: ServerCapabilities): boolean {\n    return this.canAdaptV2(serverCapabilities) || this.canAdaptV3(serverCapabilities)\n  }\n\n  private static canAdaptV2(serverCapabilities: ServerCapabilities): boolean {\n    return (\n      serverCapabilities.textDocumentSync === TextDocumentSyncKind.Incremental ||\n      serverCapabilities.textDocumentSync === TextDocumentSyncKind.Full\n    )\n  }\n\n  private static canAdaptV3(serverCapabilities: ServerCapabilities): boolean {\n    const options = serverCapabilities.textDocumentSync\n    return (\n      options !== null &&\n      typeof options === \"object\" &&\n      (options.change === TextDocumentSyncKind.Incremental || options.change === TextDocumentSyncKind.Full)\n    )\n  }\n\n  /**\n   * Public: Create a new {DocumentSyncAdapter} for the given language server.\n   *\n   * @param _connection A {LanguageClientConnection} to the language server to be kept in sync.\n   * @param documentSync The document syncing options.\n   * @param _editorSelector A predicate function that takes a {TextEditor} and returns a {boolean} indicating whether\n   *   this adapter should care about the contents of the editor.\n   * @param _getLanguageIdFromEditor A function that returns a {string} of `languageId` used for `textDocument/didOpen`\n   *   notification.\n   */\n  constructor(\n    private _connection: LanguageClientConnection,\n    private _editorSelector: (editor: TextEditor) => boolean,\n    documentSync: TextDocumentSyncOptions | TextDocumentSyncKind | undefined,\n    private _reportBusyWhile: Utils.ReportBusyWhile,\n    private _getLanguageIdFromEditor: (editor: TextEditor) => string\n  ) {\n    if (typeof documentSync === \"object\") {\n      this._documentSync = documentSync\n    } else {\n      this._documentSync = {\n        change: documentSync || TextDocumentSyncKind.Full,\n      }\n    }\n    this._disposable.add(atom.textEditors.observe(this.observeTextEditor.bind(this)))\n  }\n\n  /** Dispose this adapter ensuring any resources are freed and events unhooked. */\n  public dispose(): void {\n    this._disposable.dispose()\n  }\n\n  /**\n   * Examine a {TextEditor} and decide if we wish to observe it. If so ensure that we stop observing it when it is\n   * closed or otherwise destroyed.\n   *\n   * @param editor A {TextEditor} to consider for observation.\n   */\n  public observeTextEditor(editor: TextEditor): void {\n    const listener = editor.observeGrammar((_grammar) => this._handleGrammarChange(editor))\n    this._disposable.add(\n      editor.onDidDestroy(() => {\n        this._disposable.remove(listener)\n        listener.dispose()\n      })\n    )\n    this._disposable.add(listener)\n    if (!this._editors.has(editor) && this._editorSelector(editor)) {\n      this._handleNewEditor(editor)\n    }\n  }\n\n  private _handleGrammarChange(editor: TextEditor): void {\n    const sync = this._editors.get(editor)\n    if (sync != null && !this._editorSelector(editor)) {\n      this._editors.delete(editor)\n      this._disposable.remove(sync)\n      sync.didClose()\n      sync.dispose()\n    } else if (sync == null && this._editorSelector(editor)) {\n      this._handleNewEditor(editor)\n    }\n  }\n\n  private _handleNewEditor(editor: TextEditor): void {\n    const sync = new TextEditorSyncAdapter(\n      editor,\n      this._connection,\n      this._documentSync,\n      this._versions,\n      this._reportBusyWhile,\n      this._getLanguageIdFromEditor\n    )\n    this._editors.set(editor, sync)\n    this._disposable.add(sync)\n    this._disposable.add(\n      editor.onDidDestroy(() => {\n        const destroyedSync = this._editors.get(editor)\n        if (destroyedSync) {\n          this._editors.delete(editor)\n          this._disposable.remove(destroyedSync)\n          destroyedSync.dispose()\n        }\n      })\n    )\n  }\n\n  public getEditorSyncAdapter(editor: TextEditor): TextEditorSyncAdapter | undefined {\n    return this._editors.get(editor)\n  }\n}\n\n/** Public: Keep a single {TextEditor} in sync with a given language server. */\nexport class TextEditorSyncAdapter {\n  private _disposable = new CompositeDisposable()\n  private _currentUri: string\n  private _fakeDidChangeWatchedFiles: boolean\n\n  /**\n   * Public: Create a {TextEditorSyncAdapter} in sync with a given language server.\n   *\n   * @param _editor A {TextEditor} to keep in sync.\n   * @param _connection A {LanguageClientConnection} to a language server to keep in sync.\n   * @param _documentSync The document syncing options.\n   */\n  constructor(\n    private _editor: TextEditor,\n    private _connection: LanguageClientConnection,\n    private _documentSync: TextDocumentSyncOptions,\n    private _versions: Map<string, number>,\n    private _reportBusyWhile: Utils.ReportBusyWhile,\n    private _getLanguageIdFromEditor: (editor: TextEditor) => string\n  ) {\n    this._fakeDidChangeWatchedFiles = atom.project.onDidChangeFiles == null\n\n    const changeTracking = this.setupChangeTracking(_documentSync)\n    if (changeTracking != null) {\n      this._disposable.add(changeTracking)\n    }\n\n    // These handlers are attached only if server supports them\n    if (_documentSync.willSave) {\n      this._disposable.add(_editor.getBuffer().onWillSave(this.willSave.bind(this)))\n    }\n    if (_documentSync.willSaveWaitUntil) {\n      this._disposable.add(_editor.getBuffer().onWillSave(this.willSaveWaitUntil.bind(this)))\n    }\n    // Send close notifications unless it's explicitly disabled\n    if (_documentSync.openClose !== false) {\n      this._disposable.add(_editor.onDidDestroy(this.didClose.bind(this)))\n    }\n    this._disposable.add(_editor.onDidSave(this.didSave.bind(this)), _editor.onDidChangePath(this.didRename.bind(this)))\n\n    this._currentUri = this.getEditorUri()\n\n    if (_documentSync.openClose !== false) {\n      this.didOpen()\n    }\n  }\n\n  /** The change tracking disposable listener that will ensure that changes are sent to the language server as appropriate. */\n  public setupChangeTracking(documentSync: TextDocumentSyncOptions): Disposable | null {\n    switch (documentSync.change) {\n      case TextDocumentSyncKind.Full:\n        return this._editor.onDidChange(this.sendFullChanges.bind(this))\n      case TextDocumentSyncKind.Incremental:\n        return this._editor.getBuffer().onDidChangeText(this.sendIncrementalChanges.bind(this))\n    }\n    return null\n  }\n\n  /** Dispose this adapter ensuring any resources are freed and events unhooked. */\n  public dispose(): void {\n    this._disposable.dispose()\n  }\n\n  /** Get the languageId field that will be sent to the language server by simply using the `_getLanguageIdFromEditor`. */\n  public getLanguageId(): string {\n    return this._getLanguageIdFromEditor(this._editor)\n  }\n\n  /**\n   * Public: Create a {VersionedTextDocumentIdentifier} for the document observed by this adapter including both the Uri\n   * and the current Version.\n   */\n  public getVersionedTextDocumentIdentifier(): VersionedTextDocumentIdentifier {\n    return {\n      uri: this.getEditorUri(),\n      version: this._getVersion(this._editor.getPath() || \"\"),\n    }\n  }\n\n  /** Public: Send the entire document to the language server. This is used when operating in Full (1) sync mode. */\n  public sendFullChanges(): void {\n    if (!this._isPrimaryAdapter()) {\n      return\n    } // Multiple editors, we are not first\n\n    this._bumpVersion()\n    this._connection.didChangeTextDocument({\n      textDocument: this.getVersionedTextDocumentIdentifier(),\n      contentChanges: [{ text: this._editor.getText() }],\n    })\n  }\n\n  /**\n   * Public: Send the incremental text changes to the language server. This is used when operating in Incremental (2) sync mode.\n   *\n   * @param event The event fired by Atom to indicate the document has stopped changing including a list of changes\n   *   since the last time this event fired for this text editor. NOTE: The order of changes in the event is guaranteed\n   *   top to bottom. Language server expects this in reverse.\n   */\n  public sendIncrementalChanges(event: BufferStoppedChangingEvent): void {\n    if (event.changes.length > 0) {\n      if (!this._isPrimaryAdapter()) {\n        return\n      } // Multiple editors, we are not first\n\n      this._bumpVersion()\n      this._connection.didChangeTextDocument({\n        textDocument: this.getVersionedTextDocumentIdentifier(),\n        contentChanges: event.changes.map(TextEditorSyncAdapter.textEditToContentChange).reverse(),\n      })\n    }\n  }\n\n  /**\n   * Public: Convert an Atom {TextEditEvent} to a language server {TextDocumentContentChangeEvent} object.\n   *\n   * @param change The Atom {TextEditEvent} to convert.\n   * @returns A {TextDocumentContentChangeEvent} that represents the converted {TextEditEvent}.\n   */\n  public static textEditToContentChange(change: TextChange): TextDocumentContentChangeEvent {\n    return {\n      range: Convert.atomRangeToLSRange(change.oldRange),\n      rangeLength: change.oldText.length,\n      text: change.newText,\n    }\n  }\n\n  private _isPrimaryAdapter(): boolean {\n    const lowestIdForBuffer = Math.min(\n      ...atom.workspace\n        .getTextEditors()\n        .filter((t) => t.getBuffer() === this._editor.getBuffer())\n        .map((t) => t.id)\n    )\n    return lowestIdForBuffer === this._editor.id\n  }\n\n  private _bumpVersion(): void {\n    const filePath = this._editor.getPath()\n    if (filePath == null) {\n      return\n    }\n    this._versions.set(filePath, this._getVersion(filePath) + 1)\n  }\n\n  /**\n   * Ensure when the document is opened we send notification to the language server so it can load it in and keep track\n   * of diagnostics etc.\n   */\n  private didOpen(): void {\n    const filePath = this._editor.getPath()\n    if (filePath == null) {\n      return\n    } // Not yet saved\n\n    if (!this._isPrimaryAdapter()) {\n      return\n    } // Multiple editors, we are not first\n\n    this._connection.didOpenTextDocument({\n      textDocument: {\n        uri: this.getEditorUri(),\n        languageId: this.getLanguageId().toLowerCase(),\n        version: this._getVersion(filePath),\n        text: this._editor.getText(),\n      },\n    })\n  }\n\n  private _getVersion(filePath: string): number {\n    return this._versions.get(filePath) || 1\n  }\n\n  /** Called when the {TextEditor} is closed and sends the 'didCloseTextDocument' notification to the connected language server. */\n  public didClose(): void {\n    if (this._editor.getPath() == null) {\n      return\n    } // Not yet saved\n\n    const fileStillOpen = atom.workspace.getTextEditors().find((t) => t.getBuffer() === this._editor.getBuffer())\n    if (fileStillOpen) {\n      return // Other windows or editors still have this file open\n    }\n\n    this._connection.didCloseTextDocument({ textDocument: { uri: this.getEditorUri() } })\n  }\n\n  /** Called just before the {TextEditor} saves and sends the 'willSaveTextDocument' notification to the connected language server. */\n  public willSave(): void {\n    if (!this._isPrimaryAdapter()) {\n      return\n    }\n\n    const uri = this.getEditorUri()\n    this._connection.willSaveTextDocument({\n      textDocument: { uri },\n      reason: TextDocumentSaveReason.Manual,\n    })\n  }\n\n  /**\n   * Called just before the {TextEditor} saves, sends the 'willSaveWaitUntilTextDocument' request to the connected\n   * language server and waits for the response before saving the buffer.\n   */\n  public async willSaveWaitUntil(): Promise<void> {\n    if (!this._isPrimaryAdapter()) {\n      return Promise.resolve()\n    }\n\n    const buffer = this._editor.getBuffer()\n    const uri = this.getEditorUri()\n    const title = this._editor.getLongTitle()\n\n    const applyEditsOrTimeout = Utils.promiseWithTimeout(\n      2500, // 2.5 seconds timeout\n      this._connection.willSaveWaitUntilTextDocument({\n        textDocument: { uri },\n        reason: TextDocumentSaveReason.Manual,\n      })\n    )\n      .then((edits) => {\n        const cursor = this._editor.getCursorBufferPosition()\n        ApplyEditAdapter.applyEdits(buffer, Convert.convertLsTextEdits(edits))\n        this._editor.setCursorBufferPosition(cursor)\n      })\n      .catch((err) => {\n        atom.notifications.addError(\"On-save action failed\", {\n          description: `Failed to apply edits to ${title}`,\n          detail: err.message,\n        })\n        return\n      })\n\n    const withBusySignal = this._reportBusyWhile(`Applying on-save edits for ${title}`, () => applyEditsOrTimeout)\n    return withBusySignal || applyEditsOrTimeout\n  }\n\n  /**\n   * Called when the {TextEditor} saves and sends the 'didSaveTextDocument' notification to the connected language\n   * server. Note: Right now this also sends the `didChangeWatchedFiles` notification as well but that will be sent from\n   * elsewhere soon.\n   */\n  public didSave(): void {\n    if (!this._isPrimaryAdapter()) {\n      return\n    }\n\n    const uri = this.getEditorUri()\n    const didSaveNotification = {\n      textDocument: { uri, version: this._getVersion(uri) },\n    } as DidSaveTextDocumentParams\n    if (typeof this._documentSync.save === \"object\" && this._documentSync.save.includeText) {\n      didSaveNotification.text = this._editor.getText()\n    }\n    this._connection.didSaveTextDocument(didSaveNotification)\n    if (this._fakeDidChangeWatchedFiles) {\n      this._connection.didChangeWatchedFiles({\n        changes: [{ uri, type: FileChangeType.Changed }],\n      })\n    }\n  }\n\n  public didRename(): void {\n    if (!this._isPrimaryAdapter()) {\n      return\n    }\n\n    const oldUri = this._currentUri\n    this._currentUri = this.getEditorUri()\n    if (!oldUri) {\n      return // Didn't previously have a name\n    }\n\n    if (this._documentSync.openClose !== false) {\n      this._connection.didCloseTextDocument({ textDocument: { uri: oldUri } })\n    }\n\n    if (this._fakeDidChangeWatchedFiles) {\n      this._connection.didChangeWatchedFiles({\n        changes: [\n          { uri: oldUri, type: FileChangeType.Deleted },\n          { uri: this._currentUri, type: FileChangeType.Created },\n        ],\n      })\n    }\n\n    // Send an equivalent open event for this editor, which will now use the new\n    // file path.\n    if (this._documentSync.openClose !== false) {\n      this.didOpen()\n    }\n  }\n\n  /** Public: Obtain the current {TextEditor} path and convert it to a Uri. */\n  public getEditorUri(): string {\n    return Convert.pathToUri(this._editor.getPath() || \"\")\n  }\n}\n"]} |
\ | No newline at end of file |