Skip to content

Commit 644dca1

Browse files
committed
add TypeAtPos completion type and let it drive completion when something might've compiled already and has a type, but no explicit type annotation
1 parent b4bccca commit 644dca1

6 files changed

+102
-2
lines changed

analysis/src/CompletionBackEnd.ml

+8
Original file line numberDiff line numberDiff line change
@@ -1113,6 +1113,14 @@ and getCompletionsForContextPath ~debug ~full ~opens ~rawOpens ~pos ~env ~exact
11131113
[Completion.create "dummy" ~env ~kind:(kindFromInnerType typ)]
11141114
| None -> [])
11151115
| None -> [])
1116+
| CTypeAtPos loc -> (
1117+
match
1118+
References.getLocItem ~full ~pos:(Pos.ofLexing loc.loc_start) ~debug
1119+
with
1120+
| None -> []
1121+
| Some {locType = Typed (_, typExpr, _)} ->
1122+
[Completion.create "dummy" ~env ~kind:(Value typExpr)]
1123+
| _ -> [])
11161124

11171125
let getOpens ~debug ~rawOpens ~package ~env =
11181126
if debug && rawOpens <> [] then

analysis/src/CompletionFrontEnd.ml

+21
Original file line numberDiff line numberDiff line change
@@ -591,6 +591,27 @@ let completionWithParser1 ~currentFile ~debug ~offset ~path ~posCursor ~text =
591591
(Completable.Cexpression
592592
{contextPath = ctxPath; prefix; nested = List.rev nested})
593593
| _ -> ())
594+
| {pvb_pat = {ppat_desc = Ppat_var {loc}}; pvb_expr}
595+
when locHasCursor pvb_expr.pexp_loc -> (
596+
(* Expression without a type annotation. We can complete this if this
597+
has compiled previously and there's a type available for the identifier itself.
598+
This is nice because the type is assigned even if the assignment isn't complete.
599+
600+
E.g: let x = {name: "name", <com>}, when `x` has compiled. *)
601+
match
602+
pvb_expr
603+
|> CompletionExpressions.traverseExpr ~exprPath:[] ~pos:posBeforeCursor
604+
~firstCharBeforeCursorNoWhite
605+
with
606+
| Some (prefix, nested) ->
607+
(* This completion should be low prio, so let any deeper completion
608+
hit first, and only set this TypeAtPos completion if nothing else
609+
here hit. *)
610+
Ast_iterator.default_iterator.value_binding iterator value_binding;
611+
setResult
612+
(Completable.Cexpression
613+
{contextPath = CTypeAtPos loc; prefix; nested = List.rev nested})
614+
| _ -> ())
594615
| {
595616
pvb_pat = {ppat_desc = Ppat_constraint (_pat, coreType); ppat_loc};
596617
pvb_expr;

analysis/src/SharedTypes.ml

+3
Original file line numberDiff line numberDiff line change
@@ -592,6 +592,8 @@ module Completable = struct
592592
}
593593
| CJsxPropValue of {pathToComponent: string list; propName: string}
594594
| CPatternPath of {rootCtxPath: contextPath; nested: nestedPath list}
595+
| CTypeAtPos of Location.t
596+
(** A position holding something that might have a *compiled* type. *)
595597

596598
type patternMode = Default | Destructuring
597599

@@ -671,6 +673,7 @@ module Completable = struct
671673
^ (nested
672674
|> List.map (fun nestedPath -> nestedPathToString nestedPath)
673675
|> String.concat "->")
676+
| CTypeAtPos _loc -> "CTypeAtPos()"
674677

675678
let toString = function
676679
| Cpath cp -> "Cpath " ^ contextPathToString cp

analysis/tests/src/RecordCompletion.res

+21
Original file line numberDiff line numberDiff line change
@@ -22,3 +22,24 @@ let n = {R.name: ""}
2222

2323
// n.R. xx
2424
// ^com
25+
26+
type optRecord = {
27+
name: string,
28+
age?: int,
29+
online?: bool,
30+
}
31+
32+
let optRecord = {
33+
name: "Hello",
34+
// ^com
35+
}
36+
37+
type someVariant = One(int, optRecord)
38+
39+
let x = One(
40+
1,
41+
{
42+
name: "What",
43+
// ^com
44+
},
45+
)

analysis/tests/src/expected/Fragment.res.txt

+4
Original file line numberDiff line numberDiff line change
@@ -12,5 +12,9 @@ posCursor:[9:56] posNoWhite:[9:55] Found expr:[9:13->9:66]
1212
JSX <SectionHeader:[9:13->9:26] > _children:9:26
1313
posCursor:[9:56] posNoWhite:[9:55] Found expr:__ghost__[9:10->9:67]
1414
Pexp_construct []:__ghost__[9:10->9:67] None
15+
Completable: Cexpression CTypeAtPos()=[]->variantPayload::::($1)
16+
Package opens Pervasives.JsxModules.place holder
17+
Resolved opens 1 pervasives
18+
ContextPath CTypeAtPos()
1519
null
1620

analysis/tests/src/expected/RecordCompletion.res.txt

+45-2
Original file line numberDiff line numberDiff line change
@@ -50,8 +50,8 @@ Path Js.Array2.m
5050
}]
5151

5252
Complete src/RecordCompletion.res 19:7
53-
posCursor:[19:7] posNoWhite:[19:6] Found expr:[19:3->19:7]
54-
Pexp_field [19:3->19:4] R.:[19:5->19:7]
53+
posCursor:[19:7] posNoWhite:[19:6] Found expr:[19:3->25:4]
54+
Pexp_field [19:3->19:4] R.:[19:5->25:4]
5555
Completable: Cpath Module[R].""
5656
Package opens Pervasives.JsxModules.place holder
5757
Resolved opens 1 pervasives
@@ -81,3 +81,46 @@ Path R.
8181
"documentation": null
8282
}]
8383

84+
Complete src/RecordCompletion.res 32:17
85+
posCursor:[32:17] posNoWhite:[32:15] Found expr:[31:16->34:1]
86+
Completable: Cexpression CTypeAtPos()->recordBody
87+
Package opens Pervasives.JsxModules.place holder
88+
Resolved opens 1 pervasives
89+
ContextPath CTypeAtPos()
90+
[{
91+
"label": "age",
92+
"kind": 5,
93+
"tags": [],
94+
"detail": "age?: int\n\noptRecord",
95+
"documentation": null
96+
}, {
97+
"label": "online",
98+
"kind": 5,
99+
"tags": [],
100+
"detail": "online?: bool\n\noptRecord",
101+
"documentation": null
102+
}]
103+
104+
Complete src/RecordCompletion.res 41:18
105+
posCursor:[41:18] posNoWhite:[41:16] Found expr:[38:8->44:1]
106+
Pexp_construct One:[38:8->38:11] [38:11->44:1]
107+
posCursor:[41:18] posNoWhite:[41:16] Found expr:[38:11->44:1]
108+
posCursor:[41:18] posNoWhite:[41:16] Found expr:[40:2->43:3]
109+
Completable: Cexpression CTypeAtPos()->variantPayload::One($1), recordBody
110+
Package opens Pervasives.JsxModules.place holder
111+
Resolved opens 1 pervasives
112+
ContextPath CTypeAtPos()
113+
[{
114+
"label": "age",
115+
"kind": 5,
116+
"tags": [],
117+
"detail": "age?: int\n\noptRecord",
118+
"documentation": null
119+
}, {
120+
"label": "online",
121+
"kind": 5,
122+
"tags": [],
123+
"detail": "online?: bool\n\noptRecord",
124+
"documentation": null
125+
}]
126+

0 commit comments

Comments
 (0)