mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-27 20:35:09 +00:00
extend add_impl_members to constants and types
This commit is contained in:
parent
f15f0d1ec0
commit
e0e42095db
2 changed files with 87 additions and 50 deletions
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
|
@ -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> {
|
||||||
|
|
Loading…
Reference in a new issue