| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849 |
- /*
- * The code in this file is licensed under the CC0 license.
- *
- * http://creativecommons.org/publicdomain/zero/1.0/
- *
- * It is free to use for any purpose. No attribution, permission, or reproduction of this license is require
- */
- // Modified by Gildas Lormeau (ES5 -> ES6, removed unused code)
- // https://github.com/tabatkins/parse-css
- this.parseCss = this.parseCss || (() => {
- function between(num, first, last) { return num >= first && num <= last; }
- function digit(code) { return between(code, 0x30, 0x39); }
- function hexdigit(code) { return digit(code) || between(code, 0x41, 0x46) || between(code, 0x61, 0x66); }
- function uppercaseletter(code) { return between(code, 0x41, 0x5a); }
- function lowercaseletter(code) { return between(code, 0x61, 0x7a); }
- function letter(code) { return uppercaseletter(code) || lowercaseletter(code); }
- function nonascii(code) { return code >= 0x80; }
- function namestartchar(code) { return letter(code) || nonascii(code) || code == 0x5f; }
- function namechar(code) { return namestartchar(code) || digit(code) || code == 0x2d; }
- function nonprintable(code) { return between(code, 0, 8) || code == 0xb || between(code, 0xe, 0x1f) || code == 0x7f; }
- function newline(code) { return code == 0xa; }
- function whitespace(code) { return newline(code) || code == 9 || code == 0x20; }
- const maximumallowedcodepoint = 0x10ffff;
- const InvalidCharacterError = function (message) {
- this.message = message;
- };
- InvalidCharacterError.prototype = new Error;
- InvalidCharacterError.prototype.name = "InvalidCharacterError";
- function preprocess(str) {
- // Turn a string into an array of code points,
- // following the preprocessing cleanup rules.
- const codepoints = [];
- for (let i = 0; i < str.length; i++) {
- let code = str.charCodeAt(i);
- if (code == 0xd && str.charCodeAt(i + 1) == 0xa) {
- code = 0xa; i++;
- }
- if (code == 0xd || code == 0xc) code = 0xa;
- if (code == 0x0) code = 0xfffd;
- if (between(code, 0xd800, 0xdbff) && between(str.charCodeAt(i + 1), 0xdc00, 0xdfff)) {
- // Decode a surrogate pair into an astral codepoint.
- const lead = code - 0xd800;
- const trail = str.charCodeAt(i + 1) - 0xdc00;
- code = Math.pow(2, 20) + lead * Math.pow(2, 10) + trail;
- i++;
- }
- codepoints.push(code);
- }
- return codepoints;
- }
- function stringFromCode(code) {
- if (code <= 0xffff) return String.fromCharCode(code);
- // Otherwise, encode astral char as surrogate pair.
- code -= Math.pow(2, 20);
- const lead = Math.floor(code / Math.pow(2, 10)) + 0xd800;
- const trail = code % Math.pow(2, 10) + 0xdc00;
- return String.fromCharCode(lead) + String.fromCharCode(trail);
- }
- function tokenize(str) {
- str = preprocess(str);
- let i = -1;
- const tokens = [];
- let code;
- // Line number information.
- let line = 0;
- let column = 0;
- // The only use of lastLineLength is in reconsume().
- let lastLineLength = 0;
- const incrLineno = function () {
- line += 1;
- lastLineLength = column;
- column = 0;
- };
- const locStart = { line: line, column: column };
- const codepoint = function (i) {
- if (i >= str.length) {
- return -1;
- }
- return str[i];
- };
- const next = function (num) {
- if (num === undefined)
- num = 1;
- if (num > 3)
- throw "Spec Error: no more than three codepoints of lookahead.";
- return codepoint(i + num);
- };
- const consume = function (num) {
- if (num === undefined)
- num = 1;
- i += num;
- code = codepoint(i);
- if (newline(code)) incrLineno();
- else column += num;
- //console.log('Consume '+i+' '+String.fromCharCode(code) + ' 0x' + code.toString(16));
- return true;
- };
- const reconsume = function () {
- i -= 1;
- if (newline(code)) {
- line -= 1;
- column = lastLineLength;
- } else {
- column -= 1;
- }
- locStart.line = line;
- locStart.column = column;
- return true;
- };
- const eof = function (codepoint) {
- if (codepoint === undefined) codepoint = code;
- return codepoint == -1;
- };
- const donothing = function () { };
- const parseerror = function () { throw new Error("Parse error at index " + i + ", processing codepoint 0x" + code.toString(16) + "."); };
- const consumeAToken = function () {
- consumeComments();
- consume();
- if (whitespace(code)) {
- while (whitespace(next())) consume();
- return new WhitespaceToken;
- }
- else if (code == 0x22) return consumeAStringToken();
- else if (code == 0x23) {
- if (namechar(next()) || areAValidEscape(next(1), next(2))) {
- const token = new HashToken();
- if (wouldStartAnIdentifier(next(1), next(2), next(3))) token.type = "id";
- token.value = consumeAName();
- return token;
- } else {
- return new DelimToken(code);
- }
- }
- else if (code == 0x24) {
- if (next() == 0x3d) {
- consume();
- return new SuffixMatchToken();
- } else {
- return new DelimToken(code);
- }
- }
- else if (code == 0x27) return consumeAStringToken();
- else if (code == 0x28) return new OpenParenToken();
- else if (code == 0x29) return new CloseParenToken();
- else if (code == 0x2a) {
- if (next() == 0x3d) {
- consume();
- return new SubstringMatchToken();
- } else {
- return new DelimToken(code);
- }
- }
- else if (code == 0x2b) {
- if (startsWithANumber()) {
- reconsume();
- return consumeANumericToken();
- } else {
- return new DelimToken(code);
- }
- }
- else if (code == 0x2c) return new CommaToken();
- else if (code == 0x2d) {
- if (startsWithANumber()) {
- reconsume();
- return consumeANumericToken();
- } else if (next(1) == 0x2d && next(2) == 0x3e) {
- consume(2);
- return new CDCToken();
- } else if (startsWithAnIdentifier()) {
- reconsume();
- return consumeAnIdentlikeToken();
- } else {
- return new DelimToken(code);
- }
- }
- else if (code == 0x2e) {
- if (startsWithANumber()) {
- reconsume();
- return consumeANumericToken();
- } else {
- return new DelimToken(code);
- }
- }
- else if (code == 0x3a) return new ColonToken;
- else if (code == 0x3b) return new SemicolonToken;
- else if (code == 0x3c) {
- if (next(1) == 0x21 && next(2) == 0x2d && next(3) == 0x2d) {
- consume(3);
- return new CDOToken();
- } else {
- return new DelimToken(code);
- }
- }
- else if (code == 0x40) {
- return new DelimToken(code);
- }
- else if (code == 0x5b) return new OpenSquareToken();
- else if (code == 0x5c) {
- if (startsWithAValidEscape()) {
- reconsume();
- return consumeAnIdentlikeToken();
- } else {
- parseerror();
- return new DelimToken(code);
- }
- }
- else if (code == 0x5d) return new CloseSquareToken();
- else if (code == 0x5e) {
- if (next() == 0x3d) {
- consume();
- return new PrefixMatchToken();
- } else {
- return new DelimToken(code);
- }
- }
- else if (code == 0x7b) return new OpenCurlyToken();
- else if (code == 0x7c) {
- if (next() == 0x3d) {
- consume();
- return new DashMatchToken();
- } else if (next() == 0x7c) {
- consume();
- return new ColumnToken();
- } else {
- return new DelimToken(code);
- }
- }
- else if (code == 0x7d) return new CloseCurlyToken();
- else if (code == 0x7e) {
- if (next() == 0x3d) {
- consume();
- return new IncludeMatchToken();
- } else {
- return new DelimToken(code);
- }
- }
- else if (digit(code)) {
- reconsume();
- return consumeANumericToken();
- }
- else if (namestartchar(code)) {
- reconsume();
- return consumeAnIdentlikeToken();
- }
- else if (eof()) return new EOFToken();
- else return new DelimToken(code);
- };
- const consumeComments = function () {
- while (next(1) == 0x2f && next(2) == 0x2a) {
- consume(2);
- while (true) { // eslint-disable-line no-constant-condition
- consume();
- if (code == 0x2a && next() == 0x2f) {
- consume();
- break;
- } else if (eof()) {
- parseerror();
- return;
- }
- }
- }
- };
- const consumeANumericToken = function () {
- const num = consumeANumber();
- if (wouldStartAnIdentifier(next(1), next(2), next(3))) {
- const token = new DimensionToken();
- token.value = num.value;
- token.repr = num.repr;
- token.type = num.type;
- token.unit = consumeAName();
- return token;
- } else if (next() == 0x25) {
- consume();
- const token = new PercentageToken();
- token.value = num.value;
- token.repr = num.repr;
- return token;
- } else {
- const token = new NumberToken();
- token.value = num.value;
- token.repr = num.repr;
- token.type = num.type;
- return token;
- }
- };
- const consumeAnIdentlikeToken = function () {
- const str = consumeAName();
- if (str.toLowerCase() == "url" && next() == 0x28) {
- consume();
- while (whitespace(next(1)) && whitespace(next(2))) consume();
- if (next() == 0x22 || next() == 0x27) {
- return new FunctionToken(str);
- } else if (whitespace(next()) && (next(2) == 0x22 || next(2) == 0x27)) {
- return new FunctionToken(str);
- } else {
- return consumeAURLToken();
- }
- } else if (next() == 0x28) {
- consume();
- return new FunctionToken(str);
- } else {
- return new IdentToken(str);
- }
- };
- const consumeAStringToken = function (endingCodePoint) {
- if (endingCodePoint === undefined) endingCodePoint = code;
- let string = "";
- while (consume()) {
- if (code == endingCodePoint || eof()) {
- return new StringToken(string);
- } else if (newline(code)) {
- parseerror();
- reconsume();
- return new BadStringToken();
- } else if (code == 0x5c) {
- if (eof(next())) {
- donothing();
- } else if (newline(next())) {
- consume();
- } else {
- string += stringFromCode(consumeEscape());
- }
- } else {
- string += stringFromCode(code);
- }
- }
- };
- const consumeAURLToken = function () {
- const token = new URLToken("");
- while (whitespace(next())) consume();
- if (eof(next())) return token;
- while (consume()) {
- if (code == 0x29 || eof()) {
- return token;
- } else if (whitespace(code)) {
- while (whitespace(next())) consume();
- if (next() == 0x29 || eof(next())) {
- consume();
- return token;
- } else {
- consumeTheRemnantsOfABadURL();
- return new BadURLToken();
- }
- } else if (code == 0x22 || code == 0x27 || code == 0x28 || nonprintable(code)) {
- parseerror();
- consumeTheRemnantsOfABadURL();
- return new BadURLToken();
- } else if (code == 0x5c) {
- if (startsWithAValidEscape()) {
- token.value += stringFromCode(consumeEscape());
- } else {
- parseerror();
- consumeTheRemnantsOfABadURL();
- return new BadURLToken();
- }
- } else {
- token.value += stringFromCode(code);
- }
- }
- };
- const consumeEscape = function () {
- // Assume the the current character is the \
- // and the next code point is not a newline.
- consume();
- if (hexdigit(code)) {
- // Consume 1-6 hex digits
- const digits = [code];
- for (let total = 0; total < 5; total++) {
- if (hexdigit(next())) {
- consume();
- digits.push(code);
- } else {
- break;
- }
- }
- if (whitespace(next())) consume();
- let value = parseInt(digits.map(function (x) { return String.fromCharCode(x); }).join(""), 16);
- if (value > maximumallowedcodepoint) value = 0xfffd;
- return value;
- } else if (eof()) {
- return 0xfffd;
- } else {
- return code;
- }
- };
- const areAValidEscape = function (c1, c2) {
- if (c1 != 0x5c) return false;
- if (newline(c2)) return false;
- return true;
- };
- const startsWithAValidEscape = function () {
- return areAValidEscape(code, next());
- };
- const wouldStartAnIdentifier = function (c1, c2, c3) {
- if (c1 == 0x2d) {
- return namestartchar(c2) || c2 == 0x2d || areAValidEscape(c2, c3);
- } else if (namestartchar(c1)) {
- return true;
- } else if (c1 == 0x5c) {
- return areAValidEscape(c1, c2);
- } else {
- return false;
- }
- };
- const startsWithAnIdentifier = function () {
- return wouldStartAnIdentifier(code, next(1), next(2));
- };
- const wouldStartANumber = function (c1, c2, c3) {
- if (c1 == 0x2b || c1 == 0x2d) {
- if (digit(c2)) return true;
- if (c2 == 0x2e && digit(c3)) return true;
- return false;
- } else if (c1 == 0x2e) {
- if (digit(c2)) return true;
- return false;
- } else if (digit(c1)) {
- return true;
- } else {
- return false;
- }
- };
- const startsWithANumber = function () {
- return wouldStartANumber(code, next(1), next(2));
- };
- const consumeAName = function () {
- let result = "";
- while (consume()) {
- if (namechar(code)) {
- result += stringFromCode(code);
- } else if (startsWithAValidEscape()) {
- result += stringFromCode(consumeEscape());
- } else {
- reconsume();
- return result;
- }
- }
- };
- const consumeANumber = function () {
- let repr = [];
- let type = "integer";
- if (next() == 0x2b || next() == 0x2d) {
- consume();
- repr += stringFromCode(code);
- }
- while (digit(next())) {
- consume();
- repr += stringFromCode(code);
- }
- if (next(1) == 0x2e && digit(next(2))) {
- consume();
- repr += stringFromCode(code);
- consume();
- repr += stringFromCode(code);
- type = "number";
- while (digit(next())) {
- consume();
- repr += stringFromCode(code);
- }
- }
- const c1 = next(1), c2 = next(2), c3 = next(3);
- if ((c1 == 0x45 || c1 == 0x65) && digit(c2)) {
- consume();
- repr += stringFromCode(code);
- consume();
- repr += stringFromCode(code);
- type = "number";
- while (digit(next())) {
- consume();
- repr += stringFromCode(code);
- }
- } else if ((c1 == 0x45 || c1 == 0x65) && (c2 == 0x2b || c2 == 0x2d) && digit(c3)) {
- consume();
- repr += stringFromCode(code);
- consume();
- repr += stringFromCode(code);
- consume();
- repr += stringFromCode(code);
- type = "number";
- while (digit(next())) {
- consume();
- repr += stringFromCode(code);
- }
- }
- const value = convertAStringToANumber(repr);
- return { type: type, value: value, repr: repr };
- };
- const convertAStringToANumber = function (string) {
- // CSS's number rules are identical to JS, afaik.
- return Number(string);
- };
- const consumeTheRemnantsOfABadURL = function () {
- while (consume()) {
- if (code == 0x29 || eof()) {
- return;
- } else if (startsWithAValidEscape()) {
- consumeEscape();
- donothing();
- } else {
- donothing();
- }
- }
- };
- let iterationCount = 0;
- while (!eof(next())) {
- tokens.push(consumeAToken());
- iterationCount++;
- if (iterationCount > str.length * 2) return "I'm infinite-looping!";
- }
- return tokens;
- }
- function CSSParserToken() { }
- CSSParserToken.prototype.toString = function () { return this.tokenType; };
- function BadStringToken() { return this; }
- BadStringToken.prototype = Object.create(CSSParserToken.prototype);
- BadStringToken.prototype.tokenType = "BADSTRING";
- function BadURLToken() { return this; }
- BadURLToken.prototype = Object.create(CSSParserToken.prototype);
- BadURLToken.prototype.tokenType = "BADURL";
- function WhitespaceToken() { return this; }
- WhitespaceToken.prototype = Object.create(CSSParserToken.prototype);
- WhitespaceToken.prototype.tokenType = "WHITESPACE";
- WhitespaceToken.prototype.toString = function () { return "WS"; };
- function CDOToken() { return this; }
- CDOToken.prototype = Object.create(CSSParserToken.prototype);
- CDOToken.prototype.tokenType = "CDO";
- function CDCToken() { return this; }
- CDCToken.prototype = Object.create(CSSParserToken.prototype);
- CDCToken.prototype.tokenType = "CDC";
- function ColonToken() { return this; }
- ColonToken.prototype = Object.create(CSSParserToken.prototype);
- ColonToken.prototype.tokenType = ":";
- function SemicolonToken() { return this; }
- SemicolonToken.prototype = Object.create(CSSParserToken.prototype);
- SemicolonToken.prototype.tokenType = ";";
- function CommaToken() { return this; }
- CommaToken.prototype = Object.create(CSSParserToken.prototype);
- CommaToken.prototype.tokenType = ",";
- function GroupingToken() { throw "Abstract Base Class"; }
- GroupingToken.prototype = Object.create(CSSParserToken.prototype);
- function OpenCurlyToken() { this.value = "{"; this.mirror = "}"; return this; }
- OpenCurlyToken.prototype = Object.create(GroupingToken.prototype);
- OpenCurlyToken.prototype.tokenType = "{";
- function CloseCurlyToken() { this.value = "}"; this.mirror = "{"; return this; }
- CloseCurlyToken.prototype = Object.create(GroupingToken.prototype);
- CloseCurlyToken.prototype.tokenType = "}";
- function OpenSquareToken() { this.value = "["; this.mirror = "]"; return this; }
- OpenSquareToken.prototype = Object.create(GroupingToken.prototype);
- OpenSquareToken.prototype.tokenType = "[";
- function CloseSquareToken() { this.value = "]"; this.mirror = "["; return this; }
- CloseSquareToken.prototype = Object.create(GroupingToken.prototype);
- CloseSquareToken.prototype.tokenType = "]";
- function OpenParenToken() { this.value = "("; this.mirror = ")"; return this; }
- OpenParenToken.prototype = Object.create(GroupingToken.prototype);
- OpenParenToken.prototype.tokenType = "(";
- function CloseParenToken() { this.value = ")"; this.mirror = "("; return this; }
- CloseParenToken.prototype = Object.create(GroupingToken.prototype);
- CloseParenToken.prototype.tokenType = ")";
- function IncludeMatchToken() { return this; }
- IncludeMatchToken.prototype = Object.create(CSSParserToken.prototype);
- IncludeMatchToken.prototype.tokenType = "~=";
- function DashMatchToken() { return this; }
- DashMatchToken.prototype = Object.create(CSSParserToken.prototype);
- DashMatchToken.prototype.tokenType = "|=";
- function PrefixMatchToken() { return this; }
- PrefixMatchToken.prototype = Object.create(CSSParserToken.prototype);
- PrefixMatchToken.prototype.tokenType = "^=";
- function SuffixMatchToken() { return this; }
- SuffixMatchToken.prototype = Object.create(CSSParserToken.prototype);
- SuffixMatchToken.prototype.tokenType = "$=";
- function SubstringMatchToken() { return this; }
- SubstringMatchToken.prototype = Object.create(CSSParserToken.prototype);
- SubstringMatchToken.prototype.tokenType = "*=";
- function ColumnToken() { return this; }
- ColumnToken.prototype = Object.create(CSSParserToken.prototype);
- ColumnToken.prototype.tokenType = "||";
- function EOFToken() { return this; }
- EOFToken.prototype = Object.create(CSSParserToken.prototype);
- EOFToken.prototype.tokenType = "EOF";
- function DelimToken(code) {
- this.value = stringFromCode(code);
- return this;
- }
- DelimToken.prototype = Object.create(CSSParserToken.prototype);
- DelimToken.prototype.tokenType = "DELIM";
- DelimToken.prototype.toString = function () { return "DELIM(" + this.value + ")"; };
- function StringValuedToken() { throw "Abstract Base Class"; }
- StringValuedToken.prototype = Object.create(CSSParserToken.prototype);
- StringValuedToken.prototype.ASCIIMatch = function (str) {
- return this.value.toLowerCase() == str.toLowerCase();
- };
- function IdentToken(val) {
- this.value = val;
- }
- IdentToken.prototype = Object.create(StringValuedToken.prototype);
- IdentToken.prototype.tokenType = "IDENT";
- IdentToken.prototype.toString = function () { return "IDENT(" + this.value + ")"; };
- function FunctionToken(val) {
- this.value = val;
- this.mirror = ")";
- }
- FunctionToken.prototype = Object.create(StringValuedToken.prototype);
- FunctionToken.prototype.tokenType = "FUNCTION";
- FunctionToken.prototype.toString = function () { return "FUNCTION(" + this.value + ")"; };
- function HashToken(val) {
- this.value = val;
- this.type = "unrestricted";
- }
- HashToken.prototype = Object.create(StringValuedToken.prototype);
- HashToken.prototype.tokenType = "HASH";
- HashToken.prototype.toString = function () { return "HASH(" + this.value + ")"; };
- function StringToken(val) {
- this.value = val;
- }
- StringToken.prototype = Object.create(StringValuedToken.prototype);
- StringToken.prototype.tokenType = "STRING";
- function URLToken(val) {
- this.value = val;
- }
- URLToken.prototype = Object.create(StringValuedToken.prototype);
- URLToken.prototype.tokenType = "URL";
- URLToken.prototype.toString = function () { return "URL(" + this.value + ")"; };
- function NumberToken() {
- this.value = null;
- this.type = "integer";
- this.repr = "";
- }
- NumberToken.prototype = Object.create(CSSParserToken.prototype);
- NumberToken.prototype.tokenType = "NUMBER";
- NumberToken.prototype.toString = function () {
- if (this.type == "integer")
- return "INT(" + this.value + ")";
- return "NUMBER(" + this.value + ")";
- };
- function PercentageToken() {
- this.value = null;
- this.repr = "";
- }
- PercentageToken.prototype = Object.create(CSSParserToken.prototype);
- PercentageToken.prototype.tokenType = "PERCENTAGE";
- PercentageToken.prototype.toString = function () { return "PERCENTAGE(" + this.value + ")"; };
- function DimensionToken() {
- this.value = null;
- this.type = "integer";
- this.repr = "";
- this.unit = "";
- }
- DimensionToken.prototype = Object.create(CSSParserToken.prototype);
- DimensionToken.prototype.tokenType = "DIMENSION";
- DimensionToken.prototype.toString = function () { return "DIM(" + this.value + "," + this.unit + ")"; };
- // ---
- function TokenStream(tokens) {
- // Assume that tokens is an array.
- this.tokens = tokens;
- this.i = -1;
- }
- TokenStream.prototype.tokenAt = function (i) {
- if (i < this.tokens.length)
- return this.tokens[i];
- return new EOFToken();
- };
- TokenStream.prototype.consume = function (num) {
- if (num === undefined) num = 1;
- this.i += num;
- this.token = this.tokenAt(this.i);
- //console.log(this.i, this.token);
- return true;
- };
- TokenStream.prototype.next = function () {
- return this.tokenAt(this.i + 1);
- };
- TokenStream.prototype.reconsume = function () {
- this.i--;
- };
- function parseerror(s, msg) {
- throw new Error("Parse error at token " + s.i + ": " + s.token + ".\n" + msg);
- }
- function donothing() { return true; }
- function consumeAListOfDeclarations(s) {
- const decls = [];
- while (s.consume()) {
- if (s.token instanceof WhitespaceToken || s.token instanceof SemicolonToken) {
- donothing();
- } else if (s.token instanceof EOFToken) {
- return decls;
- } else if (s.token instanceof IdentToken) {
- const temp = [s.token];
- while (!(s.next() instanceof SemicolonToken || s.next() instanceof EOFToken))
- temp.push(consumeAComponentValue(s));
- let decl = consumeADeclaration(new TokenStream(temp));
- if (decl) decls.push(decl);
- } else {
- parseerror(s);
- s.reconsume();
- while (!(s.next() instanceof SemicolonToken || s.next() instanceof EOFToken))
- consumeAComponentValue(s);
- }
- }
- }
- function consumeADeclaration(s) {
- // Assumes that the next input token will be an ident token.
- s.consume();
- const decl = new Declaration(s.token.value);
- while (s.next() instanceof WhitespaceToken) s.consume();
- if (!(s.next() instanceof ColonToken)) {
- parseerror(s);
- return;
- } else {
- s.consume();
- }
- while (!(s.next() instanceof EOFToken)) {
- decl.value.push(consumeAComponentValue(s));
- }
- let foundImportant = false;
- for (let i = decl.value.length - 1; i >= 0; i--) {
- if (decl.value[i] instanceof WhitespaceToken) {
- continue;
- } else if (decl.value[i] instanceof IdentToken && decl.value[i].ASCIIMatch("important")) {
- foundImportant = true;
- } else if (foundImportant && decl.value[i] instanceof DelimToken && decl.value[i].value == "!") {
- decl.value.splice(i, decl.value.length);
- decl.important = true;
- break;
- } else {
- break;
- }
- }
- return decl;
- }
- function consumeAComponentValue(s) {
- s.consume();
- if (s.token instanceof FunctionToken)
- return consumeAFunction(s);
- return s.token;
- }
- function consumeAFunction(s) {
- const func = new Func(s.token.value);
- while (s.consume()) {
- if (s.token instanceof EOFToken || s.token instanceof CloseParenToken)
- return func;
- else {
- s.reconsume();
- func.value.push(consumeAComponentValue(s));
- }
- }
- }
- function normalizeInput(input) {
- if (typeof input == "string")
- return new TokenStream(tokenize(input));
- else throw SyntaxError(input);
- }
- function parseAListOfDeclarations(s) {
- s = normalizeInput(s);
- return consumeAListOfDeclarations(s);
- }
- function CSSParserRule() { throw "Abstract Base Class"; }
- function Declaration(name) {
- this.name = name;
- this.value = [];
- this.important = false;
- return this;
- }
- Declaration.prototype = Object.create(CSSParserRule.prototype);
- Declaration.prototype.type = "DECLARATION";
- function Func(name) {
- this.name = name;
- this.value = [];
- return this;
- }
- Func.prototype = Object.create(CSSParserRule.prototype);
- Func.prototype.type = "FUNCTION";
- // Exportation.
- return {
- parseAListOfDeclarations: parseAListOfDeclarations
- };
- })();
|