rust-analyzer/crates/assists/src/handlers/reorder_fields.rs

223 lines
5.6 KiB
Rust
Raw Normal View History

2020-06-29 15:07:52 +00:00
use itertools::Itertools;
use rustc_hash::FxHashMap;
2020-04-09 22:35:43 +00:00
use hir::{Adt, ModuleDef, PathResolution, Semantics, Struct};
2020-08-13 14:39:16 +00:00
use ide_db::RootDatabase;
2020-08-12 16:26:51 +00:00
use syntax::{algo, ast, match_ast, AstNode, SyntaxKind, SyntaxKind::*, SyntaxNode};
2020-06-28 22:36:05 +00:00
use crate::{AssistContext, AssistId, AssistKind, Assists};
2020-04-09 22:35:43 +00:00
2020-04-11 17:39:10 +00:00
// Assist: reorder_fields
//
// Reorder the fields of record literals and record patterns in the same order as in
// the definition.
//
// ```
// struct Foo {foo: i32, bar: i32};
// const test: Foo = <|>Foo {bar: 0, foo: 1}
// ```
// ->
// ```
// struct Foo {foo: i32, bar: i32};
2020-04-11 18:32:48 +00:00
// const test: Foo = Foo {foo: 1, bar: 0}
2020-04-11 17:39:10 +00:00
// ```
//
pub(crate) fn reorder_fields(acc: &mut Assists, ctx: &AssistContext) -> Option<()> {
2020-07-30 14:21:30 +00:00
reorder::<ast::RecordExpr>(acc, ctx).or_else(|| reorder::<ast::RecordPat>(acc, ctx))
2020-04-09 22:35:43 +00:00
}
fn reorder<R: AstNode>(acc: &mut Assists, ctx: &AssistContext) -> Option<()> {
let record = ctx.find_node_at_offset::<R>()?;
let path = record.syntax().children().find_map(ast::Path::cast)?;
2020-04-09 22:35:43 +00:00
let ranks = compute_fields_ranks(&path, &ctx)?;
2020-04-09 22:35:43 +00:00
let fields = get_fields(&record.syntax());
let sorted_fields = sorted_by_rank(&fields, |node| {
*ranks.get(&get_field_name(node)).unwrap_or(&usize::max_value())
});
2020-04-09 22:35:43 +00:00
if sorted_fields == fields {
return None;
}
let target = record.syntax().text_range();
2020-06-28 22:36:05 +00:00
acc.add(
2020-07-02 21:48:35 +00:00
AssistId("reorder_fields", AssistKind::RefactorRewrite),
2020-06-28 22:36:05 +00:00
"Reorder record fields",
target,
|edit| {
let mut rewriter = algo::SyntaxRewriter::default();
2020-06-28 22:36:05 +00:00
for (old, new) in fields.iter().zip(&sorted_fields) {
rewriter.replace(old, new);
2020-06-28 22:36:05 +00:00
}
edit.rewrite(rewriter);
2020-06-28 22:36:05 +00:00
},
)
2020-04-09 22:35:43 +00:00
}
fn get_fields_kind(node: &SyntaxNode) -> Vec<SyntaxKind> {
match node.kind() {
2020-07-30 14:21:30 +00:00
RECORD_EXPR => vec![RECORD_EXPR_FIELD],
2020-07-31 18:09:09 +00:00
RECORD_PAT => vec![RECORD_PAT_FIELD, IDENT_PAT],
_ => vec![],
}
}
fn get_field_name(node: &SyntaxNode) -> String {
let res = match_ast! {
match node {
2020-07-30 14:21:30 +00:00
ast::RecordExprField(field) => field.field_name().map(|it| it.to_string()),
2020-07-31 17:54:16 +00:00
ast::RecordPatField(field) => field.field_name().map(|it| it.to_string()),
_ => None,
}
};
res.unwrap_or_default()
}
fn get_fields(record: &SyntaxNode) -> Vec<SyntaxNode> {
let kinds = get_fields_kind(record);
record.children().flat_map(|n| n.children()).filter(|n| kinds.contains(&n.kind())).collect()
2020-04-09 22:35:43 +00:00
}
fn sorted_by_rank(
fields: &[SyntaxNode],
get_rank: impl Fn(&SyntaxNode) -> usize,
) -> Vec<SyntaxNode> {
2020-04-09 22:35:43 +00:00
fields.iter().cloned().sorted_by_key(get_rank).collect()
}
fn struct_definition(path: &ast::Path, sema: &Semantics<RootDatabase>) -> Option<Struct> {
match sema.resolve_path(path) {
Some(PathResolution::Def(ModuleDef::Adt(Adt::Struct(s)))) => Some(s),
_ => None,
}
}
2020-06-29 15:07:52 +00:00
fn compute_fields_ranks(path: &ast::Path, ctx: &AssistContext) -> Option<FxHashMap<String, usize>> {
2020-04-09 22:35:43 +00:00
Some(
struct_definition(path, &ctx.sema)?
2020-07-01 07:14:23 +00:00
.fields(ctx.db())
2020-04-09 22:35:43 +00:00
.iter()
.enumerate()
2020-07-01 07:14:23 +00:00
.map(|(idx, field)| (field.name(ctx.db()).to_string(), idx))
2020-04-09 22:35:43 +00:00
.collect(),
)
}
#[cfg(test)]
mod tests {
2020-05-06 08:16:55 +00:00
use crate::tests::{check_assist, check_assist_not_applicable};
2020-04-09 22:35:43 +00:00
use super::*;
#[test]
fn not_applicable_if_sorted() {
check_assist_not_applicable(
reorder_fields,
r#"
struct Foo {
foo: i32,
bar: i32,
}
const test: Foo = <|>Foo { foo: 0, bar: 0 };
"#,
)
}
#[test]
fn trivial_empty_fields() {
check_assist_not_applicable(
reorder_fields,
r#"
struct Foo {};
const test: Foo = <|>Foo {}
"#,
)
}
#[test]
fn reorder_struct_fields() {
check_assist(
reorder_fields,
r#"
struct Foo {foo: i32, bar: i32};
const test: Foo = <|>Foo {bar: 0, foo: 1}
"#,
r#"
struct Foo {foo: i32, bar: i32};
const test: Foo = Foo {foo: 1, bar: 0}
2020-04-09 22:35:43 +00:00
"#,
)
}
#[test]
fn reorder_struct_pattern() {
check_assist(
reorder_fields,
r#"
struct Foo { foo: i64, bar: i64, baz: i64 }
fn f(f: Foo) -> {
match f {
<|>Foo { baz: 0, ref mut bar, .. } => (),
_ => ()
}
}
"#,
r#"
struct Foo { foo: i64, bar: i64, baz: i64 }
fn f(f: Foo) -> {
match f {
Foo { ref mut bar, baz: 0, .. } => (),
2020-04-09 22:35:43 +00:00
_ => ()
}
}
"#,
)
}
#[test]
fn reorder_with_extra_field() {
check_assist(
reorder_fields,
r#"
struct Foo {
foo: String,
bar: String,
}
impl Foo {
fn new() -> Foo {
let foo = String::new();
<|>Foo {
bar: foo.clone(),
extra: "Extra field",
foo,
}
}
}
"#,
r#"
struct Foo {
foo: String,
bar: String,
}
impl Foo {
fn new() -> Foo {
let foo = String::new();
Foo {
2020-04-09 22:35:43 +00:00
foo,
bar: foo.clone(),
extra: "Extra field",
}
}
}
"#,
)
}
}