Uni-push 推送实现全过程 (2022 最新 最全 最细)
Uni-push 推送实现全过程
序言:
公司使用的uniapp开发的项目需要推送功能,看了下官方文档,经过几天努力,打通了推送功能。下图是ios和安卓(oppo和荣耀手机),在线或离线获取的推送通知。
简介
UniPush是DCloud联合个推公司推出的集成型统一推送服务,内建了苹果、华为、小米、OPPO、VIVO、魅族、谷歌FCM等手机厂商的系统级推送和个推等第三方推送。
第一步:开通UniPush推送服务
云端打包
应用开通 Uni Push 功能以后,只需要在 HBuilderX 中打开manifest.json,选择“App SDK配置”,向下找到推送功能配置,勾选 “DCloud UniPush” 即可。
打开你的项目App模块配置,点击配置进入DCloud开发者中心。
应用开通 Uni Push 功能时,需要提交应用相关信息,如下图所示:
**注意:**在申请开通时,需要确保输入的 Android包名
或 iOS Bundle ID
必须与打包时配置的一致,否则可能会导致无法收到推送消息。
如果已经开通 Uni Push,会看到如下页面:
iOS推送证书配置
iOS 平台还需要在 “配置管理”-“应用配置” 页面上传推送证书
如何获取推送证书请参考个推官方文档教程 iOS证书配置指南
厂商推送参数设置(可选,应用离线时可提高推送成功率)
Uni Push 集成并统一了各个手机厂商的系统级推送,目前支持魅族、OPPO、华为、小米、VIVO。如果需要使用厂商推送,需要先在各厂商开发者平台申请。
申请通过后厂商会提供推送的相关参数,需要将这些参数配置到 DCloud 开发者中心后台。
厂商参数配置页面如下图所示:
公司App也是上架了oppo,华为和小米的平台,所以这三个厂商推送平台都有填写。可以通过打开各自厂商的推送后台获取到这些信息,
到此移动端的配置都已经完成!
第二步:服务端下发推送消息
使用代码通过服务端接口推送消息
服务端集成时首先需要获取AppId、AppKey、MasterSecret参数,登录DCloud开发者中心,在“Uni Push”下的“应用配置”页面中获取,如下图所示:
如果使用uniCloud开发服务器
已经有封装好的易用插件负责服务器端开发,见:https://ext.dcloud.net.cn/plugin?id=1680
如果使用传统服务器开发
参考“个推·消息推送”的服务端快速集成文档http://docs.getui.com/
通过服务端接口下发推送消息,需要封装个推侧Rest-V2接口。应用在前台,透传内容若使用Dcloud侧要求的格式,则安卓客户端不进透传receive回调,自动创建通知栏通知;IOS开发者则需要在客户端receive透传回调中自己处理创建本地通知;应用在后台使用厂商通道需配置push_channel中的相关参数才可生效。
以下是Java-sdk封装Rest-V2厂商推送示例:
public class push2 { public static void main(String[] args) { GtApiConfiguration apiConfiguration = new GtApiConfiguration(); //填写应用配置 apiConfiguration.setAppId(""); apiConfiguration.setAppKey(""); apiConfiguration.setMasterSecret(""); apiConfiguration.setDomain("https://restapi.getui.com/v2/"); // 实例化ApiHelper对象,用于创建接口对象 ApiHelper apiHelper = ApiHelper.build(apiConfiguration); // 创建对象,建议复用。目前有PushApi、StatisticApi、UserApi PushApi pushApi = apiHelper.creatApi(PushApi.class); //根据cid进行单推 PushDTO<Audience> pushDTO = new PushDTO<Audience>(); // 设置推送参数 pushDTO.setRequestId(System.currentTimeMillis() + "");//requestid需要每次变化唯一 //配置推送条件 // 1: 表示该消息在用户在线时推送个推通道,用户离线时推送厂商通道; // 2: 表示该消息只通过厂商通道策略下发,不考虑用户是否在线; // 3: 表示该消息只通过个推通道下发,不考虑用户是否在线; // 4: 表示该消息优先从厂商通道下发,若消息内容在厂商通道代发失败后会从个推通道下发。 Strategy strategy=new Strategy(); strategy.setDef(1); Settings settings=new Settings(); settings.setStrategy(strategy); pushDTO.setSettings(settings); settings.setTtl(3600000);//消息有效期,走厂商消息需要设置该值
//推送苹果离线通知标题内容 Alert alert=new Alert(); alert.setTitle("苹果离线通知栏标题"); alert.setBody("苹果离线通知栏内容"); Aps aps = new Aps(); //1表示静默推送(无通知栏消息),静默推送时不需要填写其他参数。 //苹果建议1小时最多推送3条静默消息 aps.setContentAvailable(0); aps.setSound("default"); aps.setAlert(alert); IosDTO iosDTO = new IosDTO(); iosDTO.setAps(aps); iosDTO.setType("notify"); PushChannel pushChannel = new PushChannel(); pushChannel.setIos(iosDTO); //安卓离线厂商通道推送消息体 PushChannel pushChannel = new PushChannel(); AndroidDTO androidDTO = new AndroidDTO(); Ups ups = new Ups(); ThirdNotification notification1 = new ThirdNotification();; ups.setNotification(notification1); notification1.setTitle("安卓离线展示的标题"); notification1.setBody("安卓离线展示的内容"); notification1.setClickType("intent"); notification1.setIntent("intent:#Intent;launchFlags=0x04000000;action=android.intent.action.oppopush;component=io.dcloud.HBuilder/io.dcloud.PandoraEntry;S.UP-OL-SU=true;S.title=测试标题;S.content=测试内容;S.payload=test;end"); //各厂商自有功能单项设置
//ups.addOption("HW", "/message/android/notification/badge/class", "io.dcloud.PandoraEntry "); //ups.addOption("HW", "/message/android/notification/badge/add_num", 1); //ups.addOption("HW", "/message/android/notification/importance", "HIGH");
//ups.addOption("VV","classification",1); androidDTO.setUps(ups); pushChannel.setAndroid(androidDTO); pushDTO.setPushChannel(pushChannel); // PushMessage在线走个推通道才会起作用的消息体 PushMessage pushMessage = new PushMessage(); pushDTO.setPushMessage(pushMessage); pushMessage.setTransmission(" {title:\"标题\",content:\"内容\",payload:\"自定义数据\"}"); // 设置接收人信息 Audience audience = new Audience(); pushDTO.setAudience(audience); audience.addCid("请输入clientid"); // 进行cid单推 ApiResult<Map<String, Map<String, String>>> apiResult = pushApi.pushToSingleByCid(pushDTO); if (apiResult.isSuccess()) { // success System.out.println(apiResult.getData()); } else { // failed System.out.println("code:" + apiResult.getCode() + ", msg: " + apiResult.getMsg()); } }
}
注意:把$intent变量赋值字符串中的io.dcloud.HBuilder换成自己应用的包名
完整服务端厂商推送教程参考:
https://docs.getui.com/getui/server/rest_v2/common_args/
项目推送逻辑
项目需求是针对用户在审核过程中下发到指定某人,此人就可以在手机上获得推送消息通知,以便不会错过审核信息。逻辑也比较简单,App端获取到cid,通过登录接口发给服务端,服务端再通过集成的个推sdk把对应的cid,下发给App端,来完成交互。至于你需要个推,批量推,还是群推,可根据自己业务需求来定。
不同的推送规则可以参照教程查看
第三步:客户端处理推送消息
UniPush推送服务已经封装好iOS&Android平台的原生集成工作,开发者只需要调用JS代码处理推送消息的业务逻辑:
在App.vue页面:
onLaunch: function() {// #ifdef APP-PLUS/* 接收消息通知并跳转到tab页面 */const _self = this;const _handlePush = function(message) {_self.updatePushMessage(message);};plus.push.addEventListener('click', function(message) {plus.nativeUI.toast('push click');_handlePush(message);uni.reLaunch({url: '/pages/tab/tab'});}, true);plus.push.addEventListener('receive', function(message) {plus.nativeUI.toast('push receive');_handlePush(message);});// 获取App端cidlet pinf = plus.push.getClientInfo();if (pinf && pinf.clientid) uni.setStorageSync('cid', pinf.clientid);else {var obtainingCIDTimer = setInterval(function() {pinf = plus.push.getClientInfo();if (pinf.clientid) {uni.setStorageSync('cid', pinf.clientid);clearInterval(obtainingCIDTimer);}}, 50);}setTimeout(() => {console.log('cid', uni.getStorageSync('cid'));}, 500);},
两个监听click很明显,就是点击时候出触发的。
receive有两种情况会触发
1.ios应用已经打开的情况,这种情况通知栏不会有消息。可以自己写这种情况的处理逻辑,一般会弹出一个弹窗问需不需要跳转,我的方式是用plus.push.createMessage本地创建一条消息。
2.android接收到不符合格式的推送(不符合{title:“xxxx”,content:“xxxx”,payload:“xxxxx”}),这个是服务端来控制的。
有可能取不到clientId,或者为‘undefined’ ‘null’ 等字符串,可以通过定时器获取,参照上图代码。
获取cid码特别注意一点,真机调试时要在自定义打包基座中获取,否则获取的cid码无效!
第四步:bug问题分析
过程中遇到的问题
1.服务端推送ios接收不到消息,ios or android cannot be null报错
2.安卓点击消息图标不能唤醒应用
3.ios上桌面图标显示角标, 并且不消失
解决方法
1.可能是没有仔细看个推文档,感觉参数都对也不缺,试了半小时都接收不到,最后只能加个推技术微信
然后就顺利解决了,所以说直接问技术总比瞎浪费时间好。
2.离线推送时,安卓点击推送消息图标,没有任何反应,也无法唤醒App。后面查看文档,发现需要payload参数
3.ios角标的小红点一直存在,对于强迫症真的很难受,而且点击过角标红点一直都在,重新卸载也没有办法。后面还是用了代码来解决
var UIApplication = plus.ios.import("UIApplication");var app = UIApplication.sharedApplication();//获取应用图标的数量var oldNum = app.applicationIconBadgeNumber();if(oldNum!=0){console.log("oldNum:"+oldNum);var newNum = oldNum - 1;console.log("newNum:"+newNum);//设置应用图标的数量plus.runtime.setBadgeNumber(newNum);//导入个推原生类var GeTuiSdk = plus.ios.importClass('GeTuiSdk');GeTuiSdk.setBadge(newNum);}
总结
实现简单的uni-push并不是很难,都是集成封装好的, 代码量也不多,官网也有很详细的文章,就是开通各大厂商步骤繁琐,需要些耐心,遇到bug还是仔细多看几遍文档吧。此篇是博主一点点经验分享,也希望能对大家有点点帮助,谢谢!
推送功能官方相关资源
客户端调用的js API见:https://www.html5plus.org/doc/zh_cn/push.html
如果使用传统服务器开发,文档仍然是个推的服务器文档http://docs.getui.com/
检查应用是否被授予推送权限:https://ext.dcloud.net.cn/plugin?id=594
开启关闭推送服务:https://ext.dcloud.net.cn/plugin?id=727
自定义iOS推送铃声:https://ext.dcloud.net.cn/plugin?id=690
如何自定义推送通知的图标:https://ask.dcloud.net.cn/article/35537
Uni-push 推送实现全过程 (2022 最新 最全 最细)相关推荐
- 一篇文章带你了解APP PUSH推送机制
本文为PMCAFF专栏作者卓别木出品 写作目的: 本文主要讲解关于APP PUSH的流程.机制及相关经验,一是为了方便各位可以针对APP迅速制定PUSH消息推送方案,实现0到1的推送功能搭建,二是可以 ...
- iPhone 的 Push(推送通知)功能原理浅析
第一部分:Push原理 (以下绝大多数内容参考自.图片来自iPhone OS Reference Library) 机制简介 Push 的工作机制可以简单的概括为下图 图中, Provider是指某个 ...
- 变更数据推送java_idea 团队成员修改工程后push推送
idea 团队成员修改工程后push推送 当团队成员修改了工程后,可以进行commit和push操作: 比如我们代码里,加了一段输出: 我们先把项目提交到本地库: 右击项目 -> Git -&g ...
- 为什么 PUSH 推送要经常背锅?
来源 | Java3y 头图 | CSDN付费下载自视觉中国 自从做了推送以后,每隔一段时间就发现有各大的公司推送事故出现. 你问我做开发的慌不慌,我当然慌得一批了. 为什么经常会有推送事故? 为什么 ...
- push推送服务设计
PUSH系统架构设计简述 一.网络传输协议的选择 PUSH系统协议选取: UDP协议实时性更好,但是如何处理安全可靠的传输并且处理不同客户端之间的消息交互是个难题,实现起来过于复杂,那就非TCP协议莫 ...
- iPhone的Push(推送通知)功能原理浅析
来自http://blog.csdn.net/omgle/archive/2011/06/29/6574313.aspx 第一部分:Push原理 (以下绝大多数内容参考自.图片来自iPhone OS ...
- 58同城高性能移动Push推送平台架构演进之路
本文详细讲述58同城高性能移动Push推送平台架构演进的三个阶段,并介绍了什么是移动Push推送,为什么需要,原理和方案对比:移动Push推送第一阶段(单平台)架构如何设计:移动Push推送典型性能问 ...
- 2021-06-10-APP PUSH推送机制
APP PUSH推送机制 一.APP PUSH定义与价值 二.APP推送分类 三. PUSH流程 四.底层通道说明 五.下发推送 六.数据上报 七.PUSH特点 八.触达率的提升 一.APP PUSH ...
- iPhone的Push(推送通知)功能原理浅析[转]
转自:http://xiaolife.com/wordpress/an-introduce-to-iphone-push/ 第一部分:Push原理 (以下绝大多数内容参考自.图片来自iPhone OS ...
最新文章
- 【单调队列】【DP】城市交通(jzoj 1749)
- java 可视化_可视化Java 9模块关系
- 【渝粤教育】 国家开放大学2020年春季 2411中国现代文学 参考试题
- linux 解压tar.jz,linux系统压缩文件和解压缩命令
- require mysql.php_require和include经典一例抛析_php
- frameset标签设计页面
- [工具] PicGo + smms 构建图床
- CF 317D Game with Powers
- Windows7开机加速全攻略
- Iterator死循环
- python太阳花画法_Python——教你画朵太阳花
- Unable to read entire header,0 bytes read;expected 512 bytes
- 深刻认识差模电压和共模电压
- vue文件在服务器上乱码,解决vue-pdf查看pdf文件及打印乱码的问题
- 用大白话聊聊JavaSE -- 如何理解Java Bean
- 基于机智云平台的厨房智能监控系统
- 算法题|-灯泡开关问题
- 网站图片怎么优化搜索排名
- 计算机毕业设计Java物流信息管理系统录像演示(源码+系统+mysql数据库+Lw文档)
- 达人实测:天玑1000和骁龙765g哪个好-天玑1000和骁龙765g对比跑分
热门文章
- windows系统搭建WEB服务器(IIS)
- JZOJ6384【NOIP2019模拟2019.10.23】珂学家
- Linux上利用nginx搭建一个简单的rtmp视频流服务器(不涉及直播)
- php怎么将字符串转为数字类型,利用PHP怎么将字符串转换成数字
- chrony服务配置
- CAN总线的数据解析
- 3dMax利用粒子流创建烟花效果详细教程
- Could not parse multipart servlet request; nested exception is java.io.IOException: The temporary up
- H5 自动播放背景音乐
- vrPlus(神之眼)拓展屏(裸眼和左右)+zSpace方案