欧美一级特黄大片做受成人-亚洲成人一区二区电影-激情熟女一区二区三区-日韩专区欧美专区国产专区

VueJSScopedCSS的實現(xiàn)原理分析-創(chuàng)新互聯(lián)

這篇文章主要介紹VueJS Scoped CSS的實現(xiàn)原理分析,文中介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們一定要看完!

創(chuàng)新互聯(lián)于2013年成立,是專業(yè)互聯(lián)網(wǎng)技術服務公司,擁有項目成都做網(wǎng)站、網(wǎng)站設計網(wǎng)站策劃,項目實施與項目整合能力。我們以讓每一個夢想脫穎而出為使命,1280元克東做網(wǎng)站,已為上家服務,為克東各地企業(yè)和個人服務,聯(lián)系電話:028-86922220

使用VueJS進行應用開發(fā), 脫離不了對應用間的模塊進行拆分, 將大塊界面拆解為組件的過程. 我們可以很方便的在單文件中使用<template>塊維護組件的視圖, 使用<script>維護組件的邏輯部分, 使用<style>維護組件的樣式. 在我們編寫 VueJS 組件樣式時, 不得忽略的一點就是樣式污染.

樣式污染產(chǎn)生原因

提及樣式污染, 主要要追溯到Webpack對CSS文件的打包過程, 這里我們以Vue-Element-Admin中的Webpack配置項舉例:

const webpackConfig = merge(baseWebpackConfig, {
 plugins: [
 new MiniCssExtractPlugin({
     filename: utils.assetsPath('css/[name].[contenthash:8].css'),
     chunkFilename: utils.assetsPath('css/[name].[contenthash:8].css')
    }),
 ]
})

Webpack 使用 MiniCssExtractPlugin 插件, 將文件(如Vue單文件組件)中的CSS代碼, 經(jīng)過處理后, 分離到形如app.hash2234.css的單獨的CSS文件:

VueJS Scoped CSS的實現(xiàn)原理分析

如果沒有加入防止樣式污染的措施的同時, 項目中存在了大量的同名 ClassName, 那么可能會產(chǎn)生意想不到的CSS選擇器權重覆蓋. 這可能使后文件中某部分選擇器權重更高的類影響整個應用, 而此過程通常發(fā)生在組件的編寫中, 所以一般稱之為組件樣式污染.

Webpack & Vue SFC Object

對于 Vue 項目而言, 使用 Webpack 將極大的優(yōu)化了工作流程, 因為通過Vue Loader, Vue 單文件組件能很好的融合進 Webpack 工作流中. 通過跟蹤源碼, 可以發(fā)現(xiàn), 我們寫的單文件組件都被處理為了SFC對象, 即包含了單個HTML模塊, 單個腳本模塊, 一個或多個樣式模塊, 一個或多個自定義模塊的對象:

// vue-loader/index.js
const descriptor = parse({
  source,
  compiler: options.compiler || loadTemplateCompiler(),
  filename,
  sourceRoot,
  needMap: sourceMap
})

// vuejs/component-compiler-utils/index.js
function parse(options) {
  const { compiler } = options
  output = compiler.parseComponent(source, compilerParseOptions)
  return output
}

// vue.js
function parseComponent(content, options) {
 // ...
  var sfc = {
    template: null,
    script: null,
    styles: [],
    customBlocks: []
  }
  // ...
  return sfc
}

我們可以將SFC結構融合到Webpack進行開發(fā)的過程成中, 主要有這幾點影響:

  • 允許為 Vue 組件的每個部分使用其它的 webpack loader,例如在 <style>的部分使用 Sass Loader , 在 <customBlocks>的部分使用自定義 Loader

  • 使用 webpack loader 將 <style>和 <template> 中引用的資源當作模塊依賴來處理

  • 模擬 Scoped CSS

  • 在開發(fā)過程中使用熱重載來保持狀態(tài)

以下主要介紹Scoped CSS的原理.

Scoped CSS

大白話版本之 Scoped CSS 原理

通過 Webpack 調(diào)用 VueJS 中相應 Loader , 給組件HTML模板添加自定義屬性 (Attribute) data-v-x, 以及給組件內(nèi)CSS選擇器添加對應的屬性選擇器 (Attribute Selector) [data-v-x], 達到組件內(nèi)樣式只能生效與組件內(nèi)HTML的效果, 代碼效果如下:

<div class='lionad' data-v-lionad></div>
<style>
.lionad[data-v-lionad] {
 background: @tiger-orange;
}
</style>

源碼跟蹤

Webpack 使用其它 CSS Loader 處理 VueJS 中對應 CSS 代碼之前, Vue Loader 已經(jīng)替我們做了一層簡單的處理, 如果組件中 style 塊包含了 scoped 屬性:

<!-- 某個VueJS組件中 -->
<template>
  <div class='lionad'></div>
</template>
<style lang="scss" scoped>
  .lionad {
    background: @tiger-orange;
  }
</style>

下代碼即判斷當前SFC對象樣式塊中是否有scoped屬性, 并插入用于 query 中, 順帶一提, 每個單文件組件被解析后, 都會生成對應組件ID, ID主要以生產(chǎn)/開發(fā)環(huán)境做區(qū)分, 通過文件路徑+源碼或是文件路徑的值作為哈希特征值的形式生成, 如下:

// vue-loader/index.js
const id = hash(isProduction (shortFilePath + '\n' + source) : shortFilePath)
const hasScoped = descriptor.styles.some(s => s.scoped)
const query = `? vue&type=template${idQuery}${scopedQuery}`
const request = templateRequest = stringifyRequest(src + query)
templateImport = `import { render, staticRenderFns } from ${request}`

HTML模板處理

在用于處理SFC結構中HTML模板的 templateLoader 中, 我們可以得知, query 中所設置的參數(shù)將合并為 loader options 經(jīng)由 Webpack 轉(zhuǎn)交 templateLoader 再轉(zhuǎn)交 @vue/component-compiler-utils.compileTemplate 處理:

// vue-loader/templateLoader.js
const query = qs.parse(this.resourceQuery)
const { id } = query
const compilerOptions = Object.assign({}, options.compilerOptions, {
  scopeId: query.scoped ? `data-v-${id}` : null
})
const compiled = compileTemplate({ compilerOptions })

實際 compileTemplate 函數(shù)在處理內(nèi)容時, 編譯函數(shù)使用的是 query 中的 compiler 或 vue-template-compiler, 后者會將模板文本轉(zhuǎn)換成為 JavaScript 渲染函數(shù), 大致如下:

  1. 從HTML模版轉(zhuǎn)換為AST(虛擬語法樹)

  2. AST優(yōu)化,處理靜態(tài)模版與動態(tài)模板

  3. 生成JS函數(shù),用于在運行時運行時生成純HTML

代碼分別對應:

// vue-template-compiler/build.js/createCompilerCreator
var ast = parse(template.trim(), options)
optimize(ast, options)
var code = generate(ast, options)

先前我們的組件ID在 parse 階段解析開始標簽時就會被推入內(nèi)部儲存的數(shù)據(jù)結構中:

function elementToOpenTagSegments (el, state) {
 var segments = [{ type: RAW, value: ("<" + (el.tag)) }]
 // _scopedId
 if (state.options.scopeId) {
  segments.push({ type: RAW, value: (" " + (state.options.scopeId)) })
 }
 segments.push({ type: RAW, value: ">" })
 return segments
}

先前我們的HTML模板 <div class='lionad'></div> 中開始標簽會被轉(zhuǎn)換成如下數(shù)據(jù)結構:

[
  { type: RAW, value: '<div' },
  { type: RAW, value: 'class=lionad' },
  { type: RAW, value: 'data-v-xxxxxx' },
  { type: RAW, value: '>' },
]

樣式模板處理

與 HTML Template 解析的過程類似, 通過 Webpack 將樣式模板轉(zhuǎn)交 stylePostLoader 進行處理, 處理邏輯主要引用了 @vue/component-compiler-utils 中的 compileStyle 部分, 后者對樣式模板進行解析的過程中, 將會對含 scoped 標記的模板引入插件 stylePlugins/scoped.js, scoped.js 將 data-v-xxxxxx 添加到選擇器末尾的過程如下:

selectors.each((selector) => {
  selector.each((n) => {
    if (n.value === '::v-deep' || n.value === '>>>' || n.value === '/deep/') {
      return false;
    }
  });
  selector.insertAfter(node, selectorParser.attribute({
    attribute: id
  }))
})

題外話, 通過以上代碼, 我們發(fā)現(xiàn)當當前處理到三種特定類型選擇器會終止循環(huán), 停止將 data-v-xxx 添加到選擇器末尾:

  1. 偽類 ::v-deep

  2. 選擇器 >>>

  3. 選擇器 /deep/

我們可以利用這個特征, 在組件中寫樣式穿透, 即內(nèi)部組件影響外部組件樣式 (ε=ε=ε=┏(゜ロ゜;)┛ 主動樣式污染), 當然這在特定的情境下是有用的, 比如當我們想主動覆蓋第三方UI組件框架的樣式, 卻不想引入新的CSS文件, 或不想寫非 Scoped CSS 模板的時候.

以上是“VueJS Scoped CSS的實現(xiàn)原理分析”這篇文章的所有內(nèi)容,感謝各位的閱讀!希望分享的內(nèi)容對大家有幫助,更多相關知識,歡迎關注創(chuàng)新互聯(lián)行業(yè)資訊頻道!

分享文章:VueJSScopedCSS的實現(xiàn)原理分析-創(chuàng)新互聯(lián)
網(wǎng)頁網(wǎng)址:http://aaarwkj.com/article42/hsjhc.html

成都網(wǎng)站建設公司_創(chuàng)新互聯(lián),為您提供面包屑導航靜態(tài)網(wǎng)站、微信公眾號、企業(yè)網(wǎng)站制作動態(tài)網(wǎng)站、網(wǎng)站收錄

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

商城網(wǎng)站建設
国产亚洲成人精品久久| 可以看黄片的在线观看 | 91熟女激情五月综合| 在线精品91国产在线观看| 正在播放蜜臀av在线| 天堂av日韩在线播放| 亚洲一区二区三区色偷偷| 91久久国产香蕉熟女| 亚洲高清中文字幕一区二三区| 日本不卡一区二区在线视频| 国产一区精品在线免费看| 亚洲乱码一区二区av| 偷拍视频在线一区二区| 五十路八十路息与子交尾| 深夜福利视频一区二区| 亚洲精品日韩av专区| 欧美日韩成人精品一区二区| 中文字幕人妻久久一区| 久久精品国产一区二区三区91| 欧美激情欧美狂野欧美精品| 一区二区中文字幕精品| 91口爆吞精国产对白| 色呦呦视频在线免费观看| 国产剧情av在线播放| 日韩一区二区三区av观看| 性生活自制视频网站麻豆| 国产精品久久久毛片av| 欧美日韩精品免费在线观看| 国产成人免费自拍一区| 亚洲毛片高清一区二区三区| 超碰91人人草人人干| 六月综合激情丁香婷婷色| 色哟哟亚洲精品一区二区| 日韩在线免费色视频| 成人欧美一区二区三区av| 欧美性大片一区二区三区| 在线激情视频一区二区| 青青久久精品国产亚洲av| 日韩免费的黄色片网站| 久久精品国产欧美日韩| 中文字幕人妻久久精品一区|