1use syntax::{AstNode, SyntaxKind, T, ast, syntax_editor::Position};
2
3use crate::{AssistContext, AssistId, Assists};
4
5pub(crate) fn remove_parentheses(acc: &mut Assists, ctx: &AssistContext<'_, '_>) -> Option<()> {
21 let parens = ctx.find_node_at_offset::<ast::ParenExpr>()?;
22
23 let cursor_in_range =
24 parens.l_paren_token()?.text_range().contains_range(ctx.selection_trimmed())
25 || parens.r_paren_token()?.text_range().contains_range(ctx.selection_trimmed());
26 if !cursor_in_range {
27 return None;
28 }
29
30 let expr = parens.expr()?;
31
32 let parent = parens.syntax().parent()?;
33 if expr.needs_parens_in(&parent) {
34 return None;
35 }
36
37 let target = parens.syntax().text_range();
38 acc.add(
39 AssistId::refactor("remove_parentheses"),
40 "Remove redundant parentheses",
41 target,
42 |builder| {
43 let editor = builder.make_editor(parens.syntax());
44 let make = editor.make();
45 let prev_token = parens.syntax().first_token().and_then(|it| it.prev_token());
46 let need_to_add_ws = match prev_token {
47 Some(it) => {
48 let tokens = [T![&], T![!], T!['('], T!['['], T!['{']];
49 it.kind() != SyntaxKind::WHITESPACE && !tokens.contains(&it.kind())
50 }
51 None => false,
52 };
53 if need_to_add_ws {
54 editor.insert(Position::before(parens.syntax()), make.whitespace(" "));
55 }
56 editor.replace(parens.syntax(), expr.syntax());
57 builder.add_file_edits(ctx.vfs_file_id(), editor);
58 },
59 )
60}
61
62#[cfg(test)]
63mod tests {
64 use crate::tests::{check_assist, check_assist_not_applicable};
65
66 use super::*;
67
68 #[test]
69 fn remove_parens_space() {
70 check_assist(
71 remove_parentheses,
72 r#"fn f() { match$0(true) {} }"#,
73 r#"fn f() { match true {} }"#,
74 );
75 }
76
77 #[test]
78 fn remove_parens_simple() {
79 check_assist(remove_parentheses, r#"fn f() { $0(2) + 2; }"#, r#"fn f() { 2 + 2; }"#);
80 check_assist(remove_parentheses, r#"fn f() { ($02) + 2; }"#, r#"fn f() { 2 + 2; }"#);
81 check_assist(remove_parentheses, r#"fn f() { (2)$0 + 2; }"#, r#"fn f() { 2 + 2; }"#);
82 check_assist(remove_parentheses, r#"fn f() { (2$0) + 2; }"#, r#"fn f() { 2 + 2; }"#);
83 }
84
85 #[test]
86 fn remove_parens_closure() {
87 check_assist(remove_parentheses, r#"fn f() { &$0(|| 42) }"#, r#"fn f() { &|| 42 }"#);
88
89 check_assist_not_applicable(remove_parentheses, r#"fn f() { $0(|| 42).f() }"#);
90 }
91
92 #[test]
93 fn remove_parens_if_let_chains() {
94 check_assist_not_applicable(
95 remove_parentheses,
96 r#"fn f() { if let true = $0(true && true) {} }"#,
97 );
98 }
99
100 #[test]
101 fn remove_parens_associativity() {
102 check_assist(
103 remove_parentheses,
104 r#"fn f() { $0(2 + 2) + 2; }"#,
105 r#"fn f() { 2 + 2 + 2; }"#,
106 );
107 check_assist_not_applicable(remove_parentheses, r#"fn f() { 2 + $0(2 + 2); }"#);
108 }
109
110 #[test]
111 fn remove_parens_precedence() {
112 check_assist(
113 remove_parentheses,
114 r#"fn f() { $0(2 * 3) + 1; }"#,
115 r#"fn f() { 2 * 3 + 1; }"#,
116 );
117 check_assist(remove_parentheses, r#"fn f() { ( $0(2) ); }"#, r#"fn f() { ( 2 ); }"#);
118 check_assist(remove_parentheses, r#"fn f() { $0(2?)?; }"#, r#"fn f() { 2??; }"#);
119 check_assist(remove_parentheses, r#"fn f() { f(($02 + 2)); }"#, r#"fn f() { f(2 + 2); }"#);
120 check_assist(
121 remove_parentheses,
122 r#"fn f() { (1<2) &&$0(3>4); }"#,
123 r#"fn f() { (1<2) && 3>4; }"#,
124 );
125 }
126
127 #[test]
128 fn remove_parens_doesnt_apply_precedence() {
129 check_assist_not_applicable(remove_parentheses, r#"fn f() { $0(2 + 2) * 8; }"#);
130 check_assist_not_applicable(remove_parentheses, r#"fn f() { $0(2 + 2).f(); }"#);
131 check_assist_not_applicable(remove_parentheses, r#"fn f() { $0(2 + 2).await; }"#);
132 check_assist_not_applicable(remove_parentheses, r#"fn f() { $0!(2..2); }"#);
133 }
134
135 #[test]
136 fn remove_parens_doesnt_apply_with_cursor_not_on_paren() {
137 check_assist_not_applicable(remove_parentheses, r#"fn f() { (2 +$0 2) }"#);
138 check_assist_not_applicable(remove_parentheses, r#"fn f() {$0 (2 + 2) }"#);
139 }
140
141 #[test]
142 fn remove_parens_doesnt_apply_when_expr_would_be_turned_into_a_statement() {
143 check_assist_not_applicable(remove_parentheses, r#"fn x() -> u8 { $0({ 0 } + 1) }"#);
144 check_assist_not_applicable(
145 remove_parentheses,
146 r#"fn x() -> u8 { $0(if true { 0 } else { 1 } + 1) }"#,
147 );
148 check_assist_not_applicable(remove_parentheses, r#"fn x() -> u8 { $0(loop {} + 1) }"#);
149 }
150
151 #[test]
152 fn remove_parens_doesnt_apply_weird_syntax_and_edge_cases() {
153 check_assist_not_applicable(remove_parentheses, r#"fn f() { for _ in $0(0..{3}) {} }"#);
155 check_assist_not_applicable(remove_parentheses, r#"fn f() { for _ in $0(S {}) {} }"#);
156 check_assist_not_applicable(remove_parentheses, r#"fn f() { if $0(S {} == 2) {} }"#);
157 check_assist_not_applicable(remove_parentheses, r#"fn f() { if $0(return) {} }"#);
158 }
159
160 #[test]
161 fn remove_parens_prefix_with_ret_like_prefix() {
162 check_assist(remove_parentheses, r#"fn f() { !$0(return) }"#, r#"fn f() { !return }"#);
163 check_assist(remove_parentheses, r#"fn f() { !$0(break) }"#, r#"fn f() { !break }"#);
165 check_assist(remove_parentheses, r#"fn f() { !$0(continue) }"#, r#"fn f() { !continue }"#);
166 check_assist(
167 remove_parentheses,
168 r#"fn f() { !$0(return false) }"#,
169 r#"fn f() { !return false }"#,
170 );
171
172 check_assist(
174 remove_parentheses,
175 r#"fn f() { true || $0(return) }"#,
176 r#"fn f() { true || return }"#,
177 );
178 check_assist(
179 remove_parentheses,
180 r#"fn f() { cond && $0(return) }"#,
181 r#"fn f() { cond && return }"#,
182 );
183 }
184
185 #[test]
186 fn remove_parens_return_with_value_followed_by_block() {
187 check_assist(
188 remove_parentheses,
189 r#"fn f() { if $0(return ()) {} }"#,
190 r#"fn f() { if return () {} }"#,
191 );
192 }
193
194 #[test]
195 fn remove_exprs_let_else_restrictions() {
196 check_assist_not_applicable(
198 remove_parentheses,
199 r#"fn f() { let _ = $0(S{}) else { return }; }"#,
200 );
201
202 check_assist_not_applicable(
204 remove_parentheses,
205 r#"fn f() { let _ = $0(false || false) else { return }; }"#,
206 );
207 check_assist_not_applicable(
208 remove_parentheses,
209 r#"fn f() { let _ = $0(true && true) else { return }; }"#,
210 );
211 }
212
213 #[test]
214 fn remove_parens_weird_places() {
215 check_assist(
216 remove_parentheses,
217 r#"fn f() { match () { _ =>$0(()) } }"#,
218 r#"fn f() { match () { _ => () } }"#,
219 );
220
221 check_assist(
222 remove_parentheses,
223 r#"fn x() -> u8 { { [$0({ 0 } + 1)] } }"#,
224 r#"fn x() -> u8 { { [{ 0 } + 1] } }"#,
225 );
226 }
227
228 #[test]
229 fn remove_parens_return_dot_f() {
230 check_assist(
231 remove_parentheses,
232 r#"fn f() { $0(return).f() }"#,
233 r#"fn f() { return.f() }"#,
234 );
235 }
236
237 #[test]
238 fn remove_parens_prefix_then_return_something() {
239 check_assist(
240 remove_parentheses,
241 r#"fn f() { &$0(return ()) }"#,
242 r#"fn f() { &return () }"#,
243 );
244 }
245
246 #[test]
247 fn remove_parens_return_in_unary_not() {
248 check_assist(
249 remove_parentheses,
250 r#"fn f() { cond && !$0(return) }"#,
251 r#"fn f() { cond && !return }"#,
252 );
253 check_assist(
254 remove_parentheses,
255 r#"fn f() { cond && !$0(return false) }"#,
256 r#"fn f() { cond && !return false }"#,
257 );
258 }
259
260 #[test]
261 fn remove_parens_return_in_disjunction_with_closure_risk() {
262 check_assist_not_applicable(
264 remove_parentheses,
265 r#"fn f() { let _x = true && $0(return) || true; }"#,
266 );
267 check_assist_not_applicable(
268 remove_parentheses,
269 r#"fn f() { let _x = true && !$0(return) || true; }"#,
270 );
271 check_assist_not_applicable(
272 remove_parentheses,
273 r#"fn f() { let _x = true && $0(return false) || true; }"#,
274 );
275 check_assist_not_applicable(
276 remove_parentheses,
277 r#"fn f() { let _x = true && !$0(return false) || true; }"#,
278 );
279 check_assist_not_applicable(
280 remove_parentheses,
281 r#"fn f() { let _x = true && $0(return) && true; }"#,
282 );
283 check_assist_not_applicable(
284 remove_parentheses,
285 r#"fn f() { let _x = true && !$0(return) && true; }"#,
286 );
287 check_assist_not_applicable(
288 remove_parentheses,
289 r#"fn f() { let _x = true && $0(return false) && true; }"#,
290 );
291 check_assist_not_applicable(
292 remove_parentheses,
293 r#"fn f() { let _x = true && !$0(return false) && true; }"#,
294 );
295 check_assist_not_applicable(
296 remove_parentheses,
297 r#"fn f() { let _x = $0(return) || true; }"#,
298 );
299 check_assist_not_applicable(
300 remove_parentheses,
301 r#"fn f() { let _x = $0(return) && true; }"#,
302 );
303 }
304
305 #[test]
306 fn remove_parens_return_in_disjunction_is_ok() {
307 check_assist(
308 remove_parentheses,
309 r#"fn f() { let _x = true || $0(return); }"#,
310 r#"fn f() { let _x = true || return; }"#,
311 );
312 check_assist(
313 remove_parentheses,
314 r#"fn f() { let _x = true && $0(return); }"#,
315 r#"fn f() { let _x = true && return; }"#,
316 );
317 }
318
319 #[test]
320 fn remove_parens_conflict_cast_before_l_angle() {
321 check_assist_not_applicable(remove_parentheses, r#"fn f() { _ = $0(1 as u32) << 10; }"#);
322 check_assist_not_applicable(remove_parentheses, r#"fn f() { _ = $0(1 as u32) < 10; }"#);
323 }
324
325 #[test]
326 fn remove_parens_double_paren_stmt() {
327 check_assist(
328 remove_parentheses,
329 r#"fn x() -> u8 { $0(({ 0 } + 1)) }"#,
330 r#"fn x() -> u8 { ({ 0 } + 1) }"#,
331 );
332
333 check_assist(
334 remove_parentheses,
335 r#"fn x() -> u8 { (($0{ 0 } + 1)) }"#,
336 r#"fn x() -> u8 { ({ 0 } + 1) }"#,
337 );
338 }
339
340 #[test]
341 fn remove_parens_im_tired_of_naming_tests() {
342 check_assist(
343 remove_parentheses,
344 r#"fn f() { 2 + $0(return 2) }"#,
345 r#"fn f() { 2 + return 2 }"#,
346 );
347
348 check_assist_not_applicable(remove_parentheses, r#"fn f() { $0(return 2) + 2 }"#);
349 }
350
351 #[test]
352 fn remove_parens_indirect_calls() {
353 check_assist(
354 remove_parentheses,
355 r#"fn f(call: fn(usize), arg: usize) { $0(call)(arg); }"#,
356 r#"fn f(call: fn(usize), arg: usize) { call(arg); }"#,
357 );
358 check_assist(
359 remove_parentheses,
360 r#"fn f<F>(call: F, arg: usize) where F: Fn(usize) { $0(call)(arg); }"#,
361 r#"fn f<F>(call: F, arg: usize) where F: Fn(usize) { call(arg); }"#,
362 );
363
364 check_assist_not_applicable(
366 remove_parentheses,
367 r#"
368struct Foo<T> {
369 t: T,
370}
371
372impl Foo<fn(usize)> {
373 fn foo(&self, arg: usize) {
374 $0(self.t)(arg);
375 }
376}"#,
377 );
378 }
379}