mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-13 13:48:50 +00:00
Enable delegation generation for complex types
This commit is contained in:
parent
c9882c8002
commit
680dd9d952
3 changed files with 133 additions and 34 deletions
|
@ -1,5 +1,5 @@
|
|||
use hir::{self, HasCrate, HasSource, HirDisplay};
|
||||
use syntax::ast::{self, make, AstNode, HasName, HasVisibility};
|
||||
use syntax::ast::{self, make, AstNode, HasGenericParams, HasName, HasVisibility};
|
||||
|
||||
use crate::{
|
||||
utils::{find_struct_impl, render_snippet, Cursor},
|
||||
|
@ -7,29 +7,44 @@ use crate::{
|
|||
};
|
||||
use syntax::ast::edit::AstNodeEdit;
|
||||
|
||||
// Assist: generate_setter
|
||||
// Assist: generate_delegate
|
||||
//
|
||||
// Generate a setter method.
|
||||
// Generate a delegate method.
|
||||
//
|
||||
// ```
|
||||
// struct Age(u8);
|
||||
// impl Age {
|
||||
// fn age(&self) -> u8 {
|
||||
// self.0
|
||||
// }
|
||||
// }
|
||||
//
|
||||
// struct Person {
|
||||
// nam$0e: String,
|
||||
// ag$0e: Age,
|
||||
// }
|
||||
// ```
|
||||
// ->
|
||||
// ```
|
||||
// struct Age(u8);
|
||||
// impl Age {
|
||||
// fn age(&self) -> u8 {
|
||||
// self.0
|
||||
// }
|
||||
// }
|
||||
//
|
||||
// struct Person {
|
||||
// name: String,
|
||||
// age: Age,
|
||||
// }
|
||||
//
|
||||
// impl Person {
|
||||
// /// Set the person's name.
|
||||
// fn set_name(&mut self, name: String) {
|
||||
// self.name = name;
|
||||
// $0fn age(&self) -> u8 {
|
||||
// self.age.age()
|
||||
// }
|
||||
// }
|
||||
// ```
|
||||
pub(crate) fn generate_delegate(acc: &mut Assists, ctx: &AssistContext) -> Option<()> {
|
||||
let cap = ctx.config.snippet_cap?;
|
||||
|
||||
let strukt = ctx.find_node_at_offset::<ast::Struct>()?;
|
||||
let strukt_name = strukt.name()?;
|
||||
|
||||
|
@ -62,7 +77,7 @@ pub(crate) fn generate_delegate(acc: &mut Assists, ctx: &AssistContext) -> Optio
|
|||
format!("Generate a delegate method for '{}'", method.name(ctx.db())),
|
||||
target,
|
||||
|builder| {
|
||||
// make function
|
||||
// Create the function
|
||||
let method_source = match method.source(ctx.db()) {
|
||||
Some(source) => source.value,
|
||||
None => return,
|
||||
|
@ -70,29 +85,31 @@ pub(crate) fn generate_delegate(acc: &mut Assists, ctx: &AssistContext) -> Optio
|
|||
let method_name = method.name(ctx.db());
|
||||
let vis = method_source.visibility();
|
||||
let name = make::name(&method.name(ctx.db()).to_string());
|
||||
let type_params = None;
|
||||
let self_ty = method
|
||||
.self_param(ctx.db())
|
||||
.map(|s| s.source(ctx.db()).map(|s| s.value))
|
||||
.flatten();
|
||||
let params = make::param_list(self_ty, []);
|
||||
let params =
|
||||
method_source.param_list().unwrap_or_else(|| make::param_list(None, []));
|
||||
let tail_expr = make::expr_method_call(
|
||||
field_from_idents(["self", &field_name.to_string()]).unwrap(),
|
||||
make::ext::field_from_idents(["self", &field_name.to_string()]).unwrap(), // This unwrap is ok because we have at least 1 arg in the list
|
||||
make::name_ref(&method_name.to_string()),
|
||||
make::arg_list([]),
|
||||
);
|
||||
let type_params = method_source.generic_param_list();
|
||||
let body = make::block_expr([], Some(tail_expr));
|
||||
let ret_type = &method.ret_type(ctx.db()).display(ctx.db()).to_string();
|
||||
let ret_type = Some(make::ret_type(make::ty(ret_type)));
|
||||
let is_async = false;
|
||||
let ret_type = method.ret_type(ctx.db());
|
||||
let ret_type = if ret_type.is_unknown() {
|
||||
Some(make::ret_type(make::ty_placeholder()))
|
||||
} else {
|
||||
let ret_type = &ret_type.display(ctx.db()).to_string();
|
||||
Some(make::ret_type(make::ty(ret_type)))
|
||||
};
|
||||
let is_async = method_source.async_token().is_some();
|
||||
let f = make::fn_(vis, name, type_params, params, body, ret_type, is_async)
|
||||
.indent(ast::edit::IndentLevel(1))
|
||||
.clone_for_update();
|
||||
|
||||
let cursor = Cursor::Before(f.syntax());
|
||||
let cap = ctx.config.snippet_cap.unwrap(); // FIXME.
|
||||
|
||||
// Create or update an impl block, and attach the function to it.
|
||||
// Create or update an impl block, attach the function to it,
|
||||
// then insert into our code.
|
||||
match impl_def {
|
||||
Some(impl_def) => {
|
||||
// Remember where in our source our `impl` block lives.
|
||||
|
@ -110,7 +127,10 @@ pub(crate) fn generate_delegate(acc: &mut Assists, ctx: &AssistContext) -> Optio
|
|||
None => {
|
||||
// Attach the function to the impl block
|
||||
let name = &strukt_name.to_string();
|
||||
let impl_def = make::impl_(make::ext::ident_path(name)).clone_for_update();
|
||||
let params = strukt.generic_param_list();
|
||||
let ty_params = params.clone();
|
||||
let impl_def = make::impl_(make::ext::ident_path(name), params, ty_params)
|
||||
.clone_for_update();
|
||||
let assoc_items = impl_def.get_or_create_assoc_item_list();
|
||||
assoc_items.add_item(f.clone().into());
|
||||
|
||||
|
@ -127,15 +147,6 @@ pub(crate) fn generate_delegate(acc: &mut Assists, ctx: &AssistContext) -> Optio
|
|||
Some(())
|
||||
}
|
||||
|
||||
pub fn field_from_idents<'a>(
|
||||
parts: impl std::iter::IntoIterator<Item = &'a str>,
|
||||
) -> Option<ast::Expr> {
|
||||
let mut iter = parts.into_iter();
|
||||
let base = make::expr_path(make::ext::ident_path(iter.next()?));
|
||||
let expr = iter.fold(base, |base, s| make::expr_field(base, s));
|
||||
Some(expr)
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use crate::tests::check_assist;
|
||||
|
@ -213,6 +224,36 @@ impl Person {
|
|||
}"#,
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_generate_delegate_enable_all_attributes() {
|
||||
check_assist(
|
||||
generate_delegate,
|
||||
r#"
|
||||
struct Age<T>(T);
|
||||
impl<T> Age<T> {
|
||||
pub(crate) async fn age<J, 'a>(&'a mut self, ty: T, arg: J) -> T {
|
||||
self.0
|
||||
}
|
||||
}
|
||||
|
||||
struct Person<T> {
|
||||
ag$0e: Age<T>,
|
||||
}"#,
|
||||
r#"
|
||||
struct Age<T>(T);
|
||||
impl<T> Age<T> {
|
||||
pub(crate) async fn age<J, 'a>(&'a mut self, ty: T, arg: J) -> T {
|
||||
self.0
|
||||
}
|
||||
}
|
||||
|
||||
struct Person<T> {
|
||||
age: Age<T>,
|
||||
}
|
||||
|
||||
impl<T> Person<T> {
|
||||
$0pub(crate) async fn age<J, 'a>(&'a mut self, ty: T, arg: J) -> _ {
|
||||
self.age.age()
|
||||
}
|
||||
}"#,
|
||||
|
|
|
@ -728,6 +728,43 @@ impl Default for Example {
|
|||
)
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn doctest_generate_delegate() {
|
||||
check_doc_test(
|
||||
"generate_delegate",
|
||||
r#####"
|
||||
struct Age(u8);
|
||||
impl Age {
|
||||
fn age(&self) -> u8 {
|
||||
self.0
|
||||
}
|
||||
}
|
||||
|
||||
struct Person {
|
||||
ag$0e: Age,
|
||||
}
|
||||
"#####,
|
||||
r#####"
|
||||
struct Age(u8);
|
||||
impl Age {
|
||||
fn age(&self) -> u8 {
|
||||
self.0
|
||||
}
|
||||
}
|
||||
|
||||
struct Person {
|
||||
age: Age,
|
||||
}
|
||||
|
||||
impl Person {
|
||||
$0fn age(&self) -> u8 {
|
||||
self.age.age()
|
||||
}
|
||||
}
|
||||
"#####,
|
||||
)
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn doctest_generate_deref() {
|
||||
check_doc_test(
|
||||
|
|
|
@ -44,6 +44,15 @@ pub mod ext {
|
|||
Some(path)
|
||||
}
|
||||
|
||||
pub fn field_from_idents<'a>(
|
||||
parts: impl std::iter::IntoIterator<Item = &'a str>,
|
||||
) -> Option<ast::Expr> {
|
||||
let mut iter = parts.into_iter();
|
||||
let base = expr_path(ext::ident_path(iter.next()?));
|
||||
let expr = iter.fold(base, |base, s| expr_field(base, s));
|
||||
Some(expr)
|
||||
}
|
||||
|
||||
pub fn expr_unreachable() -> ast::Expr {
|
||||
expr_from_text("unreachable!()")
|
||||
}
|
||||
|
@ -124,8 +133,20 @@ pub fn assoc_item_list() -> ast::AssocItemList {
|
|||
ast_from_text("impl C for D {}")
|
||||
}
|
||||
|
||||
pub fn impl_(ty: ast::Path) -> ast::Impl {
|
||||
ast_from_text(&format!("impl {} {{}}", ty))
|
||||
pub fn impl_(
|
||||
ty: ast::Path,
|
||||
params: Option<ast::GenericParamList>,
|
||||
ty_params: Option<ast::GenericParamList>,
|
||||
) -> ast::Impl {
|
||||
let params = match params {
|
||||
Some(params) => params.to_string(),
|
||||
None => String::new(),
|
||||
};
|
||||
let ty_params = match ty_params {
|
||||
Some(params) => params.to_string(),
|
||||
None => String::new(),
|
||||
};
|
||||
ast_from_text(&format!("impl{} {}{} {{}}", params, ty, ty_params))
|
||||
}
|
||||
|
||||
pub fn impl_trait(trait_: ast::Path, ty: ast::Path) -> ast::Impl {
|
||||
|
@ -649,7 +670,7 @@ pub fn fn_(
|
|||
is_async: bool,
|
||||
) -> ast::Fn {
|
||||
let type_params = match type_params {
|
||||
Some(type_params) => format!("<{}>", type_params),
|
||||
Some(type_params) => format!("{}", type_params),
|
||||
None => "".into(),
|
||||
};
|
||||
let ret_type = match ret_type {
|
||||
|
|
Loading…
Reference in a new issue