2021-04-20 14:16:23 +00:00
|
|
|
use either::Either;
|
|
|
|
use hir::PathResolution;
|
|
|
|
use ide_db::{base_db::FileId, defs::Definition, search::FileReference};
|
2021-02-12 20:30:55 +00:00
|
|
|
use rustc_hash::FxHashMap;
|
2020-08-12 16:26:51 +00:00
|
|
|
use syntax::{
|
2021-04-20 14:16:23 +00:00
|
|
|
ast::{self, AstNode, AstToken, NameOwner},
|
2019-03-25 17:02:06 +00:00
|
|
|
TextRange,
|
|
|
|
};
|
2019-03-24 17:05:11 +00:00
|
|
|
|
2020-05-06 16:45:35 +00:00
|
|
|
use crate::{
|
|
|
|
assist_context::{AssistContext, Assists},
|
2020-06-28 22:36:05 +00:00
|
|
|
AssistId, AssistKind,
|
2020-05-06 16:45:35 +00:00
|
|
|
};
|
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() {
|
2021-01-06 20:15:48 +00:00
|
|
|
// let x$0 = 1 + 2;
|
2019-10-26 16:08:13 +00:00
|
|
|
// x * 4;
|
|
|
|
// }
|
|
|
|
// ```
|
|
|
|
// ->
|
|
|
|
// ```
|
|
|
|
// fn main() {
|
|
|
|
// (1 + 2) * 4;
|
|
|
|
// }
|
|
|
|
// ```
|
2020-05-06 16:45:35 +00:00
|
|
|
pub(crate) fn inline_local_variable(acc: &mut Assists, ctx: &AssistContext) -> Option<()> {
|
2021-04-20 14:16:23 +00:00
|
|
|
let InlineData { let_stmt, delete_let, replace_usages, target } =
|
|
|
|
inline_let(ctx).or_else(|| inline_usage(ctx))?;
|
2019-07-19 08:24:41 +00:00
|
|
|
let initializer_expr = let_stmt.initializer()?;
|
2020-03-04 13:25:22 +00:00
|
|
|
|
2021-04-20 14:16:23 +00:00
|
|
|
let delete_range = if delete_let {
|
|
|
|
if let Some(whitespace) = let_stmt
|
|
|
|
.syntax()
|
|
|
|
.next_sibling_or_token()
|
|
|
|
.and_then(|it| ast::Whitespace::cast(it.as_token()?.clone()))
|
|
|
|
{
|
|
|
|
Some(TextRange::new(
|
|
|
|
let_stmt.syntax().text_range().start(),
|
|
|
|
whitespace.syntax().text_range().end(),
|
|
|
|
))
|
|
|
|
} else {
|
|
|
|
Some(let_stmt.syntax().text_range())
|
|
|
|
}
|
2019-03-24 17:05:11 +00:00
|
|
|
} else {
|
2021-04-20 14:16:23 +00:00
|
|
|
None
|
2019-03-24 17:05:11 +00:00
|
|
|
};
|
|
|
|
|
2021-04-20 14:16:23 +00:00
|
|
|
let wrap_in_parens = replace_usages
|
2021-02-07 17:38:12 +00:00
|
|
|
.iter()
|
|
|
|
.map(|(&file_id, refs)| {
|
|
|
|
refs.iter()
|
|
|
|
.map(|&FileReference { range, .. }| {
|
|
|
|
let usage_node = ctx
|
|
|
|
.covering_node_for_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,
|
2021-02-20 22:45:30 +00:00
|
|
|
None => return Some(false),
|
2021-02-07 17:38:12 +00:00
|
|
|
};
|
|
|
|
|
2021-02-20 22:45:30 +00:00
|
|
|
Some(!matches!(
|
2021-02-07 17:38:12 +00:00
|
|
|
(&initializer_expr, usage_parent),
|
2021-06-17 15:37:14 +00:00
|
|
|
(
|
|
|
|
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(_)
|
|
|
|
| ast::Expr::EffectExpr(_),
|
|
|
|
_
|
|
|
|
) | (
|
|
|
|
_,
|
|
|
|
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(_)
|
|
|
|
)
|
2021-02-07 17:38:12 +00:00
|
|
|
))
|
|
|
|
})
|
2021-02-20 22:45:30 +00:00
|
|
|
.collect::<Option<_>>()
|
2021-02-07 17:38:12 +00:00
|
|
|
.map(|b| (file_id, b))
|
2021-01-11 23:05:07 +00:00
|
|
|
})
|
2021-02-20 22:45:30 +00:00
|
|
|
.collect::<Option<FxHashMap<_, Vec<_>>>>()?;
|
2019-03-25 17:02:06 +00:00
|
|
|
|
2019-07-19 08:24:41 +00:00
|
|
|
let init_str = initializer_expr.syntax().text().to_string();
|
2019-03-25 17:02:06 +00:00
|
|
|
let init_in_paren = format!("({})", &init_str);
|
|
|
|
|
2021-04-20 14:16:23 +00:00
|
|
|
let target = match target {
|
|
|
|
ast::NameOrNameRef::Name(it) => it.syntax().text_range(),
|
|
|
|
ast::NameOrNameRef::NameRef(it) => it.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| {
|
2021-04-20 14:16:23 +00:00
|
|
|
if let Some(range) = delete_range {
|
|
|
|
builder.delete(range);
|
|
|
|
}
|
|
|
|
for (file_id, references) in replace_usages {
|
2021-02-07 17:38:12 +00:00
|
|
|
for (&should_wrap, reference) in wrap_in_parens[&file_id].iter().zip(references) {
|
|
|
|
let replacement =
|
|
|
|
if should_wrap { init_in_paren.clone() } else { init_str.clone() };
|
2021-02-09 15:03:39 +00:00
|
|
|
match reference.name.as_name_ref() {
|
2021-02-07 17:38:12 +00:00
|
|
|
Some(name_ref)
|
|
|
|
if ast::RecordExprField::for_field_name(name_ref).is_some() =>
|
|
|
|
{
|
2021-03-08 20:19:44 +00:00
|
|
|
cov_mark::hit!(inline_field_shorthand);
|
2021-02-07 17:38:12 +00:00
|
|
|
builder.insert(reference.range.end(), format!(": {}", replacement));
|
|
|
|
}
|
|
|
|
_ => builder.replace(reference.range, replacement),
|
2020-08-26 11:03:14 +00:00
|
|
|
}
|
|
|
|
}
|
2020-06-28 22:36:05 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
)
|
2019-03-24 17:05:11 +00:00
|
|
|
}
|
|
|
|
|
2021-04-20 14:16:23 +00:00
|
|
|
struct InlineData {
|
|
|
|
let_stmt: ast::LetStmt,
|
|
|
|
delete_let: bool,
|
|
|
|
target: ast::NameOrNameRef,
|
|
|
|
replace_usages: FxHashMap<FileId, Vec<FileReference>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
fn inline_let(ctx: &AssistContext) -> Option<InlineData> {
|
|
|
|
let let_stmt = ctx.find_node_at_offset::<ast::LetStmt>()?;
|
|
|
|
let bind_pat = match let_stmt.pat()? {
|
|
|
|
ast::Pat::IdentPat(pat) => pat,
|
|
|
|
_ => return None,
|
|
|
|
};
|
|
|
|
if bind_pat.mut_token().is_some() {
|
|
|
|
cov_mark::hit!(test_not_inline_mut_variable);
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
if !bind_pat.syntax().text_range().contains_inclusive(ctx.offset()) {
|
|
|
|
cov_mark::hit!(not_applicable_outside_of_bind_pat);
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
|
|
|
let def = ctx.sema.to_def(&bind_pat)?;
|
|
|
|
let def = Definition::Local(def);
|
|
|
|
let usages = def.usages(&ctx.sema).all();
|
|
|
|
if usages.is_empty() {
|
|
|
|
cov_mark::hit!(test_not_applicable_if_variable_unused);
|
|
|
|
return None;
|
|
|
|
};
|
|
|
|
|
|
|
|
Some(InlineData {
|
|
|
|
let_stmt,
|
|
|
|
delete_let: true,
|
|
|
|
target: ast::NameOrNameRef::Name(bind_pat.name()?),
|
|
|
|
replace_usages: usages.references,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
fn inline_usage(ctx: &AssistContext) -> Option<InlineData> {
|
|
|
|
let path_expr = ctx.find_node_at_offset::<ast::PathExpr>()?;
|
|
|
|
let path = path_expr.path()?;
|
|
|
|
let name = match path.as_single_segment()?.kind()? {
|
|
|
|
ast::PathSegmentKind::Name(name) => name,
|
|
|
|
_ => return None,
|
|
|
|
};
|
|
|
|
|
|
|
|
let local = match ctx.sema.resolve_path(&path)? {
|
|
|
|
PathResolution::Local(local) => local,
|
|
|
|
_ => return None,
|
|
|
|
};
|
2021-06-04 18:36:43 +00:00
|
|
|
if local.is_mut(ctx.sema.db) {
|
|
|
|
cov_mark::hit!(test_not_inline_mut_variable_use);
|
|
|
|
return None;
|
|
|
|
}
|
2021-04-20 14:16:23 +00:00
|
|
|
|
|
|
|
let bind_pat = match local.source(ctx.db()).value {
|
|
|
|
Either::Left(ident) => ident,
|
|
|
|
_ => return None,
|
|
|
|
};
|
|
|
|
|
|
|
|
let let_stmt = ast::LetStmt::cast(bind_pat.syntax().parent()?)?;
|
|
|
|
|
|
|
|
let def = Definition::Local(local);
|
|
|
|
let mut usages = def.usages(&ctx.sema).all();
|
|
|
|
|
|
|
|
let delete_let = usages.references.values().map(|v| v.len()).sum::<usize>() == 1;
|
|
|
|
|
|
|
|
for references in usages.references.values_mut() {
|
|
|
|
references.retain(|reference| reference.name.as_name_ref() == Some(&name));
|
|
|
|
}
|
|
|
|
|
|
|
|
Some(InlineData {
|
|
|
|
let_stmt,
|
|
|
|
delete_let,
|
|
|
|
target: ast::NameOrNameRef::NameRef(name),
|
|
|
|
replace_usages: usages.references,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-03-24 17:05:11 +00:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
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() {
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = 1;
|
2019-03-24 17:05:11 +00:00
|
|
|
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() {
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = 1 + 1;
|
2019-03-24 17:05:11 +00:00
|
|
|
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;
|
2019-03-25 17:02:06 +00:00
|
|
|
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() {
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = bar(1);
|
2019-03-24 17:05:11 +00:00
|
|
|
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() {
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = bar(1) as u64;
|
2019-03-24 17:05:11 +00:00
|
|
|
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;
|
2019-03-25 17:02:06 +00:00
|
|
|
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() {
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = { 10 + 1 };
|
2019-03-24 17:05:11 +00:00
|
|
|
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() {
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = ( 10 + 1 );
|
2019-03-24 17:05:11 +00:00
|
|
|
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() {
|
2021-03-08 20:19:44 +00:00
|
|
|
cov_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() {
|
2021-01-06 20:15:48 +00:00
|
|
|
let mut a$0 = 1 + 1;
|
2019-03-24 17:05:11 +00:00
|
|
|
a + 1;
|
2019-03-25 17:02:06 +00:00
|
|
|
}",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-06-04 18:36:43 +00:00
|
|
|
#[test]
|
|
|
|
fn test_not_inline_mut_variable_use() {
|
|
|
|
cov_mark::check!(test_not_inline_mut_variable_use);
|
|
|
|
check_assist_not_applicable(
|
|
|
|
inline_local_variable,
|
|
|
|
r"
|
|
|
|
fn foo() {
|
|
|
|
let mut a = 1 + 1;
|
|
|
|
a$0 + 1;
|
|
|
|
}",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-03-25 17:02:06 +00:00
|
|
|
#[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"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = bar(10 + 1);
|
2019-03-25 17:02:06 +00:00
|
|
|
let b = a * 10;
|
|
|
|
let c = a as usize;
|
|
|
|
}",
|
2020-03-03 16:03:46 +00:00
|
|
|
r"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
2020-05-20 21:07:17 +00:00
|
|
|
let b = bar(10 + 1) * 10;
|
2019-03-25 17:02:06 +00:00
|
|
|
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"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
|
|
|
let x = vec![1, 2, 3];
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = x[0];
|
2019-03-25 17:02:06 +00:00
|
|
|
let b = a * 10;
|
|
|
|
let c = a as usize;
|
|
|
|
}",
|
2020-03-03 16:03:46 +00:00
|
|
|
r"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
|
|
|
let x = vec![1, 2, 3];
|
2020-05-20 21:07:17 +00:00
|
|
|
let b = x[0] * 10;
|
2019-03-25 17:02:06 +00:00
|
|
|
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"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
|
|
|
let bar = vec![1];
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = bar.len();
|
2019-03-25 17:02:06 +00:00
|
|
|
let b = a * 10;
|
|
|
|
let c = a as usize;
|
|
|
|
}",
|
2020-03-03 16:03:46 +00:00
|
|
|
r"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
|
|
|
let bar = vec![1];
|
2020-05-20 21:07:17 +00:00
|
|
|
let b = bar.len() * 10;
|
2019-03-25 17:02:06 +00:00
|
|
|
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"
|
2019-03-25 17:02:06 +00:00
|
|
|
struct Bar {
|
|
|
|
foo: usize
|
|
|
|
}
|
|
|
|
|
|
|
|
fn foo() {
|
|
|
|
let bar = Bar { foo: 1 };
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = bar.foo;
|
2019-03-25 17:02:06 +00:00
|
|
|
let b = a * 10;
|
|
|
|
let c = a as usize;
|
|
|
|
}",
|
2020-03-03 16:03:46 +00:00
|
|
|
r"
|
2019-03-25 17:02:06 +00:00
|
|
|
struct Bar {
|
|
|
|
foo: usize
|
|
|
|
}
|
|
|
|
|
|
|
|
fn foo() {
|
|
|
|
let bar = Bar { foo: 1 };
|
2020-05-20 21:07:17 +00:00
|
|
|
let b = bar.foo * 10;
|
2019-03-25 17:02:06 +00:00
|
|
|
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"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() -> Option<usize> {
|
|
|
|
let bar = Some(1);
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = bar?;
|
2019-03-25 17:02:06 +00:00
|
|
|
let b = a * 10;
|
|
|
|
let c = a as usize;
|
|
|
|
None
|
|
|
|
}",
|
2020-03-03 16:03:46 +00:00
|
|
|
r"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() -> Option<usize> {
|
|
|
|
let bar = Some(1);
|
2020-05-20 21:07:17 +00:00
|
|
|
let b = bar? * 10;
|
2019-03-25 17:02:06 +00:00
|
|
|
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"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
|
|
|
let bar = 10;
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = &bar;
|
2019-03-25 17:02:06 +00:00
|
|
|
let b = a * 10;
|
|
|
|
}",
|
2020-03-03 16:03:46 +00:00
|
|
|
r"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
|
|
|
let bar = 10;
|
2020-05-20 21:07:17 +00:00
|
|
|
let b = &bar * 10;
|
2019-03-25 17:02:06 +00:00
|
|
|
}",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[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"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = (10, 20);
|
2019-03-25 17:02:06 +00:00
|
|
|
let b = a[0];
|
|
|
|
}",
|
2020-03-03 16:03:46 +00:00
|
|
|
r"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
2020-05-20 21:07:17 +00:00
|
|
|
let b = (10, 20)[0];
|
2019-03-25 17:02:06 +00:00
|
|
|
}",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[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"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = [1, 2, 3];
|
2019-03-25 17:02:06 +00:00
|
|
|
let b = a.len();
|
|
|
|
}",
|
2020-03-03 16:03:46 +00:00
|
|
|
r"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
2020-05-20 21:07:17 +00:00
|
|
|
let b = [1, 2, 3].len();
|
2019-03-25 17:02:06 +00:00
|
|
|
}",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[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"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = (10 + 20);
|
2019-03-25 17:02:06 +00:00
|
|
|
let b = a * 10;
|
|
|
|
let c = a as usize;
|
|
|
|
}",
|
2020-03-03 16:03:46 +00:00
|
|
|
r"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
2020-05-20 21:07:17 +00:00
|
|
|
let b = (10 + 20) * 10;
|
2019-03-25 17:02:06 +00:00
|
|
|
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"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
|
|
|
let d = 10;
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = d;
|
2019-03-25 17:02:06 +00:00
|
|
|
let b = a * 10;
|
|
|
|
let c = a as usize;
|
|
|
|
}",
|
2020-03-03 16:03:46 +00:00
|
|
|
r"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
|
|
|
let d = 10;
|
2020-05-20 21:07:17 +00:00
|
|
|
let b = d * 10;
|
2019-03-25 17:02:06 +00:00
|
|
|
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"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = { 10 };
|
2019-03-25 17:02:06 +00:00
|
|
|
let b = a * 10;
|
|
|
|
let c = a as usize;
|
|
|
|
}",
|
2020-03-03 16:03:46 +00:00
|
|
|
r"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
2020-05-20 21:07:17 +00:00
|
|
|
let b = { 10 } * 10;
|
2019-03-25 17:02:06 +00:00
|
|
|
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"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = 10 + 20;
|
2019-03-25 17:02:06 +00:00
|
|
|
let b = a * 10;
|
|
|
|
let c = (a, 20);
|
|
|
|
let d = [a, 10];
|
|
|
|
let e = (a);
|
|
|
|
}",
|
2020-03-03 16:03:46 +00:00
|
|
|
r"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
2020-05-20 21:07:17 +00:00
|
|
|
let b = (10 + 20) * 10;
|
2019-03-25 17:02:06 +00:00
|
|
|
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"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = vec![10, 20];
|
2019-03-25 17:02:06 +00:00
|
|
|
for i in a {}
|
|
|
|
}",
|
2020-03-03 16:03:46 +00:00
|
|
|
r"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
2020-05-20 21:07:17 +00:00
|
|
|
for i in vec![10, 20] {}
|
2019-03-25 17:02:06 +00:00
|
|
|
}",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[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"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = 1 > 0;
|
2019-03-25 17:02:06 +00:00
|
|
|
while a {}
|
|
|
|
}",
|
2020-03-03 16:03:46 +00:00
|
|
|
r"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
2020-05-20 21:07:17 +00:00
|
|
|
while 1 > 0 {}
|
2019-03-25 17:02:06 +00:00
|
|
|
}",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[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"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = 1 + 1;
|
2019-03-25 17:02:06 +00:00
|
|
|
loop {
|
|
|
|
break a;
|
|
|
|
}
|
|
|
|
}",
|
2020-03-03 16:03:46 +00:00
|
|
|
r"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
2020-05-20 21:07:17 +00:00
|
|
|
loop {
|
2019-03-25 17:02:06 +00:00
|
|
|
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"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = 1 > 0;
|
2019-03-25 17:02:06 +00:00
|
|
|
return a;
|
|
|
|
}",
|
2020-03-03 16:03:46 +00:00
|
|
|
r"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
2020-05-20 21:07:17 +00:00
|
|
|
return 1 > 0;
|
2019-03-25 17:02:06 +00:00
|
|
|
}",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[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"
|
2019-03-25 17:02:06 +00:00
|
|
|
fn foo() {
|
2021-01-06 20:15:48 +00:00
|
|
|
let a$0 = 1 > 0;
|
2019-03-25 17:02:06 +00:00
|
|
|
match a {}
|
|
|
|
}",
|
2020-03-03 16:03:46 +00:00
|
|
|
r"
|
2019-03-25 17:02:06 +00:00
|
|
|
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
|
|
|
|
2020-08-26 11:03:14 +00:00
|
|
|
#[test]
|
|
|
|
fn inline_field_shorthand() {
|
2021-03-08 20:19:44 +00:00
|
|
|
cov_mark::check!(inline_field_shorthand);
|
2020-08-26 11:03:14 +00:00
|
|
|
check_assist(
|
|
|
|
inline_local_variable,
|
|
|
|
r"
|
|
|
|
struct S { foo: i32}
|
|
|
|
fn main() {
|
2021-01-06 20:15:48 +00:00
|
|
|
let $0foo = 92;
|
2020-08-26 11:03:14 +00:00
|
|
|
S { foo }
|
|
|
|
}
|
|
|
|
",
|
|
|
|
r"
|
|
|
|
struct S { foo: i32}
|
|
|
|
fn main() {
|
|
|
|
S { foo: 92 }
|
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-01-25 20:07:21 +00:00
|
|
|
#[test]
|
|
|
|
fn test_not_applicable_if_variable_unused() {
|
2021-03-08 20:19:44 +00:00
|
|
|
cov_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() {
|
2021-01-06 20:15:48 +00:00
|
|
|
let $0a = 0;
|
2020-01-25 20:07:21 +00:00
|
|
|
}
|
|
|
|
",
|
|
|
|
)
|
|
|
|
}
|
2020-03-03 15:56:42 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn not_applicable_outside_of_bind_pat() {
|
2021-03-08 20:19:44 +00:00
|
|
|
cov_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() {
|
2021-01-06 20:15:48 +00:00
|
|
|
let x = $01 + 2;
|
2020-03-03 15:56:42 +00:00
|
|
|
x * 4;
|
|
|
|
}
|
|
|
|
",
|
|
|
|
)
|
|
|
|
}
|
2021-04-20 14:16:23 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn works_on_local_usage() {
|
|
|
|
check_assist(
|
|
|
|
inline_local_variable,
|
|
|
|
r#"
|
|
|
|
fn f() {
|
|
|
|
let xyz = 0;
|
|
|
|
xyz$0;
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
r#"
|
|
|
|
fn f() {
|
|
|
|
0;
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn does_not_remove_let_when_multiple_usages() {
|
|
|
|
check_assist(
|
|
|
|
inline_local_variable,
|
|
|
|
r#"
|
|
|
|
fn f() {
|
|
|
|
let xyz = 0;
|
|
|
|
xyz$0;
|
|
|
|
xyz;
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
r#"
|
|
|
|
fn f() {
|
|
|
|
let xyz = 0;
|
|
|
|
0;
|
|
|
|
xyz;
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn not_applicable_with_non_ident_pattern() {
|
|
|
|
check_assist_not_applicable(
|
|
|
|
inline_local_variable,
|
|
|
|
r#"
|
|
|
|
fn main() {
|
|
|
|
let (x, y) = (0, 1);
|
|
|
|
x$0;
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn not_applicable_on_local_usage_in_macro() {
|
|
|
|
check_assist_not_applicable(
|
|
|
|
inline_local_variable,
|
|
|
|
r#"
|
|
|
|
macro_rules! m {
|
|
|
|
($i:ident) => { $i }
|
|
|
|
}
|
|
|
|
fn f() {
|
|
|
|
let xyz = 0;
|
|
|
|
m!(xyz$0); // replacing it would break the macro
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
check_assist_not_applicable(
|
|
|
|
inline_local_variable,
|
|
|
|
r#"
|
|
|
|
macro_rules! m {
|
|
|
|
($i:ident) => { $i }
|
|
|
|
}
|
|
|
|
fn f() {
|
|
|
|
let xyz$0 = 0;
|
|
|
|
m!(xyz); // replacing it would break the macro
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
}
|
2019-03-24 17:05:11 +00:00
|
|
|
}
|