- web/: Vue 3 + Vite + UnoCSS + Pinia, dark tactical theme (amber/#0d1117) - AppNav, ListingCard, SearchView with filters/sort, composables (useSnipeMode, useKonamiCode, useMotion), Pinia search store - Steal shimmer, auction countdown, Snipe Mode easter egg all native in Vue - docker/web/: nginx + multi-stage Dockerfile (node build → nginx serve) - compose.yml: api (8510) + web (8509) services - Dockerfile CMD updated to uvicorn for upcoming FastAPI layer - Clean build: 0 TS errors, 380 modules
53 lines
1.2 KiB
JavaScript
53 lines
1.2 KiB
JavaScript
const noop = function() {};
|
|
|
|
function ensureFunction(value) {
|
|
return typeof value === 'function' ? value : noop;
|
|
}
|
|
|
|
export function walk(node, options, context) {
|
|
function walk(node) {
|
|
enter.call(context, node);
|
|
|
|
switch (node.type) {
|
|
case 'Group':
|
|
node.terms.forEach(walk);
|
|
break;
|
|
|
|
case 'Multiplier':
|
|
case 'Boolean':
|
|
walk(node.term);
|
|
break;
|
|
|
|
case 'Type':
|
|
case 'Property':
|
|
case 'Keyword':
|
|
case 'AtKeyword':
|
|
case 'Function':
|
|
case 'String':
|
|
case 'Token':
|
|
case 'Comma':
|
|
break;
|
|
|
|
default:
|
|
throw new Error('Unknown type: ' + node.type);
|
|
}
|
|
|
|
leave.call(context, node);
|
|
}
|
|
|
|
let enter = noop;
|
|
let leave = noop;
|
|
|
|
if (typeof options === 'function') {
|
|
enter = options;
|
|
} else if (options) {
|
|
enter = ensureFunction(options.enter);
|
|
leave = ensureFunction(options.leave);
|
|
}
|
|
|
|
if (enter === noop && leave === noop) {
|
|
throw new Error('Neither `enter` nor `leave` walker handler is set or both aren\'t a function');
|
|
}
|
|
|
|
walk(node, context);
|
|
};
|