diff options
| author | bors[bot] <26634292+bors[bot]@users.noreply.github.com> | 2021-12-03 20:21:55 +0000 |
|---|---|---|
| committer | GitHub <noreply@github.com> | 2021-12-03 20:21:55 +0000 |
| commit | bf7622614895ec6cbed7ea5a6b6b2616680fe50f (patch) | |
| tree | 4f3b733bea85d0a581215388ac44db4129db7ec5 | |
| parent | 3f3289df2aa44a2f33907e6e2abad75fa9787d01 (diff) | |
| parent | 79b4e67b4ffdbdb58ad5b02398fd174e929609f8 (diff) | |
| download | rust-bf7622614895ec6cbed7ea5a6b6b2616680fe50f.tar.gz rust-bf7622614895ec6cbed7ea5a6b6b2616680fe50f.zip | |
Merge #10916
10916: feat: ide: Convert while let to loop r=Veykril a=rainy-me close #10901 Co-authored-by: rainy-me <github@yue.coffee>
| -rw-r--r-- | crates/ide_assists/src/handlers/convert_while_to_loop.rs | 63 |
1 files changed, 33 insertions, 30 deletions
diff --git a/crates/ide_assists/src/handlers/convert_while_to_loop.rs b/crates/ide_assists/src/handlers/convert_while_to_loop.rs index 92dc2ee73a3..2bc64e77a38 100644 --- a/crates/ide_assists/src/handlers/convert_while_to_loop.rs +++ b/crates/ide_assists/src/handlers/convert_while_to_loop.rs @@ -39,16 +39,10 @@ use crate::{ // ``` pub(crate) fn convert_while_to_loop(acc: &mut Assists, ctx: &AssistContext) -> Option<()> { let while_kw = ctx.find_token_syntax_at_offset(T![while])?; - let while_expr: ast::WhileExpr = while_kw.parent().and_then(ast::WhileExpr::cast)?; + let while_expr = while_kw.parent().and_then(ast::WhileExpr::cast)?; let while_body = while_expr.loop_body()?; - let cond = while_expr.condition()?; - - // Don't handle while let - if cond.pat().is_some() { - return None; - }; - - let cond_expr = cond.expr()?; + let while_cond = while_expr.condition()?; + let while_cond_expr = while_cond.expr()?; let target = while_expr.syntax().text_range(); acc.add( @@ -58,27 +52,25 @@ pub(crate) fn convert_while_to_loop(acc: &mut Assists, ctx: &AssistContext) -> O |edit| { let while_indent_level = IndentLevel::from_node(while_expr.syntax()); - let replacement = { - let if_expr = { - let cond = invert_boolean_expression(cond_expr); - let then_branch = make::block_expr( - once(make::expr_stmt(make::expr_break(None)).into()), - None, - ); - - make::expr_if(make::condition(cond, None), then_branch, None) - }; - - let if_expr = if_expr.indent(while_indent_level); - let stmts = once(make::expr_stmt(if_expr).into()).chain(while_body.statements()); - - let block_expr = make::block_expr(stmts, while_body.tail_expr()); - - let block_expr = block_expr.indent(while_indent_level); - - make::expr_loop(block_expr) + let break_block = + make::block_expr(once(make::expr_stmt(make::expr_break(None)).into()), None) + .indent(while_indent_level); + let block_expr = match while_cond.pat() { + Some(_) => { + let if_expr = make::expr_if(while_cond, while_body, Some(break_block.into())); + let stmts = once(make::expr_stmt(if_expr).into()); + make::block_expr(stmts, None) + } + None => { + let if_cond = make::condition(invert_boolean_expression(while_cond_expr), None); + let if_expr = make::expr_if(if_cond, break_block, None); + let stmts = + once(make::expr_stmt(if_expr).into()).chain(while_body.statements()); + make::block_expr(stmts, while_body.tail_expr()) + } }; + let replacement = make::expr_loop(block_expr.indent(while_indent_level)); edit.replace(target, replacement.syntax().text()) }, ) @@ -160,8 +152,8 @@ fn main() { } #[test] - fn ignore_while_let() { - check_assist_not_applicable( + fn convert_while_let() { + check_assist( convert_while_to_loop, r#" fn main() { @@ -170,6 +162,17 @@ fn main() { } } "#, + r#" +fn main() { + loop { + if let Some(_) = foo() { + bar(); + } else { + break; + } + } +} +"#, ); } |
