详解Vue微信授权登录前后端分离较为优雅的解决方案


Posted in Javascript onJune 29, 2018

微信授权登录是一个非常常见的场景,利用微信授权登录,我们可以很容易获取用户的一些信息,通过用户对公众号的唯一openid从而建立数据库绑定用户身份.

微信授权登录的机制这里不做详述,微信官方文档已有详述,简述就是通过跳转微信授权的页面,用户点击确认后,微信会跳到回调页面,此时回调页面url上会携带code参数,通过code参数,后端可以拿code换取拥护openid,或者用户信息

在vue项目中,通常是一个SPA应用,即所有的页面都是同一个html,通常现在开发也是前后端彻底分离的,vue打包后生成的纯静态文件,甚至可以不经过服务器,所以通过后端弄跳转之类的都不太优雅,本文即介绍此类场景的微信授权登录

对于一个vue的SPA应用,我们通常可能有很多页面,在微信公众号上我们可能配置多个菜单,多个菜单对应vue的路由页面,但是可能并不是每个页面都需要用户授权才能进入,有些页面用户不登录也需要可以预览,此时我们可以通过vue router来实现前端路由拦截

router.beforeEach(async (to, from, next) => {
 if (to.matched.some(recode => recode.meta.noAuth)) {
  next()
 } else {
  // store已存在用户信息直接进入页面
  if (store.state.userInfo.nickname) {
   next()
   return
  }
  const code = getUrl(window.location.href).code // 截取url上的code ,可能没有,则返回''空字符串
  let res = await api.post('/imsl/user/user-auth', [code]) // 获取用户信息,后端可首先通过cookie,session等判断,没有信息则通过code获取
  console.log(res)
  // 返回用户信息
  if (res.code === 200 && res.data.is_auth) {
   store.commit('setUserInfo', res.data)
   next()
  } else {
   // 此状态根据业务需求 可能不存在
   if (res.code === 201) {
    const openid = res.data.openid
    console.log(openid)
    store.commit('setOpenid', openid)
    localStorage.setItem('openid', openid)
    next('/enlist-info')
   }
   // 上面的获取用户信息接口,如果cookie,session拿不到用户信息,且传递的code为空,则跳转到微信授权页面
   if (res.code === 202) {
    console.log(window.location.origin)
    console.log(to.fullPath)
    // 这个redirectUrl用 当前页路径或者tof.fullPath(将要进入的路径)
    let redirectUrl = window.location.href
    redirectUrl = encodeURIComponent(redirectUrl)
    console.log(redirectUrl)
    const appid='wxdff0642c2120ea39'
    window.location.href = `https://open.weixin.qq.com/connect/oauth2/authorize?appid=${appid}&redirect_uri=${redirectUrl}&response_type=code&scope=snsapi_userinfo&state=123#wechat_redirect`
   }
  }
 }
})

上述代码基本阐述了一个授权的过程,首先,我们在配置vue路由的时候,设置此路由是否需要登录即给router的meta加一个noAuth:true的属性,这个是处理不需要登录的页面,通过router.beforeEach进行判断,如果是不需要登录的页面:noAuth,则直接next()让其进入相应页面.对于需要登录的页面,则让后端配合,此时,后端写一个获取用户信息的接口,前端则直接调用获取用户信息的接口,当然,不需要每个页面都调用,获取一次之后可以将用户信息存入vuex中,所以通过判断vuex里面有没有用户信息,如果已存在用户信息,则进入页面.没有用户信息那就调用后端获取用户信息的接口,说到这里,终于回到此文主题了,用户信息是通过微信授权登录拿到的,此时后端如何拿到用户信息呢?这里,可以跟后端商议好,用户绑定身份后,后端则可以通过设置cookie,token之类的保存这个用户登录状态,如果有相关状态,那么后端则可以直接返回用户信息. 如果是首次进入,或者cookie,token之类的已失效,那么此时则会调用微信授权登录了,如上述代码所述,分为三种情况,

1. 通过cookie,token等,后端直接拿到了用户信息,此时则拿到用户信息直接进入页面,同时把用户信息存入vuex中

2. 没有用户信息的情况,此时也没有cookie,token,那就需要重新调用微信授权登录了,上面给出的两种返回码code=201,code=202的情况,当code=2是则由前端直接跳转到微信授权页面,而redirectUri则为将要进入的页面,此时会发生什么呢?会跳到微信授权页面,用户点击之后又回到了这个页面,不同的是此时url上面已经携带了code,前端通过字符串截取拿到code,发送给后端,后端即可通过code换取openid以及用户信息了.

总结:

  1. 项目采用前后端完全分离方式,即打包后给的纯静态文件放在服务器,访问index.html
  2. 后端不在接口处拦截,不需要后端跳转微信授权登录页面,由前端路由来拦截跳转,实现方法如3.
  3. 前端在需要用户身份才能进入的页面通过vue-router的 router.beforeEach钩子函数拦截,此时调用获取用户信息接口,后端首先通过获取cookie,token等判断用户,无相关信息返回201或202,当返回202的时候,前端跳转到微信授权页,redirecturi即为要进入的页面的url,跳转授权后微信会在url上面携带code回到当前页,此时前端截取url上的code传给后端,后端通过code在后端处理拿到用户信息,openid等实现了授权登录

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
JavaScript学习笔记之Function对象
Jan 22 Javascript
浅谈jQuery页面的滚动位置scrollTop、scrollLeft
May 19 Javascript
js+html5通过canvas指定开始和结束点绘制线条的方法
Jun 05 Javascript
jquery判断至少有一个checkbox被选中的方法
Jun 05 Javascript
漫谈JS引擎的运行机制 你应该知道什么
Jun 15 Javascript
轻松掌握JavaScript享元模式
Aug 27 Javascript
angularjs之$timeout指令详解
Jun 13 Javascript
Node做中转服务器转发接口
Oct 18 Javascript
jQuery实现参数自定义的文字跑马灯效果
Aug 15 jQuery
vue进入页面时不在顶部,检测滚动返回顶部按钮问题及解决方法
Oct 30 Javascript
Windows上node.js的多版本管理工具用法实例分析
Nov 06 Javascript
Vue组件更新数据v-model不生效的解决
Apr 02 Vue.js
JavaScript实现仿Clock ISO时钟
Jun 29 #Javascript
vue ssr 指南详读
Jun 29 #Javascript
jQuery实现获取动态添加的标签对象示例
Jun 28 #jQuery
Vue实现textarea固定输入行数与添加下划线样式的思路详解
Jun 28 #Javascript
详解swipe使用及竖屏页面滚动方法
Jun 28 #Javascript
微信小程序wx.uploadfile 本地文件转base64的实现代码
Jun 28 #Javascript
浅谈vue首屏加载优化
Jun 28 #Javascript
You might like
星际争霸, 教主第一视角, ZvT经典龙蛇演义
2020/03/02 星际争霸
php和js如何通过json互相传递数据相关问题探讨
2013/02/26 PHP
浅谈laravel-admin form中的数据,在提交后,保存前,获取并进行编辑
2019/10/21 PHP
js 手机号码合法性验证代码集合
2012/09/29 Javascript
jquery Mobile入门—外部链接切换示例代码
2013/01/08 Javascript
html组件不可输入(只读)同时任何组件都有效
2013/04/01 Javascript
js 实现 input type="file" 文件上传示例代码
2013/08/07 Javascript
jquery实现的伪分页效果代码
2015/10/29 Javascript
jQuery on()绑定动态元素出现的问题小结
2016/02/19 Javascript
JavaScript关于提高网站性能的几点建议(一)
2016/07/24 Javascript
JS实现列表的响应式排版(推荐)
2016/09/01 Javascript
原生js简单实现放大镜特效
2017/05/16 Javascript
gulp解决跨域的配置文件问题
2017/06/08 Javascript
使用vuex的state状态对象的5种方式
2018/04/19 Javascript
create-react-app 修改为多入口编译的方法
2018/08/01 Javascript
解决vue 项目引入字体图标报错、不显示等问题
2018/09/01 Javascript
vue+element实现图片上传及裁剪功能
2020/06/29 Javascript
Python编码爬坑指南(必看)
2016/06/10 Python
分享Python开发中要注意的十个小贴士
2016/08/30 Python
对python:循环定义多个变量的实例详解
2019/01/20 Python
详解Python正则表达式re模块
2019/03/19 Python
python文件绝对路径写法介绍(windows)
2019/12/25 Python
解决Jupyter因卸载重装导致的问题修复
2020/04/10 Python
分享unittest单元测试框架中几种常用的用例加载方法
2020/12/02 Python
Python排序函数的使用方法详解
2020/12/11 Python
欧洲高端品牌直销店:Fashionesta
2016/08/31 全球购物
应届大学生自荐信格式
2013/09/21 职场文书
教师现实表现材料
2014/02/14 职场文书
八项规定整改方案
2014/02/21 职场文书
总经理的岗位职责
2014/02/23 职场文书
经典公益广告词
2014/03/13 职场文书
一分钟演讲稿
2014/04/30 职场文书
3.15消费者权益日活动总结
2015/02/09 职场文书
奖学金申请个人主要事迹材料
2015/11/04 职场文书
2016年教师党员创先争优承诺书
2016/03/24 职场文书
springboot 自定义配置 解决Boolean属性不生效
2022/03/18 Java/Android