uniapp-网络请求 request 封装+ 拦截器(ts版)

5 篇文章 0 订阅
5 篇文章 0 订阅
/**
 * @author wn
 * @date 2023/07/25 11:32:45
 * @description:uniapp 网络请求
 */

export const baseURL = 'api'

/**
 * 添加拦截器:
 *   拦截 request 请求
 *   拦截 uploadFile 文件上传
 *
 * TODO:
 *   1. 非 http 开头需拼接地址
 *   2. 请求超时
 *   3. 添加小程序端请求头标识
 *   4. 添加 token 请求头标识
 *   5. 成功 失败 完成(调用成功、失败都会执行)
 */
const httpInterceptor = {
	// 拦截前触发
	invoke(options: UniApp.RequestOptions) {
		// 1. 非 http 开头需拼接地址
		if (!options.url.startsWith('http')) {
			options.url = baseURL + options.url
		}
		// 2. 请求超时, 默认 60s
		options.timeout = 5000
		// 3. 添加小程序端请求头标识
		options.header = {
			...options.header,
			'source-client': 'miniapp',
		}
		// 4. 添加 token 请求头标识
		const memberStore =  useMemberStore()
		const token = memberStore.profile?.token
		if (token) {
			options.header.Authorization = token
		}
	},
	success(args: UniApp.RequestSuccessCallbackResult) {
		// 请求成功后,修改code值为1
		// args.data.code = 1
	},
	fail(err: UniApp.GeneralCallbackResult) {
		// console.log('interceptor-fail', err)
	},
	complete(res: UniApp.GeneralCallbackResult) {
		// console.log('interceptor-complete', res)
	},
}

/**
 * 添加拦截器
 * 名字 对应  uni.xxx 请求
 */
uni.addInterceptor('request', httpInterceptor)
uni.addInterceptor('uploadFile', httpInterceptor)

/**
 * 请求返回 基类 封装
 */
interface BaseResponse<T> {
	code: string
	msg: string
	result: T
}
/**
 * uniapp request 请求 封装
 * @param  UniApp.RequestOptions--{ method, url, data }
 * @returns Promise
 *  1. 返回 Promise 对象
 *  2. 获取数据成功
 *    2.1 提取核心数据 res.data
 *    2.2 添加类型,支持泛型
 *  3. 获取数据失败
 *    3.1 401错误  -> 清理用户信息,跳转到登录页
 *    3.2 其他错误 -> 根据后端错误信息轻提示
 *    3.3 网络错误 -> 提示用户换网络
 */
export const request = <T = any>(options: UniApp.RequestOptions) =>
	new Promise<T>((resolve, reject) => {
		uni.request({
			...options,
			// 响应成功
			success(res) {
				// 状态码 2xx, axios 就是这样设计的
				if (res.statusCode >= 200 && res.statusCode < 300) {
					// 2.1 提取核心数据 res.data
					resolve((res.data as BaseResponse<T>).result)
				} else if (res.statusCode === 401) {
					// 401错误  -> 清理用户信息,跳转到登录页
					const memberStore = useMemberStore()
					memberStore.clearProfile()
					uni.navigateTo({ url: '/pages/login/login' })
					reject(res)
				} else {
					// 其他错误 -> 根据后端错误信息轻提示
					uni.showToast({
						icon: 'none',
						title: (res.data as BaseResponse<T>).msg || '请求错误',
					})
					reject(res)
				}
			},
			// 响应失败
			fail(err) {
				uni.showToast({
					icon: 'none',
					title: '网络错误,换个网络试试',
				})
				reject(err)
			},
		})
	})

  • 0
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值