Resolve most of corner cases

This commit is contained in:
Aleksei Sidorov 2020-09-03 14:46:28 +03:00
parent fe3170dc34
commit 7259cc82f3
2 changed files with 124 additions and 12 deletions

View file

@ -13,9 +13,6 @@ pub(crate) fn replace_impl_trait_with_generic(
let type_param = type_impl_trait.syntax().parent().and_then(ast::Param::cast)?; let type_param = type_impl_trait.syntax().parent().and_then(ast::Param::cast)?;
let type_fn = type_param.syntax().ancestors().nth(2).and_then(ast::Fn::cast)?; let type_fn = type_param.syntax().ancestors().nth(2).and_then(ast::Fn::cast)?;
let generic_param_list =
type_fn.generic_param_list().unwrap_or_else(|| make::generic_param_list(None));
let impl_trait_ty = type_impl_trait let impl_trait_ty = type_impl_trait
.syntax() .syntax()
.descendants() .descendants()
@ -31,11 +28,16 @@ pub(crate) fn replace_impl_trait_with_generic(
target, target,
|edit| { |edit| {
let generic_letter = impl_trait_ty[..1].to_string(); let generic_letter = impl_trait_ty[..1].to_string();
edit.replace_ast::<ast::Type>(type_impl_trait.into(), make::ty(&generic_letter));
let new_params = generic_param_list let generic_param_list = type_fn
.append_param(make::generic_param(generic_letter, Some(impl_trait_ty))); .generic_param_list()
let new_type_fn = type_fn.replace_descendant(generic_param_list, new_params); .unwrap_or_else(|| make::generic_param_list(None))
.append_param(make::generic_param(generic_letter.clone(), Some(impl_trait_ty)));
let new_type_fn = type_fn
.replace_descendant::<ast::Type>(type_impl_trait.into(), make::ty(&generic_letter))
.with_generic_params(generic_param_list);
edit.replace_ast(type_fn.clone(), new_type_fn); edit.replace_ast(type_fn.clone(), new_type_fn);
}, },
) )
@ -48,7 +50,7 @@ mod tests {
use crate::tests::check_assist; use crate::tests::check_assist;
#[test] #[test]
fn replace_with_generic_params() { fn replace_impl_trait_with_generic_params() {
check_assist( check_assist(
replace_impl_trait_with_generic, replace_impl_trait_with_generic,
r#" r#"
@ -59,4 +61,96 @@ mod tests {
"#, "#,
); );
} }
#[test]
fn replace_impl_trait_without_generic_params() {
check_assist(
replace_impl_trait_with_generic,
r#"
fn foo(bar: <|>impl Bar) {}
"#,
r#"
fn foo<B: Bar>(bar: B) {}
"#,
);
}
#[test]
fn replace_two_impl_trait_with_generic_params() {
check_assist(
replace_impl_trait_with_generic,
r#"
fn foo<G>(foo: impl Foo, bar: <|>impl Bar) {}
"#,
r#"
fn foo<G, B: Bar>(foo: impl Foo, bar: B) {}
"#,
);
}
#[test]
fn replace_impl_trait_with_empty_generic_params() {
check_assist(
replace_impl_trait_with_generic,
r#"
fn foo<>(bar: <|>impl Bar) {}
"#,
r#"
fn foo<B: Bar>(bar: B) {}
"#,
);
}
#[test]
fn replace_impl_trait_with_empty_multiline_generic_params() {
// FIXME: It would be more correct to place the generic parameter
// on the next line after the left angle.
check_assist(
replace_impl_trait_with_generic,
r#"
fn foo<
>(bar: <|>impl Bar) {}
"#,
r#"
fn foo<B: Bar,
>(bar: B) {}
"#,
);
}
#[test]
#[ignore = "This case is very rare but there is no simple solutions to fix it."]
fn replace_impl_trait_with_exist_generic_letter() {
check_assist(
replace_impl_trait_with_generic,
r#"
fn foo<B>(bar: <|>impl Bar) {}
"#,
r#"
fn foo<B, C: Bar>(bar: C) {}
"#,
);
}
#[test]
fn replace_impl_trait_with_multiline_generic_params() {
check_assist(
replace_impl_trait_with_generic,
r#"
fn foo<
G: Foo,
F,
H,
>(bar: <|>impl Bar) {}
"#,
r#"
fn foo<
G: Foo,
F,
H,
B: Bar,
>(bar: B) {}
"#,
);
}
} }

View file

@ -13,7 +13,7 @@ use crate::{
ast::{ ast::{
self, self,
make::{self, tokens}, make::{self, tokens},
AstNode, TypeBoundsOwner, AstNode, GenericParamsOwner, NameOwner, TypeBoundsOwner,
}, },
AstToken, Direction, InsertPosition, SmolStr, SyntaxElement, SyntaxKind, AstToken, Direction, InsertPosition, SmolStr, SyntaxElement, SyntaxKind,
SyntaxKind::{ATTR, COMMENT, WHITESPACE}, SyntaxKind::{ATTR, COMMENT, WHITESPACE},
@ -46,6 +46,19 @@ impl ast::Fn {
to_insert.push(body.syntax().clone().into()); to_insert.push(body.syntax().clone().into());
self.replace_children(single_node(old_body_or_semi), to_insert) self.replace_children(single_node(old_body_or_semi), to_insert)
} }
#[must_use]
pub fn with_generic_params(&self, generic_args: ast::GenericParamList) -> ast::Fn {
if let Some(old) = self.generic_param_list() {
return self.replace_descendant(old, generic_args);
}
let anchor = self.name().expect("The function must have a name").syntax().clone();
let mut to_insert: ArrayVec<[SyntaxElement; 1]> = ArrayVec::new();
to_insert.push(generic_args.syntax().clone().into());
self.insert_children(InsertPosition::After(anchor.into()), to_insert)
}
} }
fn make_multiline<N>(node: N) -> N fn make_multiline<N>(node: N) -> N
@ -461,14 +474,17 @@ impl ast::MatchArmList {
impl ast::GenericParamList { impl ast::GenericParamList {
#[must_use] #[must_use]
pub fn append_params(&self, params: impl IntoIterator<Item = ast::GenericParam>) -> Self { pub fn append_params(
&self,
params: impl IntoIterator<Item = ast::GenericParam>,
) -> ast::GenericParamList {
let mut res = self.clone(); let mut res = self.clone();
params.into_iter().for_each(|it| res = res.append_param(it)); params.into_iter().for_each(|it| res = res.append_param(it));
res res
} }
#[must_use] #[must_use]
pub fn append_param(&self, item: ast::GenericParam) -> Self { pub fn append_param(&self, item: ast::GenericParam) -> ast::GenericParamList {
let is_multiline = self.syntax().text().contains_char('\n'); let is_multiline = self.syntax().text().contains_char('\n');
let ws; let ws;
let space = if is_multiline { let space = if is_multiline {
@ -482,7 +498,9 @@ impl ast::GenericParamList {
}; };
let mut to_insert: ArrayVec<[SyntaxElement; 4]> = ArrayVec::new(); let mut to_insert: ArrayVec<[SyntaxElement; 4]> = ArrayVec::new();
to_insert.push(space.into()); if self.generic_params().next().is_some() {
to_insert.push(space.into());
}
to_insert.push(item.syntax().clone().into()); to_insert.push(item.syntax().clone().into());
to_insert.push(make::token(T![,]).into()); to_insert.push(make::token(T![,]).into());