rust-analyzer/crates/ide-assists/src/handlers/introduce_named_generic.rs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

179 lines
4.7 KiB
Rust
Raw Normal View History

use syntax::{
ast::{self, edit_in_place::GenericParamsOwnerEdit, make, AstNode, HasGenericParams},
ted,
};
2021-05-09 14:58:03 +00:00
use crate::{utils::suggest_name, AssistContext, AssistId, AssistKind, Assists};
2021-09-20 21:53:05 +00:00
// Assist: introduce_named_generic
//
// Replaces `impl Trait` function argument with the named generic.
2020-09-04 12:24:36 +00:00
//
// ```
2021-01-06 20:15:48 +00:00
// fn foo(bar: $0impl Bar) {}
2020-09-04 12:24:36 +00:00
// ```
// ->
// ```
// fn foo<$0B: Bar>(bar: B) {}
2020-09-04 12:24:36 +00:00
// ```
2022-07-20 13:02:08 +00:00
pub(crate) fn introduce_named_generic(acc: &mut Assists, ctx: &AssistContext<'_>) -> Option<()> {
2021-05-09 14:58:03 +00:00
let impl_trait_type = ctx.find_node_at_offset::<ast::ImplTraitType>()?;
2024-01-18 12:59:49 +00:00
let param = impl_trait_type.syntax().ancestors().find_map(ast::Param::cast)?;
2021-05-09 14:58:03 +00:00
let fn_ = param.syntax().ancestors().find_map(ast::Fn::cast)?;
2021-05-09 14:58:03 +00:00
let type_bound_list = impl_trait_type.type_bound_list()?;
2021-05-09 14:58:03 +00:00
let target = fn_.syntax().text_range();
acc.add(
2021-09-20 21:53:05 +00:00
AssistId("introduce_named_generic", AssistKind::RefactorRewrite),
"Replace impl trait with generic",
target,
|edit| {
2021-05-16 11:18:49 +00:00
let impl_trait_type = edit.make_mut(impl_trait_type);
let fn_ = edit.make_mut(fn_);
let fn_generic_param_list = fn_.get_or_create_generic_param_list();
let type_param_name =
suggest_name::for_impl_trait_as_generic(&impl_trait_type, &fn_generic_param_list);
2020-09-03 11:46:28 +00:00
2021-05-09 16:11:42 +00:00
let type_param = make::type_param(make::name(&type_param_name), Some(type_bound_list))
.clone_for_update();
let new_ty = make::ty(&type_param_name).clone_for_update();
2020-09-03 11:46:28 +00:00
ted::replace(impl_trait_type.syntax(), new_ty.syntax());
fn_generic_param_list.add_generic_param(type_param.into());
if let Some(cap) = ctx.config.snippet_cap {
if let Some(generic_param) =
fn_.generic_param_list().and_then(|it| it.generic_params().last())
{
edit.add_tabstop_before(cap, generic_param);
}
}
},
)
}
#[cfg(test)]
mod tests {
use super::*;
use crate::tests::check_assist;
#[test]
2021-09-20 21:53:05 +00:00
fn introduce_named_generic_params() {
check_assist(
2021-09-20 21:53:05 +00:00
introduce_named_generic,
2021-05-09 14:41:25 +00:00
r#"fn foo<G>(bar: $0impl Bar) {}"#,
r#"fn foo<G, $0B: Bar>(bar: B) {}"#,
);
}
2020-09-03 11:46:28 +00:00
#[test]
fn replace_impl_trait_without_generic_params() {
check_assist(
2021-09-20 21:53:05 +00:00
introduce_named_generic,
2021-05-09 14:41:25 +00:00
r#"fn foo(bar: $0impl Bar) {}"#,
r#"fn foo<$0B: Bar>(bar: B) {}"#,
2020-09-03 11:46:28 +00:00
);
}
#[test]
fn replace_two_impl_trait_with_generic_params() {
check_assist(
2021-09-20 21:53:05 +00:00
introduce_named_generic,
2021-05-09 14:41:25 +00:00
r#"fn foo<G>(foo: impl Foo, bar: $0impl Bar) {}"#,
r#"fn foo<G, $0B: Bar>(foo: impl Foo, bar: B) {}"#,
2020-09-03 11:46:28 +00:00
);
}
#[test]
fn replace_impl_trait_with_empty_generic_params() {
check_assist(
2021-09-20 21:53:05 +00:00
introduce_named_generic,
2021-05-09 14:41:25 +00:00
r#"fn foo<>(bar: $0impl Bar) {}"#,
r#"fn foo<$0B: Bar>(bar: B) {}"#,
2020-09-03 11:46:28 +00:00
);
}
#[test]
fn replace_impl_trait_with_empty_multiline_generic_params() {
check_assist(
2021-09-20 21:53:05 +00:00
introduce_named_generic,
2020-09-03 11:46:28 +00:00
r#"
2021-05-09 14:41:25 +00:00
fn foo<
>(bar: $0impl Bar) {}
"#,
2020-09-03 11:46:28 +00:00
r#"
fn foo<$0B: Bar
2021-05-09 14:41:25 +00:00
>(bar: B) {}
"#,
2020-09-03 11:46:28 +00:00
);
}
#[test]
fn replace_impl_trait_with_exist_generic_letter() {
check_assist(
2021-09-20 21:53:05 +00:00
introduce_named_generic,
2021-05-09 14:41:25 +00:00
r#"fn foo<B>(bar: $0impl Bar) {}"#,
r#"fn foo<B, $0B0: Bar>(bar: B0) {}"#,
);
}
#[test]
fn replace_impl_trait_with_more_exist_generic_letter() {
check_assist(
introduce_named_generic,
r#"fn foo<B, B0, B1, B3>(bar: $0impl Bar) {}"#,
r#"fn foo<B, B0, B1, B3, $0B2: Bar>(bar: B2) {}"#,
2020-09-03 11:46:28 +00:00
);
}
#[test]
fn replace_impl_trait_with_multiline_generic_params() {
check_assist(
2021-09-20 21:53:05 +00:00
introduce_named_generic,
2020-09-03 11:46:28 +00:00
r#"
2021-05-09 14:41:25 +00:00
fn foo<
G: Foo,
F,
H,
>(bar: $0impl Bar) {}
"#,
r#"
fn foo<
G: Foo,
F,
H, $0B: Bar,
2021-05-09 14:41:25 +00:00
>(bar: B) {}
"#,
2020-09-03 11:46:28 +00:00
);
}
2020-09-04 14:55:27 +00:00
#[test]
fn replace_impl_trait_multiple() {
check_assist(
2021-09-20 21:53:05 +00:00
introduce_named_generic,
2021-05-09 14:41:25 +00:00
r#"fn foo(bar: $0impl Foo + Bar) {}"#,
r#"fn foo<$0F: Foo + Bar>(bar: F) {}"#,
2020-09-04 14:55:27 +00:00
);
}
#[test]
fn replace_impl_with_mut() {
check_assist(
introduce_named_generic,
r#"fn f(iter: &mut $0impl Iterator<Item = i32>) {}"#,
r#"fn f<$0I: Iterator<Item = i32>>(iter: &mut I) {}"#,
);
}
#[test]
fn replace_impl_inside() {
check_assist(
introduce_named_generic,
r#"fn f(x: &mut Vec<$0impl Iterator<Item = i32>>) {}"#,
r#"fn f<$0I: Iterator<Item = i32>>(x: &mut Vec<I>) {}"#,
);
}
}