blob: 9af8411f4cb6378f809c85162a4174e2e92d1b1d [file] [log] [blame]
use std::iter;
use ast::edit::IndentLevel;
use hir::HasAttrs;
use ide_db::base_db::AnchoredPathBuf;
use itertools::Itertools;
use stdx::format_to;
use syntax::{
ast::{self, edit::AstNodeEdit, HasName},
AstNode, SmolStr, TextRange,
};
use crate::{AssistContext, AssistId, AssistKind, Assists};
// Assist: move_module_to_file
//
// Moves inline module's contents to a separate file.
//
// ```
// mod $0foo {
// fn t() {}
// }
// ```
// ->
// ```
// mod foo;
// ```
pub(crate) fn move_module_to_file(acc: &mut Assists, ctx: &AssistContext<'_>) -> Option<()> {
let module_ast = ctx.find_node_at_offset::<ast::Module>()?;
let module_items = module_ast.item_list()?;
let l_curly_offset = module_items.syntax().text_range().start();
if l_curly_offset <= ctx.offset() {
cov_mark::hit!(available_before_curly);
return None;
}
let target = TextRange::new(module_ast.syntax().text_range().start(), l_curly_offset);
let module_name = module_ast.name()?;
// get to the outermost module syntax so we can grab the module of file we are in
let outermost_mod_decl =
iter::successors(Some(module_ast.clone()), |module| module.parent()).last()?;
let module_def = ctx.sema.to_def(&outermost_mod_decl)?;
let parent_module = module_def.parent(ctx.db())?;
acc.add(
AssistId("move_module_to_file", AssistKind::RefactorExtract),
"Extract module to file",
target,
|builder| {
let path = {
let mut buf = String::from("./");
let db = ctx.db();
match parent_module.name(db) {
Some(name)
if !parent_module.is_mod_rs(db)
&& parent_module
.attrs(db)
.by_key("path")
.string_value_unescape()
.is_none() =>
{
format_to!(buf, "{}/", name.display(db))
}
_ => (),
}
let segments = iter::successors(Some(module_ast.clone()), |module| module.parent())
.filter_map(|it| it.name())
.map(|name| SmolStr::from(name.text().trim_start_matches("r#")))
.collect::<Vec<_>>();
format_to!(buf, "{}", segments.into_iter().rev().format("/"));
// We need to special case mod named `r#mod` and place the file in a
// subdirectory as "mod.rs" would be of its parent module otherwise.
if module_name.text() == "r#mod" {
format_to!(buf, "/mod.rs");
} else {
format_to!(buf, ".rs");
}
buf
};
let contents = {
let items = module_items.dedent(IndentLevel(1)).to_string();
let mut items =
items.trim_start_matches('{').trim_end_matches('}').trim().to_owned();
if !items.is_empty() {
items.push('\n');
}
items
};
let buf = format!("mod {module_name};");
let replacement_start = match module_ast.mod_token() {
Some(mod_token) => mod_token.text_range(),
None => module_ast.syntax().text_range(),
}
.start();
builder.replace(
TextRange::new(replacement_start, module_ast.syntax().text_range().end()),
buf,
);
let dst = AnchoredPathBuf { anchor: ctx.file_id(), path };
builder.create_file(dst, contents);
},
)
}
#[cfg(test)]
mod tests {
use crate::tests::{check_assist, check_assist_not_applicable};
use super::*;
#[test]
fn extract_with_specified_path_attr() {
check_assist(
move_module_to_file,
r#"
//- /main.rs
#[path="parser/__mod.rs"]
mod parser;
//- /parser/__mod.rs
fn test() {}
mod $0expr {
struct A {}
}
"#,
r#"
//- /parser/__mod.rs
fn test() {}
mod expr;
//- /parser/expr.rs
struct A {}
"#,
);
check_assist(
move_module_to_file,
r#"
//- /main.rs
#[path="parser/a/__mod.rs"]
mod parser;
//- /parser/a/__mod.rs
fn test() {}
mod $0expr {
struct A {}
}
"#,
r#"
//- /parser/a/__mod.rs
fn test() {}
mod expr;
//- /parser/a/expr.rs
struct A {}
"#,
);
check_assist(
move_module_to_file,
r#"
//- /main.rs
#[path="a.rs"]
mod parser;
//- /a.rs
fn test() {}
mod $0expr {
struct A {}
}
"#,
r#"
//- /a.rs
fn test() {}
mod expr;
//- /expr.rs
struct A {}
"#,
);
}
#[test]
fn extract_from_root() {
check_assist(
move_module_to_file,
r#"
mod $0tests {
#[test] fn t() {}
}
"#,
r#"
//- /main.rs
mod tests;
//- /tests.rs
#[test] fn t() {}
"#,
);
}
#[test]
fn extract_from_submodule() {
check_assist(
move_module_to_file,
r#"
//- /main.rs
mod submod;
//- /submod.rs
$0mod inner {
fn f() {}
}
fn g() {}
"#,
r#"
//- /submod.rs
mod inner;
fn g() {}
//- /submod/inner.rs
fn f() {}
"#,
);
}
#[test]
fn extract_from_mod_rs() {
check_assist(
move_module_to_file,
r#"
//- /main.rs
mod submodule;
//- /submodule/mod.rs
mod inner$0 {
fn f() {}
}
fn g() {}
"#,
r#"
//- /submodule/mod.rs
mod inner;
fn g() {}
//- /submodule/inner.rs
fn f() {}
"#,
);
}
#[test]
fn extract_public() {
check_assist(
move_module_to_file,
r#"
pub mod $0tests {
#[test] fn t() {}
}
"#,
r#"
//- /main.rs
pub mod tests;
//- /tests.rs
#[test] fn t() {}
"#,
);
}
#[test]
fn extract_public_crate() {
check_assist(
move_module_to_file,
r#"
pub(crate) mod $0tests {
#[test] fn t() {}
}
"#,
r#"
//- /main.rs
pub(crate) mod tests;
//- /tests.rs
#[test] fn t() {}
"#,
);
}
#[test]
fn available_before_curly() {
cov_mark::check!(available_before_curly);
check_assist_not_applicable(move_module_to_file, r#"mod m { $0 }"#);
}
#[test]
fn keep_outer_comments_and_attributes() {
check_assist(
move_module_to_file,
r#"
/// doc comment
#[attribute]
mod $0tests {
#[test] fn t() {}
}
"#,
r#"
//- /main.rs
/// doc comment
#[attribute]
mod tests;
//- /tests.rs
#[test] fn t() {}
"#,
);
}
#[test]
fn extract_nested() {
check_assist(
move_module_to_file,
r#"
//- /lib.rs
mod foo;
//- /foo.rs
mod bar {
mod baz {
mod qux$0 {}
}
}
"#,
r#"
//- /foo.rs
mod bar {
mod baz {
mod qux;
}
}
//- /foo/bar/baz/qux.rs
"#,
);
}
#[test]
fn extract_mod_with_raw_ident() {
check_assist(
move_module_to_file,
r#"
//- /main.rs
mod $0r#static {}
"#,
r#"
//- /main.rs
mod r#static;
//- /static.rs
"#,
)
}
#[test]
fn extract_r_mod() {
check_assist(
move_module_to_file,
r#"
//- /main.rs
mod $0r#mod {}
"#,
r#"
//- /main.rs
mod r#mod;
//- /mod/mod.rs
"#,
)
}
#[test]
fn extract_r_mod_from_mod_rs() {
check_assist(
move_module_to_file,
r#"
//- /main.rs
mod foo;
//- /foo/mod.rs
mod $0r#mod {}
"#,
r#"
//- /foo/mod.rs
mod r#mod;
//- /foo/mod/mod.rs
"#,
)
}
#[test]
fn extract_nested_r_mod() {
check_assist(
move_module_to_file,
r#"
//- /main.rs
mod r#mod {
mod foo {
mod $0r#mod {}
}
}
"#,
r#"
//- /main.rs
mod r#mod {
mod foo {
mod r#mod;
}
}
//- /mod/foo/mod/mod.rs
"#,
)
}
}