日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

【Vite】1371- 手把手开发 Vite 插件

發布時間:2023/12/29 编程问答 24 豆豆
生活随笔 收集整理的這篇文章主要介紹了 【Vite】1371- 手把手开发 Vite 插件 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

前言

大家好,我是易師傅,在現如今?vite?工具快開始盛行之下,我們是不是可以去做一件有意義的事呢,比如寫一個?vite 插件,你覺得怎么樣?

剛好我們可以趁?vite 插件?生態還未很成熟階段,做一個讓自己順心,讓領導賞心,讓社區開心的插件,與之攜手共進。

如果大家對 vite 感興趣可以去看看專欄:?《Vite 從入門到精通》:juejin.cn/column/7074954144817086472[1]

通過本文你可以學到

  • 如何創建一個?vite 插件模板

  • vite 插件的?各個鉤子作用

  • vite 插件的?鉤子執行順序

  • 如何寫一個自己的插件

了解 vite 插件

1. 什么是 vite 插件

vite?其實就是一個由原生?ES Module?驅動的新型 Web 開發前端構建工具。

vite 插件?就可以很好的擴展?vite?自身不能做到的事情,比如?文件圖片的壓縮、?對 commonjs 的支持、?打包進度條?等等。

2. 為什么要寫 vite 插件

相信在座的每位同學,到現在對?webpack?的相關配置以及常用插件都了如指掌了吧;

vite?作為一個新型的前端構建工具,它還很年輕,也有很多擴展性,那么為什么我們不趁現在與它一起攜手前進呢?做一些于你于我于大家更有意義的事呢?

快速體驗

要想寫一個插件,那必須從創建一個項目開始,下面的?vite 插件通用模板?大家以后寫插件可以直接clone使用;

插件通用模板 github:體驗入口:github.com/jeddygong/vite-templates/tree/master/vite-plugin-template[2]

插件 github:體驗入口:github.com/jeddygong/vite-plugin-progress[3]

建議包管理器使用優先級:pnpm > yarn > npm > cnpm

長話短說,直接開干 ~

創建?vite 插件通用模板

1. 初始化

1.1?創建一個文件夾并且初始化:初始化按照提示操作即可

mkdir?vite-plugin-progress?&&?cd?vite-plugin-progress?&&?pnpm?init? 復制代碼

1.2?安裝?typescript

pnpm?i?typescript?@types/node?-D 復制代碼

1.3?配置?tsconfig.json

{"compilerOptions":?{"module":?"ESNext","target":?"esnext","moduleResolution":?"node","strict":?true,"declaration":?true,"noUnusedLocals":?true,"esModuleInterop":?true,"outDir":?"dist","lib":?["ESNext"],"sourceMap":?false,"noEmitOnError":?true,"noImplicitAny":?false},"include":?["src/*","*.d.ts"],"exclude":?["node_modules","examples","dist"] } 復制代碼

1.4?安裝?vite

//?進入?package.json {..."devDependencies":?{"vite":?"*"}... } 復制代碼

2. 配置?eslint?和?prettier(可選)

  • 安裝?eslint

  • pnpm?i?eslint?@typescript-eslint/parser?@typescript-eslint/eslint-plugin?--save-dev
  • 配置?.eslintrc:配置連接[4]

  • 安裝?prettier?(可選)

  • pnpm?i?prettier?eslint-config-prettier?eslint-plugin-prettier?--save-dev
  • 配置?.prettierrc?:配置連接[5]

  • 3. 新增?src/index.ts?入口

    import type { PluginOption } from 'vite';export default function vitePluginTemplate(): PluginOption {return {// 插件名稱name: 'vite-plugin-template',// pre 會較于 post 先執行enforce: 'pre', // post// 指明它們僅在 'build' 或 'serve' 模式時調用apply: 'build', // apply 亦可以是一個函數config(config, { command }) {console.log('這里是config鉤子');},configResolved(resolvedConfig) {console.log('這里是configResolved鉤子');},configureServer(server) {console.log('這里是configureServer鉤子');},transformIndexHtml(html) {console.log('這里是transformIndexHtml鉤子');},} } 復制代碼

    其中的 vite 插件函數鉤子會在下面詳細詳解 ~

    到這里,那么我們的基本模版就建好了,但是我們現在思考一下,我們應該怎么去運行這個插件呢?

    那么我們就需要創建一些?examples?例子來運行這個代碼了;

    4. 創建 examples 目錄

    我這里創建了三套項目 demo,大家直接 copy 就行了,這里就不詳細介紹了

  • vite-react:github.com/jeddygong/vite-templates/tree/master/vite-plugin-template/examples/vite-react[6]

  • vite-vue2:github.com/jeddygong/vite-templates/tree/master/vite-plugin-template/examples/vite-vue2[7]

  • vite-vue3:github.com/jeddygong/vite-templates/tree/master/vite-plugin-template/examples/vite-vue3[8]

  • 如果你的插件需要多跑一些 demo,自行創建項目即可;

    那么下面我們就需要配置 examples 下的項目與當前根目錄的插件做一個聯調了(下面以?examples/vite-vue3?為例)。

    5. 配置?examples/vite-vue3?項目

  • 修改?examples/vite-vue3/package.json

  • {..."devDependencies":?{..."vite":?"link:../../node_modules/vite","vite-plugin-template":?"link:../../"} }

    上面意思就是說:

    • 要把?examples/vite-vue3?項目中的 vite 版本與根目錄?vite-plugin-template?的版本一致;

    • 同時要把?examples/vite-vue3?項目中的?vite-plugin-template?指向你當前根目錄所開發的插件;

  • 引入插件:?examples/vite-vue3/vite.config.ts

  • import template from 'vite-plugin-template';export default defineConfig({...plugins: [vue(), template()],... });
  • 安裝:?cd examples/vite-vue3 && pnpm install

  • cd?examples/vite-vue3?&&?pnpm?install

    注意:?examples/vite-vue2?和?examples/vite-react?的配置與這一致

    思考:

    到這里,我們再思考一下,我們把?examples/vite-vue3?中的項目配置好了,但是我們應該怎么去運行呢?

    直接去?examples/vite-vue3?目錄下運行?pnpm run build?或者?pnpm run dev??

    這樣顯然是不能運行成功的,因為我們的根目錄下的?src/index.ts?是沒法直接運行的,所以我們需要把?.ts?文件轉義成?.js?文件;

    那么我們怎么處理呢?

    那么我們不得不去試著用用一個輕小且無需配置的工具?tsup?了。

    6. 安裝?tsup?配置運行命令

    tsup?是一個輕小且無需配置的,由?esbuild?支持的構建工具;

    同時它可以直接把?.ts、.tsx?轉成不同格式?esm、cjs、iife?的工具;

  • 安裝?tsup

  • pnpm?i?tsup?-D
  • 在根目錄下的?package.json?中配置

  • {..."scripts":?{"dev":?"pnpm?run?build?--?--watch?--ignore-watch?examples","build":?"tsup?src/index.ts?--dts?--format?cjs,esm","example:react":?"cd?examples/vite-react?&&?pnpm?run?build","example:vue2":?"cd?examples/vite-vue2?&&?pnpm?run?build","example:vue3":?"cd?examples/vite-vue3?&&?pnpm?run?build"},... }

    7. 開發環境運行

  • 開發環境運行:實時監聽文件修改后重新打包(熱更新)

  • pnpm?run?dev
  • 運行?examples?中的任意一個項目(以 vite-vue3 為例)

  • pnpm?run?example:vue3

    注意:

    如果你的插件只會在 build 時運行,那就設置?"example:vue3": "cd examples/vite-vue3 && pnpm run build"?;

    反之就運行?pnpm run dev

  • 輸出:

  • Untitled.png

    到這里你就可以?邊開發邊運行?了,尤雨溪看了都說爽歪歪 ~

    8. 發布

  • 安裝 `bumpp` 添加版本控制與 tag
  • pnpm?i?bumpp?-D 復制代碼
  • 配置 `package.json`
  • {..."scripts":?{..."prepublishOnly":?"pnpm?run?build","release":?"npx?bumpp?--push?--tag?--commit?&&?pnpm?publish",},... } 復制代碼
  • 開發完插件后運行發布
  • #?第一步 pnpm?run?prepublishOnly#?第二步 pnpm?run?release 復制代碼

    那么到這里,我們的?vite 插件模板?就已經寫好了,大家可以直接克隆?vite-plugin-template 模板[9]?使用;

    如果你對?vite 的插件鉤子?和?實現一個真正的 vite 插件?感興趣可以繼續往下面看;

    vite 的插件鉤子 hooks 們

    1. vite 獨有的鉤子

  • enforce?:值可以是pre?或?post?,?pre?會較于?post?先執行;

  • apply?:值可以是?build?或?serve?亦可以是一個函數,指明它們僅在?build?或?serve?模式時調用;

  • config(config, env)?:可以在 vite 被解析之前修改 vite 的相關配置。鉤子接收原始用戶配置 config 和一個描述配置環境的變量env;

  • configResolved(resolvedConfig)?:在解析 vite 配置后調用。使用這個鉤子讀取和存儲最終解析的配置。當插件需要根據運行的命令做一些不同的事情時,它很有用。

  • configureServer(server)?:主要用來配置開發服務器,為 dev-server (connect 應用程序) 添加自定義的中間件;

  • transformIndexHtml(html)?:轉換 index.html 的專用鉤子。鉤子接收當前的 HTML 字符串和轉換上下文;

  • handleHotUpdate(ctx):執行自定義HMR更新,可以通過ws往客戶端發送自定義的事件;

  • 2. vite 與 rollup 的通用鉤子之構建階段

  • options(options)?:在服務器啟動時被調用:獲取、操縱Rollup選項,嚴格意義上來講,它執行于屬于構建階段之前;

  • buildStart(options):在每次開始構建時調用;

  • resolveId(source, importer, options):在每個傳入模塊請求時被調用,創建自定義確認函數,可以用來定位第三方依賴;

  • load(id):在每個傳入模塊請求時被調用,可以自定義加載器,可用來返回自定義的內容;

  • transform(code, id):在每個傳入模塊請求時被調用,主要是用來轉換單個模塊;

  • buildEnd():在構建階段結束后被調用,此處構建結束只是代表所有模塊轉義完成;

  • 3. vite 與 rollup 的通用鉤子之輸出階段

  • outputOptions(options):接受輸出參數;

  • renderStart(outputOptions, inputOptions):每次 bundle.generate 和 bundle.write 調用時都會被觸發;

  • augmentChunkHash(chunkInfo):用來給 chunk 增加 hash;

  • renderChunk(code, chunk, options):轉譯單個的chunk時觸發。rollup 輸出每一個chunk文件的時候都會調用;

  • generateBundle(options, bundle, isWrite):在調用 bundle.write 之前立即觸發這個 hook;

  • writeBundle(options, bundle):在調用 bundle.write后,所有的chunk都寫入文件后,最后會調用一次 writeBundle;

  • closeBundle():在服務器關閉時被調用

  • 4. 插件鉤子函數 hooks 的執行順序(如下圖)

    vite插件開發鉤子函數 (1).png

    5. 插件的執行順序

  • 別名處理Alias

  • 用戶插件設置enforce: 'pre'

  • vite 核心插件

  • 用戶插件未設置enforce

  • vite 構建插件

  • 用戶插件設置enforce: 'post'

  • vite 構建后置插件(minify, manifest, reporting)

  • 手擼一個 vite 插件

    下面以?vite 打包進度條?插件為例;

    demo.gif

    插件地址:github[10]?如果您覺得不錯歡迎 star ??

    該插件已被 vite 官方收集至官方文檔:鏈接地址[11]

    因為文章的重點不在于這個插件的詳細實現過程,所以本文只會貼上源代碼供大家參考,詳細介紹會在下一篇文章中講解,請大家拭目以待吧!

  • `inde.ts`
  • import type { PluginOption } from 'vite'; import colors from 'picocolors'; import progress from 'progress'; import rd from 'rd'; import { isExists, getCacheData, setCacheData } from './cache';type Omit<T, K extends keyof T> = Pick<T, Exclude<keyof T, K>>; type Merge<M, N> = Omit<M, Extract<keyof M, keyof N>> & N;type PluginOptions = Merge<ProgressBar.ProgressBarOptions,{/*** total number of ticks to complete* @default 100*/total?: number;/*** The format of the progress bar*/format?: string;} >;export default function viteProgressBar(options?: PluginOptions): PluginOption {const { cacheTransformCount, cacheChunkCount } = getCacheData()let bar: progress;const stream = options?.stream || process.stderr;let outDir: string;let transformCount = 0let chunkCount = 0let transformed = 0let fileCount = 0let lastPercent = 0let percent = 0return {name: 'vite-plugin-progress',enforce: 'pre',apply: 'build',config(config, { command }) {if (command === 'build') {config.logLevel = 'silent';outDir = config.build?.outDir || 'dist';options = {width: 40,complete: '\u2588',incomplete: '\u2591',...options};options.total = options?.total || 100;const transforming = isExists ? `${colors.magenta('Transforms:')} :transformCur/:transformTotal | ` : ''const chunks = isExists ? `${colors.magenta('Chunks:')} :chunkCur/:chunkTotal | ` : ''const barText = `${colors.cyan(`[:bar]`)}`const barFormat =options.format ||`${colors.green('Bouilding')} ${barText} :percent | ${transforming}${chunks}Time: :elapseds`delete options.format;bar = new progress(barFormat, options as ProgressBar.ProgressBarOptions);// not cache: Loop files in src directoryif (!isExists) {const readDir = rd.readSync('src');const reg = /\.(vue|ts|js|jsx|tsx|css|scss||sass|styl|less)$/gi;readDir.forEach((item) => reg.test(item) && fileCount++);}}},transform(code, id) {transformCount++// not cacheif(!isExists) {const reg = /node_modules/gi;if (!reg.test(id) && percent < 0.25) {transformed++percent = +(transformed / (fileCount * 2)).toFixed(2)percent < 0.8 && (lastPercent = percent)}if (percent >= 0.25 && lastPercent <= 0.65) {lastPercent = +(lastPercent + 0.001).toFixed(4)} }// go cacheif (isExists) runCachedData()bar.update(lastPercent, {transformTotal: cacheTransformCount,transformCur: transformCount,chunkTotal: cacheChunkCount,chunkCur: 0,})return {code,map: null};},renderChunk() {chunkCount++if (lastPercent <= 0.95) isExists ? runCachedData() : (lastPercent = +(lastPercent + 0.005).toFixed(4))bar.update(lastPercent, {transformTotal: cacheTransformCount,transformCur: transformCount,chunkTotal: cacheChunkCount,chunkCur: chunkCount,})return null},closeBundle() {// close progressbar.update(1)bar.terminate()// set cache datasetCacheData({cacheTransformCount: transformCount,cacheChunkCount: chunkCount,})// out successful messagestream.write(`${colors.cyan(colors.bold(`Build successful. Please see ${outDir} directory`))}`);stream.write('\n');stream.write('\n');}};/*** run cache data of progress*/function runCachedData() {if (transformCount === 1) {stream.write('\n');bar.tick({transformTotal: cacheTransformCount,transformCur: transformCount,chunkTotal: cacheChunkCount,chunkCur: 0,})}transformed++percent = lastPercent = +(transformed / (cacheTransformCount + cacheChunkCount)).toFixed(2)} }復制代碼
  • `cache.ts`
  • import fs from 'fs'; import path from 'path';const dirPath = path.join(process.cwd(), 'node_modules', '.progress'); const filePath = path.join(dirPath, 'index.json');export interface ICacheData {/*** Transform all count*/cacheTransformCount: number;/*** chunk all count*/cacheChunkCount: number }/*** It has been cached* @return boolean*/ export const isExists = fs.existsSync(filePath) || false;/*** Get cached data* @returns ICacheData*/ export const getCacheData = (): ICacheData => {if (!isExists) return {cacheTransformCount: 0,cacheChunkCount: 0};return JSON.parse(fs.readFileSync(filePath, 'utf8')); };/*** Set the data to be cached* @returns */ export const setCacheData = (data: ICacheData) => {!isExists && fs.mkdirSync(dirPath);fs.writeFileSync(filePath, JSON.stringify(data)); };復制代碼

    最后

    該系列會是一個持續更新系列,關于整個《Vite 從入門到精通》專欄[12],我主要會從如下圖幾個方面講解,請大家拭目以待吧!!!

    Untitled.png

    寶貝們,都看到這里了,要不點個贊唄 👍

    關于本文

    來自:易師傅

    https://juejin.cn/post/7103165205483356168

    總結

    以上是生活随笔為你收集整理的【Vite】1371- 手把手开发 Vite 插件的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

    主站蜘蛛池模板: 精品福利在线视频 | 6080av| 九九热免费 | 精品人妻人人做人人爽夜夜爽 | 国产乱子伦农村叉叉叉 | 先锋影音av资源站 | 亚洲系列在线观看 | 好吊视频一区二区三区 | 手机在线看片你懂的 | 国产精品三级在线 | 中文字幕123区 | 亚洲精品小视频 | 婷婷一区二区三区 | 美女又爽又黄免费 | www.四虎.com| 熟女一区二区三区视频 | 天堂av2020 | 中文字幕人妻一区二区 | 体感预报日剧 | 久久九九国产精品 | av片在线免费观看 | 黄色片aaaa | 久久精品99国产 | 亚洲淫| 国产理论在线观看 | 国产一卡二卡三卡四卡 | 蜜臀av性久久久久蜜臀aⅴ | 亚洲视频在线免费 | 另类色综合 | 蜜桃久久精品成人无码av | 一本免费视频 | 91手机在线播放 | 日本美女裸体视频 | 豆花免费跳转入口官网 | 禁断介护av一区二区 | 欧美美女黄色 | 蜜桃av乱码一区二区三区 | 手机天堂av | 日本黄色a级片 | 亚洲日本久久久 | 91精品免费看 | 秋霞精品一区二区三区 | www黄色在线观看 | www.av天天| 天天插夜夜 | 天天色天天射天天干 | 国产在线不卡 | 国产精品久久九九 | 探花系列在线观看 | 日韩欧美视频一区二区 | 专业操老外 | 国产一级免费 | 久久免费公开视频 | 日本免费无人高清 | 欧美乱大交xxxxx潮喷l头像 | 精品免费囯产一区二区三区 | 三级色网站 | 精品人妻伦九区久久aaa片 | 国产一级免费观看 | 四虎影院国产精品 | www.爱色av.com | www.蜜臀| 国产人人看 | 亚洲乱色 | 亚洲欧美日韩国产一区二区 | 伊人操| 亚洲av无码一区二区三区观看 | 日韩亚洲欧美一区二区三区 | 免费视频色 | 欧洲性开放大片 | 国语对白永久免费 | 国产精品久久av无码一区二区 | 国产精品成人久久久 | 亚洲视屏一区 | 国产一级生活片 | 国产专区在线视频 | 国产婷婷色一区二区在线观看 | 亚洲最大免费视频 | 天堂av.com| 日韩欧美一区二区一幕 | 亚洲va欧美 | 国产精品情侣自拍 | 亚洲综合免费观看高清完整版 | 欧美第一页在线 | 国产精品成久久久久三级 | 久久一级黄色片 | 色欲亚洲Av无码精品天堂 | 在线观看a级片 | 在线中文字幕观看 | 久久精品毛片 | 日本视频www | 亚洲欧美日韩一区 | 免费黄色网址在线观看 | 不卡国产视频 | 色综合婷婷| 一级全黄裸体免费视频 | 国产精品久久久一区二区 | 91免费成人 | 日本久久久久久久久久 |