Remove imports when inlining all calls in a file

This commit is contained in:
Lukas Wirth 2021-09-26 16:01:54 +02:00
parent 79c70d0ad3
commit 215a077ee4
2 changed files with 59 additions and 22 deletions

View file

@ -1,13 +1,15 @@
use ast::make; use ast::make;
use either::Either;
use hir::{db::HirDatabase, HasSource, PathResolution, Semantics, TypeInfo}; use hir::{db::HirDatabase, HasSource, PathResolution, Semantics, TypeInfo};
use ide_db::{ use ide_db::{
base_db::{FileId, FileRange}, base_db::{FileId, FileRange},
defs::Definition, defs::Definition,
helpers::insert_use::remove_path_if_in_use_stmt,
path_transform::PathTransform, path_transform::PathTransform,
search::{FileReference, SearchScope}, search::{FileReference, SearchScope},
RootDatabase, RootDatabase,
}; };
use itertools::izip; use itertools::{izip, Itertools};
use syntax::{ use syntax::{
ast::{self, edit_in_place::Indent, ArgListOwner}, ast::{self, edit_in_place::Indent, ArgListOwner},
ted, AstNode, SyntaxNode, ted, AstNode, SyntaxNode,
@ -96,26 +98,45 @@ pub(crate) fn inline_into_callers(acc: &mut Assists, ctx: &AssistContext) -> Opt
let mut inline_refs_for_file = |file_id, refs: Vec<FileReference>| { let mut inline_refs_for_file = |file_id, refs: Vec<FileReference>| {
builder.edit_file(file_id); builder.edit_file(file_id);
let count = refs.len(); let count = refs.len();
let name_refs = refs.into_iter().filter_map(|file_ref| match file_ref.name { // The collects are required as we are otherwise iterating while mutating 🙅‍♀️🙅‍♂️
ast::NameLike::NameRef(name_ref) => Some(name_ref), let (name_refs, name_refs_use): (Vec<_>, Vec<_>) = refs
_ => None, .into_iter()
}); .filter_map(|file_ref| match file_ref.name {
let call_infos = name_refs.filter_map(CallInfo::from_name_ref); ast::NameLike::NameRef(name_ref) => Some(name_ref),
let replaced = call_infos _ => None,
})
.partition_map(|name_ref| {
match name_ref.syntax().ancestors().find_map(ast::UseTree::cast) {
Some(use_tree) => Either::Right(builder.make_mut(use_tree)),
None => Either::Left(name_ref),
}
});
let call_infos: Vec<_> = name_refs
.into_iter()
.filter_map(CallInfo::from_name_ref)
.map(|call_info| { .map(|call_info| {
let mut_node = builder.make_syntax_mut(call_info.node.syntax().clone());
(call_info, mut_node)
})
.collect();
let replaced = call_infos
.into_iter()
.map(|(call_info, mut_node)| {
let replacement = let replacement =
inline(&ctx.sema, def_file, function, &func_body, &params, &call_info); inline(&ctx.sema, def_file, function, &func_body, &params, &call_info);
ted::replace(mut_node, replacement.syntax());
builder.replace_ast(
match call_info.node {
CallExprNode::Call(it) => ast::Expr::CallExpr(it),
CallExprNode::MethodCallExpr(it) => ast::Expr::MethodCallExpr(it),
},
replacement,
);
}) })
.count(); .count();
remove_def &= replaced == count; if replaced + name_refs_use.len() == count {
// we replaced all usages in this file, so we can remove the imports
name_refs_use.into_iter().for_each(|use_tree| {
if let Some(path) = use_tree.path() {
remove_path_if_in_use_stmt(&path);
}
})
} else {
remove_def = false;
}
}; };
for (file_id, refs) in usages.into_iter() { for (file_id, refs) in usages.into_iter() {
inline_refs_for_file(file_id, refs); inline_refs_for_file(file_id, refs);
@ -915,7 +936,10 @@ fn foo() {
} }
"#, "#,
r#" r#"
use super::do_the_math; //- /lib.rs
mod foo;
//- /foo.rs
fn foo() { fn foo() {
{ {
let foo = 10; let foo = 10;
@ -954,10 +978,7 @@ fn foo() {
r#" r#"
//- /lib.rs //- /lib.rs
mod foo; mod foo;
fn do_the_math(b: u32) -> u32 {
let foo = 10;
foo * b + foo
}
fn bar(a: u32, b: u32) -> u32 { fn bar(a: u32, b: u32) -> u32 {
{ {
let foo = 10; let foo = 10;
@ -965,7 +986,6 @@ fn bar(a: u32, b: u32) -> u32 {
}; };
} }
//- /foo.rs //- /foo.rs
use super::do_the_math;
fn foo() { fn foo() {
{ {
let foo = 10; let foo = 10;

View file

@ -220,6 +220,23 @@ pub fn insert_use(scope: &ImportScope, path: ast::Path, cfg: &InsertUseConfig) {
insert_use_(scope, &path, cfg.group, use_item); insert_use_(scope, &path, cfg.group, use_item);
} }
pub fn remove_path_if_in_use_stmt(path: &ast::Path) {
// FIXME: improve this
if path.parent_path().is_some() {
return;
}
if let Some(use_tree) = path.syntax().parent().and_then(ast::UseTree::cast) {
if use_tree.use_tree_list().is_some() || use_tree.star_token().is_some() {
return;
}
if let Some(use_) = use_tree.syntax().parent().and_then(ast::Use::cast) {
use_.remove();
return;
}
use_tree.remove();
}
}
#[derive(Eq, PartialEq, PartialOrd, Ord)] #[derive(Eq, PartialEq, PartialOrd, Ord)]
enum ImportGroup { enum ImportGroup {
// the order here defines the order of new group inserts // the order here defines the order of new group inserts