@@ -115,16 +115,25 @@ pub fn format_expr(
115
115
match expr_type {
116
116
ExprType :: Statement => {
117
117
if is_unsafe_block ( block) {
118
- block. rewrite ( context, shape)
119
- } else if let rw @ Some ( _) = rewrite_empty_block ( context, block, shape) {
118
+ rewrite_block ( block, Some ( & expr. attrs ) , context, shape)
119
+ } else if let rw @ Some ( _) =
120
+ rewrite_empty_block ( context, block, Some ( & expr. attrs ) , "" , shape)
121
+ {
120
122
// Rewrite block without trying to put it in a single line.
121
123
rw
122
124
} else {
123
125
let prefix = block_prefix ( context, block, shape) ?;
124
- rewrite_block_with_visitor ( context, & prefix, block, shape, true )
126
+ rewrite_block_with_visitor (
127
+ context,
128
+ & prefix,
129
+ block,
130
+ Some ( & expr. attrs ) ,
131
+ shape,
132
+ true ,
133
+ )
125
134
}
126
135
}
127
- ExprType :: SubExpression => block. rewrite ( context, shape) ,
136
+ ExprType :: SubExpression => rewrite_block ( block, Some ( & expr . attrs ) , context, shape) ,
128
137
}
129
138
}
130
139
ast:: ExprKind :: Match ( ref cond, ref arms) => {
@@ -290,15 +299,22 @@ pub fn format_expr(
290
299
Some ( context. snippet ( expr. span ) . to_owned ( ) )
291
300
}
292
301
ast:: ExprKind :: Catch ( ref block) => {
293
- if let rw @ Some ( _) = rewrite_single_line_block ( context, "do catch " , block, shape) {
302
+ if let rw @ Some ( _) =
303
+ rewrite_single_line_block ( context, "do catch " , block, Some ( & expr. attrs ) , shape)
304
+ {
294
305
rw
295
306
} else {
296
307
// 9 = `do catch `
297
308
let budget = shape. width . checked_sub ( 9 ) . unwrap_or ( 0 ) ;
298
309
Some ( format ! (
299
310
"{}{}" ,
300
311
"do catch " ,
301
- block. rewrite( context, Shape :: legacy( budget, shape. indent) ) ?
312
+ rewrite_block(
313
+ block,
314
+ Some ( & expr. attrs) ,
315
+ context,
316
+ Shape :: legacy( budget, shape. indent)
317
+ ) ?
302
318
) )
303
319
}
304
320
}
@@ -347,7 +363,7 @@ where
347
363
let lhs_result = lhs. rewrite ( context, lhs_shape)
348
364
. map ( |lhs_str| format ! ( "{}{}" , pp. prefix, lhs_str) ) ?;
349
365
350
- // Try to the both lhs and rhs on the same line.
366
+ // Try to put both lhs and rhs on the same line.
351
367
let rhs_orig_result = shape
352
368
. offset_left ( last_line_width ( & lhs_result) + pp. infix . len ( ) )
353
369
. and_then ( |s| s. sub_width ( pp. suffix . len ( ) ) )
@@ -564,11 +580,17 @@ fn nop_block_collapse(block_str: Option<String>, budget: usize) -> Option<String
564
580
fn rewrite_empty_block (
565
581
context : & RewriteContext ,
566
582
block : & ast:: Block ,
583
+ attrs : Option < & [ ast:: Attribute ] > ,
584
+ prefix : & str ,
567
585
shape : Shape ,
568
586
) -> Option < String > {
587
+ if attrs. map_or ( false , |a| !inner_attributes ( a) . is_empty ( ) ) {
588
+ return None ;
589
+ }
590
+
569
591
if block. stmts . is_empty ( ) && !block_contains_comment ( block, context. codemap ) && shape. width >= 2
570
592
{
571
- return Some ( "{}" . to_owned ( ) ) ;
593
+ return Some ( format ! ( "{}{{}}" , prefix ) ) ;
572
594
}
573
595
574
596
// If a block contains only a single-line comment, then leave it on one line.
@@ -579,7 +601,7 @@ fn rewrite_empty_block(
579
601
if block. stmts . is_empty ( ) && !comment_str. contains ( '\n' ) && !comment_str. starts_with ( "//" )
580
602
&& comment_str. len ( ) + 4 <= shape. width
581
603
{
582
- return Some ( format ! ( "{{ {} }}" , comment_str) ) ;
604
+ return Some ( format ! ( "{}{{ {} }}" , prefix , comment_str) ) ;
583
605
}
584
606
}
585
607
@@ -618,9 +640,10 @@ fn rewrite_single_line_block(
618
640
context : & RewriteContext ,
619
641
prefix : & str ,
620
642
block : & ast:: Block ,
643
+ attrs : Option < & [ ast:: Attribute ] > ,
621
644
shape : Shape ,
622
645
) -> Option < String > {
623
- if is_simple_block ( block, context. codemap ) {
646
+ if is_simple_block ( block, attrs , context. codemap ) {
624
647
let expr_shape = shape. offset_left ( last_line_width ( prefix) ) ?;
625
648
let expr_str = block. stmts [ 0 ] . rewrite ( context, expr_shape) ?;
626
649
let result = format ! ( "{}{{ {} }}" , prefix, expr_str) ;
@@ -635,10 +658,11 @@ pub fn rewrite_block_with_visitor(
635
658
context : & RewriteContext ,
636
659
prefix : & str ,
637
660
block : & ast:: Block ,
661
+ attrs : Option < & [ ast:: Attribute ] > ,
638
662
shape : Shape ,
639
663
has_braces : bool ,
640
664
) -> Option < String > {
641
- if let rw @ Some ( _) = rewrite_empty_block ( context, block, shape) {
665
+ if let rw @ Some ( _) = rewrite_empty_block ( context, block, attrs , prefix , shape) {
642
666
return rw;
643
667
}
644
668
@@ -654,31 +678,41 @@ pub fn rewrite_block_with_visitor(
654
678
ast:: BlockCheckMode :: Default => visitor. last_pos = block. span . lo ( ) ,
655
679
}
656
680
657
- visitor. visit_block ( block, None , has_braces) ;
681
+ let inner_attrs = attrs. map ( inner_attributes) ;
682
+ visitor. visit_block ( block, inner_attrs. as_ref ( ) . map ( |a| & * * a) , has_braces) ;
658
683
Some ( format ! ( "{}{}" , prefix, visitor. buffer) )
659
684
}
660
685
661
686
impl Rewrite for ast:: Block {
662
687
fn rewrite ( & self , context : & RewriteContext , shape : Shape ) -> Option < String > {
663
- // shape.width is used only for the single line case: either the empty block `{}`,
664
- // or an unsafe expression `unsafe { e }`.
665
- if let rw @ Some ( _) = rewrite_empty_block ( context, self , shape) {
666
- return rw;
667
- }
688
+ rewrite_block ( self , None , context, shape)
689
+ }
690
+ }
668
691
669
- let prefix = block_prefix ( context, self , shape) ?;
692
+ fn rewrite_block (
693
+ block : & ast:: Block ,
694
+ attrs : Option < & [ ast:: Attribute ] > ,
695
+ context : & RewriteContext ,
696
+ shape : Shape ,
697
+ ) -> Option < String > {
698
+ let prefix = block_prefix ( context, block, shape) ?;
670
699
671
- let result = rewrite_block_with_visitor ( context, & prefix, self , shape, true ) ;
672
- if let Some ( ref result_str) = result {
673
- if result_str. lines ( ) . count ( ) <= 3 {
674
- if let rw @ Some ( _) = rewrite_single_line_block ( context, & prefix, self , shape) {
675
- return rw;
676
- }
700
+ // shape.width is used only for the single line case: either the empty block `{}`,
701
+ // or an unsafe expression `unsafe { e }`.
702
+ if let rw @ Some ( _) = rewrite_empty_block ( context, block, attrs, & prefix, shape) {
703
+ return rw;
704
+ }
705
+
706
+ let result = rewrite_block_with_visitor ( context, & prefix, block, attrs, shape, true ) ;
707
+ if let Some ( ref result_str) = result {
708
+ if result_str. lines ( ) . count ( ) <= 3 {
709
+ if let rw @ Some ( _) = rewrite_single_line_block ( context, & prefix, block, attrs, shape) {
710
+ return rw;
677
711
}
678
712
}
679
-
680
- result
681
713
}
714
+
715
+ result
682
716
}
683
717
684
718
impl Rewrite for ast:: Stmt {
@@ -889,8 +923,8 @@ impl<'a> ControlFlow<'a> {
889
923
let fixed_cost = self . keyword . len ( ) + " { } else { }" . len ( ) ;
890
924
891
925
if let ast:: ExprKind :: Block ( ref else_node) = else_block. node {
892
- if !is_simple_block ( self . block , context. codemap )
893
- || !is_simple_block ( else_node, context. codemap )
926
+ if !is_simple_block ( self . block , None , context. codemap )
927
+ || !is_simple_block ( else_node, None , context. codemap )
894
928
|| pat_expr_str. contains ( '\n' )
895
929
{
896
930
return None ;
@@ -1123,7 +1157,7 @@ impl<'a> Rewrite for ControlFlow<'a> {
1123
1157
let mut block_context = context. clone ( ) ;
1124
1158
block_context. is_if_else_block = self . else_block . is_some ( ) ;
1125
1159
let block_str =
1126
- rewrite_block_with_visitor ( & block_context, "" , self . block , block_shape, true ) ?;
1160
+ rewrite_block_with_visitor ( & block_context, "" , self . block , None , block_shape, true ) ?;
1127
1161
1128
1162
let mut result = format ! ( "{}{}" , cond_str, block_str) ;
1129
1163
@@ -1234,22 +1268,39 @@ pub fn block_contains_comment(block: &ast::Block, codemap: &CodeMap) -> bool {
1234
1268
contains_comment ( & snippet)
1235
1269
}
1236
1270
1237
- // Checks that a block contains no statements, an expression and no comments.
1271
+ // Checks that a block contains no statements, an expression and no comments or
1272
+ // attributes.
1238
1273
// FIXME: incorrectly returns false when comment is contained completely within
1239
1274
// the expression.
1240
- pub fn is_simple_block ( block : & ast:: Block , codemap : & CodeMap ) -> bool {
1275
+ pub fn is_simple_block (
1276
+ block : & ast:: Block ,
1277
+ attrs : Option < & [ ast:: Attribute ] > ,
1278
+ codemap : & CodeMap ,
1279
+ ) -> bool {
1241
1280
( block. stmts . len ( ) == 1 && stmt_is_expr ( & block. stmts [ 0 ] )
1242
- && !block_contains_comment ( block, codemap) )
1281
+ && !block_contains_comment ( block, codemap) && attrs . map_or ( true , |a| a . is_empty ( ) ) )
1243
1282
}
1244
1283
1245
- /// Checks whether a block contains at most one statement or expression, and no comments.
1246
- pub fn is_simple_block_stmt ( block : & ast:: Block , codemap : & CodeMap ) -> bool {
1284
+ /// Checks whether a block contains at most one statement or expression, and no
1285
+ /// comments or attributes.
1286
+ pub fn is_simple_block_stmt (
1287
+ block : & ast:: Block ,
1288
+ attrs : Option < & [ ast:: Attribute ] > ,
1289
+ codemap : & CodeMap ,
1290
+ ) -> bool {
1247
1291
block. stmts . len ( ) <= 1 && !block_contains_comment ( block, codemap)
1292
+ && attrs. map_or ( true , |a| a. is_empty ( ) )
1248
1293
}
1249
1294
1250
- /// Checks whether a block contains no statements, expressions, or comments.
1251
- pub fn is_empty_block ( block : & ast:: Block , codemap : & CodeMap ) -> bool {
1295
+ /// Checks whether a block contains no statements, expressions, comments, or
1296
+ /// inner attributes.
1297
+ pub fn is_empty_block (
1298
+ block : & ast:: Block ,
1299
+ attrs : Option < & [ ast:: Attribute ] > ,
1300
+ codemap : & CodeMap ,
1301
+ ) -> bool {
1252
1302
block. stmts . is_empty ( ) && !block_contains_comment ( block, codemap)
1303
+ && attrs. map_or ( true , |a| inner_attributes ( a) . is_empty ( ) )
1253
1304
}
1254
1305
1255
1306
pub fn stmt_is_expr ( stmt : & ast:: Stmt ) -> bool {
@@ -1577,7 +1628,8 @@ fn rewrite_match_pattern(
1577
1628
fn flatten_arm_body < ' a > ( context : & ' a RewriteContext , body : & ' a ast:: Expr ) -> ( bool , & ' a ast:: Expr ) {
1578
1629
match body. node {
1579
1630
ast:: ExprKind :: Block ( ref block)
1580
- if !is_unsafe_block ( block) && is_simple_block ( block, context. codemap ) =>
1631
+ if !is_unsafe_block ( block)
1632
+ && is_simple_block ( block, Some ( & body. attrs ) , context. codemap ) =>
1581
1633
{
1582
1634
if let ast:: StmtKind :: Expr ( ref expr) = block. stmts [ 0 ] . node {
1583
1635
(
@@ -1605,7 +1657,10 @@ fn rewrite_match_body(
1605
1657
) -> Option < String > {
1606
1658
let ( extend, body) = flatten_arm_body ( context, body) ;
1607
1659
let ( is_block, is_empty_block) = if let ast:: ExprKind :: Block ( ref block) = body. node {
1608
- ( true , is_empty_block ( block, context. codemap ) )
1660
+ (
1661
+ true ,
1662
+ is_empty_block ( block, Some ( & body. attrs ) , context. codemap ) ,
1663
+ )
1609
1664
} else {
1610
1665
( false , false )
1611
1666
} ;
0 commit comments