如何搭建 Vue3 + Vite + Tailwind + Typescript + Vue Router4 + Pinia 的6个,简单步骤

 

 https://www.youtube.com/watch?v=0jk3aleuwUM

 

 

 项目地址:https://gitee.com/ChinaGoogles/Vue3ViteTailwindTypescript

 

 

 

 

 

 

 

 

Vite

Vue3

Vue Router4

Typescript

Pinia

Tailwind CSS

Fonts & Icons - (As needed)

 

 

第一步

npm create vite@latest

 

vue3 和 Typescript  在设置vite时候会被选择为选项

 

1.1   项目名字为viteVueSetup2023

 

 

 选择Vue

 

 

 选择TypeScript

 

 

 执行这3个

 

 

 进入ViteVueSetup2023项目之后,我们看看包有什么

 

 

 

 

 

二. 安装Vue Router

npm install vue-router@4

 

 

 

 

 

三.  安装Pinia

npm install pinia

 

 

 

 

 

四. 安装Tailwind CSS

npm install -D tailwindcss postcss autoprefixer

-D:带有此标志的包将作为开发依赖项安装。这意味着,它们只在开发环境中使用,而不会在最终的生产应用程序包中发布

实用工具优先的CSS框架构建自定义Ul

 

 

 

 

 

五. 初始化包

  • 安装所有节点包
  • 初始化Tailwind CSS
  1. 设置PostCSS模块导出
  2. 安装TailwindCSS智能感知
  3. 设置漂亮的插件(如果需要)
  •  设置Vite / Typescript别名(方便文件夹路径)(如果需要的话)

 

 

1.1  执行

npm install 

 

1.2  再执行,生成一个tailwind.config.cjs文件

npx tailwindcss init

 

 

1.3   新建一个postcss.config.cjs文件

 

 输入

module.exports={
    plugins:{
        tailwindcss:{},
        autoprefixer:{},
    }
}

 

 

1.4  安装插件 tailwind css

 

 安装插件Prettier - Code formatter

 

 

1.5  npm 安装prettier

npm install -D prettier prettier-plugin-tailwindcss

 

 

 

1.6 新建一个.prettierrc.json

 

 

1.7  在tsconfig.json文件中添加baseUrl和paths

{
  "compilerOptions": {
    "baseUrl": "./src/",
    "target": "ESNext",
    "useDefineForClassFields": true,
    "module": "ESNext",
    "moduleResolution": "Node",
    "strict": true,
    "jsx": "preserve",
    "resolveJsonModule": true,
    "isolatedModules": true,
    "esModuleInterop": true,
    "lib": ["ESNext", "DOM"],
    "skipLibCheck": true,
    "noEmit": true,
    "paths": {
      //Direct
      //Sub Locations
      "@/*":["*"],
      "@components/*":["components/*"]
    }
  },
  "include": ["src/**/*.ts", "src/**/*.d.ts", "src/**/*.tsx", "src/**/*.vue"],
  "references": [{ "path": "./tsconfig.node.json" }]
}

 

1.8  设置vite.config.ts 文件

先执行

npm install -D @types/node

import { defineConfig } from 'vite'
//Install node types before calling below import
import {fileURLToPath} from "url";
import path from 'path';
import vue from '@vitejs/plugin-vue'

// https://vitejs.dev/config/
export default defineConfig({
  plugins: [vue()],
  base:"./",
  resolve:{
    alias:{
      //Two methods available
      //Method 1 : using fireURLtoPath(keep adding other paths as needed)
      "@":fileURLToPath(new URL("./src",import.meta.url)),

      //Method 2 : using path
      "@components":path.resolve(__dirname,"src/components"),
    }
  }
})

 

 

1.9  执行   npm  run dev

 

 

 

 

 

 

 

六 .  配置使用tailwind

1.1  原因

在这一点上,我们只是安装tailwind。

步骤1:必须指示加载文件格式(应该访问顺风的地方)

步骤2:CSS应该包括tailwindimports。   

我们运行 npm run dev的时候发生警告

警告 - Tailwind CSS 配置中的“内容”选项缺失或为空。
警告 - 配置内容源,否则生成的 CSS 将缺少样式。
警告 - https://tailwindcss.com/docs/content-configuration

 

1.2   打开  tailwind.config.cjs文件

/** @type {import('tailwindcss').Config} */
module.exports = {
  //Specify the file formats where tailwind shoudl work
  content: ["./src/**/*.{html,js,jsx,tsx,vue}"],
  theme: {
    extend: {},
  },
  plugins: [],
}

 

打开 style.css

@tailwind base;
@tailwind components;
@tailwind utilities;

:root {
  font-family: Inter, system-ui, Avenir, Helvetica, Arial, sans-serif;
  line-height: 1.5;
  font-weight: 400;
}

 

 

1.3  在tsconfig中包含vite配置

{
  "compilerOptions": {
    "baseUrl": "./src/",
    "target": "ESNext",
    "useDefineForClassFields": true,
    "module": "ESNext",
    "moduleResolution": "Node",
    "strict": true,
    "jsx": "preserve",
    "resolveJsonModule": true,
    "isolatedModules": true,
    "esModuleInterop": true,
    "lib": ["ESNext", "DOM"],
    "skipLibCheck": true,
    "noEmit": true,
    "paths": {
      //Direct
      //Sub Locations
      "@/*":["*"],
      "@components/*":["components/*"]
    }
  },
  "include": [
    "src/**/*.ts", 
    "src/**/*.d.ts", 
    "src/**/*.tsx", 
    "src/**/*.vue",
    "vite.config.*"
  ],
  "references": [{ "path": "./tsconfig.node.json" }]
}

 

 

 

我们已经准备好开始在vue 3中使用tailwind和typescript

 

 

 

 

 

七,使用 tailwind

1.1 添加类

 

 

 

 

1.2 在style.css文件使用

@tailwind base;
@tailwind components;
@tailwind utilities;

:root {
  font-family: Inter, system-ui, Avenir, Helvetica, Arial, sans-serif;
  line-height: 1.5;
  font-weight: 400;
}

body{
  @apply bg-green-100 p-5 m-0 w-full h-screen;
}

 

import 可以修改

 

 改成

import HelloWorld from '@components/HelloWorld.vue'

 

 

WORKS    Typescript 

WORKS   Tailwind    ----------------完

 

 

 
 
 
 
 
Setup Vue Router & Store
设置vue路由和存储
 
 
 
1.1  在设置路由器之前,安装一些有用的vue扩展(用于vscode)(如果需要的话),
安装2个插件

TypeScript Vue Plugin (Volar)

Vue VSCode Snippets

 
 
1.2  对于路由器,创建一些示例页面来检查它是否工作。
新建一个@Pages文件夹,里面有Dashboard.vue, DemoPage.vue, NotFoundPage.vue
Pages,P是大写

 

 Dashboard.vue

<template>
    <div>
This is dashboard
    </div>
</template>

<script setup lang="ts">

</script>

<style scoped>

</style>

 

NotFoundPage.vue

<template>
    <div>
Page not found
    </div>
</template>

<script setup lang="ts">

</script>

<style scoped>

</style>

 

DemoPage.vue

<script setup lang="ts">
import HelloWorld from '@components/HelloWorld.vue'
</script>

<template>
  <div class="bg-pink-50">
    <a href="https://vitejs.dev" target="_blank">
      <img src="/vite.svg" class="logo" alt="Vite logo" />
    </a>
    <a href="https://vuejs.org/" target="_blank">
      <img src="./assets/vue.svg" class="logo vue" alt="Vue logo" />
    </a>
  </div>
  <HelloWorld msg="Vite + Vue" />
</template>

<style scoped>
.logo {
  height: 6em;
  padding: 1.5em;
  will-change: filter;
  transition: filter 300ms;
}
.logo:hover {
  filter: drop-shadow(0 0 2em #646cffaa);
}
.logo.vue:hover {
  filter: drop-shadow(0 0 2em #42b883aa);
}
</style>

 

App.vue换成

<template>
  <div>
This is Langding App components
  </div>
</template>

<script setup lang="ts">

</script>

<style scoped>

</style>

 

 

1.3 在src下面新建一个文件夹 routes, 再建一个router.ts文件

 

 

vite.config.ts 文件添加     "@pages":path.resolve(__dirname,"src/@Pages"),

 

 

tsconfig.json  添加     "@pages/*":["@Pages/*"],

 

 

 

routers文件夹下的 reouter.ts ,---------------------》history:vR.createMemoryHistory()这个模式最好看文档,注意一下,看不同需求,目前我用的是history:vR.createWebHistory()

import * as vR from 'vue-router';
import DashboardVue from '@pages/Dashboard.vue';
import Pnf from "@pages/NotFoundPage.vue"


//Create a router and then export it

const _routes:Array<vR.RouteRecordRaw> = [
    //Path  6 Component are mandatory
    {
        path:"/",
        component:DashboardVue,
        name:"home"
    },
    {
        path:"/demo",
        component:()=>import("@pages/DemoPage.vue"),//Lazy Loading to defer loading only when required
        name:"home.demo"  //Use defined kepp dot notation to have  elegant design
    },
    {
        path:"/nf", //need to change with regex to catch all non-defined
        component:Pnf,
        name:"home.missing"
    },
];


const router = vR.createRouter({
    history:vR.createMemoryHistory(),//webHash history will  add hash before every route
    routes:_routes, //Send the routes here.
})


export default router;

 

 
1.4   要让路由器工作,请设置 routerlink。在任何需要的地方查看路由器。
在App.vue 文件    加上<RouterView />

 

 

1.5   我们仅仅创建了路由器。需要在main.ts中加载
 
修改main.ts文件
import { createApp } from 'vue'
import './style.css'
import App from './App.vue'
import router from './routers/router'

createApp(App)
.use(router)
.mount('#app')

 

 
1.6    在APP.vue中,修改成
<template>
  <div >
This is Langding App components
    <ul>
      <RouterLink to="/" >
        <li>Home</li>
      </RouterLink>
      <RouterLink to="/demo" >
        <li>Demo</li>
      </RouterLink>
      <RouterLink to="/auth" >
        <li>Auth</li>
      </RouterLink>
    </ul>
  </div>
  <RouterView />
</template>

<script setup lang="ts">

</script>

<style scoped>

</style>

 

在@Pages文件夹添加多一个文件,AuthPage.vue

<template>
    <div>
        Authorize me
    </div>
</template>

<script setup lang="ts">

</script>

<style scoped>

</style>

 

router.ts文件

{
        path:"/auth",
        component:()=>import("@pages/AuthPage.vue"),//Lazy Loading to defer loading only when required
        name:"home.auth"  //Use defined kepp dot notation to have  elegant design
    },

 

 

最后切换

 

 

 

 

Router ----------WORKS   完

 

 

 

 

 

 

 

 

 

 

 

 

 

Pinia---- start

 使用pinia可以设置多个stores

1.1  src下新建@Store文件夹和authStore.ts文件

 

 

 authStore.ts文件内容

import * as pin from 'pinia';
import {ref} from "vue";

const useAuthStore =pin.defineStore('auth',()=>{
    const is_authorized = ref<Boolean>(false);
    const username=ref<string>();
    function authenticate(status:boolean,user_name?:string){
        is_authorized.value = status;
        username.value=user_name;
        console.log(is_authorized,user_name);
        
    }

    return{
        //State
        is_authorized,
        username,
        //Actions
        authenticate,
    };
})


export default{
    useAuthStore
}

 

 

1.2  执行命令

npm install -D unplugin-auto-import

 

 

 

1.3  到vite.config.ts文件,引入使用

import { defineConfig } from 'vite'
//Install node types before calling below import
import {fileURLToPath} from "url";
import path from 'path';
import vue from '@vitejs/plugin-vue'

import AutoImport from "unplugin-auto-import/vite";


// https://vitejs.dev/config/
export default defineConfig({
  plugins: [
    vue(),
    AutoImport({
      //Targets(file extensions)
      include:[
        /\.[tj]sx?$/,
        //Regex which says t or j followed by sx. tsx jsx ,? next to x says it can happen zero or one time js ,ts
        /\.vue$/,
        // $ at the end says it ends with vue
      ],
      //globals (libraries)
      imports:["vue","vue-router","pinia"],
      //other settings/files/dirs to import
      dts:true,//Autoimport all the files that ends with d.t 
      //Autoimport inside vue template
      vueTemplate:true,
      eslintrc:{enabled:true}
    })
  ],
  base:"./",
  resolve:{
    alias:{
      //Two methods available
      //Method 1 : using fireURLtoPath(keep adding other paths as needed)
      "@":fileURLToPath(new URL("./src",import.meta.url)),

      //Method 2 : using path
      "@components":path.resolve(__dirname,"src/components"),
      "@pages":path.resolve(__dirname,"src/@Pages"),
    }
  }
})

 

 

 

 

 

 

 

1.4 转到AuthPage.vue页面

<template>
    <div class="bg-green-200">
        <input @click="authorize" type="button" class="bg-blue-300 p-1 m-1 cursor-pointer hover:bg-blue-500 hover:text-white" value="Authorize me">
        <span :class="auth.is_authorized? 'text-green-500':'text-red-500'" class="text-red-500">{{ auth.is_authorized }}</span>
    </div>
</template>

<script setup lang="ts">
//we can import ref, router ,route and other itens from vue , vue-router
//pinia each time. Or use auto plugin to do it once.
//Now , we don't need to import ref keyword

import authStore from '@/@Stores/authStore';
const auth =authStore.useAuthStore();




// const isAuthorized = ref<boolean>(false);
//this state remains only in this component.So store it in pini store to share 
//with other cpmps

function getRandomString():string{
    return (Math.random().toString(36).substring(2,10));
}



function authorize(){
    console.log(getRandomString());
    let uname = getRandomString();
    auth.authenticate(!auth.is_authorized,uname);//either give direct value or invert it .

}

</script>

<style scoped>

</style>

 

 

 

 

 

 

1.5 自动导入文件生成的所有必需的输入。在tsconfig中包含这个auto importfile

打开 tsconfig.json 文件,在“include”,加入“auto-imports.d.ts”

 

 

 

 

 

 

 

 1.6  像路由器一样,它也需要导入main.ts
import { createApp } from 'vue'
import { createPinia } from 'pinia'
import './style.css'
import App from './App.vue'
import router from './routers/router'

const _pinia = createPinia();

createApp(App)
.use(_pinia)
.use(router)
.mount('#app')

 

 
 
1.7  转到Dashboard.vue文件
<template>
    <div class="bg-green-200">
        <input @click="authorize" type="button" class="bg-blue-300 p-1 m-1 cursor-pointer hover:bg-blue-500 hover:text-white" value="Authorize me">
        <span :class="auth.is_authorized? 'text-green-500':'text-red-500'" class="text-red-500">{{ auth.is_authorized }}</span>
    </div>
</template>

<script setup lang="ts">
//we can import ref, router ,route and other itens from vue , vue-router
//pinia each time. Or use auto plugin to do it once.
//Now , we don't need to import ref keyword

import authStore from '@/@Stores/authStore';
const auth =authStore.useAuthStore();




// const isAuthorized = ref<boolean>(false);
//this state remains only in this component.So store it in pini store to share 
//with other cpmps

function getRandomString():string{
    return (Math.random().toString(36).substring(2,10));
}



function authorize(){
    console.log(getRandomString());
    let uname = getRandomString();
    auth.authenticate(!auth.is_authorized,uname);//either give direct value or invert it .

}

</script>

<style scoped>

</style>

 

 
 
 
最后我们看看效果
 
 
刚打开
1. 

 

 

2. 

 

 

 3. 

 

 

 
 4.

 

 

 另外一个页面多了这个
 
 
 
 
 
 
-------完-------
 
 
 
 
 
 
 
 
 
 
 
 
 
posted @ 2023-03-24 01:14  漫漫长路</>  阅读(1241)  评论(0编辑  收藏  举报