From 2b461c50d7a05c76c996d75e858cfe3c8a7390b7 Mon Sep 17 00:00:00 2001 From: Lukas Wirth Date: Thu, 29 Jul 2021 15:43:44 +0200 Subject: [PATCH] Refine extraction targets of extract_function assist --- crates/ide/src/rename.rs | 24 ++--- .../src/handlers/extract_function.rs | 90 +++++++++---------- crates/ide_db/src/rename.rs | 14 +-- 3 files changed, 64 insertions(+), 64 deletions(-) diff --git a/crates/ide/src/rename.rs b/crates/ide/src/rename.rs index f58be6d3fd..d0a49d1f37 100644 --- a/crates/ide/src/rename.rs +++ b/crates/ide/src/rename.rs @@ -617,22 +617,22 @@ fn main() { #[test] fn test_rename_struct_field() { check( - "j", + "foo", r#" -struct Foo { i$0: i32 } +struct Foo { field$0: i32 } impl Foo { fn new(i: i32) -> Self { - Self { i: i } + Self { field: i } } } "#, r#" -struct Foo { j: i32 } +struct Foo { foo: i32 } impl Foo { fn new(i: i32) -> Self { - Self { j: i } + Self { foo: i } } } "#, @@ -643,22 +643,22 @@ impl Foo { fn test_rename_field_in_field_shorthand() { cov_mark::check!(test_rename_field_in_field_shorthand); check( - "j", + "field", r#" -struct Foo { i$0: i32 } +struct Foo { foo$0: i32 } impl Foo { - fn new(i: i32) -> Self { - Self { i } + fn new(foo: i32) -> Self { + Self { foo } } } "#, r#" -struct Foo { j: i32 } +struct Foo { field: i32 } impl Foo { - fn new(i: i32) -> Self { - Self { j: i } + fn new(foo: i32) -> Self { + Self { field: foo } } } "#, diff --git a/crates/ide_assists/src/handlers/extract_function.rs b/crates/ide_assists/src/handlers/extract_function.rs index 4f0b1f5b79..abf8329f01 100644 --- a/crates/ide_assists/src/handlers/extract_function.rs +++ b/crates/ide_assists/src/handlers/extract_function.rs @@ -16,7 +16,7 @@ use syntax::{ AstNode, }, ted, - SyntaxKind::{self, BLOCK_EXPR, BREAK_EXPR, COMMENT, PATH_EXPR, RETURN_EXPR}, + SyntaxKind::{self, COMMENT}, SyntaxNode, SyntaxToken, TextRange, TextSize, TokenAtOffset, WalkEvent, T, }; @@ -466,22 +466,17 @@ enum FunctionBody { } impl FunctionBody { - fn from_whole_node(node: SyntaxNode) -> Option { - match node.kind() { - PATH_EXPR => None, - BREAK_EXPR => ast::BreakExpr::cast(node).and_then(|e| e.expr()).map(Self::Expr), - RETURN_EXPR => ast::ReturnExpr::cast(node).and_then(|e| e.expr()).map(Self::Expr), - BLOCK_EXPR => ast::BlockExpr::cast(node) - .filter(|it| it.is_standalone()) - .map(Into::into) - .map(Self::Expr), - _ => ast::Expr::cast(node).map(Self::Expr), + fn from_expr(expr: ast::Expr) -> Option { + match expr { + ast::Expr::BreakExpr(it) => it.expr().map(Self::Expr), + ast::Expr::ReturnExpr(it) => it.expr().map(Self::Expr), + ast::Expr::BlockExpr(it) if !it.is_standalone() => None, + expr => Some(Self::Expr(expr)), } } - fn from_range(node: SyntaxNode, text_range: TextRange) -> Option { - let block = ast::BlockExpr::cast(node)?; - Some(Self::Span { parent: block, text_range }) + fn from_range(parent: ast::BlockExpr, text_range: TextRange) -> FunctionBody { + Self::Span { parent, text_range } } fn indent_level(&self) -> IndentLevel { @@ -592,44 +587,39 @@ struct OutlivedLocal { /// ``` /// fn extraction_target(node: &SyntaxNode, selection_range: TextRange) -> Option { - // we have selected exactly the expr node - // wrap it before anything else + if let Some(stmt) = ast::Stmt::cast(node.clone()) { + return match stmt { + ast::Stmt::Item(_) => None, + ast::Stmt::ExprStmt(_) | ast::Stmt::LetStmt(_) => Some(FunctionBody::from_range( + node.parent().and_then(ast::BlockExpr::cast)?, + node.text_range(), + )), + }; + } + + let expr = ast::Expr::cast(node.clone())?; + // A node got selected fully if node.text_range() == selection_range { - let body = FunctionBody::from_whole_node(node.clone()); - if body.is_some() { - return body; - } + return FunctionBody::from_expr(expr.clone()); } - // we have selected a few statements in a block - // so covering_element returns the whole block - if node.kind() == BLOCK_EXPR { + // Covering element returned the parent block of one or multiple statements that have been selected + if let ast::Expr::BlockExpr(block) = expr { // Extract the full statements. - let statements_range = node - .children() - .filter(|c| selection_range.intersect(c.text_range()).is_some()) - .fold(selection_range, |acc, c| acc.cover(c.text_range())); - let body = FunctionBody::from_range(node.clone(), statements_range); - if body.is_some() { - return body; + let mut statements_range = block + .statements() + .filter(|stmt| selection_range.intersect(stmt.syntax().text_range()).is_some()) + .fold(selection_range, |acc, stmt| acc.cover(stmt.syntax().text_range())); + if let Some(e) = block + .tail_expr() + .filter(|it| selection_range.intersect(it.syntax().text_range()).is_some()) + { + statements_range = statements_range.cover(e.syntax().text_range()); } + return Some(FunctionBody::from_range(block, statements_range)); } - // we have selected single statement - // `from_whole_node` failed because (let) statement is not and expression - // so we try to expand covering_element to parent and repeat the previous - if let Some(parent) = node.parent() { - if parent.kind() == BLOCK_EXPR { - // Extract the full statement. - let body = FunctionBody::from_range(parent, node.text_range()); - if body.is_some() { - return body; - } - } - } - - // select the closest containing expr (both ifs are used) - std::iter::once(node.clone()).chain(node.ancestors()).find_map(FunctionBody::from_whole_node) + node.ancestors().find_map(ast::Expr::cast).and_then(FunctionBody::from_expr) } /// list local variables that are referenced in `body` @@ -3743,6 +3733,16 @@ async fn $0fun_name() { async fn some_function() { } +"#, + ); + } + + #[test] + fn extract_does_not_extract_standalone_blocks() { + check_assist_not_applicable( + extract_function, + r#" +fn main() $0{}$0 "#, ); } diff --git a/crates/ide_db/src/rename.rs b/crates/ide_db/src/rename.rs index 643e677812..4ca3260877 100644 --- a/crates/ide_db/src/rename.rs +++ b/crates/ide_db/src/rename.rs @@ -229,7 +229,7 @@ fn rename_reference( let ident_kind = IdentifierKind::classify(new_name)?; if matches!( - def, // is target a lifetime? + def, Definition::GenericParam(hir::GenericParam::LifetimeParam(_)) | Definition::Label(_) ) { match ident_kind { @@ -240,13 +240,13 @@ fn rename_reference( IdentifierKind::Lifetime => cov_mark::hit!(rename_lifetime), } } else { - match (ident_kind, def) { - (IdentifierKind::Lifetime, _) => { + match ident_kind { + IdentifierKind::Lifetime => { cov_mark::hit!(rename_not_an_ident_ref); bail!("Invalid name `{}`: not an identifier", new_name); } - (IdentifierKind::Ident, _) => cov_mark::hit!(rename_non_local), - (IdentifierKind::Underscore, _) => (), + IdentifierKind::Ident => cov_mark::hit!(rename_non_local), + IdentifierKind::Underscore => (), } } @@ -325,6 +325,8 @@ pub fn source_edit_from_references( fn source_edit_from_name(name: &ast::Name, new_name: &str) -> Option<(TextRange, String)> { if let Some(_) = ast::RecordPatField::for_field_name(name) { + // FIXME: instead of splitting the shorthand, recursively trigger a rename of the + // other name https://github.com/rust-analyzer/rust-analyzer/issues/6547 if let Some(ident_pat) = name.syntax().parent().and_then(ast::IdentPat::cast) { return Some(( TextRange::empty(ident_pat.syntax().text_range().start()), @@ -368,8 +370,6 @@ fn source_edit_from_name_ref( None } // init shorthand - // FIXME: instead of splitting the shorthand, recursively trigger a rename of the - // other name https://github.com/rust-analyzer/rust-analyzer/issues/6547 (None, Some(_)) if matches!(def, Definition::Field(_)) => { cov_mark::hit!(test_rename_field_in_field_shorthand); let s = name_ref.syntax().text_range().start();