2020-08-12 16:26:51 +00:00
|
|
|
use syntax::{
|
2021-01-30 15:19:21 +00:00
|
|
|
ast::{self, edit_in_place::GenericParamsOwnerEdit, make, AstNode, NameOwner, TypeBoundsOwner},
|
2020-03-18 19:51:47 +00:00
|
|
|
match_ast,
|
2019-09-04 16:48:45 +00:00
|
|
|
};
|
|
|
|
|
2020-06-28 22:36:05 +00:00
|
|
|
use crate::{AssistContext, AssistId, AssistKind, Assists};
|
2019-09-04 16:48:45 +00:00
|
|
|
|
2019-10-27 08:26:46 +00:00
|
|
|
// Assist: move_bounds_to_where_clause
|
|
|
|
//
|
|
|
|
// Moves inline type bounds to a where clause.
|
|
|
|
//
|
|
|
|
// ```
|
2021-01-06 20:15:48 +00:00
|
|
|
// fn apply<T, U, $0F: FnOnce(T) -> U>(f: F, x: T) -> U {
|
2019-10-27 08:26:46 +00:00
|
|
|
// f(x)
|
|
|
|
// }
|
|
|
|
// ```
|
|
|
|
// ->
|
|
|
|
// ```
|
|
|
|
// fn apply<T, U, F>(f: F, x: T) -> U where F: FnOnce(T) -> U {
|
|
|
|
// f(x)
|
|
|
|
// }
|
|
|
|
// ```
|
2020-05-06 16:45:35 +00:00
|
|
|
pub(crate) fn move_bounds_to_where_clause(acc: &mut Assists, ctx: &AssistContext) -> Option<()> {
|
2021-01-30 15:19:21 +00:00
|
|
|
let type_param_list = ctx.find_node_at_offset::<ast::GenericParamList>()?.clone_for_update();
|
2019-09-04 16:48:45 +00:00
|
|
|
|
|
|
|
let mut type_params = type_param_list.type_params();
|
|
|
|
if type_params.all(|p| p.type_bound_list().is_none()) {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
|
|
|
let parent = type_param_list.syntax().parent()?;
|
2021-01-30 15:19:21 +00:00
|
|
|
let original_parent_range = parent.text_range();
|
2019-09-04 16:48:45 +00:00
|
|
|
|
2020-05-06 10:51:28 +00:00
|
|
|
let target = type_param_list.syntax().text_range();
|
2020-06-28 22:36:05 +00:00
|
|
|
acc.add(
|
2020-07-02 21:48:35 +00:00
|
|
|
AssistId("move_bounds_to_where_clause", AssistKind::RefactorRewrite),
|
2020-06-28 22:36:05 +00:00
|
|
|
"Move to where clause",
|
|
|
|
target,
|
|
|
|
|edit| {
|
2021-01-30 15:19:21 +00:00
|
|
|
let where_clause: ast::WhereClause = match_ast! {
|
|
|
|
match parent {
|
|
|
|
ast::Fn(it) => it.get_or_create_where_clause(),
|
2021-03-16 19:28:04 +00:00
|
|
|
ast::Trait(it) => it.get_or_create_where_clause(),
|
2021-01-30 15:19:21 +00:00
|
|
|
ast::Impl(it) => it.get_or_create_where_clause(),
|
2021-03-16 19:28:04 +00:00
|
|
|
ast::Enum(it) => it.get_or_create_where_clause(),
|
2021-01-30 15:19:21 +00:00
|
|
|
ast::Struct(it) => it.get_or_create_where_clause(),
|
|
|
|
_ => return,
|
|
|
|
}
|
2020-06-28 22:36:05 +00:00
|
|
|
};
|
2019-09-04 16:48:45 +00:00
|
|
|
|
2021-01-30 15:19:21 +00:00
|
|
|
for type_param in type_param_list.type_params() {
|
|
|
|
if let Some(tbl) = type_param.type_bound_list() {
|
|
|
|
if let Some(predicate) = build_predicate(type_param.clone()) {
|
|
|
|
where_clause.add_predicate(predicate.clone_for_update())
|
|
|
|
}
|
|
|
|
tbl.remove()
|
2020-06-28 22:36:05 +00:00
|
|
|
}
|
2021-01-30 15:19:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
edit.replace(original_parent_range, parent.to_string())
|
2020-06-28 22:36:05 +00:00
|
|
|
},
|
|
|
|
)
|
2019-09-04 16:48:45 +00:00
|
|
|
}
|
|
|
|
|
2019-09-25 14:57:12 +00:00
|
|
|
fn build_predicate(param: ast::TypeParam) -> Option<ast::WherePred> {
|
2020-02-29 10:55:36 +00:00
|
|
|
let path = {
|
|
|
|
let name_ref = make::name_ref(¶m.name()?.syntax().to_string());
|
|
|
|
let segment = make::path_segment(name_ref);
|
2020-02-29 12:50:47 +00:00
|
|
|
make::path_unqualified(segment)
|
2020-02-29 10:55:36 +00:00
|
|
|
};
|
2019-09-26 09:18:26 +00:00
|
|
|
let predicate = make::where_pred(path, param.type_bound_list()?.bounds());
|
2019-09-04 16:48:45 +00:00
|
|
|
Some(predicate)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
|
2020-05-06 08:16:55 +00:00
|
|
|
use crate::tests::check_assist;
|
2019-09-04 16:48:45 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn move_bounds_to_where_clause_fn() {
|
|
|
|
check_assist(
|
|
|
|
move_bounds_to_where_clause,
|
2021-03-16 19:28:04 +00:00
|
|
|
r#"fn foo<T: u32, $0F: FnOnce(T) -> T>() {}"#,
|
|
|
|
r#"fn foo<T, F>() where T: u32, F: FnOnce(T) -> T {}"#,
|
2019-09-04 16:48:45 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn move_bounds_to_where_clause_impl() {
|
|
|
|
check_assist(
|
|
|
|
move_bounds_to_where_clause,
|
2021-03-16 19:28:04 +00:00
|
|
|
r#"impl<U: u32, $0T> A<U, T> {}"#,
|
|
|
|
r#"impl<U, T> A<U, T> where U: u32 {}"#,
|
2019-09-04 16:48:45 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn move_bounds_to_where_clause_struct() {
|
|
|
|
check_assist(
|
|
|
|
move_bounds_to_where_clause,
|
2021-03-16 19:28:04 +00:00
|
|
|
r#"struct A<$0T: Iterator<Item = u32>> {}"#,
|
|
|
|
r#"struct A<T> where T: Iterator<Item = u32> {}"#,
|
2019-09-04 16:48:45 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn move_bounds_to_where_clause_tuple_struct() {
|
|
|
|
check_assist(
|
|
|
|
move_bounds_to_where_clause,
|
2021-03-16 19:28:04 +00:00
|
|
|
r#"struct Pair<$0T: u32>(T, T);"#,
|
|
|
|
r#"struct Pair<T>(T, T) where T: u32;"#,
|
2019-09-04 16:48:45 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|