Skip to content

Commit 05f5c77

Browse files
committed
Auto merge of rust-lang#17928 - roife:fix-issue-17869, r=Veykril
fix: keep comments in convert_while_to_loop Fix rust-lang#17869.
2 parents 85f6d15 + 036affc commit 05f5c77

File tree

1 file changed

+89
-8
lines changed

1 file changed

+89
-8
lines changed

src/tools/rust-analyzer/crates/ide-assists/src/handlers/convert_while_to_loop.rs

+89-8
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
1-
use std::iter::once;
1+
use std::iter;
22

3+
use either::Either;
34
use ide_db::syntax_helpers::node_ext::is_pattern_cond;
45
use syntax::{
56
ast::{
@@ -52,18 +53,30 @@ pub(crate) fn convert_while_to_loop(acc: &mut Assists, ctx: &AssistContext<'_>)
5253
|edit| {
5354
let while_indent_level = IndentLevel::from_node(while_expr.syntax());
5455

55-
let break_block =
56-
make::block_expr(once(make::expr_stmt(make::expr_break(None, None)).into()), None)
57-
.indent(while_indent_level);
56+
let break_block = make::block_expr(
57+
iter::once(make::expr_stmt(make::expr_break(None, None)).into()),
58+
None,
59+
)
60+
.indent(while_indent_level);
5861
let block_expr = if is_pattern_cond(while_cond.clone()) {
5962
let if_expr = make::expr_if(while_cond, while_body, Some(break_block.into()));
60-
let stmts = once(make::expr_stmt(if_expr).into());
63+
let stmts = iter::once(make::expr_stmt(if_expr).into());
6164
make::block_expr(stmts, None)
6265
} else {
6366
let if_cond = invert_boolean_expression(while_cond);
64-
let if_expr = make::expr_if(if_cond, break_block, None);
65-
let stmts = once(make::expr_stmt(if_expr).into()).chain(while_body.statements());
66-
make::block_expr(stmts, while_body.tail_expr())
67+
let if_expr = make::expr_if(if_cond, break_block, None).syntax().clone().into();
68+
let elements = while_body.stmt_list().map_or_else(
69+
|| Either::Left(iter::empty()),
70+
|stmts| {
71+
Either::Right(stmts.syntax().children_with_tokens().filter(|node_or_tok| {
72+
// Filter out the trailing expr
73+
!node_or_tok
74+
.as_node()
75+
.is_some_and(|node| ast::Expr::can_cast(node.kind()))
76+
}))
77+
},
78+
);
79+
make::hacky_block_expr(iter::once(if_expr).chain(elements), while_body.tail_expr())
6780
};
6881

6982
let replacement = make::expr_loop(block_expr.indent(while_indent_level));
@@ -182,6 +195,74 @@ fn main() {
182195
bar();
183196
}
184197
}
198+
"#,
199+
);
200+
}
201+
202+
#[test]
203+
fn preserve_comments() {
204+
check_assist(
205+
convert_while_to_loop,
206+
r#"
207+
fn main() {
208+
let mut i = 0;
209+
210+
$0while i < 5 {
211+
// comment 1
212+
dbg!(i);
213+
// comment 2
214+
i += 1;
215+
// comment 3
216+
}
217+
}
218+
"#,
219+
r#"
220+
fn main() {
221+
let mut i = 0;
222+
223+
loop {
224+
if i >= 5 {
225+
break;
226+
}
227+
// comment 1
228+
dbg!(i);
229+
// comment 2
230+
i += 1;
231+
// comment 3
232+
}
233+
}
234+
"#,
235+
);
236+
237+
check_assist(
238+
convert_while_to_loop,
239+
r#"
240+
fn main() {
241+
let v = vec![1, 2, 3];
242+
let iter = v.iter();
243+
244+
$0while let Some(i) = iter.next() {
245+
// comment 1
246+
dbg!(i);
247+
// comment 2
248+
}
249+
}
250+
"#,
251+
r#"
252+
fn main() {
253+
let v = vec![1, 2, 3];
254+
let iter = v.iter();
255+
256+
loop {
257+
if let Some(i) = iter.next() {
258+
// comment 1
259+
dbg!(i);
260+
// comment 2
261+
} else {
262+
break;
263+
}
264+
}
265+
}
185266
"#,
186267
);
187268
}

0 commit comments

Comments
 (0)