从零打造基于vue3+ts+tsx+vite的antdv中后台管理系统(一) 项目概览和环境搭建

8/12/2022 vue3tslessantdtsxviteantdv模板

# 一、前言

接触和使用 vue3 快 1 年了, 时至今日, vue3 的生态已经趋于成熟, 已结陆陆续续有不少公司开始使用 vue3 开发项目, 学习和使用 vue3 已是大势所趋. vue3-ts-antd-admin是基于 vue3、ts、tsx、vite 开发的一套中后台管理系统模板, 简洁轻量, 适合中小型中后台项目的开发.

本文会介绍该模板的功能和技术栈, 并从零开始介绍该系统的构造过程.

Github: 传送门

演示地址:传送门

编辑器预览:

预览.jpg 1661504529243.jpg

# 二、技术栈

  • 前端

    vue3: 核心框架
    pinia: 状态管理
    less: css 预编译器
    ant design: 组件库
    ts/tsx: 开发和模板语言
    vite2: 前端构建工具

  • 服务端

    koa2: 核心框架
    mysql: 数据库
    redis: 缓存
    ts:开发语言

本文主要涉及前端部分的设计和开发, 服务端的开发在我的掘金专栏 超细致 nodejs + koa2 + ts + mysql + redis 后端框架搭建 里有详细介绍, 感兴趣的可以看看

# 三、功能

本项目并不复杂, 只实现和保留了(我认为)最核心的部分. 后续会按重要程度和普适性增加功能, 如果有需要的功能也可以给我留言.

目前项目功能如下:

从业务上看, 系统包含:

  • 登录注册

  • 完整的权限管理模块, 包括 用户管理、角色管理、菜单管理等

  • 系统风格样式设置, 如主题色切换、布局切换、整体风格切换、色弱模式等等

从开发的角度上, 系统也对常见功能做了封装:

  • ts、tsx 支持

  • antdv 按需引入

  • 基于 eslint、prettier、husky 的完整的校验和格式化支持

  • 环境变量支持

  • vue-router 路由支持

  • pinia 状态管理

  • 自定义 icon 引入

  • less 支持

  • 表单封装

  • 图片预览封装

  • 常见 筛选表单 + 按钮组 + 表格页面封装

  • axios 请求封装和全局拦截

  • gzip

# 四、目录结构

├── .husky husky # 脚本
├── .vscode vscode # 配置
├── public
│ └── favicon.ico # 网站图标
├── src
│ ├── api # Api ajax 等
│ ├── assets # 本地静态资源
│ ├── config # 项目全局设置
│ ├── components # 通用组件
│ ├── core # 自定义指令等
│ ├── lib # 依赖包引入
│ ├── router # Vue-Router
│ ├── store # Pinia
│ ├── typings # .d.ts 描述文件
│ ├── utils # 工具库
│ ├── views # 业务页面入口和常用模板
│ └── App.vue # Vue 模板入口
│ └── env.d.ts # 环境变量定义文件
│ └── main.ts # Vue 入口 ts
│ └── permission.ts # 路由守卫(路由权限控制)
└── .env # 环境变量
└── .env.development # 开发环境变量
└── .env.production # 生产环境变量
└── .eslintrc.json # eslint 配置
└── .prettierrc.json # prettier 配置
└── deploy.sh # 部署脚本
└── index.html # Vue 入口模板
└── README.md
└── package.json
└── tsconfig.json
└── tsconfig.node.json
└── vite.config.ts

# 五、环境搭建

# 1. 项目创建

使用 vite 社区模板, 即可轻松创建基于 vue3 和 ts 的项目模板

npm init vite@latest

然后按照提示, 依次选择 vue 、 vue-ts , 即可创建 vue3 + ts + vite 项目

# 2. eslint、 prettier 和 husky 配置

# 安装依赖
yarn add eslint eslint-config-prettier eslint-plugin-prettier eslint-plugin-vue @typescript-eslint/eslint-plugin @typescript-eslint/parser prettier -D
# 配置 eslint 和 prettier
  1. 新增.eslintrc.json
// .eslintrc.json
{
  "env": {
    "browser": true,
    "es2021": true,
    "node": true,
    "vue/setup-compiler-macros": true
  },
  "extends": [
    "eslint:recommended",
    "plugin:vue/vue3-recommended",
    "plugin:@typescript-eslint/recommended",
    "plugin:prettier/recommended"
  ],
  "parser": "vue-eslint-parser",
  "parserOptions": {
    "ecmaVersion": "latest",
    "parser": "@typescript-eslint/parser",
    "sourceType": "module"
  },
  "plugins": ["vue", "@typescript-eslint"],
  "rules": {
    "vue/comment-directive": "off",
    "prettier/prettier": "off",
    // 允许单字单词作为组件名
    "vue/multi-word-component-names": "off",
    "@typescript-eslint/no-non-null-assertion": "off"
  }
}
  1. 新增.prettierrc.json
// .prettierrc.js
{
  "printWidth": 80,
  "tabWidth": 2,
  "useTabs": false,
  "semi": true,
  "singleQuote": true
}
  1. 安装 vscode 插件

安装下列 vscode 插件(已安装可跳过)

  • Vue Language Features (Volar)
  • Prettier - Code formatter

Volar 可以简单理解为是 vue3 的 Vetur, 如果是既有 vue2 项目又有 vue3 项目的, 可在工作区修改设置

新建.vscode 文件夹, 在.vscode 新建 extensions.json 和 settings.json

// .vscode/extensions.json
{
  "recommendations": ["johnsoncodehk.volar"]
}
// .vscode/settings.json
{
  "editor.formatOnSave": true,
  "editor.defaultFormatter": "esbenp.prettier-vscode",
  "editor.codeActionsOnSave": {
    "source.fixAll": true
  },
  "vetur.format.enable": true,
  "vetur.validation.script": false,
  "vetur.validation.style": false,
  "vetur.validation.template": false
}
  1. 安装 husky

husky 支持所有 git 钩子, 可以用来在提交校验代码

# 安装 husky

husky 是一款 git-hooks 执行工具, 帮助我们在提交代码时执行一些代码检查或其他的操作, 这里只用到 pre-commit 钩子。

yarn add husky -D
# husky 配置

在根目录新建 .husky 文件夹, 在 .husky 下新建 pre-commit 文件

# pre-commit

#!/bin/sh#!/bin/sh
. "$(dirname "$0")/_/husky.sh"

npx --no-install lint-staged

# 配置 lint 相关命令

修改 package.json

// package.json
...
"scripts": {
  "dev": "vite --mode development",
  "build": "vue-tsc --noEmit && vite build --mode production",
  "preview": "vite preview",
  "lint": "eslint . --ext .vue,.ts,.jsx,.tsx --fix",
  "format": "prettier --write ./**/*.{vue,ts,tsx,js,jsx,css,less,scss,json,md}",
  "prepare": "husky install"
}

执行 npm run prepare 或者当每次执行 npm install 时, 会初始化 husky, 即在.husky 目录生成*_*文件夹

当我们提交代码时, 会对代码进行全量检查

# 配置 lint-staged
// package.json

"lint-staged": {
  "*.{js,vue,ts,jsx,tsx}": [
    "prettier --write",
    "eslint --fix"
  ],
  "*.{html,css,less,scss,md}": [
    "prettier --write"
  ]
}

lint-staged 的作用在于可以只对 git staged(暂存区) 中要提交的代码进行检查

# 3. 路径别名配置 和 文件后缀省略

// vite.config.ts
import { defineConfig } from "vite";
import vue from "@vitejs/plugin-vue";
import path from "path";

// https://vitejs.dev/config/
export default defineConfig({
  plugins: [vue()],
  resolve: {
    alias: {
      "@": resolve("./src"),
      "~@": resolve("./src"),
    },
    // 省略文件后缀
    extensions: [".js", ".ts", ".json", ".jsx", ".tsx"],
  },
});

# 4. 添加 css 预处理器 less

# 安装 less
yarn add less -D
# 自定义全局 less 变量

在 assets 新建 styles 目录存放样式, 在 styles 下新建 common 目录存放公共样式和变量, 在 common 下新建 common.less、var.css 和 var.less

/* src/assets/styles/common/var.css 存放css变量 */
:root {
  --ant-primary-color: #18a058;
}
/* src/assets/styles/common/var.less 存放less变量 */
@primary-color: var(--ant-primary-color, #18a058);
/* src/assets/styles/common/common.less 存放公共样式 */
a {
  color: @primary-color;
}

在 vite.config.ts 引入 var.less, var.less 里定义的 less 变量便可以在全局使用

// vite.config.ts
import { defineConfig } from "vite";
import vue from "@vitejs/plugin-vue";
import path from "path";

function resolve(url: string): string {
  return path.resolve(__dirname, url);
}

// https://vitejs.dev/config/
export default defineConfig({
  plugins: [vue()],
  resolve: {
    alias: {
      "@": resolve("./src"),
      "~@": resolve("./src"),
    },
  },
  css: {
    preprocessorOptions: {
      less: {
        // 全局添加less
        additionalData: `@import '@/assets/styles/common/var.less';`,
        javascriptEnabled: true,
      },
    },
  },
});

此时 ts 并不能识别 nodejs 的模块, path 会报错, 需要安装@types/node

yarn add @types/node -D

然后修改 tsconfig.node.json

// tsconfig.node.json
{
  "compilerOptions": {
    "composite": true,
    "module": "esnext",
    "moduleResolution": "node",
    "allowSyntheticDefaultImports": true
  },
  "include": ["vite.config.ts"]
}

在 App.vue 里引入 var.css 和 common.less

<!-- src/App.vue -->
<script setup lang="ts">
// This starter template is using Vue 3 <script setup> SFCs
// Check out https://vuejs.org/api/sfc-script-setup.html#script-setup
import HelloWorld from "./components/HelloWorld.vue";
</script>

<template>
  <HelloWorld msg="Hello Vue 3 + TypeScript + Vite" />
</template>

<style>
@import url("@/assets/styles/common/var.css");
@import url("@/assets/styles/common/common.less");
</style>

# 5. 开发服务器配置

// vite.config.ts

export default defineConfig({
  ...
  server: {
    port: 8001, // 端口
    proxy: { // 请求转发
      '/api': {
        target: 'http://localhost:9002',
        ws: false,
        changeOrigin: true,
        rewrite: (path) => path.replace(/^\/api/, '/api/v1'),
      },
      '/resource': {
        target: 'https://static.fhtwl.cc',
        ws: false,
        changeOrigin: true,
        rewrite: (path) => path.replace(/^\/resource/, ''),
      },
    },
  },
});

# 6. ts/tsx 配置

在创建项目时, 我们选择 ts - vue-ts ,此时项目已经支持 ts , 支持 tsx 还需要安装插件

yarn add @vitejs/pugin-vue-jsx

配置插件

// vite.config.ts

import { defineConfig } from "vite";
import vue from "@vitejs/plugin-vue";
import vueJsx from "@vitejs/plugin-vue-jsx";

export default defineConfig({
  plugins: [vue(), vueJsx()],
});

# 7. antdv 按需引入

# 安装 antdv
yarn add ant-design-vue
yarn add @ant-design/icons-vue -D
# 按需加载

安装按需加载插件

yarn add unplugin-vue-components -D

引入插件

// vite.config.ts
import Components from "unplugin-vue-components/vite";
import { AntDesignVueResolver } from "unplugin-vue-components/resolvers";
import { defineConfig } from "vite";
import vue from "@vitejs/plugin-vue";
import vueJsx from "@vitejs/plugin-vue-jsx";

export default defineConfig({
  plugins: [
    vue(),
    vueJsx(),
    // 按需加载
    Components({
      include: [/\.vue$/, /\.tsx$/], // 支持 vue和tsx
      resolvers: [
        AntDesignVueResolver({
          // 不加载css, 而是手动加载css. 通过手动加载less文件并将less变量绑定到css变量上, 即可实现动态主题色
          importStyle: false,
          // 加载所有icon
          resolveIcons: true,
        }),
      ],
    }),
  ],
});

unplugin-vue-components 在监测到项目使用 ts 后, 会在项目根目录生成一个 components.d.ts, 里面即是根据组件内容生成的类型声明, 为 volar 提供类型提示, 他会自动根据文件变动进行更新(注意: compnents.d.ts 文件会在 vue-tsc 运行的时候进行更新,建议把他加入 gitignore 中,以免出现频繁更改导致 git 监测到项目内容一直变动的问题)

unplugin-vue-components 还支持项目中开发的公共组件的自动引入, 即 dirs 配置, 其默认值 就是 src/components, 也就是说, src/components 下的组件同样会被自动引入

# 挂载全局变量

在 src/lib 下新建 ant-design-vue 目录, 在 ant-design-vue 下新建 index.ts

// src/lib/ant-design-vue/index.ts

import { message, Modal, notification } from "ant-design-vue";

import { App } from "vue";

export default function setupAtnd(app: App<Element>) {
  app.config.globalProperties.$message = message;
  app.config.globalProperties.$confirm = Modal.confirm;
  app.config.globalProperties.$notification = notification;
}

然后在 main.ts 里执行

// main.ts

import { createApp } from "vue";
import App from "./App.vue";
import setupAtnd from "@/lib/ant-design-vue";

const app = createApp(App);

setupAtnd(app);

app.mount("#app");

此时使用 this.$message 会报错, 提示不存在 $message 这个属性, 因此还需要对这些自定义全局变量进行声明

// src/typings/common/index.d.ts
import { MessageApi } from "ant-design-vue/lib/message";
import { ModalFunc } from "ant-design-vue/lib/modal/Modal";
import { NotificationApi } from "ant-design-vue/lib/notification";

declare module "@vue/runtime-core" {
  // 定义绑定在全局vue示例上的变量
  interface ComponentCustomProperties {
    $message: MessageApi;
    $confirm: ModalFunc;
    $notification: NotificationApi;
  }
}

# 总结

本文是 从零打造基于 vue3+ts+tsx+vite 的 antdv 中后台管理系统 系列的第一篇, 主要对项目进行了一个整体的介绍以及对开发环境的搭建, 下一步就可以正常开始开发

文中若有错误或者可优化之处, 望请不吝赐教. 如果对你有帮助的话, 麻烦点个赞