vue单页面在微信下只能分享落地页的解决方案

43次阅读

共计 1763 个字符,预计需要花费 5 分钟才能阅读完成。

实际上关键词叫微信 pushState 只能分享落地页更贴切一点
应用场景:

vue + vue-router
vue-router 使用 hash 模式(history 模式没试过)
不使用微信的 js-sdk(因为我这个项目是可配置域名的商城,比较特殊,不能使用微信 sdk)

这个方案并不是最优秀的,会对性能造成一定的影响
HTML5 history.pushState
vue-router 的内部是通过 history.pushState 和 history.replaceState 实现的。但是 iOS 设备的微信浏览器不会去检测它们的变化。但是我们可以通过更新 location.href 让微信浏览器识别到当前的 url。
// vue-router/src/util/push-state.js

export function pushState (url?: string, replace?: boolean) {
saveScrollPosition()
// try…catch the pushState call to get around Safari
// DOM Exception 18 where it limits to 100 pushState calls
const history = window.history
try {
if (replace) {
history.replaceState({key: _key}, ”, url)
} else {
_key = genKey()
history.pushState({key: _key}, ”, url)
}
} catch (e) {
window.location[replace ? ‘replace’ : ‘assign’](url)
}
}

export function replaceState (url?: string) {
pushState(url, true)
}

解决方法
window.location.href = window.location.href,这段代码可以让微信记录当前的 url,且不会刷新页面。可以在 app.vue 中 watch $route 在每次页面更新的时候执行一次。
// app.vue

watch: {
$route: {
immediate: true,
deep: true,
handler(to) {
// 微信浏览器判断
const WECHAT_BROWSER = navigator.userAgent.toLowerCase().includes(‘micromessenger’)

// 解决 iOS 微信浏览器分享地址只能是落地页的问题,这个操作是不会刷新页面的,query 参数改变也会执行
if (WECHAT_BROWSER) {
// eslint-disable-next-line
window.location.href = window.location.href
}
}
},
使用了上述方法可以解决这个问题,但是这会引出一个很奇葩的问题,在真机上进入 http://192.168.1.5:8080 和 http://192.168.1.5:8080/#/ 这两个页面,其中有一个链接的 bug 依然存在。原因具体不清楚,经过测试可以在入口文件 (main.js) 中在页面还没有展示内容前刷新一次页面,即可解决这个问题。
// main.js

// 微信浏览器判断
const WECHAT_BROWSER = navigator.userAgent.toLowerCase().includes(‘micromessenger’)

// 在 url 插入的 search 参数,可以随意,但是必须要
// 例:http://192.168.1.5:8080/?_wx_=1#/
const wxQuery = ‘_wx_=1’
const isRepeatQuery = location.search.includes(wxQuery)
if (WECHAT_BROWSER && !isRepeatQuery) {
const unit = (location.search && location.search !== ‘?’) ? ‘&’ : ‘?’
location.search += unit + wxQuery // 添加_wx_参数,该操作会刷新页面
}
上面的代码之所以要在 hash 前面加一个?_wx_= 1 参数,为了方便刷新页面给一个标志位判断是否已刷新。参数的 key-value 随意。

正文完
 0