rust-analyzer/crates/assists/src/handlers
bors[bot] ac5958485e
Merge #7535
7535: Extract function assist r=cpud36 a=cpud36

This PR adds `extract function/method` assist. closes #5409.

# Supported features
Assist should support extracting from expressions(`1`, `2 + 2`, `loop { }`) and from a series of statements, e.g.:
```rust
foo();
$0bar();
baz();$0
quix();
```
Assist also supports extracting parameters, like:
```rust
fn foo() -> i32 {
  let n = 1;
  $0n + 1$0
}

// -
fn foo() -> i32 {
  let n = 1;
  fun_name(n)
}

fn fun_name(n: i32) -> i32 {
  n + 1
}
```

Extracting methods also generally works.

Assist allows referencing outer variables, both mutably and immutably, and handles handles access to variables local to extracted function:
```rust
fn foo() {
  let mut n = 1;
  let mut m = 2;
  let mut moved_v = Vec::new();
  let mut ref_mut_v = Vec::new();
  $0
  n += 1;
  let k = 1;
  moved_v.push(n);
  let r = &mut m;
  ref_mut_v.push(*r);
  let h = 3;
  $0
  n = ref_mut_v.len() + k;
  n -= h + m;
}

// -
fn foo() {
  let mut n = 1;
  let mut m = 2;
  let mut moved_v = Vec::new();
  let mut ref_mut_v = Vec::new();
 
  let (k, h) =  fun_name(&mut n, moved_v, &mut m, &mut ref_mut_v);

  n = ref_mut_v.len() + k;
  n -= h + m;
}

fn fun_name(n: &mut i32, mut moved_v: Vec<i32>, m: &mut i32, ref_mut_v: &mut Vec<i32>) -> (i32, i32) {
  *n += 1;
  let k = 1;
  moved_v.push(*n);
  let r = m;
  ref_mut_v.push(*r);
  let h = 3;
  (k, h)
}
```

So we handle both input and output paramters

# Showcase

![extract_cursor_in_range_3](https://user-images.githubusercontent.com/4218373/106980190-c9870800-6770-11eb-83d9-3d36b2550ff6.gif)
![fill_match_arms_discard_wildcard](https://user-images.githubusercontent.com/4218373/106980197-cbe96200-6770-11eb-96b0-14c27894fac0.gif)
![ide_db_helpers_handle_kind](https://user-images.githubusercontent.com/4218373/106980201-cdb32580-6770-11eb-9e6e-6ac8155d65ac.gif)
![ide_db_imports_location_local_query](https://user-images.githubusercontent.com/4218373/106980205-cf7ce900-6770-11eb-8516-653c8fcca807.gif)

# Working with non-`Copy` types

Consider the following example:
```rust
fn foo() {
  let v = Vec::new();
  $0
  let n = v.len();
  $0
  let is_empty = v.is_empty();
}
```
`v` must be a parameter to extracted function. 
The question is, what type should it have.
It could be `v: Vec<i32>`, or `v: &Vec<i32>`. 
The former is incorrect for `Vec<i32>`, but the later is silly for `i32`.

To resolve this we need to know if the type implements `Copy` trait.

I didn't find any api available from assists to query this. 
`hir_ty::method_resolution::implements` seems relevant, but is isn't publicly re-exported from `hir`.

# Star(`*`) token and pointer dereference

If I understand correctly, in order to create expression like `*p`, one should use `ast::make::expr_prefix(T![*], ...)`, which
in turn calls `token(T![*])`.

`token` does not have star in `tokens::SOURCE_FILE`, so this panics.
I had to add `*` to `SOURCE_FILE` to make it work.

Correct me if this is not intended way to do this.

# Lowering access `value -> mut ref -> shared ref`

Consider the following example:
```rust
fn foo() {
  let v = Vec::new();
  $0 let n = v.len(); $0
}
```
`v` is not used after extracted function body, so both `v: &Vec<i32>` and `v: Vec<i32>` would work.
Currently the later would be chosen.

We can however check the body of extracted function and conclude that `v: &Vec<i32>` is sufficient.
Using `v: &Vec<i32>`(that is a minimal required access level) might be a better default.
I am unsure.

# Cleanup
The assist seems to be reasonably handling most of common cases.
If there are no concerns with code it produces(i.e. with test cases), I will start cleaning up

[edit]
added showcase


Co-authored-by: Vladyslav Katasonov <cpud47@gmail.com>
2021-02-05 02:55:56 +00:00
..
add_explicit_type.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
add_lifetime_to_type.rs Add assist: add lifetime to type #7200 2021-01-21 09:53:24 +01:00
add_missing_impl_members.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
add_turbo_fish.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
apply_demorgan.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
auto_import.rs Add flyimport completion for trait assoc items 2021-01-16 20:44:12 +02:00
change_visibility.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
convert_integer_literal.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
early_return.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
expand_glob_import.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
extract_function.rs allow extracted body to be indented(dedent it) 2021-02-05 05:44:08 +03:00
extract_struct_from_enum_variant.rs Make ModPath's representation private 2021-02-04 20:49:24 +01:00
extract_variable.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
fill_match_arms.rs Enable fill_match_arms in macros 2021-01-27 22:32:40 +01:00
fix_visibility.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
flip_binexpr.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
flip_comma.rs Better fixture highlight 2021-01-07 19:11:27 +03:00
flip_trait_bound.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
generate_default_from_enum_variant.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
generate_derive.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
generate_from_impl_for_enum.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
generate_function.rs Treat BlockExpr as a potential module origin 2021-01-20 20:05:48 +01:00
generate_impl.rs . 2021-01-20 01:56:11 +03:00
generate_new.rs . 2021-01-20 01:56:11 +03:00
infer_function_return_type.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
inline_function.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
inline_local_variable.rs Add support for yiled keyword 2021-01-15 23:35:17 +09:00
introduce_named_lifetime.rs Wrap remaining self/super/crate in Name{Ref} 2021-01-15 22:18:43 +01:00
invert_if.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
merge_imports.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
merge_match_arms.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
move_bounds.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
move_guard.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
move_module_to_file.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
pull_assignment_up.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
qualify_path.rs Show deprecated completions for deprecated traits 2021-01-19 01:08:59 +02:00
raw_string.rs . 2021-01-20 01:56:11 +03:00
remove_dbg.rs Replace SyntaxKind usage with T! macro where applicable 2021-01-10 17:14:01 +01:00
remove_mut.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
remove_unused_param.rs Ensure uniqueness of file ids in reference search via hashmap 2021-01-12 15:51:02 +01:00
reorder_fields.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
reorder_impl.rs Sort impls by trait definition 2021-01-13 16:04:37 -07:00
replace_derive_with_manual_impl.rs . 2021-01-20 01:56:11 +03:00
replace_if_let_with_match.rs Deduplicate variant matching 2021-01-22 23:39:43 +01:00
replace_impl_trait_with_generic.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
replace_let_with_if_let.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
replace_qualified_name_with_use.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
replace_string_with_char.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
replace_unwrap_with_match.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
split_import.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
toggle_ignore.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
unmerge_use.rs Add test mark for skipping single use item 2021-01-15 22:57:10 +03:00
unwrap_block.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00
wrap_return_type_in_result.rs Change <|> to $0 - Rebase 2021-01-07 12:09:23 +00:00