mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-14 22:24:14 +00:00
Use new HirDisplay variant in add_function assist
This commit is contained in:
parent
fe93675e8a
commit
64e6b8200b
1 changed files with 116 additions and 58 deletions
|
@ -43,16 +43,12 @@ pub(crate) fn add_function(acc: &mut Assists, ctx: &AssistContext) -> Option<()>
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
|
|
||||||
let target_module = if let Some(qualifier) = path.qualifier() {
|
let target_module = match path.qualifier() {
|
||||||
if let Some(hir::PathResolution::Def(hir::ModuleDef::Module(module))) =
|
Some(qualifier) => match ctx.sema.resolve_path(&qualifier) {
|
||||||
ctx.sema.resolve_path(&qualifier)
|
Some(hir::PathResolution::Def(hir::ModuleDef::Module(module))) => Some(module),
|
||||||
{
|
_ => return None,
|
||||||
Some(module.definition_source(ctx.sema.db))
|
},
|
||||||
} else {
|
None => None,
|
||||||
return None;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
};
|
};
|
||||||
|
|
||||||
let function_builder = FunctionBuilder::from_call(&ctx, &call, &path, target_module)?;
|
let function_builder = FunctionBuilder::from_call(&ctx, &call, &path, target_module)?;
|
||||||
|
@ -83,25 +79,29 @@ struct FunctionBuilder {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl FunctionBuilder {
|
impl FunctionBuilder {
|
||||||
/// Prepares a generated function that matches `call` in `generate_in`
|
/// Prepares a generated function that matches `call`.
|
||||||
/// (or as close to `call` as possible, if `generate_in` is `None`)
|
/// The function is generated in `target_module` or next to `call`
|
||||||
fn from_call(
|
fn from_call(
|
||||||
ctx: &AssistContext,
|
ctx: &AssistContext,
|
||||||
call: &ast::CallExpr,
|
call: &ast::CallExpr,
|
||||||
path: &ast::Path,
|
path: &ast::Path,
|
||||||
target_module: Option<hir::InFile<hir::ModuleSource>>,
|
target_module: Option<hir::Module>,
|
||||||
) -> Option<Self> {
|
) -> Option<Self> {
|
||||||
let needs_pub = target_module.is_some();
|
|
||||||
let mut file = ctx.frange.file_id;
|
let mut file = ctx.frange.file_id;
|
||||||
let target = if let Some(target_module) = target_module {
|
let target = match &target_module {
|
||||||
let (in_file, target) = next_space_for_fn_in_module(ctx.sema.db, target_module)?;
|
Some(target_module) => {
|
||||||
file = in_file;
|
let module_source = target_module.definition_source(ctx.db);
|
||||||
target
|
let (in_file, target) = next_space_for_fn_in_module(ctx.sema.db, &module_source)?;
|
||||||
} else {
|
file = in_file;
|
||||||
next_space_for_fn_after_call_site(&call)?
|
target
|
||||||
|
}
|
||||||
|
None => next_space_for_fn_after_call_site(&call)?,
|
||||||
};
|
};
|
||||||
|
let needs_pub = target_module.is_some();
|
||||||
|
let target_module = target_module.or_else(|| ctx.sema.scope(target.syntax()).module())?;
|
||||||
let fn_name = fn_name(&path)?;
|
let fn_name = fn_name(&path)?;
|
||||||
let (type_params, params) = fn_args(ctx, &call)?;
|
let (type_params, params) = fn_args(ctx, target_module, &call)?;
|
||||||
|
|
||||||
Some(Self { target, fn_name, type_params, params, file, needs_pub })
|
Some(Self { target, fn_name, type_params, params, file, needs_pub })
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -144,6 +144,15 @@ enum GeneratedFunctionTarget {
|
||||||
InEmptyItemList(ast::ItemList),
|
InEmptyItemList(ast::ItemList),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl GeneratedFunctionTarget {
|
||||||
|
fn syntax(&self) -> &SyntaxNode {
|
||||||
|
match self {
|
||||||
|
GeneratedFunctionTarget::BehindItem(it) => it,
|
||||||
|
GeneratedFunctionTarget::InEmptyItemList(it) => it.syntax(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn fn_name(call: &ast::Path) -> Option<ast::Name> {
|
fn fn_name(call: &ast::Path) -> Option<ast::Name> {
|
||||||
let name = call.segment()?.syntax().to_string();
|
let name = call.segment()?.syntax().to_string();
|
||||||
Some(ast::make::name(&name))
|
Some(ast::make::name(&name))
|
||||||
|
@ -152,17 +161,17 @@ fn fn_name(call: &ast::Path) -> Option<ast::Name> {
|
||||||
/// Computes the type variables and arguments required for the generated function
|
/// Computes the type variables and arguments required for the generated function
|
||||||
fn fn_args(
|
fn fn_args(
|
||||||
ctx: &AssistContext,
|
ctx: &AssistContext,
|
||||||
|
target_module: hir::Module,
|
||||||
call: &ast::CallExpr,
|
call: &ast::CallExpr,
|
||||||
) -> Option<(Option<ast::TypeParamList>, ast::ParamList)> {
|
) -> Option<(Option<ast::TypeParamList>, ast::ParamList)> {
|
||||||
let mut arg_names = Vec::new();
|
let mut arg_names = Vec::new();
|
||||||
let mut arg_types = Vec::new();
|
let mut arg_types = Vec::new();
|
||||||
for arg in call.arg_list()?.args() {
|
for arg in call.arg_list()?.args() {
|
||||||
let arg_name = match fn_arg_name(&arg) {
|
arg_names.push(match fn_arg_name(&arg) {
|
||||||
Some(name) => name,
|
Some(name) => name,
|
||||||
None => String::from("arg"),
|
None => String::from("arg"),
|
||||||
};
|
});
|
||||||
arg_names.push(arg_name);
|
arg_types.push(match fn_arg_type(ctx, target_module, &arg) {
|
||||||
arg_types.push(match fn_arg_type(ctx, &arg) {
|
|
||||||
Some(ty) => ty,
|
Some(ty) => ty,
|
||||||
None => String::from("()"),
|
None => String::from("()"),
|
||||||
});
|
});
|
||||||
|
@ -218,12 +227,21 @@ fn fn_arg_name(fn_arg: &ast::Expr) -> Option<String> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn fn_arg_type(ctx: &AssistContext, fn_arg: &ast::Expr) -> Option<String> {
|
fn fn_arg_type(
|
||||||
|
ctx: &AssistContext,
|
||||||
|
target_module: hir::Module,
|
||||||
|
fn_arg: &ast::Expr,
|
||||||
|
) -> Option<String> {
|
||||||
let ty = ctx.sema.type_of_expr(fn_arg)?;
|
let ty = ctx.sema.type_of_expr(fn_arg)?;
|
||||||
if ty.is_unknown() {
|
if ty.is_unknown() {
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
Some(ty.display(ctx.sema.db).to_string())
|
|
||||||
|
if let Ok(rendered) = ty.display_source_code(ctx.db, target_module.into()) {
|
||||||
|
Some(rendered)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns the position inside the current mod or file
|
/// Returns the position inside the current mod or file
|
||||||
|
@ -252,10 +270,10 @@ fn next_space_for_fn_after_call_site(expr: &ast::CallExpr) -> Option<GeneratedFu
|
||||||
|
|
||||||
fn next_space_for_fn_in_module(
|
fn next_space_for_fn_in_module(
|
||||||
db: &dyn hir::db::AstDatabase,
|
db: &dyn hir::db::AstDatabase,
|
||||||
module: hir::InFile<hir::ModuleSource>,
|
module_source: &hir::InFile<hir::ModuleSource>,
|
||||||
) -> Option<(FileId, GeneratedFunctionTarget)> {
|
) -> Option<(FileId, GeneratedFunctionTarget)> {
|
||||||
let file = module.file_id.original_file(db);
|
let file = module_source.file_id.original_file(db);
|
||||||
let assist_item = match module.value {
|
let assist_item = match &module_source.value {
|
||||||
hir::ModuleSource::SourceFile(it) => {
|
hir::ModuleSource::SourceFile(it) => {
|
||||||
if let Some(last_item) = it.items().last() {
|
if let Some(last_item) = it.items().last() {
|
||||||
GeneratedFunctionTarget::BehindItem(last_item.syntax().clone())
|
GeneratedFunctionTarget::BehindItem(last_item.syntax().clone())
|
||||||
|
@ -599,8 +617,33 @@ fn bar(foo: impl Foo) {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
#[ignore]
|
fn borrowed_arg() {
|
||||||
// FIXME print paths properly to make this test pass
|
check_assist(
|
||||||
|
add_function,
|
||||||
|
r"
|
||||||
|
struct Baz;
|
||||||
|
fn baz() -> Baz { todo!() }
|
||||||
|
|
||||||
|
fn foo() {
|
||||||
|
bar<|>(&baz())
|
||||||
|
}
|
||||||
|
",
|
||||||
|
r"
|
||||||
|
struct Baz;
|
||||||
|
fn baz() -> Baz { todo!() }
|
||||||
|
|
||||||
|
fn foo() {
|
||||||
|
bar(&baz())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn bar(baz: &Baz) {
|
||||||
|
<|>todo!()
|
||||||
|
}
|
||||||
|
",
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
fn add_function_with_qualified_path_arg() {
|
fn add_function_with_qualified_path_arg() {
|
||||||
check_assist(
|
check_assist(
|
||||||
add_function,
|
add_function,
|
||||||
|
@ -609,10 +652,8 @@ mod Baz {
|
||||||
pub struct Bof;
|
pub struct Bof;
|
||||||
pub fn baz() -> Bof { Bof }
|
pub fn baz() -> Bof { Bof }
|
||||||
}
|
}
|
||||||
mod Foo {
|
fn foo() {
|
||||||
fn foo() {
|
<|>bar(Baz::baz())
|
||||||
<|>bar(super::Baz::baz())
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
",
|
",
|
||||||
r"
|
r"
|
||||||
|
@ -620,14 +661,12 @@ mod Baz {
|
||||||
pub struct Bof;
|
pub struct Bof;
|
||||||
pub fn baz() -> Bof { Bof }
|
pub fn baz() -> Bof { Bof }
|
||||||
}
|
}
|
||||||
mod Foo {
|
fn foo() {
|
||||||
fn foo() {
|
bar(Baz::baz())
|
||||||
bar(super::Baz::baz())
|
}
|
||||||
}
|
|
||||||
|
|
||||||
fn bar(baz: super::Baz::Bof) {
|
fn bar(baz: Baz::Bof) {
|
||||||
<|>todo!()
|
<|>todo!()
|
||||||
}
|
|
||||||
}
|
}
|
||||||
",
|
",
|
||||||
)
|
)
|
||||||
|
@ -808,6 +847,40 @@ fn foo() {
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[ignore]
|
||||||
|
// Ignored until local imports are supported.
|
||||||
|
// See https://github.com/rust-analyzer/rust-analyzer/issues/1165
|
||||||
|
fn qualified_path_uses_correct_scope() {
|
||||||
|
check_assist(
|
||||||
|
add_function,
|
||||||
|
"
|
||||||
|
mod foo {
|
||||||
|
pub struct Foo;
|
||||||
|
}
|
||||||
|
fn bar() {
|
||||||
|
use foo::Foo;
|
||||||
|
let foo = Foo;
|
||||||
|
baz<|>(foo)
|
||||||
|
}
|
||||||
|
",
|
||||||
|
"
|
||||||
|
mod foo {
|
||||||
|
pub struct Foo;
|
||||||
|
}
|
||||||
|
fn bar() {
|
||||||
|
use foo::Foo;
|
||||||
|
let foo = Foo;
|
||||||
|
baz(foo)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn baz(foo: foo::Foo) {
|
||||||
|
<|>todo!()
|
||||||
|
}
|
||||||
|
",
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn add_function_in_module_containing_other_items() {
|
fn add_function_in_module_containing_other_items() {
|
||||||
check_assist(
|
check_assist(
|
||||||
|
@ -919,21 +992,6 @@ fn bar(baz: ()) {}
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn add_function_not_applicable_if_function_path_not_singleton() {
|
|
||||||
// In the future this assist could be extended to generate functions
|
|
||||||
// if the path is in the same crate (or even the same workspace).
|
|
||||||
// For the beginning, I think this is fine.
|
|
||||||
check_assist_not_applicable(
|
|
||||||
add_function,
|
|
||||||
r"
|
|
||||||
fn foo() {
|
|
||||||
other_crate::bar<|>();
|
|
||||||
}
|
|
||||||
",
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
#[ignore]
|
#[ignore]
|
||||||
fn create_method_with_no_args() {
|
fn create_method_with_no_args() {
|
||||||
|
|
Loading…
Reference in a new issue