aboutsummaryrefslogtreecommitdiffstats
path: root/vanilla/node_modules/css-tree/lib/syntax/config/mix.js
diff options
context:
space:
mode:
authorAdam Mathes <adam@adammathes.com>2026-02-13 21:34:48 -0800
committerAdam Mathes <adam@adammathes.com>2026-02-13 21:34:48 -0800
commit76cb9c2a39d477a64824a985ade40507e3bbade1 (patch)
tree41e997aa9c6f538d3a136af61dae9424db2005a9 /vanilla/node_modules/css-tree/lib/syntax/config/mix.js
parent819a39a21ac992b1393244a4c283bbb125208c69 (diff)
downloadneko-76cb9c2a39d477a64824a985ade40507e3bbade1.tar.gz
neko-76cb9c2a39d477a64824a985ade40507e3bbade1.tar.bz2
neko-76cb9c2a39d477a64824a985ade40507e3bbade1.zip
feat(vanilla): add testing infrastructure and tests (NK-wjnczv)
Diffstat (limited to 'vanilla/node_modules/css-tree/lib/syntax/config/mix.js')
-rw-r--r--vanilla/node_modules/css-tree/lib/syntax/config/mix.js123
1 files changed, 123 insertions, 0 deletions
diff --git a/vanilla/node_modules/css-tree/lib/syntax/config/mix.js b/vanilla/node_modules/css-tree/lib/syntax/config/mix.js
new file mode 100644
index 0000000..f722ceb
--- /dev/null
+++ b/vanilla/node_modules/css-tree/lib/syntax/config/mix.js
@@ -0,0 +1,123 @@
+function appendOrSet(a, b) {
+ if (typeof b === 'string' && /^\s*\|/.test(b)) {
+ return typeof a === 'string'
+ ? a + b
+ : b.replace(/^\s*\|\s*/, '');
+ }
+
+ return b || null;
+}
+
+function sliceProps(obj, props) {
+ const result = Object.create(null);
+
+ for (const [key, value] of Object.entries(obj)) {
+ if (value) {
+ result[key] = {};
+ for (const prop of Object.keys(value)) {
+ if (props.includes(prop)) {
+ result[key][prop] = value[prop];
+ }
+ }
+ }
+ }
+
+ return result;
+}
+
+export default function mix(dest, src) {
+ const result = { ...dest };
+
+ for (const [prop, value] of Object.entries(src)) {
+ switch (prop) {
+ case 'generic':
+ result[prop] = Boolean(value);
+ break;
+
+ case 'cssWideKeywords':
+ result[prop] = dest[prop]
+ ? [...dest[prop], ...value]
+ : value || [];
+ break;
+
+ case 'units':
+ result[prop] = { ...dest[prop] };
+ for (const [name, patch] of Object.entries(value)) {
+ result[prop][name] = Array.isArray(patch) ? patch : [];
+ }
+ break;
+
+ case 'atrules':
+ result[prop] = { ...dest[prop] };
+
+ for (const [name, atrule] of Object.entries(value)) {
+ const exists = result[prop][name] || {};
+ const current = result[prop][name] = {
+ prelude: exists.prelude || null,
+ descriptors: {
+ ...exists.descriptors
+ }
+ };
+
+ if (!atrule) {
+ continue;
+ }
+
+ current.prelude = atrule.prelude
+ ? appendOrSet(current.prelude, atrule.prelude)
+ : current.prelude || null;
+
+ for (const [descriptorName, descriptorValue] of Object.entries(atrule.descriptors || {})) {
+ current.descriptors[descriptorName] = descriptorValue
+ ? appendOrSet(current.descriptors[descriptorName], descriptorValue)
+ : null;
+ }
+
+ if (!Object.keys(current.descriptors).length) {
+ current.descriptors = null;
+ }
+ }
+ break;
+
+ case 'types':
+ case 'properties':
+ result[prop] = { ...dest[prop] };
+ for (const [name, syntax] of Object.entries(value)) {
+ result[prop][name] = appendOrSet(result[prop][name], syntax);
+ }
+ break;
+
+ case 'scope':
+ case 'features':
+ result[prop] = { ...dest[prop] };
+ for (const [name, props] of Object.entries(value)) {
+ result[prop][name] = { ...result[prop][name], ...props };
+ }
+ break;
+
+ case 'parseContext':
+ result[prop] = {
+ ...dest[prop],
+ ...value
+ };
+ break;
+
+ case 'atrule':
+ case 'pseudo':
+ result[prop] = {
+ ...dest[prop],
+ ...sliceProps(value, ['parse'])
+ };
+ break;
+
+ case 'node':
+ result[prop] = {
+ ...dest[prop],
+ ...sliceProps(value, ['name', 'structure', 'parse', 'generate', 'walkContext'])
+ };
+ break;
+ }
+ }
+
+ return result;
+}