環信3.x集成的一些問題

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最前面,不然后來的環信相關配置都失效

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容

  • 1)項目里面不需要環信SDK的太多功能,只是想要聊天和好友功能,其他都不用,那SDK一定要總是跟著更新么? a.環...
    DefaultYuan閱讀 26,648評論 17 59
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 173,532評論 25 708
  • 最近項目中用到環信(項目中使用的是沒有音視頻的SDK),這邊就從導入SDK、引入依賴庫后開始說起,網上找了些資料都...
    Nest0912閱讀 534評論 0 1
  • 點擊查看原文 Web SDK 開發手冊 SDK 概述 網易云信 SDK 為 Web 應用提供一個完善的 IM 系統...
    layjoy閱讀 13,951評論 0 15
  • 夢萬鄉 我把親手揉碎的夢舒展 平鋪桌上 心想著再夢一場 生命是本書 我遺失的是其中一頁 小小一片 卻落得我殘缺不全...
    逐月馬尾閱讀 176評論 0 2