背景

vue-cli项目。路由是hash模式。需要授权的场景有:项目入口处(App.vue),指定页面(创建时、methods 方法内);可以携带参数

思路

由于hash模式# 号的存在,授权后链接会被扰乱。所以 我希望在 授权前 将重定向的链接 即 redirect_uri

即:两步操作

1、授权前 将redirect_uri 改为没有# 的url

2、项目入口处,将没有#的url改回到 带#的url,跳到指定页面。

主要代码

1、授权前,redirect_uri 重置

这里会将当前页面的原有参数信息、额外参数、codepath参数 作为 redirect_uri 的新参数。其中codepath是 用来指定 跳转页面的,通常为 授权发起页。

/**
   * 开始设置页面链接,进行code 授权重定向
   *
   * @param scope 授权作用域
   * @param path  授权所在页面 path
   * @param extraPm 额外参数,需要在 配置中 进行参数配置
   * @returns {string}
   */
  get_codeV3(scope, path, extraPm={}) {
        if (process.env.NODE_ENV === 'development' && config.isMockWeChat)
          return '0713eFVW0AmFP12kGVTW0oGDVW03eFV2'

      let homeUrl = '', searchOb = {}, searchstr = ''
      let oriUrls = window.location.href.split('?')
      let baseShareURl = window.location.origin + window.location.pathname
      homeUrl = baseShareURl
      if (oriUrls.length > 1) {
        let searchUrls = oriUrls[1].split('#')
        let searchUrlReal = searchUrls[0]
        searchOb = qs.parse(searchUrlReal)
        let {code} = searchOb
        if (code)
          delete searchOb.code
      }
      searchOb.codepath = path
      Object.assign(searchOb, extraPm)
      searchstr = `?${qs.stringify(searchOb)}`
      homeUrl += searchstr

      let redirect_uri = encodeURIComponent(homeUrl);

      let response_type = 'code';
      // 应用授权作用域,
      // snsapi_base (不弹出授权页面,直接跳转,只能获取用户openid),
      // snsapi_userinfo (弹出授权页面,可通过openid拿到昵称、性别、所在地。并且, 即使在未关注的情况下,只要用户授权,也能获取其信息 )

      // 重定向后会带上state参数,开发者可以填写a-zA-Z0-9的参数值,最多128字节
      let state = '';
      let url = `https://open.weixin.qq.com/connect/oauth2/authorize?appid=${appid}&redirect_uri=${redirect_uri}&response_type=${response_type}&scope=${scope}&state=${state}#wechat_redirect`;
      window.location.href = url;
    }

2、链接前置判断

在项目入口处,发现有 codepath 参数,即 将该链接视为 授权后的redirect_uri 链接。进行页面重定向,跳转到 codepath 指定页面

/**
   * 页面重定向(授权)-确保页面链接正常
   * 支持多参、无参、指定页面
  * code 在 返回的search 里
   * @returns {{needRedirect: boolean, homeUrl: string, search: {}}}
   */
  urlResetForWxCode: function () {
    //需要重定向时:codepath 参数 与 code 参数同在。

    let needRedirect = false, homeUrl = '', searchOb = {}, searchstr = ''
    let oriUrls = window.location.href.split('?')
    let baseShareURl = window.location.origin + window.location.pathname
    homeUrl = baseShareURl
    if (oriUrls.length > 1) {
      let searchUrls = oriUrls[1].split('#')
      let searchUrlReal = searchUrls[0]
      let searchObReal = {}
      searchOb = qs.parse(searchUrlReal)
      let { codepath } = searchOb
      needRedirect = codepath
      if (codepath) {
        config.platform.wxCode[codepath].forEach(item => searchObReal[item] = searchOb[item])
        if (config.platform.wxCode[codepath].length)
          searchstr = `?${qs.stringify(searchObReal)}`
        homeUrl += `#${codepath}` + (searchstr.length > 1 ? searchstr : '')
      }
    }
    return {
      needRedirect,
      homeUrl,
      search: searchOb
    }
  }

// 配置信息

config = {
  platform: {
    wxCode: {'/': ['hello', 'yiha'], '/spe': ['spp']}, // 微信授权-需要授权的页面: [页面用到的参数]
  }
}

 

使用

1、授权发起位置通常有 项目入口处、页面create 时,页面 指定方法内。

if (!storage.getToken()) {// code 授权标志 保存在localstorage。
        wx_jssdk.get_codeV3('snsapi_userinfo', '/')
        return
      }

2、项目入口处进行前置判断

// App.vue

created(){

      // 可以在项目入口处 进行code授权
      if (!storage.getToken()) {// code 授权标志 保存在localstorage。
        wx_jssdk.get_codeV3('snsapi_userinfo', '/')
        return
      }
      // 项目只要用到授权,则需要进行重定向判断。
      let {needRedirect, homeUrl, search} = browserUtil.urlResetForWxCode()
      if (needRedirect) {
        storage.setToken({code: search.code}) // 保存code
        window.location.href = homeUrl
      }

    }

 

 写在最后

想着尽量能够做到配置化、可扩展,目前很low。凑合用吧😄

配置化的体现:

链接前置判断方法内,组装链接参数时,我是只拿配置信息里的存在的参数。(方法里飘红的代码)。这是为了参数可控,避免不必要的bug。

可扩展的体现:

在redirect_uri 重置方法 参数里,有一个额外参数,原本 想着只截取授权发起页的原始search参数就好了。但一想 会不会有这样的使用场景:

表单提交页,在提交时 才去发起授权。这样的话 可能授权后 原表单信息已丢失,即需要重新再来;体验并不好。那么可不可以 将表单数据作为 授权时的一部分参数,然后在授权成功后,页面重新create时 检测到有表单数据时 直接提交。(当然,你也可以通过local保存数据来实现这个功能,或者其他)

注:该额外参数 也需要在配置里注明,不然 前置判断里 会不予理会。