2019-01-03 12:08:32 +00:00
|
|
|
use ra_syntax::{
|
2020-04-09 11:00:09 +00:00
|
|
|
ast::{self, AstNode},
|
2019-07-04 20:05:17 +00:00
|
|
|
SyntaxKind::{
|
2019-09-02 18:41:50 +00:00
|
|
|
BLOCK_EXPR, BREAK_EXPR, COMMENT, LAMBDA_EXPR, LOOP_EXPR, MATCH_ARM, PATH_EXPR, RETURN_EXPR,
|
|
|
|
WHITESPACE,
|
2019-07-04 20:05:17 +00:00
|
|
|
},
|
2020-05-20 21:07:17 +00:00
|
|
|
SyntaxNode,
|
2019-01-03 12:08:32 +00:00
|
|
|
};
|
2020-03-28 10:01:25 +00:00
|
|
|
use stdx::format_to;
|
2020-05-20 10:59:20 +00:00
|
|
|
use test_utils::mark;
|
2019-01-03 12:08:32 +00:00
|
|
|
|
2020-05-06 16:45:35 +00:00
|
|
|
use crate::{AssistContext, AssistId, Assists};
|
2019-01-03 12:08:32 +00:00
|
|
|
|
2020-06-26 23:21:43 +00:00
|
|
|
// Assist: extract_variable
|
2019-10-26 16:58:18 +00:00
|
|
|
//
|
|
|
|
// Extracts subexpression into a variable.
|
|
|
|
//
|
|
|
|
// ```
|
|
|
|
// fn main() {
|
|
|
|
// <|>(1 + 2)<|> * 4;
|
|
|
|
// }
|
|
|
|
// ```
|
|
|
|
// ->
|
|
|
|
// ```
|
|
|
|
// fn main() {
|
2020-05-20 21:07:17 +00:00
|
|
|
// let $0var_name = (1 + 2);
|
2019-10-26 16:58:18 +00:00
|
|
|
// var_name * 4;
|
|
|
|
// }
|
|
|
|
// ```
|
2020-06-26 23:21:43 +00:00
|
|
|
pub(crate) fn extract_variable(acc: &mut Assists, ctx: &AssistContext) -> Option<()> {
|
2019-02-24 11:18:10 +00:00
|
|
|
if ctx.frange.range.is_empty() {
|
|
|
|
return None;
|
|
|
|
}
|
2019-03-30 10:25:53 +00:00
|
|
|
let node = ctx.covering_element();
|
2019-02-24 11:22:25 +00:00
|
|
|
if node.kind() == COMMENT {
|
2020-06-26 23:21:43 +00:00
|
|
|
mark::hit!(extract_var_in_comment_is_not_applicable);
|
2019-01-30 20:36:49 +00:00
|
|
|
return None;
|
|
|
|
}
|
2019-02-24 11:22:25 +00:00
|
|
|
let expr = node.ancestors().find_map(valid_target_expr)?;
|
2019-07-19 08:24:41 +00:00
|
|
|
let (anchor_stmt, wrap_in_block) = anchor_stmt(expr.clone())?;
|
|
|
|
let indent = anchor_stmt.prev_sibling_or_token()?.as_token()?.clone();
|
2019-01-03 12:08:32 +00:00
|
|
|
if indent.kind() != WHITESPACE {
|
|
|
|
return None;
|
|
|
|
}
|
2020-05-06 10:51:28 +00:00
|
|
|
let target = expr.syntax().text_range();
|
2020-06-26 23:21:43 +00:00
|
|
|
acc.add(AssistId("extract_variable"), "Extract into variable", target, move |edit| {
|
2020-06-25 22:51:34 +00:00
|
|
|
let field_shorthand = match expr.syntax().parent().and_then(ast::RecordField::cast) {
|
|
|
|
Some(field) => field.name_ref(),
|
|
|
|
None => None,
|
|
|
|
};
|
|
|
|
|
2019-01-03 12:08:32 +00:00
|
|
|
let mut buf = String::new();
|
|
|
|
|
2020-06-25 22:51:34 +00:00
|
|
|
let var_name = match &field_shorthand {
|
|
|
|
Some(it) => it.to_string(),
|
|
|
|
None => "var_name".to_string(),
|
|
|
|
};
|
|
|
|
let expr_range = match &field_shorthand {
|
|
|
|
Some(it) => it.syntax().text_range().cover(expr.syntax().text_range()),
|
|
|
|
None => expr.syntax().text_range(),
|
|
|
|
};
|
|
|
|
|
2020-05-20 21:07:17 +00:00
|
|
|
if wrap_in_block {
|
2020-06-25 22:51:34 +00:00
|
|
|
format_to!(buf, "{{ let {} = ", var_name);
|
2019-01-28 14:12:07 +00:00
|
|
|
} else {
|
2020-06-25 22:51:34 +00:00
|
|
|
format_to!(buf, "let {} = ", var_name);
|
2019-01-28 14:12:07 +00:00
|
|
|
};
|
2020-03-28 10:01:25 +00:00
|
|
|
format_to!(buf, "{}", expr.syntax());
|
2020-05-20 21:07:17 +00:00
|
|
|
|
2019-07-19 08:24:41 +00:00
|
|
|
let full_stmt = ast::ExprStmt::cast(anchor_stmt.clone());
|
|
|
|
let is_full_stmt = if let Some(expr_stmt) = &full_stmt {
|
|
|
|
Some(expr.syntax().clone()) == expr_stmt.expr().map(|e| e.syntax().clone())
|
2019-01-03 12:08:32 +00:00
|
|
|
} else {
|
|
|
|
false
|
|
|
|
};
|
|
|
|
if is_full_stmt {
|
2020-06-26 23:21:43 +00:00
|
|
|
mark::hit!(test_extract_var_expr_stmt);
|
2020-04-10 08:11:05 +00:00
|
|
|
if full_stmt.unwrap().semicolon_token().is_none() {
|
2019-01-16 04:27:15 +00:00
|
|
|
buf.push_str(";");
|
2019-01-14 02:22:53 +00:00
|
|
|
}
|
2020-05-20 21:07:17 +00:00
|
|
|
match ctx.config.snippet_cap {
|
|
|
|
Some(cap) => {
|
2020-06-25 22:51:34 +00:00
|
|
|
let snip =
|
|
|
|
buf.replace(&format!("let {}", var_name), &format!("let $0{}", var_name));
|
|
|
|
edit.replace_snippet(cap, expr_range, snip)
|
2020-05-20 21:07:17 +00:00
|
|
|
}
|
2020-06-25 22:51:34 +00:00
|
|
|
None => edit.replace(expr_range, buf),
|
2019-02-09 11:41:03 +00:00
|
|
|
}
|
2020-05-20 21:07:17 +00:00
|
|
|
return;
|
|
|
|
}
|
2019-02-09 11:41:03 +00:00
|
|
|
|
2020-05-20 21:07:17 +00:00
|
|
|
buf.push_str(";");
|
|
|
|
|
|
|
|
// We want to maintain the indent level,
|
|
|
|
// but we do not want to duplicate possible
|
|
|
|
// extra newlines in the indent block
|
|
|
|
let text = indent.text();
|
|
|
|
if text.starts_with('\n') {
|
|
|
|
buf.push_str("\n");
|
|
|
|
buf.push_str(text.trim_start_matches('\n'));
|
|
|
|
} else {
|
|
|
|
buf.push_str(text);
|
|
|
|
}
|
|
|
|
|
2020-06-25 22:51:34 +00:00
|
|
|
edit.replace(expr_range, var_name.clone());
|
2020-05-20 21:07:17 +00:00
|
|
|
let offset = anchor_stmt.text_range().start();
|
|
|
|
match ctx.config.snippet_cap {
|
|
|
|
Some(cap) => {
|
2020-06-25 22:51:34 +00:00
|
|
|
let snip =
|
|
|
|
buf.replace(&format!("let {}", var_name), &format!("let $0{}", var_name));
|
2020-05-20 21:07:17 +00:00
|
|
|
edit.insert_snippet(cap, offset, snip)
|
2019-01-28 14:12:07 +00:00
|
|
|
}
|
2020-05-20 21:07:17 +00:00
|
|
|
None => edit.insert(offset, buf),
|
|
|
|
}
|
|
|
|
|
|
|
|
if wrap_in_block {
|
|
|
|
edit.insert(anchor_stmt.text_range().end(), " }");
|
2019-01-03 12:08:32 +00:00
|
|
|
}
|
2019-10-27 14:35:37 +00:00
|
|
|
})
|
2019-01-03 15:59:17 +00:00
|
|
|
}
|
2019-01-03 12:08:32 +00:00
|
|
|
|
2019-02-08 23:34:05 +00:00
|
|
|
/// Check whether the node is a valid expression which can be extracted to a variable.
|
2019-01-30 20:36:49 +00:00
|
|
|
/// In general that's true for any expression, but in some cases that would produce invalid code.
|
2019-07-19 08:24:41 +00:00
|
|
|
fn valid_target_expr(node: SyntaxNode) -> Option<ast::Expr> {
|
2019-02-06 20:50:26 +00:00
|
|
|
match node.kind() {
|
2019-09-02 18:41:50 +00:00
|
|
|
PATH_EXPR | LOOP_EXPR => None,
|
2019-01-30 20:36:49 +00:00
|
|
|
BREAK_EXPR => ast::BreakExpr::cast(node).and_then(|e| e.expr()),
|
|
|
|
RETURN_EXPR => ast::ReturnExpr::cast(node).and_then(|e| e.expr()),
|
2019-09-02 18:41:50 +00:00
|
|
|
BLOCK_EXPR => {
|
|
|
|
ast::BlockExpr::cast(node).filter(|it| it.is_standalone()).map(ast::Expr::from)
|
|
|
|
}
|
2019-01-30 20:36:49 +00:00
|
|
|
_ => ast::Expr::cast(node),
|
2019-02-06 20:50:26 +00:00
|
|
|
}
|
2019-01-30 20:36:49 +00:00
|
|
|
}
|
|
|
|
|
2020-06-26 23:21:43 +00:00
|
|
|
/// Returns the syntax node which will follow the freshly extractd var
|
2019-01-28 14:12:07 +00:00
|
|
|
/// and a boolean indicating whether we have to wrap it within a { } block
|
|
|
|
/// to produce correct code.
|
|
|
|
/// It can be a statement, the last in a block expression or a wanna be block
|
2019-02-08 23:34:05 +00:00
|
|
|
/// expression like a lambda or match arm.
|
2019-07-19 08:24:41 +00:00
|
|
|
fn anchor_stmt(expr: ast::Expr) -> Option<(SyntaxNode, bool)> {
|
2019-01-28 14:12:07 +00:00
|
|
|
expr.syntax().ancestors().find_map(|node| {
|
2020-05-01 23:18:19 +00:00
|
|
|
if let Some(expr) = node.parent().and_then(ast::BlockExpr::cast).and_then(|it| it.expr()) {
|
2019-07-19 08:24:41 +00:00
|
|
|
if expr.syntax() == &node {
|
2020-06-26 23:21:43 +00:00
|
|
|
mark::hit!(test_extract_var_last_expr);
|
2019-01-28 14:12:07 +00:00
|
|
|
return Some((node, false));
|
2019-01-03 12:08:32 +00:00
|
|
|
}
|
2019-01-03 15:59:17 +00:00
|
|
|
}
|
2019-01-28 14:12:07 +00:00
|
|
|
|
|
|
|
if let Some(parent) = node.parent() {
|
2019-01-30 20:36:49 +00:00
|
|
|
if parent.kind() == MATCH_ARM || parent.kind() == LAMBDA_EXPR {
|
2019-01-28 14:12:07 +00:00
|
|
|
return Some((node, true));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-19 19:15:49 +00:00
|
|
|
if ast::Stmt::cast(node.clone()).is_some() {
|
2019-07-19 08:24:41 +00:00
|
|
|
return Some((node, false));
|
|
|
|
}
|
|
|
|
|
2019-01-28 14:12:07 +00:00
|
|
|
None
|
2019-01-03 15:59:17 +00:00
|
|
|
})
|
2019-01-03 12:08:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2020-05-20 10:59:20 +00:00
|
|
|
use test_utils::mark;
|
2019-02-24 11:22:25 +00:00
|
|
|
|
2020-05-06 08:16:55 +00:00
|
|
|
use crate::tests::{check_assist, check_assist_not_applicable, check_assist_target};
|
2019-01-03 12:08:32 +00:00
|
|
|
|
2019-02-24 11:22:25 +00:00
|
|
|
use super::*;
|
|
|
|
|
2019-01-03 12:08:32 +00:00
|
|
|
#[test]
|
2020-06-26 23:21:43 +00:00
|
|
|
fn test_extract_var_simple() {
|
2020-02-25 17:57:47 +00:00
|
|
|
check_assist(
|
2020-06-26 23:21:43 +00:00
|
|
|
extract_variable,
|
2020-05-20 21:07:17 +00:00
|
|
|
r#"
|
2019-01-03 12:08:32 +00:00
|
|
|
fn foo() {
|
|
|
|
foo(<|>1 + 1<|>);
|
2020-05-20 21:07:17 +00:00
|
|
|
}"#,
|
|
|
|
r#"
|
2019-01-03 12:08:32 +00:00
|
|
|
fn foo() {
|
2020-05-20 21:07:17 +00:00
|
|
|
let $0var_name = 1 + 1;
|
2019-01-03 12:08:32 +00:00
|
|
|
foo(var_name);
|
2020-05-20 21:07:17 +00:00
|
|
|
}"#,
|
2019-01-03 12:08:32 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-02-24 11:22:25 +00:00
|
|
|
#[test]
|
2020-06-26 23:21:43 +00:00
|
|
|
fn extract_var_in_comment_is_not_applicable() {
|
|
|
|
mark::check!(extract_var_in_comment_is_not_applicable);
|
|
|
|
check_assist_not_applicable(extract_variable, "fn main() { 1 + /* <|>comment<|> */ 1; }");
|
2019-02-24 11:22:25 +00:00
|
|
|
}
|
|
|
|
|
2019-01-03 12:08:32 +00:00
|
|
|
#[test]
|
2020-06-26 23:21:43 +00:00
|
|
|
fn test_extract_var_expr_stmt() {
|
|
|
|
mark::check!(test_extract_var_expr_stmt);
|
2020-02-25 17:57:47 +00:00
|
|
|
check_assist(
|
2020-06-26 23:21:43 +00:00
|
|
|
extract_variable,
|
2020-05-20 21:07:17 +00:00
|
|
|
r#"
|
2019-01-03 12:08:32 +00:00
|
|
|
fn foo() {
|
|
|
|
<|>1 + 1<|>;
|
2020-05-20 21:07:17 +00:00
|
|
|
}"#,
|
|
|
|
r#"
|
2019-01-03 12:08:32 +00:00
|
|
|
fn foo() {
|
2020-05-20 21:07:17 +00:00
|
|
|
let $0var_name = 1 + 1;
|
|
|
|
}"#,
|
2019-02-24 11:22:25 +00:00
|
|
|
);
|
2020-02-25 17:57:47 +00:00
|
|
|
check_assist(
|
2020-06-26 23:21:43 +00:00
|
|
|
extract_variable,
|
2019-02-24 11:22:25 +00:00
|
|
|
"
|
|
|
|
fn foo() {
|
|
|
|
<|>{ let x = 0; x }<|>
|
|
|
|
something_else();
|
|
|
|
}",
|
|
|
|
"
|
|
|
|
fn foo() {
|
2020-05-20 21:07:17 +00:00
|
|
|
let $0var_name = { let x = 0; x };
|
2019-02-24 11:22:25 +00:00
|
|
|
something_else();
|
2019-01-03 12:08:32 +00:00
|
|
|
}",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-06-26 23:21:43 +00:00
|
|
|
fn test_extract_var_part_of_expr_stmt() {
|
2020-02-25 17:57:47 +00:00
|
|
|
check_assist(
|
2020-06-26 23:21:43 +00:00
|
|
|
extract_variable,
|
2019-01-03 12:08:32 +00:00
|
|
|
"
|
|
|
|
fn foo() {
|
|
|
|
<|>1<|> + 1;
|
|
|
|
}",
|
|
|
|
"
|
|
|
|
fn foo() {
|
2020-05-20 21:07:17 +00:00
|
|
|
let $0var_name = 1;
|
2019-01-03 12:08:32 +00:00
|
|
|
var_name + 1;
|
|
|
|
}",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-06-26 23:21:43 +00:00
|
|
|
fn test_extract_var_last_expr() {
|
|
|
|
mark::check!(test_extract_var_last_expr);
|
2020-02-25 17:57:47 +00:00
|
|
|
check_assist(
|
2020-06-26 23:21:43 +00:00
|
|
|
extract_variable,
|
2020-06-23 22:30:34 +00:00
|
|
|
r#"
|
2019-01-03 12:08:32 +00:00
|
|
|
fn foo() {
|
|
|
|
bar(<|>1 + 1<|>)
|
2020-06-23 22:30:34 +00:00
|
|
|
}
|
|
|
|
"#,
|
|
|
|
r#"
|
2019-01-03 12:08:32 +00:00
|
|
|
fn foo() {
|
2020-05-20 21:07:17 +00:00
|
|
|
let $0var_name = 1 + 1;
|
2019-01-03 12:08:32 +00:00
|
|
|
bar(var_name)
|
2020-06-23 22:30:34 +00:00
|
|
|
}
|
|
|
|
"#,
|
2019-01-03 12:08:32 +00:00
|
|
|
);
|
2020-02-25 17:57:47 +00:00
|
|
|
check_assist(
|
2020-06-26 23:21:43 +00:00
|
|
|
extract_variable,
|
2020-06-23 22:30:34 +00:00
|
|
|
r#"
|
2019-01-03 12:08:32 +00:00
|
|
|
fn foo() {
|
|
|
|
<|>bar(1 + 1)<|>
|
2020-06-23 22:30:34 +00:00
|
|
|
}
|
|
|
|
"#,
|
|
|
|
r#"
|
2019-01-03 12:08:32 +00:00
|
|
|
fn foo() {
|
2020-05-20 21:07:17 +00:00
|
|
|
let $0var_name = bar(1 + 1);
|
2019-01-03 12:08:32 +00:00
|
|
|
var_name
|
2020-06-23 22:30:34 +00:00
|
|
|
}
|
|
|
|
"#,
|
2019-02-24 11:22:25 +00:00
|
|
|
)
|
2019-01-14 01:59:56 +00:00
|
|
|
}
|
2019-01-28 14:12:07 +00:00
|
|
|
|
|
|
|
#[test]
|
2020-06-26 23:21:43 +00:00
|
|
|
fn test_extract_var_in_match_arm_no_block() {
|
2020-02-25 17:57:47 +00:00
|
|
|
check_assist(
|
2020-06-26 23:21:43 +00:00
|
|
|
extract_variable,
|
2019-01-28 14:12:07 +00:00
|
|
|
"
|
|
|
|
fn main() {
|
|
|
|
let x = true;
|
|
|
|
let tuple = match x {
|
|
|
|
true => (<|>2 + 2<|>, true)
|
|
|
|
_ => (0, false)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
",
|
|
|
|
"
|
|
|
|
fn main() {
|
|
|
|
let x = true;
|
|
|
|
let tuple = match x {
|
2020-05-20 21:07:17 +00:00
|
|
|
true => { let $0var_name = 2 + 2; (var_name, true) }
|
2019-01-28 14:12:07 +00:00
|
|
|
_ => (0, false)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-06-26 23:21:43 +00:00
|
|
|
fn test_extract_var_in_match_arm_with_block() {
|
2020-02-25 17:57:47 +00:00
|
|
|
check_assist(
|
2020-06-26 23:21:43 +00:00
|
|
|
extract_variable,
|
2019-01-28 14:12:07 +00:00
|
|
|
"
|
|
|
|
fn main() {
|
|
|
|
let x = true;
|
|
|
|
let tuple = match x {
|
|
|
|
true => {
|
|
|
|
let y = 1;
|
|
|
|
(<|>2 + y<|>, true)
|
|
|
|
}
|
|
|
|
_ => (0, false)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
",
|
|
|
|
"
|
|
|
|
fn main() {
|
|
|
|
let x = true;
|
|
|
|
let tuple = match x {
|
|
|
|
true => {
|
|
|
|
let y = 1;
|
2020-05-20 21:07:17 +00:00
|
|
|
let $0var_name = 2 + y;
|
2019-01-28 14:12:07 +00:00
|
|
|
(var_name, true)
|
|
|
|
}
|
|
|
|
_ => (0, false)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-06-26 23:21:43 +00:00
|
|
|
fn test_extract_var_in_closure_no_block() {
|
2020-02-25 17:57:47 +00:00
|
|
|
check_assist(
|
2020-06-26 23:21:43 +00:00
|
|
|
extract_variable,
|
2019-01-28 14:12:07 +00:00
|
|
|
"
|
|
|
|
fn main() {
|
|
|
|
let lambda = |x: u32| <|>x * 2<|>;
|
|
|
|
}
|
|
|
|
",
|
|
|
|
"
|
|
|
|
fn main() {
|
2020-05-20 21:07:17 +00:00
|
|
|
let lambda = |x: u32| { let $0var_name = x * 2; var_name };
|
2019-01-28 14:12:07 +00:00
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-06-26 23:21:43 +00:00
|
|
|
fn test_extract_var_in_closure_with_block() {
|
2020-02-25 17:57:47 +00:00
|
|
|
check_assist(
|
2020-06-26 23:21:43 +00:00
|
|
|
extract_variable,
|
2019-01-28 14:12:07 +00:00
|
|
|
"
|
|
|
|
fn main() {
|
|
|
|
let lambda = |x: u32| { <|>x * 2<|> };
|
|
|
|
}
|
|
|
|
",
|
|
|
|
"
|
|
|
|
fn main() {
|
2020-05-20 21:07:17 +00:00
|
|
|
let lambda = |x: u32| { let $0var_name = x * 2; var_name };
|
2019-01-28 14:12:07 +00:00
|
|
|
}
|
2019-01-30 20:36:49 +00:00
|
|
|
",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-06-26 23:21:43 +00:00
|
|
|
fn test_extract_var_path_simple() {
|
2020-02-25 17:57:47 +00:00
|
|
|
check_assist(
|
2020-06-26 23:21:43 +00:00
|
|
|
extract_variable,
|
2019-01-30 20:36:49 +00:00
|
|
|
"
|
|
|
|
fn main() {
|
2019-02-24 11:18:10 +00:00
|
|
|
let o = <|>Some(true)<|>;
|
2019-01-30 20:36:49 +00:00
|
|
|
}
|
|
|
|
",
|
|
|
|
"
|
|
|
|
fn main() {
|
2020-05-20 21:07:17 +00:00
|
|
|
let $0var_name = Some(true);
|
2019-01-30 20:36:49 +00:00
|
|
|
let o = var_name;
|
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-06-26 23:21:43 +00:00
|
|
|
fn test_extract_var_path_method() {
|
2020-02-25 17:57:47 +00:00
|
|
|
check_assist(
|
2020-06-26 23:21:43 +00:00
|
|
|
extract_variable,
|
2019-01-30 20:36:49 +00:00
|
|
|
"
|
|
|
|
fn main() {
|
2019-02-24 11:18:10 +00:00
|
|
|
let v = <|>bar.foo()<|>;
|
2019-01-30 20:36:49 +00:00
|
|
|
}
|
|
|
|
",
|
|
|
|
"
|
|
|
|
fn main() {
|
2020-05-20 21:07:17 +00:00
|
|
|
let $0var_name = bar.foo();
|
2019-01-30 20:36:49 +00:00
|
|
|
let v = var_name;
|
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-06-26 23:21:43 +00:00
|
|
|
fn test_extract_var_return() {
|
2020-02-25 17:57:47 +00:00
|
|
|
check_assist(
|
2020-06-26 23:21:43 +00:00
|
|
|
extract_variable,
|
2019-01-30 20:36:49 +00:00
|
|
|
"
|
|
|
|
fn foo() -> u32 {
|
2019-02-24 11:18:10 +00:00
|
|
|
<|>return 2 + 2<|>;
|
2019-01-30 20:36:49 +00:00
|
|
|
}
|
|
|
|
",
|
|
|
|
"
|
|
|
|
fn foo() -> u32 {
|
2020-05-20 21:07:17 +00:00
|
|
|
let $0var_name = 2 + 2;
|
2019-02-09 11:41:03 +00:00
|
|
|
return var_name;
|
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-06-26 23:21:43 +00:00
|
|
|
fn test_extract_var_does_not_add_extra_whitespace() {
|
2020-02-25 17:57:47 +00:00
|
|
|
check_assist(
|
2020-06-26 23:21:43 +00:00
|
|
|
extract_variable,
|
2019-02-09 11:41:03 +00:00
|
|
|
"
|
|
|
|
fn foo() -> u32 {
|
|
|
|
|
|
|
|
|
2019-02-24 11:18:10 +00:00
|
|
|
<|>return 2 + 2<|>;
|
2019-02-09 11:41:03 +00:00
|
|
|
}
|
|
|
|
",
|
|
|
|
"
|
|
|
|
fn foo() -> u32 {
|
|
|
|
|
|
|
|
|
2020-05-20 21:07:17 +00:00
|
|
|
let $0var_name = 2 + 2;
|
2019-02-09 11:41:03 +00:00
|
|
|
return var_name;
|
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
|
|
|
|
2020-02-25 17:57:47 +00:00
|
|
|
check_assist(
|
2020-06-26 23:21:43 +00:00
|
|
|
extract_variable,
|
2019-02-09 11:41:03 +00:00
|
|
|
"
|
|
|
|
fn foo() -> u32 {
|
|
|
|
|
2019-02-24 11:18:10 +00:00
|
|
|
<|>return 2 + 2<|>;
|
2019-02-09 11:41:03 +00:00
|
|
|
}
|
|
|
|
",
|
|
|
|
"
|
|
|
|
fn foo() -> u32 {
|
|
|
|
|
2020-05-20 21:07:17 +00:00
|
|
|
let $0var_name = 2 + 2;
|
2019-02-09 11:41:03 +00:00
|
|
|
return var_name;
|
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
|
|
|
|
2020-02-25 17:57:47 +00:00
|
|
|
check_assist(
|
2020-06-26 23:21:43 +00:00
|
|
|
extract_variable,
|
2019-02-09 11:41:03 +00:00
|
|
|
"
|
|
|
|
fn foo() -> u32 {
|
|
|
|
let foo = 1;
|
|
|
|
|
|
|
|
// bar
|
|
|
|
|
|
|
|
|
2019-02-24 11:18:10 +00:00
|
|
|
<|>return 2 + 2<|>;
|
2019-02-09 11:41:03 +00:00
|
|
|
}
|
|
|
|
",
|
|
|
|
"
|
|
|
|
fn foo() -> u32 {
|
|
|
|
let foo = 1;
|
|
|
|
|
|
|
|
// bar
|
|
|
|
|
|
|
|
|
2020-05-20 21:07:17 +00:00
|
|
|
let $0var_name = 2 + 2;
|
2019-01-30 20:36:49 +00:00
|
|
|
return var_name;
|
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-06-26 23:21:43 +00:00
|
|
|
fn test_extract_var_break() {
|
2020-02-25 17:57:47 +00:00
|
|
|
check_assist(
|
2020-06-26 23:21:43 +00:00
|
|
|
extract_variable,
|
2019-01-30 20:36:49 +00:00
|
|
|
"
|
|
|
|
fn main() {
|
|
|
|
let result = loop {
|
2019-02-24 11:18:10 +00:00
|
|
|
<|>break 2 + 2<|>;
|
2019-01-30 20:36:49 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
",
|
|
|
|
"
|
|
|
|
fn main() {
|
|
|
|
let result = loop {
|
2020-05-20 21:07:17 +00:00
|
|
|
let $0var_name = 2 + 2;
|
2019-01-30 20:36:49 +00:00
|
|
|
break var_name;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-06-26 23:21:43 +00:00
|
|
|
fn test_extract_var_for_cast() {
|
2020-02-25 17:57:47 +00:00
|
|
|
check_assist(
|
2020-06-26 23:21:43 +00:00
|
|
|
extract_variable,
|
2019-01-30 20:36:49 +00:00
|
|
|
"
|
|
|
|
fn main() {
|
2019-02-24 11:18:10 +00:00
|
|
|
let v = <|>0f32 as u32<|>;
|
2019-01-30 20:36:49 +00:00
|
|
|
}
|
|
|
|
",
|
|
|
|
"
|
|
|
|
fn main() {
|
2020-05-20 21:07:17 +00:00
|
|
|
let $0var_name = 0f32 as u32;
|
2019-01-30 20:36:49 +00:00
|
|
|
let v = var_name;
|
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-06-25 22:51:34 +00:00
|
|
|
#[test]
|
2020-06-26 23:21:43 +00:00
|
|
|
fn extract_var_field_shorthand() {
|
2020-06-25 22:51:34 +00:00
|
|
|
check_assist(
|
2020-06-26 23:21:43 +00:00
|
|
|
extract_variable,
|
2020-06-25 22:51:34 +00:00
|
|
|
r#"
|
|
|
|
struct S {
|
|
|
|
foo: i32
|
|
|
|
}
|
|
|
|
|
|
|
|
fn main() {
|
|
|
|
S { foo: <|>1 + 1<|> }
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
r#"
|
|
|
|
struct S {
|
|
|
|
foo: i32
|
|
|
|
}
|
|
|
|
|
|
|
|
fn main() {
|
|
|
|
let $0foo = 1 + 1;
|
|
|
|
S { foo }
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2019-01-30 20:36:49 +00:00
|
|
|
#[test]
|
2020-06-26 23:21:43 +00:00
|
|
|
fn test_extract_var_for_return_not_applicable() {
|
|
|
|
check_assist_not_applicable(extract_variable, "fn foo() { <|>return<|>; } ");
|
2019-01-30 20:36:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-06-26 23:21:43 +00:00
|
|
|
fn test_extract_var_for_break_not_applicable() {
|
|
|
|
check_assist_not_applicable(extract_variable, "fn main() { loop { <|>break<|>; }; }");
|
2019-01-30 20:36:49 +00:00
|
|
|
}
|
|
|
|
|
2019-02-08 23:34:05 +00:00
|
|
|
// FIXME: This is not quite correct, but good enough(tm) for the sorting heuristic
|
|
|
|
#[test]
|
2020-06-26 23:21:43 +00:00
|
|
|
fn extract_var_target() {
|
|
|
|
check_assist_target(extract_variable, "fn foo() -> u32 { <|>return 2 + 2<|>; }", "2 + 2");
|
2019-02-08 23:34:05 +00:00
|
|
|
|
2020-02-25 17:57:47 +00:00
|
|
|
check_assist_target(
|
2020-06-26 23:21:43 +00:00
|
|
|
extract_variable,
|
2019-02-08 23:34:05 +00:00
|
|
|
"
|
|
|
|
fn main() {
|
|
|
|
let x = true;
|
|
|
|
let tuple = match x {
|
|
|
|
true => (<|>2 + 2<|>, true)
|
|
|
|
_ => (0, false)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
",
|
|
|
|
"2 + 2",
|
|
|
|
);
|
|
|
|
}
|
2019-01-03 12:08:32 +00:00
|
|
|
}
|