@@ -80,8 +80,6 @@ pub struct FormatSpec<'a> {
80
80
/// Enum describing where an argument for a format can be located.
81
81
#[ derive( Copy , Clone , PartialEq ) ]
82
82
pub enum Position < ' a > {
83
- /// The argument will be in the next position. This is the default.
84
- ArgumentNext ,
85
83
/// The argument is located at a specific index.
86
84
ArgumentIs ( usize ) ,
87
85
/// The argument has a name.
@@ -127,8 +125,6 @@ pub enum Count<'a> {
127
125
CountIsName ( & ' a str ) ,
128
126
/// The count is specified by the argument at the given index.
129
127
CountIsParam ( usize ) ,
130
- /// The count is specified by the next parameter.
131
- CountIsNextParam ,
132
128
/// The count is implied and cannot be explicitly specified.
133
129
CountImplied ,
134
130
}
@@ -144,6 +140,8 @@ pub struct Parser<'a> {
144
140
cur : iter:: Peekable < str:: CharIndices < ' a > > ,
145
141
/// Error messages accumulated during parsing
146
142
pub errors : Vec < string:: String > ,
143
+ /// Current position of implicit positional argument pointer
144
+ curarg : usize ,
147
145
}
148
146
149
147
impl < ' a > Iterator for Parser < ' a > {
@@ -186,6 +184,7 @@ impl<'a> Parser<'a> {
186
184
input : s,
187
185
cur : s. char_indices ( ) . peekable ( ) ,
188
186
errors : vec ! [ ] ,
187
+ curarg : 0 ,
189
188
}
190
189
}
191
190
@@ -259,21 +258,40 @@ impl<'a> Parser<'a> {
259
258
/// Parses an Argument structure, or what's contained within braces inside
260
259
/// the format string
261
260
fn argument ( & mut self ) -> Argument < ' a > {
261
+ let pos = self . position ( ) ;
262
+ let format = self . format ( ) ;
263
+
264
+ // Resolve position after parsing format spec.
265
+ let pos = match pos {
266
+ Some ( position) => position,
267
+ None => {
268
+ let i = self . curarg ;
269
+ self . curarg += 1 ;
270
+ ArgumentIs ( i)
271
+ }
272
+ } ;
273
+
262
274
Argument {
263
- position : self . position ( ) ,
264
- format : self . format ( ) ,
275
+ position : pos ,
276
+ format : format,
265
277
}
266
278
}
267
279
268
280
/// Parses a positional argument for a format. This could either be an
269
281
/// integer index of an argument, a named argument, or a blank string.
270
- fn position ( & mut self ) -> Position < ' a > {
282
+ /// Returns `Some(parsed_position)` if the position is not implicitly
283
+ /// consuming a macro argument, `None` if it's the case.
284
+ fn position ( & mut self ) -> Option < Position < ' a > > {
271
285
if let Some ( i) = self . integer ( ) {
272
- ArgumentIs ( i)
286
+ Some ( ArgumentIs ( i) )
273
287
} else {
274
288
match self . cur . peek ( ) {
275
- Some ( & ( _, c) ) if c. is_alphabetic ( ) => ArgumentNamed ( self . word ( ) ) ,
276
- _ => ArgumentNext ,
289
+ Some ( & ( _, c) ) if c. is_alphabetic ( ) => Some ( ArgumentNamed ( self . word ( ) ) ) ,
290
+
291
+ // This is an `ArgumentNext`.
292
+ // Record the fact and do the resolution after parsing the
293
+ // format spec, to make things like `{:.*}` work.
294
+ _ => None ,
277
295
}
278
296
}
279
297
}
@@ -340,7 +358,11 @@ impl<'a> Parser<'a> {
340
358
}
341
359
if self . consume ( '.' ) {
342
360
if self . consume ( '*' ) {
343
- spec. precision = CountIsNextParam ;
361
+ // Resolve `CountIsNextParam`.
362
+ // We can do this immediately as `position` is resolved later.
363
+ let i = self . curarg ;
364
+ self . curarg += 1 ;
365
+ spec. precision = CountIsParam ( i) ;
344
366
} else {
345
367
spec. precision = self . count ( ) ;
346
368
}
@@ -487,7 +509,7 @@ mod tests {
487
509
fn format_nothing ( ) {
488
510
same ( "{}" ,
489
511
& [ NextArgument ( Argument {
490
- position : ArgumentNext ,
512
+ position : ArgumentIs ( 0 ) ,
491
513
format : fmtdflt ( ) ,
492
514
} ) ] ) ;
493
515
}
@@ -565,7 +587,7 @@ mod tests {
565
587
fn format_counts ( ) {
566
588
same ( "{:10s}" ,
567
589
& [ NextArgument ( Argument {
568
- position : ArgumentNext ,
590
+ position : ArgumentIs ( 0 ) ,
569
591
format : FormatSpec {
570
592
fill : None ,
571
593
align : AlignUnknown ,
@@ -577,7 +599,7 @@ mod tests {
577
599
} ) ] ) ;
578
600
same ( "{:10$.10s}" ,
579
601
& [ NextArgument ( Argument {
580
- position : ArgumentNext ,
602
+ position : ArgumentIs ( 0 ) ,
581
603
format : FormatSpec {
582
604
fill : None ,
583
605
align : AlignUnknown ,
@@ -589,19 +611,19 @@ mod tests {
589
611
} ) ] ) ;
590
612
same ( "{:.*s}" ,
591
613
& [ NextArgument ( Argument {
592
- position : ArgumentNext ,
614
+ position : ArgumentIs ( 1 ) ,
593
615
format : FormatSpec {
594
616
fill : None ,
595
617
align : AlignUnknown ,
596
618
flags : 0 ,
597
- precision : CountIsNextParam ,
619
+ precision : CountIsParam ( 0 ) ,
598
620
width : CountImplied ,
599
621
ty : "s" ,
600
622
} ,
601
623
} ) ] ) ;
602
624
same ( "{:.10$s}" ,
603
625
& [ NextArgument ( Argument {
604
- position : ArgumentNext ,
626
+ position : ArgumentIs ( 0 ) ,
605
627
format : FormatSpec {
606
628
fill : None ,
607
629
align : AlignUnknown ,
@@ -613,7 +635,7 @@ mod tests {
613
635
} ) ] ) ;
614
636
same ( "{:a$.b$s}" ,
615
637
& [ NextArgument ( Argument {
616
- position : ArgumentNext ,
638
+ position : ArgumentIs ( 0 ) ,
617
639
format : FormatSpec {
618
640
fill : None ,
619
641
align : AlignUnknown ,
@@ -628,7 +650,7 @@ mod tests {
628
650
fn format_flags ( ) {
629
651
same ( "{:-}" ,
630
652
& [ NextArgument ( Argument {
631
- position : ArgumentNext ,
653
+ position : ArgumentIs ( 0 ) ,
632
654
format : FormatSpec {
633
655
fill : None ,
634
656
align : AlignUnknown ,
@@ -640,7 +662,7 @@ mod tests {
640
662
} ) ] ) ;
641
663
same ( "{:+#}" ,
642
664
& [ NextArgument ( Argument {
643
- position : ArgumentNext ,
665
+ position : ArgumentIs ( 0 ) ,
644
666
format : FormatSpec {
645
667
fill : None ,
646
668
align : AlignUnknown ,
0 commit comments