@jesscss/core 2.0.0-alpha.5 → 2.0.0-alpha.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (637) hide show
  1. package/lib/index.cjs +20159 -0
  2. package/lib/index.d.cts +5993 -0
  3. package/lib/index.d.cts.map +1 -0
  4. package/lib/index.d.ts +5992 -21
  5. package/lib/index.d.ts.map +1 -1
  6. package/lib/index.js +19926 -22
  7. package/lib/index.js.map +1 -1
  8. package/package.json +15 -14
  9. package/src/__tests__/define-function-record.test.ts +58 -0
  10. package/src/__tests__/define-function-simple.test.ts +55 -0
  11. package/src/__tests__/define-function-split-sequence.test.ts +547 -0
  12. package/src/__tests__/define-function-type-parity.test.ts +9 -0
  13. package/src/__tests__/define-function.test.ts +763 -0
  14. package/src/__tests__/num-operations.test.ts +91 -0
  15. package/src/__tests__/safe-parse.test.ts +374 -0
  16. package/src/context.ts +896 -0
  17. package/src/conversions.ts +282 -0
  18. package/src/debug-log.ts +29 -0
  19. package/src/define-function.ts +1006 -0
  20. package/src/deprecation.ts +67 -0
  21. package/src/globals.d.ts +26 -0
  22. package/src/index.ts +31 -0
  23. package/src/jess-error.ts +773 -0
  24. package/src/logger/deprecation-processing.ts +109 -0
  25. package/src/logger.ts +31 -0
  26. package/src/plugin.ts +292 -0
  27. package/src/tree/LOOKUP_CHAINS.md +35 -0
  28. package/src/tree/README.md +18 -0
  29. package/src/tree/__tests__/__snapshots__/extend-eval-integration.test.ts.snap +1455 -0
  30. package/src/tree/__tests__/ampersand.test.ts +382 -0
  31. package/src/tree/__tests__/at-rule.test.ts +2047 -0
  32. package/src/tree/__tests__/basic-render.test.ts +212 -0
  33. package/src/tree/__tests__/block.test.ts +40 -0
  34. package/src/tree/__tests__/call.test.ts +346 -0
  35. package/src/tree/__tests__/color.test.ts +537 -0
  36. package/src/tree/__tests__/condition.test.ts +186 -0
  37. package/src/tree/__tests__/control.test.ts +564 -0
  38. package/src/tree/__tests__/declaration.test.ts +253 -0
  39. package/src/tree/__tests__/dependency-graph.test.ts +177 -0
  40. package/src/tree/__tests__/detached-rulesets.test.ts +213 -0
  41. package/src/tree/__tests__/dimension.test.ts +236 -0
  42. package/src/tree/__tests__/expression.test.ts +73 -0
  43. package/src/tree/__tests__/ext-node.test.ts +31 -0
  44. package/src/tree/__tests__/extend-eval-integration.test.ts +1033 -0
  45. package/src/tree/__tests__/extend-import-style.test.ts +929 -0
  46. package/src/tree/__tests__/extend-less-fixtures.test.ts +851 -0
  47. package/src/tree/__tests__/extend-list.test.ts +31 -0
  48. package/src/tree/__tests__/extend-roots.test.ts +1045 -0
  49. package/src/tree/__tests__/extend-rules.test.ts +740 -0
  50. package/src/tree/__tests__/func.test.ts +171 -0
  51. package/src/tree/__tests__/import-js.test.ts +33 -0
  52. package/src/tree/__tests__/import-style-test-helpers.ts +56 -0
  53. package/src/tree/__tests__/import-style.test.ts +1967 -0
  54. package/src/tree/__tests__/interpolated-reference.test.ts +44 -0
  55. package/src/tree/__tests__/interpolated.test.ts +41 -0
  56. package/src/tree/__tests__/list.test.ts +177 -0
  57. package/src/tree/__tests__/log.test.ts +83 -0
  58. package/src/tree/__tests__/mixin-recursion.test.ts +639 -0
  59. package/src/tree/__tests__/mixin.test.ts +2171 -0
  60. package/src/tree/__tests__/negative.test.ts +45 -0
  61. package/src/tree/__tests__/nesting-collapse.test.ts +519 -0
  62. package/src/tree/__tests__/node-flags-perf.test.ts +195 -0
  63. package/src/tree/__tests__/node-flags.test.ts +410 -0
  64. package/src/tree/__tests__/node-graph.test.ts +598 -0
  65. package/src/tree/__tests__/node-mutation.test.ts +182 -0
  66. package/src/tree/__tests__/operation.test.ts +18 -0
  67. package/src/tree/__tests__/paren.test.ts +90 -0
  68. package/src/tree/__tests__/preserve-mode-output.test.ts +50 -0
  69. package/src/tree/__tests__/quoted.test.ts +72 -0
  70. package/src/tree/__tests__/range.test.ts +59 -0
  71. package/src/tree/__tests__/reference.test.ts +743 -0
  72. package/src/tree/__tests__/rest.test.ts +29 -0
  73. package/src/tree/__tests__/rules-raw.test.ts +14 -0
  74. package/src/tree/__tests__/rules.test.ts +1271 -0
  75. package/src/tree/__tests__/ruleset.test.ts +597 -0
  76. package/src/tree/__tests__/selector-attr.test.ts +50 -0
  77. package/src/tree/__tests__/selector-basic.test.ts +44 -0
  78. package/src/tree/__tests__/selector-capture.test.ts +22 -0
  79. package/src/tree/__tests__/selector-complex.test.ts +120 -0
  80. package/src/tree/__tests__/selector-compound.test.ts +74 -0
  81. package/src/tree/__tests__/selector-interpolated.test.ts +50 -0
  82. package/src/tree/__tests__/selector-list.test.ts +59 -0
  83. package/src/tree/__tests__/selector-pseudo.test.ts +23 -0
  84. package/src/tree/__tests__/selector.test.ts +182 -0
  85. package/src/tree/__tests__/sequence.test.ts +226 -0
  86. package/src/tree/__tests__/serialize-types.test.ts +529 -0
  87. package/src/tree/__tests__/spaced.test.ts +8 -0
  88. package/src/tree/__tests__/url.test.ts +72 -0
  89. package/src/tree/__tests__/var-declaration.test.ts +90 -0
  90. package/src/tree/ampersand.ts +538 -0
  91. package/src/tree/any.ts +169 -0
  92. package/src/tree/at-rule.ts +760 -0
  93. package/src/tree/block.ts +72 -0
  94. package/src/tree/bool.ts +46 -0
  95. package/src/tree/call.ts +593 -0
  96. package/src/tree/collection.ts +52 -0
  97. package/src/tree/color.ts +629 -0
  98. package/src/tree/combinator.ts +30 -0
  99. package/src/tree/comment.ts +36 -0
  100. package/src/tree/condition.ts +194 -0
  101. package/src/tree/control.ts +452 -0
  102. package/src/tree/declaration-custom.ts +56 -0
  103. package/src/tree/declaration-var.ts +87 -0
  104. package/src/tree/declaration.ts +742 -0
  105. package/src/tree/default-guard.ts +35 -0
  106. package/src/tree/dimension.ts +392 -0
  107. package/src/tree/expression.ts +97 -0
  108. package/src/tree/extend-list.ts +51 -0
  109. package/src/tree/extend.ts +391 -0
  110. package/src/tree/function.ts +254 -0
  111. package/src/tree/import-js.ts +130 -0
  112. package/src/tree/import-style.ts +875 -0
  113. package/{lib/tree/index.js → src/tree/index.ts} +49 -22
  114. package/src/tree/interpolated.ts +346 -0
  115. package/src/tree/js-array.ts +21 -0
  116. package/src/tree/js-expr.ts +50 -0
  117. package/src/tree/js-function.ts +31 -0
  118. package/src/tree/js-object.ts +22 -0
  119. package/src/tree/list.ts +415 -0
  120. package/src/tree/log.ts +89 -0
  121. package/src/tree/mixin.ts +331 -0
  122. package/src/tree/negative.ts +58 -0
  123. package/src/tree/nil.ts +57 -0
  124. package/src/tree/node-base.ts +1716 -0
  125. package/src/tree/node-type.ts +122 -0
  126. package/src/tree/node.ts +118 -0
  127. package/src/tree/number.ts +54 -0
  128. package/src/tree/operation.ts +187 -0
  129. package/src/tree/paren.ts +132 -0
  130. package/src/tree/query-condition.ts +47 -0
  131. package/src/tree/quoted.ts +119 -0
  132. package/src/tree/range.ts +101 -0
  133. package/src/tree/reference.ts +1099 -0
  134. package/src/tree/rest.ts +55 -0
  135. package/src/tree/rules-raw.ts +52 -0
  136. package/src/tree/rules.ts +2896 -0
  137. package/src/tree/ruleset.ts +1217 -0
  138. package/src/tree/selector-attr.ts +172 -0
  139. package/src/tree/selector-basic.ts +75 -0
  140. package/src/tree/selector-capture.ts +85 -0
  141. package/src/tree/selector-complex.ts +189 -0
  142. package/src/tree/selector-compound.ts +205 -0
  143. package/src/tree/selector-interpolated.ts +95 -0
  144. package/src/tree/selector-list.ts +245 -0
  145. package/src/tree/selector-pseudo.ts +173 -0
  146. package/src/tree/selector-simple.ts +10 -0
  147. package/src/tree/selector.ts +152 -0
  148. package/src/tree/sequence.ts +463 -0
  149. package/src/tree/tree.ts +130 -0
  150. package/src/tree/url.ts +95 -0
  151. package/src/tree/util/EXTEND_ARCHITECTURE_ANALYSIS.md +215 -0
  152. package/src/tree/util/EXTEND_AUDIT.md +233 -0
  153. package/src/tree/util/EXTEND_BASELINE.md +64 -0
  154. package/src/tree/util/EXTEND_CALL_GRAPH_ANALYSIS.md +244 -0
  155. package/src/tree/util/EXTEND_DOCS.md +24 -0
  156. package/src/tree/util/EXTEND_FINAL_SUMMARY.md +95 -0
  157. package/src/tree/util/EXTEND_FUNCTION_AUDIT.md +1433 -0
  158. package/src/tree/util/EXTEND_OPTIMIZATION_PLAN.md +114 -0
  159. package/src/tree/util/EXTEND_REFACTORING_SUMMARY.md +152 -0
  160. package/src/tree/util/EXTEND_RULES.md +74 -0
  161. package/src/tree/util/EXTEND_UNUSED_FUNCTIONS.md +127 -0
  162. package/src/tree/util/EXTEND_UNUSED_FUNCTIONS_ANALYSIS.md +227 -0
  163. package/src/tree/util/NODE_COPY_REDUCTION_PLAN.md +12 -0
  164. package/src/tree/util/__tests__/EXTEND_TEST_INDEX.md +59 -0
  165. package/src/tree/util/__tests__/OPTIMIZATION-ANALYSIS.md +130 -0
  166. package/src/tree/util/__tests__/WALK_AND_CONSUME_DESIGN.md +138 -0
  167. package/src/tree/util/__tests__/_archive/2026-02-09__OPTIMIZATION-ANALYSIS.md +9 -0
  168. package/src/tree/util/__tests__/_archive/README.md +4 -0
  169. package/src/tree/util/__tests__/bitset.test.ts +142 -0
  170. package/src/tree/util/__tests__/debug-log.ts +50 -0
  171. package/src/tree/util/__tests__/extend-comment-handling.test.ts +187 -0
  172. package/src/tree/util/__tests__/extend-core-unit.test.ts +941 -0
  173. package/src/tree/util/__tests__/extend-pipeline-bench.test.ts +154 -0
  174. package/src/tree/util/__tests__/extend-pipeline-bench.ts +190 -0
  175. package/src/tree/util/__tests__/fast-reject.test.ts +377 -0
  176. package/src/tree/util/__tests__/is-node.test.ts +63 -0
  177. package/src/tree/util/__tests__/list-like.test.ts +63 -0
  178. package/src/tree/util/__tests__/outputwriter.test.ts +523 -0
  179. package/src/tree/util/__tests__/print.test.ts +183 -0
  180. package/src/tree/util/__tests__/process-extends.test.ts +226 -0
  181. package/src/tree/util/__tests__/process-leading-is.test.ts +205 -0
  182. package/src/tree/util/__tests__/recursion-helper.test.ts +184 -0
  183. package/src/tree/util/__tests__/selector-match-unit.test.ts +1427 -0
  184. package/src/tree/util/__tests__/sourcemap.test.ts +117 -0
  185. package/src/tree/util/ampersand-template.ts +9 -0
  186. package/src/tree/util/bitset.ts +194 -0
  187. package/src/tree/util/calculate.ts +11 -0
  188. package/src/tree/util/cast.ts +89 -0
  189. package/src/tree/util/cloning.ts +8 -0
  190. package/src/tree/util/collections.ts +299 -0
  191. package/src/tree/util/compare.ts +90 -0
  192. package/src/tree/util/cursor.ts +171 -0
  193. package/src/tree/util/extend-core.ts +2139 -0
  194. package/src/tree/util/extend-roots.ts +1108 -0
  195. package/src/tree/util/field-helpers.ts +354 -0
  196. package/src/tree/util/is-node.ts +43 -0
  197. package/src/tree/util/list-like.ts +93 -0
  198. package/src/tree/util/mixin-instance-primitives.ts +2020 -0
  199. package/src/tree/util/print.ts +303 -0
  200. package/src/tree/util/process-leading-is.ts +421 -0
  201. package/src/tree/util/recursion-helper.ts +54 -0
  202. package/src/tree/util/regex.ts +2 -0
  203. package/src/tree/util/registry-utils.ts +1953 -0
  204. package/src/tree/util/ruleset-trace.ts +17 -0
  205. package/src/tree/util/scoped-body-eval.ts +320 -0
  206. package/src/tree/util/selector-match-core.ts +2005 -0
  207. package/src/tree/util/selector-utils.ts +757 -0
  208. package/src/tree/util/serialize-helper.ts +535 -0
  209. package/src/tree/util/serialize-types.ts +318 -0
  210. package/src/tree/util/should-operate.ts +78 -0
  211. package/src/tree/util/sourcemap.ts +37 -0
  212. package/src/types/config.ts +247 -0
  213. package/src/types/index.ts +12 -0
  214. package/{lib/types/modes.d.ts → src/types/modes.ts} +2 -1
  215. package/src/types.d.ts +9 -0
  216. package/src/types.ts +68 -0
  217. package/src/use-webpack-resolver.ts +56 -0
  218. package/src/visitor/__tests__/visitor.test.ts +136 -0
  219. package/src/visitor/index.ts +263 -0
  220. package/{lib/visitor/less-visitor.js → src/visitor/less-visitor.ts} +3 -2
  221. package/lib/context.d.ts +0 -352
  222. package/lib/context.d.ts.map +0 -1
  223. package/lib/context.js +0 -636
  224. package/lib/context.js.map +0 -1
  225. package/lib/conversions.d.ts +0 -73
  226. package/lib/conversions.d.ts.map +0 -1
  227. package/lib/conversions.js +0 -253
  228. package/lib/conversions.js.map +0 -1
  229. package/lib/debug-log.d.ts +0 -2
  230. package/lib/debug-log.d.ts.map +0 -1
  231. package/lib/debug-log.js +0 -27
  232. package/lib/debug-log.js.map +0 -1
  233. package/lib/define-function.d.ts +0 -587
  234. package/lib/define-function.d.ts.map +0 -1
  235. package/lib/define-function.js +0 -726
  236. package/lib/define-function.js.map +0 -1
  237. package/lib/deprecation.d.ts +0 -34
  238. package/lib/deprecation.d.ts.map +0 -1
  239. package/lib/deprecation.js +0 -57
  240. package/lib/deprecation.js.map +0 -1
  241. package/lib/jess-error.d.ts +0 -343
  242. package/lib/jess-error.d.ts.map +0 -1
  243. package/lib/jess-error.js +0 -508
  244. package/lib/jess-error.js.map +0 -1
  245. package/lib/logger/deprecation-processing.d.ts +0 -41
  246. package/lib/logger/deprecation-processing.d.ts.map +0 -1
  247. package/lib/logger/deprecation-processing.js +0 -81
  248. package/lib/logger/deprecation-processing.js.map +0 -1
  249. package/lib/logger.d.ts +0 -10
  250. package/lib/logger.d.ts.map +0 -1
  251. package/lib/logger.js +0 -20
  252. package/lib/logger.js.map +0 -1
  253. package/lib/plugin.d.ts +0 -94
  254. package/lib/plugin.d.ts.map +0 -1
  255. package/lib/plugin.js +0 -174
  256. package/lib/plugin.js.map +0 -1
  257. package/lib/tree/ampersand.d.ts +0 -98
  258. package/lib/tree/ampersand.d.ts.map +0 -1
  259. package/lib/tree/ampersand.js +0 -319
  260. package/lib/tree/ampersand.js.map +0 -1
  261. package/lib/tree/any.d.ts +0 -58
  262. package/lib/tree/any.d.ts.map +0 -1
  263. package/lib/tree/any.js +0 -104
  264. package/lib/tree/any.js.map +0 -1
  265. package/lib/tree/at-rule.d.ts +0 -53
  266. package/lib/tree/at-rule.d.ts.map +0 -1
  267. package/lib/tree/at-rule.js +0 -503
  268. package/lib/tree/at-rule.js.map +0 -1
  269. package/lib/tree/block.d.ts +0 -22
  270. package/lib/tree/block.d.ts.map +0 -1
  271. package/lib/tree/block.js +0 -24
  272. package/lib/tree/block.js.map +0 -1
  273. package/lib/tree/bool.d.ts +0 -18
  274. package/lib/tree/bool.d.ts.map +0 -1
  275. package/lib/tree/bool.js +0 -28
  276. package/lib/tree/bool.js.map +0 -1
  277. package/lib/tree/call.d.ts +0 -66
  278. package/lib/tree/call.d.ts.map +0 -1
  279. package/lib/tree/call.js +0 -306
  280. package/lib/tree/call.js.map +0 -1
  281. package/lib/tree/collection.d.ts +0 -30
  282. package/lib/tree/collection.d.ts.map +0 -1
  283. package/lib/tree/collection.js +0 -37
  284. package/lib/tree/collection.js.map +0 -1
  285. package/lib/tree/color.d.ts +0 -101
  286. package/lib/tree/color.d.ts.map +0 -1
  287. package/lib/tree/color.js +0 -513
  288. package/lib/tree/color.js.map +0 -1
  289. package/lib/tree/combinator.d.ts +0 -13
  290. package/lib/tree/combinator.d.ts.map +0 -1
  291. package/lib/tree/combinator.js +0 -12
  292. package/lib/tree/combinator.js.map +0 -1
  293. package/lib/tree/comment.d.ts +0 -20
  294. package/lib/tree/comment.d.ts.map +0 -1
  295. package/lib/tree/comment.js +0 -19
  296. package/lib/tree/comment.js.map +0 -1
  297. package/lib/tree/condition.d.ts +0 -31
  298. package/lib/tree/condition.d.ts.map +0 -1
  299. package/lib/tree/condition.js +0 -103
  300. package/lib/tree/condition.js.map +0 -1
  301. package/lib/tree/control.d.ts +0 -104
  302. package/lib/tree/control.d.ts.map +0 -1
  303. package/lib/tree/control.js +0 -430
  304. package/lib/tree/control.js.map +0 -1
  305. package/lib/tree/declaration-custom.d.ts +0 -18
  306. package/lib/tree/declaration-custom.d.ts.map +0 -1
  307. package/lib/tree/declaration-custom.js +0 -24
  308. package/lib/tree/declaration-custom.js.map +0 -1
  309. package/lib/tree/declaration-var.d.ts +0 -35
  310. package/lib/tree/declaration-var.d.ts.map +0 -1
  311. package/lib/tree/declaration-var.js +0 -63
  312. package/lib/tree/declaration-var.js.map +0 -1
  313. package/lib/tree/declaration.d.ts +0 -78
  314. package/lib/tree/declaration.d.ts.map +0 -1
  315. package/lib/tree/declaration.js +0 -286
  316. package/lib/tree/declaration.js.map +0 -1
  317. package/lib/tree/default-guard.d.ts +0 -15
  318. package/lib/tree/default-guard.d.ts.map +0 -1
  319. package/lib/tree/default-guard.js +0 -19
  320. package/lib/tree/default-guard.js.map +0 -1
  321. package/lib/tree/dimension.d.ts +0 -34
  322. package/lib/tree/dimension.d.ts.map +0 -1
  323. package/lib/tree/dimension.js +0 -294
  324. package/lib/tree/dimension.js.map +0 -1
  325. package/lib/tree/expression.d.ts +0 -25
  326. package/lib/tree/expression.d.ts.map +0 -1
  327. package/lib/tree/expression.js +0 -32
  328. package/lib/tree/expression.js.map +0 -1
  329. package/lib/tree/extend-list.d.ts +0 -23
  330. package/lib/tree/extend-list.d.ts.map +0 -1
  331. package/lib/tree/extend-list.js +0 -23
  332. package/lib/tree/extend-list.js.map +0 -1
  333. package/lib/tree/extend.d.ts +0 -47
  334. package/lib/tree/extend.d.ts.map +0 -1
  335. package/lib/tree/extend.js +0 -296
  336. package/lib/tree/extend.js.map +0 -1
  337. package/lib/tree/function.d.ts +0 -48
  338. package/lib/tree/function.d.ts.map +0 -1
  339. package/lib/tree/function.js +0 -74
  340. package/lib/tree/function.js.map +0 -1
  341. package/lib/tree/import-js.d.ts +0 -35
  342. package/lib/tree/import-js.d.ts.map +0 -1
  343. package/lib/tree/import-js.js +0 -45
  344. package/lib/tree/import-js.js.map +0 -1
  345. package/lib/tree/import-style.d.ts +0 -156
  346. package/lib/tree/import-style.d.ts.map +0 -1
  347. package/lib/tree/import-style.js +0 -566
  348. package/lib/tree/import-style.js.map +0 -1
  349. package/lib/tree/index.d.ts +0 -71
  350. package/lib/tree/index.d.ts.map +0 -1
  351. package/lib/tree/index.js.map +0 -1
  352. package/lib/tree/interpolated-reference.d.ts +0 -24
  353. package/lib/tree/interpolated-reference.d.ts.map +0 -1
  354. package/lib/tree/interpolated-reference.js +0 -37
  355. package/lib/tree/interpolated-reference.js.map +0 -1
  356. package/lib/tree/interpolated.d.ts +0 -62
  357. package/lib/tree/interpolated.d.ts.map +0 -1
  358. package/lib/tree/interpolated.js +0 -204
  359. package/lib/tree/interpolated.js.map +0 -1
  360. package/lib/tree/js-array.d.ts +0 -10
  361. package/lib/tree/js-array.d.ts.map +0 -1
  362. package/lib/tree/js-array.js +0 -10
  363. package/lib/tree/js-array.js.map +0 -1
  364. package/lib/tree/js-expr.d.ts +0 -23
  365. package/lib/tree/js-expr.d.ts.map +0 -1
  366. package/lib/tree/js-expr.js +0 -28
  367. package/lib/tree/js-expr.js.map +0 -1
  368. package/lib/tree/js-function.d.ts +0 -20
  369. package/lib/tree/js-function.d.ts.map +0 -1
  370. package/lib/tree/js-function.js +0 -16
  371. package/lib/tree/js-function.js.map +0 -1
  372. package/lib/tree/js-object.d.ts +0 -10
  373. package/lib/tree/js-object.d.ts.map +0 -1
  374. package/lib/tree/js-object.js +0 -10
  375. package/lib/tree/js-object.js.map +0 -1
  376. package/lib/tree/list.d.ts +0 -38
  377. package/lib/tree/list.d.ts.map +0 -1
  378. package/lib/tree/list.js +0 -83
  379. package/lib/tree/list.js.map +0 -1
  380. package/lib/tree/log.d.ts +0 -29
  381. package/lib/tree/log.d.ts.map +0 -1
  382. package/lib/tree/log.js +0 -56
  383. package/lib/tree/log.js.map +0 -1
  384. package/lib/tree/mixin.d.ts +0 -87
  385. package/lib/tree/mixin.d.ts.map +0 -1
  386. package/lib/tree/mixin.js +0 -112
  387. package/lib/tree/mixin.js.map +0 -1
  388. package/lib/tree/negative.d.ts +0 -17
  389. package/lib/tree/negative.d.ts.map +0 -1
  390. package/lib/tree/negative.js +0 -22
  391. package/lib/tree/negative.js.map +0 -1
  392. package/lib/tree/nil.d.ts +0 -30
  393. package/lib/tree/nil.d.ts.map +0 -1
  394. package/lib/tree/nil.js +0 -35
  395. package/lib/tree/nil.js.map +0 -1
  396. package/lib/tree/node-base.d.ts +0 -361
  397. package/lib/tree/node-base.d.ts.map +0 -1
  398. package/lib/tree/node-base.js +0 -930
  399. package/lib/tree/node-base.js.map +0 -1
  400. package/lib/tree/node.d.ts +0 -10
  401. package/lib/tree/node.d.ts.map +0 -1
  402. package/lib/tree/node.js +0 -45
  403. package/lib/tree/node.js.map +0 -1
  404. package/lib/tree/number.d.ts +0 -21
  405. package/lib/tree/number.d.ts.map +0 -1
  406. package/lib/tree/number.js +0 -27
  407. package/lib/tree/number.js.map +0 -1
  408. package/lib/tree/operation.d.ts +0 -26
  409. package/lib/tree/operation.d.ts.map +0 -1
  410. package/lib/tree/operation.js +0 -103
  411. package/lib/tree/operation.js.map +0 -1
  412. package/lib/tree/paren.d.ts +0 -19
  413. package/lib/tree/paren.d.ts.map +0 -1
  414. package/lib/tree/paren.js +0 -92
  415. package/lib/tree/paren.js.map +0 -1
  416. package/lib/tree/query-condition.d.ts +0 -17
  417. package/lib/tree/query-condition.d.ts.map +0 -1
  418. package/lib/tree/query-condition.js +0 -39
  419. package/lib/tree/query-condition.js.map +0 -1
  420. package/lib/tree/quoted.d.ts +0 -28
  421. package/lib/tree/quoted.d.ts.map +0 -1
  422. package/lib/tree/quoted.js +0 -75
  423. package/lib/tree/quoted.js.map +0 -1
  424. package/lib/tree/range.d.ts +0 -33
  425. package/lib/tree/range.d.ts.map +0 -1
  426. package/lib/tree/range.js +0 -47
  427. package/lib/tree/range.js.map +0 -1
  428. package/lib/tree/reference.d.ts +0 -76
  429. package/lib/tree/reference.d.ts.map +0 -1
  430. package/lib/tree/reference.js +0 -521
  431. package/lib/tree/reference.js.map +0 -1
  432. package/lib/tree/rest.d.ts +0 -15
  433. package/lib/tree/rest.d.ts.map +0 -1
  434. package/lib/tree/rest.js +0 -32
  435. package/lib/tree/rest.js.map +0 -1
  436. package/lib/tree/rules-raw.d.ts +0 -17
  437. package/lib/tree/rules-raw.d.ts.map +0 -1
  438. package/lib/tree/rules-raw.js +0 -37
  439. package/lib/tree/rules-raw.js.map +0 -1
  440. package/lib/tree/rules.d.ts +0 -262
  441. package/lib/tree/rules.d.ts.map +0 -1
  442. package/lib/tree/rules.js +0 -2359
  443. package/lib/tree/rules.js.map +0 -1
  444. package/lib/tree/ruleset.d.ts +0 -92
  445. package/lib/tree/ruleset.d.ts.map +0 -1
  446. package/lib/tree/ruleset.js +0 -528
  447. package/lib/tree/ruleset.js.map +0 -1
  448. package/lib/tree/selector-attr.d.ts +0 -31
  449. package/lib/tree/selector-attr.d.ts.map +0 -1
  450. package/lib/tree/selector-attr.js +0 -99
  451. package/lib/tree/selector-attr.js.map +0 -1
  452. package/lib/tree/selector-basic.d.ts +0 -24
  453. package/lib/tree/selector-basic.d.ts.map +0 -1
  454. package/lib/tree/selector-basic.js +0 -38
  455. package/lib/tree/selector-basic.js.map +0 -1
  456. package/lib/tree/selector-capture.d.ts +0 -23
  457. package/lib/tree/selector-capture.d.ts.map +0 -1
  458. package/lib/tree/selector-capture.js +0 -34
  459. package/lib/tree/selector-capture.js.map +0 -1
  460. package/lib/tree/selector-complex.d.ts +0 -40
  461. package/lib/tree/selector-complex.d.ts.map +0 -1
  462. package/lib/tree/selector-complex.js +0 -143
  463. package/lib/tree/selector-complex.js.map +0 -1
  464. package/lib/tree/selector-compound.d.ts +0 -16
  465. package/lib/tree/selector-compound.d.ts.map +0 -1
  466. package/lib/tree/selector-compound.js +0 -114
  467. package/lib/tree/selector-compound.js.map +0 -1
  468. package/lib/tree/selector-interpolated.d.ts +0 -23
  469. package/lib/tree/selector-interpolated.d.ts.map +0 -1
  470. package/lib/tree/selector-interpolated.js +0 -27
  471. package/lib/tree/selector-interpolated.js.map +0 -1
  472. package/lib/tree/selector-list.d.ts +0 -17
  473. package/lib/tree/selector-list.d.ts.map +0 -1
  474. package/lib/tree/selector-list.js +0 -174
  475. package/lib/tree/selector-list.js.map +0 -1
  476. package/lib/tree/selector-pseudo.d.ts +0 -42
  477. package/lib/tree/selector-pseudo.d.ts.map +0 -1
  478. package/lib/tree/selector-pseudo.js +0 -204
  479. package/lib/tree/selector-pseudo.js.map +0 -1
  480. package/lib/tree/selector-simple.d.ts +0 -5
  481. package/lib/tree/selector-simple.d.ts.map +0 -1
  482. package/lib/tree/selector-simple.js +0 -6
  483. package/lib/tree/selector-simple.js.map +0 -1
  484. package/lib/tree/selector.d.ts +0 -43
  485. package/lib/tree/selector.d.ts.map +0 -1
  486. package/lib/tree/selector.js +0 -56
  487. package/lib/tree/selector.js.map +0 -1
  488. package/lib/tree/sequence.d.ts +0 -43
  489. package/lib/tree/sequence.d.ts.map +0 -1
  490. package/lib/tree/sequence.js +0 -151
  491. package/lib/tree/sequence.js.map +0 -1
  492. package/lib/tree/tree.d.ts +0 -87
  493. package/lib/tree/tree.d.ts.map +0 -1
  494. package/lib/tree/tree.js +0 -2
  495. package/lib/tree/tree.js.map +0 -1
  496. package/lib/tree/url.d.ts +0 -18
  497. package/lib/tree/url.d.ts.map +0 -1
  498. package/lib/tree/url.js +0 -35
  499. package/lib/tree/url.js.map +0 -1
  500. package/lib/tree/util/__tests__/debug-log.d.ts +0 -1
  501. package/lib/tree/util/__tests__/debug-log.d.ts.map +0 -1
  502. package/lib/tree/util/__tests__/debug-log.js +0 -36
  503. package/lib/tree/util/__tests__/debug-log.js.map +0 -1
  504. package/lib/tree/util/calculate.d.ts +0 -3
  505. package/lib/tree/util/calculate.d.ts.map +0 -1
  506. package/lib/tree/util/calculate.js +0 -10
  507. package/lib/tree/util/calculate.js.map +0 -1
  508. package/lib/tree/util/cast.d.ts +0 -10
  509. package/lib/tree/util/cast.d.ts.map +0 -1
  510. package/lib/tree/util/cast.js +0 -87
  511. package/lib/tree/util/cast.js.map +0 -1
  512. package/lib/tree/util/cloning.d.ts +0 -4
  513. package/lib/tree/util/cloning.d.ts.map +0 -1
  514. package/lib/tree/util/cloning.js +0 -8
  515. package/lib/tree/util/cloning.js.map +0 -1
  516. package/lib/tree/util/collections.d.ts +0 -57
  517. package/lib/tree/util/collections.d.ts.map +0 -1
  518. package/lib/tree/util/collections.js +0 -136
  519. package/lib/tree/util/collections.js.map +0 -1
  520. package/lib/tree/util/compare.d.ts +0 -11
  521. package/lib/tree/util/compare.d.ts.map +0 -1
  522. package/lib/tree/util/compare.js +0 -89
  523. package/lib/tree/util/compare.js.map +0 -1
  524. package/lib/tree/util/extend-helpers.d.ts +0 -2
  525. package/lib/tree/util/extend-helpers.d.ts.map +0 -1
  526. package/lib/tree/util/extend-helpers.js +0 -2
  527. package/lib/tree/util/extend-helpers.js.map +0 -1
  528. package/lib/tree/util/extend-roots.d.ts +0 -37
  529. package/lib/tree/util/extend-roots.d.ts.map +0 -1
  530. package/lib/tree/util/extend-roots.js +0 -700
  531. package/lib/tree/util/extend-roots.js.map +0 -1
  532. package/lib/tree/util/extend-roots.old.d.ts +0 -132
  533. package/lib/tree/util/extend-roots.old.d.ts.map +0 -1
  534. package/lib/tree/util/extend-roots.old.js +0 -2272
  535. package/lib/tree/util/extend-roots.old.js.map +0 -1
  536. package/lib/tree/util/extend-trace-debug.d.ts +0 -13
  537. package/lib/tree/util/extend-trace-debug.d.ts.map +0 -1
  538. package/lib/tree/util/extend-trace-debug.js +0 -34
  539. package/lib/tree/util/extend-trace-debug.js.map +0 -1
  540. package/lib/tree/util/extend-walk.d.ts +0 -53
  541. package/lib/tree/util/extend-walk.d.ts.map +0 -1
  542. package/lib/tree/util/extend-walk.js +0 -881
  543. package/lib/tree/util/extend-walk.js.map +0 -1
  544. package/lib/tree/util/extend.d.ts +0 -218
  545. package/lib/tree/util/extend.d.ts.map +0 -1
  546. package/lib/tree/util/extend.js +0 -3182
  547. package/lib/tree/util/extend.js.map +0 -1
  548. package/lib/tree/util/find-extendable-locations.d.ts +0 -2
  549. package/lib/tree/util/find-extendable-locations.d.ts.map +0 -1
  550. package/lib/tree/util/find-extendable-locations.js +0 -2
  551. package/lib/tree/util/find-extendable-locations.js.map +0 -1
  552. package/lib/tree/util/format.d.ts +0 -20
  553. package/lib/tree/util/format.d.ts.map +0 -1
  554. package/lib/tree/util/format.js +0 -67
  555. package/lib/tree/util/format.js.map +0 -1
  556. package/lib/tree/util/is-node.d.ts +0 -13
  557. package/lib/tree/util/is-node.d.ts.map +0 -1
  558. package/lib/tree/util/is-node.js +0 -43
  559. package/lib/tree/util/is-node.js.map +0 -1
  560. package/lib/tree/util/print.d.ts +0 -80
  561. package/lib/tree/util/print.d.ts.map +0 -1
  562. package/lib/tree/util/print.js +0 -205
  563. package/lib/tree/util/print.js.map +0 -1
  564. package/lib/tree/util/process-leading-is.d.ts +0 -25
  565. package/lib/tree/util/process-leading-is.d.ts.map +0 -1
  566. package/lib/tree/util/process-leading-is.js +0 -364
  567. package/lib/tree/util/process-leading-is.js.map +0 -1
  568. package/lib/tree/util/recursion-helper.d.ts +0 -15
  569. package/lib/tree/util/recursion-helper.d.ts.map +0 -1
  570. package/lib/tree/util/recursion-helper.js +0 -43
  571. package/lib/tree/util/recursion-helper.js.map +0 -1
  572. package/lib/tree/util/regex.d.ts +0 -4
  573. package/lib/tree/util/regex.d.ts.map +0 -1
  574. package/lib/tree/util/regex.js +0 -4
  575. package/lib/tree/util/regex.js.map +0 -1
  576. package/lib/tree/util/registry-utils.d.ts +0 -192
  577. package/lib/tree/util/registry-utils.d.ts.map +0 -1
  578. package/lib/tree/util/registry-utils.js +0 -1214
  579. package/lib/tree/util/registry-utils.js.map +0 -1
  580. package/lib/tree/util/ruleset-trace.d.ts +0 -4
  581. package/lib/tree/util/ruleset-trace.d.ts.map +0 -1
  582. package/lib/tree/util/ruleset-trace.js +0 -14
  583. package/lib/tree/util/ruleset-trace.js.map +0 -1
  584. package/lib/tree/util/selector-compare.d.ts +0 -2
  585. package/lib/tree/util/selector-compare.d.ts.map +0 -1
  586. package/lib/tree/util/selector-compare.js +0 -2
  587. package/lib/tree/util/selector-compare.js.map +0 -1
  588. package/lib/tree/util/selector-match-core.d.ts +0 -184
  589. package/lib/tree/util/selector-match-core.d.ts.map +0 -1
  590. package/lib/tree/util/selector-match-core.js +0 -1603
  591. package/lib/tree/util/selector-match-core.js.map +0 -1
  592. package/lib/tree/util/selector-utils.d.ts +0 -30
  593. package/lib/tree/util/selector-utils.d.ts.map +0 -1
  594. package/lib/tree/util/selector-utils.js +0 -100
  595. package/lib/tree/util/selector-utils.js.map +0 -1
  596. package/lib/tree/util/serialize-helper.d.ts +0 -13
  597. package/lib/tree/util/serialize-helper.d.ts.map +0 -1
  598. package/lib/tree/util/serialize-helper.js +0 -387
  599. package/lib/tree/util/serialize-helper.js.map +0 -1
  600. package/lib/tree/util/serialize-types.d.ts +0 -9
  601. package/lib/tree/util/serialize-types.d.ts.map +0 -1
  602. package/lib/tree/util/serialize-types.js +0 -216
  603. package/lib/tree/util/serialize-types.js.map +0 -1
  604. package/lib/tree/util/should-operate.d.ts +0 -23
  605. package/lib/tree/util/should-operate.d.ts.map +0 -1
  606. package/lib/tree/util/should-operate.js +0 -46
  607. package/lib/tree/util/should-operate.js.map +0 -1
  608. package/lib/tree/util/sourcemap.d.ts +0 -7
  609. package/lib/tree/util/sourcemap.d.ts.map +0 -1
  610. package/lib/tree/util/sourcemap.js +0 -25
  611. package/lib/tree/util/sourcemap.js.map +0 -1
  612. package/lib/types/config.d.ts +0 -205
  613. package/lib/types/config.d.ts.map +0 -1
  614. package/lib/types/config.js +0 -2
  615. package/lib/types/config.js.map +0 -1
  616. package/lib/types/index.d.ts +0 -15
  617. package/lib/types/index.d.ts.map +0 -1
  618. package/lib/types/index.js +0 -3
  619. package/lib/types/index.js.map +0 -1
  620. package/lib/types/modes.d.ts.map +0 -1
  621. package/lib/types/modes.js +0 -2
  622. package/lib/types/modes.js.map +0 -1
  623. package/lib/types.d.ts +0 -61
  624. package/lib/types.d.ts.map +0 -1
  625. package/lib/types.js +0 -2
  626. package/lib/types.js.map +0 -1
  627. package/lib/use-webpack-resolver.d.ts +0 -9
  628. package/lib/use-webpack-resolver.d.ts.map +0 -1
  629. package/lib/use-webpack-resolver.js +0 -41
  630. package/lib/use-webpack-resolver.js.map +0 -1
  631. package/lib/visitor/index.d.ts +0 -136
  632. package/lib/visitor/index.d.ts.map +0 -1
  633. package/lib/visitor/index.js +0 -135
  634. package/lib/visitor/index.js.map +0 -1
  635. package/lib/visitor/less-visitor.d.ts +0 -7
  636. package/lib/visitor/less-visitor.d.ts.map +0 -1
  637. package/lib/visitor/less-visitor.js.map +0 -1
@@ -0,0 +1,1033 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import { Context } from '../../context.js';
3
+ import {
4
+ any,
5
+ amp,
6
+ attr,
7
+ atrule,
8
+ co,
9
+ color,
10
+ comment,
11
+ compound,
12
+ decl,
13
+ el,
14
+ ExtendFlag,
15
+ extend,
16
+ keyword,
17
+ nil,
18
+ paren,
19
+ pseudo,
20
+ query,
21
+ quoted,
22
+ rules,
23
+ ruleset,
24
+ sel,
25
+ sellist,
26
+ spaced
27
+ } from '../index.js';
28
+ import { serializeTypes } from '../util/serialize-types.js';
29
+
30
+ describe('extend integration (eval -> toString)', () => {
31
+ it('extends selectors inside nested rulesets (Less extend-selector replace case)', async () => {
32
+ // Expected Less behavior (extend-selector.css): inner block outputs .replace, .rep_ace, .c.
33
+ // Correct fix: do NOT flatten ampersand in extend target; DO flatten in extendWith when different context.
34
+ // No sourceNode for header — serialization uses selector; implicit & is omitted in toTrimmedString.
35
+ // Progressive reproduction:
36
+ // - Step 0: no extends → nested output
37
+ // - Step 1: add `.rep_ace:extend(.replace all)` → Less hoists/mixes using `:is(...)`; inner block must show .replace, .rep_ace, .c
38
+
39
+ const makeRoot = (includeRepAceExtend: boolean) => rules([
40
+ ruleset({
41
+ selector: sellist([
42
+ compound([el('.replace'), el('.replace')]),
43
+ sel([compound([el('.c'), el('.replace')]), co('+'), el('.replace')])
44
+ ]),
45
+ rules: rules([
46
+ ruleset({
47
+ selector: sellist([el('.replace'), el('.c')]),
48
+ rules: rules([
49
+ decl({ name: 'prop', value: any('copy-paste-replace') })
50
+ ])
51
+ })
52
+ ])
53
+ }),
54
+ ...(includeRepAceExtend
55
+ ? [
56
+ ruleset({
57
+ selector: el('.rep_ace'),
58
+ rules: rules([
59
+ // Less `all` (partial=true)
60
+ extend({ target: el('.replace'), flag: ExtendFlag.All })
61
+ ])
62
+ })
63
+ ]
64
+ : [])
65
+ ]);
66
+
67
+ // Step 0
68
+ {
69
+ const context = new Context({ collapseNesting: false });
70
+ const evald = await makeRoot(false).eval(context);
71
+ const css = evald.render(context);
72
+ expect(css).toBeString(`
73
+ .replace.replace,
74
+ .c.replace + .replace {
75
+ .replace,
76
+ .c {
77
+ prop: copy-paste-replace;
78
+ }
79
+ }
80
+ `);
81
+ }
82
+
83
+ // Step 1 (expected Less output, from `tests-unit/extend-selector/extend-selector.css`)
84
+ {
85
+ const context = new Context({ collapseNesting: false });
86
+ const evald = await makeRoot(true).eval(context);
87
+ const css = evald.render(context);
88
+ expect(css).toBeString(`
89
+ :is(.replace, .rep_ace):is(.replace, .rep_ace),
90
+ .c:is(.replace, .rep_ace) + :is(.replace, .rep_ace) {
91
+ .replace,
92
+ .c,
93
+ .rep_ace {
94
+ prop: copy-paste-replace;
95
+ }
96
+ }
97
+ `);
98
+ }
99
+ });
100
+
101
+ it('extends nested ruleset selector across parent boundary (header/footer)', async () => {
102
+ // Represents:
103
+ // .header {
104
+ // .header-nav {
105
+ // background: red;
106
+ // &:before { background: blue; }
107
+ // }
108
+ // }
109
+ //
110
+ // .footer {
111
+ // .footer-nav {
112
+ // &:extend(.header .header-nav all);
113
+ // }
114
+ // }
115
+ const root = rules([
116
+ ruleset({
117
+ selector: el('.header'),
118
+ rules: rules([
119
+ ruleset({
120
+ selector: el('.header-nav'),
121
+ rules: rules([
122
+ decl({ name: 'background', value: any('red') }),
123
+ ruleset({
124
+ selector: sel([amp({}), pseudo({ name: ':before' })]) as any,
125
+ rules: rules([
126
+ decl({ name: 'background', value: any('blue') })
127
+ ])
128
+ })
129
+ ])
130
+ })
131
+ ])
132
+ }),
133
+ ruleset({
134
+ selector: el('.footer'),
135
+ rules: rules([
136
+ ruleset({
137
+ selector: el('.footer-nav'),
138
+ rules: rules([
139
+ extend({
140
+ target: sel([el('.header'), co(' '), el('.header-nav')]),
141
+ flag: ExtendFlag.All
142
+ })
143
+ ])
144
+ })
145
+ ])
146
+ })
147
+ ]);
148
+
149
+ const context = new Context({ collapseNesting: false });
150
+ const evald = await root.eval(context);
151
+ const css = evald.render(context);
152
+
153
+ expect(css).toBeString(`
154
+ .header .header-nav,
155
+ .footer .footer-nav {
156
+ background: red;
157
+ &:before {
158
+ background: blue;
159
+ }
160
+ }
161
+ `);
162
+ });
163
+
164
+ it('extends attribute selectors without duplicating implicit parent prefix (Less extend-selector attributes)', async () => {
165
+ // Represents:
166
+ // .attributes {
167
+ // [data="test"] { extend: attributes; }
168
+ // .attribute-test { &:extend([data="test"] all); }
169
+ // }
170
+ const dataTest = attr({ name: 'data', op: '=', value: quoted('test') });
171
+
172
+ const root = rules([
173
+ ruleset({
174
+ selector: el('.attributes'),
175
+ rules: rules([
176
+ ruleset({
177
+ selector: dataTest,
178
+ rules: rules([decl({ name: 'extend', value: any('attributes') })])
179
+ }),
180
+ ruleset({
181
+ selector: el('.attribute-test'),
182
+ rules: rules([
183
+ extend({ target: dataTest, flag: ExtendFlag.All })
184
+ ])
185
+ })
186
+ ])
187
+ })
188
+ ]);
189
+
190
+ const context = new Context({ collapseNesting: false });
191
+ const evald = await root.eval(context);
192
+ const css = evald.render(context);
193
+
194
+ expect(css).toBeString(`
195
+ .attributes {
196
+ [data="test"],
197
+ .attribute-test {
198
+ extend: attributes;
199
+ }
200
+ }
201
+ `);
202
+ });
203
+
204
+ it('extend.less .ee:extend(.dd all,.bb) does NOT add .ee to inner .bb (only outer .bb)', async () => {
205
+ // Replicate extend.less: .bb { background: red; .bb { color: black; } } .ee:extend(.dd all,.bb) {}
206
+ // .bb in the extend has NO "all", so exact match only. Inner ruleset has selector .bb .bb (implicit &).
207
+ // So .ee must be added to outer .bb only; inner .bb must stay .bb (and only .ff gets added there via .bb all).
208
+ const innerBbRuleset = ruleset({
209
+ selector: el('.bb'),
210
+ rules: rules([decl({ name: 'color', value: any('black') })])
211
+ });
212
+ const root = rules([
213
+ ruleset({
214
+ selector: el('.bb'),
215
+ rules: rules([
216
+ decl({ name: 'background', value: any('red') }),
217
+ innerBbRuleset
218
+ ])
219
+ }),
220
+ ruleset({
221
+ selector: el('.ee'),
222
+ rules: rules([
223
+ extend({ target: el('.dd'), flag: ExtendFlag.All }),
224
+ extend({ target: el('.bb'), flag: ExtendFlag.Exact })
225
+ ])
226
+ })
227
+ ]);
228
+ const context = new Context({ collapseNesting: false });
229
+ const evald = await root.eval(context);
230
+ // Find the inner ruleset in the evald tree (ruleset that has decl color and is nested inside .bb)
231
+ const evaldRoot = evald as import('../rules.js').Rules;
232
+ const outerBb = evaldRoot.value.find(
233
+ (n: any) =>
234
+ n?.type === 'Ruleset'
235
+ && Array.isArray(n.rules?.value)
236
+ && n.rules.value.some((r: any) => r?.type === 'Declaration' && (r.name?.valueOf?.() ?? r.name) === 'background')
237
+ && n.rules.value.some((r: any) => r?.type === 'Ruleset')
238
+ );
239
+ expect(outerBb).toBeTruthy();
240
+ const inner = (outerBb as any).rules.value.find(
241
+ (n: any) => n?.type === 'Ruleset' && n.rules?.value?.some((d: any) => (d?.name?.valueOf?.() ?? d?.name) === 'color')
242
+ );
243
+ expect(inner).toBeTruthy();
244
+ const innerSelectorStr = typeof (inner as any).selector?.valueOf === 'function' ? (inner as any).selector.valueOf() : '';
245
+ // Inner selector must be .bb .bb (or equivalent), must NOT contain .ee
246
+ expect(innerSelectorStr).toContain('.bb');
247
+ expect(innerSelectorStr).not.toContain('.ee');
248
+ });
249
+
250
+ it('extend-media: .all:extend(.ext1 all) at root merges with .ext1 inside and outside @media (Less extend-media.less)', async () => {
251
+ // .ext1 .ext2 { background: black }
252
+ // @media (tv) { .ext1 .ext3 { color: inherit }, .tv-lowres :extend(.ext1 all) { background: blue },
253
+ // @media (hires) { .ext1 .ext4 { color: green }, .tv-hires :extend(.ext1 all) { background: red } } }
254
+ // .all:extend(.ext1 all) {}
255
+ const root = rules([
256
+ ruleset({
257
+ selector: sellist([sel([el('.ext1'), co(' '), el('.ext2')])]),
258
+ rules: rules([decl({ name: 'background', value: any('black') })])
259
+ }),
260
+ atrule({
261
+ name: any('@media'),
262
+ prelude: any('(tv)'),
263
+ rules: rules([
264
+ ruleset({
265
+ selector: sellist([sel([el('.ext1'), co(' '), el('.ext3')])]),
266
+ rules: rules([decl({ name: 'color', value: any('inherit') })])
267
+ }),
268
+ ruleset({
269
+ selector: el('.tv-lowres'),
270
+ rules: rules([
271
+ decl({ name: 'background', value: any('blue') }),
272
+ extend({ target: el('.ext1'), flag: ExtendFlag.All })
273
+ ])
274
+ }),
275
+ atrule({
276
+ name: any('@media'),
277
+ prelude: any('(hires)'),
278
+ rules: rules([
279
+ ruleset({
280
+ selector: sellist([sel([el('.ext1'), co(' '), el('.ext4')])]),
281
+ rules: rules([decl({ name: 'color', value: any('green') })])
282
+ }),
283
+ ruleset({
284
+ selector: el('.tv-hires'),
285
+ rules: rules([
286
+ decl({ name: 'background', value: any('red') }),
287
+ extend({ target: el('.ext1'), flag: ExtendFlag.All })
288
+ ])
289
+ })
290
+ ])
291
+ })
292
+ ])
293
+ }),
294
+ ruleset({
295
+ selector: el('.all'),
296
+ rules: rules([extend({ target: el('.ext1'), flag: ExtendFlag.All })])
297
+ })
298
+ ]);
299
+
300
+ const context = new Context({ collapseNesting: false });
301
+ const evald = await root.eval(context);
302
+ const css = evald.render(context);
303
+
304
+ expect(css).toBeString(`
305
+ :is(.ext1, .all) .ext2 {
306
+ background: black;
307
+ }
308
+ @media (tv) {
309
+ :is(.ext1, .tv-lowres, .all) .ext3 {
310
+ color: inherit;
311
+ }
312
+ .tv-lowres {
313
+ background: blue;
314
+ }
315
+ @media (hires) {
316
+ :is(.ext1, .tv-lowres, .tv-hires, .all) .ext4 {
317
+ color: green;
318
+ }
319
+ .tv-hires {
320
+ background: red;
321
+ }
322
+ }
323
+ }
324
+ `);
325
+ });
326
+
327
+ it('extend-chaining media with context.root set before eval (simulates jess getTree)', async () => {
328
+ // Same structure as below; eval sets context.root during preEval (no manual set).
329
+ const root = rules([
330
+ ruleset({
331
+ selector: sellist([sel([el('.a')])]),
332
+ rules: rules([decl({ name: 'color', value: any('black') })])
333
+ }),
334
+ atrule({
335
+ name: any('@media'),
336
+ prelude: any('(tv)'),
337
+ rules: rules([
338
+ ruleset({
339
+ selector: el('.ma'),
340
+ rules: rules([
341
+ decl({ name: 'color', value: any('black') }),
342
+ extend({ target: el('.a') }),
343
+ extend({ target: el('.md') })
344
+ ])
345
+ }),
346
+ ruleset({
347
+ selector: el('.md'),
348
+ rules: rules([decl({ name: 'color', value: any('inherit') })])
349
+ })
350
+ ])
351
+ }),
352
+ ruleset({
353
+ selector: el('.mb'),
354
+ rules: rules([extend({ target: el('.ma') })])
355
+ }),
356
+ ruleset({
357
+ selector: el('.mc'),
358
+ rules: rules([extend({ target: el('.mb') })])
359
+ })
360
+ ]);
361
+ const context = new Context({ collapseNesting: false });
362
+ const evald = await root.eval(context);
363
+ const css = evald.render(context);
364
+ expect(css).toBeString(`
365
+ .a {
366
+ color: black;
367
+ }
368
+ @media (tv) {
369
+ .ma,
370
+ .mb,
371
+ .mc {
372
+ color: black;
373
+ }
374
+ .md,
375
+ .ma,
376
+ .mb,
377
+ .mc {
378
+ color: inherit;
379
+ }
380
+ }
381
+ `);
382
+ });
383
+
384
+ it('extend-chaining media: inside @media extends outside and .md; outside extends .ma/.mb inside (Less extend-chaining.less media block)', async () => {
385
+ // .a { color: black }
386
+ // @media (tv) { .ma:extend(.a, .md) { color: black }, .md { color: inherit } }
387
+ // .mb:extend(.ma) {} .mc:extend(.mb) {}
388
+ const root = rules([
389
+ ruleset({
390
+ selector: sellist([sel([el('.a')])]),
391
+ rules: rules([decl({ name: 'color', value: any('black') })])
392
+ }),
393
+ atrule({
394
+ name: any('@media'),
395
+ prelude: any('(tv)'),
396
+ rules: rules([
397
+ ruleset({
398
+ selector: el('.ma'),
399
+ rules: rules([
400
+ decl({ name: 'color', value: any('black') }),
401
+ extend({ target: el('.a') }),
402
+ extend({ target: el('.md') })
403
+ ])
404
+ }),
405
+ ruleset({
406
+ selector: el('.md'),
407
+ rules: rules([decl({ name: 'color', value: any('inherit') })])
408
+ })
409
+ ])
410
+ }),
411
+ ruleset({
412
+ selector: el('.mb'),
413
+ rules: rules([extend({ target: el('.ma') })])
414
+ }),
415
+ ruleset({
416
+ selector: el('.mc'),
417
+ rules: rules([extend({ target: el('.mb') })])
418
+ })
419
+ ]);
420
+
421
+ const context = new Context({ collapseNesting: false });
422
+ const evald = await root.eval(context);
423
+ const css = evald.render(context);
424
+
425
+ expect(css).toBeString(`
426
+ .a {
427
+ color: black;
428
+ }
429
+ @media (tv) {
430
+ .ma,
431
+ .mb,
432
+ .mc {
433
+ color: black;
434
+ }
435
+ .md,
436
+ .ma,
437
+ .mb,
438
+ .mc {
439
+ color: inherit;
440
+ }
441
+ }
442
+ `);
443
+ });
444
+
445
+ it('extend-chaining media with SelectorList target (same AST shape as Less parser for .ma:extend(.a,.md))', async () => {
446
+ // Replicate parsed extend-chaining.less: one Extend with target SelectorList([.a, .md])
447
+ // instead of two separate Extend nodes. Ensures processExtend handles SelectorList target
448
+ // when extend is inside @media and targets are at document root.
449
+ const root = rules([
450
+ ruleset({
451
+ selector: sellist([sel([el('.a')])]),
452
+ rules: rules([decl({ name: 'color', value: any('black') })])
453
+ }),
454
+ atrule({
455
+ name: any('@media'),
456
+ prelude: any('(tv)'),
457
+ rules: rules([
458
+ ruleset({
459
+ selector: el('.ma'),
460
+ rules: rules([
461
+ decl({ name: 'color', value: any('black') }),
462
+ extend({ target: sellist([el('.a'), el('.md')]) })
463
+ ])
464
+ }),
465
+ ruleset({
466
+ selector: el('.md'),
467
+ rules: rules([decl({ name: 'color', value: any('inherit') })])
468
+ })
469
+ ])
470
+ }),
471
+ ruleset({
472
+ selector: el('.mb'),
473
+ rules: rules([extend({ target: el('.ma') })])
474
+ }),
475
+ ruleset({
476
+ selector: el('.mc'),
477
+ rules: rules([extend({ target: el('.mb') })])
478
+ })
479
+ ]);
480
+
481
+ const context = new Context({ collapseNesting: false });
482
+ const evald = await root.eval(context);
483
+ const css = evald.render(context);
484
+
485
+ expect(css).toBeString(`
486
+ .a {
487
+ color: black;
488
+ }
489
+ @media (tv) {
490
+ .ma,
491
+ .mb,
492
+ .mc {
493
+ color: black;
494
+ }
495
+ .md,
496
+ .ma,
497
+ .mb,
498
+ .mc {
499
+ color: inherit;
500
+ }
501
+ }
502
+ `);
503
+ });
504
+
505
+ it('extend-chaining.less AST shape: same nodes and selector shapes as parsed (start through @media (tv))', async () => {
506
+ // Same nodes and AST shape as Jess/Less parser for extend-chaining.less from document start
507
+ // through end of @media (tv), plus .mb/.mc so the extend chain resolves. Selector shape:
508
+ // BasicSelector for ruleset selectors; Paren(QueryCondition(Keyword)) for @media prelude;
509
+ // one Extend with SelectorList target for .ma; Extend before Declaration in .ma rules.
510
+ const blackColor = color({ node: 'black', format: 0, rgb: [0, 0, 0], alpha: 1 });
511
+ const maExtendTarget = sellist([
512
+ el('.a'),
513
+ el('.b'),
514
+ el('.c'),
515
+ el('.d'),
516
+ el('.e'),
517
+ el('.f'),
518
+ el('.g'),
519
+ el('.h'),
520
+ el('.i'),
521
+ el('.j'),
522
+ el('.k'),
523
+ el('.l'),
524
+ el('.m'),
525
+ el('.n'),
526
+ el('.o'),
527
+ el('.p'),
528
+ el('.q'),
529
+ el('.r'),
530
+ el('.s'),
531
+ el('.t'),
532
+ el('.u'),
533
+ el('.v'),
534
+ el('.w'),
535
+ el('.x'),
536
+ el('.y'),
537
+ el('.z'),
538
+ el('.md')
539
+ ]);
540
+ const root = rules([
541
+ comment('//very simple chaining'),
542
+ ruleset({
543
+ selector: el('.a'),
544
+ rules: rules([decl({ name: 'color', value: blackColor })])
545
+ }),
546
+ ruleset({ selector: el('.b'), rules: rules([extend({ target: el('.a'), flag: ExtendFlag.Exact })]) }),
547
+ ruleset({ selector: el('.c'), rules: rules([extend({ target: el('.b'), flag: ExtendFlag.Exact })]) }),
548
+ comment('//very simple chaining, ordering not important'),
549
+ ruleset({ selector: el('.d'), rules: rules([extend({ target: el('.e'), flag: ExtendFlag.Exact })]) }),
550
+ ruleset({ selector: el('.e'), rules: rules([extend({ target: el('.f'), flag: ExtendFlag.Exact })]) }),
551
+ ruleset({
552
+ selector: el('.f'),
553
+ rules: rules([decl({ name: 'color', value: blackColor })])
554
+ }),
555
+ comment('//extend with all'),
556
+ ruleset({
557
+ selector: compound([el('.g'), el('.h')]),
558
+ rules: rules([decl({ name: 'color', value: blackColor })])
559
+ }),
560
+ ruleset({
561
+ selector: compound([el('.i'), el('.j')]),
562
+ rules: rules([
563
+ extend({ target: el('.g'), flag: ExtendFlag.All }),
564
+ decl({ name: 'color', value: any('inherit') })
565
+ ])
566
+ }),
567
+ ruleset({
568
+ selector: el('.k'),
569
+ rules: rules([extend({ target: el('.i'), flag: ExtendFlag.All })])
570
+ }),
571
+ comment('//extend multi-chaining'),
572
+ ruleset({
573
+ selector: el('.l'),
574
+ rules: rules([decl({ name: 'color', value: blackColor })])
575
+ }),
576
+ ruleset({ selector: el('.m'), rules: rules([extend({ target: el('.l'), flag: ExtendFlag.Exact })]) }),
577
+ ruleset({ selector: el('.n'), rules: rules([extend({ target: el('.m'), flag: ExtendFlag.Exact })]) }),
578
+ ruleset({ selector: el('.o'), rules: rules([extend({ target: el('.n'), flag: ExtendFlag.Exact })]) }),
579
+ ruleset({ selector: el('.p'), rules: rules([extend({ target: el('.o'), flag: ExtendFlag.Exact })]) }),
580
+ ruleset({ selector: el('.q'), rules: rules([extend({ target: el('.p'), flag: ExtendFlag.Exact })]) }),
581
+ ruleset({ selector: el('.r'), rules: rules([extend({ target: el('.q'), flag: ExtendFlag.Exact })]) }),
582
+ ruleset({ selector: el('.s'), rules: rules([extend({ target: el('.r'), flag: ExtendFlag.Exact })]) }),
583
+ ruleset({ selector: el('.t'), rules: rules([extend({ target: el('.s'), flag: ExtendFlag.Exact })]) }),
584
+ comment('// self referencing is ignored'),
585
+ ruleset({
586
+ selector: el('.u'),
587
+ rules: rules([decl({ name: 'color', value: blackColor })])
588
+ }),
589
+ ruleset({
590
+ selector: compound([el('.v'), el('.u'), el('.v')]),
591
+ rules: rules([extend({ target: el('.u'), flag: ExtendFlag.All })])
592
+ }),
593
+ comment('// circular reference because the new extend product will match the existing extend'),
594
+ ruleset({
595
+ selector: el('.w'),
596
+ rules: rules([
597
+ extend({ target: el('.w'), flag: ExtendFlag.Exact }),
598
+ decl({ name: 'color', value: blackColor })
599
+ ])
600
+ }),
601
+ ruleset({
602
+ selector: compound([el('.v'), el('.w'), el('.v')]),
603
+ rules: rules([extend({ target: el('.w'), flag: ExtendFlag.All })])
604
+ }),
605
+ comment('// classic circular references'),
606
+ ruleset({
607
+ selector: el('.x'),
608
+ rules: rules([
609
+ extend({ target: el('.z'), flag: ExtendFlag.Exact }),
610
+ decl({ name: 'color', value: any('x') })
611
+ ])
612
+ }),
613
+ ruleset({
614
+ selector: el('.y'),
615
+ rules: rules([
616
+ extend({ target: el('.x'), flag: ExtendFlag.Exact }),
617
+ decl({ name: 'color', value: any('y') })
618
+ ])
619
+ }),
620
+ ruleset({
621
+ selector: el('.z'),
622
+ rules: rules([
623
+ extend({ target: el('.y'), flag: ExtendFlag.Exact }),
624
+ decl({ name: 'color', value: any('z') })
625
+ ])
626
+ }),
627
+ comment('//very simple chaining, but with the extend inside the ruleset'),
628
+ ruleset({
629
+ selector: el('.va'),
630
+ rules: rules([decl({ name: 'color', value: blackColor })])
631
+ }),
632
+ ruleset({
633
+ selector: el('.vb'),
634
+ rules: rules([
635
+ extend({ target: el('.va'), flag: ExtendFlag.Exact }),
636
+ nil(),
637
+ decl({ name: 'color', value: any('inherit') })
638
+ ])
639
+ }),
640
+ ruleset({
641
+ selector: el('.vc'),
642
+ rules: rules([extend({ target: el('.vb'), flag: ExtendFlag.Exact }), nil()])
643
+ }),
644
+ comment('// media queries - don\'t extend outside, do extend inside'),
645
+ atrule({
646
+ name: any('@media'),
647
+ prelude: paren(query([keyword('tv')])),
648
+ rules: rules([
649
+ ruleset({
650
+ selector: el('.ma'),
651
+ rules: rules([
652
+ extend({ target: maExtendTarget, flag: ExtendFlag.Exact }),
653
+ decl({ name: 'color', value: blackColor })
654
+ ])
655
+ }),
656
+ ruleset({
657
+ selector: el('.md'),
658
+ rules: rules([decl({ name: 'color', value: any('inherit') })])
659
+ }),
660
+ atrule({
661
+ name: any('@media'),
662
+ prelude: paren(query([keyword('plasma')])),
663
+ rules: rules([
664
+ // Parsed structure: inner Rules wrapping Extend then Ruleset (same as snapshot)
665
+ rules([
666
+ extend({
667
+ selector: sellist([el('.me'), el('.mf')]),
668
+ target: sellist([el('.mb'), el('.md')]),
669
+ flag: ExtendFlag.Exact
670
+ }),
671
+ ruleset({
672
+ selector: sellist([el('.me'), el('.mf')]),
673
+ rules: rules([
674
+ nil(),
675
+ decl({
676
+ name: 'background',
677
+ value: color({ node: 'red', format: 0, rgb: [255, 0, 0], alpha: 1 })
678
+ })
679
+ ])
680
+ })
681
+ ])
682
+ ])
683
+ })
684
+ ])
685
+ }),
686
+ ruleset({
687
+ selector: el('.mb'),
688
+ rules: rules([extend({ target: el('.ma'), flag: ExtendFlag.Exact })])
689
+ }),
690
+ ruleset({
691
+ selector: el('.mc'),
692
+ rules: rules([extend({ target: el('.mb'), flag: ExtendFlag.Exact })])
693
+ })
694
+ ]);
695
+ const serializeOpts = { showValues: true, maxStringLength: 120 };
696
+ const preEvalSerialized = serializeTypes(root, serializeOpts);
697
+ expect(typeof preEvalSerialized).toBe('string');
698
+ expect(preEvalSerialized).toMatchSnapshot();
699
+
700
+ const context = new Context({ collapseNesting: false });
701
+ const evald = await root.eval(context);
702
+ const postEvalSerialized = serializeTypes(evald, serializeOpts);
703
+ expect(typeof postEvalSerialized).toBe('string');
704
+ expect(postEvalSerialized).toMatchSnapshot();
705
+
706
+ const css = evald.render(context);
707
+ // Large parsed-shape parity test: keep deterministic string checks without regex/snapshot churn.
708
+ expect(css).toContain(` .ma,
709
+ .mb,
710
+ .mc {
711
+ color: black;
712
+ }`);
713
+ expect(css).toContain(` .md,
714
+ .ma,
715
+ .mb,
716
+ .mc {
717
+ color: inherit;
718
+ }`);
719
+ });
720
+
721
+ it('extend-chaining media with collapseNesting: true - merge must still apply (replicates Jess all-less bug)', async () => {
722
+ // Same structure as "extend-chaining media" but collapseNesting: true.
723
+ // In Jess all-less, extend-chaining.less was run with collapseNesting: true and got unmerged
724
+ // .ma / .md in the media block; extend-chaining-ast-compare used false and passed.
725
+ // Extend merging must be correct regardless of collapseNesting.
726
+ const root = rules([
727
+ ruleset({
728
+ selector: sellist([sel([el('.a')])]),
729
+ rules: rules([decl({ name: 'color', value: any('black') })])
730
+ }),
731
+ atrule({
732
+ name: any('@media'),
733
+ prelude: any('(tv)'),
734
+ rules: rules([
735
+ ruleset({
736
+ selector: el('.ma'),
737
+ rules: rules([
738
+ decl({ name: 'color', value: any('black') }),
739
+ extend({ target: el('.a') }),
740
+ extend({ target: el('.md') })
741
+ ])
742
+ }),
743
+ ruleset({
744
+ selector: el('.md'),
745
+ rules: rules([decl({ name: 'color', value: any('inherit') })])
746
+ })
747
+ ])
748
+ }),
749
+ ruleset({
750
+ selector: el('.mb'),
751
+ rules: rules([extend({ target: el('.ma') })])
752
+ }),
753
+ ruleset({
754
+ selector: el('.mc'),
755
+ rules: rules([extend({ target: el('.mb') })])
756
+ })
757
+ ]);
758
+ const context = new Context({ collapseNesting: true });
759
+ const evald = await root.eval(context);
760
+ const css = evald.render(context);
761
+ expect(css).toBeString(`
762
+ .a {
763
+ color: black;
764
+ }
765
+ @media (tv) {
766
+ .ma,
767
+ .mb,
768
+ .mc {
769
+ color: black;
770
+ }
771
+ .md,
772
+ .ma,
773
+ .mb,
774
+ .mc {
775
+ color: inherit;
776
+ }
777
+ }
778
+ `);
779
+ });
780
+
781
+ it('PARITY: extend-chaining media with SelectorList target + collapseNesting true keeps merged selectors', async () => {
782
+ // Failure-point parity from jess all-less extend-chaining.less:
783
+ // .ma:extend(.a, .md) parsed as a single Extend with SelectorList target,
784
+ // and compiler defaults to collapseNesting true.
785
+ const root = rules([
786
+ ruleset({
787
+ selector: sellist([sel([el('.a')])]),
788
+ rules: rules([decl({ name: 'color', value: any('black') })])
789
+ }),
790
+ atrule({
791
+ name: any('@media'),
792
+ prelude: any('(tv)'),
793
+ rules: rules([
794
+ ruleset({
795
+ selector: el('.ma'),
796
+ rules: rules([
797
+ decl({ name: 'color', value: any('black') }),
798
+ extend({ target: sellist([el('.a'), el('.md')]) })
799
+ ])
800
+ }),
801
+ ruleset({
802
+ selector: el('.md'),
803
+ rules: rules([decl({ name: 'color', value: any('inherit') })])
804
+ })
805
+ ])
806
+ }),
807
+ ruleset({
808
+ selector: el('.mb'),
809
+ rules: rules([extend({ target: el('.ma') })])
810
+ }),
811
+ ruleset({
812
+ selector: el('.mc'),
813
+ rules: rules([extend({ target: el('.mb') })])
814
+ })
815
+ ]);
816
+
817
+ const context = new Context({ collapseNesting: true });
818
+ const evald = await root.eval(context);
819
+ const css = evald.render(context);
820
+
821
+ expect(css).toBeString(`
822
+ .a {
823
+ color: black;
824
+ }
825
+ @media (tv) {
826
+ .ma,
827
+ .mb,
828
+ .mc {
829
+ color: black;
830
+ }
831
+ .md,
832
+ .ma,
833
+ .mb,
834
+ .mc {
835
+ color: inherit;
836
+ }
837
+ }
838
+ `);
839
+ });
840
+
841
+ /**
842
+ * Minimal tests for extend direction and scope across @media (Less semantics).
843
+ * These assert exact CSS so core matches real Less behavior; Jess must pass these.
844
+ */
845
+ describe('extend and @media: direction and scope (Less semantics)', () => {
846
+ /**
847
+ * A: .b:extend(.a) inside @media cannot reach OUT to root .a.
848
+ * Root .a stays unchanged; .b inside @media does NOT get .a's declarations (extend from inside @media does not copy decls from root).
849
+ */
850
+ it('A: .b:extend(.a) inside @media cannot reach out - root .a unchanged, .b in media does not get .a decls', async () => {
851
+ const root = rules([
852
+ ruleset({
853
+ selector: el('.a'),
854
+ rules: rules([decl({ name: 'color', value: spaced([any('red')]) })])
855
+ }),
856
+ atrule({
857
+ name: any('@media'),
858
+ prelude: any('screen'),
859
+ rules: rules([
860
+ ruleset({
861
+ selector: el('.b'),
862
+ rules: rules([
863
+ decl({ name: 'background', value: spaced([any('blue')]) }),
864
+ extend({ target: el('.a') })
865
+ ])
866
+ })
867
+ ])
868
+ })
869
+ ]);
870
+ const context = new Context({ collapseNesting: false });
871
+ const evald = await root.eval(context);
872
+ const css = evald.render(context);
873
+ // Less: .b:extend(.a) inside @media does NOT copy .a's declarations into .b. Root .a unchanged; .b has only its own decls.
874
+ expect(css).toBeString(`
875
+ .a {
876
+ color: red;
877
+ }
878
+ @media screen {
879
+ .b {
880
+ background: blue;
881
+ }
882
+ }
883
+ `);
884
+ });
885
+
886
+ /**
887
+ * B: .a:extend(.b) at root CAN reach IN to @media and merge with .b.
888
+ */
889
+ it('B: .a:extend(.b) at root can reach in - .a merged with .b inside @media', async () => {
890
+ const root = rules([
891
+ atrule({
892
+ name: any('@media'),
893
+ prelude: any('screen'),
894
+ rules: rules([
895
+ ruleset({
896
+ selector: el('.b'),
897
+ rules: rules([decl({ name: 'color', value: spaced([any('red')]) })])
898
+ })
899
+ ])
900
+ }),
901
+ ruleset({
902
+ selector: el('.a'),
903
+ rules: rules([
904
+ decl({ name: 'background', value: spaced([any('blue')]) }),
905
+ extend({ target: el('.b') })
906
+ ])
907
+ })
908
+ ]);
909
+ const context = new Context({ collapseNesting: false });
910
+ const evald = await root.eval(context);
911
+ const css = evald.render(context);
912
+ expect(css).toBeString(`
913
+ @media screen {
914
+ .b,
915
+ .a {
916
+ color: red;
917
+ }
918
+ }
919
+ .a {
920
+ background: blue;
921
+ }
922
+ `);
923
+ });
924
+
925
+ /**
926
+ * C: .c:extend(.b) inside @media extends sibling .b (same extend root).
927
+ */
928
+ it('C: .c:extend(.b) inside @media extends sibling .b - same extend root', async () => {
929
+ const root = rules([
930
+ atrule({
931
+ name: any('@media'),
932
+ prelude: any('screen'),
933
+ rules: rules([
934
+ ruleset({
935
+ selector: el('.b'),
936
+ rules: rules([decl({ name: 'color', value: spaced([any('red')]) })])
937
+ }),
938
+ ruleset({
939
+ selector: el('.c'),
940
+ rules: rules([
941
+ decl({ name: 'background', value: spaced([any('blue')]) }),
942
+ extend({ target: el('.b') })
943
+ ])
944
+ })
945
+ ])
946
+ })
947
+ ]);
948
+ const context = new Context({ collapseNesting: false });
949
+ const evald = await root.eval(context);
950
+ const css = evald.render(context);
951
+ expect(css).toBeString(`
952
+ @media screen {
953
+ .b,
954
+ .c {
955
+ color: red;
956
+ }
957
+ .c {
958
+ background: blue;
959
+ }
960
+ }
961
+ `);
962
+ });
963
+ });
964
+
965
+ it('PARITY: extend-selector nested all keeps parent prefix for footer/header and issue-2586 content', async () => {
966
+ // Failure-point parity from jess all-less extend-selector.less:
967
+ // with collapseNesting true, nested extends must keep resolved parent context.
968
+ const root = rules([
969
+ ruleset({
970
+ selector: el('.header'),
971
+ rules: rules([
972
+ ruleset({
973
+ selector: el('.header-nav'),
974
+ rules: rules([
975
+ decl({ name: 'background', value: any('red') }),
976
+ ruleset({
977
+ selector: sel([amp({}), pseudo({ name: ':before' })]) as any,
978
+ rules: rules([decl({ name: 'background', value: any('blue') })])
979
+ })
980
+ ])
981
+ })
982
+ ])
983
+ }),
984
+ ruleset({
985
+ selector: el('.footer'),
986
+ rules: rules([
987
+ ruleset({
988
+ selector: el('.footer-nav'),
989
+ rules: rules([
990
+ extend({
991
+ target: sel([el('.header'), co(' '), el('.header-nav')]),
992
+ flag: ExtendFlag.All
993
+ })
994
+ ])
995
+ })
996
+ ])
997
+ }),
998
+ ruleset({
999
+ selector: el('.issue-2586-bordered'),
1000
+ rules: rules([decl({ name: 'border', value: any('solid 1px black') })])
1001
+ }),
1002
+ ruleset({
1003
+ selector: el('.issue-2586-somepage'),
1004
+ rules: rules([
1005
+ ruleset({
1006
+ selector: el('.content'),
1007
+ rules: rules([
1008
+ extend({ target: el('.issue-2586-bordered'), flag: ExtendFlag.All })
1009
+ ])
1010
+ })
1011
+ ])
1012
+ })
1013
+ ]);
1014
+
1015
+ const context = new Context({ collapseNesting: true });
1016
+ const evald = await root.eval(context);
1017
+ const css = evald.render(context);
1018
+
1019
+ expect(css).toBeString(`
1020
+ .header .header-nav,
1021
+ .footer .footer-nav {
1022
+ background: red;
1023
+ }
1024
+ :is(.header .header-nav, .footer .footer-nav):before {
1025
+ background: blue;
1026
+ }
1027
+ .issue-2586-bordered,
1028
+ .issue-2586-somepage .content {
1029
+ border: solid 1px black;
1030
+ }
1031
+ `);
1032
+ });
1033
+ });