這是xue源碼學習記錄,如有錯誤請指出,謝謝!相互學習相互進步。
vue源碼目錄為vue ├── src #vue源碼 ├── flow #flow定義的數據類型庫(vue通過flow來檢測數據類型是否正確) ├── examples #demo ├── scripts #vue構建命令 ├── ...
vue內部代碼模塊比較清晰,這邊主要分析scripts內部代碼,講解vue是如何進行構建的.
首先你必須要懂一些rollup,vue內部是通過rollup來進行構建的,rollup是一款js的構建工具,
將各個小模塊打包成一個總的模塊(只針對js文件,比較輕量,不會有css,img等壓縮,比較適合開發插件,
如果是ui組件庫的話,還是webpack構建會比較好。)
rollup說明文檔:https://rollupjs.cn/
文件主要是scripts下的alias.js,config.js和build.js三個文件組成
alias主要就是提供文件對應的路徑
const path = require("path") const resolve = p => path.resolve(__dirname, "../", p) // 以下是設置別名,與對應的真實文件路徑 module.exports = { vue: resolve("src/platforms/web/entry-runtime-with-compiler"), compiler: resolve("src/compiler"), core: resolve("src/core"), shared: resolve("src/shared"), web: resolve("src/platforms/web"), weex: resolve("src/platforms/weex"), server: resolve("src/server"), entries: resolve("src/entries"), sfc: resolve("src/sfc"), test: resolve("src/test") // 這個是測試目錄是自己添加的 }
其中test是我自己加的,為了測試打包
configconfig是為了提供打包的基礎配置(即rollup打包配置文件格式),由于打包內容比較多,
所以做成可配置的
const path = require("path") const buble = require("rollup-plugin-buble") // 提供modules名稱的 alias 和reslove 功能 const alias = require("rollup-plugin-alias") // 將CommonJS模塊轉換為 ES2015供Rollup 處理 const cjs = require("rollup-plugin-commonjs") // 變量替換,可以將動態設置的變量提取出來在配置文件中設置 const replace = require("rollup-plugin-replace") // 幫助 Rollup 查找外部模塊,然后安裝 const node = require("rollup-plugin-node-resolve") const flow = require("rollup-plugin-flow-no-whitespace") const version = process.env.VERSION || require("../package.json").version // 下面是weex服務器端代碼,不需要管 // const weexVersion = process.env.WEEX_VERSION || require("../packages/weex-vue-framework/package.json").version // 這邊是打包完成后模塊外部首行注釋代碼 const banner = `/* * test-vue.js v${version} * (C) 2014-${new Date().getFullYear()} Enan You * @author zhengjie */ ` // 獲取文件夾路徑別名 const aliases = require("./alias") // 尋找路徑 const resolve = p => { const base = p.split("/")[0] if (aliases[base]) { return path.resolve(aliases[base], p.slice(base.length + 1)) } else { return path.resolve(__dirname, "../", p) } } // 設置打包參數 // 這邊把原先vue打包文件去除,替換成知己的一個測試文件 const builds = { "test-cjs": { entry: resolve("test/main.js"), dest: resolve("dist/test-cjs.js"), format: "cjs", // csj格式 module.exports banner }, "test-es": { entry: resolve("test/main.js"), dest: resolve("dist/test-es.js"), format: "es", // es格式 export default banner }, "test-umd": { entry: resolve("test/main.js"), dest: resolve("dist/test-umd.js"), format: "umd", // 瀏覽器格式 return banner } } // 根據上面builds打包轉換成rollup打包格式 function getConfig(name) { const opts = builds[name] const config = { input: opts.entry, external: opts.external, plugins: [ replace({ __WEEX__: !!opts.weex, __WEEX_VERSION__: null, __VERSION__: version }), flow(), buble(), alias(Object.assign({}, aliases, opts.alias)) ].concat(opts.plugins || []), output: { file: opts.dest, format: opts.format, banner: opts.banner, name: opts.moduleName || "Vue" } } // 如果是開發模式 if (opts.env) { config.plugins.push(replace({ "process.env.NODE_ENV": JSON.stringify(opts.env) })) } // 增加屬性 Object.defineProperty(config, "_name", { enumerable: false, value: name }) return config } if (process.env.TARGET) { module.exports = getConfig(process.env.TARGET) } else { exports.getBuild = getConfig exports.getAllBuilds = () => Object.keys(builds).map(getConfig) }
其中里面的builds已被替換成我自己的測試文件,用于測試打敗es,模塊化和瀏覽器的不同格式。
buildbuild文件就是根據配置文件進行打包,打包模式分為全部打包,或者是可配置打包,
如果運行npm run build,將會打包所有的配置,
而運行npm run build "參數", 則根據參數配置進行打包
const fs = require("fs") const path = require("path") const zlib = require("zlib") const rollup = require("rollup") const uglify = require("uglify-js") // 檢測是否有dist文件 if (!fs.existsSync("dist")) { fs.mkdirSync("dist") } let builds = require("./config").getAllBuilds() // build后面輸入的參數 if (process.argv[2]) { // 過濾出需要打包的數組 const filters = process.argv[2].split(",") builds = builds.filter(b => { return filters.some(f => b.output.file.indexOf(f) > -1 || b._name.indexOf(f) > -1) }) } else { // 這邊過濾出weex,不需要管 builds = builds.filter(b => { return b.output.file.indexOf("weex") === -1 }) } // 把需要打包的打包出來 build(builds) // 打包函數 function build(builds) { let built = 0 const total = builds.length const next = () => { buildEntry(builds[built]).then(() => { built++ if (built < total) { next() } }).catch(logError) } next() } // 單個配置文件打包 function buildEntry(config) { const output = config.output const {file, banner} = output // 是否為壓縮文件 const isProd = /min.js$/.test(file) return rollup.rollup(config) .then(bundle => bundle.generate(output)) .then(({code}) => { // 壓縮我文件 if (isProd) { var minified = (banner ? banner + " " : "") + uglify.minify(code, { output: { ascii_only: true }, compress: { pure_funcs: ["makeMap"] } }).code return write(file, minified, true) } else { return write(file, code) } }) } function write(dest, code, zip) { return new Promise((resolve, reject) => { function report (extra) { console.log(blue(path.relative(process.cwd(), dest)) + " " + getSize(code) + (extra || "")) resolve() } fs.writeFile(dest, code, err => { if (err) return reject(err) if (zip) { zlib.gzip(code, (err, zipped) => { if (err) return reject(err) report(" (gzipped: " + getSize(zipped) + ")") }) } else { report() } }) }) } // 計算文件大小 function getSize (code) { return (code.length / 1024).toFixed(2) + "kb" } // 輸入錯誤信息 function logError (e) { console.log(e) } function blue (str) { return "x1b[1mx1b[34m" + str + "x1b[39mx1b[22m" }測試文件
main.js
import foo from "./foo"; export default function () { console.log(foo) }
foo.js
export default "hello rollup"
通過npm run build打包測試,會在dist文件打包出test-cjs.js,test-es.js,test-umd.js
文件內容如下:
test-cjs.js
/* * test-vue.js v1.0.0 * (C) 2014-2018 Enan You * @author zhengjie */ "use strict"; var foo = "hello rollup" function main () { console.log(foo); } module.exports = main;
test-es.js
/* * test-vue.js v1.0.0 * (C) 2014-2018 Enan You * @author zhengjie */ var foo = "hello rollup" function main () { console.log(foo); } export default main;
test-umd.js
/* * test-vue.js v1.0.0 * (C) 2014-2018 Enan You * @author zhengjie */ (function (global, factory) { typeof exports === "object" && typeof module !== "undefined" ? module.exports = factory() : typeof define === "function" && define.amd ? define(factory) : (global.Vue = factory()); }(this, (function () { "use strict"; var foo = "hello rollup" function main () { console.log(foo); } return main; })));
這個構建的過程不難,比起webpack的配置文件要容易懂很多
懂得了vue的構建,接下來就可以開始vue源碼的學習了
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/98401.html
摘要:至此算是找到了源碼位置。至此進入過渡的部分完畢。在動畫結束后,調用了由組件生命周期傳入的方法,把這個元素的副本移出了文檔流。這篇并沒有去分析相關的內容,推薦一篇講非常不錯的文章,對構造函數如何來的感興趣的同學可以看這里 Vue transition源碼分析 本來打算自己造一個transition的輪子,所以決定先看看源碼,理清思路。Vue的transition組件提供了一系列鉤子函數,...
摘要:框架使用了作為類型檢查,來保證項目的可讀性和維護性。的主目錄下有的配置文件,還有目錄,指定了各種自定義類型。表示構建出來的文件遵循規范。 Flow vue框架使用了Flow作為類型檢查,來保證項目的可讀性和維護性。vue.js的主目錄下有Flow的配置.flowconfig文件,還有flow目錄,指定了各種自定義類型。 在學習源碼前可以先看下Flow的語法 官方文檔 目錄結構 vue....
摘要:源碼目錄結構打包相關的配置文件,其中最重要的是。主要是根據不同的入口,打包為不同的文件。這個目錄下的代碼邏輯會把文件內容解析成一個的對象。 源碼目錄結構 VUE 2.6.10 ├── scripts # 打包相關的配置文件,其中最重要的是config.js。主要是根據不同的入口,打 包為不同的文件。 ├── dist # 打包之后文...
摘要:了解的構建,可以從文件開始命令內關于構建的命令如下版本從命令看出是通過來進行構建的同樣的構建工具更強大一些,可以處理圖片等但是只做的處理,相比之下更輕量所以更適合框架的打包總共有三個版本的構建從版本的為起點來了解,版本的構建 了解Vue的構建,可以從package.json文件開始sprits命令內關于構建的命令如下 { ... build: node scri...
摘要:運行時用來創建實例渲染并處理虛擬等的代碼。基本上就是除去編譯器的其它一切。版本可以通過標簽直接用在瀏覽器中。為這些打包工具提供的默認文件是只有運行時的構建。為瀏覽器提供的用于在現代瀏覽器中通過直接導入。 Vue版本:2.6.9 源碼結構圖 ├─ .circleci // 包含CircleCI持續集成/持續部署工具的配置文件 ├─ .github ...
閱讀 2088·2021-11-23 10:13
閱讀 2795·2021-11-09 09:47
閱讀 2742·2021-09-22 15:08
閱讀 3322·2021-09-03 10:46
閱讀 2235·2019-08-30 15:54
閱讀 916·2019-08-28 18:09
閱讀 2432·2019-08-26 18:26
閱讀 2346·2019-08-26 13:48