iOS IM SDK客户端开发文档

消息

消息:IM 交互实体,在 SDK 中对应的类型是 TDMessage。TDMessage 由 TDMessageBody 组成。

// 消息构建部分
#param mark - 消息基类
TDMessage.h
TDMessageBody.h

#param mark - 1V1消息类
TDTextMessageBody.h
TDImageMessageBody.h
TDVoiceMessageBody.h
TDVideoMessageBody.h
TDFileMessageBody.h
TDLocationMessageBody.h

#param mark - 群消息类
TDCreateGroupMessageBody.h
TDGroupAddMessageBody.h
TDGroupRemoveUidsMessageBody.h
TDDeleteGroupMessageBody.h
TDQuitGroupMessageBody.h
TDReplaceOwnerIdMessageBody.h
TDGroupSucessMessageBody.h

#param mark - 特殊消息类
TDCmdMessageBody.h
TDKeyBoardSoundMessageBody.h
TDConferenceMessageBody.h

/** 消息方法调用部分,比如添加代理,移除代理,发送消息等,
同时也包含会话相关的操作,比如创建或获取会话,获取会话列表等*/
TDChatManager.h
// 消息的协议部分,比如监听接收消息的回调方法等
ChatManagerDelegate

构造消息

以下构造消息的示例中,使用到的“初始化消息实例”方法的说明:

/// 初始化消息实例
/// @param body 消息体实例
/// @param fromUid 发送方Uid
/// @param toUid 接收方Uid
/// @param chatType 聊天类型 单聊/群聊
- (id) initMessageWithBody:(TDMessageBody *)body 
                           fromUid:(NSString *)fromUid 
                           toUid:(NSString *)toUid 
                           ChatType:(TDChatType)chatType; 

注意: 如果是给群组发消息,那么toUid为群组id

构造文字消息

/// 初始化文本消息体
/// @param aText 文本内容
- (instancetype)initWithText:(NSString *)aText;

// 调用:
TDTextMessageBody *body = [[TDTextMessageBody alloc] initWithText:@"要发送的消息"];
// 获取当前登录的环信id
NSString *from = [[TDWLClient sharedClient] currentUserId];

//生成Message
TDMessage *message = [[TDMessage alloc] initMessageWithBody:body fromUid:from toUid:@"接收者Uid" ChatType:chatType];
// chatType = SingleChat; // 单聊消息
// chatType = GroupChat;// 群聊消息

构造图片消息

/// 初始化文件消息体
/// @param aData 文件data数据
/// @param aDisplayName 文件名
- (instancetype)initWithData:(NSData *)aData
                  displayName:(NSString *)aDisplayName;
                 
// 调用:               
TDImageMessageBody *body = [[TDImageMessageBody alloc] initWithData:aImageData displayName:@"image"];

NSString *from = [[EMClient sharedClient] currentUsername];

//生成Message
TDMessage *message = [[TDMessage alloc] initMessageWithBody:body fromUid:from toUid:@"接收者Uid" ChatType:chatType];
// chatType = SingleChat; // 单聊消息
// chatType = GroupChat;// 群聊消息

构造位置消息

/// 实例化位置消息体
/// @param aLatitude 纬度
/// @param aLongitude 经度
/// @param aAddress 地理位置信息
- (instancetype)initWithLatitude:(double)aLatitude longitude:(double)aLongitude address:(NSString *)aAddress;

//调用:
TDLocationMessageBody *body = [[TDLocationMessageBody alloc] initWithLatitude:39 longitude:116 address:@"地址"]; 

NSString *from = [[EMClient sharedClient] currentUsername];

//生成Message
TDMessage *message = [[TDMessage alloc] initMessageWithBody:body fromUid:from toUid:@"接收者Uid" ChatType:chatType];
// chatType = SingleChat; // 单聊消息
// chatType = GroupChat;// 群聊消息

构造语音消息

/// 初始化文件消息体
/// @param aLocalPath 本地文件路径
/// @param aDisplayName 文件名
- (instancetype)initWithLocalPath:(NSString *)aLocalPath
                       displayName:(NSString *)aDisplayName;

// 调用:
TDVoiceMessageBody *body = [[TDVoiceMessageBody alloc] initWithLocalPath:aPath displayName:@"audio"];
body.duration = duration;

NSString *from = [[EMClient sharedClient] currentUsername];

//生成Message
TDMessage *message = [[TDMessage alloc] initMessageWithBody:body fromUid:from toUid:@"接收者Uid" ChatType:chatType];
// chatType = SingleChat; // 单聊消息
// chatType = GroupChat;// 群聊消息

构造视频消息

/// 初始化文件消息体
/// @param aLocalPath 本地文件路径
/// @param aDisplayName 文件名
- (instancetype)initWithLocalPath:(NSString *)aLocalPath
                       displayName:(NSString *)aDisplayName;

// 调用:
TDVideoMessageBody *body = [[TDVideoMessageBody alloc] initWithLocalPath:[aUrl path] displayName:@"video.mp4"];

NSString *from = [[EMClient sharedClient] currentUsername];

//生成Message
TDMessage *message = [[TDMessage alloc] initMessageWithBody:body fromUid:from toUid:@"接收者Uid" ChatType:chatType];
// chatType = SingleChat; // 单聊消息
// chatType = GroupChat;// 群聊消息

构造文件消息

/// 初始化文件消息体
/// @param aLocalPath 本地文件路径
/// @param aDisplayName 文件名
- (instancetype)initWithLocalPath:(NSString *)aLocalPath
                       displayName:(NSString *)aDisplayName;

// 调用:       
TDFileMessageBody *body = [[TDFileMessageBody alloc] initWithLocalPath:[aUrl path] displayName:@"file"];

NSString *from = [[EMClient sharedClient] currentUsername];

//生成Message
TDMessage *message = [[TDMessage alloc] initMessageWithBody:body fromUid:from toUid:@"接收者Uid" ChatType:chatType];
// chatType = SingleChat; // 单聊消息
// chatType = GroupChat;// 群聊消息

构造透传消息

SDK 提供的一种特殊类型的消息,即 CMD,不会存 db,也不会走 APNS 推送,类似一种指令型的消息。比如您的服务器要通知客户端做某些操作,您可以服务器和客户端提前约定好某个字段,当客户端收到约定好的字段时,执行某种特殊操作。

/// 初始化透传消息
/// @param aText 透传消息text 用户自定义
- (instancetype)initWithCmdText:(NSString *)aText;

// 调用:
TDCmdMessageBody *body = [[TDCmdMessageBody alloc] initWithCmdText:aCMDText];

NSString *from = [[EMClient sharedClient] currentUsername];

//生成Message
TDMessage *message = [[TDMessage alloc] initMessageWithBody:body fromUid:from toUid:@"接收者Uid" ChatType:chatType];
// chatType = SingleChat; // 单聊消息
// chatType = GroupChat;// 群聊消息

会话

// 主要用到的类
// 会话方法调用部分,包含创建或获取会话,获取会话列表等
TDChatManager.h

/**
获取所有会话
@param completion 完成的回调 返回会话数组
 数组中存的为TDSessionModel *类型
*/
- (void) getAllConversations:(LoadAllConversationComplection)completion;

// 调用:
[[TDWLClient sharedClient].chatManager getAllConversations:^(NSArray * _Nonnull contacts, NSError * _Nullable error) {
    if (error) {
        NSLog(@"get AllConversations error = %@",error);
        return;
    }

    // 这里根据你的需要去拿到contacts,进行业务操作
    .......
}];
/**
 删除一个会话
 @param sessionId  会话id  是NSString类型
 */
- (BOOL) deleteConversationWithSessionId:(NSString *)sessionId;

// 调用:
BOOL deleteResult = [[TDWLClient sharedClient].chatManager deleteConversationWithSessionId:@"要删除的会话id"];

if (deleteResult) {
        // 这里已经删除成功,根据开发业务需求去刷新列表或者做其他操作
}
/**
 更新会话内部消息全部已读
 @param sessionId 会话id  NSString类型
 */
- (BOOL) updateRecentSessionReadAllWithSessionId:(NSString *)sessionId;

// 调用:
[[TDWLClient sharedClient].chatManager updateRecentSessionReadAllWithSessionId:@"会话的id"];

聊天

登录成功之后才能进行聊天操作。发消息时,单聊和群聊调用的是统一接口,区别是 chatType不同。

发送消息

/**
 发送消息
 @param message  TDMessage类型
 */
- (void) sendMessage:(TDMessage *)message Completion:(void(^)(BOOL sendSuccess, NSError * _Nullable error))completion;

// 调用:
[[TDWLClient sharedClient].chatManager sendMessage:message Completion:^(BOOL sendSuccess, NSError * _Nullable error) {
        if (error) {
            // 失败
        }else {
            // 成功
        }
}];

接收消息

建议在自己工程的根控制器中注册消息代理,监听接收普通消息的回调方法,用于不在聊天页面时接收消息。

协议 ChatManagerDelegate

代理:
// 设置代理
[TDWLClient sharedClient].chatManager.chatDelegate = self;

// 移除代理
[TDWLClient sharedClient].chatManager.chatDelegate = nil;

接收普通消息会走以下回调:

/// 用于接收消息 传出delegate数据 传出数据为NSArray*类型 数据为TDMessage*类型
- (void) messagesDidReceive:(NSArray *)aMessageArray;

接收透传(cmd)消息会走以下回调:

/// 用于接收CMD消息 传出delegate数据 传出数据为NSArray *类型
- (void) CMDMessageDidReceive:(NSArray *)cmdArray;

接收按键音(KeyBorad)消息会走以下回调:

/// 用于接收按键音消息 传出delegate数据  传出数据为NSArray *类型
- (void) KeyBoradMessageDidReceive:(NSArray *)KeyBoradArray;

接收创建群(CreateGroup)的通知会走以下回调:
ps:此时为收到建群通知,还没加入群聊

/// 用于接收创建群的通知 传出delegate数据 传出数据为NSArray *类型
- (void) CreateGroupMessageDidReceive:(NSArray *)CreateGroupArray;

接收解散群的通知 会走以下回调:

/// 用于接收解散群的通知  传出delegate数据 传出数据为NSArray *类型
- (void) DeleteGroupMessageDidReceive:(NSArray *)DeleteGroupArray;

收到消息解析示例:

#pragma mark -----ChatManagerDelegate,CMDMessageDelegate,KeyBoradMessageDelegate,GroupNoticeDelegate-----
// 收到普通消息 看情况做处理
- (void) messagesDidReceive:(NSArray *)aMessageArray {

    NSLog(@"收到普通消息");

    __weak typeof(self) weakself = self;
    dispatch_async(self.msgQueue, ^{
          
        NSMutableArray *receveArray = [[NSMutableArray alloc] init];
        for (int i = 0; i < aMessageArray.count; i++) {
            TDMessage *message = aMessageArray[i];
            if (message.chatType == GroupChat) {
                if ([message.to_uid isEqualToString:self.aSessionModel.sessionId]) {
                    [receveArray addObject:message];
                }
            }else {
                if ([message.from_uid isEqualToString:self.aSessionModel.sessionId]) {
                    [receveArray addObject:message];
                }
            }
        }
        
        NSArray *formated = [weakself _formatMessages:receveArray];
        [weakself.dataArray addObjectsFromArray:formated];
        
        dispatch_async(dispatch_get_main_queue(), ^{
            [weakself.tableView reloadData];
            [weakself _scrollToBottomRow];
        });
    });
}
- (void) CMDMessageDidReceive:(NSArray *)cmdArray {
    NSLog(@"收到了CMD消息:%@",cmdArray);
    //根据业务需求进行解析,做后续操作
    ......
}
- (void) KeyBoradMessageDidReceive:(NSArray *)KeyBoradArray {
    NSLog(@"收到了按键音消息:%@",KeyBoradArray);
    //根据业务需求进行解析,做后续操作
    ......
}
- (void) CreateGroupMessageDidReceive:(NSArray *)CreateGroupArray {
    NSLog(@"收到了建群通知:%@",CreateGroupArray);
    //根据业务需求进行解析,做后续操作
    ......
}
- (void) DeleteGroupMessageDidReceive:(NSArray *)DeleteGroupArray {
    NSLog(@"收到了删除群的通知");
    //根据业务需求进行解析,做后续操作
    ......
}

下载消息中的附件

/**
 * 文件下载方法
 * @param downloadURL 文件链接
 * @param progress 进度
 * @param completion 完成的回调
 */
- (void)downloadFileWithURL:(NSString *) downloadURL progress:(void (^)(NSProgress *downloadProgress))progress Completion:(void(^)(NSString *_Nullable filePath, NSError * _Nullable error))completion;

/// 判断文件是否已经在沙盒中存在
/// @param fileName 文件名
-(BOOL) isFileExist:(NSString *)fileName;

/// 获取文件沙盒路径
/// @param fileName 文件名
- (NSString *) getFileCachesPathWith:(NSString *)fileName;

示例:下载图片消息中的原图:

// 点击图片下载原图
- (void) _imageMessageCellDidSelected:(TDMessageCell *)aCell {
    
    __weak typeof(self) weakself = self;
    
    TDImageMessageBody *body = (TDImageMessageBody*)aCell.model.emModel.body;
    
    // 判断原图是否存在
    BOOL isFileExist = [[TDWLClient sharedClient].chatManager isFileExist:body.imgUrl];
    if (isFileExist) {
        // 获取原图路径
        NSString *imgPath = [[TDWLClient sharedClient].chatManager getFileCachesPathWith:body.imgUrl];
        NSLog(@"imgUrl:%@",body.imgUrl);
        NSLog(@"imgPath:%@",imgPath);
        UIImage *image = [UIImage imageWithContentsOfFile:imgPath];
        if (image) {
            [[TDImageBrowser sharedBrowser] showImages:@[image] fromController:self];
        } else {
            [TDAlertController showErrorAlert:@"获取原图失败T_T"];
        }
    }else {
        // 下载原图
        [self showHudInView:self.view hint:@"下载原图..."];
        [[TDWLClient sharedClient].chatManager downloadFileWithURL:body.imgUrl progress:^(NSProgress * _Nonnull downloadProgress) {
            NSLog(@"进度:%@",downloadProgress);
        } Completion:^(NSString * _Nullable filePath, NSError * _Nullable error) {
            [weakself hideHud];
            if (error) {
                [TDAlertController showErrorAlert:@"下载原图失败T_T"];
            } else {
                NSString *localPath = [[[NSString stringWithFormat:@"%@",filePath] componentsSeparatedByString:@"://"] lastObject];
                UIImage *image = [UIImage imageWithContentsOfFile:localPath];
                if (image) {
                    [[TDImageBrowser sharedBrowser] showImages:@[image] fromController:self];
                } else {
                    [TDAlertController showErrorAlert:@"下载原图失败T_T"];
                }
            }
        }];
    }
}

好友管理

注:通达IM不是好友也可以聊天,不推荐使用通达IM的好友机制。如果你有自己的服务器或好友关系,请自己维护好友关系。

好友管理涉及到的SDK头文件如下:

TDWLUserManager.h

好友管理提供的方法如下:

/**
 添加一个好友
 @param uid 用户uid  NSSting *类型
 */
- (BOOL) addFriendWithUid:(NSString *)uid;

/**
 删除一个好友
 @param uid 用户uid  NSSting *类型
 */
- (BOOL) deleteFriendWithUid:(NSString *)uid;

/**
 获取所有好友列表
 @param completion 完成后的回调
 */
- (void) getAllFriend:(LoadAllFriendCompletion)completion;

/**
 获取所有在黑名单里面的好友
 @param completion 完成后的回调
 */
- (void)getAllBlackListUser:(LoadAllFriendCompletion)completion;

/**
 拉黑某个用户
 @param uid 用户uid
 */
- (BOOL)blackUserWithUid:(NSString *)uid;

/**
 将某个用户从黑名单释放
 @param uid 用户uid
 */
- (BOOL)unBlackUserWithUid:(NSString *)uid;

/**
 判断某个用户是否在黑名单
 @param uid uid 用户uid
 */
- (BOOL)isUserInBlackListWithUid:(NSString *)uid;

注:以上方法调用全部用[TDWLClient sharedClient].contactManager调用
添加好友调用示例:

BOOL addFriendSuccess = [[TDWLClient sharedClient].contactManager addFriendWithUid:@"对方Uid"];

if (addFriendSuccess) {
    // 添加好友成功
}else {
    // 添加好友失败
}

群组管理

群涉及的方法如下:

/**
 建群
 @param uidsArr  uidsArr中为想要添加的群成员uid  注: uidsArr.count >= 2  否则建群失败;
 @param name 群名 NSString *类型
 @param completion 完成的回调
 */
- (void) createGroup:(NSArray *)uidsArr andGroupName:(NSString *)name complet:(void(^)(BOOL isSuccess,NSError *_Nullable error))completion;


/**
 添加群成员
 @param gid 被添加用户的群id;
 @param uids NSArray类型  添加的用户列表
 @param completion 完成的回调
 */
- (void) addGroupMembersWithGId:(NSString *)gid andUidsArray:(NSArray *)uids complet:(void(^)(BOOL isSuccess,NSError *_Nullable error))completion;


/**
 删除群成员
 @param gid 被删除用户的群id;
 @param uids NSArray类型 删除的用户列表。
 返回类型  BOOL
 */
- (void) removeGroupMembersWithGid:(NSString *)gid andUidsArray:(NSArray *)uids complet:(void(^)(BOOL isSuccess,NSError *_Nullable error))completion;


/**
 销毁群
 @param gid 被销毁的群id。
 返回类型  BOOL
 */
- (void) destoryGroup:(NSString *)gid complet:(void(^)(BOOL isSuccess,NSError *_Nullable error))completion;

/**
 退群
 @param gid 群id  NSString类型
 */
- (void) quitGroup:(NSString *)gid complet:(void(^)(BOOL isSuccess,NSError *_Nullable error))completion;

/**
 更换群主
 @param gid 群id NSString类型
 */
- (void) changeGroupOwnerWithGroup:(NSString *)gid targetOwnerId:(NSString *)targetOwnerId complet:(void(^)(BOOL isSuccess,NSError *_Nullable error))completion;


/**
 修改群昵称(本地数据库修改,服务器暂无此功能)
 @param gid 群id NSString类型
 @param newName 新群名 NSString类型
 */
- (BOOL) changeGroupNameWithGroupId:(NSString *)gid newName:(NSString *)newName;


/// 根据群ID查询某条群组的信息
/// @param groupId 群Id NSString类型
- (NSDictionary * _Nullable) getGroupInfoWithGroupId:(NSString *)groupId;

注: 群操作中、建群/添加群成员/删除群成员/解散群/退群/更换群主 这几种功能,将会以普通消息的形式插入在聊天消息中。可根据需求选择是否显示此类消息。

音视频通话

适用场景

1对1通话API用于实现1对1语音通话和1对1视频通话,典型场景包括社交交友,远程心理咨询、远程医疗、一对一在线教育、远程视频辅助等。
其API设计思想类似于传统的电话通话,呼叫过程使用了即时通讯的消息机制,所以和即时通讯功能是紧耦合的。

发起实时通话

用户可以调用发起语音或者视频API向在线用户发起实时通话。

主要用到的类:

ITDWLCallManager.h

需要用到的代理方法,建议在开启SDK时,同时添加回调代理

/*!
 *  \~chinese
 *  添加回调代理
 *
 *  @param aDelegate  要添加的代理
 */
- (void)addDelegate:(id)aDelegate;

/*!
 *  \~chinese
 *  移除回调代理
 *
 *  @param aDelegate  要移除的代理
 */
- (void)removeDelegate:(id)aDelegate;

主要方法

/*!
 *  \~chinese
 *  发起实时会话
 *
 *  @param aType                     通话类型
 *  @param aRemoteName      被呼叫用户的IM账号(不能与自己通话)
 *  @param aCompletionBlock 完成的回调
 *
 */
- (void)startCall:(TDWLCallType)aType
       remoteName:(NSString *)aRemoteName
       completion:(void (^)(TDWLCallSession * _Nullable aCallSession,TDWLError * _Nullable error))aCompletionBlock;

注意: TDWLCallType 为枚举类型
TDWLCallTypeAudio = 0,    /*! \~chinese 实时语音 \~english Voice call */
TDWLCallTypeVideo,        /*! \~chinese 实时视频 \~english Video call */

/*!
 *  \~chinese
 *  接收方同意通话请求
 *
 *  @param  aCallId     通话ID
 *
 *  @result 错误信息
 */
- (TDWLError *)answerIncomingCall:(NSString *)aCallId;

/*!
 *  \~chinese
 *  结束通话
 *
 *  @param aCallSession  aCallSession   通话对象
 *  @param aReason       结束原因
 *
 *  @result 错误
 */
- (TDWLError *)endCall:(TDWLCallSession *)aCallSession
                reason:(TDWLCallEndReason)aReason;

扩展

/*!
 *  \~chinese
 *  切换前后摄像头
 *
 */
- (void)updateSipCallWithSwitchCamera;

/*!
 *  \~chinese
 *  设置是否静音
 *
 */
- (void)updateSipCallIsMute:(BOOL)aIsMute;

/*!
 *  \~chinese
 *  设置视频是否可用
 *
 */
- (void)updateSipCallIsEnableVideo:(BOOL)aEnableVideo;

示例:

[[TDWLClient sharedClient].callManager startCall:aType
                                          remoteName:aUsername
                                          completion:^(TDWLCallSession * _Nullable aCallSession, TDWLError * _Nullable error) {
        if (error == nil) {
            // 成功,根据业务需求进行后续处理
        } else {
            // 失败,根据业务需求进行后续处理
        }
}];

外网呼叫

外网呼叫API用于打外网电话使用,场景包括客服、销售等
其API设计思想类似于传统的电话通话。

主要用到的类:

ITDWLSipCallManager.h

需要用到的代理方法,建议在开启SDK时,同时添加回调代理

/*!
 *  \~chinese
 *  添加回调代理
 *
 *  @param aDelegate  要添加的代理
 */
- (void)addDelegate:(id)aDelegate;

/*!
 *  \~chinese
 *  移除回调代理
 *
 *  @param aDelegate  要移除的代理
 */
- (void)removeDelegate:(id)aDelegate;

主要方法:

/*!
 *  \~chinese
 *  发起SIP实时音频会话
 *
 *  @param sipNumber        被呼叫用户的Sip账号 可以是手机号
 *  @param remoteName       被呼叫用户的昵称
 *  @param aCompletionBlock  完成的回调
 *
 */
- (void)startAudioCallWithSipNumber:(NSString *)sipNumber
                          remoteName:(NSString * _Nullable)remoteName
                          completion:(void (^_Nonnull)(TDWLError * _Nullable error))aCompletionBlock;


/*!
 *  \~chinese
 *  设置是否静音
 *
 */
- (void)updateSipCallIsMute:(BOOL)aIsMute;

示例:

[[TDWLClient sharedClient].sipManager startAudioCallWithSipNumber:@"手机号或者对方sip号" remoteName:aUsername
                                                               completion:^(TDWLError * _Nullable error) {
        if (error == nil) {
            // 成功
        } else {
            // 失败
        }
}];

更新于:2020.07.11
by 张宗琦

你可能感兴趣的:(iOS IM SDK客户端开发文档)