Skip to content

Commit

Permalink
Navigating Typed Holes
Browse files Browse the repository at this point in the history
Co-authored-by: Ulysse Gérard <thevoodoos@gmail.com>
  • Loading branch information
2 people authored and smorimoto committed Dec 11, 2024
1 parent 22478a3 commit c1363af
Show file tree
Hide file tree
Showing 7 changed files with 157 additions and 1 deletion.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
# Unreleased

- Add `ocaml.search-by-type` to search for values using their type signature (#1626)
- Add `ocaml.navigate-typed-holes` to navigate to different typed holes. (#1666)

## 1.24.0

Expand Down
14 changes: 14 additions & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -267,6 +267,11 @@
"command": "ocaml.search-by-type",
"category": "OCaml",
"title": "Search a value by type or polarity"
},
{
"command": "ocaml.navigate-typed-holes",
"category": "OCaml",
"title": "Lists typed holes in the file for navigation."
}
],
"configuration": {
Expand Down Expand Up @@ -311,6 +316,11 @@
"default": false,
"markdownDescription": "Enable/Disable syntax documentation"
},
"ocaml.commands.typedHoles.constructAfterNavigate": {
"type": "boolean",
"default": false,
"markdownDescription": "When enabled, list values that can fill a typed hole after navigating to it."
},
"ocaml.commands.construct.recursiveCalls": {
"type": "boolean",
"default": true,
Expand Down Expand Up @@ -1081,6 +1091,10 @@
{
"command": "ocaml.search-by-type",
"when": "editorLangId == ocaml || editorLangId == ocaml.interface || editorLangId == reason || editorLangId == ocaml.ocamllex"
},
{
"command": "ocaml.navigate-typed-holes",
"when": "editorLangId == ocaml || editorLangId == ocaml.interface || editorLangId == reason"
}
],
"editor/title": [
Expand Down
107 changes: 107 additions & 0 deletions src/extension_commands.ml
Original file line number Diff line number Diff line change
Expand Up @@ -950,6 +950,113 @@ module Search_by_type = struct
;;
end

module Navigate_holes = struct
let extension_name = "Navigate between Typed Holes"

let ocaml_lsp_doesnt_support_typed_holes ocaml_lsp =
not (Ocaml_lsp.can_handle_typed_holes ocaml_lsp)
;;

let is_valid_text_doc textdoc =
match TextDocument.languageId textdoc with
| "ocaml" | "ocaml.interface" | "reason" -> true
| _ -> false
;;

let send_request_to_lsp client doc =
let uri = TextDocument.uri doc in
Custom_requests.send_request client Custom_requests.typedHoles uri
;;

let display_results (results : Range.t list) text_document =
let quickPickItems =
List.map results ~f:(fun res ->
let line = Position.line @@ Range.end_ res in
( QuickPickItem.create
~label:(Printf.sprintf "Line %d" line)
~detail:
(Printf.sprintf
"%s"
(TextLine.text @@ TextDocument.lineAt ~line text_document))
()
, (res, ()) ))
in
let quickPickOptions = QuickPickOptions.create ~title:"Typed Holes" () in
Window.showQuickPickItems ~choices:quickPickItems ~options:quickPickOptions ()
;;

let jump_to_hole range text_editor =
let open Promise.Syntax in
let+ _ =
Window.showTextDocument
~document:(TextEditor.document text_editor)
~preserveFocus:true
()
in
let new_selection =
let anchor = Range.start range in
let active = Range.end_ range in
Selection.makePositions ~anchor ~active
in
TextEditor.set_selection text_editor new_selection;
TextEditor.revealRange
text_editor
~range
~revealType:TextEditorRevealType.InCenterIfOutsideViewport
()
;;

let handle_hole_navigation text_editor client instance =
let open Promise.Syntax in
let doc = TextEditor.document text_editor in
let* hole_positions = send_request_to_lsp client doc in
match hole_positions with
| [] ->
show_message `Info "No typed holes found in the file.";
Promise.return ()
| holes ->
let* selected_hole = display_results holes doc in
(match selected_hole with
| Some (range, ()) ->
let* () = jump_to_hole range text_editor in
(match Settings.(get server_typedHolesConstructAfterNavigate_setting) with
| Some true ->
Construct.process_construct (Range.end_ range) text_editor client instance
| Some false | None -> Promise.return ())
| None -> Promise.return ())
;;

let _holes =
let handler (instance : Extension_instance.t) ~args:_ =
match Window.activeTextEditor () with
| None ->
Extension_consts.Command_errors.text_editor_must_be_active
extension_name
~expl:
"This command only works in an active editor because it's based on the \
content of the editor"
|> show_message `Error "%s"
| Some text_editor when not (is_valid_text_doc (TextEditor.document text_editor)) ->
show_message
`Error
"Invalid file type. This command only works in ocaml files, ocaml interface \
files, reason files."
| Some text_editor ->
(match Extension_instance.lsp_client instance with
| None -> show_message `Warn "ocamllsp is not running"
| Some (_client, ocaml_lsp) when ocaml_lsp_doesnt_support_typed_holes ocaml_lsp
->
show_message
`Warn
"The installed version of `ocamllsp` does not support typed hole navigation"
| Some (client, _) ->
let _ = handle_hole_navigation text_editor client instance in
())
in
command Extension_consts.Commands.navigate_typed_holes handler
;;
end

let register extension instance = function
| Command { id; handler } ->
let callback = handler instance in
Expand Down
1 change: 1 addition & 0 deletions src/extension_consts.ml
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@ module Commands = struct
let construct = ocaml_prefixed "construct"
let merlin_jump = ocaml_prefixed "jump"
let search_by_type = ocaml_prefixed "search-by-type"
let navigate_typed_holes = ocaml_prefixed "navigate-typed-holes"
end

module Command_errors = struct
Expand Down
8 changes: 8 additions & 0 deletions src/settings.ml
Original file line number Diff line number Diff line change
Expand Up @@ -149,6 +149,14 @@ let server_syntaxDocumentation_setting =
~to_json:Jsonoo.Encode.bool
;;

let server_typedHolesConstructAfterNavigate_setting =
create_setting
~scope:ConfigurationTarget.Workspace
~key:"ocaml.commands.typedHoles.constructAfterNavigate"
~of_json:Jsonoo.Decode.bool
~to_json:Jsonoo.Encode.bool
;;

let server_constructRecursiveCalls_setting =
create_setting
~scope:ConfigurationTarget.Workspace
Expand Down
1 change: 1 addition & 0 deletions src/settings.mli
Original file line number Diff line number Diff line change
Expand Up @@ -39,4 +39,5 @@ val server_codelens_setting : bool setting
val server_extendedHover_setting : bool setting
val server_duneDiagnostics_setting : bool setting
val server_syntaxDocumentation_setting : bool setting
val server_typedHolesConstructAfterNavigate_setting : bool setting
val server_constructRecursiveCalls_setting : bool setting
26 changes: 25 additions & 1 deletion src/treeview_commands.ml
Original file line number Diff line number Diff line change
Expand Up @@ -84,8 +84,32 @@ let type_search_item =
item
;;

let navigate_holes_item =
let icon = `ThemeIcon (Vscode.ThemeIcon.make ~id:"breakpoints-activate" ()) in
let label =
`TreeItemLabel
(Vscode.TreeItemLabel.create ~label:"Navigate to different typed holes" ())
in
let item = Vscode.TreeItem.make_label ~label () in
let command =
Vscode.Command.create
~title:"Navigate typed holes"
~command:"ocaml.navigate-typed-holes"
()
in
Vscode.TreeItem.set_iconPath item icon;
Vscode.TreeItem.set_command item command;
item
;;

let items =
[ select_sandbox_item; terminal_item; construct_item; jump_item; type_search_item ]
[ select_sandbox_item
; terminal_item
; construct_item
; jump_item
; type_search_item
; navigate_holes_item
]
;;

let getTreeItem ~element = `Value element
Expand Down

0 comments on commit c1363af

Please sign in to comment.