完美搭建一个vue3+ts项目(一篇文章搞定你的所有疑惑)

news2024/12/28 2:56:56

目录

一、创建vite项目 

二、启动vite项目 

三、处理一些配置问题

四、增加工程化插件

1、安装sass

2、安装vue-router 

3、安装pinia

4、安装element-plus

5、安装axios

6、设置路径别名,将相对路径改为绝对路径


一、创建vite项目 

1、在一个文件夹下通过cmd打开,输入命令

 npm create vite@latest

 

2、接着选择ts

 3、创建好之后,结构目录如下:

二、启动vite项目 

 1、启动项目前需要先 npm i ,从上图可以发现 vite 不会自动帮助我们install相关的依赖,需要我们手动去安装

npm i

 2、此时再通过npm run dev就启动好了

 

 三、处理一些配置问题

1、当我们打开一个引入vue的页面,会发现如下报错,如何解决?

 翻译过来是找不到 vue模块 需要将 moduleResolution设置成node

 解决方式:

找到 tsconfig.json 文件, 将 moduleResolution设置成 node,报红就没了

2、在main.ts中可能也会报如下错,虽然不影响项目运行,但看着难受

 解决方式:

找到 vite-env.d.ts  文件,在里面追加如下代码,保存即可:

declare module "*.vue" {
  import type { DefineComponent } from "vue";
  const vueComponent: DefineComponent<{}, {}, any>;
  export default vueComponent;
}

3、在写ts语法的时候,一些变量名未使用也会报错,想消除怎么办?

解决方式: 

在 tsconfig.json 中将noUnusedLocals改为false,不去校验未使用的变量。

 这样就不会报错了

 四、增加工程化插件

1、安装sass

这是因为 sass 模块是使用 scss 文件必备的工具,所以我们需要安装 sass,

注意:增加 -D 参数,在开发环境中才使用 sass,因为生产环境,scss 文件会被编译成 css 文件,这个插件是用不到的

npm install sass -D

此时我们就可以创建一个common.scss文件,然后在main.ts中引入即可

 2、安装路由 

npm i vue-router

 成功之后,我们来使用一下:

(1)在src下创建pages文件夹,存放路由页面

在pages下面创建一个login页面,点击登录进入index页面

<template>
      <div @click="tologin">登录</div>
</template>
<script lang="ts" setup>

import { useRoute, useRouter } from 'vue-router'

const router = useRouter()
function tologin() {
      router.push({
            name: 'index'
      })
}
</script>

 (2)写好页面之后,我们来创建路由,在 src 路径下新建 router/index.ts 

index.ts

import { createRouter, createWebHistory, } from 'vue-router'

export const routes = [
      {
            path: '/',
            redirect: '/login',
      },
      {
            name: 'login',
            path: '/login',
            component: () => import("../pages/login.vue")
      },
      {
            name: 'index',
            path: '/index',
            component: () => import("../pages/index.vue")
      }

]

const router = createRouter({
      scrollBehavior: () => ({ left: 0, top: 0 }),
      history: createWebHistory(),
      routes,
})


router.beforeEach((to, from, next) => {
      next()
})


export default router

(3)在main.ts中引入

 (4)在 app.vue 设置 routerview

将app.vue中不必要的东西删除,template只写一个routerview,就可以看到登录页了

 

 

3、安装pinia

本文大概讲一下,不详细介绍了,详情看文章 

vue3【抛弃vuex,使用pinia】_vue3替代vuex_suoh's Blog的博客-CSDN博客 

(1)安装pinia 

npm i pinia

(2)创建文件,在 src 路径下增加 store/index.ts

import { defineStore, acceptHMRUpdate } from "pinia";

export const useStore = defineStore({
      id: "index",
      state: () => ({
            name: "old name",
      }),
      getters: {
            myName: (state) => {
                  return `getters ${state.name}`
            }
      },
      actions: {
            changeName(name: string) {
                  this.name = name
            }
      },
});

if (import.meta.hot) {
      import.meta.hot.accept(acceptHMRUpdate(useStore, import.meta.hot))
}

 注意:这里有个import.meta.hot判断作用如下:

Pinia 是 vuex 新替代方案。Pinia 中热更新实现,借助 import.meta

热更新:(Hot Module Replacement,简写 HMR)代码会自动更新页面。当修改代码时,HMR 能够在不刷新页面的情况下,把页面中发生变化的模块,替换成新的模块,同时不影响其他模块的正常运作。

Pinia 支持热更新,所以你可以编辑你的 store,并直接在你的应用中与它们互动,而不需要重新加载页面,允许你保持当前的state、并添加甚至删除stateactionsgetters

目前,只有Vite被官方支持,不过任何实现import.meta.hot规范的构建工具都应该能正常工作。你只需要在任何 store 声明旁边添加这段代码。比方说,你有三个 store:auth.jscart.jschat.js,你必须在每个 store 声明后都添加(和调整)这段代码即可。即:

if (import.meta.hot) {
      import.meta.hot.accept(acceptHMRUpdate(useStore, import.meta.hot))
}

(3)在main.ts中引入

 (4)在页面中使用

 (5)查看效果,点击修改名称

 4、安装element-plus

npm install element-plus

 (1)安装完成之后,在main.ts中引入

import { createPinia } from 'pinia'
import ElementPlus from 'element-plus'
app.use(ElementPlus)

(2)使用

引入一个树形控件

<template>
      <div @click="tologin">登录</div>
      <h2> {{ store.name }}</h2>
      <h1 @click="changeName">修改名称</h1>
      <el-tree :data="data" :props="defaultProps" @node-click="handleNodeClick" />
</template>
<script lang="ts" setup>
import { useRoute, useRouter } from 'vue-router'
import { useStore } from "../store/index";
const router = useRouter()
const store = useStore()
function tologin() {
      router.push({
            name: 'index'
      })
}
function changeName(): void {
      store.name = '索索'
}
interface Tree {
      label: string
      children?: Tree[]
}

const handleNodeClick = (data: Tree) => {
      console.log(data)
}

const data: Tree[] = [
      {
            label: 'Level one 1',
            children: [
                  {
                        label: 'Level two 1-1',
                        children: [
                              {
                                    label: 'Level three 1-1-1',
                              },
                        ],
                  },
            ],
      },
      {
            label: 'Level one 2',
            children: [
                  {
                        label: 'Level two 2-1',
                        children: [
                              {
                                    label: 'Level three 2-1-1',
                              },
                        ],
                  },
                  {
                        label: 'Level two 2-2',
                        children: [
                              {
                                    label: 'Level three 2-2-1',
                              },
                        ],
                  },
            ],
      },
]

const defaultProps = {
      children: 'children',
      label: 'label',
}
</script>

(3)看效果

 补充:对于下图这个ts的interface不太懂的小盆友,看这里  https://www.cnblogs.com/xuelanying/p/14411850.html  

5、安装axios

npm install axios --save

(1) 安装好之后,在src下创建utils/request.ts封装axios请求

request.ts

/**axios封装
 * 请求拦截、相应拦截、错误统一处理
 */
import axios from 'axios';
import router from '../router/index'
//  let protocol = window.location.protocol; //协议
//  let host = window.location.host; //主机
//  axios.defaults.baseURL = protocol + "//" + host;
axios.defaults.baseURL = '/api'

axios.interceptors.request.use( //响应拦截
    async config => {
        // 每次发送请求之前判断vuex中是否存在token        
        // 如果存在,则统一在http请求的header都加上token,这样后台根据token判断你的登录情况
        // 即使本地存在token,也有可能token是过期的,所以在响应拦截器中要对返回状态进行判断 
        config.headers.token = sessionStorage.getItem('token')
        return config;
    },
    error => {
        return Promise.reject(error);
    })
// 响应拦截器
axios.interceptors.response.use(
    response => {
        if (response.status === 200) {
            return Promise.resolve(response); //进行中        
        } else {
            return Promise.reject(response); //失败       
        }
    },
    // 服务器状态码不是200的情况    
    error => {
        if (error.response.status) {
            switch (error.response.status) {
                // 401: 未登录                
                // 未登录则跳转登录页面,并携带当前页面的路径                
                // 在登录成功后返回当前页面,这一步需要在登录页操作。                
                case 401:
                    // 自定义过期之后的操作
                    break
                // 403 token过期                
                // 登录过期对用户进行提示                
                // 清除本地token和清空vuex中token对象                
                // 跳转登录页面                
                case 403:
                    sessionStorage.clear()
                    break
                // 404请求不存在                
                case 404:
                    break;
                // 其他错误,直接抛出错误提示                
                default:
            }
            return Promise.reject(error.response);
        }
    }
);
/** 
 * get方法,对应get请求 
 * @param {String} url [请求的url地址] 
 * @param {Object} params [请求时携带的参数] 
 */
const $get = (url: string, params: object) => {
    return new Promise((resolve, reject) => {
        axios.get(url, {
            params: params,
        })
            .then(res => {
                resolve(res.data);
            })
            .catch(err => {
                reject(err.data)
            })
    });
}
/** 
 * post方法,对应post请求 
 * @param {String} url [请求的url地址] 
 * @param {Object} params [请求时携带的参数] 
 */
const $post = (url: string, params: object) => {
    return new Promise((resolve, reject) => {
        axios.post(url, params) //是将对象 序列化成URL的形式,以&进行拼接   
            .then(res => {
                resolve(res.data);
            })
            .catch(err => {
                reject(err.data)
            })
    });
}
// 下面是将get和post方法挂载到vue原型上供全局使用、
// vue2.x中是通 Vue.prototype 来绑定的,像这样Vue.prototype.$toast = Toast。在vue3中取消了Vue.prototype,推荐使用globalProperties来绑定,
export default {
    install: (app: any) => {
        app.config.globalProperties['$get'] = $get;
        app.config.globalProperties['$post'] = $post;
        app.config.globalProperties['$axios'] = axios;
    }
}

(2)在main.ts中引入即可

import http from './utils/request.js';
app.use(http)

(3)在页面中使用请求,调登录接口

import { getCurrentInstance } from 'vue'
const { proxy } = getCurrentInstance() as any;

function login(): void {
      let data = {
            roleId: "A",
            username: "dpc",
            password: "dpc12345",
            sysType: "zhfw",
      }
      proxy.$post("/index/login", data).then((response: any) => {
            console.log(response)
            router.push({
                  name: 'index'
            })
      })
}
function tologin() {
      login()
}

6、设置路径别名,将相对路径改为绝对路径

@types/node 模块 在我们使用 node 方法(比如 path.resolve)时提供 ts 类型声明,否则编辑器会报错,虽然不影响代码运行,但是会有红线很影响心情,如下图所示。

(1)安装 @types/node

npm i @types/node -D

(2)在 tsconfig.json 中设置 path

    "paths": {
      "@/*": ["./src/*"]
    }

 (3)在vite.config.ts中添加如下配置

import { resolve } from 'path'

export default defineConfig({
  plugins: [vue()],
  resolve: {
    alias: {
      '@': resolve(__dirname, './src')
    },
    //extensions: [".ts", ".js", ".vue", ".json", ".mjs"],
    extensions: [".mjs", ".js", ".ts", ".jsx", ".tsx", ".json", ".vue"],
  },
})

 (4)此时就不会报红了

 

 到这里,一个基本的vue3项目所需要的技术栈插件都配置好了,期待你的一键三连哦~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/650505.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Vision Pro:为什么空间音频是AR的绝杀武器?

Apple Vision Pro&#xff0c;不仅仅是苹果全新的重磅品类&#xff0c;而且在它身上也融合了苹果过去几乎所有新技术&#xff0c;比如空间音频就是其中一个例子。 苹果表示&#xff0c;Vision Pro中空间音频可以很好的应用在&#xff1a;影视节目、游戏内容、3D空间照片、3D空…

金融风控项目实战-银行信用卡流失预测模型_基于ANN神经网络_金融培训_论文科研_毕业设计

业务背景 根据央行公布的数据显示&#xff0c;全国性银行信用卡和借贷合一卡的发卡量增速从2017年同比增速26.35%的高点逐年下降&#xff0c;截至2020年同比增速降至4.26%。银行信用卡发卡增速明显放缓的背景下&#xff0c;预防老客户流失的问题变得愈发重要。 假设一家消费信…

pytest+allure

知识点1&#xff1a; 1、测试结果信息阅读 passed表示通过&#xff0c;有个简写. failed表示失败&#xff0c;有个简写F 2、命令行参数 -h&#xff1a;帮助 -version&#xff1a;版本信息 3、测试用例命名规则&#xff1a; 测试函数必须以test开头 测试类必须以Test开头…

我为开放原子全球开源峰会助力:共建开源之梦

我为开放原子全球开源峰会助力&#xff1a;共建开源之梦 6月11日&#xff0c;以“开源赋能&#xff0c;普惠未来”为主题的2023开放原子全球开源峰会开幕式暨高峰论坛在北京成功举办。 开源的力量与魅力 开源是当今软件行业中不可忽视的力量&#xff0c;它为技术的快速发展和…

入职滴滴和字节的2 年里,我感觉忒真实了……

引言 先简单交代一下背景吧&#xff0c;某不知名985的本硕&#xff0c;17年毕业加入滴滴&#xff0c;之后跳槽到了头条&#xff0c;一直从事软件测试相关的工作。之前没有实习经历&#xff0c;算是两年半的工作经验吧。 这两年半之间完成了一次晋升&#xff0c;换了一家公司&…

Ubuntu如何安装vmtools?

虚拟机-安装VMware Tools 然后在Ubuntu中找到设备中的VMware Tools&#xff0c;将这个文件来复制到桌面上去。 选择提取到此处 可以看到桌面上多了一个VMware Tools的文件夹 使用cd命令进入桌面上的这个VMware tools的文件夹 使用sudo ./安装命令 对vmware-tools-distrib文件夹…

政企市场,「观望」AI大模型

数据安全等“刚需”下&#xff0c;私有化成为政企市场的准入门槛&#xff0c;然而私有化下&#xff0c;行业模型局限性尚未可知&#xff0c;加之信创化等因素&#xff0c;厂商仍需取长补短&#xff0c;为政企客户提供全方位的解决方案。 作者|斗斗 编辑|皮爷 出品|产业家 政…

采用UWB定位技术开发的室内定位系统源码

UWB精准定位系统源码 UWB是什么&#xff1f; UWB(Ultra Wideband)超宽带技术是一种全新的、与传统通信技术有极大差异的通信新技术。它不需要使用传统通信体制中的载波&#xff0c;而是通过发送和接收具有纳秒或纳秒级以下的极窄脉冲来传输数据&#xff0c;实现精准定位。 技术…

一名高水平的程序员,为什么面试总是失败?

目录 场景1 场景2 场景3 1 从发现和分析问题方面准备。 2 从linux命令和日志层面分析问题的步骤。 1 事先准备很重要 2 面试开始是自我介绍&#xff0c;这时要表露出&#xff0c;自己解决过redis,dubbo等组件的线上问题 3 一旦开始说第一个组件层面解决过的问题时&…

Django高级扩展之文件上传

文件上传是一个比较常用的网站功能&#xff0c;在服务器端&#xff0c;Django会使用一个叫作request.FILES的对象来处理上传的文件。 目录 存储路径 创建存储目录 配置settings.py 上传单文件 配置url 上传文件模板 视图方法 显示上传页面 上传文件处理 上传效果 1.…

盘点五种最常见加密算法!

大家好&#xff0c;我是Martin。 今天&#xff0c;就给大家来盘点一下最常见的5种加密算法。 大家平时的工作中&#xff0c;可能也在很多地方用到了加密、解密&#xff0c;比如&#xff1a; 用户的密码不能明文存储&#xff0c;要存储加密后的密文 用户的银行卡号、身份证号…

2024年天津农学院专升本专业课参考书目

天津农学院2024年高职升本科拟招生专业参考书目 天津农学院专升本专业课参考教材&#xff08;我校不提供专业课参考教材&#xff09; 人力资源管理专业参考教材&#xff1a; 1、《人力资源管理实用教程》 (第2版)&#xff0c;吴宝华&#xff0c;北京大学出版社 2、《人力资源…

运维(SRE)成长之路-第3天 文本处理三剑客之 grep

1.命令作用以及参数详解 grep: 全局搜索正则表达式并打印行(Global search REgular expression and Print out the line)作用&#xff1a;文本搜索工具&#xff0c;根据用户指定的“模式”对目标文本逐行进行匹配检查&#xff1b;打印匹配到的行模式&#xff1a;由正则表达式字…

GitHub Copilot 最全安装、使用

GitHub Copilot 最全安装、使用教程 一、温馨提示 GitHub Copilot 目前为止可以免费试用一个月&#xff0c;但是试用的前提是必须要绑定银行卡&#xff0c;因为后续会自动扣费&#xff0c;所以请注意试用结束日期&#xff0c;自己定好闹钟关闭订阅。 订阅价格为每月10美刀&a…

【Pinia 状态管理篇】Vite + Vue3 组合式 API 模式下的 Store 数据处理

文章目录 Pinia 状态管理一、 Pinia 安装与使用1.1 安装1.2 注册 pinia 实例到全局1.3 创建一个 Store1.4 组件内使用 Store 二、Pinia 核心概念展开学习Store 的定义和使用2.1 State2.2 Getter2.3 Action 附&#xff1a;1. 什么是魔法字符串&#xff1f; Pinia 状态管理 一、 …

Nacos配置中心使用(Spring Cloud版)

目标 向项目中集成Nacos配置。原项目是一个SpringBoot项目。这里假设我们无法修改原有项目的SpringBoot版本。 注意 在不动SpringBoot版本的前提下&#xff0c;根据SpringBoot的版本&#xff0c;确定Spring Cloud和Nacos版本。Nacos版本其实就是Spring Cloud Alibaba版本。在…

【Linux】网络结构模式

目录 网络结构模式C/S结构B/S结构 MAC地址IP地址端口网络模型OSI七层模型TCP/IP四层模型 通信过程数据包封装协议以太网协议ARP协议IP数据报格式UDP协议格式TCP协议格式封装分用TCP详解TCP和UDPTCP通信流程TCP三次握手 网络结构模式 C/S结构 客户机-服务器(client-server)结构…

uboot读取adc,通过cmdline传给kernel解析数值,不同硬件模块进行处理

uboot读取adc&#xff0c;通过cmdline传给kernel解析数值&#xff0c;不同硬件模块进行处理 uboot开发&#xff0c;要想读取adc在哪个时间点读取以及如何传给cmdline&#xff0c;需要清楚2点&#xff0c; uboot启动得大致流程及相关启动功能&#xff0c;uboot各个硬件模块得接…

主机安全各个方面应该怎么做?

主机安全是指保护主机系统不受恶意攻击和未经授权访问的威胁&#xff0c;以保证系统的稳定性、可用性和数据的安全性。 以下是主机安全涉及的几个方面做法&#xff1a; 1. 操作系统安全&#xff1a;安装操作系统时需要选择合适的版本&#xff0c;及时打补丁更新&#xff0c;配置…

入职Linux驱动工程师后,我才知道的真相…

大家好&#xff0c;我是ST。 做Linux驱动工程师也有一段时间了&#xff0c;今天分享一下我曾经入职才知道的一些事情&#xff0c;算是一个菜鸟的经历吧&#xff01; 设备树 起初学习Linux驱动&#xff0c;是从最简单的一个.c文件开始。 在.c中实现module_init和module_exit这…