container.js 10.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433
  1. 'use strict'
  2. let { isClean, my } = require('./symbols')
  3. let Declaration = require('./declaration')
  4. let Comment = require('./comment')
  5. let Node = require('./node')
  6. let parse, Rule, AtRule
  7. function cleanSource(nodes) {
  8. return nodes.map(i => {
  9. if (i.nodes) i.nodes = cleanSource(i.nodes)
  10. delete i.source
  11. return i
  12. })
  13. }
  14. function markDirtyUp(node) {
  15. node[isClean] = false
  16. if (node.proxyOf.nodes) {
  17. for (let i of node.proxyOf.nodes) {
  18. markDirtyUp(i)
  19. }
  20. }
  21. }
  22. class Container extends Node {
  23. push(child) {
  24. child.parent = this
  25. this.proxyOf.nodes.push(child)
  26. return this
  27. }
  28. each(callback) {
  29. if (!this.proxyOf.nodes) return undefined
  30. let iterator = this.getIterator()
  31. let index, result
  32. while (this.indexes[iterator] < this.proxyOf.nodes.length) {
  33. index = this.indexes[iterator]
  34. result = callback(this.proxyOf.nodes[index], index)
  35. if (result === false) break
  36. this.indexes[iterator] += 1
  37. }
  38. delete this.indexes[iterator]
  39. return result
  40. }
  41. walk(callback) {
  42. return this.each((child, i) => {
  43. let result
  44. try {
  45. result = callback(child, i)
  46. } catch (e) {
  47. throw child.addToError(e)
  48. }
  49. if (result !== false && child.walk) {
  50. result = child.walk(callback)
  51. }
  52. return result
  53. })
  54. }
  55. walkDecls(prop, callback) {
  56. if (!callback) {
  57. callback = prop
  58. return this.walk((child, i) => {
  59. if (child.type === 'decl') {
  60. return callback(child, i)
  61. }
  62. })
  63. }
  64. if (prop instanceof RegExp) {
  65. return this.walk((child, i) => {
  66. if (child.type === 'decl' && prop.test(child.prop)) {
  67. return callback(child, i)
  68. }
  69. })
  70. }
  71. return this.walk((child, i) => {
  72. if (child.type === 'decl' && child.prop === prop) {
  73. return callback(child, i)
  74. }
  75. })
  76. }
  77. walkRules(selector, callback) {
  78. if (!callback) {
  79. callback = selector
  80. return this.walk((child, i) => {
  81. if (child.type === 'rule') {
  82. return callback(child, i)
  83. }
  84. })
  85. }
  86. if (selector instanceof RegExp) {
  87. return this.walk((child, i) => {
  88. if (child.type === 'rule' && selector.test(child.selector)) {
  89. return callback(child, i)
  90. }
  91. })
  92. }
  93. return this.walk((child, i) => {
  94. if (child.type === 'rule' && child.selector === selector) {
  95. return callback(child, i)
  96. }
  97. })
  98. }
  99. walkAtRules(name, callback) {
  100. if (!callback) {
  101. callback = name
  102. return this.walk((child, i) => {
  103. if (child.type === 'atrule') {
  104. return callback(child, i)
  105. }
  106. })
  107. }
  108. if (name instanceof RegExp) {
  109. return this.walk((child, i) => {
  110. if (child.type === 'atrule' && name.test(child.name)) {
  111. return callback(child, i)
  112. }
  113. })
  114. }
  115. return this.walk((child, i) => {
  116. if (child.type === 'atrule' && child.name === name) {
  117. return callback(child, i)
  118. }
  119. })
  120. }
  121. walkComments(callback) {
  122. return this.walk((child, i) => {
  123. if (child.type === 'comment') {
  124. return callback(child, i)
  125. }
  126. })
  127. }
  128. append(...children) {
  129. for (let child of children) {
  130. let nodes = this.normalize(child, this.last)
  131. for (let node of nodes) this.proxyOf.nodes.push(node)
  132. }
  133. this.markDirty()
  134. return this
  135. }
  136. prepend(...children) {
  137. children = children.reverse()
  138. for (let child of children) {
  139. let nodes = this.normalize(child, this.first, 'prepend').reverse()
  140. for (let node of nodes) this.proxyOf.nodes.unshift(node)
  141. for (let id in this.indexes) {
  142. this.indexes[id] = this.indexes[id] + nodes.length
  143. }
  144. }
  145. this.markDirty()
  146. return this
  147. }
  148. cleanRaws(keepBetween) {
  149. super.cleanRaws(keepBetween)
  150. if (this.nodes) {
  151. for (let node of this.nodes) node.cleanRaws(keepBetween)
  152. }
  153. }
  154. insertBefore(exist, add) {
  155. exist = this.index(exist)
  156. let type = exist === 0 ? 'prepend' : false
  157. let nodes = this.normalize(add, this.proxyOf.nodes[exist], type).reverse()
  158. for (let node of nodes) this.proxyOf.nodes.splice(exist, 0, node)
  159. let index
  160. for (let id in this.indexes) {
  161. index = this.indexes[id]
  162. if (exist <= index) {
  163. this.indexes[id] = index + nodes.length
  164. }
  165. }
  166. this.markDirty()
  167. return this
  168. }
  169. insertAfter(exist, add) {
  170. exist = this.index(exist)
  171. let nodes = this.normalize(add, this.proxyOf.nodes[exist]).reverse()
  172. for (let node of nodes) this.proxyOf.nodes.splice(exist + 1, 0, node)
  173. let index
  174. for (let id in this.indexes) {
  175. index = this.indexes[id]
  176. if (exist < index) {
  177. this.indexes[id] = index + nodes.length
  178. }
  179. }
  180. this.markDirty()
  181. return this
  182. }
  183. removeChild(child) {
  184. child = this.index(child)
  185. this.proxyOf.nodes[child].parent = undefined
  186. this.proxyOf.nodes.splice(child, 1)
  187. let index
  188. for (let id in this.indexes) {
  189. index = this.indexes[id]
  190. if (index >= child) {
  191. this.indexes[id] = index - 1
  192. }
  193. }
  194. this.markDirty()
  195. return this
  196. }
  197. removeAll() {
  198. for (let node of this.proxyOf.nodes) node.parent = undefined
  199. this.proxyOf.nodes = []
  200. this.markDirty()
  201. return this
  202. }
  203. replaceValues(pattern, opts, callback) {
  204. if (!callback) {
  205. callback = opts
  206. opts = {}
  207. }
  208. this.walkDecls(decl => {
  209. if (opts.props && !opts.props.includes(decl.prop)) return
  210. if (opts.fast && !decl.value.includes(opts.fast)) return
  211. decl.value = decl.value.replace(pattern, callback)
  212. })
  213. this.markDirty()
  214. return this
  215. }
  216. every(condition) {
  217. return this.nodes.every(condition)
  218. }
  219. some(condition) {
  220. return this.nodes.some(condition)
  221. }
  222. index(child) {
  223. if (typeof child === 'number') return child
  224. if (child.proxyOf) child = child.proxyOf
  225. return this.proxyOf.nodes.indexOf(child)
  226. }
  227. get first() {
  228. if (!this.proxyOf.nodes) return undefined
  229. return this.proxyOf.nodes[0]
  230. }
  231. get last() {
  232. if (!this.proxyOf.nodes) return undefined
  233. return this.proxyOf.nodes[this.proxyOf.nodes.length - 1]
  234. }
  235. normalize(nodes, sample) {
  236. if (typeof nodes === 'string') {
  237. nodes = cleanSource(parse(nodes).nodes)
  238. } else if (Array.isArray(nodes)) {
  239. nodes = nodes.slice(0)
  240. for (let i of nodes) {
  241. if (i.parent) i.parent.removeChild(i, 'ignore')
  242. }
  243. } else if (nodes.type === 'root' && this.type !== 'document') {
  244. nodes = nodes.nodes.slice(0)
  245. for (let i of nodes) {
  246. if (i.parent) i.parent.removeChild(i, 'ignore')
  247. }
  248. } else if (nodes.type) {
  249. nodes = [nodes]
  250. } else if (nodes.prop) {
  251. if (typeof nodes.value === 'undefined') {
  252. throw new Error('Value field is missed in node creation')
  253. } else if (typeof nodes.value !== 'string') {
  254. nodes.value = String(nodes.value)
  255. }
  256. nodes = [new Declaration(nodes)]
  257. } else if (nodes.selector) {
  258. nodes = [new Rule(nodes)]
  259. } else if (nodes.name) {
  260. nodes = [new AtRule(nodes)]
  261. } else if (nodes.text) {
  262. nodes = [new Comment(nodes)]
  263. } else {
  264. throw new Error('Unknown node type in node creation')
  265. }
  266. let processed = nodes.map(i => {
  267. /* c8 ignore next */
  268. if (!i[my]) Container.rebuild(i)
  269. i = i.proxyOf
  270. if (i.parent) i.parent.removeChild(i)
  271. if (i[isClean]) markDirtyUp(i)
  272. if (typeof i.raws.before === 'undefined') {
  273. if (sample && typeof sample.raws.before !== 'undefined') {
  274. i.raws.before = sample.raws.before.replace(/\S/g, '')
  275. }
  276. }
  277. i.parent = this
  278. return i
  279. })
  280. return processed
  281. }
  282. getProxyProcessor() {
  283. return {
  284. set(node, prop, value) {
  285. if (node[prop] === value) return true
  286. node[prop] = value
  287. if (prop === 'name' || prop === 'params' || prop === 'selector') {
  288. node.markDirty()
  289. }
  290. return true
  291. },
  292. get(node, prop) {
  293. if (prop === 'proxyOf') {
  294. return node
  295. } else if (!node[prop]) {
  296. return node[prop]
  297. } else if (
  298. prop === 'each' ||
  299. (typeof prop === 'string' && prop.startsWith('walk'))
  300. ) {
  301. return (...args) => {
  302. return node[prop](
  303. ...args.map(i => {
  304. if (typeof i === 'function') {
  305. return (child, index) => i(child.toProxy(), index)
  306. } else {
  307. return i
  308. }
  309. })
  310. )
  311. }
  312. } else if (prop === 'every' || prop === 'some') {
  313. return cb => {
  314. return node[prop]((child, ...other) =>
  315. cb(child.toProxy(), ...other)
  316. )
  317. }
  318. } else if (prop === 'root') {
  319. return () => node.root().toProxy()
  320. } else if (prop === 'nodes') {
  321. return node.nodes.map(i => i.toProxy())
  322. } else if (prop === 'first' || prop === 'last') {
  323. return node[prop].toProxy()
  324. } else {
  325. return node[prop]
  326. }
  327. }
  328. }
  329. }
  330. getIterator() {
  331. if (!this.lastEach) this.lastEach = 0
  332. if (!this.indexes) this.indexes = {}
  333. this.lastEach += 1
  334. let iterator = this.lastEach
  335. this.indexes[iterator] = 0
  336. return iterator
  337. }
  338. }
  339. Container.registerParse = dependant => {
  340. parse = dependant
  341. }
  342. Container.registerRule = dependant => {
  343. Rule = dependant
  344. }
  345. Container.registerAtRule = dependant => {
  346. AtRule = dependant
  347. }
  348. module.exports = Container
  349. Container.default = Container
  350. /* c8 ignore start */
  351. Container.rebuild = node => {
  352. if (node.type === 'atrule') {
  353. Object.setPrototypeOf(node, AtRule.prototype)
  354. } else if (node.type === 'rule') {
  355. Object.setPrototypeOf(node, Rule.prototype)
  356. } else if (node.type === 'decl') {
  357. Object.setPrototypeOf(node, Declaration.prototype)
  358. } else if (node.type === 'comment') {
  359. Object.setPrototypeOf(node, Comment.prototype)
  360. }
  361. node[my] = true
  362. if (node.nodes) {
  363. node.nodes.forEach(child => {
  364. Container.rebuild(child)
  365. })
  366. }
  367. }
  368. /* c8 ignore stop */