公司最近有微信公众号的需求,那么微信登录授权和如何使用WX-JSSDk实现分享等等肯定是最头疼的问题。本人也是第一次开发微信公众号,在网上看了很多篇博客,最终选定了两种方法,并且亲测有效。

一、通过全局,在router.afterEach中定义

1.首先通过yarn add weixin-js-sdk/ npm i weixin-js-sdk
2.将微信jsdk挂载到全局上
在utils目录下新建WechatPlugin.js
WechatPlugin.js

  1. import wx from 'weixin-js-sdk'
  2. const plugin = {
  3. install(Vue) {
  4. Vue.prototype.$wechat = wx
  5. Vue.wechat = wx
  6. },
  7. $wechat: wx
  8. }
  9. export default plugin
  10. export const install = plugin.install

main.js中

  1. import WechatPlugin from './utils/WechatPlugin'
  2. // 全局注册微信jsdk
  3. Vue.use(WechatPlugin)

3.router.afterEach中

  1. import wechatUtil from '@/utils/wechatUtil' // 在此文件中定义微信的一些方法
  2. router.afterEach((to, from) => {
  3. let path = to.fullPath.slice(1) // 去除'/'
  4. let url
  5. const jsApiList = [
  6. 'onMenuShareAppMessage',
  7. 'onMenuShareTimeline',
  8. 'chooseWXPay',
  9. 'showOptionMenu',
  10. "updateAppMessageShareData",
  11. "hideMenuItems",
  12. "showMenuItems"
  13. ]
  14. if (!sessionStorage.getItem('initLink')) {
  15. // 解决ios微信下,分享签名不成功的问题,将第一次的进入的url缓存起来。
  16. sessionStorage.setItem('initLink', document.URL)
  17. }
  18. if (!!window.__wxjs_is_wkwebview) {
  19. // ios
  20. url = sessionStorage.getItem('initLink')
  21. wechatUtil.setWeChatConfig(url, jsApiList)
  22. } else {
  23. // 安卓
  24. url = location.origin + process.env.BASE_URL + path
  25. // setTimeout(() => {
  26. wechatUtil.setWeChatConfig(url, jsApiList)
  27. // }, 0)
  28. }
  29. })

3.wechatUtil.js中

  1. import Vue from 'vue'
  2. export default {
  3. appid: process.env.VUE_APP_WECHAT_APPID, // 可以在根据不同环境配置appid
  4. setWeChatConfig(url, jsApiList) {
  5. getSignature(decodeURIComponent(url)) // getSignature需要你自己跟后端约定请求签名时的接口
  6. .then(data => {
  7. Vue.wechat.config({
  8. debug: false,
  9. signature: data.signature,
  10. nonceStr: data.nonceStr,
  11. timestamp: data.timestamp,
  12. appId: data.appId,
  13. jsApiList
  14. })
  15. })
  16. .catch(err => {
  17. console.log(err)
  18. })
  19. }
  20. }

上面方法虽然全局可以使用,但是会遇到一个问题,在单个页面调用微信jsddk中的updateAppMessageShareData方法或者其他方法时,有时成功有时失败,这可能是微信jsdk异步的问题,因此,需要你在单个页面中使用的时候加上setTimeout(()=>{ “这里调取微信的接口” },500)。
下面的第二种方法我觉得是最方便也是最自定义能力最好的,在需要的页面的调取。

二、方法二通过new promise封装成统一的入口,在单个页面中调用

我们还是要在router.afterEach中将进入的url记录下来,我是放在vuex上的(这里要特别注意苹果手机和安卓手机的区别,这里我就不多做讲解,原因是苹果浏览器中的url是第一次进来的url)
1.在router.afterEach中

  1. import store from '@/store'
  2. router.afterEach((to, from) => {
  3. let path = to.fullPath.slice(1) // 去除'/'
  4. if (!sessionStorage.getItem('initLink')) {
  5. // 解决ios微信下,分享签名不成功的问题,将第一次的进入的url缓存起来。
  6. sessionStorage.setItem('initLink', document.URL)
  7. }
  8. let url
  9. if (!!window.__wxjs_is_wkwebview) {
  10. // ios
  11. url = sessionStorage.getItem('initLink')
  12. } else {
  13. // 安卓 process.env.BASE_URL 自己定义各个环境下域名变量
  14. url = location.origin + process.env.BASE_URL + path
  15. }
  16. store.commit('page/setInitLink', url)
  17. })

2.在store/page.js中

  1. const state = {
  2. initLink: ''
  3. }
  4. const mutations = {
  5. setInitLink (state, initLink) {
  6. state.initLink = initLink
  7. }
  8. }
  9. export default {
  10. namespaced: true,
  11. state,
  12. mutations
  13. }

3.在utils/wechatUtil.js定义初始化方法

  1. import wx from 'weixin-js-sdk'
  2. import store from '@/store'
  3. export default {
  4. /* 初始化wxjsdk各种接口 */
  5. init(apiList = [], url) {
  6. //需要使用的api列表
  7. return new Promise((resolve, reject) => {
  8. getSignature(store.state.page.initLink).then(res => {
  9. if (res.appId) {
  10. wx.config({
  11. // debug: true,
  12. appId: res.appId,
  13. timestamp: res.timestamp,
  14. nonceStr: res.nonceStr,
  15. signature: res.signature,
  16. jsApiList: apiList
  17. })
  18. wx.ready(res => {
  19. // 微信SDK准备就绪后执行的回调。
  20. resolve(wx, res)
  21. })
  22. } else {
  23. reject(res)
  24. }
  25. })
  26. })
  27. }
  28. }

4.在页面中的使用

  1. import wechatUtil from '@/utils/wechatUtil'
  2. wechatUtil
  3. .init([
  4. 'updateAppMessageShareData',
  5. 'onMenuShareAppMessage',
  6. 'onMenuShareTimeline',
  7. 'updateTimelineShareData'
  8. ])
  9. .then((wx, res) => {
  10. // 这里写微信的接口
  11. })

总结:最后我个人推荐第二种方法,第一种方法虽然很方便,但是每次路由跳转都调取了微信获取签名初始化的方法,而且自定义的扩展性不是很强,而且还会有微信接口异步的问题,需要用到微信中的debu:true调试。第二种方法使用和定义起来比较简单。