SDK集成的一些問題
1、只添加環信IM功能
1.1 兩個SDK HyphenateLite.framework 和Hyphenate.framework,后者包含實時音視頻功能
1.2 環信demo有個EaseUI的包拖進工程,對界面要求不大的直接使用就可以了
2、只添加環信客服功能
1.1 環信客服SDK HelpDesk.framework,由于底層基于HyphenateLite.framework實現通信功能,所以兩個庫都要導入,1.2 HelpDeskUI拖進工程,界面基本就不用自己搭建了
3、同時接入環信IM和客服功能
3.1 不包含音視頻的話,直接添加HyphenateLite.framework和HelpDesk.framework兩個庫,包含的話就添加Hyphenate.framework和HelpDesk.framework兩個庫
3.2 由于客服和IM的聊天界面不同,且是根據不同的庫獲取的數據進行展示,所以UI要導入兩套EaseUI和HelpDeskUI
3.3 環信的注冊和登錄對于兩個庫都是一樣的,所以只需要使用客服的庫HelpDesk進行用戶系統的建成就可以了,包括SDK的初始化都是用客服的就可以
4、最好設置成自動登錄,以免每次進入進入會話列表沒有數據
5、登錄前一定要確定已經注冊,當然注冊是授權注冊最好放在后臺
6、通知回調的deviceToken通過環信SDK綁定的時候,最好確認用戶已登錄環信,且綁定代碼要放在異步線程,否則會造成進入app卡頓
會話列表和聊天顯示用戶真實昵稱和頭像
由于環信服務器不存儲任何用戶的個人資料,所以不管是發送接收消息還是后來的通知內容,都無法獲取聊天對方的真實資料
1、EMMessage消息類,包含了消息本身、發送和接受方的id,以及其他一些信息,其中有一個ext的屬性,是一個NSDictionary類型,通過它可以使消息額外附帶一些你想要的數據,其中就可以把用的name和headerImage存入進去,有兩個特定字段
NSDictionary *dic = @{@"userAvatar":[WechatUserInfo shared].headimgurl,@"userNick":[WechatUserInfo shared].nickname};
2、會話列表展示中,鑒于以上ext存儲了發送方信息,可以通過會話的lastReceivedMessage獲取到ext字段,從而獲取發送方的信息; 而對于用戶主動發出卻沒有收到回復的消息就沒辦法了,如果能在用戶發起聊天的消息ext字段中同時存儲自己的信息和對方的信息,那么在消息列表展示的時候,及時沒有收到對方消息(即沒有lastReceivedMessage可以獲取對方信息),但是自己發送的消息lastMessage中是有對方信息的呀
環信通知
1、嚴格按照步驟制作相關證書和文件,從keychain導出p12
文件的時候一定要設置密碼
2、首次測試推送,一定要殺死app,因為app在前臺或者后臺未殺死狀態,需要自己去做本地推送
3、因為app在前臺或者后臺未殺死狀態,需要自己去做本地推送;對于app在后臺未被殺死的時候,新消息會執行代理的方法,在方法內部自己做一個本地推送;環信demo中有此方法
- (void)messagesDidReceive:(NSArray *)aMessages{
//刷新列表
[[NSNotificationCenter defaultCenter] postNotificationName:NOTI_MESSAGE_RECEIVE object:nil];
UIApplicationState state = [[UIApplication sharedApplication] applicationState];
switch (state) {
case UIApplicationStateActive:
break;
case UIApplicationStateInactive:
break;
case UIApplicationStateBackground:
[self showNotificationWithMessage:aMessages.lastObject];
break;
default:
break;
}
}
4、在通知被用戶點擊觸發的相關方法中,做相應處理
5、如果打包ad_hoc收不到推送,一定先檢查自己的證書是否正確
環信消息撤銷
一條消息的撤銷涉及到兩端用戶對消息的處理,所以對于接收端是需要受到一個撤銷消息的,所以A端撤銷消息,需要將撤銷的消息封裝成透傳消息發送給B端
1、撤銷方,發送透傳消息,發送成功將本地數據清除,刷新列表
- (void)revokeMessageWithMessageId:(NSString *)aMessageId conversationId:(NSString *)conversationId {
EMCmdMessageBody *body = [[EMCmdMessageBody alloc] initWithAction:@"REVOKE_FLAG"];
NSDictionary *ext = @{@"msgId":aMessageId};
NSString *currentUsername = [EMClient sharedClient].currentUsername;
EMMessage *message = [[EMMessage alloc] initWithConversationID:conversationId from:currentUsername to:conversationId body:body ext:ext];
if (self.conversation.type == EMConversationTypeGroupChat){
message.chatType = EMChatTypeGroupChat;
} else {
message.chatType = EMChatTypeChat;
}
//發送cmd消息
[[EMClient sharedClient].chatManager sendMessage:message progress:nil completion:^(EMMessage *message, EMError *error) {
if (!error) {
NSLog(@"發送成功");
// 需要自己從dataArray里將聊天消息刪掉, 還有self.conversation里,最后刷新
NSMutableIndexSet *indexs = [NSMutableIndexSet indexSetWithIndex:self.menuIndexPath.row];
NSMutableArray *indexPaths = [NSMutableArray arrayWithObjects:self.menuIndexPath, nil];
[self.conversation deleteMessageWithId:aMessageId error:nil];
[self.messsagesSource removeObject:message];
if (self.menuIndexPath.row - 1 >= 0) {
id nextMessage = nil;
id prevMessage = [self.dataArray objectAtIndex:(self.menuIndexPath.row - 1)];
if (self.menuIndexPath.row + 1 < [self.dataArray count]) {
nextMessage = [self.dataArray objectAtIndex:(self.menuIndexPath.row + 1)];
}
if ((!nextMessage || [nextMessage isKindOfClass:[NSString class]]) && [prevMessage isKindOfClass:[NSString class]]) {
[indexs addIndex:self.menuIndexPath.row - 1];
[indexPaths addObject:[NSIndexPath indexPathForRow:(self.menuIndexPath.row - 1) inSection:0]];
}
}
[self.dataArray removeObjectAtIndex:self.menuIndexPath.row];
[self.messsagesSource enumerateObjectsWithOptions:NSEnumerationReverse usingBlock:^(EMMessage *obj, NSUInteger idx, BOOL *stop) {
if ([obj isKindOfClass:[EMMessage class]] && [obj.messageId isEqualToString:aMessageId]) {
[self.messsagesSource removeObject:obj];
}
}];
// [self.tableView beginUpdates];
[self.tableView reloadData];
// [self.tableView endUpdates];
if ([self.dataArray count] == 0) {
self.messageTimeIntervalTag = -1;
}
}else {
NSLog(@"發送失敗");
}
}];
}
2、接收方:EMChatManagerDelegate代理方法, 接收透傳消息,進行處理,全局的消息工具類和會話類中都需要實現該代理方法,更新相關數據及頁面
- (void)cmdMessagesDidReceive:(NSArray *)aCmdMessages
{
BOOL isRefreshCons = YES;
for (EMMessage *cmdMessage in aCmdMessages) {
EMCmdMessageBody *body = (EMCmdMessageBody *)cmdMessage.body;
if ([body.action isEqualToString:@"REVOKE_FLAG"]) {
NSString *revokeMessageId = cmdMessage.ext[@"msgId"];
BOOL isSuccess = [self removeRevokeMessageWithChatter:cmdMessage.conversationId conversationType:(EMConversationType)cmdMessage.chatType messageId:revokeMessageId];
//發送刷新會話列表通知
[[NSNotificationCenter defaultCenter] postNotificationName:@"conversationListRefresh" object:nil];
[self.tableView reloadData];
}
}
}
- (BOOL)removeRevokeMessageWithChatter:(NSString *)aChatter
conversationType:(EMConversationType)type
messageId:(NSString *)messageId{
EMConversation *conversation = [[EMClient sharedClient].chatManager getConversation:aChatter type:type createIfNotExist:YES];
[conversation deleteMessageWithId:messageId error:nil];
[self.messsagesSource enumerateObjectsWithOptions:NSEnumerationReverse usingBlock:^(EMMessage *obj, NSUInteger idx, BOOL *stop) {
if ([obj isKindOfClass:[EMMessage class]] && [obj.messageId isEqualToString:messageId]) {
[self.messsagesSource removeObject:obj];
}
}];
NSArray *formattedMessages = [self formatMessages:self.messsagesSource];
[self.dataArray removeAllObjects];
[self.dataArray addObjectsFromArray:formattedMessages]; return YES;
}
環信IM中的多代理的使用
[[EMClient sharedClient] addDelegate:self delegateQueue:nil];
[[EMClient sharedClient].groupManager addDelegate:self delegateQueue:nil];
[[EMClient sharedClient].contactManager addDelegate:self delegateQueue:nil];
[[EMClient sharedClient].roomManager addDelegate:self delegateQueue:nil];
[[EMClient sharedClient].chatManager addDelegate:self delegateQueue:nil];
在設置代理的過程中發現,環信可以設置多代理,但請注意:
1、addDelegate一個類只能一次,不然當前類代理方法調用的次數就是你add的次數(猜想,addDelegate就只添加代理,沒有去重,當調用代理方法的時候,遍歷了所有的delegate,所以同一個類的delegate方法會執行多次)
2、環信注冊一定放在didFinishLaunch最前面,不然后來的環信相關配置都失效