热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

uniapp技术分享|uniapp转小程序实时消息

uni-app技术分享|uni-app转小程序-实时消息-微信小程序实现实时消息与uniapp转码成微信小程序实现实时消息两者是一样的,区别仅仅是一个是原生小程序一个是uniapp

微信小程序 实现实时消息与 uniapp 转码成微信小程序 实现实时消息两者是一样的,区别仅仅是一个是原生小程序一个是 uniapp 转码成小程序。
本文主要简单实现点对点消息与呼叫邀请等相关功能实现。
uniapp转码成小程序逻辑与小程序逻辑基本一致。

引入 RTM SDK

使用 web RTM-SDK 即可,小程序的实时消息与 WEB 的实时消息共用 SDK。

使用 1.0.5 版本,否则转成小程序时可能会无法正常使用
npm i ar-rtm-sdk@1.0.5
// 引入 anyRTM 
import ArRTM from "ar-rtm-sdk"

代码封装

你可以在页面里边直接调用,也可单独封装成一个 js。
本人喜欢将 RTM 封装成一个 js 文件。

本地存放

const Store = {
    // RTM 客户端
    rtmClient: null,

    // 主叫邀请实例
    localInvitation: null,
    // 被叫收到的邀请实例
    remoteInvitation: null,
}

回调封装

本文仅进行简单封装,如需更复杂逻辑请自行更改。

// RTM 监听事件
const rtmEvent = {
    // 主叫:被叫已收到呼叫邀请
    localInvitationReceivedByPeer: () => {
        uni.hideToast();
        uni.showToast({
            title: '被叫已收到呼叫邀请',
            icon: 'none',
            duration: 2000,
            mask: true,
        });

    },
    // 主叫:被叫已接受呼叫邀请
    localInvitationAccepted: async (response) => {
        console.log("主叫:被叫已接受呼叫邀请", response);
        uni.hideToast();
        uni.showToast({
            title: '被叫接受呼叫邀请',
            icon: 'none',
            duration: 2000,
            mask: true,
        });

    },
    // 主叫:被叫拒绝了你的呼叫邀请
    localInvitationRefused: (response) => {
        console.log("主叫:被叫拒绝了你的呼叫邀请", response);
        uni.hideToast();
        uni.showToast({
            title: '被叫拒绝呼叫邀请',
            icon: 'none',
            duration: 2000,
            mask: true,
        });
    },
    // 主叫:呼叫邀请进程失败
    localInvitationFailure: (response) => {
        console.log("主叫:呼叫邀请进程失败", response);
        uni.hideToast();
        uni.showToast({
            title: '呼叫邀请失败',
            icon: 'error',
            duration: 2000,
            mask: true,
        });
    },
    // 主叫:呼叫邀请已被成功取消 (主动挂断)
    localInvitationCanceled: () => {
        console.log("主叫:呼叫邀请已被成功取消 (主动挂断)");
    },

    // 被叫:监听收到来自主叫的呼叫邀请
    RemoteInvitationReceived: async (remoteInvitation) => {
        console.log("监听收到来自主叫的呼叫邀请", remoteInvitation);
        // 监听回调
        rtmInternal.inviteProcessing(remoteInvitation)
        // 显示模态弹窗
        uni.showModal({
            title: '提示',
            content: '收到来自主叫的呼叫邀请',
            cancelText: '拒绝',
            confirmText: '接听',
            success: function(res) {
                if (res.confirm) {
                    console.log('用户点击确定');
                    remoteInvitation.accept();
                } else if (res.cancel) {
                    console.log('用户点击取消');
                    remoteInvitation.refuse();
                }
            }
        });
    },
    // 被叫:监听接受呼叫邀请
    RemoteInvitationAccepted: async () => {
        console.log("被叫 接受呼叫邀请");
        uni.hideToast();
        uni.showToast({
            title: '接受呼叫邀请',
            icon: 'success',
            duration: 2000,
            mask: true,
        });
    },
    // 被叫:监听拒绝呼叫邀请
    RemoteInvitationRefused: () => {
        console.log("被叫 拒绝呼叫邀请");
        uni.hideToast();
        uni.showToast({
            title: '拒绝呼叫邀请',
            icon: 'success',
            duration: 2000,
            mask: true,
        });
    },
    // 被叫:监听主叫取消呼叫邀请
    RemoteInvitationCanceled: () => {
        console.log("主叫 取消呼叫邀请");
        uni.hideToast();
        uni.showToast({
            title: '主叫取消呼叫',
            icon: 'success',
            duration: 2000,
            mask: true,
        });
    },
    // 被叫:监听呼叫邀请进程失败
    RemoteInvitationFailure: () => {
        console.log("被叫 呼叫邀请进程失败");
        uni.hideToast();
        uni.showToast({
            title: '呼叫邀请失败',
            icon: 'error',
            duration: 2000,
            mask: true,
        });
    },


    // 收到来自对端的点对点消息
    MessageFromPeer: (message, peerId) => {
        console.log("收到来自对端的点对点消息", message, peerId);
        uni.showToast({
            title: '收到' + peerId + '的点对点消息:' + message.text,
            icon: 'none',
            duration: 1000 * 5
        })

    },
    // 通知 SDK 与 RTM 系统的连接状态发生了改变
    ConnectionStateChanged: (newState, reason) => {
        console.log("系统的连接状态发生了改变", newState);

        switch (newState) {
            case "CONNECTED":
                uni.hideLoading();
                //  SDK 已登录 RTM 系统
                uni.showToast({
                    title: 'RTM 连接成功',
                    icon: 'success',
                    mask: true,
                })
                break;
            case "ABORTED":
                uni.showToast({
                    title: 'RTM 停止登录',
                    icon: 'error',
                    mask: true,
                });
                console.log("RTM 停止登录,重新登录");

                break;
            default:
                wx.showLoading({
                    title: 'RTM 连接中',
                    mask: true,
                })
                break;
        }
    }
}

登录 RTM 系统

所有与 rtm 相关操作都需要登录后才可正常使用
成功登录后即可监听相关回调
// 初始化
export const InItRtm = async (Config) => {
    // 创建 RTM 客户端
    Store.rtmClient = await ArRTM.createInstance(Config.AppID);

    // RTM 版本
    console.log("RTM 版本", ArRTM.VERSION);

    uni.showLoading({
        title: '登录中',
        mask: true
    })

    // 登录 RTM
    await Store.rtmClient.login({
        token: "",
        uid: Config.userId
    }).then(() => {
        uni.hideLoading();
        uni.showToast({
            title: '登录成功',
            icon: 'success',
            duration: 2000
        })
        console.log("登录成功");

        // 监听收到来自主叫的呼叫邀请
        Store.rtmClient.on(
            "RemoteInvitationReceived",
            rtmEvent.RemoteInvitationReceived
        );
        // 监听收到来自对端的点对点消息
        Store.rtmClient.on("MessageFromPeer", rtmEvent.MessageFromPeer);
        // 通知 SDK 与 RTM 系统的连接状态发生了改变
        Store.rtmClient.on(
            "ConnectionStateChanged",
            rtmEvent.ConnectionStateChanged
        );

    }).catch((err) => {
        Store.userId = "";
        uni.hideLoading();
        uni.showToast({
            icon: 'error',
            title: 'RTM 登录失败',
            mask: true,
            duration: 2000
        });
        console.log("RTM 登录失败", err);
    });



}

逻辑方法封装

// RTM 内部逻辑
export const rtmInternal = {
...
}

查询呼叫用户是否在线

// 查询呼叫用户是否在线
    peerUserQuery: async (uid) => {
        const oUserStatus = await Store.rtmClient.queryPeersOnlineStatus([uid]);
        if (!oUserStatus[uid]) {
            uni.showToast({
                title: '用户不在线',
                icon: 'error',
                duration: 2000,
                mask: true,
            });
            return false;
        }
        return true;
    },

发起呼叫

// 主叫发起呼叫
    inviteSend: async (peerUserId) => {

        Store.localInvitation = await Store.rtmClient.createLocalInvitation(
            peerUserId
        )
        // 设置邀请内容
        // Store.localInvitation.cOntent= JSON.stringify({});

        // 事件监听
        // 监听被叫已收到呼叫邀请
        Store.localInvitation.on(
            "LocalInvitationReceivedByPeer",
            rtmEvent.localInvitationReceivedByPeer
        );
        // 监听被叫已接受呼叫邀请
        Store.localInvitation.on(
            "LocalInvitationAccepted",
            rtmEvent.localInvitationAccepted
        );
        // 监听被叫拒绝了你的呼叫邀请
        Store.localInvitation.on(
            "LocalInvitationRefused",
            rtmEvent.localInvitationRefused
        );
        // 监听呼叫邀请进程失败
        Store.localInvitation.on(
            "LocalInvitationFailure",
            rtmEvent.localInvitationFailure
        );
        // 监听呼叫邀请已被成功取消
        Store.localInvitation.on(
            "LocalInvitationCanceled",
            rtmEvent.localInvitationCanceled
        );

        // 发送邀请
        Store.localInvitation.send();
    },

取消呼叫

发起者主动取消呼叫邀请

callCancel: () => {
        if (Store.localInvitation) {
            Store.localInvitation.cancel()
        }
    },

被叫邀请回调绑定

// 被叫收到呼叫邀请处理(给收到的邀请实例绑定事件)
    inviteProcessing: async (remoteInvitation) => {
        // 监听接受呼叫邀请
        remoteInvitation.on(
            "RemoteInvitationAccepted",
            rtmEvent.RemoteInvitationAccepted
        );
        // 监听拒绝呼叫邀请
        remoteInvitation.on(
            "RemoteInvitationRefused",
            rtmEvent.RemoteInvitationRefused
        );
        // 监听主叫取消呼叫邀请
        remoteInvitation.on(
            "RemoteInvitationCanceled",
            rtmEvent.RemoteInvitationCanceled
        );
        // 监听呼叫邀请进程失败
        remoteInvitation.on(
            "RemoteInvitationFailure",
            rtmEvent.RemoteInvitationFailure
        );
    },

点对点消息发送

// 发送消息
    sendMessage: (uid, message) => {
        console.log("发送消息", uid, message);
        Store.rtmClient && Store.rtmClient.sendMessageToPeer({
            text: JSON.stringify(message)
        }, uid).catch(err => {
            console.log("发送消息失败", err);
        });
    },

简单页面

html


        
            用户 ID:{{userId}}
        
        
            
            
            
            
                
                
            
            
            
                
                
                
                
            
            
            
                
                
                
            
        
    

js

    import {
        generateNumber
    } from "../../until/until.js"; // 生成随机数
    import {
        InItRtm,
        rtmInternal
    } from "../../until/rtm.js"
    export default {
        data() {
            return {
                page: 0,
                // 本地用户
                userId: '',
                // 远端用户
                peerId: '',
                // 发送的信息
                sendText: ''
            }
        },
        created() {
            // 用户 UID
            this.userId = generateNumber(4) + ''
        },
        methods: {
            /** 登录 RTM 系统 */
            async loginRTM() {
                const info = {
                    /** 
                     * 必填 anyRTC 为 App 开发者签发的 App ID。每个项目都应该有一个独一无二的 App ID。
                     * 如果你的开发包里没有 App ID,请从anyRTC官网(https://www.anyrtc.io)申请一个新的 App ID
                     */
                    AppID: '',
                    userId: this.userId
                }
                await InItRtm(info);
                this.page = 1
            },

            /** 呼叫邀请 */
            async invitationCall() {
                if (this.peerId === '') return uni.showToast({
                    title: '请输入远端用户',
                    icon: 'error',
                });
                if (this.peerId === this.userId) return uni.showToast({
                    title: '禁止远端用户与本地用户一致',
                    icon: 'none',
                });

                // 查询用户是否在线
                const state = await rtmInternal.peerUserQuery(this.peerId);
                if (state) {
                    rtmInternal.inviteSend(this.peerId)
                } else {
                    return uni.showToast({
                        title: '用户不在线',
                        icon: 'error',
                    });
                }
            },
            invitationCallOff() {
                rtmInternal.callCancel()
            },

            /** 发送消息 */
            async sendMessage() {
                if (this.peerId === '') return uni.showToast({
                    title: '请输入远端用户',
                    icon: 'error',
                });
                if (this.peerId === this.userId) return uni.showToast({
                    title: '禁止远端用户与本地用户一致',
                    icon: 'none',
                });
                if (this.sendText === '') return uni.showToast({
                    title: '请输入发送信息',
                    icon: 'error',
                });
                // 查询用户是否在线
                const state = await rtmInternal.peerUserQuery(this.peerId);
                if (state) {
                    rtmInternal.sendMessage(this.peerId, this.sendText)
                } else {
                    return uni.showToast({
                        title: '用户不在线',
                        icon: 'error',
                    });
                }
            }
        }
    }

style

.input_automatic {
        border: 1px solid;
        height: 40px;
        border-radius: 4px;
        padding: 0 12px;

    }


推荐阅读
  • 小程序自动授权和手动接入的方式及操作步骤
    本文介绍了小程序支持的两种接入方式:自动授权和手动接入,并详细说明了它们的操作步骤。同时还介绍了如何在两种方式之间切换,以及手动接入后如何下载代码包和提交审核。 ... [详细]
  • 本文介绍了Web学习历程记录中关于Tomcat的基本概念和配置。首先解释了Web静态Web资源和动态Web资源的概念,以及C/S架构和B/S架构的区别。然后介绍了常见的Web服务器,包括Weblogic、WebSphere和Tomcat。接着详细讲解了Tomcat的虚拟主机、web应用和虚拟路径映射的概念和配置过程。最后简要介绍了http协议的作用。本文内容详实,适合初学者了解Tomcat的基础知识。 ... [详细]
  • Java序列化对象传给PHP的方法及原理解析
    本文介绍了Java序列化对象传给PHP的方法及原理,包括Java对象传递的方式、序列化的方式、PHP中的序列化用法介绍、Java是否能反序列化PHP的数据、Java序列化的原理以及解决Java序列化中的问题。同时还解释了序列化的概念和作用,以及代码执行序列化所需要的权限。最后指出,序列化会将对象实例的所有字段都进行序列化,使得数据能够被表示为实例的序列化数据,但只有能够解释该格式的代码才能够确定数据的内容。 ... [详细]
  • [译]技术公司十年经验的职场生涯回顾
    本文是一位在技术公司工作十年的职场人士对自己职业生涯的总结回顾。她的职业规划与众不同,令人深思又有趣。其中涉及到的内容有机器学习、创新创业以及引用了女性主义者在TED演讲中的部分讲义。文章表达了对职业生涯的愿望和希望,认为人类有能力不断改善自己。 ... [详细]
  • 利用Visual Basic开发SAP接口程序初探的方法与原理
    本文介绍了利用Visual Basic开发SAP接口程序的方法与原理,以及SAP R/3系统的特点和二次开发平台ABAP的使用。通过程序接口自动读取SAP R/3的数据表或视图,在外部进行处理和利用水晶报表等工具生成符合中国人习惯的报表样式。具体介绍了RFC调用的原理和模型,并强调本文主要不讨论SAP R/3函数的开发,而是针对使用SAP的公司的非ABAP开发人员提供了初步的接口程序开发指导。 ... [详细]
  • flowable工作流 流程变量_信也科技工作流平台的技术实践
    1背景随着公司业务发展及内部业务流程诉求的增长,目前信息化系统不能够很好满足期望,主要体现如下:目前OA流程引擎无法满足企业特定业务流程需求,且移动端体 ... [详细]
  • 本文讨论了在openwrt-17.01版本中,mt7628设备上初始化启动时eth0的mac地址总是随机生成的问题。每次随机生成的eth0的mac地址都会写到/sys/class/net/eth0/address目录下,而openwrt-17.01原版的SDK会根据随机生成的eth0的mac地址再生成eth0.1、eth0.2等,生成后的mac地址会保存在/etc/config/network下。 ... [详细]
  • 本文讨论了在手机移动端如何使用HTML5和JavaScript实现视频上传并压缩视频质量,或者降低手机摄像头拍摄质量的问题。作者指出HTML5和JavaScript无法直接压缩视频,只能通过将视频传送到服务器端由后端进行压缩。对于控制相机拍摄质量,只有使用JAVA编写Android客户端才能实现压缩。此外,作者还解释了在交作业时使用zip格式压缩包导致CSS文件和图片音乐丢失的原因,并提供了解决方法。最后,作者还介绍了一个用于处理图片的类,可以实现图片剪裁处理和生成缩略图的功能。 ... [详细]
  • Imtryingtofigureoutawaytogeneratetorrentfilesfromabucket,usingtheAWSSDKforGo.我正 ... [详细]
  • 本文介绍了前端人员必须知道的三个问题,即前端都做哪些事、前端都需要哪些技术,以及前端的发展阶段。初级阶段包括HTML、CSS、JavaScript和jQuery的基础知识。进阶阶段涵盖了面向对象编程、响应式设计、Ajax、HTML5等新兴技术。高级阶段包括架构基础、模块化开发、预编译和前沿规范等内容。此外,还介绍了一些后端服务,如Node.js。 ... [详细]
  • 小程序wxs中的时间格式化以及格式化时间和date时间互转
    本文介绍了在小程序wxs中进行时间格式化操作的问题,并提供了解决方法。同时还介绍了格式化时间和date时间的互相转换的方法。 ... [详细]
  • 本文介绍了PHP常量的定义和使用方法,包括常量的命名规则、大小写敏感性、全局范围和标量数据的限制。同时还提到了应尽量避免定义resource常量,并给出了使用define()函数定义常量的示例。 ... [详细]
  • 本文介绍了关系型数据库和NoSQL数据库的概念和特点,列举了主流的关系型数据库和NoSQL数据库,同时描述了它们在新闻、电商抢购信息和微博热点信息等场景中的应用。此外,还提供了MySQL配置文件的相关内容。 ... [详细]
  • 本文介绍了在sqoop1.4.*版本中,如何实现自定义分隔符的方法及步骤。通过修改sqoop生成的java文件,并重新编译,可以满足实际开发中对分隔符的需求。具体步骤包括修改java文件中的一行代码,重新编译所需的hadoop包等。详细步骤和编译方法在本文中都有详细说明。 ... [详细]
  • 微信开放外链的第二阶段:腾讯和阿里巴巴的博弈
    2021年11月30日,微信开始进行“开放外链”的第二阶段,允许在微信个人会话中打开外部链接和在微信群中打开电商链接。虽然这是腾讯和阿里巴巴都能接受的阶段性结果,但双方都不会太满意。接下来几个月,腾讯和阿里将展开复杂的博弈,我们作为外人很难看清全过程。工信部从未要求腾讯无条件开放微信API,本次开放的也只是普通的HTTP链接。 ... [详细]
author-avatar
书生
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有