Skip to content

Commit

Permalink
fix(lsp): Improve lsp behaviour on goto definition (#1893)
Browse files Browse the repository at this point in the history
Co-authored-by: Oscar Spencer <[email protected]>
  • Loading branch information
spotandjake and ospencer authored Jul 28, 2024
1 parent 639856f commit d822c87
Showing 1 changed file with 52 additions and 21 deletions.
73 changes: 52 additions & 21 deletions compiler/src/language_server/definition.re
Original file line number Diff line number Diff line change
Expand Up @@ -53,19 +53,18 @@ let send_definition =
}),
);
};
type check_position =
| Forward
| Backward;
let rec find_definition =
(
~check_position=Forward,
sourcetree: Sourcetree.sourcetree,
position: Protocol.position,
) => {
let results = Sourcetree.query(position, sourcetree);

let process =
(
~id: Protocol.message_id,
~compiled_code: Hashtbl.t(Protocol.uri, Lsp_types.code),
~documents: Hashtbl.t(Protocol.uri, string),
params: RequestParams.t,
) => {
switch (Hashtbl.find_opt(compiled_code, params.text_document.uri)) {
| None => send_no_result(~id)
| Some({program, sourcetree}) =>
let results = Sourcetree.query(params.position, sourcetree);

let result =
switch (results) {
| [Value({definition}), ..._]
| [Pattern({definition}), ..._]
Expand All @@ -74,18 +73,50 @@ let process =
| [Exception({definition}), ..._]
| [Module({definition}), ..._] =>
switch (definition) {
| None => send_no_result(~id)
| None => None
| Some(loc) =>
let uri = Utils.filename_to_uri(loc.loc_start.pos_fname);

send_definition(
~id,
~range=Utils.loc_to_range(loc),
~target_uri=uri,
Utils.loc_to_range(loc),
);
Some((loc, uri));
}
| _ => send_no_result(~id)
| _ => None
};
switch (result) {
| None =>
if (check_position == Forward && position.character > 0) {
// If a user selects from left to right, their pointer ends up after the identifier
// this tries to check if the identifier was selected.
find_definition(
~check_position=Backward,
sourcetree,
{line: position.line, character: position.character - 1},
);
} else {
None;
}
| Some((loc, uri)) => result
};
};

let process =
(
~id: Protocol.message_id,
~compiled_code: Hashtbl.t(Protocol.uri, Lsp_types.code),
~documents: Hashtbl.t(Protocol.uri, string),
params: RequestParams.t,
) => {
switch (Hashtbl.find_opt(compiled_code, params.text_document.uri)) {
| None => send_no_result(~id)
| Some({program, sourcetree}) =>
let result = find_definition(sourcetree, params.position);
switch (result) {
| None => send_no_result(~id)
| Some((loc, uri)) =>
send_definition(
~id,
~range=Utils.loc_to_range(loc),
~target_uri=uri,
Utils.loc_to_range(loc),
)
};
};
};

0 comments on commit d822c87

Please sign in to comment.