持续创作,加速成长!这是我参与「日新计划 6 月更文挑战」的第1天,点击查看活动详情

前言

大家好,我是易师傅,在现如今 vite 工具快开始盛行之下,我们是不是可以去做一件有意义的事呢,比如写一个 vite 插件,你觉得怎么样?

刚好我们可以趁 vite 插件 生态前端工程师还未很成熟阶段,做一前端工程师个让自己顺心,让领导赏心,让社区开心的插件,与之携手共进。

如果大家对 vite 感兴趣可以去看看专栏:《Vite 从入门到精通》

通过本文你可以学到

  • 如何创建一个 vite 插件模板
  • vite 插件的 各个钩子作仓鼠饲养八大禁忌
  • vite 插件的 钩子执行顺序
  • 如何写一个自己的插件

了解 vite 插件

1. 什么是 vite 插件

vite 其实就是一个由原生ES Module驱动的新型 Web 开发前端构建工具。

vite 插件 就可以很好的扩优先级是什么意思vite 自身不能做到的事情,比如优先级英文 文件图片的压缩对 commonjs 的支持打包进度条 等等。

2. 为什么要写 vite 插件

相信在座的每位同学,到现在对 webpack 的相关配置以及常用插件都了如指掌了吧;

vite 作为一个新型的前端和后端的区别前端构建工具,它还很年轻,也有很多扩展性,那么为什么我们不趁现在与它一起携手前进呢?做一些于你于我于大家更有giticomfort是什么轮胎意义的事呢?

快速体验

要想写一个插件,那必须从创建一个项目开始,下面的 viESLintte 插件通用模板 大家以后写插件可以直接clone使用;

插件通用模板 github:体验入口

插件 github:体验入口

建议包管理器使用优先级:pnpm > yarn > npm > cnpm

长话短说,直接开干 ~

创建 vite 插件通用模板

1. 初始化

1.1 创建一个文件夹并且初始化:初始化按照提示操作即可

mkdir vite-plugin-progress && cd vite-plugin-progress && pnpm init

1.2giticomfort是什么轮胎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. 配置 eslintprettier(可选)

  1. 安装 esliGitnt

    pnpm i eslint @typescript-eslint/parser @typescript-eslint/eslint-plugin --save-dev
    
  2. 配置 .eslintrc:配置连接

  3. 安装 prettier (可选)

    pnpm i prettier eslint-config-prettier eslint-plugin-prettier --save-dev
    
  4. 配置 .prettierrc :配置连接

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 就行了,这里就不详细介绍了

  1. vite-r优先级英文eact
  2. vite-vue2
  3. vite-vue3

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

那么下面我们就需要配置 examples 下的项目与当前根目录的插件做一个联调了(下面以 examp优先级排序les/vite-vue3 为例)。

5. 配置 examples/vite-vue3 项目

  1. 修改 examples/vite-vue3/package.json

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

前端和后端哪个工资高面意思就是说:

  • 要把 examgit教程ples/vite-vue3 项目中的 vite 版本与根目录 vite-plugin-template 的版本一致;

  • 同时要把 examples/vite-vue常熟市疫情3 项目中的 vite-plugin-templat优先级调度算法e 指向你当前根目录所开发的插件;

  1. 引入插件: examples/vite-vue3/vite前端和后端哪个工资高.config.ts

    import template from 'vite-plugin-template';
    export default defineConfig({
        ...
        plugins: [vue(), template()],
        ...
    });
    
  2. 安装: cd examples/vite-vue3 && pnpm instal优先级英文l

    cd examples/vite-vue3 && pnpm install
    

注意: examples/vite-vue2examples/vite-react 的配置与这一致

思考:

到这里,我们再思考一下,我们把 exa长生十万年mples/vitgitlabe-vue3 中的项目配置好了,但是我们应该怎么去运行呢?

直接去 examples/vite-vue3 目录下运行仓鼠饲养八大禁忌 pnpm run build 或者 pnpm run dev

这样显然是不能运行成功的,因为我们的根目录下的 src/indgitiex.ts 是没法直接运行的,所以我们需要把 .ts 文件转义成 .js 文件;

那么我们怎么处理呢?

那么我们不得不去试着用用一个轻小且无需配置的工具 tsup 了。

6. 安装 tsup 配置运行命令

tsup 是一个轻小且无需配置的,由 esbuild 支持的构建工具;

同时它可以直接把优先级英文 .ts、.tsx 转成不同格式 esm、cjs、iife 的工具;

  1. 安装 t优先级队列sup

    pnpm i tsup -D
    
  2. 在根目录下的 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. 开发环境运行

  1. 开发环境运行:实时监听文件修改后重新打包(热更新)

    pnpm run dev
    
  2. 运行 examples 中的任意一个项目(以 vite-vue3 为例)

    pnpm run example:vue3
    

注意:

如果你的插件只会在 build 时运行,优先级越小越优先吗那就设置 "example:vue3": "cd examples/vite-vue3 && pnpm run build"

反之就运行 pn前端开发需要掌握什么技术pm run dev

  1. 输出:

你还不会写 vite 插件吗?没关系,我教你啊!

到这里你就可以 边开发边运行 了,尤雨溪看了都说爽歪歪 ~

8. 发布

  1. 安装 bu优先级表mpp 添加版本控制与 t优先级排序ag
pnpm i bumpp -D
  1. 配置 package.json
{
  ...
  "scripts": {
    ...
    "prepublishOnly": "pnpm run build",
    "release": "npx bumpp --push --tag --commit && pnpm publish",
  },
  ...
}
  1. 开发完插件后运行发布
# 第一步
pnpm run prepublishOnly
# 第二步
pnpm run release

那么到这里,我们的 vite 插件模板 就已经写好了,大家可以直接克隆 vite-plugin-template 模板 使用;

如果你对 vite 的插件钩子实现一个真正的 vite 插件 感兴趣可以继续往下面看;

vite 的插件钩子 hooks 们

1. vite 独有的钩子

  1. enforce :值可以是prepos前端开发需要学什么tpre 会较于 posgiti轮胎t 先执行;
  2. appl优先级排序c语言y :值可以是 buildserve 亦可以是一个函数,指明它们仅在 build优先级是什么意思 serve 模式时调用;
  3. c优先级c语言onfig(config, env) :可以在 veslint报错怎么解决ite 被解析之前修改 vite 的相关配置。钩子接收原始用gitee户配置 config 和一个giti描述配置环境的变量env;
  4. configResolved(resolvedConfig) :在解析 vite 配置后调用。使用这个钩子读取和存储最终解析的配置。当插件需长沙市天气要根据运行的命令做一些不同的事情时,它很有用。
  5. config长生十万年ureServer(server) :主要用来配置开优先级越小越优先吗发服务器,为 dev-server (connect 应用程序) 添加自定义的中间件;
  6. transformIndexHtml(html) :转换 index.html 的专用钩子。钩子接收当前的 HTML 字符串和转换上下文;
  7. handleHot长沙师范学院Update(ctx):执行自定义HMR更新,eslint是什么可以通过ws往客户端发送前端工程师自定义的事件;

2. vite 与 rollup 的通用钩子之构建阶段

  1. options(o前端和后端ptions) :在优先级是什么意思服务器启动时被调用:获取、操纵Rollup选项,严格意义上来讲,它执行于属于构建阶段之前;
  2. bu优先级英文ildStart(options):在每次开始构建时调用;
  3. resolveId(优先级越小越优先吗source, importer, options):在每个传入模块请求时被调用,创建自定义确认函数,可以用来定位第三方依赖;
  4. load(id):在每个传入模块请求时被调用,可以自定义加载器,可用来返回自定义的内容;
  5. transfogiti轮胎rm(code, id):在每个传入模块请求时被调用优先级越小越优先吗,主要是用来转换单个模块;
  6. buildEnd():在构建阶段结束后被调用,此处构建结束只是代表所有模块转义完成;

3. vite长沙市天气 与 rollup 的通用钩子之输出阶段

  1. outputOptions(options):接受长沙市疫情最新情况输出参数;
  2. renderStart(outputESLintOp前端开发需要掌握什么技术tions, inputOptions):每次 bundle.generate 和 bundle优先级越小越优先吗.write 调用时都会被触发;
  3. augmentChunkHa优先级队列sh(chunkInfo):用来给优先级越小越优先吗 chunk 增加 haseslintrc.js配置h;
  4. renderChunk(code, chunk, options):转译单个的chunk时触发。rollup 输出每一个c长沙师范学院hunk文件前端面试题的时候都会调用;
  5. gener优先级ateBundle(options, bundle, isWriteESLint):在调用 bundle.write 之前立即触发这个 hook;
  6. writeBundle(options, bundle):在调用 bundle.write后,所有的chunk都写入文件后,最后会调用一次 writeBundle;
  7. closeBundle():在服务器长沙市疫情最新情况关闭时被调用

4. 插件钩子函数 hooks 的执行顺序(如下图)

你还不会写 vite 插件吗?没关系,我教你啊!

5. 插件的执行顺序

  1. 别名处理Aliaeslintrc.js配置s
  2. 优先级队列户插件设置enforce: 'pre'
  3. vitegithub 核心插件
  4. 用户插件未设置enforce
  5. vite 构建插件
  6. 用户插件设置enforce: 'post'
  7. vite 构建后置插件(minify, manifgit命令est, reporting)

手撸一个 veslint怎么配置ite 插件

下面以优先级排序 vite 打包进度条陈涉世家翻译及原文件为例;

你还不会写 vite 插件吗?没关系,我教你啊!

插件地址:github 如果您觉得不错欢迎 star ⭐️

该插件已被 vite 官方收集至官方文档:链接地址

因为文章的重点不在于这个插件的详前端开发细实现过程,所以本文只会贴上源代码供大家参考,详细介绍会在前端开发下一篇文前端和后端章中讲解,请大家拭目以待吧!

  1. 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 = 0
    let chunkCount = 0
    let transformed = 0
    let fileCount = 0
    let lastPercent = 0
    let percent = 0
    return {
        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 directory
                if (!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 cache
            if(!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 cache
            if (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 progress
            bar.update(1)
            bar.terminate()
            // set cache data
            setCacheData({
                cacheTransformCount: transformCount,
                cacheChunkCount: chunkCount,
            })
            // out successful message
            stream.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)
    }
}
  1. 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 从入长沙师范学院门到精通》专栏,我主要会从如下图几个方面讲解,请大家拭目以待吧!!!

你还不会写 vite 插件吗?没关系,我教你啊!

宝贝们,都看到这里了,要不点个赞呗