rust/crates/assists/src/handlers/replace_qualified_name_with_use.rs

679 lines
12 KiB
Rust
Raw Normal View History

2020-11-28 16:30:39 +02:00
use ide_db::helpers::insert_use::{insert_use, ImportScope};
2020-09-16 21:29:48 +02:00
use syntax::{algo::SyntaxRewriter, ast, match_ast, AstNode, SyntaxNode};
use test_utils::mark;
use crate::{AssistContext, AssistId, AssistKind, Assists};
2020-02-07 23:35:34 +02:00
// Assist: replace_qualified_name_with_use
2019-10-27 17:49:39 +03:00
//
2020-02-07 23:35:34 +02:00
// Adds a use statement for a given fully-qualified name.
2019-10-27 17:49:39 +03:00
//
// ```
// fn process(map: std::collections::<|>HashMap<String, String>) {}
// ```
// ->
// ```
// use std::collections::HashMap;
//
// fn process(map: HashMap<String, String>) {}
// ```
pub(crate) fn replace_qualified_name_with_use(
acc: &mut Assists,
ctx: &AssistContext,
) -> Option<()> {
let path: ast::Path = ctx.find_node_at_offset()?;
// We don't want to mess with use statements
2020-07-30 14:12:04 +02:00
if path.syntax().ancestors().find_map(ast::Use::cast).is_some() {
return None;
}
if path.qualifier().is_none() {
mark::hit!(dont_import_trivial_paths);
return None;
}
let target = path.syntax().text_range();
2020-09-18 23:40:11 +03:00
let scope = ImportScope::find_insert_use_container(path.syntax(), &ctx.sema)?;
let syntax = scope.as_syntax_node();
acc.add(
2020-07-02 17:48:35 -04:00
AssistId("replace_qualified_name_with_use", AssistKind::RefactorRewrite),
2020-02-07 23:35:34 +02:00
"Replace qualified path with use",
target,
|builder| {
// Now that we've brought the name into scope, re-qualify all paths that could be
// affected (that is, all paths inside the node we added the `use` to).
let mut rewriter = SyntaxRewriter::default();
2020-09-16 21:29:48 +02:00
shorten_paths(&mut rewriter, syntax.clone(), &path);
if let Some(ref import_scope) = ImportScope::from(syntax.clone()) {
rewriter += insert_use(import_scope, path, ctx.config.insert_use.merge);
builder.rewrite(rewriter);
}
2020-02-28 21:53:20 +01:00
},
)
}
2020-11-27 12:26:20 +02:00
/// Adds replacements to `re` that shorten `path` in all descendants of `node`.
2020-09-16 21:29:48 +02:00
fn shorten_paths(rewriter: &mut SyntaxRewriter<'static>, node: SyntaxNode, path: &ast::Path) {
for child in node.children() {
match_ast! {
match child {
// Don't modify `use` items, as this can break the `use` item when injecting a new
// import into the use tree.
2020-07-30 14:12:04 +02:00
ast::Use(_it) => continue,
// Don't descend into submodules, they don't have the same `use` items in scope.
ast::Module(_it) => continue,
ast::Path(p) => {
match maybe_replace_path(rewriter, p.clone(), path.clone()) {
Some(()) => {},
2020-09-16 21:29:48 +02:00
None => shorten_paths(rewriter, p.syntax().clone(), path),
}
},
2020-09-16 21:29:48 +02:00
_ => shorten_paths(rewriter, child, path),
}
}
}
}
fn maybe_replace_path(
rewriter: &mut SyntaxRewriter<'static>,
path: ast::Path,
target: ast::Path,
) -> Option<()> {
2020-07-19 14:26:24 -04:00
if !path_eq(path.clone(), target) {
return None;
}
// Shorten `path`, leaving only its last segment.
if let Some(parent) = path.qualifier() {
rewriter.delete(parent.syntax());
}
if let Some(double_colon) = path.coloncolon_token() {
rewriter.delete(&double_colon);
}
Some(())
}
fn path_eq(lhs: ast::Path, rhs: ast::Path) -> bool {
let mut lhs_curr = lhs;
let mut rhs_curr = rhs;
loop {
match (lhs_curr.segment(), rhs_curr.segment()) {
(Some(lhs), Some(rhs)) if lhs.syntax().text() == rhs.syntax().text() => (),
_ => return false,
}
match (lhs_curr.qualifier(), rhs_curr.qualifier()) {
(Some(lhs), Some(rhs)) => {
lhs_curr = lhs;
rhs_curr = rhs;
}
(None, None) => return true,
_ => return false,
}
}
}
2019-02-06 15:34:37 +01:00
#[cfg(test)]
mod tests {
2020-05-06 10:16:55 +02:00
use crate::tests::{check_assist, check_assist_not_applicable};
2019-02-06 15:34:37 +01:00
2019-10-27 17:26:51 +03:00
use super::*;
#[test]
fn test_replace_already_imported() {
check_assist(
replace_qualified_name_with_use,
r"use std::fs;
fn main() {
std::f<|>s::Path
}",
r"use std::fs;
fn main() {
fs::Path
}",
)
}
2019-02-06 15:34:37 +01:00
#[test]
2020-02-07 23:35:34 +02:00
fn test_replace_add_use_no_anchor() {
2019-02-06 15:34:37 +01:00
check_assist(
2020-02-07 23:35:34 +02:00
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
2019-02-06 15:34:37 +01:00
std::fmt::Debug<|>
",
2020-06-15 22:39:26 +02:00
r"
2019-02-06 15:34:37 +01:00
use std::fmt::Debug;
Debug
",
);
}
#[test]
2020-02-07 23:35:34 +02:00
fn test_replace_add_use_no_anchor_with_item_below() {
check_assist(
2020-02-07 23:35:34 +02:00
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
std::fmt::Debug<|>
fn main() {
}
",
2020-06-15 22:39:26 +02:00
r"
use std::fmt::Debug;
Debug
fn main() {
}
",
);
}
#[test]
2020-02-07 23:35:34 +02:00
fn test_replace_add_use_no_anchor_with_item_above() {
check_assist(
2020-02-07 23:35:34 +02:00
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
fn main() {
}
std::fmt::Debug<|>
",
2020-06-15 22:39:26 +02:00
r"
use std::fmt::Debug;
fn main() {
}
Debug
2019-02-06 15:34:37 +01:00
",
);
}
#[test]
2020-02-07 23:35:34 +02:00
fn test_replace_add_use_no_anchor_2seg() {
check_assist(
2020-02-07 23:35:34 +02:00
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
std::fmt<|>::Debug
",
2020-06-15 22:39:26 +02:00
r"
use std::fmt;
fmt::Debug
",
);
}
2019-02-06 15:34:37 +01:00
#[test]
2020-02-07 23:35:34 +02:00
fn test_replace_add_use() {
2019-02-06 15:34:37 +01:00
check_assist(
2020-02-07 23:35:34 +02:00
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
2019-02-06 15:34:37 +01:00
use stdx;
impl std::fmt::Debug<|> for Foo {
}
",
2020-06-15 22:39:26 +02:00
r"
2019-02-06 15:34:37 +01:00
use std::fmt::Debug;
use stdx;
impl Debug for Foo {
2019-02-06 15:34:37 +01:00
}
",
);
}
#[test]
2020-02-07 23:35:34 +02:00
fn test_replace_file_use_other_anchor() {
2019-02-06 15:34:37 +01:00
check_assist(
2020-02-07 23:35:34 +02:00
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
2019-02-06 15:34:37 +01:00
impl std::fmt::Debug<|> for Foo {
}
",
2020-06-15 22:39:26 +02:00
r"
2019-02-06 15:34:37 +01:00
use std::fmt::Debug;
impl Debug for Foo {
2019-02-06 15:34:37 +01:00
}
",
);
}
#[test]
2020-02-07 23:35:34 +02:00
fn test_replace_add_use_other_anchor_indent() {
2019-02-06 15:34:37 +01:00
check_assist(
2020-02-07 23:35:34 +02:00
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
2019-02-06 15:34:37 +01:00
impl std::fmt::Debug<|> for Foo {
}
",
2020-06-15 22:39:26 +02:00
r"
2019-02-06 15:34:37 +01:00
use std::fmt::Debug;
impl Debug for Foo {
2019-02-06 15:34:37 +01:00
}
",
);
}
#[test]
2020-02-07 23:35:34 +02:00
fn test_replace_split_different() {
2019-02-06 15:34:37 +01:00
check_assist(
2020-02-07 23:35:34 +02:00
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
2019-02-06 15:34:37 +01:00
use std::fmt;
impl std::io<|> for Foo {
}
",
2020-06-15 22:39:26 +02:00
r"
use std::{fmt, io};
2019-02-06 15:34:37 +01:00
impl io for Foo {
2019-02-06 15:34:37 +01:00
}
",
);
}
#[test]
2020-02-07 23:35:34 +02:00
fn test_replace_split_self_for_use() {
2019-02-06 15:34:37 +01:00
check_assist(
2020-02-07 23:35:34 +02:00
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
2019-02-06 15:34:37 +01:00
use std::fmt;
impl std::fmt::Debug<|> for Foo {
}
",
2020-06-15 22:39:26 +02:00
r"
use std::fmt::{self, Debug};
2019-02-06 15:34:37 +01:00
impl Debug for Foo {
2019-02-06 15:34:37 +01:00
}
",
);
}
#[test]
2020-02-07 23:35:34 +02:00
fn test_replace_split_self_for_target() {
2019-02-06 15:34:37 +01:00
check_assist(
2020-02-07 23:35:34 +02:00
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
2019-02-06 15:34:37 +01:00
use std::fmt::Debug;
impl std::fmt<|> for Foo {
}
",
2020-06-15 22:39:26 +02:00
r"
use std::fmt::{self, Debug};
2019-02-06 15:34:37 +01:00
impl fmt for Foo {
2019-02-06 15:34:37 +01:00
}
",
);
}
#[test]
2020-02-07 23:35:34 +02:00
fn test_replace_add_to_nested_self_nested() {
2019-02-06 15:34:37 +01:00
check_assist(
2020-02-07 23:35:34 +02:00
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
2019-02-06 15:34:37 +01:00
use std::fmt::{Debug, nested::{Display}};
impl std::fmt::nested<|> for Foo {
}
",
2020-06-15 22:39:26 +02:00
r"
use std::fmt::{Debug, nested::{self, Display}};
2019-02-06 15:34:37 +01:00
impl nested for Foo {
2019-02-06 15:34:37 +01:00
}
",
);
}
#[test]
2020-02-07 23:35:34 +02:00
fn test_replace_add_to_nested_self_already_included() {
2019-02-06 15:34:37 +01:00
check_assist(
2020-02-07 23:35:34 +02:00
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
2019-02-06 15:34:37 +01:00
use std::fmt::{Debug, nested::{self, Display}};
impl std::fmt::nested<|> for Foo {
}
",
2020-06-15 22:39:26 +02:00
r"
use std::fmt::{Debug, nested::{self, Display}};
2019-02-06 15:34:37 +01:00
impl nested for Foo {
2019-02-06 15:34:37 +01:00
}
",
);
}
#[test]
2020-02-07 23:35:34 +02:00
fn test_replace_add_to_nested_nested() {
2019-02-06 15:34:37 +01:00
check_assist(
2020-02-07 23:35:34 +02:00
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
2019-02-06 15:34:37 +01:00
use std::fmt::{Debug, nested::{Display}};
impl std::fmt::nested::Debug<|> for Foo {
}
",
2020-06-15 22:39:26 +02:00
r"
use std::fmt::{Debug, nested::{Debug, Display}};
2019-02-06 15:34:37 +01:00
impl Debug for Foo {
2019-02-06 15:34:37 +01:00
}
",
);
}
#[test]
2020-02-07 23:35:34 +02:00
fn test_replace_split_common_target_longer() {
check_assist(
2020-02-07 23:35:34 +02:00
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
use std::fmt::Debug;
impl std::fmt::nested::Display<|> for Foo {
}
",
2020-06-15 22:39:26 +02:00
r"
use std::fmt::{Debug, nested::Display};
impl Display for Foo {
}
",
);
}
#[test]
2020-02-07 23:35:34 +02:00
fn test_replace_split_common_use_longer() {
check_assist(
2020-02-07 23:35:34 +02:00
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
use std::fmt::nested::Debug;
impl std::fmt::Display<|> for Foo {
}
",
2020-06-15 22:39:26 +02:00
r"
use std::fmt::{Display, nested::Debug};
impl Display for Foo {
}
",
);
}
#[test]
2020-02-07 23:35:34 +02:00
fn test_replace_use_nested_import() {
check_assist(
2020-02-07 23:35:34 +02:00
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
use crate::{
ty::{Substs, Ty},
AssocItem,
};
fn foo() { crate::ty::lower<|>::trait_env() }
",
2020-06-15 22:39:26 +02:00
r"
use crate::{AssocItem, ty::{Substs, Ty, lower}};
fn foo() { lower::trait_env() }
",
);
}
#[test]
2020-02-07 23:35:34 +02:00
fn test_replace_alias() {
2019-02-06 15:34:37 +01:00
check_assist(
2020-02-07 23:35:34 +02:00
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
2019-02-06 15:34:37 +01:00
use std::fmt as foo;
impl foo::Debug<|> for Foo {
}
",
2020-06-15 22:39:26 +02:00
r"
2019-02-06 15:34:37 +01:00
use std::fmt as foo;
use foo::Debug;
impl Debug for Foo {
2019-02-06 15:34:37 +01:00
}
",
);
}
#[test]
fn dont_import_trivial_paths() {
mark::check!(dont_import_trivial_paths);
2019-02-06 15:34:37 +01:00
check_assist_not_applicable(
2020-02-07 23:35:34 +02:00
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
2019-02-06 15:34:37 +01:00
impl foo<|> for Foo {
}
",
);
}
#[test]
2020-02-07 23:35:34 +02:00
fn test_replace_not_applicable_in_use() {
check_assist_not_applicable(
2020-02-07 23:35:34 +02:00
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
use std::fmt<|>;
2019-02-06 15:34:37 +01:00
",
);
}
#[test]
2020-02-07 23:35:34 +02:00
fn test_replace_add_use_no_anchor_in_mod_mod() {
check_assist(
2020-02-07 23:35:34 +02:00
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
mod foo {
mod bar {
std::fmt::Debug<|>
}
}
",
2020-06-15 22:39:26 +02:00
r"
mod foo {
mod bar {
use std::fmt::Debug;
Debug
}
}
",
);
}
#[test]
fn inserts_imports_after_inner_attributes() {
check_assist(
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
#![allow(dead_code)]
fn main() {
std::fmt::Debug<|>
}
",
2020-06-15 22:39:26 +02:00
r"
#![allow(dead_code)]
use std::fmt::Debug;
fn main() {
Debug
}
",
);
}
#[test]
fn replaces_all_affected_paths() {
check_assist(
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
fn main() {
std::fmt::Debug<|>;
let x: std::fmt::Debug = std::fmt::Debug;
}
",
2020-06-15 22:39:26 +02:00
r"
use std::fmt::Debug;
fn main() {
Debug;
let x: Debug = Debug;
}
",
);
}
#[test]
fn replaces_all_affected_paths_mod() {
check_assist(
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
mod m {
fn f() {
std::fmt::Debug<|>;
let x: std::fmt::Debug = std::fmt::Debug;
}
fn g() {
std::fmt::Debug;
}
}
fn f() {
std::fmt::Debug;
}
",
2020-06-15 22:39:26 +02:00
r"
mod m {
use std::fmt::Debug;
fn f() {
Debug;
let x: Debug = Debug;
}
fn g() {
Debug;
}
}
fn f() {
std::fmt::Debug;
}
",
);
}
#[test]
fn does_not_replace_in_submodules() {
check_assist(
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
fn main() {
std::fmt::Debug<|>;
}
mod sub {
fn f() {
std::fmt::Debug;
}
}
",
2020-06-15 22:39:26 +02:00
r"
use std::fmt::Debug;
fn main() {
Debug;
}
mod sub {
fn f() {
std::fmt::Debug;
}
}
",
);
}
#[test]
fn does_not_replace_in_use() {
check_assist(
replace_qualified_name_with_use,
2020-06-15 22:39:26 +02:00
r"
use std::fmt::Display;
fn main() {
std::fmt<|>;
}
",
2020-06-15 22:39:26 +02:00
r"
use std::fmt::{self, Display};
fn main() {
fmt;
2020-08-03 12:17:05 -07:00
}
",
);
}
#[test]
fn does_not_replace_pub_use() {
check_assist(
replace_qualified_name_with_use,
r"
pub use std::fmt;
impl std::io<|> for Foo {
}
",
r"
pub use std::fmt;
use std::io;
2020-08-03 12:17:05 -07:00
impl io for Foo {
}
",
);
}
#[test]
fn does_not_replace_pub_crate_use() {
check_assist(
replace_qualified_name_with_use,
r"
pub(crate) use std::fmt;
impl std::io<|> for Foo {
}
",
r"
pub(crate) use std::fmt;
use std::io;
2020-08-03 12:17:05 -07:00
impl io for Foo {
}
",
);
}
2019-02-06 15:34:37 +01:00
}