util.js 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  1. 'use strict'
  2. var config
  3. var CHAR_COMMENT_REPLACEMENT = '\uFFFD' // �
  4. var CHAR_TOKEN_REPLACEMENT = '\u00A4'// ¤
  5. var CHAR_TOKEN_START = '\u00AB' // «
  6. var CHAR_TOKEN_END = '\u00BB' // »
  7. var REGEX_COMMENT_REPLACEMENT = new RegExp(CHAR_COMMENT_REPLACEMENT, 'ig')
  8. var REGEX_TOKEN_REPLACEMENT = new RegExp(CHAR_TOKEN_REPLACEMENT, 'ig')
  9. var PATTERN_NUMBER = '\\-?(\\d*?\\.\\d+|\\d+)'
  10. var PATTERN_NUMBER_WITH_CALC = '(calc' + CHAR_TOKEN_REPLACEMENT + ')|(' + PATTERN_NUMBER + ')(?!d\\()'
  11. var PATTERN_TOKEN = CHAR_TOKEN_START + '\\d+:\\d+' + CHAR_TOKEN_END // «offset:index»
  12. var PATTERN_TOKEN_WITH_NAME = '\\w*?' + CHAR_TOKEN_START + '\\d+:\\d+' + CHAR_TOKEN_END // «offset:index»
  13. var REGEX_COMMENT = /\/\*[^]*?\*\//igm // none-greedy
  14. var REGEX_DIRECTIVE = /\/\*(?:!)?rtl:[^]*?\*\//img
  15. var REGEX_ESCAPE = /[.*+?^${}()|[\]\\]/g
  16. var REGEX_FUNCTION = /\([^\(\)]+\)/i
  17. var REGEX_HEX_COLOR = /#[a-f0-9]{3,6}/ig
  18. var REGEX_CALC = /calc/
  19. var REGEX_TOKENS = new RegExp(PATTERN_TOKEN, 'ig')
  20. var REGEX_TOKENS_WITH_NAME = new RegExp(PATTERN_TOKEN_WITH_NAME, 'ig')
  21. var REGEX_COMPLEMENT = new RegExp(PATTERN_NUMBER_WITH_CALC, 'i')
  22. var REGEX_NEGATE_ALL = new RegExp(PATTERN_NUMBER_WITH_CALC, 'ig')
  23. var REGEX_NEGATE_ONE = new RegExp(PATTERN_NUMBER_WITH_CALC, 'i')
  24. var DEFAULT_STRING_MAP_OPTIONS = { scope: '*', ignoreCase: true }
  25. var TOKEN_ID = 0
  26. function compare (what, to, ignoreCase) {
  27. if (ignoreCase) {
  28. return what.toLowerCase() === to.toLowerCase()
  29. }
  30. return what === to
  31. }
  32. function escapeRegExp (string) {
  33. return string.replace(REGEX_ESCAPE, '\\$&')
  34. }
  35. module.exports = {
  36. extend: function (dest, src) {
  37. if (typeof dest === 'undefined' || typeof dest !== 'object') {
  38. dest = {}
  39. }
  40. for (var prop in src) {
  41. if (!dest.hasOwnProperty(prop)) {
  42. dest[prop] = src[prop]
  43. }
  44. }
  45. return dest
  46. },
  47. swap: function (value, a, b, options) {
  48. var expr = escapeRegExp(a) + '|' + escapeRegExp(b)
  49. options = options || DEFAULT_STRING_MAP_OPTIONS
  50. var greedy = options.hasOwnProperty('greedy') ? options.greedy : config.greedy
  51. if (!greedy) {
  52. expr = '\\b(' + expr + ')\\b'
  53. }
  54. var flags = options.ignoreCase ? 'img' : 'mg'
  55. return value.replace(new RegExp(expr, flags), function (m) { return compare(m, a, options.ignoreCase) ? b : a })
  56. },
  57. swapLeftRight: function (value) {
  58. return this.swap(value, 'left', 'right')
  59. },
  60. swapLtrRtl: function (value) {
  61. return this.swap(value, 'ltr', 'rtl')
  62. },
  63. applyStringMap: function (value, isUrl) {
  64. var result = value
  65. for (var x = 0; x < config.stringMap.length; x++) {
  66. var map = config.stringMap[x]
  67. var options = this.extend(map.options, DEFAULT_STRING_MAP_OPTIONS)
  68. if (options.scope === '*' || (isUrl && options.scope === 'url') || (!isUrl && options.scope === 'selector')) {
  69. if (Array.isArray(map.search) && Array.isArray(map.replace)) {
  70. for (var mapIndex = 0; mapIndex < map.search.length; mapIndex++) {
  71. result = this.swap(result, map.search[mapIndex], map.replace[mapIndex % map.search.length], options)
  72. }
  73. } else {
  74. result = this.swap(result, map.search, map.replace, options)
  75. }
  76. if (map.exclusive === true) {
  77. break
  78. }
  79. }
  80. }
  81. return result
  82. },
  83. negate: function (value) {
  84. var state = this.saveTokens(value)
  85. state.value = state.value.replace(REGEX_NEGATE_ONE, function (num) {
  86. return REGEX_TOKEN_REPLACEMENT.test(num) ? num.replace(REGEX_TOKEN_REPLACEMENT, function (m) { return '(-1*' + m + ')' }) : parseFloat(num, 10) * -1
  87. })
  88. return this.restoreTokens(state)
  89. },
  90. negateAll: function (value) {
  91. var state = this.saveTokens(value)
  92. state.value = state.value.replace(REGEX_NEGATE_ALL, function (num) {
  93. return REGEX_TOKEN_REPLACEMENT.test(num) ? num.replace(REGEX_TOKEN_REPLACEMENT, function (m) { return '(-1*' + m + ')' }) : parseFloat(num, 10) * -1
  94. })
  95. return this.restoreTokens(state)
  96. },
  97. complement: function (value) {
  98. var state = this.saveTokens(value)
  99. state.value = state.value.replace(REGEX_COMPLEMENT, function (num) {
  100. return REGEX_TOKEN_REPLACEMENT.test(num) ? num.replace(REGEX_TOKEN_REPLACEMENT, function (m) { return '(100% - ' + m + ')' }) : 100 - parseFloat(num, 10)
  101. })
  102. return this.restoreTokens(state)
  103. },
  104. save: function (what, who, replacement, restorer, exclude) {
  105. var state = {
  106. value: who,
  107. store: [],
  108. replacement: replacement,
  109. restorer: restorer
  110. }
  111. state.value = state.value.replace(what, function (c) {
  112. if (exclude && c.match(exclude)) {
  113. return c
  114. } else {
  115. state.store.push(c); return state.replacement
  116. }
  117. })
  118. return state
  119. },
  120. restore: function (state) {
  121. var index = 0
  122. var result = state.value.replace(state.restorer, function () {
  123. return state.store[index++]
  124. })
  125. state.store.length = 0
  126. return result
  127. },
  128. saveComments: function (value) {
  129. return this.save(REGEX_COMMENT, value, CHAR_COMMENT_REPLACEMENT, REGEX_COMMENT_REPLACEMENT)
  130. },
  131. restoreComments: function (state) {
  132. return this.restore(state)
  133. },
  134. saveTokens: function (value, excludeCalc) {
  135. return excludeCalc === true
  136. ? this.save(REGEX_TOKENS_WITH_NAME, value, CHAR_TOKEN_REPLACEMENT, REGEX_TOKEN_REPLACEMENT, REGEX_CALC)
  137. : this.save(REGEX_TOKENS, value, CHAR_TOKEN_REPLACEMENT, REGEX_TOKEN_REPLACEMENT)
  138. },
  139. restoreTokens: function (state) {
  140. return this.restore(state)
  141. },
  142. guard: function (what, who, indexed) {
  143. var state = {
  144. value: who,
  145. store: [],
  146. offset: TOKEN_ID++,
  147. token: CHAR_TOKEN_START + TOKEN_ID,
  148. indexed: indexed === true
  149. }
  150. if (state.indexed === true) {
  151. while (what.test(state.value)) {
  152. state.value = state.value.replace(what, function (m) { state.store.push(m); return state.token + ':' + state.store.length + CHAR_TOKEN_END })
  153. }
  154. } else {
  155. state.value = state.value.replace(what, function (m) { state.store.push(m); return state.token + CHAR_TOKEN_END })
  156. }
  157. return state
  158. },
  159. unguard: function (state, callback) {
  160. if (state.indexed === true) {
  161. var detokenizer = new RegExp('(\\w*?)' + state.token + ':(\\d+)' + CHAR_TOKEN_END, 'i')
  162. while (detokenizer.test(state.value)) {
  163. state.value = state.value.replace(detokenizer, function (match, name, index) {
  164. var value = state.store[index - 1]
  165. if (typeof callback === 'function') {
  166. return name + callback(value, name)
  167. }
  168. return name + value
  169. })
  170. }
  171. return state.value
  172. } else {
  173. return state.value.replace(new RegExp('(\\w*?)' + state.token + CHAR_TOKEN_END, 'i'), function (match, name) {
  174. var value = state.store.shift()
  175. if (typeof callback === 'function') {
  176. return name + callback(value, name)
  177. }
  178. return name + value
  179. })
  180. }
  181. },
  182. guardHexColors: function (value) {
  183. return this.guard(REGEX_HEX_COLOR, value, true)
  184. },
  185. unguardHexColors: function (state, callback) {
  186. return this.unguard(state, callback)
  187. },
  188. guardFunctions: function (value) {
  189. return this.guard(REGEX_FUNCTION, value, true)
  190. },
  191. unguardFunctions: function (state, callback) {
  192. return this.unguard(state, callback)
  193. },
  194. trimDirective: function (value) {
  195. return value.replace(REGEX_DIRECTIVE, '')
  196. },
  197. regexCache: {},
  198. regexDirective: function (name) {
  199. // /(?:\/\*(?:!)?rtl:ignore(?::)?)([^]*?)(?:\*\/)/img
  200. this.regexCache[name] = this.regexCache[name] || new RegExp('(?:\\/\\*(?:!)?rtl:' + (name ? escapeRegExp(name) + '(?::)?' : '') + ')([^]*?)(?:\\*\\/)', 'img')
  201. return this.regexCache[name]
  202. },
  203. regex: function (what, options) {
  204. what = what || []
  205. var expression = ''
  206. for (var x = 0; x < what.length; x++) {
  207. switch (what[x]) {
  208. case 'percent':
  209. expression += '|(' + PATTERN_NUMBER + '%)'
  210. break
  211. case 'length':
  212. expression += '|(' + PATTERN_NUMBER + ')(?:ex|ch|r?em|vh|vw|vmin|vmax|px|mm|cm|in|pt|pc)?'
  213. break
  214. case 'number':
  215. expression += '|(' + PATTERN_NUMBER + ')'
  216. break
  217. case 'position':
  218. expression += '|(left|center|right|top|bottom)'
  219. break
  220. case 'calc':
  221. expression += '|(calc' + PATTERN_TOKEN + ')'
  222. break
  223. }
  224. }
  225. return new RegExp(expression.slice(1), options)
  226. },
  227. isLastOfType: function (node) {
  228. var isLast = true
  229. var next = node.next()
  230. while (next) {
  231. if (next && next.type === node.type) {
  232. isLast = false
  233. break
  234. }
  235. next = next.next()
  236. }
  237. return isLast
  238. },
  239. /**
  240. * Simple breakable each: returning false in the callback will break the loop
  241. * returns false if the loop was broken, otherwise true
  242. */
  243. each: function (array, callback) {
  244. for (var len = 0; len < array.length; len++) {
  245. if (callback(array[len]) === false) {
  246. return false
  247. }
  248. }
  249. return true
  250. }
  251. }
  252. module.exports.configure = function (configuration) {
  253. config = configuration
  254. return this
  255. }