/* @flow */ "use strict" const _ = require("lodash") const balancedMatch = require("balanced-match") /** * Replace all of the characters that are arguments to a certain * CSS function with some innocuous character. * * This is useful if you need to use a RegExp to find a string * but want to ignore matches in certain functions (e.g. `url()`, * which might contain all kinds of false positives). * * For example: * blurFunctionArguments("abc url(abc) abc", "url") === "abc url(```) abc" * * @param {string} source * @param {string} functionName * @param {[string]} blurChar="`" * @return {string} - The result string, with the function arguments "blurred" */ module.exports = function (source/*: string*/, functionName/*: string*/)/*: string*/ { const blurChar/*: string*/ = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : "`" const nameWithParen = `${functionName.toLowerCase()}(` const lowerCaseSource = source.toLowerCase() if (!_.includes(lowerCaseSource, nameWithParen)) { return source } const functionNameLength/*: number*/ = functionName.length let result = source let searchStartIndex = 0 while (lowerCaseSource.indexOf(nameWithParen, searchStartIndex) !== -1) { const openingParenIndex = lowerCaseSource.indexOf(nameWithParen, searchStartIndex) + functionNameLength const closingParenIndex = balancedMatch("(", ")", lowerCaseSource.slice(openingParenIndex)).end + openingParenIndex const argumentsLength = closingParenIndex - openingParenIndex - 1 result = result.slice(0, openingParenIndex + 1) + _.repeat(blurChar, argumentsLength) + result.slice(closingParenIndex) searchStartIndex = closingParenIndex } return result }