Spaces:
Running
Running
| var openParentheses = "(".charCodeAt(0); | |
| var closeParentheses = ")".charCodeAt(0); | |
| var singleQuote = "'".charCodeAt(0); | |
| var doubleQuote = '"'.charCodeAt(0); | |
| var backslash = "\\".charCodeAt(0); | |
| var slash = "/".charCodeAt(0); | |
| var comma = ",".charCodeAt(0); | |
| var colon = ":".charCodeAt(0); | |
| var star = "*".charCodeAt(0); | |
| var uLower = "u".charCodeAt(0); | |
| var uUpper = "U".charCodeAt(0); | |
| var plus = "+".charCodeAt(0); | |
| var isUnicodeRange = /^[a-f0-9?-]+$/i; | |
| module.exports = function(input) { | |
| var tokens = []; | |
| var value = input; | |
| var next, | |
| quote, | |
| prev, | |
| token, | |
| escape, | |
| escapePos, | |
| whitespacePos, | |
| parenthesesOpenPos; | |
| var pos = 0; | |
| var code = value.charCodeAt(pos); | |
| var max = value.length; | |
| var stack = [{ nodes: tokens }]; | |
| var balanced = 0; | |
| var parent; | |
| var name = ""; | |
| var before = ""; | |
| var after = ""; | |
| while (pos < max) { | |
| // Whitespaces | |
| if (code <= 32) { | |
| next = pos; | |
| do { | |
| next += 1; | |
| code = value.charCodeAt(next); | |
| } while (code <= 32); | |
| token = value.slice(pos, next); | |
| prev = tokens[tokens.length - 1]; | |
| if (code === closeParentheses && balanced) { | |
| after = token; | |
| } else if (prev && prev.type === "div") { | |
| prev.after = token; | |
| prev.sourceEndIndex += token.length; | |
| } else if ( | |
| code === comma || | |
| code === colon || | |
| (code === slash && | |
| value.charCodeAt(next + 1) !== star && | |
| (!parent || | |
| (parent && parent.type === "function" && parent.value !== "calc"))) | |
| ) { | |
| before = token; | |
| } else { | |
| tokens.push({ | |
| type: "space", | |
| sourceIndex: pos, | |
| sourceEndIndex: next, | |
| value: token | |
| }); | |
| } | |
| pos = next; | |
| // Quotes | |
| } else if (code === singleQuote || code === doubleQuote) { | |
| next = pos; | |
| quote = code === singleQuote ? "'" : '"'; | |
| token = { | |
| type: "string", | |
| sourceIndex: pos, | |
| quote: quote | |
| }; | |
| do { | |
| escape = false; | |
| next = value.indexOf(quote, next + 1); | |
| if (~next) { | |
| escapePos = next; | |
| while (value.charCodeAt(escapePos - 1) === backslash) { | |
| escapePos -= 1; | |
| escape = !escape; | |
| } | |
| } else { | |
| value += quote; | |
| next = value.length - 1; | |
| token.unclosed = true; | |
| } | |
| } while (escape); | |
| token.value = value.slice(pos + 1, next); | |
| token.sourceEndIndex = token.unclosed ? next : next + 1; | |
| tokens.push(token); | |
| pos = next + 1; | |
| code = value.charCodeAt(pos); | |
| // Comments | |
| } else if (code === slash && value.charCodeAt(pos + 1) === star) { | |
| next = value.indexOf("*/", pos); | |
| token = { | |
| type: "comment", | |
| sourceIndex: pos, | |
| sourceEndIndex: next + 2 | |
| }; | |
| if (next === -1) { | |
| token.unclosed = true; | |
| next = value.length; | |
| token.sourceEndIndex = next; | |
| } | |
| token.value = value.slice(pos + 2, next); | |
| tokens.push(token); | |
| pos = next + 2; | |
| code = value.charCodeAt(pos); | |
| // Operation within calc | |
| } else if ( | |
| (code === slash || code === star) && | |
| parent && | |
| parent.type === "function" && | |
| parent.value === "calc" | |
| ) { | |
| token = value[pos]; | |
| tokens.push({ | |
| type: "word", | |
| sourceIndex: pos - before.length, | |
| sourceEndIndex: pos + token.length, | |
| value: token | |
| }); | |
| pos += 1; | |
| code = value.charCodeAt(pos); | |
| // Dividers | |
| } else if (code === slash || code === comma || code === colon) { | |
| token = value[pos]; | |
| tokens.push({ | |
| type: "div", | |
| sourceIndex: pos - before.length, | |
| sourceEndIndex: pos + token.length, | |
| value: token, | |
| before: before, | |
| after: "" | |
| }); | |
| before = ""; | |
| pos += 1; | |
| code = value.charCodeAt(pos); | |
| // Open parentheses | |
| } else if (openParentheses === code) { | |
| // Whitespaces after open parentheses | |
| next = pos; | |
| do { | |
| next += 1; | |
| code = value.charCodeAt(next); | |
| } while (code <= 32); | |
| parenthesesOpenPos = pos; | |
| token = { | |
| type: "function", | |
| sourceIndex: pos - name.length, | |
| value: name, | |
| before: value.slice(parenthesesOpenPos + 1, next) | |
| }; | |
| pos = next; | |
| if (name === "url" && code !== singleQuote && code !== doubleQuote) { | |
| next -= 1; | |
| do { | |
| escape = false; | |
| next = value.indexOf(")", next + 1); | |
| if (~next) { | |
| escapePos = next; | |
| while (value.charCodeAt(escapePos - 1) === backslash) { | |
| escapePos -= 1; | |
| escape = !escape; | |
| } | |
| } else { | |
| value += ")"; | |
| next = value.length - 1; | |
| token.unclosed = true; | |
| } | |
| } while (escape); | |
| // Whitespaces before closed | |
| whitespacePos = next; | |
| do { | |
| whitespacePos -= 1; | |
| code = value.charCodeAt(whitespacePos); | |
| } while (code <= 32); | |
| if (parenthesesOpenPos < whitespacePos) { | |
| if (pos !== whitespacePos + 1) { | |
| token.nodes = [ | |
| { | |
| type: "word", | |
| sourceIndex: pos, | |
| sourceEndIndex: whitespacePos + 1, | |
| value: value.slice(pos, whitespacePos + 1) | |
| } | |
| ]; | |
| } else { | |
| token.nodes = []; | |
| } | |
| if (token.unclosed && whitespacePos + 1 !== next) { | |
| token.after = ""; | |
| token.nodes.push({ | |
| type: "space", | |
| sourceIndex: whitespacePos + 1, | |
| sourceEndIndex: next, | |
| value: value.slice(whitespacePos + 1, next) | |
| }); | |
| } else { | |
| token.after = value.slice(whitespacePos + 1, next); | |
| token.sourceEndIndex = next; | |
| } | |
| } else { | |
| token.after = ""; | |
| token.nodes = []; | |
| } | |
| pos = next + 1; | |
| token.sourceEndIndex = token.unclosed ? next : pos; | |
| code = value.charCodeAt(pos); | |
| tokens.push(token); | |
| } else { | |
| balanced += 1; | |
| token.after = ""; | |
| token.sourceEndIndex = pos + 1; | |
| tokens.push(token); | |
| stack.push(token); | |
| tokens = token.nodes = []; | |
| parent = token; | |
| } | |
| name = ""; | |
| // Close parentheses | |
| } else if (closeParentheses === code && balanced) { | |
| pos += 1; | |
| code = value.charCodeAt(pos); | |
| parent.after = after; | |
| parent.sourceEndIndex += after.length; | |
| after = ""; | |
| balanced -= 1; | |
| stack[stack.length - 1].sourceEndIndex = pos; | |
| stack.pop(); | |
| parent = stack[balanced]; | |
| tokens = parent.nodes; | |
| // Words | |
| } else { | |
| next = pos; | |
| do { | |
| if (code === backslash) { | |
| next += 1; | |
| } | |
| next += 1; | |
| code = value.charCodeAt(next); | |
| } while ( | |
| next < max && | |
| !( | |
| code <= 32 || | |
| code === singleQuote || | |
| code === doubleQuote || | |
| code === comma || | |
| code === colon || | |
| code === slash || | |
| code === openParentheses || | |
| (code === star && | |
| parent && | |
| parent.type === "function" && | |
| parent.value === "calc") || | |
| (code === slash && | |
| parent.type === "function" && | |
| parent.value === "calc") || | |
| (code === closeParentheses && balanced) | |
| ) | |
| ); | |
| token = value.slice(pos, next); | |
| if (openParentheses === code) { | |
| name = token; | |
| } else if ( | |
| (uLower === token.charCodeAt(0) || uUpper === token.charCodeAt(0)) && | |
| plus === token.charCodeAt(1) && | |
| isUnicodeRange.test(token.slice(2)) | |
| ) { | |
| tokens.push({ | |
| type: "unicode-range", | |
| sourceIndex: pos, | |
| sourceEndIndex: next, | |
| value: token | |
| }); | |
| } else { | |
| tokens.push({ | |
| type: "word", | |
| sourceIndex: pos, | |
| sourceEndIndex: next, | |
| value: token | |
| }); | |
| } | |
| pos = next; | |
| } | |
| } | |
| for (pos = stack.length - 1; pos; pos -= 1) { | |
| stack[pos].unclosed = true; | |
| stack[pos].sourceEndIndex = value.length; | |
| } | |
| return stack[0].nodes; | |
| }; | |