小程序实现微信服务通知(消息/模板通知)

小程序端

将下面代码放在mixin里面

export default {

  methods:{

     saveFormIds() {

        /*

        * 调保存接口有两种情况:

        * 1: 退出小程序

        * 2: formID数组达到一定数量(此处设为30个)

        */

        console.log('saveFormIds')

        let formIds=this.$storage('formIds')

        if(formIds.length>0) {

            this.$storage('formIds', [])

        }

        console.log('formIds:',formIds)

        if(formIds.length>0) {

            this.$api.saveFormIds({

            formIds:formIds

        }).then(res=>{

            console.log(res)

        }).catch(err=>{

            console.error(err)

        })

    }

  },

dealFormIds(formId) {

    console.log('dealFormIds')

    console.log('click formId:',formId)

    // 开发工具上点击按钮生成的mock FormId不收集

    if(!formId||formId==='the formId is a mock one') return

        let formIds=this.$storage('formIds')

        let expire=parseInt(newDate().getTime()/1000)+604800// 计算七天后过期的时间戳

        if(!formIds) formIds=[]

        let item={

          formId:formId,

          expire:expire

        }

        formIds.push(item)

        this.$storage('formIds',formIds)

        // formID 长度大于30时,调用接口保存formId

        if(formIds.length>=30) {

            console.log('长度大于30啦啦啦啦啦啦啦啦啦啦啦')

            this.saveFormIds()

        }

 }

  }

}

app.js里面引入全局mixin:

import mixin from '@/mixin'

Vue.mixin(mixin)

小程序端formId收集示例(一般加在按钮上面):

<form report-submit='true'

    @submit="go2($event)">

    <button hover-class="none"

            formType="submit">

        button内容

    </button>

</form>

// go2方法

go2(type,event) {

    let formId=event.target.formId

      // 存储formId

    this.dealFormIds(formId)

}

在关闭小程序页面或者formId收集到30个的时候调用保存formId的接口

onHide() {

    this.saveFormIds()

    console.log('app hide')

}

node端(node+mongodb)

1.保存formId

// 保存formId

static async saveFormId(ctx) {

    let params=ctx.request.body

    console.log('==================保存formId==================')

    console.log('params',params)

    let userId=await ctx.getUserId()

    if(!userId) {

        return ctx.sendError('000002','用户未登录')

    }


    // 判断是否存在这个用户的记录

    let check=await TemplateMsg.findOne({userId:userId})

    let update

    let add

    if(check) {

    // 存在,更新

        update=await TemplateMsg.updateOne({userId:userId},

            {

                $push:{

                    formIdList:{

                        $each:params.formIds

                    }

            }

        })

    }else{

        // 不存在,添加

        let item={

        userId:userId,

        formIdList:params.formIds

        }

        add=await TemplateMsg(item).save()

    }

    if(update&&update.ok===1||add) {

        return ctx.send(null,'保存formId成功')

    }else{

        return ctx.sendError('000002','保存formId失败')

    }

}

2.发送模板消息

const sendTemplateMsg=async(template_id,

                            wx_url,

                            data,

                            touserOpenId,

                            userId)=>{

    // 获取并判断token是否失效

    let token=await getToken()

    // 获取有效的formId

    let formId=await getFormId(userId)

    if(!formId) {

        console.log('没有formId或者formId都已经过期了')

        return false

    }

    console.log('token',token)

    let options={

        method:'POST',

        uri:`https://api.weixin.qq.com/cgi-bin/message/wxopen/template/send?access_token=${token}`,

        body:{

            'touser':touserOpenId,// 待推送用户ID

            'template_id':template_id,// 模板ID

            'page':wx_url,// 小程序页面URL

            'form_id':formId,

            'data':data

        },

        json:true// Automatically stringifies the body to JSON

    }

    // 模板消息推送

    request(options)

    .then(parsedBody=>{

        console.log('推送结果',parsedBody)

        // POST succeeded...

    })

    .catch(function(err) {

        console.log(err)

    // POST failed...

    })

}

//判断token是否过期

function isTokenVaild(token) {

    return newPromise(async(resolve,reject)=>{

    let isVaild=true

    if(token.expires_in>parseInt(newDate().getTime()/1000)) {

    // 未过期

    console.log('没过期')

    console.log('token',token.access_token)

    // 判断是否失效

    await request({

        uri:`https://api.weixin.qq.com/cgi-bin/getcallbackip?access_token=${token.access_token}`,

        json:true

    }).then(function(res) {

        if(res.errcode) {

            isVaild=false

            console.log('失效')

        }else{

            isVaild=true

        }

    })

    }else{

        // 过期

        console.log('过期')

        isVaild=false

    }

    resolve(isVaild)

    })

}

// 获取并判断token是否失效

async function getToken() {

    let token

    let isVaild=false

    if(global.token) {

        isVaild=await isTokenVaild(global.token)

    }

    return new Promise(async(resolve,reject)=>{

        // 判断token是否失效/过期

        if(isVaild) {

            token=global.token

        }else{

            // 失效/过期/不存在重新获取

            await request({

                uri:`https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=${config.APP_ID}&secret=${config.APP_SECRET}`,

                json:true

            }).then((res)=>{

                console.log(res)

                res.expires_in=parseInt(newDate().getTime()/1000)+res.expires_in// access_token过期时间

                console.log('存储token',res)

                // 存储token

                token={

                    access_token:res.access_token,

                    expires_in:res.expires_in

                }

                global.token=token

            })

            .catch(function(err) {

                console.log(err)

            })

        }

        console.log('最后的access_token',token.access_token)

        resolve(token.access_token)

    })

}

// 获取有效的formId

function getFormId(userId) {

    return new Promise(async(resolve,reject)=>{

        let form=await TemplateMsg.findOne({userId:userId})

        let formId

        if(form.formIdList.length<0) {

            resolve(false)

        }

        let updateFormIdList=JSON.parse(JSON.stringify(form.formIdList))

        let isExist=false

        form.formIdList.forEach((item,index)=>{

            if(item.expire<=parseInt(newDate().getTime())/1000) {

            // 过期

            console.log('过期',item)

            updateFormIdList.splice(index,1)

            }else{

            // 没过期

                isExist=true

                formId=item.formId

                updateFormIdList.splice(index,1)

                return

            }

        })

        console.log('updateFormIdList',updateFormIdList)

        // 更新去除过期的formId和将被用掉的formId

        let update=await TemplateMsg.updateOne({userId:userId},{$set:{

            formIdList:updateFormIdList

        }})

        if(update.ok===1&&isExist) {

            resolve(formId)

        }else{

            resolve(false)

        }

    })

}

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