@@ -114,6 +114,10 @@ abstract class AbstractScanner implements Scanner {
114
114
@override
115
115
bool hasErrors = false ;
116
116
117
+ Token ? openBraceWithMissingEndForPossibleRecovery;
118
+
119
+ int ? offsetForCurlyBracketRecoveryStart;
120
+
117
121
/**
118
122
* A pointer to the stream of comment tokens created by this scanner
119
123
* before they are assigned to the [Token] precedingComments field
@@ -352,9 +356,136 @@ abstract class AbstractScanner implements Scanner {
352
356
appendToken (new KeywordToken (keyword, tokenStart, comments));
353
357
}
354
358
359
+ /// Find the indentation of the logical line of [token] .
360
+ ///
361
+ /// By logical take this as an example:
362
+ ///
363
+ /// ```
364
+ /// if (a &&
365
+ /// b) {
366
+ /// }
367
+ /// ```
368
+ ///
369
+ /// the indentation of `{` should logically be the same as the indentation of
370
+ /// `a` because it's the indentation of the `if` , even though the _line_ of a
371
+ /// has a different indentation of the _line_ of `{` .
372
+ int _spacesAtStartOfLogicalLineOf (Token token) {
373
+ if (lineStarts.isEmpty) return - 1 ;
374
+
375
+ // If the previous token is a `)`, e.g. if this is the start curly brace in
376
+ // an if, we find the first token before the corresponding `(` (in this case
377
+ // the if) in an attempt to find "the right" token to get the indentation
378
+ // for - e.g. if the if is spread over several lines the given token itself
379
+ // will - if formatted by the formatter - be indented more than the "if".
380
+ if (token.isA (TokenType .OPEN_CURLY_BRACKET )) {
381
+ if (token.previous == null ) return - 1 ;
382
+ Token previous = token.previous! ;
383
+ bool foundWanted = false ;
384
+ if (previous.isA (TokenType .CLOSE_PAREN )) {
385
+ Token closeParen = token.previous! ;
386
+ Token ? candidate = closeParen.previous;
387
+ while (candidate != null && candidate.endGroup != closeParen) {
388
+ candidate = candidate.previous;
389
+ }
390
+ if (candidate? .endGroup == closeParen && candidate! .previous != null ) {
391
+ token = candidate.previous! ;
392
+ if (token.isA (Keyword .IF ) ||
393
+ token.isA (Keyword .FOR ) ||
394
+ token.isA (Keyword .WHILE ) ||
395
+ token.isA (Keyword .SWITCH ) ||
396
+ token.isA (Keyword .CATCH )) {
397
+ foundWanted = true ;
398
+ }
399
+ }
400
+ } else if (previous.isA (Keyword .ELSE ) ||
401
+ previous.isA (Keyword .TRY ) ||
402
+ previous.isA (Keyword .FINALLY )) {
403
+ foundWanted = true ;
404
+ } else if (previous.isA (TokenType .EQ ) &&
405
+ (previous.previous? .isA (TokenType .IDENTIFIER ) ?? false )) {
406
+ // `someIdentifier = {`
407
+ foundWanted = true ;
408
+ } else if (previous.isA (Keyword .CONST ) &&
409
+ (previous.previous? .isA (TokenType .EQ ) ?? false ) &&
410
+ (previous.previous? .previous? .isA (TokenType .IDENTIFIER ) ?? false )) {
411
+ // `someIdentifier = const {`
412
+ foundWanted = true ;
413
+ }
414
+ if (! foundWanted) return - 1 ;
415
+ }
416
+
417
+ // Now find the line of [token].
418
+ final int offset = token.offset;
419
+ int low = 0 , high = lineStarts.length - 1 ;
420
+ while (low < high) {
421
+ int mid = high - ((high - low) >> 1 ); // Get middle, rounding up.
422
+ int pivot = lineStarts[mid];
423
+ if (pivot <= offset) {
424
+ low = mid;
425
+ } else {
426
+ high = mid - 1 ;
427
+ }
428
+ }
429
+ final int lineIndex = low;
430
+ if (lineIndex == 0 ) {
431
+ // On first line.
432
+ return tokens.next? .charOffset ?? - 1 ;
433
+ }
434
+
435
+ // Find the first token of the line.
436
+ int lineStartOfToken = lineStarts[lineIndex];
437
+ Token ? candidate = token.previous;
438
+ while (candidate != null && candidate.offset >= lineStartOfToken) {
439
+ candidate = candidate.previous;
440
+ }
441
+ if (candidate != null ) {
442
+ // candidate.next is the first token of the line.
443
+ return candidate.next! .offset - lineStartOfToken;
444
+ }
445
+
446
+ return - 1 ;
447
+ }
448
+
449
+ /// If there was a single missing `}` when tokenizing, try to find the actual
450
+ /// `{` that is missing the `}` .
451
+ ///
452
+ /// This is done by checking all `{` tokens between the one (currently)
453
+ /// missing a `}` and the end of the token stream. For each we try to identify
454
+ /// the indentation of the `{` and the indentation of the endGroup (i.e. the
455
+ /// `}` it has been matched with). If the indentations mismatch we assume this
456
+ /// is a bad match. There might be more bad matches because of how the curly
457
+ /// braces are nested and we pick the last one, which should be the inner-most
458
+ /// one and thus the one introducing the error: That is going to be the one
459
+ /// that has "misaligned" the rest of the end-braces.
460
+ int ? getOffsetForCurlyBracketRecoveryStart () {
461
+ if (openBraceWithMissingEndForPossibleRecovery == null ) return null ;
462
+ Token ? next = openBraceWithMissingEndForPossibleRecovery! .next;
463
+ Token ? lastMismatch;
464
+ while (next != null && ! next.isEof) {
465
+ if (next.isA (TokenType .OPEN_CURLY_BRACKET )) {
466
+ int indentOfNext = _spacesAtStartOfLogicalLineOf (next);
467
+ if (indentOfNext >= 0 &&
468
+ indentOfNext != _spacesAtStartOfLogicalLineOf (next.endGroup! )) {
469
+ lastMismatch = next;
470
+ }
471
+ }
472
+ next = next.next;
473
+ }
474
+ if (lastMismatch != null ) {
475
+ return lastMismatch.offset;
476
+ }
477
+ return null ;
478
+ }
479
+
355
480
void appendEofToken () {
356
481
beginToken ();
357
482
discardOpenLt ();
483
+ if (! groupingStack.isEmpty &&
484
+ groupingStack.head.isA (TokenType .OPEN_CURLY_BRACKET ) &&
485
+ groupingStack.tail! .isEmpty) {
486
+ // We have a single `{` that's missing a `}`. Maybe the user is typing?
487
+ openBraceWithMissingEndForPossibleRecovery = groupingStack.head;
488
+ }
358
489
while (! groupingStack.isEmpty) {
359
490
unmatchedBeginGroup (groupingStack.head);
360
491
groupingStack = groupingStack.tail! ;
@@ -870,6 +1001,15 @@ abstract class AbstractScanner implements Scanner {
870
1001
}
871
1002
872
1003
if (next == $CLOSE_CURLY_BRACKET ) {
1004
+ if (offsetForCurlyBracketRecoveryStart != null &&
1005
+ ! groupingStack.isEmpty &&
1006
+ groupingStack.head.isA (TokenType .OPEN_CURLY_BRACKET ) &&
1007
+ groupingStack.head.offset == offsetForCurlyBracketRecoveryStart) {
1008
+ // This instance of the scanner was instructed to recover this
1009
+ // opening curly bracket.
1010
+ unmatchedBeginGroup (groupingStack.head);
1011
+ groupingStack = groupingStack.tail! ;
1012
+ }
873
1013
return appendEndGroup (
874
1014
TokenType .CLOSE_CURLY_BRACKET , OPEN_CURLY_BRACKET_TOKEN );
875
1015
}
0 commit comments