diff --git a/packages/vant-cli/package.json b/packages/vant-cli/package.json index 0461f1387..750046e1a 100644 --- a/packages/vant-cli/package.json +++ b/packages/vant-cli/package.json @@ -74,6 +74,7 @@ "eslint": "^6.8.0", "fast-glob": "^3.2.2", "gh-pages": "2.0.1", + "hash-sum": "^2.0.0", "html-webpack-plugin": "3.2.0", "husky": "^4.2.3", "jest": "^25.1.0", diff --git a/packages/vant-cli/src/compiler/compile-sfc.ts b/packages/vant-cli/src/compiler/compile-sfc.ts index 1cb6badbd..0a162e3c9 100644 --- a/packages/vant-cli/src/compiler/compile-sfc.ts +++ b/packages/vant-cli/src/compiler/compile-sfc.ts @@ -1,5 +1,6 @@ import * as compiler from 'vue-template-compiler'; import * as compileUtils from '@vue/component-compiler-utils'; +import hash from 'hash-sum'; import { parse } from 'path'; import { remove, writeFileSync, readFileSync } from 'fs-extra'; import { replaceExt } from '../common'; @@ -34,6 +35,10 @@ function injectRender(script: string, render: string) { ); } +function injectScopeId(script: string, scopeId: string) { + return script.replace(EXPORT, `${EXPORT}\n _scopeId: '${scopeId}',\n\n`); +} + function injectStyle( script: string, styles: compileUtils.SFCBlock[], @@ -63,10 +68,6 @@ function compileTemplate(template: string) { return result.code; } -type CompileSfcOptions = { - skipStyle?: boolean; -}; - export function parseSfc(filePath: string) { const source = readFileSync(filePath, 'utf-8'); @@ -79,15 +80,16 @@ export function parseSfc(filePath: string) { return descriptor; } -export async function compileSfc( - filePath: string, - options: CompileSfcOptions = {} -): Promise { +export async function compileSfc(filePath: string): Promise { const tasks = [remove(filePath)]; + const source = readFileSync(filePath, 'utf-8'); const jsFilePath = replaceExt(filePath, '.js'); const descriptor = parseSfc(filePath); const { template, styles } = descriptor; + const hasScoped = styles.some(s => s.scoped); + const scopeId = hasScoped ? `data-v-${hash(source)}` : ''; + // compile js part if (descriptor.script) { tasks.push( @@ -100,6 +102,10 @@ export async function compileSfc( script = injectRender(script, render); } + if (scopeId) { + script = injectScopeId(script, scopeId); + } + writeFileSync(jsFilePath, script); compileJs(jsFilePath) .then(resolve) @@ -109,21 +115,27 @@ export async function compileSfc( } // compile style part - if (!options.skipStyle) { - tasks.push( - ...styles.map((style, index: number) => { - const cssFilePath = getSfcStylePath( - filePath, - style.lang || 'css', - index - ); + tasks.push( + ...styles.map((style, index: number) => { + const cssFilePath = getSfcStylePath(filePath, style.lang || 'css', index); - writeFileSync(cssFilePath, trim(style.content)); + let styleSource = trim(style.content); - return compileStyle(cssFilePath); - }) - ); - } + if (style.scoped) { + styleSource = compileUtils.compileStyle({ + id: scopeId, + scoped: true, + source: styleSource, + filename: cssFilePath, + preprocessLang: style.lang, + }).code; + } + + writeFileSync(cssFilePath, styleSource); + + return compileStyle(cssFilePath); + }) + ); return Promise.all(tasks); } diff --git a/packages/vant-cli/src/module.d.ts b/packages/vant-cli/src/module.d.ts index ca3ef7e5d..99188db52 100644 --- a/packages/vant-cli/src/module.d.ts +++ b/packages/vant-cli/src/module.d.ts @@ -2,6 +2,7 @@ declare module 'less'; declare module 'sass'; declare module 'execa'; +declare module 'hash-sum'; declare module 'clean-css'; declare module 'webpackbar'; declare module 'release-it'; diff --git a/packages/vant-cli/yarn.lock b/packages/vant-cli/yarn.lock index 76eb6d731..5f064d6c3 100644 --- a/packages/vant-cli/yarn.lock +++ b/packages/vant-cli/yarn.lock @@ -5370,6 +5370,11 @@ hash-sum@^1.0.2: resolved "https://registry.yarnpkg.com/hash-sum/-/hash-sum-1.0.2.tgz#33b40777754c6432573c120cc3808bbd10d47f04" integrity sha1-M7QHd3VMZDJXPBIMw4CLvRDUfwQ= +hash-sum@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/hash-sum/-/hash-sum-2.0.0.tgz#81d01bb5de8ea4a214ad5d6ead1b523460b0b45a" + integrity sha512-WdZTbAByD+pHfl/g9QSsBIIwy8IT+EsPiKDs0KNX+zSHhdDLFKdZu0BQHljvO+0QI/BasbMSUa8wYNCZTvhslg== + hash.js@^1.0.0, hash.js@^1.0.3: version "1.1.7" resolved "https://registry.yarnpkg.com/hash.js/-/hash.js-1.1.7.tgz#0babca538e8d4ee4a0f8988d68866537a003cf42"