uni-app小程序登录后…
前情
最近新接了一个全新项目,是类似商城的小程序项目,我负责从0开始搭建小程序,我选用的技术栈是uni-app技术栈,其中就有一个用户登录功能,小程序部分页面是需要登录才可以查看的,对于未登录的用户需要引导用户去登录页面,再back回来重新渲染当前页面,以让用户正常使用
思考
问题也不复杂,就是判断登录状态而已,需要登录的页面没登录就引导去登录再回来,回来后再重新渲染页面数据即可
这里有二个动作:一个是判断登录态,一个是重新渲染页面数据,有动作就有触发时机,对于判断登录态,我们是在跳转前判断,还是在跳转进需要登录的页面再判断,对于重新渲染数据当然是进入页面的时候再重新渲染,但是怎么去实现重新渲染了,对于小程序我们第一时间想到的是通过生命钩子来做,那当然就是onShow了
解决方案
基于上述的思考,我想到如下二种解决方案:

应该能解决的方案应该有很多,这只是我在实现这个需求的时候想到的二种方案
场景方案1:跳转进需要登录的页面再判断是否是登录态,同时登录回来后通过onShow生命钩子重新渲染页面
此方案优点:就是判断登录态你不需要特定代码去判断,在服务端接口这一块做下处理即可,如果返回状态码是401或者是你和服务端沟通好的错误码时再引导去登录页,这样全局做请求拦截就行,项目中我就是有做这一块的处理,使用的是我封装好的工具方法:常用工具方法 - DCloud 插件市场已分享到插件市场,欢迎使用
此方案的缺点:在未登录的状态下,用户会看到明显的页面跳转,跳转到一个空白页面,突然又跳转到登录页,用户体验不是特别好,同时在onShow生命周期钩子里做数据重新渲染会有一个问题,这样会造成过多的网络请求,如果用户量不小的话会对于服务器造成一些压力
场景方案2:跳转进需要登录的页面前判断登录态,并记录正在跳转的页面,登录后重定向到前面已经记录的跳转页面
此方案优点:避免了onShow频繁触发导致服务器渲染浪费的问题,缺点就是你需要在每一个跳转需要登录的页面前做登录态判断,会导致代码冗余工作量增加,后期维护不是特别好
方案选择
我选择的是场景方案2
二种方案都有优缺点,方案1有一点是用户感受最直接的,就是闪跳的用户体验那一点,至于onShow会导致接口频繁请求问题是有方法解决的,后面会提到;方案2只要想办法解决代码冗余的问题即可
方案实现细节
解决代码冗余问题,我们使用uni-app的拦截api来做下路由拦截即可,根据跳转的URL和当前登录态判断要不要先跳登录页做登录,在main.js中增加路由拦截,关键代码如下:
...
/**
* 需要登录才能跳转的页面
*/
const needLoginPages = [
'/orders/detail/detail',
'/orders/orderList/orderList',
'/orders/confirmOrder/confirmOrder',
'/orders/orderSuccess/orderSuccess',
'/other/address/address',
'/other/address/addAddress'
]
// 要拦截的路由方法
const interceptors = ['navigateTo', 'reLaunch', 'redirectTo']
const globalStoreInstance = globalStore(pinia);
// 路由拦截
interceptors.forEach(interceptor => {
uni.addInterceptor(interceptor, {
invoke(e) {
// 判断当前页面是否是要需要登录才能跳转的页面里
const needLogin = needLoginPages.findIndex(item => e.url.includes(item)) !== -1;
if (needLogin && !storage.get(TOKEN)) {
// 记录要跳转的页面
globalStoreInstance.setNeedLoginBackPage(e.url);
uni.navigateTo({
url: '/other/login/login'
})
return false
}
return true
}
})
})
...
我在写这篇博客的时候,我发现这里代码其实有一个可优化点,你发现了吗?欢迎留言讨论
同时在登录页登录成功后需要做一下跳转逻辑,关键代码如下:
...
// 解决登录后跳转的问题
if (globalStoreInstance.needLoginBackPage) {
uni.redirectTo({
url: globalStoreInstance.needLoginBackPage,
complete: () => {
globalStoreInstance.setNeedLoginBackPage('');
}
})
} else {
// 解决登录回去页面数据丢失的问题
const pages = getCurrentPages();
if (pages.length >= 2) {
// 获取前一个页面实例
const prevPage = pages[pages.length - 2];
// 调用前一个页面的onLoad方法
if (prevPage?.onLoad) {
prevPage.onLoad(prevPage.options || {}); // 传递原始参数
}
}
uni.navigateBack();
}
...
看代码除了跳转还处理了back,这一段back逻辑也是我在实现的时候发现它可以解决方案1的onShow问题,也就是说方案1也就是只有一个体验问题,所以二种方案我觉得都是可行的,同时我提供方案1接口拦截的代码:
import { Request, storage } from '@/uni_modules/hbxw-utils/js_sdk/hbxw-utils.js';
import { BASE_URL } from '@/config/http';
import { TOKEN } from '@/config/common';
const request = new Request({
isLogin: true,
});
request.baseUrl = BASE_URL;
/**
* 请求拦截,可以通过add方法添加多个
* 参数为请求配置,可以对请求参数做一些特殊处理
*/
request.requestIntercept.add((requestConfig) => {
// 如果有传就用传的,没有就去取,为了解决登录默认token
console.log('---- requestConfig ----:', requestConfig)
if (!requestConfig.header) {
requestConfig.header = {}
}
// 如果header中没有Accept,则设置为application/json
if (!requestConfig.header?.Accept) {
requestConfig.header.Accept = 'application/json';
}
if (!requestConfig.header?.Authorization) {
let Authorization = ''
try {
Authorization = storage.get(TOKEN) || '';
} catch (err) {
console.log(err)
}
// 添加Authorization到header中用于服务端登录判断
if (Authorization) {
if (!requestConfig.header) {
requestConfig.header = {}
}
requestConfig.header.Authorization = Authorization;
}
}
// 如果返回true则请求会中断
// return true;
});
/**
* 响应拦截,可以通过add方法添加多个
* 第一个参数为请求响应体
* 第二个参数为请求配置信息
*/
request.responIntercept.add((response, requestConfig) => {
console.log('---- response ----:', response)
// 如果接状态码为401,而且当前接口是需要判断登录状态的
if (response.statusCode == 401 && requestConfig.isLogin) {
uni.navigateTo({
url: '/other/login/login'
})
// 返回true 中断后面处理
return true;
}
// 通用错误处理
if (response.statusCode !== 200 || response.data.code !== 200) {
uni.showToast({
title: response.data.message || '请求失败,请稍后再试',
icon: 'none'
})
return true;
}
});
export default request;
期望
解决问题的方法千千万,上述是我是解决登录跳转逻辑的处理方案,如果在上面二种方案中,你会选择哪一种了?聪明的你也一定有别的更好的方案,期待你的分享和留言,共同进步。
uni-app小程序登录后…的更多相关文章
- 完整且易读的最新版小程序登录态和检验注册过没的app.js写法
目录 0.可参考的官方页面 1.流程 2.app.js代码 3.java后台怎么通过code获取openId 0.可参考的官方页面 获取登录凭证:https://developers.weixin.q ...
- 微信小程序登录(包括获取不到unionid的情况)
我们一般都是先获取到微信的 unionid,然后再通过 unionid 去登录自己的网站,就可以关联到用户在自己网站上的 user_id,但是在小程序登录中,有时候可以获取到 unionid,有时候获 ...
- ASP.NET WebAPI 双向token实现对接小程序登录逻辑
最近在学习用asp.net webapi搭建小程序的后台服务,因为基于小程序端和后台二者的通信,不像OAuth(开放授权),存在第三方应用.所以这个token是双向的,一个是对用户的,一个是对接口的. ...
- 微信小程序登录对接Django后端实现JWT方式验证登录
先上效果图 点击授权按钮后可以显示部分资料和头像,点击修改资料可以修改部分资料. 流程 1.使用微信小程序登录和获取用户信息Api接口 2.把Api获取的用户资料和code发送给django后端 3. ...
- 微信小程序 - 登录(后端实现) | 授权(后端实现)
登录与授权 官方文档 一.登录 登录流程时序 说明: 调用 wx.login() 获取 临时登录凭证code ,并回传到开发者服务器. 调用 code2Session 接口,换取 用户唯一标识 Ope ...
- 微信小程序登录那些事
最近团队在开发一款小程序,都是新手,一边看文档,一边开发.在开发中会遇到各种问题,今天把小程序登录这块的流程整理下,做个记录. 小程序的登录跟平时自己APP这种登录验证还不太一样,多了一个角色,那就是 ...
- Flask与微信小程序登录(后端)
开发微信小程序时,接入小程序的授权登录可以快速实现用户注册登录的步骤,是快速建立用户体系的重要一步.这篇文章将介绍 python + flask + 微信小程序实现用户快速注册登录方案(本文主要进行后 ...
- 使用 IdentityService4 集成小程序登录一种尝试
1 场景介绍 主要业务是通过 App 承载,在 App 中可以注册和登录,为了更好的发展业务引入了微信小程序,于是如何让这两个入口的用户互通便成了需要解决的问题. 看了一下其它 App 大致地思路是两 ...
- uni-app微信小程序登录授权
微信小程序授权是非常简单和常用的功能,但为了方便,还是在此记录一下要点: 首先是需要用到一个授权按钮来触发获取用户信息授权: 关键在于 open-type 为 getUserInfo , 然后有个@g ...
- 微信小程序开发:python+sanic 实现小程序登录注册
开发微信小程序时,接入小程序的授权登录可以快速实现用户注册登录的步骤,是快速建立用户体系的重要一步.这篇文章将介绍 python + sanic + 微信小程序实现用户快速注册登录全栈方案. 微信小程 ...
随机推荐
- 奥特曼框架autMan对接微信(千寻、西瓜)框架的详细教程
教程只写常用的两:西瓜.千寻,都运行在windows平台上. 1.千寻对接 文章底部下载千寻微信框架 解压至win电脑 电脑安装微信3.6.0.18并关闭自动更新 运行千寻微信框架 5.千寻框架设置 ...
- Linux - 关于yum源 file & ftp & http 的三种配置方式
一.环境准备 1.两台Centos服务器:node1.node2 2.配置ip:node1(192.168.2.111).node2(192.168.2.112) 3.关闭防火墙 systemctl ...
- Spark SQL (一)
Spark SQL Spark与Hive的比较,Hive用一句话总结是,传入一条交互式sql在海量数据中查找结果,Spark可以将其结果转化成RDD来来进一步操作. 1.0以前: Shark 1.1. ...
- 分布式锁—2.Redisson的可重入锁
大纲 1.Redisson可重入锁RedissonLock概述 2.可重入锁源码之创建RedissonClient实例 3.可重入锁源码之lua脚本加锁逻辑 4.可重入锁源码之WatchDog维持加锁 ...
- npm 如何更新项目最新依赖包
NPM 是什么? Node 软件包管理器(NPM)提供了各种功能来帮助你安装和维护项目的依赖关系. 由于错误修复.新功能和其他更新,依赖关系可能会随着时间的推移而变得过时.你的项目依赖越多,就越难跟上 ...
- Lambda表达式--java进阶day03
1.Lambda表达式 2.Lambda表达式格式 ()放我们要重写的方法的形参,{}放我们要写的逻辑 show()没有形参,所以空着,我们将打印语句写入{}中 如图,匿名内部类和Lambda都出现了 ...
- 接口新特性--java进阶day03
1.接口新特性 在JDk8和JDK9开始,接口可以定义普通方法 这时就会感到很奇怪,明明之前说好接口只是用来制定规则的,为什么现在又可以定义普通方法了呢? 我们以一个公司案例进行讲解,公司1.0上线了 ...
- 【Linux】Vim 设置
[Linux]Vim 设置 零.起因 刚学Linux,有时候会重装Linux系统,然后默认的vi不太好用,需要进行一些设置,本文简述如何配置一个好用的Vim. 壹.软件安装 sudo apt-get ...
- 学习Linux只要学会这个命令就够了!
大家好,我是良许. 这段时间又是搬家,又是找新办公室,现在终于安顿下来了,有时间给大家分享干货了. 今天给大家介绍一个 Linux 超级实用命令,有了这个命令,你就可以愉快使用 Linux 上几乎所有 ...
- EvoSuite使用总结
1.安装EvoSuite插件 以IDEA为例,在Plugins栏搜索EvoSuite后点击install,安装完成后重启IDEA 2.使用EvoSuite 选中文件右键选择Run EvoSuite 生 ...