用vite搭建一个vue3 + ts项目

vite + ts 项目搭建

创建项目

// 1. 全局安装vite
npm i vite -g
// 2. 创建项目
npm init vite@latest
// 3. 输入项目名称,选择模板
// 4. 按提示运行npm install , npm run dev
image-20220106103432239.png

看上面的截图,这里面没有vue+ts 所以后面要手动添加ts,如果有vue+ts直接选择就好。

vite.config.js

import { defineConfig } from 'vite'
// import path from 'path'
import vue from '@vitejs/plugin-vue'
import vueJsx from '@vitejs/plugin-vue-jsx'
import { resolve } from 'path'

// https://vitejs.dev/config/
export default defineConfig({
  server: {
    host: '0.0.0.0', //use `--host` to expose
    port: 8099,
    open: true
  },
  resolve: {
    alias: {
      "@": resolve(__dirname, 'src')
    },
  },
  plugins: [vue(), vueJsx()]
})

集成ts

  1. 安装ts

    npm i typescript -S

  2. 创建tsconfig.json 文件

    npx tsc --init

  3. 将main.js改成main.ts,index.html引入也改成mian.ts

  4. 在项目根目录创建shim.d.ts文件,文件内 写入以下文件,用于ts识别.vue文件

declare module "*.vue" {
    import { Component } from "vue";
    const component: Compoent;
    export default component;
}

tsconfig.json

{
  "compilerOptions": {
      "target": "es2016", 
      "jsx": "preserve",
      "module": "esnext",
      "moduleResolution": "node",
      "baseUrl": "./", 
      "paths": {
          "@/*": [
            "src/*"
          ]
       }, 
      "strict": true,
       "skipLibCheck": true
  }
}

集成router

  1. 安装

    npm install vue-router@4

  2. 创建router/index.ts文件

   import { createRouter, createWebHashHistory, RouteRecordRaw } from 'vue-router'
   import HelloWorld from '@/components/HelloWorld.vue'
   const routes: Array<RouteRecordRaw> = [
     {
       path: '/',
       name: 'home',
       component: HelloWorld
     }
   ]
   
   const router = createRouter({
     history: createWebHashHistory(),
     routes
   })
   export default router
   
  1. 在main.ts中导入使用

    import router from './router/index'

    app.use(router)

如果以模块(module)的形式使用router

  1. 在router中创建module目录

  2. 在module中创建一个??榈膔outer文件

    如:setting.ts

import type { RouteRecordRaw } from 'vue-router';

const routes: RouteRecordRaw[] = [
  {
    component: () => import('@/pages/Settting.vue'),
    path: '/setting',
    redirect: { name: 'authority' },
    children: [
      {
        path: 'authority',
        name: 'Authority',
        component: () => import('@/pages/settting/Authority.vue'),
      },
      {
        path: 'userList',
        name: 'UserList',
        component: () => import('@/pages/settting/UserList.vue'),
      },
    ],
  },
];

export default routes;
  1. 在router目录下创建index.ts文件,并导入???/li>
  import type { RouteRecordRaw } from 'vue-router';
  import { createRouter, createWebHashHistory } from 'vue-router';
  // vite2
  const routes: RouteRecordRaw[] = [];
  
  const modules = import.meta.globEager('./module/*.ts');
  for (const path in modules) {
    routes.push(...modules[path].default);
  }
  
  const router = createRouter({
    history: createWebHashHistory(),
    routes: routes,
  });
  
  export default router;

这段代码的意思是获取当前module文件夹下的所有ts结尾的文件自动导入

const modules = import.meta.globEager('./module/*.ts');
for (const path in modules) {
   routes.push(...modules[path].default);
}

相当于在webpack中的这段代码

// 参数:1. 目录  2. 是否加载子目录  3. 加载的正则匹配
const importFn = require.context('./module/', false, /\.ts$/)
importFn.keys().forEach(key => {
    const rt = importFn(key).default
    routes.push(rt);
})

集成vuex

  1. 安装

npm install vuex@next --save

  1. 创建store/index.ts文件
// store.ts
import { InjectionKey } from 'vue'
import { createStore, useStore as baseUseStore, Store } from 'vuex'

export interface State {
  count: number
}

export const key: InjectionKey<Store<State>> = Symbol()

export const store = createStore<State>({
  state: {
    count: 0
  },
  mutations: {
    setCount(state: State, count: number) {
      state.count = count;
    }
  },
  getters: {
    getCount(state: State) {
      return state.count
    }
  }
})

// 定义自己的 `useStore` 组合式函数
export function useStore() {
  return baseUseStore(key)
}
  1. 在main.ts中导入

import { store, key } from './store/index'

app.use(store, key)

推荐使用pinia,不再使用vuex

使用sass

  1. 安装

npm install -D sass

  1. 配置公共变量

在vite.config.js文件中添加

css:{
     preprocessorOptions: {
      scss: {
        additionalData: `@import "style/_style.scss";`
      }
    }
} 

集成Axios

安装npm install axios

创建request.ts

import axios from 'axios';
// 里面就是定义了一些常量  NO_PERMISSION=401 ;OK_CODE=200
import { NO_PERMISSION, OK_CODE } from '@/app/keys';
import router from '@/router';
// 可以从其他文件导入的
export interface UserInfo {
  id: number;
  username: string;
  role: string;
  email: string;
  token: string;
}

const requests = axios.create({
  baseURL: import.meta.env.VITE_API_URL,
  timeout: 10000,
});

//拦截器
requests.interceptors.request.use((config) => {
  config = config || {};
  //pinia
  try {
    const user = JSON.parse(localStorage.getItem('user') || '') as UserInfo;
    if (user.token) {
      config.headers!['Authorization'] = `Bearer ${user.token}`;
    }
  } catch (e) {}
  return config;
});

requests.interceptors.response.use(
  (resp) => {
    const { code, msg } = resp.data || {};
    if (code !== OK_CODE) {
      return Promise.reject(msg);
    }
    if (code === NO_PERMISSION) {
      router.push({ name: 'Login' }).then();
      return Promise.reject(msg);
    }
    return Promise.resolve(resp);
  },
  (error) => {
    return Promise.reject(error);
  }
);

export default requests;

这里有用到import.meta.env.VITE_API_URL,这是vite提供的用于区分开发和生产环境的

在根目录创建.env.env.production分别表示开发时变量和生产时变量

# .env
VITE_API_URL=http://localhost:3031

# .env.producttion
VITE_API_URL=http://localhost:1212

创建useHttp.ts

import { Method } from 'axios';
import requests from '@/api/requests';

export interface HTTPConfig {
  url: string;
  method: Method;
  data?: { [key: string]: unknown };
  params?: { [key: string]: unknown };
}
const useHttp = <T>(config: HTTPConfig): Promise<T> => {
  return new Promise<T>((resolve, reject) => {
    requests({
      url: config.url,
      method: config.method,
      data: config.data || {},
      params: config.params || {},
    })
      .then((resp) => {
        resolve(resp.data);
      })
      .catch((err) => {
        reject(err);
      });
  });
};

export default useHttp;

编写接口文件

import useHttp from '@/api/useHttp';
import { BasicResp } from '@/api/types';
// BasicResp 是返回数据的外层通用的类型
export interface BasicResp<T> {
  code: number;
  data: T;
  msg: string;
}

export interface RegParams {
  username: string;
  password: string;
  email: string;
}
export const reqUserRegister = (params: RegParams) => {
  //axios http
  return useHttp<BasicResp<null>>({
    url: `/v1/user`,
    method: 'post',
  });
};

export interface LoginParams {
  username: string;
  password: string;
}

export interface LoginData {
  info: UserInfo;
  token: string;
}

export const reqUserLogin = (params: LoginParams) => {
  //axios http
  return useHttp<BasicResp<LoginData>>({
    url: `/v1/user/login`,
    method: 'post',
    data: { ...params },
  });
};
最后编辑于
?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,100评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,308评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事?!?“怎么了?”我有些...
    开封第一讲书人阅读 159,718评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,275评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,376评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,454评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,464评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,248评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,686评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,974评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,150评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,817评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,484评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,140评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,374评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,012评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,041评论 2 351

推荐阅读更多精彩内容