// Copyright 2016 Google Inc. Use of this source code is governed by an // MIT-style license that can be found in the LICENSE file or at // https://opensource.org/licenses/MIT. import 'package:charcode/charcode.dart'; import 'package:string_scanner/string_scanner.dart'; import '../ast/sass.dart'; import '../util/character.dart'; import 'stylesheet.dart'; class SassParser extends StylesheetParser { var _currentIndentation = 0; int _nextIndentation; LineScannerState _nextIndentationEnd; bool _spaces; bool get indented => true; SassParser(String contents, {url}) : super(contents, url: url); bool atEndOfStatement() { var next = scanner.peekChar(); return next == null || isNewline(next); } bool lookingAtChildren() => atEndOfStatement() && _peekIndentation() > _currentIndentation; List children(Statement child()) { var children = []; _whileIndentedLower(() { children.add(_child(child)); }); return children; } List statements(Statement statement()) { var first = scanner.peekChar(); if (first == $tab || first == $space) { scanner.error("Indenting at the beginning of the document is illegal.", position: 0, length: scanner.position); } var statements = []; while (!scanner.isDone) { statements.add(_child(statement)); var indentation = _readIndentation(); assert(indentation == 0); } return statements; } Statement _child(Statement child()) { switch (scanner.peekChar()) { case $dollar: return variableDeclaration(); break; case $slash: switch (scanner.peekChar(1)) { case $slash: return _silentComment(); break; case $asterisk: return _loudComment(); break; default: return child(); break; } break; default: return child(); break; } } Comment _silentComment() { var start = scanner.state; scanner.expect("//"); var buffer = new StringBuffer(); var parentIndentation = _currentIndentation; while (true) { buffer.write("//"); // Skip the first two indentation characters because we're already writing // "//". for (var i = 2; i < _currentIndentation - parentIndentation; i++) { buffer.writeCharCode($space); } while (!scanner.isDone && !isNewline(scanner.peekChar())) { buffer.writeCharCode(scanner.readChar()); } buffer.writeln(); if (_peekIndentation() <= parentIndentation) break; _readIndentation(); } return new Comment(buffer.toString(), scanner.spanFrom(start), silent: true); } Comment _loudComment() { var start = scanner.state; scanner.expect("/*"); var first = true; var buffer = new StringBuffer("/*"); var parentIndentation = _currentIndentation; while (true) { if (!first) { buffer.writeln(); buffer.write(" * "); } first = false; for (var i = 3; i < _currentIndentation - parentIndentation; i++) { buffer.writeCharCode($space); } while (!scanner.isDone && !isNewline(scanner.peekChar())) { buffer.writeCharCode(scanner.readChar()); } if (_peekIndentation() <= parentIndentation) break; _readIndentation(); } buffer.write(" */"); print(buffer); return new Comment(buffer.toString(), scanner.spanFrom(start), silent: false); } // Doesn't consume newlines, doesn't support comments. void whitespace() { while (!scanner.isDone) { var next = scanner.peekChar(); if (next != $tab && next != $space) break; scanner.readChar(); } if (scanner.peekChar() == $slash && scanner.peekChar(1) == $slash) { silentComment(); } } void _whileIndentedLower(void body()) { var parentIndentation = _currentIndentation; int childIndentation; while (_peekIndentation() > parentIndentation) { var indentation = _readIndentation(); childIndentation ??= indentation; if (childIndentation != indentation) { scanner.error( "Inconsistent indentation, expected $childIndentation spaces.", position: scanner.position - scanner.column, length: scanner.column); } body(); } } int _readIndentation() { if (_nextIndentation == null) _peekIndentation(); _currentIndentation = _nextIndentation; scanner.state = _nextIndentationEnd; _nextIndentation = null; _nextIndentationEnd = null; return _currentIndentation; } int _peekIndentation() { if (_nextIndentation != null) return _nextIndentation; if (scanner.isDone) { _nextIndentation = 0; _nextIndentationEnd = scanner.state; return 0; } var start = scanner.state; if (!scanCharIf(isNewline)) { scanner.error("Expected newline.", position: scanner.position); } bool containsTab; bool containsSpace; do { containsTab = false; containsSpace = false; _nextIndentation = 0; while (true) { var next = scanner.peekChar(); if (next == $space) { containsSpace = true; } else if (next == $tab) { containsTab = true; } else { break; } _nextIndentation++; scanner.readChar(); } if (scanner.isDone) { _nextIndentation = 0; _nextIndentationEnd = scanner.state; scanner.state = start; return 0; } } while (scanCharIf(isNewline)); _checkIndentationConsistency(containsTab, containsSpace); _spaces ??= containsSpace; _nextIndentationEnd = scanner.state; scanner.state = start; return _nextIndentation; } void _checkIndentationConsistency(bool containsTab, bool containsSpace) { if (containsTab) { if (containsSpace) { scanner.error("Tabs and spaces may not be mixed.", position: scanner.position - scanner.column, length: scanner.column); } else if (_spaces == true) { scanner.error("Expected spaces, was tabs.", position: scanner.position - scanner.column, length: scanner.column); } } else if (_spaces == false) { scanner.error("Expected tabs, was spaces.", position: scanner.position - scanner.column, length: scanner.column); } } }