Vue項目history模式下微信分享總結

每回遇到微信分享都是一個坑,目前的商城項目使用Vue開發,採用history的路由模式,配置微信分享又遇到了不少問題,最後終於解決了,現將解決的過程分享一下。vue

原文justyeh.top/post/39android

技術要點

Vue,historyios

常見問題及說明

debug模式下報falsegit

這個沒得說,就是調用wx.config方法的參數錯誤形成的,請確認如下事項:github

  1. 是否成功綁定了域名(域名校驗文件要能被訪問到)
  2. 使用最新的js-sdk文件,由於微信會改部分api
  3. config方法的參數是否傳正確了(拼寫錯誤、大小寫...)
  4. 須要使用的方法是否寫在了jsApiList
  5. 獲取簽名的url須要decodeURIComponent
  6. 後臺的生成簽名的加密方法須要對照官方文檔

debug返回ok,分享不成功web

  1. 確保代碼拼寫正確
  2. 分享連接域名或路徑必須與當前頁面對應的公衆號JS安全域名一致
  3. 接口調用須要放在wx.ready方法中

單頁項目(SPA)中的一些要點

全部須要使用JS-SDK的頁面必須先注入配置信息,不然將沒法調用(同一個url僅需調用一次,對於變化url的SPA的web app可在每次url變化時進行調用,目前Android微信客戶端不支持pushState的H5新特性,因此使用pushState來實現web app的頁面會致使簽名失敗,此問題會在Android6.2中修復)。vue-router

上面那段話摘自官方文檔npm

開發者須要注意的事項:segmentfault

  1. android和ios須要分開處理
  2. 須要在頁面url變化的時候從新調用wx.config方法,android獲取簽名的url就傳window.location.href
  3. Vue項目在切換頁面時,IOS中瀏覽器的url並不會改變,依舊是第一次進入頁面的地址,因此IOS獲取簽名的url須要傳第一次進入的頁面url

Code

router/index.jsapi

......
import { wechatAuth } from "@/common/wechatConfig.js";
......

const router = new Router({
    mode: "history",
    base: process.env.BASE_URL,
    routes: [
        {
            path: "/",
            name: "home",
            meta: {
                title: "首頁",
                showTabbar: true,
                allowShare: true
            },
        },
        {
            path: "/cart",
            name: "cart",
            meta: {
                title: "購物車",
                showTabbar: true
            },
            component: () => import("./views/cart/index.vue")
        }
        ......
    ]
});


router.afterEach((to, from) => {
    let authUrl = `${window.location.origin}${to.fullPath}`;
    let allowShare = !!to.meta.allowShare;

    if (!!window.__wxjs_is_wkwebview) {// IOS
        if (window.entryUrl == "" || window.entryUrl == undefined) {
            window.entryUrl = authUrl; // 將後面的參數去除
        }
        wechatAuth(authUrl, "ios", allowShare);
    } else {
        // 安卓
        setTimeout(function () {
            wechatAuth(authUrl, "android", allowShare);
        }, 500);
    }
});
複製代碼

代碼要點:

  1. meta中的allowShare用於判斷頁面是否可分享
  2. window.__wxjs_is_wkwebview可用來判斷是不是微信IOS瀏覽器
  3. entryUrl是項目第一次進入的頁面的地址,將其緩存在window對象上
  4. 爲何安卓的時候要增長一個延時器,由於安卓會存在一些狀況,就是即使簽名成功,可是仍是會喚不起功能,這個貌似是一個比較穩妥的解決辦法

wechatConfig.js

import http from "../api/http";
import store from "../store/store";

export const wechatAuth = async (authUrl, device, allowShare) => {
    let shareConfig = {
        title: "xx一站式服務平臺",
        desc: "xxxx",
        link: allowShare ? authUrl : window.location.origin,
        imgUrl: window.location.origin + "/share.png"
    };

    let authRes = await http.get("/pfront/wxauth/jsconfig", {
        params: {
            url: decodeURIComponent(device == "ios" ? window.entryUrl : authUrl)
        }
    });

    if (authRes && authRes.code == 101) {
        wx.config({
            //debug: true,
            appId: authRes.data.appId,
            timestamp: authRes.data.timestamp,
            nonceStr: authRes.data.nonceStr,
            signature: authRes.data.signature,
            jsApiList: ["updateAppMessageShareData", "updateTimelineShareData", "onMenuShareAppMessage", "onMenuShareTimeline"]
        });

        wx.ready(() => {
            wx.updateAppMessageShareData({
                title: shareConfig.title,
                desc: shareConfig.desc,
                link: shareConfig.link,
                imgUrl: shareConfig.imgUrl,
                success: function () {//設置成功
                    //shareSuccessCallback();
                }
            });
            wx.updateTimelineShareData({
                title: shareConfig.title,
                link: shareConfig.link,
                imgUrl: shareConfig.imgUrl,
                success: function () {//設置成功
                    //shareSuccessCallback();
                }
            });
            wx.onMenuShareTimeline({
                title: shareConfig.title,
                link: shareConfig.link,
                imgUrl: shareConfig.imgUrl,
                success: function () {
                    shareSuccessCallback();
                }
            });
            wx.onMenuShareAppMessage({
                title: shareConfig.title,
                desc: shareConfig.desc,
                link: shareConfig.link,
                imgUrl: shareConfig.imgUrl,
                success: function () {
                    shareSuccessCallback();
                }
            });
        });
    }
};

function shareSuccessCallback() {
    if (!store.state.user.uid) {
        return false;
    }
    store.state.cs.stream({
        eid: "share",
        tpc: "all",
        data: {
            uid: store.state.user.uid,
            truename: store.state.user.truename || ""
        }
    });
    http.get("/pfront/member/share_score", {
        params: {
            uid: store.state.user.uid
        }
    });
}

複製代碼

總結

原先計劃不能分享的頁面就使用hideMenuItems方法隱藏掉相關按鈕,在ios下試了一下,有些bug:顯示按鈕的頁面切換的影藏按鈕的頁面,分享按鈕有時依然存在,刷新就沒問題,估計又是一個深坑,沒精力在折騰了,就改成隱私頁面分享到首頁,公共頁面分享原地址,若是有什麼好的解決辦法,請聯繫我!

一開始我有參考sf上的一篇博客segmentfault.com/a/119000001…,按照上面的代碼,android手機都能成功,可是IOS有一個奇怪的問題,就是分享間歇性的失效,同一個頁面,剛剛調起分享成功,再試一次就失敗(沒有圖標、title,只能跳轉到首頁),通過「不斷」努力的嘗試,應該是解決了問題,說一下過程:

  1. 一開始覺得是異步喚起沒成功的問題,就和android同樣給IOS調用wechatAuth方法也加了個定時器,測了一遍沒效果,放棄

  2. 起始js-sdk是經過npm安裝的,版本上帶了個test,有點不放心,改成直接使用script標籤引用官方的版本

  3. 從新讀了一遍文檔,發現onMenuShareTimeline等方法即將廢棄,就把jsApiList改成jsApiList:['updateAppMessageShareData','updateTimelineShareData'],改後就變成了IOS能夠成功,android分享失敗

  4. 百度updateAppMessageShareData安卓失敗緣由,參考這個連接www.jianshu.com/p/1b6e04c29…,把老的api也加到jsApiList中,仔細、反覆試了試兩種設備都ok,好像是成功了,說"好像"是由於內心沒底啊,各類「魔法」代碼!

最後,在這裏但願騰訊官方能不能走點心,更新文檔及時點,demo能不能提供完整點....

參考連接

segmentfault.com/a/119000001… www.jianshu.com/p/1b6e04c29… segmentfault.com/a/119000001… github.com/vuejs/vue-r…

相關文章
相關標籤/搜索