草庐IT

uniapp集成腾讯即时通信IM,实现一对一聊天,支持文字、表情、语音、图片、视频

一码代码库 2023-04-25 原文

uniapp集成腾讯即时通信IM,实现一对一聊天,支持文字、表情、语音、图片、视频

原则

拿来即用,节省开发时间

介绍

腾讯的给的例子内容比较乱,我花了好长时间才集成出来,然后对聊天页面做了UI美化。

效果图

uniapp集成腾讯即时通信IM,实现一对一聊天,支持文字、

使用方式

  1. 将文件放到相应的位置
  2. app配置
<script>
	import TIM from 'tim-wx-sdk';
	import COS from 'cos-wx-sdk-v5';
	import TIMUploadPlugin from 'tim-upload-plugin';
	import logger from './utils/logger'; // app.js
	import {
		SDKAppID
	} from './debug/GenerateTestUserSig.js';
	import {genTestUserSig} from "./debug/GenerateTestUserSig.js"
	
	
	export default {
		onLaunch: function() {
			uni.setStorageSync(`TIM_${SDKAppID}_isTUIKit`, true);
			// 重点注意: 为了 uni-app 更好地接入使用 tim,快速定位和解决问题,请勿修改 uni.$TUIKit 命名。
			// 如果您已经接入 tim ,请将 uni.tim 修改为 uni.$TUIKit。
			uni.$TUIKit = TIM.create({
				SDKAppID: SDKAppID
			});
			
			// 注册 COS SDK 插件
			uni.$TUIKit.registerPlugin({
				'cos-wx-sdk': COS,
				'tim-upload-plugin': TIMUploadPlugin
			});
			
			// 如果您已创建了 tim,请将 tim 实例挂载在 wx 上,且不可以修改 wx.$TIM(修改变量可能导致 TUICalling 组件无法正常使用), 完成 TUICalling 初始化,
			// 如果您没有创建,可以不传
			// #ifdef MP-WEIXIN
			wx.$TIM = uni.$TUIKit;
			// #endif
			uni.$TUIKitTIM = TIM;
			uni.$TUIKitEvent = TIM.EVENT;
			uni.$TUIKitVersion = TIM.VERSION;
			uni.$TUIKitTypes = TIM.TYPES; // 监听系统级事件
			uni.$TUIKit.on(uni.$TUIKitEvent.SDK_READY, this.onSDKReady);
			uni.$TUIKit.on(uni.$TUIKitEvent.SDK_NOT_READY, this.onSdkNotReady);
			uni.$TUIKit.on(uni.$TUIKitEvent.KICKED_OUT, this.onKickedOut);
			uni.$TUIKit.on(uni.$TUIKitEvent.ERROR, this.onTIMError);
			uni.$TUIKit.on(uni.$TUIKitEvent.NET_STATE_CHANGE, this.onNetStateChange);
			uni.$TUIKit.on(uni.$TUIKitEvent.SDK_RELOAD, this.onSDKReload);
			
			//重新设置用户信息
			uni.$LTResetLoginData = this.resetLoginData;
			uni.$LTLogin = this.liaotianLogin;
			uni.$LTUpdateMyInfo = this.ltUpdateMyInfo;
		},
		globalData: {
					// userInfo: userID userSig token phone
					userInfo: {
						userID:'',
						userSig:""
					},
					// 个人信息
					userProfile: {
					},
					isTUIKit: true,
					headerHeight: 0,
					statusBarHeight: 0,
					SDKAppID: SDKAppID,
					isSDKReady:false
				},
		methods: {
			onTIMError(event) {
				console.info("聊天-SDK遇到错误时触发",event)
				uni.showModal({
					title:"系统提示",
					content:'初始化聊天错误:' + event.data.code + "," + event.data.message,
				})
			},
			onSDKReady({name}) {
				console.info("聊天-SDK进入ready状态时触发",name)
				const isSDKReady = name === uni.$TUIKitEvent.SDK_READY ? true : false
				getApp().globalData.isSDKReady = true
				this.ltUpdateMyInfo()
			},
			onSdkNotReady() {
				console.info("聊天-SDK进入not ready状态时触发")
				getApp().globalData.isSDKReady = false
			},
			onNetStateChange(event) {
				console.info("聊天-网络状态改变通知",event)
				// v2.5.0 起支持
				  // event.data.state 当前网络状态,枚举值及说明如下:
				  // TIM.TYPES.NET_STATE_CONNECTED - 已接入网络
				  // TIM.TYPES.NET_STATE_CONNECTING - 连接中。很可能遇到网络抖动,SDK 在重试。接入侧可根据此状态提示“当前网络不稳定”或“连接中”
				  // TIM.TYPES.NET_STATE_DISCONNECTED - 未接入网络。接入侧可根据此状态提示“当前网络不可用”。SDK 仍会继续重试,若用户网络恢复,SDK 会自动同步消息
				  
			},
			onSDKReload() {
				console.info("聊天-SDK重启")
			},
			onKickedOut() {
				console.info("聊天-用户被踢下线时触发")
				uni.showModal({
					title:"系统提示",
					content:'您的聊天账号在另一台设备登陆,若不是本人,请联系管理员',
				})
				
				
			},
			liaotianLogin(event){
				if(getApp().globalData.isSDKReady === true){
					if(this.isNotNull(event)){
						event(true)
					}
					return
				}
				this.resetLoginData()
				console.info("签名",getApp().globalData.userInfo)
				uni.$TUIKit.login({userID: getApp().globalData.userInfo.userID, userSig: getApp().globalData.userInfo.userSig})
				.then((imResponse)=> {
				  console.log("登陆结果",imResponse.data); // 登录成功
				  if (imResponse.data.repeatLogin === true) {
				    // 标识帐号已登录,本次登录操作为重复登录。v2.5.1 起支持
				    console.log("登陆成功",imResponse.data.errorInfo);
				  }else{
					 let interval = setInterval(()=> {
					 	console.info("sdk是否准备",getApp().globalData.isSDKReady)
					       if(getApp().globalData.isSDKReady === true){
					 		  if(this.isNotNull(event)){
					 			  clearInterval(interval)
					 			  event(true)
					 		  }
					 	  }
					 }, 100) 
				  }	
				})
				.catch((imError)=> {
					console.warn('登陆失败:', imError); // 登录失败的相关信息
					uni.showModal({
						title:"系统提示",
						content:"聊天登陆失败,请稍后重试",
					})
				  event(false)
				});
			},
			// 聊天 - 初始化数据
			resetLoginData() {
				//从数据库获取到用户信息
				let userInfo = {
					fbNickName:"闫伟",
					fbAvatarUrl:"https://thirdwx.qlogo.cn/mmopen/vi_32/LWicoUend7QOfiabWDRjfpPDB51LzhbzVzjibicqDD2ztqy7BlpicRn8HToMoDVg9g1XfNiayHStXfC2f6yGNHgLueEw/132",
					liaotianId:"lt402882f280bb92810180bbb3f40a000f",
				}
				getApp().globalData.expiresIn = '';
				getApp().globalData.sessionID = '';
				getApp().globalData.userInfo = {
					userID: userInfo.liaotianId,
					userSig: genTestUserSig(userInfo.liaotianId).userSig,
				};
				getApp().globalData.userProfile = {
					userID:userInfo.liaotianId,
					nick: userInfo.fbNickName,
					avatar:userInfo.fbAvatarUrl
				};
			},
			ltUpdateMyInfo(){
				if(getApp().globalData.isSDKReady === true){
					let userInfo = this.$user.getUserInfo()
					// 修改个人标配资料
					let promise = uni.$TIM.updateMyProfile({
					  nick: userInfo.fbNickName,
					  avatar: userInfo.fbAvatarUrl,
					  // allowType: uni.$TIM.TYPES.ALLOW_TYPE_ALLOW_ANY
					});
					promise.then(function(imResponse) {
					  console.log("更新资料成功",imResponse.data); // 更新资料成功
					}).catch(function(imError) {
					  console.warn('updateMyProfile error:', imError); // 更新资料失败的相关信息
					});
				}
			},
			/**
			 * 是否不为空
			 * @param {Object} params
			 */
			isNotNull(params){
			      if (typeof(params) === 'undefined'){
					  return false
				  }else if(params == null){
					  return false
				  }
				  
				  return true
			}
		},
		onShow: function(res) {
			
		},
		onHide: function() {
		
		}
	}
</script>

<style>
	@import "colorui/main.css";
	@import "colorui/icon.css";
	
	body{
		height: 100%;
		width: 100%;
		background-color: #ededed;
		color: #333333;
		font-family: 'ukijtor'
	}
	
	.full-back-color-gray{
		height: 100%;
		background-color: #f1f1f1;
	}
	
	.full-back-color-white{
		height: 100%;
		background-color: white;
	}
	.nav-list {
		display: flex;
		flex-wrap: wrap;
		padding: 0px 40upx 0px;
		justify-content: space-between;
	}
	
	.nav-li {
		padding: 30upx;
		border-radius: 12upx;
		width: 45%;
		margin: 0 2.5% 40upx;
		background-image: url(https://cdn.nlark.com/yuque/0/2019/png/280374/1552996358352-assets/web-upload/cc3b1807-c684-4b83-8f80-80e5b8a6b975.png);
		background-size: cover;
		background-position: center;
		position: relative;
		z-index: 1;
	}
	
	.nav-li::after {
		content: "";
		position: absolute;
		z-index: -1;
		background-color: inherit;
		width: 100%;
		height: 100%;
		left: 0;
		bottom: -10%;
		border-radius: 10upx;
		opacity: 0.2;
		transform: scale(0.9, 0.9);
	}
	
	.nav-li.cur {
		color: #fff;
		background: rgb(94, 185, 94);
		box-shadow: 4upx 4upx 6upx rgba(94, 185, 94, 0.4);
	}
	
	.nav-title {
		font-size: 32upx;
		font-weight: 300;
	}
	
	.nav-title::first-letter {
		font-size: 40upx;
		margin-right: 4upx;
	}
	
	.nav-name {
		font-size: 28upx;
		text-transform: Capitalize;
		margin-top: 20upx;
		position: relative;
	}
	
	.nav-name::before {
		content: "";
		position: absolute;
		display: block;
		width: 40upx;
		height: 6upx;
		background: #fff;
		bottom: 0;
		right: 0;
		opacity: 0.5;
	}
	
	.nav-name::after {
		content: "";
		position: absolute;
		display: block;
		width: 100upx;
		height: 1px;
		background: #fff;
		bottom: 0;
		right: 40upx;
		opacity: 0.3;
	}
	
	.nav-name::first-letter {
		font-weight: bold;
		font-size: 36upx;
		margin-right: 1px;
	}
	
	.nav-li text {
		position: absolute;
		right: 30upx;
		top: 30upx;
		font-size: 52upx;
		width: 60upx;
		height: 60upx;
		text-align: center;
		line-height: 60upx;
	}
	
	.text-light {
		font-weight: 300;
	}
	
	@keyframes show {
		0% {
			transform: translateY(-50px);
		}
	
		60% {
			transform: translateY(40upx);
		}
	
		100% {
			transform: translateY(0px);
		}
	}
	
	@-webkit-keyframes show {
		0% {
			transform: translateY(-50px);
		}
	
		60% {
			transform: translateY(40upx);
		}
	
		100% {
			transform: translateY(0px);
		}
	}
</style>

  1. main.js 配置
// 全局mixins,用于实现setData等功能';
import Mixin from './polyfill/mixins';
Vue.mixin(Mixin);
  1. pages.json 配置
{
	"pages": [
		{
		    "path" : "pages/TUI-Conversation/conversation/conversation",
		    "style" :                                                                                    
		    {
		        "navigationBarTitleText": "消息",
		        "enablePullDownRefresh": false
		    }
		},
		{
		    "path" : "pages/TUI-Chat/chat",
		    "style" :                                                                                    
		    {
				"navigationBarTitleText": "聊天",
				"enablePullDownRefresh": false
		    }
		}
    ],
	"globalStyle": {
		"navigationBarTextStyle": "black",
		"navigationBarTitleText": "集成腾讯im",
		"navigationBarBackgroundColor": "#F8F8F8",
		"backgroundColor": "#F8F8F8",
		"usingComponents": {}		
	}
}

  1. GenerateTestUserSig.js配置SDKAPPID和SECRETKEY
  2. 安装 npm i
  3. 运行

源码

下载源码

联系方式

可关注后发私信

文章转载

查看文章

有关uniapp集成腾讯即时通信IM,实现一对一聊天,支持文字、表情、语音、图片、视频的更多相关文章

  1. ruby - 即时确定方法的可见性 - 2

    我正在编写一个方法,它将在一个类中定义一个实例方法;类似于attr_accessor:classFoocustom_method(:foo)end我通过将custom_method函数添加到Module模块并使用define_method定义方法来实现它,效果很好。但我无法弄清楚如何考虑类(class)的可见性属性。例如,在下面的类中classFoocustom_method(:foo)privatecustom_method(:bar)end第一个生成的方法(foo)必须是公共(public)的,第二个(bar)必须是私有(private)的。我怎么做?或者,如何找到调用我的cust

  2. ruby - 如何根据特征实现 FactoryGirl 的条件行为 - 2

    我有一个用户工厂。我希望默认情况下确认用户。但是鉴于unconfirmed特征,我不希望它们被确认。虽然我有一个基于实现细节而不是抽象的工作实现,但我想知道如何正确地做到这一点。factory:userdoafter(:create)do|user,evaluator|#unwantedimplementationdetailshereunlessFactoryGirl.factories[:user].defined_traits.map(&:name).include?(:unconfirmed)user.confirm!endendtrait:unconfirmeddoenden

  3. ruby-on-rails - 如何使辅助方法在 Rails 集成测试中可用? - 2

    我在app/helpers/sessions_helper.rb中有一个帮助程序文件,其中包含一个方法my_preference,它返回当前登录用户的首选项。我想在集成测试中访问该方法。例如,这样我就可以在测试中使用getuser_path(my_preference)。在其他帖子中,我读到这可以通过在测试文件中包含requiresessions_helper来实现,但我仍然收到错误NameError:undefinedlocalvariableormethod'my_preference'.我做错了什么?require'test_helper'require'sessions_hel

  4. ruby-on-rails - 我如何将 Hoptoad 与 DelayedJob 和 DaemonSpawn 集成? - 2

    我一直很高兴地使用DelayedJob习惯用法:foo.send_later(:bar)这会调用DelayedJob进程中对象foo的方法bar。我一直在使用DaemonSpawn在我的服务器上启动DelayedJob进程。但是...如果foo抛出异常,Hoptoad不会捕获它。这是任何这些包中的错误...还是我需要更改某些配置...或者我是否需要在DS或DJ中插入一些异常处理来调用Hoptoad通知程序?回应下面的第一条评论。classDelayedJobWorker 最佳答案 尝试monkeypatchingDelayed::W

  5. ruby-on-rails - Ruby on Rails - 为文本区域和图片生成列 - 2

    我是Rails的新手,所以请原谅简单的问题。我正在为一家公司创建一个网站。那家公司想在网站上展示它的客户。我想让客户自己管理这个。我正在为“客户”生成一个表格,我想要的三列是:公司名称、公司描述和Logo。对于名称,我使用的是name:string但不确定如何在脚本/生成脚手架终端命令中最好地创建描述列(因为我打算将其设置为文本区域)和图片。我怀疑描述(我想成为一个文本区域)应该仍然是描述:字符串,然后以实际形式进行调整。不确定如何处理图片字段。那么……说来话长:我在脚手架命令中输入什么来生成描述和图片列? 最佳答案 对于“文本”数

  6. 华为OD机试用Python实现 -【明明的随机数】 2023Q1A - 2

    华为OD机试题本篇题目:明明的随机数题目输入描述输出描述:示例1输入输出说明代码编写思路最近更新的博客华为od2023|什么是华为od,od薪资待遇,od机试题清单华为OD机试真题大全,用Python解华为机试题|机试宝典【华为OD机试】全流程解析+经验分享,题型分享,防作弊指南华为o

  7. 基于C#实现简易绘图工具【100010177】 - 2

    C#实现简易绘图工具一.引言实验目的:通过制作窗体应用程序(C#画图软件),熟悉基本的窗体设计过程以及控件设计,事件处理等,熟悉使用C#的winform窗体进行绘图的基本步骤,对于面向对象编程有更加深刻的体会.Tutorial任务设计一个具有基本功能的画图软件**·包括简单的新建文件,保存,重新绘图等功能**·实现一些基本图形的绘制,包括铅笔和基本形状等,学习橡皮工具的创建**·设计一个合理舒适的UI界面**注明:你可能需要先了解一些关于winform窗体应用程序绘图的基本知识,以及关于GDI+类和结构的知识二.实验环境Windows系统下的visualstudio2017C#窗体应用程序三.

  8. jenkins部署1--jenkins+gitee持续集成 - 2

    前置步骤我们都操作完了,这篇开始介绍jenkins的集成。话不多说,看操作1、登录进入jenkins后会让你选择安装插件,选择第一个默认的就行。安装完成后设置账号密码,重新登录。2、配置JDK和Git都需要执行路径,所以需要先把执行路径找到,先进入服务器的docker容器,2.1JDK的路径root@69eef9ee86cf:/usr/bin#echo$JAVA_HOME/usr/local/openjdk-82.2Git的路径root@69eef9ee86cf:/#whichgit/usr/bin/git3、先配置JDK和Git。点击:ManageJenkins>>GlobalToolCon

  9. MIMO-OFDM无线通信技术及MATLAB实现(1)无线信道:传播和衰落 - 2

     MIMO技术的优缺点优点通过下面三个增益来总体概括:阵列增益。阵列增益是指由于接收机通过对接收信号的相干合并而活得的平均SNR的提高。在发射机不知道信道信息的情况下,MIMO系统可以获得的阵列增益与接收天线数成正比复用增益。在采用空间复用方案的MIMO系统中,可以获得复用增益,即信道容量成倍增加。信道容量的增加与min(Nt,Nr)成正比分集增益。在采用空间分集方案的MIMO系统中,可以获得分集增益,即可靠性性能的改善。分集增益用独立衰落支路数来描述,即分集指数。在使用了空时编码的MIMO系统中,由于接收天线或发射天线之间的间距较远,可认为它们各自的大尺度衰落是相互独立的,因此分布式MIMO

  10. 动漫制作技巧如何制作动漫视频 - 2

    动漫制作技巧是很多新人想了解的问题,今天小编就来解答与大家分享一下动漫制作流程,为了帮助有兴趣的同学理解,大多数人会选择动漫培训机构,那么今天小编就带大家来看看动漫制作要掌握哪些技巧?一、动漫作品首先完成草图设计和原型制作。设计草图要有目的、有对象、有步骤、要形象、要简单、符合实际。设计图要一致性,以保证制作的顺利进行。二、原型制作是根据设计图纸和制作材料,可以是手绘也可以是3d软件创建。在此步骤中,要注意的问题是色彩和平面布局。三、动漫制作制作完成后,加工成型。完成不同的表现形式后,就要对设计稿进行加工处理,使加工的难易度降低,并得到一些基本准确的概念,以便于后续的大样、准确的尺寸制定。四、

随机推荐