vue + node + express 实现微信分享全攻略

本人自己写完全栈小项目,想在微信分享时带标题、描述和缩略图,没想到搞了3天,才搞定!所以写篇文章,记录浪费的时间,以此为戒,以后要多读书。。。

技术栈:vue 2.9.2, node v8.11.1, express 4.15.5

一、前期准备:

一定要看到微信的开发者文档,本人最后发现掉进去的坑,都是开发文档没看仔细。。


在这个位置处,填写js安全域名


上传到域名根目录后,一定要检查能否访问


在服务器配置里,后台要写相应的接口,供微信调用,稍后上代码。


要把服务器的ip地址添加到白名单里



同时要去接口权限内,检查公众号是否有分享权限,我之前找了一个没有权限的公众号号,被坑了半个小时。

二、动手实践

1、后端代码

/getwe 这个接口是用来,接受微信服务器验证的;/getsignature是供前端调用的接口,可以获取签名等信息。

var express = require('express');

var crypto = require('crypto')

var router = express.Router();

var sha1 = require('sha1');

var wxShare = require('./wxShare')

var token = "403106220Asd"

/* GET home page. */

router.get('/getwe', function(req, res, next) {

  console.log();

    var signature = req.query.signature;

    var timestamp = req.query.timestamp;

    var nonce = req.query.nonce;

    var echostr = req.query.echostr;

    console.log(signature, timestamp, nonce, echostr);

    /*  加密/校验流程如下: */

    //1. 将token、timestamp、nonce三个参数进行字典序排序

    var array = new Array(token,timestamp,nonce);

    array.sort();

    var str = array.toString().replace(/,/g,"");

    //2. 将三个参数字符串拼接成一个字符串进行sha1加密

    var sha1Code = crypto.createHash("sha1");

    var code = sha1Code.update(str,'utf-8').digest("hex");

    //3. 开发者获得加密后的字符串可与signature对比,标识该请求来源于微信

    if(code===signature){

        res.send(echostr)

    }else{

        res.send("error");

    }

});

/**

* 分享

*/

router.post('/getsignature', function(req, res, next) {

    let hrefURL = req.body.urlhref;

    wxShare.prototype.accessToken(hrefURL, function(data) {

        res.json(data);

    });

});

module.exports = router;


wxShare:

var url = require('url');

var request = require('request');

var sha1 = require('sha1');

let config = {

        appID: "",// 微信公众号ID

        appSecret: "" //微信公众号里有

    },

    configEnd = {

        appID: '',

        access_token: '',

        ticket: '',

        timestamp: '', // 必填,生成签名的时间戳

        nonceStr: '', // 必填,生成签名的随机串

        signature: '', // 必填,签名,见附录1

    };

/**

* 微信分享

*/

class wxShare {

    /**

    * 请求获取access_token 方法入口

    * @param {* URL链接} hrefURL

    * @param {* 回调请求方法} callback

    */

    accessToken(hrefURL, callback) { // 获取access_token

        let _this = this;

        var tokenUrl = 'https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=' + config.appID + '&secret=' + config.appSecret;

        request(tokenUrl, function(error, response, body) {

            if (response.statusCode && response.statusCode === 200) {

                body = JSON.parse(body);

                configEnd.access_token = body.access_token;

                _this.upJsapiTicket(hrefURL, body.access_token, callback)

            }

        });

    };

    /**

    * 获取Jsapi_Ticket

    * @param {* URL链接} hrefURL

    * @param {* token} access_Ttoken

    * @param {* 回调请求方法} callback

    */

    upJsapiTicket(hrefURL, access_Ttoken, callback) { // Jsapi_ticket

        let _this = this;

        var ticketUrl = 'https://api.weixin.qq.com/cgi-bin/ticket/getticket?access_token=' + access_Ttoken + '&type=jsapi';

        request(ticketUrl, function(err, response, content) {

            content = JSON.parse(content);

            if (content.errcode == 0) {

                configEnd.appID = config.appID;

                configEnd.ticket = content.ticket; // ticket

                configEnd.timestamp = _this.createTimestamp(); // 时间戳

                configEnd.nonceStr = _this.createNonceStr(); // 随机数

                configEnd.signature = _this.sign(hrefURL); // 签名

                callback && callback(configEnd); // 回调前端JS方法

            }

        })

    };

    /**

    * 随机字符串

    */

    createNonceStr() {

        return Math.random().toString(36).substr(2, 15);

    };

    /**

    * 时间戳

    */

    createTimestamp() {

        return parseInt(new Date().getTime() / 1000).toString();

    };

    /**

    * 拼接字符串

    * @param {*} args

    */

    rawString(args) {

        var keys = Object.keys(args);

        keys = keys.sort()

        var newArgs = {};

        keys.forEach(function(key) {

            newArgs[key.toLowerCase()] = args[key];

        });

        var string = '';

        for (var k in newArgs) {

            string += '&' + k + '=' + newArgs[k];

        }

        string = string.substr(1);

        return string;

    };

    /**

    * 签名

    * @param {*} url

    */

    sign(url) {

        let _this = this;

        var ret = {

            jsapi_ticket: configEnd.ticket,

            nonceStr: configEnd.nonceStr,

            timestamp: configEnd.timestamp,

            url: url

        };

        var string = _this.rawString(ret);

        var shaObjs = sha1(string);

        return shaObjs;

    };

}

module.exports = wxShare;

2、前端代码

在想要分享的页面的mounted写上调用,同时引入已经封装好的js。

这里有个小坑就是,当用npm i weixin-js-sdk  在2019年2月17日下载的是1.40 test版本,可能不是很稳定,在测试时候,wx.config总是显示错误信息error massage “config ok”,最后发现目前开发文档推荐的updateAppMessageShareData和updateTimelineShareData根本返回不了数据,所以用以前版本的接口函数,就通过了。

import wxshare from '@/utils/wxshare'

export default {

  data() {

    return {

      // jsonurl: ''

    }

  },

  // created() {

  //  // console.log('self.router', this.$route)

  //  this.jsonurl = this.$route.query

  // },

  mounted() {

    wxshare.do(this.$route.query, '缩略图url', '想用的标题', '这是一段描述')

  }


wxshare:

import axios from 'axios'
import wx from 'weixin-js-sdk'

const wxshare = {
  do(queryId, icon = '', title = '', desc = '') {
    const url = location.href.split('#')[0] // ruoter是hash模式的时候 获取锚点之前的链接
    console.log('前端传输前的url地址', location.href)
    console.log('前端传输前的url地址', url)
    axios.post('https:/..../getsignature', { // 服务端获取配置jssdk 签名等 文件

      urlhref: url

    }).then(response => {
      const res = response.data
      console.log('调用微信js接口返回的签名:', res)

      this.wxInit(res, queryId, icon, title, desc)
    })
  },
  // 微信分享
  wxInit(res, queryId, icon = '', title = '', desc = '') {
    const url = location.href.split('#')[0] // 获取锚点之前的链接

    // let links = url+'#/Food/' + this.$route.params.id;    //用于签名的url 和 用于微信分享的url可以不同
    // const links = url + '#/?jsonurl=' + queryId

    const links = url

    console.log('url link:', links)

    wx.config({
      debug: false,
      appId: res.appID,
      timestamp: res.timestamp,
      nonceStr: res.nonceStr,
      signature: res.signature,
      jsApiList: ['onMenuShareAppMessage', 'onMenuShareTimeline']
    })
    wx.ready(function() {
      wx.onMenuShareAppMessage({
        title: title, // 分享标题
        desc: desc, // 分享描述
        link: links, // 分享链接,该链接域名或路径必须与当前页面对应的公众号JS安全域名一致
        imgUrl: icon, // 分享图标
        success: function() {
          // 设置成功
          console.log('设置成功')
        }
      })
      // 微信分享菜单测试
      wx.onMenuShareTimeline({
        title: title, // 分享标题
        link: links, // 分享链接,该链接域名或路径必须与当前页面对应的公众号JS安全域名一致
        imgUrl: icon, // 分享图标
        success: function() {
          // 设置成功
          console.log('设置成功')
        }
      })
    })
    wx.error(function(err) {
      alert(JSON.stringify(err))
    })
  }
}

export default wxshare

PS:本人此次也是站在了巨人的肩膀上,所以避免了二次分享的坑、router是否是history的坑……如可以更优化代码地方,可以指正,共同进步!

文末奉上友情链接:

这位大神已经把vue前端如何设置写的明明白白 https://github.com/SevenChen/VueWxShare

后台部分借鉴了 https://blog.csdn.net/baidu_31333625/article/details/72673958

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,117评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,328评论 1 293
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,839评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,007评论 0 206
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,384评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,629评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,880评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,593评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,313评论 1 243
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,575评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,066评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,392评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,052评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,082评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,844评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,662评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,575评论 2 270

推荐阅读更多精彩内容

  • 先引入JS 文件 this.wxShare() 在created里调用 首先登陆微信公众号 JSSDK使用步骤 步...
    寄鱼予海与你阅读 6,584评论 1 3
  • 久违的新文章发布。。我的锅这篇文章是在写的太长,可以慢慢看一下,伪代码,毕竟简书上面,望大家见谅!!!先看官方文档...
    wyatt_plus阅读 1,519评论 0 2
  • 人生总是奇妙的, 一旦你努力去做一件事, 即便这件事听起来就像痴人说梦, 也一定会有开花结果的那一天。 也许结果不...
    净含量cy阅读 198评论 0 0
  • “嘀!” 清脆的水滴声响起。微弱的回音从四面八方反复回响,逐渐传遍整条通道,随即传向笼罩在大片漆黑中的远方。 一点...
    生还者阅读 463评论 1 12
  • 小说非原创,作者不知。个人喜爱,供大家欣赏。 阿悔(二) 5、 想不到阿悔住在这么深的山里。 一路上傅秋抱怨不迭,...
    记号晴系阅读 400评论 0 0