Skip to content

Commit 3dc6eed

Browse files
authored
Merge pull request rust-lang#2816 from moxian/enum-discrim
Add option to vertically align enum discriminants.
2 parents 86fff9e + 65ae0b9 commit 3dc6eed

File tree

5 files changed

+146
-3
lines changed

5 files changed

+146
-3
lines changed

Configurations.md

+47
Original file line numberDiff line numberDiff line change
@@ -885,6 +885,53 @@ impl Lorem {
885885
See also [`brace_style`](#brace_style), [`control_brace_style`](#control_brace_style).
886886

887887

888+
## `enum_discrim_align_threshold`
889+
890+
The maximum length of enum variant having discriminant, that gets vertically aligned with others.
891+
Variants without discriminants would be ignored for the purpose of alignment.
892+
893+
Note that this is not how much whitespace is inserted, but instead the longest variant name that
894+
doesn't get ignored when aligning.
895+
896+
- **Default value** : 0
897+
- **Possible values**: any positive integer
898+
- **Stable**: No
899+
900+
#### `0` (default):
901+
902+
```rust
903+
enum Bar {
904+
A = 0,
905+
Bb = 1,
906+
RandomLongVariantGoesHere = 10,
907+
Ccc = 71,
908+
}
909+
910+
enum Bar {
911+
VeryLongVariantNameHereA = 0,
912+
VeryLongVariantNameHereBb = 1,
913+
VeryLongVariantNameHereCcc = 2,
914+
}
915+
```
916+
917+
#### `20`:
918+
919+
```rust
920+
enum Foo {
921+
A = 0,
922+
Bb = 1,
923+
RandomLongVariantGoesHere = 10,
924+
Ccc = 2,
925+
}
926+
927+
enum Bar {
928+
VeryLongVariantNameHereA = 0,
929+
VeryLongVariantNameHereBb = 1,
930+
VeryLongVariantNameHereCcc = 2,
931+
}
932+
```
933+
934+
888935
## `fn_single_line`
889936

890937
Put single-expression functions on a single line

src/config/mod.rs

+2
Original file line numberDiff line numberDiff line change
@@ -89,6 +89,8 @@ create_config! {
8989
combine_control_expr: bool, true, false, "Combine control expressions with function calls";
9090
struct_field_align_threshold: usize, 0, false, "Align struct fields if their diffs fits within \
9191
threshold";
92+
enum_discrim_align_threshold: usize, 0, false,
93+
"Align enum variants discrims, if their diffs fit within threshold";
9294
match_arm_blocks: bool, true, false, "Wrap the body of arms in blocks when it does not fit on \
9395
the same line with the pattern of arms";
9496
force_multiline_blocks: bool, false, false,

src/items.rs

+29-3
Original file line numberDiff line numberDiff line change
@@ -500,6 +500,23 @@ impl<'a> FmtVisitor<'a> {
500500
let original_offset = self.block_indent;
501501
self.block_indent = self.block_indent.block_indent(self.config);
502502

503+
// If enum variants have discriminants, try to vertically align those,
504+
// provided the discrims are not shifted too much to the right
505+
let align_threshold: usize = self.config.enum_discrim_align_threshold();
506+
let discr_ident_lens: Vec<usize> = enum_def
507+
.variants
508+
.iter()
509+
.filter(|var| var.node.disr_expr.is_some())
510+
.map(|var| rewrite_ident(&self.get_context(), var.node.ident).len())
511+
.collect();
512+
// cut the list at the point of longest discrim shorter than the threshold
513+
// All of the discrims under the threshold will get padded, and all above - left as is.
514+
let pad_discrim_ident_to = *discr_ident_lens
515+
.iter()
516+
.filter(|&l| *l <= align_threshold)
517+
.max()
518+
.unwrap_or(&0);
519+
503520
let itemize_list_with = |one_line_width: usize| {
504521
itemize_list(
505522
self.snippet_provider,
@@ -514,7 +531,7 @@ impl<'a> FmtVisitor<'a> {
514531
}
515532
},
516533
|f| f.span.hi(),
517-
|f| self.format_variant(f, one_line_width),
534+
|f| self.format_variant(f, one_line_width, pad_discrim_ident_to),
518535
body_lo,
519536
body_hi,
520537
false,
@@ -543,7 +560,12 @@ impl<'a> FmtVisitor<'a> {
543560
}
544561

545562
// Variant of an enum.
546-
fn format_variant(&self, field: &ast::Variant, one_line_width: usize) -> Option<String> {
563+
fn format_variant(
564+
&self,
565+
field: &ast::Variant,
566+
one_line_width: usize,
567+
pad_discrim_ident_to: usize,
568+
) -> Option<String> {
547569
if contains_skip(&field.node.attrs) {
548570
let lo = field.node.attrs[0].span.lo();
549571
let span = mk_sp(lo, field.span.hi());
@@ -570,7 +592,11 @@ impl<'a> FmtVisitor<'a> {
570592
)?,
571593
ast::VariantData::Unit(..) => {
572594
if let Some(ref expr) = field.node.disr_expr {
573-
let lhs = format!("{} =", rewrite_ident(&context, field.node.ident));
595+
let lhs = format!(
596+
"{:1$} =",
597+
rewrite_ident(&context, field.node.ident),
598+
pad_discrim_ident_to
599+
);
574600
rewrite_assign_rhs(&context, lhs, &*expr.value, shape)?
575601
} else {
576602
rewrite_ident(&context, field.node.ident).to_owned()
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
// rustfmt-enum_discrim_align_threshold: 40
2+
3+
enum Standard {
4+
A = 1,
5+
Bcdef = 2,
6+
}
7+
8+
enum NoDiscrims {
9+
ThisIsAFairlyLongEnumVariantWithoutDiscrimLongerThan40,
10+
A = 1,
11+
ThisIsAnotherFairlyLongEnumVariantWithoutDiscrimLongerThan40,
12+
Bcdef = 2,
13+
}
14+
15+
enum TooLong {
16+
ThisOneHasDiscrimAaaaaaaaaaaaaaaaaaaaaaChar40 = 10,
17+
A = 1,
18+
Bcdef = 2,
19+
}
20+
21+
enum Borderline {
22+
ThisOneHasDiscrimAaaaaaaaaaaaaaaaaaaaaa = 10,
23+
A = 1,
24+
Bcdef = 2,
25+
}
26+
27+
// Live specimen from #1686
28+
enum LongWithSmallDiff {
29+
SceneColorimetryEstimates = 0x73636F65,
30+
SceneAppearanceEstimates = 0x73617065,
31+
FocalPlaneColorimetryEstimates = 0x66706365,
32+
ReflectionHardcopyOriginalColorimetry = 0x72686F63,
33+
ReflectionPrintOutputColorimetry = 0x72706F63,
34+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
// rustfmt-enum_discrim_align_threshold: 40
2+
3+
enum Standard {
4+
A = 1,
5+
Bcdef = 2,
6+
}
7+
8+
enum NoDiscrims {
9+
ThisIsAFairlyLongEnumVariantWithoutDiscrimLongerThan40,
10+
A = 1,
11+
ThisIsAnotherFairlyLongEnumVariantWithoutDiscrimLongerThan40,
12+
Bcdef = 2,
13+
}
14+
15+
enum TooLong {
16+
ThisOneHasDiscrimAaaaaaaaaaaaaaaaaaaaaaChar40 = 10,
17+
A = 1,
18+
Bcdef = 2,
19+
}
20+
21+
enum Borderline {
22+
ThisOneHasDiscrimAaaaaaaaaaaaaaaaaaaaaa = 10,
23+
A = 1,
24+
Bcdef = 2,
25+
}
26+
27+
// Live specimen from #1686
28+
enum LongWithSmallDiff {
29+
SceneColorimetryEstimates = 0x73636F65,
30+
SceneAppearanceEstimates = 0x73617065,
31+
FocalPlaneColorimetryEstimates = 0x66706365,
32+
ReflectionHardcopyOriginalColorimetry = 0x72686F63,
33+
ReflectionPrintOutputColorimetry = 0x72706F63,
34+
}

0 commit comments

Comments
 (0)