rust-analyzer/crates/ra_assists/src/handlers/inline_local_variable.rs

696 lines
12 KiB
Rust
Raw Normal View History

2020-03-04 13:25:22 +00:00
use ra_ide_db::defs::Definition;
2020-08-12 16:26:51 +00:00
use syntax::{
2019-08-19 11:11:09 +00:00
ast::{self, AstNode, AstToken},
TextRange,
};
2020-05-20 10:59:20 +00:00
use test_utils::mark;
2019-03-24 17:05:11 +00:00
use crate::{
assist_context::{AssistContext, Assists},
2020-06-28 22:36:05 +00:00
AssistId, AssistKind,
};
2019-03-24 17:05:11 +00:00
2019-10-26 16:08:13 +00:00
// Assist: inline_local_variable
//
// Inlines local variable.
//
// ```
// fn main() {
// let x<|> = 1 + 2;
// x * 4;
// }
// ```
// ->
// ```
// fn main() {
// (1 + 2) * 4;
// }
// ```
pub(crate) fn inline_local_variable(acc: &mut Assists, ctx: &AssistContext) -> Option<()> {
let let_stmt = ctx.find_node_at_offset::<ast::LetStmt>()?;
2019-08-19 11:11:09 +00:00
let bind_pat = match let_stmt.pat()? {
2020-07-31 18:09:09 +00:00
ast::Pat::IdentPat(pat) => pat,
2019-03-24 17:05:11 +00:00
_ => return None,
};
2020-04-09 21:35:05 +00:00
if bind_pat.mut_token().is_some() {
2020-05-20 10:59:20 +00:00
mark::hit!(test_not_inline_mut_variable);
2020-03-03 15:56:42 +00:00
return None;
}
if !bind_pat.syntax().text_range().contains_inclusive(ctx.offset()) {
2020-05-20 10:59:20 +00:00
mark::hit!(not_applicable_outside_of_bind_pat);
2019-03-24 17:05:11 +00:00
return None;
}
2019-07-19 08:24:41 +00:00
let initializer_expr = let_stmt.initializer()?;
2020-03-04 13:25:22 +00:00
let def = ctx.sema.to_def(&bind_pat)?;
let def = Definition::Local(def);
2020-07-01 12:11:34 +00:00
let refs = def.find_usages(&ctx.sema, None);
2020-03-04 13:25:22 +00:00
if refs.is_empty() {
2020-05-20 10:59:20 +00:00
mark::hit!(test_not_applicable_if_variable_unused);
2020-03-04 13:25:22 +00:00
return None;
};
2019-03-30 10:25:53 +00:00
let delete_range = if let Some(whitespace) = let_stmt
.syntax()
.next_sibling_or_token()
2019-07-19 08:24:41 +00:00
.and_then(|it| ast::Whitespace::cast(it.as_token()?.clone()))
2019-03-24 17:05:11 +00:00
{
2020-04-24 21:40:41 +00:00
TextRange::new(
2019-07-20 09:58:27 +00:00
let_stmt.syntax().text_range().start(),
whitespace.syntax().text_range().end(),
)
2019-03-24 17:05:11 +00:00
} else {
2019-07-20 09:58:27 +00:00
let_stmt.syntax().text_range()
2019-03-24 17:05:11 +00:00
};
let mut wrap_in_parens = vec![true; refs.len()];
for (i, desc) in refs.iter().enumerate() {
2020-03-04 13:25:22 +00:00
let usage_node = ctx
.covering_node_for_range(desc.file_range.range)
.ancestors()
.find_map(ast::PathExpr::cast)?;
let usage_parent_option = usage_node.syntax().parent().and_then(ast::Expr::cast);
let usage_parent = match usage_parent_option {
Some(u) => u,
None => {
wrap_in_parens[i] = false;
continue;
}
};
2019-08-19 11:11:09 +00:00
wrap_in_parens[i] = match (&initializer_expr, usage_parent) {
(ast::Expr::CallExpr(_), _)
| (ast::Expr::IndexExpr(_), _)
| (ast::Expr::MethodCallExpr(_), _)
| (ast::Expr::FieldExpr(_), _)
| (ast::Expr::TryExpr(_), _)
| (ast::Expr::RefExpr(_), _)
| (ast::Expr::Literal(_), _)
| (ast::Expr::TupleExpr(_), _)
| (ast::Expr::ArrayExpr(_), _)
| (ast::Expr::ParenExpr(_), _)
| (ast::Expr::PathExpr(_), _)
| (ast::Expr::BlockExpr(_), _)
2020-05-01 23:18:19 +00:00
| (ast::Expr::EffectExpr(_), _)
2019-08-19 11:11:09 +00:00
| (_, ast::Expr::CallExpr(_))
| (_, ast::Expr::TupleExpr(_))
| (_, ast::Expr::ArrayExpr(_))
| (_, ast::Expr::ParenExpr(_))
| (_, ast::Expr::ForExpr(_))
| (_, ast::Expr::WhileExpr(_))
| (_, ast::Expr::BreakExpr(_))
| (_, ast::Expr::ReturnExpr(_))
| (_, ast::Expr::MatchExpr(_)) => false,
_ => true,
};
}
2019-07-19 08:24:41 +00:00
let init_str = initializer_expr.syntax().text().to_string();
let init_in_paren = format!("({})", &init_str);
let target = bind_pat.syntax().text_range();
2020-06-28 22:36:05 +00:00
acc.add(
2020-07-02 21:48:35 +00:00
AssistId("inline_local_variable", AssistKind::RefactorInline),
2020-06-28 22:36:05 +00:00
"Inline variable",
target,
move |builder| {
builder.delete(delete_range);
for (desc, should_wrap) in refs.iter().zip(wrap_in_parens) {
let replacement =
if should_wrap { init_in_paren.clone() } else { init_str.clone() };
builder.replace(desc.file_range.range, replacement)
}
},
)
2019-03-24 17:05:11 +00:00
}
#[cfg(test)]
mod tests {
2020-05-20 10:59:20 +00:00
use test_utils::mark;
2020-03-03 15:56:42 +00:00
2020-05-06 08:16:55 +00:00
use crate::tests::{check_assist, check_assist_not_applicable};
2019-03-24 17:05:11 +00:00
use super::*;
#[test]
fn test_inline_let_bind_literal_expr() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
2019-03-24 17:05:11 +00:00
fn bar(a: usize) {}
fn foo() {
let a<|> = 1;
a + 1;
if a > 10 {
}
while a > 10 {
}
let b = a * 10;
bar(a);
}",
2020-03-03 16:03:46 +00:00
r"
2019-03-24 17:05:11 +00:00
fn bar(a: usize) {}
fn foo() {
2020-05-20 21:07:17 +00:00
1 + 1;
2019-03-24 17:05:11 +00:00
if 1 > 10 {
}
while 1 > 10 {
}
let b = 1 * 10;
bar(1);
}",
);
}
#[test]
fn test_inline_let_bind_bin_expr() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
2019-03-24 17:05:11 +00:00
fn bar(a: usize) {}
fn foo() {
let a<|> = 1 + 1;
a + 1;
if a > 10 {
}
while a > 10 {
}
let b = a * 10;
bar(a);
}",
2020-03-03 16:03:46 +00:00
r"
2019-03-24 17:05:11 +00:00
fn bar(a: usize) {}
fn foo() {
2020-05-20 21:07:17 +00:00
(1 + 1) + 1;
2019-03-24 17:05:11 +00:00
if (1 + 1) > 10 {
}
while (1 + 1) > 10 {
}
let b = (1 + 1) * 10;
bar(1 + 1);
2019-03-24 17:05:11 +00:00
}",
);
}
#[test]
fn test_inline_let_bind_function_call_expr() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
2019-03-24 17:05:11 +00:00
fn bar(a: usize) {}
fn foo() {
let a<|> = bar(1);
a + 1;
if a > 10 {
}
while a > 10 {
}
let b = a * 10;
bar(a);
}",
2020-03-03 16:03:46 +00:00
r"
2019-03-24 17:05:11 +00:00
fn bar(a: usize) {}
fn foo() {
2020-05-20 21:07:17 +00:00
bar(1) + 1;
2019-03-24 17:05:11 +00:00
if bar(1) > 10 {
}
while bar(1) > 10 {
}
let b = bar(1) * 10;
bar(bar(1));
}",
);
}
#[test]
fn test_inline_let_bind_cast_expr() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
2019-03-24 17:05:11 +00:00
fn bar(a: usize): usize { a }
fn foo() {
let a<|> = bar(1) as u64;
a + 1;
if a > 10 {
}
while a > 10 {
}
let b = a * 10;
bar(a);
}",
2020-03-03 16:03:46 +00:00
r"
2019-03-24 17:05:11 +00:00
fn bar(a: usize): usize { a }
fn foo() {
2020-05-20 21:07:17 +00:00
(bar(1) as u64) + 1;
2019-03-24 17:05:11 +00:00
if (bar(1) as u64) > 10 {
}
while (bar(1) as u64) > 10 {
}
let b = (bar(1) as u64) * 10;
bar(bar(1) as u64);
2019-03-24 17:05:11 +00:00
}",
);
}
#[test]
fn test_inline_let_bind_block_expr() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
2019-03-24 17:05:11 +00:00
fn foo() {
let a<|> = { 10 + 1 };
a + 1;
if a > 10 {
}
while a > 10 {
}
let b = a * 10;
bar(a);
}",
2020-03-03 16:03:46 +00:00
r"
2019-03-24 17:05:11 +00:00
fn foo() {
2020-05-20 21:07:17 +00:00
{ 10 + 1 } + 1;
2019-03-24 17:05:11 +00:00
if { 10 + 1 } > 10 {
}
while { 10 + 1 } > 10 {
}
let b = { 10 + 1 } * 10;
bar({ 10 + 1 });
}",
);
}
#[test]
fn test_inline_let_bind_paren_expr() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
2019-03-24 17:05:11 +00:00
fn foo() {
let a<|> = ( 10 + 1 );
a + 1;
if a > 10 {
}
while a > 10 {
}
let b = a * 10;
bar(a);
}",
2020-03-03 16:03:46 +00:00
r"
2019-03-24 17:05:11 +00:00
fn foo() {
2020-05-20 21:07:17 +00:00
( 10 + 1 ) + 1;
2019-03-24 17:05:11 +00:00
if ( 10 + 1 ) > 10 {
}
while ( 10 + 1 ) > 10 {
}
let b = ( 10 + 1 ) * 10;
bar(( 10 + 1 ));
}",
);
}
#[test]
fn test_not_inline_mut_variable() {
2020-05-20 10:59:20 +00:00
mark::check!(test_not_inline_mut_variable);
2019-03-24 17:05:11 +00:00
check_assist_not_applicable(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 15:56:42 +00:00
r"
2019-03-24 17:05:11 +00:00
fn foo() {
let mut a<|> = 1 + 1;
a + 1;
}",
);
}
#[test]
fn test_call_expr() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
fn foo() {
let a<|> = bar(10 + 1);
let b = a * 10;
let c = a as usize;
}",
2020-03-03 16:03:46 +00:00
r"
fn foo() {
2020-05-20 21:07:17 +00:00
let b = bar(10 + 1) * 10;
let c = bar(10 + 1) as usize;
}",
);
}
#[test]
fn test_index_expr() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
fn foo() {
let x = vec![1, 2, 3];
let a<|> = x[0];
let b = a * 10;
let c = a as usize;
}",
2020-03-03 16:03:46 +00:00
r"
fn foo() {
let x = vec![1, 2, 3];
2020-05-20 21:07:17 +00:00
let b = x[0] * 10;
let c = x[0] as usize;
}",
);
}
#[test]
fn test_method_call_expr() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
fn foo() {
let bar = vec![1];
let a<|> = bar.len();
let b = a * 10;
let c = a as usize;
}",
2020-03-03 16:03:46 +00:00
r"
fn foo() {
let bar = vec![1];
2020-05-20 21:07:17 +00:00
let b = bar.len() * 10;
let c = bar.len() as usize;
}",
);
}
#[test]
fn test_field_expr() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
struct Bar {
foo: usize
}
fn foo() {
let bar = Bar { foo: 1 };
let a<|> = bar.foo;
let b = a * 10;
let c = a as usize;
}",
2020-03-03 16:03:46 +00:00
r"
struct Bar {
foo: usize
}
fn foo() {
let bar = Bar { foo: 1 };
2020-05-20 21:07:17 +00:00
let b = bar.foo * 10;
let c = bar.foo as usize;
}",
);
}
#[test]
fn test_try_expr() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
fn foo() -> Option<usize> {
let bar = Some(1);
let a<|> = bar?;
let b = a * 10;
let c = a as usize;
None
}",
2020-03-03 16:03:46 +00:00
r"
fn foo() -> Option<usize> {
let bar = Some(1);
2020-05-20 21:07:17 +00:00
let b = bar? * 10;
let c = bar? as usize;
None
}",
);
}
#[test]
fn test_ref_expr() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
fn foo() {
let bar = 10;
let a<|> = &bar;
let b = a * 10;
}",
2020-03-03 16:03:46 +00:00
r"
fn foo() {
let bar = 10;
2020-05-20 21:07:17 +00:00
let b = &bar * 10;
}",
);
}
#[test]
fn test_tuple_expr() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
fn foo() {
let a<|> = (10, 20);
let b = a[0];
}",
2020-03-03 16:03:46 +00:00
r"
fn foo() {
2020-05-20 21:07:17 +00:00
let b = (10, 20)[0];
}",
);
}
#[test]
fn test_array_expr() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
fn foo() {
let a<|> = [1, 2, 3];
let b = a.len();
}",
2020-03-03 16:03:46 +00:00
r"
fn foo() {
2020-05-20 21:07:17 +00:00
let b = [1, 2, 3].len();
}",
);
}
#[test]
fn test_paren() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
fn foo() {
let a<|> = (10 + 20);
let b = a * 10;
let c = a as usize;
}",
2020-03-03 16:03:46 +00:00
r"
fn foo() {
2020-05-20 21:07:17 +00:00
let b = (10 + 20) * 10;
let c = (10 + 20) as usize;
}",
);
}
#[test]
fn test_path_expr() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
fn foo() {
let d = 10;
let a<|> = d;
let b = a * 10;
let c = a as usize;
}",
2020-03-03 16:03:46 +00:00
r"
fn foo() {
let d = 10;
2020-05-20 21:07:17 +00:00
let b = d * 10;
let c = d as usize;
}",
);
}
#[test]
fn test_block_expr() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
fn foo() {
let a<|> = { 10 };
let b = a * 10;
let c = a as usize;
}",
2020-03-03 16:03:46 +00:00
r"
fn foo() {
2020-05-20 21:07:17 +00:00
let b = { 10 } * 10;
let c = { 10 } as usize;
}",
);
}
#[test]
fn test_used_in_different_expr1() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
fn foo() {
let a<|> = 10 + 20;
let b = a * 10;
let c = (a, 20);
let d = [a, 10];
let e = (a);
}",
2020-03-03 16:03:46 +00:00
r"
fn foo() {
2020-05-20 21:07:17 +00:00
let b = (10 + 20) * 10;
let c = (10 + 20, 20);
let d = [10 + 20, 10];
let e = (10 + 20);
}",
);
}
#[test]
fn test_used_in_for_expr() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
fn foo() {
let a<|> = vec![10, 20];
for i in a {}
}",
2020-03-03 16:03:46 +00:00
r"
fn foo() {
2020-05-20 21:07:17 +00:00
for i in vec![10, 20] {}
}",
);
}
#[test]
fn test_used_in_while_expr() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
fn foo() {
let a<|> = 1 > 0;
while a {}
}",
2020-03-03 16:03:46 +00:00
r"
fn foo() {
2020-05-20 21:07:17 +00:00
while 1 > 0 {}
}",
);
}
#[test]
fn test_used_in_break_expr() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
fn foo() {
let a<|> = 1 + 1;
loop {
break a;
}
}",
2020-03-03 16:03:46 +00:00
r"
fn foo() {
2020-05-20 21:07:17 +00:00
loop {
break 1 + 1;
}
}",
);
}
#[test]
fn test_used_in_return_expr() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
fn foo() {
let a<|> = 1 > 0;
return a;
}",
2020-03-03 16:03:46 +00:00
r"
fn foo() {
2020-05-20 21:07:17 +00:00
return 1 > 0;
}",
);
}
#[test]
fn test_used_in_match_expr() {
check_assist(
2020-01-19 16:39:53 +00:00
inline_local_variable,
2020-03-03 16:03:46 +00:00
r"
fn foo() {
let a<|> = 1 > 0;
match a {}
}",
2020-03-03 16:03:46 +00:00
r"
fn foo() {
2020-05-20 21:07:17 +00:00
match 1 > 0 {}
2019-03-24 17:05:11 +00:00
}",
);
}
2020-01-25 20:07:21 +00:00
#[test]
fn test_not_applicable_if_variable_unused() {
2020-05-20 10:59:20 +00:00
mark::check!(test_not_applicable_if_variable_unused);
2020-01-25 20:07:21 +00:00
check_assist_not_applicable(
inline_local_variable,
2020-03-03 15:56:42 +00:00
r"
2020-01-25 20:07:21 +00:00
fn foo() {
let <|>a = 0;
}
",
)
}
2020-03-03 15:56:42 +00:00
#[test]
fn not_applicable_outside_of_bind_pat() {
2020-05-20 10:59:20 +00:00
mark::check!(not_applicable_outside_of_bind_pat);
2020-03-03 15:56:42 +00:00
check_assist_not_applicable(
inline_local_variable,
r"
fn main() {
let x = <|>1 + 2;
x * 4;
}
",
)
}
2019-03-24 17:05:11 +00:00
}