2019-09-04 11:48:45 -05:00
|
|
|
use ra_syntax::{
|
2020-03-18 10:41:24 -05:00
|
|
|
ast::{self, edit::AstNodeEdit, make, AstNode, NameOwner, TypeBoundsOwner},
|
2020-03-18 14:51:47 -05:00
|
|
|
match_ast,
|
2019-09-04 11:48:45 -05:00
|
|
|
SyntaxKind::*,
|
2020-04-10 10:06:57 -05:00
|
|
|
T,
|
2019-09-04 11:48:45 -05:00
|
|
|
};
|
|
|
|
|
2019-09-30 02:05:12 -05:00
|
|
|
use crate::{Assist, AssistCtx, AssistId};
|
2019-09-04 11:48:45 -05:00
|
|
|
|
2019-10-27 03:26:46 -05:00
|
|
|
// Assist: move_bounds_to_where_clause
|
|
|
|
//
|
|
|
|
// Moves inline type bounds to a where clause.
|
|
|
|
//
|
|
|
|
// ```
|
|
|
|
// fn apply<T, U, <|>F: FnOnce(T) -> U>(f: F, x: T) -> U {
|
|
|
|
// f(x)
|
|
|
|
// }
|
|
|
|
// ```
|
|
|
|
// ->
|
|
|
|
// ```
|
|
|
|
// fn apply<T, U, F>(f: F, x: T) -> U where F: FnOnce(T) -> U {
|
|
|
|
// f(x)
|
|
|
|
// }
|
|
|
|
// ```
|
2020-02-06 09:58:57 -06:00
|
|
|
pub(crate) fn move_bounds_to_where_clause(ctx: AssistCtx) -> Option<Assist> {
|
2019-10-27 03:48:40 -05:00
|
|
|
let type_param_list = ctx.find_node_at_offset::<ast::TypeParamList>()?;
|
2019-09-04 11:48:45 -05: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()?;
|
2019-10-11 14:55:45 -05:00
|
|
|
if parent.children_with_tokens().any(|it| it.kind() == WHERE_CLAUSE) {
|
2019-09-04 11:48:45 -05:00
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
2020-03-18 14:51:47 -05:00
|
|
|
let anchor = match_ast! {
|
|
|
|
match parent {
|
|
|
|
ast::FnDef(it) => it.body()?.syntax().clone().into(),
|
|
|
|
ast::TraitDef(it) => it.item_list()?.syntax().clone().into(),
|
|
|
|
ast::ImplDef(it) => it.item_list()?.syntax().clone().into(),
|
|
|
|
ast::EnumDef(it) => it.variant_list()?.syntax().clone().into(),
|
|
|
|
ast::StructDef(it) => {
|
|
|
|
it.syntax().children_with_tokens()
|
2020-04-10 10:06:57 -05:00
|
|
|
.find(|it| it.kind() == RECORD_FIELD_DEF_LIST || it.kind() == T![;])?
|
2020-03-18 14:51:47 -05:00
|
|
|
},
|
|
|
|
_ => return None
|
|
|
|
}
|
2019-09-04 11:48:45 -05:00
|
|
|
};
|
|
|
|
|
2020-01-14 11:32:26 -06:00
|
|
|
ctx.add_assist(AssistId("move_bounds_to_where_clause"), "Move to where clause", |edit| {
|
2019-10-27 09:35:37 -05:00
|
|
|
let new_params = type_param_list
|
|
|
|
.type_params()
|
|
|
|
.filter(|it| it.type_bound_list().is_some())
|
|
|
|
.map(|type_param| {
|
|
|
|
let without_bounds = type_param.remove_bounds();
|
|
|
|
(type_param, without_bounds)
|
|
|
|
});
|
2019-09-04 11:48:45 -05:00
|
|
|
|
2020-03-18 10:41:24 -05:00
|
|
|
let new_type_param_list = type_param_list.replace_descendants(new_params);
|
2019-10-27 09:35:37 -05:00
|
|
|
edit.replace_ast(type_param_list.clone(), new_type_param_list);
|
2019-09-04 11:48:45 -05:00
|
|
|
|
2019-10-27 09:35:37 -05:00
|
|
|
let where_clause = {
|
|
|
|
let predicates = type_param_list.type_params().filter_map(build_predicate);
|
|
|
|
make::where_clause(predicates)
|
|
|
|
};
|
2019-09-04 11:48:45 -05:00
|
|
|
|
2019-10-27 09:35:37 -05:00
|
|
|
let to_insert = match anchor.prev_sibling_or_token() {
|
|
|
|
Some(ref elem) if elem.kind() == WHITESPACE => format!("{} ", where_clause.syntax()),
|
|
|
|
_ => format!(" {}", where_clause.syntax()),
|
|
|
|
};
|
|
|
|
edit.insert(anchor.text_range().start(), to_insert);
|
|
|
|
edit.target(type_param_list.syntax().text_range());
|
|
|
|
})
|
2019-09-04 11:48:45 -05:00
|
|
|
}
|
|
|
|
|
2019-09-25 09:57:12 -05:00
|
|
|
fn build_predicate(param: ast::TypeParam) -> Option<ast::WherePred> {
|
2020-02-29 04:55:36 -06:00
|
|
|
let path = {
|
|
|
|
let name_ref = make::name_ref(¶m.name()?.syntax().to_string());
|
|
|
|
let segment = make::path_segment(name_ref);
|
2020-02-29 06:50:47 -06:00
|
|
|
make::path_unqualified(segment)
|
2020-02-29 04:55:36 -06:00
|
|
|
};
|
2019-09-26 04:18:26 -05:00
|
|
|
let predicate = make::where_pred(path, param.type_bound_list()?.bounds());
|
2019-09-04 11:48:45 -05:00
|
|
|
Some(predicate)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
use crate::helpers::check_assist;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn move_bounds_to_where_clause_fn() {
|
|
|
|
check_assist(
|
|
|
|
move_bounds_to_where_clause,
|
|
|
|
r#"
|
|
|
|
fn foo<T: u32, <|>F: FnOnce(T) -> T>() {}
|
|
|
|
"#,
|
|
|
|
r#"
|
|
|
|
fn foo<T, <|>F>() where T: u32, F: FnOnce(T) -> T {}
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn move_bounds_to_where_clause_impl() {
|
|
|
|
check_assist(
|
|
|
|
move_bounds_to_where_clause,
|
|
|
|
r#"
|
|
|
|
impl<U: u32, <|>T> A<U, T> {}
|
|
|
|
"#,
|
|
|
|
r#"
|
|
|
|
impl<U, <|>T> A<U, T> where U: u32 {}
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn move_bounds_to_where_clause_struct() {
|
|
|
|
check_assist(
|
|
|
|
move_bounds_to_where_clause,
|
|
|
|
r#"
|
|
|
|
struct A<<|>T: Iterator<Item = u32>> {}
|
|
|
|
"#,
|
|
|
|
r#"
|
|
|
|
struct A<<|>T> where T: Iterator<Item = u32> {}
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn move_bounds_to_where_clause_tuple_struct() {
|
|
|
|
check_assist(
|
|
|
|
move_bounds_to_where_clause,
|
|
|
|
r#"
|
|
|
|
struct Pair<<|>T: u32>(T, T);
|
|
|
|
"#,
|
|
|
|
r#"
|
|
|
|
struct Pair<<|>T>(T, T) where T: u32;
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|