diff --git a/config/vue3migration/compiler.js b/config/vue3migration/compiler.js new file mode 100644 index 0000000000000000000000000000000000000000..bb92e1e2356a8ef637dc6447c5a5cd4563212d7c --- /dev/null +++ b/config/vue3migration/compiler.js @@ -0,0 +1,50 @@ +const { parse, compile: compilerDomCompile } = require('@vue/compiler-dom'); + +const getPropIndex = (node, prop) => node.props?.findIndex((p) => p.name === prop) ?? -1; + +function modifyKeysInsideTemplateTag(templateNode) { + let keyCandidate = null; + for (const node of templateNode.children) { + const keyBindingIndex = node.props + ? node.props.findIndex((prop) => prop.arg && prop.arg.content === 'key') + : -1; + + if (keyBindingIndex !== -1 && getPropIndex(node, 'for') === -1) { + if (!keyCandidate) { + keyCandidate = node.props[keyBindingIndex]; + } + node.props.splice(keyBindingIndex, 1); + } + } + + if (keyCandidate) { + templateNode.props.push(keyCandidate); + } +} + +module.exports = { + parse, + compile(template, options) { + const rootNode = parse(template, options); + const pendingNodes = [rootNode]; + while (pendingNodes.length) { + const currentNode = pendingNodes.pop(); + if (getPropIndex(currentNode, 'for') !== -1) { + if (currentNode.tag === 'template') { + // This one will be dropped all together with compiler when we drop Vue.js 2 support + modifyKeysInsideTemplateTag(currentNode); + } + + // This one will be dropped when https://github.com/vuejs/core/issues/7725 will be fixed + const vOncePropIndex = getPropIndex(currentNode, 'once'); + if (vOncePropIndex !== -1) { + currentNode.props.splice(vOncePropIndex, 1); + } + } + + currentNode.children?.forEach((child) => pendingNodes.push(child)); + } + + return compilerDomCompile(rootNode, options); + }, +}; diff --git a/config/webpack.config.js b/config/webpack.config.js index e60440e74b3f16f4d424bfdff8f95f59d4c46463..53479a5a9ba57f63ac8f0f6dffb775c8d61447b1 100644 --- a/config/webpack.config.js +++ b/config/webpack.config.js @@ -301,12 +301,15 @@ if (WEBPACK_USE_ESBUILD_LOADER) { } const vueLoaderOptions = { + ident: 'vue-loader-options', + cacheDirectory: path.join(CACHE_PATH, 'vue-loader'), cacheIdentifier: [ process.env.NODE_ENV || 'development', webpack.version, VUE_VERSION, VUE_LOADER_VERSION, + EXPLICIT_VUE_VERSION, ].join('|'), }; @@ -334,6 +337,8 @@ if (USE_VUE3) { Object.assign(alias, { vue: '@vue/compat', }); + + vueLoaderOptions.compiler = require.resolve('./vue3migration/compiler'); } module.exports = { diff --git a/jest.config.base.js b/jest.config.base.js index 611853c019cc3e1a4a4a6ae897fbc008f83d573d..50fd91b2e38e4291df740d905a5a532decc8aaac 100644 --- a/jest.config.base.js +++ b/jest.config.base.js @@ -44,6 +44,7 @@ module.exports = (path, options = {}) => { Object.assign(globals, { 'vue-jest': { experimentalCSSCompile: false, + compiler: require.resolve('./config/vue3migration/compiler'), compilerOptions: { compatConfig: { MODE: 2, diff --git a/patches/@vue+vue3-jest+29.2.3.patch b/patches/@vue+vue3-jest+29.2.3.patch new file mode 100644 index 0000000000000000000000000000000000000000..3b0ced984000d858c9b23ca16efaaa5ebd8cc833 --- /dev/null +++ b/patches/@vue+vue3-jest+29.2.3.patch @@ -0,0 +1,22 @@ +diff --git a/node_modules/@vue/vue3-jest/lib/process.js b/node_modules/@vue/vue3-jest/lib/process.js +index a8d1c5c..a6b2036 100644 +--- a/node_modules/@vue/vue3-jest/lib/process.js ++++ b/node_modules/@vue/vue3-jest/lib/process.js +@@ -108,12 +108,17 @@ function processTemplate(descriptor, filename, config) { + (descriptor.script && descriptor.script.lang) + const isTS = /^typescript$|tsx?$/.test(lang) + ++ const compiler = typeof vueJestConfig.compiler === 'string' ++ ? require(vueJestConfig.compiler) ++ : vueJestConfig.compiler ++ + const result = compileTemplate({ + id: filename, + source: template.content, + filename, + preprocessLang: template.lang, + preprocessOptions: vueJestConfig[template.lang], ++ compiler, + compilerOptions: { + bindingMetadata: bindings, + mode: 'module',