extend add_impl_members to constants and types

This commit is contained in:
Ekaterina Babshukova 2019-07-03 16:17:18 +03:00
parent f15f0d1ec0
commit e0e42095db
2 changed files with 87 additions and 50 deletions

View file

@ -2,9 +2,10 @@ use crate::{Assist, AssistId, AssistCtx, ast_editor::{AstEditor, AstBuilder}};
use hir::{HasSource, db::HirDatabase}; use hir::{HasSource, db::HirDatabase};
use ra_syntax::{SmolStr, TreeArc}; use ra_syntax::{SmolStr, TreeArc};
use ra_syntax::ast::{self, AstNode, FnDef, ImplItem, ImplItemKind, NameOwner}; use ra_syntax::ast::{self, AstNode, ImplItem, ImplItemKind, NameOwner};
use ra_db::FilePosition; use ra_db::FilePosition;
#[derive(PartialEq)]
enum AddMissingImplMembersMode { enum AddMissingImplMembersMode {
DefaultMethodsOnly, DefaultMethodsOnly,
NoDefaultMethods, NoDefaultMethods,
@ -45,39 +46,50 @@ fn add_missing_impl_members_inner(
resolve_target_trait_def(ctx.db, &analyzer, impl_node)? resolve_target_trait_def(ctx.db, &analyzer, impl_node)?
}; };
let missing_fns: Vec<_> = { let def_name = |kind| -> Option<&SmolStr> {
let fn_def_opt = |kind| if let ImplItemKind::FnDef(def) = kind { Some(def) } else { None }; match kind {
let def_name = |def| -> Option<&SmolStr> { FnDef::name(def).map(ast::Name::text) }; ImplItemKind::FnDef(def) => def.name(),
ImplItemKind::TypeAliasDef(def) => def.name(),
let trait_items = ImplItemKind::ConstDef(def) => def.name(),
trait_def.syntax().descendants().find_map(ast::ItemList::cast)?.impl_items(); }
let impl_items = impl_item_list.impl_items(); .map(ast::Name::text)
let trait_fns = trait_items.map(ImplItem::kind).filter_map(fn_def_opt);
let impl_fns = impl_items.map(ImplItem::kind).filter_map(fn_def_opt).collect::<Vec<_>>();
trait_fns
.filter(|t| def_name(t).is_some())
.filter(|t| match mode {
AddMissingImplMembersMode::DefaultMethodsOnly => t.body().is_some(),
AddMissingImplMembersMode::NoDefaultMethods => t.body().is_none(),
})
.filter(|t| impl_fns.iter().all(|i| def_name(i) != def_name(t)))
.collect()
}; };
if missing_fns.is_empty() {
let trait_items = trait_def.item_list()?.impl_items();
let impl_items = impl_item_list.impl_items().collect::<Vec<_>>();
let missing_items: Vec<_> = trait_items
.filter(|t| def_name(t.kind()).is_some())
.filter(|t| match t.kind() {
ImplItemKind::FnDef(def) => match mode {
AddMissingImplMembersMode::DefaultMethodsOnly => def.body().is_some(),
AddMissingImplMembersMode::NoDefaultMethods => def.body().is_none(),
},
_ => mode == AddMissingImplMembersMode::NoDefaultMethods,
})
.filter(|t| impl_items.iter().all(|i| def_name(i.kind()) != def_name(t.kind())))
.collect();
if missing_items.is_empty() {
return None; return None;
} }
ctx.add_action(AssistId(assist_id), label, |edit| { ctx.add_action(AssistId(assist_id), label, |edit| {
let n_existing_items = impl_item_list.impl_items().count(); let n_existing_items = impl_item_list.impl_items().count();
let fns = missing_fns.into_iter().map(add_body_and_strip_docstring).collect::<Vec<_>>();
let mut ast_editor = AstEditor::new(impl_item_list); let mut ast_editor = AstEditor::new(impl_item_list);
if n_existing_items == 0 { if n_existing_items == 0 {
ast_editor.make_multiline(); ast_editor.make_multiline();
} }
ast_editor.append_functions(fns.iter().map(|it| &**it));
for item in missing_items {
let it = match item.kind() {
ImplItemKind::FnDef(def) => {
strip_docstring(ImplItem::cast(add_body(def).syntax()).unwrap())
}
_ => strip_docstring(item),
};
ast_editor.append_item(&it)
}
let first_new_item = ast_editor.ast().impl_items().nth(n_existing_items).unwrap(); let first_new_item = ast_editor.ast().impl_items().nth(n_existing_items).unwrap();
let cursor_poisition = first_new_item.syntax().range().start(); let cursor_poisition = first_new_item.syntax().range().start();
ast_editor.into_text_edit(edit.text_edit_builder()); ast_editor.into_text_edit(edit.text_edit_builder());
@ -88,14 +100,19 @@ fn add_missing_impl_members_inner(
ctx.build() ctx.build()
} }
fn add_body_and_strip_docstring(fn_def: &ast::FnDef) -> TreeArc<ast::FnDef> { fn strip_docstring(item: &ast::ImplItem) -> TreeArc<ast::ImplItem> {
let mut ast_editor = AstEditor::new(item);
ast_editor.strip_attrs_and_docs();
ast_editor.ast().to_owned()
}
fn add_body(fn_def: &ast::FnDef) -> TreeArc<ast::FnDef> {
let mut ast_editor = AstEditor::new(fn_def); let mut ast_editor = AstEditor::new(fn_def);
if fn_def.body().is_none() { if fn_def.body().is_none() {
ast_editor.set_body(&AstBuilder::<ast::Block>::single_expr( ast_editor.set_body(&AstBuilder::<ast::Block>::single_expr(
&AstBuilder::<ast::Expr>::unimplemented(), &AstBuilder::<ast::Expr>::unimplemented(),
)); ));
} }
ast_editor.strip_attrs_and_docs();
ast_editor.ast().to_owned() ast_editor.ast().to_owned()
} }
@ -126,6 +143,10 @@ mod tests {
add_missing_impl_members, add_missing_impl_members,
" "
trait Foo { trait Foo {
type Output;
const CONST: usize = 42;
fn foo(&self); fn foo(&self);
fn bar(&self); fn bar(&self);
fn baz(&self); fn baz(&self);
@ -139,6 +160,10 @@ impl Foo for S {
}", }",
" "
trait Foo { trait Foo {
type Output;
const CONST: usize = 42;
fn foo(&self); fn foo(&self);
fn bar(&self); fn bar(&self);
fn baz(&self); fn baz(&self);
@ -148,7 +173,9 @@ struct S;
impl Foo for S { impl Foo for S {
fn bar(&self) {} fn bar(&self) {}
<|>fn foo(&self) { unimplemented!() } <|>type Output;
const CONST: usize = 42;
fn foo(&self) { unimplemented!() }
fn baz(&self) { unimplemented!() } fn baz(&self) { unimplemented!() }
}", }",
@ -256,6 +283,8 @@ impl Foo for S { <|> }",
#[doc(alias = "test alias")] #[doc(alias = "test alias")]
trait Foo { trait Foo {
/// doc string /// doc string
type Output;
#[must_use] #[must_use]
fn foo(&self); fn foo(&self);
} }
@ -265,12 +294,15 @@ impl Foo for S {}<|>"#,
#[doc(alias = "test alias")] #[doc(alias = "test alias")]
trait Foo { trait Foo {
/// doc string /// doc string
type Output;
#[must_use] #[must_use]
fn foo(&self); fn foo(&self);
} }
struct S; struct S;
impl Foo for S { impl Foo for S {
<|>fn foo(&self) { unimplemented!() } <|>type Output;
fn foo(&self) { unimplemented!() }
}"#, }"#,
) )
} }
@ -281,6 +313,10 @@ impl Foo for S {
add_missing_default_members, add_missing_default_members,
" "
trait Foo { trait Foo {
type Output;
const CONST: usize = 42;
fn valid(some: u32) -> bool { false } fn valid(some: u32) -> bool { false }
fn foo(some: u32) -> bool; fn foo(some: u32) -> bool;
} }
@ -288,6 +324,10 @@ struct S;
impl Foo for S { <|> }", impl Foo for S { <|> }",
" "
trait Foo { trait Foo {
type Output;
const CONST: usize = 42;
fn valid(some: u32) -> bool { false } fn valid(some: u32) -> bool { false }
fn foo(some: u32) -> bool; fn foo(some: u32) -> bool;
} }

View file

@ -163,11 +163,7 @@ impl AstEditor<ast::ItemList> {
self.do_make_multiline() self.do_make_multiline()
} }
pub fn append_functions<'a>(&mut self, fns: impl Iterator<Item = &'a ast::FnDef>) { pub fn append_item(&mut self, item: &ast::ImplItem) {
fns.for_each(|it| self.append_function(it))
}
pub fn append_function(&mut self, fn_def: &ast::FnDef) {
let (indent, position) = match self.ast().impl_items().last() { let (indent, position) = match self.ast().impl_items().last() {
Some(it) => ( Some(it) => (
leading_indent(it.syntax()).unwrap_or("").to_string(), leading_indent(it.syntax()).unwrap_or("").to_string(),
@ -182,8 +178,7 @@ impl AstEditor<ast::ItemList> {
}, },
}; };
let ws = tokens::WsBuilder::new(&format!("\n{}", indent)); let ws = tokens::WsBuilder::new(&format!("\n{}", indent));
let to_insert: ArrayVec<[SyntaxElement; 2]> = let to_insert: ArrayVec<[SyntaxElement; 2]> = [ws.ws().into(), item.syntax().into()].into();
[ws.ws().into(), fn_def.syntax().into()].into();
self.ast = self.insert_children(position, to_insert.into_iter()); self.ast = self.insert_children(position, to_insert.into_iter());
} }
@ -192,6 +187,23 @@ impl AstEditor<ast::ItemList> {
} }
} }
impl AstEditor<ast::ImplItem> {
pub fn strip_attrs_and_docs(&mut self) {
while let Some(start) = self
.ast()
.syntax()
.children_with_tokens()
.find(|it| it.kind() == ATTR || it.kind() == COMMENT)
{
let end = match start.next_sibling_or_token() {
Some(el) if el.kind() == WHITESPACE => el,
Some(_) | None => start,
};
self.ast = self.replace_children(RangeInclusive::new(start, end), iter::empty());
}
}
}
impl AstEditor<ast::FnDef> { impl AstEditor<ast::FnDef> {
pub fn set_body(&mut self, body: &ast::Block) { pub fn set_body(&mut self, body: &ast::Block) {
let mut to_insert: ArrayVec<[SyntaxElement; 2]> = ArrayVec::new(); let mut to_insert: ArrayVec<[SyntaxElement; 2]> = ArrayVec::new();
@ -210,21 +222,6 @@ impl AstEditor<ast::FnDef> {
let replace_range = RangeInclusive::new(old_body_or_semi, old_body_or_semi); let replace_range = RangeInclusive::new(old_body_or_semi, old_body_or_semi);
self.ast = self.replace_children(replace_range, to_insert.into_iter()) self.ast = self.replace_children(replace_range, to_insert.into_iter())
} }
pub fn strip_attrs_and_docs(&mut self) {
while let Some(start) = self
.ast()
.syntax()
.children_with_tokens()
.find(|it| it.kind() == ATTR || it.kind() == COMMENT)
{
let end = match start.next_sibling_or_token() {
Some(el) if el.kind() == WHITESPACE => el,
Some(_) | None => start,
};
self.ast = self.replace_children(RangeInclusive::new(start, end), iter::empty());
}
}
} }
pub struct AstBuilder<N: AstNode> { pub struct AstBuilder<N: AstNode> {