UNPKG

16.1 kBJavaScriptView Raw
1"use strict";
2var __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};
11Object.defineProperty(exports, "__esModule", { value: true });
12const convert_1 = require("../convert");
13const Utils = require("../utils");
14const languageclient_1 = require("../languageclient");
15const atom_1 = require("atom");
16/**
17 * Public: Adapts the language server definition provider to the Atom IDE UI Definitions package for 'Go To Definition'
18 * functionality.
19 */
20class DefinitionAdapter {
21 /**
22 * Public: Determine whether this adapter can be used to adapt a language server based on the serverCapabilities
23 * matrix containing a definitionProvider.
24 *
25 * @param serverCapabilities The {ServerCapabilities} of the language server to consider.
26 * @returns A {Boolean} indicating adapter can adapt the server based on the given serverCapabilities.
27 */
28 static canAdapt(serverCapabilities) {
29 return serverCapabilities.definitionProvider === true;
30 }
31 /**
32 * Public: Get the definitions for a symbol at a given {Point} within a {TextEditor} including optionally highlighting
33 * all other references within the document if the langauge server also supports highlighting.
34 *
35 * @param connection A {LanguageClientConnection} to the language server that will provide definitions and highlights.
36 * @param serverCapabilities The {ServerCapabilities} of the language server that will be used.
37 * @param languageName The name of the programming language.
38 * @param editor The Atom {TextEditor} containing the symbol and potential highlights.
39 * @param point The Atom {Point} containing the position of the text that represents the symbol for which the
40 * definition and highlights should be provided.
41 * @returns A {Promise} indicating adapter can adapt the server based on the given serverCapabilities.
42 */
43 getDefinition(connection, serverCapabilities, languageName, editor, point) {
44 return __awaiter(this, void 0, void 0, function* () {
45 const documentPositionParams = convert_1.default.editorToTextDocumentPositionParams(editor, point);
46 const definitionLocations = DefinitionAdapter.normalizeLocations(yield connection.gotoDefinition(documentPositionParams));
47 if (definitionLocations == null || definitionLocations.length === 0) {
48 return null;
49 }
50 let queryRange;
51 if (serverCapabilities.documentHighlightProvider) {
52 const highlights = yield connection.documentHighlight(documentPositionParams);
53 if (highlights != null && highlights.length > 0) {
54 queryRange = highlights.map((h) => convert_1.default.lsRangeToAtomRange(h.range));
55 }
56 }
57 return {
58 queryRange: queryRange || [Utils.getWordAtPosition(editor, point)],
59 definitions: DefinitionAdapter.convertLocationsToDefinitions(definitionLocations, languageName),
60 };
61 });
62 }
63 /**
64 * Public: Normalize the locations so a single {Location} becomes an {Array} of just one. The language server protocol
65 * return either as the protocol evolved between v1 and v2.
66 *
67 * @param locationResult Either a single {Location} object or an {Array} of {Locations}.
68 * @returns An {Array} of {Location}s or {null} if the locationResult was null.
69 */
70 static normalizeLocations(locationResult) {
71 if (locationResult == null) {
72 // TODO use ===
73 return null;
74 }
75 // TODO `d.targetRange.start` never becomes `null` according to the types
76 if (isLocationLinkArray(locationResult)) {
77 return locationResult.filter((d) => d.targetRange.start != null);
78 }
79 return (Array.isArray(locationResult) ? locationResult : [locationResult]).filter((d) => d.range.start != null);
80 }
81 /**
82 * Public: Convert an {Array} of {Location} objects into an Array of {Definition}s.
83 *
84 * @param locations An {Array} of {Location} objects to be converted.
85 * @param languageName The name of the language these objects are written in.
86 * @returns An {Array} of {Definition}s that represented the converted {Location}s.
87 */
88 static convertLocationsToDefinitions(locations, languageName) {
89 if (isLocationLinkArray(locations)) {
90 return locations.map((d) => ({
91 path: convert_1.default.uriToPath(d.targetUri),
92 position: convert_1.default.positionToPoint(d.targetRange.start),
93 range: atom_1.Range.fromObject(convert_1.default.lsRangeToAtomRange(d.targetRange)),
94 language: languageName,
95 }));
96 }
97 return locations.map((d) => ({
98 path: convert_1.default.uriToPath(d.uri),
99 position: convert_1.default.positionToPoint(d.range.start),
100 range: atom_1.Range.fromObject(convert_1.default.lsRangeToAtomRange(d.range)),
101 language: languageName,
102 }));
103 }
104}
105exports.default = DefinitionAdapter;
106function isLocationLinkArray(value) {
107 return Array.isArray(value) && languageclient_1.LocationLink.is(value[0]);
108}
109//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"definition-adapter.js","sourceRoot":"","sources":["../../../lib/adapters/definition-adapter.ts"],"names":[],"mappings":"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","sourcesContent":["import type * as atomIde from \"atom-ide-base\"\nimport Convert from \"../convert\"\nimport * as Utils from \"../utils\"\nimport { LanguageClientConnection, Location, LocationLink, ServerCapabilities } from \"../languageclient\"\nimport { Point, TextEditor, Range } from \"atom\"\n\n/**\n * Public: Adapts the language server definition provider to the Atom IDE UI Definitions package for 'Go To Definition'\n * functionality.\n */\nexport default class DefinitionAdapter {\n  /**\n   * Public: Determine whether this adapter can be used to adapt a language server based on the serverCapabilities\n   * matrix containing a definitionProvider.\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 serverCapabilities.definitionProvider === true\n  }\n\n  /**\n   * Public: Get the definitions for a symbol at a given {Point} within a {TextEditor} including optionally highlighting\n   * all other references within the document if the langauge server also supports highlighting.\n   *\n   * @param connection A {LanguageClientConnection} to the language server that will provide definitions and highlights.\n   * @param serverCapabilities The {ServerCapabilities} of the language server that will be used.\n   * @param languageName The name of the programming language.\n   * @param editor The Atom {TextEditor} containing the symbol and potential highlights.\n   * @param point The Atom {Point} containing the position of the text that represents the symbol for which the\n   *   definition and highlights should be provided.\n   * @returns A {Promise} indicating adapter can adapt the server based on the given serverCapabilities.\n   */\n  public async getDefinition(\n    connection: LanguageClientConnection,\n    serverCapabilities: ServerCapabilities,\n    languageName: string,\n    editor: TextEditor,\n    point: Point\n  ): Promise<atomIde.DefinitionQueryResult | null> {\n    const documentPositionParams = Convert.editorToTextDocumentPositionParams(editor, point)\n    const definitionLocations = DefinitionAdapter.normalizeLocations(\n      await connection.gotoDefinition(documentPositionParams)\n    )\n    if (definitionLocations == null || definitionLocations.length === 0) {\n      return null\n    }\n\n    let queryRange\n    if (serverCapabilities.documentHighlightProvider) {\n      const highlights = await connection.documentHighlight(documentPositionParams)\n      if (highlights != null && highlights.length > 0) {\n        queryRange = highlights.map((h) => Convert.lsRangeToAtomRange(h.range))\n      }\n    }\n\n    return {\n      queryRange: queryRange || [Utils.getWordAtPosition(editor, point)],\n      definitions: DefinitionAdapter.convertLocationsToDefinitions(definitionLocations, languageName),\n    }\n  }\n\n  /**\n   * Public: Normalize the locations so a single {Location} becomes an {Array} of just one. The language server protocol\n   * return either as the protocol evolved between v1 and v2.\n   *\n   * @param locationResult Either a single {Location} object or an {Array} of {Locations}.\n   * @returns An {Array} of {Location}s or {null} if the locationResult was null.\n   */\n  public static normalizeLocations(\n    locationResult: Location | Location[] | LocationLink[] | null\n  ): Location[] | LocationLink[] | null {\n    if (locationResult == null) {\n      // TODO use ===\n      return null\n    }\n    // TODO `d.targetRange.start` never becomes `null` according to the types\n    if (isLocationLinkArray(locationResult)) {\n      return locationResult.filter((d) => d.targetRange.start != null)\n    }\n    return (Array.isArray(locationResult) ? locationResult : [locationResult]).filter((d) => d.range.start != null)\n  }\n\n  /**\n   * Public: Convert an {Array} of {Location} objects into an Array of {Definition}s.\n   *\n   * @param locations An {Array} of {Location} objects to be converted.\n   * @param languageName The name of the language these objects are written in.\n   * @returns An {Array} of {Definition}s that represented the converted {Location}s.\n   */\n  public static convertLocationsToDefinitions(\n    locations: Location[] | LocationLink[],\n    languageName: string\n  ): atomIde.Definition[] {\n    if (isLocationLinkArray(locations)) {\n      return locations.map((d) => ({\n        path: Convert.uriToPath(d.targetUri),\n        position: Convert.positionToPoint(d.targetRange.start),\n        range: Range.fromObject(Convert.lsRangeToAtomRange(d.targetRange)),\n        language: languageName,\n      }))\n    }\n    return locations.map((d) => ({\n      path: Convert.uriToPath(d.uri),\n      position: Convert.positionToPoint(d.range.start),\n      range: Range.fromObject(Convert.lsRangeToAtomRange(d.range)),\n      language: languageName,\n    }))\n  }\n}\n\nfunction isLocationLinkArray(value: any): value is LocationLink[] {\n  return Array.isArray(value) && LocationLink.is(value[0])\n}\n"]}
\No newline at end of file