当前位置: 首页 > article >正文

从零搭建uniapp项目

目录

创建uni-app项目

基础架构

安装 uni-ui 组件库

安装sass依赖

easycom配置组件自动导入

配置view等标签高亮声明

配置uni-ui组件类型声明

解决 标签 错误

关于tsconfig.json中提示报错

关于非原生标签错误(看运气)

安装 uview-plus 组件库

​编辑

Pinia 配置

pinia 依赖安装

测试

pinia 持久化配置


创建uni-app项目

npx degit dcloudio/uni-preset-vue#vite-ts new-project-name

new-project-name:这里更改一下为自己的项目名称

基础架构

安装 uni-ui 组件库

官网:uni-app官网

通过官网说明文档相对应的搭建

pnpm i @dcloudio/uni-ui

安装sass依赖

pnpm i sass@1.63.2 -D
pnpm i sass-loader@10.4.1 -D

easycom配置组件自动导入

// pages.json
{// uniapp 配置文件"easycom": {"autoscan": true,// 以正则方式自定义组件匹配规则"custom": {// uni-ui 规则如下配置"^uni-(.*)": "@dcloudio/uni-ui/lib/uni-$1/uni-$1.vue",// 以serive开头的组件,在 components 文件夹中查找引入(需要重启服务器)//"^serive(.*)": "@/components/serive/serive$1.vue"}},// 其他内容pages:[// ...]
}

配置view等标签高亮声明

pnpm i -D miniprogram-api-typings @uni-helper/uni-app-types

安装成功后复制对应的依赖包名放置在 tsconfig.json中的type字段

同时加上 vueCompilerOptions 配置

  "vueCompilerOptions": {"experimentalRuntimeMode": "runtime-uni-app"},

配置uni-ui组件类型声明

组件声明类型依赖官网(注意:如果缓慢需要魔法):@uni-helper/uni-ui-types - npm

pnpm i -D @uni-helper/uni-ui-types

安装成功后复制对应的依赖包名放置在 tsconfig.json中的type字段

解决 标签 错误

关于tsconfig.json中提示报错

选项“importsNotUsedAsValues”已删除。请从配置中删除它。请改用“verbatimModuleSyntax”

主要是我们使用的tsconfig版本太低了

步骤1:

更新@vue/tsconfig版本,在package.json文件中找到@vue/tsconfig,把版本号改为0.7.0

 "@vue/tsconfig": "^0.7.0",

随后运行pnpm 更新依赖

pnpm install

最后更改 tsconfig 地址 

  "extends": "@vue/tsconfig/tsconfig.dom.json",

这样就不报错了

关于非原生标签错误(看运气)

当我们发现正常标签报错

更改我们的 tsconfig.json,参考一下我下面的配置

{"extends": "@vue/tsconfig/tsconfig.dom.json","compilerOptions": {"sourceMap": true,"baseUrl": ".","paths": {"@/*": ["./src/*"]},"lib": ["esnext", "dom"],// 类型声明文件"types": ["@uni-helper/uni-ui-types",   // uni-ui 组件类型"@dcloudio/types", // uni-app API 类型"miniprogram-api-typings", // 原生微信小程序类型"@uni-helper/uni-app-types" // uni-app 组件类型]},// vue 编译器类型,校验标签类型"vueCompilerOptions": {//新增加的配置 experimentalRuntimeMode 已废弃,现调整为 nativeTags,请升级 Volar 插件至最新版本"nativeTags": ["block", "component", "template", "slot"],// experimentalRuntimeMode 已废弃,请升级 Vue - Official 插件至最新版本"plugins": ["@uni-helper/uni-app-types/volar-plugin"]},"include": ["src/**/*.ts", "src/**/*.d.ts", "src/**/*.tsx", "src/**/*.vue"]
}

如果还是爆红,多次尝试重启 Vue - Official 插件,以及重启vscode,我就是这样不明不白的改好了。

安装 uview-plus 组件库

执行命令安装 uview 依赖

pnpm install uview-plus

挂载使用

import uviewPlus from 'uview-plus'app.use(uviewPlus);

由于uview-plus 不是原生组件,所以我们要自定义声明类型,这样就不会爆红波浪线了

uview-plus.d.ts

// src/types/uview-plus.d.ts
declare module 'uview-plus' {import { Plugin } from 'vue';const install: Plugin;export default install;
}

全局样式配置

 /* src\uni.scss */
@import  'uview-plus/theme.scss';
// src\App.vue
<style lang="scss">/* 注意要写在第一行,同时给style标签加入lang="scss"属性 */@import "uview-plus/index.scss";
</style>

自动导入

// pages.json
{"easycom": {"autoscan": true,// 注意一定要放在custom里,否则无效,https://ask.dcloud.net.cn/question/131175"custom": {"^u--(.*)": "@/uni_modules/uview-plus/components/u-$1/u-$1.vue","^up-(.*)": "@/uni_modules/uview-plus/components/u-$1/u-$1.vue","^u-([^-].*)": "@/uni_modules/uview-plus/components/u-$1/u-$1.vue"}},// 此为本身已有的内容"pages": [// ......]
}

Pinia 配置

Pinia官网(注意:如果缓慢需要魔法):Pinia | The intuitive store for Vue.js

pinia 依赖安装

执行命令安装 Pinia 依赖

pnpm install pinia

分包创建目录

index.ts配置pinia

import { createPinia } from "pinia";
const pinia = createPinia();    // 创建 Pinia 实例
// 默认导出,给 main.ts 使用
export default pinia;
// 模块统一导出
export * from "./modules/counter";

main.ts 使用 pinia

import { createSSRApp } from "vue";
import App from "./App.vue";
import pinia from "./stores";
export function createApp() {const app = createSSRApp(App);app.use(pinia); // 挂载piniareturn {app,};
}
测试

counter.ts

import { defineStore } from "pinia";
import { computed, ref } from "vue";
export const useCounterStore = defineStore("counter", () => {// 声明 数据类型const count = ref(0);// 定义 计数器方法const increment = () => {count.value++;}// 声明基于数据派生的计算属性const double = computed(() => {return count.value * 2;})// 返回 计数器数据、方法、计算属性return {count,double,increment,}},
);

Test.vue

<script lang="ts" setup>
import { useCounterStore } from '@/stores';
import { storeToRefs } from 'pinia';
const counter = useCounterStore();const { count } = storeToRefs(counter);
const {double} = storeToRefs(counter)
const {increment} =  counter</script><template><div class="Test"><uni-card><text>计数:{{ count }}</text><br><text>双倍:{{ double }}</text></uni-card><up-button @tap="increment">+</up-button></div>
</template><style lang="scss" scoped></style>

pinia 持久化配置

执行命令安装 pinia-plugin-persistedstate 依赖

pnpm i pinia-plugin-persistedstate

 index.ts配置持久化

import { createPinia } from "pinia";
import piniaPluginPersistedstate from "pinia-plugin-persistedstate";
const pinia = createPinia();    // 创建 Pinia 实例pinia.use(piniaPluginPersistedstate);   // 使用 pinia-plugin-persistedstate 持久化配置// 默认导出,给 main.ts 使用
export default pinia;// 模块统一导出
export * from "./modules/counter";

配置持久化 counter.ts


import { defineStore} from "pinia";
import { computed, ref } from "vue";export const useCounterStore = defineStore("counter", () => {// 声明 数据类型const count = ref(100);// 定义 计数器方法const increment = () => {count.value++;console.log('增加了:',count.value);}// 声明基于数据派生的计算属性const double = computed(() => {return count.value * 2;})// 返回 计数器数据、方法、计算属性return {count,double,increment,}  
},// 配置项 {// 网页端的 持久化配置// persist: true// 移动端的 持久化配置// persist: {//     storage: localStorage,//     key: 'count',// }// 小程序的 持久化配置, 同时兼容大部分浏览器H5persist: {storage: {getItem(key) {return uni.getStorageSync(key);},setItem(key, value) {return uni.setStorageSync(key, value);},}}}
);

报错 X [ERROR] Could not resolve "destr"

解决方法:

降低 持久化依赖的版本

  "pinia-plugin-persistedstate": "^3.2.3",

测试效果,成功持久化存储到本地

请求配置

添加拦截器

// src/utils/http.tsimport { useUserStore } from "@/stores/modules/user"
// 基础的请求地址
const baseUrl = "http://localhost:3000"
// 拦截器配置
const httpInterceptor = {// 拦截前触发invoke(options: UniApp.RequestOptions) {// 1、非 http 开头需拼接地址if (!options.url.startsWith('http')) {options.url = baseUrl + options.url}// 2、请求超时,如果没有配置则默认 60soptions.timeout = options.timeout || 10000// 3、添加小程序端请求头标识options.header = options.header || {...options.header,  // 如果请求设置方法前配置了 header 则保留之前配置过的'source-client': 'uniapp'}// 4、添加 token 请求标识const userStore = useUserStore();// 这里 ?. 表示 可选链操作符,如果 userStore.userInfo 为空,则 token 为 undefinedconst token = userStore.userInfo?.token;if  (token) {options.header['Authorization'] = 'Bearer ' + token;}}
}
// 添加拦截器,这里共用一份拦截器
uni.addInterceptor('request', httpInterceptor)
uni.addInterceptor('uploadFile', httpInterceptor)

添加封装请求函数

// src/utils/http.ts// .....  接着添加// 2.3 指定响应数据结构
interface Data<T>{code: number,msg: string,result: T
}
// 2.2 添加类型,支持泛型
const http = <T>(options: UniApp.RequestOptions) => {// 返回 Promise 对象return new Promise<Data<T>>((resolve, reject) => {uni.request({...options,// 2、请求成功success: (res) => {// 2.1 提取核心数据 res.data// 类型断言指定数据类型resolve(res.data as Data<T>)},fail: (err: UniApp.GeneralCallbackResult) => {reject(err)}})})
}

添加响应拦截

// src/utils/http.ts// .....  接着添加// 2.2 添加类型,支持泛型
const http = <T>(options: UniApp.RequestOptions) => {// 返回 Promise 对象return new Promise<Data<T>>((resolve, reject) => {uni.request({...options,// 2、请求成功success: (res) => {// 2.1 提取核心数据 res.data// 状态码 2xx,axios 就是这样设计的if (res.statusCode >= 200 && res.statusCode < 300) {// 类型断言指定数据类型resolve(res.data as Data<T>)} else if (res.statusCode === 401) {// 401 错误 -> 清理用户信息,跳转登录页const userStore = useUserStore();userStore.clearUserInfo();uni.navigateTo({        // 跳转登录页url: '/pages/login/login'});reject(res);} else {// 其他错误 --> 根据后端错误信息轻提示uni.showToast({title: (res.data as Data<T>).msg || '请求错误',icon: 'none'});reject(res);}},// 响应失败fail: (err: UniApp.GeneralCallbackResult) => {uni.showToast({title:'网络请求错误,请尝试切换网络',icon: 'none'});reject(err)}})})
}

--------------------------------- 持续更新中---------------------------------

http://www.lryc.cn/news/2403716.html

相关文章:

  • 数据库密码加密
  • GaLore:基于梯度低秩投影的大语言模型高效训练方法详解一
  • OpenCV CUDA模块图像处理------图像融合函数blendLinear()
  • Linux服务器如何安装wps?
  • 图片压缩工具 | 图片生成PDF文档
  • Python的浅拷贝与深拷贝
  • VSCode - VSCode 放大与缩小代码
  • 消息队列处理模式:流式与批处理的艺术
  • 11-Oracle 23ai Vector Embbeding和ONNX
  • Build a Large Language Model (From Scratch) 序章
  • 【HarmonyOS 5】教育开发实践详解以及详细代码案例
  • NoSQL 之Redis哨兵
  • 【nano与Vim】常用命令
  • OpenCV 图像色彩空间转换与抠图
  • Amazing晶焱科技:电子系统产品在多次静电放电测试后的退化案例
  • Go 中的 Map 与字符处理指南
  • 互联网大厂Java求职面试:云原生架构下的微服务网关与可观测性设计
  • C++中const关键字详解:不同情况下的使用方式
  • Java 2D 图形类总结与分类
  • C# 快速检测 PDF 是否加密,并验证正确密码
  • 服务器信任质询
  • 华为云Flexus+DeepSeek征文| 华为云Flexus X实例单机部署Dify-LLM应用开发平台全流程指南
  • Python: 操作 Excel折叠
  • IBM官网新闻爬虫代码示例
  • Java持久层技术对比:Hibernate、MyBatis与JPA的选择与应用
  • Spring Boot实现接口时间戳鉴权
  • 视觉SLAM基础补盲
  • STM32外设问题总结
  • Vue-3-前端框架Vue基础入门之VSCode开发环境配置和Tomcat部署Vue项目
  • 动态IP与静态IP:数字世界的“变脸术”与“身份证”