2020-08-12 08:04:06 -05:00
|
|
|
use itertools::Itertools;
|
2020-08-12 11:26:51 -05:00
|
|
|
use syntax::{
|
2019-10-15 13:29:20 -05:00
|
|
|
ast::{self, AstNode},
|
|
|
|
Direction, SmolStr,
|
|
|
|
SyntaxKind::{IDENT, WHITESPACE},
|
2020-04-24 16:40:41 -05:00
|
|
|
TextRange, TextSize,
|
2019-10-15 13:29:20 -05:00
|
|
|
};
|
|
|
|
|
2020-05-06 11:45:35 -05:00
|
|
|
use crate::{
|
|
|
|
assist_context::{AssistContext, Assists},
|
2020-06-28 17:36:05 -05:00
|
|
|
AssistId, AssistKind,
|
2020-05-06 11:45:35 -05:00
|
|
|
};
|
2020-03-19 06:36:33 -05:00
|
|
|
|
2019-10-19 06:19:06 -05:00
|
|
|
// Assist: add_custom_impl
|
|
|
|
//
|
|
|
|
// Adds impl block for derived trait.
|
|
|
|
//
|
|
|
|
// ```
|
|
|
|
// #[derive(Deb<|>ug, Display)]
|
|
|
|
// struct S;
|
|
|
|
// ```
|
|
|
|
// ->
|
|
|
|
// ```
|
|
|
|
// #[derive(Display)]
|
|
|
|
// struct S;
|
|
|
|
//
|
|
|
|
// impl Debug for S {
|
2020-05-17 07:21:24 -05:00
|
|
|
// $0
|
2019-10-19 06:19:06 -05:00
|
|
|
// }
|
|
|
|
// ```
|
2020-05-06 11:45:35 -05:00
|
|
|
pub(crate) fn add_custom_impl(acc: &mut Assists, ctx: &AssistContext) -> Option<()> {
|
2020-07-30 13:16:04 -05:00
|
|
|
let attr = ctx.find_node_at_offset::<ast::Attr>()?;
|
|
|
|
let input = attr.token_tree()?;
|
2019-10-15 13:29:20 -05:00
|
|
|
|
|
|
|
let attr_name = attr
|
|
|
|
.syntax()
|
|
|
|
.descendants_with_tokens()
|
|
|
|
.filter(|t| t.kind() == IDENT)
|
|
|
|
.find_map(|i| i.into_token())
|
2020-03-28 05:01:25 -05:00
|
|
|
.filter(|t| *t.text() == "derive")?
|
2019-10-15 13:29:20 -05:00
|
|
|
.text()
|
|
|
|
.clone();
|
|
|
|
|
|
|
|
let trait_token =
|
2020-02-18 07:32:19 -06:00
|
|
|
ctx.token_at_offset().find(|t| t.kind() == IDENT && *t.text() != attr_name)?;
|
2019-10-15 13:29:20 -05:00
|
|
|
|
2020-02-18 06:53:02 -06:00
|
|
|
let annotated = attr.syntax().siblings(Direction::Next).find_map(ast::Name::cast)?;
|
2019-10-15 13:29:20 -05:00
|
|
|
let annotated_name = annotated.syntax().text().to_string();
|
|
|
|
let start_offset = annotated.syntax().parent()?.text_range().end();
|
|
|
|
|
2020-01-14 08:07:28 -06:00
|
|
|
let label =
|
2020-05-13 18:06:07 -05:00
|
|
|
format!("Add custom impl `{}` for `{}`", trait_token.text().as_str(), annotated_name);
|
2020-01-14 08:07:28 -06:00
|
|
|
|
2020-05-06 05:51:28 -05:00
|
|
|
let target = attr.syntax().text_range();
|
2020-07-02 16:48:35 -05:00
|
|
|
acc.add(AssistId("add_custom_impl", AssistKind::Refactor), label, target, |builder| {
|
2019-10-15 13:29:20 -05:00
|
|
|
let new_attr_input = input
|
|
|
|
.syntax()
|
|
|
|
.descendants_with_tokens()
|
|
|
|
.filter(|t| t.kind() == IDENT)
|
|
|
|
.filter_map(|t| t.into_token().map(|t| t.text().clone()))
|
|
|
|
.filter(|t| t != trait_token.text())
|
|
|
|
.collect::<Vec<SmolStr>>();
|
2020-02-18 06:53:02 -06:00
|
|
|
let has_more_derives = !new_attr_input.is_empty();
|
2019-10-15 13:29:20 -05:00
|
|
|
|
2020-05-17 07:21:24 -05:00
|
|
|
if has_more_derives {
|
2020-08-12 08:04:06 -05:00
|
|
|
let new_attr_input = format!("({})", new_attr_input.iter().format(", "));
|
2020-05-17 07:21:24 -05:00
|
|
|
builder.replace(input.syntax().text_range(), new_attr_input);
|
2019-10-15 13:29:20 -05:00
|
|
|
} else {
|
|
|
|
let attr_range = attr.syntax().text_range();
|
2020-05-17 07:21:24 -05:00
|
|
|
builder.delete(attr_range);
|
2019-10-15 13:29:20 -05:00
|
|
|
|
|
|
|
let line_break_range = attr
|
|
|
|
.syntax()
|
|
|
|
.next_sibling_or_token()
|
|
|
|
.filter(|t| t.kind() == WHITESPACE)
|
|
|
|
.map(|t| t.text_range())
|
2020-04-24 16:40:41 -05:00
|
|
|
.unwrap_or_else(|| TextRange::new(TextSize::from(0), TextSize::from(0)));
|
2020-05-17 07:21:24 -05:00
|
|
|
builder.delete(line_break_range);
|
|
|
|
}
|
|
|
|
|
|
|
|
match ctx.config.snippet_cap {
|
|
|
|
Some(cap) => {
|
|
|
|
builder.insert_snippet(
|
|
|
|
cap,
|
|
|
|
start_offset,
|
|
|
|
format!("\n\nimpl {} for {} {{\n $0\n}}", trait_token, annotated_name),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
None => {
|
|
|
|
builder.insert(
|
|
|
|
start_offset,
|
|
|
|
format!("\n\nimpl {} for {} {{\n\n}}", trait_token, annotated_name),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2019-10-15 13:29:20 -05:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2020-05-06 03:16:55 -05:00
|
|
|
use crate::tests::{check_assist, check_assist_not_applicable};
|
2019-10-15 13:29:20 -05:00
|
|
|
|
2020-03-28 05:01:25 -05:00
|
|
|
use super::*;
|
|
|
|
|
2019-10-15 13:29:20 -05:00
|
|
|
#[test]
|
|
|
|
fn add_custom_impl_for_unique_input() {
|
|
|
|
check_assist(
|
|
|
|
add_custom_impl,
|
|
|
|
"
|
|
|
|
#[derive(Debu<|>g)]
|
|
|
|
struct Foo {
|
|
|
|
bar: String,
|
|
|
|
}
|
|
|
|
",
|
|
|
|
"
|
|
|
|
struct Foo {
|
|
|
|
bar: String,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Debug for Foo {
|
2020-05-17 07:21:24 -05:00
|
|
|
$0
|
2019-10-15 13:29:20 -05:00
|
|
|
}
|
|
|
|
",
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn add_custom_impl_for_with_visibility_modifier() {
|
|
|
|
check_assist(
|
|
|
|
add_custom_impl,
|
|
|
|
"
|
|
|
|
#[derive(Debug<|>)]
|
|
|
|
pub struct Foo {
|
|
|
|
bar: String,
|
|
|
|
}
|
|
|
|
",
|
|
|
|
"
|
|
|
|
pub struct Foo {
|
|
|
|
bar: String,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Debug for Foo {
|
2020-05-17 07:21:24 -05:00
|
|
|
$0
|
2019-10-15 13:29:20 -05:00
|
|
|
}
|
|
|
|
",
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn add_custom_impl_when_multiple_inputs() {
|
|
|
|
check_assist(
|
|
|
|
add_custom_impl,
|
|
|
|
"
|
|
|
|
#[derive(Display, Debug<|>, Serialize)]
|
|
|
|
struct Foo {}
|
|
|
|
",
|
|
|
|
"
|
|
|
|
#[derive(Display, Serialize)]
|
|
|
|
struct Foo {}
|
|
|
|
|
|
|
|
impl Debug for Foo {
|
2020-05-17 07:21:24 -05:00
|
|
|
$0
|
2019-10-15 13:29:20 -05:00
|
|
|
}
|
|
|
|
",
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_ignore_derive_macro_without_input() {
|
|
|
|
check_assist_not_applicable(
|
|
|
|
add_custom_impl,
|
|
|
|
"
|
|
|
|
#[derive(<|>)]
|
|
|
|
struct Foo {}
|
|
|
|
",
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_ignore_if_cursor_on_param() {
|
|
|
|
check_assist_not_applicable(
|
|
|
|
add_custom_impl,
|
|
|
|
"
|
|
|
|
#[derive<|>(Debug)]
|
|
|
|
struct Foo {}
|
|
|
|
",
|
|
|
|
);
|
|
|
|
|
|
|
|
check_assist_not_applicable(
|
|
|
|
add_custom_impl,
|
|
|
|
"
|
|
|
|
#[derive(Debug)<|>]
|
|
|
|
struct Foo {}
|
|
|
|
",
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_ignore_if_not_derive() {
|
|
|
|
check_assist_not_applicable(
|
|
|
|
add_custom_impl,
|
|
|
|
"
|
|
|
|
#[allow(non_camel_<|>case_types)]
|
|
|
|
struct Foo {}
|
|
|
|
",
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|