微信小程序订阅

摘要

1.基于promise封装微信小程序订阅

2.解决由于微信基础库版本低下的兼容

3.解决“总是保持以上选择,不再询问”的取消状态

主要运用API:

requestSubscribeMessage
getSetting
openSetting

主要功能:

用户强制订阅,无论怎么取消都能订阅,总代码在下方,具体流程思路有耐心可以往下看...,可以拷贝直接用。引入即可,当然我比较菜,大家可以再改改。

总代码

/**
* @desc 小程序订阅功能
* @param {Array} tmplIds 订阅列表
* @returns {Promise} 返回promise
*/ // 永久关闭订阅则代表点击了我不在接受此消息,再次订阅是无法弹起授权窗口的。普通关闭时可以再次弹起授权窗口。 export const Subscribe = (tmplIds = []) => {
return new Promise((resolve, reject) => {
// 判断是否为微信小程序,不是的不做订阅进行跳过
let isWx = false;
// #ifdef MP-WEIXIN
isWx = true;
// #endif
if (!isWx) resolve(1);
console.log('进来了哟')
// 判断基本库是否在2.8.3,低于的暂时不做订阅进行跳过
const versionCan = compareVersion("2.8.3");
if (versionCan === -1) resolve(1);
// 主流程
requestSubscribe(tmplIds, resolve, reject)
}) }
// 申请订阅功能
function requestSubscribe(tmplIds, resolve, reject) {
uni.requestSubscribeMessage({
tmplIds,
success: async res => {
// 检查订阅数量
let checkSubscribeBool = await checkSubscribeAll(tmplIds, res);
if (checkSubscribeBool) {
// 用户完成订阅
console.log("[用户完成订阅]")
resolve(1);
} else {
// 跳去检查永久关闭订阅还是普通关闭订阅
guidSubscribeMessageAuthAfter(tmplIds, resolve, reject);
}
},
fail: res => {
console.log(res, "订阅,失败");
if (res.errCode == 20004) {
// console.log(res, 'fail:用户关闭了主开关,无法进行订阅,引导开启---');
guideOpenSubscribeMessage(tmplIds, resolve, reject);
}
return
}
})
} // 计算用户订阅消息的数量
function checkSubscribeAll(tmplIds, res) {
// 将accept的生成一个数组,判断申请的订阅消息是不是已经订阅消息的子集
let arr = [];
for (const key of Object.keys(res)) {
if (res[key] === 'accept') {
arr.push(key);
}
}
if (arr.length == tmplIds.length) {
console.log('订阅完毕')
return true
} else {
console.log('没订阅或者少订阅')
return false
} } // 检查用户是否授权完毕(检查时永久关闭还是普通关闭)
function guidSubscribeMessageAuthAfter(tmplIds, resolve, reject) {
uni.getSetting({
withSubscriptions: true,
success: async res => {
let {
authSetting = {},
subscriptionsSetting: { mainSwitch = false, itemSettings = {} } = {}
} = res;
if (Object.keys(itemSettings).length == 0) { // 这种情况是普通关闭
uni.showModal({
title: "温馨提示",
content: "同意订阅才能及时获知完成进度领取优惠",
confirmText: "重新订阅",
cancelText: "我再看看",
success: res => {
if (res.confirm) {
// 重新调起授权订阅
requestSubscribe(tmplIds, resolve, reject);
} else if (res.cancel) {
//没成功订阅,返回reject
reject(2);
}
}
});
} else { // 这种是订阅成功或永久关闭
let checkSubscribeBool = await checkSubscribeAll(tmplIds, itemSettings);
if (
authSetting["scope.subscribeMessage"] ||
(mainSwitch && checkSubscribeBool)
) {
//成功
console.log("用户手动开启同意了,订阅消息");
resolve(1);
} else {
//失败,永久关闭
guideOpenSubscribeMessage(tmplIds, resolve, reject);
}
}
}
});
} //引导用户重新授权(永久关闭的方法)
function guideOpenSubscribeMessage(tmplIds, resolve, reject) {
// console.log(resolve, reject, 'rescovavasr1')
uni.showModal({
title: "温馨提示",
content: "检测到您没有开启全部订阅消息的权限,是否去设置?",
success: res => { if (res.confirm) {
uni.openSetting({
success: res => {
// 在检查是否全部订阅完毕
guidSubscribeMessageAuthAfter(tmplIds, resolve, reject);
}
});
} else if (res.cancel) {
// console.log(resolve, reject, 'rescovavasr2')
uni.showModal({
title: "温馨提示",
content: "同意订阅才能及时获知完成进度领取优惠",
showCancel: false,
confirmText: "我知道了"
});
reject(2);
}
}
});
} // 比较版本号
function compareVersion(v2) {
let { SDKVersion: v1 } = uni.getSystemInfoSync();
v1 = v1.split(".");
v2 = v2.split(".");
const len = Math.max(v1.length, v2.length); while (v1.length < len) {
v1.push("0");
}
while (v2.length < len) {
v2.push("0");
} for (let i = 0; i < len; i++) {
const num1 = parseInt(v1[i]);
const num2 = parseInt(v2[i]); if (num1 > num2) {
return 1;
} else if (num1 < num2) {
return -1;
}
} return 0;
}

思路流程解析

这是第一步,封装promise,返回订阅成功和订阅失败的回调。以及判断版本号是否低于2.8.3,低于2.8.3是只能使用一个模板id的,可以考虑做兼容。这边内部原因就跳过不做订阅。然后走入主流程requestSubscribe(tmplIds, resolve, reject),带入模板id和俩个回调

   return new Promise((resolve, reject) => {
// 判断是否为微信小程序,不是的不做订阅进行跳过
let isWx = false;
// #ifdef MP-WEIXIN
isWx = true;
// #endif
if (!isWx) resolve(1);
console.log('进来了哟')
// 判断基本库是否在2.8.3,低于的暂时不做订阅进行跳过
const versionCan = compareVersion("2.8.3");
if (versionCan === -1) resolve(1);
// 主流程
requestSubscribe(tmplIds, resolve, reject)
})

这是第二步,订阅api。api走通就success,报错走fail。 主要看success,返回一个res,大家可以打印看看,里面就一个对象,键是id,订阅成功的值是accept,执行一波判断数量函数,符合长度说明用户全部订阅,就放行。

当然点击了“总是保持以上选择,不再询问”也是走到这里面,然后值都是订阅失败的。这里走去检查看看是否点击了“总是保持以上选择,不再询问”。

// 申请订阅功能
function requestSubscribe(tmplIds, resolve, reject) {
uni.requestSubscribeMessage({
tmplIds,
success: async res => {
// 检查订阅数量
let checkSubscribeBool = await checkSubscribeAll(tmplIds, res);
if (checkSubscribeBool) {
// 用户完成订阅
console.log("[用户完成订阅]")
resolve(1);
} else {
// 跳去检查永久关闭订阅还是普通关闭订阅
guidSubscribeMessageAuthAfter(tmplIds, resolve, reject);
}
},
fail: res => {
console.log(res, "订阅,失败");
if (res.errCode == 20004) {
// console.log(res, 'fail:用户关闭了主开关,无法进行订阅,引导开启---');
guideOpenSubscribeMessage(tmplIds, resolve, reject);
}
return
}
})
}

第三步 getsettingApi,检查用户是否点击了“总是保持以上选择,不再询问”,判断的条件就是返回的res有没有itemSettings,当然大家可以找俩种情况比对一下就知道了,永久关闭的是有的那个对象的,普通关闭是没有的。

判断这个目的,是可以直接在调起订阅界面。可以引导用户重新点击。当然如果调不起就走guideOpenSubscribeMessage函数。

// 检查用户是否授权完毕(检查时永久关闭还是普通关闭)
function guidSubscribeMessageAuthAfter(tmplIds, resolve, reject) {
uni.getSetting({
withSubscriptions: true,
success: async res => {
let {
authSetting = {},
subscriptionsSetting: { mainSwitch = false, itemSettings = {} } = {}
} = res;
if (Object.keys(itemSettings).length == 0) { // 这种情况是普通关闭
uni.showModal({
title: "温馨提示",
content: "同意订阅才能及时获知完成进度领取优惠",
confirmText: "重新订阅",
cancelText: "我再看看",
success: res => {
if (res.confirm) {
// 重新调起授权订阅
requestSubscribe(tmplIds, resolve, reject);
} else if (res.cancel) {
//没成功订阅,返回reject
reject(2);
}
}
});
} else { // 这种是订阅成功或永久关闭
let checkSubscribeBool = await checkSubscribeAll(tmplIds, itemSettings);
if (
authSetting["scope.subscribeMessage"] ||
(mainSwitch && checkSubscribeBool)
) {
//成功
console.log("用户手动开启同意了,订阅消息");
resolve(1);
} else {
//失败,永久关闭
guideOpenSubscribeMessage(tmplIds, resolve, reject);
}
}
}
});
}

第四步,用户永久关闭了订阅,我们可以用openSetting打开。这时候订阅消息是会出来没订阅的模板的。重新引导...这样就结束了这个封装啦!

//引导用户重新授权(永久关闭的方法)
function guideOpenSubscribeMessage(tmplIds, resolve, reject) {
// console.log(resolve, reject, 'rescovavasr1')
uni.showModal({
title: "温馨提示",
content: "检测到您没有开启全部订阅消息的权限,是否去设置?",
success: res => { if (res.confirm) {
uni.openSetting({
success: res => {
// 在检查是否全部订阅完毕
guidSubscribeMessageAuthAfter(tmplIds, resolve, reject);
}
});
} else if (res.cancel) {
// console.log(resolve, reject, 'rescovavasr2')
uni.showModal({
title: "温馨提示",
content: "同意订阅才能及时获知完成进度领取优惠",
showCancel: false,
confirmText: "我知道了"
});
reject(2);
}
}
});
}
 
 
 

大家觉得有什么奇怪的,欢迎和我讨论

 

微信小程序订阅的更多相关文章

  1. 微信小程序订阅消息,我踩过的坑都在这里了!

    旧的模板消息将在 2020 年 1 月 10 号全面下架,也就是今天,不过貌似现在还可以用!!!我已经改好了,只不过还没有上线,准备坚持到最后一天! 0.订阅消息 简单介绍一下订阅消息的特点: 用户授 ...

  2. 微信小程序订阅消息

    概述 消息能力是小程序能力中的重要组成,我们为开发者提供了订阅消息能力,以便实现服务的闭环和更优的体验. 订阅消息推送位置:服务通知 订阅消息下发条件:用户自主订阅 订阅消息卡片跳转能力:点击查看详情 ...

  3. 微信小程序订阅消息调研

    相关资料 背景:微信模板消息已正式下架,改为订阅消息,详情如下: 服务变更通知 订阅消息:订阅消息相关内容如下: 订阅消息 接口设计 获取接口访问凭证 :根据appId和secret获取接口访问凭证a ...

  4. 微信小程序、应用号、订阅号、服务号、企业号小总结

    微信小程序是现在微信推出的一个新的项目,但是很多人都不是很清楚微信小程序是怎么一回事,不明白到底怎样分别微信小程序和别的公众号.订阅号等的区别,那么让小编来给你介绍一下. 微信小程序目前是内侧阶段,是 ...

  5. 微信小程序如何发送订阅消息,正确姿势来了,建议收藏!

    小程序订阅消息公测已经有些日子,今天以世界上最好的语言(PHP)为例,说一下如何发送订阅消息. 1.订阅消息 其实如果用过模板消息的话,改用订阅消息挺简单的,看一下官方文档稍加摸索就能使用. 但是对于 ...

  6. 微信小程序的模板消息与小程序订阅消息

    小程序订阅消息 功能介绍 消息能力是小程序能力中的重要组成,我们为开发者提供了订阅消息能力,以便实现服务的闭环和更优的体验. 订阅消息推送位置:服务通知 订阅消息下发条件:用户自主订阅 订阅消息卡片跳 ...

  7. .netcore 3.1 C# 微信小程序发送订阅消息

    一.appsettings.json定义小程序配置信息 "WX": { "AppId": "wx88822730803edd44", &qu ...

  8. 来自于微信小程序的一封简讯

    9月21晚间,微信向部分公众号发出公众平台-微信应用号(小程序)的内测邀请,向来较为低调的微信在这一晚没人再忽视它了. 来自个人博客:Damonare的个人博客 一夜之间火了的微信应用号你真的知道吗? ...

  9. 张小龙宣布微信小程序1月9日发布,并回答了大家最关心的8个问题

    2016 年 12 月 28 日,张小龙在微信公开课 PRO 版的会场上,宣布了微信小程序的正式发布时间. 微信小程序将于 2017 年 1 月 9 号正式上线. 同时他解释称,小程序就像PC时代的网 ...

随机推荐

  1. 图论--网络流--最大流 HDU 3572 Task Schedule(限流建图,超级源汇)

    Problem Description Our geometry princess XMM has stoped her study in computational geometry to conc ...

  2. idea配置tomcat运行参数,防止中文乱码和内存问题

    点击选择 Edit Configurations... 修改 VM options ,参数在最后可直接复制 -Xms550m -Xmx1250m -Dfile.encoding=UTF-8

  3. B. Sorted Adjacent Differences(思维构造)

    \(给出n个数字,要求构造一个由这n个数组成的序列,使得|a_1-a_2|<=|a_2-a_3|...<=|a_{n-1}-a_n|\) \(排序后,从数列中间取个数,然后从左右分别循环取 ...

  4. Linux文件操作命令并举例说明其作用

    ls ,常用于查看当前文件下有工作中需要的文件 cd, 常用于进行切换文件的位置 vim,常用于编辑软件系统相关的配置文件 ps –ef|grep jdk,常用语显示跟jdk有关的进程   |:表示 ...

  5. Dockerfile-Namespace

    Docker核心-Namespaces(命名空间) 1)概念: 命令空间是Linux内核的一个强大的特性.每个容器都有自己单独的命令空间,运行在其中的应用都是独立在操作系用中运行一样.命名空间保证了容 ...

  6. Arthas 使用(二) —— 应用场景

    1. ognl获取bean SpringContextUtil,通常代码中会有类似这样的工具类用来获取 bean 实例 @Component public class SpringContextUti ...

  7. lrzsz-神一样的上传下载工具

    yum list lrzsz rz sz filename

  8. mysql+redis缓存策略常见的错误

    什么时候应该更新缓存 应该是从数据库读取数据后,再更新缓存,从缓存读取到数据,就不需要再重新写缓存了,一个常见的错误是,每次访问接口都更新缓存,这样的话,如果接口一直有流量,那么db中的数据,就一直没 ...

  9. spark机器学习从0到1逻辑斯蒂回归之(四)

      逻辑斯蒂回归 一.概念 逻辑斯蒂回归(logistic regression)是统计学习中的经典分类方法,属于对数线性模型.logistic回归的因变量可以是二分类的,也可以是多分类的.logis ...

  10. 微信小程序实战篇-电商(一)

    我想大家对电商一定不陌生,一般电商的底部导航栏有以下几个首页.分类.购物车.个人中心.所以我们按照这个来做吧. app.json是用来配置page路径以及导航栏属性的,那我们要做首页.分类.购物车.个 ...