1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
open Std
open Typedtree
open Browse_raw
type direction = Prev | Next
let is_node_fun = function
| Expression { exp_desc = Texp_function _; _ } -> true
| _ -> false
;;
let is_node_let = function
| Value_binding _ -> true
| _ -> false
;;
let is_node_pattern = function
| Case _ -> true
| _ -> false
;;
let fun_pred = fun all ->
let rec normalize_fun = function
| node1 :: node2 :: node3 :: tail when is_node_fun node3 ->
assert (is_node_fun node1);
assert (is_node_pattern node2);
normalize_fun (node3 :: tail)
| node1 :: node2 :: _ when is_node_let node2 ->
assert (is_node_fun node1);
node2
| node :: _ ->
assert (is_node_fun node);
node
| _ ->
assert false
in
match all with
| node :: _ when is_node_fun node -> Some (normalize_fun all)
| _ -> None
;;
let let_pred = function
| node :: _ when is_node_let node -> Some node
| _ -> None
;;
let module_pred = function
| (Module_binding _ as node) :: _ -> Some node
| _ -> None
;;
let module_type_pred = function
| (Module_type_declaration _ as node) :: _ -> Some node
| _ -> None
let match_pred = function
| (Expression { exp_desc = Texp_match _ ; _ } as node) :: _ -> Some node
| _ -> None
;;
let rec find_map ~f = function
| [] -> None
| head :: tail ->
match f head with
| Some v -> Some v
| None -> find_map tail ~f
;;
exception No_matching_target
exception No_predicate of string
exception No_next_match_case
exception No_prev_match_case
let rec find_node preds nodes =
match nodes with
| [] -> raise No_matching_target
| _ :: tail ->
match find_map preds ~f:(fun pred -> pred nodes) with
| Some node -> node
| None -> find_node preds tail
;;
let rec skip_non_moving pos = function
| (node :: tail) as all ->
let node_loc = Browse_raw.node_real_loc Location.none node in
let loc_start = node_loc.Location.loc_start in
if pos.Lexing.pos_lnum = loc_start.Lexing.pos_lnum then
skip_non_moving pos tail
else
all
| [] -> []
;;
let get_cases_from_match node =
match node with
| Expression { exp_desc = Texp_match (_, cases, _); _ } -> cases
| _ -> []
let find_case_pos cases pos direction =
let rec find_pos pos cases direction =
match cases with
| [] -> None
| { c_lhs = { pat_loc; _ }; _ } :: tail ->
let check =
match direction with
| Prev ->
pos.Lexing.pos_cnum > pat_loc.loc_start.pos_cnum
| Next ->
pos.Lexing.pos_cnum < pat_loc.loc_start.pos_cnum
in
if check then
Some pat_loc.loc_start
else
find_pos pos tail direction
in
let case = find_pos pos cases direction in
match case with
| Some location -> `Found location
| None ->
(match direction with
| Next -> raise No_next_match_case
| Prev -> raise No_prev_match_case)
let get typed_tree pos target =
let roots = Mbrowse.of_typedtree typed_tree in
let enclosings =
match Mbrowse.enclosing pos [roots] with
| [] -> []
| l -> List.map ~f:snd l
in
let all_preds = [
"fun", fun_pred;
"let", let_pred;
"module", module_pred;
"module-type", module_type_pred;
"match", match_pred;
"match-next-case", match_pred;
"match-prev-case", match_pred;
] in
let targets = Str.split (Str.regexp "[, ]") target in
try
let preds =
List.map targets ~f:(fun target ->
match List.find_some all_preds ~f:(fun (name, _) -> name = target) with
| Some (_, f) -> f
| None -> raise (No_predicate target)
)
in
if String.length target = 0 then
`Error "Specify target"
else
let nodes = skip_non_moving pos enclosings in
let node = find_node preds nodes in
match target with
| "match-next-case" -> find_case_pos (get_cases_from_match node) pos Next
| "match-prev-case" ->
find_case_pos (List.rev (get_cases_from_match node)) pos Prev
| _ ->
let node_loc = Browse_raw.node_real_loc Location.none node in
`Found node_loc.Location.loc_start
with
| No_predicate target ->
`Error ("No predicate for " ^ target)
| No_matching_target ->
`Error "No matching target"
| No_next_match_case ->
`Error "No next case found"
| No_prev_match_case ->
`Error "No previous case found"
let phrase typed_tree pos target =
let roots = Mbrowse.of_typedtree typed_tree in
let enclosing = match Mbrowse.enclosing pos [roots] with
| (env, (Browse_raw.Module_expr _ as node)) :: enclosing ->
Browse_raw.fold_node (fun env node enclosing -> (env,node) :: enclosing)
env node enclosing
| enclosing -> enclosing
in
let enclosing = List.map ~f:snd enclosing in
let find_item x xs = match target with
| `Prev -> List.rev (List.take_while ~f:((!=)x) xs)
| `Next -> match List.drop_while ~f:((!=)x) xs with _::xs -> xs | [] -> []
in
let find_pos prj xs =
match target with
| `Prev ->
let f x = Location_aux.compare_pos pos (prj x) > 0 in
List.rev (List.take_while ~f xs)
| `Next ->
let f x = Location_aux.compare_pos pos (prj x) >= 0 in
List.drop_while ~f xs
in
let rec seek_item = function
| [] -> None
| Browse_raw.Signature xs :: tail ->
begin match find_pos (fun x -> x.Typedtree.sig_loc) xs.Typedtree.sig_items with
| [] -> seek_item tail
| y :: _ -> Some y.Typedtree.sig_loc
end
| Browse_raw.Structure xs :: tail ->
begin match find_pos (fun x -> x.Typedtree.str_loc) xs.Typedtree.str_items with
| [] -> seek_item tail
| y :: _ -> Some y.Typedtree.str_loc
end
| Browse_raw.Signature_item (x,_) :: Browse_raw.Signature xs :: tail ->
begin match find_item x xs.Typedtree.sig_items with
| [] -> seek_item tail
| y :: _ -> Some y.Typedtree.sig_loc
end
| Browse_raw.Structure_item (x,_) :: Browse_raw.Structure xs :: tail ->
begin match find_item x xs.Typedtree.str_items with
| [] -> seek_item tail
| y :: _ -> Some y.Typedtree.str_loc
end
| _ :: xs -> seek_item xs
in
match seek_item enclosing, target with
| Some loc, _ -> `Logical (Lexing.split_pos loc.Location.loc_start)
| None, `Prev -> `Start
| None, `Next -> `End